leds.c 2.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  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/export.h>
  11. #include <linux/init.h>
  12. #include <linux/sysdev.h>
  13. #include <linux/syscore_ops.h>
  14. #include <linux/string.h>
  15. #include <asm/leds.h>
  16. static void dummy_leds_event(led_event_t evt)
  17. {
  18. }
  19. void (*leds_event)(led_event_t) = dummy_leds_event;
  20. struct leds_evt_name {
  21. const char name[8];
  22. int on;
  23. int off;
  24. };
  25. static const struct leds_evt_name evt_names[] = {
  26. { "amber", led_amber_on, led_amber_off },
  27. { "blue", led_blue_on, led_blue_off },
  28. { "green", led_green_on, led_green_off },
  29. { "red", led_red_on, led_red_off },
  30. };
  31. static ssize_t leds_store(struct sys_device *dev,
  32. struct sysdev_attribute *attr,
  33. const char *buf, size_t size)
  34. {
  35. int ret = -EINVAL, len = strcspn(buf, " ");
  36. if (len > 0 && buf[len] == '\0')
  37. len--;
  38. if (strncmp(buf, "claim", len) == 0) {
  39. leds_event(led_claim);
  40. ret = size;
  41. } else if (strncmp(buf, "release", len) == 0) {
  42. leds_event(led_release);
  43. ret = size;
  44. } else {
  45. int i;
  46. for (i = 0; i < ARRAY_SIZE(evt_names); i++) {
  47. if (strlen(evt_names[i].name) != len ||
  48. strncmp(buf, evt_names[i].name, len) != 0)
  49. continue;
  50. if (strncmp(buf+len, " on", 3) == 0) {
  51. leds_event(evt_names[i].on);
  52. ret = size;
  53. } else if (strncmp(buf+len, " off", 4) == 0) {
  54. leds_event(evt_names[i].off);
  55. ret = size;
  56. }
  57. break;
  58. }
  59. }
  60. return ret;
  61. }
  62. static SYSDEV_ATTR(event, 0200, NULL, leds_store);
  63. static struct sysdev_class leds_sysclass = {
  64. .name = "leds",
  65. };
  66. static struct sys_device leds_device = {
  67. .id = 0,
  68. .cls = &leds_sysclass,
  69. };
  70. static int leds_suspend(void)
  71. {
  72. leds_event(led_stop);
  73. return 0;
  74. }
  75. static void leds_resume(void)
  76. {
  77. leds_event(led_start);
  78. }
  79. static void leds_shutdown(void)
  80. {
  81. leds_event(led_halted);
  82. }
  83. static struct syscore_ops leds_syscore_ops = {
  84. .shutdown = leds_shutdown,
  85. .suspend = leds_suspend,
  86. .resume = leds_resume,
  87. };
  88. static int __init leds_init(void)
  89. {
  90. int ret;
  91. ret = sysdev_class_register(&leds_sysclass);
  92. if (ret == 0)
  93. ret = sysdev_register(&leds_device);
  94. if (ret == 0)
  95. ret = sysdev_create_file(&leds_device, &attr_event);
  96. if (ret == 0)
  97. register_syscore_ops(&leds_syscore_ops);
  98. return ret;
  99. }
  100. device_initcall(leds_init);
  101. EXPORT_SYMBOL(leds_event);