/* * 53c710 driver. Modified from Drew Eckhardts driver * for 53c810 by Richard Hirst [richard@sleepie.demon.co.uk] * Check out PERM_OPTIONS and EXPECTED_CLOCK, which may be defined in the * relevant machine specific file (eg. mvme16x.[ch], amiga7xx.[ch]). * There are also currently some defines at the top of 53c7xx.scr. * The chip type is #defined in script_asm.pl, as well as the Makefile. * Host scsi ID expected to be 7 - see NCR53c7x0_init(). * * I have removed the PCI code and some of the 53c8xx specific code - * simply to make this file smaller and easier to manage. * * MVME16x issues: * Problems trying to read any chip registers in NCR53c7x0_init(), as they * may never have been set by 16xBug (eg. If kernel has come in over tftp). */ /* * Adapted for Linux/m68k Amiga platforms for the A4000T/A4091 and * WarpEngine SCSI controllers. * By Alan Hourihane <alanh@fairlite.demon.co.uk> * Thanks to Richard Hirst for making it possible with the MVME additions */ /* * 53c710 rev 0 doesn't support add with carry. Rev 1 and 2 does. To * overcome this problem you can define FORCE_DSA_ALIGNMENT, which ensures * that the DSA address is always xxxxxx00. If disconnection is not allowed, * then the script only ever tries to add small (< 256) positive offsets to * DSA, so lack of carry isn't a problem. FORCE_DSA_ALIGNMENT can, of course, * be defined for all chip revisions at a small cost in memory usage. */ #define FORCE_DSA_ALIGNMENT /* * Selection timer does not always work on the 53c710, depending on the * timing at the last disconnect, if this is a problem for you, try * using validids as detailed below. * * Options for the NCR7xx driver * * noasync:0 - disables sync and asynchronous negotiation * nosync:0 - disables synchronous negotiation (does async) * nodisconnect:0 - disables disconnection * validids:0x?? - Bitmask field that disallows certain ID's. * - e.g. 0x03 allows ID 0,1 * - 0x1F allows ID 0,1,2,3,4 * opthi:n - replace top word of options with 'n' * optlo:n - replace bottom word of options with 'n' * - ALWAYS SPECIFY opthi THEN optlo <<<<<<<<<< */ /* * PERM_OPTIONS are driver options which will be enabled for all NCR boards * in the system at driver initialization time. * * Don't THINK about touching these in PERM_OPTIONS : * OPTION_MEMORY_MAPPED * 680x0 doesn't have an IO map! * * OPTION_DEBUG_TEST1 * Test 1 does bus mastering and interrupt tests, which will help weed * out brain damaged main boards. * * Other PERM_OPTIONS settings are listed below. Note the actual options * required are set in the relevant file (mvme16x.c, amiga7xx.c, etc): * * OPTION_NO_ASYNC * Don't negotiate for asynchronous transfers on the first command * when OPTION_ALWAYS_SYNCHRONOUS is set. Useful for dain bramaged * devices which do something bad rather than sending a MESSAGE * REJECT back to us like they should if they can't cope. * * OPTION_SYNCHRONOUS * Enable support for synchronous transfers. Target negotiated * synchronous transfers will be responded to. To initiate * a synchronous transfer request, call * * request_synchronous (hostno, target) * * from within KGDB. * * OPTION_ALWAYS_SYNCHRONOUS * Negotiate for synchronous transfers with every target after * driver initialization or a SCSI bus reset. This is a bit dangerous, * since there are some dain bramaged SCSI devices which will accept * SDTR messages but keep talking asynchronously. * * OPTION_DISCONNECT * Enable support for disconnect/reconnect. To change the * default setting on a given host adapter, call * * request_disconnect (hostno, allow) * * where allow is non-zero to allow, 0 to disallow. * * If you really want to run 10MHz FAST SCSI-II transfers, you should * know that the NCR driver currently ignores parity information. Most * systems do 5MHz SCSI fine. I've seen a lot that have problems faster * than 8MHz. To play it safe, we only request 5MHz transfers. * * If you'd rather get 10MHz transfers, edit sdtr_message and change * the fourth byte from 50 to 25. */ /* * Sponsored by * iX Multiuser Multitasking Magazine * Hannover, Germany * hm@ix.de * * Copyright 1993, 1994, 1995 Drew Eckhardt * Visionary Computing * (Unix and Linux consulting and custom programming) * drew@PoohSticks.ORG * +1 (303) 786-7975 * * TolerANT and SCSI SCRIPTS are registered trademarks of NCR Corporation. * * For more information, please consult * * NCR53C810 * SCSI I/O Processor * Programmer's Guide * * NCR 53C810 * PCI-SCSI I/O Processor * Data Manual * * NCR 53C810/53C820 * PCI-SCSI I/O Processor Design In Guide * * For literature on Symbios Logic Inc. formerly NCR, SCSI, * and Communication products please call (800) 334-5454 or * (719) 536-3300. * * PCI BIOS Specification Revision * PCI Local Bus Specification * PCI System Design Guide * * PCI Special Interest Group * M/S HF3-15A * 5200 N.E. Elam Young Parkway * Hillsboro, Oregon 97124-6497 * +1 (503) 696-2000 * +1 (800) 433-5177 */ /* * Design issues : * The cumulative latency needed to propagate a read/write request * through the file system, buffer cache, driver stacks, SCSI host, and * SCSI device is ultimately the limiting factor in throughput once we * have a sufficiently fast host adapter. * * So, to maximize performance we want to keep the ratio of latency to data * transfer time to a minimum by * 1. Minimizing the total number of commands sent (typical command latency * including drive and bus mastering host overhead is as high as 4.5ms) * to transfer a given amount of data. * * This is accomplished by placing no arbitrary limit on the number * of scatter/gather buffers supported, since we can transfer 1K * per scatter/gather buffer without Eric's cluster patches, * 4K with. * * 2. Minimizing the number of fatal interrupts serviced, since * fatal interrupts halt the SCSI I/O processor. Basically, * this means offloading the practical maximum amount of processing * to the SCSI chip. * * On the NCR53c810/820/720, this is accomplished by using * interrupt-on-the-fly signals when commands complete, * and only handling fatal errors and SDTR / WDTR messages * in the host code. * * On the NCR53c710, interrupts are generated as on the NCR53c8x0, * only the lack of a interrupt-on-the-fly facility complicates * things. Also, SCSI ID registers and commands are * bit fielded rather than binary encoded. * * On the NCR53c700 and NCR53c700-66, operations that are done via * indirect, table mode on the more advanced chips must be * replaced by calls through a jump table which * acts as a surrogate for the DSA. Unfortunately, this * will mean that we must service an interrupt for each * disconnect/reconnect. * * 3. Eliminating latency by pipelining operations at the different levels. * * This driver allows a configurable number of commands to be enqueued * for each target/lun combination (experimentally, I have discovered * that two seems to work best) and will ultimately allow for * SCSI-II tagged queuing. * * * Architecture : * This driver is built around a Linux queue of commands waiting to * be executed, and a shared Linux/NCR array of commands to start. Commands * are transferred to the array by the run_process_issue_queue() function * which is called whenever a command completes. * * As commands are completed, the interrupt routine is triggered, * looks for commands in the linked list of completed commands with * valid status, removes these commands from a list of running commands, * calls the done routine, and flags their target/luns as not busy. * * Due to limitations in the intelligence of the NCR chips, certain * concessions are made. In many cases, it is easier to dynamically * generate/fix-up code rather than calculate on the NCR at run time. * So, code is generated or fixed up for * * - Handling data transfers, using a variable number of MOVE instructions * interspersed with CALL MSG_IN, WHEN MSGIN instructions. * * The DATAIN and DATAOUT routines are separate, so that an incorrect * direction can be trapped, and space isn't wasted. * * It may turn out that we're better off using some sort * of table indirect instruction in a loop with a variable * sized table on the NCR53c710 and newer chips. * * - Checking for reselection (NCR53c710 and better) * * - Handling the details of SCSI context switches (NCR53c710 and better), * such as reprogramming appropriate synchronous parameters, * removing the dsa structure from the NCR's queue of outstanding * commands, etc. * */ #include <linux/module.h> #include <linux/config.h> #include <linux/types.h> #include <asm/setup.h> #include <asm/dma.h> #include <asm/io.h> #include <asm/system.h> #include <linux/delay.h> #include <linux/signal.h> #include <linux/sched.h> #include <linux/errno.h> #include <linux/string.h> #include <linux/slab.h> #include <linux/vmalloc.h> #include <linux/mm.h> #include <linux/ioport.h> #include <linux/time.h> #include <linux/blkdev.h> #include <linux/spinlock.h> #include <linux/interrupt.h> #include <asm/pgtable.h> #ifdef CONFIG_AMIGA #include <asm/amigahw.h> #include <asm/amigaints.h> #include <asm/irq.h> #define BIG_ENDIAN #define NO_IO_SPACE #endif #ifdef CONFIG_MVME16x #include <asm/mvme16xhw.h> #define BIG_ENDIAN #define NO_IO_SPACE #define VALID_IDS #endif #ifdef CONFIG_BVME6000 #include <asm/bvme6000hw.h> #define BIG_ENDIAN #define NO_IO_SPACE #define VALID_IDS #endif #include "scsi.h" #include <scsi/scsi_dbg.h> #include <scsi/scsi_host.h> #include <scsi/scsi_transport_spi.h> #include "53c7xx.h" #include <linux/stat.h> #include <linux/stddef.h> #ifdef NO_IO_SPACE /* * The following make the definitions in 53c7xx.h (write8, etc) smaller, * we don't have separate i/o space anyway. */ #undef inb #undef outb #undef inw #undef outw #undef inl #undef outl #define inb(x) 1 #define inw(x) 1 #define inl(x) 1 #define outb(x,y) 1 #define outw(x,y) 1 #define outl(x,y) 1 #endif static int check_address (unsigned long addr, int size); static void dump_events (struct Scsi_Host *host, int count); static Scsi_Cmnd * return_outstanding_commands (struct Scsi_Host *host, int free, int issue); static void hard_reset (struct Scsi_Host *host); static void ncr_scsi_reset (struct Scsi_Host *host); static void print_lots (struct Scsi_Host *host); static void set_synchronous (struct Scsi_Host *host, int target, int sxfer, int scntl3, int now_connected); static int datapath_residual (struct Scsi_Host *host); static const char * sbcl_to_phase (int sbcl); static void print_progress (Scsi_Cmnd *cmd); static void print_queues (struct Scsi_Host *host); static void process_issue_queue (unsigned long flags); static int shutdown (struct Scsi_Host *host); static void abnormal_finished (struct NCR53c7x0_cmd *cmd, int result); static int disable (struct Scsi_Host *host); static int NCR53c7xx_run_tests (struct Scsi_Host *host); static irqreturn_t NCR53c7x0_intr(int irq, void *dev_id, struct pt_regs * regs); static void NCR53c7x0_intfly (struct Scsi_Host *host); static int ncr_halt (struct Scsi_Host *host); static void intr_phase_mismatch (struct Scsi_Host *host, struct NCR53c7x0_cmd *cmd); static void intr_dma (struct Scsi_Host *host, struct NCR53c7x0_cmd *cmd); static void print_dsa (struct Scsi_Host *host, u32 *dsa, const char *prefix); static int print_insn (struct Scsi_Host *host, const u32 *insn, const char *prefix, int kernel); static void NCR53c7xx_dsa_fixup (struct NCR53c7x0_cmd *cmd); static void NCR53c7x0_init_fixup (struct Scsi_Host *host); static int NCR53c7x0_dstat_sir_intr (struct Scsi_Host *host, struct NCR53c7x0_cmd *cmd); static void NCR53c7x0_soft_reset (struct Scsi_Host *host); /* Size of event list (per host adapter) */ static int track_events = 0; static struct Scsi_Host *first_host = NULL; /* Head of list of NCR boards */ static struct scsi_host_template *the_template = NULL; /* NCR53c710 script handling code */ #include "53c7xx_d.h" #ifdef A_int_debug_sync #define DEBUG_SYNC_INTR A_int_debug_sync #endif int NCR53c7xx_script_len = sizeof (SCRIPT); int NCR53c7xx_dsa_len = A_dsa_end + Ent_dsa_zero - Ent_dsa_code_template; #ifdef FORCE_DSA_ALIGNMENT int CmdPageStart = (0 - Ent_dsa_zero - sizeof(struct NCR53c7x0_cmd)) & 0xff; #endif static char *setup_strings[] = {"","","","","","","",""}; #define MAX_SETUP_STRINGS (sizeof(setup_strings) / sizeof(char *)) #define SETUP_BUFFER_SIZE 200 static char setup_buffer[SETUP_BUFFER_SIZE]; static char setup_used[MAX_SETUP_STRINGS]; void ncr53c7xx_setup (char *str, int *ints) { int i; char *p1, *p2; p1 = setup_buffer; *p1 = '\0'; if (str) strncpy(p1, str, SETUP_BUFFER_SIZE - strlen(setup_buffer)); setup_buffer[SETUP_BUFFER_SIZE - 1] = '\0'; p1 = setup_buffer; i = 0; while (*p1 && (i < MAX_SETUP_STRINGS)) { p2 = strchr(p1, ','); if (p2) { *p2 = '\0'; if (p1 != p2) setup_strings[i] = p1; p1 = p2 + 1; i++; } else { setup_strings[i] = p1; break; } } for (i=0; i<MAX_SETUP_STRINGS; i++) setup_used[i] = 0; } /* check_setup_strings() returns index if key found, 0 if not */ static int check_setup_strings(char *key, int *flags, int *val, char *buf) { int x; char *cp; for (x=0; x<MAX_SETUP_STRINGS; x++) { if (setup_used[x]) continue; if (!strncmp(setup_strings[x], key, strlen(key))) break; if (!strncmp(setup_strings[x], "next", strlen("next"))) return 0; } if (x == MAX_SETUP_STRINGS) return 0; setup_used[x] = 1; cp = setup_strings[x] + strlen(key); *val = -1; if (*cp != ':') return ++x; cp++; if ((*cp >= '0') && (*cp <= '9')) { *val = simple_strtoul(cp,NULL,0); } return ++x; } /* * KNOWN BUGS : * - There is some sort of conflict when the PPP driver is compiled with * support for 16 channels? * * - On systems which predate the 1.3.x initialization order change, * the NCR driver will cause Cannot get free page messages to appear. * These are harmless, but I don't know of an easy way to avoid them. * * - With OPTION_DISCONNECT, on two systems under unknown circumstances, * we get a PHASE MISMATCH with DSA set to zero (suggests that we * are occurring somewhere in the reselection code) where * DSP=some value DCMD|DBC=same value. * * Closer inspection suggests that we may be trying to execute * some portion of the DSA? * scsi0 : handling residual transfer (+ 0 bytes from DMA FIFO) * scsi0 : handling residual transfer (+ 0 bytes from DMA FIFO) * scsi0 : no current command : unexpected phase MSGIN. * DSP=0x1c46cc, DCMD|DBC=0x1c46ac, DSA=0x0 * DSPS=0x0, TEMP=0x1c3e70, DMODE=0x80 * scsi0 : DSP-> * 001c46cc : 0x001c46cc 0x00000000 * 001c46d4 : 0x001c5ea0 0x000011f8 * * Changed the print code in the phase_mismatch handler so * that we call print_lots to try to diagnose this. * */ /* * Possible future direction of architecture for max performance : * * We're using a single start array for the NCR chip. This is * sub-optimal, because we cannot add a command which would conflict with * an executing command to this start queue, and therefore must insert the * next command for a given I/T/L combination after the first has completed; * incurring our interrupt latency between SCSI commands. * * To allow further pipelining of the NCR and host CPU operation, we want * to set things up so that immediately on termination of a command destined * for a given LUN, we get that LUN busy again. * * To do this, we need to add a 32 bit pointer to which is jumped to * on completion of a command. If no new command is available, this * would point to the usual DSA issue queue select routine. * * If one were, it would point to a per-NCR53c7x0_cmd select routine * which starts execution immediately, inserting the command at the head * of the start queue if the NCR chip is selected or reselected. * * We would change so that we keep a list of outstanding commands * for each unit, rather than a single running_list. We'd insert * a new command into the right running list; if the NCR didn't * have something running for that yet, we'd put it in the * start queue as well. Some magic needs to happen to handle the * race condition between the first command terminating before the * new one is written. * * Potential for profiling : * Call do_gettimeofday(struct timeval *tv) to get 800ns resolution. */ /* * TODO : * 1. To support WIDE transfers, not much needs to happen. We * should do CHMOVE instructions instead of MOVEs when * we have scatter/gather segments of uneven length. When * we do this, we need to handle the case where we disconnect * between segments. * * 2. Currently, when Icky things happen we do a FATAL(). Instead, * we want to do an integrity check on the parts of the NCR hostdata * structure which were initialized at boot time; FATAL() if that * fails, and otherwise try to recover. Keep track of how many * times this has happened within a single SCSI command; if it * gets excessive, then FATAL(). * * 3. Parity checking is currently disabled, and a few things should * happen here now that we support synchronous SCSI transfers : * 1. On soft-reset, we shoould set the EPC (Enable Parity Checking) * and AAP (Assert SATN/ on parity error) bits in SCNTL0. * * 2. We should enable the parity interrupt in the SIEN0 register. * * 3. intr_phase_mismatch() needs to believe that message out is * always an "acceptable" phase to have a mismatch in. If * the old phase was MSG_IN, we should send a MESSAGE PARITY * error. If the old phase was something else, we should send * a INITIATOR_DETECTED_ERROR message. Note that this could * cause a RESTORE POINTERS message; so we should handle that * correctly first. Instead, we should probably do an * initiator_abort. * * 4. MPEE bit of CTEST4 should be set so we get interrupted if * we detect an error. * * * 5. The initial code has been tested on the NCR53c810. I don't * have access to NCR53c700, 700-66 (Forex boards), NCR53c710 * (NCR Pentium systems), NCR53c720, NCR53c820, or NCR53c825 boards to * finish development on those platforms. * * NCR53c820/825/720 - need to add wide transfer support, including WDTR * negotiation, programming of wide transfer capabilities * on reselection and table indirect selection. * * NCR53c710 - need to add fatal interrupt or GEN code for * command completion signaling. Need to modify all * SDID, SCID, etc. registers, and table indirect select code * since these use bit fielded (ie 1<<target) instead of * binary encoded target ids. Need to accommodate * different register mappings, probably scan through * the SCRIPT code and change the non SFBR register operand * of all MOVE instructions. * * It is rather worse than this actually, the 710 corrupts * both TEMP and DSA when you do a MOVE MEMORY. This * screws you up all over the place. MOVE MEMORY 4 with a * destination of DSA seems to work OK, which helps some. * Richard Hirst richard@sleepie.demon.co.uk * * NCR53c700/700-66 - need to add code to refix addresses on * every nexus change, eliminate all table indirect code, * very messy. * * 6. The NCR53c7x0 series is very popular on other platforms that * could be running Linux - ie, some high performance AMIGA SCSI * boards use it. * * So, I should include #ifdef'd code so that it is * compatible with these systems. * * Specifically, the little Endian assumptions I made in my * bit fields need to change, and if the NCR doesn't see memory * the right way, we need to provide options to reverse words * when the scripts are relocated. * * 7. Use vremap() to access memory mapped boards. */ /* * Allow for simultaneous existence of multiple SCSI scripts so we * can have a single driver binary for all of the family. * * - one for NCR53c700 and NCR53c700-66 chips (not yet supported) * - one for rest (only the NCR53c810, 815, 820, and 825 are currently * supported) * * So that we only need two SCSI scripts, we need to modify things so * that we fixup register accesses in READ/WRITE instructions, and * we'll also have to accommodate the bit vs. binary encoding of IDs * with the 7xx chips. */ #define ROUNDUP(adr,type) \ ((void *) (((long) (adr) + sizeof(type) - 1) & ~(sizeof(type) - 1))) /* * Function: issue_to_cmd * * Purpose: convert jump instruction in issue array to NCR53c7x0_cmd * structure pointer. * * Inputs; issue - pointer to start of NOP or JUMP instruction * in issue array. * * Returns: pointer to command on success; 0 if opcode is NOP. */ static inline struct NCR53c7x0_cmd * issue_to_cmd (struct Scsi_Host *host, struct NCR53c7x0_hostdata *hostdata, u32 *issue) { return (issue[0] != hostdata->NOP_insn) ? /* * If the IF TRUE bit is set, it's a JUMP instruction. The * operand is a bus pointer to the dsa_begin routine for this DSA. The * dsa field of the NCR53c7x0_cmd structure starts with the * DSA code template. By converting to a virtual address, * subtracting the code template size, and offset of the * dsa field, we end up with a pointer to the start of the * structure (alternatively, we could use the * dsa_cmnd field, an anachronism from when we weren't * sure what the relationship between the NCR structures * and host structures were going to be. */ (struct NCR53c7x0_cmd *) ((char *) bus_to_virt (issue[1]) - (hostdata->E_dsa_code_begin - hostdata->E_dsa_code_template) - offsetof(struct NCR53c7x0_cmd, dsa)) /* If the IF TRUE bit is not set, it's a NOP */ : NULL; } /* * FIXME: we should junk these, in favor of synchronous_want and * wide_want in the NCR53c7x0_hostdata structure. */ /* Template for "preferred" synchronous transfer parameters. */ static const unsigned char sdtr_message[] = { #ifdef CONFIG_SCSI_NCR53C7xx_FAST EXTENDED_MESSAGE, 3 /* length */, EXTENDED_SDTR, 25 /* *4ns */, 8 /* off */ #else EXTENDED_MESSAGE, 3 /* length */, EXTENDED_SDTR, 50 /* *4ns */, 8 /* off */ #endif }; /* Template to request asynchronous transfers */ static const unsigned char async_message[] = { EXTENDED_MESSAGE, 3 /* length */, EXTENDED_SDTR, 0, 0 /* asynchronous */ }; /* Template for "preferred" WIDE transfer parameters */ static const unsigned char wdtr_message[] = { EXTENDED_MESSAGE, 2 /* length */, EXTENDED_WDTR, 1 /* 2^1 bytes */ }; #if 0 /* * Function : struct Scsi_Host *find_host (int host) * * Purpose : KGDB support function which translates a host number * to a host structure. * * Inputs : host - number of SCSI host * * Returns : NULL on failure, pointer to host structure on success. */ static struct Scsi_Host * find_host (int host) { struct Scsi_Host *h; for (h = first_host; h && h->host_no != host; h = h->next); if (!h) { printk (KERN_ALERT "scsi%d not found\n", host); return NULL; } else if (h->hostt != the_template) { printk (KERN_ALERT "scsi%d is not a NCR board\n", host); return NULL; } return h; } #if 0 /* * Function : request_synchronous (int host, int target) * * Purpose : KGDB interface which will allow us to negotiate for * synchronous transfers. This ill be replaced with a more * integrated function; perhaps a new entry in the scsi_host * structure, accessible via an ioctl() or perhaps /proc/scsi. * * Inputs : host - number of SCSI host; target - number of target. * * Returns : 0 when negotiation has been setup for next SCSI command, * -1 on failure. */ static int request_synchronous (int host, int target) { struct Scsi_Host *h; struct NCR53c7x0_hostdata *hostdata; unsigned long flags; if (target < 0) { printk (KERN_ALERT "target %d is bogus\n", target); return -1; } if (!(h = find_host (host))) return -1; else if (h->this_id == target) { printk (KERN_ALERT "target %d is host ID\n", target); return -1; } else if (target > h->max_id) { printk (KERN_ALERT "target %d exceeds maximum of %d\n", target, h->max_id); return -1; } hostdata = (struct NCR53c7x0_hostdata *)h->hostdata[0]; local_irq_save(flags); if (hostdata->initiate_sdtr & (1 << target)) { local_irq_restore(flags); printk (KERN_ALERT "target %d already doing SDTR\n", target); return -1; } hostdata->initiate_sdtr |= (1 << target); local_irq_restore(flags); return 0; } #endif /* * Function : request_disconnect (int host, int on_or_off) * * Purpose : KGDB support function, tells us to allow or disallow * disconnections. * * Inputs : host - number of SCSI host; on_or_off - non-zero to allow, * zero to disallow. * * Returns : 0 on success, * -1 on failure. */ static int request_disconnect (int host, int on_or_off) { struct Scsi_Host *h; struct NCR53c7x0_hostdata *hostdata; if (!(h = find_host (host))) return -1; hostdata = (struct NCR53c7x0_hostdata *) h->hostdata[0]; if (on_or_off) hostdata->options |= OPTION_DISCONNECT; else hostdata->options &= ~OPTION_DISCONNECT; return 0; } #endif /* * Function : static void NCR53c7x0_driver_init (struct Scsi_Host *host) * * Purpose : Initialize internal structures, as required on startup, or * after a SCSI bus reset. * * Inputs : host - pointer to this host adapter's structure */ static void NCR53c7x0_driver_init (struct Scsi_Host *host) { struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; int i, j; u32 *ncrcurrent; for (i = 0; i < 16; ++i) { hostdata->request_sense[i] = 0; for (j = 0; j < 8; ++j) hostdata->busy[i][j] = 0; set_synchronous (host, i, /* sxfer */ 0, hostdata->saved_scntl3, 0); } hostdata->issue_queue = NULL; hostdata->running_list = hostdata->finished_queue = hostdata->ncrcurrent = NULL; for (i = 0, ncrcurrent = (u32 *) hostdata->schedule; i < host->can_queue; ++i, ncrcurrent += 2) { ncrcurrent[0] = hostdata->NOP_insn; ncrcurrent[1] = 0xdeadbeef; } ncrcurrent[0] = ((DCMD_TYPE_TCI|DCMD_TCI_OP_JUMP) << 24) | DBC_TCI_TRUE; ncrcurrent[1] = (u32) virt_to_bus (hostdata->script) + hostdata->E_wait_reselect; hostdata->reconnect_dsa_head = 0; hostdata->addr_reconnect_dsa_head = (u32) virt_to_bus((void *) &(hostdata->reconnect_dsa_head)); hostdata->expecting_iid = 0; hostdata->expecting_sto = 0; if (hostdata->options & OPTION_ALWAYS_SYNCHRONOUS) hostdata->initiate_sdtr = 0xffff; else hostdata->initiate_sdtr = 0; hostdata->talked_to = 0; hostdata->idle = 1; } /* * Function : static int clock_to_ccf_710 (int clock) * * Purpose : Return the clock conversion factor for a given SCSI clock. * * Inputs : clock - SCSI clock expressed in Hz. * * Returns : ccf on success, -1 on failure. */ static int clock_to_ccf_710 (int clock) { if (clock <= 16666666) return -1; if (clock <= 25000000) return 2; /* Divide by 1.0 */ else if (clock <= 37500000) return 1; /* Divide by 1.5 */ else if (clock <= 50000000) return 0; /* Divide by 2.0 */ else if (clock <= 66000000) return 3; /* Divide by 3.0 */ else return -1; } /* * Function : static int NCR53c7x0_init (struct Scsi_Host *host) * * Purpose : initialize the internal structures for a given SCSI host * * Inputs : host - pointer to this host adapter's structure * * Preconditions : when this function is called, the chip_type * field of the hostdata structure MUST have been set. * * Returns : 0 on success, -1 on failure. */ int NCR53c7x0_init (struct Scsi_Host *host) { NCR53c7x0_local_declare(); int i, ccf; unsigned char revision; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; /* * There are some things which we need to know about in order to provide * a semblance of support. Print 'em if they aren't what we expect, * otherwise don't add to the noise. * * -1 means we don't know what to expect. */ int val, flags; char buf[32]; int expected_id = -1; int expected_clock = -1; int uninitialized = 0; #ifdef NO_IO_SPACE int expected_mapping = OPTION_MEMORY_MAPPED; #else int expected_mapping = OPTION_IO_MAPPED; #endif for (i=0;i<7;i++) hostdata->valid_ids[i] = 1; /* Default all ID's to scan */ /* Parse commandline flags */ if (check_setup_strings("noasync",&flags,&val,buf)) { hostdata->options |= OPTION_NO_ASYNC; hostdata->options &= ~(OPTION_SYNCHRONOUS | OPTION_ALWAYS_SYNCHRONOUS); } if (check_setup_strings("nosync",&flags,&val,buf)) { hostdata->options &= ~(OPTION_SYNCHRONOUS | OPTION_ALWAYS_SYNCHRONOUS); } if (check_setup_strings("nodisconnect",&flags,&val,buf)) hostdata->options &= ~OPTION_DISCONNECT; if (check_setup_strings("validids",&flags,&val,buf)) { for (i=0;i<7;i++) hostdata->valid_ids[i] = val & (1<<i); } if ((i = check_setup_strings("next",&flags,&val,buf))) { while (i) setup_used[--i] = 1; } if (check_setup_strings("opthi",&flags,&val,buf)) hostdata->options = (long long)val << 32; if (check_setup_strings("optlo",&flags,&val,buf)) hostdata->options |= val; NCR53c7x0_local_setup(host); switch (hostdata->chip) { case 710: case 770: hostdata->dstat_sir_intr = NCR53c7x0_dstat_sir_intr; hostdata->init_save_regs = NULL; hostdata->dsa_fixup = NCR53c7xx_dsa_fixup; hostdata->init_fixup = NCR53c7x0_init_fixup; hostdata->soft_reset = NCR53c7x0_soft_reset; hostdata->run_tests = NCR53c7xx_run_tests; expected_clock = hostdata->scsi_clock; expected_id = 7; break; default: printk ("scsi%d : chip type of %d is not supported yet, detaching.\n", host->host_no, hostdata->chip); scsi_unregister (host); return -1; } /* Assign constants accessed by NCR */ hostdata->NCR53c7xx_zero = 0; hostdata->NCR53c7xx_msg_reject = MESSAGE_REJECT; hostdata->NCR53c7xx_msg_abort = ABORT; hostdata->NCR53c7xx_msg_nop = NOP; hostdata->NOP_insn = (DCMD_TYPE_TCI|DCMD_TCI_OP_JUMP) << 24; if (expected_mapping == -1 || (hostdata->options & (OPTION_MEMORY_MAPPED)) != (expected_mapping & OPTION_MEMORY_MAPPED)) printk ("scsi%d : using %s mapped access\n", host->host_no, (hostdata->options & OPTION_MEMORY_MAPPED) ? "memory" : "io"); hostdata->dmode = (hostdata->chip == 700 || hostdata->chip == 70066) ? DMODE_REG_00 : DMODE_REG_10; hostdata->istat = ((hostdata->chip / 100) == 8) ? ISTAT_REG_800 : ISTAT_REG_700; /* We have to assume that this may be the first access to the chip, so * we must set EA in DCNTL. */ NCR53c7x0_write8 (DCNTL_REG, DCNTL_10_EA|DCNTL_10_COM); /* Only the ISTAT register is readable when the NCR is running, so make sure it's halted. */ ncr_halt(host); /* * XXX - the NCR53c700 uses bitfielded registers for SCID, SDID, etc, * as does the 710 with one bit per SCSI ID. Conversely, the NCR * uses a normal, 3 bit binary representation of these values. * * Get the rest of the NCR documentation, and FIND OUT where the change * was. */ #if 0 /* May not be able to do this - chip my not have been set up yet */ tmp = hostdata->this_id_mask = NCR53c7x0_read8(SCID_REG); for (host->this_id = 0; tmp != 1; tmp >>=1, ++host->this_id); #else host->this_id = 7; #endif /* * Note : we should never encounter a board setup for ID0. So, * if we see ID0, assume that it was uninitialized and set it * to the industry standard 7. */ if (!host->this_id) { printk("scsi%d : initiator ID was %d, changing to 7\n", host->host_no, host->this_id); host->this_id = 7; hostdata->this_id_mask = 1 << 7; uninitialized = 1; }; if (expected_id == -1 || host->this_id != expected_id) printk("scsi%d : using initiator ID %d\n", host->host_no, host->this_id); /* * Save important registers to allow a soft reset. */ /* * CTEST7 controls cache snooping, burst mode, and support for * external differential drivers. This isn't currently used - the * default value may not be optimal anyway. * Even worse, it may never have been set up since reset. */ hostdata->saved_ctest7 = NCR53c7x0_read8(CTEST7_REG) & CTEST7_SAVE; revision = (NCR53c7x0_read8(CTEST8_REG) & 0xF0) >> 4; switch (revision) { case 1: revision = 0; break; case 2: revision = 1; break; case 4: revision = 2; break; case 8: revision = 3; break; default: revision = 255; break; } printk("scsi%d: Revision 0x%x\n",host->host_no,revision); if ((revision == 0 || revision == 255) && (hostdata->options & (OPTION_SYNCHRONOUS|OPTION_DISCONNECT|OPTION_ALWAYS_SYNCHRONOUS))) { printk ("scsi%d: Disabling sync working and disconnect/reselect\n", host->host_no); hostdata->options &= ~(OPTION_SYNCHRONOUS|OPTION_DISCONNECT|OPTION_ALWAYS_SYNCHRONOUS); } /* * On NCR53c700 series chips, DCNTL controls the SCSI clock divisor, * on 800 series chips, it allows for a totem-pole IRQ driver. * NOTE saved_dcntl currently overwritten in init function. * The value read here may be garbage anyway, MVME16x board at least * does not initialise chip if kernel arrived via tftp. */ hostdata->saved_dcntl = NCR53c7x0_read8(DCNTL_REG); /* * DMODE controls DMA burst length, and on 700 series chips, * 286 mode and bus width * NOTE: On MVME16x, chip may have been reset, so this could be a * power-on/reset default value. */ hostdata->saved_dmode = NCR53c7x0_read8(hostdata->dmode); /* * Now that burst length and enabled/disabled status is known, * clue the user in on it. */ ccf = clock_to_ccf_710 (expected_clock); for (i = 0; i < 16; ++i) hostdata->cmd_allocated[i] = 0; if (hostdata->init_save_regs) hostdata->init_save_regs (host); if (hostdata->init_fixup) hostdata->init_fixup (host); if (!the_template) { the_template = host->hostt; first_host = host; } /* * Linux SCSI drivers have always been plagued with initialization * problems - some didn't work with the BIOS disabled since they expected * initialization from it, some didn't work when the networking code * was enabled and registers got scrambled, etc. * * To avoid problems like this, in the future, we will do a soft * reset on the SCSI chip, taking it back to a sane state. */ hostdata->soft_reset (host); #if 1 hostdata->debug_count_limit = -1; #else hostdata->debug_count_limit = 1; #endif hostdata->intrs = -1; hostdata->resets = -1; memcpy ((void *) hostdata->synchronous_want, (void *) sdtr_message, sizeof (hostdata->synchronous_want)); NCR53c7x0_driver_init (host); if (request_irq(host->irq, NCR53c7x0_intr, SA_SHIRQ, "53c7xx", host)) { printk("scsi%d : IRQ%d not free, detaching\n", host->host_no, host->irq); goto err_unregister; } if ((hostdata->run_tests && hostdata->run_tests(host) == -1) || (hostdata->options & OPTION_DEBUG_TESTS_ONLY)) { /* XXX Should disable interrupts, etc. here */ goto err_free_irq; } else { if (host->io_port) { host->n_io_port = 128; if (!request_region (host->io_port, host->n_io_port, "ncr53c7xx")) goto err_free_irq; } } if (NCR53c7x0_read8 (SBCL_REG) & SBCL_BSY) { printk ("scsi%d : bus wedge, doing SCSI reset\n", host->host_no); hard_reset (host); } return 0; err_free_irq: free_irq(host->irq, NCR53c7x0_intr); err_unregister: scsi_unregister(host); return -1; } /* * Function : int ncr53c7xx_init(struct scsi_host_template *tpnt, int board, int chip, * unsigned long base, int io_port, int irq, int dma, long long options, * int clock); * * Purpose : initializes a NCR53c7,8x0 based on base addresses, * IRQ, and DMA channel. * * Inputs : tpnt - Template for this SCSI adapter, board - board level * product, chip - 710 * * Returns : 0 on success, -1 on failure. * */ int ncr53c7xx_init (struct scsi_host_template *tpnt, int board, int chip, unsigned long base, int io_port, int irq, int dma, long long options, int clock) { struct Scsi_Host *instance; struct NCR53c7x0_hostdata *hostdata; char chip_str[80]; int script_len = 0, dsa_len = 0, size = 0, max_cmd_size = 0, schedule_size = 0, ok = 0; void *tmp; unsigned long page; switch (chip) { case 710: case 770: schedule_size = (tpnt->can_queue + 1) * 8 /* JUMP instruction size */; script_len = NCR53c7xx_script_len; dsa_len = NCR53c7xx_dsa_len; options |= OPTION_INTFLY; sprintf (chip_str, "NCR53c%d", chip); break; default: printk("scsi-ncr53c7xx : unsupported SCSI chip %d\n", chip); return -1; } printk("scsi-ncr53c7xx : %s at memory 0x%lx, io 0x%x, irq %d", chip_str, base, io_port, irq); if (dma == DMA_NONE) printk("\n"); else printk(", dma %d\n", dma); if (options & OPTION_DEBUG_PROBE_ONLY) { printk ("scsi-ncr53c7xx : probe only enabled, aborting initialization\n"); return -1; } max_cmd_size = sizeof(struct NCR53c7x0_cmd) + dsa_len + /* Size of dynamic part of command structure : */ 2 * /* Worst case : we don't know if we need DATA IN or DATA out */ ( 2 * /* Current instructions per scatter/gather segment */ tpnt->sg_tablesize + 3 /* Current startup / termination required per phase */ ) * 8 /* Each instruction is eight bytes */; /* Allocate fixed part of hostdata, dynamic part to hold appropriate SCSI SCRIPT(tm) plus a single, maximum-sized NCR53c7x0_cmd structure. We need a NCR53c7x0_cmd structure for scan_scsis() when we are not loaded as a module, and when we're loaded as a module, we can't use a non-dynamically allocated structure because modules are vmalloc()'d, which can allow structures to cross page boundaries and breaks our physical/virtual address assumptions for DMA. So, we stick it past the end of our hostdata structure. ASSUMPTION : Regardless of how many simultaneous SCSI commands we allow, the probe code only executes a _single_ instruction at a time, so we only need one here, and don't need to allocate NCR53c7x0_cmd structures for each target until we are no longer in scan_scsis and kmalloc() has become functional (memory_init() happens after all device driver initialization). */ size = sizeof(struct NCR53c7x0_hostdata) + script_len + /* Note that alignment will be guaranteed, since we put the command allocated at probe time after the fixed-up SCSI script, which consists of 32 bit words, aligned on a 32 bit boundary. But on a 64bit machine we need 8 byte alignment for hostdata->free, so we add in another 4 bytes to take care of potential misalignment */ (sizeof(void *) - sizeof(u32)) + max_cmd_size + schedule_size; page = __get_free_pages(GFP_ATOMIC,1); if(page==0) { printk(KERN_ERR "53c7xx: out of memory.\n"); return -ENOMEM; } #ifdef FORCE_DSA_ALIGNMENT /* * 53c710 rev.0 doesn't have an add-with-carry instruction. * Ensure we allocate enough memory to force DSA alignment. */ size += 256; #endif /* Size should be < 8K, so we can fit it in two pages. */ if (size > 8192) { printk(KERN_ERR "53c7xx: hostdata > 8K\n"); return -1; } instance = scsi_register (tpnt, 4); if (!instance) { free_page(page); return -1; } instance->hostdata[0] = page; memset((void *)instance->hostdata[0], 0, 8192); cache_push(virt_to_phys((void *)(instance->hostdata[0])), 8192); cache_clear(virt_to_phys((void *)(instance->hostdata[0])), 8192); kernel_set_cachemode((void *)instance->hostdata[0], 8192, IOMAP_NOCACHE_SER); /* FIXME : if we ever support an ISA NCR53c7xx based board, we need to check if the chip is running in a 16 bit mode, and if so unregister it if it is past the 16M (0x1000000) mark */ hostdata = (struct NCR53c7x0_hostdata *)instance->hostdata[0]; hostdata->size = size; hostdata->script_count = script_len / sizeof(u32); hostdata->board = board; hostdata->chip = chip; /* * Being memory mapped is more desirable, since * * - Memory accesses may be faster. * * - The destination and source address spaces are the same for * all instructions, meaning we don't have to twiddle dmode or * any other registers. * * So, we try for memory mapped, and if we don't get it, * we go for port mapped, and that failing we tell the user * it can't work. */ if (base) { instance->base = base; /* Check for forced I/O mapping */ if (!(options & OPTION_IO_MAPPED)) { options |= OPTION_MEMORY_MAPPED; ok = 1; } } else { options &= ~OPTION_MEMORY_MAPPED; } if (io_port) { instance->io_port = io_port; options |= OPTION_IO_MAPPED; ok = 1; } else { options &= ~OPTION_IO_MAPPED; } if (!ok) { printk ("scsi%d : not initializing, no I/O or memory mapping known \n", instance->host_no); scsi_unregister (instance); return -1; } instance->irq = irq; instance->dma_channel = dma; hostdata->options = options; hostdata->dsa_len = dsa_len; hostdata->max_cmd_size = max_cmd_size; hostdata->num_cmds = 1; hostdata->scsi_clock = clock; /* Initialize single command */ tmp = (hostdata->script + hostdata->script_count); #ifdef FORCE_DSA_ALIGNMENT { void *t = ROUNDUP(tmp, void *); if (((u32)t & 0xff) > CmdPageStart) t = (void *)((u32)t + 255); t = (void *)(((u32)t & ~0xff) + CmdPageStart); hostdata->free = t; #if 0 printk ("scsi: Registered size increased by 256 to %d\n", size); printk ("scsi: CmdPageStart = 0x%02x\n", CmdPageStart); printk ("scsi: tmp = 0x%08x, hostdata->free set to 0x%08x\n", (u32)tmp, (u32)t); #endif } #else hostdata->free = ROUNDUP(tmp, void *); #endif hostdata->free->real = tmp; hostdata->free->size = max_cmd_size; hostdata->free->free = NULL; hostdata->free->next = NULL; hostdata->extra_allocate = 0; /* Allocate command start code space */ hostdata->schedule = (chip == 700 || chip == 70066) ? NULL : (u32 *) ((char *)hostdata->free + max_cmd_size); /* * For diagnostic purposes, we don't really care how fast things blaze. * For profiling, we want to access the 800ns resolution system clock, * using a 'C' call on the host processor. * * Therefore, there's no need for the NCR chip to directly manipulate * this data, and we should put it wherever is most convenient for * Linux. */ if (track_events) hostdata->events = (struct NCR53c7x0_event *) (track_events ? vmalloc (sizeof (struct NCR53c7x0_event) * track_events) : NULL); else hostdata->events = NULL; if (hostdata->events) { memset ((void *) hostdata->events, 0, sizeof(struct NCR53c7x0_event) * track_events); hostdata->event_size = track_events; hostdata->event_index = 0; } else hostdata->event_size = 0; return NCR53c7x0_init(instance); } /* * Function : static void NCR53c7x0_init_fixup (struct Scsi_Host *host) * * Purpose : copy and fixup the SCSI SCRIPTS(tm) code for this device. * * Inputs : host - pointer to this host adapter's structure * */ static void NCR53c7x0_init_fixup (struct Scsi_Host *host) { NCR53c7x0_local_declare(); struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; unsigned char tmp; int i, ncr_to_memory, memory_to_ncr; u32 base; NCR53c7x0_local_setup(host); /* XXX - NOTE : this code MUST be made endian aware */ /* Copy code into buffer that was allocated at detection time. */ memcpy ((void *) hostdata->script, (void *) SCRIPT, sizeof(SCRIPT)); /* Fixup labels */ for (i = 0; i < PATCHES; ++i) hostdata->script[LABELPATCHES[i]] += virt_to_bus(hostdata->script); /* Fixup addresses of constants that used to be EXTERNAL */ patch_abs_32 (hostdata->script, 0, NCR53c7xx_msg_abort, virt_to_bus(&(hostdata->NCR53c7xx_msg_abort))); patch_abs_32 (hostdata->script, 0, NCR53c7xx_msg_reject, virt_to_bus(&(hostdata->NCR53c7xx_msg_reject))); patch_abs_32 (hostdata->script, 0, NCR53c7xx_zero, virt_to_bus(&(hostdata->NCR53c7xx_zero))); patch_abs_32 (hostdata->script, 0, NCR53c7xx_sink, virt_to_bus(&(hostdata->NCR53c7xx_sink))); patch_abs_32 (hostdata->script, 0, NOP_insn, virt_to_bus(&(hostdata->NOP_insn))); patch_abs_32 (hostdata->script, 0, schedule, virt_to_bus((void *) hostdata->schedule)); /* Fixup references to external variables: */ for (i = 0; i < EXTERNAL_PATCHES_LEN; ++i) hostdata->script[EXTERNAL_PATCHES[i].offset] += virt_to_bus(EXTERNAL_PATCHES[i].address); /* * Fixup absolutes set at boot-time. * * All non-code absolute variables suffixed with "dsa_" and "int_" * are constants, and need no fixup provided the assembler has done * it for us (I don't know what the "real" NCR assembler does in * this case, my assembler does the right magic). */ patch_abs_rwri_data (hostdata->script, 0, dsa_save_data_pointer, Ent_dsa_code_save_data_pointer - Ent_dsa_zero); patch_abs_rwri_data (hostdata->script, 0, dsa_restore_pointers, Ent_dsa_code_restore_pointers - Ent_dsa_zero); patch_abs_rwri_data (hostdata->script, 0, dsa_check_reselect, Ent_dsa_code_check_reselect - Ent_dsa_zero); /* * Just for the hell of it, preserve the settings of * Burst Length and Enable Read Line bits from the DMODE * register. Make sure SCRIPTS start automagically. */ #if defined(CONFIG_MVME16x) || defined(CONFIG_BVME6000) /* We know better what we want than 16xBug does! */ tmp = DMODE_10_BL_8 | DMODE_10_FC2; #else tmp = NCR53c7x0_read8(DMODE_REG_10); tmp &= (DMODE_BL_MASK | DMODE_10_FC2 | DMODE_10_FC1 | DMODE_710_PD | DMODE_710_UO); #endif if (!(hostdata->options & OPTION_MEMORY_MAPPED)) { base = (u32) host->io_port; memory_to_ncr = tmp|DMODE_800_DIOM; ncr_to_memory = tmp|DMODE_800_SIOM; } else { base = virt_to_bus((void *)host->base); memory_to_ncr = ncr_to_memory = tmp; } /* SCRATCHB_REG_10 == SCRATCHA_REG_800, as it happens */ patch_abs_32 (hostdata->script, 0, addr_scratch, base + SCRATCHA_REG_800); patch_abs_32 (hostdata->script, 0, addr_temp, base + TEMP_REG); patch_abs_32 (hostdata->script, 0, addr_dsa, base + DSA_REG); /* * I needed some variables in the script to be accessible to * both the NCR chip and the host processor. For these variables, * I made the arbitrary decision to store them directly in the * hostdata structure rather than in the RELATIVE area of the * SCRIPTS. */ patch_abs_rwri_data (hostdata->script, 0, dmode_memory_to_memory, tmp); patch_abs_rwri_data (hostdata->script, 0, dmode_memory_to_ncr, memory_to_ncr); patch_abs_rwri_data (hostdata->script, 0, dmode_ncr_to_memory, ncr_to_memory); patch_abs_32 (hostdata->script, 0, msg_buf, virt_to_bus((void *)&(hostdata->msg_buf))); patch_abs_32 (hostdata->script, 0, reconnect_dsa_head, virt_to_bus((void *)&(hostdata->reconnect_dsa_head))); patch_abs_32 (hostdata->script, 0, addr_reconnect_dsa_head, virt_to_bus((void *)&(hostdata->addr_reconnect_dsa_head))); patch_abs_32 (hostdata->script, 0, reselected_identify, virt_to_bus((void *)&(hostdata->reselected_identify))); /* reselected_tag is currently unused */ #if 0 patch_abs_32 (hostdata->script, 0, reselected_tag, virt_to_bus((void *)&(hostdata->reselected_tag))); #endif patch_abs_32 (hostdata->script, 0, test_dest, virt_to_bus((void*)&hostdata->test_dest)); patch_abs_32 (hostdata->script, 0, test_src, virt_to_bus(&hostdata->test_source)); patch_abs_32 (hostdata->script, 0, saved_dsa, virt_to_bus((void *)&hostdata->saved2_dsa)); patch_abs_32 (hostdata->script, 0, emulfly, virt_to_bus((void *)&hostdata->emulated_intfly)); patch_abs_rwri_data (hostdata->script, 0, dsa_check_reselect, (unsigned char)(Ent_dsa_code_check_reselect - Ent_dsa_zero)); /* These are for event logging; the ncr_event enum contains the actual interrupt numbers. */ #ifdef A_int_EVENT_SELECT patch_abs_32 (hostdata->script, 0, int_EVENT_SELECT, (u32) EVENT_SELECT); #endif #ifdef A_int_EVENT_DISCONNECT patch_abs_32 (hostdata->script, 0, int_EVENT_DISCONNECT, (u32) EVENT_DISCONNECT); #endif #ifdef A_int_EVENT_RESELECT patch_abs_32 (hostdata->script, 0, int_EVENT_RESELECT, (u32) EVENT_RESELECT); #endif #ifdef A_int_EVENT_COMPLETE patch_abs_32 (hostdata->script, 0, int_EVENT_COMPLETE, (u32) EVENT_COMPLETE); #endif #ifdef A_int_EVENT_IDLE patch_abs_32 (hostdata->script, 0, int_EVENT_IDLE, (u32) EVENT_IDLE); #endif #ifdef A_int_EVENT_SELECT_FAILED patch_abs_32 (hostdata->script, 0, int_EVENT_SELECT_FAILED, (u32) EVENT_SELECT_FAILED); #endif #ifdef A_int_EVENT_BEFORE_SELECT patch_abs_32 (hostdata->script, 0, int_EVENT_BEFORE_SELECT, (u32) EVENT_BEFORE_SELECT); #endif #ifdef A_int_EVENT_RESELECT_FAILED patch_abs_32 (hostdata->script, 0, int_EVENT_RESELECT_FAILED, (u32) EVENT_RESELECT_FAILED); #endif /* * Make sure the NCR and Linux code agree on the location of * certain fields. */ hostdata->E_accept_message = Ent_accept_message; hostdata->E_command_complete = Ent_command_complete; hostdata->E_cmdout_cmdout = Ent_cmdout_cmdout; hostdata->E_data_transfer = Ent_data_transfer; hostdata->E_debug_break = Ent_debug_break; hostdata->E_dsa_code_template = Ent_dsa_code_template; hostdata->E_dsa_code_template_end = Ent_dsa_code_template_end; hostdata->E_end_data_transfer = Ent_end_data_transfer; hostdata->E_initiator_abort = Ent_initiator_abort; hostdata->E_msg_in = Ent_msg_in; hostdata->E_other_transfer = Ent_other_transfer; hostdata->E_other_in = Ent_other_in; hostdata->E_other_out = Ent_other_out; hostdata->E_reject_message = Ent_reject_message; hostdata->E_respond_message = Ent_respond_message; hostdata->E_select = Ent_select; hostdata->E_select_msgout = Ent_select_msgout; hostdata->E_target_abort = Ent_target_abort; #ifdef Ent_test_0 hostdata->E_test_0 = Ent_test_0; #endif hostdata->E_test_1 = Ent_test_1; hostdata->E_test_2 = Ent_test_2; #ifdef Ent_test_3 hostdata->E_test_3 = Ent_test_3; #endif hostdata->E_wait_reselect = Ent_wait_reselect; hostdata->E_dsa_code_begin = Ent_dsa_code_begin; hostdata->dsa_cmdout = A_dsa_cmdout; hostdata->dsa_cmnd = A_dsa_cmnd; hostdata->dsa_datain = A_dsa_datain; hostdata->dsa_dataout = A_dsa_dataout; hostdata->dsa_end = A_dsa_end; hostdata->dsa_msgin = A_dsa_msgin; hostdata->dsa_msgout = A_dsa_msgout; hostdata->dsa_msgout_other = A_dsa_msgout_other; hostdata->dsa_next = A_dsa_next; hostdata->dsa_select = A_dsa_select; hostdata->dsa_start = Ent_dsa_code_template - Ent_dsa_zero; hostdata->dsa_status = A_dsa_status; hostdata->dsa_jump_dest = Ent_dsa_code_fix_jump - Ent_dsa_zero + 8 /* destination operand */; /* sanity check */ if (A_dsa_fields_start != Ent_dsa_code_template_end - Ent_dsa_zero) printk("scsi%d : NCR dsa_fields start is %d not %d\n", host->host_no, A_dsa_fields_start, Ent_dsa_code_template_end - Ent_dsa_zero); printk("scsi%d : NCR code relocated to 0x%lx (virt 0x%p)\n", host->host_no, virt_to_bus(hostdata->script), hostdata->script); } /* * Function : static int NCR53c7xx_run_tests (struct Scsi_Host *host) * * Purpose : run various verification tests on the NCR chip, * including interrupt generation, and proper bus mastering * operation. * * Inputs : host - a properly initialized Scsi_Host structure * * Preconditions : the NCR chip must be in a halted state. * * Returns : 0 if all tests were successful, -1 on error. * */ static int NCR53c7xx_run_tests (struct Scsi_Host *host) { NCR53c7x0_local_declare(); struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; unsigned long timeout; u32 start; int failed, i; unsigned long flags; NCR53c7x0_local_setup(host); /* The NCR chip _must_ be idle to run the test scripts */ local_irq_save(flags); if (!hostdata->idle) { printk ("scsi%d : chip not idle, aborting tests\n", host->host_no); local_irq_restore(flags); return -1; } /* * Check for functional interrupts, this could work as an * autoprobe routine. */ if ((hostdata->options & OPTION_DEBUG_TEST1) && hostdata->state != STATE_DISABLED) { hostdata->idle = 0; hostdata->test_running = 1; hostdata->test_completed = -1; hostdata->test_dest = 0; hostdata->test_source = 0xdeadbeef; start = virt_to_bus (hostdata->script) + hostdata->E_test_1; hostdata->state = STATE_RUNNING; printk ("scsi%d : test 1", host->host_no); NCR53c7x0_write32 (DSP_REG, start); if (hostdata->options & OPTION_DEBUG_TRACE) NCR53c7x0_write8 (DCNTL_REG, hostdata->saved_dcntl | DCNTL_SSM | DCNTL_STD); printk (" started\n"); local_irq_restore(flags); /* * This is currently a .5 second timeout, since (in theory) no slow * board will take that long. In practice, we've seen one * pentium which occassionally fails with this, but works with * 10 times as much? */ timeout = jiffies + 5 * HZ / 10; while ((hostdata->test_completed == -1) && time_before(jiffies, timeout)) barrier(); failed = 1; if (hostdata->test_completed == -1) printk ("scsi%d : driver test 1 timed out%s\n",host->host_no , (hostdata->test_dest == 0xdeadbeef) ? " due to lost interrupt.\n" " Please verify that the correct IRQ is being used for your board,\n" : ""); else if (hostdata->test_completed != 1) printk ("scsi%d : test 1 bad interrupt value (%d)\n", host->host_no, hostdata->test_completed); else failed = (hostdata->test_dest != 0xdeadbeef); if (hostdata->test_dest != 0xdeadbeef) { printk ("scsi%d : driver test 1 read 0x%x instead of 0xdeadbeef indicating a\n" " probable cache invalidation problem. Please configure caching\n" " as write-through or disabled\n", host->host_no, hostdata->test_dest); } if (failed) { printk ("scsi%d : DSP = 0x%p (script at 0x%p, start at 0x%x)\n", host->host_no, bus_to_virt(NCR53c7x0_read32(DSP_REG)), hostdata->script, start); printk ("scsi%d : DSPS = 0x%x\n", host->host_no, NCR53c7x0_read32(DSPS_REG)); local_irq_restore(flags); return -1; } hostdata->test_running = 0; } if ((hostdata->options & OPTION_DEBUG_TEST2) && hostdata->state != STATE_DISABLED) { u32 dsa[48]; unsigned char identify = IDENTIFY(0, 0); unsigned char cmd[6]; unsigned char data[36]; unsigned char status = 0xff; unsigned char msg = 0xff; cmd[0] = INQUIRY; cmd[1] = cmd[2] = cmd[3] = cmd[5] = 0; cmd[4] = sizeof(data); dsa[2] = 1; dsa[3] = virt_to_bus(&identify); dsa[4] = 6; dsa[5] = virt_to_bus(&cmd); dsa[6] = sizeof(data); dsa[7] = virt_to_bus(&data); dsa[8] = 1; dsa[9] = virt_to_bus(&status); dsa[10] = 1; dsa[11] = virt_to_bus(&msg); for (i = 0; i < 6; ++i) { #ifdef VALID_IDS if (!hostdata->valid_ids[i]) continue; #endif local_irq_disable(); if (!hostdata->idle) { printk ("scsi%d : chip not idle, aborting tests\n", host->host_no); local_irq_restore(flags); return -1; } /* 710: bit mapped scsi ID, async */ dsa[0] = (1 << i) << 16; hostdata->idle = 0; hostdata->test_running = 2; hostdata->test_completed = -1; start = virt_to_bus(hostdata->script) + hostdata->E_test_2; hostdata->state = STATE_RUNNING; NCR53c7x0_write32 (DSA_REG, virt_to_bus(dsa)); NCR53c7x0_write32 (DSP_REG, start); if (hostdata->options & OPTION_DEBUG_TRACE) NCR53c7x0_write8 (DCNTL_REG, hostdata->saved_dcntl | DCNTL_SSM | DCNTL_STD); local_irq_restore(flags); timeout = jiffies + 5 * HZ; /* arbitrary */ while ((hostdata->test_completed == -1) && time_before(jiffies, timeout)) barrier(); NCR53c7x0_write32 (DSA_REG, 0); if (hostdata->test_completed == 2) { data[35] = 0; printk ("scsi%d : test 2 INQUIRY to target %d, lun 0 : %s\n", host->host_no, i, data + 8); printk ("scsi%d : status ", host->host_no); scsi_print_status (status); printk ("\nscsi%d : message ", host->host_no); spi_print_msg(&msg); printk ("\n"); } else if (hostdata->test_completed == 3) { printk("scsi%d : test 2 no connection with target %d\n", host->host_no, i); if (!hostdata->idle) { printk("scsi%d : not idle\n", host->host_no); local_irq_restore(flags); return -1; } } else if (hostdata->test_completed == -1) { printk ("scsi%d : test 2 timed out\n", host->host_no); local_irq_restore(flags); return -1; } hostdata->test_running = 0; } } local_irq_restore(flags); return 0; } /* * Function : static void NCR53c7xx_dsa_fixup (struct NCR53c7x0_cmd *cmd) * * Purpose : copy the NCR53c8xx dsa structure into cmd's dsa buffer, * performing all necessary relocation. * * Inputs : cmd, a NCR53c7x0_cmd structure with a dsa area large * enough to hold the NCR53c8xx dsa. */ static void NCR53c7xx_dsa_fixup (struct NCR53c7x0_cmd *cmd) { Scsi_Cmnd *c = cmd->cmd; struct Scsi_Host *host = c->device->host; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; int i; memcpy (cmd->dsa, hostdata->script + (hostdata->E_dsa_code_template / 4), hostdata->E_dsa_code_template_end - hostdata->E_dsa_code_template); /* * Note : within the NCR 'C' code, dsa points to the _start_ * of the DSA structure, and _not_ the offset of dsa_zero within * that structure used to facilitate shorter signed offsets * for the 8 bit ALU. * * The implications of this are that * * - 32 bit A_dsa_* absolute values require an additional * dsa_zero added to their value to be correct, since they are * relative to dsa_zero which is in essentially a separate * space from the code symbols. * * - All other symbols require no special treatment. */ patch_abs_tci_data (cmd->dsa, Ent_dsa_code_template / sizeof(u32), dsa_temp_lun, c->device->lun); patch_abs_32 (cmd->dsa, Ent_dsa_code_template / sizeof(u32), dsa_temp_addr_next, virt_to_bus(&cmd->dsa_next_addr)); patch_abs_32 (cmd->dsa, Ent_dsa_code_template / sizeof(u32), dsa_temp_next, virt_to_bus(cmd->dsa) + Ent_dsa_zero - Ent_dsa_code_template + A_dsa_next); patch_abs_32 (cmd->dsa, Ent_dsa_code_template / sizeof(u32), dsa_temp_sync, virt_to_bus((void *)hostdata->sync[c->device->id].script)); patch_abs_32 (cmd->dsa, Ent_dsa_code_template / sizeof(u32), dsa_sscf_710, virt_to_bus((void *)&hostdata->sync[c->device->id].sscf_710)); patch_abs_tci_data (cmd->dsa, Ent_dsa_code_template / sizeof(u32), dsa_temp_target, 1 << c->device->id); /* XXX - new pointer stuff */ patch_abs_32 (cmd->dsa, Ent_dsa_code_template / sizeof(u32), dsa_temp_addr_saved_pointer, virt_to_bus(&cmd->saved_data_pointer)); patch_abs_32 (cmd->dsa, Ent_dsa_code_template / sizeof(u32), dsa_temp_addr_saved_residual, virt_to_bus(&cmd->saved_residual)); patch_abs_32 (cmd->dsa, Ent_dsa_code_template / sizeof(u32), dsa_temp_addr_residual, virt_to_bus(&cmd->residual)); /* XXX - new start stuff */ patch_abs_32 (cmd->dsa, Ent_dsa_code_template / sizeof(u32), dsa_temp_addr_dsa_value, virt_to_bus(&cmd->dsa_addr)); } /* * Function : run_process_issue_queue (void) * * Purpose : insure that the coroutine is running and will process our * request. process_issue_queue_running is checked/set here (in an * inline function) rather than in process_issue_queue itself to reduce * the chances of stack overflow. * */ static volatile int process_issue_queue_running = 0; static __inline__ void run_process_issue_queue(void) { unsigned long flags; local_irq_save(flags); if (!process_issue_queue_running) { process_issue_queue_running = 1; process_issue_queue(flags); /* * process_issue_queue_running is cleared in process_issue_queue * once it can't do more work, and process_issue_queue exits with * interrupts disabled. */ } local_irq_restore(flags); } /* * Function : static void abnormal_finished (struct NCR53c7x0_cmd *cmd, int * result) * * Purpose : mark SCSI command as finished, OR'ing the host portion * of the result word into the result field of the corresponding * Scsi_Cmnd structure, and removing it from the internal queues. * * Inputs : cmd - command, result - entire result field * * Preconditions : the NCR chip should be in a halted state when * abnormal_finished is run, since it modifies structures which * the NCR expects to have exclusive access to. */ static void abnormal_finished (struct NCR53c7x0_cmd *cmd, int result) { Scsi_Cmnd *c = cmd->cmd; struct Scsi_Host *host = c->device->host; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; unsigned long flags; int left, found; volatile struct NCR53c7x0_cmd * linux_search; volatile struct NCR53c7x0_cmd * volatile *linux_prev; volatile u32 *ncr_prev, *ncrcurrent, ncr_search; #if 0 printk ("scsi%d: abnormal finished\n", host->host_no); #endif local_irq_save(flags); found = 0; /* * Traverse the NCR issue array until we find a match or run out * of instructions. Instructions in the NCR issue array are * either JUMP or NOP instructions, which are 2 words in length. */ for (found = 0, left = host->can_queue, ncrcurrent = hostdata->schedule; left > 0; --left, ncrcurrent += 2) { if (issue_to_cmd (host, hostdata, (u32 *) ncrcurrent) == cmd) { ncrcurrent[0] = hostdata->NOP_insn; ncrcurrent[1] = 0xdeadbeef; ++found; break; } } /* * Traverse the NCR reconnect list of DSA structures until we find * a pointer to this dsa or have found too many command structures. * We let prev point at the next field of the previous element or * head of the list, so we don't do anything different for removing * the head element. */ for (left = host->can_queue, ncr_search = hostdata->reconnect_dsa_head, ncr_prev = &hostdata->reconnect_dsa_head; left >= 0 && ncr_search && ((char*)bus_to_virt(ncr_search) + hostdata->dsa_start) != (char *) cmd->dsa; ncr_prev = (u32*) ((char*)bus_to_virt(ncr_search) + hostdata->dsa_next), ncr_search = *ncr_prev, --left); if (left < 0) printk("scsi%d: loop detected in ncr reconncect list\n", host->host_no); else if (ncr_search) { if (found) printk("scsi%d: scsi %ld in ncr issue array and reconnect lists\n", host->host_no, c->pid); else { volatile u32 * next = (u32 *) ((char *)bus_to_virt(ncr_search) + hostdata->dsa_next); *ncr_prev = *next; /* If we're at the tail end of the issue queue, update that pointer too. */ found = 1; } } /* * Traverse the host running list until we find this command or discover * we have too many elements, pointing linux_prev at the next field of the * linux_previous element or head of the list, search at this element. */ for (left = host->can_queue, linux_search = hostdata->running_list, linux_prev = &hostdata->running_list; left >= 0 && linux_search && linux_search != cmd; linux_prev = &(linux_search->next), linux_search = linux_search->next, --left); if (left < 0) printk ("scsi%d: loop detected in host running list for scsi pid %ld\n", host->host_no, c->pid); else if (linux_search) { *linux_prev = linux_search->next; --hostdata->busy[c->device->id][c->device->lun]; } /* Return the NCR command structure to the free list */ cmd->next = hostdata->free; hostdata->free = cmd; c->host_scribble = NULL; /* And return */ c->result = result; c->scsi_done(c); local_irq_restore(flags); run_process_issue_queue(); } /* * Function : static void intr_break (struct Scsi_Host *host, * struct NCR53c7x0_cmd *cmd) * * Purpose : Handler for breakpoint interrupts from a SCSI script * * Inputs : host - pointer to this host adapter's structure, * cmd - pointer to the command (if any) dsa was pointing * to. * */ static void intr_break (struct Scsi_Host *host, struct NCR53c7x0_cmd *cmd) { NCR53c7x0_local_declare(); struct NCR53c7x0_break *bp; #if 0 Scsi_Cmnd *c = cmd ? cmd->cmd : NULL; #endif u32 *dsp; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; unsigned long flags; NCR53c7x0_local_setup(host); /* * Find the break point corresponding to this address, and * dump the appropriate debugging information to standard * output. */ local_irq_save(flags); dsp = (u32 *) bus_to_virt(NCR53c7x0_read32(DSP_REG)); for (bp = hostdata->breakpoints; bp && bp->address != dsp; bp = bp->next); if (!bp) panic("scsi%d : break point interrupt from %p with no breakpoint!", host->host_no, dsp); /* * Configure the NCR chip for manual start mode, so that we can * point the DSP register at the instruction that follows the * INT int_debug_break instruction. */ NCR53c7x0_write8 (hostdata->dmode, NCR53c7x0_read8(hostdata->dmode)|DMODE_MAN); /* * And update the DSP register, using the size of the old * instruction in bytes. */ local_irq_restore(flags); } /* * Function : static void print_synchronous (const char *prefix, * const unsigned char *msg) * * Purpose : print a pretty, user and machine parsable representation * of a SDTR message, including the "real" parameters, data * clock so we can tell transfer rate at a glance. * * Inputs ; prefix - text to prepend, msg - SDTR message (5 bytes) */ static void print_synchronous (const char *prefix, const unsigned char *msg) { if (msg[4]) { int Hz = 1000000000 / (msg[3] * 4); int integer = Hz / 1000000; int fraction = (Hz - (integer * 1000000)) / 10000; printk ("%speriod %dns offset %d %d.%02dMHz %s SCSI%s\n", prefix, (int) msg[3] * 4, (int) msg[4], integer, fraction, (((msg[3] * 4) < 200) ? "FAST" : "synchronous"), (((msg[3] * 4) < 200) ? "-II" : "")); } else printk ("%sasynchronous SCSI\n", prefix); } /* * Function : static void set_synchronous (struct Scsi_Host *host, * int target, int sxfer, int scntl3, int now_connected) * * Purpose : reprogram transfers between the selected SCSI initiator and * target with the given register values; in the indirect * select operand, reselection script, and chip registers. * * Inputs : host - NCR53c7,8xx SCSI host, target - number SCSI target id, * sxfer and scntl3 - NCR registers. now_connected - if non-zero, * we should reprogram the registers now too. * * NOTE: For 53c710, scntl3 is actually used for SCF bits from * SBCL, as we don't have a SCNTL3. */ static void set_synchronous (struct Scsi_Host *host, int target, int sxfer, int scntl3, int now_connected) { NCR53c7x0_local_declare(); struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; u32 *script; NCR53c7x0_local_setup(host); /* These are eight bit registers */ sxfer &= 0xff; scntl3 &= 0xff; hostdata->sync[target].sxfer_sanity = sxfer; hostdata->sync[target].scntl3_sanity = scntl3; /* * HARD CODED : synchronous script is EIGHT words long. This * must agree with 53c7.8xx.h */ if ((hostdata->chip != 700) && (hostdata->chip != 70066)) { hostdata->sync[target].select_indirect = (1 << target) << 16 | (sxfer << 8); hostdata->sync[target].sscf_710 = scntl3; script = (u32 *) hostdata->sync[target].script; /* XXX - add NCR53c7x0 code to reprogram SCF bits if we want to */ script[0] = ((DCMD_TYPE_RWRI | DCMD_RWRI_OPC_MODIFY | DCMD_RWRI_OP_MOVE) << 24) | (SBCL_REG << 16) | (scntl3 << 8); script[1] = 0; script += 2; script[0] = ((DCMD_TYPE_RWRI | DCMD_RWRI_OPC_MODIFY | DCMD_RWRI_OP_MOVE) << 24) | (SXFER_REG << 16) | (sxfer << 8); script[1] = 0; script += 2; #ifdef DEBUG_SYNC_INTR if (hostdata->options & OPTION_DEBUG_DISCONNECT) { script[0] = ((DCMD_TYPE_TCI|DCMD_TCI_OP_INT) << 24) | DBC_TCI_TRUE; script[1] = DEBUG_SYNC_INTR; script += 2; } #endif script[0] = ((DCMD_TYPE_TCI|DCMD_TCI_OP_RETURN) << 24) | DBC_TCI_TRUE; script[1] = 0; script += 2; } if (hostdata->options & OPTION_DEBUG_SYNCHRONOUS) printk ("scsi%d : target %d sync parameters are sxfer=0x%x, scntl3=0x%x\n", host->host_no, target, sxfer, scntl3); if (now_connected) { NCR53c7x0_write8(SBCL_REG, scntl3); NCR53c7x0_write8(SXFER_REG, sxfer); } } /* * Function : static int asynchronous (struct Scsi_Host *host, int target) * * Purpose : reprogram between the selected SCSI Host adapter and target * (assumed to be currently connected) for asynchronous transfers. * * Inputs : host - SCSI host structure, target - numeric target ID. * * Preconditions : the NCR chip should be in one of the halted states */ static void asynchronous (struct Scsi_Host *host, int target) { NCR53c7x0_local_declare(); struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; NCR53c7x0_local_setup(host); set_synchronous (host, target, /* no offset */ 0, hostdata->saved_scntl3, 1); printk ("scsi%d : setting target %d to asynchronous SCSI\n", host->host_no, target); } /* * XXX - do we want to go out of our way (ie, add extra code to selection * in the NCR53c710/NCR53c720 script) to reprogram the synchronous * conversion bits, or can we be content in just setting the * sxfer bits? I chose to do so [richard@sleepie.demon.co.uk] */ /* Table for NCR53c8xx synchronous values */ /* This table is also correct for 710, allowing that scf=4 is equivalent * of SSCF=0 (ie use DCNTL, divide by 3) for a 50.01-66.00MHz clock. * For any other clock values, we cannot use entries with SCF values of * 4. I guess that for a 66MHz clock, the slowest it will set is 2MHz, * and for a 50MHz clock, the slowest will be 2.27Mhz. Should check * that a device doesn't try and negotiate sync below these limits! */ static const struct { int div; /* Total clock divisor * 10 */ unsigned char scf; /* */ unsigned char tp; /* 4 + tp = xferp divisor */ } syncs[] = { /* div scf tp div scf tp div scf tp */ { 40, 1, 0}, { 50, 1, 1}, { 60, 1, 2}, { 70, 1, 3}, { 75, 2, 1}, { 80, 1, 4}, { 90, 1, 5}, { 100, 1, 6}, { 105, 2, 3}, { 110, 1, 7}, { 120, 2, 4}, { 135, 2, 5}, { 140, 3, 3}, { 150, 2, 6}, { 160, 3, 4}, { 165, 2, 7}, { 180, 3, 5}, { 200, 3, 6}, { 210, 4, 3}, { 220, 3, 7}, { 240, 4, 4}, { 270, 4, 5}, { 300, 4, 6}, { 330, 4, 7} }; /* * Function : static void synchronous (struct Scsi_Host *host, int target, * char *msg) * * Purpose : reprogram transfers between the selected SCSI initiator and * target for synchronous SCSI transfers such that the synchronous * offset is less than that requested and period at least as long * as that requested. Also modify *msg such that it contains * an appropriate response. * * Inputs : host - NCR53c7,8xx SCSI host, target - number SCSI target id, * msg - synchronous transfer request. */ static void synchronous (struct Scsi_Host *host, int target, char *msg) { struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; int desire, divisor, i, limit; unsigned char scntl3, sxfer; /* The diagnostic message fits on one line, even with max. width integers */ char buf[80]; /* Desired transfer clock in Hz */ desire = 1000000000L / (msg[3] * 4); /* Scale the available SCSI clock by 10 so we get tenths */ divisor = (hostdata->scsi_clock * 10) / desire; /* NCR chips can handle at most an offset of 8 */ if (msg[4] > 8) msg[4] = 8; if (hostdata->options & OPTION_DEBUG_SDTR) printk("scsi%d : optimal synchronous divisor of %d.%01d\n", host->host_no, divisor / 10, divisor % 10); limit = (sizeof(syncs) / sizeof(syncs[0]) -1); for (i = 0; (i < limit) && (divisor > syncs[i].div); ++i); if (hostdata->options & OPTION_DEBUG_SDTR) printk("scsi%d : selected synchronous divisor of %d.%01d\n", host->host_no, syncs[i].div / 10, syncs[i].div % 10); msg[3] = ((1000000000L / hostdata->scsi_clock) * syncs[i].div / 10 / 4); if (hostdata->options & OPTION_DEBUG_SDTR) printk("scsi%d : selected synchronous period of %dns\n", host->host_no, msg[3] * 4); scntl3 = syncs[i].scf; sxfer = (msg[4] << SXFER_MO_SHIFT) | (syncs[i].tp << 4); if (hostdata->options & OPTION_DEBUG_SDTR) printk ("scsi%d : sxfer=0x%x scntl3=0x%x\n", host->host_no, (int) sxfer, (int) scntl3); set_synchronous (host, target, sxfer, scntl3, 1); sprintf (buf, "scsi%d : setting target %d to ", host->host_no, target); print_synchronous (buf, msg); } /* * Function : static int NCR53c7x0_dstat_sir_intr (struct Scsi_Host *host, * struct NCR53c7x0_cmd *cmd) * * Purpose : Handler for INT generated instructions for the * NCR53c810/820 SCSI SCRIPT * * Inputs : host - pointer to this host adapter's structure, * cmd - pointer to the command (if any) dsa was pointing * to. * */ static int NCR53c7x0_dstat_sir_intr (struct Scsi_Host *host, struct NCR53c7x0_cmd *cmd) { NCR53c7x0_local_declare(); int print; Scsi_Cmnd *c = cmd ? cmd->cmd : NULL; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; u32 dsps,*dsp; /* Argument of the INT instruction */ NCR53c7x0_local_setup(host); dsps = NCR53c7x0_read32(DSPS_REG); dsp = (u32 *) bus_to_virt(NCR53c7x0_read32(DSP_REG)); /* RGH 150597: Frig. Commands which fail with Check Condition are * Flagged as successful - hack dsps to indicate check condition */ #if 0 /* RGH 200597: Need to disable for BVME6000, as it gets Check Conditions * and then dies. Seems to handle Check Condition at startup, but * not mid kernel build. */ if (dsps == A_int_norm_emulateintfly && cmd && cmd->result == 2) dsps = A_int_err_check_condition; #endif if (hostdata->options & OPTION_DEBUG_INTR) printk ("scsi%d : DSPS = 0x%x\n", host->host_no, dsps); switch (dsps) { case A_int_msg_1: print = 1; switch (hostdata->msg_buf[0]) { /* * Unless we've initiated synchronous negotiation, I don't * think that this should happen. */ case MESSAGE_REJECT: hostdata->dsp = hostdata->script + hostdata->E_accept_message / sizeof(u32); hostdata->dsp_changed = 1; if (cmd && (cmd->flags & CMD_FLAG_SDTR)) { printk ("scsi%d : target %d rejected SDTR\n", host->host_no, c->device->id); cmd->flags &= ~CMD_FLAG_SDTR; asynchronous (host, c->device->id); print = 0; } break; case INITIATE_RECOVERY: printk ("scsi%d : extended contingent allegiance not supported yet, rejecting\n", host->host_no); /* Fall through to default */ hostdata->dsp = hostdata->script + hostdata->E_reject_message / sizeof(u32); hostdata->dsp_changed = 1; break; default: printk ("scsi%d : unsupported message, rejecting\n", host->host_no); hostdata->dsp = hostdata->script + hostdata->E_reject_message / sizeof(u32); hostdata->dsp_changed = 1; } if (print) { printk ("scsi%d : received message", host->host_no); if (c) printk (" from target %d lun %d ", c->device->id, c->device->lun); spi_print_msg((unsigned char *) hostdata->msg_buf); printk("\n"); } return SPECIFIC_INT_NOTHING; case A_int_msg_sdtr: /* * At this point, hostdata->msg_buf contains * 0 EXTENDED MESSAGE * 1 length * 2 SDTR * 3 period * 4ns * 4 offset */ if (cmd) { char buf[80]; sprintf (buf, "scsi%d : target %d %s ", host->host_no, c->device->id, (cmd->flags & CMD_FLAG_SDTR) ? "accepting" : "requesting"); print_synchronous (buf, (unsigned char *) hostdata->msg_buf); /* * Initiator initiated, won't happen unless synchronous * transfers are enabled. If we get a SDTR message in * response to our SDTR, we should program our parameters * such that * offset <= requested offset * period >= requested period */ if (cmd->flags & CMD_FLAG_SDTR) { cmd->flags &= ~CMD_FLAG_SDTR; if (hostdata->msg_buf[4]) synchronous (host, c->device->id, (unsigned char *) hostdata->msg_buf); else asynchronous (host, c->device->id); hostdata->dsp = hostdata->script + hostdata->E_accept_message / sizeof(u32); hostdata->dsp_changed = 1; return SPECIFIC_INT_NOTHING; } else { if (hostdata->options & OPTION_SYNCHRONOUS) { cmd->flags |= CMD_FLAG_DID_SDTR; synchronous (host, c->device->id, (unsigned char *) hostdata->msg_buf); } else { hostdata->msg_buf[4] = 0; /* 0 offset = async */ asynchronous (host, c->device->id); } patch_dsa_32 (cmd->dsa, dsa_msgout_other, 0, 5); patch_dsa_32 (cmd->dsa, dsa_msgout_other, 1, (u32) virt_to_bus ((void *)&hostdata->msg_buf)); hostdata->dsp = hostdata->script + hostdata->E_respond_message / sizeof(u32); hostdata->dsp_changed = 1; } return SPECIFIC_INT_NOTHING; } /* Fall through to abort if we couldn't find a cmd, and therefore a dsa structure to twiddle */ case A_int_msg_wdtr: hostdata->dsp = hostdata->script + hostdata->E_reject_message / sizeof(u32); hostdata->dsp_changed = 1; return SPECIFIC_INT_NOTHING; case A_int_err_unexpected_phase: if (hostdata->options & OPTION_DEBUG_INTR) printk ("scsi%d : unexpected phase\n", host->host_no); return SPECIFIC_INT_ABORT; case A_int_err_selected: if ((hostdata->chip / 100) == 8) printk ("scsi%d : selected by target %d\n", host->host_no, (int) NCR53c7x0_read8(SDID_REG_800) &7); else printk ("scsi%d : selected by target LCRC=0x%02x\n", host->host_no, (int) NCR53c7x0_read8(LCRC_REG_10)); hostdata->dsp = hostdata->script + hostdata->E_target_abort / sizeof(u32); hostdata->dsp_changed = 1; return SPECIFIC_INT_NOTHING; case A_int_err_unexpected_reselect: if ((hostdata->chip / 100) == 8) printk ("scsi%d : unexpected reselect by target %d lun %d\n", host->host_no, (int) NCR53c7x0_read8(SDID_REG_800) & 7, hostdata->reselected_identify & 7); else printk ("scsi%d : unexpected reselect LCRC=0x%02x\n", host->host_no, (int) NCR53c7x0_read8(LCRC_REG_10)); hostdata->dsp = hostdata->script + hostdata->E_initiator_abort / sizeof(u32); hostdata->dsp_changed = 1; return SPECIFIC_INT_NOTHING; /* * Since contingent allegiance conditions are cleared by the next * command issued to a target, we must issue a REQUEST SENSE * command after receiving a CHECK CONDITION status, before * another command is issued. * * Since this NCR53c7x0_cmd will be freed after use, we don't * care if we step on the various fields, so modify a few things. */ case A_int_err_check_condition: #if 0 if (hostdata->options & OPTION_DEBUG_INTR) #endif printk ("scsi%d : CHECK CONDITION\n", host->host_no); if (!c) { printk("scsi%d : CHECK CONDITION with no SCSI command\n", host->host_no); return SPECIFIC_INT_PANIC; } /* * FIXME : this uses the normal one-byte selection message. * We may want to renegotiate for synchronous & WIDE transfers * since these could be the crux of our problem. * hostdata->NOP_insn* FIXME : once SCSI-II tagged queuing is implemented, we'll * have to set this up so that the rest of the DSA * agrees with this being an untagged queue'd command. */ patch_dsa_32 (cmd->dsa, dsa_msgout, 0, 1); /* * Modify the table indirect for COMMAND OUT phase, since * Request Sense is a six byte command. */ patch_dsa_32 (cmd->dsa, dsa_cmdout, 0, 6); /* * The CDB is now mirrored in our local non-cached * structure, but keep the old structure up to date as well, * just in case anyone looks at it. */ /* * XXX Need to worry about data buffer alignment/cache state * XXX here, but currently never get A_int_err_check_condition, * XXX so ignore problem for now. */ cmd->cmnd[0] = c->cmnd[0] = REQUEST_SENSE; cmd->cmnd[0] = c->cmnd[1] &= 0xe0; /* Zero all but LUN */ cmd->cmnd[0] = c->cmnd[2] = 0; cmd->cmnd[0] = c->cmnd[3] = 0; cmd->cmnd[0] = c->cmnd[4] = sizeof(c->sense_buffer); cmd->cmnd[0] = c->cmnd[5] = 0; /* * Disable dataout phase, and program datain to transfer to the * sense buffer, and add a jump to other_transfer after the * command so overflow/underrun conditions are detected. */ patch_dsa_32 (cmd->dsa, dsa_dataout, 0, virt_to_bus(hostdata->script) + hostdata->E_other_transfer); patch_dsa_32 (cmd->dsa, dsa_datain, 0, virt_to_bus(cmd->data_transfer_start)); cmd->data_transfer_start[0] = (((DCMD_TYPE_BMI | DCMD_BMI_OP_MOVE_I | DCMD_BMI_IO)) << 24) | sizeof(c->sense_buffer); cmd->data_transfer_start[1] = (u32) virt_to_bus(c->sense_buffer); cmd->data_transfer_start[2] = ((DCMD_TYPE_TCI | DCMD_TCI_OP_JUMP) << 24) | DBC_TCI_TRUE; cmd->data_transfer_start[3] = (u32) virt_to_bus(hostdata->script) + hostdata->E_other_transfer; /* * Currently, this command is flagged as completed, ie * it has valid status and message data. Reflag it as * incomplete. Q - need to do something so that original * status, etc are used. */ cmd->result = cmd->cmd->result = 0xffff; /* * Restart command as a REQUEST SENSE. */ hostdata->dsp = (u32 *) hostdata->script + hostdata->E_select / sizeof(u32); hostdata->dsp_changed = 1; return SPECIFIC_INT_NOTHING; case A_int_debug_break: return SPECIFIC_INT_BREAK; case A_int_norm_aborted: hostdata->dsp = (u32 *) hostdata->schedule; hostdata->dsp_changed = 1; if (cmd) abnormal_finished (cmd, DID_ERROR << 16); return SPECIFIC_INT_NOTHING; case A_int_norm_emulateintfly: NCR53c7x0_intfly(host); return SPECIFIC_INT_NOTHING; case A_int_test_1: case A_int_test_2: hostdata->idle = 1; hostdata->test_completed = (dsps - A_int_test_1) / 0x00010000 + 1; if (hostdata->options & OPTION_DEBUG_INTR) printk("scsi%d : test%d complete\n", host->host_no, hostdata->test_completed); return SPECIFIC_INT_NOTHING; #ifdef A_int_debug_reselected_ok case A_int_debug_reselected_ok: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR| OPTION_DEBUG_DISCONNECT)) { /* * Note - this dsa is not based on location relative to * the command structure, but to location relative to the * DSA register */ u32 *dsa; dsa = (u32 *) bus_to_virt (NCR53c7x0_read32(DSA_REG)); printk("scsi%d : reselected_ok (DSA = 0x%x (virt 0x%p)\n", host->host_no, NCR53c7x0_read32(DSA_REG), dsa); printk("scsi%d : resume address is 0x%x (virt 0x%p)\n", host->host_no, cmd->saved_data_pointer, bus_to_virt(cmd->saved_data_pointer)); print_insn (host, hostdata->script + Ent_reselected_ok / sizeof(u32), "", 1); if ((hostdata->chip / 100) == 8) printk ("scsi%d : sxfer=0x%x, scntl3=0x%x\n", host->host_no, NCR53c7x0_read8(SXFER_REG), NCR53c7x0_read8(SCNTL3_REG_800)); else printk ("scsi%d : sxfer=0x%x, cannot read SBCL\n", host->host_no, NCR53c7x0_read8(SXFER_REG)); if (c) { print_insn (host, (u32 *) hostdata->sync[c->device->id].script, "", 1); print_insn (host, (u32 *) hostdata->sync[c->device->id].script + 2, "", 1); } } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_reselect_check case A_int_debug_reselect_check: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR)) { u32 *dsa; #if 0 u32 *code; #endif /* * Note - this dsa is not based on location relative to * the command structure, but to location relative to the * DSA register */ dsa = bus_to_virt (NCR53c7x0_read32(DSA_REG)); printk("scsi%d : reselected_check_next (DSA = 0x%lx (virt 0x%p))\n", host->host_no, virt_to_bus(dsa), dsa); if (dsa) { printk("scsi%d : resume address is 0x%x (virt 0x%p)\n", host->host_no, cmd->saved_data_pointer, bus_to_virt (cmd->saved_data_pointer)); #if 0 printk("scsi%d : template code :\n", host->host_no); for (code = dsa + (Ent_dsa_code_check_reselect - Ent_dsa_zero) / sizeof(u32); code < (dsa + Ent_dsa_zero / sizeof(u32)); code += print_insn (host, code, "", 1)); #endif } print_insn (host, hostdata->script + Ent_reselected_ok / sizeof(u32), "", 1); } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_dsa_schedule case A_int_debug_dsa_schedule: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR)) { u32 *dsa; /* * Note - this dsa is not based on location relative to * the command structure, but to location relative to the * DSA register */ dsa = (u32 *) bus_to_virt (NCR53c7x0_read32(DSA_REG)); printk("scsi%d : dsa_schedule (old DSA = 0x%lx (virt 0x%p))\n", host->host_no, virt_to_bus(dsa), dsa); if (dsa) printk("scsi%d : resume address is 0x%x (virt 0x%p)\n" " (temp was 0x%x (virt 0x%p))\n", host->host_no, cmd->saved_data_pointer, bus_to_virt (cmd->saved_data_pointer), NCR53c7x0_read32 (TEMP_REG), bus_to_virt (NCR53c7x0_read32(TEMP_REG))); } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_scheduled case A_int_debug_scheduled: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR)) { printk("scsi%d : new I/O 0x%x (virt 0x%p) scheduled\n", host->host_no, NCR53c7x0_read32(DSA_REG), bus_to_virt(NCR53c7x0_read32(DSA_REG))); } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_idle case A_int_debug_idle: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR)) { printk("scsi%d : idle\n", host->host_no); } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_cmd case A_int_debug_cmd: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR)) { printk("scsi%d : command sent\n"); } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_dsa_loaded case A_int_debug_dsa_loaded: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR)) { printk("scsi%d : DSA loaded with 0x%x (virt 0x%p)\n", host->host_no, NCR53c7x0_read32(DSA_REG), bus_to_virt(NCR53c7x0_read32(DSA_REG))); } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_reselected case A_int_debug_reselected: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR| OPTION_DEBUG_DISCONNECT)) { if ((hostdata->chip / 100) == 8) printk("scsi%d : reselected by target %d lun %d\n", host->host_no, (int) NCR53c7x0_read8(SDID_REG_800) & ~0x80, (int) hostdata->reselected_identify & 7); else printk("scsi%d : reselected by LCRC=0x%02x lun %d\n", host->host_no, (int) NCR53c7x0_read8(LCRC_REG_10), (int) hostdata->reselected_identify & 7); print_queues(host); } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_disconnect_msg case A_int_debug_disconnect_msg: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR)) { if (c) printk("scsi%d : target %d lun %d disconnecting\n", host->host_no, c->device->id, c->device->lun); else printk("scsi%d : unknown target disconnecting\n", host->host_no); } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_disconnected case A_int_debug_disconnected: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR| OPTION_DEBUG_DISCONNECT)) { printk ("scsi%d : disconnected, new queues are\n", host->host_no); print_queues(host); #if 0 /* Not valid on ncr53c710! */ printk ("scsi%d : sxfer=0x%x, scntl3=0x%x\n", host->host_no, NCR53c7x0_read8(SXFER_REG), NCR53c7x0_read8(SCNTL3_REG_800)); #endif if (c) { print_insn (host, (u32 *) hostdata->sync[c->device->id].script, "", 1); print_insn (host, (u32 *) hostdata->sync[c->device->id].script + 2, "", 1); } } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_panic case A_int_debug_panic: printk("scsi%d : int_debug_panic received\n", host->host_no); print_lots (host); return SPECIFIC_INT_PANIC; #endif #ifdef A_int_debug_saved case A_int_debug_saved: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR| OPTION_DEBUG_DISCONNECT)) { printk ("scsi%d : saved data pointer 0x%x (virt 0x%p)\n", host->host_no, cmd->saved_data_pointer, bus_to_virt (cmd->saved_data_pointer)); print_progress (c); } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_restored case A_int_debug_restored: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR| OPTION_DEBUG_DISCONNECT)) { if (cmd) { int size; printk ("scsi%d : restored data pointer 0x%x (virt 0x%p)\n", host->host_no, cmd->saved_data_pointer, bus_to_virt ( cmd->saved_data_pointer)); size = print_insn (host, (u32 *) bus_to_virt(cmd->saved_data_pointer), "", 1); size = print_insn (host, (u32 *) bus_to_virt(cmd->saved_data_pointer) + size, "", 1); print_progress (c); } #if 0 printk ("scsi%d : datapath residual %d\n", host->host_no, datapath_residual (host)) ; #endif } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_sync case A_int_debug_sync: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR| OPTION_DEBUG_DISCONNECT|OPTION_DEBUG_SDTR)) { unsigned char sxfer = NCR53c7x0_read8 (SXFER_REG), scntl3; if ((hostdata->chip / 100) == 8) { scntl3 = NCR53c7x0_read8 (SCNTL3_REG_800); if (c) { if (sxfer != hostdata->sync[c->device->id].sxfer_sanity || scntl3 != hostdata->sync[c->device->id].scntl3_sanity) { printk ("scsi%d : sync sanity check failed sxfer=0x%x, scntl3=0x%x", host->host_no, sxfer, scntl3); NCR53c7x0_write8 (SXFER_REG, sxfer); NCR53c7x0_write8 (SCNTL3_REG_800, scntl3); } } else printk ("scsi%d : unknown command sxfer=0x%x, scntl3=0x%x\n", host->host_no, (int) sxfer, (int) scntl3); } else { if (c) { if (sxfer != hostdata->sync[c->device->id].sxfer_sanity) { printk ("scsi%d : sync sanity check failed sxfer=0x%x", host->host_no, sxfer); NCR53c7x0_write8 (SXFER_REG, sxfer); NCR53c7x0_write8 (SBCL_REG, hostdata->sync[c->device->id].sscf_710); } } else printk ("scsi%d : unknown command sxfer=0x%x\n", host->host_no, (int) sxfer); } } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_datain case A_int_debug_datain: if (hostdata->options & (OPTION_DEBUG_SCRIPT|OPTION_DEBUG_INTR| OPTION_DEBUG_DISCONNECT|OPTION_DEBUG_SDTR)) { int size; if ((hostdata->chip / 100) == 8) printk ("scsi%d : In do_datain (%s) sxfer=0x%x, scntl3=0x%x\n" " datapath residual=%d\n", host->host_no, sbcl_to_phase (NCR53c7x0_read8 (SBCL_REG)), (int) NCR53c7x0_read8(SXFER_REG), (int) NCR53c7x0_read8(SCNTL3_REG_800), datapath_residual (host)) ; else printk ("scsi%d : In do_datain (%s) sxfer=0x%x\n" " datapath residual=%d\n", host->host_no, sbcl_to_phase (NCR53c7x0_read8 (SBCL_REG)), (int) NCR53c7x0_read8(SXFER_REG), datapath_residual (host)) ; print_insn (host, dsp, "", 1); size = print_insn (host, (u32 *) bus_to_virt(dsp[1]), "", 1); print_insn (host, (u32 *) bus_to_virt(dsp[1]) + size, "", 1); } return SPECIFIC_INT_RESTART; #endif #ifdef A_int_debug_check_dsa case A_int_debug_check_dsa: if (NCR53c7x0_read8 (SCNTL1_REG) & SCNTL1_CON) { int sdid; int tmp; char *where; if (hostdata->chip / 100 == 8) sdid = NCR53c7x0_read8 (SDID_REG_800) & 15; else { tmp = NCR53c7x0_read8 (SDID_REG_700); if (!tmp) panic ("SDID_REG_700 = 0"); tmp >>= 1; sdid = 0; while (tmp) { tmp >>= 1; sdid++; } } where = dsp - NCR53c7x0_insn_size(NCR53c7x0_read8 (DCMD_REG)) == hostdata->script + Ent_select_check_dsa / sizeof(u32) ? "selection" : "reselection"; if (c && sdid != c->device->id) { printk ("scsi%d : SDID target %d != DSA target %d at %s\n", host->host_no, sdid, c->device->id, where); print_lots(host); dump_events (host, 20); return SPECIFIC_INT_PANIC; } } return SPECIFIC_INT_RESTART; #endif default: if ((dsps & 0xff000000) == 0x03000000) { printk ("scsi%d : misc debug interrupt 0x%x\n", host->host_no, dsps); return SPECIFIC_INT_RESTART; } else if ((dsps & 0xff000000) == 0x05000000) { if (hostdata->events) { struct NCR53c7x0_event *event; ++hostdata->event_index; if (hostdata->event_index >= hostdata->event_size) hostdata->event_index = 0; event = (struct NCR53c7x0_event *) hostdata->events + hostdata->event_index; event->event = (enum ncr_event) dsps; event->dsa = bus_to_virt(NCR53c7x0_read32(DSA_REG)); if (NCR53c7x0_read8 (SCNTL1_REG) & SCNTL1_CON) { if (hostdata->chip / 100 == 8) event->target = NCR53c7x0_read8(SSID_REG_800); else { unsigned char tmp, sdid; tmp = NCR53c7x0_read8 (SDID_REG_700); if (!tmp) panic ("SDID_REG_700 = 0"); tmp >>= 1; sdid = 0; while (tmp) { tmp >>= 1; sdid++; } event->target = sdid; } } else event->target = 255; if (event->event == EVENT_RESELECT) event->lun = hostdata->reselected_identify & 0xf; else if (c) event->lun = c->device->lun; else event->lun = 255; do_gettimeofday(&(event->time)); if (c) { event->pid = c->pid; memcpy ((void *) event->cmnd, (void *) c->cmnd, sizeof (event->cmnd)); } else { event->pid = -1; } } return SPECIFIC_INT_RESTART; } printk ("scsi%d : unknown user interrupt 0x%x\n", host->host_no, (unsigned) dsps); return SPECIFIC_INT_PANIC; } } /* * XXX - the stock NCR assembler won't output the scriptu.h file, * which undefine's all #define'd CPP symbols from the script.h * file, which will create problems if you use multiple scripts * with the same symbol names. * * If you insist on using NCR's assembler, you could generate * scriptu.h from script.h using something like * * grep #define script.h | \ * sed 's/#define[ ][ ]*\([_a-zA-Z][_a-zA-Z0-9]*\).*$/#undefine \1/' \ * > scriptu.h */ #include "53c7xx_u.h" /* XXX - add alternate script handling code here */ /* * Function : static void NCR537xx_soft_reset (struct Scsi_Host *host) * * Purpose : perform a soft reset of the NCR53c7xx chip * * Inputs : host - pointer to this host adapter's structure * * Preconditions : NCR53c7x0_init must have been called for this * host. * */ static void NCR53c7x0_soft_reset (struct Scsi_Host *host) { NCR53c7x0_local_declare(); unsigned long flags; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; NCR53c7x0_local_setup(host); local_irq_save(flags); /* Disable scsi chip and s/w level 7 ints */ #ifdef CONFIG_MVME16x if (MACH_IS_MVME16x) { volatile unsigned long v; v = *(volatile unsigned long *)0xfff4006c; v &= ~0x8000; *(volatile unsigned long *)0xfff4006c = v; v = *(volatile unsigned long *)0xfff4202c; v &= ~0x10; *(volatile unsigned long *)0xfff4202c = v; } #endif /* Anything specific for your hardware? */ /* * Do a soft reset of the chip so that everything is * reinitialized to the power-on state. * * Basically follow the procedure outlined in the NCR53c700 * data manual under Chapter Six, How to Use, Steps Necessary to * Start SCRIPTS, with the exception of actually starting the * script and setting up the synchronous transfer gunk. */ /* Should we reset the scsi bus here??????????????????? */ NCR53c7x0_write8(ISTAT_REG_700, ISTAT_10_SRST); NCR53c7x0_write8(ISTAT_REG_700, 0); /* * saved_dcntl is set up in NCR53c7x0_init() before it is overwritten * here. We should have some better way of working out the CF bit * setting.. */ hostdata->saved_dcntl = DCNTL_10_EA|DCNTL_10_COM; if (hostdata->scsi_clock > 50000000) hostdata->saved_dcntl |= DCNTL_700_CF_3; else if (hostdata->scsi_clock > 37500000) hostdata->saved_dcntl |= DCNTL_700_CF_2; #if 0 else /* Any clocks less than 37.5MHz? */ #endif if (hostdata->options & OPTION_DEBUG_TRACE) NCR53c7x0_write8(DCNTL_REG, hostdata->saved_dcntl | DCNTL_SSM); else NCR53c7x0_write8(DCNTL_REG, hostdata->saved_dcntl); /* Following disables snooping - snooping is not required, as non- * cached pages are used for shared data, and appropriate use is * made of cache_push/cache_clear. Indeed, for 68060 * enabling snooping causes disk corruption of ext2fs free block * bitmaps and the like. If you have a 68060 with snooping hardwared * on, then you need to enable CONFIG_060_WRITETHROUGH. */ NCR53c7x0_write8(CTEST7_REG, CTEST7_10_TT1|CTEST7_STD); /* Actually burst of eight, according to my 53c710 databook */ NCR53c7x0_write8(hostdata->dmode, DMODE_10_BL_8 | DMODE_10_FC2); NCR53c7x0_write8(SCID_REG, 1 << host->this_id); NCR53c7x0_write8(SBCL_REG, 0); NCR53c7x0_write8(SCNTL1_REG, SCNTL1_ESR_700); NCR53c7x0_write8(SCNTL0_REG, ((hostdata->options & OPTION_PARITY) ? SCNTL0_EPC : 0) | SCNTL0_EPG_700 | SCNTL0_ARB1 | SCNTL0_ARB2); /* * Enable all interrupts, except parity which we only want when * the user requests it. */ NCR53c7x0_write8(DIEN_REG, DIEN_700_BF | DIEN_ABRT | DIEN_SSI | DIEN_SIR | DIEN_700_OPC); NCR53c7x0_write8(SIEN_REG_700, ((hostdata->options & OPTION_PARITY) ? SIEN_PAR : 0) | SIEN_700_STO | SIEN_RST | SIEN_UDC | SIEN_SGE | SIEN_MA); #ifdef CONFIG_MVME16x if (MACH_IS_MVME16x) { volatile unsigned long v; /* Enable scsi chip and s/w level 7 ints */ v = *(volatile unsigned long *)0xfff40080; v = (v & ~(0xf << 28)) | (4 << 28); *(volatile unsigned long *)0xfff40080 = v; v = *(volatile unsigned long *)0xfff4006c; v |= 0x8000; *(volatile unsigned long *)0xfff4006c = v; v = *(volatile unsigned long *)0xfff4202c; v = (v & ~0xff) | 0x10 | 4; *(volatile unsigned long *)0xfff4202c = v; } #endif /* Anything needed for your hardware? */ local_irq_restore(flags); } /* * Function static struct NCR53c7x0_cmd *allocate_cmd (Scsi_Cmnd *cmd) * * Purpose : Return the first free NCR53c7x0_cmd structure (which are * reused in a LIFO manner to minimize cache thrashing). * * Side effects : If we haven't yet scheduled allocation of NCR53c7x0_cmd * structures for this device, do so. Attempt to complete all scheduled * allocations using get_zeroed_page(), putting NCR53c7x0_cmd structures on * the free list. Teach programmers not to drink and hack. * * Inputs : cmd - SCSI command * * Returns : NCR53c7x0_cmd structure allocated on behalf of cmd; * NULL on failure. */ static void my_free_page (void *addr, int dummy) { /* XXX This assumes default cache mode to be IOMAP_FULL_CACHING, which * XXX may be invalid (CONFIG_060_WRITETHROUGH) */ kernel_set_cachemode((void *)addr, 4096, IOMAP_FULL_CACHING); free_page ((u32)addr); } static struct NCR53c7x0_cmd * allocate_cmd (Scsi_Cmnd *cmd) { struct Scsi_Host *host = cmd->device->host; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; u32 real; /* Real address */ int size; /* Size of *tmp */ struct NCR53c7x0_cmd *tmp; unsigned long flags; if (hostdata->options & OPTION_DEBUG_ALLOCATION) printk ("scsi%d : num_cmds = %d, can_queue = %d\n" " target = %d, lun = %d, %s\n", host->host_no, hostdata->num_cmds, host->can_queue, cmd->device->id, cmd->device->lun, (hostdata->cmd_allocated[cmd->device->id] & (1 << cmd->device->lun)) ? "already allocated" : "not allocated"); /* * If we have not yet reserved commands for this I_T_L nexus, and * the device exists (as indicated by permanent Scsi_Cmnd structures * being allocated under 1.3.x, or being outside of scan_scsis in * 1.2.x), do so now. */ if (!(hostdata->cmd_allocated[cmd->device->id] & (1 << cmd->device->lun)) && cmd->device && cmd->device->has_cmdblocks) { if ((hostdata->extra_allocate + hostdata->num_cmds) < host->can_queue) hostdata->extra_allocate += host->cmd_per_lun; hostdata->cmd_allocated[cmd->device->id] |= (1 << cmd->device->lun); } for (; hostdata->extra_allocate > 0 ; --hostdata->extra_allocate, ++hostdata->num_cmds) { /* historically, kmalloc has returned unaligned addresses; pad so we have enough room to ROUNDUP */ size = hostdata->max_cmd_size + sizeof (void *); #ifdef FORCE_DSA_ALIGNMENT /* * 53c710 rev.0 doesn't have an add-with-carry instruction. * Ensure we allocate enough memory to force alignment. */ size += 256; #endif /* FIXME: for ISA bus '7xx chips, we need to or GFP_DMA in here */ if (size > 4096) { printk (KERN_ERR "53c7xx: allocate_cmd size > 4K\n"); return NULL; } real = get_zeroed_page(GFP_ATOMIC); if (real == 0) return NULL; memset((void *)real, 0, 4096); cache_push(virt_to_phys((void *)real), 4096); cache_clear(virt_to_phys((void *)real), 4096); kernel_set_cachemode((void *)real, 4096, IOMAP_NOCACHE_SER); tmp = ROUNDUP(real, void *); #ifdef FORCE_DSA_ALIGNMENT { if (((u32)tmp & 0xff) > CmdPageStart) tmp = (struct NCR53c7x0_cmd *)((u32)tmp + 255); tmp = (struct NCR53c7x0_cmd *)(((u32)tmp & ~0xff) + CmdPageStart); #if 0 printk ("scsi: size = %d, real = 0x%08x, tmp set to 0x%08x\n", size, real, (u32)tmp); #endif } #endif tmp->real = (void *)real; tmp->size = size; tmp->free = ((void (*)(void *, int)) my_free_page); local_irq_save(flags); tmp->next = hostdata->free; hostdata->free = tmp; local_irq_restore(flags); } local_irq_save(flags); tmp = (struct NCR53c7x0_cmd *) hostdata->free; if (tmp) { hostdata->free = tmp->next; } local_irq_restore(flags); if (!tmp) printk ("scsi%d : can't allocate command for target %d lun %d\n", host->host_no, cmd->device->id, cmd->device->lun); return tmp; } /* * Function static struct NCR53c7x0_cmd *create_cmd (Scsi_Cmnd *cmd) * * * Purpose : allocate a NCR53c7x0_cmd structure, initialize it based on the * Scsi_Cmnd structure passed in cmd, including dsa and Linux field * initialization, and dsa code relocation. * * Inputs : cmd - SCSI command * * Returns : NCR53c7x0_cmd structure corresponding to cmd, * NULL on failure. */ static struct NCR53c7x0_cmd * create_cmd (Scsi_Cmnd *cmd) { NCR53c7x0_local_declare(); struct Scsi_Host *host = cmd->device->host; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; struct NCR53c7x0_cmd *tmp; /* NCR53c7x0_cmd structure for this command */ int datain, /* Number of instructions per phase */ dataout; int data_transfer_instructions, /* Count of dynamic instructions */ i; /* Counter */ u32 *cmd_datain, /* Address of datain/dataout code */ *cmd_dataout; /* Incremented as we assemble */ #ifdef notyet unsigned char *msgptr; /* Current byte in select message */ int msglen; /* Length of whole select message */ #endif unsigned long flags; u32 exp_select_indirect; /* Used in sanity check */ NCR53c7x0_local_setup(cmd->device->host); if (!(tmp = allocate_cmd (cmd))) return NULL; /* * Copy CDB and initialised result fields from Scsi_Cmnd to NCR53c7x0_cmd. * We do this because NCR53c7x0_cmd may have a special cache mode * selected to cope with lack of bus snooping, etc. */ memcpy(tmp->cmnd, cmd->cmnd, 12); tmp->result = cmd->result; /* * Decide whether we need to generate commands for DATA IN, * DATA OUT, neither, or both based on the SCSI command */ switch (cmd->cmnd[0]) { /* These commands do DATA IN */ case INQUIRY: case MODE_SENSE: case READ_6: case READ_10: case READ_CAPACITY: case REQUEST_SENSE: case READ_BLOCK_LIMITS: case READ_TOC: datain = 2 * (cmd->use_sg ? cmd->use_sg : 1) + 3; dataout = 0; break; /* These commands do DATA OUT */ case MODE_SELECT: case WRITE_6: case WRITE_10: #if 0 printk("scsi%d : command is ", host->host_no); __scsi_print_command(cmd->cmnd); #endif #if 0 printk ("scsi%d : %d scatter/gather segments\n", host->host_no, cmd->use_sg); #endif datain = 0; dataout = 2 * (cmd->use_sg ? cmd->use_sg : 1) + 3; #if 0 hostdata->options |= OPTION_DEBUG_INTR; #endif break; /* * These commands do no data transfer, we should force an * interrupt if a data phase is attempted on them. */ case TEST_UNIT_READY: case ALLOW_MEDIUM_REMOVAL: case START_STOP: datain = dataout = 0; break; /* * We don't know about these commands, so generate code to handle * both DATA IN and DATA OUT phases. More efficient to identify them * and add them to the above cases. */ default: printk("scsi%d : datain+dataout for command ", host->host_no); __scsi_print_command(cmd->cmnd); datain = dataout = 2 * (cmd->use_sg ? cmd->use_sg : 1) + 3; } /* * New code : so that active pointers work correctly regardless * of where the saved data pointer is at, we want to immediately * enter the dynamic code after selection, and on a non-data * phase perform a CALL to the non-data phase handler, with * returns back to this address. * * If a phase mismatch is encountered in the middle of a * Block MOVE instruction, we want to _leave_ that instruction * unchanged as the current case is, modify a temporary buffer, * and point the active pointer (TEMP) at that. * * Furthermore, we want to implement a saved data pointer, * set by the SAVE_DATA_POINTERs message. * * So, the data transfer segments will change to * CALL data_transfer, WHEN NOT data phase * MOVE x, x, WHEN data phase * ( repeat ) * JUMP other_transfer */ data_transfer_instructions = datain + dataout; /* * When we perform a request sense, we overwrite various things, * including the data transfer code. Make sure we have enough * space to do that. */ if (data_transfer_instructions < 2) data_transfer_instructions = 2; /* * The saved data pointer is set up so that a RESTORE POINTERS message * will start the data transfer over at the beginning. */ tmp->saved_data_pointer = virt_to_bus (hostdata->script) + hostdata->E_data_transfer; /* * Initialize Linux specific fields. */ tmp->cmd = cmd; tmp->next = NULL; tmp->flags = 0; tmp->dsa_next_addr = virt_to_bus(tmp->dsa) + hostdata->dsa_next - hostdata->dsa_start; tmp->dsa_addr = virt_to_bus(tmp->dsa) - hostdata->dsa_start; /* * Calculate addresses of dynamic code to fill in DSA */ tmp->data_transfer_start = tmp->dsa + (hostdata->dsa_end - hostdata->dsa_start) / sizeof(u32); tmp->data_transfer_end = tmp->data_transfer_start + 2 * data_transfer_instructions; cmd_datain = datain ? tmp->data_transfer_start : NULL; cmd_dataout = dataout ? (datain ? cmd_datain + 2 * datain : tmp-> data_transfer_start) : NULL; /* * Fill in the NCR53c7x0_cmd structure as follows * dsa, with fixed up DSA code * datain code * dataout code */ /* Copy template code into dsa and perform all necessary fixups */ if (hostdata->dsa_fixup) hostdata->dsa_fixup(tmp); patch_dsa_32(tmp->dsa, dsa_next, 0, 0); /* * XXX is this giving 53c710 access to the Scsi_Cmnd in some way? * Do we need to change it for caching reasons? */ patch_dsa_32(tmp->dsa, dsa_cmnd, 0, virt_to_bus(cmd)); if (hostdata->options & OPTION_DEBUG_SYNCHRONOUS) { exp_select_indirect = ((1 << cmd->device->id) << 16) | (hostdata->sync[cmd->device->id].sxfer_sanity << 8); if (hostdata->sync[cmd->device->id].select_indirect != exp_select_indirect) { printk ("scsi%d : sanity check failed select_indirect=0x%x\n", host->host_no, hostdata->sync[cmd->device->id].select_indirect); FATAL(host); } } patch_dsa_32(tmp->dsa, dsa_select, 0, hostdata->sync[cmd->device->id].select_indirect); /* * Right now, we'll do the WIDE and SYNCHRONOUS negotiations on * different commands; although it should be trivial to do them * both at the same time. */ if (hostdata->initiate_wdtr & (1 << cmd->device->id)) { memcpy ((void *) (tmp->select + 1), (void *) wdtr_message, sizeof(wdtr_message)); patch_dsa_32(tmp->dsa, dsa_msgout, 0, 1 + sizeof(wdtr_message)); local_irq_save(flags); hostdata->initiate_wdtr &= ~(1 << cmd->device->id); local_irq_restore(flags); } else if (hostdata->initiate_sdtr & (1 << cmd->device->id)) { memcpy ((void *) (tmp->select + 1), (void *) sdtr_message, sizeof(sdtr_message)); patch_dsa_32(tmp->dsa, dsa_msgout, 0, 1 + sizeof(sdtr_message)); tmp->flags |= CMD_FLAG_SDTR; local_irq_save(flags); hostdata->initiate_sdtr &= ~(1 << cmd->device->id); local_irq_restore(flags); } #if 1 else if (!(hostdata->talked_to & (1 << cmd->device->id)) && !(hostdata->options & OPTION_NO_ASYNC)) { memcpy ((void *) (tmp->select + 1), (void *) async_message, sizeof(async_message)); patch_dsa_32(tmp->dsa, dsa_msgout, 0, 1 + sizeof(async_message)); tmp->flags |= CMD_FLAG_SDTR; } #endif else patch_dsa_32(tmp->dsa, dsa_msgout, 0, 1); hostdata->talked_to |= (1 << cmd->device->id); tmp->select[0] = (hostdata->options & OPTION_DISCONNECT) ? IDENTIFY (1, cmd->device->lun) : IDENTIFY (0, cmd->device->lun); patch_dsa_32(tmp->dsa, dsa_msgout, 1, virt_to_bus(tmp->select)); patch_dsa_32(tmp->dsa, dsa_cmdout, 0, cmd->cmd_len); patch_dsa_32(tmp->dsa, dsa_cmdout, 1, virt_to_bus(tmp->cmnd)); patch_dsa_32(tmp->dsa, dsa_dataout, 0, cmd_dataout ? virt_to_bus (cmd_dataout) : virt_to_bus (hostdata->script) + hostdata->E_other_transfer); patch_dsa_32(tmp->dsa, dsa_datain, 0, cmd_datain ? virt_to_bus (cmd_datain) : virt_to_bus (hostdata->script) + hostdata->E_other_transfer); /* * XXX - need to make endian aware, should use separate variables * for both status and message bytes. */ patch_dsa_32(tmp->dsa, dsa_msgin, 0, 1); /* * FIXME : these only works for little endian. We probably want to * provide message and status fields in the NCR53c7x0_cmd * structure, and assign them to cmd->result when we're done. */ #ifdef BIG_ENDIAN patch_dsa_32(tmp->dsa, dsa_msgin, 1, virt_to_bus(&tmp->result) + 2); patch_dsa_32(tmp->dsa, dsa_status, 0, 1); patch_dsa_32(tmp->dsa, dsa_status, 1, virt_to_bus(&tmp->result) + 3); #else patch_dsa_32(tmp->dsa, dsa_msgin, 1, virt_to_bus(&tmp->result) + 1); patch_dsa_32(tmp->dsa, dsa_status, 0, 1); patch_dsa_32(tmp->dsa, dsa_status, 1, virt_to_bus(&tmp->result)); #endif patch_dsa_32(tmp->dsa, dsa_msgout_other, 0, 1); patch_dsa_32(tmp->dsa, dsa_msgout_other, 1, virt_to_bus(&(hostdata->NCR53c7xx_msg_nop))); /* * Generate code for zero or more of the DATA IN, DATA OUT phases * in the format * * CALL data_transfer, WHEN NOT phase * MOVE first buffer length, first buffer address, WHEN phase * ... * MOVE last buffer length, last buffer address, WHEN phase * JUMP other_transfer */ /* * See if we're getting to data transfer by generating an unconditional * interrupt. */ #if 0 if (datain) { cmd_datain[0] = 0x98080000; cmd_datain[1] = 0x03ffd00d; cmd_datain += 2; } #endif /* * XXX - I'm undecided whether all of this nonsense is faster * in the long run, or whether I should just go and implement a loop * on the NCR chip using table indirect mode? * * In any case, this is how it _must_ be done for 53c700/700-66 chips, * so this stays even when we come up with something better. * * When we're limited to 1 simultaneous command, no overlapping processing, * we're seeing 630K/sec, with 7% CPU usage on a slow Syquest 45M * drive. * * Not bad, not good. We'll see. */ tmp->bounce.len = 0; /* Assume aligned buffer */ for (i = 0; cmd->use_sg ? (i < cmd->use_sg) : !i; cmd_datain += 4, cmd_dataout += 4, ++i) { u32 vbuf = cmd->use_sg ? (u32)page_address(((struct scatterlist *)cmd->buffer)[i].page)+ ((struct scatterlist *)cmd->buffer)[i].offset : (u32)(cmd->request_buffer); u32 bbuf = virt_to_bus((void *)vbuf); u32 count = cmd->use_sg ? ((struct scatterlist *)cmd->buffer)[i].length : cmd->request_bufflen; /* * If we have buffers which are not aligned with 16 byte cache * lines, then we just hope nothing accesses the other parts of * those cache lines while the transfer is in progress. That would * fill the cache, and subsequent reads of the dma data would pick * up the wrong thing. * XXX We need a bounce buffer to handle that correctly. */ if (((bbuf & 15) || (count & 15)) && (datain || dataout)) { /* Bounce buffer needed */ if (cmd->use_sg) printk ("53c7xx: Non-aligned buffer with use_sg\n"); else if (datain && dataout) printk ("53c7xx: Non-aligned buffer with datain && dataout\n"); else if (count > 256) printk ("53c7xx: Non-aligned transfer > 256 bytes\n"); else { if (datain) { tmp->bounce.len = count; tmp->bounce.addr = vbuf; bbuf = virt_to_bus(tmp->bounce.buf); tmp->bounce.buf[0] = 0xff; tmp->bounce.buf[1] = 0xfe; tmp->bounce.buf[2] = 0xfd; tmp->bounce.buf[3] = 0xfc; } if (dataout) { memcpy ((void *)tmp->bounce.buf, (void *)vbuf, count); bbuf = virt_to_bus(tmp->bounce.buf); } } } if (datain) { cache_clear(virt_to_phys((void *)vbuf), count); /* CALL other_in, WHEN NOT DATA_IN */ cmd_datain[0] = ((DCMD_TYPE_TCI | DCMD_TCI_OP_CALL | DCMD_TCI_IO) << 24) | DBC_TCI_WAIT_FOR_VALID | DBC_TCI_COMPARE_PHASE; cmd_datain[1] = virt_to_bus (hostdata->script) + hostdata->E_other_in; /* MOVE count, buf, WHEN DATA_IN */ cmd_datain[2] = ((DCMD_TYPE_BMI | DCMD_BMI_OP_MOVE_I | DCMD_BMI_IO) << 24) | count; cmd_datain[3] = bbuf; #if 0 print_insn (host, cmd_datain, "dynamic ", 1); print_insn (host, cmd_datain + 2, "dynamic ", 1); #endif } if (dataout) { cache_push(virt_to_phys((void *)vbuf), count); /* CALL other_out, WHEN NOT DATA_OUT */ cmd_dataout[0] = ((DCMD_TYPE_TCI | DCMD_TCI_OP_CALL) << 24) | DBC_TCI_WAIT_FOR_VALID | DBC_TCI_COMPARE_PHASE; cmd_dataout[1] = virt_to_bus(hostdata->script) + hostdata->E_other_out; /* MOVE count, buf, WHEN DATA+OUT */ cmd_dataout[2] = ((DCMD_TYPE_BMI | DCMD_BMI_OP_MOVE_I) << 24) | count; cmd_dataout[3] = bbuf; #if 0 print_insn (host, cmd_dataout, "dynamic ", 1); print_insn (host, cmd_dataout + 2, "dynamic ", 1); #endif } } /* * Install JUMP instructions after the data transfer routines to return * control to the do_other_transfer routines. */ if (datain) { cmd_datain[0] = ((DCMD_TYPE_TCI | DCMD_TCI_OP_JUMP) << 24) | DBC_TCI_TRUE; cmd_datain[1] = virt_to_bus(hostdata->script) + hostdata->E_other_transfer; #if 0 print_insn (host, cmd_datain, "dynamic jump ", 1); #endif cmd_datain += 2; } #if 0 if (datain) { cmd_datain[0] = 0x98080000; cmd_datain[1] = 0x03ffdeed; cmd_datain += 2; } #endif if (dataout) { cmd_dataout[0] = ((DCMD_TYPE_TCI | DCMD_TCI_OP_JUMP) << 24) | DBC_TCI_TRUE; cmd_dataout[1] = virt_to_bus(hostdata->script) + hostdata->E_other_transfer; #if 0 print_insn (host, cmd_dataout, "dynamic jump ", 1); #endif cmd_dataout += 2; } return tmp; } /* * Function : int NCR53c7xx_queue_command (Scsi_Cmnd *cmd, * void (*done)(Scsi_Cmnd *)) * * Purpose : enqueues a SCSI command * * Inputs : cmd - SCSI command, done - function called on completion, with * a pointer to the command descriptor. * * Returns : 0 * * Side effects : * cmd is added to the per instance driver issue_queue, with major * twiddling done to the host specific fields of cmd. If the * process_issue_queue coroutine isn't running, it is restarted. * * NOTE : we use the host_scribble field of the Scsi_Cmnd structure to * hold our own data, and pervert the ptr field of the SCp field * to create a linked list. */ int NCR53c7xx_queue_command (Scsi_Cmnd *cmd, void (* done)(Scsi_Cmnd *)) { struct Scsi_Host *host = cmd->device->host; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; unsigned long flags; Scsi_Cmnd *tmp; cmd->scsi_done = done; cmd->host_scribble = NULL; cmd->SCp.ptr = NULL; cmd->SCp.buffer = NULL; #ifdef VALID_IDS /* Ignore commands on invalid IDs */ if (!hostdata->valid_ids[cmd->device->id]) { printk("scsi%d : ignoring target %d lun %d\n", host->host_no, cmd->device->id, cmd->device->lun); cmd->result = (DID_BAD_TARGET << 16); done(cmd); return 0; } #endif local_irq_save(flags); if ((hostdata->options & (OPTION_DEBUG_INIT_ONLY|OPTION_DEBUG_PROBE_ONLY)) || ((hostdata->options & OPTION_DEBUG_TARGET_LIMIT) && !(hostdata->debug_lun_limit[cmd->device->id] & (1 << cmd->device->lun))) #ifdef LINUX_1_2 || cmd->device->id > 7 #else || cmd->device->id > host->max_id #endif || cmd->device->id == host->this_id || hostdata->state == STATE_DISABLED) { printk("scsi%d : disabled or bad target %d lun %d\n", host->host_no, cmd->device->id, cmd->device->lun); cmd->result = (DID_BAD_TARGET << 16); done(cmd); local_irq_restore(flags); return 0; } if ((hostdata->options & OPTION_DEBUG_NCOMMANDS_LIMIT) && (hostdata->debug_count_limit == 0)) { printk("scsi%d : maximum commands exceeded\n", host->host_no); cmd->result = (DID_BAD_TARGET << 16); done(cmd); local_irq_restore(flags); return 0; } if (hostdata->options & OPTION_DEBUG_READ_ONLY) { switch (cmd->cmnd[0]) { case WRITE_6: case WRITE_10: printk("scsi%d : WRITE attempted with NO_WRITE debugging flag set\n", host->host_no); cmd->result = (DID_BAD_TARGET << 16); done(cmd); local_irq_restore(flags); return 0; } } if ((hostdata->options & OPTION_DEBUG_TARGET_LIMIT) && hostdata->debug_count_limit != -1) --hostdata->debug_count_limit; cmd->result = 0xffff; /* The NCR will overwrite message and status with valid data */ cmd->host_scribble = (unsigned char *) tmp = create_cmd (cmd); /* * REQUEST SENSE commands are inserted at the head of the queue * so that we do not clear the contingent allegiance condition * they may be looking at. */ if (!(hostdata->issue_queue) || (cmd->cmnd[0] == REQUEST_SENSE)) { cmd->SCp.ptr = (unsigned char *) hostdata->issue_queue; hostdata->issue_queue = cmd; } else { for (tmp = (Scsi_Cmnd *) hostdata->issue_queue; tmp->SCp.ptr; tmp = (Scsi_Cmnd *) tmp->SCp.ptr); tmp->SCp.ptr = (unsigned char *) cmd; } local_irq_restore(flags); run_process_issue_queue(); return 0; } /* * Function : void to_schedule_list (struct Scsi_Host *host, * struct NCR53c7x0_hostdata * hostdata, Scsi_Cmnd *cmd) * * Purpose : takes a SCSI command which was just removed from the * issue queue, and deals with it by inserting it in the first * free slot in the schedule list or by terminating it immediately. * * Inputs : * host - SCSI host adapter; hostdata - hostdata structure for * this adapter; cmd - a pointer to the command; should have * the host_scribble field initialized to point to a valid * * Side effects : * cmd is added to the per instance schedule list, with minor * twiddling done to the host specific fields of cmd. * */ static __inline__ void to_schedule_list (struct Scsi_Host *host, struct NCR53c7x0_hostdata *hostdata, struct NCR53c7x0_cmd *cmd) { NCR53c7x0_local_declare(); Scsi_Cmnd *tmp = cmd->cmd; unsigned long flags; /* dsa start is negative, so subtraction is used */ volatile u32 *ncrcurrent; int i; NCR53c7x0_local_setup(host); #if 0 printk("scsi%d : new dsa is 0x%lx (virt 0x%p)\n", host->host_no, virt_to_bus(hostdata->dsa), hostdata->dsa); #endif local_irq_save(flags); /* * Work around race condition : if an interrupt fired and we * got disabled forget about this command. */ if (hostdata->state == STATE_DISABLED) { printk("scsi%d : driver disabled\n", host->host_no); tmp->result = (DID_BAD_TARGET << 16); cmd->next = (struct NCR53c7x0_cmd *) hostdata->free; hostdata->free = cmd; tmp->scsi_done(tmp); local_irq_restore(flags); return; } for (i = host->can_queue, ncrcurrent = hostdata->schedule; i > 0 && ncrcurrent[0] != hostdata->NOP_insn; --i, ncrcurrent += 2 /* JUMP instructions are two words */); if (i > 0) { ++hostdata->busy[tmp->device->id][tmp->device->lun]; cmd->next = hostdata->running_list; hostdata->running_list = cmd; /* Restore this instruction to a NOP once the command starts */ cmd->dsa [(hostdata->dsa_jump_dest - hostdata->dsa_start) / sizeof(u32)] = (u32) virt_to_bus ((void *)ncrcurrent); /* Replace the current jump operand. */ ncrcurrent[1] = virt_to_bus ((void *) cmd->dsa) + hostdata->E_dsa_code_begin - hostdata->E_dsa_code_template; /* Replace the NOP instruction with a JUMP */ ncrcurrent[0] = ((DCMD_TYPE_TCI|DCMD_TCI_OP_JUMP) << 24) | DBC_TCI_TRUE; } else { printk ("scsi%d: no free slot\n", host->host_no); disable(host); tmp->result = (DID_ERROR << 16); cmd->next = (struct NCR53c7x0_cmd *) hostdata->free; hostdata->free = cmd; tmp->scsi_done(tmp); local_irq_restore(flags); return; } /* * If the NCR chip is in an idle state, start it running the scheduler * immediately. Otherwise, signal the chip to jump to schedule as * soon as it is idle. */ if (hostdata->idle) { hostdata->idle = 0; hostdata->state = STATE_RUNNING; NCR53c7x0_write32 (DSP_REG, virt_to_bus ((void *)hostdata->schedule)); if (hostdata->options & OPTION_DEBUG_TRACE) NCR53c7x0_write8 (DCNTL_REG, hostdata->saved_dcntl | DCNTL_SSM | DCNTL_STD); } else { NCR53c7x0_write8(hostdata->istat, ISTAT_10_SIGP); } local_irq_restore(flags); } /* * Function : busyp (struct Scsi_Host *host, struct NCR53c7x0_hostdata * *hostdata, Scsi_Cmnd *cmd) * * Purpose : decide if we can pass the given SCSI command on to the * device in question or not. * * Returns : non-zero when we're busy, 0 when we aren't. */ static __inline__ int busyp (struct Scsi_Host *host, struct NCR53c7x0_hostdata *hostdata, Scsi_Cmnd *cmd) { /* FIXME : in the future, this needs to accommodate SCSI-II tagged queuing, and we may be able to play with fairness here a bit. */ return hostdata->busy[cmd->device->id][cmd->device->lun]; } /* * Function : process_issue_queue (void) * * Purpose : transfer commands from the issue queue to NCR start queue * of each NCR53c7/8xx in the system, avoiding kernel stack * overflows when the scsi_done() function is invoked recursively. * * NOTE : process_issue_queue exits with interrupts *disabled*, so the * caller must reenable them if it desires. * * NOTE : process_issue_queue should be called from both * NCR53c7x0_queue_command() and from the interrupt handler * after command completion in case NCR53c7x0_queue_command() * isn't invoked again but we've freed up resources that are * needed. */ static void process_issue_queue (unsigned long flags) { Scsi_Cmnd *tmp, *prev; struct Scsi_Host *host; struct NCR53c7x0_hostdata *hostdata; int done; /* * We run (with interrupts disabled) until we're sure that none of * the host adapters have anything that can be done, at which point * we set process_issue_queue_running to 0 and exit. * * Interrupts are enabled before doing various other internal * instructions, after we've decided that we need to run through * the loop again. * */ do { local_irq_disable(); /* Freeze request queues */ done = 1; for (host = first_host; host && host->hostt == the_template; host = host->next) { hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; local_irq_disable(); if (hostdata->issue_queue) { if (hostdata->state == STATE_DISABLED) { tmp = (Scsi_Cmnd *) hostdata->issue_queue; hostdata->issue_queue = (Scsi_Cmnd *) tmp->SCp.ptr; tmp->result = (DID_BAD_TARGET << 16); if (tmp->host_scribble) { ((struct NCR53c7x0_cmd *)tmp->host_scribble)->next = hostdata->free; hostdata->free = (struct NCR53c7x0_cmd *)tmp->host_scribble; tmp->host_scribble = NULL; } tmp->scsi_done (tmp); done = 0; } else for (tmp = (Scsi_Cmnd *) hostdata->issue_queue, prev = NULL; tmp; prev = tmp, tmp = (Scsi_Cmnd *) tmp->SCp.ptr) if (!tmp->host_scribble || !busyp (host, hostdata, tmp)) { if (prev) prev->SCp.ptr = tmp->SCp.ptr; else hostdata->issue_queue = (Scsi_Cmnd *) tmp->SCp.ptr; tmp->SCp.ptr = NULL; if (tmp->host_scribble) { if (hostdata->options & OPTION_DEBUG_QUEUES) printk ("scsi%d : moving command for target %d lun %d to start list\n", host->host_no, tmp->device->id, tmp->device->lun); to_schedule_list (host, hostdata, (struct NCR53c7x0_cmd *) tmp->host_scribble); } else { if (((tmp->result & 0xff) == 0xff) || ((tmp->result & 0xff00) == 0xff00)) { printk ("scsi%d : danger Will Robinson!\n", host->host_no); tmp->result = DID_ERROR << 16; disable (host); } tmp->scsi_done(tmp); } done = 0; } /* if target/lun is not busy */ } /* if hostdata->issue_queue */ if (!done) local_irq_restore(flags); } /* for host */ } while (!done); process_issue_queue_running = 0; } /* * Function : static void intr_scsi (struct Scsi_Host *host, * struct NCR53c7x0_cmd *cmd) * * Purpose : handle all SCSI interrupts, indicated by the setting * of the SIP bit in the ISTAT register. * * Inputs : host, cmd - host and NCR command causing the interrupt, cmd * may be NULL. */ static void intr_scsi (struct Scsi_Host *host, struct NCR53c7x0_cmd *cmd) { NCR53c7x0_local_declare(); struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; unsigned char sstat0_sist0, sist1, /* Registers */ fatal; /* Did a fatal interrupt occur ? */ NCR53c7x0_local_setup(host); fatal = 0; sstat0_sist0 = NCR53c7x0_read8(SSTAT0_REG); sist1 = 0; if (hostdata->options & OPTION_DEBUG_INTR) printk ("scsi%d : SIST0 0x%0x, SIST1 0x%0x\n", host->host_no, sstat0_sist0, sist1); /* 250ms selection timeout */ if (sstat0_sist0 & SSTAT0_700_STO) { fatal = 1; if (hostdata->options & OPTION_DEBUG_INTR) { printk ("scsi%d : Selection Timeout\n", host->host_no); if (cmd) { printk("scsi%d : target %d, lun %d, command ", host->host_no, cmd->cmd->device->id, cmd->cmd->device->lun); __scsi_print_command (cmd->cmd->cmnd); printk("scsi%d : dsp = 0x%x (virt 0x%p)\n", host->host_no, NCR53c7x0_read32(DSP_REG), bus_to_virt(NCR53c7x0_read32(DSP_REG))); } else { printk("scsi%d : no command\n", host->host_no); } } /* * XXX - question : how do we want to handle the Illegal Instruction * interrupt, which may occur before or after the Selection Timeout * interrupt? */ if (1) { hostdata->idle = 1; hostdata->expecting_sto = 0; if (hostdata->test_running) { hostdata->test_running = 0; hostdata->test_completed = 3; } else if (cmd) { abnormal_finished(cmd, DID_BAD_TARGET << 16); } #if 0 hostdata->intrs = 0; #endif } } /* * FIXME : in theory, we can also get a UDC when a STO occurs. */ if (sstat0_sist0 & SSTAT0_UDC) { fatal = 1; if (cmd) { printk("scsi%d : target %d lun %d unexpected disconnect\n", host->host_no, cmd->cmd->device->id, cmd->cmd->device->lun); print_lots (host); abnormal_finished(cmd, DID_ERROR << 16); } else printk("scsi%d : unexpected disconnect (no command)\n", host->host_no); hostdata->dsp = (u32 *) hostdata->schedule; hostdata->dsp_changed = 1; } /* SCSI PARITY error */ if (sstat0_sist0 & SSTAT0_PAR) { fatal = 1; if (cmd && cmd->cmd) { printk("scsi%d : target %d lun %d parity error.\n", host->host_no, cmd->cmd->device->id, cmd->cmd->device->lun); abnormal_finished (cmd, DID_PARITY << 16); } else printk("scsi%d : parity error\n", host->host_no); /* Should send message out, parity error */ /* XXX - Reduce synchronous transfer rate! */ hostdata->dsp = hostdata->script + hostdata->E_initiator_abort / sizeof(u32); hostdata->dsp_changed = 1; /* SCSI GROSS error */ } if (sstat0_sist0 & SSTAT0_SGE) { fatal = 1; printk("scsi%d : gross error, saved2_dsa = 0x%x\n", host->host_no, (unsigned int)hostdata->saved2_dsa); print_lots (host); /* * A SCSI gross error may occur when we have * * - A synchronous offset which causes the SCSI FIFO to be overwritten. * * - A REQ which causes the maximum synchronous offset programmed in * the SXFER register to be exceeded. * * - A phase change with an outstanding synchronous offset. * * - Residual data in the synchronous data FIFO, with a transfer * other than a synchronous receive is started.$# */ /* XXX Should deduce synchronous transfer rate! */ hostdata->dsp = hostdata->script + hostdata->E_initiator_abort / sizeof(u32); hostdata->dsp_changed = 1; /* Phase mismatch */ } if (sstat0_sist0 & SSTAT0_MA) { fatal = 1; if (hostdata->options & OPTION_DEBUG_INTR) printk ("scsi%d : SSTAT0_MA\n", host->host_no); intr_phase_mismatch (host, cmd); } #if 0 if (sstat0_sist0 & SIST0_800_RSL) printk ("scsi%d : Oh no Mr. Bill!\n", host->host_no); #endif /* * If a fatal SCSI interrupt occurs, we must insure that the DMA and * SCSI FIFOs were flushed. */ if (fatal) { if (!hostdata->dstat_valid) { hostdata->dstat = NCR53c7x0_read8(DSTAT_REG); hostdata->dstat_valid = 1; } if (!(hostdata->dstat & DSTAT_DFE)) { printk ("scsi%d : DMA FIFO not empty\n", host->host_no); /* * Really need to check this code for 710 RGH. * Havn't seen any problems, but maybe we should FLUSH before * clearing sometimes. */ NCR53c7x0_write8 (CTEST8_REG, CTEST8_10_CLF); while (NCR53c7x0_read8 (CTEST8_REG) & CTEST8_10_CLF) ; hostdata->dstat |= DSTAT_DFE; } } } #ifdef CYCLIC_TRACE /* * The following implements a cyclic log of instructions executed, if you turn * TRACE on. It will also print the log for you. Very useful when debugging * 53c710 support, possibly not really needed any more. */ u32 insn_log[4096]; u32 insn_log_index = 0; void log1 (u32 i) { insn_log[insn_log_index++] = i; if (insn_log_index == 4096) insn_log_index = 0; } void log_insn (u32 *ip) { log1 ((u32)ip); log1 (*ip); log1 (*(ip+1)); if (((*ip >> 24) & DCMD_TYPE_MASK) == DCMD_TYPE_MMI) log1 (*(ip+2)); } void dump_log(void) { int cnt = 0; int i = insn_log_index; int size; struct Scsi_Host *host = first_host; while (cnt < 4096) { printk ("%08x (+%6x): ", insn_log[i], (insn_log[i] - (u32)&(((struct NCR53c7x0_hostdata *)host->hostdata[0])->script))/4); if (++i == 4096) i = 0; cnt++; if (((insn_log[i] >> 24) & DCMD_TYPE_MASK) == DCMD_TYPE_MMI) size = 3; else size = 2; while (size--) { printk ("%08x ", insn_log[i]); if (++i == 4096) i = 0; cnt++; } printk ("\n"); } } #endif /* * Function : static void NCR53c7x0_intfly (struct Scsi_Host *host) * * Purpose : Scan command queue for specified host, looking for completed * commands. * * Inputs : Scsi_Host pointer. * * This is called from the interrupt handler, when a simulated INTFLY * interrupt occurs. */ static void NCR53c7x0_intfly (struct Scsi_Host *host) { NCR53c7x0_local_declare(); struct NCR53c7x0_hostdata *hostdata; /* host->hostdata[0] */ struct NCR53c7x0_cmd *cmd, /* command which halted */ **cmd_prev_ptr; unsigned long flags; char search_found = 0; /* Got at least one ? */ hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; NCR53c7x0_local_setup(host); if (hostdata->options & OPTION_DEBUG_INTR) printk ("scsi%d : INTFLY\n", host->host_no); /* * Traverse our list of running commands, and look * for those with valid (non-0xff ff) status and message * bytes encoded in the result which signify command * completion. */ local_irq_save(flags); restart: for (cmd_prev_ptr = (struct NCR53c7x0_cmd **)&(hostdata->running_list), cmd = (struct NCR53c7x0_cmd *) hostdata->running_list; cmd ; cmd_prev_ptr = (struct NCR53c7x0_cmd **) &(cmd->next), cmd = (struct NCR53c7x0_cmd *) cmd->next) { Scsi_Cmnd *tmp; if (!cmd) { printk("scsi%d : very weird.\n", host->host_no); break; } if (!(tmp = cmd->cmd)) { printk("scsi%d : weird. NCR53c7x0_cmd has no Scsi_Cmnd\n", host->host_no); continue; } /* Copy the result over now; may not be complete, * but subsequent tests may as well be done on * cached memory. */ tmp->result = cmd->result; if (((tmp->result & 0xff) == 0xff) || ((tmp->result & 0xff00) == 0xff00)) continue; search_found = 1; if (cmd->bounce.len) memcpy ((void *)cmd->bounce.addr, (void *)cmd->bounce.buf, cmd->bounce.len); /* Important - remove from list _before_ done is called */ if (cmd_prev_ptr) *cmd_prev_ptr = (struct NCR53c7x0_cmd *) cmd->next; --hostdata->busy[tmp->device->id][tmp->device->lun]; cmd->next = hostdata->free; hostdata->free = cmd; tmp->host_scribble = NULL; if (hostdata->options & OPTION_DEBUG_INTR) { printk ("scsi%d : command complete : pid %lu, id %d,lun %d result 0x%x ", host->host_no, tmp->pid, tmp->device->id, tmp->device->lun, tmp->result); __scsi_print_command (tmp->cmnd); } tmp->scsi_done(tmp); goto restart; } local_irq_restore(flags); if (!search_found) { printk ("scsi%d : WARNING : INTFLY with no completed commands.\n", host->host_no); } else { run_process_issue_queue(); } return; } /* * Function : static irqreturn_t NCR53c7x0_intr (int irq, void *dev_id, struct pt_regs * regs) * * Purpose : handle NCR53c7x0 interrupts for all NCR devices sharing * the same IRQ line. * * Inputs : Since we're using the SA_INTERRUPT interrupt handler * semantics, irq indicates the interrupt which invoked * this handler. * * On the 710 we simualte an INTFLY with a script interrupt, and the * script interrupt handler will call back to this function. */ static irqreturn_t NCR53c7x0_intr (int irq, void *dev_id, struct pt_regs * regs) { NCR53c7x0_local_declare(); struct Scsi_Host *host; /* Host we are looking at */ unsigned char istat; /* Values of interrupt regs */ struct NCR53c7x0_hostdata *hostdata; /* host->hostdata[0] */ struct NCR53c7x0_cmd *cmd; /* command which halted */ u32 *dsa; /* DSA */ int handled = 0; #ifdef NCR_DEBUG char buf[80]; /* Debugging sprintf buffer */ size_t buflen; /* Length of same */ #endif host = (struct Scsi_Host *)dev_id; hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; NCR53c7x0_local_setup(host); /* * Only read istat once per loop, since reading it again will unstack * interrupts */ while ((istat = NCR53c7x0_read8(hostdata->istat)) & (ISTAT_SIP|ISTAT_DIP)) { handled = 1; hostdata->dsp_changed = 0; hostdata->dstat_valid = 0; hostdata->state = STATE_HALTED; if (NCR53c7x0_read8 (SSTAT2_REG) & SSTAT2_FF_MASK) printk ("scsi%d : SCSI FIFO not empty\n", host->host_no); /* * NCR53c700 and NCR53c700-66 change the current SCSI * process, hostdata->ncrcurrent, in the Linux driver so * cmd = hostdata->ncrcurrent. * * With other chips, we must look through the commands * executing and find the command structure which * corresponds to the DSA register. */ if (hostdata->options & OPTION_700) { cmd = (struct NCR53c7x0_cmd *) hostdata->ncrcurrent; } else { dsa = bus_to_virt(NCR53c7x0_read32(DSA_REG)); for (cmd = (struct NCR53c7x0_cmd *) hostdata->running_list; cmd && (dsa + (hostdata->dsa_start / sizeof(u32))) != cmd->dsa; cmd = (struct NCR53c7x0_cmd *)(cmd->next)) ; } if (hostdata->options & OPTION_DEBUG_INTR) { if (cmd) { printk("scsi%d : interrupt for pid %lu, id %d, lun %d ", host->host_no, cmd->cmd->pid, (int) cmd->cmd->device->id, (int) cmd->cmd->device->lun); __scsi_print_command (cmd->cmd->cmnd); } else { printk("scsi%d : no active command\n", host->host_no); } } if (istat & ISTAT_SIP) { if (hostdata->options & OPTION_DEBUG_INTR) printk ("scsi%d : ISTAT_SIP\n", host->host_no); intr_scsi (host, cmd); } if (istat & ISTAT_DIP) { if (hostdata->options & OPTION_DEBUG_INTR) printk ("scsi%d : ISTAT_DIP\n", host->host_no); intr_dma (host, cmd); } if (!hostdata->dstat_valid) { hostdata->dstat = NCR53c7x0_read8(DSTAT_REG); hostdata->dstat_valid = 1; } if (!(hostdata->dstat & DSTAT_DFE)) { printk ("scsi%d : DMA FIFO not empty\n", host->host_no); /* Really need to check this out for 710 RGH */ NCR53c7x0_write8 (CTEST8_REG, CTEST8_10_CLF); while (NCR53c7x0_read8 (CTEST8_REG) & CTEST8_10_CLF) ; hostdata->dstat |= DSTAT_DFE; } if (!hostdata->idle && hostdata->state == STATE_HALTED) { if (!hostdata->dsp_changed) hostdata->dsp = (u32 *)bus_to_virt(NCR53c7x0_read32(DSP_REG)); #if 0 printk("scsi%d : new dsp is 0x%lx (virt 0x%p)\n", host->host_no, virt_to_bus(hostdata->dsp), hostdata->dsp); #endif hostdata->state = STATE_RUNNING; NCR53c7x0_write32 (DSP_REG, virt_to_bus(hostdata->dsp)); if (hostdata->options & OPTION_DEBUG_TRACE) { #ifdef CYCLIC_TRACE log_insn (hostdata->dsp); #else print_insn (host, hostdata->dsp, "t ", 1); #endif NCR53c7x0_write8 (DCNTL_REG, hostdata->saved_dcntl | DCNTL_SSM | DCNTL_STD); } } } return IRQ_HANDLED; } /* * Function : static int abort_connected (struct Scsi_Host *host) * * Purpose : Assuming that the NCR SCSI processor is currently * halted, break the currently established nexus. Clean * up of the NCR53c7x0_cmd and Scsi_Cmnd structures should * be done on receipt of the abort interrupt. * * Inputs : host - SCSI host * */ static int abort_connected (struct Scsi_Host *host) { #ifdef NEW_ABORT NCR53c7x0_local_declare(); #endif struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; /* FIXME : this probably should change for production kernels; at the least, counter should move to a per-host structure. */ static int counter = 5; #ifdef NEW_ABORT int sstat, phase, offset; u32 *script; NCR53c7x0_local_setup(host); #endif if (--counter <= 0) { disable(host); return 0; } printk ("scsi%d : DANGER : abort_connected() called \n", host->host_no); #ifdef NEW_ABORT /* * New strategy : Rather than using a generic abort routine, * we'll specifically try to source or sink the appropriate * amount of data for the phase we're currently in (taking into * account the current synchronous offset) */ sstat = (NCR53c8x0_read8 (SSTAT2_REG); offset = OFFSET (sstat & SSTAT2_FF_MASK) >> SSTAT2_FF_SHIFT; phase = sstat & SSTAT2_PHASE_MASK; /* * SET ATN * MOVE source_or_sink, WHEN CURRENT PHASE * < repeat for each outstanding byte > * JUMP send_abort_message */ script = hostdata->abort_script = kmalloc ( 8 /* instruction size */ * ( 1 /* set ATN */ + (!offset ? 1 : offset) /* One transfer per outstanding byte */ + 1 /* send abort message */), GFP_ATOMIC); #else /* def NEW_ABORT */ hostdata->dsp = hostdata->script + hostdata->E_initiator_abort / sizeof(u32); #endif /* def NEW_ABORT */ hostdata->dsp_changed = 1; /* XXX - need to flag the command as aborted after the abort_connected code runs */ return 0; } /* * Function : static int datapath_residual (Scsi_Host *host) * * Purpose : return residual data count of what's in the chip. * * Inputs : host - SCSI host */ static int datapath_residual (struct Scsi_Host *host) { NCR53c7x0_local_declare(); int count, synchronous, sstat; unsigned int ddir; NCR53c7x0_local_setup(host); /* COMPAT : the 700 and 700-66 need to use DFIFO_00_BO_MASK */ count = ((NCR53c7x0_read8 (DFIFO_REG) & DFIFO_10_BO_MASK) - (NCR53c7x0_read32 (DBC_REG) & DFIFO_10_BO_MASK)) & DFIFO_10_BO_MASK; synchronous = NCR53c7x0_read8 (SXFER_REG) & SXFER_MO_MASK; /* COMPAT : DDIR is elsewhere on non-'8xx chips. */ ddir = NCR53c7x0_read8 (CTEST0_REG_700) & CTEST0_700_DDIR; if (ddir) { /* Receive */ if (synchronous) count += (NCR53c7x0_read8 (SSTAT2_REG) & SSTAT2_FF_MASK) >> SSTAT2_FF_SHIFT; else if (NCR53c7x0_read8 (SSTAT1_REG) & SSTAT1_ILF) ++count; } else { /* Send */ sstat = NCR53c7x0_read8 (SSTAT1_REG); if (sstat & SSTAT1_OLF) ++count; if (synchronous && (sstat & SSTAT1_ORF)) ++count; } return count; } /* * Function : static const char * sbcl_to_phase (int sbcl)_ * * Purpose : Convert SBCL register to user-parsable phase representation * * Inputs : sbcl - value of sbcl register */ static const char * sbcl_to_phase (int sbcl) { switch (sbcl & SBCL_PHASE_MASK) { case SBCL_PHASE_DATAIN: return "DATAIN"; case SBCL_PHASE_DATAOUT: return "DATAOUT"; case SBCL_PHASE_MSGIN: return "MSGIN"; case SBCL_PHASE_MSGOUT: return "MSGOUT"; case SBCL_PHASE_CMDOUT: return "CMDOUT"; case SBCL_PHASE_STATIN: return "STATUSIN"; default: return "unknown"; } } /* * Function : static const char * sstat2_to_phase (int sstat)_ * * Purpose : Convert SSTAT2 register to user-parsable phase representation * * Inputs : sstat - value of sstat register */ static const char * sstat2_to_phase (int sstat) { switch (sstat & SSTAT2_PHASE_MASK) { case SSTAT2_PHASE_DATAIN: return "DATAIN"; case SSTAT2_PHASE_DATAOUT: return "DATAOUT"; case SSTAT2_PHASE_MSGIN: return "MSGIN"; case SSTAT2_PHASE_MSGOUT: return "MSGOUT"; case SSTAT2_PHASE_CMDOUT: return "CMDOUT"; case SSTAT2_PHASE_STATIN: return "STATUSIN"; default: return "unknown"; } } /* * Function : static void intr_phase_mismatch (struct Scsi_Host *host, * struct NCR53c7x0_cmd *cmd) * * Purpose : Handle phase mismatch interrupts * * Inputs : host, cmd - host and NCR command causing the interrupt, cmd * may be NULL. * * Side effects : The abort_connected() routine is called or the NCR chip * is restarted, jumping to the command_complete entry point, or * patching the address and transfer count of the current instruction * and calling the msg_in entry point as appropriate. */ static void intr_phase_mismatch (struct Scsi_Host *host, struct NCR53c7x0_cmd *cmd) { NCR53c7x0_local_declare(); u32 dbc_dcmd, *dsp, *dsp_next; unsigned char dcmd, sbcl; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; int residual; enum {ACTION_ABORT, ACTION_ABORT_PRINT, ACTION_CONTINUE} action = ACTION_ABORT_PRINT; const char *where = NULL; NCR53c7x0_local_setup(host); /* * Corrective action is based on where in the SCSI SCRIPT(tm) the error * occurred, as well as which SCSI phase we are currently in. */ dsp_next = bus_to_virt(NCR53c7x0_read32(DSP_REG)); /* * Fetch the current instruction, and remove the operands for easier * interpretation. */ dbc_dcmd = NCR53c7x0_read32(DBC_REG); dcmd = (dbc_dcmd & 0xff000000) >> 24; /* * Like other processors, the NCR adjusts the instruction pointer before * instruction decode. Set the DSP address back to what it should * be for this instruction based on its size (2 or 3 32 bit words). */ dsp = dsp_next - NCR53c7x0_insn_size(dcmd); /* * Read new SCSI phase from the SBCL lines. Since all of our code uses * a WHEN conditional instead of an IF conditional, we don't need to * wait for a new REQ. */ sbcl = NCR53c7x0_read8(SBCL_REG) & SBCL_PHASE_MASK; if (!cmd) { action = ACTION_ABORT_PRINT; where = "no current command"; /* * The way my SCSI SCRIPTS(tm) are architected, recoverable phase * mismatches should only occur where we're doing a multi-byte * BMI instruction. Specifically, this means * * - select messages (a SCSI-I target may ignore additional messages * after the IDENTIFY; any target may reject a SDTR or WDTR) * * - command out (targets may send a message to signal an error * condition, or go into STATUSIN after they've decided * they don't like the command. * * - reply_message (targets may reject a multi-byte message in the * middle) * * - data transfer routines (command completion with buffer space * left, disconnect message, or error message) */ } else if (((dsp >= cmd->data_transfer_start && dsp < cmd->data_transfer_end)) || dsp == (cmd->residual + 2)) { if ((dcmd & (DCMD_TYPE_MASK|DCMD_BMI_OP_MASK|DCMD_BMI_INDIRECT| DCMD_BMI_MSG|DCMD_BMI_CD)) == (DCMD_TYPE_BMI| DCMD_BMI_OP_MOVE_I)) { residual = datapath_residual (host); if (hostdata->options & OPTION_DEBUG_DISCONNECT) printk ("scsi%d : handling residual transfer (+ %d bytes from DMA FIFO)\n", host->host_no, residual); /* * The first instruction is a CALL to the alternate handler for * this data transfer phase, so we can do calls to * munge_msg_restart as we would if control were passed * from normal dynamic code. */ if (dsp != cmd->residual + 2) { cmd->residual[0] = ((DCMD_TYPE_TCI | DCMD_TCI_OP_CALL | ((dcmd & DCMD_BMI_IO) ? DCMD_TCI_IO : 0)) << 24) | DBC_TCI_WAIT_FOR_VALID | DBC_TCI_COMPARE_PHASE; cmd->residual[1] = virt_to_bus(hostdata->script) + ((dcmd & DCMD_BMI_IO) ? hostdata->E_other_in : hostdata->E_other_out); } /* * The second instruction is the a data transfer block * move instruction, reflecting the pointer and count at the * time of the phase mismatch. */ cmd->residual[2] = dbc_dcmd + residual; cmd->residual[3] = NCR53c7x0_read32(DNAD_REG) - residual; /* * The third and final instruction is a jump to the instruction * which follows the instruction which had to be 'split' */ if (dsp != cmd->residual + 2) { cmd->residual[4] = ((DCMD_TYPE_TCI|DCMD_TCI_OP_JUMP) << 24) | DBC_TCI_TRUE; cmd->residual[5] = virt_to_bus(dsp_next); } /* * For the sake of simplicity, transfer control to the * conditional CALL at the start of the residual buffer. */ hostdata->dsp = cmd->residual; hostdata->dsp_changed = 1; action = ACTION_CONTINUE; } else { where = "non-BMI dynamic DSA code"; action = ACTION_ABORT_PRINT; } } else if (dsp == (hostdata->script + hostdata->E_select_msgout / 4 + 2)) { /* RGH 290697: Added +2 above, to compensate for the script * instruction which disables the selection timer. */ /* Release ATN */ NCR53c7x0_write8 (SOCL_REG, 0); switch (sbcl) { /* * Some devices (SQ555 come to mind) grab the IDENTIFY message * sent on selection, and decide to go into COMMAND OUT phase * rather than accepting the rest of the messages or rejecting * them. Handle these devices gracefully. */ case SBCL_PHASE_CMDOUT: hostdata->dsp = dsp + 2 /* two _words_ */; hostdata->dsp_changed = 1; printk ("scsi%d : target %d ignored SDTR and went into COMMAND OUT\n", host->host_no, cmd->cmd->device->id); cmd->flags &= ~CMD_FLAG_SDTR; action = ACTION_CONTINUE; break; case SBCL_PHASE_MSGIN: hostdata->dsp = hostdata->script + hostdata->E_msg_in / sizeof(u32); hostdata->dsp_changed = 1; action = ACTION_CONTINUE; break; default: where="select message out"; action = ACTION_ABORT_PRINT; } /* * Some SCSI devices will interpret a command as they read the bytes * off the SCSI bus, and may decide that the command is Bogus before * they've read the entire command off the bus. */ } else if (dsp == hostdata->script + hostdata->E_cmdout_cmdout / sizeof (u32)) { hostdata->dsp = hostdata->script + hostdata->E_data_transfer / sizeof (u32); hostdata->dsp_changed = 1; action = ACTION_CONTINUE; /* FIXME : we need to handle message reject, etc. within msg_respond. */ #ifdef notyet } else if (dsp == hostdata->script + hostdata->E_reply_message) { switch (sbcl) { /* Any other phase mismatches abort the currently executing command. */ #endif } else { where = "unknown location"; action = ACTION_ABORT_PRINT; } /* Flush DMA FIFO */ if (!hostdata->dstat_valid) { hostdata->dstat = NCR53c7x0_read8(DSTAT_REG); hostdata->dstat_valid = 1; } if (!(hostdata->dstat & DSTAT_DFE)) { /* Really need to check this out for 710 RGH */ NCR53c7x0_write8 (CTEST8_REG, CTEST8_10_CLF); while (NCR53c7x0_read8 (CTEST8_REG) & CTEST8_10_CLF); hostdata->dstat |= DSTAT_DFE; } switch (action) { case ACTION_ABORT_PRINT: printk("scsi%d : %s : unexpected phase %s.\n", host->host_no, where ? where : "unknown location", sbcl_to_phase(sbcl)); print_lots (host); /* Fall through to ACTION_ABORT */ case ACTION_ABORT: abort_connected (host); break; case ACTION_CONTINUE: break; } #if 0 if (hostdata->dsp_changed) { printk("scsi%d: new dsp 0x%p\n", host->host_no, hostdata->dsp); print_insn (host, hostdata->dsp, "", 1); } #endif } /* * Function : static void intr_bf (struct Scsi_Host *host, * struct NCR53c7x0_cmd *cmd) * * Purpose : handle BUS FAULT interrupts * * Inputs : host, cmd - host and NCR command causing the interrupt, cmd * may be NULL. */ static void intr_bf (struct Scsi_Host *host, struct NCR53c7x0_cmd *cmd) { NCR53c7x0_local_declare(); u32 *dsp, *next_dsp, /* Current dsp */ *dsa, dbc_dcmd; /* DCMD (high eight bits) + DBC */ char *reason = NULL; /* Default behavior is for a silent error, with a retry until we've exhausted retries. */ enum {MAYBE, ALWAYS, NEVER} retry = MAYBE; int report = 0; NCR53c7x0_local_setup(host); dbc_dcmd = NCR53c7x0_read32 (DBC_REG); next_dsp = bus_to_virt (NCR53c7x0_read32(DSP_REG)); dsp = next_dsp - NCR53c7x0_insn_size ((dbc_dcmd >> 24) & 0xff); /* FIXME - check chip type */ dsa = bus_to_virt (NCR53c7x0_read32(DSA_REG)); /* * Bus faults can be caused by either a Bad Address or * Target Abort. We should check the Received Target Abort * bit of the PCI status register and Master Abort Bit. * * - Master Abort bit indicates that no device claimed * the address with DEVSEL within five clocks * * - Target Abort bit indicates that a target claimed it, * but changed its mind once it saw the byte enables. * */ /* 53c710, not PCI system */ report = 1; reason = "Unknown"; #ifndef notyet report = 1; #endif if (report && reason) { printk(KERN_ALERT "scsi%d : BUS FAULT reason = %s\n", host->host_no, reason ? reason : "unknown"); print_lots (host); } #ifndef notyet retry = NEVER; #endif /* * TODO : we should attempt to recover from any spurious bus * faults. After X retries, we should figure that things are * sufficiently wedged, and call NCR53c7xx_reset. * * This code should only get executed once we've decided that we * cannot retry. */ if (retry == NEVER) { printk(KERN_ALERT " mail richard@sleepie.demon.co.uk\n"); FATAL (host); } } /* * Function : static void intr_dma (struct Scsi_Host *host, * struct NCR53c7x0_cmd *cmd) * * Purpose : handle all DMA interrupts, indicated by the setting * of the DIP bit in the ISTAT register. * * Inputs : host, cmd - host and NCR command causing the interrupt, cmd * may be NULL. */ static void intr_dma (struct Scsi_Host *host, struct NCR53c7x0_cmd *cmd) { NCR53c7x0_local_declare(); struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; unsigned char dstat; /* DSTAT */ u32 *dsp, *next_dsp, /* Current dsp */ *dsa, dbc_dcmd; /* DCMD (high eight bits) + DBC */ int tmp; unsigned long flags; NCR53c7x0_local_setup(host); if (!hostdata->dstat_valid) { hostdata->dstat = NCR53c7x0_read8(DSTAT_REG); hostdata->dstat_valid = 1; } dstat = hostdata->dstat; if (hostdata->options & OPTION_DEBUG_INTR) printk("scsi%d : DSTAT=0x%x\n", host->host_no, (int) dstat); dbc_dcmd = NCR53c7x0_read32 (DBC_REG); next_dsp = bus_to_virt(NCR53c7x0_read32(DSP_REG)); dsp = next_dsp - NCR53c7x0_insn_size ((dbc_dcmd >> 24) & 0xff); /* XXX - check chip type */ dsa = bus_to_virt(NCR53c7x0_read32(DSA_REG)); /* * DSTAT_ABRT is the aborted interrupt. This is set whenever the * SCSI chip is aborted. * * With NCR53c700 and NCR53c700-66 style chips, we should only * get this when the chip is currently running the accept * reselect/select code and we have set the abort bit in the * ISTAT register. * */ if (dstat & DSTAT_ABRT) { #if 0 /* XXX - add code here to deal with normal abort */ if ((hostdata->options & OPTION_700) && (hostdata->state == STATE_ABORTING)) { } else #endif { printk(KERN_ALERT "scsi%d : unexpected abort interrupt at\n" " ", host->host_no); print_insn (host, dsp, KERN_ALERT "s ", 1); FATAL (host); } } /* * DSTAT_SSI is the single step interrupt. Should be generated * whenever we have single stepped or are tracing. */ if (dstat & DSTAT_SSI) { if (hostdata->options & OPTION_DEBUG_TRACE) { /* Don't print instr. until we write DSP at end of intr function */ } else if (hostdata->options & OPTION_DEBUG_SINGLE) { print_insn (host, dsp, "s ", 0); local_irq_save(flags); /* XXX - should we do this, or can we get away with writing dsp? */ NCR53c7x0_write8 (DCNTL_REG, (NCR53c7x0_read8(DCNTL_REG) & ~DCNTL_SSM) | DCNTL_STD); local_irq_restore(flags); } else { printk(KERN_ALERT "scsi%d : unexpected single step interrupt at\n" " ", host->host_no); print_insn (host, dsp, KERN_ALERT "", 1); printk(KERN_ALERT " mail drew@PoohSticks.ORG\n"); FATAL (host); } } /* * DSTAT_IID / DSTAT_OPC (same bit, same meaning, only the name * is different) is generated whenever an illegal instruction is * encountered. * * XXX - we may want to emulate INTFLY here, so we can use * the same SCSI SCRIPT (tm) for NCR53c710 through NCR53c810 * chips. */ if (dstat & DSTAT_OPC) { /* * Ascertain if this IID interrupts occurred before or after a STO * interrupt. Since the interrupt handling code now leaves * DSP unmodified until _after_ all stacked interrupts have been * processed, reading the DSP returns the original DSP register. * This means that if dsp lies between the select code, and * message out following the selection code (where the IID interrupt * would have to have occurred by due to the implicit wait for REQ), * we have an IID interrupt resulting from a STO condition and * can ignore it. */ if (((dsp >= (hostdata->script + hostdata->E_select / sizeof(u32))) && (dsp <= (hostdata->script + hostdata->E_select_msgout / sizeof(u32) + 8))) || (hostdata->test_running == 2)) { if (hostdata->options & OPTION_DEBUG_INTR) printk ("scsi%d : ignoring DSTAT_IID for SSTAT_STO\n", host->host_no); if (hostdata->expecting_iid) { hostdata->expecting_iid = 0; hostdata->idle = 1; if (hostdata->test_running == 2) { hostdata->test_running = 0; hostdata->test_completed = 3; } else if (cmd) abnormal_finished (cmd, DID_BAD_TARGET << 16); } else { hostdata->expecting_sto = 1; } /* * We can't guarantee we'll be able to execute the WAIT DISCONNECT * instruction within the 3.4us of bus free and arbitration delay * that a target can RESELECT in and assert REQ after we've dropped * ACK. If this happens, we'll get an illegal instruction interrupt. * Doing away with the WAIT DISCONNECT instructions broke everything, * so instead I'll settle for moving one WAIT DISCONNECT a few * instructions closer to the CLEAR ACK before it to minimize the * chances of this happening, and handle it if it occurs anyway. * * Simply continue with what we were doing, and control should * be transferred to the schedule routine which will ultimately * pass control onto the reselection or selection (not yet) * code. */ } else if (dbc_dcmd == 0x48000000 && (NCR53c7x0_read8 (SBCL_REG) & SBCL_REQ)) { if (!(hostdata->options & OPTION_NO_PRINT_RACE)) { printk("scsi%d: REQ before WAIT DISCONNECT IID\n", host->host_no); hostdata->options |= OPTION_NO_PRINT_RACE; } } else { printk(KERN_ALERT "scsi%d : invalid instruction\n", host->host_no); print_lots (host); printk(KERN_ALERT " mail Richard@sleepie.demon.co.uk with ALL\n" " boot messages and diagnostic output\n"); FATAL (host); } } /* * DSTAT_BF are bus fault errors. DSTAT_800_BF is valid for 710 also. */ if (dstat & DSTAT_800_BF) { intr_bf (host, cmd); } /* * DSTAT_SIR interrupts are generated by the execution of * the INT instruction. Since the exact values available * are determined entirely by the SCSI script running, * and are local to a particular script, a unique handler * is called for each script. */ if (dstat & DSTAT_SIR) { if (hostdata->options & OPTION_DEBUG_INTR) printk ("scsi%d : DSTAT_SIR\n", host->host_no); switch ((tmp = hostdata->dstat_sir_intr (host, cmd))) { case SPECIFIC_INT_NOTHING: case SPECIFIC_INT_RESTART: break; case SPECIFIC_INT_ABORT: abort_connected(host); break; case SPECIFIC_INT_PANIC: printk(KERN_ALERT "scsi%d : failure at ", host->host_no); print_insn (host, dsp, KERN_ALERT "", 1); printk(KERN_ALERT " dstat_sir_intr() returned SPECIFIC_INT_PANIC\n"); FATAL (host); break; case SPECIFIC_INT_BREAK: intr_break (host, cmd); break; default: printk(KERN_ALERT "scsi%d : failure at ", host->host_no); print_insn (host, dsp, KERN_ALERT "", 1); printk(KERN_ALERT" dstat_sir_intr() returned unknown value %d\n", tmp); FATAL (host); } } } /* * Function : static int print_insn (struct Scsi_Host *host, * u32 *insn, int kernel) * * Purpose : print numeric representation of the instruction pointed * to by insn to the debugging or kernel message buffer * as appropriate. * * If desired, a user level program can interpret this * information. * * Inputs : host, insn - host, pointer to instruction, prefix - * string to prepend, kernel - use printk instead of debugging buffer. * * Returns : size, in u32s, of instruction printed. */ /* * FIXME: should change kernel parameter so that it takes an ENUM * specifying severity - either KERN_ALERT or KERN_PANIC so * all panic messages are output with the same severity. */ static int print_insn (struct Scsi_Host *host, const u32 *insn, const char *prefix, int kernel) { char buf[160], /* Temporary buffer and pointer. ICKY arbitrary length. */ *tmp; unsigned char dcmd; /* dcmd register for *insn */ int size; /* * Check to see if the instruction pointer is not bogus before * indirecting through it; avoiding red-zone at start of * memory. * * FIXME: icky magic needs to happen here on non-intel boxes which * don't have kernel memory mapped in like this. Might be reasonable * to use vverify()? */ if (virt_to_phys((void *)insn) < PAGE_SIZE || virt_to_phys((void *)(insn + 8)) > virt_to_phys(high_memory) || ((((dcmd = (insn[0] >> 24) & 0xff) & DCMD_TYPE_MMI) == DCMD_TYPE_MMI) && virt_to_phys((void *)(insn + 12)) > virt_to_phys(high_memory))) { size = 0; sprintf (buf, "%s%p: address out of range\n", prefix, insn); } else { /* * FIXME : (void *) cast in virt_to_bus should be unnecessary, because * it should take const void * as argument. */ #if !defined(CONFIG_MVME16x) && !defined(CONFIG_BVME6000) sprintf(buf, "%s0x%lx (virt 0x%p) : 0x%08x 0x%08x (virt 0x%p)", (prefix ? prefix : ""), virt_to_bus((void *) insn), insn, insn[0], insn[1], bus_to_virt (insn[1])); #else /* Remove virtual addresses to reduce output, as they are the same */ sprintf(buf, "%s0x%x (+%x) : 0x%08x 0x%08x", (prefix ? prefix : ""), (u32)insn, ((u32)insn - (u32)&(((struct NCR53c7x0_hostdata *)host->hostdata[0])->script))/4, insn[0], insn[1]); #endif tmp = buf + strlen(buf); if ((dcmd & DCMD_TYPE_MASK) == DCMD_TYPE_MMI) { #if !defined(CONFIG_MVME16x) && !defined(CONFIG_BVME6000) sprintf (tmp, " 0x%08x (virt 0x%p)\n", insn[2], bus_to_virt(insn[2])); #else /* Remove virtual addr to reduce output, as it is the same */ sprintf (tmp, " 0x%08x\n", insn[2]); #endif size = 3; } else { sprintf (tmp, "\n"); size = 2; } } if (kernel) printk ("%s", buf); #ifdef NCR_DEBUG else { size_t len = strlen(buf); debugger_kernel_write(host, buf, len); } #endif return size; } /* * Function : int NCR53c7xx_abort (Scsi_Cmnd *cmd) * * Purpose : Abort an errant SCSI command, doing all necessary * cleanup of the issue_queue, running_list, shared Linux/NCR * dsa issue and reconnect queues. * * Inputs : cmd - command to abort, code - entire result field * * Returns : 0 on success, -1 on failure. */ int NCR53c7xx_abort (Scsi_Cmnd *cmd) { NCR53c7x0_local_declare(); struct Scsi_Host *host = cmd->device->host; struct NCR53c7x0_hostdata *hostdata = host ? (struct NCR53c7x0_hostdata *) host->hostdata[0] : NULL; unsigned long flags; struct NCR53c7x0_cmd *curr, **prev; Scsi_Cmnd *me, **last; #if 0 static long cache_pid = -1; #endif if (!host) { printk ("Bogus SCSI command pid %ld; no host structure\n", cmd->pid); return SCSI_ABORT_ERROR; } else if (!hostdata) { printk ("Bogus SCSI host %d; no hostdata\n", host->host_no); return SCSI_ABORT_ERROR; } NCR53c7x0_local_setup(host); /* * CHECK : I don't think that reading ISTAT will unstack any interrupts, * since we need to write the INTF bit to clear it, and SCSI/DMA * interrupts don't clear until we read SSTAT/SIST and DSTAT registers. * * See that this is the case. Appears to be correct on the 710, at least. * * I suspect that several of our failures may be coming from a new fatal * interrupt (possibly due to a phase mismatch) happening after we've left * the interrupt handler, but before the PIC has had the interrupt condition * cleared. */ if (NCR53c7x0_read8(hostdata->istat) & (ISTAT_DIP|ISTAT_SIP)) { printk ("scsi%d : dropped interrupt for command %ld\n", host->host_no, cmd->pid); NCR53c7x0_intr (host->irq, NULL, NULL); return SCSI_ABORT_BUSY; } local_irq_save(flags); #if 0 if (cache_pid == cmd->pid) panic ("scsi%d : bloody fetus %d\n", host->host_no, cmd->pid); else cache_pid = cmd->pid; #endif /* * The command could be hiding in the issue_queue. This would be very * nice, as commands can't be moved from the high level driver's issue queue * into the shared queue until an interrupt routine is serviced, and this * moving is atomic. * * If this is the case, we don't have to worry about anything - we simply * pull the command out of the old queue, and call it aborted. */ for (me = (Scsi_Cmnd *) hostdata->issue_queue, last = (Scsi_Cmnd **) &(hostdata->issue_queue); me && me != cmd; last = (Scsi_Cmnd **)&(me->SCp.ptr), me = (Scsi_Cmnd *)me->SCp.ptr); if (me) { *last = (Scsi_Cmnd *) me->SCp.ptr; if (me->host_scribble) { ((struct NCR53c7x0_cmd *)me->host_scribble)->next = hostdata->free; hostdata->free = (struct NCR53c7x0_cmd *) me->host_scribble; me->host_scribble = NULL; } cmd->result = DID_ABORT << 16; cmd->scsi_done(cmd); printk ("scsi%d : found command %ld in Linux issue queue\n", host->host_no, me->pid); local_irq_restore(flags); run_process_issue_queue(); return SCSI_ABORT_SUCCESS; } /* * That failing, the command could be in our list of already executing * commands. If this is the case, drastic measures are called for. */ for (curr = (struct NCR53c7x0_cmd *) hostdata->running_list, prev = (struct NCR53c7x0_cmd **) &(hostdata->running_list); curr && curr->cmd != cmd; prev = (struct NCR53c7x0_cmd **) &(curr->next), curr = (struct NCR53c7x0_cmd *) curr->next); if (curr) { if ((curr->result & 0xff) != 0xff && (curr->result & 0xff00) != 0xff00) { cmd->result = curr->result; if (prev) *prev = (struct NCR53c7x0_cmd *) curr->next; curr->next = (struct NCR53c7x0_cmd *) hostdata->free; cmd->host_scribble = NULL; hostdata->free = curr; cmd->scsi_done(cmd); printk ("scsi%d : found finished command %ld in running list\n", host->host_no, cmd->pid); local_irq_restore(flags); return SCSI_ABORT_NOT_RUNNING; } else { printk ("scsi%d : DANGER : command running, can not abort.\n", cmd->device->host->host_no); local_irq_restore(flags); return SCSI_ABORT_BUSY; } } /* * And if we couldn't find it in any of our queues, it must have been * a dropped interrupt. */ curr = (struct NCR53c7x0_cmd *) cmd->host_scribble; if (curr) { curr->next = hostdata->free; hostdata->free = curr; cmd->host_scribble = NULL; } if (curr == NULL || ((curr->result & 0xff00) == 0xff00) || ((curr->result & 0xff) == 0xff)) { printk ("scsi%d : did this command ever run?\n", host->host_no); cmd->result = DID_ABORT << 16; } else { printk ("scsi%d : probably lost INTFLY, normal completion\n", host->host_no); cmd->result = curr->result; /* * FIXME : We need to add an additional flag which indicates if a * command was ever counted as BUSY, so if we end up here we can * decrement the busy count if and only if it is necessary. */ --hostdata->busy[cmd->device->id][cmd->device->lun]; } local_irq_restore(flags); cmd->scsi_done(cmd); /* * We need to run process_issue_queue since termination of this command * may allow another queued command to execute first? */ return SCSI_ABORT_NOT_RUNNING; } /* * Function : int NCR53c7xx_reset (Scsi_Cmnd *cmd) * * Purpose : perform a hard reset of the SCSI bus and NCR * chip. * * Inputs : cmd - command which caused the SCSI RESET * * Returns : 0 on success. */ int NCR53c7xx_reset (Scsi_Cmnd *cmd, unsigned int reset_flags) { NCR53c7x0_local_declare(); unsigned long flags; int found = 0; struct NCR53c7x0_cmd * c; Scsi_Cmnd *tmp; /* * When we call scsi_done(), it's going to wake up anything sleeping on the * resources which were in use by the aborted commands, and we'll start to * get new commands. * * We can't let this happen until after we've re-initialized the driver * structures, and can't reinitialize those structures until after we've * dealt with their contents. * * So, we need to find all of the commands which were running, stick * them on a linked list of completed commands (we'll use the host_scribble * pointer), do our reinitialization, and then call the done function for * each command. */ Scsi_Cmnd *nuke_list = NULL; struct Scsi_Host *host = cmd->device->host; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; NCR53c7x0_local_setup(host); local_irq_save(flags); ncr_halt (host); print_lots (host); dump_events (host, 30); ncr_scsi_reset (host); for (tmp = nuke_list = return_outstanding_commands (host, 1 /* free */, 0 /* issue */ ); tmp; tmp = (Scsi_Cmnd *) tmp->SCp.buffer) if (tmp == cmd) { found = 1; break; } /* * If we didn't find the command which caused this reset in our running * list, then we've lost it. See that it terminates normally anyway. */ if (!found) { c = (struct NCR53c7x0_cmd *) cmd->host_scribble; if (c) { cmd->host_scribble = NULL; c->next = hostdata->free; hostdata->free = c; } else printk ("scsi%d: lost command %ld\n", host->host_no, cmd->pid); cmd->SCp.buffer = (struct scatterlist *) nuke_list; nuke_list = cmd; } NCR53c7x0_driver_init (host); hostdata->soft_reset (host); if (hostdata->resets == 0) disable(host); else if (hostdata->resets != -1) --hostdata->resets; local_irq_restore(flags); for (; nuke_list; nuke_list = tmp) { tmp = (Scsi_Cmnd *) nuke_list->SCp.buffer; nuke_list->result = DID_RESET << 16; nuke_list->scsi_done (nuke_list); } local_irq_restore(flags); return SCSI_RESET_SUCCESS; } /* * The NCR SDMS bios follows Annex A of the SCSI-CAM draft, and * therefore shares the scsicam_bios_param function. */ /* * Function : int insn_to_offset (Scsi_Cmnd *cmd, u32 *insn) * * Purpose : convert instructions stored at NCR pointer into data * pointer offset. * * Inputs : cmd - SCSI command; insn - pointer to instruction. Either current * DSP, or saved data pointer. * * Returns : offset on success, -1 on failure. */ static int insn_to_offset (Scsi_Cmnd *cmd, u32 *insn) { struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) cmd->device->host->hostdata[0]; struct NCR53c7x0_cmd *ncmd = (struct NCR53c7x0_cmd *) cmd->host_scribble; int offset = 0, buffers; struct scatterlist *segment; char *ptr; int found = 0; /* * With the current code implementation, if the insn is inside dynamically * generated code, the data pointer will be the instruction preceding * the next transfer segment. */ if (!check_address ((unsigned long) ncmd, sizeof (struct NCR53c7x0_cmd)) && ((insn >= ncmd->data_transfer_start && insn < ncmd->data_transfer_end) || (insn >= ncmd->residual && insn < (ncmd->residual + sizeof(ncmd->residual))))) { ptr = bus_to_virt(insn[3]); if ((buffers = cmd->use_sg)) { for (offset = 0, segment = (struct scatterlist *) cmd->buffer; buffers && !((found = ((ptr >= (char *)page_address(segment->page)+segment->offset) && (ptr < ((char *)page_address(segment->page)+segment->offset+segment->length))))); --buffers, offset += segment->length, ++segment) #if 0 printk("scsi%d: comparing 0x%p to 0x%p\n", cmd->device->host->host_no, saved, page_address(segment->page+segment->offset); #else ; #endif offset += ptr - ((char *)page_address(segment->page)+segment->offset); } else { found = 1; offset = ptr - (char *) (cmd->request_buffer); } } else if ((insn >= hostdata->script + hostdata->E_data_transfer / sizeof(u32)) && (insn <= hostdata->script + hostdata->E_end_data_transfer / sizeof(u32))) { found = 1; offset = 0; } return found ? offset : -1; } /* * Function : void print_progress (Scsi_Cmnd *cmd) * * Purpose : print the current location of the saved data pointer * * Inputs : cmd - command we are interested in * */ static void print_progress (Scsi_Cmnd *cmd) { NCR53c7x0_local_declare(); struct NCR53c7x0_cmd *ncmd = (struct NCR53c7x0_cmd *) cmd->host_scribble; int offset, i; char *where; u32 *ptr; NCR53c7x0_local_setup (cmd->device->host); if (check_address ((unsigned long) ncmd,sizeof (struct NCR53c7x0_cmd)) == 0) { printk("\nNCR53c7x0_cmd fields:\n"); printk(" bounce.len=0x%x, addr=0x%0x, buf[]=0x%02x %02x %02x %02x\n", ncmd->bounce.len, ncmd->bounce.addr, ncmd->bounce.buf[0], ncmd->bounce.buf[1], ncmd->bounce.buf[2], ncmd->bounce.buf[3]); printk(" result=%04x, cdb[0]=0x%02x\n", ncmd->result, ncmd->cmnd[0]); } for (i = 0; i < 2; ++i) { if (check_address ((unsigned long) ncmd, sizeof (struct NCR53c7x0_cmd)) == -1) continue; if (!i) { where = "saved"; ptr = bus_to_virt(ncmd->saved_data_pointer); } else { where = "active"; ptr = bus_to_virt (NCR53c7x0_read32 (DSP_REG) - NCR53c7x0_insn_size (NCR53c7x0_read8 (DCMD_REG)) * sizeof(u32)); } offset = insn_to_offset (cmd, ptr); if (offset != -1) printk ("scsi%d : %s data pointer at offset %d\n", cmd->device->host->host_no, where, offset); else { int size; printk ("scsi%d : can't determine %s data pointer offset\n", cmd->device->host->host_no, where); if (ncmd) { size = print_insn (cmd->device->host, bus_to_virt(ncmd->saved_data_pointer), "", 1); print_insn (cmd->device->host, bus_to_virt(ncmd->saved_data_pointer) + size * sizeof(u32), "", 1); } } } } static void print_dsa (struct Scsi_Host *host, u32 *dsa, const char *prefix) { struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; int i, len; char *ptr; Scsi_Cmnd *cmd; if (check_address ((unsigned long) dsa, hostdata->dsa_end - hostdata->dsa_start) == -1) { printk("scsi%d : bad dsa virt 0x%p\n", host->host_no, dsa); return; } printk("%sscsi%d : dsa at phys 0x%lx (virt 0x%p)\n" " + %d : dsa_msgout length = %u, data = 0x%x (virt 0x%p)\n" , prefix ? prefix : "", host->host_no, virt_to_bus (dsa), dsa, hostdata->dsa_msgout, dsa[hostdata->dsa_msgout / sizeof(u32)], dsa[hostdata->dsa_msgout / sizeof(u32) + 1], bus_to_virt (dsa[hostdata->dsa_msgout / sizeof(u32) + 1])); /* * Only print messages if they're sane in length so we don't * blow the kernel printk buffer on something which won't buy us * anything. */ if (dsa[hostdata->dsa_msgout / sizeof(u32)] < sizeof (hostdata->free->select)) for (i = dsa[hostdata->dsa_msgout / sizeof(u32)], ptr = bus_to_virt (dsa[hostdata->dsa_msgout / sizeof(u32) + 1]); i > 0 && !check_address ((unsigned long) ptr, 1); ptr += len, i -= len) { printk(" "); len = spi_print_msg(ptr); printk("\n"); if (!len) break; } printk(" + %d : select_indirect = 0x%x\n", hostdata->dsa_select, dsa[hostdata->dsa_select / sizeof(u32)]); cmd = (Scsi_Cmnd *) bus_to_virt(dsa[hostdata->dsa_cmnd / sizeof(u32)]); printk(" + %d : dsa_cmnd = 0x%x ", hostdata->dsa_cmnd, (u32) virt_to_bus(cmd)); /* XXX Maybe we should access cmd->host_scribble->result here. RGH */ if (cmd) { printk(" result = 0x%x, target = %d, lun = %d, cmd = ", cmd->result, cmd->device->id, cmd->device->lun); __scsi_print_command(cmd->cmnd); } else printk("\n"); printk(" + %d : dsa_next = 0x%x\n", hostdata->dsa_next, dsa[hostdata->dsa_next / sizeof(u32)]); if (cmd) { printk("scsi%d target %d : sxfer_sanity = 0x%x, scntl3_sanity = 0x%x\n" " script : ", host->host_no, cmd->device->id, hostdata->sync[cmd->device->id].sxfer_sanity, hostdata->sync[cmd->device->id].scntl3_sanity); for (i = 0; i < (sizeof(hostdata->sync[cmd->device->id].script) / 4); ++i) printk ("0x%x ", hostdata->sync[cmd->device->id].script[i]); printk ("\n"); print_progress (cmd); } } /* * Function : void print_queues (Scsi_Host *host) * * Purpose : print the contents of the NCR issue and reconnect queues * * Inputs : host - SCSI host we are interested in * */ static void print_queues (struct Scsi_Host *host) { struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; u32 *dsa, *next_dsa; volatile u32 *ncrcurrent; int left; Scsi_Cmnd *cmd, *next_cmd; unsigned long flags; printk ("scsi%d : issue queue\n", host->host_no); for (left = host->can_queue, cmd = (Scsi_Cmnd *) hostdata->issue_queue; left >= 0 && cmd; cmd = next_cmd) { next_cmd = (Scsi_Cmnd *) cmd->SCp.ptr; local_irq_save(flags); if (cmd->host_scribble) { if (check_address ((unsigned long) (cmd->host_scribble), sizeof (cmd->host_scribble)) == -1) printk ("scsi%d: scsi pid %ld bad pointer to NCR53c7x0_cmd\n", host->host_no, cmd->pid); /* print_dsa does sanity check on address, no need to check */ else print_dsa (host, ((struct NCR53c7x0_cmd *) cmd->host_scribble) -> dsa, ""); } else printk ("scsi%d : scsi pid %ld for target %d lun %d has no NCR53c7x0_cmd\n", host->host_no, cmd->pid, cmd->device->id, cmd->device->lun); local_irq_restore(flags); } if (left <= 0) { printk ("scsi%d : loop detected in issue queue\n", host->host_no); } /* * Traverse the NCR reconnect and start DSA structures, printing out * each element until we hit the end or detect a loop. Currently, * the reconnect structure is a linked list; and the start structure * is an array. Eventually, the reconnect structure will become a * list as well, since this simplifies the code. */ printk ("scsi%d : schedule dsa array :\n", host->host_no); for (left = host->can_queue, ncrcurrent = hostdata->schedule; left > 0; ncrcurrent += 2, --left) if (ncrcurrent[0] != hostdata->NOP_insn) /* FIXME : convert pointer to dsa_begin to pointer to dsa. */ print_dsa (host, bus_to_virt (ncrcurrent[1] - (hostdata->E_dsa_code_begin - hostdata->E_dsa_code_template)), ""); printk ("scsi%d : end schedule dsa array\n", host->host_no); printk ("scsi%d : reconnect_dsa_head :\n", host->host_no); for (left = host->can_queue, dsa = bus_to_virt (hostdata->reconnect_dsa_head); left >= 0 && dsa; dsa = next_dsa) { local_irq_save(flags); if (check_address ((unsigned long) dsa, sizeof(dsa)) == -1) { printk ("scsi%d: bad DSA pointer 0x%p", host->host_no, dsa); next_dsa = NULL; } else { next_dsa = bus_to_virt(dsa[hostdata->dsa_next / sizeof(u32)]); print_dsa (host, dsa, ""); } local_irq_restore(flags); } printk ("scsi%d : end reconnect_dsa_head\n", host->host_no); if (left < 0) printk("scsi%d: possible loop in ncr reconnect list\n", host->host_no); } static void print_lots (struct Scsi_Host *host) { NCR53c7x0_local_declare(); struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; u32 *dsp_next, *dsp, *dsa, dbc_dcmd; unsigned char dcmd, sbcl; int i, size; NCR53c7x0_local_setup(host); if ((dsp_next = bus_to_virt(NCR53c7x0_read32 (DSP_REG)))) { dbc_dcmd = NCR53c7x0_read32(DBC_REG); dcmd = (dbc_dcmd & 0xff000000) >> 24; dsp = dsp_next - NCR53c7x0_insn_size(dcmd); dsa = bus_to_virt(NCR53c7x0_read32(DSA_REG)); sbcl = NCR53c7x0_read8 (SBCL_REG); /* * For the 53c710, the following will report value 0 for SCNTL3 * and STEST0 - we don't have these registers. */ printk ("scsi%d : DCMD|DBC=0x%x, DNAD=0x%x (virt 0x%p)\n" " DSA=0x%lx (virt 0x%p)\n" " DSPS=0x%x, TEMP=0x%x (virt 0x%p), DMODE=0x%x\n" " SXFER=0x%x, SCNTL3=0x%x\n" " %s%s%sphase=%s, %d bytes in SCSI FIFO\n" " SCRATCH=0x%x, saved2_dsa=0x%0lx\n", host->host_no, dbc_dcmd, NCR53c7x0_read32(DNAD_REG), bus_to_virt(NCR53c7x0_read32(DNAD_REG)), virt_to_bus(dsa), dsa, NCR53c7x0_read32(DSPS_REG), NCR53c7x0_read32(TEMP_REG), bus_to_virt (NCR53c7x0_read32(TEMP_REG)), (int) NCR53c7x0_read8(hostdata->dmode), (int) NCR53c7x0_read8(SXFER_REG), ((hostdata->chip / 100) == 8) ? (int) NCR53c7x0_read8(SCNTL3_REG_800) : 0, (sbcl & SBCL_BSY) ? "BSY " : "", (sbcl & SBCL_SEL) ? "SEL " : "", (sbcl & SBCL_REQ) ? "REQ " : "", sstat2_to_phase(NCR53c7x0_read8 (((hostdata->chip / 100) == 8) ? SSTAT1_REG : SSTAT2_REG)), (NCR53c7x0_read8 ((hostdata->chip / 100) == 8 ? SSTAT1_REG : SSTAT2_REG) & SSTAT2_FF_MASK) >> SSTAT2_FF_SHIFT, ((hostdata->chip / 100) == 8) ? NCR53c7x0_read8 (STEST0_REG_800) : NCR53c7x0_read32(SCRATCHA_REG_800), hostdata->saved2_dsa); printk ("scsi%d : DSP 0x%lx (virt 0x%p) ->\n", host->host_no, virt_to_bus(dsp), dsp); for (i = 6; i > 0; --i, dsp += size) size = print_insn (host, dsp, "", 1); if (NCR53c7x0_read8 (SCNTL1_REG) & SCNTL1_CON) { if ((hostdata->chip / 100) == 8) printk ("scsi%d : connected (SDID=0x%x, SSID=0x%x)\n", host->host_no, NCR53c7x0_read8 (SDID_REG_800), NCR53c7x0_read8 (SSID_REG_800)); else printk ("scsi%d : connected (SDID=0x%x)\n", host->host_no, NCR53c7x0_read8 (SDID_REG_700)); print_dsa (host, dsa, ""); } #if 1 print_queues (host); #endif } } /* * Function : static int shutdown (struct Scsi_Host *host) * * Purpose : does a clean (we hope) shutdown of the NCR SCSI * chip. Use prior to dumping core, unloading the NCR driver, * * Returns : 0 on success */ static int shutdown (struct Scsi_Host *host) { NCR53c7x0_local_declare(); unsigned long flags; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; NCR53c7x0_local_setup(host); local_irq_save(flags); /* Get in a state where we can reset the SCSI bus */ ncr_halt (host); ncr_scsi_reset (host); hostdata->soft_reset(host); disable (host); local_irq_restore(flags); return 0; } /* * Function : void ncr_scsi_reset (struct Scsi_Host *host) * * Purpose : reset the SCSI bus. */ static void ncr_scsi_reset (struct Scsi_Host *host) { NCR53c7x0_local_declare(); unsigned long flags; NCR53c7x0_local_setup(host); local_irq_save(flags); NCR53c7x0_write8(SCNTL1_REG, SCNTL1_RST); udelay(25); /* Minimum amount of time to assert RST */ NCR53c7x0_write8(SCNTL1_REG, 0); local_irq_restore(flags); } /* * Function : void hard_reset (struct Scsi_Host *host) * */ static void hard_reset (struct Scsi_Host *host) { struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; unsigned long flags; local_irq_save(flags); ncr_scsi_reset(host); NCR53c7x0_driver_init (host); if (hostdata->soft_reset) hostdata->soft_reset (host); local_irq_restore(flags); } /* * Function : Scsi_Cmnd *return_outstanding_commands (struct Scsi_Host *host, * int free, int issue) * * Purpose : return a linked list (using the SCp.buffer field as next, * so we don't perturb hostdata. We don't use a field of the * NCR53c7x0_cmd structure since we may not have allocated one * for the command causing the reset.) of Scsi_Cmnd structures that * had propagated below the Linux issue queue level. If free is set, * free the NCR53c7x0_cmd structures which are associated with * the Scsi_Cmnd structures, and clean up any internal * NCR lists that the commands were on. If issue is set, * also return commands in the issue queue. * * Returns : linked list of commands * * NOTE : the caller should insure that the NCR chip is halted * if the free flag is set. */ static Scsi_Cmnd * return_outstanding_commands (struct Scsi_Host *host, int free, int issue) { struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; struct NCR53c7x0_cmd *c; int i; u32 *ncrcurrent; Scsi_Cmnd *list = NULL, *tmp; for (c = (struct NCR53c7x0_cmd *) hostdata->running_list; c; c = (struct NCR53c7x0_cmd *) c->next) { if (c->cmd->SCp.buffer) { printk ("scsi%d : loop detected in running list!\n", host->host_no); break; } else { printk ("Duh? Bad things happening in the NCR driver\n"); break; } c->cmd->SCp.buffer = (struct scatterlist *) list; list = c->cmd; if (free) { c->next = hostdata->free; hostdata->free = c; } } if (free) { for (i = 0, ncrcurrent = (u32 *) hostdata->schedule; i < host->can_queue; ++i, ncrcurrent += 2) { ncrcurrent[0] = hostdata->NOP_insn; ncrcurrent[1] = 0xdeadbeef; } hostdata->ncrcurrent = NULL; } if (issue) { for (tmp = (Scsi_Cmnd *) hostdata->issue_queue; tmp; tmp = tmp->next) { if (tmp->SCp.buffer) { printk ("scsi%d : loop detected in issue queue!\n", host->host_no); break; } tmp->SCp.buffer = (struct scatterlist *) list; list = tmp; } if (free) hostdata->issue_queue = NULL; } return list; } /* * Function : static int disable (struct Scsi_Host *host) * * Purpose : disables the given NCR host, causing all commands * to return a driver error. Call this so we can unload the * module during development and try again. Eventually, * we should be able to find clean workarounds for these * problems. * * Inputs : host - hostadapter to twiddle * * Returns : 0 on success. */ static int disable (struct Scsi_Host *host) { struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; unsigned long flags; Scsi_Cmnd *nuke_list, *tmp; local_irq_save(flags); if (hostdata->state != STATE_HALTED) ncr_halt (host); nuke_list = return_outstanding_commands (host, 1 /* free */, 1 /* issue */); hard_reset (host); hostdata->state = STATE_DISABLED; local_irq_restore(flags); printk ("scsi%d : nuking commands\n", host->host_no); for (; nuke_list; nuke_list = tmp) { tmp = (Scsi_Cmnd *) nuke_list->SCp.buffer; nuke_list->result = DID_ERROR << 16; nuke_list->scsi_done(nuke_list); } printk ("scsi%d : done. \n", host->host_no); printk (KERN_ALERT "scsi%d : disabled. Unload and reload\n", host->host_no); return 0; } /* * Function : static int ncr_halt (struct Scsi_Host *host) * * Purpose : halts the SCSI SCRIPTS(tm) processor on the NCR chip * * Inputs : host - SCSI chip to halt * * Returns : 0 on success */ static int ncr_halt (struct Scsi_Host *host) { NCR53c7x0_local_declare(); unsigned long flags; unsigned char istat, tmp; struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; int stage; NCR53c7x0_local_setup(host); local_irq_save(flags); /* Stage 0 : eat all interrupts Stage 1 : set ABORT Stage 2 : eat all but abort interrupts Stage 3 : eat all interrupts */ for (stage = 0;;) { if (stage == 1) { NCR53c7x0_write8(hostdata->istat, ISTAT_ABRT); ++stage; } istat = NCR53c7x0_read8 (hostdata->istat); if (istat & ISTAT_SIP) { tmp = NCR53c7x0_read8(SSTAT0_REG); } else if (istat & ISTAT_DIP) { tmp = NCR53c7x0_read8(DSTAT_REG); if (stage == 2) { if (tmp & DSTAT_ABRT) { NCR53c7x0_write8(hostdata->istat, 0); ++stage; } else { printk(KERN_ALERT "scsi%d : could not halt NCR chip\n", host->host_no); disable (host); } } } if (!(istat & (ISTAT_SIP|ISTAT_DIP))) { if (stage == 0) ++stage; else if (stage == 3) break; } } hostdata->state = STATE_HALTED; local_irq_restore(flags); #if 0 print_lots (host); #endif return 0; } /* * Function: event_name (int event) * * Purpose: map event enum into user-readable strings. */ static const char * event_name (int event) { switch (event) { case EVENT_NONE: return "none"; case EVENT_ISSUE_QUEUE: return "to issue queue"; case EVENT_START_QUEUE: return "to start queue"; case EVENT_SELECT: return "selected"; case EVENT_DISCONNECT: return "disconnected"; case EVENT_RESELECT: return "reselected"; case EVENT_COMPLETE: return "completed"; case EVENT_IDLE: return "idle"; case EVENT_SELECT_FAILED: return "select failed"; case EVENT_BEFORE_SELECT: return "before select"; case EVENT_RESELECT_FAILED: return "reselect failed"; default: return "unknown"; } } /* * Function : void dump_events (struct Scsi_Host *host, count) * * Purpose : print last count events which have occurred. */ static void dump_events (struct Scsi_Host *host, int count) { struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; struct NCR53c7x0_event event; int i; unsigned long flags; if (hostdata->events) { if (count > hostdata->event_size) count = hostdata->event_size; for (i = hostdata->event_index; count > 0; i = (i ? i - 1 : hostdata->event_size -1), --count) { /* * By copying the event we're currently examining with interrupts * disabled, we can do multiple printk(), etc. operations and * still be guaranteed that they're happening on the same * event structure. */ local_irq_save(flags); #if 0 event = hostdata->events[i]; #else memcpy ((void *) &event, (void *) &(hostdata->events[i]), sizeof(event)); #endif local_irq_restore(flags); printk ("scsi%d : %s event %d at %ld secs %ld usecs target %d lun %d\n", host->host_no, event_name (event.event), count, (long) event.time.tv_sec, (long) event.time.tv_usec, event.target, event.lun); if (event.dsa) printk (" event for dsa 0x%lx (virt 0x%p)\n", virt_to_bus(event.dsa), event.dsa); if (event.pid != -1) { printk (" event for pid %ld ", event.pid); __scsi_print_command (event.cmnd); } } } } /* * Function: check_address * * Purpose: Check to see if a possibly corrupt pointer will fault the * kernel. * * Inputs: addr - address; size - size of area * * Returns: 0 if area is OK, -1 on error. * * NOTES: should be implemented in terms of vverify on kernels * that have it. */ static int check_address (unsigned long addr, int size) { return (virt_to_phys((void *)addr) < PAGE_SIZE || virt_to_phys((void *)(addr + size)) > virt_to_phys(high_memory) ? -1 : 0); } #ifdef MODULE int NCR53c7x0_release(struct Scsi_Host *host) { struct NCR53c7x0_hostdata *hostdata = (struct NCR53c7x0_hostdata *) host->hostdata[0]; struct NCR53c7x0_cmd *cmd, *tmp; shutdown (host); if (host->irq != SCSI_IRQ_NONE) { int irq_count; struct Scsi_Host *tmp; for (irq_count = 0, tmp = first_host; tmp; tmp = tmp->next) if (tmp->hostt == the_template && tmp->irq == host->irq) ++irq_count; if (irq_count == 1) free_irq(host->irq, NULL); } if (host->dma_channel != DMA_NONE) free_dma(host->dma_channel); if (host->io_port) release_region(host->io_port, host->n_io_port); for (cmd = (struct NCR53c7x0_cmd *) hostdata->free; cmd; cmd = tmp, --hostdata->num_cmds) { tmp = (struct NCR53c7x0_cmd *) cmd->next; /* * If we're going to loop, try to stop it to get a more accurate * count of the leaked commands. */ cmd->next = NULL; if (cmd->free) cmd->free ((void *) cmd->real, cmd->size); } if (hostdata->num_cmds) printk ("scsi%d : leaked %d NCR53c7x0_cmd structures\n", host->host_no, hostdata->num_cmds); vfree(hostdata->events); /* XXX This assumes default cache mode to be IOMAP_FULL_CACHING, which * XXX may be invalid (CONFIG_060_WRITETHROUGH) */ kernel_set_cachemode((void *)hostdata, 8192, IOMAP_FULL_CACHING); free_pages ((u32)hostdata, 1); return 1; } #endif /* def MODULE */