亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频

? 歡迎來到蟲蟲下載站! | ?? 資源下載 ?? 資源專輯 ?? 關于我們
? 蟲蟲下載站

?? wd7000.c

?? 內核是系統的心臟
?? C
?? 第 1 頁 / 共 2 頁
字號:
/* $Id: wd7000.c,v 1.2 1994/01/15 06:02:32 drew Exp $
 *  linux/kernel/wd7000.c
 *
 *  Copyright (C) 1992  Thomas Wuensche
 *	closely related to the aha1542 driver from Tommy Thorn
 *	( as close as different hardware allows on a lowlevel-driver :-) )
 *
 *  Revised (and renamed) by John Boyd <boyd@cis.ohio-state.edu> to
 *  accomodate Eric Youngdale's modifications to scsi.c.  Nov 1992.
 *
 *  Additional changes to support scatter/gather.  Dec. 1992.  tw/jb
 */

#include <stdarg.h>
#include <linux/kernel.h>
#include <linux/head.h>
#include <linux/types.h>
#include <linux/string.h>
#include <linux/sched.h>
#include <asm/system.h>
#include <asm/dma.h>
#include <asm/io.h>
#include <linux/ioport.h>

#include "../block/blk.h"
#include "scsi.h"
#include "hosts.h"

/* #define DEBUG  */

#include "wd7000.h"


#ifdef DEBUG
#define DEB(x) x
#else
#define DEB(x)
#endif


/*
   Driver data structures:
   - mb and scbs are required for interfacing with the host adapter.
     An SCB has extra fields not visible to the adapter; mb's
     _cannot_ do this, since the adapter assumes they are contiguous in
     memory, 4 bytes each, with ICMBs following OGMBs, and uses this fact
     to access them.
   - An icb is for host-only (non-SCSI) commands.  ICBs are 16 bytes each;
     the additional bytes are used only by the driver.
   - For now, a pool of SCBs are kept in global storage by this driver,
     and are allocated and freed as needed.

  The 7000-FASST2 marks OGMBs empty as soon as it has _started_ a command,
  not when it has finished.  Since the SCB must be around for completion,
  problems arise when SCBs correspond to OGMBs, which may be reallocated
  earlier (or delayed unnecessarily until a command completes).
  Mailboxes are used as transient data structures, simply for
  carrying SCB addresses to/from the 7000-FASST2.

  Note also since SCBs are not "permanently" associated with mailboxes,
  there is no need to keep a global list of Scsi_Cmnd pointers indexed
  by OGMB.   Again, SCBs reference their Scsi_Cmnds directly, so mailbox
  indices need not be involved.
*/

static struct {
       struct wd_mailbox ogmb[OGMB_CNT]; 
       struct wd_mailbox icmb[ICMB_CNT];
} mb;
static int next_ogmb = 0;   /* to reduce contention at mailboxes */

static Scb scbs[MAX_SCBS];
static Scb *scbfree = NULL;

static int wd7000_host = 0;
static unchar controlstat = 0;

static unchar rev_1 = 0, rev_2 = 0;  /* filled in by wd7000_revision */

#define wd7000_intr_ack()  outb(0,INTR_ACK)

#define WAITnexttimeout 3000000


static inline void wd7000_enable_intr(void)
{
    controlstat |= INT_EN;
    outb(controlstat,CONTROL);
}


static inline void wd7000_enable_dma(void)
{
    controlstat |= DMA_EN;
    outb(controlstat,CONTROL);
    set_dma_mode(DMA_CH, DMA_MODE_CASCADE);
    enable_dma(DMA_CH);
}


#define WAIT(port, mask, allof, noneof)					\
 { register WAITbits;							\
   register WAITtimeout = WAITnexttimeout;				\
   while (1) {								\
     WAITbits = inb(port) & (mask);					\
     if ((WAITbits & (allof)) == (allof) && ((WAITbits & (noneof)) == 0)) \
       break;                                                         	\
     if (--WAITtimeout == 0) goto fail;					\
   }									\
 }


static inline void delay( unsigned how_long )
{
     unsigned long time = jiffies + how_long;

     while (jiffies < time);
}


static inline int command_out(unchar *cmdp, int len)
{
    while (len--)  {
        WAIT(ASC_STAT, STATMASK, CMD_RDY, 0);
	outb(*cmdp++, COMMAND);
    }
    return 1;

fail:
    printk("wd7000_out WAIT failed(%d): ", len+1);
    return 0;
}

static inline Scb *alloc_scb(void)
{
    Scb *scb;
    unsigned long flags;

    save_flags(flags);
    cli();

    if (scbfree == NULL)  {
        panic("wd7000: can't allocate free SCB.\n");
	restore_flags(flags);
	return NULL;
    }
    scb = scbfree;  scbfree = scb->next;
    memset(scb, 0, sizeof(Scb));  scb->next = NULL;

    restore_flags(flags);

    return scb;
}


static inline void free_scb( Scb *scb )
{
    unsigned long flags;

    save_flags(flags);
    cli();

    memset(scb, 0, sizeof(Scb));
    scb->next = scbfree;  scbfree = scb;

    restore_flags(flags);
}


static inline void init_scbs(void)
{
    int i;
    unsigned long flags;

    save_flags(flags);
    cli();

    scbfree = &(scbs[0]);
    for (i = 0;  i < MAX_SCBS-1;  i++)  scbs[i].next = &(scbs[i+1]);
    scbs[MAX_SCBS-1].next = NULL;

    restore_flags(flags);
}    
    

static int mail_out( Scb *scbptr )
/*
 *  Note: this can also be used for ICBs; just cast to the parm type.
 */
{
    int i, ogmb;
    unsigned long flags;

    DEB(printk("wd7000_scb_out: %06x");)

    /* We first look for a free outgoing mailbox */
    save_flags(flags);
    cli();
    ogmb = next_ogmb;
    for (i = 0; i < OGMB_CNT; i++) {
	if (mb.ogmb[ogmb].status == 0)  {
	    DEB(printk(" using OGMB %x",ogmb));
	    mb.ogmb[ogmb].status = 1;
	    any2scsi(mb.ogmb[ogmb].scbptr, scbptr);

	    next_ogmb = (ogmb+1) % OGMB_CNT;
	    break;
	}  else
	    ogmb = (++ogmb) % OGMB_CNT;
    }
    restore_flags(flags);
    DEB(printk(", scb is %x",scbptr);)

    if (i >= OGMB_CNT) {
        DEB(printk(", no free OGMBs.\n");)
	/* Alternatively, issue "interrupt on free OGMB", and sleep... */
        return 0;
    }

    wd7000_enable_intr(); 
    do  {
        WAIT(ASC_STAT,STATMASK,CMD_RDY,0);
	outb(START_OGMB|ogmb, COMMAND);
	WAIT(ASC_STAT,STATMASK,CMD_RDY,0);
    }  while (inb(ASC_STAT) & CMD_REJ);

    DEB(printk(", awaiting interrupt.\n");)
    return 1;

fail:
    DEB(printk(", WAIT timed out.\n");)
    return 0;
}


int make_code(unsigned hosterr, unsigned scsierr)
{   
#ifdef DEBUG
    int in_error = hosterr;
#endif

    switch ((hosterr>>8)&0xff){
	case 0:	/* Reserved */
		hosterr = DID_ERROR;
		break;
	case 1:	/* Command Complete, no errors */
		hosterr = DID_OK;
		break;
	case 2: /* Command complete, error logged in scb status (scsierr) */ 
		hosterr = DID_OK;
		break;
	case 4:	/* Command failed to complete - timeout */
		hosterr = DID_TIME_OUT;
		break;
	case 5:	/* Command terminated; Bus reset by external device */
		hosterr = DID_RESET;
		break;
	case 6:	/* Unexpected Command Received w/ host as target */
		hosterr = DID_BAD_TARGET;
		break;
	case 80: /* Unexpected Reselection */
        case 81: /* Unexpected Selection */
		hosterr = DID_BAD_INTR;
		break;
        case 82: /* Abort Command Message  */
		hosterr = DID_ABORT;
		break;
	case 83: /* SCSI Bus Software Reset */
	case 84: /* SCSI Bus Hardware Reset */
		hosterr = DID_RESET;
		break;
        default: /* Reserved */
		hosterr = DID_ERROR;
		break;
	}
#ifdef DEBUG
    if (scsierr||hosterr)
        printk("\nSCSI command error: SCSI %02x host %04x return %d",
	       scsierr,in_error,hosterr);
#endif
    return scsierr | (hosterr << 16);
}


static void wd7000_scsi_done(Scsi_Cmnd * SCpnt)
{
    DEB(printk("wd7000_scsi_done: %06x\n",SCpnt);)
    SCpnt->SCp.phase = 0;
}


void wd7000_intr_handle(int irq)
{
    int flag, icmb, errstatus, icmb_status;
    int host_error, scsi_error;
    Scb *scb;             /* for SCSI commands */
    unchar *icb;          /* for host commands */
    Scsi_Cmnd *SCpnt;

    flag = inb(INTR_STAT);
    DEB(printk("wd7000_intr_handle: intr stat = %02x",flag);)

    if (!(inb(ASC_STAT)&0x80)){ 
	DEB(printk("\nwd7000_intr_handle: phantom interrupt...\n");)
	wd7000_intr_ack();
	return; 
    }

    /* check for an incoming mailbox */
    if ((flag & 0x40) == 0) {
        /*  for a free OGMB - need code for this case... */
        DEB(printk("wd7000_intr_handle: free outgoing mailbox\n");)

?? 快捷鍵說明

復制代碼 Ctrl + C
搜索代碼 Ctrl + F
全屏模式 F11
切換主題 Ctrl + Shift + D
顯示快捷鍵 ?
增大字號 Ctrl + =
減小字號 Ctrl + -
亚洲欧美第一页_禁久久精品乱码_粉嫩av一区二区三区免费野_久草精品视频
亚洲午夜久久久久久久久电影院 | 亚洲综合精品自拍| 成人avav影音| 欧美精彩视频一区二区三区| 久久成人久久鬼色| 日韩午夜电影av| 久久国产精品一区二区| 欧美一区二区三区男人的天堂| 一区二区三区国产精华| 91丨九色丨蝌蚪丨老版| 亚洲人午夜精品天堂一二香蕉| 97精品超碰一区二区三区| 中文字幕一区二区三| 99精品视频在线观看| 一色屋精品亚洲香蕉网站| 在线亚洲精品福利网址导航| 一区二区三区不卡视频| 欧美午夜一区二区三区| 午夜精品久久久久久| 制服视频三区第一页精品| 日韩精品一级中文字幕精品视频免费观看 | 蜜臀91精品一区二区三区 | 欧美日韩国产大片| 美女脱光内衣内裤视频久久网站 | 欧美午夜片在线观看| 日韩成人精品在线| 国产亚洲午夜高清国产拍精品| 国产精品一区二区久久不卡| 亚洲特黄一级片| 欧美一区二区三级| 日本久久电影网| 国产精品天美传媒沈樵| 色综合久久天天| 日本不卡一二三区黄网| 国产欧美一区二区三区在线看蜜臀| eeuss鲁片一区二区三区在线看| 午夜激情一区二区三区| 国产日韩欧美麻豆| 日韩欧美一二三四区| 无码av免费一区二区三区试看| 91精品国产综合久久香蕉的特点| 国产精品影视在线观看| 夜夜爽夜夜爽精品视频| 久久五月婷婷丁香社区| 欧美亚日韩国产aⅴ精品中极品| 久久国产尿小便嘘嘘| 亚洲美女免费视频| 国产午夜一区二区三区| 欧美日韩久久久| 99久久精品免费看| 国产一区视频网站| 日韩国产欧美三级| 亚洲一区二区欧美日韩| 国产精品久久久久一区二区三区 | 全国精品久久少妇| 亚洲成人三级小说| 亚洲午夜精品在线| 一区二区三区影院| 亚洲人成电影网站色mp4| 国产精品超碰97尤物18| 国产精品乱人伦| 欧美激情一区二区三区在线| 久久免费视频一区| 国产日产精品一区| 久久精品亚洲精品国产欧美kt∨ | 国产综合色精品一区二区三区| 青青草91视频| 蜜桃视频在线观看一区二区| 五月天网站亚洲| 免费观看一级欧美片| 国产在线播精品第三| 狠狠色丁香久久婷婷综合_中| 久久99国产乱子伦精品免费| 国产日韩欧美麻豆| 亚洲国产精品99久久久久久久久| 久久综合久久综合亚洲| 精品国产露脸精彩对白| 精品久久久网站| 国产精品视频免费| 1000部国产精品成人观看| 国产精品久久久久影视| 久久久亚洲高清| 精品91自产拍在线观看一区| 日韩欧美国产一区二区三区| 精品久久久久香蕉网| 中文字幕免费一区| 亚洲国产综合色| 久久国产精品99精品国产| 粉嫩一区二区三区在线看| 色欲综合视频天天天| 在线不卡a资源高清| 国产日本欧美一区二区| 亚洲欧美日韩在线| 午夜亚洲福利老司机| 国产一区二区三区最好精华液| 成人av在线电影| 欧美一区二区三区婷婷月色| 中文字幕免费一区| 日韩不卡免费视频| 91在线看国产| 国产三级欧美三级| 日韩成人精品视频| 色先锋资源久久综合| 精品处破学生在线二十三| 一区二区三区精品| 成人蜜臀av电影| 日韩欧美国产精品| 亚洲一区视频在线| 99国产精品视频免费观看| 久久综合一区二区| 久久99深爱久久99精品| 欧美精品乱码久久久久久按摩| 国产亚洲欧洲997久久综合| 天堂成人国产精品一区| 91极品视觉盛宴| 一区二区在线观看视频| 国产精品亚洲一区二区三区在线| 日韩一二三区不卡| 免费精品视频在线| 91精品在线一区二区| 视频在线观看一区| 欧美在线不卡视频| 偷偷要91色婷婷| 日韩一级片在线观看| 免费看日韩a级影片| 欧美成人国产一区二区| 国产伦精品一区二区三区免费迷 | 精品久久99ma| 国产一区二区三区av电影| 日韩精品一区二区三区在线观看| 人人爽香蕉精品| 久久久高清一区二区三区| 久久97超碰国产精品超碰| 久久久www成人免费毛片麻豆 | 欧美videos大乳护士334| 免费一级片91| 欧美日韩国产欧美日美国产精品| 欧美一级在线视频| 亚洲制服欧美中文字幕中文字幕| 在线一区二区观看| 午夜私人影院久久久久| 欧美高清精品3d| 蜜桃免费网站一区二区三区| 日韩免费一区二区三区在线播放| 韩日av一区二区| 国产精品成人一区二区艾草| 欧美在线色视频| 久久99精品久久久久久国产越南 | 欧美一级免费大片| 懂色av一区二区夜夜嗨| 一区二区在线观看免费| 精品理论电影在线| 欧美中文字幕一二三区视频| 久久国产精品99精品国产| 国产欧美在线观看一区| 成人av影院在线| 日韩av一区二区在线影视| 精品国产伦一区二区三区观看体验 | 亚洲美腿欧美偷拍| 欧美精品一区二区在线播放| 一本久久精品一区二区| 麻豆精品久久久| 亚洲夂夂婷婷色拍ww47| 久久精品这里都是精品| 欧美丰满一区二区免费视频| 99视频超级精品| 国产成人精品免费视频网站| 日本aⅴ亚洲精品中文乱码| 亚洲人成亚洲人成在线观看图片| 欧美zozo另类异族| 欧美剧情片在线观看| 色欧美乱欧美15图片| 成人午夜免费视频| 国产精品1区2区3区| 麻豆精品国产91久久久久久| 亚洲aⅴ怡春院| 一区二区三区中文字幕精品精品| 久久久一区二区三区捆绑**| 欧美一区二区性放荡片| 欧美日韩情趣电影| 欧美色网一区二区| 欧美人妇做爰xxxⅹ性高电影| 色婷婷综合久久久中文字幕| 99re这里都是精品| 99精品1区2区| 色噜噜狠狠成人网p站| 在线看日本不卡| 欧美欧美欧美欧美首页| 欧美精品123区| 精品久久久三级丝袜| 久久免费午夜影院| 国产精品久久久久久久蜜臀 | 欧美一区二区在线观看| 欧美成人国产一区二区| 久久午夜国产精品| 国产精品久久久久天堂| 亚洲欧美日韩国产手机在线| 亚洲在线视频一区| 美国三级日本三级久久99| 国产精品一品二品|