forked from Inverted/tasbot_eyes
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathled.c
147 lines (124 loc) · 4.2 KB
/
led.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include "led.h"
#include "color.h"
#include "arguments.h"
#include "tasbot.h"
int brightness = BRIGHTNESS;
int dataPin = GPIO_PIN;
ws2811_led_t* buffer;
ws2811_t display;
pthread_t renderThread;
/**
* Initialize the LEDs and their data structure
* @return Infos about, if initialization was successful
*/
void initLEDs() {
if (activateLEDModule) {
//Setup display
memset(&display, 0, sizeof(ws2811_t));
display.freq = TARGET_FREQ;
display.dmanum = DMA;
ws2811_channel_t* channel = calloc(1, sizeof(ws2811_channel_t));
channel->gpionum = dataPin;
channel->count = LED_COUNT;
channel->invert = INVERTED;
channel->brightness = brightness;
channel->strip_type = STRIP_TYPE;
display.channel[0] = *channel; //todo: free at end
//Setup color array
buffer = malloc(sizeof(ws2811_led_t) * LED_WIDTH * LED_HEIGHT);
if (!buffer) {
fprintf(stderr, "[ERROR] initLEDs: Failed to allocate memory for render buffer");
exit(EXIT_FAILURE);
}
//Initialize hardware
ws2811_return_t r;
if ((r = ws2811_init(&display)) != WS2811_SUCCESS) {
fprintf(stderr, "[ERROR] ws2811_init failed. Couldn't initialize LEDs: %s\n", ws2811_get_return_t_str(r));
} else {
if (verbose) {
printf("[INFO] Initialized LEDs with code %d\n", r);
}
}
clearLEDs();
if (r != WS2811_SUCCESS) {
fprintf(stderr, "[ERROR] Can't run program. Did you started with root privileges?\n");
exit(EXIT_FAILURE);
}
} //else
if (verbose) {
printf("[INFO] Starting program without LED module\n");
}
}
/**
* Updates the display's hardware LEDs color to the local buffer variables array
* @return Infos about, if the LEDs where rendered successful
*/
ws2811_return_t renderLEDs(){
for (int x = 0; x < LED_WIDTH; x++) {
for (int y = 0; y < LED_HEIGHT; y++) {
//rainbow mode color fade
if (rainbowMode && buffer[(y * LED_WIDTH) + x] == 0xffffff){
float rgb[3];
hsv2rgb(hueToFloat(hue), 1, 1, rgb);
GifColorType rgbColor;
rgbColor.Red = valueToInt(rgb[0]);
rgbColor.Green = valueToInt(rgb[1]);
rgbColor.Blue = valueToInt(rgb[2]);
ws2811_led_t color = translateColor(&rgbColor, useGammaCorrection);
//set buffer to output
display.channel[0].leds[(y * LED_WIDTH) + x] = color;
} else {
display.channel[0].leds[(y * LED_WIDTH) + x] = buffer[(y * LED_WIDTH) + x];
}
}
}
ws2811_return_t r;
if ((r = ws2811_render(&display)) != WS2811_SUCCESS) {
fprintf(stderr, "[ERROR] Failed to render: %s\n", ws2811_get_return_t_str(r));
} else {
if (verbose) {
printf("[INFO] Rendered LEDs with code %d\n", r);
}
}
return r;
}
void* runRenderThread(void* vargp){
while (running){
renderLEDs();
usleep(RENDER_DELAY * 1000);
}
}
void startRenderThread() {
if (activateLEDModule){
pthread_create(&renderThread, NULL, runRenderThread, NULL);
if (verbose) {
printf("[INFO] Started thread for rendering with TID %lu\n", renderThread);
}
}
}
/**
* Clears all the LEDs by setting their color to black and renders it
*/
ws2811_return_t clearLEDs() {
for (size_t i = 0; i < (size_t) LED_COUNT; i++) {
buffer[i] = 0;
}
return renderLEDs();
}
/**
* Translate the RGB (255,255,255) color structure into a hexadecimal value
* @param _color The RGB color, that is to convert
* @return The convert hexadecimal color
*/
ws2811_led_t translateColor(GifColorType* _color, bool _useGammaCorrection) {
if (_useGammaCorrection) { //TODO: when used, things break
_color->Red = gamma8[_color->Red];
_color->Green = gamma8[_color->Green];
_color->Blue = gamma8[_color->Blue];
}
return ((_color->Red & 0xff) << 16) + ((_color->Green & 0xff) << 8) + (_color->Blue & 0xff);
}