337 lines
11 KiB
Rust
337 lines
11 KiB
Rust
use core::borrow::BorrowMut;
|
|
use std::sync::Arc;
|
|
use std::fmt::Debug;
|
|
use std::sync::Mutex;
|
|
|
|
use chrono::DateTime;
|
|
use chrono::Utc;
|
|
|
|
use esp_idf_svc::eventloop::{EspSubscription, EspSystemEventLoop, System};
|
|
use esp_idf_svc::hal::modem::Modem;
|
|
use esp_idf_svc::hal::prelude::Peripherals;
|
|
use esp_idf_svc::hal::task::thread::ThreadSpawnConfiguration;
|
|
use esp_idf_svc::netif::IpEvent;
|
|
use esp_idf_svc::nvs::{EspDefaultNvsPartition, EspNvsPartition, NvsDefault};
|
|
use esp_idf_svc::sntp::EspSntp;
|
|
use esp_idf_svc::sys::esp_efuse_mac_get_default;
|
|
use esp_idf_svc::wifi::{AuthMethod, ClientConfiguration, Configuration, EspWifi, WifiEvent};
|
|
use rgb::Rgb;
|
|
|
|
use super::smart_leds_lib::rmt::FastWs2812Esp32Rmt;
|
|
use super::smart_leds_lib::StrideOutput;
|
|
use super::Board;
|
|
|
|
use crate::buffers::BufferedSurfacePool;
|
|
use crate::buffers::Pixbuf;
|
|
use crate::mappings::StrideMapping;
|
|
use crate::task::FixedSizeScheduler;
|
|
use crate::task::Task;
|
|
use crate::time::Periodically;
|
|
|
|
pub mod i2s {
|
|
use esp_idf_svc::hal::i2s::*;
|
|
use rgb::ComponentBytes;
|
|
use rgb::Rgb;
|
|
|
|
use crate::mappings::*;
|
|
use crate::buffers::Pixbuf;
|
|
use crate::render::Output;
|
|
use crate::render::Sample;
|
|
|
|
pub struct I2SOutput<'d> {
|
|
driver: I2sDriver<'d, I2sTx>,
|
|
pixbuf: [Rgb<u8>; 310],
|
|
pixmap: StrideMapping,
|
|
}
|
|
|
|
impl<'d> I2SOutput<'d> {
|
|
fn new(driver: I2sDriver<'d, I2sTx>) -> Self {
|
|
I2SOutput {
|
|
driver,
|
|
pixbuf: Pixbuf::new(),
|
|
pixmap: StrideMapping::new_jar()
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'d> Output for I2SOutput<'d> {
|
|
fn blank(&mut self) {
|
|
self.pixbuf.blank();
|
|
}
|
|
|
|
fn commit(&mut self) {
|
|
let bytes = self.pixbuf.as_bytes();
|
|
let mut written = self.driver.preload_data(bytes).unwrap();
|
|
self.driver.tx_enable().unwrap();
|
|
while written < bytes.len() {
|
|
let next = &bytes[written..];
|
|
written += self.driver.write(next, 0).unwrap();
|
|
}
|
|
self.driver.tx_disable().unwrap();
|
|
}
|
|
}
|
|
|
|
impl<'d> Sample for I2SOutput<'d> {
|
|
type Pixel = Rgb<u8>;
|
|
fn sample(&mut self, rect: &crate::geometry::Rectangle<crate::geometry::Virtual>) -> impl crate::render::PixelView<Pixel = Self::Pixel> {
|
|
StrideSampler::new(&mut self.pixbuf, self.pixmap.select(rect))
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct Esp32Board {
|
|
sys_loop: EspSystemEventLoop,
|
|
modem: Option<Modem>,
|
|
surfaces: BufferedSurfacePool,
|
|
output: Option<<Self as Board>::Output>
|
|
}
|
|
|
|
impl Board for Esp32Board {
|
|
type Output = StrideOutput<[Rgb<u8>; 310], FastWs2812Esp32Rmt<'static>>;
|
|
type Surfaces = BufferedSurfacePool;
|
|
type Scheduler = FixedSizeScheduler<2>;
|
|
|
|
fn take() -> Self {
|
|
// It is necessary to call this function once. Otherwise some patches to the runtime
|
|
// implemented by esp-idf-sys might not link properly. See https://github.com/esp-rs/esp-idf-template/issues/71
|
|
esp_idf_svc::sys::link_patches();
|
|
|
|
// Bind the log crate to the ESP Logging facilities
|
|
esp_idf_svc::log::EspLogger::initialize_default();
|
|
|
|
let peripherals = Peripherals::take().unwrap();
|
|
let sys_loop = EspSystemEventLoop::take().unwrap();
|
|
|
|
let mut chip_id: [u8; 8] = [0; 8];
|
|
unsafe {
|
|
esp_efuse_mac_get_default(&mut chip_id as *mut u8);
|
|
}
|
|
|
|
log::info!("Setting up output for chip ID {:x?}", chip_id);
|
|
|
|
const POWER_VOLTS : u32 = 5;
|
|
const POWER_MA : u32 = 500;
|
|
const MAX_POWER_MW : u32 = POWER_VOLTS * POWER_MA;
|
|
let pins = peripherals.pins;
|
|
|
|
ThreadSpawnConfiguration {
|
|
pin_to_core: Some(esp_idf_svc::hal::cpu::Core::Core1),
|
|
..Default::default()
|
|
}.set().unwrap();
|
|
// Wifi driver creates too many interrupts on core0, so we need to use RMT on core1.
|
|
// But the implementation spawns a thread based on the core the driver was created in,
|
|
// so we create the driver in another thread briefly.
|
|
// Fun stuff.
|
|
let output = match chip_id {
|
|
[72, 202, 67, 89, 145, 204, 0, 0] => {
|
|
StrideOutput::new(
|
|
Pixbuf::new(),
|
|
StrideMapping::new_panel(),
|
|
std::thread::spawn(move || { FastWs2812Esp32Rmt::new(peripherals.rmt.channel0, pins.gpio5).unwrap() }).join().unwrap(),
|
|
MAX_POWER_MW
|
|
)
|
|
},
|
|
[0x8C, 0xAA, 0xB5, 0x83, 0x5f, 0x74, 0x0, 0x0] => {
|
|
StrideOutput::new(
|
|
Pixbuf::new(),
|
|
StrideMapping::new_jar(),
|
|
std::thread::spawn(move || { FastWs2812Esp32Rmt::new(peripherals.rmt.channel0, pins.gpio14).unwrap() }).join().unwrap(),
|
|
MAX_POWER_MW
|
|
)
|
|
},
|
|
[0x4a, 0xca, 0x43, 0x59, 0x85, 0x58, 0x0, 0x0] => { // Albus the tree
|
|
StrideOutput::new(
|
|
Pixbuf::new(),
|
|
StrideMapping::new_jar(),
|
|
std::thread::spawn(move || { FastWs2812Esp32Rmt::new(peripherals.rmt.channel0, pins.gpio5).unwrap() }).join().unwrap(),
|
|
MAX_POWER_MW
|
|
)
|
|
},
|
|
[0x48, 0xca, 0x43, 0x59, 0x9d, 0x48, 0x0, 0x0] => { // kitchen cabinets
|
|
StrideOutput::new(
|
|
Pixbuf::new(),
|
|
StrideMapping::new_fairylights(),
|
|
std::thread::spawn(move || { FastWs2812Esp32Rmt::new(peripherals.rmt.channel0, pins.gpio5).unwrap() }).join().unwrap(),
|
|
MAX_POWER_MW
|
|
)
|
|
},
|
|
[0x48, 0xca, 0x43, 0x59, 0x9e, 0xdc, 0x0, 0x0] => { // front window
|
|
StrideOutput::new(
|
|
Pixbuf::new(),
|
|
StrideMapping::new_fairylights(),
|
|
std::thread::spawn(move || { FastWs2812Esp32Rmt::new(peripherals.rmt.channel0, pins.gpio5).unwrap() }).join().unwrap(),
|
|
MAX_POWER_MW
|
|
)
|
|
}
|
|
_ => {
|
|
StrideOutput::new(
|
|
Pixbuf::new(),
|
|
StrideMapping::new(),
|
|
std::thread::spawn(move || { FastWs2812Esp32Rmt::new(peripherals.rmt.channel0, pins.gpio5).unwrap() }).join().unwrap(),
|
|
MAX_POWER_MW
|
|
)
|
|
}
|
|
};
|
|
ThreadSpawnConfiguration {
|
|
..Default::default()
|
|
}.set().unwrap();
|
|
|
|
Esp32Board {
|
|
modem: Some(peripherals.modem),
|
|
sys_loop: sys_loop.clone(),
|
|
surfaces: BufferedSurfacePool::new(),
|
|
output: Some(output)
|
|
}
|
|
}
|
|
|
|
fn output(&mut self) -> Self::Output {
|
|
self.output.take().unwrap()
|
|
}
|
|
|
|
fn surfaces(&mut self) -> Self::Surfaces {
|
|
self.surfaces.clone()
|
|
}
|
|
|
|
fn system_tasks(&mut self) -> Self::Scheduler {
|
|
let nvs = EspDefaultNvsPartition::take().unwrap();
|
|
FixedSizeScheduler::new([
|
|
Box::new(WifiTask::new(self.modem.take().unwrap(), self.sys_loop.clone(), &nvs)),
|
|
Box::new(self.surfaces.clone())
|
|
])
|
|
}
|
|
}
|
|
|
|
impl Debug for WifiTask {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
f.debug_struct("WifiTask").finish()
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, PartialEq, Clone, Copy)]
|
|
enum WifiState {
|
|
Stopped,
|
|
Disconnected,
|
|
Connecting,
|
|
Connected
|
|
}
|
|
|
|
struct WifiTask {
|
|
wifi: EspWifi<'static>,
|
|
ntp: EspSntp<'static>,
|
|
connection_check: Periodically,
|
|
state: Arc<Mutex<WifiState>>,
|
|
last_state: WifiState,
|
|
sys_loop: EspSystemEventLoop,
|
|
wifi_sub: Option<EspSubscription<'static, System>>,
|
|
ip_sub: Option<EspSubscription<'static, System>>
|
|
}
|
|
|
|
impl WifiTask {
|
|
fn new(modem: Modem, sys_loop: EspSystemEventLoop, nvs: &EspNvsPartition<NvsDefault>) -> Self {
|
|
log::info!("Installing wifi driver");
|
|
let mut wifi = EspWifi::new(
|
|
modem,
|
|
sys_loop.clone(),
|
|
Some(nvs.clone())
|
|
).unwrap();
|
|
let wifi_config = Configuration::Client(ClientConfiguration {
|
|
ssid: "The Frequency".try_into().unwrap(),
|
|
bssid: None,
|
|
auth_method: AuthMethod::WPA2Personal,
|
|
password: "thepasswordkenneth".try_into().unwrap(),
|
|
channel: None,
|
|
..Default::default()
|
|
});
|
|
wifi.set_configuration(&wifi_config).unwrap();
|
|
|
|
WifiTask {
|
|
wifi,
|
|
ntp: EspSntp::new_default().unwrap(),
|
|
connection_check: Periodically::new_every_n_seconds(1),
|
|
state: Arc::new(Mutex::new(WifiState::Stopped)),
|
|
last_state: WifiState::Stopped,
|
|
sys_loop,
|
|
wifi_sub: None,
|
|
ip_sub: None
|
|
}
|
|
}
|
|
|
|
fn connect(&mut self) {
|
|
log::info!("Connecting wifi");
|
|
self.wifi.start().unwrap();
|
|
self.wifi.connect().unwrap();
|
|
}
|
|
|
|
fn disconnect(&mut self) {
|
|
log::info!("Disconnecting wifi");
|
|
self.wifi.disconnect().unwrap();
|
|
self.wifi.stop().unwrap();
|
|
}
|
|
}
|
|
|
|
impl Task for WifiTask {
|
|
fn start(&mut self) {
|
|
log::info!("Starting wifi!");
|
|
|
|
let wifi_state = self.state.clone();
|
|
self.wifi_sub = Some(self.sys_loop.subscribe::<WifiEvent, _>( move |evt| {
|
|
log::warn!("wifi event {:?}", evt);
|
|
let next_state = match evt {
|
|
WifiEvent::StaStopped => Some(WifiState::Disconnected),
|
|
WifiEvent::StaDisconnected => Some(WifiState::Disconnected),
|
|
WifiEvent::StaStarted => Some(WifiState::Connecting),
|
|
WifiEvent::StaConnected => Some(WifiState::Connecting),
|
|
_ => None
|
|
};
|
|
|
|
match next_state {
|
|
Some(s) => {
|
|
let mut state = wifi_state.lock().unwrap();
|
|
*state = s
|
|
},
|
|
None => ()
|
|
}
|
|
}).unwrap());
|
|
let ip_state = self.state.clone();
|
|
self.ip_sub = Some(self.sys_loop.subscribe::<IpEvent, _>(move |evt| {
|
|
log::warn!("ip event {:?}", evt);
|
|
match evt {
|
|
IpEvent::DhcpIpAssigned(_) => {
|
|
let mut state = ip_state.lock().unwrap();
|
|
*state = WifiState::Connected;
|
|
},
|
|
_ => ()
|
|
}
|
|
}).unwrap());
|
|
|
|
self.connect();
|
|
}
|
|
|
|
fn tick(&mut self ) {
|
|
if self.connection_check.tick() {
|
|
let cur_state = *self.state.lock().unwrap();
|
|
|
|
if self.last_state != cur_state {
|
|
match cur_state {
|
|
WifiState::Connected => log::info!("Wifi connected!"),
|
|
WifiState::Connecting => log::info!("Connecting!"),
|
|
WifiState::Stopped => log::info!("Stopped!"),
|
|
WifiState::Disconnected => log::info!("Disconnected!")
|
|
}
|
|
|
|
log::info!("online: {:?}", cur_state);
|
|
|
|
self.last_state = cur_state;
|
|
}
|
|
|
|
let now: DateTime<Utc> = std::time::SystemTime::now().into();
|
|
log::info!("Current time: {} status={:?}", now.format("%d/%m/%Y %T"), self.ntp.get_sync_status());
|
|
}
|
|
}
|
|
|
|
fn stop(&mut self) {
|
|
log::info!("Stopping wifi");
|
|
self.disconnect();
|
|
}
|
|
}
|