#![no_std]
#![feature(stdarch_arm_hints)]
#![feature(stdarch_arm_neon_intrinsics)]
#![feature(generic_const_exprs)]
#![feature(associated_type_defaults)]
#![warn(missing_docs)]
#![doc = include_str!("../../README.md")]
use embassy_sync::mutex::{Mutex, MutexGuard};
use embassy_sync::signal::Signal;
use crate::hw::platform::RawMutex;
pub(crate) trait StaticArray {
const LEN: usize;
}
impl<T, const N: usize> StaticArray for [T; N] {
const LEN: usize = N;
}
trait LEDEffect {
fn is_animated(&self) -> bool;
fn is_reactive(&self) -> bool;
}
trait Cycle {
fn increment(&mut self);
fn decrement(&mut self);
}
pub struct State<'a, T: Clone + PartialEq> {
data: Mutex<RawMutex, T>,
listeners: &'a [&'a Signal<RawMutex, ()>],
}
impl<'a, T: Clone + PartialEq> State<'a, T> {
pub const fn new(data: T, listeners: &'a [&'a Signal<RawMutex, ()>]) -> State<'a, T> {
Self {
data: Mutex::new(data),
listeners,
}
}
pub async fn get(&self) -> T {
self.data.lock().await.clone()
}
async fn set_inner(&self, value: T) -> bool {
let mut data = self.data.lock().await;
let changed = *data != value;
*data = value;
changed
}
pub async fn set(&self, value: T) {
if self.set_inner(value).await {
self.notify_listeners();
}
}
pub async fn quiet_set(&self, value: T) {
self.set_inner(value).await;
}
async fn update_inner<R>(
&self,
updater: impl FnOnce(&mut MutexGuard<'_, RawMutex, T>) -> R,
) -> (bool, R) {
let mut data = self.data.lock().await;
let old = data.clone();
let update_result = updater(&mut data);
(old != *data, update_result)
}
pub async fn update<R>(
&self,
updater: impl FnOnce(&mut MutexGuard<'_, RawMutex, T>) -> R,
) -> R {
let (changed, update_result) = self.update_inner(updater).await;
if changed {
self.notify_listeners();
}
update_result
}
pub async fn quiet_update<R>(
&self,
updater: impl FnOnce(&mut MutexGuard<'_, RawMutex, T>) -> R,
) -> R {
let (_changed, update_result) = self.update_inner(updater).await;
update_result
}
fn notify_listeners(&self) {
for listener in self.listeners.iter() {
listener.signal(());
}
}
}
pub use keyberon;
pub use once_cell;
pub use rumcake_macros::keyboard_main as keyboard;
pub mod keyboard;
mod math;
#[cfg(feature = "storage")]
pub mod storage;
#[cfg(feature = "lighting")]
pub mod lighting;
#[cfg(feature = "usb")]
pub mod usb;
#[cfg(feature = "via")]
pub mod via;
#[cfg(feature = "vial")]
pub mod vial;
#[cfg(any(feature = "split-peripheral", feature = "split-central"))]
pub mod split;
#[cfg(feature = "bluetooth")]
pub mod bluetooth;
#[cfg(feature = "display")]
pub mod display;
pub mod hw;
pub mod drivers;
pub mod tasks {
pub use crate::hw::output_switcher;
pub use crate::keyboard::{ec11_encoders_poll, layout_collect, matrix_poll};
#[cfg(all(feature = "lighting", feature = "storage"))]
pub use crate::lighting::lighting_storage_task;
#[cfg(feature = "lighting")]
pub use crate::lighting::lighting_task;
#[cfg(feature = "display")]
pub use crate::display::display_task;
#[cfg(feature = "usb")]
pub use crate::usb::{start_usb, usb_hid_consumer_write_task, usb_hid_kb_write_task};
#[cfg(all(feature = "via", feature = "usb"))]
pub use crate::usb::usb_hid_via_read_task;
#[cfg(all(feature = "via", feature = "usb"))]
pub use crate::usb::usb_hid_via_write_task;
#[cfg(feature = "via")]
pub use crate::via::via_process_task;
#[cfg(feature = "vial")]
pub use crate::vial::vial_process_task;
#[cfg(feature = "split-central")]
pub use crate::split::central::central_task;
#[cfg(feature = "split-peripheral")]
pub use crate::split::peripheral::peripheral_task;
#[cfg(feature = "nrf")]
pub use crate::hw::platform::adc_task;
#[cfg(feature = "nrf-ble")]
pub use crate::hw::platform::softdevice_task;
#[cfg(all(feature = "nrf", feature = "bluetooth"))]
pub use crate::bluetooth::nrf_ble::nrf_ble_task;
#[cfg(all(feature = "nrf-ble", feature = "split-central"))]
pub use crate::drivers::nrf_ble::central::nrf_ble_central_task;
#[cfg(all(feature = "nrf-ble", feature = "split-peripheral"))]
pub use crate::drivers::nrf_ble::peripheral::nrf_ble_peripheral_task;
}