blob: 6770ec84c6ace7955b8d51e2dae6110681be9d85 [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
60#define WLED_EN_MASK 0x01
61#define WLED_CP_SELECT_MAX 0x03
62#define WLED_CP_SELECT_MASK 0x03
63#define WLED_DIG_MOD_GEN_MASK 0x70
64#define WLED_CS_OUT_MASK 0x0E
65#define WLED_CTL_DLY_STEP 200
66#define WLED_CTL_DLY_MAX 1400
67#define WLED_CTL_DLY_MASK 0xE0
68#define WLED_CTL_DLY_BIT_SHFT 0x05
69#define WLED_MAX_CURR 25
70#define WLED_MAX_CURR_MASK 0x1F
71#define WLED_OP_FDBCK_MASK 0x1C
72#define WLED_OP_FDBCK_BIT_SHFT 0x02
73
74#define WLED_MAX_LEVEL 100
75#define WLED_8_BIT_MASK 0xFF
76#define WLED_8_BIT_SHFT 0x08
77#define WLED_MAX_DUTY_CYCLE 0xFFF
78
79#define WLED_SYNC_VAL 0x07
80#define WLED_SYNC_RESET_VAL 0x00
81
Amy Malochec17c3732012-02-27 18:34:07 -080082#define SSBI_REG_ADDR_RGB_CNTL1 0x12D
83#define SSBI_REG_ADDR_RGB_CNTL2 0x12E
84
85#define PM8XXX_DRV_RGB_RED_LED BIT(2)
86#define PM8XXX_DRV_RGB_GREEN_LED BIT(1)
87#define PM8XXX_DRV_RGB_BLUE_LED BIT(0)
88
Jay Chokshi12e49bf2011-07-22 16:24:39 -070089#define MAX_FLASH_LED_CURRENT 300
90#define MAX_LC_LED_CURRENT 40
91#define MAX_KP_BL_LED_CURRENT 300
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070092
Jay Chokshi12e49bf2011-07-22 16:24:39 -070093#define PM8XXX_ID_LED_CURRENT_FACTOR 2 /* Iout = x * 2mA */
94#define PM8XXX_ID_FLASH_CURRENT_FACTOR 20 /* Iout = x * 20mA */
95
96#define PM8XXX_FLASH_MODE_DBUS1 1
97#define PM8XXX_FLASH_MODE_DBUS2 2
98#define PM8XXX_FLASH_MODE_PWM 3
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070099
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700100#define MAX_LC_LED_BRIGHTNESS 20
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700101#define MAX_FLASH_BRIGHTNESS 15
102#define MAX_KB_LED_BRIGHTNESS 15
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700103
Jay Chokshide4cefb2011-08-04 18:10:44 -0700104#define PM8XXX_LED_OFFSET(id) ((id) - PM8XXX_ID_LED_0)
105
Jay Chokshi868312e2011-09-16 13:57:13 -0700106#define PM8XXX_LED_PWM_FLAGS (PM_PWM_LUT_LOOP | PM_PWM_LUT_RAMP_UP)
107
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700108#define LED_MAP(_version, _kb, _led0, _led1, _led2, _flash_led0, _flash_led1, \
Amy Malochec17c3732012-02-27 18:34:07 -0800109 _wled, _rgb_led_red, _rgb_led_green, _rgb_led_blue)\
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700110 {\
111 .version = _version,\
112 .supported = _kb << PM8XXX_ID_LED_KB_LIGHT | \
113 _led0 << PM8XXX_ID_LED_0 | _led1 << PM8XXX_ID_LED_1 | \
114 _led2 << PM8XXX_ID_LED_2 | \
115 _flash_led0 << PM8XXX_ID_FLASH_LED_0 | \
116 _flash_led1 << PM8XXX_ID_FLASH_LED_1 | \
Amy Malochec17c3732012-02-27 18:34:07 -0800117 _wled << PM8XXX_ID_WLED | \
118 _rgb_led_red << PM8XXX_ID_RGB_LED_RED | \
119 _rgb_led_green << PM8XXX_ID_RGB_LED_GREEN | \
120 _rgb_led_blue << PM8XXX_ID_RGB_LED_BLUE, \
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700121 }
122
123/**
124 * supported_leds - leds supported for each PMIC version
125 * @version - version of PMIC
126 * @supported - which leds are supported on version
127 */
128
129struct supported_leds {
130 enum pm8xxx_version version;
131 u32 supported;
132};
133
134static const struct supported_leds led_map[] = {
Amy Malochec17c3732012-02-27 18:34:07 -0800135 LED_MAP(PM8XXX_VERSION_8058, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
136 LED_MAP(PM8XXX_VERSION_8921, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
137 LED_MAP(PM8XXX_VERSION_8018, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0),
138 LED_MAP(PM8XXX_VERSION_8922, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1),
139 LED_MAP(PM8XXX_VERSION_8038, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1),
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700140};
141
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700142/**
143 * struct pm8xxx_led_data - internal led data structure
144 * @led_classdev - led class device
145 * @id - led index
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700146 * @work - workqueue for led
147 * @lock - to protect the transactions
148 * @reg - cached value of led register
Jay Chokshi868312e2011-09-16 13:57:13 -0700149 * @pwm_dev - pointer to PWM device if LED is driven using PWM
150 * @pwm_channel - PWM channel ID
151 * @pwm_period_us - PWM period in micro seconds
152 * @pwm_duty_cycles - struct that describes PWM duty cycles info
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700153 */
154struct pm8xxx_led_data {
155 struct led_classdev cdev;
156 int id;
157 u8 reg;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700158 struct device *dev;
159 struct work_struct work;
160 struct mutex lock;
Jay Chokshi868312e2011-09-16 13:57:13 -0700161 struct pwm_device *pwm_dev;
162 int pwm_channel;
163 u32 pwm_period_us;
164 struct pm8xxx_pwm_duty_cycles *pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530165 struct wled_config_data *wled_cfg;
166 int max_current;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700167};
168
169static void led_kp_set(struct pm8xxx_led_data *led, enum led_brightness value)
170{
171 int rc;
172 u8 level;
173
174 level = (value << PM8XXX_DRV_KEYPAD_BL_SHIFT) &
175 PM8XXX_DRV_KEYPAD_BL_MASK;
176
177 led->reg &= ~PM8XXX_DRV_KEYPAD_BL_MASK;
178 led->reg |= level;
179
180 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_DRV_KEYPAD,
181 led->reg);
182 if (rc < 0)
183 dev_err(led->cdev.dev,
184 "can't set keypad backlight level rc=%d\n", rc);
185}
186
187static void led_lc_set(struct pm8xxx_led_data *led, enum led_brightness value)
188{
189 int rc, offset;
190 u8 level;
191
192 level = (value << PM8XXX_DRV_LED_CTRL_SHIFT) &
193 PM8XXX_DRV_LED_CTRL_MASK;
194
195 offset = PM8XXX_LED_OFFSET(led->id);
196
197 led->reg &= ~PM8XXX_DRV_LED_CTRL_MASK;
198 led->reg |= level;
199
200 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_LED_CTRL(offset),
201 led->reg);
202 if (rc)
203 dev_err(led->cdev.dev, "can't set (%d) led value rc=%d\n",
204 led->id, rc);
205}
206
207static void
208led_flash_set(struct pm8xxx_led_data *led, enum led_brightness value)
209{
210 int rc;
211 u8 level;
212 u16 reg_addr;
213
214 level = (value << PM8XXX_DRV_FLASH_SHIFT) &
215 PM8XXX_DRV_FLASH_MASK;
216
217 led->reg &= ~PM8XXX_DRV_FLASH_MASK;
218 led->reg |= level;
219
220 if (led->id == PM8XXX_ID_FLASH_LED_0)
221 reg_addr = SSBI_REG_ADDR_FLASH_DRV0;
222 else
223 reg_addr = SSBI_REG_ADDR_FLASH_DRV1;
224
225 rc = pm8xxx_writeb(led->dev->parent, reg_addr, led->reg);
226 if (rc < 0)
227 dev_err(led->cdev.dev, "can't set flash led%d level rc=%d\n",
228 led->id, rc);
229}
230
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530231static int
232led_wled_set(struct pm8xxx_led_data *led, enum led_brightness value)
233{
234 int rc, duty;
235 u8 val, i, num_wled_strings;
236
237 if (value > WLED_MAX_LEVEL)
238 value = WLED_MAX_LEVEL;
239
240 duty = (WLED_MAX_DUTY_CYCLE * value) / WLED_MAX_LEVEL;
241
242 num_wled_strings = led->wled_cfg->num_strings;
243
244 /* program brightness control registers */
245 for (i = 0; i < num_wled_strings; i++) {
246 rc = pm8xxx_readb(led->dev->parent,
247 WLED_BRIGHTNESS_CNTL_REG1(i), &val);
248 if (rc) {
249 dev_err(led->dev->parent, "can't read wled brightnes ctrl"
250 " register1 rc=%d\n", rc);
251 return rc;
252 }
253
254 val = (val & ~WLED_MAX_CURR_MASK) | (duty >> WLED_8_BIT_SHFT);
255 rc = pm8xxx_writeb(led->dev->parent,
256 WLED_BRIGHTNESS_CNTL_REG1(i), val);
257 if (rc) {
258 dev_err(led->dev->parent, "can't write wled brightness ctrl"
259 " register1 rc=%d\n", rc);
260 return rc;
261 }
262
263 val = duty & WLED_8_BIT_MASK;
264 rc = pm8xxx_writeb(led->dev->parent,
265 WLED_BRIGHTNESS_CNTL_REG2(i), val);
266 if (rc) {
267 dev_err(led->dev->parent, "can't write wled brightness ctrl"
268 " register2 rc=%d\n", rc);
269 return rc;
270 }
271 }
272
273 /* sync */
274 val = WLED_SYNC_VAL;
275 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
276 if (rc) {
277 dev_err(led->dev->parent,
278 "can't read wled sync register rc=%d\n", rc);
279 return rc;
280 }
281
282 val = WLED_SYNC_RESET_VAL;
283 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
284 if (rc) {
285 dev_err(led->dev->parent,
286 "can't read wled sync register rc=%d\n", rc);
287 return rc;
288 }
289 return 0;
290}
291
292static void wled_dump_regs(struct pm8xxx_led_data *led)
293{
294 int i;
295 u8 val;
296
297 for (i = 1; i < 17; i++) {
298 pm8xxx_readb(led->dev->parent,
299 SSBI_REG_ADDR_WLED_CTRL(i), &val);
300 pr_debug("WLED_CTRL_%d = 0x%x\n", i, val);
301 }
302}
303
Amy Malochec17c3732012-02-27 18:34:07 -0800304static void
305led_rgb_set(struct pm8xxx_led_data *led, enum led_brightness value)
306{
307 int rc;
308 u8 val, mask;
309
310 rc = pm8xxx_readb(led->dev->parent, SSBI_REG_ADDR_RGB_CNTL2, &val);
311 if (rc) {
312 dev_err(led->cdev.dev, "can't read rgb ctrl register rc=%d\n",
313 rc);
314 return;
315 }
316
317 switch (led->id) {
318 case PM8XXX_ID_RGB_LED_RED:
319 mask = PM8XXX_DRV_RGB_RED_LED;
320 break;
321 case PM8XXX_ID_RGB_LED_GREEN:
322 mask = PM8XXX_DRV_RGB_GREEN_LED;
323 break;
324 case PM8XXX_ID_RGB_LED_BLUE:
325 mask = PM8XXX_DRV_RGB_BLUE_LED;
326 break;
327 default:
328 return;
329 }
330
331 if (value)
332 val |= mask;
333 else
334 val &= ~mask;
335
336 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_RGB_CNTL2, val);
337 if (rc < 0)
338 dev_err(led->cdev.dev, "can't set rgb led %d level rc=%d\n",
339 led->id, rc);
340}
341
Jay Chokshi868312e2011-09-16 13:57:13 -0700342static int pm8xxx_led_pwm_work(struct pm8xxx_led_data *led)
343{
344 int duty_us;
345 int rc = 0;
346
347 if (led->pwm_duty_cycles == NULL) {
348 duty_us = (led->pwm_period_us * led->cdev.brightness) /
349 LED_FULL;
350 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
351 if (led->cdev.brightness)
352 rc = pwm_enable(led->pwm_dev);
353 else
354 pwm_disable(led->pwm_dev);
355 } else {
356 rc = pm8xxx_pwm_lut_enable(led->pwm_dev, led->cdev.brightness);
357 }
358
359 return rc;
360}
361
Jay Chokshide4cefb2011-08-04 18:10:44 -0700362static void __pm8xxx_led_work(struct pm8xxx_led_data *led,
363 enum led_brightness level)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700364{
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530365 int rc;
366
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700367 mutex_lock(&led->lock);
368
369 switch (led->id) {
370 case PM8XXX_ID_LED_KB_LIGHT:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700371 led_kp_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530372 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700373 case PM8XXX_ID_LED_0:
374 case PM8XXX_ID_LED_1:
375 case PM8XXX_ID_LED_2:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700376 led_lc_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530377 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700378 case PM8XXX_ID_FLASH_LED_0:
379 case PM8XXX_ID_FLASH_LED_1:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700380 led_flash_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530381 break;
382 case PM8XXX_ID_WLED:
383 rc = led_wled_set(led, level);
384 if (rc < 0)
385 pr_err("wled brightness set failed %d\n", rc);
386 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800387 case PM8XXX_ID_RGB_LED_RED:
388 case PM8XXX_ID_RGB_LED_GREEN:
389 case PM8XXX_ID_RGB_LED_BLUE:
390 led_rgb_set(led, level);
391 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530392 default:
393 dev_err(led->cdev.dev, "unknown led id %d", led->id);
394 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700395 }
396
397 mutex_unlock(&led->lock);
398}
399
Jay Chokshide4cefb2011-08-04 18:10:44 -0700400static void pm8xxx_led_work(struct work_struct *work)
401{
Jay Chokshi868312e2011-09-16 13:57:13 -0700402 int rc;
403
Jay Chokshide4cefb2011-08-04 18:10:44 -0700404 struct pm8xxx_led_data *led = container_of(work,
405 struct pm8xxx_led_data, work);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700406
Jay Chokshi868312e2011-09-16 13:57:13 -0700407 if (led->pwm_dev == NULL) {
408 __pm8xxx_led_work(led, led->cdev.brightness);
409 } else {
410 rc = pm8xxx_led_pwm_work(led);
411 if (rc)
412 pr_err("could not configure PWM mode for LED:%d\n",
413 led->id);
414 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700415}
416
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700417static void pm8xxx_led_set(struct led_classdev *led_cdev,
418 enum led_brightness value)
419{
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700420 struct pm8xxx_led_data *led;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700421
422 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
423
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700424 if (value < LED_OFF || value > led->cdev.max_brightness) {
425 dev_err(led->cdev.dev, "Invalid brightness value exceeds");
426 return;
427 }
428
429 led->cdev.brightness = value;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700430 schedule_work(&led->work);
431}
432
Jay Chokshide4cefb2011-08-04 18:10:44 -0700433static int pm8xxx_set_led_mode_and_max_brightness(struct pm8xxx_led_data *led,
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700434 enum pm8xxx_led_modes led_mode, int max_current)
435{
Jay Chokshide4cefb2011-08-04 18:10:44 -0700436 switch (led->id) {
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700437 case PM8XXX_ID_LED_0:
438 case PM8XXX_ID_LED_1:
439 case PM8XXX_ID_LED_2:
440 led->cdev.max_brightness = max_current /
441 PM8XXX_ID_LED_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700442 if (led->cdev.max_brightness > MAX_LC_LED_BRIGHTNESS)
443 led->cdev.max_brightness = MAX_LC_LED_BRIGHTNESS;
Willie Ruana39d1d42011-08-19 13:54:34 -0700444 led->reg = led_mode;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700445 break;
446 case PM8XXX_ID_LED_KB_LIGHT:
447 case PM8XXX_ID_FLASH_LED_0:
448 case PM8XXX_ID_FLASH_LED_1:
449 led->cdev.max_brightness = max_current /
450 PM8XXX_ID_FLASH_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700451 if (led->cdev.max_brightness > MAX_FLASH_BRIGHTNESS)
452 led->cdev.max_brightness = MAX_FLASH_BRIGHTNESS;
453
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700454 switch (led_mode) {
455 case PM8XXX_LED_MODE_PWM1:
456 case PM8XXX_LED_MODE_PWM2:
457 case PM8XXX_LED_MODE_PWM3:
Willie Ruana39d1d42011-08-19 13:54:34 -0700458 led->reg = PM8XXX_FLASH_MODE_PWM;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700459 break;
460 case PM8XXX_LED_MODE_DTEST1:
Willie Ruana39d1d42011-08-19 13:54:34 -0700461 led->reg = PM8XXX_FLASH_MODE_DBUS1;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700462 break;
463 case PM8XXX_LED_MODE_DTEST2:
Willie Ruana39d1d42011-08-19 13:54:34 -0700464 led->reg = PM8XXX_FLASH_MODE_DBUS2;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700465 break;
466 default:
Willie Ruana39d1d42011-08-19 13:54:34 -0700467 led->reg = PM8XXX_LED_MODE_MANUAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700468 break;
469 }
470 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530471 case PM8XXX_ID_WLED:
472 led->cdev.max_brightness = WLED_MAX_LEVEL;
473 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800474 case PM8XXX_ID_RGB_LED_RED:
475 case PM8XXX_ID_RGB_LED_GREEN:
476 case PM8XXX_ID_RGB_LED_BLUE:
477 led->cdev.max_brightness = LED_FULL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700478 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800479 default:
480 dev_err(led->cdev.dev, "LED Id is invalid");
481 return -EINVAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700482 }
483
Amy Malochec17c3732012-02-27 18:34:07 -0800484 return 0;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700485}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700486
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700487static enum led_brightness pm8xxx_led_get(struct led_classdev *led_cdev)
488{
489 struct pm8xxx_led_data *led;
490
491 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
492
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700493 return led->cdev.brightness;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700494}
495
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530496static int __devinit init_wled(struct pm8xxx_led_data *led)
497{
498 int rc, i;
499 u8 val, num_wled_strings;
500
501 num_wled_strings = led->wled_cfg->num_strings;
502
503 /* program over voltage protection threshold */
504 if (led->wled_cfg->ovp_val > WLED_OVP_37V) {
505 dev_err(led->dev->parent, "Invalid ovp value");
506 return -EINVAL;
507 }
508
509 rc = pm8xxx_readb(led->dev->parent, WLED_OVP_CFG_REG, &val);
510 if (rc) {
511 dev_err(led->dev->parent, "can't read wled ovp config"
512 " register rc=%d\n", rc);
513 return rc;
514 }
515
516 val = (val & ~WLED_OVP_VAL_MASK) |
517 (led->wled_cfg->ovp_val << WLED_OVP_VAL_BIT_SHFT);
518
519 rc = pm8xxx_writeb(led->dev->parent, WLED_OVP_CFG_REG, val);
520 if (rc) {
521 dev_err(led->dev->parent, "can't write wled ovp config"
522 " register rc=%d\n", rc);
523 return rc;
524 }
525
526 /* program current boost limit and output feedback*/
527 if (led->wled_cfg->boost_curr_lim > WLED_CURR_LIMIT_1680mA) {
528 dev_err(led->dev->parent, "Invalid boost current limit");
529 return -EINVAL;
530 }
531
532 rc = pm8xxx_readb(led->dev->parent, WLED_BOOST_CFG_REG, &val);
533 if (rc) {
534 dev_err(led->dev->parent, "can't read wled boost config"
535 " register rc=%d\n", rc);
536 return rc;
537 }
538
539 val = (val & ~WLED_BOOST_LIMIT_MASK) |
540 (led->wled_cfg->boost_curr_lim << WLED_BOOST_LIMIT_BIT_SHFT);
541
542 val = (val & ~WLED_OP_FDBCK_MASK) |
543 (led->wled_cfg->op_fdbck << WLED_OP_FDBCK_BIT_SHFT);
544
545 rc = pm8xxx_writeb(led->dev->parent, WLED_BOOST_CFG_REG, val);
546 if (rc) {
547 dev_err(led->dev->parent, "can't write wled boost config"
548 " register rc=%d\n", rc);
549 return rc;
550 }
551
552 /* program high pole capacitance */
553 if (led->wled_cfg->cp_select > WLED_CP_SELECT_MAX) {
554 dev_err(led->dev->parent, "Invalid pole capacitance");
555 return -EINVAL;
556 }
557
558 rc = pm8xxx_readb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, &val);
559 if (rc) {
560 dev_err(led->dev->parent, "can't read wled high pole"
561 " capacitance register rc=%d\n", rc);
562 return rc;
563 }
564
565 val = (val & ~WLED_CP_SELECT_MASK) | led->wled_cfg->cp_select;
566
567 rc = pm8xxx_writeb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, val);
568 if (rc) {
569 dev_err(led->dev->parent, "can't write wled high pole"
570 " capacitance register rc=%d\n", rc);
571 return rc;
572 }
573
574 /* program activation delay and maximum current */
575 for (i = 0; i < num_wled_strings; i++) {
576 rc = pm8xxx_readb(led->dev->parent,
577 WLED_MAX_CURR_CFG_REG(i + 2), &val);
578 if (rc) {
579 dev_err(led->dev->parent, "can't read wled max current"
580 " config register rc=%d\n", rc);
581 return rc;
582 }
583
584 if ((led->wled_cfg->ctrl_delay_us % WLED_CTL_DLY_STEP) ||
585 (led->wled_cfg->ctrl_delay_us > WLED_CTL_DLY_MAX)) {
586 dev_err(led->dev->parent, "Invalid control delay\n");
587 return rc;
588 }
589
590 val = val / WLED_CTL_DLY_STEP;
591 val = (val & ~WLED_CTL_DLY_MASK) |
592 (led->wled_cfg->ctrl_delay_us << WLED_CTL_DLY_BIT_SHFT);
593
594 if ((led->max_current > WLED_MAX_CURR)) {
595 dev_err(led->dev->parent, "Invalid max current\n");
596 return -EINVAL;
597 }
598
599 val = (val & ~WLED_MAX_CURR_MASK) | led->max_current;
600
601 rc = pm8xxx_writeb(led->dev->parent,
602 WLED_MAX_CURR_CFG_REG(i + 2), val);
603 if (rc) {
604 dev_err(led->dev->parent, "can't write wled max current"
605 " config register rc=%d\n", rc);
606 return rc;
607 }
608 }
609
610 /* program digital module generator, cs out and enable the module */
611 rc = pm8xxx_readb(led->dev->parent, WLED_MOD_CTRL_REG, &val);
612 if (rc) {
613 dev_err(led->dev->parent, "can't read wled module ctrl"
614 " register rc=%d\n", rc);
615 return rc;
616 }
617
618 if (led->wled_cfg->dig_mod_gen_en)
619 val |= WLED_DIG_MOD_GEN_MASK;
620
621 if (led->wled_cfg->cs_out_en)
622 val |= WLED_CS_OUT_MASK;
623
624 val |= WLED_EN_MASK;
625
626 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG, val);
627 if (rc) {
628 dev_err(led->dev->parent, "can't write wled module ctrl"
629 " register rc=%d\n", rc);
630 return rc;
631 }
632
633 /* dump wled registers */
634 wled_dump_regs(led);
635
636 return 0;
637}
638
Amy Malochec17c3732012-02-27 18:34:07 -0800639static int __devinit init_rgb_led(struct pm8xxx_led_data *led)
640{
641 int rc;
642 u8 val;
643
644 rc = pm8xxx_readb(led->dev->parent, SSBI_REG_ADDR_RGB_CNTL1, &val);
645 if (rc) {
646 dev_err(led->cdev.dev, "can't read rgb ctrl1 register rc=%d\n",
647 rc);
648 return rc;
649 }
650
651 switch (led->id) {
652 case PM8XXX_ID_RGB_LED_RED:
653 val |= PM8XXX_DRV_RGB_RED_LED;
654 break;
655 case PM8XXX_ID_RGB_LED_GREEN:
656 val |= PM8XXX_DRV_RGB_GREEN_LED;
657 break;
658 case PM8XXX_ID_RGB_LED_BLUE:
659 val |= PM8XXX_DRV_RGB_BLUE_LED;
660 break;
661 default:
662 return -EINVAL;
663 }
664
665 return pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_RGB_CNTL1, val);
666}
667
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700668static int __devinit get_init_value(struct pm8xxx_led_data *led, u8 *val)
669{
670 int rc, offset;
671 u16 addr;
672
673 switch (led->id) {
674 case PM8XXX_ID_LED_KB_LIGHT:
675 addr = SSBI_REG_ADDR_DRV_KEYPAD;
676 break;
677 case PM8XXX_ID_LED_0:
678 case PM8XXX_ID_LED_1:
679 case PM8XXX_ID_LED_2:
680 offset = PM8XXX_LED_OFFSET(led->id);
681 addr = SSBI_REG_ADDR_LED_CTRL(offset);
682 break;
683 case PM8XXX_ID_FLASH_LED_0:
684 addr = SSBI_REG_ADDR_FLASH_DRV0;
685 break;
686 case PM8XXX_ID_FLASH_LED_1:
687 addr = SSBI_REG_ADDR_FLASH_DRV1;
688 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530689 case PM8XXX_ID_WLED:
690 rc = init_wled(led);
691 if (rc)
692 dev_err(led->cdev.dev, "can't initialize wled rc=%d\n",
693 rc);
694 return rc;
Amy Malochec17c3732012-02-27 18:34:07 -0800695 case PM8XXX_ID_RGB_LED_RED:
696 case PM8XXX_ID_RGB_LED_GREEN:
697 case PM8XXX_ID_RGB_LED_BLUE:
698 rc = init_rgb_led(led);
699 if (rc) {
700 dev_err(led->cdev.dev, "can't initialize rgb rc=%d\n",
701 rc);
702 return rc;
703 }
704 addr = SSBI_REG_ADDR_RGB_CNTL1;
705 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530706 default:
707 dev_err(led->cdev.dev, "unknown led id %d", led->id);
708 return -EINVAL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700709 }
710
711 rc = pm8xxx_readb(led->dev->parent, addr, val);
712 if (rc)
713 dev_err(led->cdev.dev, "can't get led(%d) level rc=%d\n",
714 led->id, rc);
715
716 return rc;
717}
718
Jay Chokshi868312e2011-09-16 13:57:13 -0700719static int pm8xxx_led_pwm_configure(struct pm8xxx_led_data *led)
720{
721 int start_idx, idx_len, duty_us, rc;
722
723 led->pwm_dev = pwm_request(led->pwm_channel,
724 led->cdev.name);
725
726 if (IS_ERR_OR_NULL(led->pwm_dev)) {
727 pr_err("could not acquire PWM Channel %d, "
728 "error %ld\n", led->pwm_channel,
729 PTR_ERR(led->pwm_dev));
730 led->pwm_dev = NULL;
731 return -ENODEV;
732 }
733
734 if (led->pwm_duty_cycles != NULL) {
735 start_idx = led->pwm_duty_cycles->start_idx;
736 idx_len = led->pwm_duty_cycles->num_duty_pcts;
737
738 if (idx_len >= PM_PWM_LUT_SIZE && start_idx) {
739 pr_err("Wrong LUT size or index\n");
740 return -EINVAL;
741 }
742 if ((start_idx + idx_len) > PM_PWM_LUT_SIZE) {
743 pr_err("Exceed LUT limit\n");
744 return -EINVAL;
745 }
746
747 rc = pm8xxx_pwm_lut_config(led->pwm_dev, led->pwm_period_us,
748 led->pwm_duty_cycles->duty_pcts,
749 led->pwm_duty_cycles->duty_ms,
750 start_idx, idx_len, 0, 0,
751 PM8XXX_LED_PWM_FLAGS);
752 } else {
753 duty_us = led->pwm_period_us;
754 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
755 }
756
757 return rc;
758}
759
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530760
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700761static int __devinit pm8xxx_led_probe(struct platform_device *pdev)
762{
Jay Chokshi8994e392011-09-14 18:20:39 -0700763 const struct pm8xxx_led_platform_data *pdata = pdev->dev.platform_data;
764 const struct led_platform_data *pcore_data;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700765 struct led_info *curr_led;
766 struct pm8xxx_led_data *led, *led_dat;
Jay Chokshi8994e392011-09-14 18:20:39 -0700767 struct pm8xxx_led_config *led_cfg;
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700768 enum pm8xxx_version version;
769 bool found = false;
770 int rc, i, j;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700771
772 if (pdata == NULL) {
773 dev_err(&pdev->dev, "platform data not supplied\n");
774 return -EINVAL;
775 }
776
Jay Chokshi8994e392011-09-14 18:20:39 -0700777 pcore_data = pdata->led_core;
778
779 if (pcore_data->num_leds != pdata->num_configs) {
780 dev_err(&pdev->dev, "#no. of led configs and #no. of led"
781 "entries are not equal\n");
782 return -EINVAL;
783 }
784
785 led = kcalloc(pcore_data->num_leds, sizeof(*led), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700786 if (led == NULL) {
787 dev_err(&pdev->dev, "failed to alloc memory\n");
788 return -ENOMEM;
789 }
790
Jay Chokshi8994e392011-09-14 18:20:39 -0700791 for (i = 0; i < pcore_data->num_leds; i++) {
792 curr_led = &pcore_data->leds[i];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700793 led_dat = &led[i];
Jay Chokshi8994e392011-09-14 18:20:39 -0700794 led_cfg = &pdata->configs[i];
795
796 led_dat->id = led_cfg->id;
Jay Chokshi868312e2011-09-16 13:57:13 -0700797 led_dat->pwm_channel = led_cfg->pwm_channel;
798 led_dat->pwm_period_us = led_cfg->pwm_period_us;
799 led_dat->pwm_duty_cycles = led_cfg->pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530800 led_dat->wled_cfg = led_cfg->wled_cfg;
801 led_dat->max_current = led_cfg->max_current;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700802
803 if (!((led_dat->id >= PM8XXX_ID_LED_KB_LIGHT) &&
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530804 (led_dat->id < PM8XXX_ID_MAX))) {
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700805 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
806 led_dat->id);
807 rc = -EINVAL;
808 goto fail_id_check;
809
810 }
811
812 found = false;
813 version = pm8xxx_get_version(pdev->dev.parent);
814 for (j = 0; j < ARRAY_SIZE(led_map); j++) {
815 if (version == led_map[j].version
816 && (led_map[j].supported & (1 << led_dat->id))) {
817 found = true;
818 break;
819 }
820 }
821
822 if (!found) {
823 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
824 led_dat->id);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700825 rc = -EINVAL;
826 goto fail_id_check;
827 }
828
829 led_dat->cdev.name = curr_led->name;
830 led_dat->cdev.default_trigger = curr_led->default_trigger;
831 led_dat->cdev.brightness_set = pm8xxx_led_set;
832 led_dat->cdev.brightness_get = pm8xxx_led_get;
833 led_dat->cdev.brightness = LED_OFF;
Jay Chokshi8994e392011-09-14 18:20:39 -0700834 led_dat->cdev.flags = curr_led->flags;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700835 led_dat->dev = &pdev->dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700836
837 rc = get_init_value(led_dat, &led_dat->reg);
838 if (rc < 0)
839 goto fail_id_check;
840
Jay Chokshi8994e392011-09-14 18:20:39 -0700841 rc = pm8xxx_set_led_mode_and_max_brightness(led_dat,
842 led_cfg->mode, led_cfg->max_current);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700843 if (rc < 0)
844 goto fail_id_check;
845
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700846 mutex_init(&led_dat->lock);
847 INIT_WORK(&led_dat->work, pm8xxx_led_work);
848
849 rc = led_classdev_register(&pdev->dev, &led_dat->cdev);
850 if (rc) {
851 dev_err(&pdev->dev, "unable to register led %d,rc=%d\n",
852 led_dat->id, rc);
853 goto fail_id_check;
854 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700855
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530856 /* configure default state */
857 if (led_cfg->default_state)
858 led->cdev.brightness = led_dat->cdev.max_brightness;
859 else
860 led->cdev.brightness = LED_OFF;
861
Jay Chokshi868312e2011-09-16 13:57:13 -0700862 if (led_cfg->mode != PM8XXX_LED_MODE_MANUAL) {
Amy Malochec17c3732012-02-27 18:34:07 -0800863 if (led_dat->id == PM8XXX_ID_RGB_LED_RED ||
864 led_dat->id == PM8XXX_ID_RGB_LED_GREEN ||
865 led_dat->id == PM8XXX_ID_RGB_LED_BLUE)
866 __pm8xxx_led_work(led_dat, 0);
867 else
868 __pm8xxx_led_work(led_dat,
Jay Chokshide4cefb2011-08-04 18:10:44 -0700869 led_dat->cdev.max_brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700870
871 if (led_dat->pwm_channel != -1) {
872 led_dat->cdev.max_brightness = LED_FULL;
873 rc = pm8xxx_led_pwm_configure(led_dat);
874 if (rc) {
875 dev_err(&pdev->dev, "failed to "
876 "configure LED, error: %d\n", rc);
877 goto fail_id_check;
878 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530879 schedule_work(&led->work);
Jay Chokshi868312e2011-09-16 13:57:13 -0700880 }
881 } else {
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530882 __pm8xxx_led_work(led_dat, led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700883 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700884 }
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700885
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700886 platform_set_drvdata(pdev, led);
887
888 return 0;
889
890fail_id_check:
891 if (i > 0) {
892 for (i = i - 1; i >= 0; i--) {
893 mutex_destroy(&led[i].lock);
894 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -0700895 if (led[i].pwm_dev != NULL)
896 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700897 }
898 }
899 kfree(led);
900 return rc;
901}
902
903static int __devexit pm8xxx_led_remove(struct platform_device *pdev)
904{
905 int i;
906 const struct led_platform_data *pdata =
907 pdev->dev.platform_data;
908 struct pm8xxx_led_data *led = platform_get_drvdata(pdev);
909
910 for (i = 0; i < pdata->num_leds; i++) {
911 cancel_work_sync(&led[i].work);
912 mutex_destroy(&led[i].lock);
913 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -0700914 if (led[i].pwm_dev != NULL)
915 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700916 }
917
918 kfree(led);
919
920 return 0;
921}
922
923static struct platform_driver pm8xxx_led_driver = {
924 .probe = pm8xxx_led_probe,
925 .remove = __devexit_p(pm8xxx_led_remove),
926 .driver = {
927 .name = PM8XXX_LEDS_DEV_NAME,
928 .owner = THIS_MODULE,
929 },
930};
931
932static int __init pm8xxx_led_init(void)
933{
934 return platform_driver_register(&pm8xxx_led_driver);
935}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700936subsys_initcall(pm8xxx_led_init);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700937
938static void __exit pm8xxx_led_exit(void)
939{
940 platform_driver_unregister(&pm8xxx_led_driver);
941}
942module_exit(pm8xxx_led_exit);
943
944MODULE_DESCRIPTION("PM8XXX LEDs driver");
945MODULE_LICENSE("GPL v2");
946MODULE_VERSION("1.0");
947MODULE_ALIAS("platform:pm8xxx-led");