5 #define UNSAFE                  /* Leave interrupts enabled during pseudo-dma I/O */
 
   6 #define xNDEBUG (NDEBUG_INTR+NDEBUG_RESELECTION+\
 
   7                  NDEBUG_SELECTION+NDEBUG_ARBITRATION)
 
   8 #define DMA_WORKS_RIGHT
 
  12  * DTC 3180/3280 driver, by
 
  13  *      Ray Van Tassle  rayvt@comm.mot.com
 
  16  *      Trantor T128/T128F/T228 driver by...
 
  20  *      (Unix and Linux consulting and custom programming)
 
  24  * DISTRIBUTION RELEASE 1.
 
  26  * For more information, please consult 
 
  29  * SCSI Protocol Controller
 
  35  * AUTOSENSE - if defined, REQUEST SENSE will be performed automatically
 
  36  *      for commands that return with a CHECK CONDITION status. 
 
  38  * PSEUDO_DMA - enables PSEUDO-DMA hardware, should give a 3-4X performance
 
  39  * increase compared to polled I/O.
 
  41  * PARITY - enable parity checking.  Not supported.
 
  43  * UNSAFE - leave interrupts enabled during pseudo-DMA transfers. 
 
  44  *              You probably want this.
 
  46  * The card is detected and initialized in one of several ways : 
 
  47  * 1.  Autoprobe (default) - since the board is memory mapped, 
 
  48  *     a BIOS signature is scanned for to locate the registers.
 
  49  *     An interrupt is triggered to autoprobe for the interrupt
 
  52  * 2.  With command line overrides - dtc=address,irq may be 
 
  53  *     used on the LILO command line to override the defaults.
 
  57 /*----------------------------------------------------------------*/
 
  58 /* the following will set the monitor border color (useful to find
 
  59  where something crashed or gets stuck at */
 
  69 #define rtrc(i) {inb(0x3da); outb(0x31, 0x3c0); outb((i), 0x3c0);}
 
  75 #include <asm/system.h>
 
  76 #include <linux/module.h>
 
  77 #include <linux/signal.h>
 
  78 #include <linux/sched.h>
 
  79 #include <linux/blkdev.h>
 
  80 #include <linux/delay.h>
 
  81 #include <linux/stat.h>
 
  82 #include <linux/string.h>
 
  83 #include <linux/init.h>
 
  84 #include <linux/interrupt.h>
 
  87 #include <scsi/scsi_host.h>
 
  93 #define DTC_PUBLIC_RELEASE 2
 
  96  * The DTC3180 & 3280 boards are memory mapped.
 
 102 /* Offset from DTC_5380_OFFSET */
 
 103 #define DTC_CONTROL_REG         0x100   /* rw */
 
 104 #define D_CR_ACCESS             0x80    /* ro set=can access 3280 registers */
 
 105 #define CSR_DIR_READ            0x40    /* rw direction, 1 = read 0 = write */
 
 107 #define CSR_RESET              0x80     /* wo  Resets 53c400 */
 
 108 #define CSR_5380_REG           0x80     /* ro  5380 registers can be accessed */
 
 109 #define CSR_TRANS_DIR          0x40     /* rw  Data transfer direction */
 
 110 #define CSR_SCSI_BUFF_INTR     0x20     /* rw  Enable int on transfer ready */
 
 111 #define CSR_5380_INTR          0x10     /* rw  Enable 5380 interrupts */
 
 112 #define CSR_SHARED_INTR        0x08     /* rw  Interrupt sharing */
 
 113 #define CSR_HOST_BUF_NOT_RDY   0x04     /* ro  Host buffer not ready */
 
 114 #define CSR_SCSI_BUF_RDY       0x02     /* ro  SCSI buffer ready */
 
 115 #define CSR_GATED_5380_IRQ     0x01     /* ro  Last block xferred */
 
 116 #define CSR_INT_BASE (CSR_SCSI_BUFF_INTR | CSR_5380_INTR)
 
 119 #define DTC_BLK_CNT             0x101   /* rw 
 
 120                                          * # of 128-byte blocks to transfer */
 
 123 #define D_CR_ACCESS             0x80    /* ro set=can access 3280 registers */
 
 125 #define DTC_SWITCH_REG          0x3982  /* ro - DIP switches */
 
 126 #define DTC_RESUME_XFER         0x3982  /* wo - resume data xfer 
 
 127                                          * after disconnect/reconnect*/
 
 129 #define DTC_5380_OFFSET         0x3880  /* 8 registers here, see NCR5380.h */
 
 131 /*!!!! for dtc, it's a 128 byte buffer at 3900 !!! */
 
 132 #define DTC_DATA_BUF            0x3900  /* rw 128 bytes long */
 
 134 static struct override {
 
 135         unsigned int address;
 
 139 [] __initdata = OVERRIDE;
 
 148 #define NO_OVERRIDES ARRAY_SIZE(overrides)
 
 151         unsigned long address;
 
 153 } bases[] __initdata = {
 
 160 #define NO_BASES ARRAY_SIZE(bases)
 
 162 static const struct signature {
 
 166         {"DATA TECHNOLOGY CORPORATION BIOS", 0x25},
 
 169 #define NO_SIGNATURES ARRAY_SIZE(signatures)
 
 173  * Function : dtc_setup(char *str, int *ints)
 
 175  * Purpose : LILO command line initialization of the overrides array,
 
 177  * Inputs : str - unused, ints - array of integer parameters with ints[0]
 
 178  *      equal to the number of ints.
 
 182 static void __init dtc_setup(char *str, int *ints)
 
 184         static int commandline_current = 0;
 
 187                 printk("dtc_setup: usage dtc=address,irq\n");
 
 188         else if (commandline_current < NO_OVERRIDES) {
 
 189                 overrides[commandline_current].address = ints[1];
 
 190                 overrides[commandline_current].irq = ints[2];
 
 191                 for (i = 0; i < NO_BASES; ++i)
 
 192                         if (bases[i].address == ints[1]) {
 
 196                 ++commandline_current;
 
 202  * Function : int dtc_detect(struct scsi_host_template * tpnt)
 
 204  * Purpose : detects and initializes DTC 3180/3280 controllers
 
 205  *      that were autoprobed, overridden on the LILO command line, 
 
 206  *      or specified at compile time.
 
 208  * Inputs : tpnt - template for this SCSI adapter.
 
 210  * Returns : 1 if a host adapter was found, 0 if not.
 
 214 static int __init dtc_detect(struct scsi_host_template * tpnt)
 
 216         static int current_override = 0, current_base = 0;
 
 217         struct Scsi_Host *instance;
 
 222         tpnt->proc_name = "dtc3x80";
 
 223         tpnt->proc_info = &dtc_proc_info;
 
 225         for (count = 0; current_override < NO_OVERRIDES; ++current_override) {
 
 229                 if (overrides[current_override].address) {
 
 230                         addr = overrides[current_override].address;
 
 231                         base = ioremap(addr, 0x2000);
 
 235                         for (; !addr && (current_base < NO_BASES); ++current_base) {
 
 236 #if (DTCDEBUG & DTCDEBUG_INIT)
 
 237                                 printk("scsi-dtc : probing address %08x\n", bases[current_base].address);
 
 239                                 if (bases[current_base].noauto)
 
 241                                 base = ioremap(bases[current_base].address, 0x2000);
 
 244                                 for (sig = 0; sig < NO_SIGNATURES; ++sig) {
 
 245                                         if (check_signature(base + signatures[sig].offset, signatures[sig].string, strlen(signatures[sig].string))) {
 
 246                                                 addr = bases[current_base].address;
 
 247 #if (DTCDEBUG & DTCDEBUG_INIT)
 
 248                                                 printk("scsi-dtc : detected board.\n");
 
 256 #if defined(DTCDEBUG) && (DTCDEBUG & DTCDEBUG_INIT)
 
 257                 printk("scsi-dtc : base = %08x\n", addr);
 
 264                 instance = scsi_register(tpnt, sizeof(struct NCR5380_hostdata));
 
 265                 if (instance == NULL)
 
 268                 instance->base = addr;
 
 269                 ((struct NCR5380_hostdata *)(instance)->hostdata)->base = base;
 
 271                 NCR5380_init(instance, 0);
 
 273                 NCR5380_write(DTC_CONTROL_REG, CSR_5380_INTR);  /* Enable int's */
 
 274                 if (overrides[current_override].irq != IRQ_AUTO)
 
 275                         instance->irq = overrides[current_override].irq;
 
 277                         instance->irq = NCR5380_probe_irq(instance, DTC_IRQS);
 
 279 #ifndef DONT_USE_INTR
 
 280                 /* With interrupts enabled, it will sometimes hang when doing heavy
 
 281                  * reads. So better not enable them until I finger it out. */
 
 282                 if (instance->irq != SCSI_IRQ_NONE)
 
 283                         if (request_irq(instance->irq, dtc_intr, SA_INTERRUPT, "dtc", instance)) {
 
 284                                 printk(KERN_ERR "scsi%d : IRQ%d not free, interrupts disabled\n", instance->host_no, instance->irq);
 
 285                                 instance->irq = SCSI_IRQ_NONE;
 
 288                 if (instance->irq == SCSI_IRQ_NONE) {
 
 289                         printk(KERN_WARNING "scsi%d : interrupts not enabled. for better interactive performance,\n", instance->host_no);
 
 290                         printk(KERN_WARNING "scsi%d : please jumper the board for a free IRQ.\n", instance->host_no);
 
 293                 if (instance->irq != SCSI_IRQ_NONE)
 
 294                         printk(KERN_WARNING "scsi%d : interrupts not used. Might as well not jumper it.\n", instance->host_no);
 
 295                 instance->irq = SCSI_IRQ_NONE;
 
 297 #if defined(DTCDEBUG) && (DTCDEBUG & DTCDEBUG_INIT)
 
 298                 printk("scsi%d : irq = %d\n", instance->host_no, instance->irq);
 
 301                 printk(KERN_INFO "scsi%d : at 0x%05X", instance->host_no, (int) instance->base);
 
 302                 if (instance->irq == SCSI_IRQ_NONE)
 
 303                         printk(" interrupts disabled");
 
 305                         printk(" irq %d", instance->irq);
 
 306                 printk(" options CAN_QUEUE=%d  CMD_PER_LUN=%d release=%d", CAN_QUEUE, CMD_PER_LUN, DTC_PUBLIC_RELEASE);
 
 307                 NCR5380_print_options(instance);
 
 317  * Function : int dtc_biosparam(Disk * disk, struct block_device *dev, int *ip)
 
 319  * Purpose : Generates a BIOS / DOS compatible H-C-S mapping for 
 
 320  *      the specified device / size.
 
 322  * Inputs : size = size of device in sectors (512 bytes), dev = block device
 
 323  *      major / minor, ip[] = {heads, sectors, cylinders}  
 
 325  * Returns : always 0 (success), initializes ip
 
 330  * XXX Most SCSI boards use this mapping, I could be incorrect.  Some one
 
 331  * using hard disks on a trantor should verify that this mapping corresponds
 
 332  * to that used by the BIOS / ASPI driver by running the linux fdisk program
 
 333  * and matching the H_C_S coordinates to what DOS uses.
 
 336 static int dtc_biosparam(struct scsi_device *sdev, struct block_device *dev,
 
 337                          sector_t capacity, int *ip)
 
 348 /****************************************************************
 
 349  * Function : int NCR5380_pread (struct Scsi_Host *instance, 
 
 350  *      unsigned char *dst, int len)
 
 352  * Purpose : Fast 5380 pseudo-dma read function, reads len bytes to 
 
 355  * Inputs : dst = destination, len = length in bytes
 
 357  * Returns : 0 on success, non zero on a failure such as a watchdog 
 
 361 static int dtc_maxi = 0;
 
 362 static int dtc_wmaxi = 0;
 
 364 static inline int NCR5380_pread(struct Scsi_Host *instance, unsigned char *dst, int len)
 
 366         unsigned char *d = dst;
 
 367         int i;                  /* For counting time spent in the poll-loop */
 
 368         NCR5380_local_declare();
 
 369         NCR5380_setup(instance);
 
 372         NCR5380_read(RESET_PARITY_INTERRUPT_REG);
 
 373         NCR5380_write(MODE_REG, MR_ENABLE_EOP_INTR | MR_DMA_MODE);
 
 374         if (instance->irq == SCSI_IRQ_NONE)
 
 375                 NCR5380_write(DTC_CONTROL_REG, CSR_DIR_READ);
 
 377                 NCR5380_write(DTC_CONTROL_REG, CSR_DIR_READ | CSR_INT_BASE);
 
 378         NCR5380_write(DTC_BLK_CNT, len >> 7);   /* Block count */
 
 382                 while (NCR5380_read(DTC_CONTROL_REG) & CSR_HOST_BUF_NOT_RDY)
 
 385                 memcpy_fromio(d, base + DTC_DATA_BUF, 128);
 
 389                 /*** with int's on, it sometimes hangs after here.
 
 390                  * Looks like something makes HBNR go away. */
 
 393         while (!(NCR5380_read(DTC_CONTROL_REG) & D_CR_ACCESS))
 
 395         NCR5380_write(MODE_REG, 0);     /* Clear the operating mode */
 
 397         NCR5380_read(RESET_PARITY_INTERRUPT_REG);
 
 403 /****************************************************************
 
 404  * Function : int NCR5380_pwrite (struct Scsi_Host *instance, 
 
 405  *      unsigned char *src, int len)
 
 407  * Purpose : Fast 5380 pseudo-dma write function, transfers len bytes from
 
 410  * Inputs : src = source, len = length in bytes
 
 412  * Returns : 0 on success, non zero on a failure such as a watchdog 
 
 416 static inline int NCR5380_pwrite(struct Scsi_Host *instance, unsigned char *src, int len)
 
 419         NCR5380_local_declare();
 
 420         NCR5380_setup(instance);
 
 422         NCR5380_read(RESET_PARITY_INTERRUPT_REG);
 
 423         NCR5380_write(MODE_REG, MR_ENABLE_EOP_INTR | MR_DMA_MODE);
 
 424         /* set direction (write) */
 
 425         if (instance->irq == SCSI_IRQ_NONE)
 
 426                 NCR5380_write(DTC_CONTROL_REG, 0);
 
 428                 NCR5380_write(DTC_CONTROL_REG, CSR_5380_INTR);
 
 429         NCR5380_write(DTC_BLK_CNT, len >> 7);   /* Block count */
 
 430         for (i = 0; len > 0; ++i) {
 
 432                 /* Poll until the host buffer can accept data. */
 
 433                 while (NCR5380_read(DTC_CONTROL_REG) & CSR_HOST_BUF_NOT_RDY)
 
 436                 memcpy_toio(base + DTC_DATA_BUF, src, 128);
 
 441         while (!(NCR5380_read(DTC_CONTROL_REG) & D_CR_ACCESS))
 
 444         /* Wait until the last byte has been sent to the disk */
 
 445         while (!(NCR5380_read(TARGET_COMMAND_REG) & TCR_LAST_BYTE_SENT))
 
 448         /* Check for parity error here. fixme. */
 
 449         NCR5380_write(MODE_REG, 0);     /* Clear the operating mode */
 
 456 MODULE_LICENSE("GPL");
 
 460 static int dtc_release(struct Scsi_Host *shost)
 
 462         NCR5380_local_declare();
 
 463         NCR5380_setup(shost);
 
 465                 free_irq(shost->irq, NULL);
 
 467         if (shost->io_port && shost->n_io_port)
 
 468                 release_region(shost->io_port, shost->n_io_port);
 
 469         scsi_unregister(shost);
 
 474 static struct scsi_host_template driver_template = {
 
 475         .name                           = "DTC 3180/3280 ",
 
 476         .detect                         = dtc_detect,
 
 477         .release                        = dtc_release,
 
 478         .queuecommand                   = dtc_queue_command,
 
 479         .eh_abort_handler               = dtc_abort,
 
 480         .eh_bus_reset_handler           = dtc_bus_reset,
 
 481         .bios_param                     = dtc_biosparam,
 
 482         .can_queue                      = CAN_QUEUE,
 
 484         .sg_tablesize                   = SG_ALL,
 
 485         .cmd_per_lun                    = CMD_PER_LUN,
 
 486         .use_clustering                 = DISABLE_CLUSTERING,
 
 488 #include "scsi_module.c"