blob: 7fe1faf3b69d5edb3b326d3fd6b7d696a0fa519d [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;
Devin Kim232d6652012-08-09 20:41:00 -0700175 int adjust_brightness;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700176};
177
178static void led_kp_set(struct pm8xxx_led_data *led, enum led_brightness value)
179{
180 int rc;
181 u8 level;
182
183 level = (value << PM8XXX_DRV_KEYPAD_BL_SHIFT) &
184 PM8XXX_DRV_KEYPAD_BL_MASK;
185
186 led->reg &= ~PM8XXX_DRV_KEYPAD_BL_MASK;
187 led->reg |= level;
188
189 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_DRV_KEYPAD,
190 led->reg);
191 if (rc < 0)
192 dev_err(led->cdev.dev,
193 "can't set keypad backlight level rc=%d\n", rc);
194}
195
196static void led_lc_set(struct pm8xxx_led_data *led, enum led_brightness value)
197{
198 int rc, offset;
199 u8 level;
200
201 level = (value << PM8XXX_DRV_LED_CTRL_SHIFT) &
202 PM8XXX_DRV_LED_CTRL_MASK;
203
204 offset = PM8XXX_LED_OFFSET(led->id);
205
206 led->reg &= ~PM8XXX_DRV_LED_CTRL_MASK;
207 led->reg |= level;
208
209 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_LED_CTRL(offset),
210 led->reg);
211 if (rc)
212 dev_err(led->cdev.dev, "can't set (%d) led value rc=%d\n",
213 led->id, rc);
214}
215
216static void
217led_flash_set(struct pm8xxx_led_data *led, enum led_brightness value)
218{
219 int rc;
220 u8 level;
221 u16 reg_addr;
222
223 level = (value << PM8XXX_DRV_FLASH_SHIFT) &
224 PM8XXX_DRV_FLASH_MASK;
225
226 led->reg &= ~PM8XXX_DRV_FLASH_MASK;
227 led->reg |= level;
228
229 if (led->id == PM8XXX_ID_FLASH_LED_0)
230 reg_addr = SSBI_REG_ADDR_FLASH_DRV0;
231 else
232 reg_addr = SSBI_REG_ADDR_FLASH_DRV1;
233
234 rc = pm8xxx_writeb(led->dev->parent, reg_addr, led->reg);
235 if (rc < 0)
236 dev_err(led->cdev.dev, "can't set flash led%d level rc=%d\n",
237 led->id, rc);
238}
239
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530240static int
241led_wled_set(struct pm8xxx_led_data *led, enum led_brightness value)
242{
243 int rc, duty;
244 u8 val, i, num_wled_strings;
245
246 if (value > WLED_MAX_LEVEL)
247 value = WLED_MAX_LEVEL;
248
Amy Maloche56913f52012-05-11 10:47:24 -0700249 if (value == 0) {
250 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
251 WLED_BOOST_OFF);
252 if (rc) {
253 dev_err(led->dev->parent, "can't write wled ctrl config"
254 " register rc=%d\n", rc);
255 return rc;
256 }
257 } else {
258 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
259 led->wled_mod_ctrl_val);
260 if (rc) {
261 dev_err(led->dev->parent, "can't write wled ctrl config"
262 " register rc=%d\n", rc);
263 return rc;
264 }
265 }
266
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530267 duty = (WLED_MAX_DUTY_CYCLE * value) / WLED_MAX_LEVEL;
268
269 num_wled_strings = led->wled_cfg->num_strings;
270
271 /* program brightness control registers */
272 for (i = 0; i < num_wled_strings; i++) {
273 rc = pm8xxx_readb(led->dev->parent,
274 WLED_BRIGHTNESS_CNTL_REG1(i), &val);
275 if (rc) {
276 dev_err(led->dev->parent, "can't read wled brightnes ctrl"
277 " register1 rc=%d\n", rc);
278 return rc;
279 }
280
281 val = (val & ~WLED_MAX_CURR_MASK) | (duty >> WLED_8_BIT_SHFT);
282 rc = pm8xxx_writeb(led->dev->parent,
283 WLED_BRIGHTNESS_CNTL_REG1(i), val);
284 if (rc) {
285 dev_err(led->dev->parent, "can't write wled brightness ctrl"
286 " register1 rc=%d\n", rc);
287 return rc;
288 }
289
290 val = duty & WLED_8_BIT_MASK;
291 rc = pm8xxx_writeb(led->dev->parent,
292 WLED_BRIGHTNESS_CNTL_REG2(i), val);
293 if (rc) {
294 dev_err(led->dev->parent, "can't write wled brightness ctrl"
295 " register2 rc=%d\n", rc);
296 return rc;
297 }
298 }
299
300 /* sync */
301 val = WLED_SYNC_VAL;
302 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
303 if (rc) {
304 dev_err(led->dev->parent,
305 "can't read wled sync register rc=%d\n", rc);
306 return rc;
307 }
308
309 val = WLED_SYNC_RESET_VAL;
310 rc = pm8xxx_writeb(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 }
316 return 0;
317}
318
319static void wled_dump_regs(struct pm8xxx_led_data *led)
320{
321 int i;
322 u8 val;
323
324 for (i = 1; i < 17; i++) {
325 pm8xxx_readb(led->dev->parent,
326 SSBI_REG_ADDR_WLED_CTRL(i), &val);
327 pr_debug("WLED_CTRL_%d = 0x%x\n", i, val);
328 }
329}
330
Amy Malochec17c3732012-02-27 18:34:07 -0800331static void
Amy Malochead3c7842012-07-20 14:51:27 -0700332led_rgb_write(struct pm8xxx_led_data *led, u16 addr, enum led_brightness value)
Amy Malochec17c3732012-02-27 18:34:07 -0800333{
334 int rc;
335 u8 val, mask;
336
Amy Malochead3c7842012-07-20 14:51:27 -0700337 if (led->id != PM8XXX_ID_RGB_LED_BLUE &&
338 led->id != PM8XXX_ID_RGB_LED_RED &&
339 led->id != PM8XXX_ID_RGB_LED_GREEN)
340 return;
341
342 rc = pm8xxx_readb(led->dev->parent, addr, &val);
Amy Malochec17c3732012-02-27 18:34:07 -0800343 if (rc) {
344 dev_err(led->cdev.dev, "can't read rgb ctrl register rc=%d\n",
345 rc);
346 return;
347 }
348
349 switch (led->id) {
350 case PM8XXX_ID_RGB_LED_RED:
351 mask = PM8XXX_DRV_RGB_RED_LED;
352 break;
353 case PM8XXX_ID_RGB_LED_GREEN:
354 mask = PM8XXX_DRV_RGB_GREEN_LED;
355 break;
356 case PM8XXX_ID_RGB_LED_BLUE:
357 mask = PM8XXX_DRV_RGB_BLUE_LED;
358 break;
359 default:
360 return;
361 }
362
363 if (value)
364 val |= mask;
365 else
366 val &= ~mask;
367
Amy Malochead3c7842012-07-20 14:51:27 -0700368 rc = pm8xxx_writeb(led->dev->parent, addr, val);
Amy Malochec17c3732012-02-27 18:34:07 -0800369 if (rc < 0)
370 dev_err(led->cdev.dev, "can't set rgb led %d level rc=%d\n",
371 led->id, rc);
372}
373
Amy Malochead3c7842012-07-20 14:51:27 -0700374static void
375led_rgb_set(struct pm8xxx_led_data *led, enum led_brightness value)
376{
377 if (value) {
378 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, value);
379 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL2, value);
380 } else {
381 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL2, value);
382 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, value);
383 }
384}
385
Devin Kim232d6652012-08-09 20:41:00 -0700386static int pm8xxx_adjust_brightness(struct led_classdev *led_cdev,
387 enum led_brightness value)
388{
389 int level = 0;
390 struct pm8xxx_led_data *led;
391 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
392
393 if (!led->adjust_brightness)
394 return value;
395
396 if (led->adjust_brightness == led->cdev.max_brightness)
397 return value;
398
399 level = (2 * value * led->adjust_brightness
400 + led->cdev.max_brightness)
401 / (2 * led->cdev.max_brightness);
402
403 if (!level && value)
404 level = 1;
405
406 return level;
407}
408
samin.ryue0041de2012-08-03 23:58:14 +0900409static int pm8xxx_led_pwm_pattern_update(struct pm8xxx_led_data * led)
410{
411 int start_idx, idx_len;
412 int *pcts = NULL;
413 int i, rc = 0;
414 int temp = 0;
Devin Kim232d6652012-08-09 20:41:00 -0700415 int pwm_max = 0;
samin.ryue0041de2012-08-03 23:58:14 +0900416
417 if (!led->pwm_duty_cycles || !led->pwm_duty_cycles->duty_pcts) {
418 dev_err(led->cdev.dev, "duty_cycles and duty_pcts is not exist\n");
419 return -EINVAL;
420 }
421
Devin Kim232d6652012-08-09 20:41:00 -0700422 pwm_max = pm8xxx_adjust_brightness(&led->cdev, led->cdev.brightness);
samin.ryue0041de2012-08-03 23:58:14 +0900423 start_idx = led->pwm_duty_cycles->start_idx;
424 idx_len = led->pwm_duty_cycles->num_duty_pcts;
425 pcts = led->pwm_duty_cycles->duty_pcts;
426
427 for (i = 0; i < idx_len; i++)
428 {
429 if (led->blink == 1) {
430 temp = ((pwm_max * i) << 1) / (idx_len - 1);
431 pcts[i] = (temp + 1) >> 1;
432 } else {
433 pcts[i] = (pwm_max);
434 }
435 }
436
437 if (idx_len >= PM_PWM_LUT_SIZE && start_idx) {
438 pr_err("Wrong LUT size or index\n");
439 return -EINVAL;
440 }
441 if ((start_idx + idx_len) > PM_PWM_LUT_SIZE) {
442 pr_err("Exceed LUT limit\n");
443 return -EINVAL;
444 }
445
446 rc = pm8xxx_pwm_lut_config(led->pwm_dev, led->pwm_period_us,
447 led->pwm_duty_cycles->duty_pcts,
448 led->pwm_duty_cycles->duty_ms,
449 start_idx, idx_len, 0, 0,
450 PM8XXX_LED_PWM_FLAGS);
451
452 return rc;
453}
454
Jay Chokshi868312e2011-09-16 13:57:13 -0700455static int pm8xxx_led_pwm_work(struct pm8xxx_led_data *led)
456{
457 int duty_us;
458 int rc = 0;
Devin Kim232d6652012-08-09 20:41:00 -0700459 int level = 0;
460
461 level = pm8xxx_adjust_brightness(&led->cdev, led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700462
463 if (led->pwm_duty_cycles == NULL) {
Devin Kim232d6652012-08-09 20:41:00 -0700464 duty_us = (led->pwm_period_us * level) / LED_FULL;
Jay Chokshi868312e2011-09-16 13:57:13 -0700465 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
Amy Malochead3c7842012-07-20 14:51:27 -0700466 if (led->cdev.brightness) {
467 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
468 led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700469 rc = pwm_enable(led->pwm_dev);
Amy Malochead3c7842012-07-20 14:51:27 -0700470 } else {
Jay Chokshi868312e2011-09-16 13:57:13 -0700471 pwm_disable(led->pwm_dev);
Amy Malochead3c7842012-07-20 14:51:27 -0700472 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
473 led->cdev.brightness);
474 }
Jay Chokshi868312e2011-09-16 13:57:13 -0700475 } else {
samin.ryue0041de2012-08-03 23:58:14 +0900476
Devin Kim232d6652012-08-09 20:41:00 -0700477 if (level) {
samin.ryue0041de2012-08-03 23:58:14 +0900478 pm8xxx_led_pwm_pattern_update(led);
Devin Kim232d6652012-08-09 20:41:00 -0700479 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, level);
samin.ryue0041de2012-08-03 23:58:14 +0900480 }
481
Devin Kim232d6652012-08-09 20:41:00 -0700482 rc = pm8xxx_pwm_lut_enable(led->pwm_dev, level);
483 if (!level)
484 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, level);
Jay Chokshi868312e2011-09-16 13:57:13 -0700485 }
486
487 return rc;
488}
489
Jay Chokshide4cefb2011-08-04 18:10:44 -0700490static void __pm8xxx_led_work(struct pm8xxx_led_data *led,
Devin Kim232d6652012-08-09 20:41:00 -0700491 enum led_brightness value)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700492{
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530493 int rc;
Devin Kim232d6652012-08-09 20:41:00 -0700494 int level = 0;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530495
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700496 mutex_lock(&led->lock);
497
Devin Kim232d6652012-08-09 20:41:00 -0700498 level = pm8xxx_adjust_brightness(&led->cdev, value);
499
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700500 switch (led->id) {
501 case PM8XXX_ID_LED_KB_LIGHT:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700502 led_kp_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530503 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700504 case PM8XXX_ID_LED_0:
505 case PM8XXX_ID_LED_1:
506 case PM8XXX_ID_LED_2:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700507 led_lc_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530508 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700509 case PM8XXX_ID_FLASH_LED_0:
510 case PM8XXX_ID_FLASH_LED_1:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700511 led_flash_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530512 break;
513 case PM8XXX_ID_WLED:
514 rc = led_wled_set(led, level);
515 if (rc < 0)
516 pr_err("wled brightness set failed %d\n", rc);
517 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800518 case PM8XXX_ID_RGB_LED_RED:
519 case PM8XXX_ID_RGB_LED_GREEN:
520 case PM8XXX_ID_RGB_LED_BLUE:
521 led_rgb_set(led, level);
522 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530523 default:
524 dev_err(led->cdev.dev, "unknown led id %d", led->id);
525 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700526 }
527
528 mutex_unlock(&led->lock);
529}
530
Jay Chokshide4cefb2011-08-04 18:10:44 -0700531static void pm8xxx_led_work(struct work_struct *work)
532{
Jay Chokshi868312e2011-09-16 13:57:13 -0700533 int rc;
534
Jay Chokshide4cefb2011-08-04 18:10:44 -0700535 struct pm8xxx_led_data *led = container_of(work,
536 struct pm8xxx_led_data, work);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700537
samin.ryue0041de2012-08-03 23:58:14 +0900538 dev_dbg(led->cdev.dev, "led %s set %d (%s mode)\n",
539 led->cdev.name, led->cdev.brightness,
540 (led->pwm_dev ? "pwm" : "manual"));
541
Jay Chokshi868312e2011-09-16 13:57:13 -0700542 if (led->pwm_dev == NULL) {
543 __pm8xxx_led_work(led, led->cdev.brightness);
544 } else {
545 rc = pm8xxx_led_pwm_work(led);
546 if (rc)
547 pr_err("could not configure PWM mode for LED:%d\n",
548 led->id);
549 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700550}
551
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700552static void pm8xxx_led_set(struct led_classdev *led_cdev,
553 enum led_brightness value)
554{
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700555 struct pm8xxx_led_data *led;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700556
557 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
558
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700559 if (value < LED_OFF || value > led->cdev.max_brightness) {
560 dev_err(led->cdev.dev, "Invalid brightness value exceeds");
561 return;
562 }
563
samin.ryue0041de2012-08-03 23:58:14 +0900564 if (!led->lock_update) {
565 schedule_work(&led->work);
566 } else {
567 dev_dbg(led->cdev.dev, "set %d pending\n",
568 value);
569 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700570}
571
Devin Kim232d6652012-08-09 20:41:00 -0700572static int pm8xxx_set_led_mode_and_adjust_brightness(struct pm8xxx_led_data *led,
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700573 enum pm8xxx_led_modes led_mode, int max_current)
574{
Jay Chokshide4cefb2011-08-04 18:10:44 -0700575 switch (led->id) {
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700576 case PM8XXX_ID_LED_0:
577 case PM8XXX_ID_LED_1:
578 case PM8XXX_ID_LED_2:
samin.ryuc650af32012-08-29 11:34:34 +0900579 led->adjust_brightness = max_current /
580 PM8XXX_ID_LED_CURRENT_FACTOR;
581 if (led->adjust_brightness > MAX_LC_LED_BRIGHTNESS)
582 led->adjust_brightness = MAX_LC_LED_BRIGHTNESS;
Willie Ruana39d1d42011-08-19 13:54:34 -0700583 led->reg = led_mode;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700584 break;
585 case PM8XXX_ID_LED_KB_LIGHT:
586 case PM8XXX_ID_FLASH_LED_0:
587 case PM8XXX_ID_FLASH_LED_1:
Devin Kim232d6652012-08-09 20:41:00 -0700588 led->adjust_brightness = max_current /
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700589 PM8XXX_ID_FLASH_CURRENT_FACTOR;
Devin Kim232d6652012-08-09 20:41:00 -0700590 if (led->adjust_brightness > MAX_FLASH_BRIGHTNESS)
591 led->adjust_brightness = MAX_FLASH_BRIGHTNESS;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700592
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700593 switch (led_mode) {
594 case PM8XXX_LED_MODE_PWM1:
595 case PM8XXX_LED_MODE_PWM2:
596 case PM8XXX_LED_MODE_PWM3:
Willie Ruana39d1d42011-08-19 13:54:34 -0700597 led->reg = PM8XXX_FLASH_MODE_PWM;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700598 break;
599 case PM8XXX_LED_MODE_DTEST1:
Willie Ruana39d1d42011-08-19 13:54:34 -0700600 led->reg = PM8XXX_FLASH_MODE_DBUS1;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700601 break;
602 case PM8XXX_LED_MODE_DTEST2:
Willie Ruana39d1d42011-08-19 13:54:34 -0700603 led->reg = PM8XXX_FLASH_MODE_DBUS2;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700604 break;
605 default:
Willie Ruana39d1d42011-08-19 13:54:34 -0700606 led->reg = PM8XXX_LED_MODE_MANUAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700607 break;
608 }
609 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530610 case PM8XXX_ID_WLED:
Devin Kim232d6652012-08-09 20:41:00 -0700611 led->adjust_brightness = WLED_MAX_LEVEL;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530612 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800613 case PM8XXX_ID_RGB_LED_RED:
614 case PM8XXX_ID_RGB_LED_GREEN:
615 case PM8XXX_ID_RGB_LED_BLUE:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700616 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800617 default:
618 dev_err(led->cdev.dev, "LED Id is invalid");
619 return -EINVAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700620 }
621
Amy Malochec17c3732012-02-27 18:34:07 -0800622 return 0;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700623}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700624
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700625static enum led_brightness pm8xxx_led_get(struct led_classdev *led_cdev)
626{
627 struct pm8xxx_led_data *led;
628
629 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
630
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700631 return led->cdev.brightness;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700632}
633
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530634static int __devinit init_wled(struct pm8xxx_led_data *led)
635{
636 int rc, i;
637 u8 val, num_wled_strings;
638
639 num_wled_strings = led->wled_cfg->num_strings;
640
641 /* program over voltage protection threshold */
642 if (led->wled_cfg->ovp_val > WLED_OVP_37V) {
643 dev_err(led->dev->parent, "Invalid ovp value");
644 return -EINVAL;
645 }
646
647 rc = pm8xxx_readb(led->dev->parent, WLED_OVP_CFG_REG, &val);
648 if (rc) {
649 dev_err(led->dev->parent, "can't read wled ovp config"
650 " register rc=%d\n", rc);
651 return rc;
652 }
653
654 val = (val & ~WLED_OVP_VAL_MASK) |
655 (led->wled_cfg->ovp_val << WLED_OVP_VAL_BIT_SHFT);
656
657 rc = pm8xxx_writeb(led->dev->parent, WLED_OVP_CFG_REG, val);
658 if (rc) {
659 dev_err(led->dev->parent, "can't write wled ovp config"
660 " register rc=%d\n", rc);
661 return rc;
662 }
663
664 /* program current boost limit and output feedback*/
665 if (led->wled_cfg->boost_curr_lim > WLED_CURR_LIMIT_1680mA) {
666 dev_err(led->dev->parent, "Invalid boost current limit");
667 return -EINVAL;
668 }
669
670 rc = pm8xxx_readb(led->dev->parent, WLED_BOOST_CFG_REG, &val);
671 if (rc) {
672 dev_err(led->dev->parent, "can't read wled boost config"
673 " register rc=%d\n", rc);
674 return rc;
675 }
676
677 val = (val & ~WLED_BOOST_LIMIT_MASK) |
678 (led->wled_cfg->boost_curr_lim << WLED_BOOST_LIMIT_BIT_SHFT);
679
680 val = (val & ~WLED_OP_FDBCK_MASK) |
681 (led->wled_cfg->op_fdbck << WLED_OP_FDBCK_BIT_SHFT);
682
683 rc = pm8xxx_writeb(led->dev->parent, WLED_BOOST_CFG_REG, val);
684 if (rc) {
685 dev_err(led->dev->parent, "can't write wled boost config"
686 " register rc=%d\n", rc);
687 return rc;
688 }
689
690 /* program high pole capacitance */
691 if (led->wled_cfg->cp_select > WLED_CP_SELECT_MAX) {
692 dev_err(led->dev->parent, "Invalid pole capacitance");
693 return -EINVAL;
694 }
695
696 rc = pm8xxx_readb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, &val);
697 if (rc) {
698 dev_err(led->dev->parent, "can't read wled high pole"
699 " capacitance register rc=%d\n", rc);
700 return rc;
701 }
702
703 val = (val & ~WLED_CP_SELECT_MASK) | led->wled_cfg->cp_select;
704
705 rc = pm8xxx_writeb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, val);
706 if (rc) {
707 dev_err(led->dev->parent, "can't write wled high pole"
708 " capacitance register rc=%d\n", rc);
709 return rc;
710 }
711
712 /* program activation delay and maximum current */
713 for (i = 0; i < num_wled_strings; i++) {
714 rc = pm8xxx_readb(led->dev->parent,
715 WLED_MAX_CURR_CFG_REG(i + 2), &val);
716 if (rc) {
717 dev_err(led->dev->parent, "can't read wled max current"
718 " config register rc=%d\n", rc);
719 return rc;
720 }
721
722 if ((led->wled_cfg->ctrl_delay_us % WLED_CTL_DLY_STEP) ||
723 (led->wled_cfg->ctrl_delay_us > WLED_CTL_DLY_MAX)) {
724 dev_err(led->dev->parent, "Invalid control delay\n");
725 return rc;
726 }
727
728 val = val / WLED_CTL_DLY_STEP;
729 val = (val & ~WLED_CTL_DLY_MASK) |
730 (led->wled_cfg->ctrl_delay_us << WLED_CTL_DLY_BIT_SHFT);
731
732 if ((led->max_current > WLED_MAX_CURR)) {
733 dev_err(led->dev->parent, "Invalid max current\n");
734 return -EINVAL;
735 }
736
737 val = (val & ~WLED_MAX_CURR_MASK) | led->max_current;
738
739 rc = pm8xxx_writeb(led->dev->parent,
740 WLED_MAX_CURR_CFG_REG(i + 2), val);
741 if (rc) {
742 dev_err(led->dev->parent, "can't write wled max current"
743 " config register rc=%d\n", rc);
744 return rc;
745 }
746 }
747
748 /* program digital module generator, cs out and enable the module */
749 rc = pm8xxx_readb(led->dev->parent, WLED_MOD_CTRL_REG, &val);
750 if (rc) {
751 dev_err(led->dev->parent, "can't read wled module ctrl"
752 " register rc=%d\n", rc);
753 return rc;
754 }
755
756 if (led->wled_cfg->dig_mod_gen_en)
757 val |= WLED_DIG_MOD_GEN_MASK;
758
759 if (led->wled_cfg->cs_out_en)
760 val |= WLED_CS_OUT_MASK;
761
762 val |= WLED_EN_MASK;
763
764 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG, val);
765 if (rc) {
766 dev_err(led->dev->parent, "can't write wled module ctrl"
767 " register rc=%d\n", rc);
768 return rc;
769 }
Amy Maloche56913f52012-05-11 10:47:24 -0700770 led->wled_mod_ctrl_val = val;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530771
772 /* dump wled registers */
773 wled_dump_regs(led);
774
775 return 0;
776}
777
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700778static int __devinit get_init_value(struct pm8xxx_led_data *led, u8 *val)
779{
780 int rc, offset;
781 u16 addr;
782
783 switch (led->id) {
784 case PM8XXX_ID_LED_KB_LIGHT:
785 addr = SSBI_REG_ADDR_DRV_KEYPAD;
786 break;
787 case PM8XXX_ID_LED_0:
788 case PM8XXX_ID_LED_1:
789 case PM8XXX_ID_LED_2:
790 offset = PM8XXX_LED_OFFSET(led->id);
791 addr = SSBI_REG_ADDR_LED_CTRL(offset);
792 break;
793 case PM8XXX_ID_FLASH_LED_0:
794 addr = SSBI_REG_ADDR_FLASH_DRV0;
795 break;
796 case PM8XXX_ID_FLASH_LED_1:
797 addr = SSBI_REG_ADDR_FLASH_DRV1;
798 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530799 case PM8XXX_ID_WLED:
800 rc = init_wled(led);
801 if (rc)
802 dev_err(led->cdev.dev, "can't initialize wled rc=%d\n",
803 rc);
804 return rc;
Amy Malochec17c3732012-02-27 18:34:07 -0800805 case PM8XXX_ID_RGB_LED_RED:
806 case PM8XXX_ID_RGB_LED_GREEN:
807 case PM8XXX_ID_RGB_LED_BLUE:
Amy Malochec17c3732012-02-27 18:34:07 -0800808 addr = SSBI_REG_ADDR_RGB_CNTL1;
809 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530810 default:
811 dev_err(led->cdev.dev, "unknown led id %d", led->id);
812 return -EINVAL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700813 }
814
815 rc = pm8xxx_readb(led->dev->parent, addr, val);
816 if (rc)
817 dev_err(led->cdev.dev, "can't get led(%d) level rc=%d\n",
818 led->id, rc);
819
820 return rc;
821}
822
Jay Chokshi868312e2011-09-16 13:57:13 -0700823static int pm8xxx_led_pwm_configure(struct pm8xxx_led_data *led)
824{
samin.ryue0041de2012-08-03 23:58:14 +0900825 int duty_us, rc;
Jay Chokshi868312e2011-09-16 13:57:13 -0700826
827 led->pwm_dev = pwm_request(led->pwm_channel,
828 led->cdev.name);
829
830 if (IS_ERR_OR_NULL(led->pwm_dev)) {
831 pr_err("could not acquire PWM Channel %d, "
832 "error %ld\n", led->pwm_channel,
833 PTR_ERR(led->pwm_dev));
834 led->pwm_dev = NULL;
835 return -ENODEV;
836 }
837
838 if (led->pwm_duty_cycles != NULL) {
samin.ryue0041de2012-08-03 23:58:14 +0900839 rc = pm8xxx_led_pwm_pattern_update(led);
Jay Chokshi868312e2011-09-16 13:57:13 -0700840 } else {
841 duty_us = led->pwm_period_us;
842 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
843 }
844
845 return rc;
846}
847
samin.ryue0041de2012-08-03 23:58:14 +0900848static ssize_t pm8xxx_led_lock_update_show(struct device *dev,
849 struct device_attribute *attr, char *buf)
850{
851 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
852 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
853 int i, n = 0;
854
855 for (i = 0; i < pdata->num_configs; i++)
856 {
857 n += sprintf(&buf[n], "%s is %s\n",
858 leds[i].cdev.name,
859 (leds[i].lock_update ? "non-updatable" : "updatable"));
860 }
861
862 return n;
863}
864
865static ssize_t pm8xxx_led_lock_update_store(struct device *dev,
866 struct device_attribute *attr, const char *buf, size_t size)
867{
868 struct pm8xxx_led_platform_data *pdata = dev->platform_data;
869 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
870 ssize_t rc = -EINVAL;
871 char *after;
872 unsigned long state = simple_strtoul(buf, &after, 10);
873 size_t count = after - buf;
874 int i;
875
876 if (isspace(*after))
877 count++;
878
879 if (count == size) {
880 rc = count;
881 for (i = 0; i < pdata->num_configs; i++)
882 {
883 leds[i].lock_update = state;
884 if (!state) {
885 dev_info(dev, "resume %s set %d\n",
886 leds[i].cdev.name, leds[i].cdev.brightness);
887 schedule_work(&leds[i].work);
888 }
889 }
890 }
891 return rc;
892}
893
894static DEVICE_ATTR(lock, 0644, pm8xxx_led_lock_update_show, pm8xxx_led_lock_update_store);
895
896static ssize_t pm8xxx_led_grppwm_show(struct device *dev,
897 struct device_attribute *attr, char *buf)
898{
899 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
900 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
901 int i, n = 0;
902
903 for (i = 0; i < pdata->num_configs; i++)
904 {
905 n += sprintf(&buf[n], "%s period_us is %d\n", leds[i].cdev.name, leds[i].pwm_period_us);
906 }
907 return n;
908}
909
910static ssize_t pm8xxx_led_grppwm_store(struct device *dev,
911 struct device_attribute *attr, const char *buf, size_t size)
912{
913 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
914 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
915 ssize_t rc = -EINVAL;
916 char *after;
917 unsigned long state = simple_strtoul(buf, &after, 10);
918 size_t count = after - buf;
919 int i;
920
921 if (isspace(*after))
922 count++;
923
924 if (count == size) {
925 rc = count;
926
927 if(state < PM8XXX_PWM_PERIOD_MIN)
928 state = PM8XXX_PWM_PERIOD_MIN;
929 if(state > PM8XXX_PWM_PERIOD_MAX)
930 state = PM8XXX_PWM_PERIOD_MAX;
931
932 for (i = 0; i < pdata->num_configs; i++)
933 {
934 leds[i].pwm_period_us = state;
935 }
936 }
937 return rc;
938}
939
940static DEVICE_ATTR(grppwm, 0644, pm8xxx_led_grppwm_show, pm8xxx_led_grppwm_store);
941
942static ssize_t pm8xxx_led_grpfreq_show(struct device *dev,
943 struct device_attribute *attr, char *buf)
944{
945 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
946 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
947 int i, n = 0;
948
949 for (i = 0; i < pdata->num_configs; i++)
950 {
951 if (leds[i].pwm_duty_cycles != NULL) {
952 n += sprintf(&buf[n], "%s freq %d\n", leds[i].cdev.name, leds[i].pwm_duty_cycles->duty_ms);
953 }
954 else
955 n += sprintf(&buf[n], "%s's duty cycles is not exist\n", leds[i].cdev.name);
956
957 }
958 return n;
959}
960
961static ssize_t pm8xxx_led_grpfreq_store(struct device *dev,
962 struct device_attribute *attr, const char *buf, size_t size)
963{
964 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
965 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
966 ssize_t rc = -EINVAL;
967 char *after;
968 unsigned long state = simple_strtoul(buf, &after, 10);
969 size_t count = after - buf;
970 int i;
971
972 if (isspace(*after))
973 count++;
974
975 if (count == size) {
976 rc = count;
977
978 if(state == 0)
979 state = 32;
980 if(state > PM_PWM_LUT_DUTY_TIME_MAX)
981 state = PM_PWM_LUT_DUTY_TIME_MAX;
982
983 for (i = 0; i < pdata->num_configs; i++)
984 {
985 if (leds[i].pwm_duty_cycles != NULL) {
986 leds[i].pwm_duty_cycles->duty_ms = state;
987 }
988 }
989 }
990 return rc;
991}
992
993static DEVICE_ATTR(grpfreq, 0644, pm8xxx_led_grpfreq_show, pm8xxx_led_grpfreq_store);
994
995static ssize_t pm8xxx_led_blink_show(struct device *dev,
996 struct device_attribute *attr, char *buf)
997{
998 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
999 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
1000 int start_idx = 0, idx_len = 0;
1001 int i, j, n = 0;
1002
1003 for (i = 0; i < pdata->num_configs; i++)
1004 {
1005 n += sprintf(&buf[n], "[%d] %s pwm pattern %d\n", i, leds[i].cdev.name, leds[i].blink);
1006 if (leds[i].pwm_duty_cycles != NULL && leds[i].pwm_duty_cycles->duty_pcts != NULL) {
1007 start_idx = leds[i].pwm_duty_cycles->start_idx;
1008 idx_len = leds[i].pwm_duty_cycles->num_duty_pcts;
1009 for (j = 0; j < idx_len; j++)
1010 {
1011 n += sprintf(&buf[n], "%d ",
1012 leds[i].pwm_duty_cycles->duty_pcts[j]);
1013 }
1014 n += sprintf(&buf[n], "\n");
1015 }
1016 else
1017 n += sprintf(&buf[n], "not exist\n");
1018
1019 }
1020 return n;
1021}
1022
1023static ssize_t pm8xxx_led_blink_store(struct device *dev,
1024 struct device_attribute *attr, const char *buf, size_t size)
1025{
1026 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
1027 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
1028 ssize_t rc = -EINVAL;
1029 char *after;
1030 unsigned long state = simple_strtoul(buf, &after, 10);
1031 size_t count = after - buf;
1032 int i;
1033
1034 if (isspace(*after))
1035 count++;
1036
1037 if (count == size) {
1038 rc = count;
1039
1040 for (i = 0; i < pdata->num_configs; i++)
1041 {
1042 if (leds[i].blink != state) {
1043 leds[i].blink = state;
1044 rc = pm8xxx_led_pwm_pattern_update(&leds[i]);
1045 }
1046 }
1047 }
1048 return rc;
1049}
1050
1051static DEVICE_ATTR(blink, 0644, pm8xxx_led_blink_show, pm8xxx_led_blink_store);
1052
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301053
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001054static int __devinit pm8xxx_led_probe(struct platform_device *pdev)
1055{
Jay Chokshi8994e392011-09-14 18:20:39 -07001056 const struct pm8xxx_led_platform_data *pdata = pdev->dev.platform_data;
1057 const struct led_platform_data *pcore_data;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001058 struct led_info *curr_led;
1059 struct pm8xxx_led_data *led, *led_dat;
Jay Chokshi8994e392011-09-14 18:20:39 -07001060 struct pm8xxx_led_config *led_cfg;
Amy Maloche3d2c24c2012-03-22 19:02:25 -07001061 enum pm8xxx_version version;
1062 bool found = false;
1063 int rc, i, j;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001064
1065 if (pdata == NULL) {
1066 dev_err(&pdev->dev, "platform data not supplied\n");
1067 return -EINVAL;
1068 }
1069
Jay Chokshi8994e392011-09-14 18:20:39 -07001070 pcore_data = pdata->led_core;
1071
1072 if (pcore_data->num_leds != pdata->num_configs) {
1073 dev_err(&pdev->dev, "#no. of led configs and #no. of led"
1074 "entries are not equal\n");
1075 return -EINVAL;
1076 }
1077
1078 led = kcalloc(pcore_data->num_leds, sizeof(*led), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001079 if (led == NULL) {
1080 dev_err(&pdev->dev, "failed to alloc memory\n");
1081 return -ENOMEM;
1082 }
1083
Jay Chokshi8994e392011-09-14 18:20:39 -07001084 for (i = 0; i < pcore_data->num_leds; i++) {
1085 curr_led = &pcore_data->leds[i];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001086 led_dat = &led[i];
Jay Chokshi8994e392011-09-14 18:20:39 -07001087 led_cfg = &pdata->configs[i];
1088
1089 led_dat->id = led_cfg->id;
Jay Chokshi868312e2011-09-16 13:57:13 -07001090 led_dat->pwm_channel = led_cfg->pwm_channel;
1091 led_dat->pwm_period_us = led_cfg->pwm_period_us;
1092 led_dat->pwm_duty_cycles = led_cfg->pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301093 led_dat->wled_cfg = led_cfg->wled_cfg;
1094 led_dat->max_current = led_cfg->max_current;
samin.ryue0041de2012-08-03 23:58:14 +09001095 led_dat->lock_update = 0;
1096 led_dat->blink = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001097
1098 if (!((led_dat->id >= PM8XXX_ID_LED_KB_LIGHT) &&
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301099 (led_dat->id < PM8XXX_ID_MAX))) {
Amy Maloche3d2c24c2012-03-22 19:02:25 -07001100 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
1101 led_dat->id);
1102 rc = -EINVAL;
1103 goto fail_id_check;
1104
1105 }
1106
1107 found = false;
1108 version = pm8xxx_get_version(pdev->dev.parent);
1109 for (j = 0; j < ARRAY_SIZE(led_map); j++) {
1110 if (version == led_map[j].version
1111 && (led_map[j].supported & (1 << led_dat->id))) {
1112 found = true;
1113 break;
1114 }
1115 }
1116
1117 if (!found) {
1118 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
1119 led_dat->id);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001120 rc = -EINVAL;
1121 goto fail_id_check;
1122 }
1123
1124 led_dat->cdev.name = curr_led->name;
1125 led_dat->cdev.default_trigger = curr_led->default_trigger;
1126 led_dat->cdev.brightness_set = pm8xxx_led_set;
1127 led_dat->cdev.brightness_get = pm8xxx_led_get;
Devin Kim232d6652012-08-09 20:41:00 -07001128 led_dat->cdev.max_brightness = LED_FULL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001129 led_dat->cdev.brightness = LED_OFF;
Jay Chokshi8994e392011-09-14 18:20:39 -07001130 led_dat->cdev.flags = curr_led->flags;
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001131 led_dat->dev = &pdev->dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001132
1133 rc = get_init_value(led_dat, &led_dat->reg);
1134 if (rc < 0)
1135 goto fail_id_check;
1136
Devin Kim232d6652012-08-09 20:41:00 -07001137 rc = pm8xxx_set_led_mode_and_adjust_brightness(led_dat,
Jay Chokshi8994e392011-09-14 18:20:39 -07001138 led_cfg->mode, led_cfg->max_current);
Jay Chokshide4cefb2011-08-04 18:10:44 -07001139 if (rc < 0)
1140 goto fail_id_check;
1141
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001142 mutex_init(&led_dat->lock);
1143 INIT_WORK(&led_dat->work, pm8xxx_led_work);
1144
1145 rc = led_classdev_register(&pdev->dev, &led_dat->cdev);
1146 if (rc) {
1147 dev_err(&pdev->dev, "unable to register led %d,rc=%d\n",
1148 led_dat->id, rc);
1149 goto fail_id_check;
1150 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001151
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301152 /* configure default state */
1153 if (led_cfg->default_state)
samin.ryue0041de2012-08-03 23:58:14 +09001154 led_dat->cdev.brightness = led_dat->cdev.max_brightness;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301155 else
samin.ryue0041de2012-08-03 23:58:14 +09001156 led_dat->cdev.brightness = LED_OFF;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301157
Jay Chokshi868312e2011-09-16 13:57:13 -07001158 if (led_cfg->mode != PM8XXX_LED_MODE_MANUAL) {
Amy Malochec17c3732012-02-27 18:34:07 -08001159 if (led_dat->id == PM8XXX_ID_RGB_LED_RED ||
1160 led_dat->id == PM8XXX_ID_RGB_LED_GREEN ||
1161 led_dat->id == PM8XXX_ID_RGB_LED_BLUE)
1162 __pm8xxx_led_work(led_dat, 0);
1163 else
1164 __pm8xxx_led_work(led_dat,
Jay Chokshide4cefb2011-08-04 18:10:44 -07001165 led_dat->cdev.max_brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -07001166
1167 if (led_dat->pwm_channel != -1) {
samin.ryuc650af32012-08-29 11:34:34 +09001168 if (led_cfg->pwm_adjust_brightness) {
1169 led_dat->adjust_brightness = led_cfg->pwm_adjust_brightness;
1170 } else {
1171 led_dat->adjust_brightness = 100;
1172 }
1173
Jay Chokshi868312e2011-09-16 13:57:13 -07001174 rc = pm8xxx_led_pwm_configure(led_dat);
1175 if (rc) {
1176 dev_err(&pdev->dev, "failed to "
1177 "configure LED, error: %d\n", rc);
1178 goto fail_id_check;
1179 }
samin.ryue0041de2012-08-03 23:58:14 +09001180 schedule_work(&led->work);
Jay Chokshi868312e2011-09-16 13:57:13 -07001181 }
1182 } else {
samin.ryue0041de2012-08-03 23:58:14 +09001183 __pm8xxx_led_work(led_dat, led_dat->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -07001184 }
Jay Chokshide4cefb2011-08-04 18:10:44 -07001185 }
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001186
samin.ryue0041de2012-08-03 23:58:14 +09001187 led->use_pwm = pdata->use_pwm;
1188
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001189 platform_set_drvdata(pdev, led);
1190
samin.ryue0041de2012-08-03 23:58:14 +09001191 rc = device_create_file(&pdev->dev, &dev_attr_lock);
1192 if (rc) {
1193 device_remove_file(&pdev->dev, &dev_attr_lock);
1194 dev_err(&pdev->dev, "failed device_create_file(lock)\n");
1195 }
1196
1197 if (led->use_pwm) {
1198 rc = device_create_file(&pdev->dev, &dev_attr_blink);
1199 if (rc) {
1200 device_remove_file(&pdev->dev, &dev_attr_blink);
1201 dev_err(&pdev->dev, "failed device_create_file(blink)\n");
1202 }
1203
1204 rc = device_create_file(&pdev->dev, &dev_attr_grppwm);
1205 if (rc) {
1206 device_remove_file(&pdev->dev, &dev_attr_grppwm);
1207 dev_err(&pdev->dev, "failed device_create_fiaild(grppwm)\n");
1208 }
1209
1210 rc = device_create_file(&pdev->dev, &dev_attr_grpfreq);
1211 if (rc) {
1212 device_remove_file(&pdev->dev, &dev_attr_grpfreq);
1213 dev_err(&pdev->dev, "failed device_create_file(grpfreq)\n");
1214 }
1215 }
1216
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001217 return 0;
1218
1219fail_id_check:
1220 if (i > 0) {
1221 for (i = i - 1; i >= 0; i--) {
1222 mutex_destroy(&led[i].lock);
1223 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -07001224 if (led[i].pwm_dev != NULL)
1225 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001226 }
1227 }
1228 kfree(led);
1229 return rc;
1230}
1231
1232static int __devexit pm8xxx_led_remove(struct platform_device *pdev)
1233{
1234 int i;
1235 const struct led_platform_data *pdata =
1236 pdev->dev.platform_data;
1237 struct pm8xxx_led_data *led = platform_get_drvdata(pdev);
1238
1239 for (i = 0; i < pdata->num_leds; i++) {
1240 cancel_work_sync(&led[i].work);
1241 mutex_destroy(&led[i].lock);
1242 led_classdev_unregister(&led[i].cdev);
samin.ryue0041de2012-08-03 23:58:14 +09001243 if (led[i].pwm_dev != NULL) {
Jay Chokshi868312e2011-09-16 13:57:13 -07001244 pwm_free(led[i].pwm_dev);
samin.ryue0041de2012-08-03 23:58:14 +09001245 }
1246 }
1247
1248 device_remove_file(&pdev->dev, &dev_attr_lock);
1249
1250 if (led->use_pwm) {
1251 device_remove_file(&pdev->dev, &dev_attr_blink);
1252 device_remove_file(&pdev->dev, &dev_attr_grppwm);
1253 device_remove_file(&pdev->dev, &dev_attr_grpfreq);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001254 }
1255
1256 kfree(led);
1257
1258 return 0;
1259}
1260
1261static struct platform_driver pm8xxx_led_driver = {
1262 .probe = pm8xxx_led_probe,
1263 .remove = __devexit_p(pm8xxx_led_remove),
1264 .driver = {
1265 .name = PM8XXX_LEDS_DEV_NAME,
1266 .owner = THIS_MODULE,
1267 },
1268};
1269
1270static int __init pm8xxx_led_init(void)
1271{
1272 return platform_driver_register(&pm8xxx_led_driver);
1273}
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001274subsys_initcall(pm8xxx_led_init);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001275
1276static void __exit pm8xxx_led_exit(void)
1277{
1278 platform_driver_unregister(&pm8xxx_led_driver);
1279}
1280module_exit(pm8xxx_led_exit);
1281
1282MODULE_DESCRIPTION("PM8XXX LEDs driver");
1283MODULE_LICENSE("GPL v2");
1284MODULE_VERSION("1.0");
1285MODULE_ALIAS("platform:pm8xxx-led");