mt7621_gpio.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. // SPDX-License-Identifier: GPL-2.0+
  2. /*
  3. * Copyright (C) 2018 Stefan Roese <sr@denx.de>
  4. *
  5. * Based on the Linux driver version which is:
  6. * Copyright (C) 2009-2011 Gabor Juhos <juhosg@openwrt.org>
  7. * Copyright (C) 2013 John Crispin <blogic@openwrt.org>
  8. */
  9. #include <common.h>
  10. #include <dm.h>
  11. #include <errno.h>
  12. #include <fdtdec.h>
  13. #include <malloc.h>
  14. #include <linux/io.h>
  15. #include <asm/io.h>
  16. #include <asm/gpio.h>
  17. #include <dm/device-internal.h>
  18. #include <dt-bindings/gpio/gpio.h>
  19. #define MTK_MAX_BANK 3
  20. #define MTK_BANK_WIDTH 32
  21. enum mediatek_gpio_reg {
  22. GPIO_REG_CTRL = 0,
  23. GPIO_REG_POL,
  24. GPIO_REG_DATA,
  25. GPIO_REG_DSET,
  26. GPIO_REG_DCLR,
  27. GPIO_REG_REDGE,
  28. GPIO_REG_FEDGE,
  29. GPIO_REG_HLVL,
  30. GPIO_REG_LLVL,
  31. GPIO_REG_STAT,
  32. GPIO_REG_EDGE,
  33. };
  34. static void __iomem *mediatek_gpio_membase;
  35. struct mediatek_gpio_platdata {
  36. char bank_name[3]; /* Name of bank, e.g. "PA", "PB" etc */
  37. int gpio_count;
  38. int bank;
  39. };
  40. static u32 reg_offs(struct mediatek_gpio_platdata *plat, int reg)
  41. {
  42. return (reg * 0x10) + (plat->bank * 0x4);
  43. }
  44. static int mediatek_gpio_get_value(struct udevice *dev, unsigned int offset)
  45. {
  46. struct mediatek_gpio_platdata *plat = dev_get_platdata(dev);
  47. return !!(ioread32(mediatek_gpio_membase +
  48. reg_offs(plat, GPIO_REG_DATA)) & BIT(offset));
  49. }
  50. static int mediatek_gpio_set_value(struct udevice *dev, unsigned int offset,
  51. int value)
  52. {
  53. struct mediatek_gpio_platdata *plat = dev_get_platdata(dev);
  54. iowrite32(BIT(offset), mediatek_gpio_membase +
  55. reg_offs(plat, value ? GPIO_REG_DSET : GPIO_REG_DCLR));
  56. return 0;
  57. }
  58. static int mediatek_gpio_direction_input(struct udevice *dev, unsigned int offset)
  59. {
  60. struct mediatek_gpio_platdata *plat = dev_get_platdata(dev);
  61. clrbits_le32(mediatek_gpio_membase + reg_offs(plat, GPIO_REG_CTRL),
  62. BIT(offset));
  63. return 0;
  64. }
  65. static int mediatek_gpio_direction_output(struct udevice *dev, unsigned int offset,
  66. int value)
  67. {
  68. struct mediatek_gpio_platdata *plat = dev_get_platdata(dev);
  69. setbits_le32(mediatek_gpio_membase + reg_offs(plat, GPIO_REG_CTRL),
  70. BIT(offset));
  71. mediatek_gpio_set_value(dev, offset, value);
  72. return 0;
  73. }
  74. static int mediatek_gpio_get_function(struct udevice *dev, unsigned int offset)
  75. {
  76. struct mediatek_gpio_platdata *plat = dev_get_platdata(dev);
  77. u32 t;
  78. t = ioread32(mediatek_gpio_membase + reg_offs(plat, GPIO_REG_CTRL));
  79. if (t & BIT(offset))
  80. return GPIOF_OUTPUT;
  81. return GPIOF_INPUT;
  82. }
  83. static const struct dm_gpio_ops gpio_mediatek_ops = {
  84. .direction_input = mediatek_gpio_direction_input,
  85. .direction_output = mediatek_gpio_direction_output,
  86. .get_value = mediatek_gpio_get_value,
  87. .set_value = mediatek_gpio_set_value,
  88. .get_function = mediatek_gpio_get_function,
  89. };
  90. static int gpio_mediatek_probe(struct udevice *dev)
  91. {
  92. struct mediatek_gpio_platdata *plat = dev_get_platdata(dev);
  93. struct gpio_dev_priv *uc_priv = dev_get_uclass_priv(dev);
  94. /* Tell the uclass how many GPIOs we have */
  95. if (plat) {
  96. uc_priv->gpio_count = plat->gpio_count;
  97. uc_priv->bank_name = plat->bank_name;
  98. }
  99. return 0;
  100. }
  101. /**
  102. * We have a top-level GPIO device with no actual GPIOs. It has a child
  103. * device for each Mediatek bank.
  104. */
  105. static int gpio_mediatek_bind(struct udevice *parent)
  106. {
  107. struct mediatek_gpio_platdata *plat = parent->platdata;
  108. ofnode node;
  109. int bank = 0;
  110. int ret;
  111. /* If this is a child device, there is nothing to do here */
  112. if (plat)
  113. return 0;
  114. mediatek_gpio_membase = dev_remap_addr(parent);
  115. if (!mediatek_gpio_membase)
  116. return -EINVAL;
  117. for (node = dev_read_first_subnode(parent); ofnode_valid(node);
  118. node = dev_read_next_subnode(node)) {
  119. struct mediatek_gpio_platdata *plat;
  120. struct udevice *dev;
  121. plat = calloc(1, sizeof(*plat));
  122. if (!plat)
  123. return -ENOMEM;
  124. plat->bank_name[0] = 'P';
  125. plat->bank_name[1] = 'A' + bank;
  126. plat->bank_name[2] = '\0';
  127. plat->gpio_count = MTK_BANK_WIDTH;
  128. plat->bank = bank;
  129. ret = device_bind(parent, parent->driver,
  130. plat->bank_name, plat, -1, &dev);
  131. if (ret)
  132. return ret;
  133. dev->node = node;
  134. bank++;
  135. }
  136. return 0;
  137. }
  138. static const struct udevice_id mediatek_gpio_ids[] = {
  139. { .compatible = "mtk,mt7621-gpio" },
  140. { }
  141. };
  142. U_BOOT_DRIVER(gpio_mediatek) = {
  143. .name = "gpio_mediatek",
  144. .id = UCLASS_GPIO,
  145. .ops = &gpio_mediatek_ops,
  146. .of_match = mediatek_gpio_ids,
  147. .bind = gpio_mediatek_bind,
  148. .probe = gpio_mediatek_probe,
  149. };