blob: aef6295db0831efd7f1115736fe5a83efb7848f2 [file] [log] [blame]
Amy Malochee7d7ef32013-01-30 16:16:35 -08001/* Copyright (c) 2010-2013, The Linux Foundation. All rights reserved.
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
Jay Chokshi12e49bf2011-07-22 16:24:39 -070013#define pr_fmt(fmt) "%s: " fmt, __func__
14
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070015#include <linux/kernel.h>
Steve Mucklef132c6c2012-06-06 18:30:57 -070016#include <linux/module.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070017#include <linux/init.h>
18#include <linux/slab.h>
19#include <linux/platform_device.h>
20#include <linux/leds.h>
21#include <linux/workqueue.h>
Jay Chokshi868312e2011-09-16 13:57:13 -070022#include <linux/err.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070023
24#include <linux/mfd/pm8xxx/core.h>
Jay Chokshi868312e2011-09-16 13:57:13 -070025#include <linux/mfd/pm8xxx/pwm.h>
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -070026#include <linux/leds-pm8xxx.h>
27
28#define SSBI_REG_ADDR_DRV_KEYPAD 0x48
29#define PM8XXX_DRV_KEYPAD_BL_MASK 0xf0
30#define PM8XXX_DRV_KEYPAD_BL_SHIFT 0x04
31
32#define SSBI_REG_ADDR_FLASH_DRV0 0x49
33#define PM8XXX_DRV_FLASH_MASK 0xf0
34#define PM8XXX_DRV_FLASH_SHIFT 0x04
35
36#define SSBI_REG_ADDR_FLASH_DRV1 0xFB
37
38#define SSBI_REG_ADDR_LED_CTRL_BASE 0x131
39#define SSBI_REG_ADDR_LED_CTRL(n) (SSBI_REG_ADDR_LED_CTRL_BASE + (n))
40#define PM8XXX_DRV_LED_CTRL_MASK 0xf8
41#define PM8XXX_DRV_LED_CTRL_SHIFT 0x03
42
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053043#define SSBI_REG_ADDR_WLED_CTRL_BASE 0x25A
44#define SSBI_REG_ADDR_WLED_CTRL(n) (SSBI_REG_ADDR_WLED_CTRL_BASE + (n) - 1)
45
46/* wled control registers */
47#define WLED_MOD_CTRL_REG SSBI_REG_ADDR_WLED_CTRL(1)
48#define WLED_MAX_CURR_CFG_REG(n) SSBI_REG_ADDR_WLED_CTRL(n + 2)
Mohan Pallaka58ff4052012-11-19 16:58:59 +053049#define WLED_BRIGHTNESS_CNTL_REG1(n) SSBI_REG_ADDR_WLED_CTRL((2 * n) + 5)
50#define WLED_BRIGHTNESS_CNTL_REG2(n) SSBI_REG_ADDR_WLED_CTRL((2 * n) + 6)
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053051#define WLED_SYNC_REG SSBI_REG_ADDR_WLED_CTRL(11)
52#define WLED_OVP_CFG_REG SSBI_REG_ADDR_WLED_CTRL(13)
53#define WLED_BOOST_CFG_REG SSBI_REG_ADDR_WLED_CTRL(14)
54#define WLED_HIGH_POLE_CAP_REG SSBI_REG_ADDR_WLED_CTRL(16)
55
Amy Maloche59cf8182012-11-30 12:27:40 -080056#define WLED_STRING_ONE 0 /* Rightmost string */
57#define WLED_STRING_TWO 1 /* Middle string */
58#define WLED_STRING_THREE 2 /* Leftmost string */
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053059#define WLED_STRINGS 0x03
60#define WLED_OVP_VAL_MASK 0x30
61#define WLED_OVP_VAL_BIT_SHFT 0x04
62#define WLED_BOOST_LIMIT_MASK 0xE0
63#define WLED_BOOST_LIMIT_BIT_SHFT 0x05
Amy Maloche56913f52012-05-11 10:47:24 -070064#define WLED_BOOST_OFF 0x00
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053065#define WLED_EN_MASK 0x01
66#define WLED_CP_SELECT_MAX 0x03
67#define WLED_CP_SELECT_MASK 0x03
68#define WLED_DIG_MOD_GEN_MASK 0x70
69#define WLED_CS_OUT_MASK 0x0E
70#define WLED_CTL_DLY_STEP 200
71#define WLED_CTL_DLY_MAX 1400
72#define WLED_CTL_DLY_MASK 0xE0
73#define WLED_CTL_DLY_BIT_SHFT 0x05
74#define WLED_MAX_CURR 25
75#define WLED_MAX_CURR_MASK 0x1F
Amy Malochebe73e192012-11-29 12:53:59 -080076#define WLED_BRIGHTNESS_MSB_MASK 0x0F
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053077#define WLED_OP_FDBCK_MASK 0x1C
78#define WLED_OP_FDBCK_BIT_SHFT 0x02
79
Chandan Uddaraju6e73f0a2012-03-08 17:32:55 -080080#define WLED_MAX_LEVEL 255
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053081#define WLED_8_BIT_MASK 0xFF
82#define WLED_8_BIT_SHFT 0x08
83#define WLED_MAX_DUTY_CYCLE 0xFFF
84
85#define WLED_SYNC_VAL 0x07
86#define WLED_SYNC_RESET_VAL 0x00
Amy Malochec366d5b2012-08-28 11:06:40 -070087#define WLED_SYNC_MASK 0xF8
88
89#define ONE_WLED_STRING 1
90#define TWO_WLED_STRINGS 2
91#define THREE_WLED_STRINGS 3
92
Amy Malochec366d5b2012-08-28 11:06:40 -070093#define WLED_CABC_SHIFT 3
Mohan Pallaka53fe1a12011-12-15 16:56:37 +053094
Amy Malochec17c3732012-02-27 18:34:07 -080095#define SSBI_REG_ADDR_RGB_CNTL1 0x12D
96#define SSBI_REG_ADDR_RGB_CNTL2 0x12E
97
98#define PM8XXX_DRV_RGB_RED_LED BIT(2)
99#define PM8XXX_DRV_RGB_GREEN_LED BIT(1)
100#define PM8XXX_DRV_RGB_BLUE_LED BIT(0)
101
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700102#define MAX_FLASH_LED_CURRENT 300
103#define MAX_LC_LED_CURRENT 40
104#define MAX_KP_BL_LED_CURRENT 300
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700105
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700106#define PM8XXX_ID_LED_CURRENT_FACTOR 2 /* Iout = x * 2mA */
107#define PM8XXX_ID_FLASH_CURRENT_FACTOR 20 /* Iout = x * 20mA */
108
109#define PM8XXX_FLASH_MODE_DBUS1 1
110#define PM8XXX_FLASH_MODE_DBUS2 2
111#define PM8XXX_FLASH_MODE_PWM 3
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700112
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700113#define MAX_LC_LED_BRIGHTNESS 20
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700114#define MAX_FLASH_BRIGHTNESS 15
115#define MAX_KB_LED_BRIGHTNESS 15
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700116
Jay Chokshide4cefb2011-08-04 18:10:44 -0700117#define PM8XXX_LED_OFFSET(id) ((id) - PM8XXX_ID_LED_0)
118
Jay Chokshi868312e2011-09-16 13:57:13 -0700119#define PM8XXX_LED_PWM_FLAGS (PM_PWM_LUT_LOOP | PM_PWM_LUT_RAMP_UP)
120
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700121#define LED_MAP(_version, _kb, _led0, _led1, _led2, _flash_led0, _flash_led1, \
Amy Malochec17c3732012-02-27 18:34:07 -0800122 _wled, _rgb_led_red, _rgb_led_green, _rgb_led_blue)\
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700123 {\
124 .version = _version,\
125 .supported = _kb << PM8XXX_ID_LED_KB_LIGHT | \
126 _led0 << PM8XXX_ID_LED_0 | _led1 << PM8XXX_ID_LED_1 | \
127 _led2 << PM8XXX_ID_LED_2 | \
128 _flash_led0 << PM8XXX_ID_FLASH_LED_0 | \
129 _flash_led1 << PM8XXX_ID_FLASH_LED_1 | \
Amy Malochec17c3732012-02-27 18:34:07 -0800130 _wled << PM8XXX_ID_WLED | \
131 _rgb_led_red << PM8XXX_ID_RGB_LED_RED | \
132 _rgb_led_green << PM8XXX_ID_RGB_LED_GREEN | \
133 _rgb_led_blue << PM8XXX_ID_RGB_LED_BLUE, \
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700134 }
135
Prasad Sodagudi5af87922013-06-05 17:33:52 +0530136#define PM8XXX_PWM_CURRENT_4MA 4
137#define PM8XXX_PWM_CURRENT_8MA 8
138#define PM8XXX_PWM_CURRENT_12MA 12
139
140
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700141/**
142 * supported_leds - leds supported for each PMIC version
143 * @version - version of PMIC
144 * @supported - which leds are supported on version
145 */
146
147struct supported_leds {
148 enum pm8xxx_version version;
149 u32 supported;
150};
151
152static const struct supported_leds led_map[] = {
Amy Malochec17c3732012-02-27 18:34:07 -0800153 LED_MAP(PM8XXX_VERSION_8058, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
154 LED_MAP(PM8XXX_VERSION_8921, 1, 1, 1, 1, 1, 1, 0, 0, 0, 0),
155 LED_MAP(PM8XXX_VERSION_8018, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0),
156 LED_MAP(PM8XXX_VERSION_8922, 0, 0, 0, 0, 1, 1, 1, 1, 1, 1),
157 LED_MAP(PM8XXX_VERSION_8038, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1),
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700158};
159
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700160/**
161 * struct pm8xxx_led_data - internal led data structure
162 * @led_classdev - led class device
163 * @id - led index
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700164 * @work - workqueue for led
165 * @lock - to protect the transactions
166 * @reg - cached value of led register
Jay Chokshi868312e2011-09-16 13:57:13 -0700167 * @pwm_dev - pointer to PWM device if LED is driven using PWM
168 * @pwm_channel - PWM channel ID
169 * @pwm_period_us - PWM period in micro seconds
170 * @pwm_duty_cycles - struct that describes PWM duty cycles info
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700171 */
172struct pm8xxx_led_data {
173 struct led_classdev cdev;
174 int id;
175 u8 reg;
Amy Maloche56913f52012-05-11 10:47:24 -0700176 u8 wled_mod_ctrl_val;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700177 struct device *dev;
178 struct work_struct work;
179 struct mutex lock;
Jay Chokshi868312e2011-09-16 13:57:13 -0700180 struct pwm_device *pwm_dev;
181 int pwm_channel;
182 u32 pwm_period_us;
183 struct pm8xxx_pwm_duty_cycles *pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530184 struct wled_config_data *wled_cfg;
185 int max_current;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700186};
187
188static void led_kp_set(struct pm8xxx_led_data *led, enum led_brightness value)
189{
190 int rc;
191 u8 level;
192
193 level = (value << PM8XXX_DRV_KEYPAD_BL_SHIFT) &
194 PM8XXX_DRV_KEYPAD_BL_MASK;
195
196 led->reg &= ~PM8XXX_DRV_KEYPAD_BL_MASK;
197 led->reg |= level;
198
199 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_DRV_KEYPAD,
200 led->reg);
201 if (rc < 0)
202 dev_err(led->cdev.dev,
203 "can't set keypad backlight level rc=%d\n", rc);
204}
205
206static void led_lc_set(struct pm8xxx_led_data *led, enum led_brightness value)
207{
208 int rc, offset;
209 u8 level;
210
211 level = (value << PM8XXX_DRV_LED_CTRL_SHIFT) &
212 PM8XXX_DRV_LED_CTRL_MASK;
213
214 offset = PM8XXX_LED_OFFSET(led->id);
215
216 led->reg &= ~PM8XXX_DRV_LED_CTRL_MASK;
217 led->reg |= level;
218
219 rc = pm8xxx_writeb(led->dev->parent, SSBI_REG_ADDR_LED_CTRL(offset),
220 led->reg);
221 if (rc)
222 dev_err(led->cdev.dev, "can't set (%d) led value rc=%d\n",
223 led->id, rc);
224}
225
226static void
227led_flash_set(struct pm8xxx_led_data *led, enum led_brightness value)
228{
229 int rc;
230 u8 level;
231 u16 reg_addr;
232
233 level = (value << PM8XXX_DRV_FLASH_SHIFT) &
234 PM8XXX_DRV_FLASH_MASK;
235
236 led->reg &= ~PM8XXX_DRV_FLASH_MASK;
237 led->reg |= level;
238
239 if (led->id == PM8XXX_ID_FLASH_LED_0)
240 reg_addr = SSBI_REG_ADDR_FLASH_DRV0;
241 else
242 reg_addr = SSBI_REG_ADDR_FLASH_DRV1;
243
244 rc = pm8xxx_writeb(led->dev->parent, reg_addr, led->reg);
245 if (rc < 0)
246 dev_err(led->cdev.dev, "can't set flash led%d level rc=%d\n",
247 led->id, rc);
248}
249
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530250static int
251led_wled_set(struct pm8xxx_led_data *led, enum led_brightness value)
252{
253 int rc, duty;
Amy Malochee7d7ef32013-01-30 16:16:35 -0800254 u8 val, i;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530255
256 if (value > WLED_MAX_LEVEL)
257 value = WLED_MAX_LEVEL;
258
Amy Maloche56913f52012-05-11 10:47:24 -0700259 if (value == 0) {
260 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
261 WLED_BOOST_OFF);
262 if (rc) {
263 dev_err(led->dev->parent, "can't write wled ctrl config"
264 " register rc=%d\n", rc);
265 return rc;
266 }
267 } else {
268 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG,
269 led->wled_mod_ctrl_val);
270 if (rc) {
271 dev_err(led->dev->parent, "can't write wled ctrl config"
272 " register rc=%d\n", rc);
273 return rc;
274 }
275 }
276
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530277 duty = (WLED_MAX_DUTY_CYCLE * value) / WLED_MAX_LEVEL;
278
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530279 /* program brightness control registers */
Amy Malochee7d7ef32013-01-30 16:16:35 -0800280 for (i = 0; i < WLED_STRINGS; i++) {
281 if (led->wled_cfg->strings && (1 << i)) {
282 rc = pm8xxx_readb(led->dev->parent,
283 WLED_BRIGHTNESS_CNTL_REG1(i), &val);
284 if (rc) {
285 dev_err(led->dev->parent,
286 "can't read wled brightnes ctrl"
287 " register1 rc=%d\n", rc);
288 return rc;
289 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530290
Amy Malochee7d7ef32013-01-30 16:16:35 -0800291 val = (val & ~WLED_MAX_CURR_MASK) |
292 (duty >> WLED_8_BIT_SHFT);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530293
Amy Malochee7d7ef32013-01-30 16:16:35 -0800294 rc = pm8xxx_writeb(led->dev->parent,
295 WLED_BRIGHTNESS_CNTL_REG1(i), val);
296 if (rc) {
297 dev_err(led->dev->parent,
298 "can't write wled brightness ctrl"
299 " register1 rc=%d\n", rc);
300 return rc;
301 }
302
303 val = duty & WLED_8_BIT_MASK;
304 rc = pm8xxx_writeb(led->dev->parent,
305 WLED_BRIGHTNESS_CNTL_REG2(i), val);
306 if (rc) {
307 dev_err(led->dev->parent,
308 "can't write wled brightness ctrl"
309 " register2 rc=%d\n", rc);
310 return rc;
311 }
312 } else
313 continue;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530314 }
Amy Malochec366d5b2012-08-28 11:06:40 -0700315 rc = pm8xxx_readb(led->dev->parent, WLED_SYNC_REG, &val);
316 if (rc) {
317 dev_err(led->dev->parent,
318 "can't read wled sync register rc=%d\n", rc);
319 return rc;
320 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530321 /* sync */
Amy Malochec366d5b2012-08-28 11:06:40 -0700322 val &= WLED_SYNC_MASK;
323 val |= WLED_SYNC_VAL;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530324 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
325 if (rc) {
326 dev_err(led->dev->parent,
327 "can't read wled sync register rc=%d\n", rc);
328 return rc;
329 }
Amy Malochec366d5b2012-08-28 11:06:40 -0700330 val &= WLED_SYNC_MASK;
331 val |= WLED_SYNC_RESET_VAL;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530332 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
333 if (rc) {
334 dev_err(led->dev->parent,
335 "can't read wled sync register rc=%d\n", rc);
336 return rc;
337 }
338 return 0;
339}
340
341static void wled_dump_regs(struct pm8xxx_led_data *led)
342{
343 int i;
344 u8 val;
345
346 for (i = 1; i < 17; i++) {
347 pm8xxx_readb(led->dev->parent,
348 SSBI_REG_ADDR_WLED_CTRL(i), &val);
349 pr_debug("WLED_CTRL_%d = 0x%x\n", i, val);
350 }
351}
352
Amy Malochec17c3732012-02-27 18:34:07 -0800353static void
Amy Malochead3c7842012-07-20 14:51:27 -0700354led_rgb_write(struct pm8xxx_led_data *led, u16 addr, enum led_brightness value)
Amy Malochec17c3732012-02-27 18:34:07 -0800355{
356 int rc;
357 u8 val, mask;
358
Amy Malochead3c7842012-07-20 14:51:27 -0700359 if (led->id != PM8XXX_ID_RGB_LED_BLUE &&
360 led->id != PM8XXX_ID_RGB_LED_RED &&
361 led->id != PM8XXX_ID_RGB_LED_GREEN)
362 return;
363
364 rc = pm8xxx_readb(led->dev->parent, addr, &val);
Amy Malochec17c3732012-02-27 18:34:07 -0800365 if (rc) {
366 dev_err(led->cdev.dev, "can't read rgb ctrl register rc=%d\n",
367 rc);
368 return;
369 }
370
371 switch (led->id) {
372 case PM8XXX_ID_RGB_LED_RED:
373 mask = PM8XXX_DRV_RGB_RED_LED;
374 break;
375 case PM8XXX_ID_RGB_LED_GREEN:
376 mask = PM8XXX_DRV_RGB_GREEN_LED;
377 break;
378 case PM8XXX_ID_RGB_LED_BLUE:
379 mask = PM8XXX_DRV_RGB_BLUE_LED;
380 break;
381 default:
382 return;
383 }
384
385 if (value)
386 val |= mask;
387 else
388 val &= ~mask;
389
Amy Malochead3c7842012-07-20 14:51:27 -0700390 rc = pm8xxx_writeb(led->dev->parent, addr, val);
Amy Malochec17c3732012-02-27 18:34:07 -0800391 if (rc < 0)
392 dev_err(led->cdev.dev, "can't set rgb led %d level rc=%d\n",
393 led->id, rc);
394}
395
Amy Malochead3c7842012-07-20 14:51:27 -0700396static void
397led_rgb_set(struct pm8xxx_led_data *led, enum led_brightness value)
398{
399 if (value) {
400 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, value);
401 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL2, value);
402 } else {
403 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL2, value);
404 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1, value);
405 }
406}
407
Jay Chokshi868312e2011-09-16 13:57:13 -0700408static int pm8xxx_led_pwm_work(struct pm8xxx_led_data *led)
409{
410 int duty_us;
411 int rc = 0;
412
413 if (led->pwm_duty_cycles == NULL) {
414 duty_us = (led->pwm_period_us * led->cdev.brightness) /
415 LED_FULL;
416 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
Amy Malochead3c7842012-07-20 14:51:27 -0700417 if (led->cdev.brightness) {
418 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
419 led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700420 rc = pwm_enable(led->pwm_dev);
Amy Malochead3c7842012-07-20 14:51:27 -0700421 } else {
Jay Chokshi868312e2011-09-16 13:57:13 -0700422 pwm_disable(led->pwm_dev);
Amy Malochead3c7842012-07-20 14:51:27 -0700423 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
424 led->cdev.brightness);
425 }
Jay Chokshi868312e2011-09-16 13:57:13 -0700426 } else {
Amy Malochead3c7842012-07-20 14:51:27 -0700427 if (led->cdev.brightness)
428 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
429 led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700430 rc = pm8xxx_pwm_lut_enable(led->pwm_dev, led->cdev.brightness);
Amy Malochead3c7842012-07-20 14:51:27 -0700431 if (!led->cdev.brightness)
432 led_rgb_write(led, SSBI_REG_ADDR_RGB_CNTL1,
433 led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700434 }
435
436 return rc;
437}
438
Jay Chokshide4cefb2011-08-04 18:10:44 -0700439static void __pm8xxx_led_work(struct pm8xxx_led_data *led,
440 enum led_brightness level)
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700441{
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530442 int rc;
443
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700444 mutex_lock(&led->lock);
445
446 switch (led->id) {
447 case PM8XXX_ID_LED_KB_LIGHT:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700448 led_kp_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530449 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700450 case PM8XXX_ID_LED_0:
451 case PM8XXX_ID_LED_1:
452 case PM8XXX_ID_LED_2:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700453 led_lc_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530454 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700455 case PM8XXX_ID_FLASH_LED_0:
456 case PM8XXX_ID_FLASH_LED_1:
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700457 led_flash_set(led, level);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530458 break;
459 case PM8XXX_ID_WLED:
460 rc = led_wled_set(led, level);
461 if (rc < 0)
462 pr_err("wled brightness set failed %d\n", rc);
463 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800464 case PM8XXX_ID_RGB_LED_RED:
465 case PM8XXX_ID_RGB_LED_GREEN:
466 case PM8XXX_ID_RGB_LED_BLUE:
467 led_rgb_set(led, level);
468 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530469 default:
470 dev_err(led->cdev.dev, "unknown led id %d", led->id);
471 break;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700472 }
473
474 mutex_unlock(&led->lock);
475}
476
Jay Chokshide4cefb2011-08-04 18:10:44 -0700477static void pm8xxx_led_work(struct work_struct *work)
478{
Jay Chokshi868312e2011-09-16 13:57:13 -0700479 int rc;
480
Jay Chokshide4cefb2011-08-04 18:10:44 -0700481 struct pm8xxx_led_data *led = container_of(work,
482 struct pm8xxx_led_data, work);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700483
Jay Chokshi868312e2011-09-16 13:57:13 -0700484 if (led->pwm_dev == NULL) {
485 __pm8xxx_led_work(led, led->cdev.brightness);
486 } else {
487 rc = pm8xxx_led_pwm_work(led);
488 if (rc)
489 pr_err("could not configure PWM mode for LED:%d\n",
490 led->id);
491 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700492}
493
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700494static void pm8xxx_led_set(struct led_classdev *led_cdev,
495 enum led_brightness value)
496{
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700497 struct pm8xxx_led_data *led;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700498
499 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
500
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700501 if (value < LED_OFF || value > led->cdev.max_brightness) {
502 dev_err(led->cdev.dev, "Invalid brightness value exceeds");
503 return;
504 }
505
506 led->cdev.brightness = value;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700507 schedule_work(&led->work);
508}
509
Jay Chokshide4cefb2011-08-04 18:10:44 -0700510static int pm8xxx_set_led_mode_and_max_brightness(struct pm8xxx_led_data *led,
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700511 enum pm8xxx_led_modes led_mode, int max_current)
512{
Jay Chokshide4cefb2011-08-04 18:10:44 -0700513 switch (led->id) {
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700514 case PM8XXX_ID_LED_0:
515 case PM8XXX_ID_LED_1:
516 case PM8XXX_ID_LED_2:
517 led->cdev.max_brightness = max_current /
518 PM8XXX_ID_LED_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700519 if (led->cdev.max_brightness > MAX_LC_LED_BRIGHTNESS)
520 led->cdev.max_brightness = MAX_LC_LED_BRIGHTNESS;
Willie Ruana39d1d42011-08-19 13:54:34 -0700521 led->reg = led_mode;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700522 break;
523 case PM8XXX_ID_LED_KB_LIGHT:
524 case PM8XXX_ID_FLASH_LED_0:
525 case PM8XXX_ID_FLASH_LED_1:
526 led->cdev.max_brightness = max_current /
527 PM8XXX_ID_FLASH_CURRENT_FACTOR;
Jay Chokshide4cefb2011-08-04 18:10:44 -0700528 if (led->cdev.max_brightness > MAX_FLASH_BRIGHTNESS)
529 led->cdev.max_brightness = MAX_FLASH_BRIGHTNESS;
530
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700531 switch (led_mode) {
532 case PM8XXX_LED_MODE_PWM1:
533 case PM8XXX_LED_MODE_PWM2:
534 case PM8XXX_LED_MODE_PWM3:
Willie Ruana39d1d42011-08-19 13:54:34 -0700535 led->reg = PM8XXX_FLASH_MODE_PWM;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700536 break;
537 case PM8XXX_LED_MODE_DTEST1:
Willie Ruana39d1d42011-08-19 13:54:34 -0700538 led->reg = PM8XXX_FLASH_MODE_DBUS1;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700539 break;
540 case PM8XXX_LED_MODE_DTEST2:
Willie Ruana39d1d42011-08-19 13:54:34 -0700541 led->reg = PM8XXX_FLASH_MODE_DBUS2;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700542 break;
543 default:
Willie Ruana39d1d42011-08-19 13:54:34 -0700544 led->reg = PM8XXX_LED_MODE_MANUAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700545 break;
546 }
547 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530548 case PM8XXX_ID_WLED:
549 led->cdev.max_brightness = WLED_MAX_LEVEL;
550 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800551 case PM8XXX_ID_RGB_LED_RED:
552 case PM8XXX_ID_RGB_LED_GREEN:
553 case PM8XXX_ID_RGB_LED_BLUE:
554 led->cdev.max_brightness = LED_FULL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700555 break;
Amy Malochec17c3732012-02-27 18:34:07 -0800556 default:
557 dev_err(led->cdev.dev, "LED Id is invalid");
558 return -EINVAL;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700559 }
560
Amy Malochec17c3732012-02-27 18:34:07 -0800561 return 0;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700562}
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700563
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700564static enum led_brightness pm8xxx_led_get(struct led_classdev *led_cdev)
565{
566 struct pm8xxx_led_data *led;
567
568 led = container_of(led_cdev, struct pm8xxx_led_data, cdev);
569
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700570 return led->cdev.brightness;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700571}
572
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530573static int __devinit init_wled(struct pm8xxx_led_data *led)
574{
575 int rc, i;
Amy Maloche59cf8182012-11-30 12:27:40 -0800576 u8 val, string_max_current;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530577
578 /* program over voltage protection threshold */
579 if (led->wled_cfg->ovp_val > WLED_OVP_37V) {
580 dev_err(led->dev->parent, "Invalid ovp value");
581 return -EINVAL;
582 }
583
584 rc = pm8xxx_readb(led->dev->parent, WLED_OVP_CFG_REG, &val);
585 if (rc) {
586 dev_err(led->dev->parent, "can't read wled ovp config"
587 " register rc=%d\n", rc);
588 return rc;
589 }
590
591 val = (val & ~WLED_OVP_VAL_MASK) |
592 (led->wled_cfg->ovp_val << WLED_OVP_VAL_BIT_SHFT);
593
594 rc = pm8xxx_writeb(led->dev->parent, WLED_OVP_CFG_REG, val);
595 if (rc) {
596 dev_err(led->dev->parent, "can't write wled ovp config"
597 " register rc=%d\n", rc);
598 return rc;
599 }
600
601 /* program current boost limit and output feedback*/
602 if (led->wled_cfg->boost_curr_lim > WLED_CURR_LIMIT_1680mA) {
603 dev_err(led->dev->parent, "Invalid boost current limit");
604 return -EINVAL;
605 }
606
607 rc = pm8xxx_readb(led->dev->parent, WLED_BOOST_CFG_REG, &val);
608 if (rc) {
609 dev_err(led->dev->parent, "can't read wled boost config"
610 " register rc=%d\n", rc);
611 return rc;
612 }
613
614 val = (val & ~WLED_BOOST_LIMIT_MASK) |
615 (led->wled_cfg->boost_curr_lim << WLED_BOOST_LIMIT_BIT_SHFT);
616
617 val = (val & ~WLED_OP_FDBCK_MASK) |
618 (led->wled_cfg->op_fdbck << WLED_OP_FDBCK_BIT_SHFT);
619
620 rc = pm8xxx_writeb(led->dev->parent, WLED_BOOST_CFG_REG, val);
621 if (rc) {
622 dev_err(led->dev->parent, "can't write wled boost config"
623 " register rc=%d\n", rc);
624 return rc;
625 }
626
627 /* program high pole capacitance */
628 if (led->wled_cfg->cp_select > WLED_CP_SELECT_MAX) {
629 dev_err(led->dev->parent, "Invalid pole capacitance");
630 return -EINVAL;
631 }
632
633 rc = pm8xxx_readb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, &val);
634 if (rc) {
635 dev_err(led->dev->parent, "can't read wled high pole"
636 " capacitance register rc=%d\n", rc);
637 return rc;
638 }
639
640 val = (val & ~WLED_CP_SELECT_MASK) | led->wled_cfg->cp_select;
641
642 rc = pm8xxx_writeb(led->dev->parent, WLED_HIGH_POLE_CAP_REG, val);
643 if (rc) {
644 dev_err(led->dev->parent, "can't write wled high pole"
645 " capacitance register rc=%d\n", rc);
646 return rc;
647 }
648
649 /* program activation delay and maximum current */
Amy Malochee7d7ef32013-01-30 16:16:35 -0800650 for (i = 0; i < WLED_STRINGS; i++) {
651 if (led->wled_cfg->strings && (1 << i)) {
652 rc = pm8xxx_readb(led->dev->parent,
Amy Maloche59cf8182012-11-30 12:27:40 -0800653 WLED_MAX_CURR_CFG_REG(i), &val);
Amy Malochee7d7ef32013-01-30 16:16:35 -0800654 if (rc) {
655 dev_err(led->dev->parent,
656 "can't read wled max current"
657 " config register rc=%d\n", rc);
658 return rc;
659 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530660
Amy Malochee7d7ef32013-01-30 16:16:35 -0800661 if ((led->wled_cfg->ctrl_delay_us % WLED_CTL_DLY_STEP)
662 || (led->wled_cfg->ctrl_delay_us >
663 WLED_CTL_DLY_MAX)) {
664 dev_err(led->dev->parent,
665 "Invalid control delay\n");
666 return rc;
667 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530668
Amy Malochee7d7ef32013-01-30 16:16:35 -0800669 val = val / WLED_CTL_DLY_STEP;
670 val = (val & ~WLED_CTL_DLY_MASK) |
671 (led->wled_cfg->ctrl_delay_us <<
672 WLED_CTL_DLY_BIT_SHFT);
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530673
Amy Malochee7d7ef32013-01-30 16:16:35 -0800674 if ((led->max_current > WLED_MAX_CURR)) {
675 dev_err(led->dev->parent,
676 "Invalid max current\n");
677 return -EINVAL;
678 }
Amy Maloche59cf8182012-11-30 12:27:40 -0800679 if (led->wled_cfg->max_current_ind) {
680 switch (i) {
681 case WLED_STRING_ONE:
682 string_max_current = led->wled_cfg->max_one;
683 break;
684 case WLED_STRING_TWO:
685 string_max_current = led->wled_cfg->max_two;
686 break;
687 case WLED_STRING_THREE:
688 string_max_current = led->wled_cfg->max_three;
689 break;
690 default:
691 return -EINVAL;
692 }
693 val = (val & ~WLED_MAX_CURR_MASK) | string_max_current;
694 } else
Amy Malochee7d7ef32013-01-30 16:16:35 -0800695 val = (val & ~WLED_MAX_CURR_MASK) | led->max_current;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530696
Amy Malochee7d7ef32013-01-30 16:16:35 -0800697 rc = pm8xxx_writeb(led->dev->parent,
Amy Maloche59cf8182012-11-30 12:27:40 -0800698 WLED_MAX_CURR_CFG_REG(i), val);
Amy Malochee7d7ef32013-01-30 16:16:35 -0800699 if (rc) {
700 dev_err(led->dev->parent,
701 "can't write wled max current"
702 " config register rc=%d\n", rc);
703 return rc;
704 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530705 }
706 }
707
Amy Malochec366d5b2012-08-28 11:06:40 -0700708 if (led->wled_cfg->cabc_en) {
709 rc = pm8xxx_readb(led->dev->parent, WLED_SYNC_REG, &val);
710 if (rc) {
711 dev_err(led->dev->parent,
712 "can't read cabc register rc=%d\n", rc);
713 return rc;
714 }
715
Amy Malochee7d7ef32013-01-30 16:16:35 -0800716 val |= (led->wled_cfg->strings << WLED_CABC_SHIFT);
Amy Malochec366d5b2012-08-28 11:06:40 -0700717
718 rc = pm8xxx_writeb(led->dev->parent, WLED_SYNC_REG, val);
719 if (rc) {
720 dev_err(led->dev->parent,
721 "can't write to enable cabc rc=%d\n", rc);
722 return rc;
723 }
724 }
725
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530726 /* program digital module generator, cs out and enable the module */
727 rc = pm8xxx_readb(led->dev->parent, WLED_MOD_CTRL_REG, &val);
728 if (rc) {
729 dev_err(led->dev->parent, "can't read wled module ctrl"
730 " register rc=%d\n", rc);
731 return rc;
732 }
733
734 if (led->wled_cfg->dig_mod_gen_en)
735 val |= WLED_DIG_MOD_GEN_MASK;
736
737 if (led->wled_cfg->cs_out_en)
738 val |= WLED_CS_OUT_MASK;
739
740 val |= WLED_EN_MASK;
741
742 rc = pm8xxx_writeb(led->dev->parent, WLED_MOD_CTRL_REG, val);
743 if (rc) {
744 dev_err(led->dev->parent, "can't write wled module ctrl"
745 " register rc=%d\n", rc);
746 return rc;
747 }
Amy Maloche56913f52012-05-11 10:47:24 -0700748 led->wled_mod_ctrl_val = val;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530749
750 /* dump wled registers */
751 wled_dump_regs(led);
752
753 return 0;
754}
755
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700756static int __devinit get_init_value(struct pm8xxx_led_data *led, u8 *val)
757{
758 int rc, offset;
759 u16 addr;
760
761 switch (led->id) {
762 case PM8XXX_ID_LED_KB_LIGHT:
763 addr = SSBI_REG_ADDR_DRV_KEYPAD;
764 break;
765 case PM8XXX_ID_LED_0:
766 case PM8XXX_ID_LED_1:
767 case PM8XXX_ID_LED_2:
768 offset = PM8XXX_LED_OFFSET(led->id);
769 addr = SSBI_REG_ADDR_LED_CTRL(offset);
770 break;
771 case PM8XXX_ID_FLASH_LED_0:
772 addr = SSBI_REG_ADDR_FLASH_DRV0;
773 break;
774 case PM8XXX_ID_FLASH_LED_1:
775 addr = SSBI_REG_ADDR_FLASH_DRV1;
776 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530777 case PM8XXX_ID_WLED:
778 rc = init_wled(led);
779 if (rc)
780 dev_err(led->cdev.dev, "can't initialize wled rc=%d\n",
781 rc);
782 return rc;
Amy Malochec17c3732012-02-27 18:34:07 -0800783 case PM8XXX_ID_RGB_LED_RED:
784 case PM8XXX_ID_RGB_LED_GREEN:
785 case PM8XXX_ID_RGB_LED_BLUE:
Amy Malochec17c3732012-02-27 18:34:07 -0800786 addr = SSBI_REG_ADDR_RGB_CNTL1;
787 break;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530788 default:
789 dev_err(led->cdev.dev, "unknown led id %d", led->id);
790 return -EINVAL;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700791 }
792
793 rc = pm8xxx_readb(led->dev->parent, addr, val);
794 if (rc)
795 dev_err(led->cdev.dev, "can't get led(%d) level rc=%d\n",
796 led->id, rc);
797
798 return rc;
799}
800
Jay Chokshi868312e2011-09-16 13:57:13 -0700801static int pm8xxx_led_pwm_configure(struct pm8xxx_led_data *led)
802{
Prasad Sodagudi5af87922013-06-05 17:33:52 +0530803 int start_idx, idx_len, duty_us, rc, flags;
Jay Chokshi868312e2011-09-16 13:57:13 -0700804
805 led->pwm_dev = pwm_request(led->pwm_channel,
806 led->cdev.name);
807
808 if (IS_ERR_OR_NULL(led->pwm_dev)) {
809 pr_err("could not acquire PWM Channel %d, "
810 "error %ld\n", led->pwm_channel,
811 PTR_ERR(led->pwm_dev));
812 led->pwm_dev = NULL;
813 return -ENODEV;
814 }
815
Prasad Sodagudi5af87922013-06-05 17:33:52 +0530816 flags = PM8XXX_LED_PWM_FLAGS;
817 switch (led->max_current) {
818 case PM8XXX_PWM_CURRENT_4MA:
819 flags |= PM_PWM_BANK_LO;
820 break;
821 case PM8XXX_PWM_CURRENT_8MA:
822 flags |= PM_PWM_BANK_HI;
823 break;
824 case PM8XXX_PWM_CURRENT_12MA:
825 flags |= (PM_PWM_BANK_LO | PM_PWM_BANK_HI);
826 break;
827 default:
828 flags |= (PM_PWM_BANK_LO | PM_PWM_BANK_HI);
829 break;
830 }
831
Jay Chokshi868312e2011-09-16 13:57:13 -0700832 if (led->pwm_duty_cycles != NULL) {
833 start_idx = led->pwm_duty_cycles->start_idx;
834 idx_len = led->pwm_duty_cycles->num_duty_pcts;
835
836 if (idx_len >= PM_PWM_LUT_SIZE && start_idx) {
837 pr_err("Wrong LUT size or index\n");
838 return -EINVAL;
839 }
840 if ((start_idx + idx_len) > PM_PWM_LUT_SIZE) {
841 pr_err("Exceed LUT limit\n");
842 return -EINVAL;
843 }
844
845 rc = pm8xxx_pwm_lut_config(led->pwm_dev, led->pwm_period_us,
846 led->pwm_duty_cycles->duty_pcts,
847 led->pwm_duty_cycles->duty_ms,
848 start_idx, idx_len, 0, 0,
Prasad Sodagudi5af87922013-06-05 17:33:52 +0530849 flags);
Jay Chokshi868312e2011-09-16 13:57:13 -0700850 } else {
851 duty_us = led->pwm_period_us;
852 rc = pwm_config(led->pwm_dev, duty_us, led->pwm_period_us);
853 }
854
855 return rc;
856}
857
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530858
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700859static int __devinit pm8xxx_led_probe(struct platform_device *pdev)
860{
Jay Chokshi8994e392011-09-14 18:20:39 -0700861 const struct pm8xxx_led_platform_data *pdata = pdev->dev.platform_data;
862 const struct led_platform_data *pcore_data;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700863 struct led_info *curr_led;
864 struct pm8xxx_led_data *led, *led_dat;
Jay Chokshi8994e392011-09-14 18:20:39 -0700865 struct pm8xxx_led_config *led_cfg;
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700866 enum pm8xxx_version version;
867 bool found = false;
868 int rc, i, j;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700869
870 if (pdata == NULL) {
871 dev_err(&pdev->dev, "platform data not supplied\n");
872 return -EINVAL;
873 }
874
Jay Chokshi8994e392011-09-14 18:20:39 -0700875 pcore_data = pdata->led_core;
876
877 if (pcore_data->num_leds != pdata->num_configs) {
878 dev_err(&pdev->dev, "#no. of led configs and #no. of led"
879 "entries are not equal\n");
880 return -EINVAL;
881 }
882
883 led = kcalloc(pcore_data->num_leds, sizeof(*led), GFP_KERNEL);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700884 if (led == NULL) {
885 dev_err(&pdev->dev, "failed to alloc memory\n");
886 return -ENOMEM;
887 }
888
Jay Chokshi8994e392011-09-14 18:20:39 -0700889 for (i = 0; i < pcore_data->num_leds; i++) {
890 curr_led = &pcore_data->leds[i];
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700891 led_dat = &led[i];
Jay Chokshi8994e392011-09-14 18:20:39 -0700892 led_cfg = &pdata->configs[i];
893
894 led_dat->id = led_cfg->id;
Jay Chokshi868312e2011-09-16 13:57:13 -0700895 led_dat->pwm_channel = led_cfg->pwm_channel;
896 led_dat->pwm_period_us = led_cfg->pwm_period_us;
897 led_dat->pwm_duty_cycles = led_cfg->pwm_duty_cycles;
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530898 led_dat->wled_cfg = led_cfg->wled_cfg;
899 led_dat->max_current = led_cfg->max_current;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700900
901 if (!((led_dat->id >= PM8XXX_ID_LED_KB_LIGHT) &&
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530902 (led_dat->id < PM8XXX_ID_MAX))) {
Amy Maloche3d2c24c2012-03-22 19:02:25 -0700903 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
904 led_dat->id);
905 rc = -EINVAL;
906 goto fail_id_check;
907
908 }
909
910 found = false;
911 version = pm8xxx_get_version(pdev->dev.parent);
912 for (j = 0; j < ARRAY_SIZE(led_map); j++) {
913 if (version == led_map[j].version
914 && (led_map[j].supported & (1 << led_dat->id))) {
915 found = true;
916 break;
917 }
918 }
919
920 if (!found) {
921 dev_err(&pdev->dev, "invalid LED ID(%d) specified\n",
922 led_dat->id);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700923 rc = -EINVAL;
924 goto fail_id_check;
925 }
926
927 led_dat->cdev.name = curr_led->name;
928 led_dat->cdev.default_trigger = curr_led->default_trigger;
929 led_dat->cdev.brightness_set = pm8xxx_led_set;
930 led_dat->cdev.brightness_get = pm8xxx_led_get;
931 led_dat->cdev.brightness = LED_OFF;
Jay Chokshi8994e392011-09-14 18:20:39 -0700932 led_dat->cdev.flags = curr_led->flags;
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700933 led_dat->dev = &pdev->dev;
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700934
935 rc = get_init_value(led_dat, &led_dat->reg);
936 if (rc < 0)
937 goto fail_id_check;
938
Jay Chokshi8994e392011-09-14 18:20:39 -0700939 rc = pm8xxx_set_led_mode_and_max_brightness(led_dat,
940 led_cfg->mode, led_cfg->max_current);
Jay Chokshide4cefb2011-08-04 18:10:44 -0700941 if (rc < 0)
942 goto fail_id_check;
943
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700944 mutex_init(&led_dat->lock);
945 INIT_WORK(&led_dat->work, pm8xxx_led_work);
946
947 rc = led_classdev_register(&pdev->dev, &led_dat->cdev);
948 if (rc) {
949 dev_err(&pdev->dev, "unable to register led %d,rc=%d\n",
950 led_dat->id, rc);
951 goto fail_id_check;
952 }
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700953
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530954 /* configure default state */
955 if (led_cfg->default_state)
956 led->cdev.brightness = led_dat->cdev.max_brightness;
957 else
958 led->cdev.brightness = LED_OFF;
959
Jay Chokshi868312e2011-09-16 13:57:13 -0700960 if (led_cfg->mode != PM8XXX_LED_MODE_MANUAL) {
Amy Malochec17c3732012-02-27 18:34:07 -0800961 if (led_dat->id == PM8XXX_ID_RGB_LED_RED ||
962 led_dat->id == PM8XXX_ID_RGB_LED_GREEN ||
963 led_dat->id == PM8XXX_ID_RGB_LED_BLUE)
964 __pm8xxx_led_work(led_dat, 0);
965 else
966 __pm8xxx_led_work(led_dat,
Jay Chokshide4cefb2011-08-04 18:10:44 -0700967 led_dat->cdev.max_brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700968
969 if (led_dat->pwm_channel != -1) {
970 led_dat->cdev.max_brightness = LED_FULL;
971 rc = pm8xxx_led_pwm_configure(led_dat);
972 if (rc) {
973 dev_err(&pdev->dev, "failed to "
974 "configure LED, error: %d\n", rc);
975 goto fail_id_check;
976 }
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530977 schedule_work(&led->work);
Jay Chokshi868312e2011-09-16 13:57:13 -0700978 }
979 } else {
Mohan Pallaka53fe1a12011-12-15 16:56:37 +0530980 __pm8xxx_led_work(led_dat, led->cdev.brightness);
Jay Chokshi868312e2011-09-16 13:57:13 -0700981 }
Jay Chokshide4cefb2011-08-04 18:10:44 -0700982 }
Jay Chokshi12e49bf2011-07-22 16:24:39 -0700983
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700984 platform_set_drvdata(pdev, led);
985
986 return 0;
987
988fail_id_check:
989 if (i > 0) {
990 for (i = i - 1; i >= 0; i--) {
991 mutex_destroy(&led[i].lock);
992 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -0700993 if (led[i].pwm_dev != NULL)
994 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -0700995 }
996 }
997 kfree(led);
998 return rc;
999}
1000
1001static int __devexit pm8xxx_led_remove(struct platform_device *pdev)
1002{
1003 int i;
1004 const struct led_platform_data *pdata =
1005 pdev->dev.platform_data;
1006 struct pm8xxx_led_data *led = platform_get_drvdata(pdev);
1007
1008 for (i = 0; i < pdata->num_leds; i++) {
1009 cancel_work_sync(&led[i].work);
1010 mutex_destroy(&led[i].lock);
1011 led_classdev_unregister(&led[i].cdev);
Jay Chokshi868312e2011-09-16 13:57:13 -07001012 if (led[i].pwm_dev != NULL)
1013 pwm_free(led[i].pwm_dev);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001014 }
1015
1016 kfree(led);
1017
1018 return 0;
1019}
1020
1021static struct platform_driver pm8xxx_led_driver = {
1022 .probe = pm8xxx_led_probe,
1023 .remove = __devexit_p(pm8xxx_led_remove),
1024 .driver = {
1025 .name = PM8XXX_LEDS_DEV_NAME,
1026 .owner = THIS_MODULE,
1027 },
1028};
1029
1030static int __init pm8xxx_led_init(void)
1031{
1032 return platform_driver_register(&pm8xxx_led_driver);
1033}
Jay Chokshi12e49bf2011-07-22 16:24:39 -07001034subsys_initcall(pm8xxx_led_init);
Bryan Huntsman3f2bc4d2011-08-16 17:27:22 -07001035
1036static void __exit pm8xxx_led_exit(void)
1037{
1038 platform_driver_unregister(&pm8xxx_led_driver);
1039}
1040module_exit(pm8xxx_led_exit);
1041
1042MODULE_DESCRIPTION("PM8XXX LEDs driver");
1043MODULE_LICENSE("GPL v2");
1044MODULE_VERSION("1.0");
1045MODULE_ALIAS("platform:pm8xxx-led");