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
|
/* Copyright 2021 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#define DT_DRV_COMPAT cros_ec_pwm_leds
#include <string.h>
#include <devicetree.h>
#if DT_HAS_COMPAT_STATUS_OKAY(DT_DRV_COMPAT)
#include "led_pwm.h"
#include "pwm.h"
BUILD_ASSERT(DT_NUM_INST_STATUS_OKAY(cros_ec_pwm_leds) <= 1,
"Multiple CrOS EC PWM LED instances defined");
BUILD_ASSERT(DT_INST_PROP_LEN(0, leds) <= 2,
"Unsupported number of LEDs defined");
#define PWM_CHANNEL_BY_IDX(node_id, prop, idx, led_ch) \
PWM_CHANNEL(DT_PWMS_CTLR_BY_IDX( \
DT_PHANDLE_BY_IDX(node_id, prop, idx), led_ch))
#define PWM_CHANNEL_BY_IDX_COND(node_id, prop, idx, led_ch) \
COND_CODE_1(DT_PROP_HAS_IDX( \
DT_PHANDLE_BY_IDX(node_id, prop, idx), pwms, led_ch), \
(PWM_CHANNEL_BY_IDX(node_id, prop, idx, led_ch)), \
(PWM_LED_NO_CHANNEL))
#define PWM_LED_INIT(node_id, prop, idx) \
[PWM_LED##idx] = { \
.ch0 = PWM_CHANNEL_BY_IDX_COND(node_id, prop, idx, 0), \
.ch1 = PWM_CHANNEL_BY_IDX_COND(node_id, prop, idx, 1), \
.ch2 = PWM_CHANNEL_BY_IDX_COND(node_id, prop, idx, 2), \
.enable = &pwm_enable, \
.set_duty = &pwm_set_duty, \
},
struct pwm_led pwm_leds[] = {
DT_INST_FOREACH_PROP_ELEM(0, leds, PWM_LED_INIT)
};
#define EC_LED_COLOR_BLANK {0}
struct pwm_led_color_map led_color_map[EC_LED_COLOR_COUNT] = {
[EC_LED_COLOR_RED] = DT_INST_PROP_OR(0, color_map_red,
EC_LED_COLOR_BLANK),
[EC_LED_COLOR_GREEN] = DT_INST_PROP_OR(0, color_map_green,
EC_LED_COLOR_BLANK),
[EC_LED_COLOR_BLUE] = DT_INST_PROP_OR(0, color_map_blue,
EC_LED_COLOR_BLANK),
[EC_LED_COLOR_YELLOW] = DT_INST_PROP_OR(0, color_map_yellow,
EC_LED_COLOR_BLANK),
[EC_LED_COLOR_WHITE] = DT_INST_PROP_OR(0, color_map_white,
EC_LED_COLOR_BLANK),
[EC_LED_COLOR_AMBER] = DT_INST_PROP_OR(0, color_map_amber,
EC_LED_COLOR_BLANK),
};
BUILD_ASSERT(DT_INST_PROP_LEN(0, brightness_range) == EC_LED_COLOR_COUNT,
"brightness_range must have exactly EC_LED_COLOR_COUNT values");
static const uint8_t dt_brigthness_range[EC_LED_COLOR_COUNT] = DT_INST_PROP(
0, brightness_range);
void led_get_brightness_range(enum ec_led_id led_id, uint8_t *brightness_range)
{
/* led_id is ignored, same ranges for all LEDs */
memcpy(brightness_range, dt_brigthness_range,
sizeof(dt_brigthness_range));
}
#define PWM_NAME_TO_ID(node_id) \
case DT_STRING_TOKEN(node_id, ec_led_name): \
pwm_id = DT_REG_ADDR(node_id); \
break;
int led_set_brightness(enum ec_led_id led_id, const uint8_t *brightness)
{
enum pwm_led_id pwm_id;
switch (led_id) {
DT_INST_FOREACH_CHILD(0, PWM_NAME_TO_ID)
default:
return EC_ERROR_UNKNOWN;
}
if (DT_INST_NODE_HAS_PROP(0, color_map_red) &&
brightness[EC_LED_COLOR_RED])
set_pwm_led_color(pwm_id, EC_LED_COLOR_RED);
else if (DT_INST_NODE_HAS_PROP(0, color_map_green) &&
brightness[EC_LED_COLOR_GREEN])
set_pwm_led_color(pwm_id, EC_LED_COLOR_GREEN);
else if (DT_INST_NODE_HAS_PROP(0, color_map_blue) &&
brightness[EC_LED_COLOR_BLUE])
set_pwm_led_color(pwm_id, EC_LED_COLOR_BLUE);
else if (DT_INST_NODE_HAS_PROP(0, color_map_yellow) &&
brightness[EC_LED_COLOR_YELLOW])
set_pwm_led_color(pwm_id, EC_LED_COLOR_YELLOW);
else if (DT_INST_NODE_HAS_PROP(0, color_map_white) &&
brightness[EC_LED_COLOR_WHITE])
set_pwm_led_color(pwm_id, EC_LED_COLOR_WHITE);
else if (DT_INST_NODE_HAS_PROP(0, color_map_amber) &&
brightness[EC_LED_COLOR_AMBER])
set_pwm_led_color(pwm_id, EC_LED_COLOR_AMBER);
else
/* Otherwise, the "color" is "off". */
set_pwm_led_color(pwm_id, -1);
return EC_SUCCESS;
}
#endif /* DT_HAS_COMPAT_STATUS_OKAY */
|