-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathled.c
154 lines (134 loc) · 4.04 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
148
149
150
151
152
153
154
#include <ch.h>
#include <hal.h>
#include "led.h"
int last_color[3] = {0};
systime_t led_blink_timer = UINT32_MAX;
static PWMConfig pwmcfg = {
8000000, /* PWM clock frequency. */
2048, /* Initial PWM period. */
NULL,
{
{PWM_OUTPUT_ACTIVE_LOW, NULL},
{PWM_OUTPUT_ACTIVE_LOW, NULL},
{PWM_OUTPUT_ACTIVE_LOW, NULL},
{PWM_OUTPUT_DISABLED, NULL}
},
0,
0
};
ioline_t led_dimmed = LINE_LED_UP;
static void pwm_dimmed_on(PWMDriver *pwmp) {
(void)pwmp;
palSetLine(led_dimmed);
}
static void pwm_dimmed_off(PWMDriver *pwmp) {
(void)pwmp;
palClearLine(led_dimmed);
}
static PWMConfig pwmcfg_updown = {
125000, /* PWM clock frequency. */
2048, /* Initial PWM period. */
pwm_dimmed_on,
{
{PWM_OUTPUT_ACTIVE_HIGH, pwm_dimmed_off},
{PWM_OUTPUT_DISABLED, NULL},
{PWM_OUTPUT_DISABLED, NULL},
{PWM_OUTPUT_DISABLED, NULL}
},
0,
0
};
/* Gamma correction to make light response close to linear.
* Input range: 0-256
* Ouput range: 0-2048
*/
int gamma8(int x) {
if (x < 33) return x;
else return ((x+1)*(x+1))/32-1;
}
void led_init(void) {
pwmStart(&PWMD4, &pwmcfg);
palSetLineMode(LINE_LED_R, PAL_MODE_ALTERNATE(2));
palSetLineMode(LINE_LED_G, PAL_MODE_ALTERNATE(2));
palSetLineMode(LINE_LED_B, PAL_MODE_ALTERNATE(2));
pwmStart(&PWMD3, &pwmcfg_updown);
pwmEnableChannel(&PWMD3, 0, 768); // duty cycle of dimmed up/down leds
}
void led_rgb(uint32_t rgb) {
int r = rgb >> 16 & 0xff;
int g = rgb >> 8 & 0xff;
int b = rgb >> 0 & 0xff;
led_rgb3(r, g, b);
}
void led_rgb3(int r, int g, int b) {
pwmEnableChannel(&PWMD4, 0, gamma8(r));
pwmEnableChannel(&PWMD4, 2, gamma8(g));
pwmEnableChannel(&PWMD4, 1, gamma8(b));
last_color[0] = r;
last_color[1] = g;
last_color[2] = b;
}
void led_rgb3_blink(int r, int g, int b, sysinterval_t time) {
pwmEnableChannel(&PWMD4, 0, gamma8(r));
pwmEnableChannel(&PWMD4, 2, gamma8(g));
pwmEnableChannel(&PWMD4, 1, gamma8(b));
led_blink_timer = chVTGetSystemTime() + time;
}
void led_tick(void) {
if (chVTGetSystemTime() > led_blink_timer) {
led_rgb3(last_color[0], last_color[1], last_color[2]);
led_blink_timer = UINT32_MAX;
}
}
inline int led_updown_state(ioline_t line, uint32_t state) {
if (state == 0xf) {
palSetLine(line);
} else if (state == 0x0) {
palClearLine(line);
} else {
led_dimmed = line;
return true;
}
return false;
}
/**
* Set up/down leds state
*
* state is 0xabcd with a=up2, b=up, c=down, d=down2
* 0 = off, f = on, anything else = dimmed.
* Only one led can be dimmed at the same time.
*/
void led_updown(uint32_t state) {
// combine all led calls in one SysLock zone to not have previous dimmed led
// turn on or off accidentally
// checks from pwmEnableChannelNotification/pwmEnablePeriodicNotification
osalDbgCheck((&PWMD3 != NULL) && (0 < PWMD3.channels));
osalSysLock();
osalDbgAssert(PWMD3.state == PWM_READY, "not ready");
osalDbgAssert(PWMD3.state == PWM_READY, "not ready");
osalDbgAssert(PWMD3.config->callback != NULL, "undefined periodic callback");
osalDbgAssert((PWMD3.enabled & ((pwmchnmsk_t)1U << (pwmchnmsk_t)0)) != 0U,
"channel not enabled");
osalDbgAssert(PWMD3.config->channels[0].callback != NULL,
"undefined channel callback");
if (led_updown_state(LINE_LED_UP2, state & 0xf) |
led_updown_state(LINE_LED_UP, state >> 4 & 0xf) |
led_updown_state(LINE_LED_DOWN, state >> 8 & 0xf) |
led_updown_state(LINE_LED_DOWN2, state >> 12 & 0xf))
{
pwmEnablePeriodicNotificationI(&PWMD3);
pwmEnableChannelNotificationI(&PWMD3, 0);
} else {
pwmDisableChannelNotificationI(&PWMD3, 0);
pwmDisablePeriodicNotificationI(&PWMD3);
}
osalSysUnlock();
}
void led_updown_dial(int angle) {
angle = ((angle + 1024) % 16) - 8; // +1024 to make % work as modulo for negative numbers
if (angle >= 0) {
led_updown(0xffff >> 2*angle);
} else {
led_updown(0xffff << 2*-angle);
}
}