Programmable IO¶
The RP2040 has hardware support for standard communication protocols like I2C, SPI and UART. For protocols where there is no hardware support, or where there is a requirement of custom I/O behaviour, Programmable Input Output (PIO) comes into play. Also, some MicroPython applications make use of a technique called bit banging in which pins are rapidly turned on and off to transmit data. This can make the entire process slow as the processor concentrates on bit banging rather than executing other logic. However, PIO allows bit banging to happen in the background while the CPU is executing the main work.
Along with the two central Cortex-M0+ processing cores, the RP2040 has two PIO blocks each of which has four independent state machines. These state machines can transfer data to/from other entities using First-In-First-Out (FIFO) buffers, which allow the state machine and main processor to work independently yet also synchronise their data. Each FIFO has four words (each of 32 bits) which can be linked to the DMA to transfer larger amounts of data.
All PIO instructions follow a common pattern:
<instruction> .side(<side_set_value>) [<delay_value>]
The side-set .side(...)
and delay [...]
parts are both optional, and if
specified allow the instruction to perform more than one operation. This keeps
PIO programs small and efficient.
There are nine instructions which perform the following tasks:
jmp()
transfers control to a different part of the codewait()
pauses until a particular action happensin_()
shifts the bits from a source (scratch register or set of pins) to the input shift registerout()
shifts the bits from the output shift register to a destinationpush()
sends data to the RX FIFOpull()
receives data from the TX FIFOmov()
moves data from a source to a destinationirq()
sets or clears an IRQ flagset()
writes a literal value to a destination
The instruction modifiers are:
.side()
sets the side-set pins at the start of the instruction[]
delays for a certain number of cycles after execution of the instruction
There are also directives:
wrap_target()
specifies where the program execution will get continued fromwrap()
specifies the instruction where the control flow of the program will get wrapped fromlabel()
sets a label for use withjmp()
instructionsword()
emits a raw 16-bit value which acts as an instruction in the program
An example¶
Take the pio_1hz.py
example for a simple understanding of how to use the PIO
and state machines. Below is the code for reference.
# Example using PIO to blink an LED and raise an IRQ at 1Hz.
import time
from machine import Pin
import rp2
@rp2.asm_pio(set_init=rp2.PIO.OUT_LOW)
def blink_1hz():
# Cycles: 1 + 1 + 6 + 32 * (30 + 1) = 1000
irq(rel(0))
set(pins, 1)
set(x, 31) [5]
label("delay_high")
nop() [29]
jmp(x_dec, "delay_high")
# Cycles: 1 + 1 + 6 + 32 * (30 + 1) = 1000
nop()
set(pins, 0)
set(x, 31) [5]
label("delay_low")
nop() [29]
jmp(x_dec, "delay_low")
# Create the StateMachine with the blink_1hz program, outputting on Pin(25).
sm = rp2.StateMachine(0, blink_1hz, freq=2000, set_base=Pin(25))
# Set the IRQ handler to print the millisecond timestamp.
sm.irq(lambda p: print(time.ticks_ms()))
# Start the StateMachine.
sm.active(1)
This creates an instance of class rp2.StateMachine
which runs the
blink_1hz
program at 2000Hz, and connects to pin 25. The blink_1hz
program uses the PIO to blink an LED connected to this pin at 1Hz, and also
raises an IRQ as the LED turns on. This IRQ then calls the lambda
function
which prints out a millisecond timestamp.
The blink_1hz
program is a PIO assembler routine. It connects to a single
pin which is configured as an output and starts out low. The instructions do
the following:
irq(rel(0))
raises the IRQ associated with the state machine.The LED is turned on via the
set(pins, 1)
instruction.The value 31 is put into register X, and then there is a delay for 5 more cycles, specified by the
[5]
.The
nop() [29]
instruction waits for 30 cycles.The
jmp(x_dec, "delay_high")
will keep looping to thedelay_high
label as long as the register X is non-zero, and will also post-decrement X. Since X starts with the value 31 this jump will happen 31 times, so thenop() [29]
runs 32 times in total (note there is also one instruction cycle taken by thejmp
for each of these 32 loops).The single
nop()
correlates with the cycle used for IRQ raise, and ensures the same number of cycles are used for LED on and LED off.set(pins, 0)
will turn the LED off by setting pin 25 low.Another 32 loops of
nop() [29]
andjmp(...)
will execute.Because
wrap_target()
andwrap()
are not specified, their default will be used and execution of the program will wrap around from the bottom to the top. This wrapping does not cost any execution cycles.
The entire routine takes exactly 2000 cycles of the state machine. Setting the frequency of the state machine to 2000Hz makes the LED blink at 1Hz.