gpio_et6416.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495
  1. /*
  2. * Copyright (c) 2018 Justin Watson
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <errno.h>
  7. #include <kernel.h>
  8. #include <device.h>
  9. #include <init.h>
  10. #include <drivers/gpio.h>
  11. #include <drivers/i2c.h>
  12. #include <board_cfg.h>
  13. #include "gpio_utils.h"
  14. #include <logging/log.h>
  15. //LOG_MODULE_REGISTER(et6416, CONFIG_GPIO_LOG_LEVEL);
  16. LOG_MODULE_REGISTER(et6416, LOG_LEVEL_DBG);
  17. #include "gpio_utils.h"
  18. /* Number of pins supported by the device */
  19. #define NUM_PINS 16
  20. /* Max to select all pins supported on the device. */
  21. #define ALL_PINS ((u16_t)BIT_MASK(NUM_PINS))
  22. /* Reset delay is 2.5 ms, round up for Zephyr resolution */
  23. #define RESET_DELAY_MS 3
  24. #define ET6416_EDGE_FALLING 1
  25. #define ET6416_EDGE_RISING 2
  26. #define ET6416_EDGE_BOTH 3
  27. /** Cache of the output configuration and data of the pins. */
  28. struct et6416_pin_state {
  29. u16_t input_dat; /* 0x00 */
  30. u16_t output_data; /* 0x02 */
  31. u16_t polarity; /* 0x04 */
  32. u16_t dir; /* 0x06, 1=input 0= output */
  33. } ;
  34. struct et6416_irq_state {
  35. u16_t interrupt_mask; /* */
  36. u32_t interrupt_sense; /* */
  37. };
  38. /** Runtime driver data */
  39. struct et6416_drv_data {
  40. /* gpio_driver_data needs to be first */
  41. struct gpio_driver_data common;
  42. const struct device *i2c_master;
  43. struct et6416_pin_state pin_state;
  44. struct k_sem lock;
  45. #ifdef CONFIG_GPIO_ET6416_INTERRUPT
  46. struct device *gpio_int;
  47. struct gpio_callback gpio_cb;
  48. struct k_work work;
  49. struct et6416_irq_state irq_state;
  50. struct device *dev;
  51. /* user ISR cb */
  52. sys_slist_t cb;
  53. /* Enabled INT pins generating a cb */
  54. u16_t cb_pins;
  55. #endif /* CONFIG_GPIO_ET6416_INTERRUPT */
  56. };
  57. /** Configuration data */
  58. struct et6416_config {
  59. /* gpio_driver_config needs to be first */
  60. struct gpio_driver_config common;
  61. const char *i2c_master_dev_name;
  62. #ifdef CONFIG_GPIO_ET6416_INTERRUPT
  63. struct gpio_cfg et6416_irq;
  64. #endif /* CONFIG_GPIO_ET6416_INTERRUPT */
  65. u16_t i2c_slave_addr;
  66. };
  67. /* Pin configuration register addresses */
  68. enum {
  69. ET6416_REG_INPUT = 0x00,
  70. ET6416_REG_OUTPUT = 0x02,
  71. ET6416_REG_POLARITY_INVERSION = 0x04,
  72. ET6416_REG_CFG = 0x06,
  73. };
  74. /**
  75. * @brief Write a word to an internal address of an I2C slave.
  76. *
  77. * @param dev Pointer to the device structure for the driver instance.
  78. * @param dev_addr Address of the I2C device for writing.
  79. * @param reg_addr Address of the internal register being written.
  80. * @param value Value to be written to internal register.
  81. *
  82. * @retval 0 If successful.
  83. * @retval -EIO General input / output error.
  84. */
  85. static inline int i2c_reg_write_word(const struct device *dev, u16_t dev_addr,
  86. u8_t reg_addr, u16_t value)
  87. {
  88. u8_t tx_buf[3] = { reg_addr, value & 0xff , value >> 8};
  89. return i2c_write(dev, tx_buf, 3, dev_addr);
  90. }
  91. static inline int i2c_reg_read_word(const struct device *dev, u16_t dev_addr,
  92. u8_t reg_addr, u16_t *value)
  93. {
  94. return i2c_burst_read(dev, dev_addr,reg_addr, (u8_t *)value, 2);
  95. }
  96. #ifdef CONFIG_GPIO_ET6416_INTERRUPT
  97. static int et6416_handle_interrupt(void *arg)
  98. {
  99. struct device *dev = (struct device *) arg;
  100. //const struct et6416_config *cfg = dev->config_info;
  101. struct et6416_drv_data *drv_data = dev->data;
  102. u16_t pin_data;
  103. int ret = 0;
  104. u16_t int_source;
  105. k_sem_take(&drv_data->lock, K_FOREVER);
  106. ret = i2c_reg_read_word(drv_data->i2c_master, cfg->i2c_slave_addr,
  107. ET6416_REG_INPUT,
  108. &pin_data);
  109. if (ret != 0) {
  110. goto out;
  111. }
  112. out:
  113. k_sem_give(&drv_data->lock);
  114. if ((ret == 0)
  115. && ((int_source & drv_data->cb_pins) != 0)) {
  116. gpio_fire_callbacks(&drv_data->cb, dev, int_source);
  117. }
  118. return ret;
  119. }
  120. static void et6416_work_handler(struct k_work *work)
  121. {
  122. struct et6416_drv_data *drv_data =
  123. CONTAINER_OF(work, struct et6416_drv_data, work);
  124. et6416_handle_interrupt(drv_data->dev);
  125. }
  126. static void et6416_int_cb(struct device *dev, struct gpio_callback *gpio_cb,
  127. u32_t pins)
  128. {
  129. struct et6416_drv_data *drv_data = CONTAINER_OF(gpio_cb,
  130. struct et6416_drv_data, gpio_cb);
  131. ARG_UNUSED(pins);
  132. k_work_submit(&drv_data->work);
  133. }
  134. #endif
  135. static int gpio_et6416_config(const struct device *dev,
  136. gpio_pin_t pin,
  137. gpio_flags_t flags)
  138. {
  139. const struct et6416_config *cfg = dev->config;
  140. struct et6416_drv_data *drv_data = dev->data;
  141. struct et6416_pin_state *pins = &drv_data->pin_state;
  142. int rc ;
  143. /* Can't do I2C bus operations from an ISR */
  144. if (k_is_in_isr()) {
  145. return -EWOULDBLOCK;
  146. }
  147. k_sem_take(&drv_data->lock, K_FOREVER);
  148. if (flags & GPIO_OUTPUT) {
  149. pins->dir &= ~BIT(pin);
  150. }else {
  151. pins->dir |= BIT(pin); // INPUT
  152. }
  153. rc = i2c_reg_write_word(drv_data->i2c_master, cfg->i2c_slave_addr,
  154. ET6416_REG_CFG, pins->dir);
  155. if (rc) {
  156. LOG_ERR("dir write err=%d\n",rc);
  157. }else{
  158. LOG_DBG("read dir=0x%x\n", drv_data->pin_state.dir);
  159. }
  160. k_sem_give(&drv_data->lock);
  161. return rc;
  162. }
  163. static int port_get(const struct device *dev,
  164. gpio_port_value_t *value)
  165. {
  166. const struct et6416_config *cfg = dev->config;
  167. struct et6416_drv_data *drv_data = dev->data;
  168. u16_t pin_data;
  169. int rc = 0;
  170. /* Can't do I2C bus operations from an ISR */
  171. if (k_is_in_isr()) {
  172. return -EWOULDBLOCK;
  173. }
  174. k_sem_take(&drv_data->lock, K_FOREVER);
  175. rc = i2c_reg_read_word(drv_data->i2c_master, cfg->i2c_slave_addr,
  176. ET6416_REG_INPUT,
  177. &pin_data);
  178. LOG_DBG("read %04x got %d", pin_data, rc);
  179. if (rc != 0) {
  180. LOG_ERR("read err=%d\n",rc);
  181. }else{
  182. *value = pin_data;
  183. drv_data->pin_state.input_dat = pin_data;
  184. }
  185. k_sem_give(&drv_data->lock);
  186. return rc;
  187. }
  188. static int port_write(const struct device *dev,
  189. gpio_port_pins_t mask,
  190. gpio_port_value_t value,
  191. gpio_port_value_t toggle)
  192. {
  193. const struct et6416_config *cfg = dev->config;
  194. struct et6416_drv_data *drv_data = dev->data;
  195. u16_t *outp = &drv_data->pin_state.output_data;
  196. /* Can't do I2C bus operations from an ISR */
  197. if (k_is_in_isr()) {
  198. return -EWOULDBLOCK;
  199. }
  200. k_sem_take(&drv_data->lock, K_FOREVER);
  201. u16_t orig_out = *outp;
  202. u16_t out = ((orig_out & ~mask) | (value & mask)) ^ toggle;
  203. int rc = i2c_reg_write_word(drv_data->i2c_master, cfg->i2c_slave_addr,
  204. ET6416_REG_OUTPUT, out);
  205. if (rc == 0) {
  206. *outp = out;
  207. }else{
  208. LOG_ERR("write err=%d\n",rc);
  209. }
  210. k_sem_give(&drv_data->lock);
  211. LOG_DBG("write %04x msk %04x val %04x => %04x: %d", orig_out, mask, value, out, rc);
  212. return rc;
  213. }
  214. static int port_set_masked(const struct device *dev,
  215. gpio_port_pins_t mask,
  216. gpio_port_value_t value)
  217. {
  218. return port_write(dev, mask, value, 0);
  219. }
  220. static int port_set_bits(const struct device *dev,
  221. gpio_port_pins_t pins)
  222. {
  223. return port_write(dev, pins, pins, 0);
  224. }
  225. static int port_clear_bits(const struct device *dev,
  226. gpio_port_pins_t pins)
  227. {
  228. return port_write(dev, pins, 0, 0);
  229. }
  230. static int port_toggle_bits(const struct device *dev,
  231. gpio_port_pins_t pins)
  232. {
  233. return port_write(dev, 0, 0, pins);
  234. }
  235. static int pin_interrupt_configure(const struct device *dev,
  236. gpio_pin_t pin,
  237. enum gpio_int_mode mode,
  238. enum gpio_int_trig trig)
  239. {
  240. int rc = 0;
  241. if (!IS_ENABLED(CONFIG_GPIO_ET6416_INTERRUPT)
  242. && (mode != GPIO_INT_MODE_DISABLED)) {
  243. return -ENOTSUP;
  244. }
  245. #ifdef CONFIG_GPIO_ET6416_INTERRUPT
  246. const struct et6416_config *cfg = dev->config;
  247. struct et6416_drv_data *drv_data = dev->data;
  248. struct et6416_irq_state *irq = &drv_data->irq_state;
  249. k_sem_take(&drv_data->lock, K_FOREVER);
  250. if (mode == GPIO_INT_MODE_DISABLED) {
  251. drv_data->cb_pins &= ~BIT(pin);
  252. irq->interrupt_mask |= BIT(pin);
  253. } else { /* GPIO_INT_MODE_EDGE */
  254. drv_data->cb_pins |= BIT(pin);
  255. irq->interrupt_mask &= ~BIT(pin);
  256. if (trig == GPIO_INT_TRIG_BOTH) {
  257. irq->interrupt_sense |= (ET6416_EDGE_BOTH <<
  258. (pin * 2));
  259. } else if (trig == GPIO_INT_TRIG_LOW) {
  260. irq->interrupt_sense |= (ET6416_EDGE_FALLING <<
  261. (pin * 2));
  262. } else if (trig == GPIO_INT_TRIG_HIGH) {
  263. irq->interrupt_sense |= (ET6416_EDGE_RISING <<
  264. (pin * 2));
  265. }
  266. }
  267. k_sem_give(&drv_data->lock);
  268. #endif /* CONFIG_GPIO_ET6416_INTERRUPT */
  269. return rc;
  270. }
  271. /**
  272. * @brief Initialization function of ET6416
  273. *
  274. * @param dev Device struct
  275. * @return 0 if successful, failed otherwise.
  276. */
  277. static int vdd18v_power_on(void)
  278. {
  279. const struct device *gpio_dev;
  280. gpio_dev = device_get_binding(CONFIG_GPIO_PIN2NAME(CONFIG_GPIO_VDD18V_POWER_ON));
  281. if (!gpio_dev){
  282. printk("%d, vdd18 gpio dev fail!\n", __LINE__);
  283. return -EINVAL;
  284. }
  285. printk("%d, vdd18 poweron!\n", __LINE__);
  286. //gpio_pin_configure(gpio_dev, CONFIG_GPIO_VDD18V_POWER_ON%32, GPIO_OUTPUT | GPIO_PULL_UP);
  287. gpio_pin_configure(gpio_dev, CONFIG_GPIO_VDD18V_POWER_ON%32, GPIO_OUTPUT);
  288. gpio_pin_set(gpio_dev, CONFIG_GPIO_VDD18V_POWER_ON%32, 1);
  289. return 0;
  290. }
  291. static int et6416_init(const struct device *dev)
  292. {
  293. const struct et6416_config *cfg = dev->config;
  294. struct et6416_drv_data *drv_data = dev->data;
  295. int rc;
  296. printk("gpio:et6416_init\n");
  297. vdd18v_power_on();
  298. drv_data->i2c_master = device_get_binding(cfg->i2c_master_dev_name);
  299. if (!drv_data->i2c_master) {
  300. LOG_ERR("%s: no bus %s", dev->name,
  301. cfg->i2c_master_dev_name);
  302. return -EINVAL;
  303. }
  304. #ifdef CONFIG_GPIO_ET6416_INTERRUPT
  305. drv_data->dev = dev;
  306. if(!cfg->et6416_irq.use_gpio){
  307. LOG_ERR("gpio irq not cfg\n");
  308. return -EINVAL
  309. }
  310. drv_data->gpio_int = device_get_binding(cfg->et6416_irq.gpio_dev_name);
  311. if (!drv_data->gpio_int) {
  312. rc = -ENOTSUP;
  313. goto out;
  314. }
  315. k_work_init(&drv_data->work, et6416_work_handler);
  316. gpio_pin_configure(drv_data->gpio_int, cfg->et6416_irq.gpion,
  317. GPIO_INPUT | cfg->et6416_irq.flag);
  318. gpio_pin_interrupt_configure(drv_data->gpio_int, cfg->et6416_irq.gpion,
  319. GPIO_INT_EDGE_TO_ACTIVE);
  320. gpio_init_callback(&drv_data->gpio_cb, et6416_int_cb,
  321. BIT(cfg->et6416_irq.gpion));
  322. gpio_add_callback(drv_data->gpio_int, &drv_data->gpio_cb);
  323. #endif
  324. //rc = i2c_reg_write_word(drv_data->i2c_master, cfg->i2c_slave_addr,
  325. //ET6416_REG_CFG,0x0000); //output
  326. //LOG_DBG("write cfg 0x0000 ,rc=%d\n", rc);
  327. rc = i2c_reg_read_word(drv_data->i2c_master, cfg->i2c_slave_addr,
  328. ET6416_REG_CFG, &drv_data->pin_state.dir);
  329. LOG_DBG("read cfg=0x%x,rc=%d\n", drv_data->pin_state.dir, rc);
  330. rc = i2c_reg_read_word(drv_data->i2c_master, cfg->i2c_slave_addr,
  331. ET6416_REG_POLARITY_INVERSION, &drv_data->pin_state.polarity);
  332. LOG_DBG("read POLARITY=0x%x,rc=%d\n", drv_data->pin_state.polarity, rc);
  333. rc = i2c_reg_read_word(drv_data->i2c_master, cfg->i2c_slave_addr,
  334. ET6416_REG_INPUT, &drv_data->pin_state.input_dat);
  335. LOG_DBG("read indata=0x%x,rc=%d\n", drv_data->pin_state.input_dat, rc);
  336. rc = i2c_reg_read_word(drv_data->i2c_master, cfg->i2c_slave_addr,
  337. ET6416_REG_OUTPUT, &drv_data->pin_state.output_data);
  338. LOG_DBG("read outdata=0x%x,rc=%d\n", drv_data->pin_state.output_data, rc);
  339. //out:
  340. if (rc != 0) {
  341. LOG_ERR("%s init failed: %d", dev->name, rc);
  342. } else {
  343. LOG_INF("%s init ok", dev->name);
  344. }
  345. return rc;
  346. }
  347. #ifdef CONFIG_GPIO_ET6416_INTERRUPT
  348. static int gpio_et6416_manage_callback(const struct device *dev,
  349. struct gpio_callback *callback,
  350. bool set)
  351. {
  352. struct et6416_drv_data *data = dev->data;
  353. return gpio_manage_callback(&data->cb, callback, set);
  354. }
  355. static int gpio_et6416_enable_callback(const struct device *dev,
  356. gpio_pin_t pin)
  357. {
  358. struct et6416_drv_data *data = dev->data;
  359. data->cb_pins |= BIT(pin);
  360. return 0;
  361. }
  362. static int gpio_et6416_disable_callback(const struct device *dev,
  363. gpio_pin_t pin)
  364. {
  365. struct et6416_drv_data *data = dev->data;
  366. data->cb_pins &= ~BIT(pin);
  367. return 0;
  368. }
  369. #endif
  370. static const struct gpio_driver_api gpio_api_table = {
  371. .pin_configure = gpio_et6416_config,
  372. .port_get_raw = port_get,
  373. .port_set_masked_raw = port_set_masked,
  374. .port_set_bits_raw = port_set_bits,
  375. .port_clear_bits_raw = port_clear_bits,
  376. .port_toggle_bits = port_toggle_bits,
  377. .pin_interrupt_configure = pin_interrupt_configure,
  378. #ifdef CONFIG_GPIO_ET6416_INTERRUPT
  379. .manage_callback = gpio_et6416_manage_callback,
  380. .enable_callback = gpio_et6416_enable_callback,
  381. .disable_callback = gpio_et6416_disable_callback,
  382. #endif
  383. };
  384. static const struct et6416_config et6416_cfg = {
  385. .common = {
  386. .port_pin_mask = 0xffff,
  387. },
  388. .i2c_master_dev_name = CONFIG_EXTEND_GPIO_I2C_NAME,
  389. #ifdef CONFIG_GPIO_ET6416_INTERRUPT
  390. .et6416_irq = CONFIG_EXTEND_GPIO_ISR_GPIO,
  391. #endif
  392. .i2c_slave_addr = 0x20,
  393. };
  394. static struct et6416_drv_data et6416_drvdata = {
  395. .lock = Z_SEM_INITIALIZER(et6416_drvdata.lock, 1, 1),
  396. };
  397. #if IS_ENABLED(CONFIG_EXTEND_GPIO)
  398. DEVICE_DEFINE(et6416, CONFIG_EXTEND_GPIO_NAME,
  399. et6416_init, NULL, &et6416_drvdata, &et6416_cfg,
  400. POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_OBJECTS,
  401. &gpio_api_table);
  402. #endif