blob: d1bfc0726ef702457248404cf0f48a300374048d [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
Jay Chokshi12e49bf2011-07-22 16:24:39 -070082#define MAX_FLASH_LED_CURRENT 300
83#define MAX_LC_LED_CURRENT 40
84#define MAX_KP_BL_LED_CURRENT 300
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070085
Jay Chokshi12e49bf2011-07-22 16:24:39 -070086#define PM8XXX_ID_LED_CURRENT_FACTOR 2 /* Iout = x * 2mA */
87#define PM8XXX_ID_FLASH_CURRENT_FACTOR 20 /* Iout = x * 20mA */
88
89#define PM8XXX_FLASH_MODE_DBUS1 1
90#define PM8XXX_FLASH_MODE_DBUS2 2
91#define PM8XXX_FLASH_MODE_PWM 3
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070092
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070093#define MAX_LC_LED_BRIGHTNESS 20
Jay Chokshi12e49bf2011-07-22 16:24:39 -070094#define MAX_FLASH_BRIGHTNESS 15
95#define MAX_KB_LED_BRIGHTNESS 15
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070096
Jay Chokshide4cefb2011-08-04 18:10:44 -070097#define PM8XXX_LED_OFFSET(id) ((id) - PM8XXX_ID_LED_0)
98
Jay Chokshi868312e2011-09-16 13:57:13 -070099#define PM8XXX_LED_PWM_FLAGS (PM_PWM_LUT_LOOP | PM_PWM_LUT_RAMP_UP)
100
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700101#define LED_MAP(_version, _kb, _led0, _led1, _led2, _flash_led0, _flash_led1, \
102 _wled)\
103 {\
104 .version = _version,\
105 .supported = _kb << PM8XXX_ID_LED_KB_LIGHT | \
106 _led0 << PM8XXX_ID_LED_0 | _led1 << PM8XXX_ID_LED_1 | \
107 _led2 << PM8XXX_ID_LED_2 | \
108 _flash_led0 << PM8XXX_ID_FLASH_LED_0 | \
109 _flash_led1 << PM8XXX_ID_FLASH_LED_1 | \
110 _wled << PM8XXX_ID_WLED, \
111 }
112
113/**
114 * supported_leds - leds supported for each PMIC version
115 * @version - version of PMIC
116 * @supported - which leds are supported on version
117 */
118
119struct supported_leds {
120 enum pm8xxx_version version;
121 u32 supported;
122};
123
124static const struct supported_leds led_map[] = {
125 LED_MAP(PM8XXX_VERSION_8058, 1, 1, 1, 1, 1, 1, 0),
126 LED_MAP(PM8XXX_VERSION_8921, 1, 1, 1, 1, 1, 1, 0),
127 LED_MAP(PM8XXX_VERSION_8018, 1, 0, 0, 0, 0, 0, 0),
128 LED_MAP(PM8XXX_VERSION_8922, 0, 0, 0, 0, 1, 1, 1),
129 LED_MAP(PM8XXX_VERSION_8038, 0, 0, 0, 0, 0, 0, 1),
130};
131
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700132/**
133 * struct pm8xxx_led_data - internal led data structure
134 * @led_classdev - led class device
135 * @id - led index
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700136 * @work - workqueue for led
137 * @lock - to protect the transactions
138 * @reg - cached value of led register
Jay Chokshi868312e2011-09-16 13:57:13 -0700139 * @pwm_dev - pointer to PWM device if LED is driven using PWM
140 * @pwm_channel - PWM channel ID
141 * @pwm_period_us - PWM period in micro seconds
142 * @pwm_duty_cycles - struct that describes PWM duty cycles info
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700143 */
144struct pm8xxx_led_data {
145 struct led_classdev cdev;
146 int id;
147 u8 reg;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700148 struct device *dev;
149 struct work_struct work;
150 struct mutex lock;
Jay Chokshi868312e2011-09-16 13:57:13 -0700151 struct pwm_device *pwm_dev;
152 int pwm_channel;
153 u32 pwm_period_us;
154 struct pm8xxx_pwm_duty_cycles *pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530155 struct wled_config_data *wled_cfg;
156 int max_current;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700157};
158
159static void led_kp_set(struct pm8xxx_led_data *led, enum led_brightness value)
160{
161 int rc;
162 u8 level;
163
164 level = (value << PM8XXX_DRV_KEYPAD_BL_SHIFT) &
165 PM8XXX_DRV_KEYPAD_BL_MASK;
166
167 led->reg &= ~PM8XXX_DRV_KEYPAD_BL_MASK;
168 led->reg |= level;
169
170 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_DRV_KEYPAD,
171 led->reg);
172 if (rc < 0)
173 dev_err(led->cdev.dev,
174 "can't set keypad backlight level rc=%d\n", rc);
175}
176
177static void led_lc_set(struct pm8xxx_led_data *led, enum led_brightness value)
178{
179 int rc, offset;
180 u8 level;
181
182 level = (value << PM8XXX_DRV_LED_CTRL_SHIFT) &
183 PM8XXX_DRV_LED_CTRL_MASK;
184
185 offset = PM8XXX_LED_OFFSET(led->id);
186
187 led->reg &= ~PM8XXX_DRV_LED_CTRL_MASK;
188 led->reg |= level;
189
190 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_LED_CTRL(offset),
191 led->reg);
192 if (rc)
193 dev_err(led->cdev.dev, "can't set (%d) led value rc=%d\n",
194 led->id, rc);
195}
196
197static void
198led_flash_set(struct pm8xxx_led_data *led, enum led_brightness value)
199{
200 int rc;
201 u8 level;
202 u16 reg_addr;
203
204 level = (value << PM8XXX_DRV_FLASH_SHIFT) &
205 PM8XXX_DRV_FLASH_MASK;
206
207 led->reg &= ~PM8XXX_DRV_FLASH_MASK;
208 led->reg |= level;
209
210 if (led->id == PM8XXX_ID_FLASH_LED_0)
211 reg_addr = SSBI_REG_ADDR_FLASH_DRV0;
212 else
213 reg_addr = SSBI_REG_ADDR_FLASH_DRV1;
214
215 rc = pm8xxx_writeb(led->dev->parent, reg_addr, led->reg);
216 if (rc < 0)
217 dev_err(led->cdev.dev, "can't set flash led%d level rc=%d\n",
218 led->id, rc);
219}
220
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530221static int
222led_wled_set(struct pm8xxx_led_data *led, enum led_brightness value)
223{
224 int rc, duty;
225 u8 val, i, num_wled_strings;
226
227 if (value > WLED_MAX_LEVEL)
228 value = WLED_MAX_LEVEL;
229
230 duty = (WLED_MAX_DUTY_CYCLE * value) / WLED_MAX_LEVEL;
231
232 num_wled_strings = led->wled_cfg->num_strings;
233
234 /* program brightness control registers */
235 for (i = 0; i < num_wled_strings; i++) {
236 rc = pm8xxx_readb(led->dev->parent,
237 WLED_BRIGHTNESS_CNTL_REG1(i), &val);
238 if (rc) {
239 dev_err(led->dev->parent, "can't read wled brightnes ctrl"
240 " register1 rc=%d\n", rc);
241 return rc;
242 }
243
244 val = (val & ~WLED_MAX_CURR_MASK) | (duty >> WLED_8_BIT_SHFT);
245 rc = pm8xxx_writeb(led->dev->parent,
246 WLED_BRIGHTNESS_CNTL_REG1(i), val);
247 if (rc) {
248 dev_err(led->dev->parent, "can't write wled brightness ctrl"
249 " register1 rc=%d\n", rc);
250 return rc;
251 }
252
253 val = duty & WLED_8_BIT_MASK;
254 rc = pm8xxx_writeb(led->dev->parent,
255 WLED_BRIGHTNESS_CNTL_REG2(i), val);
256 if (rc) {
257 dev_err(led->dev->parent, "can't write wled brightness ctrl"
258 " register2 rc=%d\n", rc);
259 return rc;
260 }
261 }
262
263 /* sync */
264 val = WLED_SYNC_VAL;
265 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
266 if (rc) {
267 dev_err(led->dev->parent,
268 "can't read wled sync register rc=%d\n", rc);
269 return rc;
270 }
271
272 val = WLED_SYNC_RESET_VAL;
273 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
274 if (rc) {
275 dev_err(led->dev->parent,
276 "can't read wled sync register rc=%d\n", rc);
277 return rc;
278 }
279 return 0;
280}
281
282static void wled_dump_regs(struct pm8xxx_led_data *led)
283{
284 int i;
285 u8 val;
286
287 for (i = 1; i < 17; i++) {
288 pm8xxx_readb(led->dev->parent,
289 SSBI_REG_ADDR_WLED_CTRL(i), &val);
290 pr_debug("WLED_CTRL_%d = 0x%x\n", i, val);
291 }
292}
293
Jay Chokshi868312e2011-09-16 13:57:13 -0700294static int pm8xxx_led_pwm_work(struct pm8xxx_led_data *led)
295{
296 int duty_us;
297 int rc = 0;
298
299 if (led->pwm_duty_cycles == NULL) {
300 duty_us = (led->pwm_period_us * led->cdev.brightness) /
301 LED_FULL;
302 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
303 if (led->cdev.brightness)
304 rc = pwm_enable(led->pwm_dev);
305 else
306 pwm_disable(led->pwm_dev);
307 } else {
308 rc = pm8xxx_pwm_lut_enable(led->pwm_dev, led->cdev.brightness);
309 }
310
311 return rc;
312}
313
Jay Chokshide4cefb2011-08-04 18:10:44 -0700314static void __pm8xxx_led_work(struct pm8xxx_led_data *led,
315 enum led_brightness level)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700316{
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530317 int rc;
318
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700319 mutex_lock(&led->lock);
320
321 switch (led->id) {
322 case PM8XXX_ID_LED_KB_LIGHT:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700323 led_kp_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530324 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700325 case PM8XXX_ID_LED_0:
326 case PM8XXX_ID_LED_1:
327 case PM8XXX_ID_LED_2:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700328 led_lc_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530329 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700330 case PM8XXX_ID_FLASH_LED_0:
331 case PM8XXX_ID_FLASH_LED_1:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700332 led_flash_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530333 break;
334 case PM8XXX_ID_WLED:
335 rc = led_wled_set(led, level);
336 if (rc < 0)
337 pr_err("wled brightness set failed %d\n", rc);
338 break;
339 default:
340 dev_err(led->cdev.dev, "unknown led id %d", led->id);
341 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700342 }
343
344 mutex_unlock(&led->lock);
345}
346
Jay Chokshide4cefb2011-08-04 18:10:44 -0700347static void pm8xxx_led_work(struct work_struct *work)
348{
Jay Chokshi868312e2011-09-16 13:57:13 -0700349 int rc;
350
Jay Chokshide4cefb2011-08-04 18:10:44 -0700351 struct pm8xxx_led_data *led = container_of(work,
352 struct pm8xxx_led_data, work);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700353
Jay Chokshi868312e2011-09-16 13:57:13 -0700354 if (led->pwm_dev == NULL) {
355 __pm8xxx_led_work(led, led->cdev.brightness);
356 } else {
357 rc = pm8xxx_led_pwm_work(led);
358 if (rc)
359 pr_err("could not configure PWM mode for LED:%d\n",
360 led->id);
361 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700362}
363
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700364static void pm8xxx_led_set(struct led_classdev *led_cdev,
365 enum led_brightness value)
366{
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700367 struct pm8xxx_led_data *led;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700368
369 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
370
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700371 if (value < LED_OFF || value > led->cdev.max_brightness) {
372 dev_err(led->cdev.dev, "Invalid brightness value exceeds");
373 return;
374 }
375
376 led->cdev.brightness = value;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700377 schedule_work(&led->work);
378}
379
Jay Chokshide4cefb2011-08-04 18:10:44 -0700380static int pm8xxx_set_led_mode_and_max_brightness(struct pm8xxx_led_data *led,
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700381 enum pm8xxx_led_modes led_mode, int max_current)
382{
383 int rc = 0;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700384
Jay Chokshide4cefb2011-08-04 18:10:44 -0700385 switch (led->id) {
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700386 case PM8XXX_ID_LED_0:
387 case PM8XXX_ID_LED_1:
388 case PM8XXX_ID_LED_2:
389 led->cdev.max_brightness = max_current /
390 PM8XXX_ID_LED_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700391 if (led->cdev.max_brightness > MAX_LC_LED_BRIGHTNESS)
392 led->cdev.max_brightness = MAX_LC_LED_BRIGHTNESS;
Willie Ruana39d1d42011-08-19 13:54:34 -0700393 led->reg = led_mode;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700394 break;
395 case PM8XXX_ID_LED_KB_LIGHT:
396 case PM8XXX_ID_FLASH_LED_0:
397 case PM8XXX_ID_FLASH_LED_1:
398 led->cdev.max_brightness = max_current /
399 PM8XXX_ID_FLASH_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700400 if (led->cdev.max_brightness > MAX_FLASH_BRIGHTNESS)
401 led->cdev.max_brightness = MAX_FLASH_BRIGHTNESS;
402
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700403 switch (led_mode) {
404 case PM8XXX_LED_MODE_PWM1:
405 case PM8XXX_LED_MODE_PWM2:
406 case PM8XXX_LED_MODE_PWM3:
Willie Ruana39d1d42011-08-19 13:54:34 -0700407 led->reg = PM8XXX_FLASH_MODE_PWM;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700408 break;
409 case PM8XXX_LED_MODE_DTEST1:
Willie Ruana39d1d42011-08-19 13:54:34 -0700410 led->reg = PM8XXX_FLASH_MODE_DBUS1;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700411 break;
412 case PM8XXX_LED_MODE_DTEST2:
Willie Ruana39d1d42011-08-19 13:54:34 -0700413 led->reg = PM8XXX_FLASH_MODE_DBUS2;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700414 break;
415 default:
Willie Ruana39d1d42011-08-19 13:54:34 -0700416 led->reg = PM8XXX_LED_MODE_MANUAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700417 break;
418 }
419 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530420 case PM8XXX_ID_WLED:
421 led->cdev.max_brightness = WLED_MAX_LEVEL;
422 break;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700423 default:
424 rc = -EINVAL;
425 pr_err("LED Id is invalid");
426 break;
427 }
428
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700429 return rc;
430}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700431
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700432static enum led_brightness pm8xxx_led_get(struct led_classdev *led_cdev)
433{
434 struct pm8xxx_led_data *led;
435
436 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
437
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700438 return led->cdev.brightness;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700439}
440
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530441static int __devinit init_wled(struct pm8xxx_led_data *led)
442{
443 int rc, i;
444 u8 val, num_wled_strings;
445
446 num_wled_strings = led->wled_cfg->num_strings;
447
448 /* program over voltage protection threshold */
449 if (led->wled_cfg->ovp_val > WLED_OVP_37V) {
450 dev_err(led->dev->parent, "Invalid ovp value");
451 return -EINVAL;
452 }
453
454 rc = pm8xxx_readb(led->dev->parent, WLED_OVP_CFG_REG, &val);
455 if (rc) {
456 dev_err(led->dev->parent, "can't read wled ovp config"
457 " register rc=%d\n", rc);
458 return rc;
459 }
460
461 val = (val & ~WLED_OVP_VAL_MASK) |
462 (led->wled_cfg->ovp_val << WLED_OVP_VAL_BIT_SHFT);
463
464 rc = pm8xxx_writeb(led->dev->parent, WLED_OVP_CFG_REG, val);
465 if (rc) {
466 dev_err(led->dev->parent, "can't write wled ovp config"
467 " register rc=%d\n", rc);
468 return rc;
469 }
470
471 /* program current boost limit and output feedback*/
472 if (led->wled_cfg->boost_curr_lim > WLED_CURR_LIMIT_1680mA) {
473 dev_err(led->dev->parent, "Invalid boost current limit");
474 return -EINVAL;
475 }
476
477 rc = pm8xxx_readb(led->dev->parent, WLED_BOOST_CFG_REG, &val);
478 if (rc) {
479 dev_err(led->dev->parent, "can't read wled boost config"
480 " register rc=%d\n", rc);
481 return rc;
482 }
483
484 val = (val & ~WLED_BOOST_LIMIT_MASK) |
485 (led->wled_cfg->boost_curr_lim << WLED_BOOST_LIMIT_BIT_SHFT);
486
487 val = (val & ~WLED_OP_FDBCK_MASK) |
488 (led->wled_cfg->op_fdbck << WLED_OP_FDBCK_BIT_SHFT);
489
490 rc = pm8xxx_writeb(led->dev->parent, WLED_BOOST_CFG_REG, val);
491 if (rc) {
492 dev_err(led->dev->parent, "can't write wled boost config"
493 " register rc=%d\n", rc);
494 return rc;
495 }
496
497 /* program high pole capacitance */
498 if (led->wled_cfg->cp_select > WLED_CP_SELECT_MAX) {
499 dev_err(led->dev->parent, "Invalid pole capacitance");
500 return -EINVAL;
501 }
502
503 rc = pm8xxx_readb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, &val);
504 if (rc) {
505 dev_err(led->dev->parent, "can't read wled high pole"
506 " capacitance register rc=%d\n", rc);
507 return rc;
508 }
509
510 val = (val & ~WLED_CP_SELECT_MASK) | led->wled_cfg->cp_select;
511
512 rc = pm8xxx_writeb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, val);
513 if (rc) {
514 dev_err(led->dev->parent, "can't write wled high pole"
515 " capacitance register rc=%d\n", rc);
516 return rc;
517 }
518
519 /* program activation delay and maximum current */
520 for (i = 0; i < num_wled_strings; i++) {
521 rc = pm8xxx_readb(led->dev->parent,
522 WLED_MAX_CURR_CFG_REG(i + 2), &val);
523 if (rc) {
524 dev_err(led->dev->parent, "can't read wled max current"
525 " config register rc=%d\n", rc);
526 return rc;
527 }
528
529 if ((led->wled_cfg->ctrl_delay_us % WLED_CTL_DLY_STEP) ||
530 (led->wled_cfg->ctrl_delay_us > WLED_CTL_DLY_MAX)) {
531 dev_err(led->dev->parent, "Invalid control delay\n");
532 return rc;
533 }
534
535 val = val / WLED_CTL_DLY_STEP;
536 val = (val & ~WLED_CTL_DLY_MASK) |
537 (led->wled_cfg->ctrl_delay_us << WLED_CTL_DLY_BIT_SHFT);
538
539 if ((led->max_current > WLED_MAX_CURR)) {
540 dev_err(led->dev->parent, "Invalid max current\n");
541 return -EINVAL;
542 }
543
544 val = (val & ~WLED_MAX_CURR_MASK) | led->max_current;
545
546 rc = pm8xxx_writeb(led->dev->parent,
547 WLED_MAX_CURR_CFG_REG(i + 2), val);
548 if (rc) {
549 dev_err(led->dev->parent, "can't write wled max current"
550 " config register rc=%d\n", rc);
551 return rc;
552 }
553 }
554
555 /* program digital module generator, cs out and enable the module */
556 rc = pm8xxx_readb(led->dev->parent, WLED_MOD_CTRL_REG, &val);
557 if (rc) {
558 dev_err(led->dev->parent, "can't read wled module ctrl"
559 " register rc=%d\n", rc);
560 return rc;
561 }
562
563 if (led->wled_cfg->dig_mod_gen_en)
564 val |= WLED_DIG_MOD_GEN_MASK;
565
566 if (led->wled_cfg->cs_out_en)
567 val |= WLED_CS_OUT_MASK;
568
569 val |= WLED_EN_MASK;
570
571 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG, val);
572 if (rc) {
573 dev_err(led->dev->parent, "can't write wled module ctrl"
574 " register rc=%d\n", rc);
575 return rc;
576 }
577
578 /* dump wled registers */
579 wled_dump_regs(led);
580
581 return 0;
582}
583
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700584static int __devinit get_init_value(struct pm8xxx_led_data *led, u8 *val)
585{
586 int rc, offset;
587 u16 addr;
588
589 switch (led->id) {
590 case PM8XXX_ID_LED_KB_LIGHT:
591 addr = SSBI_REG_ADDR_DRV_KEYPAD;
592 break;
593 case PM8XXX_ID_LED_0:
594 case PM8XXX_ID_LED_1:
595 case PM8XXX_ID_LED_2:
596 offset = PM8XXX_LED_OFFSET(led->id);
597 addr = SSBI_REG_ADDR_LED_CTRL(offset);
598 break;
599 case PM8XXX_ID_FLASH_LED_0:
600 addr = SSBI_REG_ADDR_FLASH_DRV0;
601 break;
602 case PM8XXX_ID_FLASH_LED_1:
603 addr = SSBI_REG_ADDR_FLASH_DRV1;
604 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530605 case PM8XXX_ID_WLED:
606 rc = init_wled(led);
607 if (rc)
608 dev_err(led->cdev.dev, "can't initialize wled rc=%d\n",
609 rc);
610 return rc;
611 default:
612 dev_err(led->cdev.dev, "unknown led id %d", led->id);
613 return -EINVAL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700614 }
615
616 rc = pm8xxx_readb(led->dev->parent, addr, val);
617 if (rc)
618 dev_err(led->cdev.dev, "can't get led(%d) level rc=%d\n",
619 led->id, rc);
620
621 return rc;
622}
623
Jay Chokshi868312e2011-09-16 13:57:13 -0700624static int pm8xxx_led_pwm_configure(struct pm8xxx_led_data *led)
625{
626 int start_idx, idx_len, duty_us, rc;
627
628 led->pwm_dev = pwm_request(led->pwm_channel,
629 led->cdev.name);
630
631 if (IS_ERR_OR_NULL(led->pwm_dev)) {
632 pr_err("could not acquire PWM Channel %d, "
633 "error %ld\n", led->pwm_channel,
634 PTR_ERR(led->pwm_dev));
635 led->pwm_dev = NULL;
636 return -ENODEV;
637 }
638
639 if (led->pwm_duty_cycles != NULL) {
640 start_idx = led->pwm_duty_cycles->start_idx;
641 idx_len = led->pwm_duty_cycles->num_duty_pcts;
642
643 if (idx_len >= PM_PWM_LUT_SIZE && start_idx) {
644 pr_err("Wrong LUT size or index\n");
645 return -EINVAL;
646 }
647 if ((start_idx + idx_len) > PM_PWM_LUT_SIZE) {
648 pr_err("Exceed LUT limit\n");
649 return -EINVAL;
650 }
651
652 rc = pm8xxx_pwm_lut_config(led->pwm_dev, led->pwm_period_us,
653 led->pwm_duty_cycles->duty_pcts,
654 led->pwm_duty_cycles->duty_ms,
655 start_idx, idx_len, 0, 0,
656 PM8XXX_LED_PWM_FLAGS);
657 } else {
658 duty_us = led->pwm_period_us;
659 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
660 }
661
662 return rc;
663}
664
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530665
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700666static int __devinit pm8xxx_led_probe(struct platform_device *pdev)
667{
Jay Chokshi8994e392011-09-14 18:20:39 -0700668 const struct pm8xxx_led_platform_data *pdata = pdev->dev.platform_data;
669 const struct led_platform_data *pcore_data;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700670 struct led_info *curr_led;
671 struct pm8xxx_led_data *led, *led_dat;
Jay Chokshi8994e392011-09-14 18:20:39 -0700672 struct pm8xxx_led_config *led_cfg;
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700673 enum pm8xxx_version version;
674 bool found = false;
675 int rc, i, j;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700676
677 if (pdata == NULL) {
678 dev_err(&pdev->dev, "platform data not supplied\n");
679 return -EINVAL;
680 }
681
Jay Chokshi8994e392011-09-14 18:20:39 -0700682 pcore_data = pdata->led_core;
683
684 if (pcore_data->num_leds != pdata->num_configs) {
685 dev_err(&pdev->dev, "#no. of led configs and #no. of led"
686 "entries are not equal\n");
687 return -EINVAL;
688 }
689
690 led = kcalloc(pcore_data->num_leds, sizeof(*led), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700691 if (led == NULL) {
692 dev_err(&pdev->dev, "failed to alloc memory\n");
693 return -ENOMEM;
694 }
695
Jay Chokshi8994e392011-09-14 18:20:39 -0700696 for (i = 0; i < pcore_data->num_leds; i++) {
697 curr_led = &pcore_data->leds[i];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700698 led_dat = &led[i];
Jay Chokshi8994e392011-09-14 18:20:39 -0700699 led_cfg = &pdata->configs[i];
700
701 led_dat->id = led_cfg->id;
Jay Chokshi868312e2011-09-16 13:57:13 -0700702 led_dat->pwm_channel = led_cfg->pwm_channel;
703 led_dat->pwm_period_us = led_cfg->pwm_period_us;
704 led_dat->pwm_duty_cycles = led_cfg->pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530705 led_dat->wled_cfg = led_cfg->wled_cfg;
706 led_dat->max_current = led_cfg->max_current;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700707
708 if (!((led_dat->id >= PM8XXX_ID_LED_KB_LIGHT) &&
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530709 (led_dat->id < PM8XXX_ID_MAX))) {
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700710 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
711 led_dat->id);
712 rc = -EINVAL;
713 goto fail_id_check;
714
715 }
716
717 found = false;
718 version = pm8xxx_get_version(pdev->dev.parent);
719 for (j = 0; j < ARRAY_SIZE(led_map); j++) {
720 if (version == led_map[j].version
721 && (led_map[j].supported & (1 << led_dat->id))) {
722 found = true;
723 break;
724 }
725 }
726
727 if (!found) {
728 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
729 led_dat->id);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700730 rc = -EINVAL;
731 goto fail_id_check;
732 }
733
734 led_dat->cdev.name = curr_led->name;
735 led_dat->cdev.default_trigger = curr_led->default_trigger;
736 led_dat->cdev.brightness_set = pm8xxx_led_set;
737 led_dat->cdev.brightness_get = pm8xxx_led_get;
738 led_dat->cdev.brightness = LED_OFF;
Jay Chokshi8994e392011-09-14 18:20:39 -0700739 led_dat->cdev.flags = curr_led->flags;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700740 led_dat->dev = &pdev->dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700741
742 rc = get_init_value(led_dat, &led_dat->reg);
743 if (rc < 0)
744 goto fail_id_check;
745
Jay Chokshi8994e392011-09-14 18:20:39 -0700746 rc = pm8xxx_set_led_mode_and_max_brightness(led_dat,
747 led_cfg->mode, led_cfg->max_current);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700748 if (rc < 0)
749 goto fail_id_check;
750
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700751 mutex_init(&led_dat->lock);
752 INIT_WORK(&led_dat->work, pm8xxx_led_work);
753
754 rc = led_classdev_register(&pdev->dev, &led_dat->cdev);
755 if (rc) {
756 dev_err(&pdev->dev, "unable to register led %d,rc=%d\n",
757 led_dat->id, rc);
758 goto fail_id_check;
759 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700760
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530761 /* configure default state */
762 if (led_cfg->default_state)
763 led->cdev.brightness = led_dat->cdev.max_brightness;
764 else
765 led->cdev.brightness = LED_OFF;
766
Jay Chokshi868312e2011-09-16 13:57:13 -0700767 if (led_cfg->mode != PM8XXX_LED_MODE_MANUAL) {
Jay Chokshide4cefb2011-08-04 18:10:44 -0700768 __pm8xxx_led_work(led_dat,
769 led_dat->cdev.max_brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700770
771 if (led_dat->pwm_channel != -1) {
772 led_dat->cdev.max_brightness = LED_FULL;
773 rc = pm8xxx_led_pwm_configure(led_dat);
774 if (rc) {
775 dev_err(&pdev->dev, "failed to "
776 "configure LED, error: %d\n", rc);
777 goto fail_id_check;
778 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530779 schedule_work(&led->work);
Jay Chokshi868312e2011-09-16 13:57:13 -0700780 }
781 } else {
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530782 __pm8xxx_led_work(led_dat, led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700783 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700784 }
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700785
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700786 platform_set_drvdata(pdev, led);
787
788 return 0;
789
790fail_id_check:
791 if (i > 0) {
792 for (i = i - 1; i >= 0; i--) {
793 mutex_destroy(&led[i].lock);
794 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -0700795 if (led[i].pwm_dev != NULL)
796 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700797 }
798 }
799 kfree(led);
800 return rc;
801}
802
803static int __devexit pm8xxx_led_remove(struct platform_device *pdev)
804{
805 int i;
806 const struct led_platform_data *pdata =
807 pdev->dev.platform_data;
808 struct pm8xxx_led_data *led = platform_get_drvdata(pdev);
809
810 for (i = 0; i < pdata->num_leds; i++) {
811 cancel_work_sync(&led[i].work);
812 mutex_destroy(&led[i].lock);
813 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -0700814 if (led[i].pwm_dev != NULL)
815 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700816 }
817
818 kfree(led);
819
820 return 0;
821}
822
823static struct platform_driver pm8xxx_led_driver = {
824 .probe = pm8xxx_led_probe,
825 .remove = __devexit_p(pm8xxx_led_remove),
826 .driver = {
827 .name = PM8XXX_LEDS_DEV_NAME,
828 .owner = THIS_MODULE,
829 },
830};
831
832static int __init pm8xxx_led_init(void)
833{
834 return platform_driver_register(&pm8xxx_led_driver);
835}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700836subsys_initcall(pm8xxx_led_init);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700837
838static void __exit pm8xxx_led_exit(void)
839{
840 platform_driver_unregister(&pm8xxx_led_driver);
841}
842module_exit(pm8xxx_led_exit);
843
844MODULE_DESCRIPTION("PM8XXX LEDs driver");
845MODULE_LICENSE("GPL v2");
846MODULE_VERSION("1.0");
847MODULE_ALIAS("platform:pm8xxx-led");