blob: c3a55647080908dd68878e6e0a8730e5659c554f [file] [log] [blame]
Amy Malochee7d7ef32013-01-30 16:16:35 -08001/* Copyright (c) 2010-2013, The Linux Foundation. All rights reserved.
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002 *
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
Jay Chokshi12e49bf2011-07-22 16:24:39 -070013#define pr_fmt(fmt) "%s: " fmt, __func__
14
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070015#include <linux/kernel.h>
Steve Mucklef132c6c2012-06-06 18:30:57 -070016#include <linux/module.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070017#include <linux/init.h>
18#include <linux/slab.h>
19#include <linux/platform_device.h>
20#include <linux/leds.h>
21#include <linux/workqueue.h>
Jay Chokshi868312e2011-09-16 13:57:13 -070022#include <linux/err.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070023
24#include <linux/mfd/pm8xxx/core.h>
Jay Chokshi868312e2011-09-16 13:57:13 -070025#include <linux/mfd/pm8xxx/pwm.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070026#include <linux/leds-pm8xxx.h>
27
28#define SSBI_REG_ADDR_DRV_KEYPAD 0x48
29#define PM8XXX_DRV_KEYPAD_BL_MASK 0xf0
30#define PM8XXX_DRV_KEYPAD_BL_SHIFT 0x04
31
32#define SSBI_REG_ADDR_FLASH_DRV0 0x49
33#define PM8XXX_DRV_FLASH_MASK 0xf0
34#define PM8XXX_DRV_FLASH_SHIFT 0x04
35
36#define SSBI_REG_ADDR_FLASH_DRV1 0xFB
37
38#define SSBI_REG_ADDR_LED_CTRL_BASE 0x131
39#define SSBI_REG_ADDR_LED_CTRL(n) (SSBI_REG_ADDR_LED_CTRL_BASE + (n))
40#define PM8XXX_DRV_LED_CTRL_MASK 0xf8
41#define PM8XXX_DRV_LED_CTRL_SHIFT 0x03
42
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053043#define SSBI_REG_ADDR_WLED_CTRL_BASE 0x25A
44#define SSBI_REG_ADDR_WLED_CTRL(n) (SSBI_REG_ADDR_WLED_CTRL_BASE + (n) - 1)
45
46/* wled control registers */
47#define WLED_MOD_CTRL_REG SSBI_REG_ADDR_WLED_CTRL(1)
48#define WLED_MAX_CURR_CFG_REG(n) SSBI_REG_ADDR_WLED_CTRL(n + 2)
Mohan Pallaka58ff4052012-11-19 16:58:59 +053049#define WLED_BRIGHTNESS_CNTL_REG1(n) SSBI_REG_ADDR_WLED_CTRL((2 * n) + 5)
50#define WLED_BRIGHTNESS_CNTL_REG2(n) SSBI_REG_ADDR_WLED_CTRL((2 * n) + 6)
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053051#define WLED_SYNC_REG SSBI_REG_ADDR_WLED_CTRL(11)
52#define WLED_OVP_CFG_REG SSBI_REG_ADDR_WLED_CTRL(13)
53#define WLED_BOOST_CFG_REG SSBI_REG_ADDR_WLED_CTRL(14)
54#define WLED_HIGH_POLE_CAP_REG SSBI_REG_ADDR_WLED_CTRL(16)
55
Amy Maloche59cf8182012-11-30 12:27:40 -080056#define WLED_STRING_ONE 0 /* Rightmost string */
57#define WLED_STRING_TWO 1 /* Middle string */
58#define WLED_STRING_THREE 2 /* Leftmost string */
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053059#define WLED_STRINGS 0x03
60#define WLED_OVP_VAL_MASK 0x30
61#define WLED_OVP_VAL_BIT_SHFT 0x04
62#define WLED_BOOST_LIMIT_MASK 0xE0
63#define WLED_BOOST_LIMIT_BIT_SHFT 0x05
Amy Maloche56913f52012-05-11 10:47:24 -070064#define WLED_BOOST_OFF 0x00
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053065#define WLED_EN_MASK 0x01
66#define WLED_CP_SELECT_MAX 0x03
67#define WLED_CP_SELECT_MASK 0x03
68#define WLED_DIG_MOD_GEN_MASK 0x70
69#define WLED_CS_OUT_MASK 0x0E
70#define WLED_CTL_DLY_STEP 200
71#define WLED_CTL_DLY_MAX 1400
72#define WLED_CTL_DLY_MASK 0xE0
73#define WLED_CTL_DLY_BIT_SHFT 0x05
74#define WLED_MAX_CURR 25
75#define WLED_MAX_CURR_MASK 0x1F
Amy Malochebe73e192012-11-29 12:53:59 -080076#define WLED_BRIGHTNESS_MSB_MASK 0x0F
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053077#define WLED_OP_FDBCK_MASK 0x1C
78#define WLED_OP_FDBCK_BIT_SHFT 0x02
79
Chandan Uddaraju6e73f0a2012-03-08 17:32:55 -080080#define WLED_MAX_LEVEL 255
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053081#define WLED_8_BIT_MASK 0xFF
82#define WLED_8_BIT_SHFT 0x08
83#define WLED_MAX_DUTY_CYCLE 0xFFF
84
85#define WLED_SYNC_VAL 0x07
86#define WLED_SYNC_RESET_VAL 0x00
Amy Malochec366d5b2012-08-28 11:06:40 -070087#define WLED_SYNC_MASK 0xF8
88
89#define ONE_WLED_STRING 1
90#define TWO_WLED_STRINGS 2
91#define THREE_WLED_STRINGS 3
92
Amy Malochec366d5b2012-08-28 11:06:40 -070093#define WLED_CABC_SHIFT 3
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053094
Amy Malochec17c3732012-02-27 18:34:07 -080095#define SSBI_REG_ADDR_RGB_CNTL1 0x12D
96#define SSBI_REG_ADDR_RGB_CNTL2 0x12E
97
98#define PM8XXX_DRV_RGB_RED_LED BIT(2)
99#define PM8XXX_DRV_RGB_GREEN_LED BIT(1)
100#define PM8XXX_DRV_RGB_BLUE_LED BIT(0)
101
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700102#define MAX_FLASH_LED_CURRENT 300
103#define MAX_LC_LED_CURRENT 40
104#define MAX_KP_BL_LED_CURRENT 300
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700105
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700106#define PM8XXX_ID_LED_CURRENT_FACTOR 2 /* Iout = x * 2mA */
107#define PM8XXX_ID_FLASH_CURRENT_FACTOR 20 /* Iout = x * 20mA */
108
109#define PM8XXX_FLASH_MODE_DBUS1 1
110#define PM8XXX_FLASH_MODE_DBUS2 2
111#define PM8XXX_FLASH_MODE_PWM 3
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700112
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700113#define MAX_LC_LED_BRIGHTNESS 20
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700114#define MAX_FLASH_BRIGHTNESS 15
115#define MAX_KB_LED_BRIGHTNESS 15
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700116
Jay Chokshide4cefb2011-08-04 18:10:44 -0700117#define PM8XXX_LED_OFFSET(id) ((id) - PM8XXX_ID_LED_0)
118
Jay Chokshi868312e2011-09-16 13:57:13 -0700119#define PM8XXX_LED_PWM_FLAGS (PM_PWM_LUT_LOOP | PM_PWM_LUT_RAMP_UP)
120
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700121#define LED_MAP(_version, _kb, _led0, _led1, _led2, _flash_led0, _flash_led1, \
Amy Malochec17c3732012-02-27 18:34:07 -0800122 _wled, _rgb_led_red, _rgb_led_green, _rgb_led_blue)\
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700123 {\
124 .version = _version,\
125 .supported = _kb << PM8XXX_ID_LED_KB_LIGHT | \
126 _led0 << PM8XXX_ID_LED_0 | _led1 << PM8XXX_ID_LED_1 | \
127 _led2 << PM8XXX_ID_LED_2 | \
128 _flash_led0 << PM8XXX_ID_FLASH_LED_0 | \
129 _flash_led1 << PM8XXX_ID_FLASH_LED_1 | \
Amy Malochec17c3732012-02-27 18:34:07 -0800130 _wled << PM8XXX_ID_WLED | \
131 _rgb_led_red << PM8XXX_ID_RGB_LED_RED | \
132 _rgb_led_green << PM8XXX_ID_RGB_LED_GREEN | \
133 _rgb_led_blue << PM8XXX_ID_RGB_LED_BLUE, \
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700134 }
135
136/**
137 * supported_leds - leds supported for each PMIC version
138 * @version - version of PMIC
139 * @supported - which leds are supported on version
140 */
141
142struct supported_leds {
143 enum pm8xxx_version version;
144 u32 supported;
145};
146
147static const struct supported_leds led_map[] = {
Amy Malochec17c3732012-02-27 18:34:07 -0800148 LED_MAP(PM8XXX_VERSION_8058, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
149 LED_MAP(PM8XXX_VERSION_8921, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
150 LED_MAP(PM8XXX_VERSION_8018, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0),
151 LED_MAP(PM8XXX_VERSION_8922, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1),
152 LED_MAP(PM8XXX_VERSION_8038, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1),
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700153};
154
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700155/**
156 * struct pm8xxx_led_data - internal led data structure
157 * @led_classdev - led class device
158 * @id - led index
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700159 * @work - workqueue for led
160 * @lock - to protect the transactions
161 * @reg - cached value of led register
Jay Chokshi868312e2011-09-16 13:57:13 -0700162 * @pwm_dev - pointer to PWM device if LED is driven using PWM
163 * @pwm_channel - PWM channel ID
164 * @pwm_period_us - PWM period in micro seconds
165 * @pwm_duty_cycles - struct that describes PWM duty cycles info
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700166 */
167struct pm8xxx_led_data {
168 struct led_classdev cdev;
169 int id;
170 u8 reg;
Amy Maloche56913f52012-05-11 10:47:24 -0700171 u8 wled_mod_ctrl_val;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700172 struct device *dev;
173 struct work_struct work;
174 struct mutex lock;
Jay Chokshi868312e2011-09-16 13:57:13 -0700175 struct pwm_device *pwm_dev;
176 int pwm_channel;
177 u32 pwm_period_us;
178 struct pm8xxx_pwm_duty_cycles *pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530179 struct wled_config_data *wled_cfg;
180 int max_current;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700181};
182
183static void led_kp_set(struct pm8xxx_led_data *led, enum led_brightness value)
184{
185 int rc;
186 u8 level;
187
188 level = (value << PM8XXX_DRV_KEYPAD_BL_SHIFT) &
189 PM8XXX_DRV_KEYPAD_BL_MASK;
190
191 led->reg &= ~PM8XXX_DRV_KEYPAD_BL_MASK;
192 led->reg |= level;
193
194 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_DRV_KEYPAD,
195 led->reg);
196 if (rc < 0)
197 dev_err(led->cdev.dev,
198 "can't set keypad backlight level rc=%d\n", rc);
199}
200
201static void led_lc_set(struct pm8xxx_led_data *led, enum led_brightness value)
202{
203 int rc, offset;
204 u8 level;
205
206 level = (value << PM8XXX_DRV_LED_CTRL_SHIFT) &
207 PM8XXX_DRV_LED_CTRL_MASK;
208
209 offset = PM8XXX_LED_OFFSET(led->id);
210
211 led->reg &= ~PM8XXX_DRV_LED_CTRL_MASK;
212 led->reg |= level;
213
214 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_LED_CTRL(offset),
215 led->reg);
216 if (rc)
217 dev_err(led->cdev.dev, "can't set (%d) led value rc=%d\n",
218 led->id, rc);
219}
220
221static void
222led_flash_set(struct pm8xxx_led_data *led, enum led_brightness value)
223{
224 int rc;
225 u8 level;
226 u16 reg_addr;
227
228 level = (value << PM8XXX_DRV_FLASH_SHIFT) &
229 PM8XXX_DRV_FLASH_MASK;
230
231 led->reg &= ~PM8XXX_DRV_FLASH_MASK;
232 led->reg |= level;
233
234 if (led->id == PM8XXX_ID_FLASH_LED_0)
235 reg_addr = SSBI_REG_ADDR_FLASH_DRV0;
236 else
237 reg_addr = SSBI_REG_ADDR_FLASH_DRV1;
238
239 rc = pm8xxx_writeb(led->dev->parent, reg_addr, led->reg);
240 if (rc < 0)
241 dev_err(led->cdev.dev, "can't set flash led%d level rc=%d\n",
242 led->id, rc);
243}
244
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530245static int
246led_wled_set(struct pm8xxx_led_data *led, enum led_brightness value)
247{
248 int rc, duty;
Amy Malochee7d7ef32013-01-30 16:16:35 -0800249 u8 val, i;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530250
251 if (value > WLED_MAX_LEVEL)
252 value = WLED_MAX_LEVEL;
253
Amy Maloche56913f52012-05-11 10:47:24 -0700254 if (value == 0) {
255 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
256 WLED_BOOST_OFF);
257 if (rc) {
258 dev_err(led->dev->parent, "can't write wled ctrl config"
259 " register rc=%d\n", rc);
260 return rc;
261 }
262 } else {
263 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
264 led->wled_mod_ctrl_val);
265 if (rc) {
266 dev_err(led->dev->parent, "can't write wled ctrl config"
267 " register rc=%d\n", rc);
268 return rc;
269 }
270 }
271
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530272 duty = (WLED_MAX_DUTY_CYCLE * value) / WLED_MAX_LEVEL;
273
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530274 /* program brightness control registers */
Amy Malochee7d7ef32013-01-30 16:16:35 -0800275 for (i = 0; i < WLED_STRINGS; i++) {
276 if (led->wled_cfg->strings && (1 << i)) {
277 rc = pm8xxx_readb(led->dev->parent,
278 WLED_BRIGHTNESS_CNTL_REG1(i), &val);
279 if (rc) {
280 dev_err(led->dev->parent,
281 "can't read wled brightnes ctrl"
282 " register1 rc=%d\n", rc);
283 return rc;
284 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530285
Amy Malochee7d7ef32013-01-30 16:16:35 -0800286 val = (val & ~WLED_MAX_CURR_MASK) |
287 (duty >> WLED_8_BIT_SHFT);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530288
Amy Malochee7d7ef32013-01-30 16:16:35 -0800289 rc = pm8xxx_writeb(led->dev->parent,
290 WLED_BRIGHTNESS_CNTL_REG1(i), val);
291 if (rc) {
292 dev_err(led->dev->parent,
293 "can't write wled brightness ctrl"
294 " register1 rc=%d\n", rc);
295 return rc;
296 }
297
298 val = duty & WLED_8_BIT_MASK;
299 rc = pm8xxx_writeb(led->dev->parent,
300 WLED_BRIGHTNESS_CNTL_REG2(i), val);
301 if (rc) {
302 dev_err(led->dev->parent,
303 "can't write wled brightness ctrl"
304 " register2 rc=%d\n", rc);
305 return rc;
306 }
307 } else
308 continue;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530309 }
Amy Malochec366d5b2012-08-28 11:06:40 -0700310 rc = pm8xxx_readb(led->dev->parent, WLED_SYNC_REG, &val);
311 if (rc) {
312 dev_err(led->dev->parent,
313 "can't read wled sync register rc=%d\n", rc);
314 return rc;
315 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530316 /* sync */
Amy Malochec366d5b2012-08-28 11:06:40 -0700317 val &= WLED_SYNC_MASK;
318 val |= WLED_SYNC_VAL;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530319 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
320 if (rc) {
321 dev_err(led->dev->parent,
322 "can't read wled sync register rc=%d\n", rc);
323 return rc;
324 }
Amy Malochec366d5b2012-08-28 11:06:40 -0700325 val &= WLED_SYNC_MASK;
326 val |= WLED_SYNC_RESET_VAL;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530327 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
328 if (rc) {
329 dev_err(led->dev->parent,
330 "can't read wled sync register rc=%d\n", rc);
331 return rc;
332 }
333 return 0;
334}
335
336static void wled_dump_regs(struct pm8xxx_led_data *led)
337{
338 int i;
339 u8 val;
340
341 for (i = 1; i < 17; i++) {
342 pm8xxx_readb(led->dev->parent,
343 SSBI_REG_ADDR_WLED_CTRL(i), &val);
344 pr_debug("WLED_CTRL_%d = 0x%x\n", i, val);
345 }
346}
347
Amy Malochec17c3732012-02-27 18:34:07 -0800348static void
Amy Malochead3c7842012-07-20 14:51:27 -0700349led_rgb_write(struct pm8xxx_led_data *led, u16 addr, enum led_brightness value)
Amy Malochec17c3732012-02-27 18:34:07 -0800350{
351 int rc;
352 u8 val, mask;
353
Amy Malochead3c7842012-07-20 14:51:27 -0700354 if (led->id != PM8XXX_ID_RGB_LED_BLUE &&
355 led->id != PM8XXX_ID_RGB_LED_RED &&
356 led->id != PM8XXX_ID_RGB_LED_GREEN)
357 return;
358
359 rc = pm8xxx_readb(led->dev->parent, addr, &val);
Amy Malochec17c3732012-02-27 18:34:07 -0800360 if (rc) {
361 dev_err(led->cdev.dev, "can't read rgb ctrl register rc=%d\n",
362 rc);
363 return;
364 }
365
366 switch (led->id) {
367 case PM8XXX_ID_RGB_LED_RED:
368 mask = PM8XXX_DRV_RGB_RED_LED;
369 break;
370 case PM8XXX_ID_RGB_LED_GREEN:
371 mask = PM8XXX_DRV_RGB_GREEN_LED;
372 break;
373 case PM8XXX_ID_RGB_LED_BLUE:
374 mask = PM8XXX_DRV_RGB_BLUE_LED;
375 break;
376 default:
377 return;
378 }
379
380 if (value)
381 val |= mask;
382 else
383 val &= ~mask;
384
Amy Malochead3c7842012-07-20 14:51:27 -0700385 rc = pm8xxx_writeb(led->dev->parent, addr, val);
Amy Malochec17c3732012-02-27 18:34:07 -0800386 if (rc < 0)
387 dev_err(led->cdev.dev, "can't set rgb led %d level rc=%d\n",
388 led->id, rc);
389}
390
Amy Malochead3c7842012-07-20 14:51:27 -0700391static void
392led_rgb_set(struct pm8xxx_led_data *led, enum led_brightness value)
393{
394 if (value) {
395 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, value);
396 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL2, value);
397 } else {
398 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL2, value);
399 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, value);
400 }
401}
402
Jay Chokshi868312e2011-09-16 13:57:13 -0700403static int pm8xxx_led_pwm_work(struct pm8xxx_led_data *led)
404{
405 int duty_us;
406 int rc = 0;
407
408 if (led->pwm_duty_cycles == NULL) {
409 duty_us = (led->pwm_period_us * led->cdev.brightness) /
410 LED_FULL;
411 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
Amy Malochead3c7842012-07-20 14:51:27 -0700412 if (led->cdev.brightness) {
413 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
414 led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700415 rc = pwm_enable(led->pwm_dev);
Amy Malochead3c7842012-07-20 14:51:27 -0700416 } else {
Jay Chokshi868312e2011-09-16 13:57:13 -0700417 pwm_disable(led->pwm_dev);
Amy Malochead3c7842012-07-20 14:51:27 -0700418 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
419 led->cdev.brightness);
420 }
Jay Chokshi868312e2011-09-16 13:57:13 -0700421 } else {
Amy Malochead3c7842012-07-20 14:51:27 -0700422 if (led->cdev.brightness)
423 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
424 led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700425 rc = pm8xxx_pwm_lut_enable(led->pwm_dev, led->cdev.brightness);
Amy Malochead3c7842012-07-20 14:51:27 -0700426 if (!led->cdev.brightness)
427 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
428 led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700429 }
430
431 return rc;
432}
433
Jay Chokshide4cefb2011-08-04 18:10:44 -0700434static void __pm8xxx_led_work(struct pm8xxx_led_data *led,
435 enum led_brightness level)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700436{
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530437 int rc;
438
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700439 mutex_lock(&led->lock);
440
441 switch (led->id) {
442 case PM8XXX_ID_LED_KB_LIGHT:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700443 led_kp_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530444 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700445 case PM8XXX_ID_LED_0:
446 case PM8XXX_ID_LED_1:
447 case PM8XXX_ID_LED_2:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700448 led_lc_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530449 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700450 case PM8XXX_ID_FLASH_LED_0:
451 case PM8XXX_ID_FLASH_LED_1:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700452 led_flash_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530453 break;
454 case PM8XXX_ID_WLED:
455 rc = led_wled_set(led, level);
456 if (rc < 0)
457 pr_err("wled brightness set failed %d\n", rc);
458 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800459 case PM8XXX_ID_RGB_LED_RED:
460 case PM8XXX_ID_RGB_LED_GREEN:
461 case PM8XXX_ID_RGB_LED_BLUE:
462 led_rgb_set(led, level);
463 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530464 default:
465 dev_err(led->cdev.dev, "unknown led id %d", led->id);
466 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700467 }
468
469 mutex_unlock(&led->lock);
470}
471
Jay Chokshide4cefb2011-08-04 18:10:44 -0700472static void pm8xxx_led_work(struct work_struct *work)
473{
Jay Chokshi868312e2011-09-16 13:57:13 -0700474 int rc;
475
Jay Chokshide4cefb2011-08-04 18:10:44 -0700476 struct pm8xxx_led_data *led = container_of(work,
477 struct pm8xxx_led_data, work);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700478
Jay Chokshi868312e2011-09-16 13:57:13 -0700479 if (led->pwm_dev == NULL) {
480 __pm8xxx_led_work(led, led->cdev.brightness);
481 } else {
482 rc = pm8xxx_led_pwm_work(led);
483 if (rc)
484 pr_err("could not configure PWM mode for LED:%d\n",
485 led->id);
486 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700487}
488
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700489static void pm8xxx_led_set(struct led_classdev *led_cdev,
490 enum led_brightness value)
491{
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700492 struct pm8xxx_led_data *led;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700493
494 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
495
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700496 if (value < LED_OFF || value > led->cdev.max_brightness) {
497 dev_err(led->cdev.dev, "Invalid brightness value exceeds");
498 return;
499 }
500
501 led->cdev.brightness = value;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700502 schedule_work(&led->work);
503}
504
Jay Chokshide4cefb2011-08-04 18:10:44 -0700505static int pm8xxx_set_led_mode_and_max_brightness(struct pm8xxx_led_data *led,
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700506 enum pm8xxx_led_modes led_mode, int max_current)
507{
Jay Chokshide4cefb2011-08-04 18:10:44 -0700508 switch (led->id) {
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700509 case PM8XXX_ID_LED_0:
510 case PM8XXX_ID_LED_1:
511 case PM8XXX_ID_LED_2:
512 led->cdev.max_brightness = max_current /
513 PM8XXX_ID_LED_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700514 if (led->cdev.max_brightness > MAX_LC_LED_BRIGHTNESS)
515 led->cdev.max_brightness = MAX_LC_LED_BRIGHTNESS;
Willie Ruana39d1d42011-08-19 13:54:34 -0700516 led->reg = led_mode;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700517 break;
518 case PM8XXX_ID_LED_KB_LIGHT:
519 case PM8XXX_ID_FLASH_LED_0:
520 case PM8XXX_ID_FLASH_LED_1:
521 led->cdev.max_brightness = max_current /
522 PM8XXX_ID_FLASH_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700523 if (led->cdev.max_brightness > MAX_FLASH_BRIGHTNESS)
524 led->cdev.max_brightness = MAX_FLASH_BRIGHTNESS;
525
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700526 switch (led_mode) {
527 case PM8XXX_LED_MODE_PWM1:
528 case PM8XXX_LED_MODE_PWM2:
529 case PM8XXX_LED_MODE_PWM3:
Willie Ruana39d1d42011-08-19 13:54:34 -0700530 led->reg = PM8XXX_FLASH_MODE_PWM;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700531 break;
532 case PM8XXX_LED_MODE_DTEST1:
Willie Ruana39d1d42011-08-19 13:54:34 -0700533 led->reg = PM8XXX_FLASH_MODE_DBUS1;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700534 break;
535 case PM8XXX_LED_MODE_DTEST2:
Willie Ruana39d1d42011-08-19 13:54:34 -0700536 led->reg = PM8XXX_FLASH_MODE_DBUS2;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700537 break;
538 default:
Willie Ruana39d1d42011-08-19 13:54:34 -0700539 led->reg = PM8XXX_LED_MODE_MANUAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700540 break;
541 }
542 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530543 case PM8XXX_ID_WLED:
544 led->cdev.max_brightness = WLED_MAX_LEVEL;
545 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800546 case PM8XXX_ID_RGB_LED_RED:
547 case PM8XXX_ID_RGB_LED_GREEN:
548 case PM8XXX_ID_RGB_LED_BLUE:
549 led->cdev.max_brightness = LED_FULL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700550 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800551 default:
552 dev_err(led->cdev.dev, "LED Id is invalid");
553 return -EINVAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700554 }
555
Amy Malochec17c3732012-02-27 18:34:07 -0800556 return 0;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700557}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700558
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700559static enum led_brightness pm8xxx_led_get(struct led_classdev *led_cdev)
560{
561 struct pm8xxx_led_data *led;
562
563 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
564
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700565 return led->cdev.brightness;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700566}
567
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530568static int __devinit init_wled(struct pm8xxx_led_data *led)
569{
570 int rc, i;
Amy Maloche59cf8182012-11-30 12:27:40 -0800571 u8 val, string_max_current;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530572
573 /* program over voltage protection threshold */
574 if (led->wled_cfg->ovp_val > WLED_OVP_37V) {
575 dev_err(led->dev->parent, "Invalid ovp value");
576 return -EINVAL;
577 }
578
579 rc = pm8xxx_readb(led->dev->parent, WLED_OVP_CFG_REG, &val);
580 if (rc) {
581 dev_err(led->dev->parent, "can't read wled ovp config"
582 " register rc=%d\n", rc);
583 return rc;
584 }
585
586 val = (val & ~WLED_OVP_VAL_MASK) |
587 (led->wled_cfg->ovp_val << WLED_OVP_VAL_BIT_SHFT);
588
589 rc = pm8xxx_writeb(led->dev->parent, WLED_OVP_CFG_REG, val);
590 if (rc) {
591 dev_err(led->dev->parent, "can't write wled ovp config"
592 " register rc=%d\n", rc);
593 return rc;
594 }
595
596 /* program current boost limit and output feedback*/
597 if (led->wled_cfg->boost_curr_lim > WLED_CURR_LIMIT_1680mA) {
598 dev_err(led->dev->parent, "Invalid boost current limit");
599 return -EINVAL;
600 }
601
602 rc = pm8xxx_readb(led->dev->parent, WLED_BOOST_CFG_REG, &val);
603 if (rc) {
604 dev_err(led->dev->parent, "can't read wled boost config"
605 " register rc=%d\n", rc);
606 return rc;
607 }
608
609 val = (val & ~WLED_BOOST_LIMIT_MASK) |
610 (led->wled_cfg->boost_curr_lim << WLED_BOOST_LIMIT_BIT_SHFT);
611
612 val = (val & ~WLED_OP_FDBCK_MASK) |
613 (led->wled_cfg->op_fdbck << WLED_OP_FDBCK_BIT_SHFT);
614
615 rc = pm8xxx_writeb(led->dev->parent, WLED_BOOST_CFG_REG, val);
616 if (rc) {
617 dev_err(led->dev->parent, "can't write wled boost config"
618 " register rc=%d\n", rc);
619 return rc;
620 }
621
622 /* program high pole capacitance */
623 if (led->wled_cfg->cp_select > WLED_CP_SELECT_MAX) {
624 dev_err(led->dev->parent, "Invalid pole capacitance");
625 return -EINVAL;
626 }
627
628 rc = pm8xxx_readb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, &val);
629 if (rc) {
630 dev_err(led->dev->parent, "can't read wled high pole"
631 " capacitance register rc=%d\n", rc);
632 return rc;
633 }
634
635 val = (val & ~WLED_CP_SELECT_MASK) | led->wled_cfg->cp_select;
636
637 rc = pm8xxx_writeb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, val);
638 if (rc) {
639 dev_err(led->dev->parent, "can't write wled high pole"
640 " capacitance register rc=%d\n", rc);
641 return rc;
642 }
643
644 /* program activation delay and maximum current */
Amy Malochee7d7ef32013-01-30 16:16:35 -0800645 for (i = 0; i < WLED_STRINGS; i++) {
646 if (led->wled_cfg->strings && (1 << i)) {
647 rc = pm8xxx_readb(led->dev->parent,
Amy Maloche59cf8182012-11-30 12:27:40 -0800648 WLED_MAX_CURR_CFG_REG(i), &val);
Amy Malochee7d7ef32013-01-30 16:16:35 -0800649 if (rc) {
650 dev_err(led->dev->parent,
651 "can't read wled max current"
652 " config register rc=%d\n", rc);
653 return rc;
654 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530655
Amy Malochee7d7ef32013-01-30 16:16:35 -0800656 if ((led->wled_cfg->ctrl_delay_us % WLED_CTL_DLY_STEP)
657 || (led->wled_cfg->ctrl_delay_us >
658 WLED_CTL_DLY_MAX)) {
659 dev_err(led->dev->parent,
660 "Invalid control delay\n");
661 return rc;
662 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530663
Amy Malochee7d7ef32013-01-30 16:16:35 -0800664 val = val / WLED_CTL_DLY_STEP;
665 val = (val & ~WLED_CTL_DLY_MASK) |
666 (led->wled_cfg->ctrl_delay_us <<
667 WLED_CTL_DLY_BIT_SHFT);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530668
Amy Malochee7d7ef32013-01-30 16:16:35 -0800669 if ((led->max_current > WLED_MAX_CURR)) {
670 dev_err(led->dev->parent,
671 "Invalid max current\n");
672 return -EINVAL;
673 }
Amy Maloche59cf8182012-11-30 12:27:40 -0800674 if (led->wled_cfg->max_current_ind) {
675 switch (i) {
676 case WLED_STRING_ONE:
677 string_max_current = led->wled_cfg->max_one;
678 break;
679 case WLED_STRING_TWO:
680 string_max_current = led->wled_cfg->max_two;
681 break;
682 case WLED_STRING_THREE:
683 string_max_current = led->wled_cfg->max_three;
684 break;
685 default:
686 return -EINVAL;
687 }
688 val = (val & ~WLED_MAX_CURR_MASK) | string_max_current;
689 } else
Amy Malochee7d7ef32013-01-30 16:16:35 -0800690 val = (val & ~WLED_MAX_CURR_MASK) | led->max_current;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530691
Amy Malochee7d7ef32013-01-30 16:16:35 -0800692 rc = pm8xxx_writeb(led->dev->parent,
Amy Maloche59cf8182012-11-30 12:27:40 -0800693 WLED_MAX_CURR_CFG_REG(i), val);
Amy Malochee7d7ef32013-01-30 16:16:35 -0800694 if (rc) {
695 dev_err(led->dev->parent,
696 "can't write wled max current"
697 " config register rc=%d\n", rc);
698 return rc;
699 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530700 }
701 }
702
Amy Malochec366d5b2012-08-28 11:06:40 -0700703 if (led->wled_cfg->cabc_en) {
704 rc = pm8xxx_readb(led->dev->parent, WLED_SYNC_REG, &val);
705 if (rc) {
706 dev_err(led->dev->parent,
707 "can't read cabc register rc=%d\n", rc);
708 return rc;
709 }
710
Amy Malochee7d7ef32013-01-30 16:16:35 -0800711 val |= (led->wled_cfg->strings << WLED_CABC_SHIFT);
Amy Malochec366d5b2012-08-28 11:06:40 -0700712
713 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
714 if (rc) {
715 dev_err(led->dev->parent,
716 "can't write to enable cabc rc=%d\n", rc);
717 return rc;
718 }
719 }
720
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530721 /* program digital module generator, cs out and enable the module */
722 rc = pm8xxx_readb(led->dev->parent, WLED_MOD_CTRL_REG, &val);
723 if (rc) {
724 dev_err(led->dev->parent, "can't read wled module ctrl"
725 " register rc=%d\n", rc);
726 return rc;
727 }
728
729 if (led->wled_cfg->dig_mod_gen_en)
730 val |= WLED_DIG_MOD_GEN_MASK;
731
732 if (led->wled_cfg->cs_out_en)
733 val |= WLED_CS_OUT_MASK;
734
735 val |= WLED_EN_MASK;
736
737 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG, val);
738 if (rc) {
739 dev_err(led->dev->parent, "can't write wled module ctrl"
740 " register rc=%d\n", rc);
741 return rc;
742 }
Amy Maloche56913f52012-05-11 10:47:24 -0700743 led->wled_mod_ctrl_val = val;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530744
745 /* dump wled registers */
746 wled_dump_regs(led);
747
748 return 0;
749}
750
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700751static int __devinit get_init_value(struct pm8xxx_led_data *led, u8 *val)
752{
753 int rc, offset;
754 u16 addr;
755
756 switch (led->id) {
757 case PM8XXX_ID_LED_KB_LIGHT:
758 addr = SSBI_REG_ADDR_DRV_KEYPAD;
759 break;
760 case PM8XXX_ID_LED_0:
761 case PM8XXX_ID_LED_1:
762 case PM8XXX_ID_LED_2:
763 offset = PM8XXX_LED_OFFSET(led->id);
764 addr = SSBI_REG_ADDR_LED_CTRL(offset);
765 break;
766 case PM8XXX_ID_FLASH_LED_0:
767 addr = SSBI_REG_ADDR_FLASH_DRV0;
768 break;
769 case PM8XXX_ID_FLASH_LED_1:
770 addr = SSBI_REG_ADDR_FLASH_DRV1;
771 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530772 case PM8XXX_ID_WLED:
773 rc = init_wled(led);
774 if (rc)
775 dev_err(led->cdev.dev, "can't initialize wled rc=%d\n",
776 rc);
777 return rc;
Amy Malochec17c3732012-02-27 18:34:07 -0800778 case PM8XXX_ID_RGB_LED_RED:
779 case PM8XXX_ID_RGB_LED_GREEN:
780 case PM8XXX_ID_RGB_LED_BLUE:
Amy Malochec17c3732012-02-27 18:34:07 -0800781 addr = SSBI_REG_ADDR_RGB_CNTL1;
782 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530783 default:
784 dev_err(led->cdev.dev, "unknown led id %d", led->id);
785 return -EINVAL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700786 }
787
788 rc = pm8xxx_readb(led->dev->parent, addr, val);
789 if (rc)
790 dev_err(led->cdev.dev, "can't get led(%d) level rc=%d\n",
791 led->id, rc);
792
793 return rc;
794}
795
Jay Chokshi868312e2011-09-16 13:57:13 -0700796static int pm8xxx_led_pwm_configure(struct pm8xxx_led_data *led)
797{
798 int start_idx, idx_len, duty_us, rc;
799
800 led->pwm_dev = pwm_request(led->pwm_channel,
801 led->cdev.name);
802
803 if (IS_ERR_OR_NULL(led->pwm_dev)) {
804 pr_err("could not acquire PWM Channel %d, "
805 "error %ld\n", led->pwm_channel,
806 PTR_ERR(led->pwm_dev));
807 led->pwm_dev = NULL;
808 return -ENODEV;
809 }
810
811 if (led->pwm_duty_cycles != NULL) {
812 start_idx = led->pwm_duty_cycles->start_idx;
813 idx_len = led->pwm_duty_cycles->num_duty_pcts;
814
815 if (idx_len >= PM_PWM_LUT_SIZE && start_idx) {
816 pr_err("Wrong LUT size or index\n");
817 return -EINVAL;
818 }
819 if ((start_idx + idx_len) > PM_PWM_LUT_SIZE) {
820 pr_err("Exceed LUT limit\n");
821 return -EINVAL;
822 }
823
824 rc = pm8xxx_pwm_lut_config(led->pwm_dev, led->pwm_period_us,
825 led->pwm_duty_cycles->duty_pcts,
826 led->pwm_duty_cycles->duty_ms,
827 start_idx, idx_len, 0, 0,
828 PM8XXX_LED_PWM_FLAGS);
829 } else {
830 duty_us = led->pwm_period_us;
831 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
832 }
833
834 return rc;
835}
836
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530837
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700838static int __devinit pm8xxx_led_probe(struct platform_device *pdev)
839{
Jay Chokshi8994e392011-09-14 18:20:39 -0700840 const struct pm8xxx_led_platform_data *pdata = pdev->dev.platform_data;
841 const struct led_platform_data *pcore_data;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700842 struct led_info *curr_led;
843 struct pm8xxx_led_data *led, *led_dat;
Jay Chokshi8994e392011-09-14 18:20:39 -0700844 struct pm8xxx_led_config *led_cfg;
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700845 enum pm8xxx_version version;
846 bool found = false;
847 int rc, i, j;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700848
849 if (pdata == NULL) {
850 dev_err(&pdev->dev, "platform data not supplied\n");
851 return -EINVAL;
852 }
853
Jay Chokshi8994e392011-09-14 18:20:39 -0700854 pcore_data = pdata->led_core;
855
856 if (pcore_data->num_leds != pdata->num_configs) {
857 dev_err(&pdev->dev, "#no. of led configs and #no. of led"
858 "entries are not equal\n");
859 return -EINVAL;
860 }
861
862 led = kcalloc(pcore_data->num_leds, sizeof(*led), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700863 if (led == NULL) {
864 dev_err(&pdev->dev, "failed to alloc memory\n");
865 return -ENOMEM;
866 }
867
Jay Chokshi8994e392011-09-14 18:20:39 -0700868 for (i = 0; i < pcore_data->num_leds; i++) {
869 curr_led = &pcore_data->leds[i];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700870 led_dat = &led[i];
Jay Chokshi8994e392011-09-14 18:20:39 -0700871 led_cfg = &pdata->configs[i];
872
873 led_dat->id = led_cfg->id;
Jay Chokshi868312e2011-09-16 13:57:13 -0700874 led_dat->pwm_channel = led_cfg->pwm_channel;
875 led_dat->pwm_period_us = led_cfg->pwm_period_us;
876 led_dat->pwm_duty_cycles = led_cfg->pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530877 led_dat->wled_cfg = led_cfg->wled_cfg;
878 led_dat->max_current = led_cfg->max_current;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700879
880 if (!((led_dat->id >= PM8XXX_ID_LED_KB_LIGHT) &&
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530881 (led_dat->id < PM8XXX_ID_MAX))) {
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700882 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
883 led_dat->id);
884 rc = -EINVAL;
885 goto fail_id_check;
886
887 }
888
889 found = false;
890 version = pm8xxx_get_version(pdev->dev.parent);
891 for (j = 0; j < ARRAY_SIZE(led_map); j++) {
892 if (version == led_map[j].version
893 && (led_map[j].supported & (1 << led_dat->id))) {
894 found = true;
895 break;
896 }
897 }
898
899 if (!found) {
900 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
901 led_dat->id);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700902 rc = -EINVAL;
903 goto fail_id_check;
904 }
905
906 led_dat->cdev.name = curr_led->name;
907 led_dat->cdev.default_trigger = curr_led->default_trigger;
908 led_dat->cdev.brightness_set = pm8xxx_led_set;
909 led_dat->cdev.brightness_get = pm8xxx_led_get;
910 led_dat->cdev.brightness = LED_OFF;
Jay Chokshi8994e392011-09-14 18:20:39 -0700911 led_dat->cdev.flags = curr_led->flags;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700912 led_dat->dev = &pdev->dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700913
914 rc = get_init_value(led_dat, &led_dat->reg);
915 if (rc < 0)
916 goto fail_id_check;
917
Jay Chokshi8994e392011-09-14 18:20:39 -0700918 rc = pm8xxx_set_led_mode_and_max_brightness(led_dat,
919 led_cfg->mode, led_cfg->max_current);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700920 if (rc < 0)
921 goto fail_id_check;
922
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700923 mutex_init(&led_dat->lock);
924 INIT_WORK(&led_dat->work, pm8xxx_led_work);
925
926 rc = led_classdev_register(&pdev->dev, &led_dat->cdev);
927 if (rc) {
928 dev_err(&pdev->dev, "unable to register led %d,rc=%d\n",
929 led_dat->id, rc);
930 goto fail_id_check;
931 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700932
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530933 /* configure default state */
934 if (led_cfg->default_state)
935 led->cdev.brightness = led_dat->cdev.max_brightness;
936 else
937 led->cdev.brightness = LED_OFF;
938
Jay Chokshi868312e2011-09-16 13:57:13 -0700939 if (led_cfg->mode != PM8XXX_LED_MODE_MANUAL) {
Amy Malochec17c3732012-02-27 18:34:07 -0800940 if (led_dat->id == PM8XXX_ID_RGB_LED_RED ||
941 led_dat->id == PM8XXX_ID_RGB_LED_GREEN ||
942 led_dat->id == PM8XXX_ID_RGB_LED_BLUE)
943 __pm8xxx_led_work(led_dat, 0);
944 else
945 __pm8xxx_led_work(led_dat,
Jay Chokshide4cefb2011-08-04 18:10:44 -0700946 led_dat->cdev.max_brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700947
948 if (led_dat->pwm_channel != -1) {
949 led_dat->cdev.max_brightness = LED_FULL;
950 rc = pm8xxx_led_pwm_configure(led_dat);
951 if (rc) {
952 dev_err(&pdev->dev, "failed to "
953 "configure LED, error: %d\n", rc);
954 goto fail_id_check;
955 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530956 schedule_work(&led->work);
Jay Chokshi868312e2011-09-16 13:57:13 -0700957 }
958 } else {
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530959 __pm8xxx_led_work(led_dat, led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700960 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700961 }
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700962
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700963 platform_set_drvdata(pdev, led);
964
965 return 0;
966
967fail_id_check:
968 if (i > 0) {
969 for (i = i - 1; i >= 0; i--) {
970 mutex_destroy(&led[i].lock);
971 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -0700972 if (led[i].pwm_dev != NULL)
973 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700974 }
975 }
976 kfree(led);
977 return rc;
978}
979
980static int __devexit pm8xxx_led_remove(struct platform_device *pdev)
981{
982 int i;
983 const struct led_platform_data *pdata =
984 pdev->dev.platform_data;
985 struct pm8xxx_led_data *led = platform_get_drvdata(pdev);
986
987 for (i = 0; i < pdata->num_leds; i++) {
988 cancel_work_sync(&led[i].work);
989 mutex_destroy(&led[i].lock);
990 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -0700991 if (led[i].pwm_dev != NULL)
992 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700993 }
994
995 kfree(led);
996
997 return 0;
998}
999
1000static struct platform_driver pm8xxx_led_driver = {
1001 .probe = pm8xxx_led_probe,
1002 .remove = __devexit_p(pm8xxx_led_remove),
1003 .driver = {
1004 .name = PM8XXX_LEDS_DEV_NAME,
1005 .owner = THIS_MODULE,
1006 },
1007};
1008
1009static int __init pm8xxx_led_init(void)
1010{
1011 return platform_driver_register(&pm8xxx_led_driver);
1012}
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001013subsys_initcall(pm8xxx_led_init);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001014
1015static void __exit pm8xxx_led_exit(void)
1016{
1017 platform_driver_unregister(&pm8xxx_led_driver);
1018}
1019module_exit(pm8xxx_led_exit);
1020
1021MODULE_DESCRIPTION("PM8XXX LEDs driver");
1022MODULE_LICENSE("GPL v2");
1023MODULE_VERSION("1.0");
1024MODULE_ALIAS("platform:pm8xxx-led");