Annotation of 43BSDReno/usr.bin/ex/ex_io.c, revision 1.1.1.1

1.1       root        1: /*
                      2:  * Copyright (c) 1980 Regents of the University of California.
                      3:  * All rights reserved.  The Berkeley software License Agreement
                      4:  * specifies the terms and conditions for redistribution.
                      5:  */
                      6: 
                      7: #ifndef lint
                      8: static char *sccsid = "@(#)ex_io.c     7.17 (Berkeley) 1/2/88";
                      9: #endif not lint
                     10: 
                     11: #include "ex.h"
                     12: #include "ex_argv.h"
                     13: #include "ex_temp.h"
                     14: #include "ex_tty.h"
                     15: #include "ex_vis.h"
                     16: #include <sys/file.h>
                     17: #include <sys/exec.h>
                     18: #include "pathnames.h"
                     19: 
                     20: /*
                     21:  * File input/output, source, preserve and recover
                     22:  */
                     23: 
                     24: /*
                     25:  * Following remember where . was in the previous file for return
                     26:  * on file switching.
                     27:  */
                     28: int    altdot;
                     29: int    oldadot;
                     30: bool   wasalt;
                     31: short  isalt;
                     32: 
                     33: long   cntch;                  /* Count of characters on unit io */
                     34: #ifndef VMUNIX
                     35: short  cntln;                  /* Count of lines " */
                     36: #else
                     37: int    cntln;
                     38: #endif
                     39: long   cntnull;                /* Count of nulls " */
                     40: long   cntodd;                 /* Count of non-ascii characters " */
                     41: 
                     42: #ifdef FLOCKFILE
                     43: /*
                     44:  * The alternate, saved and current file are locked the extent of the
                     45:  * time that they are active. If the saved file is exchanged
                     46:  * with the alternate file, the file descriptors are exchanged
                     47:  * and the lock is not released.
                     48:  */
                     49: int    io_savedfile, io_altfile, io_curr ;
                     50: int    lock_savedfile, lock_altfile, lock_curr ;
                     51: #endif FLOCKFILE
                     52: 
                     53: /*
                     54:  * Parse file name for command encoded by comm.
                     55:  * If comm is E then command is doomed and we are
                     56:  * parsing just so user won't have to retype the name.
                     57:  */
                     58: filename(comm)
                     59:        int comm;
                     60: {
                     61:        register int c = comm, d;
                     62:        register int i;
                     63: #ifdef FLOCKFILE
                     64:        int lock ;
                     65: 
                     66:        lock = 0 ;
                     67: #endif FLOCKFILE
                     68: 
                     69:        d = ex_getchar();
                     70:        if (endcmd(d)) {
                     71:                if (savedfile[0] == 0 && comm != 'f')
                     72:                        error("No file|No current filename");
                     73:                CP(file, savedfile);
                     74: #ifdef FLOCKFILE
                     75:                if (io_curr && io_curr != io_savedfile) close(io_curr) ;
                     76:                lock = lock_curr = lock_savedfile ;
                     77:                io_curr = io_savedfile ;
                     78: #endif FLOCKFILE
                     79:                wasalt = (isalt > 0) ? isalt-1 : 0;
                     80:                isalt = 0;
                     81:                oldadot = altdot;
                     82:                if (c == 'e' || c == 'E')
                     83:                        altdot = lineDOT();
                     84:                if (d == EOF)
                     85:                        ungetchar(d);
                     86:        } else {
                     87:                ungetchar(d);
                     88:                getone();
                     89:                eol();
                     90:                if (savedfile[0] == 0 && c != 'E' && c != 'e') {
                     91:                        c = 'e';
                     92:                        edited = 0;
                     93:                }
                     94:                wasalt = strcmp(file, altfile) == 0;
                     95:                oldadot = altdot;
                     96:                switch (c) {
                     97: 
                     98:                case 'f':
                     99:                        edited = 0;
                    100:                        /* fall into ... */
                    101: 
                    102:                case 'e':
                    103:                        if (savedfile[0]) {
                    104: #ifdef FLOCKFILE
                    105:                                if (strcmp(file,savedfile) == 0) break ;
                    106: #endif FLOCKFILE
                    107:                                altdot = lineDOT();
                    108:                                CP(altfile, savedfile);
                    109: #ifdef FLOCKFILE
                    110:                                if (io_altfile) close (io_altfile) ;
                    111:                                io_altfile = io_savedfile ;
                    112:                                lock_altfile = lock_savedfile ;
                    113:                                io_savedfile = 0 ;
                    114: #endif FLOCKFILE
                    115:                        }
                    116:                        CP(savedfile, file);
                    117: #ifdef FLOCKFILE
                    118:                        io_savedfile = io_curr ;
                    119:                        lock_savedfile = lock_curr ;
                    120:                        io_curr = 0 ;           lock = lock_curr = 0 ;
                    121: #endif FLOCKFILE
                    122:                        break;
                    123: 
                    124:                default:
                    125:                        if (file[0]) {
                    126: #ifdef FLOCKFILE
                    127:                                if (wasalt) break ;
                    128: #endif
                    129:                                if (c != 'E')
                    130:                                        altdot = lineDOT();
                    131:                                CP(altfile, file);
                    132: #ifdef FLOCKFILE
                    133:                                if (io_altfile
                    134:                                && io_altfile != io_curr) close (io_altfile) ;
                    135:                                io_altfile = io_curr ;
                    136:                                lock_altfile = lock_curr ;
                    137:                                io_curr = 0 ;           lock = lock_curr = 0 ;
                    138: #endif FLOCKFILE
                    139:                        }
                    140:                        break;
                    141:                }
                    142:        }
                    143:        if (hush && comm != 'f' || comm == 'E')
                    144:                return;
                    145:        if (file[0] != 0) {
                    146:                lprintf("\"%s\"", file);
                    147:                if (comm == 'f') {
                    148:                        if (value(READONLY))
                    149:                                ex_printf(" [Read only]");
                    150:                        if (!edited)
                    151:                                ex_printf(" [Not edited]");
                    152:                        if (tchng)
                    153:                                ex_printf(" [Modified]");
                    154: #ifdef FLOCKFILE
                    155:                        if (lock == LOCK_SH)
                    156:                                ex_printf(" [Shared lock]") ;
                    157:                        else if (lock == LOCK_EX)
                    158:                                ex_printf(" [Exclusive lock]") ;
                    159: #endif FLOCKFILE
                    160:                }
                    161:                flush();
                    162:        } else
                    163:                ex_printf("No file ");
                    164:        if (comm == 'f') {
                    165:                if (!(i = lineDOL()))
                    166:                        i++;
                    167:                ex_printf(" line %d of %d --%ld%%--", lineDOT(), lineDOL(),
                    168:                    (long) 100 * lineDOT() / i);
                    169:        }
                    170: }
                    171: 
                    172: /*
                    173:  * Get the argument words for a command into genbuf
                    174:  * expanding # and %.
                    175:  */
                    176: getargs()
                    177: {
                    178:        register int c;
                    179:        register char *cp, *fp;
                    180:        static char fpatbuf[32];        /* hence limit on :next +/pat */
                    181: 
                    182:        pastwh();
                    183:        if (peekchar() == '+') {
                    184:                for (cp = fpatbuf;;) {
                    185:                        c = *cp++ = ex_getchar();
                    186:                        if (cp >= &fpatbuf[sizeof(fpatbuf)])
                    187:                                error("Pattern too long");
                    188:                        if (c == '\\' && isspace(peekchar()))
                    189:                                c = ex_getchar();
                    190:                        if (c == EOF || isspace(c)) {
                    191:                                ungetchar(c);
                    192:                                *--cp = 0;
                    193:                                firstpat = &fpatbuf[1];
                    194:                                break;
                    195:                        }
                    196:                }
                    197:        }
                    198:        if (skipend())
                    199:                return (0);
                    200:        CP(genbuf, "echo "); cp = &genbuf[5];
                    201:        for (;;) {
                    202:                c = ex_getchar();
                    203:                if (endcmd(c)) {
                    204:                        ungetchar(c);
                    205:                        break;
                    206:                }
                    207:                switch (c) {
                    208: 
                    209:                case '\\':
                    210:                        if (any(peekchar(), "#%|"))
                    211:                                c = ex_getchar();
                    212:                        /* fall into... */
                    213: 
                    214:                default:
                    215:                        if (cp > &genbuf[LBSIZE - 2])
                    216: flong:
                    217:                                error("Argument buffer overflow");
                    218:                        *cp++ = c;
                    219:                        break;
                    220: 
                    221:                case '#':
                    222:                        fp = altfile;
                    223:                        if (*fp == 0)
                    224:                                error("No alternate filename@to substitute for #");
                    225:                        goto filexp;
                    226: 
                    227:                case '%':
                    228:                        fp = savedfile;
                    229:                        if (*fp == 0)
                    230:                                error("No current filename@to substitute for %%");
                    231: filexp:
                    232:                        while (*fp) {
                    233:                                if (cp > &genbuf[LBSIZE - 2])
                    234:                                        goto flong;
                    235:                                *cp++ = *fp++;
                    236:                        }
                    237:                        break;
                    238:                }
                    239:        }
                    240:        *cp = 0;
                    241:        return (1);
                    242: }
                    243: 
                    244: /*
                    245:  * Glob the argument words in genbuf, or if no globbing
                    246:  * is implied, just split them up directly.
                    247:  */
                    248: glob(gp)
                    249:        struct glob *gp;
                    250: {
                    251:        int pvec[2];
                    252:        register char **argv = gp->argv;
                    253:        register char *cp = gp->argspac;
                    254:        register int c;
                    255:        char ch;
                    256:        int nleft = NCARGS;
                    257: 
                    258:        gp->argc0 = 0;
                    259:        if (gscan() == 0) {
                    260:                register char *v = genbuf + 5;          /* strlen("echo ") */
                    261: 
                    262:                for (;;) {
                    263:                        while (isspace(*v))
                    264:                                v++;
                    265:                        if (!*v)
                    266:                                break;
                    267:                        *argv++ = cp;
                    268:                        while (*v && !isspace(*v))
                    269:                                *cp++ = *v++;
                    270:                        *cp++ = 0;
                    271:                        gp->argc0++;
                    272:                }
                    273:                *argv = 0;
                    274:                return;
                    275:        }
                    276:        if (pipe(pvec) < 0)
                    277:                error("Can't make pipe to glob");
                    278:        pid = vfork();
                    279:        io = pvec[0];
                    280:        if (pid < 0) {
                    281:                close(pvec[1]);
                    282:                error("Can't fork to do glob");
                    283:        }
                    284:        if (pid == 0) {
                    285:                int oerrno;
                    286: 
                    287:                if (genbuf) {
                    288:                        register char *ccp = genbuf;
                    289:                        while (*ccp)
                    290:                                *ccp++ &= TRIM;
                    291:                }
                    292:                close(1);
                    293:                dup(pvec[1]);
                    294:                close(pvec[0]);
                    295:                close(2);       /* so errors don't mess up the screen */
                    296:                ignore(open(_PATH_DEVNULL, 1));
                    297:                execl(svalue(SHELL), "sh", "-c", genbuf, 0);
                    298:                oerrno = errno;
                    299:                close(1);
                    300:                dup(2);
                    301:                errno = oerrno;
                    302:                filioerr(svalue(SHELL));
                    303:        }
                    304:        close(pvec[1]);
                    305:        do {
                    306:                *argv = cp;
                    307:                for (;;) {
                    308:                        if (read(io, &ch, 1) != 1) {
                    309:                                close(io);
                    310:                                c = -1;
                    311:                        } else
                    312:                                c = ch & TRIM;
                    313:                        if (c <= 0 || isspace(c))
                    314:                                break;
                    315:                        *cp++ = c;
                    316:                        if (--nleft <= 0)
                    317:                                error("Arg list too long");
                    318:                }
                    319:                if (cp != *argv) {
                    320:                        --nleft;
                    321:                        *cp++ = 0;
                    322:                        gp->argc0++;
                    323:                        if (gp->argc0 >= NARGS)
                    324:                                error("Arg list too long");
                    325:                        argv++;
                    326:                }
                    327:        } while (c >= 0);
                    328:        waitfor();
                    329:        if (gp->argc0 == 0)
                    330:                error("No match");
                    331: }
                    332: 
                    333: /*
                    334:  * Scan genbuf for shell metacharacters.
                    335:  * Set is union of v7 shell and csh metas.
                    336:  */
                    337: gscan()
                    338: {
                    339: #ifndef        vms                     /* Never have meta-characters in vms */
                    340:        register char *cp;
                    341: 
                    342:        for (cp = genbuf; *cp; cp++)
                    343:                if (any(*cp, "~{[*?$`'\"\\"))
                    344:                        return (1);
                    345: #endif
                    346:        return (0);
                    347: }
                    348: 
                    349: /*
                    350:  * Parse one filename into file.
                    351:  */
                    352: struct glob G;
                    353: getone()
                    354: {
                    355:        register char *str;
                    356: 
                    357:        if (getargs() == 0)
                    358:                error("Missing filename");
                    359:        glob(&G);
                    360:        if (G.argc0 > 1)
                    361:                error("Ambiguous|Too many file names");
                    362:        str = G.argv[G.argc0 - 1];
                    363:        if (strlen(str) > FNSIZE - 4)
                    364:                error("Filename too long");
                    365:        CP(file, str);
                    366: }
                    367: 
                    368: /*
                    369:  * Read a file from the world.
                    370:  * C is command, 'e' if this really an edit (or a recover).
                    371:  */
                    372: rop(c)
                    373:        int c;
                    374: {
                    375:        register int i;
                    376:        struct stat stbuf;
                    377:        struct exec head;
                    378:        static int ovro;        /* old value(READONLY) */
                    379:        static int denied;      /* 1 if READONLY was set due to file permissions */
                    380: #ifdef FLOCKFILE
                    381:        int *lp, *iop;
                    382: #endif FLOCKFILE
                    383: 
                    384:        io = open(file, 0);
                    385:        if (io < 0) {
                    386:                if (c == 'e' && errno == ENOENT) {
                    387:                        edited++;
                    388:                        /*
                    389:                         * If the user just did "ex foo" he is probably
                    390:                         * creating a new file.  Don't be an error, since
                    391:                         * this is ugly, and it screws up the + option.
                    392:                         */
                    393:                        if (!seenprompt) {
                    394:                                ex_printf(" [New file]");
                    395:                                noonl();
                    396:                                return;
                    397:                        }
                    398:                }
                    399:                syserror();
                    400:        }
                    401:        if (fstat(io, &stbuf))
                    402:                syserror();
                    403:        switch (stbuf.st_mode & S_IFMT) {
                    404: 
                    405:        case S_IFBLK:
                    406:                error(" Block special file");
                    407: 
                    408:        case S_IFCHR:
                    409:                if (isatty(io))
                    410:                        error(" Teletype");
                    411:                if (samei(&stbuf, _PATH_DEVNULL))
                    412:                        break;
                    413:                error(" Character special file");
                    414: 
                    415:        case S_IFDIR:
                    416:                error(" Directory");
                    417: 
                    418:        case S_IFREG:
                    419: #ifdef CRYPT
                    420:                if (xflag)
                    421:                        break;
                    422: #endif
                    423:                i = read(io, (char *)&head, sizeof(head));
                    424:                (void)lseek(io, 0L, L_SET);
                    425:                if (i != sizeof(head))
                    426:                        break;
                    427: #ifndef vms
                    428:                switch ((int)head.a_magic) {
                    429: 
                    430:                case 0405:      /* data overlay on exec */
                    431:                case OMAGIC:    /* unshared */
                    432:                case NMAGIC:    /* shared text */
                    433:                case 0411:      /* separate I/D */
                    434:                case ZMAGIC:    /* VM/Unix demand paged */
                    435:                case 0430:      /* PDP-11 Overlay shared */
                    436:                case 0431:      /* PDP-11 Overlay sep I/D */
                    437:                        error(" Executable");
                    438: 
                    439:                /*
                    440:                 * We do not forbid the editing of portable archives
                    441:                 * because it is reasonable to edit them, especially
                    442:                 * if they are archives of text files.  This is
                    443:                 * especially useful if you archive source files together
                    444:                 * and copy them to another system with ~%take, since
                    445:                 * the files sometimes show up munged and must be fixed.
                    446:                 */
                    447:                case 0177545:
                    448:                case 0177555:
                    449:                        error(" Archive");
                    450:                case 070707:
                    451:                        error(" Cpio file");
                    452: 
                    453:                default:
                    454:                        {
                    455:                                char *bp = (char *)&head;
                    456:                                if ((u_char)bp[0] == (u_char)'\037' &&
                    457:                                    (u_char)bp[1] == (u_char)'\235')
                    458:                                        error(" Compressed file");
                    459:                                if (!strncmp(bp, "!<arch>\n__.SYMDEF", 17)
                    460:                                    || !strncmp(bp, "!<arch>\n", 8))
                    461:                                        error(" Archive");
                    462:                        }
                    463:                        break;
                    464:                }
                    465: #endif
                    466:        }
                    467:        if (c != 'r') {
                    468:                if (value(READONLY) && denied) {
                    469:                        value(READONLY) = ovro;
                    470:                        denied = 0;
                    471:                }
                    472:                if ((stbuf.st_mode & 0222) == 0 || access(file, 2) < 0) {
                    473:                        ovro = value(READONLY);
                    474:                        denied = 1;
                    475:                        value(READONLY) = 1;
                    476:                }
                    477:        }
                    478:        if (value(READONLY)) {
                    479:                ex_printf(" [Read only]");
                    480:                flush();
                    481:        }
                    482: #ifdef FLOCKFILE
                    483:        /*
                    484:         * Attempt to lock the file. We use an sharable lock if reading
                    485:         * the file, and an exclusive lock if editting a file.
                    486:         * The lock will be released when the file is no longer being
                    487:         * referenced. At any time, the editor can have as many as
                    488:         * three files locked, and with different lock statuses.
                    489:         */
                    490:        /*
                    491:         * if this is either the saved or alternate file or current file,
                    492:         * point to the appropriate descriptor and file lock status.
                    493:         */
                    494:        if (strcmp (file,savedfile) == 0) {
                    495:                if (!io_savedfile) io_savedfile = dup(io) ;
                    496:                lp = &lock_savedfile ;  iop = &io_savedfile ;
                    497:        } else if (strcmp (file,altfile) == 0) {
                    498:                if (!io_altfile) io_altfile = dup(io) ;
                    499:                lp = &lock_altfile ;    iop = &io_altfile ;
                    500:        } else {
                    501:                /* throw away current lock, accquire new current lock */
                    502:                if (io_curr) close (io_curr) ;
                    503:                io_curr = dup(io) ;
                    504:                lp = &lock_curr ;       iop = &io_curr ;
                    505:                lock_curr = 0 ;
                    506:        }
                    507:        if (c == 'r' || value(READONLY) || *lp == 0) {
                    508:                /* if we have a lock already, don't bother */
                    509:                if (!*lp) {
                    510:                        /* try for a shared lock */
                    511:                        if (flock(*iop, LOCK_SH|LOCK_NB) < 0
                    512:                        && errno == EWOULDBLOCK) {
                    513:                                ex_printf (
                    514:                        " [FILE BEING MODIFIED BY ANOTHER PROCESS]") ;
                    515:                                flush();
                    516:                                goto fail_lock ;
                    517:                        } else *lp = LOCK_SH ;
                    518:                }
                    519:        }
                    520:        if ( c != 'r'  && !value(READONLY) && *lp != LOCK_EX) {
                    521:                /* if we are editting the file, upgrade to an exclusive lock. */
                    522:                if (flock(*iop, LOCK_EX|LOCK_NB) < 0 && errno == EWOULDBLOCK) {
                    523:                        ex_printf (" [File open by another process]") ;
                    524:                        flush();
                    525:                } else *lp = LOCK_EX ;
                    526:        }
                    527: fail_lock:
                    528: #endif FLOCKFILE
                    529:        if (c == 'r')
                    530:                setdot();
                    531:        else
                    532:                setall();
                    533:        if (FIXUNDO && inopen && c == 'r')
                    534:                undap1 = undap2 = dot + 1;
                    535:        rop2();
                    536:        rop3(c);
                    537: }
                    538: 
                    539: rop2()
                    540: {
                    541:        line *first, *last, *a;
                    542:        struct stat statb;
                    543: 
                    544:        deletenone();
                    545:        clrstats();
                    546:        first = addr2 + 1;
                    547:        if (fstat(io, &statb) < 0)
                    548:                bsize = LBSIZE;
                    549:        else {
                    550:                bsize = statb.st_blksize;
                    551:                if (bsize <= 0)
                    552:                        bsize = LBSIZE;
                    553:        }
                    554:        ignore(append(getfile, addr2));
                    555:        last = dot;
                    556:        /*
                    557:         *      if the modeline variable is set,
                    558:         *      check the first and last five lines of the file
                    559:         *      for a mode line.
                    560:         */
                    561:        if (value(MODELINE)) {
                    562:                for (a=first; a<=last; a++) {
                    563:                        if (a==first+5 && last-first > 10)
                    564:                                a = last - 4;
                    565:                        getline(*a);
                    566:                        checkmodeline(linebuf);
                    567:                }
                    568:        }
                    569: }
                    570: 
                    571: rop3(c)
                    572:        int c;
                    573: {
                    574: 
                    575:        if (iostats() == 0 && c == 'e')
                    576:                edited++;
                    577:        if (c == 'e') {
                    578:                if (wasalt || firstpat) {
                    579:                        register line *addr = zero + oldadot;
                    580: 
                    581:                        if (addr > dol)
                    582:                                addr = dol;
                    583:                        if (firstpat) {
                    584:                                globp = (*firstpat) ? firstpat : "$";
                    585:                                commands(1,1);
                    586:                                firstpat = 0;
                    587:                        } else if (addr >= one) {
                    588:                                if (inopen)
                    589:                                        dot = addr;
                    590:                                markpr(addr);
                    591:                        } else
                    592:                                goto other;
                    593:                } else
                    594: other:
                    595:                        if (dol > zero) {
                    596:                                if (inopen)
                    597:                                        dot = one;
                    598:                                markpr(one);
                    599:                        }
                    600:                if(FIXUNDO)
                    601:                        undkind = UNDNONE;
                    602:                if (inopen) {
                    603:                        vcline = 0;
                    604:                        vreplace(0, LINES, lineDOL());
                    605:                }
                    606:        }
                    607: }
                    608: 
                    609: /*
                    610:  * Are these two really the same inode?
                    611:  */
                    612: samei(sp, cp)
                    613:        struct stat *sp;
                    614:        char *cp;
                    615: {
                    616:        struct stat stb;
                    617: 
                    618:        if (stat(cp, &stb) < 0 || sp->st_dev != stb.st_dev)
                    619:                return (0);
                    620:        return (sp->st_ino == stb.st_ino);
                    621: }
                    622: 
                    623: /* Returns from edited() */
                    624: #define        EDF     0               /* Edited file */
                    625: #define        NOTEDF  -1              /* Not edited file */
                    626: #define        PARTBUF 1               /* Write of partial buffer to Edited file */
                    627: 
                    628: /*
                    629:  * Write a file.
                    630:  */
                    631: wop(dofname)
                    632: bool dofname;  /* if 1 call filename, else use savedfile */
                    633: {
                    634:        register int c, exclam, nonexist;
                    635:        line *saddr1, *saddr2;
                    636:        struct stat stbuf;
                    637: #ifdef FLOCKFILE
                    638:        int *lp, *iop ;
                    639: #endif FLOCKFILE
                    640: 
                    641:        c = 0;
                    642:        exclam = 0;
                    643:        if (dofname) {
                    644:                if (peekchar() == '!')
                    645:                        exclam++, ignchar();
                    646:                ignore(skipwh());
                    647:                while (peekchar() == '>')
                    648:                        ignchar(), c++, ignore(skipwh());
                    649:                if (c != 0 && c != 2)
                    650:                        error("Write forms are 'w' and 'w>>'");
                    651:                filename('w');
                    652:        } else {
                    653:                if (savedfile[0] == 0)
                    654:                        error("No file|No current filename");
                    655:                saddr1=addr1;
                    656:                saddr2=addr2;
                    657:                addr1=one;
                    658:                addr2=dol;
                    659:                CP(file, savedfile);
                    660:                if (inopen) {
                    661:                        vclrech(0);
                    662:                        splitw++;
                    663:                }
                    664:                lprintf("\"%s\"", file);
                    665:        }
                    666:        nonexist = stat(file, &stbuf);
                    667: #ifdef FLOCKFILE
                    668:        /*
                    669:         * if this is either the saved or alternate file or current file,
                    670:         * point to the appropriate descriptor and file lock status.
                    671:         */
                    672:        if (strcmp (file,savedfile) == 0) {
                    673:                lp = &lock_savedfile ;  iop = &io_savedfile ;
                    674:        } else if (strcmp (file,altfile) == 0) {
                    675:                lp = &lock_altfile ;    iop = &io_altfile ;
                    676:        } else {
                    677:                lp = &lock_curr ;       iop = &io_curr ;
                    678:        }
                    679:        if (!*iop && !nonexist){
                    680:                *lp = 0 ;
                    681:                if ((*iop = open(file, 1)) < 0) *iop = 0 ;
                    682:        }
                    683: #endif FLOCKFILE
                    684:        switch (c) {
                    685: 
                    686:        case 0:
                    687:                if (!exclam && (!value(WRITEANY) || value(READONLY)))
                    688:                switch (edfile()) {
                    689:                
                    690:                case NOTEDF:
                    691:                        if (nonexist)
                    692:                                break;
                    693:                        if ((stbuf.st_mode & S_IFMT) == S_IFCHR) {
                    694:                                if (samei(&stbuf, _PATH_DEVNULL))
                    695:                                        break;
                    696:                                if (samei(&stbuf, _PATH_TTY))
                    697:                                        break;
                    698:                        }
                    699:                        io = open(file, 1);
                    700:                        if (io < 0)
                    701:                                syserror();
                    702:                        if (!isatty(io))
                    703:                                serror(" File exists| File exists - use \"w! %s\" to overwrite", file);
                    704:                        close(io);
                    705:                        break;
                    706: 
                    707:                case EDF:
                    708:                        if (value(READONLY))
                    709:                                error(" File is read only");
                    710:                        break;
                    711: 
                    712:                case PARTBUF:
                    713:                        if (value(READONLY))
                    714:                                error(" File is read only");
                    715:                        error(" Use \"w!\" to write partial buffer");
                    716:                }
                    717: cre:
                    718: /*
                    719:                synctmp();
                    720: */
                    721: #ifdef FLOCKFILE
                    722:        if (*iop && !*lp != LOCK_EX && !exclam) {
                    723:                /*
                    724:                 * upgrade to a exclusive lock. if can't get, someone else 
                    725:                 * has the exclusive lock. bitch to the user.
                    726:                 */
                    727:                if (flock(*iop, LOCK_EX|LOCK_NB) < 0 && errno == EWOULDBLOCK)
                    728:        error (" File being modified by another process - use \"w!\" to write");
                    729:                 else *lp = LOCK_EX ;
                    730:        }
                    731: #endif FLOCKFILE
                    732: #ifdef V6
                    733:                io = creat(file, 0644);
                    734: #else
                    735:                io = creat(file, 0666);
                    736: #ifdef vms     /* to retain file protection modes on newer version of file */
                    737:                if (!nonexist)
                    738:                        chmod(file, stbuf.st_mode & 0777);
                    739: #endif
                    740: #endif
                    741:                if (io < 0)
                    742:                        syserror();
                    743:                writing = 1;
                    744:                if (hush == 0)
                    745:                        if (nonexist)
                    746:                                ex_printf(" [New file]");
                    747:                        else if (value(WRITEANY) && edfile() != EDF)
                    748:                                ex_printf(" [Existing file]");
                    749: #ifdef FLOCKFILE
                    750:                if (!*iop)
                    751:                        *iop = dup(io) ;
                    752: #endif FLOCKFILE
                    753:                break;
                    754: 
                    755:        case 2:
                    756:                io = open(file, 1);
                    757:                if (io < 0) {
                    758:                        if (exclam || value(WRITEANY))
                    759:                                goto cre;
                    760:                        syserror();
                    761:                }
                    762:                lseek(io, 0l, 2);
                    763: #ifdef FLOCKFILE
                    764:                if (!*iop) *iop = dup(io) ;
                    765:                if (*lp != LOCK_EX && !exclam) {
                    766:                        /*
                    767:                         * upgrade to a exclusive lock. if can't get,
                    768:                         * someone else has the exclusive lock.
                    769:                         * bitch to the user.
                    770:                         */
                    771:                        if (flock(*iop, LOCK_SH|LOCK_NB) < 0
                    772:                        && errno == EWOULDBLOCK)
                    773:                                error (
                    774: " File being modified by another process - use \"w!>>\" to write");
                    775:                        else *lp = LOCK_EX ;
                    776:                }
                    777: #endif FLOCKFILE
                    778:                break;
                    779:        }
                    780: #ifdef FLOCKFILE
                    781:        if (flock(*iop, LOCK_EX|LOCK_NB) >= 0)
                    782:                *lp = LOCK_EX ;
                    783: #endif FLOCKFILE
                    784:        putfile(0);
                    785: #ifndef        vms
                    786:        (void) fsync(io);
                    787: #endif
                    788:        ignore(iostats());
                    789:        if (c != 2 && addr1 == one && addr2 == dol) {
                    790:                if (eq(file, savedfile))
                    791:                        edited = 1;
                    792:                ex_sync();
                    793:        }
                    794:        if (!dofname) {
                    795:                addr1 = saddr1;
                    796:                addr2 = saddr2;
                    797:        }
                    798:        writing = 0;
                    799: }
                    800: 
                    801: /*
                    802:  * Is file the edited file?
                    803:  * Work here is that it is not considered edited
                    804:  * if this is a partial buffer, and distinguish
                    805:  * all cases.
                    806:  */
                    807: edfile()
                    808: {
                    809: 
                    810:        if (!edited || !eq(file, savedfile))
                    811:                return (NOTEDF);
                    812:        return (addr1 == one && addr2 == dol ? EDF : PARTBUF);
                    813: }
                    814: 
                    815: /*
                    816:  * Extract the next line from the io stream.
                    817:  */
                    818: char *nextip;
                    819: 
                    820: getfile()
                    821: {
                    822:        register short c;
                    823:        register char *lp, *fp;
                    824: 
                    825:        lp = linebuf;
                    826:        fp = nextip;
                    827:        do {
                    828:                if (--ninbuf < 0) {
                    829:                        ninbuf = read(io, genbuf, (int) bsize) - 1;
                    830:                        if (ninbuf < 0) {
                    831:                                if (lp != linebuf) {
                    832:                                        lp++;
                    833:                                        ex_printf(" [Incomplete last line]");
                    834:                                        break;
                    835:                                }
                    836:                                return (EOF);
                    837:                        }
                    838: #ifdef CRYPT
                    839:                        if (kflag) {
                    840:                                fp = genbuf;
                    841:                                while(fp < &genbuf[ninbuf]) {
                    842:                                        if (*fp++ & 0200) {
                    843:                                                crblock(perm, genbuf, ninbuf+1,
                    844:        cntch);
                    845:                                                break;
                    846:                                        }
                    847:                                }
                    848:                        }
                    849: #endif
                    850:                        fp = genbuf;
                    851:                        cntch += ninbuf+1;
                    852:                }
                    853:                if (lp >= &linebuf[LBSIZE]) {
                    854:                        error(" Line too long");
                    855:                }
                    856:                c = *fp++;
                    857:                if (c == 0) {
                    858:                        cntnull++;
                    859:                        continue;
                    860:                }
                    861:                if (c & QUOTE) {
                    862:                        cntodd++;
                    863:                        c &= TRIM;
                    864:                        if (c == 0)
                    865:                                continue;
                    866:                }
                    867:                *lp++ = c;
                    868:        } while (c != '\n');
                    869:        *--lp = 0;
                    870:        nextip = fp;
                    871:        cntln++;
                    872:        return (0);
                    873: }
                    874: 
                    875: /*
                    876:  * Write a range onto the io stream.
                    877:  */
                    878: /* ARGSUSED */
                    879: putfile(isfilter)
                    880: int isfilter;
                    881: {
                    882:        line *a1;
                    883:        register char *fp, *lp;
                    884:        register int nib;
                    885:        struct stat statb;
                    886: 
                    887:        a1 = addr1;
                    888:        clrstats();
                    889:        cntln = addr2 - a1 + 1;
                    890:        if (cntln == 0)
                    891:                return;
                    892:        if (fstat(io, &statb) < 0)
                    893:                bsize = LBSIZE;
                    894:        else {
                    895:                bsize = statb.st_blksize;
                    896:                if (bsize <= 0)
                    897:                        bsize = LBSIZE;
                    898:        }
                    899:        nib = bsize;
                    900:        fp = genbuf;
                    901:        do {
                    902:                getline(*a1++);
                    903:                lp = linebuf;
                    904:                for (;;) {
                    905:                        if (--nib < 0) {
                    906:                                nib = fp - genbuf;
                    907: #ifdef CRYPT
                    908:                                if(kflag && !isfilter)
                    909:                                         crblock(perm, genbuf, nib, cntch);
                    910: #endif
                    911:                                if (write(io, genbuf, nib) != nib) {
                    912:                                        wrerror();
                    913:                                }
                    914:                                cntch += nib;
                    915:                                nib = bsize - 1;
                    916:                                fp = genbuf;
                    917:                        }
                    918:                        if ((*fp++ = *lp++) == 0) {
                    919:                                fp[-1] = '\n';
                    920:                                break;
                    921:                        }
                    922:                }
                    923:        } while (a1 <= addr2);
                    924:        nib = fp - genbuf;
                    925: #ifdef CRYPT
                    926:        if(kflag && !isfilter)
                    927:                crblock(perm, genbuf, nib, cntch);
                    928: #endif
                    929:        if (write(io, genbuf, nib) != nib) {
                    930:                wrerror();
                    931:        }
                    932:        cntch += nib;
                    933: }
                    934: 
                    935: /*
                    936:  * A write error has occurred;  if the file being written was
                    937:  * the edited file then we consider it to have changed since it is
                    938:  * now likely scrambled.
                    939:  */
                    940: wrerror()
                    941: {
                    942: 
                    943:        if (eq(file, savedfile) && edited)
                    944:                change();
                    945:        syserror();
                    946: }
                    947: 
                    948: /*
                    949:  * Source command, handles nested sources.
                    950:  * Traps errors since it mungs unit 0 during the source.
                    951:  */
                    952: short slevel;
                    953: short ttyindes;
                    954: 
                    955: source(fil, okfail)
                    956:        char *fil;
                    957:        bool okfail;
                    958: {
                    959:        jmp_buf osetexit;
                    960:        register int saveinp, ointty, oerrno;
                    961:        char *saveglobp;
                    962:        short savepeekc;
                    963: 
                    964:        signal(SIGINT, SIG_IGN);
                    965:        saveinp = dup(0);
                    966:        savepeekc = peekc;
                    967:        saveglobp = globp;
                    968:        peekc = 0; globp = 0;
                    969:        if (saveinp < 0)
                    970:                error("Too many nested sources");
                    971:        if (slevel <= 0)
                    972:                ttyindes = saveinp;
                    973:        close(0);
                    974:        if (open(fil, 0) < 0) {
                    975:                oerrno = errno;
                    976:                setrupt();
                    977:                dup(saveinp);
                    978:                close(saveinp);
                    979:                errno = oerrno;
                    980:                if (!okfail)
                    981:                        filioerr(fil);
                    982:                return;
                    983:        }
                    984:        slevel++;
                    985:        ointty = intty;
                    986:        intty = isatty(0);
                    987:        oprompt = value(PROMPT);
                    988:        value(PROMPT) &= intty;
                    989:        getexit(osetexit);
                    990:        setrupt();
                    991:        if (setexit() == 0)
                    992:                commands(1, 1);
                    993:        else if (slevel > 1) {
                    994:                close(0);
                    995:                dup(saveinp);
                    996:                close(saveinp);
                    997:                slevel--;
                    998:                resexit(osetexit);
                    999:                reset();
                   1000:        }
                   1001:        intty = ointty;
                   1002:        value(PROMPT) = oprompt;
                   1003:        close(0);
                   1004:        dup(saveinp);
                   1005:        close(saveinp);
                   1006:        globp = saveglobp;
                   1007:        peekc = savepeekc;
                   1008:        slevel--;
                   1009:        resexit(osetexit);
                   1010: }
                   1011: 
                   1012: /*
                   1013:  * Clear io statistics before a read or write.
                   1014:  */
                   1015: clrstats()
                   1016: {
                   1017: 
                   1018:        ninbuf = 0;
                   1019:        cntch = 0;
                   1020:        cntln = 0;
                   1021:        cntnull = 0;
                   1022:        cntodd = 0;
                   1023: }
                   1024: 
                   1025: /*
                   1026:  * Io is finished, close the unit and print statistics.
                   1027:  */
                   1028: iostats()
                   1029: {
                   1030: 
                   1031:        close(io);
                   1032:        io = -1;
                   1033:        if (hush == 0) {
                   1034:                if (value(TERSE))
                   1035:                        ex_printf(" %d/%D", cntln, cntch);
                   1036:                else
                   1037:                        ex_printf(" %d line%s, %D character%s", cntln, plural((long) cntln),
                   1038:                            cntch, plural(cntch));
                   1039:                if (cntnull || cntodd) {
                   1040:                        ex_printf(" (");
                   1041:                        if (cntnull) {
                   1042:                                ex_printf("%D null", cntnull);
                   1043:                                if (cntodd)
                   1044:                                        ex_printf(", ");
                   1045:                        }
                   1046:                        if (cntodd)
                   1047:                                ex_printf("%D non-ASCII", cntodd);
                   1048:                        ex_putchar(')');
                   1049:                }
                   1050:                noonl();
                   1051:                flush();
                   1052:        }
                   1053:        return (cntnull != 0 || cntodd != 0);
                   1054: }
                   1055: 
                   1056: #ifdef USG
                   1057: # define index strchr
                   1058: # define rindex strrchr
                   1059: #endif
                   1060: #ifdef USG3TTY
                   1061: # define index strchr
                   1062: # define rindex strrchr
                   1063: #endif
                   1064: #ifdef vms
                   1065: # define index strchr
                   1066: # define rindex strrchr
                   1067: #endif
                   1068: 
                   1069: checkmodeline(l)
                   1070: char *l;
                   1071: {
                   1072:        char *beg, *end;
                   1073:        char cmdbuf[1024];
                   1074:        char *index(), *rindex(), *strncpy();
                   1075: 
                   1076:        beg = index(l, ':');
                   1077:        if (beg == NULL)
                   1078:                return;
                   1079:        if (&beg[-3] < l)
                   1080:                return;
                   1081:        if (!(  ( (beg[-3] == ' ' || beg[-3] == '\t')
                   1082:                && beg[-2] == 'e'
                   1083:                && beg[-1] == 'x')
                   1084:             || ( (beg[-3] == ' ' || beg[-3] == '\t')
                   1085:                && beg[-2] == 'v'
                   1086:                && beg[-1] == 'i'))) return;
                   1087:        strncpy(cmdbuf, beg+1, sizeof cmdbuf);
                   1088:        end = rindex(cmdbuf, ':');
                   1089:        if (end == NULL)
                   1090:                return;
                   1091:        *end = 0;
                   1092:        globp = cmdbuf;
                   1093:        commands(1, 1);
                   1094: }

unix.superglobalmegacorp.com

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