irq-ixp4xx.c 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * irqchip for the IXP4xx interrupt controller
  4. * Copyright (C) 2019 Linus Walleij <linus.walleij@linaro.org>
  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 <dsaxena@plexity.net>
  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 void __exception_irq_entry ixp4xx_handle_irq(struct pt_regs *regs)
  97. {
  98. struct ixp4xx_irq *ixi = &ixirq;
  99. unsigned long status;
  100. int i;
  101. status = __raw_readl(ixi->irqbase + IXP4XX_ICIP);
  102. for_each_set_bit(i, &status, 32)
  103. generic_handle_domain_irq(ixi->domain, i);
  104. /*
  105. * IXP465/IXP435 has an upper IRQ status register
  106. */
  107. if (ixi->is_356) {
  108. status = __raw_readl(ixi->irqbase + IXP4XX_ICIP2);
  109. for_each_set_bit(i, &status, 32)
  110. generic_handle_domain_irq(ixi->domain, i + 32);
  111. }
  112. }
  113. static int ixp4xx_irq_domain_translate(struct irq_domain *domain,
  114. struct irq_fwspec *fwspec,
  115. unsigned long *hwirq,
  116. unsigned int *type)
  117. {
  118. /* We support standard DT translation */
  119. if (is_of_node(fwspec->fwnode) && fwspec->param_count == 2) {
  120. *hwirq = fwspec->param[0];
  121. *type = fwspec->param[1];
  122. return 0;
  123. }
  124. if (is_fwnode_irqchip(fwspec->fwnode)) {
  125. if (fwspec->param_count != 2)
  126. return -EINVAL;
  127. *hwirq = fwspec->param[0];
  128. *type = fwspec->param[1];
  129. WARN_ON(*type == IRQ_TYPE_NONE);
  130. return 0;
  131. }
  132. return -EINVAL;
  133. }
  134. static int ixp4xx_irq_domain_alloc(struct irq_domain *d,
  135. unsigned int irq, unsigned int nr_irqs,
  136. void *data)
  137. {
  138. struct ixp4xx_irq *ixi = d->host_data;
  139. irq_hw_number_t hwirq;
  140. unsigned int type = IRQ_TYPE_NONE;
  141. struct irq_fwspec *fwspec = data;
  142. int ret;
  143. int i;
  144. ret = ixp4xx_irq_domain_translate(d, fwspec, &hwirq, &type);
  145. if (ret)
  146. return ret;
  147. for (i = 0; i < nr_irqs; i++) {
  148. /*
  149. * TODO: after converting IXP4xx to only device tree, set
  150. * handle_bad_irq as default handler and assume all consumers
  151. * call .set_type() as this is provided in the second cell in
  152. * the device tree phandle.
  153. */
  154. irq_domain_set_info(d,
  155. irq + i,
  156. hwirq + i,
  157. &ixi->irqchip,
  158. ixi,
  159. handle_level_irq,
  160. NULL, NULL);
  161. irq_set_probe(irq + i);
  162. }
  163. return 0;
  164. }
  165. /*
  166. * This needs to be a hierarchical irqdomain to work well with the
  167. * GPIO irqchip (which is lower in the hierarchy)
  168. */
  169. static const struct irq_domain_ops ixp4xx_irqdomain_ops = {
  170. .translate = ixp4xx_irq_domain_translate,
  171. .alloc = ixp4xx_irq_domain_alloc,
  172. .free = irq_domain_free_irqs_common,
  173. };
  174. /**
  175. * ixp4x_irq_setup() - Common setup code for the IXP4xx interrupt controller
  176. * @ixi: State container
  177. * @irqbase: Virtual memory base for the interrupt controller
  178. * @fwnode: Corresponding fwnode abstraction for this controller
  179. * @is_356: if this is an IXP43x, IXP45x or IXP46x SoC variant
  180. */
  181. static int __init ixp4xx_irq_setup(struct ixp4xx_irq *ixi,
  182. void __iomem *irqbase,
  183. struct fwnode_handle *fwnode,
  184. bool is_356)
  185. {
  186. int nr_irqs;
  187. ixi->irqbase = irqbase;
  188. ixi->is_356 = is_356;
  189. /* Route all sources to IRQ instead of FIQ */
  190. __raw_writel(0x0, ixi->irqbase + IXP4XX_ICLR);
  191. /* Disable all interrupts */
  192. __raw_writel(0x0, ixi->irqbase + IXP4XX_ICMR);
  193. if (is_356) {
  194. /* Route upper 32 sources to IRQ instead of FIQ */
  195. __raw_writel(0x0, ixi->irqbase + IXP4XX_ICLR2);
  196. /* Disable upper 32 interrupts */
  197. __raw_writel(0x0, ixi->irqbase + IXP4XX_ICMR2);
  198. nr_irqs = 64;
  199. } else {
  200. nr_irqs = 32;
  201. }
  202. ixi->irqchip.name = "IXP4xx";
  203. ixi->irqchip.irq_mask = ixp4xx_irq_mask;
  204. ixi->irqchip.irq_unmask = ixp4xx_irq_unmask;
  205. ixi->irqchip.irq_set_type = ixp4xx_set_irq_type;
  206. ixi->domain = irq_domain_create_linear(fwnode, nr_irqs,
  207. &ixp4xx_irqdomain_ops,
  208. ixi);
  209. if (!ixi->domain) {
  210. pr_crit("IXP4XX: can not add primary irqdomain\n");
  211. return -ENODEV;
  212. }
  213. set_handle_irq(ixp4xx_handle_irq);
  214. return 0;
  215. }
  216. static int __init ixp4xx_of_init_irq(struct device_node *np,
  217. struct device_node *parent)
  218. {
  219. struct ixp4xx_irq *ixi = &ixirq;
  220. void __iomem *base;
  221. struct fwnode_handle *fwnode;
  222. bool is_356;
  223. int ret;
  224. base = of_iomap(np, 0);
  225. if (!base) {
  226. pr_crit("IXP4XX: could not ioremap interrupt controller\n");
  227. return -ENODEV;
  228. }
  229. fwnode = of_node_to_fwnode(np);
  230. /* These chip variants have 64 interrupts */
  231. is_356 = of_device_is_compatible(np, "intel,ixp43x-interrupt") ||
  232. of_device_is_compatible(np, "intel,ixp45x-interrupt") ||
  233. of_device_is_compatible(np, "intel,ixp46x-interrupt");
  234. ret = ixp4xx_irq_setup(ixi, base, fwnode, is_356);
  235. if (ret)
  236. pr_crit("IXP4XX: failed to set up irqchip\n");
  237. return ret;
  238. }
  239. IRQCHIP_DECLARE(ixp42x, "intel,ixp42x-interrupt",
  240. ixp4xx_of_init_irq);
  241. IRQCHIP_DECLARE(ixp43x, "intel,ixp43x-interrupt",
  242. ixp4xx_of_init_irq);
  243. IRQCHIP_DECLARE(ixp45x, "intel,ixp45x-interrupt",
  244. ixp4xx_of_init_irq);
  245. IRQCHIP_DECLARE(ixp46x, "intel,ixp46x-interrupt",
  246. ixp4xx_of_init_irq);