irq.c 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290
  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/ptrace.h>
  26. #include <linux/sysdev.h>
  27. #include <asm/hardware.h>
  28. #include <asm/irq.h>
  29. #include <asm/io.h>
  30. #include <asm/mach/irq.h>
  31. #include <asm/arch/regs-irq.h>
  32. #include <asm/arch/regs-gpio.h>
  33. #include <asm/plat-s3c24xx/cpu.h>
  34. #include <asm/plat-s3c24xx/pm.h>
  35. #include <asm/plat-s3c24xx/irq.h>
  36. #define INTMSK(start, end) ((1 << ((end) + 1 - (start))) - 1)
  37. static inline void s3c2443_irq_demux(unsigned int irq, unsigned int len)
  38. {
  39. unsigned int subsrc, submsk;
  40. unsigned int end;
  41. struct irq_desc *mydesc;
  42. /* read the current pending interrupts, and the mask
  43. * for what it is available */
  44. subsrc = __raw_readl(S3C2410_SUBSRCPND);
  45. submsk = __raw_readl(S3C2410_INTSUBMSK);
  46. subsrc &= ~submsk;
  47. subsrc >>= (irq - S3C2410_IRQSUB(0));
  48. subsrc &= (1 << len)-1;
  49. end = len + irq;
  50. mydesc = irq_desc + irq;
  51. for (; irq < end && subsrc; irq++) {
  52. if (subsrc & 1)
  53. desc_handle_irq(irq, mydesc);
  54. mydesc++;
  55. subsrc >>= 1;
  56. }
  57. }
  58. /* WDT/AC97 sub interrupts */
  59. static void s3c2443_irq_demux_wdtac97(unsigned int irq, struct irq_desc *desc)
  60. {
  61. s3c2443_irq_demux(IRQ_S3C2443_WDT, 4);
  62. }
  63. #define INTMSK_WDTAC97 (1UL << (IRQ_WDT - IRQ_EINT0))
  64. #define SUBMSK_WDTAC97 INTMSK(IRQ_S3C2443_WDT, IRQ_S3C2443_AC97)
  65. static void s3c2443_irq_wdtac97_mask(unsigned int irqno)
  66. {
  67. s3c_irqsub_mask(irqno, INTMSK_WDTAC97, SUBMSK_WDTAC97);
  68. }
  69. static void s3c2443_irq_wdtac97_unmask(unsigned int irqno)
  70. {
  71. s3c_irqsub_unmask(irqno, INTMSK_WDTAC97);
  72. }
  73. static void s3c2443_irq_wdtac97_ack(unsigned int irqno)
  74. {
  75. s3c_irqsub_maskack(irqno, INTMSK_WDTAC97, SUBMSK_WDTAC97);
  76. }
  77. static struct irq_chip s3c2443_irq_wdtac97 = {
  78. .mask = s3c2443_irq_wdtac97_mask,
  79. .unmask = s3c2443_irq_wdtac97_unmask,
  80. .ack = s3c2443_irq_wdtac97_ack,
  81. };
  82. /* LCD sub interrupts */
  83. static void s3c2443_irq_demux_lcd(unsigned int irq, struct irq_desc *desc)
  84. {
  85. s3c2443_irq_demux(IRQ_S3C2443_LCD1, 4);
  86. }
  87. #define INTMSK_LCD (1UL << (IRQ_LCD - IRQ_EINT0))
  88. #define SUBMSK_LCD INTMSK(IRQ_S3C2443_LCD1, IRQ_S3C2443_LCD4)
  89. static void s3c2443_irq_lcd_mask(unsigned int irqno)
  90. {
  91. s3c_irqsub_mask(irqno, INTMSK_LCD, SUBMSK_LCD);
  92. }
  93. static void s3c2443_irq_lcd_unmask(unsigned int irqno)
  94. {
  95. s3c_irqsub_unmask(irqno, INTMSK_LCD);
  96. }
  97. static void s3c2443_irq_lcd_ack(unsigned int irqno)
  98. {
  99. s3c_irqsub_maskack(irqno, INTMSK_LCD, SUBMSK_LCD);
  100. }
  101. static struct irq_chip s3c2443_irq_lcd = {
  102. .mask = s3c2443_irq_lcd_mask,
  103. .unmask = s3c2443_irq_lcd_unmask,
  104. .ack = s3c2443_irq_lcd_ack,
  105. };
  106. /* DMA sub interrupts */
  107. static void s3c2443_irq_demux_dma(unsigned int irq, struct irq_desc *desc)
  108. {
  109. s3c2443_irq_demux(IRQ_S3C2443_DMA1, 6);
  110. }
  111. #define INTMSK_DMA (1UL << (IRQ_S3C2443_DMA - IRQ_EINT0))
  112. #define SUBMSK_DMA INTMSK(IRQ_S3C2443_DMA0, IRQ_S3C2443_DMA5)
  113. static void s3c2443_irq_dma_mask(unsigned int irqno)
  114. {
  115. s3c_irqsub_mask(irqno, INTMSK_DMA, SUBMSK_DMA);
  116. }
  117. static void s3c2443_irq_dma_unmask(unsigned int irqno)
  118. {
  119. s3c_irqsub_unmask(irqno, INTMSK_DMA);
  120. }
  121. static void s3c2443_irq_dma_ack(unsigned int irqno)
  122. {
  123. s3c_irqsub_maskack(irqno, INTMSK_DMA, SUBMSK_DMA);
  124. }
  125. static struct irq_chip s3c2443_irq_dma = {
  126. .mask = s3c2443_irq_dma_mask,
  127. .unmask = s3c2443_irq_dma_unmask,
  128. .ack = s3c2443_irq_dma_ack,
  129. };
  130. /* UART3 sub interrupts */
  131. static void s3c2443_irq_demux_uart3(unsigned int irq, struct irq_desc *desc)
  132. {
  133. s3c2443_irq_demux(IRQ_S3C2443_UART3, 3);
  134. }
  135. #define INTMSK_UART3 (1UL << (IRQ_S3C2443_UART3 - IRQ_EINT0))
  136. #define SUBMSK_UART3 (0xf << (IRQ_S3C2443_RX3 - S3C2410_IRQSUB(0)))
  137. static void s3c2443_irq_uart3_mask(unsigned int irqno)
  138. {
  139. s3c_irqsub_mask(irqno, INTMSK_UART3, SUBMSK_UART3);
  140. }
  141. static void s3c2443_irq_uart3_unmask(unsigned int irqno)
  142. {
  143. s3c_irqsub_unmask(irqno, INTMSK_UART3);
  144. }
  145. static void s3c2443_irq_uart3_ack(unsigned int irqno)
  146. {
  147. s3c_irqsub_maskack(irqno, INTMSK_UART3, SUBMSK_UART3);
  148. }
  149. static struct irq_chip s3c2443_irq_uart3 = {
  150. .mask = s3c2443_irq_uart3_mask,
  151. .unmask = s3c2443_irq_uart3_unmask,
  152. .ack = s3c2443_irq_uart3_ack,
  153. };
  154. /* CAM sub interrupts */
  155. static void s3c2443_irq_demux_cam(unsigned int irq, struct irq_desc *desc)
  156. {
  157. s3c2443_irq_demux(IRQ_S3C2440_CAM_C, 4);
  158. }
  159. #define INTMSK_CAM (1UL << (IRQ_CAM - IRQ_EINT0))
  160. #define SUBMSK_CAM INTMSK(IRQ_S3C2440_CAM_C, IRQ_S3C2440_CAM_P)
  161. static void s3c2443_irq_cam_mask(unsigned int irqno)
  162. {
  163. s3c_irqsub_mask(irqno, INTMSK_CAM, SUBMSK_CAM);
  164. }
  165. static void s3c2443_irq_cam_unmask(unsigned int irqno)
  166. {
  167. s3c_irqsub_unmask(irqno, INTMSK_CAM);
  168. }
  169. static void s3c2443_irq_cam_ack(unsigned int irqno)
  170. {
  171. s3c_irqsub_maskack(irqno, INTMSK_CAM, SUBMSK_CAM);
  172. }
  173. static struct irq_chip s3c2443_irq_cam = {
  174. .mask = s3c2443_irq_cam_mask,
  175. .unmask = s3c2443_irq_cam_unmask,
  176. .ack = s3c2443_irq_cam_ack,
  177. };
  178. /* IRQ initialisation code */
  179. static int __init s3c2443_add_sub(unsigned int base,
  180. void (*demux)(unsigned int,
  181. struct irq_desc *),
  182. struct irq_chip *chip,
  183. unsigned int start, unsigned int end)
  184. {
  185. unsigned int irqno;
  186. set_irq_chip(base, &s3c_irq_level_chip);
  187. set_irq_handler(base, handle_level_irq);
  188. set_irq_chained_handler(base, demux);
  189. for (irqno = start; irqno <= end; irqno++) {
  190. set_irq_chip(irqno, chip);
  191. set_irq_handler(irqno, handle_level_irq);
  192. set_irq_flags(irqno, IRQF_VALID);
  193. }
  194. return 0;
  195. }
  196. static int s3c2443_irq_add(struct sys_device *sysdev)
  197. {
  198. printk("S3C2443: IRQ Support\n");
  199. s3c2443_add_sub(IRQ_CAM, s3c2443_irq_demux_cam, &s3c2443_irq_cam,
  200. IRQ_S3C2440_CAM_C, IRQ_S3C2440_CAM_P);
  201. s3c2443_add_sub(IRQ_LCD, s3c2443_irq_demux_lcd, &s3c2443_irq_lcd,
  202. IRQ_S3C2443_LCD1, IRQ_S3C2443_LCD4);
  203. s3c2443_add_sub(IRQ_S3C2443_DMA, s3c2443_irq_demux_dma,
  204. &s3c2443_irq_dma, IRQ_S3C2443_DMA0, IRQ_S3C2443_DMA5);
  205. s3c2443_add_sub(IRQ_S3C2443_UART3, s3c2443_irq_demux_uart3,
  206. &s3c2443_irq_uart3,
  207. IRQ_S3C2443_RX3, IRQ_S3C2443_ERR3);
  208. s3c2443_add_sub(IRQ_WDT, s3c2443_irq_demux_wdtac97,
  209. &s3c2443_irq_wdtac97,
  210. IRQ_S3C2443_WDT, IRQ_S3C2443_AC97);
  211. return 0;
  212. }
  213. static struct sysdev_driver s3c2443_irq_driver = {
  214. .add = s3c2443_irq_add,
  215. };
  216. static int s3c2443_irq_init(void)
  217. {
  218. return sysdev_driver_register(&s3c2443_sysclass, &s3c2443_irq_driver);
  219. }
  220. arch_initcall(s3c2443_irq_init);