Annotation of coherent/b/kernel/coh.386/proc.c, revision 1.1

1.1     ! root        1: /* (lgl-
        !             2:  *     The information contained herein is a trade secret of Mark Williams
        !             3:  *     Company, and  is confidential information.  It is provided  under a
        !             4:  *     license agreement,  and may be  copied or disclosed  only under the
        !             5:  *     terms of  that agreement.  Any  reproduction or disclosure  of this
        !             6:  *     material without the express written authorization of Mark Williams
        !             7:  *     Company or persuant to the license agreement is unlawful.
        !             8:  *
        !             9:  *     COHERENT Version 3.x, 4.x
        !            10:  *     Copyright (c) 1982, 1993.
        !            11:  *     An unpublished work by Mark Williams Company, Chicago.
        !            12:  *     All rights reserved.
        !            13:  -lgl) */
        !            14: /*
        !            15:  * Process handling and scheduling.
        !            16:  */
        !            17: #include <sys/coherent.h>
        !            18: #include <sys/acct.h>
        !            19: #include <errno.h>
        !            20: #include <sys/inode.h>
        !            21: #include <sys/ptrace.h>
        !            22: #include <sys/sched.h>
        !            23: #include <signal.h>
        !            24: #include <sys/stat.h>
        !            25: 
        !            26: /*
        !            27:  * Initialization.
        !            28:  * Set up the hash table queues.
        !            29:  */
        !            30: pcsinit()
        !            31: {
        !            32:        register PROC *pp;
        !            33:        register PLINK *lp;
        !            34: 
        !            35:        /*
        !            36:         * Explicitly initialize everything in the first process.
        !            37:         */
        !            38:        pp = &procq;
        !            39:        SELF = pp;
        !            40:        procq.p_nforw = pp;
        !            41:        procq.p_nback = pp;
        !            42:        procq.p_lforw = pp;
        !            43:        procq.p_lback = pp;
        !            44: 
        !            45:        /* Segments are initialized in mchinit() and eveinit(). */
        !            46:        /* procq is static, so p_shmsr[] initializes to nulls.  */
        !            47: 
        !            48:        procq.p_uid = 0;                /* Effective uid */
        !            49:        procq.p_ruid = 0;               /* Real uid */
        !            50:        procq.p_rgid = 0;               /* Real gid */
        !            51:        procq.p_state = 0;              /* Scheduling state */
        !            52:        procq.p_flags = 0;              /* Flags */
        !            53:        procq.p_ssig = (sig_t) 0;       /* Signals which have been set */
        !            54: 
        !            55:        procq.p_dfsig = (sig_t) 0xffffffff;  /* All signals are defaulted.  */
        !            56:        procq.p_hsig = (sig_t) 0;       /* Signals which are being held */
        !            57:        procq.p_dsig = (sig_t) 0;       /* Signals which are being deferred */
        !            58: 
        !            59:        procq.p_isig = (sig_t) 0;       /* Signals which are being ignored */
        !            60:        procq.p_event = NULL;           /* Wakeup event channel */
        !            61:        procq.p_alarm = 0;              /* Timer for alarms */
        !            62:        procq.p_group = 0;              /* Process group */
        !            63: /*
        !            64:  * Set ttdev to null so that we do not accidentally set a tty for init.
        !            65:  */
        !            66:        procq.p_ttdev = makedev(0,0);   /* Controlling terminal */
        !            67:        procq.p_nice = 0;               /* Nice value */
        !            68:        procq.p_lctim = 0;              /* Last time cval was updated */
        !            69:        procq.p_utime = 0L;             /* User time (HZ) */
        !            70:        procq.p_stime = 0L;             /* System time */
        !            71:        procq.p_cutime = 0L;            /* Sum of childs user time */
        !            72:        procq.p_cstime = 0L;            /* Sum of childs system time */
        !            73:        procq.p_exit = 0;               /* Exit status */
        !            74:        procq.p_polls = NULL;           /* Enabled polls */
        !            75:        /* Poll  timer */
        !            76:        procq.p_polltim.t_next = NULL;
        !            77:        procq.p_polltim.t_last = NULL;
        !            78:        procq.p_polltim.t_lbolt = 0L;
        !            79:        procq.p_polltim.t_func = NULL;
        !            80:        procq.p_polltim.t_farg = NULL;
        !            81: 
        !            82:        /* Alarm timer */
        !            83:        procq.p_alrmtim.t_next = NULL;
        !            84:        procq.p_alrmtim.t_last = NULL;
        !            85:        procq.p_alrmtim.t_lbolt = 0L;
        !            86:        procq.p_alrmtim.t_func = NULL;
        !            87:        procq.p_alrmtim.t_farg = NULL;
        !            88: 
        !            89:        procq.p_prl = NULL;             /* Pending record lock */
        !            90:        procq.p_semu = NULL;            /* Semaphore undo */
        !            91:        for (lp=&linkq[0]; lp<&linkq[NHPLINK]; lp++) {
        !            92:                lp->p_lforw = lp;
        !            93:                lp->p_lback = lp;
        !            94:        }
        !            95: }
        !            96: 
        !            97: /*
        !            98:  * Initiate a process.
        !            99:  */
        !           100: PROC *
        !           101: process()
        !           102: {
        !           103:        register PROC *pp1;
        !           104:        register PROC *pp;
        !           105: 
        !           106:        if ((pp=kalloc(sizeof(PROC))) == NULL)
        !           107:                return (NULL);
        !           108: 
        !           109:        pp->p_flags = PFCORE;
        !           110:        pp->p_state = PSRUN;
        !           111:        pp->p_ttdev = NODEV;
        !           112:        pp->p_semu = NULL;              /* Semaphore undo */
        !           113: 
        !           114:        lock(pnxgate);
        !           115: next:
        !           116: 
        !           117:        /*
        !           118:         * Pick the next process id.
        !           119:         */
        !           120:        if (++cpid >= NPID)
        !           121:                cpid = 2;
        !           122:        pp->p_pid = cpid;
        !           123: 
        !           124:        /*
        !           125:         * Make sure that process id is not in use.
        !           126:         */
        !           127:        pp1 = &procq;
        !           128:        while ((pp1=pp1->p_nforw) != &procq) {
        !           129:                if (pp1->p_pid < pp->p_pid)
        !           130:                        break;
        !           131:                if (pp1->p_pid == pp->p_pid)
        !           132:                        goto next;
        !           133:        }
        !           134: 
        !           135:        /*
        !           136:         * We've got a valid pid, so let's put this process into
        !           137:         * the process table.
        !           138:         */
        !           139:        pp->p_nback = pp1->p_nback;
        !           140:        pp1->p_nback->p_nforw = pp;
        !           141:        pp->p_nforw = pp1;
        !           142:        pp1->p_nback = pp;
        !           143:        unlock(pnxgate);
        !           144:        return (pp);
        !           145: }
        !           146: 
        !           147: /*
        !           148:  * Remove a process from the next queue and release and space.
        !           149:  */
        !           150: relproc(pp)
        !           151: register PROC *pp;
        !           152: {
        !           153:        register SEG * sp;
        !           154: 
        !           155:        /*
        !           156:         * Child process still has a user-area.
        !           157:         */
        !           158:        if (sp = pp->p_segp[SIUSERP]) {
        !           159: 
        !           160:                /*
        !           161:                 * Detach user-area from child process.
        !           162:                 */
        !           163:                pp->p_segp[SIUSERP] = NULL;
        !           164: 
        !           165:                /*
        !           166:                 * Child process is swapped out.
        !           167:                 */
        !           168:                if (pp->p_flags & PFSWAP)
        !           169:                        sp->s_lrefc++;
        !           170: 
        !           171:                /*
        !           172:                 * Release child's user-area.
        !           173:                 */
        !           174:                sfree(sp);
        !           175:        }
        !           176: 
        !           177:        /*
        !           178:         * Remove process from doubly-linked list of all processes.
        !           179:         * Release space allocated for proc structure.
        !           180:         */
        !           181:        lock(pnxgate);
        !           182:        pp->p_nback->p_nforw = pp->p_nforw;
        !           183:        pp->p_nforw->p_nback = pp->p_nback;
        !           184:        unlock(pnxgate);
        !           185:        kfree(pp);
        !           186: }
        !           187: 
        !           188: /*
        !           189:  * Create a clone of ourselves.
        !           190:  *     N.B. - consave(&mcon) returns twice; anything not initialized
        !           191:  *     in automatic storage before the call to segadup() will not be
        !           192:  *     initialized when the second return from consave() commences.
        !           193:  */
        !           194: pfork()
        !           195: {
        !           196:        register PROC *cpp;
        !           197:        register PROC *pp;
        !           198:        register int s;
        !           199:        MCON mcon;
        !           200: 
        !           201:        if ((cpp=process()) == NULL) {
        !           202:                SET_U_ERROR( EAGAIN, "no more process table entries" );
        !           203:                return -1;
        !           204:        }
        !           205: 
        !           206:        pp = SELF;
        !           207:        s = sphi();     /* put current interrupt level into s before segadup */
        !           208:        spl(s);
        !           209:        /*
        !           210:         * As stated above, no auto variable may be changed between calls
        !           211:         * to segadup() and consave().
        !           212:         */
        !           213:        if (segadup(cpp) == 0) {
        !           214:                SET_U_ERROR( EAGAIN, "can not duplicate segments" );
        !           215:                relproc(cpp);
        !           216:                return -1;
        !           217:        }
        !           218:        shmDup(cpp);    /* copy shared memory info & update ref counts */
        !           219:        if (u.u_rdir)
        !           220:                u.u_rdir->i_refc++;
        !           221:        if (u.u_cdir)
        !           222:                u.u_cdir->i_refc++;
        !           223:        fdadupl();
        !           224:        cpp->p_uid   = pp->p_uid;
        !           225:        cpp->p_ruid  = pp->p_ruid;
        !           226:        cpp->p_rgid  = pp->p_rgid;
        !           227:        cpp->p_ppid  = pp->p_pid;
        !           228:        cpp->p_ttdev = pp->p_ttdev;
        !           229:        cpp->p_group = pp->p_group;
        !           230:        cpp->p_ssig  = pp->p_ssig;
        !           231: 
        !           232:        cpp->p_dfsig  = pp->p_dfsig;
        !           233:        cpp->p_dsig  = pp->p_dsig;
        !           234:        cpp->p_hsig  = pp->p_hsig;
        !           235:        cpp->p_prl = NULL;
        !           236: 
        !           237:        cpp->p_isig  = pp->p_isig;
        !           238: 
        !           239:        sphi();         /* s = sphi() was done before segadup() */
        !           240:        consave(&mcon);
        !           241:        spl(s);
        !           242: 
        !           243:        /*
        !           244:         * Parent process.
        !           245:         */
        !           246:        if ((pp = SELF) != cpp) {
        !           247:                segfinm(cpp->p_segp[SIUSERP]);
        !           248:                dmaout(sizeof(mcon), 
        !           249:                  MAPIO(cpp->p_segp[SIUSERP]->s_vmem,
        !           250:                  U_OFFSET + offset(uproc,u_syscon)),
        !           251:                  (char *)&mcon);
        !           252:                s = sphi();
        !           253:                setrun(cpp);
        !           254:                spl(s);
        !           255: 
        !           256:                u.u_rval2 = 0;
        !           257:                return(cpp->p_pid);
        !           258:        }
        !           259: 
        !           260:        /*
        !           261:         * Child process.
        !           262:         */
        !           263:        else {
        !           264:                u.u_btime = timer.t_time;
        !           265:                u.u_flag = AFORK;
        !           266: 
        !           267: #ifdef UPROC_VERSION
        !           268:                u.u_version = UPROC_VERSION;
        !           269: #endif /* UPROC_VERSION */
        !           270:                u.u_sleep[0] = '\0'; /* We are not sleeping to start with.  */
        !           271:                sproto(0);
        !           272:                segload();
        !           273:                u.u_rval2 = SELF->p_ppid;
        !           274:                return 0;
        !           275:        }
        !           276: }
        !           277: 
        !           278: /*
        !           279:  * Die.
        !           280:  */
        !           281: pexit(s)
        !           282: {
        !           283:        register PROC *pp1;
        !           284:        register PROC *pp;
        !           285:        register SEG  *sp;
        !           286:        register int n;
        !           287:        PROC *parent;
        !           288: 
        !           289:        pp = SELF;
        !           290:        T_PIGGY( 0x1, printf("%s:pexit(%x)", u.u_comm, s); );
        !           291: 
        !           292:        ndpEndProc();
        !           293: 
        !           294:        /*
        !           295:         * Cancel alarm and poll timers [if any].
        !           296:         */
        !           297:        timeout(&pp->p_alrmtim, 0, NULL, 0);
        !           298:        timeout(&pp->p_polltim, 0, NULL, 0);
        !           299: 
        !           300:        /*
        !           301:         * Write out accounting directory and close all files associated with
        !           302:         * this process.
        !           303:         */
        !           304:        setacct();
        !           305:        if (u.u_rdir)
        !           306:                ldetach(u.u_rdir);
        !           307:        if (u.u_cdir)
        !           308:                ldetach(u.u_cdir);
        !           309:        fdaclose();
        !           310: 
        !           311:        /*
        !           312:         * Free all segments in reverse order, except for user-area.
        !           313:         */
        !           314:        for (n = NUSEG; --n > 0;) {
        !           315:                if (sp = pp->p_segp[n]) {
        !           316:                        pp->p_segp[n] = NULL;
        !           317:                        sfree(sp);
        !           318:                }
        !           319:        }
        !           320: 
        !           321:        /* Detach remaining shared memory segments. */
        !           322:        shmAllDt();
        !           323: 
        !           324:        /* Adjust(undo) all semaphores. */
        !           325:        semAllAdjust(pp);
        !           326: 
        !           327:        /*
        !           328:         * Wakeup our parent.  If we have any children, init will become the
        !           329:         * new parent.  If there are any children we are tracing who are
        !           330:         * waiting for us, we wake them up.
        !           331:         */
        !           332:        pp1 = &procq;
        !           333: 
        !           334:        /* pp1 runs through the list of all processes */
        !           335:        while ((pp1=pp1->p_nforw) != &procq) {
        !           336: 
        !           337:                /* if pp1 points to parent of the current process...*/
        !           338:                if (pp1->p_pid == pp->p_ppid) {
        !           339:                        parent = pp1;   /* Remember our parent.  */
        !           340:                        if (ASLEEP(pp1) && pp1->p_event==(char *)pp1)
        !           341:                                wakeup((char *)pp1);
        !           342:                }
        !           343: 
        !           344:                /* if pp1 points to child of the current process...*/
        !           345:                if (pp1->p_ppid == pp->p_pid) {
        !           346:                        pp1->p_ppid = 1;
        !           347:                        if (pp1->p_state == PSDEAD)
        !           348:                                wakeup((char *)eprocp);
        !           349:                        if (pp1->p_flags&PFTRAC)
        !           350:                                wakeup((char *)&pts.pt_req);
        !           351:                }
        !           352:        }
        !           353: 
        !           354:        /*
        !           355:         * Mark us as dead and give up the processor forever.
        !           356:         */
        !           357:        pp->p_exit = s;
        !           358:        pp->p_state = PSDEAD;
        !           359: 
        !           360:        /*
        !           361:         * If this is a process group leader, inform all members of the group
        !           362:         * of the recent death with a HUP signal.
        !           363:         */
        !           364:        if (pp->p_group == pp->p_pid) {
        !           365:                ukill(-pp->p_pid, SIGHUP);
        !           366:        }
        !           367: 
        !           368:        /*
        !           369:         * If the parent is ignoring SIGCLD, 
        !           370:         * remove the zombie right away.
        !           371:         */
        !           372:        if (SIG_BIT(SIGCLD) & parent->p_isig) {
        !           373:                parent->p_cutime += pp->p_utime + pp->p_cutime;
        !           374:                parent->p_cstime += pp->p_stime + pp->p_cstime;
        !           375:                relproc(pp);
        !           376:        } else {
        !           377:                /*
        !           378:                 * If SIGCLD is not defaulted, notify our parent
        !           379:                 * of our demise.
        !           380:                 */
        !           381:                if (!(SIG_BIT(SIGCLD) & parent->p_dfsig)) {
        !           382:                        sendsig(SIGCLD, parent );
        !           383:                }
        !           384:        }
        !           385: 
        !           386:        dispatch();
        !           387: }
        !           388: 
        !           389: /*
        !           390:  * x_sleep()
        !           391:  *
        !           392:  * Surrender CPU while awaiting some event or resource.
        !           393:  *
        !           394:  * Arguments:
        !           395:  *     event:          key value; so wakeup() can find this sleep
        !           396:  *     schedPri:       prilo/primed/prihi/pritape/pritty/pridisk/prinet
        !           397:  *                     just copied into proc struct for scheduler to use.
        !           398:  *                     (see sys/v_types.h)
        !           399:  *     sleepPri:       slpriNoSig      - signals may not interrupt sleep
        !           400:  *                     slpriSigLjmp    - signals cause longjmp (EINTR)
        !           401:  *                     slpriSigCatch   - signals are caught
        !           402:  *                     (see sys/sched.h)
        !           403:  *     reason:         up to 10 chars of text for ps command "event"
        !           404:  *
        !           405:  * Return values:
        !           406:  *     0               wakeup received
        !           407:  *     1               signal (other than SIGSTOP/SIGCONT) received
        !           408:  *     2               SIGSTOP/SIGCONT (unimplemented now)
        !           409:  *
        !           410:  * If longjmp occurs, won't return from x_sleep!
        !           411:  */
        !           412: int
        !           413: x_sleep(event, schedPri, sleepPri, reason)
        !           414: char * event;
        !           415: int schedPri;
        !           416: int sleepPri;
        !           417: char * reason;
        !           418: {
        !           419:        int i;
        !           420:        register PROC *bp;
        !           421:        register PROC *fp;
        !           422:        register PROC *pp;
        !           423:        register int s;
        !           424: 
        !           425:        /*
        !           426:         * The descriptive string may be at most 10 characters long.
        !           427:         * It will only be NUL terminated if it has 9 or fewer characters.
        !           428:         */
        !           429:        for (i = 0; i < U_SLEEP_LEN; ++i) {
        !           430:                if ('\0' == (u.u_sleep[i] = reason[i])) {
        !           431:                        break;
        !           432:                }
        !           433:        }
        !           434: 
        !           435:        pp = SELF;
        !           436: 
        !           437:        /*
        !           438:         * Get ready to go to sleep and do so.
        !           439:         */
        !           440:        s = sphi();
        !           441:        pp->p_state = (sleepPri == slpriNoSig) ? PSSLEEP : PSSLSIG;
        !           442:        pp->p_schedPri = schedPri;
        !           443:        pp->p_event = event;
        !           444:        pp->p_lctim = utimer;
        !           445:        fp = &linkq[hash(event)];
        !           446:        bp = fp->p_lback;
        !           447:        pp->p_lforw = fp;
        !           448:        fp->p_lback = pp;
        !           449:        pp->p_lback = bp;
        !           450:        bp->p_lforw = pp;
        !           451:        spl(s);
        !           452: 
        !           453:        /* Here is sleep if signals may *not* interrupt. */
        !           454:        if (sleepPri == slpriNoSig) {
        !           455:                dispatch();
        !           456:                u.u_sleep[0] = '\0';
        !           457:                return 0;
        !           458:        }
        !           459: 
        !           460:        /* Here is sleep if signals *may* interrupt. */
        !           461:        /* Don't sleep at all if there is already a signal pending. */
        !           462:        if (!nondsig()) {
        !           463:                dispatch();
        !           464:                u.u_sleep[0] = '\0';
        !           465:                if (!nondsig()) {
        !           466:                        return 0;
        !           467:                }
        !           468:        }
        !           469: 
        !           470:        /* The process has been interrupted from sleep by a signal. */
        !           471: 
        !           472:        if (sleepPri == slpriSigCatch) {
        !           473:                return 1;
        !           474:        }
        !           475: 
        !           476:        /* Do longjmp to beginning of system call. */
        !           477:        T_HAL(8, printf("[%d]Ljmps ", SELF->p_pid));
        !           478:        sphi();
        !           479:        envrest(&u.u_sigenv);
        !           480: }
        !           481: 
        !           482: /*
        !           483:  * Defer function to wake up all processes sleeping on the event `e'.
        !           484:  */
        !           485: wakeup(e)
        !           486: char *e;
        !           487: {
        !           488:        void dwakeup();
        !           489: 
        !           490: #ifdef TRACER
        !           491:        /*
        !           492:         * In diagnostic kernel, keep return addr on queue as well.
        !           493:         */
        !           494:        int *r=(int*)(&e);
        !           495:        defer0(dwakeup, e, *(r-1));
        !           496: #else
        !           497:        defer(dwakeup, e);
        !           498: #endif
        !           499: }
        !           500: 
        !           501: /*
        !           502:  * Wake up all processes sleeping on "event".
        !           503:  */
        !           504: void
        !           505: dwakeup(event)
        !           506: char *event;
        !           507: {
        !           508:        register PROC *pp;
        !           509:        register PROC *pp1;
        !           510:        register int s;
        !           511: 
        !           512:        /*
        !           513:         * Identify event queue to check.
        !           514:         * Disable interrupts.
        !           515:         */
        !           516:        pp1 = &linkq[hash(event)];
        !           517:        pp = pp1;
        !           518:        s = sphi();
        !           519: 
        !           520:        /*
        !           521:         * Traverse doubly-linked circular event-queue.
        !           522:         */
        !           523:        while ((pp = pp->p_lforw) != pp1) {
        !           524: 
        !           525:                /*
        !           526:                 * Process is waiting on event 'event'.
        !           527:                 */
        !           528:                if (pp->p_event == event) {
        !           529:                        /*
        !           530:                         * Remove process from event queue.
        !           531:                         * Update process priority.
        !           532:                         * Insert process into run queue.
        !           533:                         */
        !           534:                        pp->p_lback->p_lforw = pp->p_lforw;
        !           535:                        pp->p_lforw->p_lback = pp->p_lback;
        !           536:                        setrun(pp);
        !           537: 
        !           538:                        /*
        !           539:                         * Enable interrupts.
        !           540:                         * Restart search at start of event queue.
        !           541:                         * Disable interrupts.
        !           542:                         */
        !           543:                        spl(s);
        !           544:                        pp = pp1;
        !           545:                        s = sphi();
        !           546:                }
        !           547:        }
        !           548:        spl(s);
        !           549: }
        !           550: 
        !           551: /*
        !           552:  * Select next process for execution, working backward from iprocp.
        !           553:  * If it is not the idle process, delete it from the run queue.
        !           554:  * If it is not the current process, consave the current process and
        !           555:  * conrest the selected process.
        !           556:  */
        !           557: dispatch()
        !           558: {
        !           559:        register PROC *pp;
        !           560:        register int s;
        !           561: 
        !           562:        s = sphi();
        !           563:        pp = iprocp->p_lback;
        !           564:        if (pp != iprocp) {
        !           565:                pp->p_lforw->p_lback = pp->p_lback;
        !           566:                pp->p_lback->p_lforw = pp->p_lforw;
        !           567:        }
        !           568:        spl(s);
        !           569: 
        !           570:        quantum = NCRTICK;
        !           571:        disflag = 0;
        !           572:        if (pp != SELF) {
        !           573:                /*
        !           574:                 * Consave() returns twice.
        !           575:                 * 1st time is after our context is saved in u.u_syscon,
        !           576:                 *      whereupon we should restore other proc's context.
        !           577:                 * 2nd time is after our context is restored by another proc.
        !           578:                 * Conrest() forces a context switch to a new process.
        !           579:                 */
        !           580:                s = sphi();
        !           581:                SELF = pp;
        !           582:                if (consave(&u.u_syscon) == 0) {
        !           583:                        conrest(*pp->p_segp[SIUSERP]->s_vmem, &u.u_syscon);
        !           584:                }
        !           585: 
        !           586:                if (SELF->p_pid) {      /* init is special! */
        !           587:                        ndpConRest();
        !           588:                        segload();
        !           589:                }
        !           590:                spl(s);
        !           591:        }
        !           592: }
        !           593: 
        !           594: /*
        !           595:  * Add a process to the run queue, just forward of iprocp.
        !           596:  * This routine must be called at high priority.
        !           597:  */
        !           598: setrun(pp)
        !           599: register PROC *pp;
        !           600: {
        !           601:        pp->p_lback = iprocp;
        !           602:        pp->p_lforw = iprocp->p_lforw;
        !           603:        pp->p_lback->p_lforw = pp;
        !           604:        pp->p_lforw->p_lback = pp;
        !           605:        pp->p_state = PSRUN;
        !           606: }
        !           607: 
        !           608: /*
        !           609:  * Wait for the gate `g' to unlock, and then lock it.
        !           610:  */
        !           611: lock(g)
        !           612: register GATE g;
        !           613: {
        !           614:        register int s;
        !           615: 
        !           616:        s = sphi();
        !           617:        while (g[0]) {
        !           618:                g[1] = 1;
        !           619:                x_sleep((char *)g, primed, slpriNoSig, "lock");
        !           620:                /* Waiting for a gate to unlock.  */
        !           621:        }
        !           622:        g[0] = 1;
        !           623:        spl(s);
        !           624: }
        !           625: 
        !           626: /*
        !           627:  * Unlock the gate `g'.
        !           628:  */
        !           629: unlock(g)
        !           630: register GATE g;
        !           631: {
        !           632:        g[0] = 0;
        !           633:        if (g[1]) {
        !           634:                g[1] = 0;
        !           635:                disflag = 1;
        !           636:                wakeup((char *)g);
        !           637:        }
        !           638: }

unix.superglobalmegacorp.com

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