led-core.c 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. /*
  2. * LED Class Core
  3. *
  4. * Copyright 2005-2006 Openedhand Ltd.
  5. *
  6. * Author: Richard Purdie <rpurdie@openedhand.com>
  7. *
  8. * This program is free software; you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License version 2 as
  10. * published by the Free Software Foundation.
  11. *
  12. */
  13. #include <linux/kernel.h>
  14. #include <linux/leds.h>
  15. #include <linux/list.h>
  16. #include <linux/module.h>
  17. #include <linux/mutex.h>
  18. #include <linux/rwsem.h>
  19. #include "leds.h"
  20. DECLARE_RWSEM(leds_list_lock);
  21. EXPORT_SYMBOL_GPL(leds_list_lock);
  22. LIST_HEAD(leds_list);
  23. EXPORT_SYMBOL_GPL(leds_list);
  24. static void led_set_software_blink(struct led_classdev *led_cdev,
  25. unsigned long delay_on,
  26. unsigned long delay_off)
  27. {
  28. int current_brightness;
  29. current_brightness = led_get_brightness(led_cdev);
  30. if (current_brightness)
  31. led_cdev->blink_brightness = current_brightness;
  32. if (!led_cdev->blink_brightness)
  33. led_cdev->blink_brightness = led_cdev->max_brightness;
  34. led_cdev->blink_delay_on = delay_on;
  35. led_cdev->blink_delay_off = delay_off;
  36. /* never on - just set to off */
  37. if (!delay_on) {
  38. __led_set_brightness(led_cdev, LED_OFF);
  39. return;
  40. }
  41. /* never off - just set to brightness */
  42. if (!delay_off) {
  43. __led_set_brightness(led_cdev, led_cdev->blink_brightness);
  44. return;
  45. }
  46. mod_timer(&led_cdev->blink_timer, jiffies + 1);
  47. }
  48. static void led_blink_setup(struct led_classdev *led_cdev,
  49. unsigned long *delay_on,
  50. unsigned long *delay_off)
  51. {
  52. if (!(led_cdev->flags & LED_BLINK_ONESHOT) &&
  53. led_cdev->blink_set &&
  54. !led_cdev->blink_set(led_cdev, delay_on, delay_off))
  55. return;
  56. /* blink with 1 Hz as default if nothing specified */
  57. if (!*delay_on && !*delay_off)
  58. *delay_on = *delay_off = 500;
  59. led_set_software_blink(led_cdev, *delay_on, *delay_off);
  60. }
  61. void led_blink_set(struct led_classdev *led_cdev,
  62. unsigned long *delay_on,
  63. unsigned long *delay_off)
  64. {
  65. del_timer_sync(&led_cdev->blink_timer);
  66. led_cdev->flags &= ~LED_BLINK_ONESHOT;
  67. led_cdev->flags &= ~LED_BLINK_ONESHOT_STOP;
  68. led_blink_setup(led_cdev, delay_on, delay_off);
  69. }
  70. EXPORT_SYMBOL(led_blink_set);
  71. void led_blink_set_oneshot(struct led_classdev *led_cdev,
  72. unsigned long *delay_on,
  73. unsigned long *delay_off,
  74. int invert)
  75. {
  76. if ((led_cdev->flags & LED_BLINK_ONESHOT) &&
  77. timer_pending(&led_cdev->blink_timer))
  78. return;
  79. led_cdev->flags |= LED_BLINK_ONESHOT;
  80. led_cdev->flags &= ~LED_BLINK_ONESHOT_STOP;
  81. if (invert)
  82. led_cdev->flags |= LED_BLINK_INVERT;
  83. else
  84. led_cdev->flags &= ~LED_BLINK_INVERT;
  85. led_blink_setup(led_cdev, delay_on, delay_off);
  86. }
  87. EXPORT_SYMBOL(led_blink_set_oneshot);
  88. void led_stop_software_blink(struct led_classdev *led_cdev)
  89. {
  90. del_timer_sync(&led_cdev->blink_timer);
  91. led_cdev->blink_delay_on = 0;
  92. led_cdev->blink_delay_off = 0;
  93. }
  94. EXPORT_SYMBOL_GPL(led_stop_software_blink);
  95. void led_set_brightness(struct led_classdev *led_cdev,
  96. enum led_brightness brightness)
  97. {
  98. /* delay brightness setting if need to stop soft-blink timer */
  99. if (led_cdev->blink_delay_on || led_cdev->blink_delay_off) {
  100. led_cdev->delayed_set_value = brightness;
  101. schedule_work(&led_cdev->set_brightness_work);
  102. return;
  103. }
  104. __led_set_brightness(led_cdev, brightness);
  105. }
  106. EXPORT_SYMBOL(led_set_brightness);
  107. int led_update_brightness(struct led_classdev *led_cdev)
  108. {
  109. int ret = 0;
  110. if (led_cdev->brightness_get) {
  111. ret = led_cdev->brightness_get(led_cdev);
  112. if (ret >= 0) {
  113. led_cdev->brightness = ret;
  114. return 0;
  115. }
  116. }
  117. return ret;
  118. }
  119. EXPORT_SYMBOL(led_update_brightness);