Skip to content

Commit

Permalink
WS2801: Support for WS2801 LEDs using SW SPI and HW SPI
Browse files Browse the repository at this point in the history
Since none of the existing LED drivers worked properly with WS2801
LED modules this driver was added.
Tested on Arduino Uno
Will not pass all regression tests due to dict files being outdated on gh.

Signed-off-by: Anders Törnberg <[email protected]>
  • Loading branch information
akrto committed Feb 6, 2024
1 parent 9f41f53 commit be44501
Show file tree
Hide file tree
Showing 4 changed files with 240 additions and 1 deletion.
34 changes: 34 additions & 0 deletions docs/Config_Reference.md
Original file line number Diff line number Diff line change
Expand Up @@ -3020,6 +3020,40 @@ PCA9632 LED support. The PCA9632 is used on the FlashForge Dreamer.
# See the "led" section for information on these parameters.
```

### [ws2801]

ws2801 LED support (one may define any number of
sections with a "ws2801" prefix). See the
[command reference](G-Codes.md#led) for more information.

```
[ws2801 my_ws2801]
#cs_pin: <mcu-name>:None
# Required when setting up SPI
#spi_speed: 1000000
# The SPI speed (in hz) to use when communicating with the chip.
# The default is 1000000.
#spi_bus:
#spi_software_sclk_pin:
#spi_software_mosi_pin:
#spi_software_miso_pin:
# See the "common SPI settings" section for a description of the
# above parameters.
# spi_software_miso_pin must be set to a pin even though it is not
# used.
# cs_pin can be set to <mcu-name>:None, it is not in use.
#chain_count:
# See the "neopixel" section for information on this parameter.
#color_order: RGB
# Set the pixel order of the LED (using a string containing the
# letters R, G, B). The default is RGB.
#initial_RED: 0.0
#initial_GREEN: 0.0
#initial_BLUE: 0.0
# See the "led" section for information on these parameters.
```


## Additional servos, buttons, and other pins

### [servo]
Expand Down
107 changes: 107 additions & 0 deletions klippy/extras/ws2801.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,107 @@
# Support for "ws2801" leds
#
# Copyright (C) 2024 Anders Törnberg <[email protected]>
#
# This file may be distributed under the terms of the GNU GPLv3 license.
import logging
from . import bus

BACKGROUND_PRIORITY_CLOCK = 0x7fffffff00000000

MAX_MCU_SIZE = 500 # Sanity check on LED chain length

class WS2801:
def __init__(self, config):
self.printer = printer = config.get_printer()
self.mutex = printer.get_reactor().mutex()
# Configure ws2801
self.spi = bus.MCU_SPI_from_config(config, 0, default_speed=1000000)
self.mcu = self.spi.get_mcu()
self.oid = self.mcu.create_oid()
self.mcu.register_config_callback(self.build_config)
self.ws8201_update_cmd = self.ws8201_send_cmd = None
# Build color map
chain_count = config.getint('chain_count', 1, minval=1)
color_order = config.getlist("color_order", ["RGB"])
if len(color_order) == 1:
color_order = [color_order[0]] * chain_count
if len(color_order) != chain_count:
raise config.error("color_order does not match chain_count")
color_indexes = []
for lidx, co in enumerate(color_order):
if sorted(co) not in (sorted("RGB"), sorted("RGB")):
raise config.error("Invalid color_order '%s'" % (co,))
color_indexes.extend([(lidx, "RGB".index(c)) for c in co])
self.color_map = list(enumerate(color_indexes))
if len(self.color_map) > MAX_MCU_SIZE:
raise config.error("WS2801 chain too long")
# Initialize color data
pled = printer.load_object(config, "led")
self.led_helper = pled.setup_helper(config, self.update_leds,
chain_count)
self.color_data = bytearray(len(self.color_map))
self.update_color_data(self.led_helper.get_status()['color_data'])
self.old_color_data = bytearray([d ^ 1 for d in self.color_data])
# Register callbacks
printer.register_event_handler("klippy:connect", self.send_data)
def build_config(self):
print(self.spi.get_oid())
self.mcu.add_config_cmd("config_ws2801 oid=%d spi_oid=%d data_size=%d"
% (self.oid, self.spi.get_oid(), len(self.color_data)))
cmd_queue = self.mcu.alloc_command_queue()
self.ws2801_update_cmd = self.mcu.lookup_command(
"ws2801_update oid=%c pos=%hu data=%*s", cq=cmd_queue)
self.ws2801_send_cmd = self.mcu.lookup_query_command(
"ws2801_send oid=%c", "ws2801_result oid=%c success=%c",
oid=self.oid, cq=cmd_queue)
def update_color_data(self, led_state):
color_data = self.color_data
for cdidx, (lidx, cidx) in self.color_map:
color_data[cdidx] = int(led_state[lidx][cidx] * 255. + .5)
def send_data(self, print_time=None):
old_data, new_data = self.old_color_data, self.color_data
if new_data == old_data:
return
# Find the position of all changed bytes in this framebuffer
diffs = [[i, 1] for i, (n, o) in enumerate(zip(new_data, old_data))
if n != o]
# Batch together changes that are close to each other
for i in range(len(diffs)-2, -1, -1):
pos, count = diffs[i]
nextpos, nextcount = diffs[i+1]
if pos + 5 >= nextpos and nextcount < 16:
diffs[i][1] = nextcount + (nextpos - pos)
del diffs[i+1]
# Transmit changes
for pos, count in diffs:
print(new_data[pos:pos+count])
ucmd = self.ws2801_update_cmd.send
for pos, count in diffs:
ucmd([self.oid, pos, new_data[pos:pos+count]],
reqclock=BACKGROUND_PRIORITY_CLOCK)
old_data[:] = new_data
#Instruct mcu to update the LEDs
minclock = 0
if print_time is not None:
minclock = self.mcu.print_time_to_clock(print_time)
scmd = self.ws2801_send_cmd.send
if self.printer.get_start_args().get('debugoutput') is not None:
return
for i in range(8):
params = scmd([self.oid], minclock=minclock,
reqclock=BACKGROUND_PRIORITY_CLOCK)
if params['success']:
break
else:
logging.info("WS2801 update did not succeed")
def update_leds(self, led_state, print_time):
def reactor_bgfunc(eventtime):
with self.mutex:
self.update_color_data(led_state)
self.send_data(print_time)
self.printer.get_reactor().register_callback(reactor_bgfunc)
def get_status(self, eventtime=None):
return self.led_helper.get_status(eventtime)

def load_config_prefix(config):
return WS2801(config)
2 changes: 1 addition & 1 deletion src/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ src-y += sched.c command.c basecmd.c debugcmds.c
src-$(CONFIG_HAVE_GPIO) += initial_pins.c gpiocmds.c stepper.c endstop.c \
trsync.c
src-$(CONFIG_HAVE_GPIO_ADC) += adccmds.c
src-$(CONFIG_HAVE_GPIO_SPI) += spicmds.c
src-$(CONFIG_HAVE_GPIO_SPI) += spicmds.c ws2801.c
src-$(CONFIG_HAVE_GPIO_SDIO) += sdiocmds.c
src-$(CONFIG_HAVE_GPIO_I2C) += i2ccmds.c
src-$(CONFIG_HAVE_GPIO_HARD_PWM) += pwmcmds.c
Expand Down
98 changes: 98 additions & 0 deletions src/ws2801.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,98 @@
// Support for SW and HW SPI commands to WS2812 type LEDs
//
// Copyright (C) 2024 Anders Törnberg <[email protected]>
//
// This file may be distributed under the terms of the GNU GPLv3 license.

#include <string.h> // memcpy
#include "autoconf.h" // CONFIG_MACH_AVR
#include "board/gpio.h" // gpio_out_write
#include "board/irq.h" // irq_poll
#include "board/misc.h" // timer_read_time
#include "basecmd.h" // oid_alloc
#include "command.h" // DECL_COMMAND
#include "sched.h" // sched_shutdown
#include "spicmds.h" // spidev_se

/****************************************************************
* Timing
****************************************************************/

typedef unsigned int ws2801_time_t;

static ws2801_time_t
nsecs_to_ticks(uint32_t ns)
{
return timer_from_us(ns * 1000) / 1000000;
}

#define MIN_TICKS_BETWEEN_REQUESTS nsecs_to_ticks(500000)

/****************************************************************
* WS2801 interface
****************************************************************/

struct ws2801_s {
struct spidev_s *spi;
uint32_t last_req_time;
uint16_t data_size;
uint8_t data[0];
};

void
command_config_ws2801(uint32_t *args)
{
uint16_t data_size = args[2];
if (data_size & 0x8000)
shutdown("Invalid ws2801 data_size");
struct ws2801_s *ws = oid_alloc(args[0], command_config_ws2801
, sizeof(*ws) + data_size);
ws->spi = spidev_oid_lookup(args[1]);
ws->data_size = data_size;
}
DECL_COMMAND(command_config_ws2801, "config_ws2801 oid=%c spi_oid=%c"
" data_size=%hu");

static int
send_data(struct ws2801_s *ws)
{
// Make sure the reset time has elapsed since last request
uint32_t last_req_time = ws->last_req_time, \
rmt = MIN_TICKS_BETWEEN_REQUESTS;
uint32_t cur = timer_read_time();
while (cur - last_req_time < rmt) {
irq_poll();
cur = timer_read_time();
}

// Transmit data
uint8_t *data = ws->data;
uint_fast16_t data_len = ws->data_size;
spidev_transfer(ws->spi, 0, data_len, data);
return 0;
}

void
command_ws2801_update(uint32_t *args)
{
uint8_t oid = args[0];
struct ws2801_s *ws = oid_lookup(oid, command_config_ws2801);
uint_fast16_t pos = args[1];
uint_fast8_t data_len = args[2];
uint8_t *data = command_decode_ptr(args[3]);
if (pos & 0x8000 || pos + data_len > ws->data_size)
shutdown("Invalid ws2801 update command");
memcpy(&ws->data[pos], data, data_len);
}
DECL_COMMAND(command_ws2801_update,
"ws2801_update oid=%c pos=%hu data=%*s");

void
command_ws2801_send(uint32_t *args)
{
uint8_t oid = args[0];
struct ws2801_s *ws = oid_lookup(oid, command_config_ws2801);
int ret = send_data(ws);
sendf("ws2801_result oid=%c success=%c", oid, ret ? 0 : 1);
}
DECL_COMMAND(command_ws2801_send, "ws2801_send oid=%c");

0 comments on commit be44501

Please sign in to comment.