irq-ixp4xx.c 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * irqchip for the IXP4xx interrupt controller
  4. * Copyright (C) 2019 Linus Walleij <[email protected]>
  5. *
  6. * Based on arch/arm/mach-ixp4xx/common.c
  7. * Copyright 2002 (C) Intel Corporation
  8. * Copyright 2003-2004 (C) MontaVista, Software, Inc.
  9. * Copyright (C) Deepak Saxena <[email protected]>
  10. */
  11. #include <linux/bitops.h>
  12. #include <linux/gpio/driver.h>
  13. #include <linux/irq.h>
  14. #include <linux/io.h>
  15. #include <linux/irqchip.h>
  16. #include <linux/irqdomain.h>
  17. #include <linux/of.h>
  18. #include <linux/of_address.h>
  19. #include <linux/of_irq.h>
  20. #include <linux/platform_device.h>
  21. #include <linux/cpu.h>
  22. #include <asm/exception.h>
  23. #include <asm/mach/irq.h>
  24. #define IXP4XX_ICPR 0x00 /* Interrupt Status */
  25. #define IXP4XX_ICMR 0x04 /* Interrupt Enable */
  26. #define IXP4XX_ICLR 0x08 /* Interrupt IRQ/FIQ Select */
  27. #define IXP4XX_ICIP 0x0C /* IRQ Status */
  28. #define IXP4XX_ICFP 0x10 /* FIQ Status */
  29. #define IXP4XX_ICHR 0x14 /* Interrupt Priority */
  30. #define IXP4XX_ICIH 0x18 /* IRQ Highest Pri Int */
  31. #define IXP4XX_ICFH 0x1C /* FIQ Highest Pri Int */
  32. /* IXP43x and IXP46x-only */
  33. #define IXP4XX_ICPR2 0x20 /* Interrupt Status 2 */
  34. #define IXP4XX_ICMR2 0x24 /* Interrupt Enable 2 */
  35. #define IXP4XX_ICLR2 0x28 /* Interrupt IRQ/FIQ Select 2 */
  36. #define IXP4XX_ICIP2 0x2C /* IRQ Status */
  37. #define IXP4XX_ICFP2 0x30 /* FIQ Status */
  38. #define IXP4XX_ICEEN 0x34 /* Error High Pri Enable */
  39. /**
  40. * struct ixp4xx_irq - state container for the Faraday IRQ controller
  41. * @irqbase: IRQ controller memory base in virtual memory
  42. * @is_356: if this is an IXP43x, IXP45x or IX46x SoC (with 64 IRQs)
  43. * @irqchip: irqchip for this instance
  44. * @domain: IRQ domain for this instance
  45. */
  46. struct ixp4xx_irq {
  47. void __iomem *irqbase;
  48. bool is_356;
  49. struct irq_chip irqchip;
  50. struct irq_domain *domain;
  51. };
  52. /* Local static state container */
  53. static struct ixp4xx_irq ixirq;
  54. /* GPIO Clocks */
  55. #define IXP4XX_GPIO_CLK_0 14
  56. #define IXP4XX_GPIO_CLK_1 15
  57. static int ixp4xx_set_irq_type(struct irq_data *d, unsigned int type)
  58. {
  59. /* All are level active high (asserted) here */
  60. if (type != IRQ_TYPE_LEVEL_HIGH)
  61. return -EINVAL;
  62. return 0;
  63. }
  64. static void ixp4xx_irq_mask(struct irq_data *d)
  65. {
  66. struct ixp4xx_irq *ixi = irq_data_get_irq_chip_data(d);
  67. u32 val;
  68. if (ixi->is_356 && d->hwirq >= 32) {
  69. val = __raw_readl(ixi->irqbase + IXP4XX_ICMR2);
  70. val &= ~BIT(d->hwirq - 32);
  71. __raw_writel(val, ixi->irqbase + IXP4XX_ICMR2);
  72. } else {
  73. val = __raw_readl(ixi->irqbase + IXP4XX_ICMR);
  74. val &= ~BIT(d->hwirq);
  75. __raw_writel(val, ixi->irqbase + IXP4XX_ICMR);
  76. }
  77. }
  78. /*
  79. * Level triggered interrupts on GPIO lines can only be cleared when the
  80. * interrupt condition disappears.
  81. */
  82. static void ixp4xx_irq_unmask(struct irq_data *d)
  83. {
  84. struct ixp4xx_irq *ixi = irq_data_get_irq_chip_data(d);
  85. u32 val;
  86. if (ixi->is_356 && d->hwirq >= 32) {
  87. val = __raw_readl(ixi->irqbase + IXP4XX_ICMR2);
  88. val |= BIT(d->hwirq - 32);
  89. __raw_writel(val, ixi->irqbase + IXP4XX_ICMR2);
  90. } else {
  91. val = __raw_readl(ixi->irqbase + IXP4XX_ICMR);
  92. val |= BIT(d->hwirq);
  93. __raw_writel(val, ixi->irqbase + IXP4XX_ICMR);
  94. }
  95. }
  96. static asmlinkage void __exception_irq_entry
  97. ixp4xx_handle_irq(struct pt_regs *regs)
  98. {
  99. struct ixp4xx_irq *ixi = &ixirq;
  100. unsigned long status;
  101. int i;
  102. status = __raw_readl(ixi->irqbase + IXP4XX_ICIP);
  103. for_each_set_bit(i, &status, 32)
  104. generic_handle_domain_irq(ixi->domain, i);
  105. /*
  106. * IXP465/IXP435 has an upper IRQ status register
  107. */
  108. if (ixi->is_356) {
  109. status = __raw_readl(ixi->irqbase + IXP4XX_ICIP2);
  110. for_each_set_bit(i, &status, 32)
  111. generic_handle_domain_irq(ixi->domain, i + 32);
  112. }
  113. }
  114. static int ixp4xx_irq_domain_translate(struct irq_domain *domain,
  115. struct irq_fwspec *fwspec,
  116. unsigned long *hwirq,
  117. unsigned int *type)
  118. {
  119. /* We support standard DT translation */
  120. if (is_of_node(fwspec->fwnode) && fwspec->param_count == 2) {
  121. *hwirq = fwspec->param[0];
  122. *type = fwspec->param[1];
  123. return 0;
  124. }
  125. if (is_fwnode_irqchip(fwspec->fwnode)) {
  126. if (fwspec->param_count != 2)
  127. return -EINVAL;
  128. *hwirq = fwspec->param[0];
  129. *type = fwspec->param[1];
  130. WARN_ON(*type == IRQ_TYPE_NONE);
  131. return 0;
  132. }
  133. return -EINVAL;
  134. }
  135. static int ixp4xx_irq_domain_alloc(struct irq_domain *d,
  136. unsigned int irq, unsigned int nr_irqs,
  137. void *data)
  138. {
  139. struct ixp4xx_irq *ixi = d->host_data;
  140. irq_hw_number_t hwirq;
  141. unsigned int type = IRQ_TYPE_NONE;
  142. struct irq_fwspec *fwspec = data;
  143. int ret;
  144. int i;
  145. ret = ixp4xx_irq_domain_translate(d, fwspec, &hwirq, &type);
  146. if (ret)
  147. return ret;
  148. for (i = 0; i < nr_irqs; i++) {
  149. /*
  150. * TODO: after converting IXP4xx to only device tree, set
  151. * handle_bad_irq as default handler and assume all consumers
  152. * call .set_type() as this is provided in the second cell in
  153. * the device tree phandle.
  154. */
  155. irq_domain_set_info(d,
  156. irq + i,
  157. hwirq + i,
  158. &ixi->irqchip,
  159. ixi,
  160. handle_level_irq,
  161. NULL, NULL);
  162. irq_set_probe(irq + i);
  163. }
  164. return 0;
  165. }
  166. /*
  167. * This needs to be a hierarchical irqdomain to work well with the
  168. * GPIO irqchip (which is lower in the hierarchy)
  169. */
  170. static const struct irq_domain_ops ixp4xx_irqdomain_ops = {
  171. .translate = ixp4xx_irq_domain_translate,
  172. .alloc = ixp4xx_irq_domain_alloc,
  173. .free = irq_domain_free_irqs_common,
  174. };
  175. /**
  176. * ixp4x_irq_setup() - Common setup code for the IXP4xx interrupt controller
  177. * @ixi: State container
  178. * @irqbase: Virtual memory base for the interrupt controller
  179. * @fwnode: Corresponding fwnode abstraction for this controller
  180. * @is_356: if this is an IXP43x, IXP45x or IXP46x SoC variant
  181. */
  182. static int __init ixp4xx_irq_setup(struct ixp4xx_irq *ixi,
  183. void __iomem *irqbase,
  184. struct fwnode_handle *fwnode,
  185. bool is_356)
  186. {
  187. int nr_irqs;
  188. ixi->irqbase = irqbase;
  189. ixi->is_356 = is_356;
  190. /* Route all sources to IRQ instead of FIQ */
  191. __raw_writel(0x0, ixi->irqbase + IXP4XX_ICLR);
  192. /* Disable all interrupts */
  193. __raw_writel(0x0, ixi->irqbase + IXP4XX_ICMR);
  194. if (is_356) {
  195. /* Route upper 32 sources to IRQ instead of FIQ */
  196. __raw_writel(0x0, ixi->irqbase + IXP4XX_ICLR2);
  197. /* Disable upper 32 interrupts */
  198. __raw_writel(0x0, ixi->irqbase + IXP4XX_ICMR2);
  199. nr_irqs = 64;
  200. } else {
  201. nr_irqs = 32;
  202. }
  203. ixi->irqchip.name = "IXP4xx";
  204. ixi->irqchip.irq_mask = ixp4xx_irq_mask;
  205. ixi->irqchip.irq_unmask = ixp4xx_irq_unmask;
  206. ixi->irqchip.irq_set_type = ixp4xx_set_irq_type;
  207. ixi->domain = irq_domain_create_linear(fwnode, nr_irqs,
  208. &ixp4xx_irqdomain_ops,
  209. ixi);
  210. if (!ixi->domain) {
  211. pr_crit("IXP4XX: can not add primary irqdomain\n");
  212. return -ENODEV;
  213. }
  214. set_handle_irq(ixp4xx_handle_irq);
  215. return 0;
  216. }
  217. static int __init ixp4xx_of_init_irq(struct device_node *np,
  218. struct device_node *parent)
  219. {
  220. struct ixp4xx_irq *ixi = &ixirq;
  221. void __iomem *base;
  222. struct fwnode_handle *fwnode;
  223. bool is_356;
  224. int ret;
  225. base = of_iomap(np, 0);
  226. if (!base) {
  227. pr_crit("IXP4XX: could not ioremap interrupt controller\n");
  228. return -ENODEV;
  229. }
  230. fwnode = of_node_to_fwnode(np);
  231. /* These chip variants have 64 interrupts */
  232. is_356 = of_device_is_compatible(np, "intel,ixp43x-interrupt") ||
  233. of_device_is_compatible(np, "intel,ixp45x-interrupt") ||
  234. of_device_is_compatible(np, "intel,ixp46x-interrupt");
  235. ret = ixp4xx_irq_setup(ixi, base, fwnode, is_356);
  236. if (ret)
  237. pr_crit("IXP4XX: failed to set up irqchip\n");
  238. return ret;
  239. }
  240. IRQCHIP_DECLARE(ixp42x, "intel,ixp42x-interrupt",
  241. ixp4xx_of_init_irq);
  242. IRQCHIP_DECLARE(ixp43x, "intel,ixp43x-interrupt",
  243. ixp4xx_of_init_irq);
  244. IRQCHIP_DECLARE(ixp45x, "intel,ixp45x-interrupt",
  245. ixp4xx_of_init_irq);
  246. IRQCHIP_DECLARE(ixp46x, "intel,ixp46x-interrupt",
  247. ixp4xx_of_init_irq);