setup_cqreek.c
上传用户:jlfgdled
上传日期:2013-04-10
资源大小:33168k
文件大小:6k
源码类别:

Linux/Unix编程

开发平台:

Unix_Linux

  1. /* $Id: setup_cqreek.c,v 1.9 2001/07/30 12:43:28 gniibe Exp $
  2.  *
  3.  * arch/sh/kernel/setup_cqreek.c
  4.  *
  5.  * Copyright (C) 2000  Niibe Yutaka
  6.  *
  7.  * CqREEK IDE/ISA Bridge Support.
  8.  *
  9.  */
  10. #include <linux/config.h>
  11. #include <linux/kernel.h>
  12. #include <linux/init.h>
  13. #include <linux/irq.h>
  14. #include <asm/io.h>
  15. #include <asm/io_generic.h>
  16. #include <asm/irq.h>
  17. #include <asm/machvec.h>
  18. #include <asm/machvec_init.h>
  19. #include <asm/rtc.h>
  20. #define BRIDGE_FEATURE 0x0002
  21. #define BRIDGE_IDE_CTRL 0x0018
  22. #define BRIDGE_IDE_INTR_LVL     0x001A
  23. #define BRIDGE_IDE_INTR_MASK 0x001C
  24. #define BRIDGE_IDE_INTR_STAT 0x001E
  25. #define BRIDGE_ISA_CTRL 0x0028
  26. #define BRIDGE_ISA_INTR_LVL     0x002A
  27. #define BRIDGE_ISA_INTR_MASK 0x002C
  28. #define BRIDGE_ISA_INTR_STAT 0x002E
  29. #define IDE_OFFSET 0xA4000000UL
  30. #define ISA_OFFSET 0xA4A00000UL
  31. static unsigned long cqreek_port2addr(unsigned long port)
  32. {
  33. if (0x0000<=port && port<=0x0040)
  34. return IDE_OFFSET + port;
  35. if ((0x01f0<=port && port<=0x01f7) || port == 0x03f6)
  36. return IDE_OFFSET + port;
  37. return ISA_OFFSET + port;
  38. }
  39. struct cqreek_irq_data {
  40. unsigned short mask_port; /* Port of Interrupt Mask Register */
  41. unsigned short stat_port; /* Port of Interrupt Status Register */
  42. unsigned short bit; /* Value of the bit */
  43. };
  44. static struct cqreek_irq_data cqreek_irq_data[NR_IRQS];
  45. static void disable_cqreek_irq(unsigned int irq)
  46. {
  47. unsigned long flags;
  48. unsigned short mask;
  49. unsigned short mask_port = cqreek_irq_data[irq].mask_port;
  50. unsigned short bit = cqreek_irq_data[irq].bit;
  51. save_and_cli(flags);
  52. /* Disable IRQ */
  53. mask = inw(mask_port) & ~bit;
  54. outw_p(mask, mask_port);
  55. restore_flags(flags);
  56. }
  57. static void enable_cqreek_irq(unsigned int irq)
  58. {
  59. unsigned long flags;
  60. unsigned short mask;
  61. unsigned short mask_port = cqreek_irq_data[irq].mask_port;
  62. unsigned short bit = cqreek_irq_data[irq].bit;
  63. save_and_cli(flags);
  64. /* Enable IRQ */
  65. mask = inw(mask_port) | bit;
  66. outw_p(mask, mask_port);
  67. restore_flags(flags);
  68. }
  69. static void mask_and_ack_cqreek(unsigned int irq)
  70. {
  71. unsigned short stat_port = cqreek_irq_data[irq].stat_port;
  72. unsigned short bit = cqreek_irq_data[irq].bit;
  73. disable_cqreek_irq(irq);
  74. /* Clear IRQ (it might be edge IRQ) */
  75. inw(stat_port);
  76. outw_p(bit, stat_port);
  77. }
  78. static void end_cqreek_irq(unsigned int irq)
  79. {
  80. if (!(irq_desc[irq].status & (IRQ_DISABLED|IRQ_INPROGRESS)))
  81. enable_cqreek_irq(irq);
  82. }
  83. static unsigned int startup_cqreek_irq(unsigned int irq)
  84. enable_cqreek_irq(irq);
  85. return 0;
  86. }
  87. static void shutdown_cqreek_irq(unsigned int irq)
  88. {
  89. disable_cqreek_irq(irq);
  90. }
  91. static struct hw_interrupt_type cqreek_irq_type = {
  92. "CqREEK-IRQ",
  93. startup_cqreek_irq,
  94. shutdown_cqreek_irq,
  95. enable_cqreek_irq,
  96. disable_cqreek_irq,
  97. mask_and_ack_cqreek,
  98. end_cqreek_irq
  99. };
  100. static int has_ide, has_isa;
  101. /* XXX: This is just for test for my NE2000 ISA board
  102.    What we really need is virtualized IRQ and demultiplexer like HP600 port */
  103. void __init init_cqreek_IRQ(void)
  104. {
  105. if (has_ide) {
  106. cqreek_irq_data[14].mask_port = BRIDGE_IDE_INTR_MASK;
  107. cqreek_irq_data[14].stat_port = BRIDGE_IDE_INTR_STAT;
  108. cqreek_irq_data[14].bit = 1;
  109. irq_desc[14].handler = &cqreek_irq_type;
  110. irq_desc[14].status = IRQ_DISABLED;
  111. irq_desc[14].action = 0;
  112. irq_desc[14].depth = 1;
  113. disable_cqreek_irq(14);
  114. }
  115. if (has_isa) {
  116. cqreek_irq_data[10].mask_port = BRIDGE_ISA_INTR_MASK;
  117. cqreek_irq_data[10].stat_port = BRIDGE_ISA_INTR_STAT;
  118. cqreek_irq_data[10].bit = (1 << 10);
  119. /* XXX: Err... we may need demultiplexer for ISA irq... */
  120. irq_desc[10].handler = &cqreek_irq_type;
  121. irq_desc[10].status = IRQ_DISABLED;
  122. irq_desc[10].action = 0;
  123. irq_desc[10].depth = 1;
  124. disable_cqreek_irq(10);
  125. }
  126. }
  127. /*
  128.  * Initialize the board
  129.  */
  130. void __init setup_cqreek(void)
  131. {
  132. int i;
  133. /* udelay is not available at setup time yet... */
  134. #define DELAY() do {for (i=0; i<10000; i++) ctrl_inw(0xa0000000);} while(0)
  135. if ((inw (BRIDGE_FEATURE) & 1)) { /* We have IDE interface */
  136. outw_p(0, BRIDGE_IDE_INTR_LVL);
  137. outw_p(0, BRIDGE_IDE_INTR_MASK);
  138. outw_p(0, BRIDGE_IDE_CTRL);
  139. DELAY();
  140. outw_p(0x8000, BRIDGE_IDE_CTRL);
  141. DELAY();
  142. outw_p(0xffff, BRIDGE_IDE_INTR_STAT); /* Clear interrupt status */
  143. outw_p(0x0f-14, BRIDGE_IDE_INTR_LVL); /* Use 14 IPR */
  144. outw_p(1, BRIDGE_IDE_INTR_MASK); /* Enable interrupt */
  145. has_ide=1;
  146. }
  147. if ((inw (BRIDGE_FEATURE) & 2)) { /* We have ISA interface */
  148. outw_p(0, BRIDGE_ISA_INTR_LVL);
  149. outw_p(0, BRIDGE_ISA_INTR_MASK);
  150. outw_p(0, BRIDGE_ISA_CTRL);
  151. DELAY();
  152. outw_p(0x8000, BRIDGE_ISA_CTRL);
  153. DELAY();
  154. outw_p(0xffff, BRIDGE_ISA_INTR_STAT); /* Clear interrupt status */
  155. outw_p(0x0f-10, BRIDGE_ISA_INTR_LVL); /* Use 10 IPR */
  156. outw_p(0xfff8, BRIDGE_ISA_INTR_MASK); /* Enable interrupt */
  157. has_isa=1;
  158. }
  159. printk(KERN_INFO "CqREEK Setup (IDE=%d, ISA=%d)...donen", has_ide, has_isa);
  160. }
  161. /*
  162.  * The Machine Vector
  163.  */
  164. struct sh_machine_vector mv_cqreek __initmv = {
  165. mv_name: "CqREEK",
  166. #if defined(__SH4__)
  167. mv_nr_irqs: 48,
  168. #elif defined(CONFIG_CPU_SUBTYPE_SH7708)
  169. mv_nr_irqs: 32,
  170. #elif defined(CONFIG_CPU_SUBTYPE_SH7709)
  171. mv_nr_irqs: 61,
  172. #endif
  173. mv_inb: generic_inb,
  174. mv_inw: generic_inw,
  175. mv_inl: generic_inl,
  176. mv_outb: generic_outb,
  177. mv_outw: generic_outw,
  178. mv_outl: generic_outl,
  179. mv_inb_p: generic_inb_p,
  180. mv_inw_p: generic_inw_p,
  181. mv_inl_p: generic_inl_p,
  182. mv_outb_p: generic_outb_p,
  183. mv_outw_p: generic_outw_p,
  184. mv_outl_p: generic_outl_p,
  185. mv_insb: generic_insb,
  186. mv_insw: generic_insw,
  187. mv_insl: generic_insl,
  188. mv_outsb: generic_outsb,
  189. mv_outsw: generic_outsw,
  190. mv_outsl: generic_outsl,
  191. mv_readb: generic_readb,
  192. mv_readw: generic_readw,
  193. mv_readl: generic_readl,
  194. mv_writeb: generic_writeb,
  195. mv_writew: generic_writew,
  196. mv_writel: generic_writel,
  197. mv_init_arch: setup_cqreek,
  198. mv_init_irq: init_cqreek_IRQ,
  199. mv_isa_port2addr: cqreek_port2addr,
  200. mv_ioremap: generic_ioremap,
  201. mv_iounmap: generic_iounmap,
  202. mv_rtc_gettimeofday: sh_rtc_gettimeofday,
  203. mv_rtc_settimeofday: sh_rtc_settimeofday,
  204. };
  205. ALIAS_MV(cqreek)