use crate::dma::Transfer;
use crate::gpio::Pull;
use crate::pio::{
Common, Config, Direction, FifoJoin, Instance, LoadedProgram, PioBatch, PioPin, ShiftConfig, ShiftDirection,
StateMachine,
};
use crate::pio_programs::clock_divider::calculate_pio_clock_divider;
use crate::{Peri, dma, interrupt};
pub struct PioI2sInProgram<'d, PIO: Instance> {
prg: LoadedProgram<'d, PIO>,
}
const PIO_I2S_IN_PROGRAM_CLK_MULTIPLIER: u32 = 2;
impl<'d, PIO: Instance> PioI2sInProgram<'d, PIO> {
pub fn new(common: &mut Common<'d, PIO>) -> Self {
let prg = pio::pio_asm! {
".side_set 2",
" set x, 14 side 0b01",
"left_data:",
" in pins, 1 side 0b00", " jmp x-- left_data side 0b01",
" in pins, 1 side 0b10", " set x, 14 side 0b11",
"right_data:",
" in pins, 1 side 0b10",
" jmp x-- right_data side 0b11",
" in pins, 1 side 0b00" };
let prg = common.load_program(&prg.program);
Self { prg }
}
}
pub struct PioI2sIn<'d, P: Instance, const S: usize> {
dma: dma::Channel<'d>,
sm: StateMachine<'d, P, S>,
}
impl<'d, P: Instance, const S: usize> PioI2sIn<'d, P, S> {
pub fn new<D: dma::ChannelInstance>(
common: &mut Common<'d, P>,
mut sm: StateMachine<'d, P, S>,
dma: Peri<'d, D>,
irq: impl interrupt::typelevel::Binding<D::Interrupt, dma::InterruptHandler<D>> + 'd,
data_pulldown: bool,
data_pin: Peri<'d, impl PioPin>,
bit_clock_pin: Peri<'d, impl PioPin>,
lr_clock_pin: Peri<'d, impl PioPin>,
sample_rate: u32,
bit_depth: u32,
channels: u32,
program: &PioI2sInProgram<'d, P>,
) -> Self {
let mut data_pin = common.make_pio_pin(data_pin);
if data_pulldown {
data_pin.set_pull(Pull::Down);
}
let bit_clock_pin = common.make_pio_pin(bit_clock_pin);
let lr_clock_pin = common.make_pio_pin(lr_clock_pin);
let cfg = {
let mut cfg = Config::default();
cfg.use_program(&program.prg, &[&bit_clock_pin, &lr_clock_pin]);
cfg.set_in_pins(&[&data_pin]);
let clock_frequency = sample_rate * bit_depth * channels;
cfg.clock_divider = calculate_pio_clock_divider(clock_frequency * PIO_I2S_IN_PROGRAM_CLK_MULTIPLIER);
cfg.shift_in = ShiftConfig {
threshold: 32,
direction: ShiftDirection::Left,
auto_fill: true,
};
cfg.fifo_join = FifoJoin::RxOnly; cfg
};
sm.set_config(&cfg);
sm.set_pin_dirs(Direction::In, &[&data_pin]);
sm.set_pin_dirs(Direction::Out, &[&lr_clock_pin, &bit_clock_pin]);
Self {
dma: dma::Channel::new(dma, irq),
sm,
}
}
pub fn start(&mut self) {
self.sm.set_enable(true);
}
pub fn stop(&mut self) {
self.sm.set_enable(true);
}
pub fn start_batched(&mut self, b: &mut PioBatch<'d, P>) {
b.set_enable(&mut self.sm, true);
}
pub fn stop_batched(&mut self, b: &mut PioBatch<'d, P>) {
b.set_enable(&mut self.sm, false);
}
pub fn read<'b>(&'b mut self, buff: &'b mut [u32]) -> Transfer<'b> {
self.sm.rx().dma_pull(&mut self.dma, buff, false)
}
}
pub struct PioI2sOutProgram<'d, PIO: Instance> {
prg: LoadedProgram<'d, PIO>,
}
const PIO_I2S_OUT_PROGRAM_CLK_MULTIPLIER: u32 = 2;
impl<'d, PIO: Instance> PioI2sOutProgram<'d, PIO> {
pub fn new(common: &mut Common<'d, PIO>) -> Self {
let prg = pio::pio_asm!(
".side_set 2", " mov x, y side 0b01", "left_data:",
" out pins, 1 side 0b00",
" jmp x-- left_data side 0b01",
" out pins, 1 side 0b10",
" mov x, y side 0b11",
"right_data:",
" out pins, 1 side 0b10",
" jmp x-- right_data side 0b11",
" out pins, 1 side 0b00",
);
let prg = common.load_program(&prg.program);
Self { prg }
}
}
pub struct PioI2sOut<'d, P: Instance, const S: usize> {
dma: dma::Channel<'d>,
sm: StateMachine<'d, P, S>,
}
impl<'d, P: Instance, const S: usize> PioI2sOut<'d, P, S> {
pub fn new<D: dma::ChannelInstance>(
common: &mut Common<'d, P>,
mut sm: StateMachine<'d, P, S>,
dma: Peri<'d, D>,
irq: impl interrupt::typelevel::Binding<D::Interrupt, dma::InterruptHandler<D>> + 'd,
data_pin: Peri<'d, impl PioPin>,
bit_clock_pin: Peri<'d, impl PioPin>,
lr_clock_pin: Peri<'d, impl PioPin>,
sample_rate: u32,
bit_depth: u32,
program: &PioI2sOutProgram<'d, P>,
) -> Self {
let data_pin = common.make_pio_pin(data_pin);
let bit_clock_pin = common.make_pio_pin(bit_clock_pin);
let lr_clock_pin = common.make_pio_pin(lr_clock_pin);
let bclk_frequency: u32 = sample_rate * bit_depth * 2;
let cfg = {
let mut cfg = Config::default();
cfg.use_program(&program.prg, &[&bit_clock_pin, &lr_clock_pin]);
cfg.set_out_pins(&[&data_pin]);
cfg.clock_divider = calculate_pio_clock_divider(bclk_frequency * PIO_I2S_OUT_PROGRAM_CLK_MULTIPLIER);
cfg.shift_out = ShiftConfig {
threshold: 32,
direction: ShiftDirection::Left,
auto_fill: true,
};
cfg.fifo_join = FifoJoin::TxOnly;
cfg
};
sm.set_config(&cfg);
sm.set_pin_dirs(Direction::Out, &[&data_pin, &lr_clock_pin, &bit_clock_pin]);
unsafe { sm.set_y(bit_depth - 2) };
Self {
dma: dma::Channel::new(dma, irq),
sm,
}
}
pub fn start(&mut self) {
self.sm.set_enable(true);
}
pub fn stop(&mut self) {
self.sm.set_enable(false);
}
pub fn start_batched(&mut self, b: &mut PioBatch<'d, P>) {
b.set_enable(&mut self.sm, true);
}
pub fn stop_batched(&mut self, b: &mut PioBatch<'d, P>) {
b.set_enable(&mut self.sm, false);
}
pub fn write<'b>(&'b mut self, buff: &'b [u32]) -> Transfer<'b> {
self.sm.tx().dma_push(&mut self.dma, buff, false)
}
}