Annotation of 43BSDTahoe/sys/tahoevba/vba.c, revision 1.1

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

unix.superglobalmegacorp.com

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