From 3335674e94c3f0f754997babf7b5473164beb47f Mon Sep 17 00:00:00 2001 From: Jimmy Date: Sun, 24 Aug 2025 19:50:04 +1200 Subject: [PATCH] Initial commit --- src/dma.py | 117 ++++++++++++++++++++++++++++++++++++++++++++++++++ src/main.py | 46 ++++++++++++++++++++ src/ws2812.py | 68 +++++++++++++++++++++++++++++ 3 files changed, 231 insertions(+) create mode 100644 src/dma.py create mode 100644 src/main.py create mode 100644 src/ws2812.py diff --git a/src/dma.py b/src/dma.py new file mode 100644 index 0000000..04abead --- /dev/null +++ b/src/dma.py @@ -0,0 +1,117 @@ +from machine import Pin +from rp2 import PIO, StateMachine, asm_pio +from time import sleep +import array +import uctypes +from uctypes import BF_POS, BF_LEN, UINT32, BFUINT32, struct + +PIO0_BASE = 0x50200000 +PIO1_BASE = 0x50300000 +DMA_BASE = 0x50000000 +DMA_CHAN_WIDTH = 0x40 +DMA_CHAN_COUNT = 12 + +DMA_SIZE_BYTE = 0x0 +DMA_SIZE_HALFWORD = 0x1 +DMA_SIZE_WORD = 0x2 + +# DMA: RP2040 datasheet 2.5.7 +DMA_CTRL_TRIG_FIELDS = { + "AHB_ERROR": 31<= 0 and sm_num < 4): + self.dma_chan.WRITE_ADDR_REG = PIO0_BASE + 0x10 + sm_num *4 + self.dma_chan.CTRL_TRIG.TREQ_SEL = sm_num + elif (sm_num < 8): + self.dma_chan.WRITE_ADDR_REG = PIO1_BASE + 0x10 + (sm_num-4) *4 + self.dma_chan.CTRL_TRIG.TREQ_SEL = sm_num + 4 + + 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 + + + + + + + + + + diff --git a/src/main.py b/src/main.py new file mode 100644 index 0000000..7dd9b36 --- /dev/null +++ b/src/main.py @@ -0,0 +1,46 @@ +from usocket import socket +from machine import Pin,SPI +import network +import time +from ws2812 import WS2812B + + +pins = [18,19,20,21,22,26,27,28] + +led = Pin(25, Pin.OUT) + +#W5x00 chip init +def w5x00_init(): + spi=SPI(1,2_000_000, mosi=Pin(11),miso=Pin(12),sck=Pin(10)) + nic = network.WIZNET5K(spi,Pin(13),Pin(9)) #spi,cs,reset pin + nic.active(True) + + #None DHCP + nic.ifconfig(('10.1.1.20','255.255.255.0','10.1.1.1','8.8.8.8')) + + #DHCP + #nic.ifconfig('dhcp') + print('IP address :', nic.ifconfig()) + + while not nic.isconnected(): + time.sleep(1) + print(nic.regs()) + + print('IP address :', nic.ifconfig()) + +w5x00_init() + + +strips = [] +num_leds = 200 +for i, pin in enumerate(pins): + print(i) + strips.append(WS2812B(num_leds, pin, i, 1)) + strips[i].fill((0,0,0)) + strips[i].show() + time.sleep(0.5) + strips[i].fill((4,4,3)) + strips[i].show() + time.sleep(0.5) + + \ No newline at end of file diff --git a/src/ws2812.py b/src/ws2812.py new file mode 100644 index 0000000..7aa5b58 --- /dev/null +++ b/src/ws2812.py @@ -0,0 +1,68 @@ + +import array, time +from machine import Pin +import rp2 +from time import sleep +import dma + +@rp2.asm_pio(sideset_init=rp2.PIO.OUT_LOW, out_shiftdir=rp2.PIO.SHIFT_LEFT, autopull=True, pull_thresh=8) +def ws2812(): + T1 = 2 + T2 = 5 + T3 = 3 + wrap_target() + label("bitloop") + out(x, 1) .side(0) [T3 - 1] + jmp(not_x, "do_zero") .side(1) [T1 - 1] + jmp("bitloop") .side(1) [T2 - 1] + label("do_zero") + nop() .side(0) [T2 - 1] + wrap() + +class WS2812B: + def __init__(self, num_leds, pin, state_machine, brightness=0.1, invert=False): + self.sm = rp2.StateMachine(state_machine, ws2812, freq=8_000_000, sideset_base=Pin(pin)) + self.sm.active(1) + self.ar = bytearray(num_leds*3) + self.num_leds = num_leds + self.brightness = brightness + self.invert = invert + self.pio_dma = dma.PIO_DMA_Transfer(state_machine+4, state_machine, 8, num_leds*3) + + def show(self): + self.pio_dma.start_transfer(self.ar) + + def set(self, i, color): + self.ar[i*3] = int(color[1]*self.brightness) + self.ar[i*3+1] = int(color[0]*self.brightness) + self.ar[i*3+2] = int(color[2]*self.brightness) + + def fill(self, color): + for i in range(self.num_leds): + self.set(i, color) + + def busy(self): + return self.pio_dma.busy() + + BLACK = (0, 0, 0) + RED = (255, 0, 0) + YELLOW = (255, 150, 0) + GREEN = (0, 255, 0) + CYAN = (0, 255, 255) + BLUE = (0, 0, 255) + PURPLE = (180, 0, 255) + WHITE = (255, 255, 255) + COLORS = (BLACK, RED, YELLOW, GREEN, CYAN, BLUE, PURPLE, WHITE) + +if __name__ == "__main__": + num_leds, pin, sm, brightness = 10, 28, 0, 1 + ws0 = WS2812B(num_leds, pin, sm, brightness) + while True: + for color in ws0.COLORS: + ws0.fill(color) + ws0.show() + time.sleep(1) + + + +