blob: f73a549e468f6141c4c893a3b1e36f47fc24f41d [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.ryudfc10982012-08-30 02:01:09 +0900109#define PM8XXX_LED_PWM_FLAGS (PM_PWM_LUT_LOOP | PM_PWM_LUT_RAMP_UP | PM_PWM_LUT_REVERSE | \
110 PM_PWM_LUT_PAUSE_LO_EN | PM_PWM_LUT_PAUSE_HI_EN)
111
112#define PM8XXX_LED_PWM_GRPFREQ_MAX 255
113#define PM8XXX_LED_PWM_GRPPWM_MAX 255
Jay Chokshi868312e2011-09-16 13:57:13 -0700114
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700115#define LED_MAP(_version, _kb, _led0, _led1, _led2, _flash_led0, _flash_led1, \
Amy Malochec17c3732012-02-27 18:34:07 -0800116 _wled, _rgb_led_red, _rgb_led_green, _rgb_led_blue)\
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700117 {\
118 .version = _version,\
119 .supported = _kb << PM8XXX_ID_LED_KB_LIGHT | \
120 _led0 << PM8XXX_ID_LED_0 | _led1 << PM8XXX_ID_LED_1 | \
121 _led2 << PM8XXX_ID_LED_2 | \
122 _flash_led0 << PM8XXX_ID_FLASH_LED_0 | \
123 _flash_led1 << PM8XXX_ID_FLASH_LED_1 | \
Amy Malochec17c3732012-02-27 18:34:07 -0800124 _wled << PM8XXX_ID_WLED | \
125 _rgb_led_red << PM8XXX_ID_RGB_LED_RED | \
126 _rgb_led_green << PM8XXX_ID_RGB_LED_GREEN | \
127 _rgb_led_blue << PM8XXX_ID_RGB_LED_BLUE, \
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700128 }
129
130/**
131 * supported_leds - leds supported for each PMIC version
132 * @version - version of PMIC
133 * @supported - which leds are supported on version
134 */
135
136struct supported_leds {
137 enum pm8xxx_version version;
138 u32 supported;
139};
140
141static const struct supported_leds led_map[] = {
Amy Malochec17c3732012-02-27 18:34:07 -0800142 LED_MAP(PM8XXX_VERSION_8058, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
143 LED_MAP(PM8XXX_VERSION_8921, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
144 LED_MAP(PM8XXX_VERSION_8018, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0),
145 LED_MAP(PM8XXX_VERSION_8922, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1),
146 LED_MAP(PM8XXX_VERSION_8038, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1),
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700147};
148
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700149/**
150 * struct pm8xxx_led_data - internal led data structure
151 * @led_classdev - led class device
152 * @id - led index
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700153 * @work - workqueue for led
154 * @lock - to protect the transactions
155 * @reg - cached value of led register
Jay Chokshi868312e2011-09-16 13:57:13 -0700156 * @pwm_dev - pointer to PWM device if LED is driven using PWM
157 * @pwm_channel - PWM channel ID
158 * @pwm_period_us - PWM period in micro seconds
159 * @pwm_duty_cycles - struct that describes PWM duty cycles info
samin.ryue0041de2012-08-03 23:58:14 +0900160 * @use_pwm - controlled by userspace
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700161 */
162struct pm8xxx_led_data {
163 struct led_classdev cdev;
164 int id;
165 u8 reg;
Amy Maloche56913f52012-05-11 10:47:24 -0700166 u8 wled_mod_ctrl_val;
samin.ryue0041de2012-08-03 23:58:14 +0900167 u8 lock_update;
168 u8 blink;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700169 struct device *dev;
170 struct work_struct work;
171 struct mutex lock;
Jay Chokshi868312e2011-09-16 13:57:13 -0700172 struct pwm_device *pwm_dev;
173 int pwm_channel;
174 u32 pwm_period_us;
175 struct pm8xxx_pwm_duty_cycles *pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530176 struct wled_config_data *wled_cfg;
177 int max_current;
samin.ryue0041de2012-08-03 23:58:14 +0900178 int use_pwm;
Devin Kim232d6652012-08-09 20:41:00 -0700179 int adjust_brightness;
samin.ryudfc10982012-08-30 02:01:09 +0900180 u16 pwm_grppwm;
181 u16 pwm_grpfreq;
182 u16 pwm_pause_hi;
183 u16 pwm_pause_lo;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700184};
185
186static void led_kp_set(struct pm8xxx_led_data *led, enum led_brightness value)
187{
188 int rc;
189 u8 level;
190
191 level = (value << PM8XXX_DRV_KEYPAD_BL_SHIFT) &
192 PM8XXX_DRV_KEYPAD_BL_MASK;
193
194 led->reg &= ~PM8XXX_DRV_KEYPAD_BL_MASK;
195 led->reg |= level;
196
197 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_DRV_KEYPAD,
198 led->reg);
199 if (rc < 0)
200 dev_err(led->cdev.dev,
201 "can't set keypad backlight level rc=%d\n", rc);
202}
203
204static void led_lc_set(struct pm8xxx_led_data *led, enum led_brightness value)
205{
206 int rc, offset;
207 u8 level;
208
209 level = (value << PM8XXX_DRV_LED_CTRL_SHIFT) &
210 PM8XXX_DRV_LED_CTRL_MASK;
211
212 offset = PM8XXX_LED_OFFSET(led->id);
213
214 led->reg &= ~PM8XXX_DRV_LED_CTRL_MASK;
215 led->reg |= level;
216
217 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_LED_CTRL(offset),
218 led->reg);
219 if (rc)
220 dev_err(led->cdev.dev, "can't set (%d) led value rc=%d\n",
221 led->id, rc);
222}
223
224static void
225led_flash_set(struct pm8xxx_led_data *led, enum led_brightness value)
226{
227 int rc;
228 u8 level;
229 u16 reg_addr;
230
231 level = (value << PM8XXX_DRV_FLASH_SHIFT) &
232 PM8XXX_DRV_FLASH_MASK;
233
234 led->reg &= ~PM8XXX_DRV_FLASH_MASK;
235 led->reg |= level;
236
237 if (led->id == PM8XXX_ID_FLASH_LED_0)
238 reg_addr = SSBI_REG_ADDR_FLASH_DRV0;
239 else
240 reg_addr = SSBI_REG_ADDR_FLASH_DRV1;
241
242 rc = pm8xxx_writeb(led->dev->parent, reg_addr, led->reg);
243 if (rc < 0)
244 dev_err(led->cdev.dev, "can't set flash led%d level rc=%d\n",
245 led->id, rc);
246}
247
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530248static int
249led_wled_set(struct pm8xxx_led_data *led, enum led_brightness value)
250{
251 int rc, duty;
252 u8 val, i, num_wled_strings;
253
254 if (value > WLED_MAX_LEVEL)
255 value = WLED_MAX_LEVEL;
256
Amy Maloche56913f52012-05-11 10:47:24 -0700257 if (value == 0) {
258 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
259 WLED_BOOST_OFF);
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 } else {
266 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
267 led->wled_mod_ctrl_val);
268 if (rc) {
269 dev_err(led->dev->parent, "can't write wled ctrl config"
270 " register rc=%d\n", rc);
271 return rc;
272 }
273 }
274
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530275 duty = (WLED_MAX_DUTY_CYCLE * value) / WLED_MAX_LEVEL;
276
277 num_wled_strings = led->wled_cfg->num_strings;
278
279 /* program brightness control registers */
280 for (i = 0; i < num_wled_strings; i++) {
281 rc = pm8xxx_readb(led->dev->parent,
282 WLED_BRIGHTNESS_CNTL_REG1(i), &val);
283 if (rc) {
284 dev_err(led->dev->parent, "can't read wled brightnes ctrl"
285 " register1 rc=%d\n", rc);
286 return rc;
287 }
288
289 val = (val & ~WLED_MAX_CURR_MASK) | (duty >> WLED_8_BIT_SHFT);
290 rc = pm8xxx_writeb(led->dev->parent,
291 WLED_BRIGHTNESS_CNTL_REG1(i), val);
292 if (rc) {
293 dev_err(led->dev->parent, "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, "can't write wled brightness ctrl"
303 " register2 rc=%d\n", rc);
304 return rc;
305 }
306 }
307
308 /* sync */
309 val = WLED_SYNC_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
317 val = WLED_SYNC_RESET_VAL;
318 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
319 if (rc) {
320 dev_err(led->dev->parent,
321 "can't read wled sync register rc=%d\n", rc);
322 return rc;
323 }
324 return 0;
325}
326
327static void wled_dump_regs(struct pm8xxx_led_data *led)
328{
329 int i;
330 u8 val;
331
332 for (i = 1; i < 17; i++) {
333 pm8xxx_readb(led->dev->parent,
334 SSBI_REG_ADDR_WLED_CTRL(i), &val);
335 pr_debug("WLED_CTRL_%d = 0x%x\n", i, val);
336 }
337}
338
Amy Malochec17c3732012-02-27 18:34:07 -0800339static void
Amy Malochead3c7842012-07-20 14:51:27 -0700340led_rgb_write(struct pm8xxx_led_data *led, u16 addr, enum led_brightness value)
Amy Malochec17c3732012-02-27 18:34:07 -0800341{
342 int rc;
343 u8 val, mask;
344
Amy Malochead3c7842012-07-20 14:51:27 -0700345 if (led->id != PM8XXX_ID_RGB_LED_BLUE &&
346 led->id != PM8XXX_ID_RGB_LED_RED &&
347 led->id != PM8XXX_ID_RGB_LED_GREEN)
348 return;
349
350 rc = pm8xxx_readb(led->dev->parent, addr, &val);
Amy Malochec17c3732012-02-27 18:34:07 -0800351 if (rc) {
352 dev_err(led->cdev.dev, "can't read rgb ctrl register rc=%d\n",
353 rc);
354 return;
355 }
356
357 switch (led->id) {
358 case PM8XXX_ID_RGB_LED_RED:
359 mask = PM8XXX_DRV_RGB_RED_LED;
360 break;
361 case PM8XXX_ID_RGB_LED_GREEN:
362 mask = PM8XXX_DRV_RGB_GREEN_LED;
363 break;
364 case PM8XXX_ID_RGB_LED_BLUE:
365 mask = PM8XXX_DRV_RGB_BLUE_LED;
366 break;
367 default:
368 return;
369 }
370
371 if (value)
372 val |= mask;
373 else
374 val &= ~mask;
375
Amy Malochead3c7842012-07-20 14:51:27 -0700376 rc = pm8xxx_writeb(led->dev->parent, addr, val);
Amy Malochec17c3732012-02-27 18:34:07 -0800377 if (rc < 0)
378 dev_err(led->cdev.dev, "can't set rgb led %d level rc=%d\n",
379 led->id, rc);
380}
381
Amy Malochead3c7842012-07-20 14:51:27 -0700382static void
383led_rgb_set(struct pm8xxx_led_data *led, enum led_brightness value)
384{
385 if (value) {
386 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, value);
387 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL2, value);
388 } else {
389 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL2, value);
390 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, value);
391 }
392}
393
Devin Kim232d6652012-08-09 20:41:00 -0700394static int pm8xxx_adjust_brightness(struct led_classdev *led_cdev,
395 enum led_brightness value)
396{
397 int level = 0;
398 struct pm8xxx_led_data *led;
399 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
400
401 if (!led->adjust_brightness)
402 return value;
403
404 if (led->adjust_brightness == led->cdev.max_brightness)
405 return value;
406
407 level = (2 * value * led->adjust_brightness
408 + led->cdev.max_brightness)
409 / (2 * led->cdev.max_brightness);
410
411 if (!level && value)
412 level = 1;
413
414 return level;
415}
416
samin.ryue0041de2012-08-03 23:58:14 +0900417static int pm8xxx_led_pwm_pattern_update(struct pm8xxx_led_data * led)
418{
419 int start_idx, idx_len;
420 int *pcts = NULL;
421 int i, rc = 0;
422 int temp = 0;
Devin Kim232d6652012-08-09 20:41:00 -0700423 int pwm_max = 0;
samin.ryudfc10982012-08-30 02:01:09 +0900424 int total_ms, on_ms;
samin.ryue0041de2012-08-03 23:58:14 +0900425
426 if (!led->pwm_duty_cycles || !led->pwm_duty_cycles->duty_pcts) {
427 dev_err(led->cdev.dev, "duty_cycles and duty_pcts is not exist\n");
428 return -EINVAL;
429 }
430
samin.ryudfc10982012-08-30 02:01:09 +0900431 if (led->pwm_grppwm > 0 && led->pwm_grpfreq > 0) {
432 total_ms = led->pwm_grpfreq * 50;
433 on_ms = (led->pwm_grppwm * total_ms) >> 8;
434 if (PM8XXX_LED_PWM_FLAGS & PM_PWM_LUT_REVERSE) {
435 led->pwm_duty_cycles->duty_ms = on_ms /
436 (led->pwm_duty_cycles->num_duty_pcts << 1);
437 led->pwm_pause_hi = on_ms %
438 (led->pwm_duty_cycles->num_duty_pcts << 1);
439 } else {
440 led->pwm_duty_cycles->duty_ms = on_ms /
441 (led->pwm_duty_cycles->num_duty_pcts);
442 led->pwm_pause_hi = on_ms %
443 (led->pwm_duty_cycles->num_duty_pcts);
444 }
445 led->pwm_pause_lo = total_ms - on_ms;
446 dev_dbg(led->cdev.dev, "duty_ms %d, pause_hi %d, pause_lo %d, total_ms %d, on_ms %d\n",
447 led->pwm_duty_cycles->duty_ms, led->pwm_pause_hi, led->pwm_pause_lo,
448 total_ms, on_ms);
449 }
450
Devin Kim232d6652012-08-09 20:41:00 -0700451 pwm_max = pm8xxx_adjust_brightness(&led->cdev, led->cdev.brightness);
samin.ryue0041de2012-08-03 23:58:14 +0900452 start_idx = led->pwm_duty_cycles->start_idx;
453 idx_len = led->pwm_duty_cycles->num_duty_pcts;
454 pcts = led->pwm_duty_cycles->duty_pcts;
455
456 for (i = 0; i < idx_len; i++)
457 {
458 if (led->blink == 1) {
459 temp = ((pwm_max * i) << 1) / (idx_len - 1);
460 pcts[i] = (temp + 1) >> 1;
461 } else {
462 pcts[i] = (pwm_max);
463 }
464 }
465
466 if (idx_len >= PM_PWM_LUT_SIZE && start_idx) {
467 pr_err("Wrong LUT size or index\n");
468 return -EINVAL;
469 }
470 if ((start_idx + idx_len) > PM_PWM_LUT_SIZE) {
471 pr_err("Exceed LUT limit\n");
472 return -EINVAL;
473 }
474
475 rc = pm8xxx_pwm_lut_config(led->pwm_dev, led->pwm_period_us,
476 led->pwm_duty_cycles->duty_pcts,
477 led->pwm_duty_cycles->duty_ms,
samin.ryudfc10982012-08-30 02:01:09 +0900478 start_idx, idx_len, led->pwm_pause_lo, led->pwm_pause_hi,
samin.ryue0041de2012-08-03 23:58:14 +0900479 PM8XXX_LED_PWM_FLAGS);
480
481 return rc;
482}
483
Jay Chokshi868312e2011-09-16 13:57:13 -0700484static int pm8xxx_led_pwm_work(struct pm8xxx_led_data *led)
485{
486 int duty_us;
487 int rc = 0;
Devin Kim232d6652012-08-09 20:41:00 -0700488 int level = 0;
489
490 level = pm8xxx_adjust_brightness(&led->cdev, led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700491
492 if (led->pwm_duty_cycles == NULL) {
Devin Kim232d6652012-08-09 20:41:00 -0700493 duty_us = (led->pwm_period_us * level) / LED_FULL;
Jay Chokshi868312e2011-09-16 13:57:13 -0700494 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
Amy Malochead3c7842012-07-20 14:51:27 -0700495 if (led->cdev.brightness) {
496 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
497 led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700498 rc = pwm_enable(led->pwm_dev);
Amy Malochead3c7842012-07-20 14:51:27 -0700499 } else {
Jay Chokshi868312e2011-09-16 13:57:13 -0700500 pwm_disable(led->pwm_dev);
Amy Malochead3c7842012-07-20 14:51:27 -0700501 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
502 led->cdev.brightness);
503 }
Jay Chokshi868312e2011-09-16 13:57:13 -0700504 } else {
Devin Kim232d6652012-08-09 20:41:00 -0700505 if (level) {
samin.ryue0041de2012-08-03 23:58:14 +0900506 pm8xxx_led_pwm_pattern_update(led);
Devin Kim232d6652012-08-09 20:41:00 -0700507 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, level);
samin.ryue0041de2012-08-03 23:58:14 +0900508 }
509
Devin Kim232d6652012-08-09 20:41:00 -0700510 rc = pm8xxx_pwm_lut_enable(led->pwm_dev, level);
511 if (!level)
512 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, level);
Jay Chokshi868312e2011-09-16 13:57:13 -0700513 }
514
515 return rc;
516}
517
Jay Chokshide4cefb2011-08-04 18:10:44 -0700518static void __pm8xxx_led_work(struct pm8xxx_led_data *led,
Devin Kim232d6652012-08-09 20:41:00 -0700519 enum led_brightness value)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700520{
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530521 int rc;
Devin Kim232d6652012-08-09 20:41:00 -0700522 int level = 0;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530523
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700524 mutex_lock(&led->lock);
525
Devin Kim232d6652012-08-09 20:41:00 -0700526 level = pm8xxx_adjust_brightness(&led->cdev, value);
527
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700528 switch (led->id) {
529 case PM8XXX_ID_LED_KB_LIGHT:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700530 led_kp_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530531 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700532 case PM8XXX_ID_LED_0:
533 case PM8XXX_ID_LED_1:
534 case PM8XXX_ID_LED_2:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700535 led_lc_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530536 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700537 case PM8XXX_ID_FLASH_LED_0:
538 case PM8XXX_ID_FLASH_LED_1:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700539 led_flash_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530540 break;
541 case PM8XXX_ID_WLED:
542 rc = led_wled_set(led, level);
543 if (rc < 0)
544 pr_err("wled brightness set failed %d\n", rc);
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_rgb_set(led, level);
550 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530551 default:
552 dev_err(led->cdev.dev, "unknown led id %d", led->id);
553 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700554 }
555
556 mutex_unlock(&led->lock);
557}
558
Jay Chokshide4cefb2011-08-04 18:10:44 -0700559static void pm8xxx_led_work(struct work_struct *work)
560{
Jay Chokshi868312e2011-09-16 13:57:13 -0700561 int rc;
562
Jay Chokshide4cefb2011-08-04 18:10:44 -0700563 struct pm8xxx_led_data *led = container_of(work,
564 struct pm8xxx_led_data, work);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700565
samin.ryue0041de2012-08-03 23:58:14 +0900566 dev_dbg(led->cdev.dev, "led %s set %d (%s mode)\n",
567 led->cdev.name, led->cdev.brightness,
568 (led->pwm_dev ? "pwm" : "manual"));
569
Jay Chokshi868312e2011-09-16 13:57:13 -0700570 if (led->pwm_dev == NULL) {
571 __pm8xxx_led_work(led, led->cdev.brightness);
572 } else {
573 rc = pm8xxx_led_pwm_work(led);
574 if (rc)
575 pr_err("could not configure PWM mode for LED:%d\n",
576 led->id);
577 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700578}
579
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700580static void pm8xxx_led_set(struct led_classdev *led_cdev,
581 enum led_brightness value)
582{
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700583 struct pm8xxx_led_data *led;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700584
585 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
586
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700587 if (value < LED_OFF || value > led->cdev.max_brightness) {
588 dev_err(led->cdev.dev, "Invalid brightness value exceeds");
589 return;
590 }
591
samin.ryue0041de2012-08-03 23:58:14 +0900592 if (!led->lock_update) {
593 schedule_work(&led->work);
594 } else {
595 dev_dbg(led->cdev.dev, "set %d pending\n",
596 value);
597 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700598}
599
Devin Kim232d6652012-08-09 20:41:00 -0700600static int pm8xxx_set_led_mode_and_adjust_brightness(struct pm8xxx_led_data *led,
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700601 enum pm8xxx_led_modes led_mode, int max_current)
602{
Jay Chokshide4cefb2011-08-04 18:10:44 -0700603 switch (led->id) {
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700604 case PM8XXX_ID_LED_0:
605 case PM8XXX_ID_LED_1:
606 case PM8XXX_ID_LED_2:
samin.ryuc650af32012-08-29 11:34:34 +0900607 led->adjust_brightness = max_current /
608 PM8XXX_ID_LED_CURRENT_FACTOR;
609 if (led->adjust_brightness > MAX_LC_LED_BRIGHTNESS)
610 led->adjust_brightness = MAX_LC_LED_BRIGHTNESS;
Willie Ruana39d1d42011-08-19 13:54:34 -0700611 led->reg = led_mode;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700612 break;
613 case PM8XXX_ID_LED_KB_LIGHT:
614 case PM8XXX_ID_FLASH_LED_0:
615 case PM8XXX_ID_FLASH_LED_1:
Devin Kim232d6652012-08-09 20:41:00 -0700616 led->adjust_brightness = max_current /
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700617 PM8XXX_ID_FLASH_CURRENT_FACTOR;
Devin Kim232d6652012-08-09 20:41:00 -0700618 if (led->adjust_brightness > MAX_FLASH_BRIGHTNESS)
619 led->adjust_brightness = MAX_FLASH_BRIGHTNESS;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700620
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700621 switch (led_mode) {
622 case PM8XXX_LED_MODE_PWM1:
623 case PM8XXX_LED_MODE_PWM2:
624 case PM8XXX_LED_MODE_PWM3:
Willie Ruana39d1d42011-08-19 13:54:34 -0700625 led->reg = PM8XXX_FLASH_MODE_PWM;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700626 break;
627 case PM8XXX_LED_MODE_DTEST1:
Willie Ruana39d1d42011-08-19 13:54:34 -0700628 led->reg = PM8XXX_FLASH_MODE_DBUS1;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700629 break;
630 case PM8XXX_LED_MODE_DTEST2:
Willie Ruana39d1d42011-08-19 13:54:34 -0700631 led->reg = PM8XXX_FLASH_MODE_DBUS2;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700632 break;
633 default:
Willie Ruana39d1d42011-08-19 13:54:34 -0700634 led->reg = PM8XXX_LED_MODE_MANUAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700635 break;
636 }
637 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530638 case PM8XXX_ID_WLED:
Devin Kim232d6652012-08-09 20:41:00 -0700639 led->adjust_brightness = WLED_MAX_LEVEL;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530640 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800641 case PM8XXX_ID_RGB_LED_RED:
642 case PM8XXX_ID_RGB_LED_GREEN:
643 case PM8XXX_ID_RGB_LED_BLUE:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700644 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800645 default:
646 dev_err(led->cdev.dev, "LED Id is invalid");
647 return -EINVAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700648 }
649
Amy Malochec17c3732012-02-27 18:34:07 -0800650 return 0;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700651}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700652
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700653static enum led_brightness pm8xxx_led_get(struct led_classdev *led_cdev)
654{
655 struct pm8xxx_led_data *led;
656
657 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
658
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700659 return led->cdev.brightness;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700660}
661
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530662static int __devinit init_wled(struct pm8xxx_led_data *led)
663{
664 int rc, i;
665 u8 val, num_wled_strings;
666
667 num_wled_strings = led->wled_cfg->num_strings;
668
669 /* program over voltage protection threshold */
670 if (led->wled_cfg->ovp_val > WLED_OVP_37V) {
671 dev_err(led->dev->parent, "Invalid ovp value");
672 return -EINVAL;
673 }
674
675 rc = pm8xxx_readb(led->dev->parent, WLED_OVP_CFG_REG, &val);
676 if (rc) {
677 dev_err(led->dev->parent, "can't read wled ovp config"
678 " register rc=%d\n", rc);
679 return rc;
680 }
681
682 val = (val & ~WLED_OVP_VAL_MASK) |
683 (led->wled_cfg->ovp_val << WLED_OVP_VAL_BIT_SHFT);
684
685 rc = pm8xxx_writeb(led->dev->parent, WLED_OVP_CFG_REG, val);
686 if (rc) {
687 dev_err(led->dev->parent, "can't write wled ovp config"
688 " register rc=%d\n", rc);
689 return rc;
690 }
691
692 /* program current boost limit and output feedback*/
693 if (led->wled_cfg->boost_curr_lim > WLED_CURR_LIMIT_1680mA) {
694 dev_err(led->dev->parent, "Invalid boost current limit");
695 return -EINVAL;
696 }
697
698 rc = pm8xxx_readb(led->dev->parent, WLED_BOOST_CFG_REG, &val);
699 if (rc) {
700 dev_err(led->dev->parent, "can't read wled boost config"
701 " register rc=%d\n", rc);
702 return rc;
703 }
704
705 val = (val & ~WLED_BOOST_LIMIT_MASK) |
706 (led->wled_cfg->boost_curr_lim << WLED_BOOST_LIMIT_BIT_SHFT);
707
708 val = (val & ~WLED_OP_FDBCK_MASK) |
709 (led->wled_cfg->op_fdbck << WLED_OP_FDBCK_BIT_SHFT);
710
711 rc = pm8xxx_writeb(led->dev->parent, WLED_BOOST_CFG_REG, val);
712 if (rc) {
713 dev_err(led->dev->parent, "can't write wled boost config"
714 " register rc=%d\n", rc);
715 return rc;
716 }
717
718 /* program high pole capacitance */
719 if (led->wled_cfg->cp_select > WLED_CP_SELECT_MAX) {
720 dev_err(led->dev->parent, "Invalid pole capacitance");
721 return -EINVAL;
722 }
723
724 rc = pm8xxx_readb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, &val);
725 if (rc) {
726 dev_err(led->dev->parent, "can't read wled high pole"
727 " capacitance register rc=%d\n", rc);
728 return rc;
729 }
730
731 val = (val & ~WLED_CP_SELECT_MASK) | led->wled_cfg->cp_select;
732
733 rc = pm8xxx_writeb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, val);
734 if (rc) {
735 dev_err(led->dev->parent, "can't write wled high pole"
736 " capacitance register rc=%d\n", rc);
737 return rc;
738 }
739
740 /* program activation delay and maximum current */
741 for (i = 0; i < num_wled_strings; i++) {
742 rc = pm8xxx_readb(led->dev->parent,
743 WLED_MAX_CURR_CFG_REG(i + 2), &val);
744 if (rc) {
745 dev_err(led->dev->parent, "can't read wled max current"
746 " config register rc=%d\n", rc);
747 return rc;
748 }
749
750 if ((led->wled_cfg->ctrl_delay_us % WLED_CTL_DLY_STEP) ||
751 (led->wled_cfg->ctrl_delay_us > WLED_CTL_DLY_MAX)) {
752 dev_err(led->dev->parent, "Invalid control delay\n");
753 return rc;
754 }
755
756 val = val / WLED_CTL_DLY_STEP;
757 val = (val & ~WLED_CTL_DLY_MASK) |
758 (led->wled_cfg->ctrl_delay_us << WLED_CTL_DLY_BIT_SHFT);
759
760 if ((led->max_current > WLED_MAX_CURR)) {
761 dev_err(led->dev->parent, "Invalid max current\n");
762 return -EINVAL;
763 }
764
765 val = (val & ~WLED_MAX_CURR_MASK) | led->max_current;
766
767 rc = pm8xxx_writeb(led->dev->parent,
768 WLED_MAX_CURR_CFG_REG(i + 2), val);
769 if (rc) {
770 dev_err(led->dev->parent, "can't write wled max current"
771 " config register rc=%d\n", rc);
772 return rc;
773 }
774 }
775
776 /* program digital module generator, cs out and enable the module */
777 rc = pm8xxx_readb(led->dev->parent, WLED_MOD_CTRL_REG, &val);
778 if (rc) {
779 dev_err(led->dev->parent, "can't read wled module ctrl"
780 " register rc=%d\n", rc);
781 return rc;
782 }
783
784 if (led->wled_cfg->dig_mod_gen_en)
785 val |= WLED_DIG_MOD_GEN_MASK;
786
787 if (led->wled_cfg->cs_out_en)
788 val |= WLED_CS_OUT_MASK;
789
790 val |= WLED_EN_MASK;
791
792 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG, val);
793 if (rc) {
794 dev_err(led->dev->parent, "can't write wled module ctrl"
795 " register rc=%d\n", rc);
796 return rc;
797 }
Amy Maloche56913f52012-05-11 10:47:24 -0700798 led->wled_mod_ctrl_val = val;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530799
800 /* dump wled registers */
801 wled_dump_regs(led);
802
803 return 0;
804}
805
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700806static int __devinit get_init_value(struct pm8xxx_led_data *led, u8 *val)
807{
808 int rc, offset;
809 u16 addr;
810
811 switch (led->id) {
812 case PM8XXX_ID_LED_KB_LIGHT:
813 addr = SSBI_REG_ADDR_DRV_KEYPAD;
814 break;
815 case PM8XXX_ID_LED_0:
816 case PM8XXX_ID_LED_1:
817 case PM8XXX_ID_LED_2:
818 offset = PM8XXX_LED_OFFSET(led->id);
819 addr = SSBI_REG_ADDR_LED_CTRL(offset);
820 break;
821 case PM8XXX_ID_FLASH_LED_0:
822 addr = SSBI_REG_ADDR_FLASH_DRV0;
823 break;
824 case PM8XXX_ID_FLASH_LED_1:
825 addr = SSBI_REG_ADDR_FLASH_DRV1;
826 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530827 case PM8XXX_ID_WLED:
828 rc = init_wled(led);
829 if (rc)
830 dev_err(led->cdev.dev, "can't initialize wled rc=%d\n",
831 rc);
832 return rc;
Amy Malochec17c3732012-02-27 18:34:07 -0800833 case PM8XXX_ID_RGB_LED_RED:
834 case PM8XXX_ID_RGB_LED_GREEN:
835 case PM8XXX_ID_RGB_LED_BLUE:
Amy Malochec17c3732012-02-27 18:34:07 -0800836 addr = SSBI_REG_ADDR_RGB_CNTL1;
837 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530838 default:
839 dev_err(led->cdev.dev, "unknown led id %d", led->id);
840 return -EINVAL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700841 }
842
843 rc = pm8xxx_readb(led->dev->parent, addr, val);
844 if (rc)
845 dev_err(led->cdev.dev, "can't get led(%d) level rc=%d\n",
846 led->id, rc);
847
848 return rc;
849}
850
Jay Chokshi868312e2011-09-16 13:57:13 -0700851static int pm8xxx_led_pwm_configure(struct pm8xxx_led_data *led)
852{
samin.ryue0041de2012-08-03 23:58:14 +0900853 int duty_us, rc;
Jay Chokshi868312e2011-09-16 13:57:13 -0700854
855 led->pwm_dev = pwm_request(led->pwm_channel,
856 led->cdev.name);
857
858 if (IS_ERR_OR_NULL(led->pwm_dev)) {
859 pr_err("could not acquire PWM Channel %d, "
860 "error %ld\n", led->pwm_channel,
861 PTR_ERR(led->pwm_dev));
862 led->pwm_dev = NULL;
863 return -ENODEV;
864 }
865
866 if (led->pwm_duty_cycles != NULL) {
samin.ryue0041de2012-08-03 23:58:14 +0900867 rc = pm8xxx_led_pwm_pattern_update(led);
Jay Chokshi868312e2011-09-16 13:57:13 -0700868 } else {
869 duty_us = led->pwm_period_us;
870 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
871 }
872
873 return rc;
874}
875
samin.ryue0041de2012-08-03 23:58:14 +0900876static ssize_t pm8xxx_led_lock_update_show(struct device *dev,
877 struct device_attribute *attr, char *buf)
878{
879 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
880 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
881 int i, n = 0;
882
883 for (i = 0; i < pdata->num_configs; i++)
884 {
885 n += sprintf(&buf[n], "%s is %s\n",
886 leds[i].cdev.name,
887 (leds[i].lock_update ? "non-updatable" : "updatable"));
888 }
889
890 return n;
891}
892
893static ssize_t pm8xxx_led_lock_update_store(struct device *dev,
894 struct device_attribute *attr, const char *buf, size_t size)
895{
896 struct pm8xxx_led_platform_data *pdata = dev->platform_data;
897 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
898 ssize_t rc = -EINVAL;
899 char *after;
900 unsigned long state = simple_strtoul(buf, &after, 10);
901 size_t count = after - buf;
902 int i;
903
904 if (isspace(*after))
905 count++;
906
907 if (count == size) {
908 rc = count;
909 for (i = 0; i < pdata->num_configs; i++)
910 {
911 leds[i].lock_update = state;
912 if (!state) {
913 dev_info(dev, "resume %s set %d\n",
914 leds[i].cdev.name, leds[i].cdev.brightness);
915 schedule_work(&leds[i].work);
916 }
917 }
918 }
919 return rc;
920}
921
922static DEVICE_ATTR(lock, 0644, pm8xxx_led_lock_update_show, pm8xxx_led_lock_update_store);
923
924static ssize_t pm8xxx_led_grppwm_show(struct device *dev,
925 struct device_attribute *attr, char *buf)
926{
927 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
928 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
929 int i, n = 0;
930
931 for (i = 0; i < pdata->num_configs; i++)
932 {
samin.ryudfc10982012-08-30 02:01:09 +0900933 n += sprintf(&buf[n], "%s period_us is %d\n", leds[i].cdev.name, leds[i].pwm_grppwm);
samin.ryue0041de2012-08-03 23:58:14 +0900934 }
935 return n;
936}
937
938static ssize_t pm8xxx_led_grppwm_store(struct device *dev,
939 struct device_attribute *attr, const char *buf, size_t size)
940{
941 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
942 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
943 ssize_t rc = -EINVAL;
944 char *after;
945 unsigned long state = simple_strtoul(buf, &after, 10);
946 size_t count = after - buf;
947 int i;
948
949 if (isspace(*after))
950 count++;
951
952 if (count == size) {
953 rc = count;
samin.ryudfc10982012-08-30 02:01:09 +0900954 if (state < 0)
955 state = 0;
956 if (state > PM8XXX_LED_PWM_GRPPWM_MAX)
957 state = PM8XXX_LED_PWM_GRPPWM_MAX;
samin.ryue0041de2012-08-03 23:58:14 +0900958
959 for (i = 0; i < pdata->num_configs; i++)
960 {
samin.ryudfc10982012-08-30 02:01:09 +0900961 leds[i].pwm_grppwm = state;
962 dev_dbg(leds[i].cdev.dev, "set grppwm %lu\n", state);
samin.ryue0041de2012-08-03 23:58:14 +0900963 }
964 }
965 return rc;
966}
967
968static DEVICE_ATTR(grppwm, 0644, pm8xxx_led_grppwm_show, pm8xxx_led_grppwm_store);
969
970static ssize_t pm8xxx_led_grpfreq_show(struct device *dev,
971 struct device_attribute *attr, char *buf)
972{
973 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
974 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
975 int i, n = 0;
976
977 for (i = 0; i < pdata->num_configs; i++)
978 {
samin.ryudfc10982012-08-30 02:01:09 +0900979 n += sprintf(&buf[n], "%s freq %d\n", leds[i].cdev.name, leds[i].pwm_grpfreq);
samin.ryue0041de2012-08-03 23:58:14 +0900980 }
981 return n;
982}
983
984static ssize_t pm8xxx_led_grpfreq_store(struct device *dev,
985 struct device_attribute *attr, const char *buf, size_t size)
986{
987 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
988 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
989 ssize_t rc = -EINVAL;
990 char *after;
991 unsigned long state = simple_strtoul(buf, &after, 10);
992 size_t count = after - buf;
993 int i;
994
995 if (isspace(*after))
996 count++;
997
998 if (count == size) {
999 rc = count;
1000
samin.ryudfc10982012-08-30 02:01:09 +09001001 if(state < 0)
1002 state = 0;
1003 if(state > PM8XXX_LED_PWM_GRPFREQ_MAX)
1004 state = PM8XXX_LED_PWM_GRPFREQ_MAX;
samin.ryue0041de2012-08-03 23:58:14 +09001005
1006 for (i = 0; i < pdata->num_configs; i++)
1007 {
samin.ryudfc10982012-08-30 02:01:09 +09001008 leds[i].pwm_grpfreq = state;
1009 dev_dbg(leds[i].cdev.dev, "set grpfreq %lu\n", state);
samin.ryue0041de2012-08-03 23:58:14 +09001010 }
1011 }
1012 return rc;
1013}
1014
1015static DEVICE_ATTR(grpfreq, 0644, pm8xxx_led_grpfreq_show, pm8xxx_led_grpfreq_store);
1016
1017static ssize_t pm8xxx_led_blink_show(struct device *dev,
1018 struct device_attribute *attr, char *buf)
1019{
1020 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
1021 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
1022 int start_idx = 0, idx_len = 0;
1023 int i, j, n = 0;
1024
1025 for (i = 0; i < pdata->num_configs; i++)
1026 {
1027 n += sprintf(&buf[n], "[%d] %s pwm pattern %d\n", i, leds[i].cdev.name, leds[i].blink);
1028 if (leds[i].pwm_duty_cycles != NULL && leds[i].pwm_duty_cycles->duty_pcts != NULL) {
1029 start_idx = leds[i].pwm_duty_cycles->start_idx;
1030 idx_len = leds[i].pwm_duty_cycles->num_duty_pcts;
1031 for (j = 0; j < idx_len; j++)
1032 {
1033 n += sprintf(&buf[n], "%d ",
1034 leds[i].pwm_duty_cycles->duty_pcts[j]);
1035 }
1036 n += sprintf(&buf[n], "\n");
1037 }
1038 else
1039 n += sprintf(&buf[n], "not exist\n");
1040
1041 }
1042 return n;
1043}
1044
1045static ssize_t pm8xxx_led_blink_store(struct device *dev,
1046 struct device_attribute *attr, const char *buf, size_t size)
1047{
1048 const struct pm8xxx_led_platform_data *pdata = dev->platform_data;
1049 struct pm8xxx_led_data *leds = (struct pm8xxx_led_data *)dev_get_drvdata(dev);
1050 ssize_t rc = -EINVAL;
1051 char *after;
1052 unsigned long state = simple_strtoul(buf, &after, 10);
1053 size_t count = after - buf;
1054 int i;
1055
1056 if (isspace(*after))
1057 count++;
1058
1059 if (count == size) {
1060 rc = count;
1061
1062 for (i = 0; i < pdata->num_configs; i++)
1063 {
1064 if (leds[i].blink != state) {
1065 leds[i].blink = state;
1066 rc = pm8xxx_led_pwm_pattern_update(&leds[i]);
1067 }
1068 }
1069 }
1070 return rc;
1071}
1072
1073static DEVICE_ATTR(blink, 0644, pm8xxx_led_blink_show, pm8xxx_led_blink_store);
1074
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301075
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001076static int __devinit pm8xxx_led_probe(struct platform_device *pdev)
1077{
Jay Chokshi8994e392011-09-14 18:20:39 -07001078 const struct pm8xxx_led_platform_data *pdata = pdev->dev.platform_data;
1079 const struct led_platform_data *pcore_data;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001080 struct led_info *curr_led;
1081 struct pm8xxx_led_data *led, *led_dat;
Jay Chokshi8994e392011-09-14 18:20:39 -07001082 struct pm8xxx_led_config *led_cfg;
Amy Maloche3d2c24c2012-03-22 19:02:25 -07001083 enum pm8xxx_version version;
1084 bool found = false;
1085 int rc, i, j;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001086
1087 if (pdata == NULL) {
1088 dev_err(&pdev->dev, "platform data not supplied\n");
1089 return -EINVAL;
1090 }
1091
Jay Chokshi8994e392011-09-14 18:20:39 -07001092 pcore_data = pdata->led_core;
1093
1094 if (pcore_data->num_leds != pdata->num_configs) {
1095 dev_err(&pdev->dev, "#no. of led configs and #no. of led"
1096 "entries are not equal\n");
1097 return -EINVAL;
1098 }
1099
1100 led = kcalloc(pcore_data->num_leds, sizeof(*led), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001101 if (led == NULL) {
1102 dev_err(&pdev->dev, "failed to alloc memory\n");
1103 return -ENOMEM;
1104 }
1105
Jay Chokshi8994e392011-09-14 18:20:39 -07001106 for (i = 0; i < pcore_data->num_leds; i++) {
1107 curr_led = &pcore_data->leds[i];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001108 led_dat = &led[i];
Jay Chokshi8994e392011-09-14 18:20:39 -07001109 led_cfg = &pdata->configs[i];
1110
1111 led_dat->id = led_cfg->id;
Jay Chokshi868312e2011-09-16 13:57:13 -07001112 led_dat->pwm_channel = led_cfg->pwm_channel;
1113 led_dat->pwm_period_us = led_cfg->pwm_period_us;
1114 led_dat->pwm_duty_cycles = led_cfg->pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301115 led_dat->wled_cfg = led_cfg->wled_cfg;
1116 led_dat->max_current = led_cfg->max_current;
samin.ryue0041de2012-08-03 23:58:14 +09001117 led_dat->lock_update = 0;
1118 led_dat->blink = 0;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001119
1120 if (!((led_dat->id >= PM8XXX_ID_LED_KB_LIGHT) &&
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301121 (led_dat->id < PM8XXX_ID_MAX))) {
Amy Maloche3d2c24c2012-03-22 19:02:25 -07001122 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
1123 led_dat->id);
1124 rc = -EINVAL;
1125 goto fail_id_check;
1126
1127 }
1128
1129 found = false;
1130 version = pm8xxx_get_version(pdev->dev.parent);
1131 for (j = 0; j < ARRAY_SIZE(led_map); j++) {
1132 if (version == led_map[j].version
1133 && (led_map[j].supported & (1 << led_dat->id))) {
1134 found = true;
1135 break;
1136 }
1137 }
1138
1139 if (!found) {
1140 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
1141 led_dat->id);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001142 rc = -EINVAL;
1143 goto fail_id_check;
1144 }
1145
1146 led_dat->cdev.name = curr_led->name;
1147 led_dat->cdev.default_trigger = curr_led->default_trigger;
1148 led_dat->cdev.brightness_set = pm8xxx_led_set;
1149 led_dat->cdev.brightness_get = pm8xxx_led_get;
Devin Kim232d6652012-08-09 20:41:00 -07001150 led_dat->cdev.max_brightness = LED_FULL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001151 led_dat->cdev.brightness = LED_OFF;
Jay Chokshi8994e392011-09-14 18:20:39 -07001152 led_dat->cdev.flags = curr_led->flags;
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001153 led_dat->dev = &pdev->dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001154
1155 rc = get_init_value(led_dat, &led_dat->reg);
1156 if (rc < 0)
1157 goto fail_id_check;
1158
Devin Kim232d6652012-08-09 20:41:00 -07001159 rc = pm8xxx_set_led_mode_and_adjust_brightness(led_dat,
Jay Chokshi8994e392011-09-14 18:20:39 -07001160 led_cfg->mode, led_cfg->max_current);
Jay Chokshide4cefb2011-08-04 18:10:44 -07001161 if (rc < 0)
1162 goto fail_id_check;
1163
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001164 mutex_init(&led_dat->lock);
1165 INIT_WORK(&led_dat->work, pm8xxx_led_work);
1166
1167 rc = led_classdev_register(&pdev->dev, &led_dat->cdev);
1168 if (rc) {
1169 dev_err(&pdev->dev, "unable to register led %d,rc=%d\n",
1170 led_dat->id, rc);
1171 goto fail_id_check;
1172 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001173
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301174 /* configure default state */
1175 if (led_cfg->default_state)
samin.ryue0041de2012-08-03 23:58:14 +09001176 led_dat->cdev.brightness = led_dat->cdev.max_brightness;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301177 else
samin.ryue0041de2012-08-03 23:58:14 +09001178 led_dat->cdev.brightness = LED_OFF;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +05301179
Jay Chokshi868312e2011-09-16 13:57:13 -07001180 if (led_cfg->mode != PM8XXX_LED_MODE_MANUAL) {
Amy Malochec17c3732012-02-27 18:34:07 -08001181 if (led_dat->id == PM8XXX_ID_RGB_LED_RED ||
1182 led_dat->id == PM8XXX_ID_RGB_LED_GREEN ||
1183 led_dat->id == PM8XXX_ID_RGB_LED_BLUE)
1184 __pm8xxx_led_work(led_dat, 0);
1185 else
1186 __pm8xxx_led_work(led_dat,
Jay Chokshide4cefb2011-08-04 18:10:44 -07001187 led_dat->cdev.max_brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -07001188
1189 if (led_dat->pwm_channel != -1) {
samin.ryuc650af32012-08-29 11:34:34 +09001190 if (led_cfg->pwm_adjust_brightness) {
1191 led_dat->adjust_brightness = led_cfg->pwm_adjust_brightness;
1192 } else {
1193 led_dat->adjust_brightness = 100;
1194 }
1195
Jay Chokshi868312e2011-09-16 13:57:13 -07001196 rc = pm8xxx_led_pwm_configure(led_dat);
1197 if (rc) {
1198 dev_err(&pdev->dev, "failed to "
1199 "configure LED, error: %d\n", rc);
1200 goto fail_id_check;
1201 }
samin.ryue0041de2012-08-03 23:58:14 +09001202 schedule_work(&led->work);
Jay Chokshi868312e2011-09-16 13:57:13 -07001203 }
1204 } else {
samin.ryue0041de2012-08-03 23:58:14 +09001205 __pm8xxx_led_work(led_dat, led_dat->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -07001206 }
Jay Chokshide4cefb2011-08-04 18:10:44 -07001207 }
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001208
samin.ryue0041de2012-08-03 23:58:14 +09001209 led->use_pwm = pdata->use_pwm;
1210
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001211 platform_set_drvdata(pdev, led);
1212
samin.ryue0041de2012-08-03 23:58:14 +09001213 rc = device_create_file(&pdev->dev, &dev_attr_lock);
1214 if (rc) {
1215 device_remove_file(&pdev->dev, &dev_attr_lock);
1216 dev_err(&pdev->dev, "failed device_create_file(lock)\n");
1217 }
1218
1219 if (led->use_pwm) {
1220 rc = device_create_file(&pdev->dev, &dev_attr_blink);
1221 if (rc) {
1222 device_remove_file(&pdev->dev, &dev_attr_blink);
1223 dev_err(&pdev->dev, "failed device_create_file(blink)\n");
1224 }
1225
1226 rc = device_create_file(&pdev->dev, &dev_attr_grppwm);
1227 if (rc) {
1228 device_remove_file(&pdev->dev, &dev_attr_grppwm);
1229 dev_err(&pdev->dev, "failed device_create_fiaild(grppwm)\n");
1230 }
1231
1232 rc = device_create_file(&pdev->dev, &dev_attr_grpfreq);
1233 if (rc) {
1234 device_remove_file(&pdev->dev, &dev_attr_grpfreq);
1235 dev_err(&pdev->dev, "failed device_create_file(grpfreq)\n");
1236 }
1237 }
1238
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001239 return 0;
1240
1241fail_id_check:
1242 if (i > 0) {
1243 for (i = i - 1; i >= 0; i--) {
1244 mutex_destroy(&led[i].lock);
1245 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -07001246 if (led[i].pwm_dev != NULL)
1247 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001248 }
1249 }
1250 kfree(led);
1251 return rc;
1252}
1253
1254static int __devexit pm8xxx_led_remove(struct platform_device *pdev)
1255{
1256 int i;
1257 const struct led_platform_data *pdata =
1258 pdev->dev.platform_data;
1259 struct pm8xxx_led_data *led = platform_get_drvdata(pdev);
1260
1261 for (i = 0; i < pdata->num_leds; i++) {
1262 cancel_work_sync(&led[i].work);
1263 mutex_destroy(&led[i].lock);
1264 led_classdev_unregister(&led[i].cdev);
samin.ryue0041de2012-08-03 23:58:14 +09001265 if (led[i].pwm_dev != NULL) {
Jay Chokshi868312e2011-09-16 13:57:13 -07001266 pwm_free(led[i].pwm_dev);
samin.ryue0041de2012-08-03 23:58:14 +09001267 }
1268 }
1269
1270 device_remove_file(&pdev->dev, &dev_attr_lock);
1271
1272 if (led->use_pwm) {
1273 device_remove_file(&pdev->dev, &dev_attr_blink);
1274 device_remove_file(&pdev->dev, &dev_attr_grppwm);
1275 device_remove_file(&pdev->dev, &dev_attr_grpfreq);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001276 }
1277
1278 kfree(led);
1279
1280 return 0;
1281}
1282
1283static struct platform_driver pm8xxx_led_driver = {
1284 .probe = pm8xxx_led_probe,
1285 .remove = __devexit_p(pm8xxx_led_remove),
1286 .driver = {
1287 .name = PM8XXX_LEDS_DEV_NAME,
1288 .owner = THIS_MODULE,
1289 },
1290};
1291
1292static int __init pm8xxx_led_init(void)
1293{
1294 return platform_driver_register(&pm8xxx_led_driver);
1295}
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001296subsys_initcall(pm8xxx_led_init);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001297
1298static void __exit pm8xxx_led_exit(void)
1299{
1300 platform_driver_unregister(&pm8xxx_led_driver);
1301}
1302module_exit(pm8xxx_led_exit);
1303
1304MODULE_DESCRIPTION("PM8XXX LEDs driver");
1305MODULE_LICENSE("GPL v2");
1306MODULE_VERSION("1.0");
1307MODULE_ALIAS("platform:pm8xxx-led");