Update dma.py
This commit is contained in:
parent
bd703c33e3
commit
1dde87f58d
262
dma.py
262
dma.py
|
@ -1,63 +1,219 @@
|
||||||
|
from machine import Pin
|
||||||
|
from rp2 import PIO, StateMachine, asm_pio
|
||||||
|
from time import sleep
|
||||||
|
import array
|
||||||
import uctypes
|
import uctypes
|
||||||
from uctypes import BF_POS, BF_LEN, BFUINT32
|
from uctypes import BF_POS, BF_LEN, UINT32, BFUINT32, struct
|
||||||
|
|
||||||
DMA_CTRL_REG = {
|
GPIO_BASE = 0x40014000
|
||||||
"AHB_ERROR": 31 << BF_POS | 1 << BF_LEN | BFUINT32,
|
GPIO_CHAN_WIDTH = 0x08
|
||||||
"READ_ERR": 30 << BF_POS | 1 << BF_LEN | BFUINT32,
|
GPIO_PIN_COUNT = 30
|
||||||
"WRITE_ERR": 29 << BF_POS | 1 << BF_LEN | BFUINT32,
|
PAD_BASE = 0x4001c000
|
||||||
"Reserved": 25 << BF_POS | 4 << BF_LEN | BFUINT32,
|
PAD_PIN_WIDTH = 0x04
|
||||||
"BUSY": 24 << BF_POS | 1 << BF_LEN | BFUINT32,
|
ADC_BASE = 0x4004c000
|
||||||
"SNIFF_EN": 23 << BF_POS | 1 << BF_LEN | BFUINT32,
|
PIO0_BASE = 0x50200000
|
||||||
"BSWAP": 22 << BF_POS | 1 << BF_LEN | BFUINT32,
|
PIO1_BASE = 0x50300000
|
||||||
"IRQ_QUIET": 21 << BF_POS | 1 << BF_LEN | BFUINT32,
|
DMA_BASE = 0x50000000
|
||||||
"TREQ_SEL": 15 << BF_POS | 6 << BF_LEN | BFUINT32,
|
DMA_CHAN_WIDTH = 0x40
|
||||||
"CHAIN_TO": 11 << BF_POS | 4 << BF_LEN | BFUINT32,
|
DMA_CHAN_COUNT = 12
|
||||||
"RING_SEL": 10 << BF_POS | 1 << BF_LEN | BFUINT32,
|
|
||||||
"RING_SIZE": 6 << BF_POS | 4 << BF_LEN | BFUINT32,
|
DMA_SIZE_BYTE = 0x0
|
||||||
"INCR_WRITE": 5 << BF_POS | 1 << BF_LEN | BFUINT32,
|
DMA_SIZE_HALFWORD = 0x1
|
||||||
"INCR_READ": 4 << BF_POS | 1 << BF_LEN | BFUINT32,
|
DMA_SIZE_WORD = 0x2
|
||||||
"DATA_SIZE": 2 << BF_POS | 2 << BF_LEN | BFUINT32,
|
|
||||||
"HIGH_PRIO": 1 << BF_POS | 1 << BF_LEN | BFUINT32,
|
# DMA: RP2040 datasheet 2.5.7
|
||||||
"EN": 0 << BF_POS | 1 << BF_LEN | BFUINT32,
|
DMA_CTRL_TRIG_FIELDS = {
|
||||||
|
"AHB_ERROR": 31<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"READ_ERROR": 30<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"WRITE_ERROR": 29<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"BUSY": 24<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"SNIFF_EN": 23<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"BSWAP": 22<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"IRQ_QUIET": 21<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"TREQ_SEL": 15<<BF_POS | 6<<BF_LEN | BFUINT32,
|
||||||
|
"CHAIN_TO": 11<<BF_POS | 4<<BF_LEN | BFUINT32,
|
||||||
|
"RING_SEL": 10<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"RING_SIZE": 6<<BF_POS | 4<<BF_LEN | BFUINT32,
|
||||||
|
"INCR_WRITE": 5<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"INCR_READ": 4<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"DATA_SIZE": 2<<BF_POS | 2<<BF_LEN | BFUINT32,
|
||||||
|
"HIGH_PRIORITY":1<<BF_POS | 1<<BF_LEN | BFUINT32,
|
||||||
|
"EN": 0<<BF_POS | 1<<BF_LEN | BFUINT32
|
||||||
|
}
|
||||||
|
# Channel-specific DMA registers
|
||||||
|
DMA_CHAN_REGS = {
|
||||||
|
"READ_ADDR_REG": 0x00|UINT32,
|
||||||
|
"WRITE_ADDR_REG": 0x04|UINT32,
|
||||||
|
"TRANS_COUNT_REG": 0x08|UINT32,
|
||||||
|
"CTRL_TRIG_REG": 0x0c|UINT32,
|
||||||
|
"CTRL_TRIG": (0x0c,DMA_CTRL_TRIG_FIELDS)
|
||||||
}
|
}
|
||||||
|
|
||||||
DMA_LAYOUT = {
|
# General DMA registers
|
||||||
"READ_ADDR": 0 | uctypes.UINT32,
|
DMA_REGS = {
|
||||||
"WRITE_ADDR": 4 | uctypes.UINT32,
|
"INTR": 0x400|UINT32,
|
||||||
"TRANS_COUNT": 8 | uctypes.UINT32,
|
"INTE0": 0x404|UINT32,
|
||||||
"CTRL_TRIG": (12, DMA_CTRL_REG),
|
"INTF0": 0x408|UINT32,
|
||||||
"CTRL_TRIG_RAW": 12 | uctypes.UINT32, # for single update of all fields
|
"INTS0": 0x40c|UINT32,
|
||||||
"AL1_CTRL": 16 | uctypes.UINT32,
|
"INTE1": 0x414|UINT32,
|
||||||
|
"INTF1": 0x418|UINT32,
|
||||||
|
"INTS1": 0x41c|UINT32,
|
||||||
|
"TIMER0": 0x420|UINT32,
|
||||||
|
"TIMER1": 0x424|UINT32,
|
||||||
|
"TIMER2": 0x428|UINT32,
|
||||||
|
"TIMER3": 0x42c|UINT32,
|
||||||
|
"MULTI_CHAN_TRIGGER": 0x430|UINT32,
|
||||||
|
"SNIFF_CTRL": 0x434|UINT32,
|
||||||
|
"SNIFF_DATA": 0x438|UINT32,
|
||||||
|
"FIFO_LEVELS": 0x440|UINT32,
|
||||||
|
"CHAN_ABORT": 0x444|UINT32
|
||||||
}
|
}
|
||||||
|
|
||||||
# create the DMA channel structs (0-11)
|
DREQ_PIO0_TX0, DREQ_PIO0_RX0, DREQ_PIO1_TX0 = 0, 4, 8
|
||||||
CHANNELS = [uctypes.struct(0x50000000 + i * 0x40, DMA_LAYOUT) for i in range(0,12)]
|
DREQ_PIO1_RX0, DREQ_SPI0_TX, DREQ_SPI0_RX = 12, 16, 17
|
||||||
|
DREQ_SPI1_TX, DREQ_SPI1_RX, DREQ_UART0_TX = 18, 19, 20
|
||||||
|
DREQ_UART0_RX, DREQ_UART1_TX, DREQ_UART1_RX = 21, 22, 23
|
||||||
|
DREQ_I2C0_TX, DREQ_I2C0_RX, DREQ_I2C1_TX = 32, 33, 34
|
||||||
|
DREQ_I2C1_RX, DREQ_ADC = 35, 36
|
||||||
|
|
||||||
# init dma channels to some default values
|
DREQ_PIO0_TX0 , DREQ_PIO0_TX1, DREQ_PIO0_TX2, DREQ_PIO0_TX3 = 0, 1, 2, 3
|
||||||
def init_channels():
|
DREQ_PIO1_TX0 , DREQ_PIO1_TX1, DREQ_PIO1_TX2, DREQ_PIO1_TX3 = 8, 9, 10, 11
|
||||||
for i, ch in enumerate(CHANNELS):
|
|
||||||
# no wraparound, sniff=0, swap_byte=0, irq_quiet=1
|
|
||||||
# unpaced transfers (=0x3f). high prio=0, data_size=word, incr_r/w = true
|
|
||||||
ch.CTRL_TRIG_RAW = 0x3f8030
|
|
||||||
# set chain to itself, to disable chaining
|
|
||||||
ch.CTRL_TRIG.CHAIN_TO = i
|
|
||||||
|
|
||||||
# example function to show mem->mem transfer
|
DMA_CHANS = [struct(DMA_BASE + n*DMA_CHAN_WIDTH, DMA_CHAN_REGS) for n in range(0,DMA_CHAN_COUNT)]
|
||||||
# dest, src can be bytearrays, size in bytes
|
DMA_DEVICE = struct(DMA_BASE, DMA_REGS)
|
||||||
def memcopy(ch, dest, src, size, enable=1):
|
|
||||||
ch.CTRL_TRIG.EN = 0
|
PIO0_TX0 = PIO0_BASE + 0x010
|
||||||
ch.WRITE_ADDR = uctypes.addressof(dest)
|
PIO0_TX1 = PIO0_BASE + 0x014
|
||||||
ch.READ_ADDR = uctypes.addressof(src)
|
PIO0_TX2 = PIO0_BASE + 0x018
|
||||||
ch.TRANS_COUNT = size // (1 << ch.CTRL_TRIG.DATA_SIZE)
|
PIO0_TX3 = PIO0_BASE + 0x01c
|
||||||
ch.CTRL_TRIG.EN = enable
|
PIO1_TX0 = PIO1_BASE + 0x010
|
||||||
|
PIO1_TX1 = PIO1_BASE + 0x014
|
||||||
|
PIO1_TX2 = PIO1_BASE + 0x018
|
||||||
|
PIO1_TX3 = PIO1_BASE + 0x01c
|
||||||
|
|
||||||
|
GPIO_FUNC_SPI, GPIO_FUNC_UART, GPIO_FUNC_I2C = 1, 2, 3
|
||||||
|
GPIO_FUNC_PWM, GPIO_FUNC_SIO, GPIO_FUNC_PIO0 = 4, 5, 6
|
||||||
|
GPIO_FUNC_NULL = 0x1f
|
||||||
|
|
||||||
|
DMA_CH0_AL3_TRANS_COUNT = DMA_BASE + 0x38
|
||||||
|
|
||||||
|
|
||||||
# can be used to temporary construct the necessary DMA CTRL values
|
class PIO_DMA_Transfer():
|
||||||
# then copy over to actual dma channel's CTRL value in one write
|
def __init__(self, dma_channel, sm_num, block_size, transfer_count):
|
||||||
# e.g.
|
self.dma_chan = DMA_CHANS[dma_channel]
|
||||||
# scratch.CHAIN_TO=2;
|
self.channel_number = dma_channel
|
||||||
# scratch.DATA_SIZE=2;
|
|
||||||
# ...
|
if (sm_num == 0):
|
||||||
# CH0.CTRL_TRIG_RAW=scratch.CTRL_TRIG_RAW
|
self.dma_chan.WRITE_ADDR_REG = PIO0_TX0
|
||||||
tmp = bytearray(uctypes.sizeof(DMA_LAYOUT))
|
self.dma_chan.CTRL_TRIG.TREQ_SEL = DREQ_PIO0_TX0
|
||||||
scratch_ctrl = uctypes.struct(uctypes.addressof(tmp), DMA_CTRL_REG)
|
elif (sm_num == 1):
|
||||||
|
self.dma_chan.WRITE_ADDR_REG = PIO0_TX1
|
||||||
|
self.dma_chan.CTRL_TRIG.TREQ_SEL = DREQ_PIO0_TX1
|
||||||
|
elif (sm_num == 2):
|
||||||
|
self.dma_chan.WRITE_ADDR_REG = PIO0_TX2
|
||||||
|
self.dma_chan.CTRL_TRIG.TREQ_SEL = DREQ_PIO0_TX2
|
||||||
|
elif (sm_num == 3):
|
||||||
|
self.dma_chan.WRITE_ADDR_REG = PIO0_TX3
|
||||||
|
self.dma_chan.CTRL_TRIG.TREQ_SEL = DREQ_PIO0_TX3
|
||||||
|
elif (sm_num == 4):
|
||||||
|
self.dma_chan.WRITE_ADDR_REG = PIO1_TX0
|
||||||
|
self.dma_chan.CTRL_TRIG.TREQ_SEL = DREQ_PIO1_TX0
|
||||||
|
elif (sm_num == 5):
|
||||||
|
self.dma_chan.WRITE_ADDR_REG = PIO1_TX1
|
||||||
|
self.dma_chan.CTRL_TRIG.TREQ_SEL = DREQ_PIO1_TX1
|
||||||
|
elif (sm_num == 6):
|
||||||
|
self.dma_chan.WRITE_ADDR_REG = PIO1_TX2
|
||||||
|
self.dma_chan.CTRL_TRIG.TREQ_SEL = DREQ_PIO1_TX2
|
||||||
|
elif (sm_num == 7):
|
||||||
|
self.dma_chan.WRITE_ADDR_REG = PIO1_TX3
|
||||||
|
self.dma_chan.CTRL_TRIG.TREQ_SEL = DREQ_PIO1_TX3
|
||||||
|
|
||||||
|
if (block_size == 8):
|
||||||
|
self.dma_chan.CTRL_TRIG.DATA_SIZE = DMA_SIZE_BYTE
|
||||||
|
if (block_size == 16):
|
||||||
|
self.dma_chan.CTRL_TRIG.DATA_SIZE = DMA_SIZE_HALFWORD
|
||||||
|
if (block_size == 32):
|
||||||
|
self.dma_chan.CTRL_TRIG.DATA_SIZE = DMA_SIZE_WORD
|
||||||
|
|
||||||
|
self.dma_chan.TRANS_COUNT_REG = transfer_count
|
||||||
|
|
||||||
|
#Do I just always want these?
|
||||||
|
self.dma_chan.CTRL_TRIG.INCR_WRITE = 0
|
||||||
|
self.dma_chan.CTRL_TRIG.INCR_READ = 1
|
||||||
|
|
||||||
|
def start_transfer(self, buffer):
|
||||||
|
self.dma_chan.READ_ADDR_REG = uctypes.addressof(buffer)
|
||||||
|
self.dma_chan.CTRL_TRIG.EN = 1
|
||||||
|
|
||||||
|
def transfer_count(self):
|
||||||
|
return self.dma_chan.TRANS_COUNT_REG
|
||||||
|
|
||||||
|
def busy(self):
|
||||||
|
if self.dma_chan.CTRL_TRIG.DATA_SIZE == 1:
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
|
def abort_transfer(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def chain_to(self, channel):
|
||||||
|
self.dma_chan.CTRL_TRIG.CHAIN_TO = channel
|
||||||
|
|
||||||
|
def get_number(self):
|
||||||
|
return self.channel_number
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#looping transfers
|
||||||
|
#note -- see datasheet 2.5.7
|
||||||
|
#location of registers is -- AL3 transcount / read address trigger.
|
||||||
|
#Writing to these (from one DMA channel) will re-trigger a second DMA channel
|
||||||
|
#need to set write ring.
|
||||||
|
#could also set read ring?
|
||||||
|
#out_buff = array.array('L', ((x if (x<1000) else (2000-x)) for x in range(NSAMPLES)))
|
||||||
|
|
||||||
|
class DMA_Control_Block:
|
||||||
|
def __init__(self, this_chan, that_chan, read_address, transfer_count, loops):
|
||||||
|
self.dma_chan = DMA_CHANS[this_chan]
|
||||||
|
|
||||||
|
#note -- need to set this up to get the right location
|
||||||
|
#but for now just always control channel 0
|
||||||
|
self.dma_chan.WRITE_ADDR_REG = DMA_CH0_AL3_TRANS_COUNT
|
||||||
|
self.dma_chan.CTRL_TRIG.DATA_SIZE = DMA_SIZE_WORD
|
||||||
|
self.dma_chan.TRANS_COUNT_REG = 2 # two transfers. One is the count, one is the read_address.
|
||||||
|
#Then pauses until the other channel chains back to this.
|
||||||
|
|
||||||
|
self.buffer = array.array('L', (x for x in range(2*loops)))
|
||||||
|
for x in range(loops):
|
||||||
|
self.buffer[2*x] = transfer_count
|
||||||
|
self.buffer[2*x+1] = read_address
|
||||||
|
|
||||||
|
self.start_address = uctypes.addressof(self.buffer)
|
||||||
|
#set up read ring
|
||||||
|
that_chan.chain_to(this_chan)
|
||||||
|
|
||||||
|
self.dma_chan.CTRL_TRIG.INCR_WRITE = 1
|
||||||
|
self.dma_chan.CTRL_TRIG.INCR_READ = 1
|
||||||
|
|
||||||
|
self.dma_chan.CTRL_TRIG.RING_SEL = 1
|
||||||
|
self.dma_chan.CTRL_TRIG.RING_SIZE = 3 # 1u<<3 bytes / 8 bytes
|
||||||
|
self.dma_chan.CTRL_TRIG.TREQ_SEL = 0x3f # unpaced transfer
|
||||||
|
|
||||||
|
def start_chain(self):
|
||||||
|
self.dma_chan.READ_ADDR_REG = self.start_address
|
||||||
|
self.dma_chan.CTRL_TRIG.EN = 1
|
||||||
|
|
||||||
|
def transfer_count(self):
|
||||||
|
return self.dma_chan.TRANS_COUNT_REG
|
||||||
|
|
||||||
|
def get_read_address(self):
|
||||||
|
return self.dma_chan.READ_ADDR_REG
|
||||||
|
|
||||||
|
def busy(self):
|
||||||
|
if self.dma_chan.CTRL_TRIG.DATA_SIZE == 1:
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue