irq.c 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289
  1. /* linux/arch/arm/mach-s3c2443/irq.c
  2. *
  3. * Copyright (c) 2007 Simtec Electronics
  4. * Ben Dooks <ben@simtec.co.uk>
  5. *
  6. * This program is free software; you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License as published by
  8. * the Free Software Foundation; either version 2 of the License, or
  9. * (at your option) any later version.
  10. *
  11. * This program is distributed in the hope that it will be useful,
  12. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. * GNU General Public License for more details.
  15. *
  16. * You should have received a copy of the GNU General Public License
  17. * along with this program; if not, write to the Free Software
  18. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  19. *
  20. */
  21. #include <linux/init.h>
  22. #include <linux/module.h>
  23. #include <linux/interrupt.h>
  24. #include <linux/ioport.h>
  25. #include <linux/sysdev.h>
  26. #include <asm/hardware.h>
  27. #include <asm/irq.h>
  28. #include <asm/io.h>
  29. #include <asm/mach/irq.h>
  30. #include <asm/arch/regs-irq.h>
  31. #include <asm/arch/regs-gpio.h>
  32. #include <asm/plat-s3c24xx/cpu.h>
  33. #include <asm/plat-s3c24xx/pm.h>
  34. #include <asm/plat-s3c24xx/irq.h>
  35. #define INTMSK(start, end) ((1 << ((end) + 1 - (start))) - 1)
  36. static inline void s3c2443_irq_demux(unsigned int irq, unsigned int len)
  37. {
  38. unsigned int subsrc, submsk;
  39. unsigned int end;
  40. struct irq_desc *mydesc;
  41. /* read the current pending interrupts, and the mask
  42. * for what it is available */
  43. subsrc = __raw_readl(S3C2410_SUBSRCPND);
  44. submsk = __raw_readl(S3C2410_INTSUBMSK);
  45. subsrc &= ~submsk;
  46. subsrc >>= (irq - S3C2410_IRQSUB(0));
  47. subsrc &= (1 << len)-1;
  48. end = len + irq;
  49. mydesc = irq_desc + irq;
  50. for (; irq < end && subsrc; irq++) {
  51. if (subsrc & 1)
  52. desc_handle_irq(irq, mydesc);
  53. mydesc++;
  54. subsrc >>= 1;
  55. }
  56. }
  57. /* WDT/AC97 sub interrupts */
  58. static void s3c2443_irq_demux_wdtac97(unsigned int irq, struct irq_desc *desc)
  59. {
  60. s3c2443_irq_demux(IRQ_S3C2443_WDT, 4);
  61. }
  62. #define INTMSK_WDTAC97 (1UL << (IRQ_WDT - IRQ_EINT0))
  63. #define SUBMSK_WDTAC97 INTMSK(IRQ_S3C2443_WDT, IRQ_S3C2443_AC97)
  64. static void s3c2443_irq_wdtac97_mask(unsigned int irqno)
  65. {
  66. s3c_irqsub_mask(irqno, INTMSK_WDTAC97, SUBMSK_WDTAC97);
  67. }
  68. static void s3c2443_irq_wdtac97_unmask(unsigned int irqno)
  69. {
  70. s3c_irqsub_unmask(irqno, INTMSK_WDTAC97);
  71. }
  72. static void s3c2443_irq_wdtac97_ack(unsigned int irqno)
  73. {
  74. s3c_irqsub_maskack(irqno, INTMSK_WDTAC97, SUBMSK_WDTAC97);
  75. }
  76. static struct irq_chip s3c2443_irq_wdtac97 = {
  77. .mask = s3c2443_irq_wdtac97_mask,
  78. .unmask = s3c2443_irq_wdtac97_unmask,
  79. .ack = s3c2443_irq_wdtac97_ack,
  80. };
  81. /* LCD sub interrupts */
  82. static void s3c2443_irq_demux_lcd(unsigned int irq, struct irq_desc *desc)
  83. {
  84. s3c2443_irq_demux(IRQ_S3C2443_LCD1, 4);
  85. }
  86. #define INTMSK_LCD (1UL << (IRQ_LCD - IRQ_EINT0))
  87. #define SUBMSK_LCD INTMSK(IRQ_S3C2443_LCD1, IRQ_S3C2443_LCD4)
  88. static void s3c2443_irq_lcd_mask(unsigned int irqno)
  89. {
  90. s3c_irqsub_mask(irqno, INTMSK_LCD, SUBMSK_LCD);
  91. }
  92. static void s3c2443_irq_lcd_unmask(unsigned int irqno)
  93. {
  94. s3c_irqsub_unmask(irqno, INTMSK_LCD);
  95. }
  96. static void s3c2443_irq_lcd_ack(unsigned int irqno)
  97. {
  98. s3c_irqsub_maskack(irqno, INTMSK_LCD, SUBMSK_LCD);
  99. }
  100. static struct irq_chip s3c2443_irq_lcd = {
  101. .mask = s3c2443_irq_lcd_mask,
  102. .unmask = s3c2443_irq_lcd_unmask,
  103. .ack = s3c2443_irq_lcd_ack,
  104. };
  105. /* DMA sub interrupts */
  106. static void s3c2443_irq_demux_dma(unsigned int irq, struct irq_desc *desc)
  107. {
  108. s3c2443_irq_demux(IRQ_S3C2443_DMA0, 6);
  109. }
  110. #define INTMSK_DMA (1UL << (IRQ_S3C2443_DMA - IRQ_EINT0))
  111. #define SUBMSK_DMA INTMSK(IRQ_S3C2443_DMA0, IRQ_S3C2443_DMA5)
  112. static void s3c2443_irq_dma_mask(unsigned int irqno)
  113. {
  114. s3c_irqsub_mask(irqno, INTMSK_DMA, SUBMSK_DMA);
  115. }
  116. static void s3c2443_irq_dma_unmask(unsigned int irqno)
  117. {
  118. s3c_irqsub_unmask(irqno, INTMSK_DMA);
  119. }
  120. static void s3c2443_irq_dma_ack(unsigned int irqno)
  121. {
  122. s3c_irqsub_maskack(irqno, INTMSK_DMA, SUBMSK_DMA);
  123. }
  124. static struct irq_chip s3c2443_irq_dma = {
  125. .mask = s3c2443_irq_dma_mask,
  126. .unmask = s3c2443_irq_dma_unmask,
  127. .ack = s3c2443_irq_dma_ack,
  128. };
  129. /* UART3 sub interrupts */
  130. static void s3c2443_irq_demux_uart3(unsigned int irq, struct irq_desc *desc)
  131. {
  132. s3c2443_irq_demux(IRQ_S3C2443_UART3, 3);
  133. }
  134. #define INTMSK_UART3 (1UL << (IRQ_S3C2443_UART3 - IRQ_EINT0))
  135. #define SUBMSK_UART3 (0xf << (IRQ_S3C2443_RX3 - S3C2410_IRQSUB(0)))
  136. static void s3c2443_irq_uart3_mask(unsigned int irqno)
  137. {
  138. s3c_irqsub_mask(irqno, INTMSK_UART3, SUBMSK_UART3);
  139. }
  140. static void s3c2443_irq_uart3_unmask(unsigned int irqno)
  141. {
  142. s3c_irqsub_unmask(irqno, INTMSK_UART3);
  143. }
  144. static void s3c2443_irq_uart3_ack(unsigned int irqno)
  145. {
  146. s3c_irqsub_maskack(irqno, INTMSK_UART3, SUBMSK_UART3);
  147. }
  148. static struct irq_chip s3c2443_irq_uart3 = {
  149. .mask = s3c2443_irq_uart3_mask,
  150. .unmask = s3c2443_irq_uart3_unmask,
  151. .ack = s3c2443_irq_uart3_ack,
  152. };
  153. /* CAM sub interrupts */
  154. static void s3c2443_irq_demux_cam(unsigned int irq, struct irq_desc *desc)
  155. {
  156. s3c2443_irq_demux(IRQ_S3C2440_CAM_C, 4);
  157. }
  158. #define INTMSK_CAM (1UL << (IRQ_CAM - IRQ_EINT0))
  159. #define SUBMSK_CAM INTMSK(IRQ_S3C2440_CAM_C, IRQ_S3C2440_CAM_P)
  160. static void s3c2443_irq_cam_mask(unsigned int irqno)
  161. {
  162. s3c_irqsub_mask(irqno, INTMSK_CAM, SUBMSK_CAM);
  163. }
  164. static void s3c2443_irq_cam_unmask(unsigned int irqno)
  165. {
  166. s3c_irqsub_unmask(irqno, INTMSK_CAM);
  167. }
  168. static void s3c2443_irq_cam_ack(unsigned int irqno)
  169. {
  170. s3c_irqsub_maskack(irqno, INTMSK_CAM, SUBMSK_CAM);
  171. }
  172. static struct irq_chip s3c2443_irq_cam = {
  173. .mask = s3c2443_irq_cam_mask,
  174. .unmask = s3c2443_irq_cam_unmask,
  175. .ack = s3c2443_irq_cam_ack,
  176. };
  177. /* IRQ initialisation code */
  178. static int __init s3c2443_add_sub(unsigned int base,
  179. void (*demux)(unsigned int,
  180. struct irq_desc *),
  181. struct irq_chip *chip,
  182. unsigned int start, unsigned int end)
  183. {
  184. unsigned int irqno;
  185. set_irq_chip(base, &s3c_irq_level_chip);
  186. set_irq_handler(base, handle_level_irq);
  187. set_irq_chained_handler(base, demux);
  188. for (irqno = start; irqno <= end; irqno++) {
  189. set_irq_chip(irqno, chip);
  190. set_irq_handler(irqno, handle_level_irq);
  191. set_irq_flags(irqno, IRQF_VALID);
  192. }
  193. return 0;
  194. }
  195. static int s3c2443_irq_add(struct sys_device *sysdev)
  196. {
  197. printk("S3C2443: IRQ Support\n");
  198. s3c2443_add_sub(IRQ_CAM, s3c2443_irq_demux_cam, &s3c2443_irq_cam,
  199. IRQ_S3C2440_CAM_C, IRQ_S3C2440_CAM_P);
  200. s3c2443_add_sub(IRQ_LCD, s3c2443_irq_demux_lcd, &s3c2443_irq_lcd,
  201. IRQ_S3C2443_LCD1, IRQ_S3C2443_LCD4);
  202. s3c2443_add_sub(IRQ_S3C2443_DMA, s3c2443_irq_demux_dma,
  203. &s3c2443_irq_dma, IRQ_S3C2443_DMA0, IRQ_S3C2443_DMA5);
  204. s3c2443_add_sub(IRQ_S3C2443_UART3, s3c2443_irq_demux_uart3,
  205. &s3c2443_irq_uart3,
  206. IRQ_S3C2443_RX3, IRQ_S3C2443_ERR3);
  207. s3c2443_add_sub(IRQ_WDT, s3c2443_irq_demux_wdtac97,
  208. &s3c2443_irq_wdtac97,
  209. IRQ_S3C2443_WDT, IRQ_S3C2443_AC97);
  210. return 0;
  211. }
  212. static struct sysdev_driver s3c2443_irq_driver = {
  213. .add = s3c2443_irq_add,
  214. };
  215. static int s3c2443_irq_init(void)
  216. {
  217. return sysdev_driver_register(&s3c2443_sysclass, &s3c2443_irq_driver);
  218. }
  219. arch_initcall(s3c2443_irq_init);