2 * Simple driver for Texas Instruments LM355x LED Flash driver chip
3 * Copyright (C) 2012 Texas Instruments
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 as
7 * published by the Free Software Foundation.
10 #include <linux/module.h>
11 #include <linux/delay.h>
12 #include <linux/i2c.h>
13 #include <linux/gpio.h>
14 #include <linux/leds.h>
15 #include <linux/slab.h>
16 #include <linux/platform_device.h>
18 #include <linux/regmap.h>
19 #include <linux/workqueue.h>
20 #include <linux/platform_data/leds-lm355x.h>
47 /* register map info. */
48 struct lm355x_reg_data
{
54 struct lm355x_chip_data
{
56 enum lm355x_type type
;
58 struct led_classdev cdev_flash
;
59 struct led_classdev cdev_torch
;
60 struct led_classdev cdev_indicator
;
62 struct work_struct work_flash
;
63 struct work_struct work_torch
;
64 struct work_struct work_indicator
;
70 struct lm355x_platform_data
*pdata
;
71 struct regmap
*regmap
;
74 unsigned int last_flag
;
75 struct lm355x_reg_data
*regs
;
78 /* specific indicator function for lm3556 */
79 enum lm3556_indic_pulse_time
{
98 enum lm3556_indic_n_blank
{
117 enum lm3556_indic_period
{
128 #define INDIC_PATTERN_SIZE 4
135 /* indicator pattern data only for lm3556 */
136 static struct indicator indicator_pattern
[INDIC_PATTERN_SIZE
] = {
137 [0] = {(INDIC_N_BLANK_1
<< 4) | PULSE_TIME_32_MS
, INDIC_PERIOD_1
},
138 [1] = {(INDIC_N_BLANK_15
<< 4) | PULSE_TIME_32_MS
, INDIC_PERIOD_2
},
139 [2] = {(INDIC_N_BLANK_10
<< 4) | PULSE_TIME_32_MS
, INDIC_PERIOD_4
},
140 [3] = {(INDIC_N_BLANK_5
<< 4) | PULSE_TIME_32_MS
, INDIC_PERIOD_7
},
143 static struct lm355x_reg_data lm3554_regs
[REG_MAX
] = {
144 [REG_FLAG
] = {0xD0, 0xBF, 0},
145 [REG_TORCH_CFG
] = {0xE0, 0x80, 7},
146 [REG_TORCH_CTRL
] = {0xA0, 0x38, 3},
147 [REG_STROBE_CFG
] = {0xE0, 0x04, 2},
148 [REG_FLASH_CTRL
] = {0xB0, 0x78, 3},
149 [REG_INDI_CFG
] = {0xE0, 0x08, 3},
150 [REG_INDI_CTRL
] = {0xA0, 0xC0, 6},
151 [REG_OPMODE
] = {0xA0, 0x03, 0},
154 static struct lm355x_reg_data lm3556_regs
[REG_MAX
] = {
155 [REG_FLAG
] = {0x0B, 0xFF, 0},
156 [REG_TORCH_CFG
] = {0x0A, 0x10, 4},
157 [REG_TORCH_CTRL
] = {0x09, 0x70, 4},
158 [REG_STROBE_CFG
] = {0x0A, 0x20, 5},
159 [REG_FLASH_CTRL
] = {0x09, 0x0F, 0},
160 [REG_INDI_CFG
] = {0xFF, 0xFF, 0},
161 [REG_INDI_CTRL
] = {0x09, 0x70, 4},
162 [REG_OPMODE
] = {0x0A, 0x03, 0},
165 static char lm355x_name
[][I2C_NAME_SIZE
] = {
166 [CHIP_LM3554
] = LM3554_NAME
,
167 [CHIP_LM3556
] = LM3556_NAME
,
170 /* chip initialize */
171 static int lm355x_chip_init(struct lm355x_chip_data
*chip
)
174 unsigned int reg_val
;
175 struct lm355x_platform_data
*pdata
= chip
->pdata
;
177 /* input and output pins configuration */
178 switch (chip
->type
) {
180 reg_val
= pdata
->pin_tx2
| pdata
->ntc_pin
;
181 ret
= regmap_update_bits(chip
->regmap
, 0xE0, 0x28, reg_val
);
184 reg_val
= pdata
->pass_mode
;
185 ret
= regmap_update_bits(chip
->regmap
, 0xA0, 0x04, reg_val
);
191 reg_val
= pdata
->pin_tx2
| pdata
->ntc_pin
| pdata
->pass_mode
;
192 ret
= regmap_update_bits(chip
->regmap
, 0x0A, 0xC4, reg_val
);
202 dev_err(chip
->dev
, "%s:i2c access fail to register\n", __func__
);
207 static void lm355x_control(struct lm355x_chip_data
*chip
,
208 u8 brightness
, enum lm355x_mode opmode
)
211 unsigned int reg_val
;
212 struct lm355x_platform_data
*pdata
= chip
->pdata
;
213 struct lm355x_reg_data
*preg
= chip
->regs
;
215 ret
= regmap_read(chip
->regmap
, preg
[REG_FLAG
].regno
, &chip
->last_flag
);
218 if (chip
->last_flag
& preg
[REG_FLAG
].mask
)
219 dev_info(chip
->dev
, "%s Last FLAG is 0x%x\n",
220 lm355x_name
[chip
->type
],
221 chip
->last_flag
& preg
[REG_FLAG
].mask
);
222 /* brightness 0 means shutdown */
229 regmap_update_bits(chip
->regmap
, preg
[REG_TORCH_CTRL
].regno
,
230 preg
[REG_TORCH_CTRL
].mask
,
232 << preg
[REG_TORCH_CTRL
].shift
);
236 if (pdata
->pin_tx1
!= LM355x_PIN_TORCH_DISABLE
) {
238 regmap_update_bits(chip
->regmap
,
239 preg
[REG_TORCH_CFG
].regno
,
240 preg
[REG_TORCH_CFG
].mask
,
242 preg
[REG_TORCH_CFG
].shift
);
247 "torch brt is set - ext. torch pin mode\n");
254 regmap_update_bits(chip
->regmap
, preg
[REG_FLASH_CTRL
].regno
,
255 preg
[REG_FLASH_CTRL
].mask
,
257 << preg
[REG_FLASH_CTRL
].shift
);
261 if (pdata
->pin_strobe
!= LM355x_PIN_STROBE_DISABLE
) {
262 if (chip
->type
== CHIP_LM3554
)
267 regmap_update_bits(chip
->regmap
,
268 preg
[REG_STROBE_CFG
].regno
,
269 preg
[REG_STROBE_CFG
].mask
,
271 preg
[REG_STROBE_CFG
].shift
);
276 "flash brt is set - ext. strobe pin mode\n");
282 regmap_update_bits(chip
->regmap
, preg
[REG_INDI_CTRL
].regno
,
283 preg
[REG_INDI_CTRL
].mask
,
285 << preg
[REG_INDI_CTRL
].shift
);
289 if (pdata
->pin_tx2
!= LM355x_PIN_TX_DISABLE
) {
291 regmap_update_bits(chip
->regmap
,
292 preg
[REG_INDI_CFG
].regno
,
293 preg
[REG_INDI_CFG
].mask
,
295 preg
[REG_INDI_CFG
].shift
);
306 /* operation mode control */
307 ret
= regmap_update_bits(chip
->regmap
, preg
[REG_OPMODE
].regno
,
308 preg
[REG_OPMODE
].mask
,
309 opmode
<< preg
[REG_OPMODE
].shift
);
314 dev_err(chip
->dev
, "%s:i2c access fail to register\n", __func__
);
319 static void lm355x_deferred_torch_brightness_set(struct work_struct
*work
)
321 struct lm355x_chip_data
*chip
=
322 container_of(work
, struct lm355x_chip_data
, work_torch
);
324 mutex_lock(&chip
->lock
);
325 lm355x_control(chip
, chip
->br_torch
, MODE_TORCH
);
326 mutex_unlock(&chip
->lock
);
329 static void lm355x_torch_brightness_set(struct led_classdev
*cdev
,
330 enum led_brightness brightness
)
332 struct lm355x_chip_data
*chip
=
333 container_of(cdev
, struct lm355x_chip_data
, cdev_torch
);
335 chip
->br_torch
= brightness
;
336 schedule_work(&chip
->work_torch
);
340 static void lm355x_deferred_strobe_brightness_set(struct work_struct
*work
)
342 struct lm355x_chip_data
*chip
=
343 container_of(work
, struct lm355x_chip_data
, work_flash
);
345 mutex_lock(&chip
->lock
);
346 lm355x_control(chip
, chip
->br_flash
, MODE_FLASH
);
347 mutex_unlock(&chip
->lock
);
350 static void lm355x_strobe_brightness_set(struct led_classdev
*cdev
,
351 enum led_brightness brightness
)
353 struct lm355x_chip_data
*chip
=
354 container_of(cdev
, struct lm355x_chip_data
, cdev_flash
);
356 chip
->br_flash
= brightness
;
357 schedule_work(&chip
->work_flash
);
361 static void lm355x_deferred_indicator_brightness_set(struct work_struct
*work
)
363 struct lm355x_chip_data
*chip
=
364 container_of(work
, struct lm355x_chip_data
, work_indicator
);
366 mutex_lock(&chip
->lock
);
367 lm355x_control(chip
, chip
->br_indicator
, MODE_INDIC
);
368 mutex_unlock(&chip
->lock
);
371 static void lm355x_indicator_brightness_set(struct led_classdev
*cdev
,
372 enum led_brightness brightness
)
374 struct lm355x_chip_data
*chip
=
375 container_of(cdev
, struct lm355x_chip_data
, cdev_indicator
);
377 chip
->br_indicator
= brightness
;
378 schedule_work(&chip
->work_indicator
);
381 /* indicator pattern only for lm3556*/
382 static ssize_t
lm3556_indicator_pattern_store(struct device
*dev
,
383 struct device_attribute
*attr
,
384 const char *buf
, size_t size
)
387 struct led_classdev
*led_cdev
= dev_get_drvdata(dev
);
388 struct lm355x_chip_data
*chip
=
389 container_of(led_cdev
, struct lm355x_chip_data
, cdev_indicator
);
392 ret
= kstrtouint(buf
, 10, &state
);
395 if (state
> INDIC_PATTERN_SIZE
- 1)
396 state
= INDIC_PATTERN_SIZE
- 1;
398 ret
= regmap_write(chip
->regmap
, 0x04,
399 indicator_pattern
[state
].blinking
);
403 ret
= regmap_write(chip
->regmap
, 0x05,
404 indicator_pattern
[state
].period_cnt
);
410 dev_err(chip
->dev
, "%s:i2c access fail to register\n", __func__
);
414 static DEVICE_ATTR(pattern
, S_IWUSR
, NULL
, lm3556_indicator_pattern_store
);
416 static const struct regmap_config lm355x_regmap
= {
419 .max_register
= 0xFF,
422 /* module initialize */
423 static int lm355x_probe(struct i2c_client
*client
,
424 const struct i2c_device_id
*id
)
426 struct lm355x_platform_data
*pdata
= dev_get_platdata(&client
->dev
);
427 struct lm355x_chip_data
*chip
;
431 if (!i2c_check_functionality(client
->adapter
, I2C_FUNC_I2C
)) {
432 dev_err(&client
->dev
, "i2c functionality check fail.\n");
437 dev_err(&client
->dev
, "needs Platform Data.\n");
441 chip
= devm_kzalloc(&client
->dev
,
442 sizeof(struct lm355x_chip_data
), GFP_KERNEL
);
446 chip
->dev
= &client
->dev
;
447 chip
->type
= id
->driver_data
;
448 switch (id
->driver_data
) {
450 chip
->regs
= lm3554_regs
;
453 chip
->regs
= lm3556_regs
;
460 chip
->regmap
= devm_regmap_init_i2c(client
, &lm355x_regmap
);
461 if (IS_ERR(chip
->regmap
)) {
462 err
= PTR_ERR(chip
->regmap
);
463 dev_err(&client
->dev
,
464 "Failed to allocate register map: %d\n", err
);
468 mutex_init(&chip
->lock
);
469 i2c_set_clientdata(client
, chip
);
471 err
= lm355x_chip_init(chip
);
476 INIT_WORK(&chip
->work_flash
, lm355x_deferred_strobe_brightness_set
);
477 chip
->cdev_flash
.name
= "flash";
478 chip
->cdev_flash
.max_brightness
= 16;
479 chip
->cdev_flash
.brightness_set
= lm355x_strobe_brightness_set
;
480 chip
->cdev_flash
.default_trigger
= "flash";
481 err
= led_classdev_register((struct device
*)
482 &client
->dev
, &chip
->cdev_flash
);
486 INIT_WORK(&chip
->work_torch
, lm355x_deferred_torch_brightness_set
);
487 chip
->cdev_torch
.name
= "torch";
488 chip
->cdev_torch
.max_brightness
= 8;
489 chip
->cdev_torch
.brightness_set
= lm355x_torch_brightness_set
;
490 chip
->cdev_torch
.default_trigger
= "torch";
491 err
= led_classdev_register((struct device
*)
492 &client
->dev
, &chip
->cdev_torch
);
494 goto err_create_torch_file
;
496 INIT_WORK(&chip
->work_indicator
,
497 lm355x_deferred_indicator_brightness_set
);
498 chip
->cdev_indicator
.name
= "indicator";
499 if (id
->driver_data
== CHIP_LM3554
)
500 chip
->cdev_indicator
.max_brightness
= 4;
502 chip
->cdev_indicator
.max_brightness
= 8;
503 chip
->cdev_indicator
.brightness_set
= lm355x_indicator_brightness_set
;
504 err
= led_classdev_register((struct device
*)
505 &client
->dev
, &chip
->cdev_indicator
);
507 goto err_create_indicator_file
;
508 /* indicator pattern control only for LM3554 */
509 if (id
->driver_data
== CHIP_LM3556
) {
511 device_create_file(chip
->cdev_indicator
.dev
,
514 goto err_create_pattern_file
;
517 dev_info(&client
->dev
, "%s is initialized\n",
518 lm355x_name
[id
->driver_data
]);
521 err_create_pattern_file
:
522 led_classdev_unregister(&chip
->cdev_indicator
);
523 err_create_indicator_file
:
524 led_classdev_unregister(&chip
->cdev_torch
);
525 err_create_torch_file
:
526 led_classdev_unregister(&chip
->cdev_flash
);
531 static int lm355x_remove(struct i2c_client
*client
)
533 struct lm355x_chip_data
*chip
= i2c_get_clientdata(client
);
534 struct lm355x_reg_data
*preg
= chip
->regs
;
536 regmap_write(chip
->regmap
, preg
[REG_OPMODE
].regno
, 0);
537 if (chip
->type
== CHIP_LM3556
)
538 device_remove_file(chip
->cdev_indicator
.dev
, &dev_attr_pattern
);
539 led_classdev_unregister(&chip
->cdev_indicator
);
540 flush_work(&chip
->work_indicator
);
541 led_classdev_unregister(&chip
->cdev_torch
);
542 flush_work(&chip
->work_torch
);
543 led_classdev_unregister(&chip
->cdev_flash
);
544 flush_work(&chip
->work_flash
);
545 dev_info(&client
->dev
, "%s is removed\n", lm355x_name
[chip
->type
]);
550 static const struct i2c_device_id lm355x_id
[] = {
551 {LM3554_NAME
, CHIP_LM3554
},
552 {LM3556_NAME
, CHIP_LM3556
},
556 MODULE_DEVICE_TABLE(i2c
, lm355x_id
);
558 static struct i2c_driver lm355x_i2c_driver
= {
561 .owner
= THIS_MODULE
,
564 .probe
= lm355x_probe
,
565 .remove
= lm355x_remove
,
566 .id_table
= lm355x_id
,
569 module_i2c_driver(lm355x_i2c_driver
);
571 MODULE_DESCRIPTION("Texas Instruments Flash Lighting driver for LM355x");
572 MODULE_AUTHOR("Daniel Jeong <daniel.jeong@ti.com>");
573 MODULE_AUTHOR("G.Shark Jeong <gshark.jeong@gmail.com>");
574 MODULE_LICENSE("GPL v2");