input_keypad.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498
  1. /*
  2. * Copyright (c) 2019 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file input manager interface
  8. */
  9. #include <os_common_api.h>
  10. #include <string.h>
  11. #include <key_hal.h>
  12. #include <app_manager.h>
  13. #include <mem_manager.h>
  14. #include <input_manager.h>
  15. #include <msg_manager.h>
  16. #include <sys_event.h>
  17. #include <srv_manager.h>
  18. #include <property_manager.h>
  19. #include <input_manager_inner.h>
  20. #ifndef CONFIG_SIMULATOR
  21. #include <board.h>
  22. #endif
  23. //K_MSGQ_DEFINE(keypad_scan_msgq, sizeof(input_dev_data_t), 10, 4);
  24. #define MAX_KEY_SUPPORT 10
  25. #define MAX_HOLD_KEY_SUPPORT 4
  26. #define MAX_MUTIPLE_CLICK_KEY_SUPPORT 6
  27. #define LONG_PRESS_TIMER 10 * 40
  28. #define SUPER_LONG_PRESS_TIMER 50 * 40 /* time */
  29. #if CONFIG_AEM_WATCH_SUPPORT
  30. #define SUPER_LONG_PRESS_6S_TIMER 100 * 50 /* time */
  31. #else
  32. #define SUPER_LONG_PRESS_6S_TIMER 150 * 40 /* time */
  33. #endif
  34. #define QUICKLY_CLICK_DURATION 300 /* ms */
  35. #define KEY_EVENT_CANCEL_DURATION 50 /* ms */
  36. #define HOLD_DELAY_TIME 500 /* ms */
  37. #define HOLD_INTERVAL_DURATION 200 /* ms */
  38. struct input_keypad_info {
  39. struct k_delayed_work report_work;
  40. struct k_delayed_work hold_key_work;
  41. struct k_delayed_work callback_work;
  42. event_trigger event_cb;
  43. uint32_t press_type;
  44. uint16_t press_code;
  45. uint16_t report_press_code;
  46. uint32_t report_key_value;
  47. uint32_t last_report_key_value;
  48. uint32_t callback_key_value;
  49. int64_t report_stamp;
  50. uint32_t press_timer;
  51. uint16_t click_num: 4;
  52. uint16_t key_hold:1;
  53. uint16_t filter_itself:1; /*过滤当前按键后续所有的事件*/
  54. uint16_t combo_mode:1;
  55. uint16_t combo_state_0:1;
  56. uint16_t combo_state_1:1;
  57. void* onoff_handle;
  58. void* adckey_handle;
  59. void* gpiokey_handle;
  60. uint16_t long_press_timer;
  61. uint16_t super_long_press_timer;
  62. uint16_t super_long_press_6s_timer;
  63. uint16_t quickly_click_duration ;
  64. uint16_t key_event_cancel_duration;
  65. uint16_t hold_delay_time;
  66. uint16_t hold_interval_time;
  67. };
  68. static struct input_keypad_info input_keypad;
  69. static struct input_keypad_info *keypad;
  70. #ifdef HOLDABLE_KEY
  71. const char support_hold_key[MAX_HOLD_KEY_SUPPORT] = HOLDABLE_KEY;
  72. #else
  73. const char support_hold_key[MAX_HOLD_KEY_SUPPORT] = {0};
  74. #endif
  75. #ifdef MUTIPLE_CLIK_KEY
  76. const char support_mutiple_key[MAX_MUTIPLE_CLICK_KEY_SUPPORT] = MUTIPLE_CLIK_KEY;
  77. #else
  78. const char support_mutiple_key[MAX_MUTIPLE_CLICK_KEY_SUPPORT] = {0};
  79. #endif
  80. void report_key_event_work_handle(struct k_work *work)
  81. {
  82. struct input_keypad_info *input = CONTAINER_OF(work, struct input_keypad_info, report_work);
  83. if (input->filter_itself) {
  84. if ((input->last_report_key_value & KEY_TYPE_SHORT_UP)
  85. || (input->last_report_key_value & KEY_TYPE_DOUBLE_CLICK)
  86. || (input->last_report_key_value & KEY_TYPE_TRIPLE_CLICK)
  87. || (input->last_report_key_value & KEY_TYPE_QUAD_CLICK)
  88. || (input->last_report_key_value & KEY_TYPE_QUINT_CLICK)
  89. || (input->last_report_key_value & KEY_TYPE_LONG_UP)) {
  90. input->filter_itself = false;
  91. }
  92. return;
  93. }
  94. input->click_num = 0;
  95. //os_printk("*******************\n\n\n\n report input 0x%x \n\n\n\n*********************\n",input->last_report_key_value);
  96. sys_event_report_input(input->last_report_key_value);
  97. }
  98. void callback_key_event_work_handle(struct k_work *work)
  99. {
  100. struct input_keypad_info *input = CONTAINER_OF(work, struct input_keypad_info, callback_work);
  101. if (input->event_cb) {
  102. input->event_cb(input->callback_key_value,EV_KEY);
  103. }
  104. }
  105. static bool is_support_hold(int key_code)
  106. {
  107. for (int i = 0 ; i < MAX_HOLD_KEY_SUPPORT; i++) {
  108. if (support_hold_key[i] == key_code) {
  109. return true;
  110. }
  111. }
  112. return false;
  113. }
  114. #ifdef CONFIG_INPUT_MUTIPLE_CLICK
  115. static bool is_support_mutiple_click(int key_code)
  116. {
  117. for (int i = 0 ; i < MAX_MUTIPLE_CLICK_KEY_SUPPORT; i++) {
  118. if (support_mutiple_key[i] == key_code) {
  119. return true;
  120. }
  121. }
  122. return false;
  123. }
  124. #endif
  125. static void check_hold_key_work_handle(struct k_work *work)
  126. {
  127. struct input_keypad_info *keypad = CONTAINER_OF(work, struct input_keypad_info, hold_key_work);
  128. int key_value = 0;
  129. if (keypad->key_hold) {
  130. os_delayed_work_submit(&keypad->hold_key_work, keypad->hold_interval_time);
  131. if (!input_manager_islock()) {
  132. key_value = KEY_TYPE_HOLD | keypad->press_code;
  133. }
  134. } else {
  135. if (!input_manager_islock()) {
  136. key_value = KEY_TYPE_HOLD_UP | keypad->press_code;
  137. }
  138. }
  139. if (key_value) {
  140. if (keypad->event_cb && !input_manager_islock()) {
  141. keypad->event_cb(key_value, EV_KEY);
  142. }
  143. if (keypad->filter_itself) {
  144. return;
  145. }
  146. sys_event_report_input(key_value);
  147. }
  148. }
  149. static void keypad_scan_callback(struct device *dev, struct input_value *val)
  150. {
  151. static input_dev_state_t last_value = KEY_VALUE_UP;
  152. bool need_report = false;
  153. int32_t time_ms;
  154. if (val->keypad.type != EV_KEY) {
  155. SYS_LOG_ERR("input type %d not support\n", val->keypad.type);
  156. return;
  157. }
  158. if (last_value != val->keypad.value) {
  159. input_manager_boost(val->keypad.value == KEY_VALUE_DOWN);
  160. last_value = val->keypad.value;
  161. }
  162. SYS_LOG_INF("type: %d, code:%d, value:%d\n", val->keypad.type, val->keypad.code, val->keypad.value);
  163. switch (val->keypad.value) {
  164. case KEY_VALUE_UP:
  165. {
  166. if(!keypad->combo_mode) {
  167. keypad->press_code = val->keypad.code;
  168. keypad->report_press_code = keypad->press_code;
  169. keypad->press_code = 0;
  170. } else {
  171. if ((keypad->press_code & 0xff) == val->keypad.code) {
  172. keypad->combo_state_0 = KEY_VALUE_UP;
  173. } else if(((keypad->press_code >> 8) & 0xff) == val->keypad.code) {
  174. keypad->combo_state_1 = KEY_VALUE_UP;
  175. }
  176. keypad->report_press_code = keypad->press_code;
  177. if (keypad->combo_state_0 == keypad->combo_state_1
  178. && keypad->combo_state_0 == KEY_VALUE_UP) {
  179. keypad->combo_mode = 0;
  180. keypad->press_code = 0;
  181. } else {
  182. return;
  183. }
  184. }
  185. keypad->key_hold = false;
  186. if (keypad->press_type == KEY_TYPE_LONG_DOWN
  187. || keypad->press_type == KEY_TYPE_LONG) {
  188. keypad->press_type = KEY_TYPE_LONG_UP;
  189. }else if(keypad->press_type == KEY_TYPE_LONG6S){
  190. keypad->press_type = KEY_TYPE_LONG6S_UP;
  191. } else {
  192. #ifdef CONFIG_INPUT_MUTIPLE_CLICK
  193. if (is_support_mutiple_click(keypad->report_press_code)) {
  194. if (keypad->last_report_key_value == (keypad->press_type | keypad->report_press_code)) {
  195. if(keypad->click_num) {
  196. os_delayed_work_cancel(&keypad->report_work);
  197. }
  198. keypad->click_num++;
  199. keypad->report_stamp = k_uptime_get();
  200. } else {
  201. if(keypad->click_num) {
  202. os_delayed_work_cancel(&keypad->report_work);
  203. }
  204. keypad->click_num = 1;
  205. keypad->report_stamp = k_uptime_get();
  206. }
  207. switch (keypad->click_num) {
  208. case 1:
  209. keypad->press_type = KEY_TYPE_SHORT_UP;
  210. break;
  211. case 2:
  212. keypad->press_type = KEY_TYPE_DOUBLE_CLICK;
  213. break;
  214. case 3:
  215. keypad->press_type = KEY_TYPE_TRIPLE_CLICK;
  216. break;
  217. case 4:
  218. keypad->press_type = KEY_TYPE_QUAD_CLICK;
  219. break;
  220. case 5:
  221. keypad->press_type = KEY_TYPE_QUINT_CLICK;
  222. break;
  223. }
  224. } else {
  225. keypad->press_type = KEY_TYPE_SHORT_UP;
  226. }
  227. #else
  228. keypad->press_type = KEY_TYPE_SHORT_UP;
  229. #endif
  230. }
  231. //keypad->press_timer = 0;
  232. keypad->press_timer = k_cyc_to_us_near32(k_cycle_get_32()) / 1000;
  233. need_report = true;
  234. break;
  235. }
  236. case KEY_VALUE_DOWN:
  237. {
  238. if (val->keypad.code != (keypad->press_code & 0xff)
  239. && val->keypad.code != ((keypad->press_code >> 8) & 0xff)) {
  240. if ((keypad->press_code & 0xff) == 0) {
  241. keypad->press_code = val->keypad.code;
  242. keypad->combo_mode = 0;
  243. keypad->combo_state_0 = KEY_VALUE_DOWN;
  244. } else if(((keypad->press_code >> 8) & 0xff) == 0) {
  245. keypad->press_code |= (val->keypad.code << 8);
  246. keypad->combo_mode = 1;
  247. keypad->combo_state_1 = KEY_VALUE_DOWN;
  248. }
  249. //keypad->press_timer = 0;
  250. keypad->press_timer = k_cyc_to_us_near32(k_cycle_get_32()) / 1000;
  251. keypad->filter_itself = false;
  252. keypad->report_press_code = keypad->press_code;
  253. keypad->press_type = KEY_TYPE_SHORT_DOWN;
  254. need_report = true;
  255. } else {
  256. //keypad->press_timer++;
  257. time_ms = (k_cyc_to_us_near32(k_cycle_get_32()) / 1000) - keypad->press_timer;
  258. if (time_ms >= keypad->super_long_press_6s_timer) {
  259. if (keypad->press_type != KEY_TYPE_LONG6S) {
  260. keypad->report_press_code = keypad->press_code;
  261. keypad->press_type = KEY_TYPE_LONG6S;
  262. need_report = true;
  263. }
  264. } else if (time_ms >= keypad->super_long_press_timer) {
  265. if (keypad->press_type != KEY_TYPE_LONG) {
  266. keypad->report_press_code = keypad->press_code;
  267. keypad->press_type = KEY_TYPE_LONG;
  268. need_report = true;
  269. }
  270. } else if (time_ms >= keypad->long_press_timer) {
  271. if (keypad->press_type != KEY_TYPE_LONG_DOWN) {
  272. keypad->report_press_code = keypad->press_code;
  273. keypad->press_type = KEY_TYPE_LONG_DOWN;
  274. if (is_support_hold(keypad->press_code)) {
  275. keypad->key_hold = true;
  276. os_delayed_work_submit(&keypad->hold_key_work, keypad->hold_delay_time);
  277. }
  278. need_report = true;
  279. }
  280. }
  281. }
  282. break;
  283. }
  284. default:
  285. break;
  286. }
  287. if (keypad->event_cb) {
  288. keypad->callback_key_value = ((val->keypad.value<<31)|val->keypad.code);
  289. os_delayed_work_submit(&keypad->callback_work, 0);
  290. }
  291. if(val->keypad.value == KEY_VALUE_DOWN && keypad->click_num) {
  292. os_delayed_work_cancel(&keypad->report_work);
  293. os_delayed_work_submit(&keypad->report_work, keypad->quickly_click_duration + keypad->key_event_cancel_duration);
  294. }
  295. if (need_report) {
  296. keypad->report_key_value = keypad->press_type
  297. | keypad->report_press_code;
  298. keypad->last_report_key_value = keypad->report_key_value;
  299. if (keypad->event_cb) {
  300. keypad->event_cb(keypad->last_report_key_value, EV_KEY);
  301. }
  302. if (input_manager_islock()) {
  303. SYS_LOG_INF("input manager locked");
  304. keypad->click_num = 0;
  305. return;
  306. }
  307. if (msg_pool_get_free_msg_num() <= (CONFIG_NUM_MBOX_ASYNC_MSGS / 2)) {
  308. SYS_LOG_INF("drop input msg ... %d", msg_pool_get_free_msg_num());
  309. return;
  310. }
  311. #ifdef CONFIG_INPUT_MUTIPLE_CLICK
  312. if (is_support_mutiple_click(keypad->report_press_code)) {
  313. keypad->report_stamp = k_uptime_get();
  314. if(keypad->press_type == KEY_TYPE_LONG_DOWN || keypad->press_type == KEY_TYPE_LONG || keypad->press_type == KEY_TYPE_LONG6S)
  315. os_delayed_work_submit(&keypad->report_work, 0);
  316. else
  317. os_delayed_work_submit(&keypad->report_work, QUICKLY_CLICK_DURATION + KEY_EVENT_CANCEL_DURATION);
  318. } else {
  319. os_delayed_work_submit(&keypad->report_work, 0);
  320. }
  321. #else
  322. os_delayed_work_submit(&keypad->report_work, 0);
  323. #endif
  324. }
  325. }
  326. int input_keypad_device_init(event_trigger event_cb)
  327. {
  328. keypad = &input_keypad;
  329. memset(keypad, 0, sizeof(struct input_keypad_info));
  330. keypad->event_cb = event_cb;
  331. os_delayed_work_init(&keypad->report_work, report_key_event_work_handle);
  332. os_delayed_work_init(&keypad->hold_key_work, check_hold_key_work_handle);
  333. os_delayed_work_init(&keypad->callback_work, callback_key_event_work_handle);
  334. keypad->adckey_handle = key_device_open(&keypad_scan_callback, ADC_KEY_DEVICE_NAME);
  335. if (!keypad->adckey_handle) {
  336. SYS_LOG_WRN("adckey devices open failed");
  337. }
  338. // if (!key_device_open(&keypad_scan_callback, TAP_KEY_DEVICE_NAME)) {
  339. // SYS_LOG_ERR("tapkey devices open failed");
  340. // }
  341. keypad->onoff_handle = key_device_open(&keypad_scan_callback, ONOFF_KEY_DEVICE_NAME);
  342. if (!keypad->onoff_handle) {
  343. SYS_LOG_WRN("onoffkey devices open failed");
  344. }
  345. keypad->gpiokey_handle = key_device_open(&keypad_scan_callback, GPIO_KEY_DEVICE_NAME);
  346. if (!keypad->gpiokey_handle) {
  347. SYS_LOG_WRN("gpiokey devices open failed");
  348. }
  349. keypad->long_press_timer = LONG_PRESS_TIMER;
  350. keypad->super_long_press_timer = SUPER_LONG_PRESS_TIMER;
  351. keypad->super_long_press_6s_timer = SUPER_LONG_PRESS_6S_TIMER;
  352. keypad->quickly_click_duration = QUICKLY_CLICK_DURATION;
  353. keypad->key_event_cancel_duration = KEY_EVENT_CANCEL_DURATION;
  354. keypad->hold_delay_time = HOLD_DELAY_TIME;
  355. keypad->hold_interval_time = HOLD_INTERVAL_DURATION;
  356. return 0;
  357. }
  358. void input_keypad_set_init_param(input_dev_init_param *data)
  359. {
  360. keypad->long_press_timer = data->long_press_timer;
  361. keypad->super_long_press_timer = data->super_long_press_timer;
  362. keypad->super_long_press_6s_timer = data->super_long_press_6s_timer;
  363. keypad->quickly_click_duration = data->quickly_click_duration;
  364. keypad->key_event_cancel_duration = data->key_event_cancel_duration;
  365. keypad->hold_delay_time = data->hold_delay_time;
  366. keypad->hold_interval_time = data->hold_interval_time;
  367. }
  368. int input_keypad_onoff_inquiry_enable(bool inner_onoff, bool enable)
  369. {
  370. if (!keypad)
  371. return -1;
  372. void* handle;
  373. if (inner_onoff)
  374. handle = keypad->onoff_handle;
  375. else
  376. handle = keypad->adckey_handle;
  377. if(!handle)
  378. return -1;
  379. keypad->press_type = 0;
  380. keypad->press_code = 0;
  381. keypad->report_press_code = 0;
  382. keypad->report_key_value = 0;
  383. keypad->callback_key_value = 0;
  384. keypad->press_timer = 0;
  385. keypad->click_num = 0;
  386. keypad->key_hold = 0;
  387. if (enable){
  388. key_device_abort_cb(handle);
  389. }else{
  390. key_device_enable_cb(handle);
  391. }
  392. return 0;
  393. }
  394. int input_keypad_onoff_inquiry(bool inner_onoff, int keycode)
  395. {
  396. if (!keypad)
  397. return -1;
  398. void* handle;
  399. if (inner_onoff)
  400. handle = keypad->onoff_handle;
  401. else
  402. handle = keypad->adckey_handle;
  403. if(!handle)
  404. return -1;
  405. struct input_value val = {0};
  406. if (!inner_onoff)
  407. {
  408. memset(&val, 0, sizeof(struct input_value));
  409. val.keypad.code = keycode;
  410. }
  411. key_device_inquiry(handle, &val);
  412. if (!inner_onoff)
  413. {
  414. SYS_LOG_ERR("keycode:0x%x, rkeycode:0x%x, value:%d",keycode,val.keypad.code,val.keypad.value);
  415. }
  416. if (val.keypad.code == keycode)
  417. {
  418. return val.keypad.value;
  419. }
  420. return KEY_VALUE_UP;
  421. }