gpio-104-idio-16.c 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. /*
  2. * GPIO driver for the ACCES 104-IDIO-16 family
  3. * Copyright (C) 2015 William Breathitt Gray
  4. *
  5. * This program is free software; you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License, version 2, as
  7. * published by the Free Software Foundation.
  8. *
  9. * This program is distributed in the hope that it will be useful, but
  10. * WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  12. * General Public License for more details.
  13. */
  14. #include <linux/bitops.h>
  15. #include <linux/device.h>
  16. #include <linux/errno.h>
  17. #include <linux/gpio/driver.h>
  18. #include <linux/io.h>
  19. #include <linux/ioport.h>
  20. #include <linux/interrupt.h>
  21. #include <linux/irqdesc.h>
  22. #include <linux/kernel.h>
  23. #include <linux/module.h>
  24. #include <linux/moduleparam.h>
  25. #include <linux/platform_device.h>
  26. #include <linux/spinlock.h>
  27. static unsigned idio_16_base;
  28. module_param(idio_16_base, uint, 0);
  29. MODULE_PARM_DESC(idio_16_base, "ACCES 104-IDIO-16 base address");
  30. static unsigned idio_16_irq;
  31. module_param(idio_16_irq, uint, 0);
  32. MODULE_PARM_DESC(idio_16_irq, "ACCES 104-IDIO-16 interrupt line number");
  33. /**
  34. * struct idio_16_gpio - GPIO device private data structure
  35. * @chip: instance of the gpio_chip
  36. * @lock: synchronization lock to prevent I/O race conditions
  37. * @irq_mask: I/O bits affected by interrupts
  38. * @base: base port address of the GPIO device
  39. * @irq: Interrupt line number
  40. * @out_state: output bits state
  41. */
  42. struct idio_16_gpio {
  43. struct gpio_chip chip;
  44. spinlock_t lock;
  45. unsigned long irq_mask;
  46. unsigned base;
  47. unsigned irq;
  48. unsigned out_state;
  49. };
  50. static int idio_16_gpio_get_direction(struct gpio_chip *chip, unsigned offset)
  51. {
  52. if (offset > 15)
  53. return 1;
  54. return 0;
  55. }
  56. static int idio_16_gpio_direction_input(struct gpio_chip *chip, unsigned offset)
  57. {
  58. return 0;
  59. }
  60. static int idio_16_gpio_direction_output(struct gpio_chip *chip,
  61. unsigned offset, int value)
  62. {
  63. chip->set(chip, offset, value);
  64. return 0;
  65. }
  66. static int idio_16_gpio_get(struct gpio_chip *chip, unsigned offset)
  67. {
  68. struct idio_16_gpio *const idio16gpio = gpiochip_get_data(chip);
  69. const unsigned mask = BIT(offset-16);
  70. if (offset < 16)
  71. return -EINVAL;
  72. if (offset < 24)
  73. return !!(inb(idio16gpio->base + 1) & mask);
  74. return !!(inb(idio16gpio->base + 5) & (mask>>8));
  75. }
  76. static void idio_16_gpio_set(struct gpio_chip *chip, unsigned offset, int value)
  77. {
  78. struct idio_16_gpio *const idio16gpio = gpiochip_get_data(chip);
  79. const unsigned mask = BIT(offset);
  80. unsigned long flags;
  81. if (offset > 15)
  82. return;
  83. spin_lock_irqsave(&idio16gpio->lock, flags);
  84. if (value)
  85. idio16gpio->out_state |= mask;
  86. else
  87. idio16gpio->out_state &= ~mask;
  88. if (offset > 7)
  89. outb(idio16gpio->out_state >> 8, idio16gpio->base + 4);
  90. else
  91. outb(idio16gpio->out_state, idio16gpio->base);
  92. spin_unlock_irqrestore(&idio16gpio->lock, flags);
  93. }
  94. static void idio_16_irq_ack(struct irq_data *data)
  95. {
  96. }
  97. static void idio_16_irq_mask(struct irq_data *data)
  98. {
  99. struct gpio_chip *chip = irq_data_get_irq_chip_data(data);
  100. struct idio_16_gpio *const idio16gpio = gpiochip_get_data(chip);
  101. const unsigned long mask = BIT(irqd_to_hwirq(data));
  102. unsigned long flags;
  103. idio16gpio->irq_mask &= ~mask;
  104. if (!idio16gpio->irq_mask) {
  105. spin_lock_irqsave(&idio16gpio->lock, flags);
  106. outb(0, idio16gpio->base + 2);
  107. spin_unlock_irqrestore(&idio16gpio->lock, flags);
  108. }
  109. }
  110. static void idio_16_irq_unmask(struct irq_data *data)
  111. {
  112. struct gpio_chip *chip = irq_data_get_irq_chip_data(data);
  113. struct idio_16_gpio *const idio16gpio = gpiochip_get_data(chip);
  114. const unsigned long mask = BIT(irqd_to_hwirq(data));
  115. const unsigned long prev_irq_mask = idio16gpio->irq_mask;
  116. unsigned long flags;
  117. idio16gpio->irq_mask |= mask;
  118. if (!prev_irq_mask) {
  119. spin_lock_irqsave(&idio16gpio->lock, flags);
  120. inb(idio16gpio->base + 2);
  121. spin_unlock_irqrestore(&idio16gpio->lock, flags);
  122. }
  123. }
  124. static int idio_16_irq_set_type(struct irq_data *data, unsigned flow_type)
  125. {
  126. /* The only valid irq types are none and both-edges */
  127. if (flow_type != IRQ_TYPE_NONE &&
  128. (flow_type & IRQ_TYPE_EDGE_BOTH) != IRQ_TYPE_EDGE_BOTH)
  129. return -EINVAL;
  130. return 0;
  131. }
  132. static struct irq_chip idio_16_irqchip = {
  133. .name = "104-idio-16",
  134. .irq_ack = idio_16_irq_ack,
  135. .irq_mask = idio_16_irq_mask,
  136. .irq_unmask = idio_16_irq_unmask,
  137. .irq_set_type = idio_16_irq_set_type
  138. };
  139. static irqreturn_t idio_16_irq_handler(int irq, void *dev_id)
  140. {
  141. struct idio_16_gpio *const idio16gpio = dev_id;
  142. struct gpio_chip *const chip = &idio16gpio->chip;
  143. int gpio;
  144. for_each_set_bit(gpio, &idio16gpio->irq_mask, chip->ngpio)
  145. generic_handle_irq(irq_find_mapping(chip->irqdomain, gpio));
  146. spin_lock(&idio16gpio->lock);
  147. outb(0, idio16gpio->base + 1);
  148. spin_unlock(&idio16gpio->lock);
  149. return IRQ_HANDLED;
  150. }
  151. static int __init idio_16_probe(struct platform_device *pdev)
  152. {
  153. struct device *dev = &pdev->dev;
  154. struct idio_16_gpio *idio16gpio;
  155. const unsigned base = idio_16_base;
  156. const unsigned extent = 8;
  157. const char *const name = dev_name(dev);
  158. int err;
  159. const unsigned irq = idio_16_irq;
  160. idio16gpio = devm_kzalloc(dev, sizeof(*idio16gpio), GFP_KERNEL);
  161. if (!idio16gpio)
  162. return -ENOMEM;
  163. if (!devm_request_region(dev, base, extent, name)) {
  164. dev_err(dev, "Unable to lock port addresses (0x%X-0x%X)\n",
  165. base, base + extent);
  166. return -EBUSY;
  167. }
  168. idio16gpio->chip.label = name;
  169. idio16gpio->chip.parent = dev;
  170. idio16gpio->chip.owner = THIS_MODULE;
  171. idio16gpio->chip.base = -1;
  172. idio16gpio->chip.ngpio = 32;
  173. idio16gpio->chip.get_direction = idio_16_gpio_get_direction;
  174. idio16gpio->chip.direction_input = idio_16_gpio_direction_input;
  175. idio16gpio->chip.direction_output = idio_16_gpio_direction_output;
  176. idio16gpio->chip.get = idio_16_gpio_get;
  177. idio16gpio->chip.set = idio_16_gpio_set;
  178. idio16gpio->base = base;
  179. idio16gpio->irq = irq;
  180. idio16gpio->out_state = 0xFFFF;
  181. spin_lock_init(&idio16gpio->lock);
  182. dev_set_drvdata(dev, idio16gpio);
  183. err = gpiochip_add_data(&idio16gpio->chip, idio16gpio);
  184. if (err) {
  185. dev_err(dev, "GPIO registering failed (%d)\n", err);
  186. return err;
  187. }
  188. /* Disable IRQ by default */
  189. outb(0, base + 2);
  190. outb(0, base + 1);
  191. err = gpiochip_irqchip_add(&idio16gpio->chip, &idio_16_irqchip, 0,
  192. handle_edge_irq, IRQ_TYPE_NONE);
  193. if (err) {
  194. dev_err(dev, "Could not add irqchip (%d)\n", err);
  195. goto err_gpiochip_remove;
  196. }
  197. err = request_irq(irq, idio_16_irq_handler, 0, name, idio16gpio);
  198. if (err) {
  199. dev_err(dev, "IRQ handler registering failed (%d)\n", err);
  200. goto err_gpiochip_remove;
  201. }
  202. return 0;
  203. err_gpiochip_remove:
  204. gpiochip_remove(&idio16gpio->chip);
  205. return err;
  206. }
  207. static int idio_16_remove(struct platform_device *pdev)
  208. {
  209. struct idio_16_gpio *const idio16gpio = platform_get_drvdata(pdev);
  210. free_irq(idio16gpio->irq, idio16gpio);
  211. gpiochip_remove(&idio16gpio->chip);
  212. return 0;
  213. }
  214. static struct platform_device *idio_16_device;
  215. static struct platform_driver idio_16_driver = {
  216. .driver = {
  217. .name = "104-idio-16"
  218. },
  219. .remove = idio_16_remove
  220. };
  221. static void __exit idio_16_exit(void)
  222. {
  223. platform_device_unregister(idio_16_device);
  224. platform_driver_unregister(&idio_16_driver);
  225. }
  226. static int __init idio_16_init(void)
  227. {
  228. int err;
  229. idio_16_device = platform_device_alloc(idio_16_driver.driver.name, -1);
  230. if (!idio_16_device)
  231. return -ENOMEM;
  232. err = platform_device_add(idio_16_device);
  233. if (err)
  234. goto err_platform_device;
  235. err = platform_driver_probe(&idio_16_driver, idio_16_probe);
  236. if (err)
  237. goto err_platform_driver;
  238. return 0;
  239. err_platform_driver:
  240. platform_device_del(idio_16_device);
  241. err_platform_device:
  242. platform_device_put(idio_16_device);
  243. return err;
  244. }
  245. module_init(idio_16_init);
  246. module_exit(idio_16_exit);
  247. MODULE_AUTHOR("William Breathitt Gray <vilhelm.gray@gmail.com>");
  248. MODULE_DESCRIPTION("ACCES 104-IDIO-16 GPIO driver");
  249. MODULE_LICENSE("GPL v2");