1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
//! [![github]](https://github.com/rafaelcaricio/lvgl-rs) [![crates-io]](https://crates.io/crates/lvgl) [![docs-rs]](crate)
//!
//! [github]: https://img.shields.io/badge/github-8da0cb?style=for-the-badge&labelColor=555555&logo=github
//! [crates-io]: https://img.shields.io/badge/crates.io-fc8d62?style=for-the-badge&labelColor=555555&logo=rust
//! [docs-rs]: https://img.shields.io/badge/docs.rs-66c2a5?style=for-the-badge&labelColor=555555&logoColor=white&logo=
//!
//! <br>
//!
//! [LVGL][1] bindings for Rust. A powerful and easy-to-use embedded GUI with many widgets, advanced visual effects, and
//! low memory footprint. This crate is compatible with `#![no_std]` environments by default.
//!
//! [1]: https://docs.lvgl.io/8.3/get-started/index.html
//!

#![cfg_attr(not(test), no_std)]
#![cfg_attr(feature = "nightly", feature(cfg_accessible))]

#[macro_use]
extern crate bitflags;

#[macro_use]
mod lv_core;

#[cfg(feature = "alloc")]
extern crate alloc;

// We can ONLY use `alloc::boxed::Box` if `lvgl_alloc` is enabled.
// That is because we use `Box` to send memory references to LVGL. Since the global allocator, when
// `lvgl_alloc` feature is enabled, is the LVGL memory manager then everything is in LVGL
// managed memory anyways. In that case we can use the Rust's provided Box definition.
#[cfg(feature = "lvgl_alloc")]
use ::alloc::boxed::Box;

#[cfg(feature = "lvgl_alloc")]
mod allocator;

#[cfg(not(feature = "lvgl_alloc"))]
pub(crate) mod mem;

// When LVGL allocator is not used on the Rust code, we need a way to add objects to the LVGL
// managed memory. We implement a very simple `Box` that has the minimal features to copy memory
// safely to the LVGL managed memory.
#[cfg(not(feature = "lvgl_alloc"))]
use crate::mem::Box;

use core::sync::atomic::{AtomicBool, Ordering};

pub use display::*;
pub use functions::*;
pub use lv_core::*;
pub use support::*;

mod display;
mod functions;
mod support;

#[cfg(feature = "drivers")]
pub mod drivers;
pub mod font;
pub mod input_device;
pub mod widgets;

#[cfg(feature = "rust_timer")]
pub mod timer;

struct RunOnce(AtomicBool);

impl RunOnce {
    const fn new() -> Self {
        Self(AtomicBool::new(false))
    }

    fn swap_and_check(&self) -> bool {
        self.0
            .compare_exchange(false, true, Ordering::Relaxed, Ordering::Relaxed)
            .is_ok()
    }
}

#[cfg(feature = "unsafe_no_autoinit")]
static LVGL_INITIALIZED: RunOnce = RunOnce::new();

/// Initializes LVGL. Call at the start of the program.
#[cfg(feature = "unsafe_no_autoinit")]
pub fn init() {
    if LVGL_INITIALIZED.swap_and_check() {
        unsafe {
            lvgl_sys::lv_init();
        }
    }
}

#[cfg(not(feature = "unsafe_no_autoinit"))]
#[ctor::ctor]
fn init() {
    unsafe {
        lvgl_sys::lv_init();
    }
}

#[cfg(test)]
pub(crate) mod tests {
    use super::*;
    use crate::display::{Display, DrawBuffer};

    pub(crate) fn initialize_test() {
        #[cfg(feature = "unsafe_no_autoinit")]
        init();

        static ONCE_INIT: RunOnce = RunOnce::new();
        const REFRESH_BUFFER_SIZE: usize = 64 * 64 / 10;
        let buffer = DrawBuffer::<REFRESH_BUFFER_SIZE>::default();

        if ONCE_INIT.swap_and_check() {
            let _ = Display::register(buffer, 240, 240, |_| {}).unwrap();
        }
    }
}