1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0+
2*4882a593Smuzhiyun //
3*4882a593Smuzhiyun // Driver for Panasonic AN30259A 3-channel LED driver
4*4882a593Smuzhiyun //
5*4882a593Smuzhiyun // Copyright (c) 2018 Simon Shields <simon@lineageos.org>
6*4882a593Smuzhiyun //
7*4882a593Smuzhiyun // Datasheet:
8*4882a593Smuzhiyun // https://www.alliedelec.com/m/d/a9d2b3ee87c2d1a535a41dd747b1c247.pdf
9*4882a593Smuzhiyun
10*4882a593Smuzhiyun #include <linux/i2c.h>
11*4882a593Smuzhiyun #include <linux/leds.h>
12*4882a593Smuzhiyun #include <linux/module.h>
13*4882a593Smuzhiyun #include <linux/mutex.h>
14*4882a593Smuzhiyun #include <linux/of.h>
15*4882a593Smuzhiyun #include <linux/regmap.h>
16*4882a593Smuzhiyun
17*4882a593Smuzhiyun #define AN30259A_MAX_LEDS 3
18*4882a593Smuzhiyun
19*4882a593Smuzhiyun #define AN30259A_REG_SRESET 0x00
20*4882a593Smuzhiyun #define AN30259A_LED_SRESET BIT(0)
21*4882a593Smuzhiyun
22*4882a593Smuzhiyun /* LED power registers */
23*4882a593Smuzhiyun #define AN30259A_REG_LED_ON 0x01
24*4882a593Smuzhiyun #define AN30259A_LED_EN(x) BIT((x) - 1)
25*4882a593Smuzhiyun #define AN30259A_LED_SLOPE(x) BIT(((x) - 1) + 4)
26*4882a593Smuzhiyun
27*4882a593Smuzhiyun #define AN30259A_REG_LEDCC(x) (0x03 + ((x) - 1))
28*4882a593Smuzhiyun
29*4882a593Smuzhiyun /* slope control registers */
30*4882a593Smuzhiyun #define AN30259A_REG_SLOPE(x) (0x06 + ((x) - 1))
31*4882a593Smuzhiyun #define AN30259A_LED_SLOPETIME1(x) (x)
32*4882a593Smuzhiyun #define AN30259A_LED_SLOPETIME2(x) ((x) << 4)
33*4882a593Smuzhiyun
34*4882a593Smuzhiyun #define AN30259A_REG_LEDCNT1(x) (0x09 + (4 * ((x) - 1)))
35*4882a593Smuzhiyun #define AN30259A_LED_DUTYMAX(x) ((x) << 4)
36*4882a593Smuzhiyun #define AN30259A_LED_DUTYMID(x) (x)
37*4882a593Smuzhiyun
38*4882a593Smuzhiyun #define AN30259A_REG_LEDCNT2(x) (0x0A + (4 * ((x) - 1)))
39*4882a593Smuzhiyun #define AN30259A_LED_DELAY(x) ((x) << 4)
40*4882a593Smuzhiyun #define AN30259A_LED_DUTYMIN(x) (x)
41*4882a593Smuzhiyun
42*4882a593Smuzhiyun /* detention time control (length of each slope step) */
43*4882a593Smuzhiyun #define AN30259A_REG_LEDCNT3(x) (0x0B + (4 * ((x) - 1)))
44*4882a593Smuzhiyun #define AN30259A_LED_DT1(x) (x)
45*4882a593Smuzhiyun #define AN30259A_LED_DT2(x) ((x) << 4)
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun #define AN30259A_REG_LEDCNT4(x) (0x0C + (4 * ((x) - 1)))
48*4882a593Smuzhiyun #define AN30259A_LED_DT3(x) (x)
49*4882a593Smuzhiyun #define AN30259A_LED_DT4(x) ((x) << 4)
50*4882a593Smuzhiyun
51*4882a593Smuzhiyun #define AN30259A_REG_MAX 0x14
52*4882a593Smuzhiyun
53*4882a593Smuzhiyun #define AN30259A_BLINK_MAX_TIME 7500 /* ms */
54*4882a593Smuzhiyun #define AN30259A_SLOPE_RESOLUTION 500 /* ms */
55*4882a593Smuzhiyun
56*4882a593Smuzhiyun #define AN30259A_NAME "an30259a"
57*4882a593Smuzhiyun
58*4882a593Smuzhiyun #define STATE_OFF 0
59*4882a593Smuzhiyun #define STATE_KEEP 1
60*4882a593Smuzhiyun #define STATE_ON 2
61*4882a593Smuzhiyun
62*4882a593Smuzhiyun struct an30259a;
63*4882a593Smuzhiyun
64*4882a593Smuzhiyun struct an30259a_led {
65*4882a593Smuzhiyun struct an30259a *chip;
66*4882a593Smuzhiyun struct fwnode_handle *fwnode;
67*4882a593Smuzhiyun struct led_classdev cdev;
68*4882a593Smuzhiyun u32 num;
69*4882a593Smuzhiyun u32 default_state;
70*4882a593Smuzhiyun bool sloping;
71*4882a593Smuzhiyun };
72*4882a593Smuzhiyun
73*4882a593Smuzhiyun struct an30259a {
74*4882a593Smuzhiyun struct mutex mutex; /* held when writing to registers */
75*4882a593Smuzhiyun struct i2c_client *client;
76*4882a593Smuzhiyun struct an30259a_led leds[AN30259A_MAX_LEDS];
77*4882a593Smuzhiyun struct regmap *regmap;
78*4882a593Smuzhiyun int num_leds;
79*4882a593Smuzhiyun };
80*4882a593Smuzhiyun
an30259a_brightness_set(struct led_classdev * cdev,enum led_brightness brightness)81*4882a593Smuzhiyun static int an30259a_brightness_set(struct led_classdev *cdev,
82*4882a593Smuzhiyun enum led_brightness brightness)
83*4882a593Smuzhiyun {
84*4882a593Smuzhiyun struct an30259a_led *led;
85*4882a593Smuzhiyun int ret;
86*4882a593Smuzhiyun unsigned int led_on;
87*4882a593Smuzhiyun
88*4882a593Smuzhiyun led = container_of(cdev, struct an30259a_led, cdev);
89*4882a593Smuzhiyun mutex_lock(&led->chip->mutex);
90*4882a593Smuzhiyun
91*4882a593Smuzhiyun ret = regmap_read(led->chip->regmap, AN30259A_REG_LED_ON, &led_on);
92*4882a593Smuzhiyun if (ret)
93*4882a593Smuzhiyun goto error;
94*4882a593Smuzhiyun
95*4882a593Smuzhiyun switch (brightness) {
96*4882a593Smuzhiyun case LED_OFF:
97*4882a593Smuzhiyun led_on &= ~AN30259A_LED_EN(led->num);
98*4882a593Smuzhiyun led_on &= ~AN30259A_LED_SLOPE(led->num);
99*4882a593Smuzhiyun led->sloping = false;
100*4882a593Smuzhiyun break;
101*4882a593Smuzhiyun default:
102*4882a593Smuzhiyun led_on |= AN30259A_LED_EN(led->num);
103*4882a593Smuzhiyun if (led->sloping)
104*4882a593Smuzhiyun led_on |= AN30259A_LED_SLOPE(led->num);
105*4882a593Smuzhiyun ret = regmap_write(led->chip->regmap,
106*4882a593Smuzhiyun AN30259A_REG_LEDCNT1(led->num),
107*4882a593Smuzhiyun AN30259A_LED_DUTYMAX(0xf) |
108*4882a593Smuzhiyun AN30259A_LED_DUTYMID(0xf));
109*4882a593Smuzhiyun if (ret)
110*4882a593Smuzhiyun goto error;
111*4882a593Smuzhiyun break;
112*4882a593Smuzhiyun }
113*4882a593Smuzhiyun
114*4882a593Smuzhiyun ret = regmap_write(led->chip->regmap, AN30259A_REG_LED_ON, led_on);
115*4882a593Smuzhiyun if (ret)
116*4882a593Smuzhiyun goto error;
117*4882a593Smuzhiyun
118*4882a593Smuzhiyun ret = regmap_write(led->chip->regmap, AN30259A_REG_LEDCC(led->num),
119*4882a593Smuzhiyun brightness);
120*4882a593Smuzhiyun
121*4882a593Smuzhiyun error:
122*4882a593Smuzhiyun mutex_unlock(&led->chip->mutex);
123*4882a593Smuzhiyun
124*4882a593Smuzhiyun return ret;
125*4882a593Smuzhiyun }
126*4882a593Smuzhiyun
an30259a_blink_set(struct led_classdev * cdev,unsigned long * delay_off,unsigned long * delay_on)127*4882a593Smuzhiyun static int an30259a_blink_set(struct led_classdev *cdev,
128*4882a593Smuzhiyun unsigned long *delay_off, unsigned long *delay_on)
129*4882a593Smuzhiyun {
130*4882a593Smuzhiyun struct an30259a_led *led;
131*4882a593Smuzhiyun int ret, num;
132*4882a593Smuzhiyun unsigned int led_on;
133*4882a593Smuzhiyun unsigned long off = *delay_off, on = *delay_on;
134*4882a593Smuzhiyun
135*4882a593Smuzhiyun led = container_of(cdev, struct an30259a_led, cdev);
136*4882a593Smuzhiyun
137*4882a593Smuzhiyun mutex_lock(&led->chip->mutex);
138*4882a593Smuzhiyun num = led->num;
139*4882a593Smuzhiyun
140*4882a593Smuzhiyun /* slope time can only be a multiple of 500ms. */
141*4882a593Smuzhiyun if (off % AN30259A_SLOPE_RESOLUTION || on % AN30259A_SLOPE_RESOLUTION) {
142*4882a593Smuzhiyun ret = -EINVAL;
143*4882a593Smuzhiyun goto error;
144*4882a593Smuzhiyun }
145*4882a593Smuzhiyun
146*4882a593Smuzhiyun /* up to a maximum of 7500ms. */
147*4882a593Smuzhiyun if (off > AN30259A_BLINK_MAX_TIME || on > AN30259A_BLINK_MAX_TIME) {
148*4882a593Smuzhiyun ret = -EINVAL;
149*4882a593Smuzhiyun goto error;
150*4882a593Smuzhiyun }
151*4882a593Smuzhiyun
152*4882a593Smuzhiyun /* if no blink specified, default to 1 Hz. */
153*4882a593Smuzhiyun if (!off && !on) {
154*4882a593Smuzhiyun *delay_off = off = 500;
155*4882a593Smuzhiyun *delay_on = on = 500;
156*4882a593Smuzhiyun }
157*4882a593Smuzhiyun
158*4882a593Smuzhiyun /* convert into values the HW will understand. */
159*4882a593Smuzhiyun off /= AN30259A_SLOPE_RESOLUTION;
160*4882a593Smuzhiyun on /= AN30259A_SLOPE_RESOLUTION;
161*4882a593Smuzhiyun
162*4882a593Smuzhiyun /* duty min should be zero (=off), delay should be zero. */
163*4882a593Smuzhiyun ret = regmap_write(led->chip->regmap, AN30259A_REG_LEDCNT2(num),
164*4882a593Smuzhiyun AN30259A_LED_DELAY(0) | AN30259A_LED_DUTYMIN(0));
165*4882a593Smuzhiyun if (ret)
166*4882a593Smuzhiyun goto error;
167*4882a593Smuzhiyun
168*4882a593Smuzhiyun /* reset detention time (no "breathing" effect). */
169*4882a593Smuzhiyun ret = regmap_write(led->chip->regmap, AN30259A_REG_LEDCNT3(num),
170*4882a593Smuzhiyun AN30259A_LED_DT1(0) | AN30259A_LED_DT2(0));
171*4882a593Smuzhiyun if (ret)
172*4882a593Smuzhiyun goto error;
173*4882a593Smuzhiyun ret = regmap_write(led->chip->regmap, AN30259A_REG_LEDCNT4(num),
174*4882a593Smuzhiyun AN30259A_LED_DT3(0) | AN30259A_LED_DT4(0));
175*4882a593Smuzhiyun if (ret)
176*4882a593Smuzhiyun goto error;
177*4882a593Smuzhiyun
178*4882a593Smuzhiyun /* slope time controls on/off cycle length. */
179*4882a593Smuzhiyun ret = regmap_write(led->chip->regmap, AN30259A_REG_SLOPE(num),
180*4882a593Smuzhiyun AN30259A_LED_SLOPETIME1(off) |
181*4882a593Smuzhiyun AN30259A_LED_SLOPETIME2(on));
182*4882a593Smuzhiyun if (ret)
183*4882a593Smuzhiyun goto error;
184*4882a593Smuzhiyun
185*4882a593Smuzhiyun /* Finally, enable slope mode. */
186*4882a593Smuzhiyun ret = regmap_read(led->chip->regmap, AN30259A_REG_LED_ON, &led_on);
187*4882a593Smuzhiyun if (ret)
188*4882a593Smuzhiyun goto error;
189*4882a593Smuzhiyun
190*4882a593Smuzhiyun led_on |= AN30259A_LED_SLOPE(num) | AN30259A_LED_EN(led->num);
191*4882a593Smuzhiyun
192*4882a593Smuzhiyun ret = regmap_write(led->chip->regmap, AN30259A_REG_LED_ON, led_on);
193*4882a593Smuzhiyun
194*4882a593Smuzhiyun if (!ret)
195*4882a593Smuzhiyun led->sloping = true;
196*4882a593Smuzhiyun error:
197*4882a593Smuzhiyun mutex_unlock(&led->chip->mutex);
198*4882a593Smuzhiyun
199*4882a593Smuzhiyun return ret;
200*4882a593Smuzhiyun }
201*4882a593Smuzhiyun
an30259a_dt_init(struct i2c_client * client,struct an30259a * chip)202*4882a593Smuzhiyun static int an30259a_dt_init(struct i2c_client *client,
203*4882a593Smuzhiyun struct an30259a *chip)
204*4882a593Smuzhiyun {
205*4882a593Smuzhiyun struct device_node *np = dev_of_node(&client->dev), *child;
206*4882a593Smuzhiyun int count, ret;
207*4882a593Smuzhiyun int i = 0;
208*4882a593Smuzhiyun const char *str;
209*4882a593Smuzhiyun struct an30259a_led *led;
210*4882a593Smuzhiyun
211*4882a593Smuzhiyun count = of_get_available_child_count(np);
212*4882a593Smuzhiyun if (!count || count > AN30259A_MAX_LEDS)
213*4882a593Smuzhiyun return -EINVAL;
214*4882a593Smuzhiyun
215*4882a593Smuzhiyun for_each_available_child_of_node(np, child) {
216*4882a593Smuzhiyun u32 source;
217*4882a593Smuzhiyun
218*4882a593Smuzhiyun ret = of_property_read_u32(child, "reg", &source);
219*4882a593Smuzhiyun if (ret != 0 || !source || source > AN30259A_MAX_LEDS) {
220*4882a593Smuzhiyun dev_err(&client->dev, "Couldn't read LED address: %d\n",
221*4882a593Smuzhiyun ret);
222*4882a593Smuzhiyun count--;
223*4882a593Smuzhiyun continue;
224*4882a593Smuzhiyun }
225*4882a593Smuzhiyun
226*4882a593Smuzhiyun led = &chip->leds[i];
227*4882a593Smuzhiyun
228*4882a593Smuzhiyun led->num = source;
229*4882a593Smuzhiyun led->chip = chip;
230*4882a593Smuzhiyun led->fwnode = of_fwnode_handle(child);
231*4882a593Smuzhiyun
232*4882a593Smuzhiyun if (!of_property_read_string(child, "default-state", &str)) {
233*4882a593Smuzhiyun if (!strcmp(str, "on"))
234*4882a593Smuzhiyun led->default_state = STATE_ON;
235*4882a593Smuzhiyun else if (!strcmp(str, "keep"))
236*4882a593Smuzhiyun led->default_state = STATE_KEEP;
237*4882a593Smuzhiyun else
238*4882a593Smuzhiyun led->default_state = STATE_OFF;
239*4882a593Smuzhiyun }
240*4882a593Smuzhiyun
241*4882a593Smuzhiyun i++;
242*4882a593Smuzhiyun }
243*4882a593Smuzhiyun
244*4882a593Smuzhiyun if (!count)
245*4882a593Smuzhiyun return -EINVAL;
246*4882a593Smuzhiyun
247*4882a593Smuzhiyun chip->num_leds = i;
248*4882a593Smuzhiyun
249*4882a593Smuzhiyun return 0;
250*4882a593Smuzhiyun }
251*4882a593Smuzhiyun
252*4882a593Smuzhiyun static const struct regmap_config an30259a_regmap_config = {
253*4882a593Smuzhiyun .reg_bits = 8,
254*4882a593Smuzhiyun .val_bits = 8,
255*4882a593Smuzhiyun .max_register = AN30259A_REG_MAX,
256*4882a593Smuzhiyun };
257*4882a593Smuzhiyun
an30259a_init_default_state(struct an30259a_led * led)258*4882a593Smuzhiyun static void an30259a_init_default_state(struct an30259a_led *led)
259*4882a593Smuzhiyun {
260*4882a593Smuzhiyun struct an30259a *chip = led->chip;
261*4882a593Smuzhiyun int led_on, err;
262*4882a593Smuzhiyun
263*4882a593Smuzhiyun switch (led->default_state) {
264*4882a593Smuzhiyun case STATE_ON:
265*4882a593Smuzhiyun led->cdev.brightness = LED_FULL;
266*4882a593Smuzhiyun break;
267*4882a593Smuzhiyun case STATE_KEEP:
268*4882a593Smuzhiyun err = regmap_read(chip->regmap, AN30259A_REG_LED_ON, &led_on);
269*4882a593Smuzhiyun if (err)
270*4882a593Smuzhiyun break;
271*4882a593Smuzhiyun
272*4882a593Smuzhiyun if (!(led_on & AN30259A_LED_EN(led->num))) {
273*4882a593Smuzhiyun led->cdev.brightness = LED_OFF;
274*4882a593Smuzhiyun break;
275*4882a593Smuzhiyun }
276*4882a593Smuzhiyun regmap_read(chip->regmap, AN30259A_REG_LEDCC(led->num),
277*4882a593Smuzhiyun &led->cdev.brightness);
278*4882a593Smuzhiyun break;
279*4882a593Smuzhiyun default:
280*4882a593Smuzhiyun led->cdev.brightness = LED_OFF;
281*4882a593Smuzhiyun }
282*4882a593Smuzhiyun
283*4882a593Smuzhiyun an30259a_brightness_set(&led->cdev, led->cdev.brightness);
284*4882a593Smuzhiyun }
285*4882a593Smuzhiyun
an30259a_probe(struct i2c_client * client)286*4882a593Smuzhiyun static int an30259a_probe(struct i2c_client *client)
287*4882a593Smuzhiyun {
288*4882a593Smuzhiyun struct an30259a *chip;
289*4882a593Smuzhiyun int i, err;
290*4882a593Smuzhiyun
291*4882a593Smuzhiyun chip = devm_kzalloc(&client->dev, sizeof(*chip), GFP_KERNEL);
292*4882a593Smuzhiyun if (!chip)
293*4882a593Smuzhiyun return -ENOMEM;
294*4882a593Smuzhiyun
295*4882a593Smuzhiyun err = an30259a_dt_init(client, chip);
296*4882a593Smuzhiyun if (err < 0)
297*4882a593Smuzhiyun return err;
298*4882a593Smuzhiyun
299*4882a593Smuzhiyun mutex_init(&chip->mutex);
300*4882a593Smuzhiyun chip->client = client;
301*4882a593Smuzhiyun i2c_set_clientdata(client, chip);
302*4882a593Smuzhiyun
303*4882a593Smuzhiyun chip->regmap = devm_regmap_init_i2c(client, &an30259a_regmap_config);
304*4882a593Smuzhiyun
305*4882a593Smuzhiyun if (IS_ERR(chip->regmap)) {
306*4882a593Smuzhiyun err = PTR_ERR(chip->regmap);
307*4882a593Smuzhiyun dev_err(&client->dev, "Failed to allocate register map: %d\n",
308*4882a593Smuzhiyun err);
309*4882a593Smuzhiyun goto exit;
310*4882a593Smuzhiyun }
311*4882a593Smuzhiyun
312*4882a593Smuzhiyun for (i = 0; i < chip->num_leds; i++) {
313*4882a593Smuzhiyun struct led_init_data init_data = {};
314*4882a593Smuzhiyun
315*4882a593Smuzhiyun an30259a_init_default_state(&chip->leds[i]);
316*4882a593Smuzhiyun chip->leds[i].cdev.brightness_set_blocking =
317*4882a593Smuzhiyun an30259a_brightness_set;
318*4882a593Smuzhiyun chip->leds[i].cdev.blink_set = an30259a_blink_set;
319*4882a593Smuzhiyun
320*4882a593Smuzhiyun init_data.fwnode = chip->leds[i].fwnode;
321*4882a593Smuzhiyun init_data.devicename = AN30259A_NAME;
322*4882a593Smuzhiyun init_data.default_label = ":";
323*4882a593Smuzhiyun
324*4882a593Smuzhiyun err = devm_led_classdev_register_ext(&client->dev,
325*4882a593Smuzhiyun &chip->leds[i].cdev,
326*4882a593Smuzhiyun &init_data);
327*4882a593Smuzhiyun if (err < 0)
328*4882a593Smuzhiyun goto exit;
329*4882a593Smuzhiyun }
330*4882a593Smuzhiyun return 0;
331*4882a593Smuzhiyun
332*4882a593Smuzhiyun exit:
333*4882a593Smuzhiyun mutex_destroy(&chip->mutex);
334*4882a593Smuzhiyun return err;
335*4882a593Smuzhiyun }
336*4882a593Smuzhiyun
an30259a_remove(struct i2c_client * client)337*4882a593Smuzhiyun static int an30259a_remove(struct i2c_client *client)
338*4882a593Smuzhiyun {
339*4882a593Smuzhiyun struct an30259a *chip = i2c_get_clientdata(client);
340*4882a593Smuzhiyun
341*4882a593Smuzhiyun mutex_destroy(&chip->mutex);
342*4882a593Smuzhiyun
343*4882a593Smuzhiyun return 0;
344*4882a593Smuzhiyun }
345*4882a593Smuzhiyun
346*4882a593Smuzhiyun static const struct of_device_id an30259a_match_table[] = {
347*4882a593Smuzhiyun { .compatible = "panasonic,an30259a", },
348*4882a593Smuzhiyun { /* sentinel */ },
349*4882a593Smuzhiyun };
350*4882a593Smuzhiyun
351*4882a593Smuzhiyun MODULE_DEVICE_TABLE(of, an30259a_match_table);
352*4882a593Smuzhiyun
353*4882a593Smuzhiyun static const struct i2c_device_id an30259a_id[] = {
354*4882a593Smuzhiyun { "an30259a", 0 },
355*4882a593Smuzhiyun { /* sentinel */ },
356*4882a593Smuzhiyun };
357*4882a593Smuzhiyun MODULE_DEVICE_TABLE(i2c, an30259a_id);
358*4882a593Smuzhiyun
359*4882a593Smuzhiyun static struct i2c_driver an30259a_driver = {
360*4882a593Smuzhiyun .driver = {
361*4882a593Smuzhiyun .name = "leds-an30259a",
362*4882a593Smuzhiyun .of_match_table = of_match_ptr(an30259a_match_table),
363*4882a593Smuzhiyun },
364*4882a593Smuzhiyun .probe_new = an30259a_probe,
365*4882a593Smuzhiyun .remove = an30259a_remove,
366*4882a593Smuzhiyun .id_table = an30259a_id,
367*4882a593Smuzhiyun };
368*4882a593Smuzhiyun
369*4882a593Smuzhiyun module_i2c_driver(an30259a_driver);
370*4882a593Smuzhiyun
371*4882a593Smuzhiyun MODULE_AUTHOR("Simon Shields <simon@lineageos.org>");
372*4882a593Smuzhiyun MODULE_DESCRIPTION("AN30259A LED driver");
373*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
374