onoff_key_acts.c 8.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. /*
  2. * Copyright (c) 2017 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Actions SoC On/Off Key driver
  9. */
  10. #include <errno.h>
  11. #include <kernel.h>
  12. #include <string.h>
  13. #include <init.h>
  14. #include <irq.h>
  15. #include <soc.h>
  16. #include <board_cfg.h>
  17. #include <drivers/input/input_dev.h>
  18. #include <sys/util.h>
  19. #include <logging/log.h>
  20. #ifdef CONFIG_CFG_DRV
  21. #include <config.h>
  22. #include <drivers/cfg_drv/driver_config.h>
  23. #endif
  24. LOG_MODULE_REGISTER(onoffkey, CONFIG_SYS_LOG_INPUT_DEV_LEVEL);
  25. #define ONOFF_KEY_RESERVED (3)
  26. #define ONOFF_KEY_CODE_INVALID (0xFF)
  27. struct onoff_key_info {
  28. uint32_t poll_interval_ms;
  29. uint32_t poll_total_ms;
  30. uint32_t timestamp;
  31. #ifdef CONFIG_ONOFF_KEY_POLL_TIMER
  32. struct k_timer timer;
  33. #else
  34. struct k_delayed_work timer;
  35. #endif
  36. input_notify_t notify;
  37. uint8_t sample_filter_cnt;
  38. uint8_t scan_count;
  39. uint8_t user_key_code;
  40. uint8_t status_check_on : 1; /* if 1 stands for the onoff key status check worker is on */
  41. uint8_t prev_key_stable_status : 2; /* previous onoff key stable status */
  42. };
  43. static void onoff_key_acts_report_key(struct onoff_key_info *onoff, int value)
  44. {
  45. struct input_value val = {0};
  46. if (onoff->notify) {
  47. val.keypad.code = onoff->user_key_code;
  48. val.keypad.type = EV_KEY;
  49. val.keypad.value = value;
  50. LOG_DBG("report onoff key code:%d value:%d", val.keypad.code, value);
  51. onoff->notify(NULL, &val);
  52. }
  53. }
  54. static void onoff_key_acts_enable(const struct device *dev)
  55. {
  56. ARG_UNUSED(dev);
  57. }
  58. static void onoff_key_acts_disable(const struct device *dev)
  59. {
  60. ARG_UNUSED(dev);
  61. }
  62. void onoff_key_acts_inquiry(const struct device *dev, struct input_value *val)
  63. {
  64. int value;
  65. struct onoff_key_info *onoff = dev->data;
  66. if (!soc_pmu_is_onoff_key_pressed()) {
  67. value = 0;
  68. } else {
  69. value = 1;
  70. }
  71. val->keypad.code = onoff->user_key_code;
  72. val->keypad.type = EV_KEY;
  73. val->keypad.value = value;
  74. }
  75. static void onoff_key_acts_register_notify(const struct device *dev, input_notify_t notify)
  76. {
  77. struct onoff_key_info *onoff = dev->data;
  78. LOG_DBG("register notify 0x%x", (uint32_t)notify);
  79. onoff->notify = notify;
  80. }
  81. static void onoff_key_acts_unregister_notify(const struct device *dev, input_notify_t notify)
  82. {
  83. struct onoff_key_info *onoff = dev->data;
  84. LOG_DBG("unregister notify 0x%x", (uint32_t)notify);
  85. onoff->notify = NULL;
  86. }
  87. static const struct input_dev_driver_api onoff_key_acts_driver_api = {
  88. .enable = onoff_key_acts_enable,
  89. .disable = onoff_key_acts_disable,
  90. .inquiry = onoff_key_acts_inquiry,
  91. .register_notify = onoff_key_acts_register_notify,
  92. .unregister_notify = onoff_key_acts_unregister_notify,
  93. };
  94. #ifdef CONFIG_ONOFF_KEY_POLL_TIMER
  95. static void onoff_key_acts_poll(struct k_timer *timer)
  96. #else
  97. static void onoff_key_acts_poll(struct k_work *work)
  98. #endif
  99. {
  100. #ifdef CONFIG_ONOFF_KEY_POLL_TIMER
  101. struct device *dev = k_timer_user_data_get(timer);
  102. struct onoff_key_info *onoff = dev->data;
  103. #else
  104. struct onoff_key_info *onoff =
  105. CONTAINER_OF(work, struct onoff_key_info, timer);
  106. #endif
  107. uint8_t is_onoff_pressed = soc_pmu_is_onoff_key_pressed();
  108. LOG_DBG("onoff_pressed:%d prev_key_stable_status:%d scan_count:%d(%d)",
  109. is_onoff_pressed, onoff->prev_key_stable_status,
  110. onoff->scan_count, onoff->sample_filter_cnt);
  111. if (is_onoff_pressed) {
  112. onoff->timestamp = k_uptime_get_32();
  113. LOG_DBG("new timestamp:%d", onoff->timestamp);
  114. }
  115. if (onoff->prev_key_stable_status != is_onoff_pressed) {
  116. if (++onoff->scan_count == onoff->sample_filter_cnt) {
  117. onoff->prev_key_stable_status = is_onoff_pressed;
  118. onoff_key_acts_report_key(onoff, is_onoff_pressed);
  119. onoff->scan_count = 0;
  120. }
  121. }
  122. if (onoff->prev_key_stable_status == 1)
  123. onoff_key_acts_report_key(onoff, 1);
  124. if ((k_uptime_get_32() - onoff->timestamp) > onoff->poll_total_ms) {
  125. onoff->status_check_on = 0;
  126. onoff->timestamp = 0;
  127. onoff->scan_count = 0;
  128. onoff->prev_key_stable_status = ONOFF_KEY_RESERVED;
  129. } else {
  130. #ifdef CONFIG_ONOFF_KEY_POLL_TIMER
  131. k_timer_start(&onoff->timer, K_MSEC(onoff->poll_interval_ms), K_NO_WAIT);
  132. #else
  133. k_delayed_work_submit(&onoff->timer, K_MSEC(onoff->poll_interval_ms));
  134. #endif
  135. }
  136. }
  137. static void onoff_key_pmu_notify(void *cb_data, int state)
  138. {
  139. struct device *dev = (struct device *)cb_data;
  140. struct onoff_key_info *onoff = dev->data;
  141. LOG_DBG("PMU notify state:%d", state);
  142. if ((state != PMU_NOTIFY_STATE_PRESSED)
  143. && (state != PMU_NOTIFY_STATE_LONG_PRESSED))
  144. return ;
  145. if (!soc_pmu_is_onoff_key_pressed())
  146. return ;
  147. if (onoff->status_check_on)
  148. return ;
  149. onoff->status_check_on = 1;
  150. onoff->timestamp = k_uptime_get_32();
  151. onoff_key_acts_report_key(onoff, 1);
  152. #ifdef CONFIG_ONOFF_KEY_POLL_TIMER
  153. k_timer_start(&onoff->timer, K_MSEC(onoff->poll_interval_ms), K_NO_WAIT);
  154. #else
  155. k_delayed_work_submit(&onoff->timer, K_MSEC(onoff->poll_interval_ms));
  156. #endif
  157. }
  158. static int onoff_key_acts_init(const struct device *dev)
  159. {
  160. struct onoff_key_info *onoff = dev->data;
  161. uint8_t long_press_time;
  162. uint8_t key_function;
  163. struct detect_param_t detect_param = {PMU_DETECT_DEV_ONOFF, onoff_key_pmu_notify, (void *)dev};
  164. LOG_DBG("init on/off key");
  165. onoff->poll_interval_ms = CONFIG_ONOFFKEY_POLL_INTERVAL_MS;
  166. onoff->poll_total_ms = CONFIG_ONOFFKEY_POLL_TOTAL_MS;
  167. onoff->status_check_on = 0;
  168. onoff->sample_filter_cnt = CONFIG_ONOFFKEY_SAMPLE_FILTER_CNT;
  169. onoff->scan_count = 0;
  170. onoff->prev_key_stable_status = ONOFF_KEY_RESERVED;
  171. long_press_time = CONFIG_ONOFFKEY_LONG_PRESS_TIME;
  172. key_function = CONFIG_ONOFFKEY_FUNCTION;
  173. #ifdef CONFIG_CFG_DRV
  174. uint8_t Use_Inner_ONOFF_Key, Reset_Time_Ms, Key_Value;
  175. uint16_t __long_press_time;
  176. uint16_t debounce_time_ms;
  177. if (cfg_get_by_key(ITEM_ONOFF_USE_INNER_ONOFF_KEY,
  178. &Use_Inner_ONOFF_Key, sizeof(Use_Inner_ONOFF_Key))) {
  179. if (!Use_Inner_ONOFF_Key) {
  180. LOG_INF("onoff key exit by config tool");
  181. return 0;
  182. }
  183. }
  184. if (cfg_get_by_key(ITEM_ONOFF_TIME_LONG_PRESS_RESET,
  185. &Reset_Time_Ms, sizeof(Reset_Time_Ms))) {
  186. if (Reset_Time_Ms == 0xFF) {
  187. key_function = 0;
  188. } else if (Reset_Time_Ms == 0) {
  189. soc_pmu_config_onoffkey_reset_time(0);
  190. key_function = 1;
  191. } else if (Reset_Time_Ms == 1) {
  192. soc_pmu_config_onoffkey_reset_time(1);
  193. key_function = 1;
  194. }
  195. }
  196. if (cfg_get_by_key(ITEM_ONOFF_TIME_PRESS_POWER_ON,
  197. &__long_press_time, sizeof(__long_press_time))) {
  198. if (__long_press_time == 250)
  199. long_press_time = 1;
  200. else if (__long_press_time == 500)
  201. long_press_time = 2;
  202. else if (__long_press_time == 1000)
  203. long_press_time = 3;
  204. else if (__long_press_time == 1500)
  205. long_press_time = 4;
  206. else if (__long_press_time == 2000)
  207. long_press_time = 5;
  208. else if (__long_press_time == 3000)
  209. long_press_time = 6;
  210. else if (__long_press_time == 4000)
  211. long_press_time = 7;
  212. }
  213. if (cfg_get_by_key(ITEM_ONOFF_DEBOUNCE_TIME_MS,
  214. &debounce_time_ms, sizeof(debounce_time_ms))) {
  215. onoff->sample_filter_cnt = (debounce_time_ms / onoff->poll_interval_ms);
  216. LOG_DBG("debounce_time_ms:%d poll_interval_ms:%d sample_filter_cnt:%d",
  217. debounce_time_ms, onoff->poll_interval_ms, onoff->sample_filter_cnt);
  218. }
  219. if (cfg_get_by_key(ITEM_ONOFF_KEY_VALUE,&Key_Value, sizeof(Key_Value))) {
  220. onoff->user_key_code = Key_Value;
  221. } else {
  222. #ifdef CONFIG_ONOFFKEY_USER_KEYCODE
  223. onoff->user_key_code = CONFIG_ONOFFKEY_USER_KEYCODE;
  224. #else
  225. onoff->user_key_code = KEY_POWER;
  226. #endif
  227. }
  228. #else
  229. #ifdef CONFIG_ONOFFKEY_USER_KEYCODE
  230. onoff->user_key_code = CONFIG_ONOFFKEY_USER_KEYCODE;
  231. #else
  232. onoff->user_key_code = KEY_POWER;
  233. #endif
  234. #endif
  235. soc_pmu_config_onoffkey_function(key_function);
  236. soc_pmu_config_onoffkey_time(long_press_time);
  237. if (soc_pmu_register_notify(&detect_param)) {
  238. LOG_ERR("failed to register pmu notify");
  239. return -ENXIO;
  240. }
  241. #ifdef CONFIG_ONOFF_KEY_POLL_TIMER
  242. k_timer_init(&onoff->timer, onoff_key_acts_poll, NULL);
  243. k_timer_user_data_set(&onoff->timer, (void *)dev);
  244. #else
  245. k_delayed_work_init(&onoff->timer, onoff_key_acts_poll);
  246. #endif
  247. return 0;
  248. }
  249. static struct onoff_key_info onoff_key_acts_ddata;
  250. #if IS_ENABLED(CONFIG_ONOFFKEY)
  251. DEVICE_DEFINE(onoff_key_acts, CONFIG_INPUT_DEV_ACTS_ONOFF_KEY_NAME,
  252. onoff_key_acts_init, NULL,
  253. &onoff_key_acts_ddata, NULL,
  254. POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_DEVICE,
  255. &onoff_key_acts_driver_api);
  256. #endif