2 * LED Driver for Dialog DA9052 PMICs.
4 * Copyright(c) 2012 Dialog Semiconductor Ltd.
6 * Author: David Dajun Chen <dchen@diasemi.com>
8 * This program is free software; you can redistribute it and/or modify it
9 * under the terms of the GNU General Public License as published by the
10 * Free Software Foundation; either version 2 of the License, or (at your
11 * option) any later version.
15 #include <linux/module.h>
16 #include <linux/kernel.h>
17 #include <linux/platform_device.h>
18 #include <linux/leds.h>
19 #include <linux/workqueue.h>
20 #include <linux/slab.h>
22 #include <linux/mfd/da9052/reg.h>
23 #include <linux/mfd/da9052/da9052.h>
24 #include <linux/mfd/da9052/pdata.h>
26 #define DA9052_OPENDRAIN_OUTPUT 2
27 #define DA9052_SET_HIGH_LVL_OUTPUT (1 << 3)
28 #define DA9052_MASK_UPPER_NIBBLE 0xF0
29 #define DA9052_MASK_LOWER_NIBBLE 0x0F
30 #define DA9052_NIBBLE_SHIFT 4
31 #define DA9052_MAX_BRIGHTNESS 0x5f
34 struct led_classdev cdev
;
35 struct work_struct work
;
36 struct da9052
*da9052
;
37 unsigned char led_index
;
42 static unsigned char led_reg
[] = {
43 DA9052_LED_CONT_4_REG
,
44 DA9052_LED_CONT_5_REG
,
47 static int da9052_set_led_brightness(struct da9052_led
*led
)
52 val
= (led
->brightness
& 0x7f) | DA9052_LED_CONT_DIM
;
54 error
= da9052_reg_write(led
->da9052
, led_reg
[led
->led_index
], val
);
56 dev_err(led
->da9052
->dev
, "Failed to set led brightness, %d\n",
61 static void da9052_led_work(struct work_struct
*work
)
63 struct da9052_led
*led
= container_of(work
, struct da9052_led
, work
);
65 da9052_set_led_brightness(led
);
68 static void da9052_led_set(struct led_classdev
*led_cdev
,
69 enum led_brightness value
)
71 struct da9052_led
*led
;
73 led
= container_of(led_cdev
, struct da9052_led
, cdev
);
74 led
->brightness
= value
;
75 schedule_work(&led
->work
);
78 static int da9052_configure_leds(struct da9052
*da9052
)
81 unsigned char register_value
= DA9052_OPENDRAIN_OUTPUT
82 | DA9052_SET_HIGH_LVL_OUTPUT
;
84 error
= da9052_reg_update(da9052
, DA9052_GPIO_14_15_REG
,
85 DA9052_MASK_LOWER_NIBBLE
,
89 dev_err(da9052
->dev
, "Failed to write GPIO 14-15 reg, %d\n",
94 error
= da9052_reg_update(da9052
, DA9052_GPIO_14_15_REG
,
95 DA9052_MASK_UPPER_NIBBLE
,
96 register_value
<< DA9052_NIBBLE_SHIFT
);
98 dev_err(da9052
->dev
, "Failed to write GPIO 14-15 reg, %d\n",
104 static int da9052_led_probe(struct platform_device
*pdev
)
106 struct da9052_pdata
*pdata
;
107 struct da9052
*da9052
;
108 struct led_platform_data
*pled
;
109 struct da9052_led
*led
= NULL
;
113 da9052
= dev_get_drvdata(pdev
->dev
.parent
);
114 pdata
= dev_get_platdata(da9052
->dev
);
116 dev_err(&pdev
->dev
, "No platform data\n");
122 dev_err(&pdev
->dev
, "No platform data for LED\n");
126 led
= devm_kzalloc(&pdev
->dev
,
127 sizeof(struct da9052_led
) * pled
->num_leds
,
134 for (i
= 0; i
< pled
->num_leds
; i
++) {
135 led
[i
].cdev
.name
= pled
->leds
[i
].name
;
136 led
[i
].cdev
.brightness_set
= da9052_led_set
;
137 led
[i
].cdev
.brightness
= LED_OFF
;
138 led
[i
].cdev
.max_brightness
= DA9052_MAX_BRIGHTNESS
;
139 led
[i
].brightness
= LED_OFF
;
140 led
[i
].led_index
= pled
->leds
[i
].flags
;
141 led
[i
].da9052
= dev_get_drvdata(pdev
->dev
.parent
);
142 INIT_WORK(&led
[i
].work
, da9052_led_work
);
144 error
= led_classdev_register(pdev
->dev
.parent
, &led
[i
].cdev
);
146 dev_err(&pdev
->dev
, "Failed to register led %d\n",
151 error
= da9052_set_led_brightness(&led
[i
]);
153 dev_err(&pdev
->dev
, "Unable to init led %d\n",
158 error
= da9052_configure_leds(led
->da9052
);
160 dev_err(&pdev
->dev
, "Failed to configure GPIO LED%d\n", error
);
164 platform_set_drvdata(pdev
, led
);
169 for (i
= i
- 1; i
>= 0; i
--) {
170 led_classdev_unregister(&led
[i
].cdev
);
171 cancel_work_sync(&led
[i
].work
);
177 static int da9052_led_remove(struct platform_device
*pdev
)
179 struct da9052_led
*led
= platform_get_drvdata(pdev
);
180 struct da9052_pdata
*pdata
;
181 struct da9052
*da9052
;
182 struct led_platform_data
*pled
;
185 da9052
= dev_get_drvdata(pdev
->dev
.parent
);
186 pdata
= dev_get_platdata(da9052
->dev
);
189 for (i
= 0; i
< pled
->num_leds
; i
++) {
190 led
[i
].brightness
= 0;
191 da9052_set_led_brightness(&led
[i
]);
192 led_classdev_unregister(&led
[i
].cdev
);
193 cancel_work_sync(&led
[i
].work
);
199 static struct platform_driver da9052_led_driver
= {
201 .name
= "da9052-leds",
202 .owner
= THIS_MODULE
,
204 .probe
= da9052_led_probe
,
205 .remove
= da9052_led_remove
,
208 module_platform_driver(da9052_led_driver
);
210 MODULE_AUTHOR("Dialog Semiconductor Ltd <dchen@diasemi.com>");
211 MODULE_DESCRIPTION("LED driver for Dialog DA9052 PMIC");
212 MODULE_LICENSE("GPL");