atari_scsi.c
来自「linux 内核源代码」· C语言 代码 · 共 1,149 行 · 第 1/3 页
C
1,149 行
/* * atari_scsi.c -- Device dependent functions for the Atari generic SCSI port * * Copyright 1994 Roman Hodek <Roman.Hodek@informatik.uni-erlangen.de> * * Loosely based on the work of Robert De Vries' team and added: * - working real DMA * - Falcon support (untested yet!) ++bjoern fixed and now it works * - lots of extensions and bug fixes. * * This file is subject to the terms and conditions of the GNU General Public * License. See the file COPYING in the main directory of this archive * for more details. * *//**************************************************************************//* *//* Notes for Falcon SCSI: *//* ---------------------- *//* *//* Since the Falcon SCSI uses the ST-DMA chip, that is shared among *//* several device drivers, locking and unlocking the access to this *//* chip is required. But locking is not possible from an interrupt, *//* since it puts the process to sleep if the lock is not available. *//* This prevents "late" locking of the DMA chip, i.e. locking it just *//* before using it, since in case of disconnection-reconnection *//* commands, the DMA is started from the reselection interrupt. *//* *//* Two possible schemes for ST-DMA-locking would be: *//* 1) The lock is taken for each command separately and disconnecting *//* is forbidden (i.e. can_queue = 1). *//* 2) The DMA chip is locked when the first command comes in and *//* released when the last command is finished and all queues are *//* empty. *//* The first alternative would result in bad performance, since the *//* interleaving of commands would not be used. The second is unfair to *//* other drivers using the ST-DMA, because the queues will seldom be *//* totally empty if there is a lot of disk traffic. *//* *//* For this reasons I decided to employ a more elaborate scheme: *//* - First, we give up the lock every time we can (for fairness), this *//* means every time a command finishes and there are no other commands *//* on the disconnected queue. *//* - If there are others waiting to lock the DMA chip, we stop *//* issuing commands, i.e. moving them onto the issue queue. *//* Because of that, the disconnected queue will run empty in a *//* while. Instead we go to sleep on a 'fairness_queue'. *//* - If the lock is released, all processes waiting on the fairness *//* queue will be woken. The first of them tries to re-lock the DMA, *//* the others wait for the first to finish this task. After that, *//* they can all run on and do their commands... *//* This sounds complicated (and it is it :-(), but it seems to be a *//* good compromise between fairness and performance: As long as no one *//* else wants to work with the ST-DMA chip, SCSI can go along as *//* usual. If now someone else comes, this behaviour is changed to a *//* "fairness mode": just already initiated commands are finished and *//* then the lock is released. The other one waiting will probably win *//* the race for locking the DMA, since it was waiting for longer. And *//* after it has finished, SCSI can go ahead again. Finally: I hope I *//* have not produced any deadlock possibilities! *//* *//**************************************************************************/#include <linux/module.h>#define NDEBUG (0)#define NDEBUG_ABORT 0x00100000#define NDEBUG_TAGS 0x00200000#define NDEBUG_MERGING 0x00400000#define AUTOSENSE/* For the Atari version, use only polled IO or REAL_DMA */#define REAL_DMA/* Support tagged queuing? (on devices that are able to... :-) */#define SUPPORT_TAGS#define MAX_TAGS 32#include <linux/types.h>#include <linux/stddef.h>#include <linux/ctype.h>#include <linux/delay.h>#include <linux/mm.h>#include <linux/blkdev.h>#include <linux/interrupt.h>#include <linux/init.h>#include <linux/nvram.h>#include <linux/bitops.h>#include <asm/setup.h>#include <asm/atarihw.h>#include <asm/atariints.h>#include <asm/page.h>#include <asm/pgtable.h>#include <asm/irq.h>#include <asm/traps.h>#include "scsi.h"#include <scsi/scsi_host.h>#include "atari_scsi.h"#include "NCR5380.h"#include <asm/atari_stdma.h>#include <asm/atari_stram.h>#include <asm/io.h>#include <linux/stat.h>#define IS_A_TT() ATARIHW_PRESENT(TT_SCSI)#define SCSI_DMA_WRITE_P(elt,val) \ do { \ unsigned long v = val; \ tt_scsi_dma.elt##_lo = v & 0xff; \ v >>= 8; \ tt_scsi_dma.elt##_lmd = v & 0xff; \ v >>= 8; \ tt_scsi_dma.elt##_hmd = v & 0xff; \ v >>= 8; \ tt_scsi_dma.elt##_hi = v & 0xff; \ } while(0)#define SCSI_DMA_READ_P(elt) \ (((((((unsigned long)tt_scsi_dma.elt##_hi << 8) | \ (unsigned long)tt_scsi_dma.elt##_hmd) << 8) | \ (unsigned long)tt_scsi_dma.elt##_lmd) << 8) | \ (unsigned long)tt_scsi_dma.elt##_lo)static inline void SCSI_DMA_SETADR(unsigned long adr){ st_dma.dma_lo = (unsigned char)adr; MFPDELAY(); adr >>= 8; st_dma.dma_md = (unsigned char)adr; MFPDELAY(); adr >>= 8; st_dma.dma_hi = (unsigned char)adr; MFPDELAY();}static inline unsigned long SCSI_DMA_GETADR(void){ unsigned long adr; adr = st_dma.dma_lo; MFPDELAY(); adr |= (st_dma.dma_md & 0xff) << 8; MFPDELAY(); adr |= (st_dma.dma_hi & 0xff) << 16; MFPDELAY(); return adr;}static inline void ENABLE_IRQ(void){ if (IS_A_TT()) atari_enable_irq(IRQ_TT_MFP_SCSI); else atari_enable_irq(IRQ_MFP_FSCSI);}static inline void DISABLE_IRQ(void){ if (IS_A_TT()) atari_disable_irq(IRQ_TT_MFP_SCSI); else atari_disable_irq(IRQ_MFP_FSCSI);}#define HOSTDATA_DMALEN (((struct NCR5380_hostdata *) \ (atari_scsi_host->hostdata))->dma_len)/* Time (in jiffies) to wait after a reset; the SCSI standard calls for 250ms, * we usually do 0.5s to be on the safe side. But Toshiba CD-ROMs once more * need ten times the standard value... */#ifndef CONFIG_ATARI_SCSI_TOSHIBA_DELAY#define AFTER_RESET_DELAY (HZ/2)#else#define AFTER_RESET_DELAY (5*HZ/2)#endif/***************************** Prototypes *****************************/#ifdef REAL_DMAstatic int scsi_dma_is_ignored_buserr(unsigned char dma_stat);static void atari_scsi_fetch_restbytes(void);static long atari_scsi_dma_residual(struct Scsi_Host *instance);static int falcon_classify_cmd(Scsi_Cmnd *cmd);static unsigned long atari_dma_xfer_len(unsigned long wanted_len, Scsi_Cmnd *cmd, int write_flag);#endifstatic irqreturn_t scsi_tt_intr(int irq, void *dummy);static irqreturn_t scsi_falcon_intr(int irq, void *dummy);static void falcon_release_lock_if_possible(struct NCR5380_hostdata *hostdata);static void falcon_get_lock(void);#ifdef CONFIG_ATARI_SCSI_RESET_BOOTstatic void atari_scsi_reset_boot(void);#endifstatic unsigned char atari_scsi_tt_reg_read(unsigned char reg);static void atari_scsi_tt_reg_write(unsigned char reg, unsigned char value);static unsigned char atari_scsi_falcon_reg_read(unsigned char reg);static void atari_scsi_falcon_reg_write(unsigned char reg, unsigned char value);/************************* End of Prototypes **************************/static struct Scsi_Host *atari_scsi_host;static unsigned char (*atari_scsi_reg_read)(unsigned char reg);static void (*atari_scsi_reg_write)(unsigned char reg, unsigned char value);#ifdef REAL_DMAstatic unsigned long atari_dma_residual, atari_dma_startaddr;static short atari_dma_active;/* pointer to the dribble buffer */static char *atari_dma_buffer;/* precalculated physical address of the dribble buffer */static unsigned long atari_dma_phys_buffer;/* != 0 tells the Falcon int handler to copy data from the dribble buffer */static char *atari_dma_orig_addr;/* size of the dribble buffer; 4k seems enough, since the Falcon cannot use * scatter-gather anyway, so most transfers are 1024 byte only. In the rare * cases where requests to physical contiguous buffers have been merged, this * request is <= 4k (one page). So I don't think we have to split transfers * just due to this buffer size... */#define STRAM_BUFFER_SIZE (4096)/* mask for address bits that can't be used with the ST-DMA */static unsigned long atari_dma_stram_mask;#define STRAM_ADDR(a) (((a) & atari_dma_stram_mask) == 0)/* number of bytes to cut from a transfer to handle NCR overruns */static int atari_read_overruns;#endifstatic 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_TAGSstatic int setup_use_tagged_queuing = -1;module_param(setup_use_tagged_queuing, int, 0);#endifstatic int setup_hostid = -1;module_param(setup_hostid, int, 0);#if defined(CONFIG_TT_DMA_EMUL)#include "atari_dma_emul.c"#endif#if defined(REAL_DMA)static int scsi_dma_is_ignored_buserr(unsigned char dma_stat){ int i; unsigned long addr = SCSI_DMA_READ_P(dma_addr), end_addr; if (dma_stat & 0x01) { /* A bus error happens when DMA-ing from the last page of a * physical memory chunk (DMA prefetch!), but that doesn't hurt. * Check for this case: */ for (i = 0; i < m68k_num_memory; ++i) { end_addr = m68k_memory[i].addr + m68k_memory[i].size; if (end_addr <= addr && addr <= end_addr + 4) return 1; } } return 0;}#if 0/* Dead code... wasn't called anyway :-) and causes some trouble, because at * end-of-DMA, both SCSI ints are triggered simultaneously, so the NCR int has * to clear the DMA int pending bit before it allows other level 6 interrupts. */static void scsi_dma_buserr(int irq, void *dummy){ unsigned char dma_stat = tt_scsi_dma.dma_ctrl; /* Don't do anything if a NCR interrupt is pending. Probably it's just * masked... */ if (atari_irq_pending(IRQ_TT_MFP_SCSI)) return; printk("Bad SCSI DMA interrupt! dma_addr=0x%08lx dma_stat=%02x dma_cnt=%08lx\n", SCSI_DMA_READ_P(dma_addr), dma_stat, SCSI_DMA_READ_P(dma_cnt)); if (dma_stat & 0x80) { if (!scsi_dma_is_ignored_buserr(dma_stat)) printk("SCSI DMA bus error -- bad DMA programming!\n"); } else { /* Under normal circumstances we never should get to this point, * since both interrupts are triggered simultaneously and the 5380 * int has higher priority. When this irq is handled, that DMA * interrupt is cleared. So a warning message is printed here. */ printk("SCSI DMA intr ?? -- this shouldn't happen!\n"); }}#endif#endifstatic irqreturn_t scsi_tt_intr(int irq, void *dummy){#ifdef REAL_DMA int dma_stat; dma_stat = tt_scsi_dma.dma_ctrl; INT_PRINTK("scsi%d: NCR5380 interrupt, DMA status = %02x\n", atari_scsi_host->host_no, dma_stat & 0xff); /* Look if it was the DMA that has interrupted: First possibility * is that a bus error occurred... */ if (dma_stat & 0x80) { if (!scsi_dma_is_ignored_buserr(dma_stat)) { printk(KERN_ERR "SCSI DMA caused bus error near 0x%08lx\n", SCSI_DMA_READ_P(dma_addr)); printk(KERN_CRIT "SCSI DMA bus error -- bad DMA programming!"); } } /* If the DMA is active but not finished, we have the case * that some other 5380 interrupt occurred within the DMA transfer. * This means we have residual bytes, if the desired end address * is not yet reached. Maybe we have to fetch some bytes from the * rest data register, too. The residual must be calculated from * the address pointer, not the counter register, because only the * addr reg counts bytes not yet written and pending in the rest * data reg! */ if ((dma_stat & 0x02) && !(dma_stat & 0x40)) { atari_dma_residual = HOSTDATA_DMALEN - (SCSI_DMA_READ_P(dma_addr) - atari_dma_startaddr); DMA_PRINTK("SCSI DMA: There are %ld residual bytes.\n", atari_dma_residual); if ((signed int)atari_dma_residual < 0) atari_dma_residual = 0; if ((dma_stat & 1) == 0) { /* * After read operations, we maybe have to * transport some rest bytes */ atari_scsi_fetch_restbytes(); } else { /* * There seems to be a nasty bug in some SCSI-DMA/NCR * combinations: If a target disconnects while a write * operation is going on, the address register of the * DMA may be a few bytes farer than it actually read. * This is probably due to DMA prefetching and a delay * between DMA and NCR. Experiments showed that the * dma_addr is 9 bytes to high, but this could vary. * The problem is, that the residual is thus calculated * wrong and the next transfer will start behind where * it should. So we round up the residual to the next * multiple of a sector size, if it isn't already a * multiple and the originally expected transfer size * was. The latter condition is there to ensure that * the correction is taken only for "real" data * transfers and not for, e.g., the parameters of some * other command. These shouldn't disconnect anyway. */ if (atari_dma_residual & 0x1ff) { DMA_PRINTK("SCSI DMA: DMA bug corrected, " "difference %ld bytes\n", 512 - (atari_dma_residual & 0x1ff)); atari_dma_residual = (atari_dma_residual + 511) & ~0x1ff; } } tt_scsi_dma.dma_ctrl = 0;
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?