Annotation of coherent/b/kernel/io.386/dmareq.c, revision 1.1.1.1

1.1       root        1: /* dmareq.c */
                      2: /* (lgl-
                      3:  *     The information contained herein is a trade secret of Mark Williams
                      4:  *     Company, and  is confidential information.  It is provided  under a
                      5:  *     license agreement,  and may be  copied or disclosed  only under the
                      6:  *     terms of  that agreement.  Any  reproduction or disclosure  of this
                      7:  *     material without the express written authorization of Mark Williams
                      8:  *     Company or persuant to the license agreement is unlawful.
                      9:  *
                     10:  *     COHERENT Version 4.1.0
                     11:  *     Copyright (c) 1993.
                     12:  *     An unpublished work by Mark Williams Company, Chicago.
                     13:  *     All rights reserved.
                     14:  -lgl) */
                     15: 
                     16: /*
                     17:  * Like ioreq, but guarantee that no DMA straddle occurs.
                     18:  * And assume we are called by fl.c, xt.c, dv.c or someone
                     19:  * else who obeys the parameter rules that they do.
                     20:  *
                     21:  */
                     22: #include <sys/coherent.h>
                     23: #include <sys/buf.h>
                     24: #include <sys/con.h>
                     25: #include <errno.h>
                     26: #include <sys/io.h>
                     27: #include <sys/proc.h>
                     28: #include <sys/sched.h>
                     29: #include <sys/seg.h>
                     30: #include <sys/stat.h>
                     31: #include <sys/dmac.h>
                     32: 
                     33: /*
                     34:  * NIGEL: Why the hell is this routine not in bio.c? It would be a whole lot
                     35:  * nicer if it was. "dold_t" is obsolete, by the way, and all instances of it
                     36:  * will disappear from "bio.c" sometime. I should merge this in as well then.
                     37:  */
                     38: typedef        unsigned char           dold_t;
                     39: 
                     40: dmareq(bp, iop, dev, req)
                     41: register BUF *bp;
                     42: register IO *iop;
                     43: dev_t dev;
                     44: {
                     45:        register int to_read;
                     46:        register SEG *sp;
                     47:        register CON *cp;
                     48:        dold_t dold;
                     49:        long next_block;        /* Next block to be read.  */
                     50:        paddr_t last;           /* Address of last byte we will read.  */
                     51:        paddr_t next_seg;       /* Beginning of next dma segment.  */
                     52:        BUF *tbp;
                     53: 
                     54:        if ((cp=drvmap(dev, &dold)) == NULL)
                     55:                return;
                     56:        lock(bp->b_gate);
                     57:        drest(dold);
                     58:        if (blocko(iop->io_seek)) {
                     59:                SET_U_ERROR(EIO, "dmareq() seek");
                     60:                goto out;
                     61:        }
                     62:        if ((sp=iomapvp(iop, bp)) == NULL) {
                     63:                SET_U_ERROR(EIO, "dmareq() iomapvp");
                     64:                goto out;
                     65:        }
                     66:        bp->b_dev = dev;
                     67:        bp->b_flag = 0;
                     68:        sp->s_lrefc++;
                     69:        /*
                     70:         * The dma address is 20 bits; 16 bit offset counter from a 4 bit
                     71:         * base segment.  Since io_ioc is limited to 32Kb positive, we
                     72:         * have at most two raw transfers separated by a block which
                     73:         * straddles the segment boundary.
                     74:         * Life would be simpler if we assumed io_ioc % BSIZE, but
                     75:         * flioctl comes through here with its short format buffer.
                     76:         */
                     77:        while (iop->io_ioc > 0 && (bp->b_flag&BFERR) == 0) {
                     78:                /*
                     79:                 * If the first and last bytes to be read are on different
                     80:                 * dma segments, we have a stradle, and must read a fraction
                     81:                 * of the entire animal.
                     82:                 *
                     83:                 * I don't understand why a fractional block at the
                     84:                 * top of a segment is not read.
                     85:                 */
                     86:                last = bp->b_paddr+iop->io_ioc-1;
                     87:                if (dmaseg(last) != dmaseg(bp->b_paddr)) {
                     88:                        /*
                     89:                         * We have a straddle--read at most to the end
                     90:                         * of the next segment.
                     91:                         */
                     92:                        next_seg = dmaseg(bp->b_paddr + DMASEG_SIZE);
                     93:                        to_read = (next_seg - bp->b_paddr) & ~((long)BSIZE-1);
                     94:                } else {
                     95:                        /* No straddle.  */
                     96:                        to_read = iop->io_ioc;
                     97:                }
                     98:                next_block = blockn(iop->io_seek);
                     99:                if (to_read == 0) {
                    100:                        /* Straddle block */
                    101:                        tbp = bp;               /* Save the raw buffer */
                    102:                        to_read = BSIZE;
                    103:                        if (to_read > iop->io_ioc)
                    104:                                to_read = iop->io_ioc;
                    105:                        bp = bclaim(dev, next_block);
                    106:                        bp->b_count = to_read;
                    107:                        bp->b_req = req;
                    108:                        if (req != BREAD)
                    109:                                ioread(iop, bp->b_vaddr, to_read);
                    110:                        dmabuf(bp, dev);
                    111:                        if ((bp->b_flag&BFERR) == 0) {
                    112:                                if (req == BREAD)
                    113:                                        iowrite(iop, bp->b_vaddr, to_read);
                    114:                        } else {
                    115:                                tbp->b_flag = bp->b_flag;
                    116:                                tbp->b_err = bp->b_err;
                    117:                                if (req != BREAD)
                    118:                                        iop->io_ioc += bp->b_resid;
                    119:                        }
                    120:                        bp->b_flag |= BFERR;
                    121:                        brelease(bp);
                    122:                        bp = tbp;               /* Reclaim raw buffer */
                    123:                } else {
                    124:                        /* Raw transfer */
                    125:                        bp->b_count = to_read;
                    126:                        bp->b_req = req;
                    127:                        bp->b_bno = next_block;
                    128:                        dmabuf(bp, dev);
                    129:                        if (bp->b_flag & BFERR)
                    130:                                to_read -= bp->b_resid;
                    131:                        iop->io_ioc -= to_read;    /* cookedio do these */
                    132:                        iop->io.pbase += to_read;  /* for everyone */
                    133:                }
                    134:                bp->b_vaddr += to_read;
                    135:                bp->b_paddr  += to_read;
                    136:                iop->io_seek += to_read;
                    137:                /* And continue for the next chunk */
                    138:        }
                    139:        sp->s_lrefc--;
                    140:        if ( stimer.t_last != 0 )
                    141:                wakeup((char *)&stimer);
                    142:        if ((bp->b_flag&BFERR) && (u.u_error = bp->b_err) == 0)
                    143:                SET_U_ERROR(EIO, "dmareq() BFERR");
                    144: out:
                    145:        unlock(bp->b_gate);
                    146: }
                    147: 
                    148: /*
                    149:  * dmabuf()
                    150:  *
                    151:  * Hand off a buf request to the block handler and go to sleep.
                    152:  */
                    153: static
                    154: dmabuf(bp, dev)
                    155: register BUF *bp;
                    156: dev_t dev;
                    157: {
                    158:        register int s;
                    159: 
                    160:        bp->b_flag = BFRAW|BFBLK|BFIOC|BFNTP;
                    161:        s = sphi();
                    162:        dblock(dev, bp);
                    163:        while (bp->b_flag&BFNTP)
                    164:                x_sleep((char *)bp, pridisk, slpriNoSig, "dmabuf");
                    165:        spl(s);
                    166: }

unix.superglobalmegacorp.com

This archive runs on limited infrastructure. Preserving old code on modern bandwidth. Automated agents are requested to crawl responsibly.