leds.c 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. /*
  2. * LED support code, ripped out of arch/arm/kernel/time.c
  3. *
  4. * Copyright (C) 1994-2001 Russell King
  5. *
  6. * This program is free software; you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License version 2 as
  8. * published by the Free Software Foundation.
  9. */
  10. #include <linux/module.h>
  11. #include <linux/init.h>
  12. #include <linux/sysdev.h>
  13. #include <linux/syscore_ops.h>
  14. #include <asm/leds.h>
  15. static void dummy_leds_event(led_event_t evt)
  16. {
  17. }
  18. void (*leds_event)(led_event_t) = dummy_leds_event;
  19. struct leds_evt_name {
  20. const char name[8];
  21. int on;
  22. int off;
  23. };
  24. static const struct leds_evt_name evt_names[] = {
  25. { "amber", led_amber_on, led_amber_off },
  26. { "blue", led_blue_on, led_blue_off },
  27. { "green", led_green_on, led_green_off },
  28. { "red", led_red_on, led_red_off },
  29. };
  30. static ssize_t leds_store(struct sys_device *dev,
  31. struct sysdev_attribute *attr,
  32. const char *buf, size_t size)
  33. {
  34. int ret = -EINVAL, len = strcspn(buf, " ");
  35. if (len > 0 && buf[len] == '\0')
  36. len--;
  37. if (strncmp(buf, "claim", len) == 0) {
  38. leds_event(led_claim);
  39. ret = size;
  40. } else if (strncmp(buf, "release", len) == 0) {
  41. leds_event(led_release);
  42. ret = size;
  43. } else {
  44. int i;
  45. for (i = 0; i < ARRAY_SIZE(evt_names); i++) {
  46. if (strlen(evt_names[i].name) != len ||
  47. strncmp(buf, evt_names[i].name, len) != 0)
  48. continue;
  49. if (strncmp(buf+len, " on", 3) == 0) {
  50. leds_event(evt_names[i].on);
  51. ret = size;
  52. } else if (strncmp(buf+len, " off", 4) == 0) {
  53. leds_event(evt_names[i].off);
  54. ret = size;
  55. }
  56. break;
  57. }
  58. }
  59. return ret;
  60. }
  61. static SYSDEV_ATTR(event, 0200, NULL, leds_store);
  62. static struct sysdev_class leds_sysclass = {
  63. .name = "leds",
  64. };
  65. static struct sys_device leds_device = {
  66. .id = 0,
  67. .cls = &leds_sysclass,
  68. };
  69. static int leds_suspend(void)
  70. {
  71. leds_event(led_stop);
  72. return 0;
  73. }
  74. static void leds_resume(void)
  75. {
  76. leds_event(led_start);
  77. }
  78. static void leds_shutdown(void)
  79. {
  80. leds_event(led_halted);
  81. }
  82. static struct syscore_ops leds_syscore_ops = {
  83. .shutdown = leds_shutdown,
  84. .suspend = leds_suspend,
  85. .resume = leds_resume,
  86. };
  87. static int __init leds_init(void)
  88. {
  89. int ret;
  90. ret = sysdev_class_register(&leds_sysclass);
  91. if (ret == 0)
  92. ret = sysdev_register(&leds_device);
  93. if (ret == 0)
  94. ret = sysdev_create_file(&leds_device, &attr_event);
  95. if (ret == 0)
  96. register_syscore_ops(&leds_syscore_ops);
  97. return ret;
  98. }
  99. device_initcall(leds_init);
  100. EXPORT_SYMBOL(leds_event);