Annotation of 43BSDReno/sys/kern/vm_swap.c, revision 1.1

1.1     ! root        1: /*
        !             2:  * Copyright (c) 1982, 1986 Regents of the University of California.
        !             3:  * All rights reserved.  The Berkeley software License Agreement
        !             4:  * specifies the terms and conditions for redistribution.
        !             5:  *
        !             6:  *     @(#)vm_swap.c   7.7 (Berkeley) 6/30/90
        !             7:  */
        !             8: 
        !             9: #include "param.h"
        !            10: #include "systm.h"
        !            11: #include "user.h"
        !            12: #include "proc.h"
        !            13: #include "text.h"
        !            14: #include "map.h"
        !            15: #include "buf.h"
        !            16: #include "cmap.h"
        !            17: #include "vm.h"
        !            18: 
        !            19: #include "machine/cpu.h"
        !            20: #include "machine/pte.h"
        !            21: #include "machine/mtpr.h"
        !            22: 
        !            23: /*
        !            24:  * Swap a process in.
        !            25:  */
        !            26: swapin(p)
        !            27:        register struct proc *p;
        !            28: {
        !            29:        register struct text *xp;
        !            30:        register int i, s;
        !            31: 
        !            32:        if (xp = p->p_textp) 
        !            33:                xlock(xp);
        !            34:        p->p_szpt = ptsize(p);
        !            35:        if (vgetpt(p, memall) == 0)
        !            36:                goto nomem;
        !            37:        if (vgetu(p, memall, Swapmap, &swaputl, (struct user *)0) == 0) {
        !            38:                vrelpt(p);
        !            39:                goto nomem;
        !            40:        }
        !            41: 
        !            42: #if defined(tahoe)
        !            43:        for (i = 0; i < UPAGES; i++)
        !            44:                mtpr(P1DC, (caddr_t)&u+i*NBPG);
        !            45: #endif
        !            46:        swdspt(p, &swaputl, B_READ);
        !            47:        /*
        !            48:         * Make sure swdspt didn't smash u. pte's
        !            49:         */
        !            50:        for (i = 0; i < UPAGES; i++) {
        !            51:                if (Swapmap[i].pg_pfnum != p->p_addr[i].pg_pfnum)
        !            52:                        panic("swapin");
        !            53:        }
        !            54:        vrelswu(p, &swaputl);
        !            55:        if (xp) {
        !            56:                xlink(p);
        !            57:                xunlock(xp);
        !            58:        }
        !            59: 
        !            60:        p->p_rssize = 0;
        !            61:        s = splclock();
        !            62:        if (p->p_stat == SRUN)
        !            63:                setrq(p);
        !            64:        p->p_flag |= SLOAD;
        !            65:        if (p->p_flag & SSWAP) {
        !            66:                swaputl.u_pcb.pcb_sswap = (int *)&u.u_ssave;
        !            67:                p->p_flag &= ~SSWAP;
        !            68:        }
        !            69:        splx(s);
        !            70:        p->p_time = 0;
        !            71:        multprog++;
        !            72:        cnt.v_swpin++;
        !            73:        return (1);
        !            74: 
        !            75: nomem:
        !            76:        if (xp)
        !            77:                xunlock(xp);
        !            78:        return (0);
        !            79: }
        !            80: 
        !            81: int    xswapwant, xswaplock;
        !            82: /*
        !            83:  * Swap out process p.
        !            84:  * ds and ss are the old data size and the stack size
        !            85:  * of the process, and are supplied during page table
        !            86:  * expansion swaps.
        !            87:  */
        !            88: swapout(p, ds, mms, ss)
        !            89:        register struct proc *p;
        !            90:        segsz_t ds, mms, ss;
        !            91: {
        !            92:        register struct pte *map;
        !            93:        register struct user *utl;
        !            94:        int s;
        !            95:        int rc = 1;
        !            96: 
        !            97:        s = 1;
        !            98:        map = Xswapmap;
        !            99:        utl = &xswaputl;
        !           100:        if (xswaplock & s)
        !           101:                if ((xswaplock & 2) == 0) {
        !           102:                        s = 2;
        !           103:                        map = Xswap2map;
        !           104:                        utl = &xswap2utl;
        !           105:                }
        !           106:        while (xswaplock & s) {
        !           107:                xswapwant |= s;
        !           108:                sleep((caddr_t)map, PSWP);
        !           109:        }
        !           110:        xswaplock |= s;
        !           111:        uaccess(p, map, utl);
        !           112:        if (vgetswu(p, utl) == 0) {
        !           113:                p->p_flag |= SLOAD;
        !           114:                rc = 0;
        !           115:                goto out;
        !           116:        }
        !           117: #if defined(tahoe)
        !           118:        { int i;
        !           119:          for (i = 0; i < UPAGES; i++)
        !           120:                mtpr(P1DC, (caddr_t)&u+i*NBPG);
        !           121:        }
        !           122: #endif
        !           123:        utl->u_ru.ru_nswap++;
        !           124:        utl->u_odsize = ds + mms;
        !           125:        utl->u_ossize = ss;
        !           126:        p->p_flag |= SLOCK;
        !           127:        if (p->p_textp) {
        !           128:                if (p->p_textp->x_ccount == 1)
        !           129:                        p->p_textp->x_swrss = p->p_textp->x_rssize;
        !           130:                xdetach(p->p_textp, p);
        !           131:        }
        !           132:        p->p_swrss = p->p_rssize;
        !           133:        vsswap(p, dptopte(p, 0), CDATA, 0, (int)ds, &utl->u_dmap);
        !           134:        vsswap(p, sptopte(p, CLSIZE-1), CSTACK, 0, (int)ss, &utl->u_smap);
        !           135:        if (p->p_rssize != 0)
        !           136:                panic("swapout rssize");
        !           137: 
        !           138:        swdspt(p, utl, B_WRITE);
        !           139:        /*
        !           140:         * If freeing the user structure and kernel stack
        !           141:         * for the current process, have to run a bit longer
        !           142:         * using the pages which are about to be freed...
        !           143:         * vrelu will then block memory allocation by raising ipl.
        !           144:         */
        !           145:        vrelu(p, 1);
        !           146:        if ((p->p_flag & SLOAD) && (p->p_stat != SRUN || p != u.u_procp))
        !           147:                panic("swapout");
        !           148:        p->p_flag &= ~SLOAD;
        !           149:        vrelpt(p);
        !           150:        p->p_flag &= ~SLOCK;
        !           151:        p->p_time = 0;
        !           152: 
        !           153:        multprog--;
        !           154:        cnt.v_swpout++;
        !           155: 
        !           156:        if (runout) {
        !           157:                runout = 0;
        !           158:                wakeup((caddr_t)&runout);
        !           159:        }
        !           160: out:
        !           161:        xswaplock &= ~s;
        !           162:        if (xswapwant & s) {
        !           163:                xswapwant &= ~s;
        !           164:                wakeup((caddr_t)map);
        !           165:        }
        !           166:        return (rc);
        !           167: }
        !           168: 
        !           169: /*
        !           170:  * Swap the data and stack page tables in or out.
        !           171:  * Only hard thing is swapping out when new pt size is different than old.
        !           172:  * If we are growing new pt pages, then we must spread pages with 2 swaps.
        !           173:  * If we are shrinking pt pages, then we must merge stack pte's into last
        !           174:  * data page so as not to lose them (and also do two swaps).
        !           175:  */
        !           176: swdspt(p, utl, rdwri)
        !           177:        register struct proc *p;
        !           178:        register struct user *utl;
        !           179: {
        !           180:        register int szpt, tsz, ssz;
        !           181:        register struct pte *pte;
        !           182:        register int i;
        !           183: 
        !           184:        szpt = ptsize(p);
        !           185:        tsz = p->p_tsize / NPTEPG;
        !           186:        if (szpt == p->p_szpt) {
        !           187:                swptstat.pteasy++;
        !           188:                swpt(rdwri, p, 0, tsz,
        !           189:                    (p->p_szpt - tsz) * NBPG - HIGHPAGES * sizeof (struct pte));
        !           190:                goto check;
        !           191:        }
        !           192:        if (szpt < p->p_szpt)
        !           193:                swptstat.ptshrink++;
        !           194:        else
        !           195:                swptstat.ptexpand++;
        !           196:        ssz = clrnd(ctopt(utl->u_ossize+HIGHPAGES));
        !           197: #if !defined(hp300) && !defined(i386)
        !           198:        if (szpt < p->p_szpt && utl->u_odsize && (utl->u_ossize+HIGHPAGES)) {
        !           199:                int tdlast, slast, tdsz;
        !           200: 
        !           201:                /*
        !           202:                 * Page tables shrinking... see if last text+data and
        !           203:                 * last stack page must be merged... if so, copy
        !           204:                 * stack pte's from last stack page to end of last
        !           205:                 * data page, and decrease size of stack pt to be swapped.
        !           206:                 */
        !           207:                tdlast = (p->p_tsize + utl->u_odsize) % (NPTEPG * CLSIZE);
        !           208:                slast = (utl->u_ossize + HIGHPAGES) % (NPTEPG * CLSIZE);
        !           209:                if (tdlast && slast && tdlast + slast <= (NPTEPG * CLSIZE)) {
        !           210:                        swptstat.ptpack++;
        !           211:                        tdsz = clrnd(ctopt(p->p_tsize + utl->u_odsize));
        !           212:                        bcopy((caddr_t)sptopte(p, utl->u_ossize - 1),
        !           213:                            (caddr_t)&p->p_p0br[tdsz * NPTEPG - slast],
        !           214:                            (unsigned)(slast * sizeof (struct pte)));
        !           215:                        ssz -= CLSIZE;
        !           216:                }
        !           217:        }
        !           218: #endif
        !           219:        if (ssz)
        !           220:                swpt(rdwri, p, szpt - ssz - tsz, p->p_szpt - ssz, ssz * NBPG);
        !           221:        if (utl->u_odsize)
        !           222:                swpt(rdwri, p, 0, tsz,
        !           223:                  (int)(clrnd(ctopt(p->p_tsize + utl->u_odsize)) - tsz) * NBPG);
        !           224: check:
        !           225:        for (i = 0; i < utl->u_odsize; i++) {
        !           226:                pte = dptopte(p, i);
        !           227: #if defined(tahoe)
        !           228:                uncache(pte);
        !           229: #endif
        !           230: #ifdef MAPMEM
        !           231:                if (pte->pg_v && pte->pg_fod)           /* mapped page */
        !           232:                        continue;
        !           233: #endif
        !           234:                if (pte->pg_v || pte->pg_fod == 0 && (pte->pg_pfnum||pte->pg_m))
        !           235:                        panic("swdspt");
        !           236:        }
        !           237:        for (i = 0; i < utl->u_ossize; i++) {
        !           238:                pte = sptopte(p, i);
        !           239: #if defined(tahoe)
        !           240:                uncache(pte);
        !           241: #endif
        !           242: #ifdef MAPMEM
        !           243:                if (pte->pg_v && pte->pg_fod)           /* mapped page */
        !           244:                        continue;
        !           245: #endif
        !           246:                if (pte->pg_v || pte->pg_fod == 0 && (pte->pg_pfnum||pte->pg_m))
        !           247:                        panic("swdspt");
        !           248:        }
        !           249: }
        !           250: 
        !           251: /*
        !           252:  * Swap a section of the page tables.
        !           253:  * Errors are handled at a lower level (by doing a panic).
        !           254:  */
        !           255: swpt(rdwri, p, doff, a, n)
        !           256:        int rdwri;
        !           257:        struct proc *p;
        !           258:        int doff, a, n;
        !           259: {
        !           260: 
        !           261:        if (n <= 0)
        !           262:                return;
        !           263:        (void) swap(p, p->p_swaddr + ctod(UPAGES) + ctod(doff),
        !           264:            (caddr_t)&p->p_p0br[a * NPTEPG], n, rdwri, B_PAGET, swapdev_vp, 0);
        !           265: }

unix.superglobalmegacorp.com

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