WIP Binary Ui

This commit is contained in:
2025-09-04 19:32:47 -06:00
parent ec25ee8601
commit ce35df64c3
2 changed files with 69 additions and 55 deletions

View File

@@ -25,6 +25,7 @@ use crate::{
keyboard::{KeyCode, KeyState, read_keyboard_fifo}, keyboard::{KeyCode, KeyState, read_keyboard_fifo},
}, },
storage::{SDCARD, SdCard}, storage::{SDCARD, SdCard},
ui::ui_handler,
usb::{ENABLE_SCSI, usb_handler}, usb::{ENABLE_SCSI, usb_handler},
}; };
use alloc::vec::Vec; use alloc::vec::Vec;
@@ -33,7 +34,7 @@ use {defmt_rtt as _, panic_probe as _};
use defmt::unwrap; use defmt::unwrap;
use embassy_executor::{Executor, Spawner}; use embassy_executor::{Executor, Spawner};
use embassy_futures::join::{join, join3}; use embassy_futures::join::{join, join3, join4};
use embassy_rp::{ use embassy_rp::{
gpio::{Input, Level, Output, Pull}, gpio::{Input, Level, Output, Pull},
i2c::{self, I2c}, i2c::{self, I2c},
@@ -45,10 +46,7 @@ use embassy_rp::{
spi::{self, Spi}, spi::{self, Spi},
usb as embassy_rp_usb, usb as embassy_rp_usb,
}; };
use embassy_sync::{ use embassy_sync::{blocking_mutex::raw::CriticalSectionRawMutex, mutex::Mutex};
blocking_mutex::{Mutex, raw::CriticalSectionRawMutex},
signal::Signal,
};
use embassy_time::{Delay, Timer}; use embassy_time::{Delay, Timer};
use embedded_hal_bus::spi::ExclusiveDevice; use embedded_hal_bus::spi::ExclusiveDevice;
use embedded_sdmmc::SdCard as SdmmcSdCard; use embedded_sdmmc::SdCard as SdmmcSdCard;
@@ -73,7 +71,12 @@ static ALLOCATOR: Talck<spin::Mutex<()>, ClaimOnOom> =
Talc::new(unsafe { ClaimOnOom::new(Span::from_array(core::ptr::addr_of!(ARENA).cast_mut())) }) Talc::new(unsafe { ClaimOnOom::new(Span::from_array(core::ptr::addr_of!(ARENA).cast_mut())) })
.lock(); .lock();
static DRIVERS_READY: Signal<CriticalSectionRawMutex, ()> = Signal::new(); static TASK_STATE: Mutex<CriticalSectionRawMutex, TaskState> = Mutex::new(TaskState::Ui);
enum TaskState {
Ui,
Kernel,
}
#[embassy_executor::main] #[embassy_executor::main]
async fn main(_spawner: Spawner) { async fn main(_spawner: Spawner) {
@@ -119,16 +122,16 @@ async fn main(_spawner: Spawner) {
// runs dynamically loaded elf files // runs dynamically loaded elf files
#[embassy_executor::task] #[embassy_executor::task]
async fn userland_task() { async fn userland_task() {
DRIVERS_READY.wait().await; // DRIVERS_READY.wait().await;
defmt::info!("Loading binary"); // defmt::info!("Loading binary");
let binary_data: &[u8] = // let binary_data: &[u8] =
include_bytes!("../../target/thumbv8m.main-none-eabihf/release/calculator"); // include_bytes!("../../target/thumbv8m.main-none-eabihf/release/calculator");
defmt::info!("Running binary"); // defmt::info!("Running binary");
let entry = unsafe { load_binary(binary_data).unwrap() }; // let entry = unsafe { load_binary(binary_data).unwrap() };
entry().await; // entry().await;
} }
struct Display { struct Display {
@@ -183,6 +186,8 @@ async fn kernel_task(display: Display, sd: Sd, mcu: Mcu, usb: USB) {
let display_fut = display_handler(display); let display_fut = display_handler(display);
let ui_fut = ui_handler();
{ {
let mut config = spi::Config::default(); let mut config = spi::Config::default();
config.frequency = 400_000; config.frequency = 400_000;
@@ -201,15 +206,15 @@ async fn kernel_task(display: Display, sd: Sd, mcu: Mcu, usb: USB) {
let usb = embassy_rp_usb::Driver::new(usb, Irqs); let usb = embassy_rp_usb::Driver::new(usb, Irqs);
let usb_fut = usb_handler(usb); let usb_fut = usb_handler(usb);
ENABLE_SCSI.store(true, core::sync::atomic::Ordering::Relaxed); let key_abi_fut = async {
DRIVERS_READY.signal(());
join3(usb_fut, display_fut, async {
loop { loop {
Timer::after_millis(100).await; Timer::after_millis(100).await;
get_keys().await get_keys().await
} }
}) };
.await;
ENABLE_SCSI.store(true, core::sync::atomic::Ordering::Relaxed);
join4(usb_fut, display_fut, key_abi_fut, ui_fut).await;
} }
static mut KEY_CACHE: Queue<KeyEvent, 32> = Queue::new(); static mut KEY_CACHE: Queue<KeyEvent, 32> = Queue::new();

View File

@@ -1,7 +1,10 @@
use crate::{ use crate::{
TASK_STATE, TaskState,
display::{FRAMEBUFFER, SCREEN_HEIGHT, SCREEN_WIDTH}, display::{FRAMEBUFFER, SCREEN_HEIGHT, SCREEN_WIDTH},
format, format,
peripherals::keyboard,
}; };
use alloc::{string::String, vec::Vec};
use core::fmt::Debug; use core::fmt::Debug;
use defmt::info; use defmt::info;
use embassy_rp::{ use embassy_rp::{
@@ -9,6 +12,11 @@ use embassy_rp::{
peripherals::{PIN_13, PIN_14, PIN_15, SPI1}, peripherals::{PIN_13, PIN_14, PIN_15, SPI1},
spi::{Async, Spi}, spi::{Async, Spi},
}; };
use embassy_sync::{
blocking_mutex::raw::{CriticalSectionRawMutex, ThreadModeRawMutex},
mutex::Mutex,
signal::Signal,
};
use embassy_time::{Delay, Timer}; use embassy_time::{Delay, Timer};
use embedded_graphics::{ use embedded_graphics::{
Drawable, Drawable,
@@ -30,38 +38,38 @@ use embedded_layout::{
object_chain::Chain, object_chain::Chain,
prelude::*, prelude::*,
}; };
use heapless::{String, Vec}; use shared::keyboard::KeyCode;
pub struct UI<const MAX_SELECTIONS: usize, const MAX_STR_LEN: usize> { static SELECTIONS: Mutex<CriticalSectionRawMutex, SelectionList> =
pub selections_list: SelectionList<MAX_SELECTIONS, MAX_STR_LEN>, Mutex::new(SelectionList::new(Vec::new()));
pub async fn ui_handler() {
loop {
let state = TASK_STATE.lock().await;
if let TaskState::Ui = *state {
let mut selections = SELECTIONS.lock().await;
if let Some(event) = keyboard::read_keyboard_fifo().await {
match event.key {
KeyCode::JoyUp => selections.up(),
KeyCode::JoyDown => selections.down(),
KeyCode::Enter | KeyCode::JoyRight => (),
_ => (),
}
}
draw_selection().await;
}
}
} }
impl<const MAX_SELECTIONS: usize, const MAX_STR_LEN: usize> UI<MAX_SELECTIONS, MAX_STR_LEN> { async fn draw_selection() {
pub fn new() -> Self {
Self {
selections_list: SelectionList::new(Vec::new()),
}
}
pub async fn draw<D: DrawTarget<Color = Rgb565>>(&mut self)
where
<D as DrawTarget>::Error: Debug,
{
self.draw_selection().await;
}
async fn draw_selection(&mut self) {
let mut fb_lock = FRAMEBUFFER.lock().await; let mut fb_lock = FRAMEBUFFER.lock().await;
let fb = fb_lock.as_mut().unwrap(); if let Some(fb) = fb_lock.as_mut() {
info!("UIINg");
let text_style = MonoTextStyle::new(&FONT_9X15, Rgb565::WHITE); let text_style = MonoTextStyle::new(&FONT_9X15, Rgb565::WHITE);
let selection = Rectangle::new( let guard = SELECTIONS.lock().await;
Point::new(0, 0), let mut file_names = guard.selections.iter();
Size::new(SCREEN_WIDTH as u32 - 1, SCREEN_HEIGHT as u32 - 1),
);
let mut file_names = self.selections_list.selections.iter();
let Some(first) = file_names.next() else { let Some(first) = file_names.next() else {
Text::new("No Programs found on SD Card\nEnsure programs end with '.bin',\nand are located in the root directory", Text::new("No Programs found on SD Card\nEnsure programs end with '.bin',\nand are located in the root directory",
@@ -72,24 +80,25 @@ impl<const MAX_SELECTIONS: usize, const MAX_STR_LEN: usize> UI<MAX_SELECTIONS, M
let chain = Chain::new(Text::new(first, Point::zero(), text_style)); let chain = Chain::new(Text::new(first, Point::zero(), text_style));
for _ in 0..10 {
LinearLayout::vertical(chain) LinearLayout::vertical(chain)
.with_alignment(horizontal::Center) .with_alignment(horizontal::Center)
.arrange() .arrange()
.align_to(&fb.bounding_box(), horizontal::Center, vertical::Center) .align_to(&fb.bounding_box(), horizontal::Center, vertical::Center)
.draw(*fb) .draw(*fb)
.unwrap(); .unwrap();
break;
}
} }
} }
pub struct SelectionList<const MAX_SELECTION: usize, const MAX_STR_LEN: usize> { pub struct SelectionList {
current_selection: u16, current_selection: u16,
selections: Vec<String<MAX_STR_LEN>, MAX_SELECTION>, selections: Vec<String>,
} }
impl<const MAX_SELECTION: usize, const MAX_STR_LEN: usize> impl SelectionList {
SelectionList<MAX_SELECTION, MAX_STR_LEN> pub const fn new(selections: Vec<String>) -> Self {
{
pub fn new(selections: Vec<String<MAX_STR_LEN>, MAX_SELECTION>) -> Self {
Self { Self {
selections, selections,
current_selection: 0, current_selection: 0,