gpio-etraxfs.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. #include <linux/kernel.h>
  2. #include <linux/init.h>
  3. #include <linux/gpio.h>
  4. #include <linux/of_gpio.h>
  5. #include <linux/io.h>
  6. #include <linux/platform_device.h>
  7. #include <linux/basic_mmio_gpio.h>
  8. #define ETRAX_FS_rw_pa_dout 0
  9. #define ETRAX_FS_r_pa_din 4
  10. #define ETRAX_FS_rw_pa_oe 8
  11. #define ETRAX_FS_rw_intr_cfg 12
  12. #define ETRAX_FS_rw_intr_mask 16
  13. #define ETRAX_FS_rw_ack_intr 20
  14. #define ETRAX_FS_r_intr 24
  15. #define ETRAX_FS_rw_pb_dout 32
  16. #define ETRAX_FS_r_pb_din 36
  17. #define ETRAX_FS_rw_pb_oe 40
  18. #define ETRAX_FS_rw_pc_dout 48
  19. #define ETRAX_FS_r_pc_din 52
  20. #define ETRAX_FS_rw_pc_oe 56
  21. #define ETRAX_FS_rw_pd_dout 64
  22. #define ETRAX_FS_r_pd_din 68
  23. #define ETRAX_FS_rw_pd_oe 72
  24. #define ETRAX_FS_rw_pe_dout 80
  25. #define ETRAX_FS_r_pe_din 84
  26. #define ETRAX_FS_rw_pe_oe 88
  27. struct etraxfs_gpio_port {
  28. const char *label;
  29. unsigned int oe;
  30. unsigned int dout;
  31. unsigned int din;
  32. unsigned int ngpio;
  33. };
  34. struct etraxfs_gpio_info {
  35. unsigned int num_ports;
  36. const struct etraxfs_gpio_port *ports;
  37. };
  38. static const struct etraxfs_gpio_port etraxfs_gpio_etraxfs_ports[] = {
  39. {
  40. .label = "A",
  41. .ngpio = 8,
  42. .oe = ETRAX_FS_rw_pa_oe,
  43. .dout = ETRAX_FS_rw_pa_dout,
  44. .din = ETRAX_FS_r_pa_din,
  45. },
  46. {
  47. .label = "B",
  48. .ngpio = 18,
  49. .oe = ETRAX_FS_rw_pb_oe,
  50. .dout = ETRAX_FS_rw_pb_dout,
  51. .din = ETRAX_FS_r_pb_din,
  52. },
  53. {
  54. .label = "C",
  55. .ngpio = 18,
  56. .oe = ETRAX_FS_rw_pc_oe,
  57. .dout = ETRAX_FS_rw_pc_dout,
  58. .din = ETRAX_FS_r_pc_din,
  59. },
  60. {
  61. .label = "D",
  62. .ngpio = 18,
  63. .oe = ETRAX_FS_rw_pd_oe,
  64. .dout = ETRAX_FS_rw_pd_dout,
  65. .din = ETRAX_FS_r_pd_din,
  66. },
  67. {
  68. .label = "E",
  69. .ngpio = 18,
  70. .oe = ETRAX_FS_rw_pe_oe,
  71. .dout = ETRAX_FS_rw_pe_dout,
  72. .din = ETRAX_FS_r_pe_din,
  73. },
  74. };
  75. static const struct etraxfs_gpio_info etraxfs_gpio_etraxfs = {
  76. .num_ports = ARRAY_SIZE(etraxfs_gpio_etraxfs_ports),
  77. .ports = etraxfs_gpio_etraxfs_ports,
  78. };
  79. static int etraxfs_gpio_of_xlate(struct gpio_chip *gc,
  80. const struct of_phandle_args *gpiospec,
  81. u32 *flags)
  82. {
  83. /*
  84. * Port numbers are A to E, and the properties are integers, so we
  85. * specify them as 0xA - 0xE.
  86. */
  87. if (gc->label[0] - 'A' + 0xA != gpiospec->args[2])
  88. return -EINVAL;
  89. return of_gpio_simple_xlate(gc, gpiospec, flags);
  90. }
  91. static const struct of_device_id etraxfs_gpio_of_table[] = {
  92. {
  93. .compatible = "axis,etraxfs-gio",
  94. .data = &etraxfs_gpio_etraxfs,
  95. },
  96. {},
  97. };
  98. static int etraxfs_gpio_probe(struct platform_device *pdev)
  99. {
  100. struct device *dev = &pdev->dev;
  101. const struct etraxfs_gpio_info *info;
  102. const struct of_device_id *match;
  103. struct bgpio_chip *chips;
  104. struct resource *res;
  105. void __iomem *regs;
  106. int ret;
  107. int i;
  108. res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
  109. regs = devm_ioremap_resource(dev, res);
  110. if (!regs)
  111. return -ENOMEM;
  112. match = of_match_node(etraxfs_gpio_of_table, dev->of_node);
  113. if (!match)
  114. return -EINVAL;
  115. info = match->data;
  116. chips = devm_kzalloc(dev, sizeof(*chips) * info->num_ports, GFP_KERNEL);
  117. if (!chips)
  118. return -ENOMEM;
  119. for (i = 0; i < info->num_ports; i++) {
  120. struct bgpio_chip *bgc = &chips[i];
  121. const struct etraxfs_gpio_port *port = &info->ports[i];
  122. ret = bgpio_init(bgc, dev, 4,
  123. regs + port->din, /* dat */
  124. regs + port->dout, /* set */
  125. NULL, /* clr */
  126. regs + port->oe, /* dirout */
  127. NULL, /* dirin */
  128. BGPIOF_UNREADABLE_REG_SET);
  129. if (ret)
  130. return ret;
  131. bgc->gc.ngpio = port->ngpio;
  132. bgc->gc.label = port->label;
  133. bgc->gc.of_node = dev->of_node;
  134. bgc->gc.of_gpio_n_cells = 3;
  135. bgc->gc.of_xlate = etraxfs_gpio_of_xlate;
  136. ret = gpiochip_add(&bgc->gc);
  137. if (ret)
  138. dev_err(dev, "Unable to register port %s\n",
  139. bgc->gc.label);
  140. }
  141. return 0;
  142. }
  143. static struct platform_driver etraxfs_gpio_driver = {
  144. .driver = {
  145. .name = "etraxfs-gpio",
  146. .of_match_table = of_match_ptr(etraxfs_gpio_of_table),
  147. },
  148. .probe = etraxfs_gpio_probe,
  149. };
  150. static int __init etraxfs_gpio_init(void)
  151. {
  152. return platform_driver_register(&etraxfs_gpio_driver);
  153. }
  154. device_initcall(etraxfs_gpio_init);