From 7cca9a71d712b50f40e89fd1a47fa18b9354b3b9 Mon Sep 17 00:00:00 2001 From: Jan Niehusmann Date: Thu, 15 Sep 2022 20:41:25 +0000 Subject: [PATCH 1/2] simplify jump offset calculation when loading PIO programs --- rp2040-hal/src/pio.rs | 23 +++++++---------------- 1 file changed, 7 insertions(+), 16 deletions(-) diff --git a/rp2040-hal/src/pio.rs b/rp2040-hal/src/pio.rs index 698d4be..61d5521 100644 --- a/rp2040-hal/src/pio.rs +++ b/rp2040-hal/src/pio.rs @@ -195,26 +195,17 @@ impl PIO

{ .iter() .cloned() .map(|instr| { - if let Some(Instruction { - operands: InstructionOperands::JMP { condition, address }, - delay, - side_set, - }) = Instruction::decode(instr, p.side_set) - { - // JMP instruction. We need to apply offset here - let address = address + offset as u8; + if instr & 0b1110_0000_0000_0000 == 0 { + // this is a JMP instruction -> add offset to address + let address = (instr & 0b11111) as usize; + let address = address + offset; assert!( - address < pio::RP2040_MAX_PROGRAM_SIZE as u8, + address < pio::RP2040_MAX_PROGRAM_SIZE, "Invalid JMP out of the program after offset addition" ); - - Instruction { - operands: InstructionOperands::JMP { condition, address }, - delay, - side_set, - } - .encode(p.side_set) + instr & (!0b11111) | address as u16 } else { + // this is not a JMP instruction -> keep it unchanged instr } }) From 6a5747b213f5803216f85e6604d605f9bf4950e9 Mon Sep 17 00:00:00 2001 From: Jan Niehusmann Date: Sun, 18 Sep 2022 10:37:16 +0000 Subject: [PATCH 2/2] Use u8 instead of usize for pio address calculation --- rp2040-hal/src/pio.rs | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/rp2040-hal/src/pio.rs b/rp2040-hal/src/pio.rs index 61d5521..e9565cc 100644 --- a/rp2040-hal/src/pio.rs +++ b/rp2040-hal/src/pio.rs @@ -156,7 +156,7 @@ impl PIO

{ } /// Tries to find an appropriate offset for the instructions, in range 0..=31. - fn find_offset_for_instructions(&self, i: &[u16], origin: Option) -> Option { + fn find_offset_for_instructions(&self, i: &[u16], origin: Option) -> Option { if i.len() > PIO_INSTRUCTION_COUNT || i.is_empty() { None } else { @@ -167,10 +167,10 @@ impl PIO

{ { None } else { - Some(origin as usize) + Some(origin) } } else { - for i in (0..=32 - i.len()).rev() { + for i in (0..=32 - (i.len() as u8)).rev() { if self.used_instruction_space & (mask << i) == 0 { return Some(i); } @@ -197,10 +197,10 @@ impl PIO

{ .map(|instr| { if instr & 0b1110_0000_0000_0000 == 0 { // this is a JMP instruction -> add offset to address - let address = (instr & 0b11111) as usize; + let address = (instr & 0b11111) as u8; let address = address + offset; assert!( - address < pio::RP2040_MAX_PROGRAM_SIZE, + address < pio::RP2040_MAX_PROGRAM_SIZE as u8, "Invalid JMP out of the program after offset addition" ); instr & (!0b11111) | address as u16 @@ -211,7 +211,8 @@ impl PIO

{ }) .enumerate() .for_each(|(i, instr)| { - self.pio.instr_mem[i + offset].write(|w| unsafe { w.instr_mem0().bits(instr) }) + self.pio.instr_mem[i + offset as usize] + .write(|w| unsafe { w.instr_mem0().bits(instr) }) }); self.used_instruction_space |= Self::instruction_mask(p.code.len()) << offset; Ok(InstalledProgram {