Annotation of 42BSD/ucb/more/more.c, revision 1.1

1.1     ! root        1: #ifndef lint
        !             2: static char *sccsid = "@(#)more.c      4.16 (Berkeley) 83/08/26";
        !             3: #endif
        !             4: 
        !             5: /*
        !             6: ** more.c - General purpose tty output filter and file perusal program
        !             7: **
        !             8: **     by Eric Shienbrood, UC Berkeley
        !             9: **
        !            10: **     modified by Geoff Peck, UCB to add underlining, single spacing
        !            11: **     modified by John Foderaro, UCB to add -c and MORE environment variable
        !            12: */
        !            13: 
        !            14: #include <stdio.h>
        !            15: #include <ctype.h>
        !            16: #include <signal.h>
        !            17: #include <errno.h>
        !            18: #include <sgtty.h>
        !            19: #include <setjmp.h>
        !            20: #include <sys/types.h>
        !            21: #include <sys/stat.h>
        !            22: 
        !            23: #define HELPFILE       "/usr/lib/more.help"
        !            24: #define VI             "/usr/ucb/vi"
        !            25: 
        !            26: #define Fopen(s,m)     (Currline = 0,file_pos=0,fopen(s,m))
        !            27: #define Ftell(f)       file_pos
        !            28: #define Fseek(f,off)   (file_pos=off,fseek(f,off,0))
        !            29: #define Getc(f)                (++file_pos, getc(f))
        !            30: #define Ungetc(c,f)    (--file_pos, ungetc(c,f))
        !            31: 
        !            32: #define MBIT   CBREAK
        !            33: #define stty(fd,argp)  ioctl(fd,TIOCSETN,argp)
        !            34: 
        !            35: #define TBUFSIZ        1024
        !            36: #define LINSIZ 256
        !            37: #define ctrl(letter)   ('letter' & 077)
        !            38: #define RUBOUT '\177'
        !            39: #define ESC    '\033'
        !            40: #define QUIT   '\034'
        !            41: 
        !            42: struct sgttyb  otty, savetty;
        !            43: long           file_pos, file_size;
        !            44: int            fnum, no_intty, no_tty, slow_tty;
        !            45: int            dum_opt, dlines, onquit(), end_it();
        !            46: int            onsusp();
        !            47: int            nscroll = 11;   /* Number of lines scrolled by 'd' */
        !            48: int            fold_opt = 1;   /* Fold long lines */
        !            49: int            stop_opt = 1;   /* Stop after form feeds */
        !            50: int            ssp_opt = 0;    /* Suppress white space */
        !            51: int            ul_opt = 1;     /* Underline as best we can */
        !            52: int            promptlen;
        !            53: int            Currline;       /* Line we are currently at */
        !            54: int            startup = 1;
        !            55: int            firstf = 1;
        !            56: int            notell = 1;
        !            57: int            bad_so; /* True if overwriting does not turn off standout */
        !            58: int            inwait, Pause, errors;
        !            59: int            within; /* true if we are within a file,
        !            60:                        false if we are between files */
        !            61: int            hard, dumb, noscroll, hardtabs, clreol;
        !            62: int            catch_susp;     /* We should catch the SIGTSTP signal */
        !            63: char           **fnames;       /* The list of file names */
        !            64: int            nfiles;         /* Number of files left to process */
        !            65: char           *shell;         /* The name of the shell to use */
        !            66: int            shellp;         /* A previous shell command exists */
        !            67: char           ch;
        !            68: jmp_buf                restore;
        !            69: char           obuf[BUFSIZ];   /* stdout buffer */
        !            70: char           Line[LINSIZ];   /* Line buffer */
        !            71: int            Lpp = 24;       /* lines per page */
        !            72: char           *Clear;         /* clear screen */
        !            73: char           *eraseln;       /* erase line */
        !            74: char           *Senter, *Sexit;/* enter and exit standout mode */
        !            75: char           *ULenter, *ULexit;      /* enter and exit underline mode */
        !            76: char           *chUL;          /* underline character */
        !            77: char           *chBS;          /* backspace character */
        !            78: char           *Home;          /* go to home */
        !            79: char           *cursorm;       /* cursor movement */
        !            80: char           cursorhome[40]; /* contains cursor movement to home */
        !            81: char           *EodClr;        /* clear rest of screen */
        !            82: char           *tgetstr();
        !            83: int            Mcol = 80;      /* number of columns */
        !            84: int            Wrap = 1;       /* set if automargins */
        !            85: long           fseek();
        !            86: char           *getenv();
        !            87: struct {
        !            88:     long chrctr, line;
        !            89: } context, screen_start;
        !            90: extern char    PC;             /* pad character */
        !            91: extern short   ospeed;
        !            92: 
        !            93: 
        !            94: main(argc, argv)
        !            95: int argc;
        !            96: char *argv[];
        !            97: {
        !            98:     register FILE      *f;
        !            99:     register char      *s;
        !           100:     register char      *p;
        !           101:     register char      ch;
        !           102:     register int       left;
        !           103:     int                        prnames = 0; 
        !           104:     int                        initopt = 0;
        !           105:     int                        srchopt = 0;
        !           106:     int                        clearit = 0;
        !           107:     int                        initline;
        !           108:     char               initbuf[80];
        !           109:     FILE               *checkf();
        !           110: 
        !           111:     nfiles = argc;
        !           112:     fnames = argv;
        !           113:     initterm ();
        !           114:     if(s = getenv("MORE")) argscan(s);
        !           115:     while (--nfiles > 0) {
        !           116:        if ((ch = (*++fnames)[0]) == '-') {
        !           117:            argscan(*fnames+1);
        !           118:        }
        !           119:        else if (ch == '+') {
        !           120:            s = *fnames;
        !           121:            if (*++s == '/') {
        !           122:                srchopt++;
        !           123:                for (++s, p = initbuf; p < initbuf + 79 && *s != '\0';)
        !           124:                    *p++ = *s++;
        !           125:                *p = '\0';
        !           126:            }
        !           127:            else {
        !           128:                initopt++;
        !           129:                for (initline = 0; *s != '\0'; s++)
        !           130:                    if (isdigit (*s))
        !           131:                        initline = initline*10 + *s -'0';
        !           132:                --initline;
        !           133:            }
        !           134:        }
        !           135:        else break;
        !           136:     }
        !           137:     /* allow clreol only if Home and eraseln and EodClr strings are
        !           138:      *  defined, and in that case, make sure we are in noscroll mode
        !           139:      */
        !           140:     if(clreol)
        !           141:     {
        !           142:        if ((*Home == '\0') || (*eraseln == '\0') || (*EodClr == '\0'))
        !           143:            clreol = 0;
        !           144:        else noscroll = 1;
        !           145:     }
        !           146: 
        !           147:     if (dlines == 0)
        !           148:        dlines = Lpp - (noscroll ? 1 : 2);
        !           149:     left = dlines;
        !           150:     if (nfiles > 1)
        !           151:        prnames++;
        !           152:     if (!no_intty && nfiles == 0) {
        !           153:        fputs("Usage: ",stderr);
        !           154:        fputs(argv[0],stderr);
        !           155:        fputs(" [-dfln] [+linenum | +/pattern] name1 name2 ...\n",stderr);
        !           156:        exit(1);
        !           157:     }
        !           158:     else
        !           159:        f = stdin;
        !           160:     if (!no_tty) {
        !           161:        signal(SIGQUIT, onquit);
        !           162:        signal(SIGINT, end_it);
        !           163:        if (signal (SIGTSTP, SIG_IGN) == SIG_DFL) {
        !           164:            signal(SIGTSTP, onsusp);
        !           165:            catch_susp++;
        !           166:        }
        !           167:        stty (2, &otty);
        !           168:     }
        !           169:     if (no_intty) {
        !           170:        if (no_tty)
        !           171:            copy_file (stdin);
        !           172:        else {
        !           173:            if ((ch = Getc (f)) == '\f')
        !           174:                doclear();
        !           175:            else {
        !           176:                Ungetc (ch, f);
        !           177:                if (noscroll && (ch != EOF)) {
        !           178:                    if (clreol)
        !           179:                        home ();
        !           180:                    else
        !           181:                        doclear ();
        !           182:                }
        !           183:            }
        !           184:            if (srchopt)
        !           185:            {
        !           186:                search (initbuf, stdin, 1);
        !           187:                if (noscroll)
        !           188:                    left--;
        !           189:            }
        !           190:            else if (initopt)
        !           191:                skiplns (initline, stdin);
        !           192:            screen (stdin, left);
        !           193:        }
        !           194:        no_intty = 0;
        !           195:        prnames++;
        !           196:        firstf = 0;
        !           197:     }
        !           198: 
        !           199:     while (fnum < nfiles) {
        !           200:        if ((f = checkf (fnames[fnum], &clearit)) != NULL) {
        !           201:            context.line = context.chrctr = 0;
        !           202:            Currline = 0;
        !           203:            if (firstf) setjmp (restore);
        !           204:            if (firstf) {
        !           205:                firstf = 0;
        !           206:                if (srchopt)
        !           207:                {
        !           208:                    search (initbuf, f, 1);
        !           209:                    if (noscroll)
        !           210:                        left--;
        !           211:                }
        !           212:                else if (initopt)
        !           213:                    skiplns (initline, f);
        !           214:            }
        !           215:            else if (fnum < nfiles && !no_tty) {
        !           216:                setjmp (restore);
        !           217:                left = command (fnames[fnum], f);
        !           218:            }
        !           219:            if (left != 0) {
        !           220:                if ((noscroll || clearit) && (file_size != 0x7fffffffffffffffL))
        !           221:                    if (clreol)
        !           222:                        home ();
        !           223:                    else
        !           224:                        doclear ();
        !           225:                if (prnames) {
        !           226:                    if (bad_so)
        !           227:                        erase (0);
        !           228:                    if (clreol)
        !           229:                        cleareol ();
        !           230:                    pr("::::::::::::::");
        !           231:                    if (promptlen > 14)
        !           232:                        erase (14);
        !           233:                    printf ("\n");
        !           234:                    if(clreol) cleareol();
        !           235:                    printf("%s\n", fnames[fnum]);
        !           236:                    if(clreol) cleareol();
        !           237:                    printf("::::::::::::::\n", fnames[fnum]);
        !           238:                    if (left > Lpp - 4)
        !           239:                        left = Lpp - 4;
        !           240:                }
        !           241:                if (no_tty)
        !           242:                    copy_file (f);
        !           243:                else {
        !           244:                    within++;
        !           245:                    screen(f, left);
        !           246:                    within = 0;
        !           247:                }
        !           248:            }
        !           249:            setjmp (restore);
        !           250:            fflush(stdout);
        !           251:            fclose(f);
        !           252:            screen_start.line = screen_start.chrctr = 0L;
        !           253:            context.line = context.chrctr = 0L;
        !           254:        }
        !           255:        fnum++;
        !           256:        firstf = 0;
        !           257:     }
        !           258:     reset_tty ();
        !           259:     exit(0);
        !           260: }
        !           261: 
        !           262: argscan(s)
        !           263: char *s;
        !           264: {
        !           265:        for (dlines = 0; *s != '\0'; s++)
        !           266:        {
        !           267:                switch (*s)
        !           268:                {
        !           269:                  case '0': case '1': case '2':
        !           270:                  case '3': case '4': case '5':
        !           271:                  case '6': case '7': case '8':
        !           272:                  case '9':
        !           273:                        dlines = dlines*10 + *s - '0';
        !           274:                        break;
        !           275:                  case 'd':
        !           276:                        dum_opt = 1;
        !           277:                        break;
        !           278:                  case 'l':
        !           279:                        stop_opt = 0;
        !           280:                        break;
        !           281:                  case 'f':
        !           282:                        fold_opt = 0;
        !           283:                        break;
        !           284:                  case 'p':
        !           285:                        noscroll++;
        !           286:                        break;
        !           287:                  case 'c':
        !           288:                        clreol++;
        !           289:                        break;
        !           290:                  case 's':
        !           291:                        ssp_opt = 1;
        !           292:                        break;
        !           293:                  case 'u':
        !           294:                        ul_opt = 0;
        !           295:                        break;
        !           296:                }
        !           297:        }
        !           298: }
        !           299: 
        !           300: 
        !           301: /*
        !           302: ** Check whether the file named by fs is an ASCII file which the user may
        !           303: ** access.  If it is, return the opened file. Otherwise return NULL.
        !           304: */
        !           305: 
        !           306: FILE *
        !           307: checkf (fs, clearfirst)
        !           308: register char *fs;
        !           309: int *clearfirst;
        !           310: {
        !           311:     struct stat stbuf;
        !           312:     register FILE *f;
        !           313:     char c;
        !           314: 
        !           315:     if (stat (fs, &stbuf) == -1) {
        !           316:        fflush(stdout);
        !           317:        if (clreol)
        !           318:            cleareol ();
        !           319:        perror(fs);
        !           320:        return (NULL);
        !           321:     }
        !           322:     if ((stbuf.st_mode & S_IFMT) == S_IFDIR) {
        !           323:        printf("\n*** %s: directory ***\n\n", fs);
        !           324:        return (NULL);
        !           325:     }
        !           326:     if ((f=Fopen(fs, "r")) == NULL) {
        !           327:        fflush(stdout);
        !           328:        perror(fs);
        !           329:        return (NULL);
        !           330:     }
        !           331:     c = Getc(f);
        !           332: 
        !           333:     /* Try to see whether it is an ASCII file */
        !           334: 
        !           335:     switch ((c | *f->_ptr << 8) & 0177777) {
        !           336:     case 0405:
        !           337:     case 0407:
        !           338:     case 0410:
        !           339:     case 0411:
        !           340:     case 0413:
        !           341:     case 0177545:
        !           342:        printf("\n******** %s: Not a text file ********\n\n", fs);
        !           343:        fclose (f);
        !           344:        return (NULL);
        !           345:     default:
        !           346:        break;
        !           347:     }
        !           348:     if (c == '\f')
        !           349:        *clearfirst = 1;
        !           350:     else {
        !           351:        *clearfirst = 0;
        !           352:        Ungetc (c, f);
        !           353:     }
        !           354:     if ((file_size = stbuf.st_size) == 0)
        !           355:        file_size = 0x7fffffffffffffffL;
        !           356:     return (f);
        !           357: }
        !           358: 
        !           359: /*
        !           360: ** A real function, for the tputs routine in termlib
        !           361: */
        !           362: 
        !           363: putch (ch)
        !           364: char ch;
        !           365: {
        !           366:     putchar (ch);
        !           367: }
        !           368: 
        !           369: /*
        !           370: ** Print out the contents of the file f, one screenful at a time.
        !           371: */
        !           372: 
        !           373: #define STOP -10
        !           374: 
        !           375: screen (f, num_lines)
        !           376: register FILE *f;
        !           377: register int num_lines;
        !           378: {
        !           379:     register int c;
        !           380:     register int nchars;
        !           381:     int length;                        /* length of current line */
        !           382:     static int prev_len = 1;   /* length of previous line */
        !           383: 
        !           384:     for (;;) {
        !           385:        while (num_lines > 0 && !Pause) {
        !           386:            if ((nchars = getline (f, &length)) == EOF)
        !           387:            {
        !           388:                if (clreol)
        !           389:                    clreos();
        !           390:                return;
        !           391:            }
        !           392:            if (ssp_opt && length == 0 && prev_len == 0)
        !           393:                continue;
        !           394:            prev_len = length;
        !           395:            if (bad_so || (Senter && *Senter == ' ') && promptlen > 0)
        !           396:                erase (0);
        !           397:            /* must clear before drawing line since tabs on some terminals
        !           398:             * do not erase what they tab over.
        !           399:             */
        !           400:            if (clreol)
        !           401:                cleareol ();
        !           402:            prbuf (Line, length);
        !           403:            if (nchars < promptlen)
        !           404:                erase (nchars); /* erase () sets promptlen to 0 */
        !           405:            else promptlen = 0;
        !           406:            /* is this needed?
        !           407:             * if (clreol)
        !           408:             *  cleareol();     /* must clear again in case we wrapped *
        !           409:             */
        !           410:            if (nchars < Mcol || !fold_opt)
        !           411:                putchar('\n');
        !           412:            if (nchars == STOP)
        !           413:                break;
        !           414:            num_lines--;
        !           415:        }
        !           416:        fflush(stdout);
        !           417:        if ((c = Getc(f)) == EOF)
        !           418:        {
        !           419:            if (clreol)
        !           420:                clreos ();
        !           421:            return;
        !           422:        }
        !           423: 
        !           424:        if (Pause && clreol)
        !           425:            clreos ();
        !           426:        Ungetc (c, f);
        !           427:        setjmp (restore);
        !           428:        Pause = 0; startup = 0;
        !           429:        if ((num_lines = command (NULL, f)) == 0)
        !           430:            return;
        !           431:        if (hard && promptlen > 0)
        !           432:                erase (0);
        !           433:        if (noscroll && num_lines >= dlines)
        !           434:        { 
        !           435:            if (clreol)
        !           436:                home();
        !           437:            else
        !           438:                doclear ();
        !           439:        }
        !           440:        screen_start.line = Currline;
        !           441:        screen_start.chrctr = Ftell (f);
        !           442:     }
        !           443: }
        !           444: 
        !           445: /*
        !           446: ** Come here if a quit signal is received
        !           447: */
        !           448: 
        !           449: onquit()
        !           450: {
        !           451:     signal(SIGQUIT, SIG_IGN);
        !           452:     if (!inwait) {
        !           453:        putchar ('\n');
        !           454:        if (!startup) {
        !           455:            signal(SIGQUIT, onquit);
        !           456:            longjmp (restore, 1);
        !           457:        }
        !           458:        else
        !           459:            Pause++;
        !           460:     }
        !           461:     else if (!dum_opt && notell) {
        !           462:        write (2, "[Use q or Q to quit]", 20);
        !           463:        promptlen += 20;
        !           464:        notell = 0;
        !           465:     }
        !           466:     signal(SIGQUIT, onquit);
        !           467: }
        !           468: 
        !           469: /*
        !           470: ** Clean up terminal state and exit. Also come here if interrupt signal received
        !           471: */
        !           472: 
        !           473: end_it ()
        !           474: {
        !           475: 
        !           476:     reset_tty ();
        !           477:     if (clreol) {
        !           478:        putchar ('\r');
        !           479:        clreos ();
        !           480:        fflush (stdout);
        !           481:     }
        !           482:     else if (!clreol && (promptlen > 0)) {
        !           483:        kill_line ();
        !           484:        fflush (stdout);
        !           485:     }
        !           486:     else
        !           487:        write (2, "\n", 1);
        !           488:     _exit(0);
        !           489: }
        !           490: 
        !           491: copy_file(f)
        !           492: register FILE *f;
        !           493: {
        !           494:     register int c;
        !           495: 
        !           496:     while ((c = getc(f)) != EOF)
        !           497:        putchar(c);
        !           498: }
        !           499: 
        !           500: /* Simplified printf function */
        !           501: 
        !           502: printf (fmt, args)
        !           503: register char *fmt;
        !           504: int args;
        !           505: {
        !           506:        register int *argp;
        !           507:        register char ch;
        !           508:        register int ccount;
        !           509: 
        !           510:        ccount = 0;
        !           511:        argp = &args;
        !           512:        while (*fmt) {
        !           513:                while ((ch = *fmt++) != '%') {
        !           514:                        if (ch == '\0')
        !           515:                                return (ccount);
        !           516:                        ccount++;
        !           517:                        putchar (ch);
        !           518:                }
        !           519:                switch (*fmt++) {
        !           520:                case 'd':
        !           521:                        ccount += printd (*argp);
        !           522:                        break;
        !           523:                case 's':
        !           524:                        ccount += pr ((char *)*argp);
        !           525:                        break;
        !           526:                case '%':
        !           527:                        ccount++;
        !           528:                        argp--;
        !           529:                        putchar ('%');
        !           530:                        break;
        !           531:                case '0':
        !           532:                        return (ccount);
        !           533:                default:
        !           534:                        break;
        !           535:                }
        !           536:                ++argp;
        !           537:        }
        !           538:        return (ccount);
        !           539: 
        !           540: }
        !           541: 
        !           542: /*
        !           543: ** Print an integer as a string of decimal digits,
        !           544: ** returning the length of the print representation.
        !           545: */
        !           546: 
        !           547: printd (n)
        !           548: int n;
        !           549: {
        !           550:     int a, nchars;
        !           551: 
        !           552:     if (a = n/10)
        !           553:        nchars = 1 + printd(a);
        !           554:     else
        !           555:        nchars = 1;
        !           556:     putchar (n % 10 + '0');
        !           557:     return (nchars);
        !           558: }
        !           559: 
        !           560: /* Put the print representation of an integer into a string */
        !           561: static char *sptr;
        !           562: 
        !           563: scanstr (n, str)
        !           564: int n;
        !           565: char *str;
        !           566: {
        !           567:     sptr = str;
        !           568:     Sprintf (n);
        !           569:     *sptr = '\0';
        !           570: }
        !           571: 
        !           572: Sprintf (n)
        !           573: {
        !           574:     int a;
        !           575: 
        !           576:     if (a = n/10)
        !           577:        Sprintf (a);
        !           578:     *sptr++ = n % 10 + '0';
        !           579: }
        !           580: 
        !           581: static char bell = ctrl(G);
        !           582: 
        !           583: strlen (s)
        !           584: char *s;
        !           585: {
        !           586:     register char *p;
        !           587: 
        !           588:     p = s;
        !           589:     while (*p++)
        !           590:        ;
        !           591:     return (p - s - 1);
        !           592: }
        !           593: 
        !           594: /* See whether the last component of the path name "path" is equal to the
        !           595: ** string "string"
        !           596: */
        !           597: 
        !           598: tailequ (path, string)
        !           599: char *path;
        !           600: register char *string;
        !           601: {
        !           602:        register char *tail;
        !           603: 
        !           604:        tail = path + strlen(path);
        !           605:        while (tail >= path)
        !           606:                if (*(--tail) == '/')
        !           607:                        break;
        !           608:        ++tail;
        !           609:        while (*tail++ == *string++)
        !           610:                if (*tail == '\0')
        !           611:                        return(1);
        !           612:        return(0);
        !           613: }
        !           614: 
        !           615: prompt (filename)
        !           616: char *filename;
        !           617: {
        !           618:     if (clreol)
        !           619:        cleareol ();
        !           620:     else if (promptlen > 0)
        !           621:        kill_line ();
        !           622:     if (!hard) {
        !           623:        promptlen = 8;
        !           624:        if (Senter && Sexit)
        !           625:            tputs (Senter, 1, putch);
        !           626:        if (clreol)
        !           627:            cleareol ();
        !           628:        pr("--More--");
        !           629:        if (filename != NULL) {
        !           630:            promptlen += printf ("(Next file: %s)", filename);
        !           631:        }
        !           632:        else if (!no_intty) {
        !           633:            promptlen += printf ("(%d%%)", (int)((file_pos * 100) / file_size));
        !           634:        }
        !           635:        if (dum_opt) {
        !           636:            promptlen += pr("[Hit space to continue, Rubout to abort]");
        !           637:        }
        !           638:        if (Senter && Sexit)
        !           639:            tputs (Sexit, 1, putch);
        !           640:        if (clreol)
        !           641:            clreos ();
        !           642:        fflush(stdout);
        !           643:     }
        !           644:     else
        !           645:        write (2, &bell, 1);
        !           646:     inwait++;
        !           647: }
        !           648: 
        !           649: /*
        !           650: ** Get a logical line
        !           651: */
        !           652: 
        !           653: getline(f, length)
        !           654: register FILE *f;
        !           655: int *length;
        !           656: {
        !           657:     register int       c;
        !           658:     register char      *p;
        !           659:     register int       column;
        !           660:     static int         colflg;
        !           661: 
        !           662:     p = Line;
        !           663:     column = 0;
        !           664:     c = Getc (f);
        !           665:     if (colflg && c == '\n') {
        !           666:        Currline++;
        !           667:        c = Getc (f);
        !           668:     }
        !           669:     while (p < &Line[LINSIZ - 1]) {
        !           670:        if (c == EOF) {
        !           671:            if (p > Line) {
        !           672:                *p = '\0';
        !           673:                *length = p - Line;
        !           674:                return (column);
        !           675:            }
        !           676:            *length = p - Line;
        !           677:            return (EOF);
        !           678:        }
        !           679:        if (c == '\n') {
        !           680:            Currline++;
        !           681:            break;
        !           682:        }
        !           683:        *p++ = c;
        !           684:        if (c == '\t')
        !           685:            if (hardtabs && column < promptlen && !hard) {
        !           686:                if (eraseln && !dumb) {
        !           687:                    column = 1 + (column | 7);
        !           688:                    tputs (eraseln, 1, putch);
        !           689:                    promptlen = 0;
        !           690:                }
        !           691:                else {
        !           692:                    for (--p; column & 7 && p < &Line[LINSIZ - 1]; column++) {
        !           693:                        *p++ = ' ';
        !           694:                    }
        !           695:                    if (column >= promptlen) promptlen = 0;
        !           696:                }
        !           697:            }
        !           698:            else
        !           699:                column = 1 + (column | 7);
        !           700:        else if (c == '\b' && column > 0)
        !           701:            column--;
        !           702:        else if (c == '\r')
        !           703:            column = 0;
        !           704:        else if (c == '\f' && stop_opt) {
        !           705:                p[-1] = '^';
        !           706:                *p++ = 'L';
        !           707:                column += 2;
        !           708:                Pause++;
        !           709:        }
        !           710:        else if (c == EOF) {
        !           711:            *length = p - Line;
        !           712:            return (column);
        !           713:        }
        !           714:        else if (c >= ' ' && c != RUBOUT)
        !           715:            column++;
        !           716:        if (column >= Mcol && fold_opt) break;
        !           717:        c = Getc (f);
        !           718:     }
        !           719:     if (column >= Mcol && Mcol > 0) {
        !           720:        if (!Wrap) {
        !           721:            *p++ = '\n';
        !           722:        }
        !           723:     }
        !           724:     colflg = column == Mcol && fold_opt;
        !           725:     *length = p - Line;
        !           726:     *p = 0;
        !           727:     return (column);
        !           728: }
        !           729: 
        !           730: /*
        !           731: ** Erase the rest of the prompt, assuming we are starting at column col.
        !           732: */
        !           733: 
        !           734: erase (col)
        !           735: register int col;
        !           736: {
        !           737: 
        !           738:     if (promptlen == 0)
        !           739:        return;
        !           740:     if (hard) {
        !           741:        putchar ('\n');
        !           742:     }
        !           743:     else {
        !           744:        if (col == 0)
        !           745:            putchar ('\r');
        !           746:        if (!dumb && eraseln)
        !           747:            tputs (eraseln, 1, putch);
        !           748:        else
        !           749:            for (col = promptlen - col; col > 0; col--)
        !           750:                putchar (' ');
        !           751:     }
        !           752:     promptlen = 0;
        !           753: }
        !           754: 
        !           755: /*
        !           756: ** Erase the current line entirely
        !           757: */
        !           758: 
        !           759: kill_line ()
        !           760: {
        !           761:     erase (0);
        !           762:     if (!eraseln || dumb) putchar ('\r');
        !           763: }
        !           764: 
        !           765: /*
        !           766:  * force clear to end of line
        !           767:  */
        !           768: cleareol()
        !           769: {
        !           770:     tputs(eraseln, 1, putch);
        !           771: }
        !           772: 
        !           773: clreos()
        !           774: {
        !           775:     tputs(EodClr, 1, putch);
        !           776: }
        !           777: 
        !           778: /*
        !           779: **  Print string and return number of characters
        !           780: */
        !           781: 
        !           782: pr(s1)
        !           783: char   *s1;
        !           784: {
        !           785:     register char      *s;
        !           786:     register char      c;
        !           787: 
        !           788:     for (s = s1; c = *s++; )
        !           789:        putchar(c);
        !           790:     return (s - s1 - 1);
        !           791: }
        !           792: 
        !           793: 
        !           794: /* Print a buffer of n characters */
        !           795: 
        !           796: prbuf (s, n)
        !           797: register char *s;
        !           798: register int n;
        !           799: {
        !           800:     char c;                            /* next ouput character */
        !           801:     register int state;                        /* next output char's UL state */
        !           802:     static int pstate = 0;             /* current terminal UL state (off) */
        !           803: 
        !           804:     while (--n >= 0)
        !           805:        if (!ul_opt)
        !           806:            putchar (*s++);
        !           807:        else {
        !           808:            if (n >= 2 && s[0] == '_' && s[1] == '\b') {
        !           809:                n -= 2;
        !           810:                s += 2;
        !           811:                c = *s++;
        !           812:                state = 1;
        !           813:            } else if (n >= 2 && s[1] == '\b' && s[2] == '_') {
        !           814:                n -= 2;
        !           815:                c = *s++;
        !           816:                s += 2;
        !           817:                state = 1;
        !           818:            } else {
        !           819:                c = *s++;
        !           820:                state = 0;
        !           821:            }
        !           822:            if (state != pstate)
        !           823:                tputs(state ? ULenter : ULexit, 1, putch);
        !           824:            pstate = state;
        !           825:            putchar(c);
        !           826:            if (state && *chUL) {
        !           827:                pr(chBS);
        !           828:                tputs(chUL, 1, putch);
        !           829:            }
        !           830:        }
        !           831: }
        !           832: 
        !           833: /*
        !           834: **  Clear the screen
        !           835: */
        !           836: 
        !           837: doclear()
        !           838: {
        !           839:     if (Clear && !hard) {
        !           840:        tputs(Clear, 1, putch);
        !           841: 
        !           842:        /* Put out carriage return so that system doesn't
        !           843:        ** get confused by escape sequences when expanding tabs
        !           844:        */
        !           845:        putchar ('\r');
        !           846:        promptlen = 0;
        !           847:     }
        !           848: }
        !           849: 
        !           850: /*
        !           851:  * Go to home position
        !           852:  */
        !           853: home()
        !           854: {
        !           855:     tputs(Home,1,putch);
        !           856: }
        !           857: 
        !           858: static int lastcmd, lastarg, lastp;
        !           859: static int lastcolon;
        !           860: char shell_line[132];
        !           861: 
        !           862: /*
        !           863: ** Read a command and do it. A command consists of an optional integer
        !           864: ** argument followed by the command character.  Return the number of lines
        !           865: ** to display in the next screenful.  If there is nothing more to display
        !           866: ** in the current file, zero is returned.
        !           867: */
        !           868: 
        !           869: command (filename, f)
        !           870: char *filename;
        !           871: register FILE *f;
        !           872: {
        !           873:     register int nlines;
        !           874:     register int retval;
        !           875:     register char c;
        !           876:     char colonch;
        !           877:     FILE *helpf;
        !           878:     int done;
        !           879:     char comchar, cmdbuf[80], *p;
        !           880: 
        !           881: #define ret(val) retval=val;done++;break
        !           882: 
        !           883:     done = 0;
        !           884:     if (!errors)
        !           885:        prompt (filename);
        !           886:     else
        !           887:        errors = 0;
        !           888:     if (MBIT == RAW && slow_tty) {
        !           889:        otty.sg_flags |= MBIT;
        !           890:        stty(2, &otty);
        !           891:     }
        !           892:     for (;;) {
        !           893:        nlines = number (&comchar);
        !           894:        lastp = colonch = 0;
        !           895:        if (comchar == '.') {   /* Repeat last command */
        !           896:                lastp++;
        !           897:                comchar = lastcmd;
        !           898:                nlines = lastarg;
        !           899:                if (lastcmd == ':')
        !           900:                        colonch = lastcolon;
        !           901:        }
        !           902:        lastcmd = comchar;
        !           903:        lastarg = nlines;
        !           904:        if (comchar == otty.sg_erase) {
        !           905:            kill_line ();
        !           906:            prompt (filename);
        !           907:            continue;
        !           908:        }
        !           909:        switch (comchar) {
        !           910:        case ':':
        !           911:            retval = colon (filename, colonch, nlines);
        !           912:            if (retval >= 0)
        !           913:                done++;
        !           914:            break;
        !           915:        case ' ':
        !           916:        case 'z':
        !           917:            if (nlines == 0) nlines = dlines;
        !           918:            else if (comchar == 'z') dlines = nlines;
        !           919:            ret (nlines);
        !           920:        case 'd':
        !           921:        case ctrl(D):
        !           922:            if (nlines != 0) nscroll = nlines;
        !           923:            ret (nscroll);
        !           924:        case RUBOUT:
        !           925:        case 'q':
        !           926:        case 'Q':
        !           927:            end_it ();
        !           928:        case 's':
        !           929:        case 'f':
        !           930:            if (nlines == 0) nlines++;
        !           931:            if (comchar == 'f')
        !           932:                nlines *= dlines;
        !           933:            putchar ('\r');
        !           934:            erase (0);
        !           935:            printf ("\n");
        !           936:            if (clreol)
        !           937:                cleareol ();
        !           938:            printf ("...skipping %d line", nlines);
        !           939:            if (nlines > 1)
        !           940:                pr ("s\n");
        !           941:            else
        !           942:                pr ("\n");
        !           943: 
        !           944:            if (clreol)
        !           945:                cleareol ();
        !           946:            pr ("\n");
        !           947: 
        !           948:            while (nlines > 0) {
        !           949:                while ((c = Getc (f)) != '\n')
        !           950:                    if (c == EOF) {
        !           951:                        retval = 0;
        !           952:                        done++;
        !           953:                        goto endsw;
        !           954:                    }
        !           955:                    Currline++;
        !           956:                    nlines--;
        !           957:            }
        !           958:            ret (dlines);
        !           959:        case '\n':
        !           960:            if (nlines != 0)
        !           961:                dlines = nlines;
        !           962:            else
        !           963:                nlines = 1;
        !           964:            ret (nlines);
        !           965:        case '\f':
        !           966:            if (!no_intty) {
        !           967:                doclear ();
        !           968:                Fseek (f, screen_start.chrctr);
        !           969:                Currline = screen_start.line;
        !           970:                ret (dlines);
        !           971:            }
        !           972:            else {
        !           973:                write (2, &bell, 1);
        !           974:                break;
        !           975:            }
        !           976:        case '\'':
        !           977:            if (!no_intty) {
        !           978:                kill_line ();
        !           979:                pr ("\n***Back***\n\n");
        !           980:                Fseek (f, context.chrctr);
        !           981:                Currline = context.line;
        !           982:                ret (dlines);
        !           983:            }
        !           984:            else {
        !           985:                write (2, &bell, 1);
        !           986:                break;
        !           987:            }
        !           988:        case '=':
        !           989:            kill_line ();
        !           990:            promptlen = printd (Currline);
        !           991:            fflush (stdout);
        !           992:            break;
        !           993:        case 'n':
        !           994:            lastp++;
        !           995:        case '/':
        !           996:            if (nlines == 0) nlines++;
        !           997:            kill_line ();
        !           998:            pr ("/");
        !           999:            promptlen = 1;
        !          1000:            fflush (stdout);
        !          1001:            if (lastp) {
        !          1002:                write (2,"\r", 1);
        !          1003:                search (NULL, f, nlines);       /* Use previous r.e. */
        !          1004:            }
        !          1005:            else {
        !          1006:                ttyin (cmdbuf, 78, '/');
        !          1007:                write (2, "\r", 1);
        !          1008:                search (cmdbuf, f, nlines);
        !          1009:            }
        !          1010:            ret (dlines-1);
        !          1011:        case '!':
        !          1012:            do_shell (filename);
        !          1013:            break;
        !          1014:        case 'h':
        !          1015:            if ((helpf = fopen (HELPFILE, "r")) == NULL)
        !          1016:                error ("Can't open help file");
        !          1017:            if (noscroll) doclear ();
        !          1018:            copy_file (helpf);
        !          1019:            close (helpf);
        !          1020:            prompt (filename);
        !          1021:            break;
        !          1022:        case 'v':       /* This case should go right before default */
        !          1023:            if (!no_intty) {
        !          1024:                kill_line ();
        !          1025:                cmdbuf[0] = '+';
        !          1026:                scanstr (Currline, &cmdbuf[1]);
        !          1027:                pr ("vi "); pr (cmdbuf); putchar (' '); pr (fnames[fnum]);
        !          1028:                execute (filename, VI, "vi", cmdbuf, fnames[fnum], 0);
        !          1029:                break;
        !          1030:            }
        !          1031:        default:
        !          1032:            write (2, &bell, 1);
        !          1033:            break;
        !          1034:        }
        !          1035:        if (done) break;
        !          1036:     }
        !          1037:     putchar ('\r');
        !          1038: endsw:
        !          1039:     inwait = 0;
        !          1040:     notell++;
        !          1041:     if (MBIT == RAW && slow_tty) {
        !          1042:        otty.sg_flags &= ~MBIT;
        !          1043:        stty(2, &otty);
        !          1044:     }
        !          1045:     return (retval);
        !          1046: }
        !          1047: 
        !          1048: char ch;
        !          1049: 
        !          1050: /*
        !          1051:  * Execute a colon-prefixed command.
        !          1052:  * Returns <0 if not a command that should cause
        !          1053:  * more of the file to be printed.
        !          1054:  */
        !          1055: 
        !          1056: colon (filename, cmd, nlines)
        !          1057: char *filename;
        !          1058: int cmd;
        !          1059: int nlines;
        !          1060: {
        !          1061:        if (cmd == 0)
        !          1062:                ch = readch ();
        !          1063:        else
        !          1064:                ch = cmd;
        !          1065:        lastcolon = ch;
        !          1066:        switch (ch) {
        !          1067:        case 'f':
        !          1068:                kill_line ();
        !          1069:                if (!no_intty)
        !          1070:                        promptlen = printf ("\"%s\" line %d", fnames[fnum], Currline);
        !          1071:                else
        !          1072:                        promptlen = printf ("[Not a file] line %d", Currline);
        !          1073:                fflush (stdout);
        !          1074:                return (-1);
        !          1075:        case 'n':
        !          1076:                if (nlines == 0) {
        !          1077:                        if (fnum >= nfiles - 1)
        !          1078:                                end_it ();
        !          1079:                        nlines++;
        !          1080:                }
        !          1081:                putchar ('\r');
        !          1082:                erase (0);
        !          1083:                skipf (nlines);
        !          1084:                return (0);
        !          1085:        case 'p':
        !          1086:                if (no_intty) {
        !          1087:                        write (2, &bell, 1);
        !          1088:                        return (-1);
        !          1089:                }
        !          1090:                putchar ('\r');
        !          1091:                erase (0);
        !          1092:                if (nlines == 0)
        !          1093:                        nlines++;
        !          1094:                skipf (-nlines);
        !          1095:                return (0);
        !          1096:        case '!':
        !          1097:                do_shell (filename);
        !          1098:                return (-1);
        !          1099:        case 'q':
        !          1100:        case 'Q':
        !          1101:                end_it ();
        !          1102:        default:
        !          1103:                write (2, &bell, 1);
        !          1104:                return (-1);
        !          1105:        }
        !          1106: }
        !          1107: 
        !          1108: /*
        !          1109: ** Read a decimal number from the terminal. Set cmd to the non-digit which
        !          1110: ** terminates the number.
        !          1111: */
        !          1112: 
        !          1113: number(cmd)
        !          1114: char *cmd;
        !          1115: {
        !          1116:        register int i;
        !          1117: 
        !          1118:        i = 0; ch = otty.sg_kill;
        !          1119:        for (;;) {
        !          1120:                ch = readch ();
        !          1121:                if (ch >= '0' && ch <= '9')
        !          1122:                        i = i*10 + ch - '0';
        !          1123:                else if (ch == otty.sg_kill)
        !          1124:                        i = 0;
        !          1125:                else {
        !          1126:                        *cmd = ch;
        !          1127:                        break;
        !          1128:                }
        !          1129:        }
        !          1130:        return (i);
        !          1131: }
        !          1132: 
        !          1133: do_shell (filename)
        !          1134: char *filename;
        !          1135: {
        !          1136:        char cmdbuf[80];
        !          1137: 
        !          1138:        kill_line ();
        !          1139:        pr ("!");
        !          1140:        fflush (stdout);
        !          1141:        promptlen = 1;
        !          1142:        if (lastp)
        !          1143:                pr (shell_line);
        !          1144:        else {
        !          1145:                ttyin (cmdbuf, 78, '!');
        !          1146:                if (expand (shell_line, cmdbuf)) {
        !          1147:                        kill_line ();
        !          1148:                        promptlen = printf ("!%s", shell_line);
        !          1149:                }
        !          1150:        }
        !          1151:        fflush (stdout);
        !          1152:        write (2, "\n", 1);
        !          1153:        promptlen = 0;
        !          1154:        shellp = 1;
        !          1155:        execute (filename, shell, shell, "-c", shell_line, 0);
        !          1156: }
        !          1157: 
        !          1158: /*
        !          1159: ** Search for nth ocurrence of regular expression contained in buf in the file
        !          1160: */
        !          1161: 
        !          1162: search (buf, file, n)
        !          1163: char buf[];
        !          1164: FILE *file;
        !          1165: register int n;
        !          1166: {
        !          1167:     long startline = Ftell (file);
        !          1168:     register long line1 = startline;
        !          1169:     register long line2 = startline;
        !          1170:     register long line3 = startline;
        !          1171:     register int lncount;
        !          1172:     int saveln, rv, re_exec();
        !          1173:     char *s, *re_comp();
        !          1174: 
        !          1175:     context.line = saveln = Currline;
        !          1176:     context.chrctr = startline;
        !          1177:     lncount = 0;
        !          1178:     if ((s = re_comp (buf)) != 0)
        !          1179:        error (s);
        !          1180:     while (!feof (file)) {
        !          1181:        line3 = line2;
        !          1182:        line2 = line1;
        !          1183:        line1 = Ftell (file);
        !          1184:        rdline (file);
        !          1185:        lncount++;
        !          1186:        if ((rv = re_exec (Line)) == 1)
        !          1187:                if (--n == 0) {
        !          1188:                    if (lncount > 3 || (lncount > 1 && no_intty))
        !          1189:                    {
        !          1190:                        pr ("\n");
        !          1191:                        if (clreol)
        !          1192:                            cleareol ();
        !          1193:                        pr("...skipping\n");
        !          1194:                    }
        !          1195:                    if (!no_intty) {
        !          1196:                        Currline -= (lncount >= 3 ? 3 : lncount);
        !          1197:                        Fseek (file, line3);
        !          1198:                        if (noscroll)
        !          1199:                            if (clreol) {
        !          1200:                                home ();
        !          1201:                                cleareol ();
        !          1202:                            } 
        !          1203:                            else
        !          1204:                                doclear ();
        !          1205:                    }
        !          1206:                    else {
        !          1207:                        kill_line ();
        !          1208:                        if (noscroll)
        !          1209:                            if (clreol) {
        !          1210:                                home (); 
        !          1211:                                cleareol ();
        !          1212:                            } 
        !          1213:                            else
        !          1214:                                doclear ();
        !          1215:                        pr (Line);
        !          1216:                        putchar ('\n');
        !          1217:                    }
        !          1218:                    break;
        !          1219:                }
        !          1220:        else if (rv == -1)
        !          1221:            error ("Regular expression botch");
        !          1222:     }
        !          1223:     if (feof (file)) {
        !          1224:        if (!no_intty) {
        !          1225:            file->_flag &= ~_IOEOF; /* why doesn't fseek do this ??!!??! */
        !          1226:            Currline = saveln;
        !          1227:            Fseek (file, startline);
        !          1228:        }
        !          1229:        else {
        !          1230:            pr ("\nPattern not found\n");
        !          1231:            end_it ();
        !          1232:        }
        !          1233:        error ("Pattern not found");
        !          1234:     }
        !          1235: }
        !          1236: 
        !          1237: execute (filename, cmd, args)
        !          1238: char *filename;
        !          1239: char *cmd, *args;
        !          1240: {
        !          1241:        int id;
        !          1242: 
        !          1243:        fflush (stdout);
        !          1244:        reset_tty ();
        !          1245:        while ((id = fork ()) < 0)
        !          1246:            sleep (5);
        !          1247:        if (id == 0) {
        !          1248:            execv (cmd, &args);
        !          1249:            write (2, "exec failed\n", 12);
        !          1250:            exit (1);
        !          1251:        }
        !          1252:        signal (SIGINT, SIG_IGN);
        !          1253:        signal (SIGQUIT, SIG_IGN);
        !          1254:        if (catch_susp)
        !          1255:            signal(SIGTSTP, SIG_DFL);
        !          1256:        wait (0);
        !          1257:        signal (SIGINT, end_it);
        !          1258:        signal (SIGQUIT, onquit);
        !          1259:        if (catch_susp)
        !          1260:            signal(SIGTSTP, onsusp);
        !          1261:        set_tty ();
        !          1262:        pr ("------------------------\n");
        !          1263:        prompt (filename);
        !          1264: }
        !          1265: /*
        !          1266: ** Skip n lines in the file f
        !          1267: */
        !          1268: 
        !          1269: skiplns (n, f)
        !          1270: register int n;
        !          1271: register FILE *f;
        !          1272: {
        !          1273:     register char c;
        !          1274: 
        !          1275:     while (n > 0) {
        !          1276:        while ((c = Getc (f)) != '\n')
        !          1277:            if (c == EOF)
        !          1278:                return;
        !          1279:            n--;
        !          1280:            Currline++;
        !          1281:     }
        !          1282: }
        !          1283: 
        !          1284: /*
        !          1285: ** Skip nskip files in the file list (from the command line). Nskip may be
        !          1286: ** negative.
        !          1287: */
        !          1288: 
        !          1289: skipf (nskip)
        !          1290: register int nskip;
        !          1291: {
        !          1292:     if (nskip == 0) return;
        !          1293:     if (nskip > 0) {
        !          1294:        if (fnum + nskip > nfiles - 1)
        !          1295:            nskip = nfiles - fnum - 1;
        !          1296:     }
        !          1297:     else if (within)
        !          1298:        ++fnum;
        !          1299:     fnum += nskip;
        !          1300:     if (fnum < 0)
        !          1301:        fnum = 0;
        !          1302:     pr ("\n...Skipping ");
        !          1303:     pr ("\n");
        !          1304:     if (clreol)
        !          1305:        cleareol ();
        !          1306:     pr ("...Skipping ");
        !          1307:     pr (nskip > 0 ? "to file " : "back to file ");
        !          1308:     pr (fnames[fnum]);
        !          1309:     pr ("\n");
        !          1310:     if (clreol)
        !          1311:        cleareol ();
        !          1312:     pr ("\n");
        !          1313:     --fnum;
        !          1314: }
        !          1315: 
        !          1316: /*----------------------------- Terminal I/O -------------------------------*/
        !          1317: 
        !          1318: initterm ()
        !          1319: {
        !          1320:     char       buf[TBUFSIZ];
        !          1321:     char       clearbuf[100];
        !          1322:     char       *clearptr, *padstr;
        !          1323:     int                ldisc;
        !          1324:     char       *term;
        !          1325: 
        !          1326:     setbuf(stdout, obuf);
        !          1327:     if (!(no_tty = gtty(1, &otty))) {
        !          1328:        if ((term = getenv("TERM")) == 0 || tgetent(buf, term) <= 0) {
        !          1329:            dumb++; ul_opt = 0;
        !          1330:        }
        !          1331:        else {
        !          1332:            if (((Lpp = tgetnum("li")) < 0) || tgetflag("hc")) {
        !          1333:                hard++; /* Hard copy terminal */
        !          1334:                Lpp = 24;
        !          1335:            }
        !          1336:            if (tailequ (fnames[0], "page") || !hard && tgetflag("ns"))
        !          1337:                noscroll++;
        !          1338:            if ((Mcol = tgetnum("co")) < 0)
        !          1339:                Mcol = 80;
        !          1340:            Wrap = tgetflag("am");
        !          1341:            bad_so = tgetflag ("xs");
        !          1342:            clearptr = clearbuf;
        !          1343:            eraseln = tgetstr("ce",&clearptr);
        !          1344:            Clear = tgetstr("cl", &clearptr);
        !          1345:            Senter = tgetstr("so", &clearptr);
        !          1346:            Sexit = tgetstr("se", &clearptr);
        !          1347: 
        !          1348:            /*
        !          1349:             *  Set up for underlining:  some terminals don't need it;
        !          1350:             *  others have start/stop sequences, still others have an
        !          1351:             *  underline char sequence which is assumed to move the
        !          1352:             *  cursor forward one character.  If underline sequence
        !          1353:             *  isn't available, settle for standout sequence.
        !          1354:             */
        !          1355: 
        !          1356:            if (tgetflag("ul") || tgetflag("os"))
        !          1357:                ul_opt = 0;
        !          1358:            if ((chUL = tgetstr("uc", &clearptr)) == NULL )
        !          1359:                chUL = "";
        !          1360:            if ((ULenter = tgetstr("us", &clearptr)) == NULL &&
        !          1361:                (!*chUL) && (ULenter = tgetstr("so", &clearptr)) == NULL)
        !          1362:                ULenter = "";
        !          1363:            if ((ULexit = tgetstr("ue", &clearptr)) == NULL &&
        !          1364:                (!*chUL) && (ULexit = tgetstr("se", &clearptr)) == NULL)
        !          1365:                ULexit = "";
        !          1366:            
        !          1367:            if (padstr = tgetstr("pc", &clearptr))
        !          1368:                PC = *padstr;
        !          1369:            Home = tgetstr("ho",&clearptr);
        !          1370:            if (Home == 0 || *Home == '\0')
        !          1371:            {
        !          1372:                if ((cursorm = tgetstr("cm", &clearptr)) != NULL) {
        !          1373:                    strcpy(cursorhome, tgoto(cursorm, 0, 0));
        !          1374:                    Home = cursorhome;
        !          1375:               }
        !          1376:            }
        !          1377:            EodClr = tgetstr("cd", &clearptr);
        !          1378:        }
        !          1379:        if ((shell = getenv("SHELL")) == NULL)
        !          1380:            shell = "/bin/sh";
        !          1381:     }
        !          1382:     no_intty = gtty(0, &otty);
        !          1383:     gtty(2, &otty);
        !          1384:     savetty = otty;
        !          1385:     ospeed = otty.sg_ospeed;
        !          1386:     slow_tty = ospeed < B1200;
        !          1387:     hardtabs =  !(otty.sg_flags & XTABS);
        !          1388:     if (!no_tty) {
        !          1389:        otty.sg_flags &= ~ECHO;
        !          1390:        if (MBIT == CBREAK || !slow_tty)
        !          1391:            otty.sg_flags |= MBIT;
        !          1392:     }
        !          1393: }
        !          1394: 
        !          1395: readch ()
        !          1396: {
        !          1397:        char ch;
        !          1398:        extern int errno;
        !          1399: 
        !          1400:        if (read (2, &ch, 1) <= 0)
        !          1401:                if (errno != EINTR)
        !          1402:                        exit(0);
        !          1403:                else
        !          1404:                        ch = otty.sg_kill;
        !          1405:        return (ch);
        !          1406: }
        !          1407: 
        !          1408: static char BS = '\b';
        !          1409: static char CARAT = '^';
        !          1410: 
        !          1411: ttyin (buf, nmax, pchar)
        !          1412: char buf[];
        !          1413: register int nmax;
        !          1414: char pchar;
        !          1415: {
        !          1416:     register char *sptr;
        !          1417:     register char ch;
        !          1418:     register int slash = 0;
        !          1419:     int        maxlen;
        !          1420:     char cbuf;
        !          1421: 
        !          1422:     sptr = buf;
        !          1423:     maxlen = 0;
        !          1424:     while (sptr - buf < nmax) {
        !          1425:        if (promptlen > maxlen) maxlen = promptlen;
        !          1426:        ch = readch ();
        !          1427:        if (ch == '\\') {
        !          1428:            slash++;
        !          1429:        }
        !          1430:        else if ((ch == otty.sg_erase) && !slash) {
        !          1431:            if (sptr > buf) {
        !          1432:                --promptlen;
        !          1433:                write (2, &BS, 1);
        !          1434:                --sptr;
        !          1435:                if ((*sptr < ' ' && *sptr != '\n') || *sptr == RUBOUT) {
        !          1436:                    --promptlen;
        !          1437:                    write (2, &BS, 1);
        !          1438:                }
        !          1439:                continue;
        !          1440:            }
        !          1441:            else {
        !          1442:                if (!eraseln) promptlen = maxlen;
        !          1443:                longjmp (restore, 1);
        !          1444:            }
        !          1445:        }
        !          1446:        else if ((ch == otty.sg_kill) && !slash) {
        !          1447:            if (hard) {
        !          1448:                show (ch);
        !          1449:                putchar ('\n');
        !          1450:                putchar (pchar);
        !          1451:            }
        !          1452:            else {
        !          1453:                putchar ('\r');
        !          1454:                putchar (pchar);
        !          1455:                if (eraseln)
        !          1456:                    erase (1);
        !          1457:                promptlen = 1;
        !          1458:            }
        !          1459:            sptr = buf;
        !          1460:            fflush (stdout);
        !          1461:            continue;
        !          1462:        }
        !          1463:        if (slash && (ch == otty.sg_kill || ch == otty.sg_erase)) {
        !          1464:            write (2, &BS, 1);
        !          1465:            --sptr;
        !          1466:        }
        !          1467:        if (ch != '\\')
        !          1468:            slash = 0;
        !          1469:        *sptr++ = ch;
        !          1470:        if ((ch < ' ' && ch != '\n' && ch != ESC) || ch == RUBOUT) {
        !          1471:            ch += ch == RUBOUT ? -0100 : 0100;
        !          1472:            write (2, &CARAT, 1);
        !          1473:            promptlen++;
        !          1474:        }
        !          1475:        cbuf = ch;
        !          1476:        if (ch != '\n' && ch != ESC) {
        !          1477:            write (2, &cbuf, 1);
        !          1478:            promptlen++;
        !          1479:        }
        !          1480:        else
        !          1481:            break;
        !          1482:     }
        !          1483:     *--sptr = '\0';
        !          1484:     if (!eraseln) promptlen = maxlen;
        !          1485:     if (sptr - buf >= nmax - 1)
        !          1486:        error ("Line too long");
        !          1487: }
        !          1488: 
        !          1489: expand (outbuf, inbuf)
        !          1490: char *outbuf;
        !          1491: char *inbuf;
        !          1492: {
        !          1493:     register char *instr;
        !          1494:     register char *outstr;
        !          1495:     register char ch;
        !          1496:     char temp[200];
        !          1497:     int changed = 0;
        !          1498: 
        !          1499:     instr = inbuf;
        !          1500:     outstr = temp;
        !          1501:     while ((ch = *instr++) != '\0')
        !          1502:        switch (ch) {
        !          1503:        case '%':
        !          1504:            if (!no_intty) {
        !          1505:                strcpy (outstr, fnames[fnum]);
        !          1506:                outstr += strlen (fnames[fnum]);
        !          1507:                changed++;
        !          1508:            }
        !          1509:            else
        !          1510:                *outstr++ = ch;
        !          1511:            break;
        !          1512:        case '!':
        !          1513:            if (!shellp)
        !          1514:                error ("No previous command to substitute for");
        !          1515:            strcpy (outstr, shell_line);
        !          1516:            outstr += strlen (shell_line);
        !          1517:            changed++;
        !          1518:            break;
        !          1519:        case '\\':
        !          1520:            if (*instr == '%' || *instr == '!') {
        !          1521:                *outstr++ = *instr++;
        !          1522:                break;
        !          1523:            }
        !          1524:        default:
        !          1525:            *outstr++ = ch;
        !          1526:        }
        !          1527:     *outstr++ = '\0';
        !          1528:     strcpy (outbuf, temp);
        !          1529:     return (changed);
        !          1530: }
        !          1531: 
        !          1532: show (ch)
        !          1533: register char ch;
        !          1534: {
        !          1535:     char cbuf;
        !          1536: 
        !          1537:     if ((ch < ' ' && ch != '\n' && ch != ESC) || ch == RUBOUT) {
        !          1538:        ch += ch == RUBOUT ? -0100 : 0100;
        !          1539:        write (2, &CARAT, 1);
        !          1540:        promptlen++;
        !          1541:     }
        !          1542:     cbuf = ch;
        !          1543:     write (2, &cbuf, 1);
        !          1544:     promptlen++;
        !          1545: }
        !          1546: 
        !          1547: error (mess)
        !          1548: char *mess;
        !          1549: {
        !          1550:     if (clreol)
        !          1551:        cleareol ();
        !          1552:     else
        !          1553:        kill_line ();
        !          1554:     promptlen += strlen (mess);
        !          1555:     if (Senter && Sexit) {
        !          1556:        tputs (Senter, 1, putch);
        !          1557:        pr(mess);
        !          1558:        tputs (Sexit, 1, putch);
        !          1559:     }
        !          1560:     else
        !          1561:        pr (mess);
        !          1562:     fflush(stdout);
        !          1563:     errors++;
        !          1564:     longjmp (restore, 1);
        !          1565: }
        !          1566: 
        !          1567: 
        !          1568: set_tty ()
        !          1569: {
        !          1570:        otty.sg_flags |= MBIT;
        !          1571:        otty.sg_flags &= ~ECHO;
        !          1572:        stty(2, &otty);
        !          1573: }
        !          1574: 
        !          1575: reset_tty ()
        !          1576: {
        !          1577:     otty.sg_flags |= ECHO;
        !          1578:     otty.sg_flags &= ~MBIT;
        !          1579:     stty(2, &savetty);
        !          1580: }
        !          1581: 
        !          1582: rdline (f)
        !          1583: register FILE *f;
        !          1584: {
        !          1585:     register char c;
        !          1586:     register char *p;
        !          1587: 
        !          1588:     p = Line;
        !          1589:     while ((c = Getc (f)) != '\n' && c != EOF && p - Line < LINSIZ - 1)
        !          1590:        *p++ = c;
        !          1591:     if (c == '\n')
        !          1592:        Currline++;
        !          1593:     *p = '\0';
        !          1594: }
        !          1595: 
        !          1596: /* Come here when we get a suspend signal from the terminal */
        !          1597: 
        !          1598: onsusp ()
        !          1599: {
        !          1600:     /* ignore SIGTTOU so we don't get stopped if csh grabs the tty */
        !          1601:     signal(SIGTTOU, SIG_IGN);
        !          1602:     reset_tty ();
        !          1603:     fflush (stdout);
        !          1604:     signal(SIGTTOU, SIG_DFL);
        !          1605:     /* Send the TSTP signal to suspend our process group */
        !          1606:     signal(SIGTSTP, SIG_DFL);
        !          1607:     sigsetmask(0);
        !          1608:     kill (0, SIGTSTP);
        !          1609:     /* Pause for station break */
        !          1610: 
        !          1611:     /* We're back */
        !          1612:     signal (SIGTSTP, onsusp);
        !          1613:     set_tty ();
        !          1614:     if (inwait)
        !          1615:            longjmp (restore);
        !          1616: }

unix.superglobalmegacorp.com

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