blob: f638bc91d9efce02e9fe860cd1a1a06e94a249af [file] [log] [blame]
Fenglin Wu3b507ad2017-11-13 10:46:02 +08001/* Copyright (c) 2018, The Linux Foundation. All rights reserved.
2 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
13#include <linux/bitops.h>
14#include <linux/device.h>
15#include <linux/err.h>
16#include <linux/init.h>
17#include <linux/kernel.h>
18#include <linux/leds.h>
19#include <linux/module.h>
20#include <linux/mutex.h>
21#include <linux/of.h>
22#include <linux/of_address.h>
23#include <linux/platform_device.h>
24#include <linux/pwm.h>
25#include <linux/regmap.h>
26#include <linux/types.h>
27
28#define TRILED_REG_TYPE 0x04
29#define TRILED_REG_SUBTYPE 0x05
30#define TRILED_REG_EN_CTL 0x46
31
32/* TRILED_REG_EN_CTL */
33#define TRILED_EN_CTL_MASK GENMASK(7, 5)
34#define TRILED_EN_CTL_MAX_BIT 7
35
36#define TRILED_TYPE 0x19
37#define TRILED_SUBTYPE_LED3H0L12 0x02
38#define TRILED_SUBTYPE_LED2H0L12 0x03
39#define TRILED_SUBTYPE_LED1H2L12 0x04
40
41#define TRILED_NUM_MAX 3
42
43#define PWM_PERIOD_DEFAULT_NS 1000000
Fenglin Wu3b507ad2017-11-13 10:46:02 +080044
45struct pwm_setting {
46 u32 pre_period_ns;
47 u32 period_ns;
48 u32 duty_ns;
49};
50
51struct led_setting {
52 u32 on_ms;
53 u32 off_ms;
54 enum led_brightness brightness;
55 bool blink;
56};
57
58struct qpnp_led_dev {
59 struct led_classdev cdev;
60 struct pwm_device *pwm_dev;
61 struct pwm_setting pwm_setting;
62 struct led_setting led_setting;
63 struct qpnp_tri_led_chip *chip;
64 struct mutex lock;
65 const char *label;
66 const char *default_trigger;
67 u8 id;
68 bool blinking;
69};
70
71struct qpnp_tri_led_chip {
72 struct device *dev;
73 struct regmap *regmap;
74 struct qpnp_led_dev *leds;
75 struct mutex bus_lock;
76 int num_leds;
77 u16 reg_base;
78 u8 subtype;
79};
80
81static int qpnp_tri_led_read(struct qpnp_tri_led_chip *chip, u16 addr, u8 *val)
82{
83 int rc;
84 unsigned int tmp;
85
86 mutex_lock(&chip->bus_lock);
87 rc = regmap_read(chip->regmap, chip->reg_base + addr, &tmp);
88 if (rc < 0)
89 dev_err(chip->dev, "Read addr 0x%x failed, rc=%d\n", addr, rc);
90 else
91 *val = (u8)tmp;
92 mutex_unlock(&chip->bus_lock);
93
94 return rc;
95}
96
97static int qpnp_tri_led_masked_write(struct qpnp_tri_led_chip *chip,
98 u16 addr, u8 mask, u8 val)
99{
100 int rc;
101
102 mutex_lock(&chip->bus_lock);
103 rc = regmap_update_bits(chip->regmap, chip->reg_base + addr, mask, val);
104 if (rc < 0)
105 dev_err(chip->dev, "Update addr 0x%x to val 0x%x with mask 0x%x failed, rc=%d\n",
106 addr, val, mask, rc);
107 mutex_unlock(&chip->bus_lock);
108
109 return rc;
110}
111
112static int __tri_led_config_pwm(struct qpnp_led_dev *led,
113 struct pwm_setting *pwm)
114{
115 struct pwm_state pstate;
116 int rc;
117
118 pwm_get_state(led->pwm_dev, &pstate);
119 pstate.enabled = !!(pwm->duty_ns != 0);
120 pstate.period = pwm->period_ns;
121 pstate.duty_cycle = pwm->duty_ns;
122 rc = pwm_apply_state(led->pwm_dev, &pstate);
123
124 if (rc < 0)
125 dev_err(led->chip->dev, "Apply PWM state for %s led failed, rc=%d\n",
126 led->cdev.name, rc);
127
128 return rc;
129}
130
131static int __tri_led_set(struct qpnp_led_dev *led)
132{
133 int rc = 0;
134 u8 val = 0, mask = 0;
135
136 rc = __tri_led_config_pwm(led, &led->pwm_setting);
137 if (rc < 0) {
138 dev_err(led->chip->dev, "Configure PWM for %s led failed, rc=%d\n",
139 led->cdev.name, rc);
140 return rc;
141 }
142
143 mask |= 1 << (TRILED_EN_CTL_MAX_BIT - led->id);
144
145 if (led->pwm_setting.duty_ns == 0)
146 val = 0;
147 else
148 val = mask;
149
150 rc = qpnp_tri_led_masked_write(led->chip, TRILED_REG_EN_CTL,
151 mask, val);
152 if (rc < 0)
153 dev_err(led->chip->dev, "Update addr 0x%x failed, rc=%d\n",
154 TRILED_REG_EN_CTL, rc);
155
156 return rc;
157}
158
159static int qpnp_tri_led_set(struct qpnp_led_dev *led)
160{
161 u32 on_ms, off_ms, period_ns, duty_ns;
162 enum led_brightness brightness = led->led_setting.brightness;
163 int rc = 0;
164
165 if (led->led_setting.blink) {
166 on_ms = led->led_setting.on_ms;
167 off_ms = led->led_setting.off_ms;
168 if (on_ms > INT_MAX / NSEC_PER_MSEC)
169 duty_ns = INT_MAX - 1;
170 else
171 duty_ns = on_ms * NSEC_PER_MSEC;
172
173 if (on_ms + off_ms > INT_MAX / NSEC_PER_MSEC) {
174 period_ns = INT_MAX;
175 duty_ns = (period_ns / (on_ms + off_ms)) * on_ms;
176 } else {
177 period_ns = (on_ms + off_ms) * NSEC_PER_MSEC;
178 }
179
180 if (period_ns < duty_ns && duty_ns != 0)
181 period_ns = duty_ns + 1;
182 } else {
183 /* Use initial period if no blinking is required */
184 period_ns = led->pwm_setting.pre_period_ns;
185
186 if (period_ns > INT_MAX / brightness)
187 duty_ns = (period_ns / LED_FULL) * brightness;
188 else
189 duty_ns = (period_ns * brightness) / LED_FULL;
190
191 if (period_ns < duty_ns && duty_ns != 0)
192 period_ns = duty_ns + 1;
193 }
194 dev_dbg(led->chip->dev, "PWM settings for %s led: period = %dns, duty = %dns\n",
195 led->cdev.name, period_ns, duty_ns);
196
197 led->pwm_setting.duty_ns = duty_ns;
198 led->pwm_setting.period_ns = period_ns;
199
200 rc = __tri_led_set(led);
201 if (rc < 0) {
202 dev_err(led->chip->dev, "__tri_led_set %s failed, rc=%d\n",
203 led->cdev.name, rc);
204 return rc;
205 }
206
207 if (led->led_setting.blink) {
208 led->cdev.brightness = LED_FULL;
209 led->blinking = true;
210 } else {
211 led->cdev.brightness = led->led_setting.brightness;
212 led->blinking = false;
213 }
214
215 return rc;
216}
217
218static int qpnp_tri_led_set_brightness(struct led_classdev *led_cdev,
219 enum led_brightness brightness)
220{
221 struct qpnp_led_dev *led =
222 container_of(led_cdev, struct qpnp_led_dev, cdev);
223 int rc = 0;
224
225 mutex_lock(&led->lock);
226 if (brightness > LED_FULL)
227 brightness = LED_FULL;
228
229 if (brightness == led->led_setting.brightness &&
230 !led->blinking) {
231 mutex_unlock(&led->lock);
232 return 0;
233 }
234
235 led->led_setting.brightness = brightness;
236 if (!!brightness)
237 led->led_setting.off_ms = 0;
238 else
239 led->led_setting.on_ms = 0;
240 led->led_setting.blink = false;
241
242 rc = qpnp_tri_led_set(led);
243 if (rc)
244 dev_err(led->chip->dev, "Set led failed for %s, rc=%d\n",
245 led->label, rc);
246
247 mutex_unlock(&led->lock);
248
249 return rc;
250}
251
252static enum led_brightness qpnp_tri_led_get_brightness(
253 struct led_classdev *led_cdev)
254{
255 return led_cdev->brightness;
256}
257
258static int qpnp_tri_led_set_blink(struct led_classdev *led_cdev,
259 unsigned long *on_ms, unsigned long *off_ms)
260{
261 struct qpnp_led_dev *led =
262 container_of(led_cdev, struct qpnp_led_dev, cdev);
263 int rc = 0;
264
265 mutex_lock(&led->lock);
266 if (led->blinking && *on_ms == led->led_setting.on_ms &&
267 *off_ms == led->led_setting.off_ms) {
268 dev_dbg(led_cdev->dev, "Ignore, on/off setting is not changed: on %lums, off %lums\n",
269 *on_ms, *off_ms);
270 mutex_unlock(&led->lock);
271 return 0;
272 }
273
274 if (*on_ms == 0) {
275 led->led_setting.blink = false;
276 led->led_setting.brightness = LED_OFF;
277 } else if (*off_ms == 0) {
278 led->led_setting.blink = false;
279 led->led_setting.brightness = led->cdev.brightness;
280 } else {
281 led->led_setting.on_ms = *on_ms;
282 led->led_setting.off_ms = *off_ms;
283 led->led_setting.blink = true;
284 }
285
286 rc = qpnp_tri_led_set(led);
287 if (rc)
288 dev_err(led->chip->dev, "Set led failed for %s, rc=%d\n",
289 led->label, rc);
290
291 mutex_unlock(&led->lock);
292 return rc;
293}
294
295static int qpnp_tri_led_register(struct qpnp_tri_led_chip *chip)
296{
297 struct qpnp_led_dev *led;
298 int rc, i, j;
299
300 for (i = 0; i < chip->num_leds; i++) {
301 led = &chip->leds[i];
302 mutex_init(&led->lock);
303 led->cdev.name = led->label;
304 led->cdev.max_brightness = LED_FULL;
305 led->cdev.brightness_set_blocking = qpnp_tri_led_set_brightness;
306 led->cdev.brightness_get = qpnp_tri_led_get_brightness;
307 led->cdev.blink_set = qpnp_tri_led_set_blink;
308 led->cdev.default_trigger = led->default_trigger;
309 led->cdev.brightness = LED_OFF;
Fenglin Wu00699452018-02-28 09:21:38 +0800310 led->cdev.flags |= LED_KEEP_TRIGGER;
Fenglin Wu3b507ad2017-11-13 10:46:02 +0800311
312 rc = devm_led_classdev_register(chip->dev, &led->cdev);
313 if (rc < 0) {
314 dev_err(chip->dev, "%s led class device registering failed, rc=%d\n",
315 led->label, rc);
316 goto destroy;
317 }
318 }
319
320 return 0;
321destroy:
322 for (j = 0; j <= i; j++)
323 mutex_destroy(&chip->leds[i].lock);
324
325 return rc;
326}
327
328static int qpnp_tri_led_hw_init(struct qpnp_tri_led_chip *chip)
329{
330 int rc = 0;
331 u8 val;
332
333 rc = qpnp_tri_led_read(chip, TRILED_REG_TYPE, &val);
334 if (rc < 0) {
335 dev_err(chip->dev, "Read REG_TYPE failed, rc=%d\n", rc);
336 return rc;
337 }
338
339 if (val != TRILED_TYPE) {
340 dev_err(chip->dev, "invalid subtype(%d)\n", val);
341 return -ENODEV;
342 }
343
344 rc = qpnp_tri_led_read(chip, TRILED_REG_SUBTYPE, &val);
345 if (rc < 0) {
346 dev_err(chip->dev, "Read REG_SUBTYPE failed, rc=%d\n", rc);
347 return rc;
348 }
349
350 chip->subtype = val;
351
352 return 0;
353}
354
355static int qpnp_tri_led_parse_dt(struct qpnp_tri_led_chip *chip)
356{
357 struct device_node *node = chip->dev->of_node, *child_node;
358 struct qpnp_led_dev *led;
359 struct pwm_args pargs;
360 const __be32 *addr;
Kiran Gunda84d7c9e2018-03-15 18:03:06 +0530361 int rc = 0, id, i = 0;
Fenglin Wu3b507ad2017-11-13 10:46:02 +0800362
363 addr = of_get_address(chip->dev->of_node, 0, NULL, NULL);
364 if (!addr) {
365 dev_err(chip->dev, "Getting address failed\n");
366 return -EINVAL;
367 }
368 chip->reg_base = be32_to_cpu(addr[0]);
369
370 chip->num_leds = of_get_available_child_count(node);
371 if (chip->num_leds == 0) {
372 dev_err(chip->dev, "No led child node defined\n");
373 return -ENODEV;
374 }
375
376 if (chip->num_leds > TRILED_NUM_MAX) {
377 dev_err(chip->dev, "can't support %d leds(max %d)\n",
378 chip->num_leds, TRILED_NUM_MAX);
379 return -EINVAL;
380 }
381
382 chip->leds = devm_kcalloc(chip->dev, chip->num_leds,
383 sizeof(struct qpnp_led_dev), GFP_KERNEL);
384 if (!chip->leds)
385 return -ENOMEM;
386
387 for_each_available_child_of_node(node, child_node) {
388 rc = of_property_read_u32(child_node, "led-sources", &id);
389 if (rc) {
390 dev_err(chip->dev, "Get led-sources failed, rc=%d\n",
391 rc);
392 return rc;
393 }
394
395 if (id >= TRILED_NUM_MAX) {
396 dev_err(chip->dev, "only support 0~%d current source\n",
397 TRILED_NUM_MAX - 1);
398 return -EINVAL;
399 }
400
401 led = &chip->leds[i++];
402 led->chip = chip;
403 led->id = id;
404 led->label =
405 of_get_property(child_node, "label", NULL) ? :
406 child_node->name;
407
408 led->pwm_dev =
409 devm_of_pwm_get(chip->dev, child_node, NULL);
410 if (IS_ERR(led->pwm_dev)) {
411 rc = PTR_ERR(led->pwm_dev);
412 if (rc != -EPROBE_DEFER)
413 dev_err(chip->dev, "Get pwm device for %s led failed, rc=%d\n",
414 led->label, rc);
415 return rc;
416 }
417
418 pwm_get_args(led->pwm_dev, &pargs);
419 if (pargs.period == 0)
420 led->pwm_setting.pre_period_ns = PWM_PERIOD_DEFAULT_NS;
421 else
422 led->pwm_setting.pre_period_ns = pargs.period;
423
424 led->default_trigger = of_get_property(child_node,
425 "linux,default-trigger", NULL);
426 }
427
428 return rc;
429}
430
431static int qpnp_tri_led_probe(struct platform_device *pdev)
432{
433 struct qpnp_tri_led_chip *chip;
434 int rc = 0;
435
436 chip = devm_kzalloc(&pdev->dev, sizeof(*chip), GFP_KERNEL);
437 if (!chip)
438 return -ENOMEM;
439
440 chip->dev = &pdev->dev;
441 chip->regmap = dev_get_regmap(chip->dev->parent, NULL);
442 if (!chip->regmap) {
443 dev_err(chip->dev, "Getting regmap failed\n");
444 return -EINVAL;
445 }
446
447 rc = qpnp_tri_led_parse_dt(chip);
448 if (rc < 0) {
449 dev_err(chip->dev, "Devicetree properties parsing failed, rc=%d\n",
450 rc);
451 return rc;
452 }
453
454 mutex_init(&chip->bus_lock);
455
456 rc = qpnp_tri_led_hw_init(chip);
457 if (rc) {
458 dev_err(chip->dev, "HW initialization failed, rc=%d\n", rc);
459 goto destroy;
460 }
461
462 dev_set_drvdata(chip->dev, chip);
463 rc = qpnp_tri_led_register(chip);
464 if (rc < 0) {
465 dev_err(chip->dev, "Registering LED class devices failed, rc=%d\n",
466 rc);
467 goto destroy;
468 }
469
470 dev_dbg(chip->dev, "Tri-led module with subtype 0x%x is detected\n",
471 chip->subtype);
472 return 0;
473destroy:
474 mutex_destroy(&chip->bus_lock);
475 dev_set_drvdata(chip->dev, NULL);
476
477 return rc;
478}
479
480static int qpnp_tri_led_remove(struct platform_device *pdev)
481{
482 int i;
483 struct qpnp_tri_led_chip *chip = dev_get_drvdata(&pdev->dev);
484
485 mutex_destroy(&chip->bus_lock);
486 for (i = 0; i < chip->num_leds; i++)
487 mutex_destroy(&chip->leds[i].lock);
488 dev_set_drvdata(chip->dev, NULL);
489 return 0;
490}
491
492static const struct of_device_id qpnp_tri_led_of_match[] = {
493 { .compatible = "qcom,tri-led",},
494 { },
495};
496
497static struct platform_driver qpnp_tri_led_driver = {
498 .driver = {
499 .name = "qcom,tri-led",
500 .of_match_table = qpnp_tri_led_of_match,
501 },
502 .probe = qpnp_tri_led_probe,
503 .remove = qpnp_tri_led_remove,
504};
505module_platform_driver(qpnp_tri_led_driver);
506
507MODULE_DESCRIPTION("QTI TRI_LED driver");
508MODULE_LICENSE("GPL v2");
509MODULE_ALIAS("leds:qpnp-tri-led");