!C99Shell v. 1.0 pre-release build #13!

Software: Apache/2.0.54 (Unix) mod_perl/1.99_09 Perl/v5.8.0 mod_ssl/2.0.54 OpenSSL/0.9.7l DAV/2 FrontPage/5.0.2.2635 PHP/4.4.0 mod_gzip/2.0.26.1a 

uname -a: Linux snow.he.net 4.4.276-v2-mono-1 #1 SMP Wed Jul 21 11:21:17 PDT 2021 i686 

uid=99(nobody) gid=98(nobody) groups=98(nobody) 

Safe-mode: OFF (not secure)

/usr/src/linux-2.4.18-xfs-1.1/drivers/scsi/   drwxr-xr-x
Free 318.3 GB of 458.09 GB (69.48%)
Home    Back    Forward    UPDIR    Refresh    Search    Buffer    Encoder    Tools    Proc.    FTP brute    Sec.    SQL    PHP-code    Update    Feedback    Self remove    Logout    


Viewing file:     sgiwd93.c (9.57 KB)      -rw-r--r--
Select action/file-type:
(+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
/*
 * sgiwd93.c: SGI WD93 scsi driver.
 *
 * Copyright (C) 1996 David S. Miller (dm@engr.sgi.com)
 *         1999 Andrew R. Baker (andrewb@uab.edu)
 *              - Support for 2nd SCSI controller on Indigo2
 * 
 * (In all truth, Jed Schimmel wrote all this code.)
 *
 * $Id: sgiwd93.c,v 1.19 2000/02/04 07:40:47 ralf Exp $
 */
#include <linux/init.h>
#include <linux/types.h>
#include <linux/mm.h>
#include <linux/blk.h>
#include <linux/version.h>
#include <linux/delay.h>
#include <linux/spinlock.h>

#include <asm/page.h>
#include <asm/pgtable.h>
#include <asm/sgialib.h>
#include <asm/sgi/sgi.h>
#include <asm/sgi/sgimc.h>
#include <asm/sgi/sgihpc.h>
#include <asm/sgi/sgint23.h>
#include <asm/irq.h>
#include <asm/io.h>

#include "scsi.h"
#include "hosts.h"
#include "wd33c93.h"
#include "sgiwd93.h"

#include <linux/stat.h>

struct hpc_chunk {
    struct hpc_dma_desc desc;
    unsigned long padding;
};

struct Scsi_Host *sgiwd93_host = NULL;
struct Scsi_Host *sgiwd93_host1 = NULL;

/* Wuff wuff, wuff, wd33c93.c, wuff wuff, object oriented, bow wow. */
static inline void write_wd33c93_count(wd33c93_regs *regp, unsigned long value)
{
    regp->SASR = WD_TRANSFER_COUNT_MSB;
    regp->SCMD = ((value >> 16) & 0xff);
    regp->SCMD = ((value >>  8) & 0xff);
    regp->SCMD = ((value >>  0) & 0xff);
}

static inline unsigned long read_wd33c93_count(wd33c93_regs *regp)
{
    unsigned long value;

    regp->SASR = WD_TRANSFER_COUNT_MSB;
    value =  ((regp->SCMD & 0xff) << 16);
    value |= ((regp->SCMD & 0xff) <<  8);
    value |= ((regp->SCMD & 0xff) <<  0);
    return value;
}

/* XXX woof! */
static void sgiwd93_intr(int irq, void *dev_id, struct pt_regs *regs)
{
    unsigned long flags;

    spin_lock_irqsave(&io_request_lock, flags);
    wd33c93_intr((struct Scsi_Host *) dev_id);
    spin_unlock_irqrestore(&io_request_lock, flags);
}

#undef DEBUG_DMA

static inline
void fill_hpc_entries (struct hpc_chunk **hcp, char *addr, unsigned long len)
{
    unsigned long physaddr;
    unsigned long count;
    
    dma_cache_wback_inv((unsigned long)addr,len);
    physaddr = PHYSADDR(addr);
    while (len) {
        /*
         * even cntinfo could be up to 16383, without
         * magic only 8192 works correctly
         */
        count = len > 8192 ? 8192 : len;
        (*hcp)->desc.pbuf = physaddr;
        (*hcp)->desc.cntinfo = count;
        (*hcp)++;
        len -= count;
        physaddr += count;
    }
}

static int dma_setup(Scsi_Cmnd *cmd, int datainp)
{
    struct WD33C93_hostdata *hdata = (struct WD33C93_hostdata *)cmd->host->hostdata;
    wd33c93_regs *regp = hdata->regp;
    struct hpc3_scsiregs *hregs = (struct hpc3_scsiregs *) cmd->host->base;
    struct hpc_chunk *hcp = (struct hpc_chunk *) hdata->dma_bounce_buffer;

#ifdef DEBUG_DMA
    printk("dma_setup: datainp<%d> hcp<%p> ",
           datainp, hcp);
#endif

    hdata->dma_dir = datainp;

    if(cmd->SCp.buffers_residual) {
        struct scatterlist *slp = cmd->SCp.buffer;
        int i, totlen = 0;

#ifdef DEBUG_DMA
        printk("SCLIST<");
#endif
        for(i = 0; i <= cmd->SCp.buffers_residual; i++) {
#ifdef DEBUG_DMA
            printk("[%p,%d]", slp[i].address, slp[i].length);
#endif
            fill_hpc_entries (&hcp, slp[i].address, slp[i].length);
            totlen += slp[i].length;
        }
#ifdef DEBUG_DMA
        printk(">tlen<%d>", totlen);
#endif
        hdata->dma_bounce_len = totlen; /* a trick... */
        write_wd33c93_count(regp, totlen);
    } else {
        /* Non-scattered dma. */
#ifdef DEBUG_DMA
        printk("ONEBUF<%p,%d>", cmd->SCp.ptr, cmd->SCp.this_residual);
#endif
        /*
         * wd33c93 shouldn't pass us bogus dma_setups, but
         * it does:-( The other wd33c93 drivers deal with
         * it the same way (which isn't that obvious).
         * IMHO a better fix would be, not to do these
         * dma setups in the first place
         */
        if (cmd->SCp.ptr == NULL)
            return 1;
        fill_hpc_entries (&hcp, cmd->SCp.ptr,cmd->SCp.this_residual);
        write_wd33c93_count(regp, cmd->SCp.this_residual);
    }

    /* To make sure, if we trip an HPC bug, that we transfer
     * every single byte, we tag on an extra zero length dma
     * descriptor at the end of the chain.
     */
    hcp->desc.pbuf = 0;
    hcp->desc.cntinfo = (HPCDMA_EOX);

#ifdef DEBUG_DMA
    printk(" HPCGO\n");
#endif

    /* Start up the HPC. */
    hregs->ndptr = PHYSADDR(hdata->dma_bounce_buffer);
    if(datainp)
        hregs->ctrl = (HPC3_SCTRL_ACTIVE);
    else
        hregs->ctrl = (HPC3_SCTRL_ACTIVE | HPC3_SCTRL_DIR);
    return 0;
}

static void dma_stop(struct Scsi_Host *instance, Scsi_Cmnd *SCpnt,
             int status)
{
    struct WD33C93_hostdata *hdata = (struct WD33C93_hostdata *)instance->hostdata;
    wd33c93_regs *regp = hdata->regp;
    struct hpc3_scsiregs *hregs;

    if (!SCpnt)
        return;

    hregs = (struct hpc3_scsiregs *) SCpnt->host->base;

#ifdef DEBUG_DMA
    printk("dma_stop: status<%d> ", status);
#endif

    /* First stop the HPC and flush it's FIFO. */
    if(hdata->dma_dir) {
        hregs->ctrl |= HPC3_SCTRL_FLUSH;
        while(hregs->ctrl & HPC3_SCTRL_ACTIVE)
            barrier();
    }
    hregs->ctrl = 0;

    /* See how far we got and update scatterlist state if necessary. */
    if(SCpnt->SCp.buffers_residual) {
        struct scatterlist *slp = SCpnt->SCp.buffer;
        int totlen, wd93_residual, transferred, i;

        /* Yep, we were doing the scatterlist thang. */
        totlen = hdata->dma_bounce_len;
        wd93_residual = read_wd33c93_count(regp);
        transferred = totlen - wd93_residual;

#ifdef DEBUG_DMA
        printk("tlen<%d>resid<%d>transf<%d> ",
               totlen, wd93_residual, transferred);
#endif

        /* Avoid long winded partial-transfer search for common case. */
        if(transferred != totlen) {
            /* This is the nut case. */
#ifdef DEBUG_DMA
            printk("Jed was here...");
#endif
            for(i = 0; i <= SCpnt->SCp.buffers_residual; i++) {
                if(slp[i].length >= transferred)
                    break;
                transferred -= slp[i].length;
            }
        } else {
            /* This is the common case. */
#ifdef DEBUG_DMA
            printk("did it all...");
#endif
            i = SCpnt->SCp.buffers_residual;
        }
        SCpnt->SCp.buffer = &slp[i];
        SCpnt->SCp.buffers_residual = SCpnt->SCp.buffers_residual - i;
        SCpnt->SCp.ptr = (char *) slp[i].address;
        SCpnt->SCp.this_residual = slp[i].length;
    }
#ifdef DEBUG_DMA
    printk("\n");
#endif
}

void sgiwd93_reset(unsigned long base)
{
    struct hpc3_scsiregs *hregs = (struct hpc3_scsiregs *) base;

    hregs->ctrl = HPC3_SCTRL_CRESET;
    udelay (50);
    hregs->ctrl = 0;
}

static inline void init_hpc_chain(uchar *buf)
{
    struct hpc_chunk *hcp = (struct hpc_chunk *) buf;
    unsigned long start, end;

    start = (unsigned long) buf;
    end = start + PAGE_SIZE;
    while(start < end) {
        hcp->desc.pnext = PHYSADDR((hcp + 1));
        hcp->desc.cntinfo = HPCDMA_EOX;
        hcp++;
        start += sizeof(struct hpc_chunk);
    };
    hcp--;
    hcp->desc.pnext = PHYSADDR(buf);
}

int __init sgiwd93_detect(Scsi_Host_Template *SGIblows)
{
    static unsigned char called = 0;
    struct hpc3_scsiregs *hregs = &hpc3c0->scsi_chan0;
    struct hpc3_scsiregs *hregs1 = &hpc3c0->scsi_chan1;
    struct WD33C93_hostdata *hdata;
    struct WD33C93_hostdata *hdata1;
    uchar *buf;
    
    if(called)
        return 0; /* Should bitch on the console about this... */

    SGIblows->proc_name = "SGIWD93";

    sgiwd93_host = scsi_register(SGIblows, sizeof(struct WD33C93_hostdata));
    if(sgiwd93_host == NULL)
        return 0;
    sgiwd93_host->base = (unsigned long) hregs;
    sgiwd93_host->irq = SGI_WD93_0_IRQ;

    buf = (uchar *) get_free_page(GFP_KERNEL);
    if (!buf) {
        printk(KERN_WARNING "sgiwd93: Could not allocate memory for host0 buffer.\n");
        scsi_unregister(sgiwd93_host);
        return 0;
    }
    init_hpc_chain(buf);
    dma_cache_wback_inv((unsigned long) buf, PAGE_SIZE);
    /* HPC_SCSI_REG0 | 0x03 | KSEG1 */
    wd33c93_init(sgiwd93_host, (wd33c93_regs *) KSEG1ADDR (0x1fbc0003),
             dma_setup, dma_stop, WD33C93_FS_16_20);

    hdata = (struct WD33C93_hostdata *)sgiwd93_host->hostdata;
    hdata->no_sync = 0;
    hdata->dma_bounce_buffer = (uchar *) (KSEG1ADDR(buf));

    if (request_irq(SGI_WD93_0_IRQ, sgiwd93_intr, 0, "SGI WD93", (void *) sgiwd93_host)) {
        printk(KERN_WARNING "sgiwd93: Could not register IRQ %d (for host 0).\n", SGI_WD93_0_IRQ);
#ifdef MODULE
        wd33c93_release();
#endif
        free_page((unsigned long)buf);
        scsi_unregister(sgiwd93_host);
        return 0;
    }
        /* set up second controller on the Indigo2 */
    if(!sgi_guiness) {
        sgiwd93_host1 = scsi_register(SGIblows, sizeof(struct WD33C93_hostdata));
        if(sgiwd93_host1 != NULL)
        {
            sgiwd93_host1->base = (unsigned long) hregs1;
            sgiwd93_host1->irq = SGI_WD93_1_IRQ;
    
            buf = (uchar *) get_free_page(GFP_KERNEL);
            if (!buf) {
                printk(KERN_WARNING "sgiwd93: Could not allocate memory for host1 buffer.\n");
                scsi_unregister(sgiwd93_host1);
                called = 1;
                return 1; /* We registered host0 so return success*/
            }
            init_hpc_chain(buf);
            dma_cache_wback_inv((unsigned long) buf, PAGE_SIZE);
            /* HPC_SCSI_REG1 | 0x03 | KSEG1 */
            wd33c93_init(sgiwd93_host1, (wd33c93_regs *) KSEG1ADDR (0x1fbc8003),
                     dma_setup, dma_stop, WD33C93_FS_16_20);
    
            hdata1 = (struct WD33C93_hostdata *)sgiwd93_host1->hostdata;
            hdata1->no_sync = 0;
            hdata1->dma_bounce_buffer = (uchar *) (KSEG1ADDR(buf));
            dma_cache_wback_inv((unsigned long) buf, PAGE_SIZE);
    
            if (request_irq(SGI_WD93_1_IRQ, sgiwd93_intr, 0, "SGI WD93", (void *) sgiwd93_host1)) {
                printk(KERN_WARNING "sgiwd93: Could not allocate irq %d (for host1).\n", SGI_WD93_1_IRQ);
#ifdef MODULE
                wd33c93_release();
#endif
                free_page((unsigned long)buf);
                scsi_unregister(sgiwd93_host1);
                /* Fall through since host0 registered OK */
            }
        }
    }
    
    called = 1;

    return 1; /* Found one. */
}

#define HOSTS_C

#include "sgiwd93.h"

static Scsi_Host_Template driver_template = SGIWD93_SCSI;

#include "scsi_module.c"

int sgiwd93_release(struct Scsi_Host *instance)
{
#ifdef MODULE
    free_irq(SGI_WD93_0_IRQ, sgiwd93_intr);
    free_page(KSEG0ADDR(hdata->dma_bounce_buffer));
    wd33c93_release();
    if(!sgi_guiness) {
        free_irq(SGI_WD93_1_IRQ, sgiwd93_intr);
        free_page(KSEG0ADDR(hdata1->dma_bounce_buffer));
        wd33c93_release();
    }
#endif
    return 1;
}

:: Command execute ::

Enter:
 
Select:
 

:: Search ::
  - regexp 

:: Upload ::
 
[ Read-Only ]

:: Make Dir ::
 
[ Read-Only ]
:: Make File ::
 
[ Read-Only ]

:: Go Dir ::
 
:: Go File ::
 

--[ c99shell v. 1.0 pre-release build #13 powered by Captain Crunch Security Team | http://ccteam.ru | Generation time: 0.009 ]--