blob: 5e0651927649ad28f66b0200596092e133403028 [file] [log] [blame]
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301/* Copyright (c) 2010-2012, Code Aurora Forum. 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>
samin.ryue0041de2012-08-03 23:58:14 +090023#include <linux/ctype.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070024
25#include <linux/mfd/pm8xxx/core.h>
Jay Chokshi868312e2011-09-16 13:57:13 -070026#include <linux/mfd/pm8xxx/pwm.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070027#include <linux/leds-pm8xxx.h>
28
29#define SSBI_REG_ADDR_DRV_KEYPAD 0x48
30#define PM8XXX_DRV_KEYPAD_BL_MASK 0xf0
31#define PM8XXX_DRV_KEYPAD_BL_SHIFT 0x04
32
33#define SSBI_REG_ADDR_FLASH_DRV0 0x49
34#define PM8XXX_DRV_FLASH_MASK 0xf0
35#define PM8XXX_DRV_FLASH_SHIFT 0x04
36
37#define SSBI_REG_ADDR_FLASH_DRV1 0xFB
38
39#define SSBI_REG_ADDR_LED_CTRL_BASE 0x131
40#define SSBI_REG_ADDR_LED_CTRL(n) (SSBI_REG_ADDR_LED_CTRL_BASE + (n))
41#define PM8XXX_DRV_LED_CTRL_MASK 0xf8
42#define PM8XXX_DRV_LED_CTRL_SHIFT 0x03
43
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053044#define SSBI_REG_ADDR_WLED_CTRL_BASE 0x25A
45#define SSBI_REG_ADDR_WLED_CTRL(n) (SSBI_REG_ADDR_WLED_CTRL_BASE + (n) - 1)
46
47/* wled control registers */
48#define WLED_MOD_CTRL_REG SSBI_REG_ADDR_WLED_CTRL(1)
49#define WLED_MAX_CURR_CFG_REG(n) SSBI_REG_ADDR_WLED_CTRL(n + 2)
50#define WLED_BRIGHTNESS_CNTL_REG1(n) SSBI_REG_ADDR_WLED_CTRL(n + 5)
51#define WLED_BRIGHTNESS_CNTL_REG2(n) SSBI_REG_ADDR_WLED_CTRL(n + 6)
52#define WLED_SYNC_REG SSBI_REG_ADDR_WLED_CTRL(11)
53#define WLED_OVP_CFG_REG SSBI_REG_ADDR_WLED_CTRL(13)
54#define WLED_BOOST_CFG_REG SSBI_REG_ADDR_WLED_CTRL(14)
55#define WLED_HIGH_POLE_CAP_REG SSBI_REG_ADDR_WLED_CTRL(16)
56
57#define WLED_STRINGS 0x03
58#define WLED_OVP_VAL_MASK 0x30
59#define WLED_OVP_VAL_BIT_SHFT 0x04
60#define WLED_BOOST_LIMIT_MASK 0xE0
61#define WLED_BOOST_LIMIT_BIT_SHFT 0x05
Amy Maloche56913f52012-05-11 10:47:24 -070062#define WLED_BOOST_OFF 0x00
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053063#define WLED_EN_MASK 0x01
64#define WLED_CP_SELECT_MAX 0x03
65#define WLED_CP_SELECT_MASK 0x03
66#define WLED_DIG_MOD_GEN_MASK 0x70
67#define WLED_CS_OUT_MASK 0x0E
68#define WLED_CTL_DLY_STEP 200
69#define WLED_CTL_DLY_MAX 1400
70#define WLED_CTL_DLY_MASK 0xE0
71#define WLED_CTL_DLY_BIT_SHFT 0x05
72#define WLED_MAX_CURR 25
73#define WLED_MAX_CURR_MASK 0x1F
74#define WLED_OP_FDBCK_MASK 0x1C
75#define WLED_OP_FDBCK_BIT_SHFT 0x02
76
Chandan Uddaraju6e73f0a2012-03-08 17:32:55 -080077#define WLED_MAX_LEVEL 255
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053078#define WLED_8_BIT_MASK 0xFF
79#define WLED_8_BIT_SHFT 0x08
80#define WLED_MAX_DUTY_CYCLE 0xFFF
81
82#define WLED_SYNC_VAL 0x07
83#define WLED_SYNC_RESET_VAL 0x00
84
Amy Malochec17c3732012-02-27 18:34:07 -080085#define SSBI_REG_ADDR_RGB_CNTL1 0x12D
86#define SSBI_REG_ADDR_RGB_CNTL2 0x12E
87
88#define PM8XXX_DRV_RGB_RED_LED BIT(2)
89#define PM8XXX_DRV_RGB_GREEN_LED BIT(1)
90#define PM8XXX_DRV_RGB_BLUE_LED BIT(0)
91
Jay Chokshi12e49bf2011-07-22 16:24:39 -070092#define MAX_FLASH_LED_CURRENT 300
93#define MAX_LC_LED_CURRENT 40
94#define MAX_KP_BL_LED_CURRENT 300
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070095
Jay Chokshi12e49bf2011-07-22 16:24:39 -070096#define PM8XXX_ID_LED_CURRENT_FACTOR 2 /* Iout = x * 2mA */
97#define PM8XXX_ID_FLASH_CURRENT_FACTOR 20 /* Iout = x * 20mA */
98
99#define PM8XXX_FLASH_MODE_DBUS1 1
100#define PM8XXX_FLASH_MODE_DBUS2 2
101#define PM8XXX_FLASH_MODE_PWM 3
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700102
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700103#define MAX_LC_LED_BRIGHTNESS 20
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700104#define MAX_FLASH_BRIGHTNESS 15
105#define MAX_KB_LED_BRIGHTNESS 15
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700106
Jay Chokshide4cefb2011-08-04 18:10:44 -0700107#define PM8XXX_LED_OFFSET(id) ((id) - PM8XXX_ID_LED_0)
108
samin.ryue0041de2012-08-03 23:58:14 +0900109#define PM8XXX_LED_PWM_FLAGS (PM_PWM_LUT_LOOP | PM_PWM_LUT_RAMP_UP | PM_PWM_LUT_REVERSE)
Jay Chokshi868312e2011-09-16 13:57:13 -0700110
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700111#define LED_MAP(_version, _kb, _led0, _led1, _led2, _flash_led0, _flash_led1, \
Amy Malochec17c3732012-02-27 18:34:07 -0800112 _wled, _rgb_led_red, _rgb_led_green, _rgb_led_blue)\
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700113 {\
114 .version = _version,\
115 .supported = _kb << PM8XXX_ID_LED_KB_LIGHT | \
116 _led0 << PM8XXX_ID_LED_0 | _led1 << PM8XXX_ID_LED_1 | \
117 _led2 << PM8XXX_ID_LED_2 | \
118 _flash_led0 << PM8XXX_ID_FLASH_LED_0 | \
119 _flash_led1 << PM8XXX_ID_FLASH_LED_1 | \
Amy Malochec17c3732012-02-27 18:34:07 -0800120 _wled << PM8XXX_ID_WLED | \
121 _rgb_led_red << PM8XXX_ID_RGB_LED_RED | \
122 _rgb_led_green << PM8XXX_ID_RGB_LED_GREEN | \
123 _rgb_led_blue << PM8XXX_ID_RGB_LED_BLUE, \
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700124 }
125
126/**
127 * supported_leds - leds supported for each PMIC version
128 * @version - version of PMIC
129 * @supported - which leds are supported on version
130 */
131
132struct supported_leds {
133 enum pm8xxx_version version;
134 u32 supported;
135};
136
137static const struct supported_leds led_map[] = {
Amy Malochec17c3732012-02-27 18:34:07 -0800138 LED_MAP(PM8XXX_VERSION_8058, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
139 LED_MAP(PM8XXX_VERSION_8921, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
140 LED_MAP(PM8XXX_VERSION_8018, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0),
141 LED_MAP(PM8XXX_VERSION_8922, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1),
142 LED_MAP(PM8XXX_VERSION_8038, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1),
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700143};
144
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700145/**
146 * struct pm8xxx_led_data - internal led data structure
147 * @led_classdev - led class device
148 * @id - led index
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700149 * @work - workqueue for led
150 * @lock - to protect the transactions
151 * @reg - cached value of led register
Jay Chokshi868312e2011-09-16 13:57:13 -0700152 * @pwm_dev - pointer to PWM device if LED is driven using PWM
153 * @pwm_channel - PWM channel ID
154 * @pwm_period_us - PWM period in micro seconds
155 * @pwm_duty_cycles - struct that describes PWM duty cycles info
samin.ryue0041de2012-08-03 23:58:14 +0900156 * @use_pwm - controlled by userspace
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700157 */
158struct pm8xxx_led_data {
159 struct led_classdev cdev;
160 int id;
161 u8 reg;
Amy Maloche56913f52012-05-11 10:47:24 -0700162 u8 wled_mod_ctrl_val;
samin.ryue0041de2012-08-03 23:58:14 +0900163 u8 lock_update;
164 u8 blink;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700165 struct device *dev;
166 struct work_struct work;
167 struct mutex lock;
Jay Chokshi868312e2011-09-16 13:57:13 -0700168 struct pwm_device *pwm_dev;
169 int pwm_channel;
170 u32 pwm_period_us;
171 struct pm8xxx_pwm_duty_cycles *pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530172 struct wled_config_data *wled_cfg;
173 int max_current;
samin.ryue0041de2012-08-03 23:58:14 +0900174 int use_pwm;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700175};
176
177static void led_kp_set(struct pm8xxx_led_data *led, enum led_brightness value)
178{
179 int rc;
180 u8 level;
181
182 level = (value << PM8XXX_DRV_KEYPAD_BL_SHIFT) &
183 PM8XXX_DRV_KEYPAD_BL_MASK;
184
185 led->reg &= ~PM8XXX_DRV_KEYPAD_BL_MASK;
186 led->reg |= level;
187
188 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_DRV_KEYPAD,
189 led->reg);
190 if (rc < 0)
191 dev_err(led->cdev.dev,
192 "can't set keypad backlight level rc=%d\n", rc);
193}
194
195static void led_lc_set(struct pm8xxx_led_data *led, enum led_brightness value)
196{
197 int rc, offset;
198 u8 level;
199
200 level = (value << PM8XXX_DRV_LED_CTRL_SHIFT) &
201 PM8XXX_DRV_LED_CTRL_MASK;
202
203 offset = PM8XXX_LED_OFFSET(led->id);
204
205 led->reg &= ~PM8XXX_DRV_LED_CTRL_MASK;
206 led->reg |= level;
207
208 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_LED_CTRL(offset),
209 led->reg);
210 if (rc)
211 dev_err(led->cdev.dev, "can't set (%d) led value rc=%d\n",
212 led->id, rc);
213}
214
215static void
216led_flash_set(struct pm8xxx_led_data *led, enum led_brightness value)
217{
218 int rc;
219 u8 level;
220 u16 reg_addr;
221
222 level = (value << PM8XXX_DRV_FLASH_SHIFT) &
223 PM8XXX_DRV_FLASH_MASK;
224
225 led->reg &= ~PM8XXX_DRV_FLASH_MASK;
226 led->reg |= level;
227
228 if (led->id == PM8XXX_ID_FLASH_LED_0)
229 reg_addr = SSBI_REG_ADDR_FLASH_DRV0;
230 else
231 reg_addr = SSBI_REG_ADDR_FLASH_DRV1;
232
233 rc = pm8xxx_writeb(led->dev->parent, reg_addr, led->reg);
234 if (rc < 0)
235 dev_err(led->cdev.dev, "can't set flash led%d level rc=%d\n",
236 led->id, rc);
237}
238
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530239static int
240led_wled_set(struct pm8xxx_led_data *led, enum led_brightness value)
241{
242 int rc, duty;
243 u8 val, i, num_wled_strings;
244
245 if (value > WLED_MAX_LEVEL)
246 value = WLED_MAX_LEVEL;
247
Amy Maloche56913f52012-05-11 10:47:24 -0700248 if (value == 0) {
249 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
250 WLED_BOOST_OFF);
251 if (rc) {
252 dev_err(led->dev->parent, "can't write wled ctrl config"
253 " register rc=%d\n", rc);
254 return rc;
255 }
256 } else {
257 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
258 led->wled_mod_ctrl_val);
259 if (rc) {
260 dev_err(led->dev->parent, "can't write wled ctrl config"
261 " register rc=%d\n", rc);
262 return rc;
263 }
264 }
265
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530266 duty = (WLED_MAX_DUTY_CYCLE * value) / WLED_MAX_LEVEL;
267
268 num_wled_strings = led->wled_cfg->num_strings;
269
270 /* program brightness control registers */
271 for (i = 0; i < num_wled_strings; i++) {
272 rc = pm8xxx_readb(led->dev->parent,
273 WLED_BRIGHTNESS_CNTL_REG1(i), &val);
274 if (rc) {
275 dev_err(led->dev->parent, "can't read wled brightnes ctrl"
276 " register1 rc=%d\n", rc);
277 return rc;
278 }
279
280 val = (val & ~WLED_MAX_CURR_MASK) | (duty >> WLED_8_BIT_SHFT);
281 rc = pm8xxx_writeb(led->dev->parent,
282 WLED_BRIGHTNESS_CNTL_REG1(i), val);
283 if (rc) {
284 dev_err(led->dev->parent, "can't write wled brightness ctrl"
285 " register1 rc=%d\n", rc);
286 return rc;
287 }
288
289 val = duty & WLED_8_BIT_MASK;
290 rc = pm8xxx_writeb(led->dev->parent,
291 WLED_BRIGHTNESS_CNTL_REG2(i), val);
292 if (rc) {
293 dev_err(led->dev->parent, "can't write wled brightness ctrl"
294 " register2 rc=%d\n", rc);
295 return rc;
296 }
297 }
298
299 /* sync */
300 val = WLED_SYNC_VAL;
301 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
302 if (rc) {
303 dev_err(led->dev->parent,
304 "can't read wled sync register rc=%d\n", rc);
305 return rc;
306 }
307
308 val = WLED_SYNC_RESET_VAL;
309 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
310 if (rc) {
311 dev_err(led->dev->parent,
312 "can't read wled sync register rc=%d\n", rc);
313 return rc;
314 }
315 return 0;
316}
317
318static void wled_dump_regs(struct pm8xxx_led_data *led)
319{
320 int i;
321 u8 val;
322
323 for (i = 1; i < 17; i++) {
324 pm8xxx_readb(led->dev->parent,
325 SSBI_REG_ADDR_WLED_CTRL(i), &val);
326 pr_debug("WLED_CTRL_%d = 0x%x\n", i, val);
327 }
328}
329
Amy Malochec17c3732012-02-27 18:34:07 -0800330static void
Amy Malochead3c7842012-07-20 14:51:27 -0700331led_rgb_write(struct pm8xxx_led_data *led, u16 addr, enum led_brightness value)
Amy Malochec17c3732012-02-27 18:34:07 -0800332{
333 int rc;
334 u8 val, mask;
335
Amy Malochead3c7842012-07-20 14:51:27 -0700336 if (led->id != PM8XXX_ID_RGB_LED_BLUE &&
337 led->id != PM8XXX_ID_RGB_LED_RED &&
338 led->id != PM8XXX_ID_RGB_LED_GREEN)
339 return;
340
341 rc = pm8xxx_readb(led->dev->parent, addr, &val);
Amy Malochec17c3732012-02-27 18:34:07 -0800342 if (rc) {
343 dev_err(led->cdev.dev, "can't read rgb ctrl register rc=%d\n",
344 rc);
345 return;
346 }
347
348 switch (led->id) {
349 case PM8XXX_ID_RGB_LED_RED:
350 mask = PM8XXX_DRV_RGB_RED_LED;
351 break;
352 case PM8XXX_ID_RGB_LED_GREEN:
353 mask = PM8XXX_DRV_RGB_GREEN_LED;
354 break;
355 case PM8XXX_ID_RGB_LED_BLUE:
356 mask = PM8XXX_DRV_RGB_BLUE_LED;
357 break;
358 default:
359 return;
360 }
361
362 if (value)
363 val |= mask;
364 else
365 val &= ~mask;
366
Amy Malochead3c7842012-07-20 14:51:27 -0700367 rc = pm8xxx_writeb(led->dev->parent, addr, val);
Amy Malochec17c3732012-02-27 18:34:07 -0800368 if (rc < 0)
369 dev_err(led->cdev.dev, "can't set rgb led %d level rc=%d\n",
370 led->id, rc);
371}
372
Amy Malochead3c7842012-07-20 14:51:27 -0700373static void
374led_rgb_set(struct pm8xxx_led_data *led, enum led_brightness value)
375{
376 if (value) {
377 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, value);
378 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL2, value);
379 } else {
380 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL2, value);
381 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, value);
382 }
383}
384
samin.ryue0041de2012-08-03 23:58:14 +0900385static int pm8xxx_led_pwm_pattern_update(struct pm8xxx_led_data * led)
386{
387 int start_idx, idx_len;
388 int *pcts = NULL;
389 int i, rc = 0;
390 int temp = 0;
391 int pwm_max = led->cdev.brightness;
392
393 if (!led->pwm_duty_cycles || !led->pwm_duty_cycles->duty_pcts) {
394 dev_err(led->cdev.dev, "duty_cycles and duty_pcts is not exist\n");
395 return -EINVAL;
396 }
397
398 start_idx = led->pwm_duty_cycles->start_idx;
399 idx_len = led->pwm_duty_cycles->num_duty_pcts;
400 pcts = led->pwm_duty_cycles->duty_pcts;
401
402 for (i = 0; i < idx_len; i++)
403 {
404 if (led->blink == 1) {
405 temp = ((pwm_max * i) << 1) / (idx_len - 1);
406 pcts[i] = (temp + 1) >> 1;
407 } else {
408 pcts[i] = (pwm_max);
409 }
410 }
411
412 if (idx_len >= PM_PWM_LUT_SIZE && start_idx) {
413 pr_err("Wrong LUT size or index\n");
414 return -EINVAL;
415 }
416 if ((start_idx + idx_len) > PM_PWM_LUT_SIZE) {
417 pr_err("Exceed LUT limit\n");
418 return -EINVAL;
419 }
420
421 rc = pm8xxx_pwm_lut_config(led->pwm_dev, led->pwm_period_us,
422 led->pwm_duty_cycles->duty_pcts,
423 led->pwm_duty_cycles->duty_ms,
424 start_idx, idx_len, 0, 0,
425 PM8XXX_LED_PWM_FLAGS);
426
427 return rc;
428}
429
Jay Chokshi868312e2011-09-16 13:57:13 -0700430static int pm8xxx_led_pwm_work(struct pm8xxx_led_data *led)
431{
432 int duty_us;
433 int rc = 0;
434
435 if (led->pwm_duty_cycles == NULL) {
436 duty_us = (led->pwm_period_us * led->cdev.brightness) /
437 LED_FULL;
438 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
Amy Malochead3c7842012-07-20 14:51:27 -0700439 if (led->cdev.brightness) {
440 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
441 led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700442 rc = pwm_enable(led->pwm_dev);
Amy Malochead3c7842012-07-20 14:51:27 -0700443 } else {
Jay Chokshi868312e2011-09-16 13:57:13 -0700444 pwm_disable(led->pwm_dev);
Amy Malochead3c7842012-07-20 14:51:27 -0700445 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
446 led->cdev.brightness);
447 }
Jay Chokshi868312e2011-09-16 13:57:13 -0700448 } else {
samin.ryue0041de2012-08-03 23:58:14 +0900449
450 if (led->cdev.brightness) {
451 pm8xxx_led_pwm_pattern_update(led);
Amy Malochead3c7842012-07-20 14:51:27 -0700452 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
453 led->cdev.brightness);
samin.ryue0041de2012-08-03 23:58:14 +0900454 }
455
Jay Chokshi868312e2011-09-16 13:57:13 -0700456 rc = pm8xxx_pwm_lut_enable(led->pwm_dev, led->cdev.brightness);
Amy Malochead3c7842012-07-20 14:51:27 -0700457 if (!led->cdev.brightness)
458 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
459 led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700460 }
461
462 return rc;
463}
464
Jay Chokshide4cefb2011-08-04 18:10:44 -0700465static void __pm8xxx_led_work(struct pm8xxx_led_data *led,
466 enum led_brightness level)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700467{
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530468 int rc;
469
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700470 mutex_lock(&led->lock);
471
472 switch (led->id) {
473 case PM8XXX_ID_LED_KB_LIGHT:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700474 led_kp_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530475 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700476 case PM8XXX_ID_LED_0:
477 case PM8XXX_ID_LED_1:
478 case PM8XXX_ID_LED_2:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700479 led_lc_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530480 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700481 case PM8XXX_ID_FLASH_LED_0:
482 case PM8XXX_ID_FLASH_LED_1:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700483 led_flash_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530484 break;
485 case PM8XXX_ID_WLED:
486 rc = led_wled_set(led, level);
487 if (rc < 0)
488 pr_err("wled brightness set failed %d\n", rc);
489 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800490 case PM8XXX_ID_RGB_LED_RED:
491 case PM8XXX_ID_RGB_LED_GREEN:
492 case PM8XXX_ID_RGB_LED_BLUE:
493 led_rgb_set(led, level);
494 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530495 default:
496 dev_err(led->cdev.dev, "unknown led id %d", led->id);
497 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700498 }
499
500 mutex_unlock(&led->lock);
501}
502
Jay Chokshide4cefb2011-08-04 18:10:44 -0700503static void pm8xxx_led_work(struct work_struct *work)
504{
Jay Chokshi868312e2011-09-16 13:57:13 -0700505 int rc;
506
Jay Chokshide4cefb2011-08-04 18:10:44 -0700507 struct pm8xxx_led_data *led = container_of(work,
508 struct pm8xxx_led_data, work);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700509
samin.ryue0041de2012-08-03 23:58:14 +0900510 dev_dbg(led->cdev.dev, "led %s set %d (%s mode)\n",
511 led->cdev.name, led->cdev.brightness,
512 (led->pwm_dev ? "pwm" : "manual"));
513
Jay Chokshi868312e2011-09-16 13:57:13 -0700514 if (led->pwm_dev == NULL) {
515 __pm8xxx_led_work(led, led->cdev.brightness);
516 } else {
517 rc = pm8xxx_led_pwm_work(led);
518 if (rc)
519 pr_err("could not configure PWM mode for LED:%d\n",
520 led->id);
521 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700522}
523
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700524static void pm8xxx_led_set(struct led_classdev *led_cdev,
525 enum led_brightness value)
526{
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700527 struct pm8xxx_led_data *led;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700528
529 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
530
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700531 if (value < LED_OFF || value > led->cdev.max_brightness) {
532 dev_err(led->cdev.dev, "Invalid brightness value exceeds");
533 return;
534 }
535
samin.ryue0041de2012-08-03 23:58:14 +0900536 if (!led->lock_update) {
537 schedule_work(&led->work);
538 } else {
539 dev_dbg(led->cdev.dev, "set %d pending\n",
540 value);
541 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700542}
543
Jay Chokshide4cefb2011-08-04 18:10:44 -0700544static int pm8xxx_set_led_mode_and_max_brightness(struct pm8xxx_led_data *led,
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700545 enum pm8xxx_led_modes led_mode, int max_current)
546{
Jay Chokshide4cefb2011-08-04 18:10:44 -0700547 switch (led->id) {
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700548 case PM8XXX_ID_LED_0:
549 case PM8XXX_ID_LED_1:
550 case PM8XXX_ID_LED_2:
551 led->cdev.max_brightness = max_current /
552 PM8XXX_ID_LED_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700553 if (led->cdev.max_brightness > MAX_LC_LED_BRIGHTNESS)
554 led->cdev.max_brightness = MAX_LC_LED_BRIGHTNESS;
Willie Ruana39d1d42011-08-19 13:54:34 -0700555 led->reg = led_mode;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700556 break;
557 case PM8XXX_ID_LED_KB_LIGHT:
558 case PM8XXX_ID_FLASH_LED_0:
559 case PM8XXX_ID_FLASH_LED_1:
560 led->cdev.max_brightness = max_current /
561 PM8XXX_ID_FLASH_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700562 if (led->cdev.max_brightness > MAX_FLASH_BRIGHTNESS)
563 led->cdev.max_brightness = MAX_FLASH_BRIGHTNESS;
564
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700565 switch (led_mode) {
566 case PM8XXX_LED_MODE_PWM1:
567 case PM8XXX_LED_MODE_PWM2:
568 case PM8XXX_LED_MODE_PWM3:
Willie Ruana39d1d42011-08-19 13:54:34 -0700569 led->reg = PM8XXX_FLASH_MODE_PWM;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700570 break;
571 case PM8XXX_LED_MODE_DTEST1:
Willie Ruana39d1d42011-08-19 13:54:34 -0700572 led->reg = PM8XXX_FLASH_MODE_DBUS1;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700573 break;
574 case PM8XXX_LED_MODE_DTEST2:
Willie Ruana39d1d42011-08-19 13:54:34 -0700575 led->reg = PM8XXX_FLASH_MODE_DBUS2;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700576 break;
577 default:
Willie Ruana39d1d42011-08-19 13:54:34 -0700578 led->reg = PM8XXX_LED_MODE_MANUAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700579 break;
580 }
581 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530582 case PM8XXX_ID_WLED:
583 led->cdev.max_brightness = WLED_MAX_LEVEL;
584 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800585 case PM8XXX_ID_RGB_LED_RED:
586 case PM8XXX_ID_RGB_LED_GREEN:
587 case PM8XXX_ID_RGB_LED_BLUE:
588 led->cdev.max_brightness = LED_FULL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700589 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800590 default:
591 dev_err(led->cdev.dev, "LED Id is invalid");
592 return -EINVAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700593 }
594
Amy Malochec17c3732012-02-27 18:34:07 -0800595 return 0;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700596}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700597
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700598static enum led_brightness pm8xxx_led_get(struct led_classdev *led_cdev)
599{
600 struct pm8xxx_led_data *led;
601
602 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
603
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700604 return led->cdev.brightness;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700605}
606
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530607static int __devinit init_wled(struct pm8xxx_led_data *led)
608{
609 int rc, i;
610 u8 val, num_wled_strings;
611
612 num_wled_strings = led->wled_cfg->num_strings;
613
614 /* program over voltage protection threshold */
615 if (led->wled_cfg->ovp_val > WLED_OVP_37V) {
616 dev_err(led->dev->parent, "Invalid ovp value");
617 return -EINVAL;
618 }
619
620 rc = pm8xxx_readb(led->dev->parent, WLED_OVP_CFG_REG, &val);
621 if (rc) {
622 dev_err(led->dev->parent, "can't read wled ovp config"
623 " register rc=%d\n", rc);
624 return rc;
625 }
626
627 val = (val & ~WLED_OVP_VAL_MASK) |
628 (led->wled_cfg->ovp_val << WLED_OVP_VAL_BIT_SHFT);
629
630 rc = pm8xxx_writeb(led->dev->parent, WLED_OVP_CFG_REG, val);
631 if (rc) {
632 dev_err(led->dev->parent, "can't write wled ovp config"
633 " register rc=%d\n", rc);
634 return rc;
635 }
636
637 /* program current boost limit and output feedback*/
638 if (led->wled_cfg->boost_curr_lim > WLED_CURR_LIMIT_1680mA) {
639 dev_err(led->dev->parent, "Invalid boost current limit");
640 return -EINVAL;
641 }
642
643 rc = pm8xxx_readb(led->dev->parent, WLED_BOOST_CFG_REG, &val);
644 if (rc) {
645 dev_err(led->dev->parent, "can't read wled boost config"
646 " register rc=%d\n", rc);
647 return rc;
648 }
649
650 val = (val & ~WLED_BOOST_LIMIT_MASK) |
651 (led->wled_cfg->boost_curr_lim << WLED_BOOST_LIMIT_BIT_SHFT);
652
653 val = (val & ~WLED_OP_FDBCK_MASK) |
654 (led->wled_cfg->op_fdbck << WLED_OP_FDBCK_BIT_SHFT);
655
656 rc = pm8xxx_writeb(led->dev->parent, WLED_BOOST_CFG_REG, val);
657 if (rc) {
658 dev_err(led->dev->parent, "can't write wled boost config"
659 " register rc=%d\n", rc);
660 return rc;
661 }
662
663 /* program high pole capacitance */
664 if (led->wled_cfg->cp_select > WLED_CP_SELECT_MAX) {
665 dev_err(led->dev->parent, "Invalid pole capacitance");
666 return -EINVAL;
667 }
668
669 rc = pm8xxx_readb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, &val);
670 if (rc) {
671 dev_err(led->dev->parent, "can't read wled high pole"
672 " capacitance register rc=%d\n", rc);
673 return rc;
674 }
675
676 val = (val & ~WLED_CP_SELECT_MASK) | led->wled_cfg->cp_select;
677
678 rc = pm8xxx_writeb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, val);
679 if (rc) {
680 dev_err(led->dev->parent, "can't write wled high pole"
681 " capacitance register rc=%d\n", rc);
682 return rc;
683 }
684
685 /* program activation delay and maximum current */
686 for (i = 0; i < num_wled_strings; i++) {
687 rc = pm8xxx_readb(led->dev->parent,
688 WLED_MAX_CURR_CFG_REG(i + 2), &val);
689 if (rc) {
690 dev_err(led->dev->parent, "can't read wled max current"
691 " config register rc=%d\n", rc);
692 return rc;
693 }
694
695 if ((led->wled_cfg->ctrl_delay_us % WLED_CTL_DLY_STEP) ||
696 (led->wled_cfg->ctrl_delay_us > WLED_CTL_DLY_MAX)) {
697 dev_err(led->dev->parent, "Invalid control delay\n");
698 return rc;
699 }
700
701 val = val / WLED_CTL_DLY_STEP;
702 val = (val & ~WLED_CTL_DLY_MASK) |
703 (led->wled_cfg->ctrl_delay_us << WLED_CTL_DLY_BIT_SHFT);
704
705 if ((led->max_current > WLED_MAX_CURR)) {
706 dev_err(led->dev->parent, "Invalid max current\n");
707 return -EINVAL;
708 }
709
710 val = (val & ~WLED_MAX_CURR_MASK) | led->max_current;
711
712 rc = pm8xxx_writeb(led->dev->parent,
713 WLED_MAX_CURR_CFG_REG(i + 2), val);
714 if (rc) {
715 dev_err(led->dev->parent, "can't write wled max current"
716 " config register rc=%d\n", rc);
717 return rc;
718 }
719 }
720
721 /* 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{
samin.ryue0041de2012-08-03 23:58:14 +0900798 int duty_us, rc;
Jay Chokshi868312e2011-09-16 13:57:13 -0700799
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) {
samin.ryue0041de2012-08-03 23:58:14 +0900812 rc = pm8xxx_led_pwm_pattern_update(led);
Jay Chokshi868312e2011-09-16 13:57:13 -0700813 } else {
814 duty_us = led->pwm_period_us;
815 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
816 }
817
818 return rc;
819}
820
samin.ryue0041de2012-08-03 23:58:14 +0900821static ssize_t pm8xxx_led_lock_update_show(struct device *dev,
822 struct device_attribute *attr, char *buf)
823{
824 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
825 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
826 int i, n = 0;
827
828 for (i = 0; i < pdata->num_configs; i++)
829 {
830 n += sprintf(&buf[n], "%s is %s\n",
831 leds[i].cdev.name,
832 (leds[i].lock_update ? "non-updatable" : "updatable"));
833 }
834
835 return n;
836}
837
838static ssize_t pm8xxx_led_lock_update_store(struct device *dev,
839 struct device_attribute *attr, const char *buf, size_t size)
840{
841 struct pm8xxx_led_platform_data *pdata = dev->platform_data;
842 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
843 ssize_t rc = -EINVAL;
844 char *after;
845 unsigned long state = simple_strtoul(buf, &after, 10);
846 size_t count = after - buf;
847 int i;
848
849 if (isspace(*after))
850 count++;
851
852 if (count == size) {
853 rc = count;
854 for (i = 0; i < pdata->num_configs; i++)
855 {
856 leds[i].lock_update = state;
857 if (!state) {
858 dev_info(dev, "resume %s set %d\n",
859 leds[i].cdev.name, leds[i].cdev.brightness);
860 schedule_work(&leds[i].work);
861 }
862 }
863 }
864 return rc;
865}
866
867static DEVICE_ATTR(lock, 0644, pm8xxx_led_lock_update_show, pm8xxx_led_lock_update_store);
868
869static ssize_t pm8xxx_led_grppwm_show(struct device *dev,
870 struct device_attribute *attr, char *buf)
871{
872 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
873 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
874 int i, n = 0;
875
876 for (i = 0; i < pdata->num_configs; i++)
877 {
878 n += sprintf(&buf[n], "%s period_us is %d\n", leds[i].cdev.name, leds[i].pwm_period_us);
879 }
880 return n;
881}
882
883static ssize_t pm8xxx_led_grppwm_store(struct device *dev,
884 struct device_attribute *attr, const char *buf, size_t size)
885{
886 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
887 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
888 ssize_t rc = -EINVAL;
889 char *after;
890 unsigned long state = simple_strtoul(buf, &after, 10);
891 size_t count = after - buf;
892 int i;
893
894 if (isspace(*after))
895 count++;
896
897 if (count == size) {
898 rc = count;
899
900 if(state < PM8XXX_PWM_PERIOD_MIN)
901 state = PM8XXX_PWM_PERIOD_MIN;
902 if(state > PM8XXX_PWM_PERIOD_MAX)
903 state = PM8XXX_PWM_PERIOD_MAX;
904
905 for (i = 0; i < pdata->num_configs; i++)
906 {
907 leds[i].pwm_period_us = state;
908 }
909 }
910 return rc;
911}
912
913static DEVICE_ATTR(grppwm, 0644, pm8xxx_led_grppwm_show, pm8xxx_led_grppwm_store);
914
915static ssize_t pm8xxx_led_grpfreq_show(struct device *dev,
916 struct device_attribute *attr, char *buf)
917{
918 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
919 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
920 int i, n = 0;
921
922 for (i = 0; i < pdata->num_configs; i++)
923 {
924 if (leds[i].pwm_duty_cycles != NULL) {
925 n += sprintf(&buf[n], "%s freq %d\n", leds[i].cdev.name, leds[i].pwm_duty_cycles->duty_ms);
926 }
927 else
928 n += sprintf(&buf[n], "%s's duty cycles is not exist\n", leds[i].cdev.name);
929
930 }
931 return n;
932}
933
934static ssize_t pm8xxx_led_grpfreq_store(struct device *dev,
935 struct device_attribute *attr, const char *buf, size_t size)
936{
937 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
938 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
939 ssize_t rc = -EINVAL;
940 char *after;
941 unsigned long state = simple_strtoul(buf, &after, 10);
942 size_t count = after - buf;
943 int i;
944
945 if (isspace(*after))
946 count++;
947
948 if (count == size) {
949 rc = count;
950
951 if(state == 0)
952 state = 32;
953 if(state > PM_PWM_LUT_DUTY_TIME_MAX)
954 state = PM_PWM_LUT_DUTY_TIME_MAX;
955
956 for (i = 0; i < pdata->num_configs; i++)
957 {
958 if (leds[i].pwm_duty_cycles != NULL) {
959 leds[i].pwm_duty_cycles->duty_ms = state;
960 }
961 }
962 }
963 return rc;
964}
965
966static DEVICE_ATTR(grpfreq, 0644, pm8xxx_led_grpfreq_show, pm8xxx_led_grpfreq_store);
967
968static ssize_t pm8xxx_led_blink_show(struct device *dev,
969 struct device_attribute *attr, char *buf)
970{
971 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
972 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
973 int start_idx = 0, idx_len = 0;
974 int i, j, n = 0;
975
976 for (i = 0; i < pdata->num_configs; i++)
977 {
978 n += sprintf(&buf[n], "[%d] %s pwm pattern %d\n", i, leds[i].cdev.name, leds[i].blink);
979 if (leds[i].pwm_duty_cycles != NULL && leds[i].pwm_duty_cycles->duty_pcts != NULL) {
980 start_idx = leds[i].pwm_duty_cycles->start_idx;
981 idx_len = leds[i].pwm_duty_cycles->num_duty_pcts;
982 for (j = 0; j < idx_len; j++)
983 {
984 n += sprintf(&buf[n], "%d ",
985 leds[i].pwm_duty_cycles->duty_pcts[j]);
986 }
987 n += sprintf(&buf[n], "\n");
988 }
989 else
990 n += sprintf(&buf[n], "not exist\n");
991
992 }
993 return n;
994}
995
996static ssize_t pm8xxx_led_blink_store(struct device *dev,
997 struct device_attribute *attr, const char *buf, size_t size)
998{
999 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
1000 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
1001 ssize_t rc = -EINVAL;
1002 char *after;
1003 unsigned long state = simple_strtoul(buf, &after, 10);
1004 size_t count = after - buf;
1005 int i;
1006
1007 if (isspace(*after))
1008 count++;
1009
1010 if (count == size) {
1011 rc = count;
1012
1013 for (i = 0; i < pdata->num_configs; i++)
1014 {
1015 if (leds[i].blink != state) {
1016 leds[i].blink = state;
1017 rc = pm8xxx_led_pwm_pattern_update(&leds[i]);
1018 }
1019 }
1020 }
1021 return rc;
1022}
1023
1024static DEVICE_ATTR(blink, 0644, pm8xxx_led_blink_show, pm8xxx_led_blink_store);
1025
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301026
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001027static int __devinit pm8xxx_led_probe(struct platform_device *pdev)
1028{
Jay Chokshi8994e392011-09-14 18:20:39 -07001029 const struct pm8xxx_led_platform_data *pdata = pdev->dev.platform_data;
1030 const struct led_platform_data *pcore_data;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001031 struct led_info *curr_led;
1032 struct pm8xxx_led_data *led, *led_dat;
Jay Chokshi8994e392011-09-14 18:20:39 -07001033 struct pm8xxx_led_config *led_cfg;
Amy Maloche3d2c24c2012-03-22 19:02:25 -07001034 enum pm8xxx_version version;
1035 bool found = false;
1036 int rc, i, j;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001037
1038 if (pdata == NULL) {
1039 dev_err(&pdev->dev, "platform data not supplied\n");
1040 return -EINVAL;
1041 }
1042
Jay Chokshi8994e392011-09-14 18:20:39 -07001043 pcore_data = pdata->led_core;
1044
1045 if (pcore_data->num_leds != pdata->num_configs) {
1046 dev_err(&pdev->dev, "#no. of led configs and #no. of led"
1047 "entries are not equal\n");
1048 return -EINVAL;
1049 }
1050
1051 led = kcalloc(pcore_data->num_leds, sizeof(*led), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001052 if (led == NULL) {
1053 dev_err(&pdev->dev, "failed to alloc memory\n");
1054 return -ENOMEM;
1055 }
1056
Jay Chokshi8994e392011-09-14 18:20:39 -07001057 for (i = 0; i < pcore_data->num_leds; i++) {
1058 curr_led = &pcore_data->leds[i];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001059 led_dat = &led[i];
Jay Chokshi8994e392011-09-14 18:20:39 -07001060 led_cfg = &pdata->configs[i];
1061
1062 led_dat->id = led_cfg->id;
Jay Chokshi868312e2011-09-16 13:57:13 -07001063 led_dat->pwm_channel = led_cfg->pwm_channel;
1064 led_dat->pwm_period_us = led_cfg->pwm_period_us;
1065 led_dat->pwm_duty_cycles = led_cfg->pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301066 led_dat->wled_cfg = led_cfg->wled_cfg;
1067 led_dat->max_current = led_cfg->max_current;
samin.ryue0041de2012-08-03 23:58:14 +09001068 led_dat->lock_update = 0;
1069 led_dat->blink = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001070
1071 if (!((led_dat->id >= PM8XXX_ID_LED_KB_LIGHT) &&
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301072 (led_dat->id < PM8XXX_ID_MAX))) {
Amy Maloche3d2c24c2012-03-22 19:02:25 -07001073 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
1074 led_dat->id);
1075 rc = -EINVAL;
1076 goto fail_id_check;
1077
1078 }
1079
1080 found = false;
1081 version = pm8xxx_get_version(pdev->dev.parent);
1082 for (j = 0; j < ARRAY_SIZE(led_map); j++) {
1083 if (version == led_map[j].version
1084 && (led_map[j].supported & (1 << led_dat->id))) {
1085 found = true;
1086 break;
1087 }
1088 }
1089
1090 if (!found) {
1091 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
1092 led_dat->id);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001093 rc = -EINVAL;
1094 goto fail_id_check;
1095 }
1096
1097 led_dat->cdev.name = curr_led->name;
1098 led_dat->cdev.default_trigger = curr_led->default_trigger;
1099 led_dat->cdev.brightness_set = pm8xxx_led_set;
1100 led_dat->cdev.brightness_get = pm8xxx_led_get;
1101 led_dat->cdev.brightness = LED_OFF;
Jay Chokshi8994e392011-09-14 18:20:39 -07001102 led_dat->cdev.flags = curr_led->flags;
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001103 led_dat->dev = &pdev->dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001104
1105 rc = get_init_value(led_dat, &led_dat->reg);
1106 if (rc < 0)
1107 goto fail_id_check;
1108
Jay Chokshi8994e392011-09-14 18:20:39 -07001109 rc = pm8xxx_set_led_mode_and_max_brightness(led_dat,
1110 led_cfg->mode, led_cfg->max_current);
Jay Chokshide4cefb2011-08-04 18:10:44 -07001111 if (rc < 0)
1112 goto fail_id_check;
1113
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001114 mutex_init(&led_dat->lock);
1115 INIT_WORK(&led_dat->work, pm8xxx_led_work);
1116
1117 rc = led_classdev_register(&pdev->dev, &led_dat->cdev);
1118 if (rc) {
1119 dev_err(&pdev->dev, "unable to register led %d,rc=%d\n",
1120 led_dat->id, rc);
1121 goto fail_id_check;
1122 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001123
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301124 /* configure default state */
1125 if (led_cfg->default_state)
samin.ryue0041de2012-08-03 23:58:14 +09001126 led_dat->cdev.brightness = led_dat->cdev.max_brightness;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301127 else
samin.ryue0041de2012-08-03 23:58:14 +09001128 led_dat->cdev.brightness = LED_OFF;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301129
Jay Chokshi868312e2011-09-16 13:57:13 -07001130 if (led_cfg->mode != PM8XXX_LED_MODE_MANUAL) {
Amy Malochec17c3732012-02-27 18:34:07 -08001131 if (led_dat->id == PM8XXX_ID_RGB_LED_RED ||
1132 led_dat->id == PM8XXX_ID_RGB_LED_GREEN ||
1133 led_dat->id == PM8XXX_ID_RGB_LED_BLUE)
1134 __pm8xxx_led_work(led_dat, 0);
1135 else
1136 __pm8xxx_led_work(led_dat,
Jay Chokshide4cefb2011-08-04 18:10:44 -07001137 led_dat->cdev.max_brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -07001138
1139 if (led_dat->pwm_channel != -1) {
samin.ryue0041de2012-08-03 23:58:14 +09001140 if (pdata->max_brightness)
1141 led_dat->cdev.max_brightness =
1142 pdata->max_brightness;
1143 else
1144 led_dat->cdev.max_brightness =
1145 LED_FULL;
Jay Chokshi868312e2011-09-16 13:57:13 -07001146 rc = pm8xxx_led_pwm_configure(led_dat);
1147 if (rc) {
1148 dev_err(&pdev->dev, "failed to "
1149 "configure LED, error: %d\n", rc);
1150 goto fail_id_check;
1151 }
samin.ryue0041de2012-08-03 23:58:14 +09001152 schedule_work(&led->work);
Jay Chokshi868312e2011-09-16 13:57:13 -07001153 }
1154 } else {
samin.ryue0041de2012-08-03 23:58:14 +09001155 __pm8xxx_led_work(led_dat, led_dat->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -07001156 }
Jay Chokshide4cefb2011-08-04 18:10:44 -07001157 }
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001158
samin.ryue0041de2012-08-03 23:58:14 +09001159 led->use_pwm = pdata->use_pwm;
1160
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001161 platform_set_drvdata(pdev, led);
1162
samin.ryue0041de2012-08-03 23:58:14 +09001163 rc = device_create_file(&pdev->dev, &dev_attr_lock);
1164 if (rc) {
1165 device_remove_file(&pdev->dev, &dev_attr_lock);
1166 dev_err(&pdev->dev, "failed device_create_file(lock)\n");
1167 }
1168
1169 if (led->use_pwm) {
1170 rc = device_create_file(&pdev->dev, &dev_attr_blink);
1171 if (rc) {
1172 device_remove_file(&pdev->dev, &dev_attr_blink);
1173 dev_err(&pdev->dev, "failed device_create_file(blink)\n");
1174 }
1175
1176 rc = device_create_file(&pdev->dev, &dev_attr_grppwm);
1177 if (rc) {
1178 device_remove_file(&pdev->dev, &dev_attr_grppwm);
1179 dev_err(&pdev->dev, "failed device_create_fiaild(grppwm)\n");
1180 }
1181
1182 rc = device_create_file(&pdev->dev, &dev_attr_grpfreq);
1183 if (rc) {
1184 device_remove_file(&pdev->dev, &dev_attr_grpfreq);
1185 dev_err(&pdev->dev, "failed device_create_file(grpfreq)\n");
1186 }
1187 }
1188
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001189 return 0;
1190
1191fail_id_check:
1192 if (i > 0) {
1193 for (i = i - 1; i >= 0; i--) {
1194 mutex_destroy(&led[i].lock);
1195 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -07001196 if (led[i].pwm_dev != NULL)
1197 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001198 }
1199 }
1200 kfree(led);
1201 return rc;
1202}
1203
1204static int __devexit pm8xxx_led_remove(struct platform_device *pdev)
1205{
1206 int i;
1207 const struct led_platform_data *pdata =
1208 pdev->dev.platform_data;
1209 struct pm8xxx_led_data *led = platform_get_drvdata(pdev);
1210
1211 for (i = 0; i < pdata->num_leds; i++) {
1212 cancel_work_sync(&led[i].work);
1213 mutex_destroy(&led[i].lock);
1214 led_classdev_unregister(&led[i].cdev);
samin.ryue0041de2012-08-03 23:58:14 +09001215 if (led[i].pwm_dev != NULL) {
Jay Chokshi868312e2011-09-16 13:57:13 -07001216 pwm_free(led[i].pwm_dev);
samin.ryue0041de2012-08-03 23:58:14 +09001217 }
1218 }
1219
1220 device_remove_file(&pdev->dev, &dev_attr_lock);
1221
1222 if (led->use_pwm) {
1223 device_remove_file(&pdev->dev, &dev_attr_blink);
1224 device_remove_file(&pdev->dev, &dev_attr_grppwm);
1225 device_remove_file(&pdev->dev, &dev_attr_grpfreq);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001226 }
1227
1228 kfree(led);
1229
1230 return 0;
1231}
1232
1233static struct platform_driver pm8xxx_led_driver = {
1234 .probe = pm8xxx_led_probe,
1235 .remove = __devexit_p(pm8xxx_led_remove),
1236 .driver = {
1237 .name = PM8XXX_LEDS_DEV_NAME,
1238 .owner = THIS_MODULE,
1239 },
1240};
1241
1242static int __init pm8xxx_led_init(void)
1243{
1244 return platform_driver_register(&pm8xxx_led_driver);
1245}
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001246subsys_initcall(pm8xxx_led_init);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001247
1248static void __exit pm8xxx_led_exit(void)
1249{
1250 platform_driver_unregister(&pm8xxx_led_driver);
1251}
1252module_exit(pm8xxx_led_exit);
1253
1254MODULE_DESCRIPTION("PM8XXX LEDs driver");
1255MODULE_LICENSE("GPL v2");
1256MODULE_VERSION("1.0");
1257MODULE_ALIAS("platform:pm8xxx-led");