Make DrawBuffer customizable using const generics

This commit is contained in:
Rafael Caricio 2021-06-04 22:29:29 +02:00
parent 00e2004822
commit de568593ef
Signed by: rafaelcaricio
GPG key ID: 3C86DBCE8E93C947
5 changed files with 55 additions and 51 deletions

View file

@ -5,10 +5,9 @@ use embedded_graphics_simulator::{
OutputSettingsBuilder, SimulatorDisplay, SimulatorEvent, Window,
};
use lvgl;
use lvgl::display::{DefaultDisplay, Display, DrawBuffer};
use lvgl::style::Style;
use lvgl::widgets::{Label, LabelAlign};
use lvgl::{Align, Color, LvError, Part, State, Widget, UI};
use lvgl::{Align, Color, DefaultDisplay, Display, DrawBuffer, LvError, Part, State, Widget};
use lvgl_sys;
use parking_lot::Mutex;
use std::sync::Arc as SyncArc;
@ -16,25 +15,32 @@ use std::thread;
use std::thread::sleep;
use std::time::{Duration, Instant};
static DRAW_BUFFER: DrawBuffer = DrawBuffer::new();
fn main() -> Result<(), LvError> {
lvgl::init();
let embedded_graphics_display: SimulatorDisplay<Rgb565> = SimulatorDisplay::new(Size::new(
lvgl_sys::LV_HOR_RES_MAX,
lvgl_sys::LV_VER_RES_MAX,
lvgl::DISP_HOR_RES as u32,
lvgl::DISP_VER_RES as u32,
));
let output_settings = OutputSettingsBuilder::new().scale(1).build();
let mut window = Window::new("PineTime", &output_settings);
// Implement and register your display:
let shared_native_display = SyncArc::new(Mutex::new(embedded_graphics_display));
let _display = Display::register_shared(&DRAW_BUFFER, &shared_native_display)?;
// LVGL-rs usage starts here
lvgl::init();
// LVGL will render the graphics here first, and seed the rendered image to the
// display. The buffer size can be set freely but 1/10 screen size is a good starting point.
const REFRESH_BUFFER_SIZE: usize = lvgl::DISP_HOR_RES * lvgl::DISP_VER_RES / 10;
static DRAW_BUFFER: DrawBuffer<REFRESH_BUFFER_SIZE> = DrawBuffer::new();
// Register your native display with LVGL. We use the `Display::register_shared()` method here,
// but that's because the Simulator needs a mutable reference to the display so it can draw
// updates. On your embedded device code, you will use `Display::register()`.
let display = Display::register_shared(&DRAW_BUFFER, &shared_native_display)?;
// Create screen and widgets
let mut screen = DefaultDisplay::get_scr_act()?;
let mut screen = display.get_scr_act()?;
println!("Before all widgets: {:?}", mem_info());

View file

@ -80,7 +80,7 @@ typedef int16_t lv_coord_t;
#define LV_MEM_CUSTOM 0
#if LV_MEM_CUSTOM == 0
/* Size of the memory used by `lv_mem_alloc` in bytes (>= 2kB)*/
# define LV_MEM_SIZE (1048576U) // 1Mb
# define LV_MEM_SIZE (14U * 1024U)
/* Compiler prefix for a big array declaration */
# define LV_MEM_ATTR

View file

@ -18,7 +18,6 @@ embedded-graphics = "0.6.2"
cstr_core = "0.2.3"
bitflags = "1.2.1"
parking_lot = "0.11.1"
lazy_static = "1.4.0"
heapless = "0.7.1"
[features]

View file

@ -1,12 +1,11 @@
use crate::functions::CoreError;
use crate::Box;
use crate::{disp_drv_register, disp_get_default, get_str_act};
use crate::{Box, RunOnce};
use crate::{Color, Obj};
use core::cell::{Cell, RefCell};
use core::cell::RefCell;
use core::marker::PhantomData;
use core::mem::MaybeUninit;
use core::ptr::NonNull;
use core::sync::atomic::{AtomicBool, Ordering};
use core::{ptr, result};
use embedded_graphics::drawable;
use embedded_graphics::prelude::*;
@ -16,10 +15,8 @@ use parking_lot::Mutex;
#[cfg(feature = "alloc")]
use alloc::sync::Arc;
// TODO: Make this an external configuration
const REFRESH_BUFFER_LEN: usize = 2;
// Declare a buffer for the refresh rate
const BUF_SIZE: usize = lvgl_sys::LV_HOR_RES_MAX as usize * REFRESH_BUFFER_LEN;
pub const DISP_HOR_RES: usize = lvgl_sys::LV_HOR_RES_MAX as usize;
pub const DISP_VER_RES: usize = lvgl_sys::LV_VER_RES_MAX as usize;
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
pub enum DisplayError {
@ -42,7 +39,10 @@ impl Display {
Self { disp }
}
pub fn register<T, C>(draw_buffer: &'static DrawBuffer, native_display: T) -> Result<Self>
pub fn register<T, C, const N: usize>(
draw_buffer: &'static DrawBuffer<N>,
native_display: T,
) -> Result<Self>
where
T: DrawTarget<C>,
C: PixelColor + From<Color>,
@ -52,8 +52,8 @@ impl Display {
}
#[cfg(feature = "alloc")]
pub fn register_shared<T, C>(
draw_buffer: &'static DrawBuffer,
pub fn register_shared<T, C, const N: usize>(
draw_buffer: &'static DrawBuffer<N>,
shared_native_display: &SharedNativeDisplay<T>,
) -> Result<Self>
where
@ -65,7 +65,7 @@ impl Display {
Ok(disp_drv_register(&mut display_diver)?)
}
pub fn get_str_act(&self) -> Result<Obj> {
pub fn get_scr_act(&self) -> Result<Obj> {
Ok(get_str_act(Some(&self))?)
}
}
@ -86,33 +86,29 @@ impl DefaultDisplay {
}
}
pub struct DrawBuffer {
initialized: AtomicBool,
refresh_buffer: Mutex<RefCell<heapless::Vec<lvgl_sys::lv_color_t, BUF_SIZE>>>,
pub struct DrawBuffer<const N: usize> {
initialized: RunOnce,
refresh_buffer: Mutex<RefCell<heapless::Vec<lvgl_sys::lv_color_t, N>>>,
}
impl DrawBuffer {
impl<const N: usize> DrawBuffer<N> {
pub const fn new() -> Self {
Self {
initialized: AtomicBool::new(false),
initialized: RunOnce::new(),
refresh_buffer: const_mutex(RefCell::new(heapless::Vec::new())),
}
}
fn get_ptr(&self) -> Option<Box<lvgl_sys::lv_disp_buf_t>> {
if self
.initialized
.compare_exchange(false, true, Ordering::Relaxed, Ordering::Relaxed)
.is_ok()
{
if self.initialized.swap_and_check() {
let mut inner: MaybeUninit<lvgl_sys::lv_disp_buf_t> = MaybeUninit::uninit();
let refresh_buffer_guard = self.refresh_buffer.lock();
let primary_buffer_guard = self.refresh_buffer.lock();
let draw_buf = unsafe {
lvgl_sys::lv_disp_buf_init(
inner.as_mut_ptr(),
refresh_buffer_guard.borrow_mut().as_mut_ptr() as *mut _ as *mut cty::c_void,
primary_buffer_guard.borrow_mut().as_mut_ptr() as *mut _ as *mut cty::c_void,
ptr::null_mut(),
lvgl_sys::LV_HOR_RES_MAX * REFRESH_BUFFER_LEN as u32,
N as u32,
);
inner.assume_init()
};
@ -138,7 +134,10 @@ where
T: DrawTarget<C>,
C: PixelColor + From<Color>,
{
pub fn new(draw_buffer: &'static DrawBuffer, native_display: T) -> Result<Self> {
pub fn new<const N: usize>(
draw_buffer: &'static DrawBuffer<N>,
native_display: T,
) -> Result<Self> {
let mut disp_drv = unsafe {
let mut inner = MaybeUninit::uninit();
lvgl_sys::lv_disp_drv_init(inner.as_mut_ptr());
@ -171,8 +170,8 @@ where
}
#[cfg(feature = "alloc")]
pub fn new_shared(
draw_buffer: &'static DrawBuffer,
pub fn new_shared<const N: usize>(
draw_buffer: &'static DrawBuffer<N>,
shared_native_display: SharedNativeDisplay<T>,
) -> Result<Self> {
let mut disp_drv = unsafe {
@ -339,7 +338,7 @@ mod tests {
let display = Display::default();
let _screen_direct = display
.get_str_act()
.get_scr_act()
.expect("Return screen directly from the display instance");
let _screen_default =
@ -352,7 +351,7 @@ mod tests {
let display = Display::default();
let _screen = display
.get_str_act()
.get_scr_act()
.expect("Return screen directly from the display instance");
Ok(())

View file

@ -17,9 +17,6 @@
#[macro_use]
extern crate bitflags;
#[macro_use]
extern crate lazy_static;
#[macro_use]
mod lv_core;
@ -47,7 +44,8 @@ pub(crate) mod mem;
#[cfg(not(feature = "lvgl_alloc"))]
use crate::mem::Box;
pub mod display;
mod display;
pub use display::*;
mod functions;
mod support;
mod ui;
@ -65,7 +63,7 @@ impl RunOnce {
Self(AtomicBool::new(false))
}
fn swap_and_get(&self) -> bool {
fn swap_and_check(&self) -> bool {
self.0
.compare_exchange(false, true, Ordering::Relaxed, Ordering::Relaxed)
.is_ok()
@ -75,7 +73,7 @@ impl RunOnce {
static LVGL_INITIALIZED: RunOnce = RunOnce::new();
pub fn init() {
if LVGL_INITIALIZED.swap_and_get() {
if LVGL_INITIALIZED.swap_and_check() {
unsafe {
lvgl_sys::lv_init();
}
@ -92,11 +90,13 @@ pub(crate) mod tests {
pub(crate) fn initialize_test() {
init();
static DRAW_BUFFER: DrawBuffer = DrawBuffer::new();
let embedded_graphics_display: MockDisplay<Rgb565> = Default::default();
const REFRESH_BUFFER_SIZE: usize = 64 * 64 / 10;
static DRAW_BUFFER: DrawBuffer<REFRESH_BUFFER_SIZE> = DrawBuffer::new();
static ONCE_INIT: RunOnce = RunOnce::new();
if ONCE_INIT.swap_and_get() {
let embedded_graphics_display: MockDisplay<Rgb565> = Default::default();
if ONCE_INIT.swap_and_check() {
let _ = Display::register(&DRAW_BUFFER, embedded_graphics_display).unwrap();
}
}