Annotation of 43BSDReno/sys/tahoevba/vba.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * Copyright (c) 1987, 1988 Regents of the University of California.
                      3:  * All rights reserved.
                      4:  *
                      5:  * Redistribution is only permitted until one year after the first shipment
                      6:  * of 4.4BSD by the Regents.  Otherwise, redistribution and use in source and
                      7:  * binary forms are permitted provided that: (1) source distributions retain
                      8:  * this entire copyright notice and comment, and (2) distributions including
                      9:  * binaries display the following acknowledgement:  This product includes
                     10:  * software developed by the University of California, Berkeley and its
                     11:  * contributors'' in the documentation or other materials provided with the
                     12:  * distribution and in all advertising materials mentioning features or use
                     13:  * of this software.  Neither the name of the University nor the names of
                     14:  * its contributors may be used to endorse or promote products derived from
                     15:  * this software without specific prior written permission.
                     16:  * THIS SOFTWARE IS PROVIDED AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
                     17:  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
                     18:  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
                     19:  *
                     20:  *     @(#)vba.c       7.5 (Berkeley) 6/28/90
                     21:  */
                     22: 
                     23: /*
                     24:  * Tahoe VERSAbus adapator support routines.
                     25:  */
                     26: 
                     27: #include "param.h"
                     28: #include "buf.h"
                     29: #include "cmap.h"
                     30: #include "conf.h"
                     31: #include "dk.h"
                     32: #include "map.h"
                     33: #include "systm.h"
                     34: #include "user.h"
                     35: #include "vmparam.h"
                     36: #include "vmmac.h"
                     37: #include "proc.h"
                     38: #include "syslog.h"
                     39: #include "malloc.h"
                     40: 
                     41: #include "../tahoe/mtpr.h"
                     42: #include "../tahoe/pte.h"
                     43: 
                     44: #include "../tahoevba/vbavar.h"
                     45: 
                     46: /*
                     47:  * Allocate private page map and intermediate buffer
                     48:  * for a VERSAbus device, large enough for maximum transfer size.
                     49:  * Intermediate buffer 
                     50:  * Make intermediate buffer uncacheable.
                     51:  */
                     52: vbainit(vb, xsize, flags)
                     53:        register struct vb_buf *vb;
                     54:        int xsize, flags;
                     55: {
                     56:        register struct pte *pte;
                     57:        register n;
                     58: 
                     59:        vb->vb_flags = flags;
                     60:        if (vbmapalloc((int)btoc(xsize) + 1, &vb->vb_map, &vb->vb_utl) == 0) {
                     61:                printf("vbmap exhausted\n");
                     62:                return (0);
                     63:        }
                     64:        n = roundup(xsize, NBPG);
                     65:        vb->vb_bufsize = n;
                     66:        if (vb->vb_rawbuf == 0)
                     67:                vb->vb_rawbuf = (caddr_t)malloc((u_long)n, M_DEVBUF, M_NOWAIT);
                     68:        if (vb->vb_rawbuf == 0) {
                     69:                printf("no memory for device buffer\n");
                     70:                return (0);
                     71:        }
                     72:        if ((int)vb->vb_rawbuf & PGOFSET)
                     73:                panic("vbinit pgoff");
                     74:        vb->vb_physbuf = (u_long)kvtophys(vb->vb_rawbuf);
                     75:        if (flags & VB_20BIT)
                     76:                vb->vb_maxphys = btoc(VB_MAXADDR20);
                     77:        else if (flags & VB_24BIT)
                     78:                vb->vb_maxphys = btoc(VB_MAXADDR24);
                     79:        else
                     80:                vb->vb_maxphys = btoc(VB_MAXADDR32);
                     81:        if (btoc(vb->vb_physbuf + n) > vb->vb_maxphys)
                     82:                panic("vbinit physbuf");
                     83:        
                     84:        /*
                     85:         * Make raw buffer pages uncacheable.
                     86:         */
                     87:        pte = kvtopte(vb->vb_rawbuf);
                     88:        for (n = btoc(n); n--; pte++)
                     89:                pte->pg_nc = 1;
                     90:        mtpr(TBIA, 0);
                     91:        return (1);
                     92: }
                     93: 
                     94: /*
                     95:  * Due to unknown hardware or software errors, some sites have problems
                     96:  * with strange crashes or corruption of text images when DMA is attempted
                     97:  * to kernel addresses spanning a page boundary, or to user addresses
                     98:  * (even if the buffer is physically contiguous).  To avoid this behavior,
                     99:  * the following toggles inhibit such transfers when set.
                    100:  *     vba_copyk: copy transfers to kernel address that span a page boundary
                    101:  *     vba_copyu: copy transfers to user addresses
                    102:  */
                    103: #ifndef VBA_TRICKY
                    104: int vba_copyk = 1;
                    105: int vba_copyu = 1;
                    106: #else
                    107: int vba_copyk = 0;
                    108: int vba_copyu = 0;
                    109: #endif
                    110: 
                    111: /*
                    112:  * Check a transfer to see whether it can be done directly
                    113:  * to the destination buffer, or whether it must be copied.
                    114:  * On Tahoe, the lack of a bus I/O map forces data to be copied
                    115:  * to a physically-contiguous buffer whenever one of the following is true:
                    116:  *     1) The data length is not a multiple of sector size.
                    117:  *        (The swapping code does this, unfortunately.)
                    118:  *     2) The buffer is not physically contiguous and the controller
                    119:  *        does not support scatter-gather operations.
                    120:  *     3) The physical address for I/O is higher than addressible
                    121:  *        by the device.
                    122:  * This routine is called by the start routine.
                    123:  * If copying is necessary, the intermediate buffer is mapped;
                    124:  * if the operation is a write, the data is copied into the buffer.
                    125:  * It returns the physical address of the first byte for DMA, to
                    126:  * be presented to the controller.
                    127:  */
                    128: u_long
                    129: vbasetup(bp, vb, sectsize)
                    130:        register struct buf *bp;
                    131:        register struct vb_buf *vb;
                    132:        int sectsize;
                    133: {
                    134:        register struct pte *spte, *dpte;
                    135:        register int p, i;
                    136:        int npf, o, v;
                    137: 
                    138:        o = (int)bp->b_un.b_addr & PGOFSET;
                    139:        npf = btoc(bp->b_bcount + o);
                    140:        vb->vb_iskernel = (((int)bp->b_un.b_addr & KERNBASE) == KERNBASE);
                    141:        if (vb->vb_iskernel) {
                    142:                spte = kvtopte(bp->b_un.b_addr);
                    143: if (vba_copyk && (o != 0 || npf > 1)) goto copy;
                    144:        } else {
                    145:                spte = vtopte((bp->b_flags&B_DIRTY) ? &proc[2] : bp->b_proc,
                    146:                    btop(bp->b_un.b_addr));
                    147: if (vba_copyu) goto copy;
                    148:        }
                    149:        if (bp->b_bcount % sectsize != 0 || (o & (sizeof(long) - 1)) != 0)
                    150:                goto copy;
                    151:        else if ((vb->vb_flags & VB_SCATTER) == 0 ||
                    152:            vb->vb_maxphys != btoc(VB_MAXADDR32)) {
                    153:                dpte = spte;
                    154:                p = (dpte++)->pg_pfnum;
                    155:                for (i = npf; --i > 0; dpte++) {
                    156:                        if ((v = dpte->pg_pfnum) != p + CLSIZE &&
                    157:                            (vb->vb_flags & VB_SCATTER) == 0)
                    158:                                goto copy;
                    159:                        if (p >= vb->vb_maxphys)
                    160:                                goto copy;
                    161:                        p = v;
                    162:                }
                    163:                if (p >= vb->vb_maxphys)
                    164:                        goto copy;
                    165:        }
                    166:        vb->vb_copy = 0;
                    167:        if (vb->vb_iskernel)
                    168:                vbastat.k_raw++;
                    169:        else
                    170:                vbastat.u_raw++;
                    171:        return ((spte->pg_pfnum << PGSHIFT) + o);
                    172: 
                    173: copy:
                    174:        vb->vb_copy = 1;
                    175:        if (bp->b_bcount > vb->vb_bufsize)
                    176:                panic("vba xfer too large");
                    177:        if (vb->vb_iskernel) {
                    178:                if ((bp->b_flags & B_READ) == 0)
                    179:                        bcopy(bp->b_un.b_addr, vb->vb_rawbuf,
                    180:                            (unsigned)bp->b_bcount);
                    181:                vbastat.k_copy++;
                    182:        } else  {
                    183:                dpte = vb->vb_map;
                    184:                for (i = npf, p = (int)vb->vb_utl; i--; p += NBPG) {
                    185:                        *(int *)dpte++ = (spte++)->pg_pfnum |
                    186:                            PG_V | PG_KW | PG_N;
                    187:                        mtpr(TBIS, p);
                    188:                }
                    189:                if ((bp->b_flags & B_READ) == 0)
                    190:                        bcopy(vb->vb_utl + o, vb->vb_rawbuf,
                    191:                            (unsigned)bp->b_bcount);
                    192:                vbastat.u_copy++;
                    193:        }
                    194:        return (vb->vb_physbuf);
                    195: }
                    196: 
                    197: /*
                    198:  * Called by the driver's interrupt routine, after DMA is completed.
                    199:  * If the operation was a read, copy data to final buffer if necessary
                    200:  * or invalidate data cache for cacheable direct buffers.
                    201:  * Similar to the vbastart routine, but in the reverse direction.
                    202:  */
                    203: vbadone(bp, vb)
                    204:        register struct buf *bp;
                    205:        register struct vb_buf *vb;
                    206: {
                    207:        register npf;
                    208:        register caddr_t v;
                    209:        int o;
                    210: 
                    211:        if (bp->b_flags & B_READ) {
                    212:                o = (int)bp->b_un.b_addr & PGOFSET;
                    213:                if (vb->vb_copy) {
                    214:                        if (vb->vb_iskernel)
                    215:                                bcopy(vb->vb_rawbuf, bp->b_un.b_addr,
                    216:                                    (unsigned)(bp->b_bcount - bp->b_resid));
                    217:                        else {
                    218:                                bcopy(vb->vb_rawbuf, vb->vb_utl + o,
                    219:                                    (unsigned)(bp->b_bcount - bp->b_resid));
                    220:                                dkeyinval(bp->b_proc);
                    221:                        }
                    222:                } else {
                    223:                        if (vb->vb_iskernel) {
                    224:                                npf = btoc(bp->b_bcount + o);
                    225:                                for (v = bp->b_un.b_addr; npf--; v += NBPG)
                    226:                                        mtpr(P1DC, (int)v);
                    227:                        } else
                    228:                                dkeyinval(bp->b_proc);
                    229:                }
                    230:        }
                    231: }
                    232: 
                    233: /*
                    234:  * Set up a scatter-gather operation for SMD/E controller.
                    235:  * This code belongs half-way between {hd,vd}.c and this file.
                    236:  */
                    237: 
                    238: #include "dk.h"
                    239: #if NVD > 0
                    240: #include "vdreg.h"
                    241: 
                    242: vd_sgsetup(bp, vb, sg)
                    243:        register struct buf *bp;
                    244:        struct vb_buf *vb;
                    245:        struct trsg *sg;
                    246: {
                    247:        register struct pte *spte;
                    248:        register struct addr_chain *adr;
                    249:        register int i;
                    250:        int o;
                    251: 
                    252:        vb->vb_iskernel = (((int)bp->b_un.b_addr & KERNBASE) == KERNBASE);
                    253:        vb->vb_copy = 0;
                    254:        if (vb->vb_iskernel) {
                    255:                spte = kvtopte(bp->b_un.b_addr);
                    256:                vbastat.k_sg++;
                    257:        } else {
                    258:                spte = vtopte((bp->b_flags&B_DIRTY) ? &proc[2] : bp->b_proc,
                    259:                    btop(bp->b_un.b_addr));
                    260:                vbastat.u_sg++;
                    261:        }
                    262: 
                    263:        o = (int)bp->b_un.b_addr & PGOFSET;
                    264:        i = min(NBPG - o, bp->b_bcount);
                    265:        sg->start_addr.wcount = i >> 1;
                    266:        sg->start_addr.memadr = ((spte++)->pg_pfnum << PGSHIFT) + o;
                    267:        i = bp->b_bcount - i;
                    268:        if (i > VDMAXPAGES * NBPG)
                    269:                panic("vba xfer too large");
                    270:        i = i >> 1;
                    271:        for (adr = sg->addr_chain; i > 0; adr++, i -= NBPG / 2) {
                    272:                adr->nxt_addr = (spte++)->pg_pfnum << PGSHIFT;
                    273:                adr->nxt_len = imin(i, NBPG / 2);
                    274:        }
                    275:        adr->nxt_addr = 0;
                    276:        adr++->nxt_len = 0;
                    277:        return ((adr - sg->addr_chain) * sizeof(*adr) / sizeof(long));
                    278: }
                    279: #endif
                    280: 
                    281: #include "hd.h"
                    282: #if NHD > 0
                    283: #include "hdreg.h"
                    284: 
                    285: hd_sgsetup(bp, vb, sg)
                    286:        register struct buf *bp;
                    287:        struct vb_buf *vb;
                    288:        struct chain *sg;
                    289: {
                    290:        register struct pte *spte;
                    291:        register struct addr_chain *adr;
                    292:        register int i, cnt;
                    293:        int o;
                    294: 
                    295:        if (bp->b_bcount > HDC_MAXBC ||
                    296:            bp->b_bcount % sizeof(long) - 1 ||
                    297:            (u_int)bp->b_un.b_addr % sizeof(long) - 1)
                    298:                return(0);
                    299: 
                    300:        vb->vb_iskernel = (((int)bp->b_un.b_addr & KERNBASE) == KERNBASE);
                    301:        vb->vb_copy = 0;
                    302:        if (vb->vb_iskernel) {
                    303:                spte = kvtopte(bp->b_un.b_addr);
                    304:                vbastat.k_sg++;
                    305:        } else {
                    306:                spte = vtopte((bp->b_flags&B_DIRTY) ? &proc[2] : bp->b_proc,
                    307:                    btop(bp->b_un.b_addr));
                    308:                vbastat.u_sg++;
                    309:        }
                    310: 
                    311:        o = (int)bp->b_un.b_addr & PGOFSET;
                    312:        i = min(NBPG - o, bp->b_bcount);
                    313:        sg->wcount = i >> 2;
                    314:        sg->memadr = ((spte++)->pg_pfnum << PGSHIFT) + o;
                    315:        cnt = 0;
                    316:        for (i = (bp->b_bcount - i) >> 2; i > 0; i -= NBPG / sizeof(long)) {
                    317:                if (++cnt == HDC_MAXCHAIN)
                    318:                        return(0);
                    319:                sg->wcount |= LWC_DATA_CHAIN;
                    320:                ++sg;
                    321:                sg->wcount = imin(i, NBPG / sizeof(long));
                    322:                sg->memadr = (spte++)->pg_pfnum << PGSHIFT;
                    323:        }
                    324:        return(1);
                    325: }
                    326: #endif

unix.superglobalmegacorp.com

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