Annotation of coherent/b/kernel/coh.386/exec.c, revision 1.1.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.