Annotation of coherent/d/PS2_KERNEL/coh.286/swap.c, revision 1.1.1.1

1.1       root        1: /* $Header: /kernel/kersrc/coh.286/RCS/swap.c,v 1.1 92/07/17 15:18:48 bin Exp Locker: bin $ */
                      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 2.3.37
                     11:  *     Copyright (c) 1982, 1983, 1984.
                     12:  *     An unpublished work by Mark Williams Company, Chicago.
                     13:  *     All rights reserved.
                     14:  -lgl) */
                     15: /*
                     16:  * Coherent.
                     17:  * Swapper.
                     18:  *
                     19:  * $Log:       swap.c,v $
                     20:  * Revision 1.1  92/07/17  15:18:48  bin
                     21:  * Initial revision
                     22:  * 
                     23:  * Revision 1.1        88/03/24  16:19:51      src
                     24:  * Initial revision
                     25:  * 
                     26:  * 87/01/05    Allan Cornish           /usr/src/sys/ker/swap.c
                     27:  * Swap() now waits for all processes to be swapped in before exit on signal.
                     28:  */
                     29: #include <coherent.h>
                     30: #include <proc.h>
                     31: #include <sched.h>
                     32: #include <sys/seg.h>
                     33: #include <sys/buf.h>
                     34: 
                     35: /*
                     36:  * Functions.
                     37:  */
                     38: SEG    *xmalloc();
                     39: SEG    *xdalloc();
                     40: 
                     41: swap()
                     42: {
                     43:        register SEG *sp;
                     44:        register PROC *pp1;
                     45:        register PROC *pp2;
                     46:        register PROC *pp3;
                     47:        register unsigned s;
                     48:        register unsigned n;
                     49:        register unsigned t;
                     50:        register unsigned v;
                     51:        register unsigned m;
                     52:        register int i;
                     53:        static unsigned ltimer;
                     54: 
                     55:        if (sexflag != 0)
                     56:                uexit(1);
                     57:        sexflag++;
                     58:        while (1) {
                     59:                lock(pnxgate);
                     60:                t = (utimer-ltimer)/NSUTICK;
                     61:                v = t*SVCLOCK;
                     62:                ltimer += t*NSUTICK;
                     63:                m = 0;
                     64:                pp2 = NULL;
                     65:                for (pp1=procq.p_nback; pp1!=&procq; pp1=pp1->p_nback) {
                     66:                        if ((pp1->p_flags&PFCORE) != 0) {
                     67:                                pp1->p_sval >>= t;
                     68:                                pp1->p_ival -= t;
                     69:                                if (pp1->p_ival < -30000)
                     70:                                        pp1->p_ival = -30000;
                     71:                                continue;
                     72:                        }
                     73:                        addu(pp1->p_sval, v);
                     74:                        if (pp1->p_state != PSRUN)
                     75:                                continue;
                     76:                        s = 0;
                     77:                        for (i=0; i<NUSEG+1; i++)
                     78:                                if ((sp=pp1->p_segp[i]) != NULL)
                     79:                                        if ((sp->s_flags&SFCORE) == 0)
                     80:                                                s += sp->s_size;
                     81:                        if ((s=ctokrd(s)) == 0)
                     82:                                s = 1;
                     83:                        n = (pp1->p_sval+pp1->p_rval)/s;
                     84:                        if (n > m) {
                     85:                                m = n;
                     86:                                pp2 = pp1;
                     87:                        }
                     88:                }
                     89:                unlock(pnxgate);
                     90:                if (pp2 == NULL) {
                     91:                        if ( SELF->p_ssig != 0 )
                     92:                                break;
                     93:                        goto con;
                     94:                }
                     95: #ifndef        NOMONITOR
                     96:                if (swmflag)
                     97:                        printf("Swapin(%p, %d)\n", pp2, pp2->p_pid);
                     98: #endif
                     99:        xxx:
                    100:                while (testcore(pp2)==0 || proccore(pp2)!=0) {
                    101:                        if ((pp2->p_flags&PFAUXM) != 0) {
                    102:                                auxmdisk(pp2);
                    103:                                goto xxx;
                    104:                        }
                    105:                        procdisk(pp2);
                    106:                        i = 32767;
                    107:                        pp3 = NULL;
                    108:                        lock(pnxgate);
                    109:                        for (pp1=procq.p_nforw; pp1!=&procq; pp1=pp1->p_nforw){
                    110:                                if (pp1->p_flags&(PFSWIO|PFLOCK|PFKERN))
                    111:                                        continue;
                    112:                                if ((pp1->p_flags&PFAUXM) != 0) {
                    113:                                        auxmdisk(pp1);
                    114:                                        unlock(pnxgate);
                    115:                                        goto xxx;
                    116:                                }
                    117:                                if ((pp1->p_flags&PFCORE) == 0) {
                    118:                                        if (procdisk(pp1) != 0) {
                    119:                                                unlock(pnxgate);
                    120:                                                goto xxx;
                    121:                                        }
                    122:                                        continue;
                    123:                                }
                    124:                                if (pp1->p_ival>-64 && pp1->p_sval!=0)
                    125:                                        continue;
                    126:                                if (pp1->p_ival < i) {
                    127:                                        i = pp1->p_ival;
                    128:                                        pp3 = pp1;
                    129:                                }
                    130:                        }
                    131:                        unlock(pnxgate);
                    132:                        if (pp3 == NULL) {
                    133: #ifndef NOMONITOR
                    134:                                if (swmflag)
                    135:                                        printf("No one to swap out\n");
                    136: #endif
                    137:                                break;
                    138:                        }
                    139:                        if (i > 0) {
                    140: #ifndef NOMONITOR
                    141:                                if (swmflag)
                    142:                                        printf("Dispatch(%p, %d)\n",
                    143:                                                pp3, pp3->p_pid);
                    144: #endif
                    145:                                pp3->p_flags |= PFDISP;
                    146:                                break;
                    147:                        }
                    148: #ifndef NOMONITOR
                    149:                        if (swmflag)
                    150:                                printf("Swapout(%p, %d)\n", pp3, pp3->p_pid);
                    151: #endif
                    152:                        procdisk(pp3);
                    153:                }
                    154: #ifndef NOMONITOR
                    155:                if (swmflag)
                    156:                        printf("Swapdone\n");
                    157: #endif
                    158:        con:
                    159:                timeout(&stimer, NSRTICK, wakeup, (char *)&stimer);
                    160:                sleep((char *)&stimer, CVSWAP, IVSWAP, SVSWAP);
                    161:        }
                    162:        --sexflag;
                    163:        uexit(1);
                    164: }
                    165: 
                    166: /*
                    167:  * See if the given process may fit in core.
                    168:  */
                    169: testcore(pp)
                    170: register PROC *pp;
                    171: {
                    172:        register SEG *sp;
                    173:        register saddr_t s;
                    174:        register saddr_t s1;
                    175:        register saddr_t s2;
                    176:        register int i;
                    177: 
                    178:        s = 0;
                    179:        for (i=0; i<NUSEG+1; i++) {
                    180:                if ((sp=pp->p_segp[i]) == NULL)
                    181:                        continue;
                    182:                if ((sp->s_flags&SFCORE) != 0)
                    183:                        continue;
                    184:                if (sp->s_size > s)
                    185:                        s = sp->s_size;
                    186:        }
                    187:        s1 = corebot;
                    188:        sp = &segmq;
                    189:        do {
                    190:                sp = sp->s_forw;
                    191:                s2 = sp->s_mbase;
                    192:                if (s2-s1 >= s)
                    193:                        return (1);
                    194:                s1 = sp->s_mbase + sp->s_size;
                    195:        } while (sp != &segmq);
                    196:        return (0);
                    197: }
                    198: 
                    199: /*
                    200:  * Swap all segments associated with a particular process into core.
                    201:  * The number of segments still swapped out is returned.
                    202:  */
                    203: proccore(pp)
                    204: register PROC *pp;
                    205: {
                    206:        register SEG *sp;
                    207:        register int i;
                    208:        register int n;
                    209:        register int f;
                    210: 
                    211:        n = 0;
                    212:        f = pp->p_flags&PFSWAP;
                    213:        for (i=0; i<NUSEG+1; i++) {
                    214:                if ((sp=pp->p_segp[i]) == NULL)
                    215:                        continue;
                    216:                if (f != 0)
                    217:                        sp->s_lrefc++;
                    218:                if ((sp->s_flags&SFCORE)==0 && segcore(sp)==0)
                    219:                        n++;
                    220:        }
                    221:        if (n == 0)
                    222:                pp->p_flags |= PFCORE;
                    223:        pp->p_flags &= ~PFSWAP;
                    224:        return (n);
                    225: }
                    226: 
                    227: /*
                    228:  * Swap out all segments associated with a given process.
                    229:  */
                    230: procdisk(pp)
                    231: register PROC *pp;
                    232: {
                    233:        register SEG *sp;
                    234:        register int i;
                    235:        register int f;
                    236:        int n;
                    237: 
                    238:        n = 0;
                    239:        f = pp->p_flags&PFSWAP;
                    240:        pp->p_flags &= ~PFCORE;
                    241:        for (i=0; i<NUSEG+1; i++) {
                    242:                if ((sp=pp->p_segp[i]) == NULL)
                    243:                        continue;
                    244:                if (f == 0)
                    245:                        --sp->s_lrefc;
                    246:                if ((sp->s_flags&SFCORE) == 0)
                    247:                        continue;
                    248:                if (sp->s_lrefc == 0)
                    249:                        if (segdisk(sp) != 0)
                    250:                                n++;
                    251:        }
                    252:        pp->p_flags |= PFSWAP;
                    253:        return (n);
                    254: }
                    255: 
                    256: /*
                    257:  * Swap out all auxiliary segments used by a process.
                    258:  */
                    259: auxmdisk(pp)
                    260: register PROC *pp;
                    261: {
                    262:        register SEG *sp;
                    263:        register int i;
                    264:        register int f;
                    265:        register int m;
                    266:        SEG *segl[NUSEG];
                    267: 
                    268: #ifndef NOMONITOR
                    269:        if (swmflag)
                    270:                printf("Auxiliary(%p, %d)\n", pp, pp->p_pid);
                    271: #endif
                    272:        sp = pp->p_segp[SIUSERP];
                    273:        if ((sp->s_flags&SFCORE) == 0) {
                    274:                panic("We may be in trouble");
                    275:                return;
                    276:        }
                    277:        m = pp->p_flags&PFCORE;
                    278:        f = pp->p_flags&PFAUXM;
                    279:        pp->p_flags &= ~(PFAUXM|PFCORE);
                    280:        skcopy(sp, offset(uproc, u_sege[0]), segl, sizeof(u.u_sege));
                    281:        for (i=0; i<NUSEG; i++) {
                    282:                if ((sp=segl[i]) == NULL)
                    283:                        continue;
                    284:                if (f != 0)
                    285:                        --sp->s_lrefc;
                    286:                if ((sp->s_flags&SFCORE) == 0)
                    287:                        continue;
                    288:                if (sp->s_lrefc == 0)
                    289:                        segdisk(sp);
                    290:        }
                    291:        pp->p_flags |= m;
                    292: }
                    293: 
                    294: /*
                    295:  * Swap the given segment into core.
                    296:  */
                    297: segcore(sp1)
                    298: register SEG *sp1;
                    299: {
                    300:        register SEG *sp2;
                    301: 
                    302:        lock(seglink);
                    303:        sp2 = xmalloc(sp1->s_size);
                    304:        unlock(seglink);
                    305:        if (sp2 == NULL)
                    306:                return (0);
                    307:        sp1->s_lrefc++;
                    308:        swapio(0, sp2->s_mbase, sp1->s_dbase, sp2->s_size);
                    309:        lock(seglink);
                    310:        sp1->s_back->s_forw = sp1->s_forw;
                    311:        sp1->s_forw->s_back = sp1->s_back;
                    312:        sp2->s_back->s_forw = sp1;
                    313:        sp1->s_back = sp2->s_back;
                    314:        sp2->s_forw->s_back = sp1;
                    315:        sp1->s_forw = sp2->s_forw;
                    316:        sp1->s_flags |= SFCORE;
                    317:        sp1->s_mbase = sp2->s_mbase;
                    318:        --sp1->s_lrefc;
                    319:        unlock(seglink);
                    320:        return (1);
                    321: }
                    322: 
                    323: /*
                    324:  * Swap the given segment out onto disk.
                    325:  */
                    326: segdisk(sp1)
                    327: register SEG *sp1;
                    328: {
                    329:        register SEG *sp2;
                    330: 
                    331:        lock(seglink);
                    332:        sp2 = xdalloc(sp1->s_size);
                    333:        unlock(seglink);
                    334:        if (sp2 == NULL)
                    335:                return (0);
                    336:        sp1->s_lrefc++;
                    337:        swapio(1, sp1->s_mbase, sp2->s_dbase, sp1->s_size);
                    338:        lock(seglink);
                    339:        sp1->s_back->s_forw = sp1->s_forw;
                    340:        sp1->s_forw->s_back = sp1->s_back;
                    341:        sp2->s_back->s_forw = sp1;
                    342:        sp1->s_back = sp2->s_back;
                    343:        sp2->s_forw->s_back = sp1;
                    344:        sp1->s_forw = sp2->s_forw;
                    345:        sp1->s_flags &= ~SFCORE;
                    346:        sp1->s_dbase = sp2->s_dbase;
                    347:        --sp1->s_lrefc;
                    348:        unlock(seglink);
                    349:        return (1);
                    350: }
                    351: 
                    352: /*
                    353:  * Allocate a segment on disk that is `n' clicks long.
                    354:  * The `seglink' gate should be locked before this routine is called.
                    355:  * This routine is the same as `sdalloc' except that we can't run out of
                    356:  * alloc space to allocate the segment and we allocate in high regions.
                    357:  */
                    358: SEG *
                    359: xdalloc(s)
                    360: saddr_t s;
                    361: {
                    362:        register SEG *sp1;
                    363:        register SEG *sp2;
                    364:        register daddr_t d;
                    365:        register daddr_t d1;
                    366:        register daddr_t d2;
                    367: 
                    368:        d = stod(s);
                    369:        d2 = swaptop;
                    370:        sp1 = &segdq;
                    371:        do {
                    372:                if ((sp1=sp1->s_back) != &segdq)
                    373:                        d1 = sp1->s_dbase + stod(sp1->s_size);
                    374:                else
                    375:                        d1 = swapbot;
                    376:                if (d2-d1 >= d) {
                    377:                        sp2 = &segswap;
                    378:                        kclear((char *)sp2, sizeof(SEG));
                    379:                        sp1->s_forw->s_back = sp2;
                    380:                        sp2->s_forw = sp1->s_forw;
                    381:                        sp1->s_forw = sp2;
                    382:                        sp2->s_back = sp1;
                    383:                        sp2->s_urefc = 1;
                    384:                        sp2->s_lrefc = 1;
                    385:                        sp2->s_size = s;
                    386:                        sp2->s_dbase = d2 - d;
                    387:                        return (sp2);
                    388:                }
                    389:                d2 = sp1->s_dbase;
                    390:        } while (sp1 != &segdq);
                    391:        return (NULL);
                    392: }
                    393: 
                    394: /*
                    395:  * Allocate a segment in memory that is `n' clicks long.
                    396:  * The `seglink' gate should be locked before this routine is called.
                    397:  * This routine is the same as `smalloc' except that we can't run out of
                    398:  * alloc space to allocate the segment.
                    399:  */
                    400: SEG *
                    401: xmalloc(s)
                    402: register saddr_t s;
                    403: {
                    404:        register SEG *sp1;
                    405:        register SEG *sp2;
                    406:        register saddr_t s1;
                    407:        register saddr_t s2;
                    408: 
                    409:        s1 = corebot;
                    410:        sp1 = &segmq;
                    411:        do {
                    412:                if ((sp1=sp1->s_forw) != &segmq)
                    413:                        s2 = sp1->s_mbase;
                    414:                else
                    415:                        s2 = coretop;
                    416:                if (s2-s1 >= s) {
                    417:                        sp2 = &segswap;
                    418:                        kclear((char *)sp2, sizeof(SEG));
                    419:                        sp1->s_back->s_forw = sp2;
                    420:                        sp2->s_back = sp1->s_back;
                    421:                        sp1->s_back = sp2;
                    422:                        sp2->s_forw = sp1;
                    423:                        sp2->s_urefc = 1;
                    424:                        sp2->s_lrefc = 1;
                    425:                        sp2->s_size = s;
                    426:                        sp2->s_mbase = s1;
                    427:                        return (sp2);
                    428:                }
                    429:                s1 = sp1->s_mbase + sp1->s_size;
                    430:        } while (sp1 != &segmq);
                    431:        return (NULL);
                    432: }

unix.superglobalmegacorp.com

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