ledtrig-input-events.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. // SPDX-License-Identifier: GPL-2.0-only
  2. /*
  3. * Input Events LED trigger
  4. *
  5. * Copyright (C) 2024 Hans de Goede <hansg@kernel.org>
  6. */
  7. #include <linux/input.h>
  8. #include <linux/jiffies.h>
  9. #include <linux/leds.h>
  10. #include <linux/module.h>
  11. #include <linux/moduleparam.h>
  12. #include <linux/slab.h>
  13. #include <linux/spinlock.h>
  14. #include <linux/workqueue.h>
  15. #include "../leds.h"
  16. static unsigned long led_off_delay_ms = 5000;
  17. module_param(led_off_delay_ms, ulong, 0644);
  18. MODULE_PARM_DESC(led_off_delay_ms,
  19. "Specify delay in ms for turning LEDs off after last input event");
  20. static struct input_events_data {
  21. struct delayed_work work;
  22. spinlock_t lock;
  23. /* To avoid repeatedly setting the brightness while there are events */
  24. bool led_on;
  25. unsigned long led_off_time;
  26. } input_events_data;
  27. static struct led_trigger *input_events_led_trigger;
  28. static void led_input_events_work(struct work_struct *work)
  29. {
  30. struct input_events_data *data =
  31. container_of(work, struct input_events_data, work.work);
  32. spin_lock_irq(&data->lock);
  33. /*
  34. * This time_after_eq() check avoids a race where this work starts
  35. * running before a new event pushed led_off_time back.
  36. */
  37. if (time_after_eq(jiffies, data->led_off_time)) {
  38. led_trigger_event(input_events_led_trigger, LED_OFF);
  39. data->led_on = false;
  40. }
  41. spin_unlock_irq(&data->lock);
  42. }
  43. static void input_events_event(struct input_handle *handle, unsigned int type,
  44. unsigned int code, int val)
  45. {
  46. struct input_events_data *data = &input_events_data;
  47. unsigned long led_off_delay = msecs_to_jiffies(led_off_delay_ms);
  48. unsigned long flags;
  49. spin_lock_irqsave(&data->lock, flags);
  50. if (!data->led_on) {
  51. led_trigger_event(input_events_led_trigger, LED_FULL);
  52. data->led_on = true;
  53. }
  54. data->led_off_time = jiffies + led_off_delay;
  55. spin_unlock_irqrestore(&data->lock, flags);
  56. mod_delayed_work(system_percpu_wq, &data->work, led_off_delay);
  57. }
  58. static int input_events_connect(struct input_handler *handler, struct input_dev *dev,
  59. const struct input_device_id *id)
  60. {
  61. struct input_handle *handle;
  62. int ret;
  63. handle = kzalloc_obj(*handle);
  64. if (!handle)
  65. return -ENOMEM;
  66. handle->dev = dev;
  67. handle->handler = handler;
  68. handle->name = KBUILD_MODNAME;
  69. ret = input_register_handle(handle);
  70. if (ret)
  71. goto err_free_handle;
  72. ret = input_open_device(handle);
  73. if (ret)
  74. goto err_unregister_handle;
  75. return 0;
  76. err_unregister_handle:
  77. input_unregister_handle(handle);
  78. err_free_handle:
  79. kfree(handle);
  80. return ret;
  81. }
  82. static void input_events_disconnect(struct input_handle *handle)
  83. {
  84. input_close_device(handle);
  85. input_unregister_handle(handle);
  86. kfree(handle);
  87. }
  88. static const struct input_device_id input_events_ids[] = {
  89. {
  90. .flags = INPUT_DEVICE_ID_MATCH_EVBIT,
  91. .evbit = { BIT_MASK(EV_KEY) },
  92. },
  93. {
  94. .flags = INPUT_DEVICE_ID_MATCH_EVBIT,
  95. .evbit = { BIT_MASK(EV_REL) },
  96. },
  97. {
  98. .flags = INPUT_DEVICE_ID_MATCH_EVBIT,
  99. .evbit = { BIT_MASK(EV_ABS) },
  100. },
  101. { }
  102. };
  103. static struct input_handler input_events_handler = {
  104. .name = KBUILD_MODNAME,
  105. .event = input_events_event,
  106. .connect = input_events_connect,
  107. .disconnect = input_events_disconnect,
  108. .id_table = input_events_ids,
  109. };
  110. static int __init input_events_init(void)
  111. {
  112. int ret;
  113. INIT_DELAYED_WORK(&input_events_data.work, led_input_events_work);
  114. spin_lock_init(&input_events_data.lock);
  115. led_trigger_register_simple("input-events", &input_events_led_trigger);
  116. ret = input_register_handler(&input_events_handler);
  117. if (ret) {
  118. led_trigger_unregister_simple(input_events_led_trigger);
  119. return ret;
  120. }
  121. return 0;
  122. }
  123. static void __exit input_events_exit(void)
  124. {
  125. input_unregister_handler(&input_events_handler);
  126. cancel_delayed_work_sync(&input_events_data.work);
  127. led_trigger_unregister_simple(input_events_led_trigger);
  128. }
  129. module_init(input_events_init);
  130. module_exit(input_events_exit);
  131. MODULE_AUTHOR("Hans de Goede <hansg@kernel.org>");
  132. MODULE_DESCRIPTION("Input Events LED trigger");
  133. MODULE_LICENSE("GPL");
  134. MODULE_ALIAS("ledtrig:input-events");