dc32-peppercon9-addon/firmware/user/src/led.c

182 lines
5.7 KiB
C

/*
* Created on: Jul 28, 2024
*/
#include "led.h"
#include "ledprog_pep.h"
#include "ledprog_rgb.h"
#include "adc.h"
#define FL3729_SW_COUNT 4 // switches utilized
#define FL3729_CS_COUNT 16 // current sink outputs used by chip
#define FL3729_G_CURRENT 15 // initial global current setting
static const uint8_t cs_currents[FL3729_CS_COUNT] = {
0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff,
0x1f, 0x1f, 0x1f, 0x1f, 0x1f, 0x1f, 0x1f
};
static const uint16_t pwm_cie_256in_1024out[] = {
0, 0, 1, 1, 2, 2, 3, 3, 4, 4, 4, 5, 5, 6, 6, 7,
7, 8, 8, 8, 9, 9, 10, 10, 11, 11, 12, 12, 13, 13, 14, 15,
15, 16, 17, 17, 18, 19, 19, 20, 21, 22, 22, 23, 24, 25, 26, 27,
28, 29, 30, 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 42, 43, 44,
45, 47, 48, 50, 51, 52, 54, 55, 57, 58, 60, 61, 63, 65, 66, 68,
70, 71, 73, 75, 77, 79, 81, 83, 84, 86, 88, 90, 93, 95, 97, 99,
101, 103, 106, 108, 110, 113, 115, 118, 120, 123, 125, 128, 130, 133, 136, 138,
141, 144, 147, 149, 152, 155, 158, 161, 164, 167, 171, 174, 177, 180, 183, 187,
190, 194, 197, 200, 204, 208, 211, 215, 218, 222, 226, 230, 234, 237, 241, 245,
249, 254, 258, 262, 266, 270, 275, 279, 283, 288, 292, 297, 301, 306, 311, 315,
320, 325, 330, 335, 340, 345, 350, 355, 360, 365, 370, 376, 381, 386, 392, 397,
403, 408, 414, 420, 425, 431, 437, 443, 449, 455, 461, 467, 473, 480, 486, 492,
499, 505, 512, 518, 525, 532, 538, 545, 552, 559, 566, 573, 580, 587, 594, 601,
609, 616, 624, 631, 639, 646, 654, 662, 669, 677, 685, 693, 701, 709, 717, 726,
734, 742, 751, 759, 768, 776, 785, 794, 802, 811, 820, 829, 838, 847, 857, 866,
875, 885, 894, 903, 913, 923, 932, 942, 952, 962, 972, 982, 992, 1002, 1013, 1023,
};
struct LedMatrix led;
uint8_t rgb[3];
static uint8_t led_matrix_updated = 0;
// helper for LED programs
void use_brightest(uint8_t *dest, uint8_t *compare, uint8_t count)
{
while (count--) {
if (*dest < *compare) *dest = *compare;
dest++;
compare++;
}
}
static uint16_t rgb_pwm_gamma(uint8_t in)
{
return pwm_cie_256in_1024out[in];
}
void led_init()
{
TIM_TimeBaseInitTypeDef timer ={0};
TIM_OCInitTypeDef pwm = {0};
uint8_t i;
// reset LED values
for (i = 0; i < FL3729_SW_COUNT * FL3729_CS_COUNT; i++) {
led.all[i] = 0;
}
rgb[0] = rgb[1] = rgb[2] = 0;
// configure matrix
is31fl3729_init(FL3729_ADDR,
FL3729_CONF_SSD_NRML | FL3729_CONF_OSDE_OFF | FL3729_CONF_MATRIX_4x16,
FL3729_G_CURRENT);
is31fl3729_set_scaling_current_multi(FL3729_ADDR, cs_currents, FL3729_CS_COUNT);
// configure rear RGBLED
timer.TIM_Period = (1 << 10) - 1; // 10-bit
timer.TIM_Prescaler = 0;
timer.TIM_ClockDivision = TIM_CKD_DIV1;
timer.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(RGB_TIM, &timer);
pwm.TIM_OCMode = TIM_OCMode_PWM1;
pwm.TIM_OutputState = TIM_OutputState_Enable;
pwm.TIM_Pulse = 0;
pwm.TIM_OCPolarity = TIM_OCPolarity_Low;
TIM_OC1Init(RGB_TIM, &pwm);
TIM_OC2Init(RGB_TIM, &pwm);
TIM_OC3Init(RGB_TIM, &pwm);
TIM_CtrlPWMOutputs(RGB_TIM, ENABLE);
TIM_OC1PreloadConfig(RGB_TIM, TIM_OCPreload_Disable);
TIM_ARRPreloadConfig(RGB_TIM, ENABLE);
TIM_Cmd(RGB_TIM, ENABLE);
}
void led_matrix_is_updated()
{
led_matrix_updated = 1;
}
void led_matrix_send()
{
int8_t i, j;
uint8_t out[64];
// only render when there's something to render
if (led_matrix_updated) {
// convert led matrix
j = 0;
// the pepper body is linear; starts from top left,
// goes counter-clockwise, and ends at the top part
for (i = 0; i < LED_PEP_COUNT; i += 4) {
out[j + (16*0)] = led.section.pep[i + 0];
out[j + (16*1)] = led.section.pep[i + 1];
out[j + (16*2)] = led.section.pep[i + 2];
out[j + (16*3)] = led.section.pep[i + 3];
j++;
}
// pepper hat is a bit more fucked up. I laid this out
// with the elements backwards. rushjob fuckup. oops.
// I would like to start from the bottom right, counter clockwise,
// ending up at pepper's top right.
// for (i = 0; i < LED_HAT_COUNT; i += 4) { // clockwise starting from bottom right
j = 15;
for (i = 0; i < LED_HAT_COUNT; i += 4) {
out[j + (16*0)] = led.section.hat[i + 0];
out[j + (16*1)] = led.section.hat[i + 1];
out[j + (16*2)] = led.section.hat[i + 2];
out[j + (16*3)] = led.section.hat[i + 3];
j--;
}
// and send
is31fl3729_set_outputs(FL3729_ADDR, 1, out, FL3729_SW_COUNT * FL3729_CS_COUNT);
led_matrix_updated = 0;
}
}
void led_rgb_update()
{
uint8_t i;
uint16_t scale[3];
uint8_t lsens = adc_get_lsens_coarse();
uint8_t max = adc_get_brightness_map(lsens);
// this isn't a matrix so we can just update whenever
// but we need to scale to ambient light level
for (i = 0; i < 3; i++) {
// first, scale to 75% of max scale
scale[i] = max * rgb[i];
scale[i] >>= 6;
// then get to 100% of max
scale[i] *= 3;
scale[i] >>= 1;
}
TIM_CtrlPWMOutputs(RGB_TIM, DISABLE);
RGB_TIM->CH1CVR = rgb_pwm_gamma(scale[2]);
RGB_TIM->CH2CVR = rgb_pwm_gamma(scale[1]);
RGB_TIM->CH3CVR = rgb_pwm_gamma(scale[0]);
TIM_CtrlPWMOutputs(RGB_TIM, ENABLE);
}