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

Linux/Unix编程

开发平台:

Unix_Linux

  1. /*
  2.  * sgiwd93.c: SGI WD93 scsi driver.
  3.  *
  4.  * Copyright (C) 1996 David S. Miller (dm@engr.sgi.com)
  5.  *  1999 Andrew R. Baker (andrewb@uab.edu)
  6.  *       - Support for 2nd SCSI controller on Indigo2
  7.  *  2001 Florian Lohoff (flo@rfc822.org)
  8.  *       - Delete HPC scatter gather (Read corruption on 
  9.  *         multiple disks)
  10.  *       - Cleanup wback cache handling
  11.  * 
  12.  * (In all truth, Jed Schimmel wrote all this code.)
  13.  *
  14.  */
  15. #include <linux/init.h>
  16. #include <linux/types.h>
  17. #include <linux/mm.h>
  18. #include <linux/blk.h>
  19. #include <linux/version.h>
  20. #include <linux/delay.h>
  21. #include <linux/spinlock.h>
  22. #include <asm/page.h>
  23. #include <asm/pgtable.h>
  24. #include <asm/sgialib.h>
  25. #include <asm/sgi/sgi.h>
  26. #include <asm/sgi/sgimc.h>
  27. #include <asm/sgi/sgihpc.h>
  28. #include <asm/sgi/sgint23.h>
  29. #include <asm/irq.h>
  30. #include <asm/io.h>
  31. #include "scsi.h"
  32. #include "hosts.h"
  33. #include "wd33c93.h"
  34. #include "sgiwd93.h"
  35. #include <linux/stat.h>
  36. struct hpc_chunk {
  37. struct hpc_dma_desc desc;
  38. u32 _padding; /* align to quadword boundary */
  39. };
  40. struct Scsi_Host *sgiwd93_host = NULL;
  41. struct Scsi_Host *sgiwd93_host1 = NULL;
  42. /* Wuff wuff, wuff, wd33c93.c, wuff wuff, object oriented, bow wow. */
  43. static inline void write_wd33c93_count(const wd33c93_regs regs,
  44.                                       unsigned long value)
  45. {
  46. *regs.SASR = WD_TRANSFER_COUNT_MSB;
  47. mb();
  48. *regs.SCMD = ((value >> 16) & 0xff);
  49. *regs.SCMD = ((value >>  8) & 0xff);
  50. *regs.SCMD = ((value >>  0) & 0xff);
  51. mb();
  52. }
  53. static inline unsigned long read_wd33c93_count(const wd33c93_regs regs)
  54. {
  55. unsigned long value;
  56. *regs.SASR = WD_TRANSFER_COUNT_MSB;
  57. mb();
  58. value =  ((*regs.SCMD & 0xff) << 16);
  59. value |= ((*regs.SCMD & 0xff) <<  8);
  60. value |= ((*regs.SCMD & 0xff) <<  0);
  61. mb();
  62. return value;
  63. }
  64. /* XXX woof! */
  65. static void sgiwd93_intr(int irq, void *dev_id, struct pt_regs *regs)
  66. {
  67. unsigned long flags;
  68. spin_lock_irqsave(&io_request_lock, flags);
  69. wd33c93_intr((struct Scsi_Host *) dev_id);
  70. spin_unlock_irqrestore(&io_request_lock, flags);
  71. }
  72. #undef DEBUG_DMA
  73. static inline
  74. void fill_hpc_entries (struct hpc_chunk **hcp, char *addr, unsigned long len)
  75. {
  76. unsigned long physaddr;
  77. unsigned long count;
  78. physaddr = PHYSADDR(addr);
  79. while (len) {
  80. /*
  81.  * even cntinfo could be up to 16383, without
  82.  * magic only 8192 works correctly
  83.  */
  84. count = len > 8192 ? 8192 : len;
  85. (*hcp)->desc.pbuf = physaddr;
  86. (*hcp)->desc.cntinfo = count;
  87. (*hcp)++;
  88. len -= count;
  89. physaddr += count;
  90. }
  91. }
  92. static int dma_setup(Scsi_Cmnd *cmd, int datainp)
  93. {
  94. struct WD33C93_hostdata *hdata = (struct WD33C93_hostdata *)cmd->host->hostdata;
  95. struct hpc3_scsiregs *hregs = (struct hpc3_scsiregs *) cmd->host->base;
  96. struct hpc_chunk *hcp = (struct hpc_chunk *) hdata->dma_bounce_buffer;
  97. #ifdef DEBUG_DMA
  98. printk("dma_setup: datainp<%d> hcp<%p> ",
  99.        datainp, hcp);
  100. #endif
  101. hdata->dma_dir = datainp;
  102. /*
  103.  * wd33c93 shouldn't pass us bogus dma_setups, but
  104.  * it does:-( The other wd33c93 drivers deal with
  105.  * it the same way (which isn't that obvious).
  106.  * IMHO a better fix would be, not to do these
  107.  * dma setups in the first place
  108.  */
  109. if (cmd->SCp.ptr == NULL)
  110. return 1;
  111. fill_hpc_entries (&hcp, cmd->SCp.ptr,cmd->SCp.this_residual);
  112. /* To make sure, if we trip an HPC bug, that we transfer
  113.  * every single byte, we tag on an extra zero length dma
  114.  * descriptor at the end of the chain.
  115.  */
  116. hcp->desc.pbuf = 0;
  117. hcp->desc.cntinfo = (HPCDMA_EOX);
  118. #ifdef DEBUG_DMA
  119. printk(" HPCGOn");
  120. #endif
  121. /* Start up the HPC. */
  122. hregs->ndptr = PHYSADDR(hdata->dma_bounce_buffer);
  123. if(datainp) {
  124. dma_cache_inv((unsigned long) cmd->SCp.ptr, cmd->SCp.this_residual);
  125. hregs->ctrl = (HPC3_SCTRL_ACTIVE);
  126. } else {
  127. dma_cache_wback_inv((unsigned long) cmd->SCp.ptr, cmd->SCp.this_residual);
  128. hregs->ctrl = (HPC3_SCTRL_ACTIVE | HPC3_SCTRL_DIR);
  129. }
  130. return 0;
  131. }
  132. static void dma_stop(struct Scsi_Host *instance, Scsi_Cmnd *SCpnt,
  133.      int status)
  134. {
  135. struct WD33C93_hostdata *hdata = (struct WD33C93_hostdata *)instance->hostdata;
  136. struct hpc3_scsiregs *hregs;
  137. if (!SCpnt)
  138. return;
  139. hregs = (struct hpc3_scsiregs *) SCpnt->host->base;
  140. #ifdef DEBUG_DMA
  141. printk("dma_stop: status<%d> ", status);
  142. #endif
  143. /* First stop the HPC and flush it's FIFO. */
  144. if(hdata->dma_dir) {
  145. hregs->ctrl |= HPC3_SCTRL_FLUSH;
  146. while(hregs->ctrl & HPC3_SCTRL_ACTIVE)
  147. barrier();
  148. }
  149. hregs->ctrl = 0;
  150. #ifdef DEBUG_DMA
  151. printk("n");
  152. #endif
  153. }
  154. void sgiwd93_reset(unsigned long base)
  155. {
  156. struct hpc3_scsiregs *hregs = (struct hpc3_scsiregs *) base;
  157. hregs->ctrl = HPC3_SCTRL_CRESET;
  158. udelay (50);
  159. hregs->ctrl = 0;
  160. }
  161. static inline void init_hpc_chain(uchar *buf)
  162. {
  163. struct hpc_chunk *hcp = (struct hpc_chunk *) buf;
  164. unsigned long start, end;
  165. start = (unsigned long) buf;
  166. end = start + PAGE_SIZE;
  167. while(start < end) {
  168. hcp->desc.pnext = PHYSADDR((hcp + 1));
  169. hcp->desc.cntinfo = HPCDMA_EOX;
  170. hcp++;
  171. start += sizeof(struct hpc_chunk);
  172. };
  173. hcp--;
  174. hcp->desc.pnext = PHYSADDR(buf);
  175. /* Force flush to memory */
  176. dma_cache_wback_inv((unsigned long) buf, PAGE_SIZE);
  177. }
  178. int __init sgiwd93_detect(Scsi_Host_Template *SGIblows)
  179. {
  180. static unsigned char called = 0;
  181. struct hpc3_scsiregs *hregs = &hpc3c0->scsi_chan0;
  182. struct hpc3_scsiregs *hregs1 = &hpc3c0->scsi_chan1;
  183. struct WD33C93_hostdata *hdata;
  184. struct WD33C93_hostdata *hdata1;
  185. wd33c93_regs regs;
  186. uchar *buf;
  187. if(called)
  188. return 0; /* Should bitch on the console about this... */
  189. SGIblows->proc_name = "SGIWD93";
  190. sgiwd93_host = scsi_register(SGIblows, sizeof(struct WD33C93_hostdata));
  191. if(sgiwd93_host == NULL)
  192. return 0;
  193. sgiwd93_host->base = (unsigned long) hregs;
  194. sgiwd93_host->irq = SGI_WD93_0_IRQ;
  195. buf = (uchar *) get_free_page(GFP_KERNEL);
  196. if (!buf) {
  197. printk(KERN_WARNING "sgiwd93: Could not allocate memory for host0 buffer.n");
  198. scsi_unregister(sgiwd93_host);
  199. return 0;
  200. }
  201. init_hpc_chain(buf);
  202. /* HPC_SCSI_REG0 | 0x03 | KSEG1 */
  203. regs.SASR = (unsigned char*) KSEG1ADDR (0x1fbc0003);
  204. regs.SCMD = (unsigned char*) KSEG1ADDR (0x1fbc0007);
  205. wd33c93_init(sgiwd93_host, regs, dma_setup, dma_stop, WD33C93_FS_16_20);
  206. hdata = (struct WD33C93_hostdata *)sgiwd93_host->hostdata;
  207. hdata->no_sync = 0;
  208. hdata->dma_bounce_buffer = (uchar *) (KSEG1ADDR(buf));
  209. if (request_irq(SGI_WD93_0_IRQ, sgiwd93_intr, 0, "SGI WD93", (void *) sgiwd93_host)) {
  210. printk(KERN_WARNING "sgiwd93: Could not register IRQ %d (for host 0).n", SGI_WD93_0_IRQ);
  211. #ifdef MODULE
  212. wd33c93_release();
  213. #endif
  214. free_page((unsigned long)buf);
  215. scsi_unregister(sgiwd93_host);
  216. return 0;
  217. }
  218.         /* set up second controller on the Indigo2 */
  219. if(!sgi_guiness) {
  220. sgiwd93_host1 = scsi_register(SGIblows, sizeof(struct WD33C93_hostdata));
  221. if(sgiwd93_host1 != NULL)
  222. {
  223. sgiwd93_host1->base = (unsigned long) hregs1;
  224. sgiwd93_host1->irq = SGI_WD93_1_IRQ;
  225. buf = (uchar *) get_free_page(GFP_KERNEL);
  226. if (!buf) {
  227. printk(KERN_WARNING "sgiwd93: Could not allocate memory for host1 buffer.n");
  228. scsi_unregister(sgiwd93_host1);
  229. called = 1;
  230. return 1; /* We registered host0 so return success*/
  231. }
  232. init_hpc_chain(buf);
  233. /* HPC_SCSI_REG1 | 0x03 | KSEG1 */
  234. regs.SASR = (unsigned char*) KSEG1ADDR(0x1fbc8003);
  235. regs.SCMD = (unsigned char*) KSEG1ADDR(0x1fbc8007);
  236. wd33c93_init(sgiwd93_host1, regs, dma_setup, dma_stop,
  237.              WD33C93_FS_16_20);
  238. hdata1 = (struct WD33C93_hostdata *)sgiwd93_host1->hostdata;
  239. hdata1->no_sync = 0;
  240. hdata1->dma_bounce_buffer = (uchar *) (KSEG1ADDR(buf));
  241. if (request_irq(SGI_WD93_1_IRQ, sgiwd93_intr, 0, "SGI WD93", (void *) sgiwd93_host1)) {
  242. printk(KERN_WARNING "sgiwd93: Could not allocate irq %d (for host1).n", SGI_WD93_1_IRQ);
  243. #ifdef MODULE
  244. wd33c93_release();
  245. #endif
  246. free_page((unsigned long)buf);
  247. scsi_unregister(sgiwd93_host1);
  248. /* Fall through since host0 registered OK */
  249. }
  250. }
  251. }
  252. called = 1;
  253. return 1; /* Found one. */
  254. }
  255. #define HOSTS_C
  256. #include "sgiwd93.h"
  257. static Scsi_Host_Template driver_template = SGIWD93_SCSI;
  258. #include "scsi_module.c"
  259. int sgiwd93_release(struct Scsi_Host *instance)
  260. {
  261. #ifdef MODULE
  262. free_irq(SGI_WD93_0_IRQ, sgiwd93_intr);
  263. free_page(KSEG0ADDR(hdata->dma_bounce_buffer));
  264. wd33c93_release();
  265. if(!sgi_guiness) {
  266. free_irq(SGI_WD93_1_IRQ, sgiwd93_intr);
  267. free_page(KSEG0ADDR(hdata1->dma_bounce_buffer));
  268. wd33c93_release();
  269. }
  270. #endif
  271. return 1;
  272. }