diff --git a/rp2040-hal/src/pio.rs b/rp2040-hal/src/pio.rs index 3326400..a9b4f8d 100644 --- a/rp2040-hal/src/pio.rs +++ b/rp2040-hal/src/pio.rs @@ -473,11 +473,11 @@ impl UninitStateMachine { }); } - /// Set the current instruction. + /// Execute the instruction immediately. // Safety: The Send trait assumes this is the only write to sm_instr while uninitialized. The // initialized `StateMachine` may also use this register. The `UnintStateMachine` is consumed // by `PIOBuilder.build` to create `StateMachine` - fn set_instruction(&mut self, instruction: u16) { + fn exec_instruction(&mut self, instruction: u16) { self.sm() .sm_instr .write(|w| unsafe { w.sm0_instr().bits(instruction) }) @@ -519,10 +519,14 @@ impl StateMachine { self.sm.sm().sm_addr.read().bits() } - /// Set the current instruction. - pub fn set_instruction(&mut self, instruction: u16) { - // TODO: Check if this function is safe to call while the state machine is running. - self.sm.set_instruction(instruction); + ///Execute the instruction immediately. + pub fn exec_instruction(&mut self, instruction: u16) { + // This is allowed even if the state machine is running. + // + // However, the datasheet says: + // "If EXEC instructions are used, instructions written to INSTR must not stall" + // It's unclear what happens if this is violated. + self.sm.exec_instruction(instruction); } /// Check if the current instruction is stalled. @@ -564,7 +568,7 @@ impl StateMachine { .sm() .sm_pinctrl .write(|w| unsafe { w.set_base().bits(pin_num).set_count().bits(1) }); - self.set_instruction( + self.exec_instruction( pio::InstructionOperands::SET { destination: pio::SetDestination::PINS, data: if PinState::High == pin_state { 1 } else { 0 }, @@ -593,7 +597,7 @@ impl StateMachine { .sm() .sm_pinctrl .write(|w| unsafe { w.set_base().bits(pinnum).set_count().bits(1) }); - self.set_instruction( + self.exec_instruction( pio::InstructionOperands::SET { destination: pio::SetDestination::PINDIRS, data: if PinDir::Output == pin_dir { 1 } else { 0 }, @@ -680,7 +684,7 @@ impl StateMachine { // pause the state machine self.sm.set_enabled(false); // revert it to its wrap target - self.sm.set_instruction( + self.sm.exec_instruction( pio::InstructionOperands::JMP { condition: pio::JmpCondition::Always, address: self.program.wrap_target(), @@ -829,7 +833,7 @@ impl Tx { } .encode(); // Safety: The only other place this register is written is - // `UninitStatemachine.set_instruction`, `Tx` is only created after init. + // `UninitStatemachine.exec_instruction`, `Tx` is only created after init. let mask = 1 << SM::id(); while self.register_block().fstat.read().txempty().bits() & mask != mask { self.register_block().sm[SM::id()] @@ -1429,9 +1433,9 @@ impl PIOBuilder

{ sm.restart(); sm.reset_clock(); - // Set starting location by setting the state machine to execute a jmp + // Set starting location by forcing the state machine to execute a jmp // to the beginning of the program we loaded in. - sm.set_instruction( + sm.exec_instruction( pio::InstructionOperands::JMP { condition: pio::JmpCondition::Always, address: offset as u8,