2017-10-17 17:20:33 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2017 Linaro Limited
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
2019-06-25 21:53:56 +02:00
|
|
|
#include <drivers/led_strip.h>
|
2017-10-17 17:20:33 +02:00
|
|
|
|
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2020-05-11 20:56:08 +02:00
|
|
|
#if DT_NODE_HAS_STATUS(DT_INST(0, colorway_lpd8806), okay)
|
2020-03-25 18:01:50 +01:00
|
|
|
#define DT_DRV_COMPAT colorway_lpd8806
|
2019-01-26 03:47:27 +01:00
|
|
|
#else
|
2020-03-25 18:01:50 +01:00
|
|
|
#define DT_DRV_COMPAT colorway_lpd8803
|
2019-01-26 03:47:27 +01:00
|
|
|
#endif
|
|
|
|
|
2018-10-09 00:23:11 +02:00
|
|
|
#define LOG_LEVEL CONFIG_LED_STRIP_LOG_LEVEL
|
|
|
|
#include <logging/log.h>
|
|
|
|
LOG_MODULE_REGISTER(lpd880x);
|
2017-10-17 17:20:33 +02:00
|
|
|
|
|
|
|
#include <zephyr.h>
|
|
|
|
#include <device.h>
|
2019-06-25 21:54:01 +02:00
|
|
|
#include <drivers/spi.h>
|
2019-06-26 16:33:55 +02:00
|
|
|
#include <sys/util.h>
|
2017-10-17 17:20:33 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* LPD880X SPI master configuration:
|
|
|
|
*
|
|
|
|
* - mode 0 (the default), 8 bit, MSB first, one-line SPI
|
|
|
|
* - no shenanigans (no CS hold, release device lock, not an EEPROM)
|
|
|
|
*/
|
|
|
|
#define LPD880X_SPI_OPERATION (SPI_OP_MODE_MASTER | \
|
|
|
|
SPI_TRANSFER_MSB | \
|
|
|
|
SPI_WORD_SET(8) | \
|
|
|
|
SPI_LINES_SINGLE)
|
|
|
|
|
|
|
|
struct lpd880x_data {
|
2018-01-30 13:49:01 +01:00
|
|
|
struct device *spi;
|
2017-10-17 17:20:33 +02:00
|
|
|
struct spi_config config;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int lpd880x_update(struct device *dev, void *data, size_t size)
|
|
|
|
{
|
|
|
|
struct lpd880x_data *drv_data = dev->driver_data;
|
|
|
|
/*
|
|
|
|
* Per the AdaFruit reverse engineering notes on the protocol,
|
|
|
|
* a zero byte propagates through at most 32 LED driver ICs.
|
|
|
|
* The LPD8803 is the worst case, at 3 output channels per IC.
|
|
|
|
*/
|
|
|
|
u8_t reset_size = ceiling_fraction(ceiling_fraction(size, 3), 32);
|
|
|
|
u8_t reset_buf[reset_size];
|
|
|
|
u8_t last = 0x00;
|
2018-01-30 13:49:01 +01:00
|
|
|
const struct spi_buf bufs[3] = {
|
|
|
|
{
|
|
|
|
/* Prepares the strip to shift in new data values. */
|
|
|
|
.buf = reset_buf,
|
|
|
|
.len = reset_size
|
|
|
|
},
|
|
|
|
{
|
|
|
|
/* Displays the serialized pixel data. */
|
|
|
|
.buf = data,
|
|
|
|
.len = size
|
|
|
|
},
|
|
|
|
{
|
|
|
|
/* Ensures the last byte of pixel data is displayed. */
|
|
|
|
.buf = &last,
|
|
|
|
.len = sizeof(last)
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
const struct spi_buf_set tx = {
|
|
|
|
.buffers = bufs,
|
|
|
|
.count = 3
|
|
|
|
};
|
2017-10-17 17:20:33 +02:00
|
|
|
size_t rc;
|
|
|
|
|
2018-09-12 04:09:03 +02:00
|
|
|
(void)memset(reset_buf, 0x00, reset_size);
|
2017-10-17 17:20:33 +02:00
|
|
|
|
2018-01-30 13:49:01 +01:00
|
|
|
rc = spi_write(drv_data->spi, &drv_data->config, &tx);
|
2017-10-17 17:20:33 +02:00
|
|
|
if (rc) {
|
2018-10-09 00:23:11 +02:00
|
|
|
LOG_ERR("can't update strip: %d", rc);
|
2017-10-17 17:20:33 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int lpd880x_strip_update_rgb(struct device *dev,
|
|
|
|
struct led_rgb *pixels,
|
|
|
|
size_t num_pixels)
|
|
|
|
{
|
|
|
|
u8_t *px = (u8_t *)pixels;
|
|
|
|
u8_t r, g, b;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Overwrite a prefix of the pixels array with its on-wire
|
|
|
|
* representation, eliminating padding/scratch garbage, if any.
|
|
|
|
*/
|
|
|
|
for (i = 0; i < num_pixels; i++) {
|
|
|
|
r = 0x80 | (pixels[i].r >> 1);
|
|
|
|
g = 0x80 | (pixels[i].g >> 1);
|
|
|
|
b = 0x80 | (pixels[i].b >> 1);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* GRB is the ordering used by commonly available
|
|
|
|
* LPD880x strips.
|
|
|
|
*/
|
|
|
|
*px++ = g;
|
|
|
|
*px++ = r;
|
|
|
|
*px++ = b;
|
|
|
|
}
|
|
|
|
|
|
|
|
return lpd880x_update(dev, pixels, 3 * num_pixels);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int lpd880x_strip_update_channels(struct device *dev, u8_t *channels,
|
|
|
|
size_t num_channels)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < num_channels; i++) {
|
|
|
|
channels[i] = 0x80 | (channels[i] >> 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
return lpd880x_update(dev, channels, num_channels);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int lpd880x_strip_init(struct device *dev)
|
|
|
|
{
|
|
|
|
struct lpd880x_data *data = dev->driver_data;
|
|
|
|
struct spi_config *config = &data->config;
|
|
|
|
|
2020-03-25 18:01:50 +01:00
|
|
|
data->spi = device_get_binding(DT_INST_BUS_LABEL(0));
|
2018-01-30 13:49:01 +01:00
|
|
|
if (!data->spi) {
|
2018-10-09 00:23:11 +02:00
|
|
|
LOG_ERR("SPI device %s not found",
|
2020-03-25 18:01:50 +01:00
|
|
|
DT_INST_BUS_LABEL(0));
|
2017-10-17 17:20:33 +02:00
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
2020-03-25 18:01:50 +01:00
|
|
|
config->frequency = DT_INST_PROP(0, spi_max_frequency);
|
2017-10-17 17:20:33 +02:00
|
|
|
config->operation = LPD880X_SPI_OPERATION;
|
2020-03-25 18:01:50 +01:00
|
|
|
config->slave = DT_INST_REG_ADDR(0); /* MOSI/CLK only; CS is not supported. */
|
2017-10-17 17:20:33 +02:00
|
|
|
config->cs = NULL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct lpd880x_data lpd880x_strip_data;
|
|
|
|
|
|
|
|
static const struct led_strip_driver_api lpd880x_strip_api = {
|
|
|
|
.update_rgb = lpd880x_strip_update_rgb,
|
|
|
|
.update_channels = lpd880x_strip_update_channels,
|
|
|
|
};
|
|
|
|
|
2020-03-25 18:01:50 +01:00
|
|
|
DEVICE_AND_API_INIT(lpd880x_strip, DT_INST_LABEL(0),
|
2017-10-17 17:20:33 +02:00
|
|
|
lpd880x_strip_init, &lpd880x_strip_data,
|
|
|
|
NULL, POST_KERNEL, CONFIG_LED_STRIP_INIT_PRIORITY,
|
|
|
|
&lpd880x_strip_api);
|