2018-02-12 22:35:20 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018 Google LLC.
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
2020-03-25 18:01:50 +01:00
|
|
|
#define DT_DRV_COMPAT apa_apa102
|
|
|
|
|
2018-02-12 22:35:20 +01:00
|
|
|
#include <errno.h>
|
2022-05-06 10:25:46 +02:00
|
|
|
#include <zephyr/drivers/led_strip.h>
|
|
|
|
#include <zephyr/drivers/spi.h>
|
|
|
|
#include <zephyr/drivers/gpio.h>
|
|
|
|
#include <zephyr/sys/util.h>
|
2020-05-14 04:24:10 +02:00
|
|
|
|
2021-08-07 05:01:52 +02:00
|
|
|
struct apa102_config {
|
|
|
|
struct spi_dt_spec bus;
|
2018-02-12 22:35:20 +01:00
|
|
|
};
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static int apa102_update(const struct device *dev, void *buf, size_t size)
|
2018-02-12 22:35:20 +01:00
|
|
|
{
|
2021-08-07 05:01:52 +02:00
|
|
|
const struct apa102_config *config = dev->config;
|
|
|
|
static const uint8_t zeros[] = { 0, 0, 0, 0 };
|
|
|
|
static const uint8_t ones[] = { 0xFF, 0xFF, 0xFF, 0xFF };
|
2018-01-30 13:49:01 +01:00
|
|
|
const struct spi_buf tx_bufs[] = {
|
2018-02-12 22:35:20 +01:00
|
|
|
{
|
|
|
|
/* Start frame: at least 32 zeros */
|
2020-05-27 18:26:57 +02:00
|
|
|
.buf = (uint8_t *)zeros,
|
2018-02-12 22:35:20 +01:00
|
|
|
.len = sizeof(zeros),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
/* LED data itself */
|
|
|
|
.buf = buf,
|
|
|
|
.len = size,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
/* End frame: at least 32 ones to clock the
|
|
|
|
* remaining bits to the LEDs at the end of
|
|
|
|
* the strip.
|
|
|
|
*/
|
2020-05-27 18:26:57 +02:00
|
|
|
.buf = (uint8_t *)ones,
|
2018-02-12 22:35:20 +01:00
|
|
|
.len = sizeof(ones),
|
|
|
|
},
|
|
|
|
};
|
2018-01-30 13:49:01 +01:00
|
|
|
const struct spi_buf_set tx = {
|
|
|
|
.buffers = tx_bufs,
|
2018-12-08 16:39:39 +01:00
|
|
|
.count = ARRAY_SIZE(tx_bufs)
|
2018-01-30 13:49:01 +01:00
|
|
|
};
|
2018-02-12 22:35:20 +01:00
|
|
|
|
2021-08-07 05:01:52 +02:00
|
|
|
return spi_write_dt(&config->bus, &tx);
|
2018-02-12 22:35:20 +01:00
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static int apa102_update_rgb(const struct device *dev, struct led_rgb *pixels,
|
2018-02-12 22:35:20 +01:00
|
|
|
size_t count)
|
|
|
|
{
|
2020-05-27 18:26:57 +02:00
|
|
|
uint8_t *p = (uint8_t *)pixels;
|
2018-02-12 22:35:20 +01:00
|
|
|
size_t i;
|
|
|
|
/* SOF (3 bits) followed by the 0 to 31 global dimming level */
|
2020-05-27 18:26:57 +02:00
|
|
|
uint8_t prefix = 0xE0 | 31;
|
2018-02-12 22:35:20 +01:00
|
|
|
|
|
|
|
/* Rewrite to the on-wire format */
|
|
|
|
for (i = 0; i < count; i++) {
|
2020-05-27 18:26:57 +02:00
|
|
|
uint8_t r = pixels[i].r;
|
|
|
|
uint8_t g = pixels[i].g;
|
|
|
|
uint8_t b = pixels[i].b;
|
2018-02-12 22:35:20 +01:00
|
|
|
|
|
|
|
*p++ = prefix;
|
|
|
|
*p++ = b;
|
|
|
|
*p++ = g;
|
|
|
|
*p++ = r;
|
|
|
|
}
|
|
|
|
|
|
|
|
BUILD_ASSERT(sizeof(struct led_rgb) == 4);
|
|
|
|
return apa102_update(dev, pixels, sizeof(struct led_rgb) * count);
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static int apa102_update_channels(const struct device *dev, uint8_t *channels,
|
2018-02-12 22:35:20 +01:00
|
|
|
size_t num_channels)
|
|
|
|
{
|
|
|
|
/* Not implemented */
|
|
|
|
return -EINVAL;
|
|
|
|
}
|
|
|
|
|
2020-04-30 20:33:38 +02:00
|
|
|
static int apa102_init(const struct device *dev)
|
2018-02-12 22:35:20 +01:00
|
|
|
{
|
2021-08-07 05:01:52 +02:00
|
|
|
const struct apa102_config *config = dev->config;
|
2018-02-12 22:35:20 +01:00
|
|
|
|
2022-12-03 14:49:05 +01:00
|
|
|
if (!spi_is_ready_dt(&config->bus)) {
|
2018-02-12 22:35:20 +01:00
|
|
|
return -ENODEV;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const struct led_strip_driver_api apa102_api = {
|
|
|
|
.update_rgb = apa102_update_rgb,
|
|
|
|
.update_channels = apa102_update_channels,
|
|
|
|
};
|
|
|
|
|
2023-05-13 16:37:01 +02:00
|
|
|
#define APA102_DEVICE(idx) \
|
|
|
|
static const struct apa102_config apa102_##idx##_config = { \
|
|
|
|
.bus = SPI_DT_SPEC_INST_GET( \
|
|
|
|
idx, \
|
|
|
|
SPI_OP_MODE_MASTER | SPI_TRANSFER_MSB | SPI_WORD_SET(8), \
|
|
|
|
0), \
|
|
|
|
}; \
|
|
|
|
\
|
|
|
|
DEVICE_DT_INST_DEFINE(idx, \
|
|
|
|
apa102_init, \
|
|
|
|
NULL, \
|
|
|
|
NULL, \
|
|
|
|
&apa102_##idx##_config, \
|
|
|
|
POST_KERNEL, \
|
|
|
|
CONFIG_LED_STRIP_INIT_PRIORITY, \
|
|
|
|
&apa102_api);
|
|
|
|
|
|
|
|
DT_INST_FOREACH_STATUS_OKAY(APA102_DEVICE)
|