mirror of
https://github.com/LegitCamper/picocalc-os-rs.git
synced 2025-12-27 15:55:25 +00:00
WIP Binary Ui
This commit is contained in:
@@ -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();
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
Reference in New Issue
Block a user