aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMilo Kim <milo.kim@ti.com>2013-08-08 01:02:29 -0400
committerBryan Wu <cooloney@gmail.com>2013-08-26 20:22:13 -0400
commit45e611bfbe18d854498c65d60703841e4a4c3c3a (patch)
tree1e7b6ad98d93d7bb1341d914433a0c098d093eca
parent224604389a5295360ef36b50ca5a51806a235fcf (diff)
leds: lp5523: restore legacy device attributes
git commit db6eaf8388a413a5ee1b4547ce78506b9c6456b0 (leds-lp5523: use generic firmware interface) causes an application conflict. This interface should be maintained for compatibility. Restored device attributes are 'engineN_mode', 'engineN_load' and 'engineN_leds'. (N = 1, 2 or 3) A 'selftest' attribute macro is replaced with LP55xx common macro. Those are accessed when a LED pattern is run by an application. Use a mutex in lp5523_update_program_memory() : This function is called when an user-application writes a 'engineN_load' file or pattern data is loaded from generic firmware interface. So, writing program memory should be protected. If an error occurs on accessing this area, just it returns as -EINVAL quickly. This error code is exact same as old driver function, lp5523_do_store_load() because it should be kept for an user-application compatibility. Even the driver is changed, we can use the application without re-compiling sources. Reported-by: Pali Rohár <pali.rohar@gmail.com> Signed-off-by: Milo Kim <milo.kim@ti.com> Signed-off-by: Bryan Wu <cooloney@gmail.com>
-rw-r--r--drivers/leds/leds-lp5523.c227
1 files changed, 223 insertions, 4 deletions
diff --git a/drivers/leds/leds-lp5523.c b/drivers/leds/leds-lp5523.c
index 29c8b19a445e..9b8be6f6c595 100644
--- a/drivers/leds/leds-lp5523.c
+++ b/drivers/leds/leds-lp5523.c
@@ -74,6 +74,9 @@
74#define LP5523_PAGE_ENG1 0 74#define LP5523_PAGE_ENG1 0
75#define LP5523_PAGE_ENG2 1 75#define LP5523_PAGE_ENG2 1
76#define LP5523_PAGE_ENG3 2 76#define LP5523_PAGE_ENG3 2
77#define LP5523_PAGE_MUX1 3
78#define LP5523_PAGE_MUX2 4
79#define LP5523_PAGE_MUX3 5
77 80
78/* Program Memory Operations */ 81/* Program Memory Operations */
79#define LP5523_MODE_ENG1_M 0x30 /* Operation Mode Register */ 82#define LP5523_MODE_ENG1_M 0x30 /* Operation Mode Register */
@@ -98,6 +101,8 @@
98#define LP5523_RUN_ENG2 0x08 101#define LP5523_RUN_ENG2 0x08
99#define LP5523_RUN_ENG3 0x02 102#define LP5523_RUN_ENG3 0x02
100 103
104#define LED_ACTIVE(mux, led) (!!(mux & (0x0001 << led)))
105
101enum lp5523_chip_id { 106enum lp5523_chip_id {
102 LP5523, 107 LP5523,
103 LP55231, 108 LP55231,
@@ -338,10 +343,20 @@ static int lp5523_update_program_memory(struct lp55xx_chip *chip,
338 goto err; 343 goto err;
339 344
340 update_size = i; 345 update_size = i;
341 for (i = 0; i < update_size; i++)
342 lp55xx_write(chip, LP5523_REG_PROG_MEM + i, pattern[i]);
343 346
344 return 0; 347 mutex_lock(&chip->lock);
348
349 for (i = 0; i < update_size; i++) {
350 ret = lp55xx_write(chip, LP5523_REG_PROG_MEM + i, pattern[i]);
351 if (ret) {
352 mutex_unlock(&chip->lock);
353 return -EINVAL;
354 }
355 }
356
357 mutex_unlock(&chip->lock);
358
359 return size;
345 360
346err: 361err:
347 dev_err(&chip->cl->dev, "wrong pattern format\n"); 362 dev_err(&chip->cl->dev, "wrong pattern format\n");
@@ -368,6 +383,192 @@ static void lp5523_firmware_loaded(struct lp55xx_chip *chip)
368 lp5523_update_program_memory(chip, fw->data, fw->size); 383 lp5523_update_program_memory(chip, fw->data, fw->size);
369} 384}
370 385
386static ssize_t show_engine_mode(struct device *dev,
387 struct device_attribute *attr,
388 char *buf, int nr)
389{
390 struct lp55xx_led *led = i2c_get_clientdata(to_i2c_client(dev));
391 struct lp55xx_chip *chip = led->chip;
392 enum lp55xx_engine_mode mode = chip->engines[nr - 1].mode;
393
394 switch (mode) {
395 case LP55XX_ENGINE_RUN:
396 return sprintf(buf, "run\n");
397 case LP55XX_ENGINE_LOAD:
398 return sprintf(buf, "load\n");
399 case LP55XX_ENGINE_DISABLED:
400 default:
401 return sprintf(buf, "disabled\n");
402 }
403}
404show_mode(1)
405show_mode(2)
406show_mode(3)
407
408static ssize_t store_engine_mode(struct device *dev,
409 struct device_attribute *attr,
410 const char *buf, size_t len, int nr)
411{
412 struct lp55xx_led *led = i2c_get_clientdata(to_i2c_client(dev));
413 struct lp55xx_chip *chip = led->chip;
414 struct lp55xx_engine *engine = &chip->engines[nr - 1];
415
416 mutex_lock(&chip->lock);
417
418 chip->engine_idx = nr;
419
420 if (!strncmp(buf, "run", 3)) {
421 lp5523_run_engine(chip, true);
422 engine->mode = LP55XX_ENGINE_RUN;
423 } else if (!strncmp(buf, "load", 4)) {
424 lp5523_stop_engine(chip);
425 lp5523_load_engine(chip);
426 engine->mode = LP55XX_ENGINE_LOAD;
427 } else if (!strncmp(buf, "disabled", 8)) {
428 lp5523_stop_engine(chip);
429 engine->mode = LP55XX_ENGINE_DISABLED;
430 }
431
432 mutex_unlock(&chip->lock);
433
434 return len;
435}
436store_mode(1)
437store_mode(2)
438store_mode(3)
439
440static int lp5523_mux_parse(const char *buf, u16 *mux, size_t len)
441{
442 u16 tmp_mux = 0;
443 int i;
444
445 len = min_t(int, len, LP5523_MAX_LEDS);
446
447 for (i = 0; i < len; i++) {
448 switch (buf[i]) {
449 case '1':
450 tmp_mux |= (1 << i);
451 break;
452 case '0':
453 break;
454 case '\n':
455 i = len;
456 break;
457 default:
458 return -1;
459 }
460 }
461 *mux = tmp_mux;
462
463 return 0;
464}
465
466static void lp5523_mux_to_array(u16 led_mux, char *array)
467{
468 int i, pos = 0;
469 for (i = 0; i < LP5523_MAX_LEDS; i++)
470 pos += sprintf(array + pos, "%x", LED_ACTIVE(led_mux, i));
471
472 array[pos] = '\0';
473}
474
475static ssize_t show_engine_leds(struct device *dev,
476 struct device_attribute *attr,
477 char *buf, int nr)
478{
479 struct lp55xx_led *led = i2c_get_clientdata(to_i2c_client(dev));
480 struct lp55xx_chip *chip = led->chip;
481 char mux[LP5523_MAX_LEDS + 1];
482
483 lp5523_mux_to_array(chip->engines[nr - 1].led_mux, mux);
484
485 return sprintf(buf, "%s\n", mux);
486}
487show_leds(1)
488show_leds(2)
489show_leds(3)
490
491static int lp5523_load_mux(struct lp55xx_chip *chip, u16 mux, int nr)
492{
493 struct lp55xx_engine *engine = &chip->engines[nr - 1];
494 int ret;
495 u8 mux_page[] = {
496 [LP55XX_ENGINE_1] = LP5523_PAGE_MUX1,
497 [LP55XX_ENGINE_2] = LP5523_PAGE_MUX2,
498 [LP55XX_ENGINE_3] = LP5523_PAGE_MUX3,
499 };
500
501 lp5523_load_engine(chip);
502
503 ret = lp55xx_write(chip, LP5523_REG_PROG_PAGE_SEL, mux_page[nr]);
504 if (ret)
505 return ret;
506
507 ret = lp55xx_write(chip, LP5523_REG_PROG_MEM , (u8)(mux >> 8));
508 if (ret)
509 return ret;
510
511 ret = lp55xx_write(chip, LP5523_REG_PROG_MEM + 1, (u8)(mux));
512 if (ret)
513 return ret;
514
515 engine->led_mux = mux;
516 return 0;
517}
518
519static ssize_t store_engine_leds(struct device *dev,
520 struct device_attribute *attr,
521 const char *buf, size_t len, int nr)
522{
523 struct lp55xx_led *led = i2c_get_clientdata(to_i2c_client(dev));
524 struct lp55xx_chip *chip = led->chip;
525 struct lp55xx_engine *engine = &chip->engines[nr - 1];
526 u16 mux = 0;
527 ssize_t ret;
528
529 if (lp5523_mux_parse(buf, &mux, len))
530 return -EINVAL;
531
532 mutex_lock(&chip->lock);
533
534 chip->engine_idx = nr;
535 ret = -EINVAL;
536
537 if (engine->mode != LP55XX_ENGINE_LOAD)
538 goto leave;
539
540 if (lp5523_load_mux(chip, mux, nr))
541 goto leave;
542
543 ret = len;
544leave:
545 mutex_unlock(&chip->lock);
546 return ret;
547}
548store_leds(1)
549store_leds(2)
550store_leds(3)
551
552static ssize_t store_engine_load(struct device *dev,
553 struct device_attribute *attr,
554 const char *buf, size_t len, int nr)
555{
556 struct lp55xx_led *led = i2c_get_clientdata(to_i2c_client(dev));
557 struct lp55xx_chip *chip = led->chip;
558
559 mutex_lock(&chip->lock);
560
561 chip->engine_idx = nr;
562 lp5523_load_engine_and_select_page(chip);
563
564 mutex_unlock(&chip->lock);
565
566 return lp5523_update_program_memory(chip, buf, len);
567}
568store_load(1)
569store_load(2)
570store_load(3)
571
371static ssize_t lp5523_selftest(struct device *dev, 572static ssize_t lp5523_selftest(struct device *dev,
372 struct device_attribute *attr, 573 struct device_attribute *attr,
373 char *buf) 574 char *buf)
@@ -467,9 +668,27 @@ static void lp5523_led_brightness_work(struct work_struct *work)
467 mutex_unlock(&chip->lock); 668 mutex_unlock(&chip->lock);
468} 669}
469 670
470static DEVICE_ATTR(selftest, S_IRUGO, lp5523_selftest, NULL); 671static LP55XX_DEV_ATTR_RW(engine1_mode, show_engine1_mode, store_engine1_mode);
672static LP55XX_DEV_ATTR_RW(engine2_mode, show_engine2_mode, store_engine2_mode);
673static LP55XX_DEV_ATTR_RW(engine3_mode, show_engine3_mode, store_engine3_mode);
674static LP55XX_DEV_ATTR_RW(engine1_leds, show_engine1_leds, store_engine1_leds);
675static LP55XX_DEV_ATTR_RW(engine2_leds, show_engine2_leds, store_engine2_leds);
676static LP55XX_DEV_ATTR_RW(engine3_leds, show_engine3_leds, store_engine3_leds);
677static LP55XX_DEV_ATTR_WO(engine1_load, store_engine1_load);
678static LP55XX_DEV_ATTR_WO(engine2_load, store_engine2_load);
679static LP55XX_DEV_ATTR_WO(engine3_load, store_engine3_load);
680static LP55XX_DEV_ATTR_RO(selftest, lp5523_selftest);
471 681
472static struct attribute *lp5523_attributes[] = { 682static struct attribute *lp5523_attributes[] = {
683 &dev_attr_engine1_mode.attr,
684 &dev_attr_engine2_mode.attr,
685 &dev_attr_engine3_mode.attr,
686 &dev_attr_engine1_load.attr,
687 &dev_attr_engine2_load.attr,
688 &dev_attr_engine3_load.attr,
689 &dev_attr_engine1_leds.attr,
690 &dev_attr_engine2_leds.attr,
691 &dev_attr_engine3_leds.attr,
473 &dev_attr_selftest.attr, 692 &dev_attr_selftest.attr,
474 NULL, 693 NULL,
475}; 694};