| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #define AUTOSENSE
 | 
					
						
							|  |  |  | #define PSEUDO_DMA
 | 
					
						
							|  |  |  | #define DONT_USE_INTR
 | 
					
						
							|  |  |  | #define UNSAFE			/* Leave interrupts enabled during pseudo-dma I/O */
 | 
					
						
							|  |  |  | #define xNDEBUG (NDEBUG_INTR+NDEBUG_RESELECTION+\
 | 
					
						
							|  |  |  | 		 NDEBUG_SELECTION+NDEBUG_ARBITRATION) | 
					
						
							|  |  |  | #define DMA_WORKS_RIGHT
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * DTC 3180/3280 driver, by | 
					
						
							|  |  |  |  *	Ray Van Tassle	rayvt@comm.mot.com | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  *	taken from ... | 
					
						
							|  |  |  |  *	Trantor T128/T128F/T228 driver by... | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 	Drew Eckhardt | 
					
						
							|  |  |  |  *	Visionary Computing | 
					
						
							|  |  |  |  *	(Unix and Linux consulting and custom programming) | 
					
						
							|  |  |  |  *	drew@colorado.edu | 
					
						
							|  |  |  |  *      +1 (303) 440-4894 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * DISTRIBUTION RELEASE 1. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For more information, please consult  | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * NCR 5380 Family | 
					
						
							|  |  |  |  * SCSI Protocol Controller | 
					
						
							|  |  |  |  * Databook | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Options :  | 
					
						
							|  |  |  |  * AUTOSENSE - if defined, REQUEST SENSE will be performed automatically | 
					
						
							|  |  |  |  *      for commands that return with a CHECK CONDITION status.  | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * PSEUDO_DMA - enables PSEUDO-DMA hardware, should give a 3-4X performance | 
					
						
							|  |  |  |  * increase compared to polled I/O. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * PARITY - enable parity checking.  Not supported. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * UNSAFE - leave interrupts enabled during pseudo-DMA transfers.  | 
					
						
							|  |  |  |  *		You probably want this. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The card is detected and initialized in one of several ways :  | 
					
						
							|  |  |  |  * 1.  Autoprobe (default) - since the board is memory mapped,  | 
					
						
							|  |  |  |  *     a BIOS signature is scanned for to locate the registers. | 
					
						
							|  |  |  |  *     An interrupt is triggered to autoprobe for the interrupt | 
					
						
							|  |  |  |  *     line. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 2.  With command line overrides - dtc=address,irq may be  | 
					
						
							|  |  |  |  *     used on the LILO command line to override the defaults. | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*----------------------------------------------------------------*/ | 
					
						
							|  |  |  | /* the following will set the monitor border color (useful to find
 | 
					
						
							|  |  |  |  where something crashed or gets stuck at */ | 
					
						
							|  |  |  | /* 1 = blue
 | 
					
						
							|  |  |  |  2 = green | 
					
						
							|  |  |  |  3 = cyan | 
					
						
							|  |  |  |  4 = red | 
					
						
							|  |  |  |  5 = magenta | 
					
						
							|  |  |  |  6 = yellow | 
					
						
							|  |  |  |  7 = white | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | #if 0
 | 
					
						
							|  |  |  | #define rtrc(i) {inb(0x3da); outb(0x31, 0x3c0); outb((i), 0x3c0);}
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #define rtrc(i) {}
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <asm/system.h>
 | 
					
						
							|  |  |  | #include <linux/module.h>
 | 
					
						
							|  |  |  | #include <linux/signal.h>
 | 
					
						
							|  |  |  | #include <linux/blkdev.h>
 | 
					
						
							|  |  |  | #include <linux/delay.h>
 | 
					
						
							|  |  |  | #include <linux/stat.h>
 | 
					
						
							|  |  |  | #include <linux/string.h>
 | 
					
						
							|  |  |  | #include <linux/init.h>
 | 
					
						
							|  |  |  | #include <linux/interrupt.h>
 | 
					
						
							| 
									
										
										
										
											2006-10-11 01:22:01 -07:00
										 |  |  | #include <linux/io.h>
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | #include "scsi.h"
 | 
					
						
							|  |  |  | #include <scsi/scsi_host.h>
 | 
					
						
							|  |  |  | #include "dtc.h"
 | 
					
						
							|  |  |  | #define AUTOPROBE_IRQ
 | 
					
						
							|  |  |  | #include "NCR5380.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DTC_PUBLIC_RELEASE 2
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * The DTC3180 & 3280 boards are memory mapped. | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | /* Offset from DTC_5380_OFFSET */ | 
					
						
							|  |  |  | #define DTC_CONTROL_REG		0x100	/* rw */
 | 
					
						
							|  |  |  | #define D_CR_ACCESS		0x80	/* ro set=can access 3280 registers */
 | 
					
						
							|  |  |  | #define CSR_DIR_READ		0x40	/* rw direction, 1 = read 0 = write */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define CSR_RESET              0x80	/* wo  Resets 53c400 */
 | 
					
						
							|  |  |  | #define CSR_5380_REG           0x80	/* ro  5380 registers can be accessed */
 | 
					
						
							|  |  |  | #define CSR_TRANS_DIR          0x40	/* rw  Data transfer direction */
 | 
					
						
							|  |  |  | #define CSR_SCSI_BUFF_INTR     0x20	/* rw  Enable int on transfer ready */
 | 
					
						
							|  |  |  | #define CSR_5380_INTR          0x10	/* rw  Enable 5380 interrupts */
 | 
					
						
							|  |  |  | #define CSR_SHARED_INTR        0x08	/* rw  Interrupt sharing */
 | 
					
						
							|  |  |  | #define CSR_HOST_BUF_NOT_RDY   0x04	/* ro  Host buffer not ready */
 | 
					
						
							|  |  |  | #define CSR_SCSI_BUF_RDY       0x02	/* ro  SCSI buffer ready */
 | 
					
						
							|  |  |  | #define CSR_GATED_5380_IRQ     0x01	/* ro  Last block xferred */
 | 
					
						
							|  |  |  | #define CSR_INT_BASE (CSR_SCSI_BUFF_INTR | CSR_5380_INTR)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DTC_BLK_CNT		0x101	/* rw 
 | 
					
						
							|  |  |  | 					 * # of 128-byte blocks to transfer */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define D_CR_ACCESS             0x80	/* ro set=can access 3280 registers */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DTC_SWITCH_REG		0x3982	/* ro - DIP switches */
 | 
					
						
							|  |  |  | #define DTC_RESUME_XFER		0x3982	/* wo - resume data xfer 
 | 
					
						
							|  |  |  | 					 * after disconnect/reconnect*/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define DTC_5380_OFFSET		0x3880	/* 8 registers here, see NCR5380.h */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*!!!! for dtc, it's a 128 byte buffer at 3900 !!! */ | 
					
						
							|  |  |  | #define DTC_DATA_BUF		0x3900	/* rw 128 bytes long */
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct override { | 
					
						
							|  |  |  | 	unsigned int address; | 
					
						
							|  |  |  | 	int irq; | 
					
						
							|  |  |  | } overrides | 
					
						
							|  |  |  | #ifdef OVERRIDE
 | 
					
						
							|  |  |  | [] __initdata = OVERRIDE; | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2007-08-10 14:50:41 -07:00
										 |  |  | [4] __initdata = { | 
					
						
							|  |  |  | 	{ 0, IRQ_AUTO }, { 0, IRQ_AUTO }, { 0, IRQ_AUTO }, { 0, IRQ_AUTO } | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-08 22:23:48 -07:00
										 |  |  | #define NO_OVERRIDES ARRAY_SIZE(overrides)
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static struct base { | 
					
						
							|  |  |  | 	unsigned long address; | 
					
						
							|  |  |  | 	int noauto; | 
					
						
							| 
									
										
										
										
											2006-06-08 22:23:48 -07:00
										 |  |  | } bases[] __initdata = { | 
					
						
							|  |  |  | 	{ 0xcc000, 0 }, | 
					
						
							|  |  |  | 	{ 0xc8000, 0 }, | 
					
						
							|  |  |  | 	{ 0xdc000, 0 }, | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 	{ 0xd8000, 0 } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-08 22:23:48 -07:00
										 |  |  | #define NO_BASES ARRAY_SIZE(bases)
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static const struct signature { | 
					
						
							|  |  |  | 	const char *string; | 
					
						
							|  |  |  | 	int offset; | 
					
						
							| 
									
										
										
										
											2006-06-08 22:23:48 -07:00
										 |  |  | } signatures[] = { | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 	{"DATA TECHNOLOGY CORPORATION BIOS", 0x25}, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-06-08 22:23:48 -07:00
										 |  |  | #define NO_SIGNATURES ARRAY_SIZE(signatures)
 | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifndef MODULE
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Function : dtc_setup(char *str, int *ints) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Purpose : LILO command line initialization of the overrides array, | 
					
						
							| 
									
										
										
										
											2006-06-08 22:23:48 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  |  * Inputs : str - unused, ints - array of integer parameters with ints[0] | 
					
						
							|  |  |  |  *	equal to the number of ints. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2007-08-10 14:50:41 -07:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void __init dtc_setup(char *str, int *ints) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	static int commandline_current = 0; | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	if (ints[0] != 2) | 
					
						
							|  |  |  | 		printk("dtc_setup: usage dtc=address,irq\n"); | 
					
						
							|  |  |  | 	else if (commandline_current < NO_OVERRIDES) { | 
					
						
							|  |  |  | 		overrides[commandline_current].address = ints[1]; | 
					
						
							|  |  |  | 		overrides[commandline_current].irq = ints[2]; | 
					
						
							|  |  |  | 		for (i = 0; i < NO_BASES; ++i) | 
					
						
							|  |  |  | 			if (bases[i].address == ints[1]) { | 
					
						
							|  |  |  | 				bases[i].noauto = 1; | 
					
						
							|  |  |  | 				break; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		++commandline_current; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 18:31:40 +01:00
										 |  |  |  * Function : int dtc_detect(struct scsi_host_template * tpnt) | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Purpose : detects and initializes DTC 3180/3280 controllers | 
					
						
							|  |  |  |  *	that were autoprobed, overridden on the LILO command line,  | 
					
						
							|  |  |  |  *	or specified at compile time. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Inputs : tpnt - template for this SCSI adapter. | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  * Returns : 1 if a host adapter was found, 0 if not. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 18:31:40 +01:00
										 |  |  | static int __init dtc_detect(struct scsi_host_template * tpnt) | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | { | 
					
						
							|  |  |  | 	static int current_override = 0, current_base = 0; | 
					
						
							|  |  |  | 	struct Scsi_Host *instance; | 
					
						
							|  |  |  | 	unsigned int addr; | 
					
						
							|  |  |  | 	void __iomem *base; | 
					
						
							|  |  |  | 	int sig, count; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	tpnt->proc_name = "dtc3x80"; | 
					
						
							|  |  |  | 	tpnt->proc_info = &dtc_proc_info; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (count = 0; current_override < NO_OVERRIDES; ++current_override) { | 
					
						
							|  |  |  | 		addr = 0; | 
					
						
							|  |  |  | 		base = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (overrides[current_override].address) { | 
					
						
							|  |  |  | 			addr = overrides[current_override].address; | 
					
						
							|  |  |  | 			base = ioremap(addr, 0x2000); | 
					
						
							|  |  |  | 			if (!base) | 
					
						
							|  |  |  | 				addr = 0; | 
					
						
							|  |  |  | 		} else | 
					
						
							|  |  |  | 			for (; !addr && (current_base < NO_BASES); ++current_base) { | 
					
						
							|  |  |  | #if (DTCDEBUG & DTCDEBUG_INIT)
 | 
					
						
							| 
									
										
										
										
											2007-08-10 14:50:41 -07:00
										 |  |  | 				printk(KERN_DEBUG "scsi-dtc : probing address %08x\n", bases[current_base].address); | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 				if (bases[current_base].noauto) | 
					
						
							|  |  |  | 					continue; | 
					
						
							|  |  |  | 				base = ioremap(bases[current_base].address, 0x2000); | 
					
						
							|  |  |  | 				if (!base) | 
					
						
							|  |  |  | 					continue; | 
					
						
							|  |  |  | 				for (sig = 0; sig < NO_SIGNATURES; ++sig) { | 
					
						
							|  |  |  | 					if (check_signature(base + signatures[sig].offset, signatures[sig].string, strlen(signatures[sig].string))) { | 
					
						
							|  |  |  | 						addr = bases[current_base].address; | 
					
						
							|  |  |  | #if (DTCDEBUG & DTCDEBUG_INIT)
 | 
					
						
							| 
									
										
										
										
											2007-09-20 15:12:11 +01:00
										 |  |  | 						printk(KERN_DEBUG "scsi-dtc : detected board.\n"); | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 						goto found; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				iounmap(base); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if defined(DTCDEBUG) && (DTCDEBUG & DTCDEBUG_INIT)
 | 
					
						
							| 
									
										
										
										
											2007-08-10 14:50:41 -07:00
										 |  |  | 		printk(KERN_DEBUG "scsi-dtc : base = %08x\n", addr); | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (!addr) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | found: | 
					
						
							|  |  |  | 		instance = scsi_register(tpnt, sizeof(struct NCR5380_hostdata)); | 
					
						
							|  |  |  | 		if (instance == NULL) | 
					
						
							|  |  |  | 			break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		instance->base = addr; | 
					
						
							|  |  |  | 		((struct NCR5380_hostdata *)(instance)->hostdata)->base = base; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		NCR5380_init(instance, 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		NCR5380_write(DTC_CONTROL_REG, CSR_5380_INTR);	/* Enable int's */ | 
					
						
							|  |  |  | 		if (overrides[current_override].irq != IRQ_AUTO) | 
					
						
							|  |  |  | 			instance->irq = overrides[current_override].irq; | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			instance->irq = NCR5380_probe_irq(instance, DTC_IRQS); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef DONT_USE_INTR
 | 
					
						
							|  |  |  | 		/* With interrupts enabled, it will sometimes hang when doing heavy
 | 
					
						
							|  |  |  | 		 * reads. So better not enable them until I finger it out. */ | 
					
						
							|  |  |  | 		if (instance->irq != SCSI_IRQ_NONE) | 
					
						
							| 
									
										
										
										
											2007-11-11 19:52:05 -05:00
										 |  |  | 			if (request_irq(instance->irq, dtc_intr, IRQF_DISABLED, | 
					
						
							|  |  |  | 					"dtc", instance)) { | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 				printk(KERN_ERR "scsi%d : IRQ%d not free, interrupts disabled\n", instance->host_no, instance->irq); | 
					
						
							|  |  |  | 				instance->irq = SCSI_IRQ_NONE; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		if (instance->irq == SCSI_IRQ_NONE) { | 
					
						
							|  |  |  | 			printk(KERN_WARNING "scsi%d : interrupts not enabled. for better interactive performance,\n", instance->host_no); | 
					
						
							|  |  |  | 			printk(KERN_WARNING "scsi%d : please jumper the board for a free IRQ.\n", instance->host_no); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 		if (instance->irq != SCSI_IRQ_NONE) | 
					
						
							|  |  |  | 			printk(KERN_WARNING "scsi%d : interrupts not used. Might as well not jumper it.\n", instance->host_no); | 
					
						
							|  |  |  | 		instance->irq = SCSI_IRQ_NONE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | #if defined(DTCDEBUG) && (DTCDEBUG & DTCDEBUG_INIT)
 | 
					
						
							|  |  |  | 		printk("scsi%d : irq = %d\n", instance->host_no, instance->irq); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		printk(KERN_INFO "scsi%d : at 0x%05X", instance->host_no, (int) instance->base); | 
					
						
							|  |  |  | 		if (instance->irq == SCSI_IRQ_NONE) | 
					
						
							|  |  |  | 			printk(" interrupts disabled"); | 
					
						
							|  |  |  | 		else | 
					
						
							|  |  |  | 			printk(" irq %d", instance->irq); | 
					
						
							|  |  |  | 		printk(" options CAN_QUEUE=%d  CMD_PER_LUN=%d release=%d", CAN_QUEUE, CMD_PER_LUN, DTC_PUBLIC_RELEASE); | 
					
						
							|  |  |  | 		NCR5380_print_options(instance); | 
					
						
							|  |  |  | 		printk("\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		++current_override; | 
					
						
							|  |  |  | 		++count; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return count; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * Function : int dtc_biosparam(Disk * disk, struct block_device *dev, int *ip) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Purpose : Generates a BIOS / DOS compatible H-C-S mapping for  | 
					
						
							|  |  |  |  *	the specified device / size. | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  * Inputs : size = size of device in sectors (512 bytes), dev = block device | 
					
						
							|  |  |  |  *	major / minor, ip[] = {heads, sectors, cylinders}   | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns : always 0 (success), initializes ip | 
					
						
							|  |  |  |  *	 | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* 
 | 
					
						
							|  |  |  |  * XXX Most SCSI boards use this mapping, I could be incorrect.  Some one | 
					
						
							|  |  |  |  * using hard disks on a trantor should verify that this mapping corresponds | 
					
						
							|  |  |  |  * to that used by the BIOS / ASPI driver by running the linux fdisk program | 
					
						
							|  |  |  |  * and matching the H_C_S coordinates to what DOS uses. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int dtc_biosparam(struct scsi_device *sdev, struct block_device *dev, | 
					
						
							|  |  |  | 			 sector_t capacity, int *ip) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int size = capacity; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ip[0] = 64; | 
					
						
							|  |  |  | 	ip[1] = 32; | 
					
						
							|  |  |  | 	ip[2] = size >> 11; | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************
 | 
					
						
							|  |  |  |  * Function : int NCR5380_pread (struct Scsi_Host *instance,  | 
					
						
							|  |  |  |  *	unsigned char *dst, int len) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Purpose : Fast 5380 pseudo-dma read function, reads len bytes to  | 
					
						
							|  |  |  |  *	dst | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  * Inputs : dst = destination, len = length in bytes | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns : 0 on success, non zero on a failure such as a watchdog  | 
					
						
							|  |  |  |  * 	timeout. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int dtc_maxi = 0; | 
					
						
							|  |  |  | static int dtc_wmaxi = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int NCR5380_pread(struct Scsi_Host *instance, unsigned char *dst, int len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	unsigned char *d = dst; | 
					
						
							|  |  |  | 	int i;			/* For counting time spent in the poll-loop */ | 
					
						
							|  |  |  | 	NCR5380_local_declare(); | 
					
						
							|  |  |  | 	NCR5380_setup(instance); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	i = 0; | 
					
						
							|  |  |  | 	NCR5380_read(RESET_PARITY_INTERRUPT_REG); | 
					
						
							|  |  |  | 	NCR5380_write(MODE_REG, MR_ENABLE_EOP_INTR | MR_DMA_MODE); | 
					
						
							|  |  |  | 	if (instance->irq == SCSI_IRQ_NONE) | 
					
						
							|  |  |  | 		NCR5380_write(DTC_CONTROL_REG, CSR_DIR_READ); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		NCR5380_write(DTC_CONTROL_REG, CSR_DIR_READ | CSR_INT_BASE); | 
					
						
							|  |  |  | 	NCR5380_write(DTC_BLK_CNT, len >> 7);	/* Block count */ | 
					
						
							|  |  |  | 	rtrc(1); | 
					
						
							|  |  |  | 	while (len > 0) { | 
					
						
							|  |  |  | 		rtrc(2); | 
					
						
							|  |  |  | 		while (NCR5380_read(DTC_CONTROL_REG) & CSR_HOST_BUF_NOT_RDY) | 
					
						
							|  |  |  | 			++i; | 
					
						
							|  |  |  | 		rtrc(3); | 
					
						
							|  |  |  | 		memcpy_fromio(d, base + DTC_DATA_BUF, 128); | 
					
						
							|  |  |  | 		d += 128; | 
					
						
							|  |  |  | 		len -= 128; | 
					
						
							|  |  |  | 		rtrc(7); | 
					
						
							|  |  |  | 		/*** with int's on, it sometimes hangs after here.
 | 
					
						
							|  |  |  | 		 * Looks like something makes HBNR go away. */ | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rtrc(4); | 
					
						
							|  |  |  | 	while (!(NCR5380_read(DTC_CONTROL_REG) & D_CR_ACCESS)) | 
					
						
							|  |  |  | 		++i; | 
					
						
							|  |  |  | 	NCR5380_write(MODE_REG, 0);	/* Clear the operating mode */ | 
					
						
							|  |  |  | 	rtrc(0); | 
					
						
							|  |  |  | 	NCR5380_read(RESET_PARITY_INTERRUPT_REG); | 
					
						
							|  |  |  | 	if (i > dtc_maxi) | 
					
						
							|  |  |  | 		dtc_maxi = i; | 
					
						
							|  |  |  | 	return (0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /****************************************************************
 | 
					
						
							|  |  |  |  * Function : int NCR5380_pwrite (struct Scsi_Host *instance,  | 
					
						
							|  |  |  |  *	unsigned char *src, int len) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Purpose : Fast 5380 pseudo-dma write function, transfers len bytes from | 
					
						
							|  |  |  |  *	src | 
					
						
							|  |  |  |  *  | 
					
						
							|  |  |  |  * Inputs : src = source, len = length in bytes | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns : 0 on success, non zero on a failure such as a watchdog  | 
					
						
							|  |  |  |  * 	timeout. | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int NCR5380_pwrite(struct Scsi_Host *instance, unsigned char *src, int len) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	int i; | 
					
						
							|  |  |  | 	NCR5380_local_declare(); | 
					
						
							|  |  |  | 	NCR5380_setup(instance); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	NCR5380_read(RESET_PARITY_INTERRUPT_REG); | 
					
						
							|  |  |  | 	NCR5380_write(MODE_REG, MR_ENABLE_EOP_INTR | MR_DMA_MODE); | 
					
						
							|  |  |  | 	/* set direction (write) */ | 
					
						
							|  |  |  | 	if (instance->irq == SCSI_IRQ_NONE) | 
					
						
							|  |  |  | 		NCR5380_write(DTC_CONTROL_REG, 0); | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		NCR5380_write(DTC_CONTROL_REG, CSR_5380_INTR); | 
					
						
							|  |  |  | 	NCR5380_write(DTC_BLK_CNT, len >> 7);	/* Block count */ | 
					
						
							|  |  |  | 	for (i = 0; len > 0; ++i) { | 
					
						
							|  |  |  | 		rtrc(5); | 
					
						
							|  |  |  | 		/* Poll until the host buffer can accept data. */ | 
					
						
							|  |  |  | 		while (NCR5380_read(DTC_CONTROL_REG) & CSR_HOST_BUF_NOT_RDY) | 
					
						
							|  |  |  | 			++i; | 
					
						
							|  |  |  | 		rtrc(3); | 
					
						
							|  |  |  | 		memcpy_toio(base + DTC_DATA_BUF, src, 128); | 
					
						
							|  |  |  | 		src += 128; | 
					
						
							|  |  |  | 		len -= 128; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	rtrc(4); | 
					
						
							|  |  |  | 	while (!(NCR5380_read(DTC_CONTROL_REG) & D_CR_ACCESS)) | 
					
						
							|  |  |  | 		++i; | 
					
						
							|  |  |  | 	rtrc(6); | 
					
						
							|  |  |  | 	/* Wait until the last byte has been sent to the disk */ | 
					
						
							|  |  |  | 	while (!(NCR5380_read(TARGET_COMMAND_REG) & TCR_LAST_BYTE_SENT)) | 
					
						
							|  |  |  | 		++i; | 
					
						
							|  |  |  | 	rtrc(7); | 
					
						
							|  |  |  | 	/* Check for parity error here. fixme. */ | 
					
						
							|  |  |  | 	NCR5380_write(MODE_REG, 0);	/* Clear the operating mode */ | 
					
						
							|  |  |  | 	rtrc(0); | 
					
						
							|  |  |  | 	if (i > dtc_wmaxi) | 
					
						
							|  |  |  | 		dtc_wmaxi = i; | 
					
						
							|  |  |  | 	return (0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MODULE_LICENSE("GPL"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "NCR5380.c"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int dtc_release(struct Scsi_Host *shost) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	NCR5380_local_declare(); | 
					
						
							|  |  |  | 	NCR5380_setup(shost); | 
					
						
							|  |  |  | 	if (shost->irq) | 
					
						
							| 
									
										
										
										
											2007-11-11 19:52:05 -05:00
										 |  |  | 		free_irq(shost->irq, shost); | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 	NCR5380_exit(shost); | 
					
						
							|  |  |  | 	if (shost->io_port && shost->n_io_port) | 
					
						
							|  |  |  | 		release_region(shost->io_port, shost->n_io_port); | 
					
						
							|  |  |  | 	scsi_unregister(shost); | 
					
						
							|  |  |  | 	iounmap(base); | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-10-31 18:31:40 +01:00
										 |  |  | static struct scsi_host_template driver_template = { | 
					
						
							| 
									
										
										
										
											2005-04-16 15:20:36 -07:00
										 |  |  | 	.name				= "DTC 3180/3280 ", | 
					
						
							|  |  |  | 	.detect				= dtc_detect, | 
					
						
							|  |  |  | 	.release			= dtc_release, | 
					
						
							|  |  |  | 	.queuecommand			= dtc_queue_command, | 
					
						
							|  |  |  | 	.eh_abort_handler		= dtc_abort, | 
					
						
							|  |  |  | 	.eh_bus_reset_handler		= dtc_bus_reset, | 
					
						
							|  |  |  | 	.bios_param     		= dtc_biosparam, | 
					
						
							|  |  |  | 	.can_queue      		= CAN_QUEUE, | 
					
						
							|  |  |  | 	.this_id        		= 7, | 
					
						
							|  |  |  | 	.sg_tablesize   		= SG_ALL, | 
					
						
							|  |  |  | 	.cmd_per_lun    		= CMD_PER_LUN, | 
					
						
							|  |  |  | 	.use_clustering 		= DISABLE_CLUSTERING, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | #include "scsi_module.c"
 |