2023-07-13 09:16:11 +00:00
|
|
|
//! This example shows powerful PIO module in the RP2040 chip to communicate with WS2812 LED modules.
|
|
|
|
//! See (https://www.sparkfun.com/categories/tags/ws2812)
|
|
|
|
|
2023-03-11 07:58:28 +00:00
|
|
|
#![no_std]
|
|
|
|
#![no_main]
|
|
|
|
#![feature(type_alias_impl_trait)]
|
|
|
|
|
|
|
|
use defmt::*;
|
|
|
|
use embassy_executor::Spawner;
|
2023-05-19 20:48:47 +00:00
|
|
|
use embassy_rp::dma::{AnyChannel, Channel};
|
2023-07-07 02:30:46 +00:00
|
|
|
use embassy_rp::peripherals::PIO0;
|
|
|
|
use embassy_rp::pio::{
|
|
|
|
Common, Config, FifoJoin, Instance, InterruptHandler, Pio, PioPin, ShiftConfig, ShiftDirection, StateMachine,
|
|
|
|
};
|
|
|
|
use embassy_rp::{bind_interrupts, clocks, into_ref, Peripheral, PeripheralRef};
|
2023-03-11 07:58:28 +00:00
|
|
|
use embassy_time::{Duration, Timer};
|
2023-05-19 20:48:47 +00:00
|
|
|
use fixed::types::U24F8;
|
2023-05-06 09:36:07 +00:00
|
|
|
use fixed_macro::fixed;
|
2023-03-11 07:58:28 +00:00
|
|
|
use smart_leds::RGB8;
|
|
|
|
use {defmt_rtt as _, panic_probe as _};
|
2023-05-19 20:48:47 +00:00
|
|
|
|
2023-07-07 02:30:46 +00:00
|
|
|
bind_interrupts!(struct Irqs {
|
|
|
|
PIO0_IRQ_0 => InterruptHandler<PIO0>;
|
|
|
|
});
|
|
|
|
|
2023-05-19 20:48:47 +00:00
|
|
|
pub struct Ws2812<'d, P: Instance, const S: usize, const N: usize> {
|
|
|
|
dma: PeripheralRef<'d, AnyChannel>,
|
2023-05-03 15:16:35 +00:00
|
|
|
sm: StateMachine<'d, P, S>,
|
2023-03-11 07:58:28 +00:00
|
|
|
}
|
|
|
|
|
2023-05-19 20:48:47 +00:00
|
|
|
impl<'d, P: Instance, const S: usize, const N: usize> Ws2812<'d, P, S, N> {
|
|
|
|
pub fn new(
|
|
|
|
pio: &mut Common<'d, P>,
|
|
|
|
mut sm: StateMachine<'d, P, S>,
|
|
|
|
dma: impl Peripheral<P = impl Channel> + 'd,
|
|
|
|
pin: impl PioPin,
|
|
|
|
) -> Self {
|
|
|
|
into_ref!(dma);
|
|
|
|
|
2023-03-11 07:58:28 +00:00
|
|
|
// Setup sm0
|
|
|
|
|
|
|
|
// prepare the PIO program
|
|
|
|
let side_set = pio::SideSet::new(false, 1, false);
|
|
|
|
let mut a: pio::Assembler<32> = pio::Assembler::new_with_side_set(side_set);
|
|
|
|
|
|
|
|
const T1: u8 = 2; // start bit
|
|
|
|
const T2: u8 = 5; // data bit
|
|
|
|
const T3: u8 = 3; // stop bit
|
|
|
|
const CYCLES_PER_BIT: u32 = (T1 + T2 + T3) as u32;
|
|
|
|
|
|
|
|
let mut wrap_target = a.label();
|
|
|
|
let mut wrap_source = a.label();
|
|
|
|
let mut do_zero = a.label();
|
|
|
|
a.set_with_side_set(pio::SetDestination::PINDIRS, 1, 0);
|
|
|
|
a.bind(&mut wrap_target);
|
|
|
|
// Do stop bit
|
|
|
|
a.out_with_delay_and_side_set(pio::OutDestination::X, 1, T3 - 1, 0);
|
|
|
|
// Do start bit
|
|
|
|
a.jmp_with_delay_and_side_set(pio::JmpCondition::XIsZero, &mut do_zero, T1 - 1, 1);
|
|
|
|
// Do data bit = 1
|
|
|
|
a.jmp_with_delay_and_side_set(pio::JmpCondition::Always, &mut wrap_target, T2 - 1, 1);
|
|
|
|
a.bind(&mut do_zero);
|
|
|
|
// Do data bit = 0
|
|
|
|
a.nop_with_delay_and_side_set(T2 - 1, 0);
|
|
|
|
a.bind(&mut wrap_source);
|
|
|
|
|
|
|
|
let prg = a.assemble_with_wrap(wrap_source, wrap_target);
|
2023-05-06 09:36:07 +00:00
|
|
|
let mut cfg = Config::default();
|
2023-03-11 07:58:28 +00:00
|
|
|
|
|
|
|
// Pin config
|
2023-04-25 18:16:27 +00:00
|
|
|
let out_pin = pio.make_pio_pin(pin);
|
2023-05-09 21:05:39 +00:00
|
|
|
cfg.set_out_pins(&[&out_pin]);
|
|
|
|
cfg.set_set_pins(&[&out_pin]);
|
2023-05-05 17:49:34 +00:00
|
|
|
|
2023-07-28 16:45:57 +00:00
|
|
|
cfg.use_program(&pio.load_program(&prg), &[&out_pin]);
|
2023-03-11 07:58:28 +00:00
|
|
|
|
2023-05-06 09:36:07 +00:00
|
|
|
// Clock config, measured in kHz to avoid overflows
|
2023-03-11 07:58:28 +00:00
|
|
|
// TODO CLOCK_FREQ should come from embassy_rp
|
2023-05-19 20:48:47 +00:00
|
|
|
let clock_freq = U24F8::from_num(clocks::clk_sys_freq() / 1000);
|
2023-05-06 09:36:07 +00:00
|
|
|
let ws2812_freq = fixed!(800: U24F8);
|
|
|
|
let bit_freq = ws2812_freq * CYCLES_PER_BIT;
|
|
|
|
cfg.clock_divider = clock_freq / bit_freq;
|
2023-03-11 07:58:28 +00:00
|
|
|
|
|
|
|
// FIFO config
|
2023-05-06 09:36:07 +00:00
|
|
|
cfg.fifo_join = FifoJoin::TxOnly;
|
|
|
|
cfg.shift_out = ShiftConfig {
|
|
|
|
auto_fill: true,
|
|
|
|
threshold: 24,
|
|
|
|
direction: ShiftDirection::Left,
|
|
|
|
};
|
|
|
|
|
|
|
|
sm.set_config(&cfg);
|
2023-03-11 07:58:28 +00:00
|
|
|
sm.set_enable(true);
|
|
|
|
|
2023-05-19 20:48:47 +00:00
|
|
|
Self {
|
|
|
|
dma: dma.map_into(),
|
|
|
|
sm,
|
|
|
|
}
|
2023-03-11 07:58:28 +00:00
|
|
|
}
|
|
|
|
|
2023-05-19 20:48:47 +00:00
|
|
|
pub async fn write(&mut self, colors: &[RGB8; N]) {
|
|
|
|
// Precompute the word bytes from the colors
|
|
|
|
let mut words = [0u32; N];
|
|
|
|
for i in 0..N {
|
|
|
|
let word = (u32::from(colors[i].g) << 24) | (u32::from(colors[i].r) << 16) | (u32::from(colors[i].b) << 8);
|
|
|
|
words[i] = word;
|
2023-03-11 07:58:28 +00:00
|
|
|
}
|
2023-05-19 20:48:47 +00:00
|
|
|
|
|
|
|
// DMA transfer
|
|
|
|
self.sm.tx().dma_push(self.dma.reborrow(), &words).await;
|
2023-03-11 07:58:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Input a value 0 to 255 to get a color value
|
|
|
|
/// The colours are a transition r - g - b - back to r.
|
|
|
|
fn wheel(mut wheel_pos: u8) -> RGB8 {
|
|
|
|
wheel_pos = 255 - wheel_pos;
|
|
|
|
if wheel_pos < 85 {
|
|
|
|
return (255 - wheel_pos * 3, 0, wheel_pos * 3).into();
|
|
|
|
}
|
|
|
|
if wheel_pos < 170 {
|
|
|
|
wheel_pos -= 85;
|
|
|
|
return (0, wheel_pos * 3, 255 - wheel_pos * 3).into();
|
|
|
|
}
|
|
|
|
wheel_pos -= 170;
|
|
|
|
(wheel_pos * 3, 255 - wheel_pos * 3, 0).into()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[embassy_executor::main]
|
|
|
|
async fn main(_spawner: Spawner) {
|
|
|
|
info!("Start");
|
|
|
|
let p = embassy_rp::init(Default::default());
|
|
|
|
|
2023-07-07 02:30:46 +00:00
|
|
|
let Pio { mut common, sm0, .. } = Pio::new(p.PIO0, Irqs);
|
2023-03-11 07:58:28 +00:00
|
|
|
|
|
|
|
// This is the number of leds in the string. Helpfully, the sparkfun thing plus and adafruit
|
|
|
|
// feather boards for the 2040 both have one built in.
|
|
|
|
const NUM_LEDS: usize = 1;
|
|
|
|
let mut data = [RGB8::default(); NUM_LEDS];
|
|
|
|
|
|
|
|
// For the thing plus, use pin 8
|
|
|
|
// For the feather, use pin 16
|
2023-05-19 20:48:47 +00:00
|
|
|
let mut ws2812 = Ws2812::new(&mut common, sm0, p.DMA_CH0, p.PIN_16);
|
2023-03-11 07:58:28 +00:00
|
|
|
|
|
|
|
// Loop forever making RGB values and pushing them out to the WS2812.
|
|
|
|
loop {
|
|
|
|
for j in 0..(256 * 5) {
|
|
|
|
debug!("New Colors:");
|
|
|
|
for i in 0..NUM_LEDS {
|
|
|
|
data[i] = wheel((((i * 256) as u16 / NUM_LEDS as u16 + j as u16) & 255) as u8);
|
|
|
|
debug!("R: {} G: {} B: {}", data[i].r, data[i].g, data[i].b);
|
|
|
|
}
|
|
|
|
ws2812.write(&data).await;
|
|
|
|
|
|
|
|
Timer::after(Duration::from_micros(5)).await;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|