162306a36Sopenharmony_ci/* $Id$ 262306a36Sopenharmony_ci * 1993/03/31 362306a36Sopenharmony_ci * linux/kernel/aha1740.c 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Based loosely on aha1542.c which is 662306a36Sopenharmony_ci * Copyright (C) 1992 Tommy Thorn and 762306a36Sopenharmony_ci * Modified by Eric Youngdale 862306a36Sopenharmony_ci * 962306a36Sopenharmony_ci * This file is aha1740.c, written and 1062306a36Sopenharmony_ci * Copyright (C) 1992,1993 Brad McLean 1162306a36Sopenharmony_ci * brad@saturn.gaylord.com or brad@bradpc.gaylord.com. 1262306a36Sopenharmony_ci * 1362306a36Sopenharmony_ci * Modifications to makecode and queuecommand 1462306a36Sopenharmony_ci * for proper handling of multiple devices courteously 1562306a36Sopenharmony_ci * provided by Michael Weller, March, 1993 1662306a36Sopenharmony_ci * 1762306a36Sopenharmony_ci * Multiple adapter support, extended translation detection, 1862306a36Sopenharmony_ci * update to current scsi subsystem changes, proc fs support, 1962306a36Sopenharmony_ci * working (!) module support based on patches from Andreas Arens, 2062306a36Sopenharmony_ci * by Andreas Degert <ad@papyrus.hamburg.com>, 2/1997 2162306a36Sopenharmony_ci * 2262306a36Sopenharmony_ci * aha1740_makecode may still need even more work 2362306a36Sopenharmony_ci * if it doesn't work for your devices, take a look. 2462306a36Sopenharmony_ci * 2562306a36Sopenharmony_ci * Reworked for new_eh and new locking by Alan Cox <alan@lxorguk.ukuu.org.uk> 2662306a36Sopenharmony_ci * 2762306a36Sopenharmony_ci * Converted to EISA and generic DMA APIs by Marc Zyngier 2862306a36Sopenharmony_ci * <maz@wild-wind.fr.eu.org>, 4/2003. 2962306a36Sopenharmony_ci * 3062306a36Sopenharmony_ci * Shared interrupt support added by Rask Ingemann Lambertsen 3162306a36Sopenharmony_ci * <rask@sygehus.dk>, 10/2003 3262306a36Sopenharmony_ci * 3362306a36Sopenharmony_ci * For the avoidance of doubt the "preferred form" of this code is one which 3462306a36Sopenharmony_ci * is in an open non patent encumbered format. Where cryptographic key signing 3562306a36Sopenharmony_ci * forms part of the process of creating an executable the information 3662306a36Sopenharmony_ci * including keys needed to generate an equivalently functional executable 3762306a36Sopenharmony_ci * are deemed to be part of the source code. 3862306a36Sopenharmony_ci */ 3962306a36Sopenharmony_ci 4062306a36Sopenharmony_ci#include <linux/blkdev.h> 4162306a36Sopenharmony_ci#include <linux/interrupt.h> 4262306a36Sopenharmony_ci#include <linux/module.h> 4362306a36Sopenharmony_ci#include <linux/kernel.h> 4462306a36Sopenharmony_ci#include <linux/types.h> 4562306a36Sopenharmony_ci#include <linux/string.h> 4662306a36Sopenharmony_ci#include <linux/ioport.h> 4762306a36Sopenharmony_ci#include <linux/proc_fs.h> 4862306a36Sopenharmony_ci#include <linux/stat.h> 4962306a36Sopenharmony_ci#include <linux/init.h> 5062306a36Sopenharmony_ci#include <linux/device.h> 5162306a36Sopenharmony_ci#include <linux/eisa.h> 5262306a36Sopenharmony_ci#include <linux/dma-mapping.h> 5362306a36Sopenharmony_ci#include <linux/gfp.h> 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_ci#include <asm/dma.h> 5662306a36Sopenharmony_ci#include <asm/io.h> 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_ci#include <scsi/scsi.h> 5962306a36Sopenharmony_ci#include <scsi/scsi_cmnd.h> 6062306a36Sopenharmony_ci#include <scsi/scsi_device.h> 6162306a36Sopenharmony_ci#include <scsi/scsi_eh.h> 6262306a36Sopenharmony_ci#include <scsi/scsi_host.h> 6362306a36Sopenharmony_ci#include <scsi/scsi_tcq.h> 6462306a36Sopenharmony_ci#include "aha1740.h" 6562306a36Sopenharmony_ci 6662306a36Sopenharmony_ci/* IF YOU ARE HAVING PROBLEMS WITH THIS DRIVER, AND WANT TO WATCH 6762306a36Sopenharmony_ci IT WORK, THEN: 6862306a36Sopenharmony_ci#define DEBUG 6962306a36Sopenharmony_ci*/ 7062306a36Sopenharmony_ci#ifdef DEBUG 7162306a36Sopenharmony_ci#define DEB(x) x 7262306a36Sopenharmony_ci#else 7362306a36Sopenharmony_ci#define DEB(x) 7462306a36Sopenharmony_ci#endif 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_cistruct aha1740_hostdata { 7762306a36Sopenharmony_ci struct eisa_device *edev; 7862306a36Sopenharmony_ci unsigned int translation; 7962306a36Sopenharmony_ci unsigned int last_ecb_used; 8062306a36Sopenharmony_ci dma_addr_t ecb_dma_addr; 8162306a36Sopenharmony_ci struct ecb ecb[AHA1740_ECBS]; 8262306a36Sopenharmony_ci}; 8362306a36Sopenharmony_ci 8462306a36Sopenharmony_cistruct aha1740_sg { 8562306a36Sopenharmony_ci struct aha1740_chain sg_chain[AHA1740_SCATTER]; 8662306a36Sopenharmony_ci dma_addr_t sg_dma_addr; 8762306a36Sopenharmony_ci dma_addr_t buf_dma_addr; 8862306a36Sopenharmony_ci}; 8962306a36Sopenharmony_ci 9062306a36Sopenharmony_ci#define HOSTDATA(host) ((struct aha1740_hostdata *) &host->hostdata) 9162306a36Sopenharmony_ci 9262306a36Sopenharmony_cistatic inline struct ecb *ecb_dma_to_cpu (struct Scsi_Host *host, 9362306a36Sopenharmony_ci dma_addr_t dma) 9462306a36Sopenharmony_ci{ 9562306a36Sopenharmony_ci struct aha1740_hostdata *hdata = HOSTDATA (host); 9662306a36Sopenharmony_ci dma_addr_t offset; 9762306a36Sopenharmony_ci 9862306a36Sopenharmony_ci offset = dma - hdata->ecb_dma_addr; 9962306a36Sopenharmony_ci 10062306a36Sopenharmony_ci return (struct ecb *)(((char *) hdata->ecb) + (unsigned int) offset); 10162306a36Sopenharmony_ci} 10262306a36Sopenharmony_ci 10362306a36Sopenharmony_cistatic inline dma_addr_t ecb_cpu_to_dma (struct Scsi_Host *host, void *cpu) 10462306a36Sopenharmony_ci{ 10562306a36Sopenharmony_ci struct aha1740_hostdata *hdata = HOSTDATA (host); 10662306a36Sopenharmony_ci dma_addr_t offset; 10762306a36Sopenharmony_ci 10862306a36Sopenharmony_ci offset = (char *) cpu - (char *) hdata->ecb; 10962306a36Sopenharmony_ci 11062306a36Sopenharmony_ci return hdata->ecb_dma_addr + offset; 11162306a36Sopenharmony_ci} 11262306a36Sopenharmony_ci 11362306a36Sopenharmony_cistatic int aha1740_show_info(struct seq_file *m, struct Scsi_Host *shpnt) 11462306a36Sopenharmony_ci{ 11562306a36Sopenharmony_ci struct aha1740_hostdata *host = HOSTDATA(shpnt); 11662306a36Sopenharmony_ci seq_printf(m, "aha174x at IO:%lx, IRQ %d, SLOT %d.\n" 11762306a36Sopenharmony_ci "Extended translation %sabled.\n", 11862306a36Sopenharmony_ci shpnt->io_port, shpnt->irq, host->edev->slot, 11962306a36Sopenharmony_ci host->translation ? "en" : "dis"); 12062306a36Sopenharmony_ci return 0; 12162306a36Sopenharmony_ci} 12262306a36Sopenharmony_ci 12362306a36Sopenharmony_cistatic int aha1740_makecode(unchar *sense, unchar *status) 12462306a36Sopenharmony_ci{ 12562306a36Sopenharmony_ci struct statusword 12662306a36Sopenharmony_ci { 12762306a36Sopenharmony_ci ushort don:1, /* Command Done - No Error */ 12862306a36Sopenharmony_ci du:1, /* Data underrun */ 12962306a36Sopenharmony_ci :1, qf:1, /* Queue full */ 13062306a36Sopenharmony_ci sc:1, /* Specification Check */ 13162306a36Sopenharmony_ci dor:1, /* Data overrun */ 13262306a36Sopenharmony_ci ch:1, /* Chaining Halted */ 13362306a36Sopenharmony_ci intr:1, /* Interrupt issued */ 13462306a36Sopenharmony_ci asa:1, /* Additional Status Available */ 13562306a36Sopenharmony_ci sns:1, /* Sense information Stored */ 13662306a36Sopenharmony_ci :1, ini:1, /* Initialization Required */ 13762306a36Sopenharmony_ci me:1, /* Major error or exception */ 13862306a36Sopenharmony_ci :1, eca:1, /* Extended Contingent alliance */ 13962306a36Sopenharmony_ci :1; 14062306a36Sopenharmony_ci } status_word; 14162306a36Sopenharmony_ci int retval = DID_OK; 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_ci status_word = * (struct statusword *) status; 14462306a36Sopenharmony_ci#ifdef DEBUG 14562306a36Sopenharmony_ci printk("makecode from %x,%x,%x,%x %x,%x,%x,%x", 14662306a36Sopenharmony_ci status[0], status[1], status[2], status[3], 14762306a36Sopenharmony_ci sense[0], sense[1], sense[2], sense[3]); 14862306a36Sopenharmony_ci#endif 14962306a36Sopenharmony_ci if (!status_word.don) { /* Anything abnormal was detected */ 15062306a36Sopenharmony_ci if ( (status[1]&0x18) || status_word.sc ) { 15162306a36Sopenharmony_ci /*Additional info available*/ 15262306a36Sopenharmony_ci /* Use the supplied info for further diagnostics */ 15362306a36Sopenharmony_ci switch ( status[2] ) { 15462306a36Sopenharmony_ci case 0x12: 15562306a36Sopenharmony_ci if ( status_word.dor ) 15662306a36Sopenharmony_ci retval=DID_ERROR; /* It's an Overrun */ 15762306a36Sopenharmony_ci /* If not overrun, assume underrun and 15862306a36Sopenharmony_ci * ignore it! */ 15962306a36Sopenharmony_ci break; 16062306a36Sopenharmony_ci case 0x00: /* No info, assume no error, should 16162306a36Sopenharmony_ci * not occur */ 16262306a36Sopenharmony_ci break; 16362306a36Sopenharmony_ci case 0x11: 16462306a36Sopenharmony_ci case 0x21: 16562306a36Sopenharmony_ci retval=DID_TIME_OUT; 16662306a36Sopenharmony_ci break; 16762306a36Sopenharmony_ci case 0x0a: 16862306a36Sopenharmony_ci retval=DID_BAD_TARGET; 16962306a36Sopenharmony_ci break; 17062306a36Sopenharmony_ci case 0x04: 17162306a36Sopenharmony_ci case 0x05: 17262306a36Sopenharmony_ci retval=DID_ABORT; 17362306a36Sopenharmony_ci /* Either by this driver or the 17462306a36Sopenharmony_ci * AHA1740 itself */ 17562306a36Sopenharmony_ci break; 17662306a36Sopenharmony_ci default: 17762306a36Sopenharmony_ci retval=DID_ERROR; /* No further 17862306a36Sopenharmony_ci * diagnostics 17962306a36Sopenharmony_ci * possible */ 18062306a36Sopenharmony_ci } 18162306a36Sopenharmony_ci } else { 18262306a36Sopenharmony_ci /* Michael suggests, and Brad concurs: */ 18362306a36Sopenharmony_ci if ( status_word.qf ) { 18462306a36Sopenharmony_ci retval = DID_TIME_OUT; /* forces a redo */ 18562306a36Sopenharmony_ci /* I think this specific one should 18662306a36Sopenharmony_ci * not happen -Brad */ 18762306a36Sopenharmony_ci printk("aha1740.c: WARNING: AHA1740 queue overflow!\n"); 18862306a36Sopenharmony_ci } else 18962306a36Sopenharmony_ci if ( status[0]&0x60 ) { 19062306a36Sopenharmony_ci /* Didn't find a better error */ 19162306a36Sopenharmony_ci retval = DID_ERROR; 19262306a36Sopenharmony_ci } 19362306a36Sopenharmony_ci /* In any other case return DID_OK so for example 19462306a36Sopenharmony_ci CONDITION_CHECKS make it through to the appropriate 19562306a36Sopenharmony_ci device driver */ 19662306a36Sopenharmony_ci } 19762306a36Sopenharmony_ci } 19862306a36Sopenharmony_ci /* Under all circumstances supply the target status -Michael */ 19962306a36Sopenharmony_ci return status[3] | retval << 16; 20062306a36Sopenharmony_ci} 20162306a36Sopenharmony_ci 20262306a36Sopenharmony_cistatic int aha1740_test_port(unsigned int base) 20362306a36Sopenharmony_ci{ 20462306a36Sopenharmony_ci if ( inb(PORTADR(base)) & PORTADDR_ENH ) 20562306a36Sopenharmony_ci return 1; /* Okay, we're all set */ 20662306a36Sopenharmony_ci 20762306a36Sopenharmony_ci printk("aha174x: Board detected, but not in enhanced mode, so disabled it.\n"); 20862306a36Sopenharmony_ci return 0; 20962306a36Sopenharmony_ci} 21062306a36Sopenharmony_ci 21162306a36Sopenharmony_ci/* A "high" level interrupt handler */ 21262306a36Sopenharmony_cistatic irqreturn_t aha1740_intr_handle(int irq, void *dev_id) 21362306a36Sopenharmony_ci{ 21462306a36Sopenharmony_ci struct Scsi_Host *host = (struct Scsi_Host *) dev_id; 21562306a36Sopenharmony_ci void (*my_done)(struct scsi_cmnd *); 21662306a36Sopenharmony_ci int errstatus, adapstat; 21762306a36Sopenharmony_ci int number_serviced; 21862306a36Sopenharmony_ci struct ecb *ecbptr; 21962306a36Sopenharmony_ci struct scsi_cmnd *SCtmp; 22062306a36Sopenharmony_ci unsigned int base; 22162306a36Sopenharmony_ci unsigned long flags; 22262306a36Sopenharmony_ci int handled = 0; 22362306a36Sopenharmony_ci struct aha1740_sg *sgptr; 22462306a36Sopenharmony_ci struct eisa_device *edev; 22562306a36Sopenharmony_ci 22662306a36Sopenharmony_ci if (!host) 22762306a36Sopenharmony_ci panic("aha1740.c: Irq from unknown host!\n"); 22862306a36Sopenharmony_ci spin_lock_irqsave(host->host_lock, flags); 22962306a36Sopenharmony_ci base = host->io_port; 23062306a36Sopenharmony_ci number_serviced = 0; 23162306a36Sopenharmony_ci edev = HOSTDATA(host)->edev; 23262306a36Sopenharmony_ci 23362306a36Sopenharmony_ci while(inb(G2STAT(base)) & G2STAT_INTPEND) { 23462306a36Sopenharmony_ci handled = 1; 23562306a36Sopenharmony_ci DEB(printk("aha1740_intr top of loop.\n")); 23662306a36Sopenharmony_ci adapstat = inb(G2INTST(base)); 23762306a36Sopenharmony_ci ecbptr = ecb_dma_to_cpu (host, inl(MBOXIN0(base))); 23862306a36Sopenharmony_ci outb(G2CNTRL_IRST,G2CNTRL(base)); /* interrupt reset */ 23962306a36Sopenharmony_ci 24062306a36Sopenharmony_ci switch ( adapstat & G2INTST_MASK ) { 24162306a36Sopenharmony_ci case G2INTST_CCBRETRY: 24262306a36Sopenharmony_ci case G2INTST_CCBERROR: 24362306a36Sopenharmony_ci case G2INTST_CCBGOOD: 24462306a36Sopenharmony_ci /* Host Ready -> Mailbox in complete */ 24562306a36Sopenharmony_ci outb(G2CNTRL_HRDY,G2CNTRL(base)); 24662306a36Sopenharmony_ci if (!ecbptr) { 24762306a36Sopenharmony_ci printk("Aha1740 null ecbptr in interrupt (%x,%x,%x,%d)\n", 24862306a36Sopenharmony_ci inb(G2STAT(base)),adapstat, 24962306a36Sopenharmony_ci inb(G2INTST(base)), number_serviced++); 25062306a36Sopenharmony_ci continue; 25162306a36Sopenharmony_ci } 25262306a36Sopenharmony_ci SCtmp = ecbptr->SCpnt; 25362306a36Sopenharmony_ci if (!SCtmp) { 25462306a36Sopenharmony_ci printk("Aha1740 null SCtmp in interrupt (%x,%x,%x,%d)\n", 25562306a36Sopenharmony_ci inb(G2STAT(base)),adapstat, 25662306a36Sopenharmony_ci inb(G2INTST(base)), number_serviced++); 25762306a36Sopenharmony_ci continue; 25862306a36Sopenharmony_ci } 25962306a36Sopenharmony_ci sgptr = (struct aha1740_sg *) SCtmp->host_scribble; 26062306a36Sopenharmony_ci scsi_dma_unmap(SCtmp); 26162306a36Sopenharmony_ci 26262306a36Sopenharmony_ci /* Free the sg block */ 26362306a36Sopenharmony_ci dma_free_coherent (&edev->dev, 26462306a36Sopenharmony_ci sizeof (struct aha1740_sg), 26562306a36Sopenharmony_ci SCtmp->host_scribble, 26662306a36Sopenharmony_ci sgptr->sg_dma_addr); 26762306a36Sopenharmony_ci 26862306a36Sopenharmony_ci /* Fetch the sense data, and tuck it away, in 26962306a36Sopenharmony_ci the required slot. The Adaptec 27062306a36Sopenharmony_ci automatically fetches it, and there is no 27162306a36Sopenharmony_ci guarantee that we will still have it in the 27262306a36Sopenharmony_ci cdb when we come back */ 27362306a36Sopenharmony_ci if ( (adapstat & G2INTST_MASK) == G2INTST_CCBERROR ) { 27462306a36Sopenharmony_ci memcpy_and_pad(SCtmp->sense_buffer, 27562306a36Sopenharmony_ci SCSI_SENSE_BUFFERSIZE, 27662306a36Sopenharmony_ci ecbptr->sense, 27762306a36Sopenharmony_ci sizeof(ecbptr->sense), 27862306a36Sopenharmony_ci 0); 27962306a36Sopenharmony_ci errstatus = aha1740_makecode(ecbptr->sense,ecbptr->status); 28062306a36Sopenharmony_ci } else 28162306a36Sopenharmony_ci errstatus = 0; 28262306a36Sopenharmony_ci DEB(if (errstatus) 28362306a36Sopenharmony_ci printk("aha1740_intr_handle: returning %6x\n", 28462306a36Sopenharmony_ci errstatus)); 28562306a36Sopenharmony_ci SCtmp->result = errstatus; 28662306a36Sopenharmony_ci my_done = ecbptr->done; 28762306a36Sopenharmony_ci memset(ecbptr,0,sizeof(struct ecb)); 28862306a36Sopenharmony_ci if ( my_done ) 28962306a36Sopenharmony_ci my_done(SCtmp); 29062306a36Sopenharmony_ci break; 29162306a36Sopenharmony_ci 29262306a36Sopenharmony_ci case G2INTST_HARDFAIL: 29362306a36Sopenharmony_ci printk(KERN_ALERT "aha1740 hardware failure!\n"); 29462306a36Sopenharmony_ci panic("aha1740.c"); /* Goodbye */ 29562306a36Sopenharmony_ci 29662306a36Sopenharmony_ci case G2INTST_ASNEVENT: 29762306a36Sopenharmony_ci printk("aha1740 asynchronous event: %02x %02x %02x %02x %02x\n", 29862306a36Sopenharmony_ci adapstat, 29962306a36Sopenharmony_ci inb(MBOXIN0(base)), 30062306a36Sopenharmony_ci inb(MBOXIN1(base)), 30162306a36Sopenharmony_ci inb(MBOXIN2(base)), 30262306a36Sopenharmony_ci inb(MBOXIN3(base))); /* Say What? */ 30362306a36Sopenharmony_ci /* Host Ready -> Mailbox in complete */ 30462306a36Sopenharmony_ci outb(G2CNTRL_HRDY,G2CNTRL(base)); 30562306a36Sopenharmony_ci break; 30662306a36Sopenharmony_ci 30762306a36Sopenharmony_ci case G2INTST_CMDGOOD: 30862306a36Sopenharmony_ci /* set immediate command success flag here: */ 30962306a36Sopenharmony_ci break; 31062306a36Sopenharmony_ci 31162306a36Sopenharmony_ci case G2INTST_CMDERROR: 31262306a36Sopenharmony_ci /* Set immediate command failure flag here: */ 31362306a36Sopenharmony_ci break; 31462306a36Sopenharmony_ci } 31562306a36Sopenharmony_ci number_serviced++; 31662306a36Sopenharmony_ci } 31762306a36Sopenharmony_ci 31862306a36Sopenharmony_ci spin_unlock_irqrestore(host->host_lock, flags); 31962306a36Sopenharmony_ci return IRQ_RETVAL(handled); 32062306a36Sopenharmony_ci} 32162306a36Sopenharmony_ci 32262306a36Sopenharmony_cistatic int aha1740_queuecommand_lck(struct scsi_cmnd *SCpnt) 32362306a36Sopenharmony_ci{ 32462306a36Sopenharmony_ci void (*done)(struct scsi_cmnd *) = scsi_done; 32562306a36Sopenharmony_ci unchar direction; 32662306a36Sopenharmony_ci unchar *cmd = (unchar *) SCpnt->cmnd; 32762306a36Sopenharmony_ci unchar target = scmd_id(SCpnt); 32862306a36Sopenharmony_ci struct aha1740_hostdata *host = HOSTDATA(SCpnt->device->host); 32962306a36Sopenharmony_ci unsigned long flags; 33062306a36Sopenharmony_ci dma_addr_t sg_dma; 33162306a36Sopenharmony_ci struct aha1740_sg *sgptr; 33262306a36Sopenharmony_ci int ecbno, nseg; 33362306a36Sopenharmony_ci DEB(int i); 33462306a36Sopenharmony_ci 33562306a36Sopenharmony_ci if(*cmd == REQUEST_SENSE) { 33662306a36Sopenharmony_ci SCpnt->result = 0; 33762306a36Sopenharmony_ci done(SCpnt); 33862306a36Sopenharmony_ci return 0; 33962306a36Sopenharmony_ci } 34062306a36Sopenharmony_ci 34162306a36Sopenharmony_ci#ifdef DEBUG 34262306a36Sopenharmony_ci if (*cmd == READ_10 || *cmd == WRITE_10) 34362306a36Sopenharmony_ci i = xscsi2int(cmd+2); 34462306a36Sopenharmony_ci else if (*cmd == READ_6 || *cmd == WRITE_6) 34562306a36Sopenharmony_ci i = scsi2int(cmd+2); 34662306a36Sopenharmony_ci else 34762306a36Sopenharmony_ci i = -1; 34862306a36Sopenharmony_ci printk("aha1740_queuecommand: dev %d cmd %02x pos %d len %d ", 34962306a36Sopenharmony_ci target, *cmd, i, bufflen); 35062306a36Sopenharmony_ci printk("scsi cmd:"); 35162306a36Sopenharmony_ci for (i = 0; i < SCpnt->cmd_len; i++) printk("%02x ", cmd[i]); 35262306a36Sopenharmony_ci printk("\n"); 35362306a36Sopenharmony_ci#endif 35462306a36Sopenharmony_ci 35562306a36Sopenharmony_ci /* locate an available ecb */ 35662306a36Sopenharmony_ci spin_lock_irqsave(SCpnt->device->host->host_lock, flags); 35762306a36Sopenharmony_ci ecbno = host->last_ecb_used + 1; /* An optimization */ 35862306a36Sopenharmony_ci if (ecbno >= AHA1740_ECBS) 35962306a36Sopenharmony_ci ecbno = 0; 36062306a36Sopenharmony_ci do { 36162306a36Sopenharmony_ci if (!host->ecb[ecbno].cmdw) 36262306a36Sopenharmony_ci break; 36362306a36Sopenharmony_ci ecbno++; 36462306a36Sopenharmony_ci if (ecbno >= AHA1740_ECBS) 36562306a36Sopenharmony_ci ecbno = 0; 36662306a36Sopenharmony_ci } while (ecbno != host->last_ecb_used); 36762306a36Sopenharmony_ci 36862306a36Sopenharmony_ci if (host->ecb[ecbno].cmdw) 36962306a36Sopenharmony_ci panic("Unable to find empty ecb for aha1740.\n"); 37062306a36Sopenharmony_ci 37162306a36Sopenharmony_ci host->ecb[ecbno].cmdw = AHA1740CMD_INIT; /* SCSI Initiator Command 37262306a36Sopenharmony_ci doubles as reserved flag */ 37362306a36Sopenharmony_ci 37462306a36Sopenharmony_ci host->last_ecb_used = ecbno; 37562306a36Sopenharmony_ci spin_unlock_irqrestore(SCpnt->device->host->host_lock, flags); 37662306a36Sopenharmony_ci 37762306a36Sopenharmony_ci#ifdef DEBUG 37862306a36Sopenharmony_ci printk("Sending command (%d %x)...", ecbno, done); 37962306a36Sopenharmony_ci#endif 38062306a36Sopenharmony_ci 38162306a36Sopenharmony_ci host->ecb[ecbno].cdblen = SCpnt->cmd_len; /* SCSI Command 38262306a36Sopenharmony_ci * Descriptor Block 38362306a36Sopenharmony_ci * Length */ 38462306a36Sopenharmony_ci 38562306a36Sopenharmony_ci direction = 0; 38662306a36Sopenharmony_ci if (*cmd == READ_10 || *cmd == READ_6) 38762306a36Sopenharmony_ci direction = 1; 38862306a36Sopenharmony_ci else if (*cmd == WRITE_10 || *cmd == WRITE_6) 38962306a36Sopenharmony_ci direction = 0; 39062306a36Sopenharmony_ci 39162306a36Sopenharmony_ci memcpy(host->ecb[ecbno].cdb, cmd, SCpnt->cmd_len); 39262306a36Sopenharmony_ci 39362306a36Sopenharmony_ci SCpnt->host_scribble = dma_alloc_coherent (&host->edev->dev, 39462306a36Sopenharmony_ci sizeof (struct aha1740_sg), 39562306a36Sopenharmony_ci &sg_dma, GFP_ATOMIC); 39662306a36Sopenharmony_ci if(SCpnt->host_scribble == NULL) { 39762306a36Sopenharmony_ci printk(KERN_WARNING "aha1740: out of memory in queuecommand!\n"); 39862306a36Sopenharmony_ci return 1; 39962306a36Sopenharmony_ci } 40062306a36Sopenharmony_ci sgptr = (struct aha1740_sg *) SCpnt->host_scribble; 40162306a36Sopenharmony_ci sgptr->sg_dma_addr = sg_dma; 40262306a36Sopenharmony_ci 40362306a36Sopenharmony_ci nseg = scsi_dma_map(SCpnt); 40462306a36Sopenharmony_ci BUG_ON(nseg < 0); 40562306a36Sopenharmony_ci if (nseg) { 40662306a36Sopenharmony_ci struct scatterlist *sg; 40762306a36Sopenharmony_ci struct aha1740_chain * cptr; 40862306a36Sopenharmony_ci int i; 40962306a36Sopenharmony_ci DEB(unsigned char * ptr); 41062306a36Sopenharmony_ci 41162306a36Sopenharmony_ci host->ecb[ecbno].sg = 1; /* SCSI Initiator Command 41262306a36Sopenharmony_ci * w/scatter-gather*/ 41362306a36Sopenharmony_ci cptr = sgptr->sg_chain; 41462306a36Sopenharmony_ci scsi_for_each_sg(SCpnt, sg, nseg, i) { 41562306a36Sopenharmony_ci cptr[i].datalen = sg_dma_len (sg); 41662306a36Sopenharmony_ci cptr[i].dataptr = sg_dma_address (sg); 41762306a36Sopenharmony_ci } 41862306a36Sopenharmony_ci host->ecb[ecbno].datalen = nseg * sizeof(struct aha1740_chain); 41962306a36Sopenharmony_ci host->ecb[ecbno].dataptr = sg_dma; 42062306a36Sopenharmony_ci#ifdef DEBUG 42162306a36Sopenharmony_ci printk("cptr %x: ",cptr); 42262306a36Sopenharmony_ci ptr = (unsigned char *) cptr; 42362306a36Sopenharmony_ci for(i=0;i<24;i++) printk("%02x ", ptr[i]); 42462306a36Sopenharmony_ci#endif 42562306a36Sopenharmony_ci } else { 42662306a36Sopenharmony_ci host->ecb[ecbno].datalen = 0; 42762306a36Sopenharmony_ci host->ecb[ecbno].dataptr = 0; 42862306a36Sopenharmony_ci } 42962306a36Sopenharmony_ci host->ecb[ecbno].lun = SCpnt->device->lun; 43062306a36Sopenharmony_ci host->ecb[ecbno].ses = 1; /* Suppress underrun errors */ 43162306a36Sopenharmony_ci host->ecb[ecbno].dir = direction; 43262306a36Sopenharmony_ci host->ecb[ecbno].ars = 1; /* Yes, get the sense on an error */ 43362306a36Sopenharmony_ci host->ecb[ecbno].senselen = 12; 43462306a36Sopenharmony_ci host->ecb[ecbno].senseptr = ecb_cpu_to_dma (SCpnt->device->host, 43562306a36Sopenharmony_ci host->ecb[ecbno].sense); 43662306a36Sopenharmony_ci host->ecb[ecbno].statusptr = ecb_cpu_to_dma (SCpnt->device->host, 43762306a36Sopenharmony_ci host->ecb[ecbno].status); 43862306a36Sopenharmony_ci host->ecb[ecbno].done = done; 43962306a36Sopenharmony_ci host->ecb[ecbno].SCpnt = SCpnt; 44062306a36Sopenharmony_ci#ifdef DEBUG 44162306a36Sopenharmony_ci { 44262306a36Sopenharmony_ci int i; 44362306a36Sopenharmony_ci printk("aha1740_command: sending.. "); 44462306a36Sopenharmony_ci for (i = 0; i < sizeof(host->ecb[ecbno]) - 10; i++) 44562306a36Sopenharmony_ci printk("%02x ", ((unchar *)&host->ecb[ecbno])[i]); 44662306a36Sopenharmony_ci } 44762306a36Sopenharmony_ci printk("\n"); 44862306a36Sopenharmony_ci#endif 44962306a36Sopenharmony_ci if (done) { 45062306a36Sopenharmony_ci /* The Adaptec Spec says the card is so fast that the loops 45162306a36Sopenharmony_ci will only be executed once in the code below. Even if this 45262306a36Sopenharmony_ci was true with the fastest processors when the spec was 45362306a36Sopenharmony_ci written, it doesn't seem to be true with today's fast 45462306a36Sopenharmony_ci processors. We print a warning if the code is executed more 45562306a36Sopenharmony_ci often than LOOPCNT_WARN. If this happens, it should be 45662306a36Sopenharmony_ci investigated. If the count reaches LOOPCNT_MAX, we assume 45762306a36Sopenharmony_ci something is broken; since there is no way to return an 45862306a36Sopenharmony_ci error (the return value is ignored by the mid-level scsi 45962306a36Sopenharmony_ci layer) we have to panic (and maybe that's the best thing we 46062306a36Sopenharmony_ci can do then anyhow). */ 46162306a36Sopenharmony_ci 46262306a36Sopenharmony_ci#define LOOPCNT_WARN 10 /* excessive mbxout wait -> syslog-msg */ 46362306a36Sopenharmony_ci#define LOOPCNT_MAX 1000000 /* mbxout deadlock -> panic() after ~ 2 sec. */ 46462306a36Sopenharmony_ci int loopcnt; 46562306a36Sopenharmony_ci unsigned int base = SCpnt->device->host->io_port; 46662306a36Sopenharmony_ci DEB(printk("aha1740[%d] critical section\n",ecbno)); 46762306a36Sopenharmony_ci 46862306a36Sopenharmony_ci spin_lock_irqsave(SCpnt->device->host->host_lock, flags); 46962306a36Sopenharmony_ci for (loopcnt = 0; ; loopcnt++) { 47062306a36Sopenharmony_ci if (inb(G2STAT(base)) & G2STAT_MBXOUT) break; 47162306a36Sopenharmony_ci if (loopcnt == LOOPCNT_WARN) { 47262306a36Sopenharmony_ci printk("aha1740[%d]_mbxout wait!\n",ecbno); 47362306a36Sopenharmony_ci } 47462306a36Sopenharmony_ci if (loopcnt == LOOPCNT_MAX) 47562306a36Sopenharmony_ci panic("aha1740.c: mbxout busy!\n"); 47662306a36Sopenharmony_ci } 47762306a36Sopenharmony_ci outl (ecb_cpu_to_dma (SCpnt->device->host, host->ecb + ecbno), 47862306a36Sopenharmony_ci MBOXOUT0(base)); 47962306a36Sopenharmony_ci for (loopcnt = 0; ; loopcnt++) { 48062306a36Sopenharmony_ci if (! (inb(G2STAT(base)) & G2STAT_BUSY)) break; 48162306a36Sopenharmony_ci if (loopcnt == LOOPCNT_WARN) { 48262306a36Sopenharmony_ci printk("aha1740[%d]_attn wait!\n",ecbno); 48362306a36Sopenharmony_ci } 48462306a36Sopenharmony_ci if (loopcnt == LOOPCNT_MAX) 48562306a36Sopenharmony_ci panic("aha1740.c: attn wait failed!\n"); 48662306a36Sopenharmony_ci } 48762306a36Sopenharmony_ci outb(ATTN_START | (target & 7), ATTN(base)); /* Start it up */ 48862306a36Sopenharmony_ci spin_unlock_irqrestore(SCpnt->device->host->host_lock, flags); 48962306a36Sopenharmony_ci DEB(printk("aha1740[%d] request queued.\n",ecbno)); 49062306a36Sopenharmony_ci } else 49162306a36Sopenharmony_ci printk(KERN_ALERT "aha1740_queuecommand: done can't be NULL\n"); 49262306a36Sopenharmony_ci return 0; 49362306a36Sopenharmony_ci} 49462306a36Sopenharmony_ci 49562306a36Sopenharmony_cistatic DEF_SCSI_QCMD(aha1740_queuecommand) 49662306a36Sopenharmony_ci 49762306a36Sopenharmony_ci/* Query the board for its irq_level and irq_type. Nothing else matters 49862306a36Sopenharmony_ci in enhanced mode on an EISA bus. */ 49962306a36Sopenharmony_ci 50062306a36Sopenharmony_cistatic void aha1740_getconfig(unsigned int base, unsigned int *irq_level, 50162306a36Sopenharmony_ci unsigned int *irq_type, 50262306a36Sopenharmony_ci unsigned int *translation) 50362306a36Sopenharmony_ci{ 50462306a36Sopenharmony_ci static int intab[] = { 9, 10, 11, 12, 0, 14, 15, 0 }; 50562306a36Sopenharmony_ci 50662306a36Sopenharmony_ci *irq_level = intab[inb(INTDEF(base)) & 0x7]; 50762306a36Sopenharmony_ci *irq_type = (inb(INTDEF(base)) & 0x8) >> 3; 50862306a36Sopenharmony_ci *translation = inb(RESV1(base)) & 0x1; 50962306a36Sopenharmony_ci outb(inb(INTDEF(base)) | 0x10, INTDEF(base)); 51062306a36Sopenharmony_ci} 51162306a36Sopenharmony_ci 51262306a36Sopenharmony_cistatic int aha1740_biosparam(struct scsi_device *sdev, 51362306a36Sopenharmony_ci struct block_device *dev, 51462306a36Sopenharmony_ci sector_t capacity, int* ip) 51562306a36Sopenharmony_ci{ 51662306a36Sopenharmony_ci int size = capacity; 51762306a36Sopenharmony_ci int extended = HOSTDATA(sdev->host)->translation; 51862306a36Sopenharmony_ci 51962306a36Sopenharmony_ci DEB(printk("aha1740_biosparam\n")); 52062306a36Sopenharmony_ci if (extended && (ip[2] > 1024)) { 52162306a36Sopenharmony_ci ip[0] = 255; 52262306a36Sopenharmony_ci ip[1] = 63; 52362306a36Sopenharmony_ci ip[2] = size / (255 * 63); 52462306a36Sopenharmony_ci } else { 52562306a36Sopenharmony_ci ip[0] = 64; 52662306a36Sopenharmony_ci ip[1] = 32; 52762306a36Sopenharmony_ci ip[2] = size >> 11; 52862306a36Sopenharmony_ci } 52962306a36Sopenharmony_ci return 0; 53062306a36Sopenharmony_ci} 53162306a36Sopenharmony_ci 53262306a36Sopenharmony_cistatic int aha1740_eh_abort_handler (struct scsi_cmnd *dummy) 53362306a36Sopenharmony_ci{ 53462306a36Sopenharmony_ci/* 53562306a36Sopenharmony_ci * From Alan Cox : 53662306a36Sopenharmony_ci * The AHA1740 has firmware handled abort/reset handling. The "head in 53762306a36Sopenharmony_ci * sand" kernel code is correct for once 8) 53862306a36Sopenharmony_ci * 53962306a36Sopenharmony_ci * So we define a dummy handler just to keep the kernel SCSI code as 54062306a36Sopenharmony_ci * quiet as possible... 54162306a36Sopenharmony_ci */ 54262306a36Sopenharmony_ci 54362306a36Sopenharmony_ci return SUCCESS; 54462306a36Sopenharmony_ci} 54562306a36Sopenharmony_ci 54662306a36Sopenharmony_cistatic const struct scsi_host_template aha1740_template = { 54762306a36Sopenharmony_ci .module = THIS_MODULE, 54862306a36Sopenharmony_ci .proc_name = "aha1740", 54962306a36Sopenharmony_ci .show_info = aha1740_show_info, 55062306a36Sopenharmony_ci .name = "Adaptec 174x (EISA)", 55162306a36Sopenharmony_ci .queuecommand = aha1740_queuecommand, 55262306a36Sopenharmony_ci .bios_param = aha1740_biosparam, 55362306a36Sopenharmony_ci .can_queue = AHA1740_ECBS, 55462306a36Sopenharmony_ci .this_id = 7, 55562306a36Sopenharmony_ci .sg_tablesize = AHA1740_SCATTER, 55662306a36Sopenharmony_ci .eh_abort_handler = aha1740_eh_abort_handler, 55762306a36Sopenharmony_ci}; 55862306a36Sopenharmony_ci 55962306a36Sopenharmony_cistatic int aha1740_probe (struct device *dev) 56062306a36Sopenharmony_ci{ 56162306a36Sopenharmony_ci int slotbase, rc; 56262306a36Sopenharmony_ci unsigned int irq_level, irq_type, translation; 56362306a36Sopenharmony_ci struct Scsi_Host *shpnt; 56462306a36Sopenharmony_ci struct aha1740_hostdata *host; 56562306a36Sopenharmony_ci struct eisa_device *edev = to_eisa_device (dev); 56662306a36Sopenharmony_ci 56762306a36Sopenharmony_ci DEB(printk("aha1740_probe: \n")); 56862306a36Sopenharmony_ci 56962306a36Sopenharmony_ci slotbase = edev->base_addr + EISA_VENDOR_ID_OFFSET; 57062306a36Sopenharmony_ci if (!request_region(slotbase, SLOTSIZE, "aha1740")) /* See if in use */ 57162306a36Sopenharmony_ci return -EBUSY; 57262306a36Sopenharmony_ci if (!aha1740_test_port(slotbase)) 57362306a36Sopenharmony_ci goto err_release_region; 57462306a36Sopenharmony_ci aha1740_getconfig(slotbase,&irq_level,&irq_type,&translation); 57562306a36Sopenharmony_ci if ((inb(G2STAT(slotbase)) & 57662306a36Sopenharmony_ci (G2STAT_MBXOUT|G2STAT_BUSY)) != G2STAT_MBXOUT) { 57762306a36Sopenharmony_ci /* If the card isn't ready, hard reset it */ 57862306a36Sopenharmony_ci outb(G2CNTRL_HRST, G2CNTRL(slotbase)); 57962306a36Sopenharmony_ci outb(0, G2CNTRL(slotbase)); 58062306a36Sopenharmony_ci } 58162306a36Sopenharmony_ci printk(KERN_INFO "Configuring slot %d at IO:%x, IRQ %u (%s)\n", 58262306a36Sopenharmony_ci edev->slot, slotbase, irq_level, irq_type ? "edge" : "level"); 58362306a36Sopenharmony_ci printk(KERN_INFO "aha174x: Extended translation %sabled.\n", 58462306a36Sopenharmony_ci translation ? "en" : "dis"); 58562306a36Sopenharmony_ci shpnt = scsi_host_alloc(&aha1740_template, 58662306a36Sopenharmony_ci sizeof(struct aha1740_hostdata)); 58762306a36Sopenharmony_ci if(shpnt == NULL) 58862306a36Sopenharmony_ci goto err_release_region; 58962306a36Sopenharmony_ci 59062306a36Sopenharmony_ci shpnt->base = 0; 59162306a36Sopenharmony_ci shpnt->io_port = slotbase; 59262306a36Sopenharmony_ci shpnt->n_io_port = SLOTSIZE; 59362306a36Sopenharmony_ci shpnt->irq = irq_level; 59462306a36Sopenharmony_ci shpnt->dma_channel = 0xff; 59562306a36Sopenharmony_ci host = HOSTDATA(shpnt); 59662306a36Sopenharmony_ci host->edev = edev; 59762306a36Sopenharmony_ci host->translation = translation; 59862306a36Sopenharmony_ci host->ecb_dma_addr = dma_map_single (&edev->dev, host->ecb, 59962306a36Sopenharmony_ci sizeof (host->ecb), 60062306a36Sopenharmony_ci DMA_BIDIRECTIONAL); 60162306a36Sopenharmony_ci if (!host->ecb_dma_addr) { 60262306a36Sopenharmony_ci printk (KERN_ERR "aha1740_probe: Couldn't map ECB, giving up\n"); 60362306a36Sopenharmony_ci goto err_host_put; 60462306a36Sopenharmony_ci } 60562306a36Sopenharmony_ci 60662306a36Sopenharmony_ci DEB(printk("aha1740_probe: enable interrupt channel %d\n",irq_level)); 60762306a36Sopenharmony_ci if (request_irq(irq_level,aha1740_intr_handle,irq_type ? 0 : IRQF_SHARED, 60862306a36Sopenharmony_ci "aha1740",shpnt)) { 60962306a36Sopenharmony_ci printk(KERN_ERR "aha1740_probe: Unable to allocate IRQ %d.\n", 61062306a36Sopenharmony_ci irq_level); 61162306a36Sopenharmony_ci goto err_unmap; 61262306a36Sopenharmony_ci } 61362306a36Sopenharmony_ci 61462306a36Sopenharmony_ci eisa_set_drvdata (edev, shpnt); 61562306a36Sopenharmony_ci 61662306a36Sopenharmony_ci rc = scsi_add_host (shpnt, dev); 61762306a36Sopenharmony_ci if (rc) 61862306a36Sopenharmony_ci goto err_irq; 61962306a36Sopenharmony_ci 62062306a36Sopenharmony_ci scsi_scan_host (shpnt); 62162306a36Sopenharmony_ci return 0; 62262306a36Sopenharmony_ci 62362306a36Sopenharmony_ci err_irq: 62462306a36Sopenharmony_ci free_irq(irq_level, shpnt); 62562306a36Sopenharmony_ci err_unmap: 62662306a36Sopenharmony_ci dma_unmap_single (&edev->dev, host->ecb_dma_addr, 62762306a36Sopenharmony_ci sizeof (host->ecb), DMA_BIDIRECTIONAL); 62862306a36Sopenharmony_ci err_host_put: 62962306a36Sopenharmony_ci scsi_host_put (shpnt); 63062306a36Sopenharmony_ci err_release_region: 63162306a36Sopenharmony_ci release_region(slotbase, SLOTSIZE); 63262306a36Sopenharmony_ci 63362306a36Sopenharmony_ci return -ENODEV; 63462306a36Sopenharmony_ci} 63562306a36Sopenharmony_ci 63662306a36Sopenharmony_cistatic int aha1740_remove (struct device *dev) 63762306a36Sopenharmony_ci{ 63862306a36Sopenharmony_ci struct Scsi_Host *shpnt = dev_get_drvdata(dev); 63962306a36Sopenharmony_ci struct aha1740_hostdata *host = HOSTDATA (shpnt); 64062306a36Sopenharmony_ci 64162306a36Sopenharmony_ci scsi_remove_host(shpnt); 64262306a36Sopenharmony_ci 64362306a36Sopenharmony_ci free_irq (shpnt->irq, shpnt); 64462306a36Sopenharmony_ci dma_unmap_single (dev, host->ecb_dma_addr, 64562306a36Sopenharmony_ci sizeof (host->ecb), DMA_BIDIRECTIONAL); 64662306a36Sopenharmony_ci release_region (shpnt->io_port, SLOTSIZE); 64762306a36Sopenharmony_ci 64862306a36Sopenharmony_ci scsi_host_put (shpnt); 64962306a36Sopenharmony_ci 65062306a36Sopenharmony_ci return 0; 65162306a36Sopenharmony_ci} 65262306a36Sopenharmony_ci 65362306a36Sopenharmony_cistatic struct eisa_device_id aha1740_ids[] = { 65462306a36Sopenharmony_ci { "ADP0000" }, /* 1740 */ 65562306a36Sopenharmony_ci { "ADP0001" }, /* 1740A */ 65662306a36Sopenharmony_ci { "ADP0002" }, /* 1742A */ 65762306a36Sopenharmony_ci { "ADP0400" }, /* 1744 */ 65862306a36Sopenharmony_ci { "" } 65962306a36Sopenharmony_ci}; 66062306a36Sopenharmony_ciMODULE_DEVICE_TABLE(eisa, aha1740_ids); 66162306a36Sopenharmony_ci 66262306a36Sopenharmony_cistatic struct eisa_driver aha1740_driver = { 66362306a36Sopenharmony_ci .id_table = aha1740_ids, 66462306a36Sopenharmony_ci .driver = { 66562306a36Sopenharmony_ci .name = "aha1740", 66662306a36Sopenharmony_ci .probe = aha1740_probe, 66762306a36Sopenharmony_ci .remove = aha1740_remove, 66862306a36Sopenharmony_ci }, 66962306a36Sopenharmony_ci}; 67062306a36Sopenharmony_ci 67162306a36Sopenharmony_cistatic __init int aha1740_init (void) 67262306a36Sopenharmony_ci{ 67362306a36Sopenharmony_ci return eisa_driver_register (&aha1740_driver); 67462306a36Sopenharmony_ci} 67562306a36Sopenharmony_ci 67662306a36Sopenharmony_cistatic __exit void aha1740_exit (void) 67762306a36Sopenharmony_ci{ 67862306a36Sopenharmony_ci eisa_driver_unregister (&aha1740_driver); 67962306a36Sopenharmony_ci} 68062306a36Sopenharmony_ci 68162306a36Sopenharmony_cimodule_init (aha1740_init); 68262306a36Sopenharmony_cimodule_exit (aha1740_exit); 68362306a36Sopenharmony_ci 68462306a36Sopenharmony_ciMODULE_LICENSE("GPL"); 685