blob: ece7b0f52dbf4b5005e18191dddf56d0dde17bb6 [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>
16#include <linux/init.h>
17#include <linux/slab.h>
18#include <linux/platform_device.h>
19#include <linux/leds.h>
20#include <linux/workqueue.h>
Jay Chokshi868312e2011-09-16 13:57:13 -070021#include <linux/err.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070022
23#include <linux/mfd/pm8xxx/core.h>
Jay Chokshi868312e2011-09-16 13:57:13 -070024#include <linux/mfd/pm8xxx/pwm.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070025#include <linux/leds-pm8xxx.h>
26
27#define SSBI_REG_ADDR_DRV_KEYPAD 0x48
28#define PM8XXX_DRV_KEYPAD_BL_MASK 0xf0
29#define PM8XXX_DRV_KEYPAD_BL_SHIFT 0x04
30
31#define SSBI_REG_ADDR_FLASH_DRV0 0x49
32#define PM8XXX_DRV_FLASH_MASK 0xf0
33#define PM8XXX_DRV_FLASH_SHIFT 0x04
34
35#define SSBI_REG_ADDR_FLASH_DRV1 0xFB
36
37#define SSBI_REG_ADDR_LED_CTRL_BASE 0x131
38#define SSBI_REG_ADDR_LED_CTRL(n) (SSBI_REG_ADDR_LED_CTRL_BASE + (n))
39#define PM8XXX_DRV_LED_CTRL_MASK 0xf8
40#define PM8XXX_DRV_LED_CTRL_SHIFT 0x03
41
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053042#define SSBI_REG_ADDR_WLED_CTRL_BASE 0x25A
43#define SSBI_REG_ADDR_WLED_CTRL(n) (SSBI_REG_ADDR_WLED_CTRL_BASE + (n) - 1)
44
45/* wled control registers */
46#define WLED_MOD_CTRL_REG SSBI_REG_ADDR_WLED_CTRL(1)
47#define WLED_MAX_CURR_CFG_REG(n) SSBI_REG_ADDR_WLED_CTRL(n + 2)
48#define WLED_BRIGHTNESS_CNTL_REG1(n) SSBI_REG_ADDR_WLED_CTRL(n + 5)
49#define WLED_BRIGHTNESS_CNTL_REG2(n) SSBI_REG_ADDR_WLED_CTRL(n + 6)
50#define WLED_SYNC_REG SSBI_REG_ADDR_WLED_CTRL(11)
51#define WLED_OVP_CFG_REG SSBI_REG_ADDR_WLED_CTRL(13)
52#define WLED_BOOST_CFG_REG SSBI_REG_ADDR_WLED_CTRL(14)
53#define WLED_HIGH_POLE_CAP_REG SSBI_REG_ADDR_WLED_CTRL(16)
54
55#define WLED_STRINGS 0x03
56#define WLED_OVP_VAL_MASK 0x30
57#define WLED_OVP_VAL_BIT_SHFT 0x04
58#define WLED_BOOST_LIMIT_MASK 0xE0
59#define WLED_BOOST_LIMIT_BIT_SHFT 0x05
Amy Maloche56913f52012-05-11 10:47:24 -070060#define WLED_BOOST_OFF 0x00
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053061#define WLED_EN_MASK 0x01
62#define WLED_CP_SELECT_MAX 0x03
63#define WLED_CP_SELECT_MASK 0x03
64#define WLED_DIG_MOD_GEN_MASK 0x70
65#define WLED_CS_OUT_MASK 0x0E
66#define WLED_CTL_DLY_STEP 200
67#define WLED_CTL_DLY_MAX 1400
68#define WLED_CTL_DLY_MASK 0xE0
69#define WLED_CTL_DLY_BIT_SHFT 0x05
70#define WLED_MAX_CURR 25
71#define WLED_MAX_CURR_MASK 0x1F
72#define WLED_OP_FDBCK_MASK 0x1C
73#define WLED_OP_FDBCK_BIT_SHFT 0x02
74
Chandan Uddaraju6e73f0a2012-03-08 17:32:55 -080075#define WLED_MAX_LEVEL 255
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053076#define WLED_8_BIT_MASK 0xFF
77#define WLED_8_BIT_SHFT 0x08
78#define WLED_MAX_DUTY_CYCLE 0xFFF
79
80#define WLED_SYNC_VAL 0x07
81#define WLED_SYNC_RESET_VAL 0x00
82
Amy Malochec17c3732012-02-27 18:34:07 -080083#define SSBI_REG_ADDR_RGB_CNTL1 0x12D
84#define SSBI_REG_ADDR_RGB_CNTL2 0x12E
85
86#define PM8XXX_DRV_RGB_RED_LED BIT(2)
87#define PM8XXX_DRV_RGB_GREEN_LED BIT(1)
88#define PM8XXX_DRV_RGB_BLUE_LED BIT(0)
89
Jay Chokshi12e49bf2011-07-22 16:24:39 -070090#define MAX_FLASH_LED_CURRENT 300
91#define MAX_LC_LED_CURRENT 40
92#define MAX_KP_BL_LED_CURRENT 300
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070093
Jay Chokshi12e49bf2011-07-22 16:24:39 -070094#define PM8XXX_ID_LED_CURRENT_FACTOR 2 /* Iout = x * 2mA */
95#define PM8XXX_ID_FLASH_CURRENT_FACTOR 20 /* Iout = x * 20mA */
96
97#define PM8XXX_FLASH_MODE_DBUS1 1
98#define PM8XXX_FLASH_MODE_DBUS2 2
99#define PM8XXX_FLASH_MODE_PWM 3
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700100
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700101#define MAX_LC_LED_BRIGHTNESS 20
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700102#define MAX_FLASH_BRIGHTNESS 15
103#define MAX_KB_LED_BRIGHTNESS 15
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700104
Jay Chokshide4cefb2011-08-04 18:10:44 -0700105#define PM8XXX_LED_OFFSET(id) ((id) - PM8XXX_ID_LED_0)
106
Jay Chokshi868312e2011-09-16 13:57:13 -0700107#define PM8XXX_LED_PWM_FLAGS (PM_PWM_LUT_LOOP | PM_PWM_LUT_RAMP_UP)
108
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700109#define LED_MAP(_version, _kb, _led0, _led1, _led2, _flash_led0, _flash_led1, \
Amy Malochec17c3732012-02-27 18:34:07 -0800110 _wled, _rgb_led_red, _rgb_led_green, _rgb_led_blue)\
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700111 {\
112 .version = _version,\
113 .supported = _kb << PM8XXX_ID_LED_KB_LIGHT | \
114 _led0 << PM8XXX_ID_LED_0 | _led1 << PM8XXX_ID_LED_1 | \
115 _led2 << PM8XXX_ID_LED_2 | \
116 _flash_led0 << PM8XXX_ID_FLASH_LED_0 | \
117 _flash_led1 << PM8XXX_ID_FLASH_LED_1 | \
Amy Malochec17c3732012-02-27 18:34:07 -0800118 _wled << PM8XXX_ID_WLED | \
119 _rgb_led_red << PM8XXX_ID_RGB_LED_RED | \
120 _rgb_led_green << PM8XXX_ID_RGB_LED_GREEN | \
121 _rgb_led_blue << PM8XXX_ID_RGB_LED_BLUE, \
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700122 }
123
124/**
125 * supported_leds - leds supported for each PMIC version
126 * @version - version of PMIC
127 * @supported - which leds are supported on version
128 */
129
130struct supported_leds {
131 enum pm8xxx_version version;
132 u32 supported;
133};
134
135static const struct supported_leds led_map[] = {
Amy Malochec17c3732012-02-27 18:34:07 -0800136 LED_MAP(PM8XXX_VERSION_8058, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
137 LED_MAP(PM8XXX_VERSION_8921, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
138 LED_MAP(PM8XXX_VERSION_8018, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0),
139 LED_MAP(PM8XXX_VERSION_8922, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1),
140 LED_MAP(PM8XXX_VERSION_8038, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1),
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700141};
142
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700143/**
144 * struct pm8xxx_led_data - internal led data structure
145 * @led_classdev - led class device
146 * @id - led index
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700147 * @work - workqueue for led
148 * @lock - to protect the transactions
149 * @reg - cached value of led register
Jay Chokshi868312e2011-09-16 13:57:13 -0700150 * @pwm_dev - pointer to PWM device if LED is driven using PWM
151 * @pwm_channel - PWM channel ID
152 * @pwm_period_us - PWM period in micro seconds
153 * @pwm_duty_cycles - struct that describes PWM duty cycles info
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700154 */
155struct pm8xxx_led_data {
156 struct led_classdev cdev;
157 int id;
158 u8 reg;
Amy Maloche56913f52012-05-11 10:47:24 -0700159 u8 wled_mod_ctrl_val;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700160 struct device *dev;
161 struct work_struct work;
162 struct mutex lock;
Jay Chokshi868312e2011-09-16 13:57:13 -0700163 struct pwm_device *pwm_dev;
164 int pwm_channel;
165 u32 pwm_period_us;
166 struct pm8xxx_pwm_duty_cycles *pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530167 struct wled_config_data *wled_cfg;
168 int max_current;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700169};
170
171static void led_kp_set(struct pm8xxx_led_data *led, enum led_brightness value)
172{
173 int rc;
174 u8 level;
175
176 level = (value << PM8XXX_DRV_KEYPAD_BL_SHIFT) &
177 PM8XXX_DRV_KEYPAD_BL_MASK;
178
179 led->reg &= ~PM8XXX_DRV_KEYPAD_BL_MASK;
180 led->reg |= level;
181
182 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_DRV_KEYPAD,
183 led->reg);
184 if (rc < 0)
185 dev_err(led->cdev.dev,
186 "can't set keypad backlight level rc=%d\n", rc);
187}
188
189static void led_lc_set(struct pm8xxx_led_data *led, enum led_brightness value)
190{
191 int rc, offset;
192 u8 level;
193
194 level = (value << PM8XXX_DRV_LED_CTRL_SHIFT) &
195 PM8XXX_DRV_LED_CTRL_MASK;
196
197 offset = PM8XXX_LED_OFFSET(led->id);
198
199 led->reg &= ~PM8XXX_DRV_LED_CTRL_MASK;
200 led->reg |= level;
201
202 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_LED_CTRL(offset),
203 led->reg);
204 if (rc)
205 dev_err(led->cdev.dev, "can't set (%d) led value rc=%d\n",
206 led->id, rc);
207}
208
209static void
210led_flash_set(struct pm8xxx_led_data *led, enum led_brightness value)
211{
212 int rc;
213 u8 level;
214 u16 reg_addr;
215
216 level = (value << PM8XXX_DRV_FLASH_SHIFT) &
217 PM8XXX_DRV_FLASH_MASK;
218
219 led->reg &= ~PM8XXX_DRV_FLASH_MASK;
220 led->reg |= level;
221
222 if (led->id == PM8XXX_ID_FLASH_LED_0)
223 reg_addr = SSBI_REG_ADDR_FLASH_DRV0;
224 else
225 reg_addr = SSBI_REG_ADDR_FLASH_DRV1;
226
227 rc = pm8xxx_writeb(led->dev->parent, reg_addr, led->reg);
228 if (rc < 0)
229 dev_err(led->cdev.dev, "can't set flash led%d level rc=%d\n",
230 led->id, rc);
231}
232
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530233static int
234led_wled_set(struct pm8xxx_led_data *led, enum led_brightness value)
235{
236 int rc, duty;
237 u8 val, i, num_wled_strings;
238
239 if (value > WLED_MAX_LEVEL)
240 value = WLED_MAX_LEVEL;
241
Amy Maloche56913f52012-05-11 10:47:24 -0700242 if (value == 0) {
243 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
244 WLED_BOOST_OFF);
245 if (rc) {
246 dev_err(led->dev->parent, "can't write wled ctrl config"
247 " register rc=%d\n", rc);
248 return rc;
249 }
250 } else {
251 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
252 led->wled_mod_ctrl_val);
253 if (rc) {
254 dev_err(led->dev->parent, "can't write wled ctrl config"
255 " register rc=%d\n", rc);
256 return rc;
257 }
258 }
259
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530260 duty = (WLED_MAX_DUTY_CYCLE * value) / WLED_MAX_LEVEL;
261
262 num_wled_strings = led->wled_cfg->num_strings;
263
264 /* program brightness control registers */
265 for (i = 0; i < num_wled_strings; i++) {
266 rc = pm8xxx_readb(led->dev->parent,
267 WLED_BRIGHTNESS_CNTL_REG1(i), &val);
268 if (rc) {
269 dev_err(led->dev->parent, "can't read wled brightnes ctrl"
270 " register1 rc=%d\n", rc);
271 return rc;
272 }
273
274 val = (val & ~WLED_MAX_CURR_MASK) | (duty >> WLED_8_BIT_SHFT);
275 rc = pm8xxx_writeb(led->dev->parent,
276 WLED_BRIGHTNESS_CNTL_REG1(i), val);
277 if (rc) {
278 dev_err(led->dev->parent, "can't write wled brightness ctrl"
279 " register1 rc=%d\n", rc);
280 return rc;
281 }
282
283 val = duty & WLED_8_BIT_MASK;
284 rc = pm8xxx_writeb(led->dev->parent,
285 WLED_BRIGHTNESS_CNTL_REG2(i), val);
286 if (rc) {
287 dev_err(led->dev->parent, "can't write wled brightness ctrl"
288 " register2 rc=%d\n", rc);
289 return rc;
290 }
291 }
292
293 /* sync */
294 val = WLED_SYNC_VAL;
295 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
296 if (rc) {
297 dev_err(led->dev->parent,
298 "can't read wled sync register rc=%d\n", rc);
299 return rc;
300 }
301
302 val = WLED_SYNC_RESET_VAL;
303 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
304 if (rc) {
305 dev_err(led->dev->parent,
306 "can't read wled sync register rc=%d\n", rc);
307 return rc;
308 }
309 return 0;
310}
311
312static void wled_dump_regs(struct pm8xxx_led_data *led)
313{
314 int i;
315 u8 val;
316
317 for (i = 1; i < 17; i++) {
318 pm8xxx_readb(led->dev->parent,
319 SSBI_REG_ADDR_WLED_CTRL(i), &val);
320 pr_debug("WLED_CTRL_%d = 0x%x\n", i, val);
321 }
322}
323
Amy Malochec17c3732012-02-27 18:34:07 -0800324static void
325led_rgb_set(struct pm8xxx_led_data *led, enum led_brightness value)
326{
327 int rc;
328 u8 val, mask;
329
330 rc = pm8xxx_readb(led->dev->parent, SSBI_REG_ADDR_RGB_CNTL2, &val);
331 if (rc) {
332 dev_err(led->cdev.dev, "can't read rgb ctrl register rc=%d\n",
333 rc);
334 return;
335 }
336
337 switch (led->id) {
338 case PM8XXX_ID_RGB_LED_RED:
339 mask = PM8XXX_DRV_RGB_RED_LED;
340 break;
341 case PM8XXX_ID_RGB_LED_GREEN:
342 mask = PM8XXX_DRV_RGB_GREEN_LED;
343 break;
344 case PM8XXX_ID_RGB_LED_BLUE:
345 mask = PM8XXX_DRV_RGB_BLUE_LED;
346 break;
347 default:
348 return;
349 }
350
351 if (value)
352 val |= mask;
353 else
354 val &= ~mask;
355
356 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_RGB_CNTL2, val);
357 if (rc < 0)
358 dev_err(led->cdev.dev, "can't set rgb led %d level rc=%d\n",
359 led->id, rc);
360}
361
Jay Chokshi868312e2011-09-16 13:57:13 -0700362static int pm8xxx_led_pwm_work(struct pm8xxx_led_data *led)
363{
364 int duty_us;
365 int rc = 0;
366
367 if (led->pwm_duty_cycles == NULL) {
368 duty_us = (led->pwm_period_us * led->cdev.brightness) /
369 LED_FULL;
370 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
371 if (led->cdev.brightness)
372 rc = pwm_enable(led->pwm_dev);
373 else
374 pwm_disable(led->pwm_dev);
375 } else {
376 rc = pm8xxx_pwm_lut_enable(led->pwm_dev, led->cdev.brightness);
377 }
378
379 return rc;
380}
381
Jay Chokshide4cefb2011-08-04 18:10:44 -0700382static void __pm8xxx_led_work(struct pm8xxx_led_data *led,
383 enum led_brightness level)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700384{
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530385 int rc;
386
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700387 mutex_lock(&led->lock);
388
389 switch (led->id) {
390 case PM8XXX_ID_LED_KB_LIGHT:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700391 led_kp_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530392 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700393 case PM8XXX_ID_LED_0:
394 case PM8XXX_ID_LED_1:
395 case PM8XXX_ID_LED_2:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700396 led_lc_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530397 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700398 case PM8XXX_ID_FLASH_LED_0:
399 case PM8XXX_ID_FLASH_LED_1:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700400 led_flash_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530401 break;
402 case PM8XXX_ID_WLED:
403 rc = led_wled_set(led, level);
404 if (rc < 0)
405 pr_err("wled brightness set failed %d\n", rc);
406 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800407 case PM8XXX_ID_RGB_LED_RED:
408 case PM8XXX_ID_RGB_LED_GREEN:
409 case PM8XXX_ID_RGB_LED_BLUE:
410 led_rgb_set(led, level);
411 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530412 default:
413 dev_err(led->cdev.dev, "unknown led id %d", led->id);
414 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700415 }
416
417 mutex_unlock(&led->lock);
418}
419
Jay Chokshide4cefb2011-08-04 18:10:44 -0700420static void pm8xxx_led_work(struct work_struct *work)
421{
Jay Chokshi868312e2011-09-16 13:57:13 -0700422 int rc;
423
Jay Chokshide4cefb2011-08-04 18:10:44 -0700424 struct pm8xxx_led_data *led = container_of(work,
425 struct pm8xxx_led_data, work);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700426
Jay Chokshi868312e2011-09-16 13:57:13 -0700427 if (led->pwm_dev == NULL) {
428 __pm8xxx_led_work(led, led->cdev.brightness);
429 } else {
430 rc = pm8xxx_led_pwm_work(led);
431 if (rc)
432 pr_err("could not configure PWM mode for LED:%d\n",
433 led->id);
434 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700435}
436
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700437static void pm8xxx_led_set(struct led_classdev *led_cdev,
438 enum led_brightness value)
439{
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700440 struct pm8xxx_led_data *led;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700441
442 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
443
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700444 if (value < LED_OFF || value > led->cdev.max_brightness) {
445 dev_err(led->cdev.dev, "Invalid brightness value exceeds");
446 return;
447 }
448
449 led->cdev.brightness = value;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700450 schedule_work(&led->work);
451}
452
Jay Chokshide4cefb2011-08-04 18:10:44 -0700453static int pm8xxx_set_led_mode_and_max_brightness(struct pm8xxx_led_data *led,
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700454 enum pm8xxx_led_modes led_mode, int max_current)
455{
Jay Chokshide4cefb2011-08-04 18:10:44 -0700456 switch (led->id) {
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700457 case PM8XXX_ID_LED_0:
458 case PM8XXX_ID_LED_1:
459 case PM8XXX_ID_LED_2:
460 led->cdev.max_brightness = max_current /
461 PM8XXX_ID_LED_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700462 if (led->cdev.max_brightness > MAX_LC_LED_BRIGHTNESS)
463 led->cdev.max_brightness = MAX_LC_LED_BRIGHTNESS;
Willie Ruana39d1d42011-08-19 13:54:34 -0700464 led->reg = led_mode;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700465 break;
466 case PM8XXX_ID_LED_KB_LIGHT:
467 case PM8XXX_ID_FLASH_LED_0:
468 case PM8XXX_ID_FLASH_LED_1:
469 led->cdev.max_brightness = max_current /
470 PM8XXX_ID_FLASH_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700471 if (led->cdev.max_brightness > MAX_FLASH_BRIGHTNESS)
472 led->cdev.max_brightness = MAX_FLASH_BRIGHTNESS;
473
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700474 switch (led_mode) {
475 case PM8XXX_LED_MODE_PWM1:
476 case PM8XXX_LED_MODE_PWM2:
477 case PM8XXX_LED_MODE_PWM3:
Willie Ruana39d1d42011-08-19 13:54:34 -0700478 led->reg = PM8XXX_FLASH_MODE_PWM;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700479 break;
480 case PM8XXX_LED_MODE_DTEST1:
Willie Ruana39d1d42011-08-19 13:54:34 -0700481 led->reg = PM8XXX_FLASH_MODE_DBUS1;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700482 break;
483 case PM8XXX_LED_MODE_DTEST2:
Willie Ruana39d1d42011-08-19 13:54:34 -0700484 led->reg = PM8XXX_FLASH_MODE_DBUS2;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700485 break;
486 default:
Willie Ruana39d1d42011-08-19 13:54:34 -0700487 led->reg = PM8XXX_LED_MODE_MANUAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700488 break;
489 }
490 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530491 case PM8XXX_ID_WLED:
492 led->cdev.max_brightness = WLED_MAX_LEVEL;
493 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800494 case PM8XXX_ID_RGB_LED_RED:
495 case PM8XXX_ID_RGB_LED_GREEN:
496 case PM8XXX_ID_RGB_LED_BLUE:
497 led->cdev.max_brightness = LED_FULL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700498 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800499 default:
500 dev_err(led->cdev.dev, "LED Id is invalid");
501 return -EINVAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700502 }
503
Amy Malochec17c3732012-02-27 18:34:07 -0800504 return 0;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700505}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700506
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700507static enum led_brightness pm8xxx_led_get(struct led_classdev *led_cdev)
508{
509 struct pm8xxx_led_data *led;
510
511 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
512
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700513 return led->cdev.brightness;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700514}
515
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530516static int __devinit init_wled(struct pm8xxx_led_data *led)
517{
518 int rc, i;
519 u8 val, num_wled_strings;
520
521 num_wled_strings = led->wled_cfg->num_strings;
522
523 /* program over voltage protection threshold */
524 if (led->wled_cfg->ovp_val > WLED_OVP_37V) {
525 dev_err(led->dev->parent, "Invalid ovp value");
526 return -EINVAL;
527 }
528
529 rc = pm8xxx_readb(led->dev->parent, WLED_OVP_CFG_REG, &val);
530 if (rc) {
531 dev_err(led->dev->parent, "can't read wled ovp config"
532 " register rc=%d\n", rc);
533 return rc;
534 }
535
536 val = (val & ~WLED_OVP_VAL_MASK) |
537 (led->wled_cfg->ovp_val << WLED_OVP_VAL_BIT_SHFT);
538
539 rc = pm8xxx_writeb(led->dev->parent, WLED_OVP_CFG_REG, val);
540 if (rc) {
541 dev_err(led->dev->parent, "can't write wled ovp config"
542 " register rc=%d\n", rc);
543 return rc;
544 }
545
546 /* program current boost limit and output feedback*/
547 if (led->wled_cfg->boost_curr_lim > WLED_CURR_LIMIT_1680mA) {
548 dev_err(led->dev->parent, "Invalid boost current limit");
549 return -EINVAL;
550 }
551
552 rc = pm8xxx_readb(led->dev->parent, WLED_BOOST_CFG_REG, &val);
553 if (rc) {
554 dev_err(led->dev->parent, "can't read wled boost config"
555 " register rc=%d\n", rc);
556 return rc;
557 }
558
559 val = (val & ~WLED_BOOST_LIMIT_MASK) |
560 (led->wled_cfg->boost_curr_lim << WLED_BOOST_LIMIT_BIT_SHFT);
561
562 val = (val & ~WLED_OP_FDBCK_MASK) |
563 (led->wled_cfg->op_fdbck << WLED_OP_FDBCK_BIT_SHFT);
564
565 rc = pm8xxx_writeb(led->dev->parent, WLED_BOOST_CFG_REG, val);
566 if (rc) {
567 dev_err(led->dev->parent, "can't write wled boost config"
568 " register rc=%d\n", rc);
569 return rc;
570 }
571
572 /* program high pole capacitance */
573 if (led->wled_cfg->cp_select > WLED_CP_SELECT_MAX) {
574 dev_err(led->dev->parent, "Invalid pole capacitance");
575 return -EINVAL;
576 }
577
578 rc = pm8xxx_readb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, &val);
579 if (rc) {
580 dev_err(led->dev->parent, "can't read wled high pole"
581 " capacitance register rc=%d\n", rc);
582 return rc;
583 }
584
585 val = (val & ~WLED_CP_SELECT_MASK) | led->wled_cfg->cp_select;
586
587 rc = pm8xxx_writeb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, val);
588 if (rc) {
589 dev_err(led->dev->parent, "can't write wled high pole"
590 " capacitance register rc=%d\n", rc);
591 return rc;
592 }
593
594 /* program activation delay and maximum current */
595 for (i = 0; i < num_wled_strings; i++) {
596 rc = pm8xxx_readb(led->dev->parent,
597 WLED_MAX_CURR_CFG_REG(i + 2), &val);
598 if (rc) {
599 dev_err(led->dev->parent, "can't read wled max current"
600 " config register rc=%d\n", rc);
601 return rc;
602 }
603
604 if ((led->wled_cfg->ctrl_delay_us % WLED_CTL_DLY_STEP) ||
605 (led->wled_cfg->ctrl_delay_us > WLED_CTL_DLY_MAX)) {
606 dev_err(led->dev->parent, "Invalid control delay\n");
607 return rc;
608 }
609
610 val = val / WLED_CTL_DLY_STEP;
611 val = (val & ~WLED_CTL_DLY_MASK) |
612 (led->wled_cfg->ctrl_delay_us << WLED_CTL_DLY_BIT_SHFT);
613
614 if ((led->max_current > WLED_MAX_CURR)) {
615 dev_err(led->dev->parent, "Invalid max current\n");
616 return -EINVAL;
617 }
618
619 val = (val & ~WLED_MAX_CURR_MASK) | led->max_current;
620
621 rc = pm8xxx_writeb(led->dev->parent,
622 WLED_MAX_CURR_CFG_REG(i + 2), val);
623 if (rc) {
624 dev_err(led->dev->parent, "can't write wled max current"
625 " config register rc=%d\n", rc);
626 return rc;
627 }
628 }
629
630 /* program digital module generator, cs out and enable the module */
631 rc = pm8xxx_readb(led->dev->parent, WLED_MOD_CTRL_REG, &val);
632 if (rc) {
633 dev_err(led->dev->parent, "can't read wled module ctrl"
634 " register rc=%d\n", rc);
635 return rc;
636 }
637
638 if (led->wled_cfg->dig_mod_gen_en)
639 val |= WLED_DIG_MOD_GEN_MASK;
640
641 if (led->wled_cfg->cs_out_en)
642 val |= WLED_CS_OUT_MASK;
643
644 val |= WLED_EN_MASK;
645
646 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG, val);
647 if (rc) {
648 dev_err(led->dev->parent, "can't write wled module ctrl"
649 " register rc=%d\n", rc);
650 return rc;
651 }
Amy Maloche56913f52012-05-11 10:47:24 -0700652 led->wled_mod_ctrl_val = val;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530653
654 /* dump wled registers */
655 wled_dump_regs(led);
656
657 return 0;
658}
659
Amy Malochec17c3732012-02-27 18:34:07 -0800660static int __devinit init_rgb_led(struct pm8xxx_led_data *led)
661{
662 int rc;
663 u8 val;
664
665 rc = pm8xxx_readb(led->dev->parent, SSBI_REG_ADDR_RGB_CNTL1, &val);
666 if (rc) {
667 dev_err(led->cdev.dev, "can't read rgb ctrl1 register rc=%d\n",
668 rc);
669 return rc;
670 }
671
672 switch (led->id) {
673 case PM8XXX_ID_RGB_LED_RED:
674 val |= PM8XXX_DRV_RGB_RED_LED;
675 break;
676 case PM8XXX_ID_RGB_LED_GREEN:
677 val |= PM8XXX_DRV_RGB_GREEN_LED;
678 break;
679 case PM8XXX_ID_RGB_LED_BLUE:
680 val |= PM8XXX_DRV_RGB_BLUE_LED;
681 break;
682 default:
683 return -EINVAL;
684 }
685
686 return pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_RGB_CNTL1, val);
687}
688
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700689static int __devinit get_init_value(struct pm8xxx_led_data *led, u8 *val)
690{
691 int rc, offset;
692 u16 addr;
693
694 switch (led->id) {
695 case PM8XXX_ID_LED_KB_LIGHT:
696 addr = SSBI_REG_ADDR_DRV_KEYPAD;
697 break;
698 case PM8XXX_ID_LED_0:
699 case PM8XXX_ID_LED_1:
700 case PM8XXX_ID_LED_2:
701 offset = PM8XXX_LED_OFFSET(led->id);
702 addr = SSBI_REG_ADDR_LED_CTRL(offset);
703 break;
704 case PM8XXX_ID_FLASH_LED_0:
705 addr = SSBI_REG_ADDR_FLASH_DRV0;
706 break;
707 case PM8XXX_ID_FLASH_LED_1:
708 addr = SSBI_REG_ADDR_FLASH_DRV1;
709 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530710 case PM8XXX_ID_WLED:
711 rc = init_wled(led);
712 if (rc)
713 dev_err(led->cdev.dev, "can't initialize wled rc=%d\n",
714 rc);
715 return rc;
Amy Malochec17c3732012-02-27 18:34:07 -0800716 case PM8XXX_ID_RGB_LED_RED:
717 case PM8XXX_ID_RGB_LED_GREEN:
718 case PM8XXX_ID_RGB_LED_BLUE:
719 rc = init_rgb_led(led);
720 if (rc) {
721 dev_err(led->cdev.dev, "can't initialize rgb rc=%d\n",
722 rc);
723 return rc;
724 }
725 addr = SSBI_REG_ADDR_RGB_CNTL1;
726 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530727 default:
728 dev_err(led->cdev.dev, "unknown led id %d", led->id);
729 return -EINVAL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700730 }
731
732 rc = pm8xxx_readb(led->dev->parent, addr, val);
733 if (rc)
734 dev_err(led->cdev.dev, "can't get led(%d) level rc=%d\n",
735 led->id, rc);
736
737 return rc;
738}
739
Jay Chokshi868312e2011-09-16 13:57:13 -0700740static int pm8xxx_led_pwm_configure(struct pm8xxx_led_data *led)
741{
742 int start_idx, idx_len, duty_us, rc;
743
744 led->pwm_dev = pwm_request(led->pwm_channel,
745 led->cdev.name);
746
747 if (IS_ERR_OR_NULL(led->pwm_dev)) {
748 pr_err("could not acquire PWM Channel %d, "
749 "error %ld\n", led->pwm_channel,
750 PTR_ERR(led->pwm_dev));
751 led->pwm_dev = NULL;
752 return -ENODEV;
753 }
754
755 if (led->pwm_duty_cycles != NULL) {
756 start_idx = led->pwm_duty_cycles->start_idx;
757 idx_len = led->pwm_duty_cycles->num_duty_pcts;
758
759 if (idx_len >= PM_PWM_LUT_SIZE && start_idx) {
760 pr_err("Wrong LUT size or index\n");
761 return -EINVAL;
762 }
763 if ((start_idx + idx_len) > PM_PWM_LUT_SIZE) {
764 pr_err("Exceed LUT limit\n");
765 return -EINVAL;
766 }
767
768 rc = pm8xxx_pwm_lut_config(led->pwm_dev, led->pwm_period_us,
769 led->pwm_duty_cycles->duty_pcts,
770 led->pwm_duty_cycles->duty_ms,
771 start_idx, idx_len, 0, 0,
772 PM8XXX_LED_PWM_FLAGS);
773 } else {
774 duty_us = led->pwm_period_us;
775 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
776 }
777
778 return rc;
779}
780
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530781
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700782static int __devinit pm8xxx_led_probe(struct platform_device *pdev)
783{
Jay Chokshi8994e392011-09-14 18:20:39 -0700784 const struct pm8xxx_led_platform_data *pdata = pdev->dev.platform_data;
785 const struct led_platform_data *pcore_data;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700786 struct led_info *curr_led;
787 struct pm8xxx_led_data *led, *led_dat;
Jay Chokshi8994e392011-09-14 18:20:39 -0700788 struct pm8xxx_led_config *led_cfg;
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700789 enum pm8xxx_version version;
790 bool found = false;
791 int rc, i, j;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700792
793 if (pdata == NULL) {
794 dev_err(&pdev->dev, "platform data not supplied\n");
795 return -EINVAL;
796 }
797
Jay Chokshi8994e392011-09-14 18:20:39 -0700798 pcore_data = pdata->led_core;
799
800 if (pcore_data->num_leds != pdata->num_configs) {
801 dev_err(&pdev->dev, "#no. of led configs and #no. of led"
802 "entries are not equal\n");
803 return -EINVAL;
804 }
805
806 led = kcalloc(pcore_data->num_leds, sizeof(*led), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700807 if (led == NULL) {
808 dev_err(&pdev->dev, "failed to alloc memory\n");
809 return -ENOMEM;
810 }
811
Jay Chokshi8994e392011-09-14 18:20:39 -0700812 for (i = 0; i < pcore_data->num_leds; i++) {
813 curr_led = &pcore_data->leds[i];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700814 led_dat = &led[i];
Jay Chokshi8994e392011-09-14 18:20:39 -0700815 led_cfg = &pdata->configs[i];
816
817 led_dat->id = led_cfg->id;
Jay Chokshi868312e2011-09-16 13:57:13 -0700818 led_dat->pwm_channel = led_cfg->pwm_channel;
819 led_dat->pwm_period_us = led_cfg->pwm_period_us;
820 led_dat->pwm_duty_cycles = led_cfg->pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530821 led_dat->wled_cfg = led_cfg->wled_cfg;
822 led_dat->max_current = led_cfg->max_current;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700823
824 if (!((led_dat->id >= PM8XXX_ID_LED_KB_LIGHT) &&
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530825 (led_dat->id < PM8XXX_ID_MAX))) {
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700826 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
827 led_dat->id);
828 rc = -EINVAL;
829 goto fail_id_check;
830
831 }
832
833 found = false;
834 version = pm8xxx_get_version(pdev->dev.parent);
835 for (j = 0; j < ARRAY_SIZE(led_map); j++) {
836 if (version == led_map[j].version
837 && (led_map[j].supported & (1 << led_dat->id))) {
838 found = true;
839 break;
840 }
841 }
842
843 if (!found) {
844 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
845 led_dat->id);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700846 rc = -EINVAL;
847 goto fail_id_check;
848 }
849
850 led_dat->cdev.name = curr_led->name;
851 led_dat->cdev.default_trigger = curr_led->default_trigger;
852 led_dat->cdev.brightness_set = pm8xxx_led_set;
853 led_dat->cdev.brightness_get = pm8xxx_led_get;
854 led_dat->cdev.brightness = LED_OFF;
Jay Chokshi8994e392011-09-14 18:20:39 -0700855 led_dat->cdev.flags = curr_led->flags;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700856 led_dat->dev = &pdev->dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700857
858 rc = get_init_value(led_dat, &led_dat->reg);
859 if (rc < 0)
860 goto fail_id_check;
861
Jay Chokshi8994e392011-09-14 18:20:39 -0700862 rc = pm8xxx_set_led_mode_and_max_brightness(led_dat,
863 led_cfg->mode, led_cfg->max_current);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700864 if (rc < 0)
865 goto fail_id_check;
866
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700867 mutex_init(&led_dat->lock);
868 INIT_WORK(&led_dat->work, pm8xxx_led_work);
869
870 rc = led_classdev_register(&pdev->dev, &led_dat->cdev);
871 if (rc) {
872 dev_err(&pdev->dev, "unable to register led %d,rc=%d\n",
873 led_dat->id, rc);
874 goto fail_id_check;
875 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700876
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530877 /* configure default state */
878 if (led_cfg->default_state)
879 led->cdev.brightness = led_dat->cdev.max_brightness;
880 else
881 led->cdev.brightness = LED_OFF;
882
Jay Chokshi868312e2011-09-16 13:57:13 -0700883 if (led_cfg->mode != PM8XXX_LED_MODE_MANUAL) {
Amy Malochec17c3732012-02-27 18:34:07 -0800884 if (led_dat->id == PM8XXX_ID_RGB_LED_RED ||
885 led_dat->id == PM8XXX_ID_RGB_LED_GREEN ||
886 led_dat->id == PM8XXX_ID_RGB_LED_BLUE)
887 __pm8xxx_led_work(led_dat, 0);
888 else
889 __pm8xxx_led_work(led_dat,
Jay Chokshide4cefb2011-08-04 18:10:44 -0700890 led_dat->cdev.max_brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700891
892 if (led_dat->pwm_channel != -1) {
893 led_dat->cdev.max_brightness = LED_FULL;
894 rc = pm8xxx_led_pwm_configure(led_dat);
895 if (rc) {
896 dev_err(&pdev->dev, "failed to "
897 "configure LED, error: %d\n", rc);
898 goto fail_id_check;
899 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530900 schedule_work(&led->work);
Jay Chokshi868312e2011-09-16 13:57:13 -0700901 }
902 } else {
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530903 __pm8xxx_led_work(led_dat, led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700904 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700905 }
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700906
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700907 platform_set_drvdata(pdev, led);
908
909 return 0;
910
911fail_id_check:
912 if (i > 0) {
913 for (i = i - 1; i >= 0; i--) {
914 mutex_destroy(&led[i].lock);
915 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -0700916 if (led[i].pwm_dev != NULL)
917 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700918 }
919 }
920 kfree(led);
921 return rc;
922}
923
924static int __devexit pm8xxx_led_remove(struct platform_device *pdev)
925{
926 int i;
927 const struct led_platform_data *pdata =
928 pdev->dev.platform_data;
929 struct pm8xxx_led_data *led = platform_get_drvdata(pdev);
930
931 for (i = 0; i < pdata->num_leds; i++) {
932 cancel_work_sync(&led[i].work);
933 mutex_destroy(&led[i].lock);
934 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -0700935 if (led[i].pwm_dev != NULL)
936 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700937 }
938
939 kfree(led);
940
941 return 0;
942}
943
944static struct platform_driver pm8xxx_led_driver = {
945 .probe = pm8xxx_led_probe,
946 .remove = __devexit_p(pm8xxx_led_remove),
947 .driver = {
948 .name = PM8XXX_LEDS_DEV_NAME,
949 .owner = THIS_MODULE,
950 },
951};
952
953static int __init pm8xxx_led_init(void)
954{
955 return platform_driver_register(&pm8xxx_led_driver);
956}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700957subsys_initcall(pm8xxx_led_init);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700958
959static void __exit pm8xxx_led_exit(void)
960{
961 platform_driver_unregister(&pm8xxx_led_driver);
962}
963module_exit(pm8xxx_led_exit);
964
965MODULE_DESCRIPTION("PM8XXX LEDs driver");
966MODULE_LICENSE("GPL v2");
967MODULE_VERSION("1.0");
968MODULE_ALIAS("platform:pm8xxx-led");