2005-04-17 02:20:36 +04:00
|
|
|
/*
|
|
|
|
* Sun3 SCSI stuff by Erik Verbruggen (erik@bigmama.xtdnet.nl)
|
|
|
|
*
|
|
|
|
* Sun3 DMA routines added by Sam Creasey (sammy@sammy.net)
|
|
|
|
*
|
2014-03-18 04:42:24 +04:00
|
|
|
* VME support added by Sam Creasey
|
|
|
|
*
|
|
|
|
* TODO: modify this driver to support multiple Sun3 SCSI VME boards
|
|
|
|
*
|
2005-04-17 02:20:36 +04:00
|
|
|
* Adapted from mac_scsinew.c:
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
* Generic Macintosh NCR5380 driver
|
|
|
|
*
|
|
|
|
* Copyright 1998, Michael Schmitz <mschmitz@lbl.gov>
|
|
|
|
*
|
|
|
|
* derived in part from:
|
|
|
|
*/
|
|
|
|
/*
|
|
|
|
* Generic Generic NCR5380 driver
|
|
|
|
*
|
|
|
|
* Copyright 1995, Russell King
|
|
|
|
*
|
|
|
|
* ALPHA RELEASE 1.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <linux/types.h>
|
|
|
|
#include <linux/stddef.h>
|
|
|
|
#include <linux/ctype.h>
|
|
|
|
#include <linux/delay.h>
|
|
|
|
|
|
|
|
#include <linux/module.h>
|
|
|
|
#include <linux/signal.h>
|
|
|
|
#include <linux/ioport.h>
|
|
|
|
#include <linux/init.h>
|
|
|
|
#include <linux/blkdev.h>
|
|
|
|
|
|
|
|
#include <asm/io.h>
|
|
|
|
|
|
|
|
#include <asm/sun3ints.h>
|
|
|
|
#include <asm/dvma.h>
|
|
|
|
#include <asm/idprom.h>
|
|
|
|
#include <asm/machines.h>
|
|
|
|
|
|
|
|
/* dma on! */
|
|
|
|
#define REAL_DMA
|
|
|
|
|
|
|
|
#include "scsi.h"
|
|
|
|
#include <scsi/scsi_host.h>
|
|
|
|
#include "sun3_scsi.h"
|
2014-03-18 04:42:23 +04:00
|
|
|
#include "NCR5380.h"
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
extern int sun3_map_test(unsigned long, char *);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/*#define RESET_BOOT */
|
|
|
|
/* #define SUPPORT_TAGS */
|
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
#define ENABLE_IRQ()
|
|
|
|
#else
|
2005-04-17 02:20:36 +04:00
|
|
|
#define ENABLE_IRQ() enable_irq( IRQ_SUN3_SCSI );
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 17:55:46 +04:00
|
|
|
static irqreturn_t scsi_sun3_intr(int irq, void *dummy);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
static int setup_can_queue = -1;
|
|
|
|
module_param(setup_can_queue, int, 0);
|
|
|
|
static int setup_cmd_per_lun = -1;
|
|
|
|
module_param(setup_cmd_per_lun, int, 0);
|
|
|
|
static int setup_sg_tablesize = -1;
|
|
|
|
module_param(setup_sg_tablesize, int, 0);
|
|
|
|
#ifdef SUPPORT_TAGS
|
|
|
|
static int setup_use_tagged_queuing = -1;
|
|
|
|
module_param(setup_use_tagged_queuing, int, 0);
|
|
|
|
#endif
|
|
|
|
static int setup_hostid = -1;
|
|
|
|
module_param(setup_hostid, int, 0);
|
|
|
|
|
2006-10-03 21:51:59 +04:00
|
|
|
static struct scsi_cmnd *sun3_dma_setup_done = NULL;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2014-05-02 12:43:01 +04:00
|
|
|
#define RESET_RUN_DONE
|
|
|
|
|
2005-04-17 02:20:36 +04:00
|
|
|
#define AFTER_RESET_DELAY (HZ/2)
|
|
|
|
|
|
|
|
/* ms to wait after hitting dma regs */
|
|
|
|
#define SUN3_DMA_DELAY 10
|
|
|
|
|
|
|
|
/* dvma buffer to allocate -- 32k should hopefully be more than sufficient */
|
|
|
|
#define SUN3_DVMA_BUFSIZE 0xe000
|
|
|
|
|
|
|
|
/* minimum number of bytes to do dma on */
|
|
|
|
#define SUN3_DMA_MINSIZE 128
|
|
|
|
|
|
|
|
static volatile unsigned char *sun3_scsi_regp;
|
|
|
|
static volatile struct sun3_dma_regs *dregs;
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifndef SUN3_SCSI_VME
|
2005-04-17 02:20:36 +04:00
|
|
|
static struct sun3_udc_regs *udc_regs = NULL;
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
static unsigned char *sun3_dma_orig_addr = NULL;
|
|
|
|
static unsigned long sun3_dma_orig_count = 0;
|
|
|
|
static int sun3_dma_active = 0;
|
|
|
|
static unsigned long last_residual = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* NCR 5380 register access functions
|
|
|
|
*/
|
|
|
|
|
|
|
|
static inline unsigned char sun3scsi_read(int reg)
|
|
|
|
{
|
|
|
|
return( sun3_scsi_regp[reg] );
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void sun3scsi_write(int reg, int value)
|
|
|
|
{
|
|
|
|
sun3_scsi_regp[reg] = value;
|
|
|
|
}
|
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifndef SUN3_SCSI_VME
|
2005-04-17 02:20:36 +04:00
|
|
|
/* dma controller register access functions */
|
|
|
|
|
|
|
|
static inline unsigned short sun3_udc_read(unsigned char reg)
|
|
|
|
{
|
|
|
|
unsigned short ret;
|
|
|
|
|
|
|
|
dregs->udc_addr = UDC_CSR;
|
|
|
|
udelay(SUN3_DMA_DELAY);
|
|
|
|
ret = dregs->udc_data;
|
|
|
|
udelay(SUN3_DMA_DELAY);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void sun3_udc_write(unsigned short val, unsigned char reg)
|
|
|
|
{
|
|
|
|
dregs->udc_addr = reg;
|
|
|
|
udelay(SUN3_DMA_DELAY);
|
|
|
|
dregs->udc_data = val;
|
|
|
|
udelay(SUN3_DMA_DELAY);
|
|
|
|
}
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* XXX: status debug
|
|
|
|
*/
|
|
|
|
static struct Scsi_Host *default_instance;
|
|
|
|
|
|
|
|
/*
|
2005-10-31 20:31:40 +03:00
|
|
|
* Function : int sun3scsi_detect(struct scsi_host_template * tpnt)
|
2005-04-17 02:20:36 +04:00
|
|
|
*
|
|
|
|
* Purpose : initializes mac NCR5380 driver based on the
|
|
|
|
* command line / compile time port and irq definitions.
|
|
|
|
*
|
|
|
|
* Inputs : tpnt - template for this SCSI adapter.
|
|
|
|
*
|
|
|
|
* Returns : 1 if a host adapter was found, 0 if not.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
static int __init sun3scsi_detect(struct scsi_host_template *tpnt)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2014-03-18 04:42:24 +04:00
|
|
|
unsigned long ioaddr, irq;
|
2005-04-17 02:20:36 +04:00
|
|
|
static int called = 0;
|
|
|
|
struct Scsi_Host *instance;
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
int i;
|
|
|
|
unsigned long addrs[3] = { IOBASE_SUN3_VMESCSI,
|
|
|
|
IOBASE_SUN3_VMESCSI + 0x4000,
|
|
|
|
0 };
|
|
|
|
unsigned long vecs[3] = { SUN3_VEC_VMESCSI0,
|
|
|
|
SUN3_VEC_VMESCSI1,
|
|
|
|
0 };
|
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/* check that this machine has an onboard 5380 */
|
|
|
|
switch(idprom->id_machtype) {
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
case SM_SUN3|SM_3_160:
|
|
|
|
case SM_SUN3|SM_3_260:
|
|
|
|
break;
|
|
|
|
#else
|
2005-04-17 02:20:36 +04:00
|
|
|
case SM_SUN3|SM_3_50:
|
|
|
|
case SM_SUN3|SM_3_60:
|
|
|
|
break;
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(called)
|
|
|
|
return 0;
|
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
tpnt->proc_name = "Sun3 5380 VME SCSI";
|
|
|
|
#else
|
2005-04-17 02:20:36 +04:00
|
|
|
tpnt->proc_name = "Sun3 5380 SCSI";
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/* setup variables */
|
|
|
|
tpnt->can_queue =
|
|
|
|
(setup_can_queue > 0) ? setup_can_queue : CAN_QUEUE;
|
|
|
|
tpnt->cmd_per_lun =
|
|
|
|
(setup_cmd_per_lun > 0) ? setup_cmd_per_lun : CMD_PER_LUN;
|
|
|
|
tpnt->sg_tablesize =
|
|
|
|
(setup_sg_tablesize >= 0) ? setup_sg_tablesize : SG_TABLESIZE;
|
|
|
|
|
|
|
|
if (setup_hostid >= 0)
|
|
|
|
tpnt->this_id = setup_hostid;
|
|
|
|
else {
|
|
|
|
/* use 7 as default */
|
|
|
|
tpnt->this_id = 7;
|
|
|
|
}
|
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
ioaddr = 0;
|
|
|
|
for (i = 0; addrs[i] != 0; i++) {
|
|
|
|
unsigned char x;
|
|
|
|
|
|
|
|
ioaddr = (unsigned long)sun3_ioremap(addrs[i], PAGE_SIZE,
|
|
|
|
SUN3_PAGE_TYPE_VME16);
|
|
|
|
irq = vecs[i];
|
|
|
|
sun3_scsi_regp = (unsigned char *)ioaddr;
|
|
|
|
|
|
|
|
dregs = (struct sun3_dma_regs *)(((unsigned char *)ioaddr) + 8);
|
|
|
|
|
|
|
|
if (sun3_map_test((unsigned long)dregs, &x)) {
|
|
|
|
unsigned short oldcsr;
|
|
|
|
|
|
|
|
oldcsr = dregs->csr;
|
|
|
|
dregs->csr = 0;
|
|
|
|
udelay(SUN3_DMA_DELAY);
|
|
|
|
if (dregs->csr == 0x1400)
|
|
|
|
break;
|
|
|
|
|
|
|
|
dregs->csr = oldcsr;
|
|
|
|
}
|
|
|
|
|
|
|
|
iounmap((void *)ioaddr);
|
|
|
|
ioaddr = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ioaddr)
|
|
|
|
return 0;
|
|
|
|
#else
|
|
|
|
irq = IRQ_SUN3_SCSI;
|
2005-04-17 02:20:36 +04:00
|
|
|
ioaddr = (unsigned long)ioremap(IOBASE_SUN3_SCSI, PAGE_SIZE);
|
|
|
|
sun3_scsi_regp = (unsigned char *)ioaddr;
|
|
|
|
|
|
|
|
dregs = (struct sun3_dma_regs *)(((unsigned char *)ioaddr) + 8);
|
|
|
|
|
|
|
|
if((udc_regs = dvma_malloc(sizeof(struct sun3_udc_regs)))
|
|
|
|
== NULL) {
|
|
|
|
printk("SUN3 Scsi couldn't allocate DVMA memory!\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#ifdef SUPPORT_TAGS
|
|
|
|
if (setup_use_tagged_queuing < 0)
|
|
|
|
setup_use_tagged_queuing = USE_TAGGED_QUEUING;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
instance = scsi_register (tpnt, sizeof(struct NCR5380_hostdata));
|
|
|
|
if(instance == NULL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
default_instance = instance;
|
|
|
|
|
|
|
|
instance->io_port = (unsigned long) ioaddr;
|
2014-03-18 04:42:24 +04:00
|
|
|
instance->irq = irq;
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
NCR5380_init(instance, 0);
|
|
|
|
|
|
|
|
instance->n_io_port = 32;
|
|
|
|
|
|
|
|
if (request_irq(instance->irq, scsi_sun3_intr,
|
2007-11-12 03:52:05 +03:00
|
|
|
0, "Sun3SCSI-5380", instance)) {
|
2005-04-17 02:20:36 +04:00
|
|
|
#ifndef REAL_DMA
|
|
|
|
printk("scsi%d: IRQ%d not free, interrupts disabled\n",
|
|
|
|
instance->host_no, instance->irq);
|
2014-11-12 08:11:56 +03:00
|
|
|
instance->irq = NO_IRQ;
|
2005-04-17 02:20:36 +04:00
|
|
|
#else
|
|
|
|
printk("scsi%d: IRQ%d not free, bailing out\n",
|
|
|
|
instance->host_no, instance->irq);
|
|
|
|
return 0;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
dregs->csr = 0;
|
|
|
|
udelay(SUN3_DMA_DELAY);
|
|
|
|
dregs->csr = CSR_SCSI | CSR_FIFO | CSR_INTR;
|
|
|
|
udelay(SUN3_DMA_DELAY);
|
|
|
|
dregs->fifo_count = 0;
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
dregs->fifo_count_hi = 0;
|
|
|
|
dregs->dma_addr_hi = 0;
|
|
|
|
dregs->dma_addr_lo = 0;
|
|
|
|
dregs->dma_count_hi = 0;
|
|
|
|
dregs->dma_count_lo = 0;
|
|
|
|
|
|
|
|
dregs->ivect = VME_DATA24 | (instance->irq & 0xff);
|
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
called = 1;
|
|
|
|
|
|
|
|
#ifdef RESET_BOOT
|
|
|
|
sun3_scsi_reset_boot(instance);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2014-11-12 08:11:51 +03:00
|
|
|
static int sun3scsi_release(struct Scsi_Host *shpnt)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2014-11-12 08:11:56 +03:00
|
|
|
if (shpnt->irq != NO_IRQ)
|
2007-11-12 03:52:05 +03:00
|
|
|
free_irq(shpnt->irq, shpnt);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
iounmap((void *)sun3_scsi_regp);
|
|
|
|
|
2011-06-13 22:39:19 +04:00
|
|
|
NCR5380_exit(shpnt);
|
2005-04-17 02:20:36 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef RESET_BOOT
|
|
|
|
/*
|
|
|
|
* Our 'bus reset on boot' function
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void sun3_scsi_reset_boot(struct Scsi_Host *instance)
|
|
|
|
{
|
|
|
|
unsigned long end;
|
|
|
|
|
|
|
|
NCR5380_local_declare();
|
|
|
|
NCR5380_setup(instance);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Do a SCSI reset to clean up the bus during initialization. No
|
|
|
|
* messing with the queues, interrupts, or locks necessary here.
|
|
|
|
*/
|
|
|
|
|
|
|
|
printk( "Sun3 SCSI: resetting the SCSI bus..." );
|
|
|
|
|
|
|
|
/* switch off SCSI IRQ - catch an interrupt without IRQ bit set else */
|
|
|
|
// sun3_disable_irq( IRQ_SUN3_SCSI );
|
|
|
|
|
|
|
|
/* get in phase */
|
|
|
|
NCR5380_write( TARGET_COMMAND_REG,
|
|
|
|
PHASE_SR_TO_TCR( NCR5380_read(STATUS_REG) ));
|
|
|
|
|
|
|
|
/* assert RST */
|
|
|
|
NCR5380_write( INITIATOR_COMMAND_REG, ICR_BASE | ICR_ASSERT_RST );
|
|
|
|
|
|
|
|
/* The min. reset hold time is 25us, so 40us should be enough */
|
|
|
|
udelay( 50 );
|
|
|
|
|
|
|
|
/* reset RST and interrupt */
|
|
|
|
NCR5380_write( INITIATOR_COMMAND_REG, ICR_BASE );
|
|
|
|
NCR5380_read( RESET_PARITY_INTERRUPT_REG );
|
|
|
|
|
|
|
|
for( end = jiffies + AFTER_RESET_DELAY; time_before(jiffies, end); )
|
|
|
|
barrier();
|
|
|
|
|
|
|
|
/* switch on SCSI IRQ again */
|
|
|
|
// sun3_enable_irq( IRQ_SUN3_SCSI );
|
|
|
|
|
|
|
|
printk( " done\n" );
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// safe bits for the CSR
|
|
|
|
#define CSR_GOOD 0x060f
|
|
|
|
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 17:55:46 +04:00
|
|
|
static irqreturn_t scsi_sun3_intr(int irq, void *dummy)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
|
|
|
unsigned short csr = dregs->csr;
|
|
|
|
int handled = 0;
|
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
dregs->csr &= ~CSR_DMA_ENABLE;
|
|
|
|
#endif
|
|
|
|
|
2005-04-17 02:20:36 +04:00
|
|
|
if(csr & ~CSR_GOOD) {
|
|
|
|
if(csr & CSR_DMA_BUSERR) {
|
|
|
|
printk("scsi%d: bus error in dma\n", default_instance->host_no);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(csr & CSR_DMA_CONFLICT) {
|
|
|
|
printk("scsi%d: dma conflict\n", default_instance->host_no);
|
|
|
|
}
|
|
|
|
handled = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(csr & (CSR_SDB_INT | CSR_DMA_INT)) {
|
IRQ: Maintain regs pointer globally rather than passing to IRQ handlers
Maintain a per-CPU global "struct pt_regs *" variable which can be used instead
of passing regs around manually through all ~1800 interrupt handlers in the
Linux kernel.
The regs pointer is used in few places, but it potentially costs both stack
space and code to pass it around. On the FRV arch, removing the regs parameter
from all the genirq function results in a 20% speed up of the IRQ exit path
(ie: from leaving timer_interrupt() to leaving do_IRQ()).
Where appropriate, an arch may override the generic storage facility and do
something different with the variable. On FRV, for instance, the address is
maintained in GR28 at all times inside the kernel as part of general exception
handling.
Having looked over the code, it appears that the parameter may be handed down
through up to twenty or so layers of functions. Consider a USB character
device attached to a USB hub, attached to a USB controller that posts its
interrupts through a cascaded auxiliary interrupt controller. A character
device driver may want to pass regs to the sysrq handler through the input
layer which adds another few layers of parameter passing.
I've build this code with allyesconfig for x86_64 and i386. I've runtested the
main part of the code on FRV and i386, though I can't test most of the drivers.
I've also done partial conversion for powerpc and MIPS - these at least compile
with minimal configurations.
This will affect all archs. Mostly the changes should be relatively easy.
Take do_IRQ(), store the regs pointer at the beginning, saving the old one:
struct pt_regs *old_regs = set_irq_regs(regs);
And put the old one back at the end:
set_irq_regs(old_regs);
Don't pass regs through to generic_handle_irq() or __do_IRQ().
In timer_interrupt(), this sort of change will be necessary:
- update_process_times(user_mode(regs));
- profile_tick(CPU_PROFILING, regs);
+ update_process_times(user_mode(get_irq_regs()));
+ profile_tick(CPU_PROFILING);
I'd like to move update_process_times()'s use of get_irq_regs() into itself,
except that i386, alone of the archs, uses something other than user_mode().
Some notes on the interrupt handling in the drivers:
(*) input_dev() is now gone entirely. The regs pointer is no longer stored in
the input_dev struct.
(*) finish_unlinks() in drivers/usb/host/ohci-q.c needs checking. It does
something different depending on whether it's been supplied with a regs
pointer or not.
(*) Various IRQ handler function pointers have been moved to type
irq_handler_t.
Signed-Off-By: David Howells <dhowells@redhat.com>
(cherry picked from 1b16e7ac850969f38b375e511e3fa2f474a33867 commit)
2006-10-05 17:55:46 +04:00
|
|
|
NCR5380_intr(irq, dummy);
|
2005-04-17 02:20:36 +04:00
|
|
|
handled = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return IRQ_RETVAL(handled);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Debug stuff - to be called on NMI, or sysrq key. Use at your own risk;
|
|
|
|
* reentering NCR5380_print_status seems to have ugly side effects
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* this doesn't seem to get used at all -- sam */
|
|
|
|
#if 0
|
|
|
|
void sun3_sun3_debug (void)
|
|
|
|
{
|
|
|
|
unsigned long flags;
|
|
|
|
NCR5380_local_declare();
|
|
|
|
|
|
|
|
if (default_instance) {
|
|
|
|
local_irq_save(flags);
|
|
|
|
NCR5380_print_status(default_instance);
|
|
|
|
local_irq_restore(flags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
/* sun3scsi_dma_setup() -- initialize the dma controller for a read/write */
|
|
|
|
static unsigned long sun3scsi_dma_setup(void *data, unsigned long count, int write_flag)
|
|
|
|
{
|
|
|
|
void *addr;
|
|
|
|
|
|
|
|
if(sun3_dma_orig_addr != NULL)
|
|
|
|
dvma_unmap(sun3_dma_orig_addr);
|
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
addr = (void *)dvma_map_vme((unsigned long) data, count);
|
|
|
|
#else
|
2005-04-17 02:20:36 +04:00
|
|
|
addr = (void *)dvma_map((unsigned long) data, count);
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
sun3_dma_orig_addr = addr;
|
|
|
|
sun3_dma_orig_count = count;
|
2014-03-18 04:42:24 +04:00
|
|
|
|
|
|
|
#ifndef SUN3_SCSI_VME
|
2005-04-17 02:20:36 +04:00
|
|
|
dregs->fifo_count = 0;
|
|
|
|
sun3_udc_write(UDC_RESET, UDC_CSR);
|
|
|
|
|
|
|
|
/* reset fifo */
|
|
|
|
dregs->csr &= ~CSR_FIFO;
|
|
|
|
dregs->csr |= CSR_FIFO;
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
/* set direction */
|
|
|
|
if(write_flag)
|
|
|
|
dregs->csr |= CSR_SEND;
|
|
|
|
else
|
|
|
|
dregs->csr &= ~CSR_SEND;
|
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
dregs->csr |= CSR_PACK_ENABLE;
|
|
|
|
|
|
|
|
dregs->dma_addr_hi = ((unsigned long)addr >> 16);
|
|
|
|
dregs->dma_addr_lo = ((unsigned long)addr & 0xffff);
|
|
|
|
|
|
|
|
dregs->dma_count_hi = 0;
|
|
|
|
dregs->dma_count_lo = 0;
|
|
|
|
dregs->fifo_count_hi = 0;
|
|
|
|
dregs->fifo_count = 0;
|
|
|
|
#else
|
2005-04-17 02:20:36 +04:00
|
|
|
/* byte count for fifo */
|
|
|
|
dregs->fifo_count = count;
|
|
|
|
|
|
|
|
sun3_udc_write(UDC_RESET, UDC_CSR);
|
|
|
|
|
|
|
|
/* reset fifo */
|
|
|
|
dregs->csr &= ~CSR_FIFO;
|
|
|
|
dregs->csr |= CSR_FIFO;
|
|
|
|
|
|
|
|
if(dregs->fifo_count != count) {
|
|
|
|
printk("scsi%d: fifo_mismatch %04x not %04x\n",
|
|
|
|
default_instance->host_no, dregs->fifo_count,
|
|
|
|
(unsigned int) count);
|
2014-03-18 04:42:16 +04:00
|
|
|
NCR5380_dprint(NDEBUG_DMA, default_instance);
|
2005-04-17 02:20:36 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* setup udc */
|
|
|
|
udc_regs->addr_hi = (((unsigned long)(addr) & 0xff0000) >> 8);
|
|
|
|
udc_regs->addr_lo = ((unsigned long)(addr) & 0xffff);
|
|
|
|
udc_regs->count = count/2; /* count in words */
|
|
|
|
udc_regs->mode_hi = UDC_MODE_HIWORD;
|
|
|
|
if(write_flag) {
|
|
|
|
if(count & 1)
|
|
|
|
udc_regs->count++;
|
|
|
|
udc_regs->mode_lo = UDC_MODE_LSEND;
|
|
|
|
udc_regs->rsel = UDC_RSEL_SEND;
|
|
|
|
} else {
|
|
|
|
udc_regs->mode_lo = UDC_MODE_LRECV;
|
|
|
|
udc_regs->rsel = UDC_RSEL_RECV;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* announce location of regs block */
|
|
|
|
sun3_udc_write(((dvma_vtob(udc_regs) & 0xff0000) >> 8),
|
|
|
|
UDC_CHN_HI);
|
|
|
|
|
|
|
|
sun3_udc_write((dvma_vtob(udc_regs) & 0xffff), UDC_CHN_LO);
|
|
|
|
|
|
|
|
/* set dma master on */
|
|
|
|
sun3_udc_write(0xd, UDC_MODE);
|
|
|
|
|
|
|
|
/* interrupt enable */
|
|
|
|
sun3_udc_write(UDC_INT_ENABLE, UDC_CSR);
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
return count;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifndef SUN3_SCSI_VME
|
2005-04-17 02:20:36 +04:00
|
|
|
static inline unsigned long sun3scsi_dma_count(struct Scsi_Host *instance)
|
|
|
|
{
|
|
|
|
unsigned short resid;
|
|
|
|
|
|
|
|
dregs->udc_addr = 0x32;
|
|
|
|
udelay(SUN3_DMA_DELAY);
|
|
|
|
resid = dregs->udc_data;
|
|
|
|
udelay(SUN3_DMA_DELAY);
|
|
|
|
resid *= 2;
|
|
|
|
|
|
|
|
return (unsigned long) resid;
|
|
|
|
}
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
static inline unsigned long sun3scsi_dma_residual(struct Scsi_Host *instance)
|
|
|
|
{
|
|
|
|
return last_residual;
|
|
|
|
}
|
|
|
|
|
2006-10-03 21:51:59 +04:00
|
|
|
static inline unsigned long sun3scsi_dma_xfer_len(unsigned long wanted,
|
|
|
|
struct scsi_cmnd *cmd,
|
|
|
|
int write_flag)
|
2005-04-17 02:20:36 +04:00
|
|
|
{
|
2010-08-07 20:17:56 +04:00
|
|
|
if (cmd->request->cmd_type == REQ_TYPE_FS)
|
2005-04-17 02:20:36 +04:00
|
|
|
return wanted;
|
|
|
|
else
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline int sun3scsi_dma_start(unsigned long count, unsigned char *data)
|
|
|
|
{
|
2014-03-18 04:42:24 +04:00
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
unsigned short csr;
|
|
|
|
|
|
|
|
csr = dregs->csr;
|
|
|
|
|
|
|
|
dregs->dma_count_hi = (sun3_dma_orig_count >> 16);
|
|
|
|
dregs->dma_count_lo = (sun3_dma_orig_count & 0xffff);
|
2005-04-17 02:20:36 +04:00
|
|
|
|
2014-03-18 04:42:24 +04:00
|
|
|
dregs->fifo_count_hi = (sun3_dma_orig_count >> 16);
|
|
|
|
dregs->fifo_count = (sun3_dma_orig_count & 0xffff);
|
|
|
|
|
|
|
|
/* if(!(csr & CSR_DMA_ENABLE))
|
|
|
|
* dregs->csr |= CSR_DMA_ENABLE;
|
|
|
|
*/
|
|
|
|
#else
|
2005-04-17 02:20:36 +04:00
|
|
|
sun3_udc_write(UDC_CHN_START, UDC_CSR);
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* clean up after our dma is done */
|
|
|
|
static int sun3scsi_dma_finish(int write_flag)
|
|
|
|
{
|
2014-03-18 04:42:24 +04:00
|
|
|
unsigned short __maybe_unused count;
|
2005-04-17 02:20:36 +04:00
|
|
|
unsigned short fifo;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
sun3_dma_active = 0;
|
2014-03-18 04:42:24 +04:00
|
|
|
|
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
dregs->csr &= ~CSR_DMA_ENABLE;
|
|
|
|
|
|
|
|
fifo = dregs->fifo_count;
|
|
|
|
if (write_flag) {
|
|
|
|
if ((fifo > 0) && (fifo < sun3_dma_orig_count))
|
|
|
|
fifo++;
|
|
|
|
}
|
|
|
|
|
|
|
|
last_residual = fifo;
|
|
|
|
/* empty bytes from the fifo which didn't make it */
|
|
|
|
if ((!write_flag) && (dregs->csr & CSR_LEFT)) {
|
|
|
|
unsigned char *vaddr;
|
|
|
|
|
|
|
|
vaddr = (unsigned char *)dvma_vmetov(sun3_dma_orig_addr);
|
|
|
|
|
|
|
|
vaddr += (sun3_dma_orig_count - fifo);
|
|
|
|
vaddr--;
|
|
|
|
|
|
|
|
switch (dregs->csr & CSR_LEFT) {
|
|
|
|
case CSR_LEFT_3:
|
|
|
|
*vaddr = (dregs->bpack_lo & 0xff00) >> 8;
|
|
|
|
vaddr--;
|
|
|
|
|
|
|
|
case CSR_LEFT_2:
|
|
|
|
*vaddr = (dregs->bpack_hi & 0x00ff);
|
|
|
|
vaddr--;
|
|
|
|
|
|
|
|
case CSR_LEFT_1:
|
|
|
|
*vaddr = (dregs->bpack_hi & 0xff00) >> 8;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
2005-04-17 02:20:36 +04:00
|
|
|
// check to empty the fifo on a read
|
|
|
|
if(!write_flag) {
|
|
|
|
int tmo = 20000; /* .2 sec */
|
|
|
|
|
|
|
|
while(1) {
|
|
|
|
if(dregs->csr & CSR_FIFO_EMPTY)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if(--tmo <= 0) {
|
|
|
|
printk("sun3scsi: fifo failed to empty!\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
udelay(10);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
count = sun3scsi_dma_count(default_instance);
|
|
|
|
|
|
|
|
fifo = dregs->fifo_count;
|
|
|
|
last_residual = fifo;
|
|
|
|
|
|
|
|
/* empty bytes from the fifo which didn't make it */
|
|
|
|
if((!write_flag) && (count - fifo) == 2) {
|
|
|
|
unsigned short data;
|
|
|
|
unsigned char *vaddr;
|
|
|
|
|
|
|
|
data = dregs->fifo_data;
|
|
|
|
vaddr = (unsigned char *)dvma_btov(sun3_dma_orig_addr);
|
|
|
|
|
|
|
|
vaddr += (sun3_dma_orig_count - fifo);
|
|
|
|
|
|
|
|
vaddr[-2] = (data & 0xff00) >> 8;
|
|
|
|
vaddr[-1] = (data & 0xff);
|
|
|
|
}
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
dvma_unmap(sun3_dma_orig_addr);
|
|
|
|
sun3_dma_orig_addr = NULL;
|
2014-03-18 04:42:24 +04:00
|
|
|
|
|
|
|
#ifdef SUN3_SCSI_VME
|
|
|
|
dregs->dma_addr_hi = 0;
|
|
|
|
dregs->dma_addr_lo = 0;
|
|
|
|
dregs->dma_count_hi = 0;
|
|
|
|
dregs->dma_count_lo = 0;
|
|
|
|
|
|
|
|
dregs->fifo_count = 0;
|
|
|
|
dregs->fifo_count_hi = 0;
|
|
|
|
|
|
|
|
dregs->csr &= ~CSR_SEND;
|
|
|
|
/* dregs->csr |= CSR_DMA_ENABLE; */
|
|
|
|
#else
|
2005-04-17 02:20:36 +04:00
|
|
|
sun3_udc_write(UDC_RESET, UDC_CSR);
|
|
|
|
dregs->fifo_count = 0;
|
|
|
|
dregs->csr &= ~CSR_SEND;
|
|
|
|
|
|
|
|
/* reset fifo */
|
|
|
|
dregs->csr &= ~CSR_FIFO;
|
|
|
|
dregs->csr |= CSR_FIFO;
|
2014-03-18 04:42:24 +04:00
|
|
|
#endif
|
2005-04-17 02:20:36 +04:00
|
|
|
|
|
|
|
sun3_dma_setup_done = NULL;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#include "sun3_NCR5380.c"
|
|
|
|
|
2005-10-31 20:31:40 +03:00
|
|
|
static struct scsi_host_template driver_template = {
|
2013-04-10 15:52:09 +04:00
|
|
|
.show_info = sun3scsi_show_info,
|
2005-04-17 02:20:36 +04:00
|
|
|
.name = SUN3_SCSI_NAME,
|
|
|
|
.detect = sun3scsi_detect,
|
|
|
|
.release = sun3scsi_release,
|
|
|
|
.info = sun3scsi_info,
|
|
|
|
.queuecommand = sun3scsi_queue_command,
|
|
|
|
.eh_abort_handler = sun3scsi_abort,
|
|
|
|
.eh_bus_reset_handler = sun3scsi_bus_reset,
|
|
|
|
.can_queue = CAN_QUEUE,
|
|
|
|
.this_id = 7,
|
|
|
|
.sg_tablesize = SG_TABLESIZE,
|
|
|
|
.cmd_per_lun = CMD_PER_LUN,
|
|
|
|
.use_clustering = DISABLE_CLUSTERING
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#include "scsi_module.c"
|
|
|
|
|
|
|
|
MODULE_LICENSE("GPL");
|