it7259_tpkey_acts.c 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377
  1. /*
  2. * Copyright (c) 2017 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief TP Keyboard driver for Actions SoC
  9. */
  10. #include <errno.h>
  11. #include <kernel.h>
  12. #include <string.h>
  13. #include <init.h>
  14. #include <irq.h>
  15. #include <drivers/adc.h>
  16. #include <drivers/input/input_dev.h>
  17. #include <sys/util.h>
  18. #include <sys/byteorder.h>
  19. #include <board.h>
  20. #include <soc_pmu.h>
  21. #include <logging/log.h>
  22. #include <device.h>
  23. #include <drivers/gpio.h>
  24. #include <soc.h>
  25. #include <string.h>
  26. #include <drivers/i2c.h>
  27. #include <board_cfg.h>
  28. LOG_MODULE_REGISTER(tpkey, CONFIG_SYS_LOG_INPUT_DEV_LEVEL);
  29. #define tp_slaver_addr (0x8C>>1)
  30. #ifdef CONFIG_TPKEY_RESET_GPIO
  31. static const struct gpio_cfg reset_gpio_cfg = CONFIG_TPKEY_RESET_GPIO;
  32. #endif
  33. #ifdef CONFIG_TPKEY_POWER_GPIO
  34. static const struct gpio_cfg power_gpio_cfg = CONFIG_TPKEY_POWER_GPIO;
  35. #endif
  36. #ifdef CONFIG_TPKEY_ISR_GPIO
  37. static const struct gpio_cfg isr_gpio_cfg = CONFIG_TPKEY_ISR_GPIO;
  38. #endif
  39. #define CONFIG_USED_TP_WORK_QUEUE 1
  40. #ifdef CONFIG_USED_TP_WORK_QUEUE
  41. #define CONFIG_TP_WORK_Q_STACK_SIZE 1536
  42. struct k_work_q tp_drv_q;
  43. K_THREAD_STACK_DEFINE(tp_work_q_stack, CONFIG_TP_WORK_Q_STACK_SIZE);
  44. #endif
  45. struct acts_tpkey_data {
  46. input_notify_t notify;
  47. struct device *i2c_dev;
  48. struct device *gpio_dev;
  49. struct device *this_dev;
  50. struct gpio_callback key_gpio_cb;
  51. struct k_delayed_work timer;
  52. };
  53. struct acts_tpkey_config {
  54. uint16_t poll_interval_ms;
  55. uint32_t poll_total_ms;
  56. };
  57. struct k_timer tpkey_inquiry_timer;
  58. u8_t tpkey_flag_timeout;
  59. static struct acts_tpkey_data tpkey_acts_ddata;
  60. static const struct acts_tpkey_config tpkey_acts_cdata = {
  61. .poll_interval_ms = 0,
  62. .poll_total_ms = 0,
  63. };
  64. static void tpkey_acts_enable(struct device *dev);
  65. static void tpkey_acts_disable(struct device *dev);
  66. static void read_touch(struct input_value *val);
  67. void tpkey_detect(struct k_timer *timer)
  68. {
  69. LOG_DBG("go into tpkey_detect\n");
  70. tpkey_flag_timeout = 1;
  71. k_timer_stop(&tpkey_inquiry_timer);
  72. }
  73. static void KEY_IRQ_callback(struct device *port, struct gpio_callback *cb, uint32_t pins)
  74. {
  75. sys_trace_void(SYS_TRACE_ID_TP_IRQ);
  76. #if CONFIG_TPKEY_LOWPOWER
  77. LOG_DBG("inquiry mode\n");
  78. tpkey_flag_timeout = 0;
  79. k_timer_start(&tpkey_inquiry_timer, K_MSEC(20), K_MSEC(20));
  80. #endif
  81. #if CONFIG_TPKEY_READ_POLL_EN
  82. k_delayed_work_submit(&tpkey_acts_ddata.timer, K_NO_WAIT);
  83. #endif
  84. sys_trace_end_call(SYS_TRACE_ID_TP_IRQ);
  85. }
  86. static void tpkey_acts_poll(struct k_work *work)
  87. {
  88. struct acts_tpkey_data *tpkey = CONTAINER_OF(work, struct acts_tpkey_data, timer);
  89. struct device *dev = tpkey->this_dev;
  90. struct input_value val;
  91. read_touch(&val);
  92. tpkey->notify(dev,&val);
  93. k_delayed_work_cancel(&tpkey->timer);
  94. }
  95. static void tpkey_acts_enable(struct device *dev)
  96. {
  97. struct acts_tpkey_data *tpkey = dev->data;
  98. const struct acts_tpkey_config *cfg = dev->config;
  99. gpio_pin_interrupt_configure(tpkey->gpio_dev , isr_gpio_cfg.gpion, GPIO_INT_EDGE_FALLING);//GPIO_INT_DISABLE
  100. LOG_DBG("enable tpkey");
  101. }
  102. static void tpkey_acts_disable(struct device *dev)
  103. {
  104. struct acts_tpkey_data *tpkey = dev->data;
  105. const struct acts_tpkey_config *cfg = dev->config;
  106. gpio_pin_interrupt_configure(tpkey->gpio_dev , isr_gpio_cfg.gpion, GPIO_INT_DISABLE);//GPIO_INT_DISABLE
  107. LOG_DBG("disable tpkey");
  108. }
  109. static void tpkey_acts_inquiry(struct device *dev, struct input_value *val)
  110. {
  111. struct acts_tpkey_data *tpkey = dev->data;
  112. if(tpkey_flag_timeout)
  113. return;
  114. read_touch(val);
  115. LOG_DBG("inquiry tpkey");
  116. }
  117. static void tpkey_acts_register_notify(struct device *dev, input_notify_t notify)
  118. {
  119. struct acts_tpkey_data *tpkey = dev->data;
  120. LOG_DBG("register notify 0x%x", (uint32_t)notify);
  121. tpkey->notify = notify;
  122. }
  123. static void tpkey_acts_unregister_notify(struct device *dev, input_notify_t notify)
  124. {
  125. struct acts_tpkey_data *tpkey = dev->data;
  126. LOG_DBG("unregister notify 0x%x", (uint32_t)notify);
  127. tpkey->notify = NULL;
  128. }
  129. static void _tpkey_acts_get_capabilities(const struct device *dev,
  130. struct input_capabilities *capabilities)
  131. {
  132. capabilities->pointer.supported_gestures = INPUT_GESTURE_DIRECTION;
  133. }
  134. const struct input_dev_driver_api tpkey_acts_driver_api = {
  135. .enable = tpkey_acts_enable,
  136. .disable = tpkey_acts_disable,
  137. .inquiry = tpkey_acts_inquiry,
  138. .register_notify = tpkey_acts_register_notify,
  139. .unregister_notify = tpkey_acts_unregister_notify,
  140. .get_capabilities = _tpkey_acts_get_capabilities,
  141. };
  142. #define mode_pin (GPIO_PULL_UP | GPIO_INPUT | GPIO_INT_DEBOUNCE)
  143. void tp_reset(struct device *dev)
  144. {
  145. struct device *gpios_temp = NULL;
  146. gpios_temp = device_get_binding(reset_gpio_cfg.gpio_dev_name);
  147. gpio_pin_configure(gpios_temp , reset_gpio_cfg.gpion, GPIO_OUTPUT| GPIO_PULL_UP);
  148. gpio_pin_set_raw(gpios_temp , reset_gpio_cfg.gpion, 1);
  149. k_busy_wait(1000*1000);
  150. gpio_pin_set_raw(gpios_temp , reset_gpio_cfg.gpion, 0);
  151. k_busy_wait(10*1000);
  152. gpio_pin_set_raw(gpios_temp , reset_gpio_cfg.gpion, 1);
  153. }
  154. static uint8_t tp_enter_bootmode(struct device *dev)
  155. {
  156. struct acts_tpkey_data *tpkey = dev->data;
  157. uint8_t retrycnt = 10;
  158. tp_reset(dev);
  159. k_busy_wait(5*1000);
  160. while(retrycnt--){
  161. uint8_t cmd[4] = {0};
  162. cmd[0] = 0xA0;
  163. cmd[1] = 0x01;
  164. cmd[2] = 0xab;
  165. if(-1 == i2c_write(tpkey->i2c_dev,cmd, 3, 0X6A)) {
  166. k_busy_wait(4*1000);
  167. continue;
  168. }
  169. cmd[0] = 0xA0;
  170. cmd[1] = 0x03;
  171. if(-1 == i2c_write(tpkey->i2c_dev,cmd, 2, 0X6A)) {
  172. k_busy_wait(4*1000);
  173. continue;
  174. }
  175. if(-1 == i2c_read(tpkey->i2c_dev,cmd, 1, 0X6A)) {
  176. k_busy_wait(2*1000);
  177. continue;
  178. } else {
  179. if(cmd[0] != 0x55) {
  180. k_busy_wait(2*1000);
  181. continue;
  182. } else {
  183. return 0;
  184. }
  185. }
  186. }
  187. return -1;
  188. }
  189. #if 0
  190. void read_touch(struct input_value *val)
  191. {
  192. uint8_t cmd[5] = {0};
  193. cmd[0] = 0x02;
  194. if(-1 == i2c_write(tpkey_acts_ddata.i2c_dev, cmd, 1, tp_slaver_addr)) {
  195. return;
  196. } else if(-1 == i2c_read(tpkey_acts_ddata.i2c_dev, cmd, 5, tp_slaver_addr)) {//tp_slaver_addr
  197. return;
  198. }
  199. val->point.loc_x = (((uint16_t)(cmd[1]&0x0f))<<8) | cmd[2];
  200. val->point.loc_y = (((uint16_t)(cmd[3]&0x0f))<<8) | cmd[4];
  201. val->point.pessure_value = cmd[0];
  202. LOG_DBG("finger_num = %d, local:(%d,%d)\n",cmd[0], val->point.loc_x, val->point.loc_y);
  203. }
  204. #endif
  205. static void read_touch(struct input_value *val)
  206. {
  207. uint8_t cmd[14] = {0};
  208. sys_trace_void(SYS_TRACE_ID_TP_READ);
  209. cmd[0] = 0xE0;
  210. // if(-1 == i2c_read(tpkey_acts_ddata.i2c_dev, cmd, 14, tp_slaver_addr))
  211. // return;
  212. if(-1 == i2c_write(tpkey_acts_ddata.i2c_dev, cmd, 1, tp_slaver_addr)) {
  213. return;
  214. } else if(-1 == i2c_read(tpkey_acts_ddata.i2c_dev, cmd, 6, tp_slaver_addr)) {//tp_slaver_addr
  215. return;
  216. }
  217. val->point.loc_x = (((uint16_t)(cmd[3]&0x0f))<<8) | cmd[2];
  218. val->point.loc_y = (((uint16_t)(cmd[3]&0xf0))<<4) | cmd[4];
  219. val->point.pessure_value = cmd[1];
  220. val->point.gesture = cmd[0];
  221. LOG_DBG("finger_num = %d, gesture = %d,local:(%d,%d)\n",cmd[0], val->point.gesture,val->point.loc_x, val->point.loc_y);
  222. sys_trace_end_call(SYS_TRACE_ID_TP_READ);
  223. }
  224. static void enter_low_power_mode(const struct device *dev)
  225. {
  226. uint8_t cmd[6] = {0};
  227. printk("go into low power mode\n");
  228. cmd[0] = 0xfe;
  229. cmd[1] = 0;
  230. i2c_write(tpkey_acts_ddata.i2c_dev, cmd, 2, tp_slaver_addr);
  231. }
  232. static void turn_off_detector_mode()
  233. {
  234. uint8_t cmd[6] = {0};
  235. cmd[0] = 0xfb;
  236. cmd[1] = 0;
  237. i2c_write(tpkey_acts_ddata.i2c_dev, cmd, 2, tp_slaver_addr);
  238. cmd[0] = 0xfc;
  239. cmd[1] = 0;
  240. i2c_write(tpkey_acts_ddata.i2c_dev, cmd, 2, tp_slaver_addr);
  241. }
  242. int tpkey_acts_init(const struct device *dev)
  243. {
  244. struct acts_tpkey_data *tpkey = dev->data;
  245. const struct acts_tpkey_config *cfg = dev->config;
  246. struct device *gpios_power = NULL;
  247. #ifdef CONFIG_TPKEY_POWER_GPIO
  248. gpios_power = device_get_binding(power_gpio_cfg.gpio_dev_name);
  249. gpio_pin_configure(gpios_power, power_gpio_cfg.gpion, GPIO_OUTPUT| GPIO_PULL_UP);
  250. gpio_pin_set_raw(gpios_power, reset_gpio_cfg.gpion, 0);
  251. #endif
  252. tpkey->i2c_dev = (struct device *)device_get_binding(CONFIG_TPKEY_I2C_NAME);
  253. tpkey->gpio_dev = (struct device *)device_get_binding(isr_gpio_cfg.gpio_dev_name);
  254. if(tpkey->i2c_dev == NULL) {
  255. printk("can not access right i2c device\n");
  256. return -1;
  257. }
  258. if(tpkey->gpio_dev == NULL) {
  259. printk("can not access right gpio device\n");
  260. return -1;
  261. }
  262. tpkey->this_dev = (struct device *)dev;
  263. gpio_pin_configure(tpkey->gpio_dev , isr_gpio_cfg.gpion, mode_pin);
  264. gpio_init_callback(&tpkey->key_gpio_cb , KEY_IRQ_callback, BIT(isr_gpio_cfg.gpion));
  265. gpio_add_callback(tpkey->gpio_dev , &tpkey->key_gpio_cb);
  266. //if(-1 == tp_enter_bootmode(dev)) {
  267. // printk("error return \n");
  268. //}
  269. tpkey_flag_timeout = 0;
  270. tp_reset(dev);
  271. k_busy_wait(100*1000);//the time waiting too short,and the device can not wake up
  272. #if CONFIG_TPKEY_LOWPOWER
  273. // enter_low_power_mode(dev);
  274. k_timer_init(&tpkey_inquiry_timer, tpkey_detect, NULL);
  275. k_timer_start(&tpkey_inquiry_timer, K_MSEC(20), K_MSEC(20));
  276. #endif
  277. // turn_off_detector_mode();
  278. k_delayed_work_init(&tpkey->timer, tpkey_acts_poll);
  279. return 0;
  280. }
  281. #if IS_ENABLED(CONFIG_TPKEY)
  282. DEVICE_DEFINE(tpkey, CONFIG_TPKEY_DEV_NAME, tpkey_acts_init,
  283. NULL, &tpkey_acts_ddata, &tpkey_acts_cdata, POST_KERNEL,
  284. 60, &tpkey_acts_driver_api);
  285. #endif