mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
107 lines
2.7 KiB
C
107 lines
2.7 KiB
C
/*
|
|
* Copyright 2020 Christian Friedrich Coors
|
|
*
|
|
* This file is subject to the terms and conditions of the GNU Lesser
|
|
* General Public License v2.1. See the file LICENSE in the top level
|
|
* directory for more details.
|
|
*/
|
|
|
|
/**
|
|
* @ingroup drivers_ws281x
|
|
*
|
|
* @{
|
|
*
|
|
* @file
|
|
* @brief Implementation of `ws281x_write_buffer()` for the ESP32 CPU
|
|
*
|
|
* @author Christian Friedrich Coors <me@ccoors.de>
|
|
*
|
|
* @}
|
|
*/
|
|
#include <assert.h>
|
|
#include <errno.h>
|
|
#include <stdint.h>
|
|
#include <string.h>
|
|
|
|
#include "ws281x.h"
|
|
#include "ws281x_params.h"
|
|
#include "ws281x_constants.h"
|
|
#include "periph_cpu.h"
|
|
#include "xtimer.h"
|
|
#include "xtensa/core-macros.h"
|
|
#include "soc/rtc.h"
|
|
|
|
#define ENABLE_DEBUG 0
|
|
#include "debug.h"
|
|
|
|
|
|
static inline __attribute__((always_inline)) uint32_t get_cycle_count(void) {
|
|
uint32_t ccount;
|
|
__asm__ __volatile__("rsr %0,ccount":"=a" (ccount));
|
|
return ccount;
|
|
}
|
|
|
|
void ws281x_write_buffer(ws281x_t *dev, const void *buf, size_t size)
|
|
{
|
|
assert(dev);
|
|
const uint8_t *pos = buf;
|
|
const uint8_t *end = pos + size;
|
|
|
|
// Cycles
|
|
uint32_t total_cycles, one_on, one_off, zero_on, zero_off, on_wait, off_wait;
|
|
|
|
// Current frequency
|
|
rtc_cpu_freq_t freq = rtc_clk_cpu_freq_value(rtc_clk_cpu_freq_get());
|
|
|
|
total_cycles = freq / (NS_PER_SEC / WS281X_T_DATA_NS);
|
|
one_on = freq / (NS_PER_SEC / WS281X_T_DATA_ONE_NS);
|
|
one_off = total_cycles - one_on;
|
|
zero_on = freq / (NS_PER_SEC / WS281X_T_DATA_ZERO_NS);
|
|
zero_off = total_cycles - zero_on;
|
|
|
|
DEBUG("[ws281x] esp32 freq=%d total=%d\n", freq, total_cycles);
|
|
DEBUG("[ws281x] esp32 cycles %d/%d/%d/%d\n", one_on, one_off, zero_on, zero_off);
|
|
|
|
uint32_t current_wait = 0, start = 0;
|
|
|
|
while (pos < end) {
|
|
uint8_t data = *pos;
|
|
for (uint8_t cnt = 8; cnt > 0; cnt--) {
|
|
if (data & 0b10000000) {
|
|
on_wait = one_on;
|
|
off_wait = one_off;
|
|
}
|
|
else {
|
|
on_wait = zero_on;
|
|
off_wait = zero_off;
|
|
}
|
|
start = get_cycle_count();
|
|
gpio_set(dev->params.pin);
|
|
current_wait = start + on_wait;
|
|
while (get_cycle_count() < current_wait) { }
|
|
gpio_clear(dev->params.pin);
|
|
start = get_cycle_count();
|
|
current_wait = start + off_wait;
|
|
while (get_cycle_count() < current_wait) { }
|
|
data <<= 1;
|
|
}
|
|
pos++;
|
|
}
|
|
}
|
|
|
|
int ws281x_init(ws281x_t *dev, const ws281x_params_t *params)
|
|
{
|
|
if (!dev || !params || !params->buf) {
|
|
return -EINVAL;
|
|
}
|
|
|
|
memset(dev, 0, sizeof(ws281x_t));
|
|
dev->params = *params;
|
|
|
|
if (gpio_init(dev->params.pin, GPIO_OUT)) {
|
|
return -EIO;
|
|
}
|
|
|
|
return 0;
|
|
}
|