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

1.1     ! root        1: /*
        !             2:  * coh.386/exec.c
        !             3:  *
        !             4:  * The exec() system call.
        !             5:  *
        !             6:  * Revised: Fri Jun  4 10:41:40 1993 CDT
        !             7:  */
        !             8: #include <sys/coherent.h>
        !             9: #include <sys/acct.h>
        !            10: #include <sys/buf.h>
        !            11: #include <canon.h>
        !            12: #include <sys/con.h>
        !            13: #include <errno.h>
        !            14: #include <fcntl.h>
        !            15: #include <sys/filsys.h>
        !            16: #include <sys/ino.h>
        !            17: #include <sys/inode.h>
        !            18: #include <a.out.h>
        !            19: #include <l.out.h>
        !            20: #include <sys/proc.h>
        !            21: #include <sys/sched.h>
        !            22: #include <sys/seg.h>
        !            23: #include <signal.h>
        !            24: #include <sys/reg.h>
        !            25: #include <sys/stat.h>
        !            26: #include <sys/fd.h>
        !            27: 
        !            28: /*
        !            29:  * Round section and segment start address to nearest lower click boundary.
        !            30:  */
        !            31: static void
        !            32: xecrnd(xsp)
        !            33: struct xecseg * xsp;
        !            34: {
        !            35:        int diff;
        !            36: 
        !            37:        diff = xsp->fbase & (NBPC-1);
        !            38:        xsp->mbase -= diff;
        !            39:        xsp->fbase -= diff;
        !            40:        xsp->size += diff;
        !            41: }
        !            42: 
        !            43: /*
        !            44:  * Pass control to an image in a file.
        !            45:  * Make sure the format is acceptable. Release
        !            46:  * the old segments. Read in the new ones. Some special
        !            47:  * care is taken so that shared and (more important) shared
        !            48:  * and separated images can be run on the 8086.
        !            49:  */
        !            50: pexece(np, argp, envp)
        !            51: char   *np;
        !            52: char   *argp[];
        !            53: char   *envp[];
        !            54: {
        !            55:        struct xechdr   head;
        !            56:        register INODE  *ip;                    /* Load file INODE */
        !            57:        register PROC   *pp;                    /* A cheap copy of SELF */
        !            58:        register SEG    *segp;
        !            59:        SEG     *ssegp;
        !            60:        register int    i;                      /* For looping over segments*/
        !            61:        int roundup;
        !            62:        int shrdsize;
        !            63:        struct xecnode * xlist = NULL;          /* list head */
        !            64:        struct xecnode * xp;
        !            65:        struct xecseg tempseg;
        !            66:        unsigned int textSize;
        !            67: 
        !            68:        pp = SELF;
        !            69:        kclear(&head, sizeof(head)); 
        !            70:        if ((ip=exlopen(&head, np, &shrdsize, &xlist)) == NULL) {
        !            71:                goto done;
        !            72:        }
        !            73: 
        !            74:        roundup = (shrdsize) & 0xf;
        !            75:        ssegp = exstack(&head,argp, envp, wdsize());
        !            76: 
        !            77:        if (!ssegp) {
        !            78:                idetach(ip);
        !            79:                goto done;
        !            80:        }
        !            81: 
        !            82:        /* Release shared memory. */
        !            83:        shmAllDt();
        !            84: 
        !            85:        /*
        !            86:         * At this point the file has been
        !            87:         * validated as an object module, and the
        !            88:         * argument list has been built. Release all of
        !            89:         * the original segments. At this point we have
        !            90:         * committed to the new image. A "sys exec" that
        !            91:         * gets an I/O error is doomed.
        !            92:         * NOTE: User-area segment is NOT released.
        !            93:         *       Segment pointer in proc is erased BEFORE invoking sfree().
        !            94:         */
        !            95:        for (i = 1; i < NUSEG; ++i) {
        !            96:                if ((segp = pp->p_segp[i])) {
        !            97:                        pp->p_segp[i] = NULL;
        !            98:                        sfree(segp);
        !            99:                }
        !           100:        }
        !           101:        pp->p_segp[SISTACK] = ssegp;
        !           102: 
        !           103:        /*
        !           104:         * Read in the loadable segments.
        !           105:         */
        !           106:        switch (head.magic) {
        !           107:        case XMAGIC(I286MAGIC,I_MAGIC):
        !           108:                u.u_regl[CS] = SEG_286_UII | R_USR;
        !           109:                u.u_regl[DS] = SEG_286_UD  | R_USR;
        !           110: 
        !           111:                if ((segp = ssalloc(ip,SFTEXT, head.segs[SISTEXT].size))
        !           112:                  == NULL)
        !           113:                        goto out; 
        !           114:                pp->p_segp[SISTEXT] = segp;
        !           115: 
        !           116:                if (!exsread(segp, ip, &head.segs[SISTEXT], 0))
        !           117:                        goto out;
        !           118: 
        !           119:                if ((segp = ssalloc(ip, 0, roundup + head.segs[SIPDATA].size 
        !           120:                  + head.segs[SIBSS].size)) == NULL)
        !           121:                        goto out;
        !           122:                pp->p_segp[SIPDATA] = segp;
        !           123: 
        !           124:                if (!exsread(segp, ip, &head.segs[SIPDATA], shrdsize)) {
        !           125:                        goto out;
        !           126:                }
        !           127:                head.segs[SIPDATA].size += roundup;
        !           128:                break;
        !           129:        case XMAGIC(I386MAGIC,Z_MAGIC):
        !           130:                u.u_regl[CS] = SEG_386_UI | R_USR;
        !           131:                u.u_regl[DS] = SEG_386_UD | R_USR;
        !           132: 
        !           133:                /*
        !           134:                 * Round segment address down to nearest click boundary.
        !           135:                 * Ciaran did this.  I'm not sure why, but will preserve
        !           136:                 * it for now. -hws-
        !           137:                 */
        !           138:                tempseg = head.segs[SISTEXT];   /* save pre-rounding value */
        !           139:                xecrnd(head.segs + SISTEXT);
        !           140:                xecrnd(head.segs + SIPDATA);
        !           141: 
        !           142:                /*
        !           143:                 * Compute text segment size by taking highest address
        !           144:                 * seen in any text section.
        !           145:                 */
        !           146:                textSize = head.segs[SISTEXT].size + head.segs[SISTEXT].mbase;
        !           147:                for (xp = xlist; xp; xp = xp->xn) {
        !           148:                        unsigned int tmpSize;
        !           149:                        if (xp->segtype != SISTEXT)
        !           150:                                continue;
        !           151:                        tmpSize = xp->xseg.size + xp->xseg.mbase;
        !           152:                        if (tmpSize > textSize)
        !           153:                                textSize = tmpSize;
        !           154:                }
        !           155: 
        !           156:                /* Entry point must be within text segment. */
        !           157:                if (head.entry >= textSize) {
        !           158:                                goto out;
        !           159:                }
        !           160: 
        !           161:                if ((segp = ssalloc(ip, SFTEXT|SFSHRX, textSize))
        !           162:                  == NULL)
        !           163:                        goto out;
        !           164:                pp->p_segp[SISTEXT] = segp;
        !           165: 
        !           166:                if (segp->s_ip==0) {
        !           167:                        if (!exsread(segp, ip, &tempseg, 0)) {
        !           168:                                goto out;
        !           169:                        }
        !           170:                        /* load additional text sections, if any */
        !           171:                        for (xp = xlist; xp; xp = xp->xn) {
        !           172:                                if (xp->segtype != SISTEXT)
        !           173:                                        continue;
        !           174:                                if (!exsread(segp, ip, &xlist->xseg, 0)) {
        !           175:                                        goto out;
        !           176:                                }
        !           177:                        }
        !           178: 
        !           179:                        segp->s_ip = ip;
        !           180:                        ip->i_refc++;
        !           181:                }
        !           182:                if ((segp = ssalloc(ip, 0, 
        !           183:                  head.segs[SIPDATA].size+head.segs[SIBSS].size)) == NULL)
        !           184:                        goto out;
        !           185:                pp->p_segp[SIPDATA] = segp;
        !           186:                if (segp->s_ip==0 &&
        !           187:                    !exsread(segp, ip, &head.segs[SIPDATA], 0)) {
        !           188:                        goto out;
        !           189:                }
        !           190: 
        !           191:                /* Deallocate nodes hooked into xlist by exlopen. */
        !           192:                while (xlist) {
        !           193:                        struct xecnode * tmp = xlist->xn;
        !           194:                        kfree(xlist);
        !           195:                        xlist = tmp;
        !           196:                }
        !           197:                break;
        !           198:        default:
        !           199:                panic("pexece");
        !           200:        }
        !           201: 
        !           202:        u.u_regl[SS] = u.u_regl[ES] = u.u_regl[DS];
        !           203:        if (sproto(&head) == 0) {
        !           204:                goto out;
        !           205:        }
        !           206: 
        !           207:        /*
        !           208:         * The new image is read in
        !           209:         * and mapped. Perform the final grunge
        !           210:         * (set-uid stuff, accounting, loading up
        !           211:         * registers, etc).
        !           212:         */
        !           213:        u.u_flag &= ~AFORK;
        !           214:        kkcopy(u.u_direct.d_name, u.u_comm, sizeof(u.u_comm));
        !           215:        if (iaccess(ip, IPR) == 0) {    /* Can't read ? no dump or trace */
        !           216:                pp->p_flags |= PFNDMP;
        !           217:                pp->p_flags &= ~PFTRAC;
        !           218:        }
        !           219: 
        !           220:        /*
        !           221:         * Norm says Frank says we need to drop this for db to work.
        !           222:         */
        !           223: #if 0
        !           224:        if (iaccess(ip, IPW) == 0) {    /* Can't write ? no trace */
        !           225:                pp->p_flags &= ~PFTRAC;
        !           226: printf("Can't write - no trace! ");
        !           227: u.u_error = 0;
        !           228:        }
        !           229: #endif
        !           230: 
        !           231:        if ((ip->i_mode&ISUID) != 0) {  /* Set user id ? no trace */
        !           232:                pp->p_uid = u.u_uid = u.u_euid = ip->i_uid;
        !           233:                pp->p_flags &= ~PFTRAC;
        !           234:        }
        !           235: 
        !           236:        if ((ip->i_mode&ISGID) != 0) {  /* Set group id ? no trace */
        !           237:                u.u_egid = u.u_gid = ip->i_gid;
        !           238:                pp->p_flags &= ~PFTRAC;
        !           239:        }
        !           240: 
        !           241:        for (i=0; i < NOFILE; i++) {
        !           242:                if (u.u_filep[i]!=NULL && (u.u_filep[i]->f_flag2&FD_CLOEXEC))  {
        !           243:                        fdclose(i);     /* close fd on exec bit set */
        !           244:                }
        !           245:        }
        !           246: 
        !           247:        /*
        !           248:         * Default every signal that is not ignored.
        !           249:         */
        !           250:        for (i = 1; i <= NSIG; ++i) {
        !           251:                if (u.u_sfunc[i - 1] != SIG_IGN) {
        !           252:                        u.u_sfunc[i - 1] = SIG_DFL;
        !           253:                        pp->p_dfsig |= SIG_BIT(i);
        !           254:                }
        !           255:        }
        !           256: 
        !           257:        if (pp->p_flags&PFTRAC) /* Being traced */
        !           258:                sendsig(SIGTRAP, pp);
        !           259:        idetach(ip);
        !           260:        msetusr(head.entry, head.initsp);
        !           261: 
        !           262:        /* initialize u area ndp fields */
        !           263:        ndpNewProc();
        !           264: 
        !           265:        segload();
        !           266:        goto done;
        !           267: 
        !           268:        /*
        !           269:         * Alas, exec() has failed..
        !           270:         */
        !           271: out:
        !           272:        /* Deallocate nodes hooked into xlist by exlopen. */
        !           273:        while (xlist) {
        !           274:                struct xecnode * tmp = xlist->xn;
        !           275:                kfree(xlist);
        !           276:                xlist = tmp;
        !           277:        }
        !           278: 
        !           279:        /* Release the INODE for the load file. */
        !           280:        idetach(ip);
        !           281: 
        !           282:        /* If we allocated a text segment, let it go. */
        !           283:        if (segp = pp->p_segp[SISTEXT]) {
        !           284:                pp->p_segp[SISTEXT] = NULL;
        !           285:                sfree(segp);
        !           286:        }
        !           287: 
        !           288:        /* If we allocated a data segment, let it go. */
        !           289:        if (segp = pp->p_segp[SIPDATA]) {
        !           290:                pp->p_segp[SIPDATA] = NULL;
        !           291:                sfree(segp);
        !           292:        }
        !           293: 
        !           294:        /*
        !           295:         * Return through the "sys exit" code with a "SIGSYS", or with the
        !           296:         * signal actually received if we are aborting due to interrupted exec.
        !           297:         */
        !           298:        if (u.u_error == EINTR)
        !           299:                pexit(nondsig());
        !           300:        pexit(SIGSYS);
        !           301: 
        !           302: done:
        !           303:        return 0;       
        !           304: }
        !           305: 
        !           306: /*
        !           307:  * Open a file, make sure it is l.out, coff, or v86 as well as
        !           308:  * executable.
        !           309:  *
        !           310:  * "xhp" points to a cleared xechdr supplied by the caller.
        !           311:  * "np" is the file name.
        !           312:  * "shrds" points to an int that will be written by exlopen().
        !           313:  *   *shrds is set nonzero only for shared l.out.
        !           314:  *
        !           315:  * If file is COFF, there may be multiple text (or data?) sections.
        !           316:  * Use "xlist" linked structure to keep track of variably many sections
        !           317:  * after the first text and data sections.
        !           318:  *
        !           319:  * return NULL if failure, else return inode pointer for the file.
        !           320:  */
        !           321: INODE *
        !           322: exlopen(xhp, np, shrds, xlist) 
        !           323: register struct xechdr *xhp;
        !           324: char *np;
        !           325: int *shrds;
        !           326: struct xecnode ** xlist;
        !           327: {
        !           328:        register INODE *ip;
        !           329:        int     i, nscn, hdrsize;
        !           330:        register BUF *bp;
        !           331:        unsigned short magic;
        !           332:        struct ldheader head;
        !           333:        struct filehdr fhead;
        !           334:        struct aouthdr ahead;   
        !           335:        struct scnhdr scnhdr;
        !           336: 
        !           337:        /*
        !           338:         * Make sure the file is executable and read the header.
        !           339:         */
        !           340:        if (ftoi(np, 'r'))
        !           341:                return NULL;
        !           342:        ip = u.u_cdiri;
        !           343:        if (iaccess(ip, IPE) == 0) {
        !           344:                idetach(ip);
        !           345:                return NULL;
        !           346:        }
        !           347: 
        !           348:        if ((ip->i_mode&(IPE|IPE<<3|IPE<<6))==0 || (ip->i_mode&IFMT)!=IFREG) {
        !           349:                u.u_error = EACCES;
        !           350:                idetach(ip);
        !           351:                return NULL;
        !           352:        }
        !           353: 
        !           354:        if ((bp=vread(ip, (daddr_t)0)) == NULL) {
        !           355:                goto bad;
        !           356:        }
        !           357: 
        !           358:        /*
        !           359:         * Copy everything we need from the l.out header and check magic
        !           360:         * number and machine type.
        !           361:         */
        !           362:        *shrds = 0;
        !           363:        kkcopy(bp->b_vaddr, &magic, sizeof(magic));
        !           364:        canint(magic);
        !           365:        switch (magic) {
        !           366:        case L_MAGIC:           /* Coherent 286 format */
        !           367:                kkcopy(bp->b_vaddr, &head, sizeof(struct ldheader));
        !           368:                canint(head.l_machine);
        !           369:                if (head.l_machine!=M_8086) {
        !           370:                        goto bad;
        !           371:                }
        !           372: 
        !           373:                for (i=0; i<NXSEG; i++) {
        !           374:                        cansize(head.l_ssize[i]);
        !           375:                }       
        !           376:                canint(head.l_flag);
        !           377:                canvaddr(head.l_entry);
        !           378: 
        !           379:                /*
        !           380:                 * If a shared and separated image
        !           381:                 * has stuff in segments that makes it impossible
        !           382:                 * to share, give an error immediately so that we don't
        !           383:                 * lose the parent.
        !           384:                 */
        !           385:                head.l_flag &= LF_SHR|LF_SEP|LF_KER;
        !           386: 
        !           387:                if ((head.l_flag&LF_SEP==0) || (head.l_flag &LF_KER)
        !           388:                  || head.l_ssize[L_PRVI] || head.l_ssize[L_BSSI]) {
        !           389:                        goto bad;
        !           390:                }
        !           391:                xhp->magic = XMAGIC(I286MAGIC,I_MAGIC);
        !           392:                xhp->entry = head.l_entry;
        !           393: 
        !           394:                xhp->segs[SISTEXT].fbase = sizeof(struct ldheader);
        !           395:                xhp->segs[SISTEXT].mbase = NBPS;
        !           396:                xhp->segs[SISTEXT].size = head.l_ssize[L_SHRI];
        !           397: 
        !           398:                xhp->segs[SIPDATA].fbase = sizeof(struct ldheader) +
        !           399:                        xhp->segs[SISTEXT].size;
        !           400:                xhp->segs[SIPDATA].mbase = 0;
        !           401:                xhp->segs[SIPDATA].size = head.l_ssize[L_SHRD] +
        !           402:                        head.l_ssize[L_PRVD];
        !           403:                if (head.l_flag & LF_SHR)
        !           404:                        *shrds = head.l_ssize[L_SHRD];
        !           405: 
        !           406:                xhp->segs[SIBSS].fbase = 0;
        !           407:                xhp->segs[SIBSS].mbase = xhp->segs[SIPDATA].size;
        !           408:                xhp->segs[SIBSS].size = head.l_ssize[L_BSSD];
        !           409: 
        !           410:                xhp->segs[SISTACK].mbase = ISP_286;     /* size 0, fbase 0 */
        !           411:                brelease(bp);
        !           412:                return ip;
        !           413: 
        !           414:        case I386MAGIC:         /* ... COFF */
        !           415:                kkcopy(bp->b_vaddr, &fhead, sizeof(struct filehdr));
        !           416:                hdrsize = sizeof(ahead) + sizeof(fhead);
        !           417:                if(fhead.f_opthdr != sizeof (ahead)
        !           418:                  || !(fhead.f_flags & F_EXEC)
        !           419:                  || fhead.f_nscns * sizeof(scnhdr) > BSIZE) {
        !           420:                        goto bad;
        !           421:                }
        !           422: 
        !           423:                kkcopy(bp->b_vaddr + sizeof(fhead), &ahead, sizeof(ahead));
        !           424:                if (ahead.magic != Z_MAGIC) {
        !           425:                        goto bad;
        !           426:                }
        !           427: 
        !           428:                xhp->magic = XMAGIC(I386MAGIC, ahead.magic);
        !           429:                xhp->entry = ahead.entry;
        !           430: 
        !           431:                for (i = 0; i < fhead.f_nscns; i++) {
        !           432:                        kkcopy(bp->b_vaddr + hdrsize + sizeof(scnhdr)*i,
        !           433:                                &scnhdr, sizeof(scnhdr));
        !           434:                        switch ((int)(scnhdr.s_flags)) {
        !           435:                        case STYP_INFO:
        !           436:                                continue;
        !           437:                        case STYP_BSS:
        !           438:                                nscn = SIBSS;
        !           439:                                break;
        !           440:                        case STYP_TEXT:
        !           441:                                nscn = SISTEXT;
        !           442:                                break;
        !           443:                        case STYP_DATA:
        !           444:                                nscn = SIPDATA;
        !           445:                                break;
        !           446:                        default:
        !           447:                                goto bad;
        !           448:                        }
        !           449: 
        !           450:                        /* Text/data shouldn't collide with stack. */
        !           451:                        if ((unsigned)scnhdr.s_vaddr >= ISP_386) {
        !           452:                                goto bad;
        !           453:                        }
        !           454: 
        !           455:                        /* Have we already seen a segment of this type? */
        !           456:                        if (xhp->segs[nscn].size) {
        !           457:                                struct xecnode * tmp;
        !           458: 
        !           459:                                if (nscn != SISTEXT) {
        !           460:                                        goto bad;
        !           461:                                }
        !           462: 
        !           463:                                /* insert new node at head of "xlist" */
        !           464:                                if (!(tmp = (struct xecnode *)
        !           465:                                  kalloc(sizeof (struct xecnode)))) {
        !           466:                                        printf("can't kalloc(xecnode)\n");
        !           467:                                        goto bad;
        !           468:                                }
        !           469:                                tmp->xn = *xlist;
        !           470:                                *xlist = tmp;
        !           471:                                tmp->segtype = nscn;
        !           472:                                tmp->xseg.mbase = scnhdr.s_vaddr;
        !           473:                                tmp->xseg.fbase = scnhdr.s_scnptr;
        !           474:                                tmp->xseg.size = scnhdr.s_size;
        !           475:                        } else {
        !           476:                                xhp->segs[nscn].mbase = scnhdr.s_vaddr;
        !           477:                                xhp->segs[nscn].fbase = scnhdr.s_scnptr;
        !           478:                                xhp->segs[nscn].size = scnhdr.s_size;
        !           479:                        }
        !           480:                }
        !           481: 
        !           482:                /* Text and data segments must both be nonempty. */
        !           483:                if (!xhp->segs[SISTEXT].size || !xhp->segs[SIPDATA].size) {
        !           484:                        goto bad;
        !           485:                }
        !           486: 
        !           487:                xhp->entry = ahead.entry;
        !           488: 
        !           489:                xhp->segs[SISTACK].mbase = ISP_386;     /* size 0, fbase 0 */
        !           490:                xhp->magic = XMAGIC(I386MAGIC,ahead.magic);
        !           491:                brelease(bp);   
        !           492:                return ip;
        !           493:        default:
        !           494:        bad:            
        !           495:                brelease(bp);
        !           496:                u.u_error = ENOEXEC;
        !           497:                idetach(ip);
        !           498:                return NULL;
        !           499:        }
        !           500: }
        !           501: 
        !           502: static SEG *
        !           503: exsread(sp, ip, xsp, shrdSz)
        !           504: register SEG *sp;
        !           505: INODE *ip;
        !           506: struct xecseg *xsp;
        !           507: int shrdSz;
        !           508: {
        !           509:        register int sa, so;
        !           510: 
        !           511:        sa = xsp->fbase;
        !           512:        so = xsp->mbase & (NBPC - 1);
        !           513: 
        !           514:        u.u_io.io_seg = IOPHY;
        !           515:        u.u_io.io_seek = sa;
        !           516:        u.u_io.io.pbase = MAPIO(sp->s_vmem, so);
        !           517:        u.u_io.io_flag = 0;
        !           518: 
        !           519:        if (shrdSz) {   /* shared l.out? */
        !           520: 
        !           521:                /* Load SHRD. */
        !           522:                u.u_io.io_ioc = shrdSz;
        !           523:                sp->s_lrefc++;
        !           524:                iread(ip, &u.u_io);
        !           525:                sp->s_lrefc--;
        !           526: 
        !           527:                if ((u.u_io.io_ioc = xsp->size - shrdSz) != 0) {
        !           528: 
        !           529:                        /* Advance file and RAM offsets past SHRD. */
        !           530:                        sa += shrdSz;
        !           531:                        so += shrdSz;
        !           532: 
        !           533:                        /* Advance RAM offset to next 16-byte boundary. */
        !           534:                        so = (so + 15) & ~15; /* round up */
        !           535: 
        !           536:                        /* Load PRVD. */
        !           537:                        u.u_io.io_seg = IOPHY;
        !           538:                        u.u_io.io_seek = sa;
        !           539:                        u.u_io.io.pbase = MAPIO(sp->s_vmem, so);
        !           540:                        u.u_io.io_flag = 0;
        !           541:                        sp->s_lrefc++;
        !           542:                        iread(ip, &u.u_io);
        !           543:                        sp->s_lrefc--;
        !           544:                }
        !           545:        } else {        /* NOT shared l.out */
        !           546:                u.u_io.io_ioc = xsp->size;
        !           547: 
        !           548:                sp->s_lrefc++;
        !           549:                iread(ip, &u.u_io);
        !           550:                sp->s_lrefc--;
        !           551:        }
        !           552:        if (nondsig())
        !           553:                u.u_error = EINTR;
        !           554: 
        !           555:        if (u.u_error == 0)
        !           556:                return (sp);
        !           557:        return NULL;
        !           558: }
        !           559: 
        !           560: struct adata {         /* Storage for arg and env data */
        !           561:        int     np;             /* Number of pointers in vector */
        !           562:        int     nc;             /* Number of characters in strings */
        !           563: };
        !           564: 
        !           565: /*
        !           566:  * Given a pointer to a list of arguments and a pointer to a list of
        !           567:  * environments, return a stack with the arguments and environments on it.
        !           568:  */
        !           569: SEG *
        !           570: exstack(xhp, argp, envp, wdin)
        !           571: register struct xechdr *xhp;
        !           572: caddr_t        argp, envp;
        !           573: {
        !           574:        register SEG *sp;               /* Stack segment pointer */
        !           575:        struct sdata {          /* To keep segment pointers */
        !           576:                caddr_t vp;             /* Argv[i], envp[i] pointer */
        !           577:                caddr_t cp;             /* Argv[i][j], envp[i][j] pointer */
        !           578:        } stk;
        !           579:        struct adata arg, env;
        !           580:        int     chrsz, vecsz, stksz, wdmask, wdout, stkoff, stktop;
        !           581:        int     stkenvp;
        !           582:        register int i;
        !           583: 
        !           584:        /* Validate and evaluate size of args and envs */
        !           585:        if (!excount(argp, &arg, wdin) || !excount(envp, &env, wdin)) {
        !           586:                return NULL;
        !           587:        }
        !           588: 
        !           589:        /* Calculate stack size and allocate it */
        !           590:        chrsz = roundu(arg.nc + env.nc, sizeof(int));
        !           591:        vecsz = (arg.np+1 + env.np+1) * sizeof(long);
        !           592:        stksz = roundu(vecsz+chrsz+ISTSIZE, NBPC);
        !           593:        if (stksz > MADSIZE || !(sp = salloc(stksz, SFDOWN))) {
        !           594:                u.u_error = E2BIG;
        !           595:                return NULL;
        !           596:        }
        !           597: 
        !           598:        /* Set up target stack */
        !           599:        stktop = xhp->segs[SISTACK].mbase;
        !           600:        stk.cp = stktop - chrsz;
        !           601:        stk.vp = stktop - chrsz - vecsz;
        !           602:        stkoff = MAPIO(sp->s_vmem, stksz - stktop);
        !           603:        u.u_argc = arg.np;
        !           604:        u.u_argp = stk.vp;
        !           605:        wdmask = -1;
        !           606:        if (wdin==sizeof(short))
        !           607:                wdmask = (unsigned short)wdmask;
        !           608: 
        !           609:        switch (stktop) {
        !           610: 
        !           611:        case ISP_386:
        !           612:                wdout = sizeof(long);
        !           613:                xhp->initsp = stk.vp - sizeof(long);
        !           614:                dmaout(sizeof(long), xhp->initsp+stkoff, &arg.np);
        !           615:                break;
        !           616: 
        !           617:        case ISP_286:
        !           618:                wdout = sizeof(short);
        !           619:                xhp->initsp = stk.vp - 3*sizeof(short);
        !           620:                stkenvp = stk.vp + (arg.np+1) * sizeof(short);
        !           621:                dmaout(sizeof(short), xhp->initsp+stkoff, &arg.np);
        !           622:                dmaout(sizeof(short), xhp->initsp+sizeof(short)+stkoff,
        !           623:                        &stk.vp);
        !           624:                dmaout(sizeof(short), xhp->initsp+2*sizeof(short)+stkoff,
        !           625:                        &stkenvp);
        !           626:                break;
        !           627: 
        !           628:        default:
        !           629:                panic("exstack");
        !           630: 
        !           631:        }
        !           632: 
        !           633:        /* Arguments */
        !           634:        for (i = 0; i<arg.np; i++, argp += wdin, stk.vp += wdout) {
        !           635:                dmaout(wdout, stk.vp+stkoff, &stk.cp);
        !           636:                stk.cp += exarg(stk.cp+stkoff, getupd(argp) & wdmask);
        !           637:        }
        !           638: 
        !           639:        /* skip null word after arguments */
        !           640:        stk.vp += wdout;
        !           641: 
        !           642:        /* Environments */
        !           643:        for (i = 0; i<env.np; i++, envp += wdin, stk.vp += wdout) {
        !           644:                dmaout(wdout, stk.vp+stkoff, &stk.cp);
        !           645:                stk.cp += exarg(stk.cp+stkoff, getupd(envp) & wdmask);
        !           646:        }
        !           647: 
        !           648:        return sp;
        !           649: }
        !           650: 
        !           651: exarg(out, in)
        !           652: caddr_t        in, out;
        !           653: {
        !           654:        char    c;
        !           655:        caddr_t init_in;
        !           656: 
        !           657:        init_in = in;
        !           658:        do {
        !           659:                c = getubd(in++);
        !           660:                dmaout(sizeof(char), out++, &c);
        !           661:        } while (c);
        !           662:        return in - init_in;
        !           663: }
        !           664: 
        !           665: /*
        !           666:  * Given a pointer to a list of arguments, a pointer to an argument count
        !           667:  * and a pointer to a byte count, count the #characters/#strings
        !           668:  * in the arguments
        !           669:  */
        !           670: excount(usrvp, adp, wdin)
        !           671: register caddr_t usrvp;
        !           672: struct adata *adp;
        !           673: {
        !           674:        register caddr_t        usrcp;
        !           675:        register int c;
        !           676:        register unsigned nb;
        !           677:        register unsigned na;
        !           678:        int     wdmask;
        !           679: 
        !           680:        wdmask = -1;
        !           681:        if (wdin==sizeof(short))
        !           682:                wdmask = (unsigned short)wdmask;
        !           683:        na = nb = 0;
        !           684:        if (usrvp != NULL) {
        !           685:                for (;;) {
        !           686:                        usrcp = getupd(usrvp) & wdmask;
        !           687:                        usrvp += wdin;
        !           688:                        if (u.u_error)
        !           689:                                return (0);
        !           690:                        if (usrcp == NULL)
        !           691:                                break;
        !           692:                        na++;
        !           693:                        for (;;) {
        !           694:                                c = getubd(usrcp++);
        !           695:                                if (u.u_error)
        !           696:                                        return (0);
        !           697:                                nb++;
        !           698:                                if (c == '\0')
        !           699:                                        break;
        !           700:                        }
        !           701:                }
        !           702:        }
        !           703:        adp->np = na;
        !           704:        adp->nc = nb;
        !           705:        return (1);
        !           706: }
        !           707: 
        !           708: /*
        !           709:  * Round up a size to a paragraph
        !           710:  * (mod 16) boundry.
        !           711:  * This is really mod 512 to make swapping work
        !           712:  */
        !           713: off_t
        !           714: exround(s)
        !           715: off_t  s;
        !           716: {
        !           717:        return ((s+15)&~0x0F);
        !           718: }
        !           719: 
        !           720: pload(np)
        !           721: char * np;
        !           722: {
        !           723:        return -1;
        !           724: 
        !           725: }
        !           726: /*
        !           727:  * Set up the first process, a small program which will exec
        !           728:  * the init program.
        !           729:  */
        !           730: extern char aicodep[];
        !           731: 
        !           732: eveinit()
        !           733: {
        !           734:        SEG *sp;
        !           735:        register PROC *pp;
        !           736:        SELF = pp = eprocp;
        !           737: /*     static struct xechdr xecinit[NUSEG+1] = { {0},{0},{0},{ISP_386}}; */ 
        !           738: 
        !           739:        /*
        !           740:         * Allocate, record, initialize code segment, make it executable.
        !           741:         */
        !           742:        if ((sp = salloc(roundu(icodes, NBPC), 0)) == NULL)
        !           743:                panic("eveinit(code)");
        !           744:        pp->p_segp[SIPDATA] = sp;
        !           745: 
        !           746:        /*
        !           747:         * Start process.
        !           748:         */
        !           749:        u.u_argp = 0;
        !           750:        if (sproto(0) == 0)
        !           751:                panic("eveinit()");
        !           752:        segload();
        !           753:        kucopy(aicodep, 0, icodes);
        !           754: }
        !           755: 
        !           756: /*
        !           757:  * Given a major number, undo the previous function.
        !           758:  */
        !           759: puload(m)
        !           760: int m;
        !           761: {
        !           762:        register CON *cp;
        !           763:        register DRV *dp;
        !           764: 
        !           765:        dp = &drvl[m];
        !           766:        lock(dp->d_gate);
        !           767:        if (m>=drvn || (cp=dp->d_conp)==NULL) {
        !           768:                u.u_error = ENXIO;
        !           769:                goto ret;
        !           770:        }
        !           771:        (*cp->c_uload)();
        !           772:        if (! u.u_error)
        !           773:                dp->d_conp = NULL;
        !           774: ret:
        !           775:        unlock(dp->d_gate);
        !           776:        return (0);
        !           777: }
        !           778: 

unix.superglobalmegacorp.com

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