/* Copyright (c) 1990, 1991 UNIX System Laboratories, Inc. */ /* Copyright (c) 1984, 1986, 1987, 1988, 1989, 1990 AT&T */ /* All Rights Reserved */ /* THIS IS UNPUBLISHED PROPRIETARY SOURCE CODE OF */ /* UNIX System Laboratories, Inc. */ /* The copyright notice above does not evidence any */ /* actual or intended publication of such source code. */ #ident "@(#)tail:tail.c 1.16.1.4" /******************************************************************* PROPRIETARY NOTICE (Combined) This source code is unpublished proprietary information constituting, or derived under license from AT&T's UNIX(r) System V. In addition, portions of such source code were derived from Berkeley 4.3 BSD under license from the Regents of the University of California. Copyright Notice Notice of copyright on this source code product does not indicate publication. (c) 1986,1987,1988,1989 Sun Microsystems, Inc (c) 1983,1984,1985,1986,1987,1988,1989 AT&T. All rights reserved. ********************************************************************/ /* tail command ** ** tail where [file] ** where is [+|-]n[type] ** - means n lines before end ** + means nth line from beginning ** type 'b' means tail n blocks, not lines ** type 'c' means tail n characters * Type 'r' means in lines in reverse order from end * (for -r, default is entire buffer ) ** option 'f' means loop endlessly trying to read more ** characters after the end of file, on the assumption ** that the file is growing */ #include #include #include #include #include #include #include #include #include #include #define LBIN 0x3ffff /* 256K */ #define BSHIFT 9 /* log2(512)=9, 512 is standard buffer size (-b) */ #define RDSIZE 0x2000 /* size to read with; must be less than LBIN */ struct stat64 statb; char bin[LBIN]; extern int errno; int follow; int piped; main(argc,argv) char **argv; { register i,j,k; off64_t n,di; int fromend; int partial,bylines; int bkwds, lastnl; char *p; char *arg; int argind; int new_options; int end_options; (void)setlocale(LC_ALL, ""); (void)setcat("uxcore"); (void)setlabel("UX:tail"); lseek64(0,(off64_t)0,1); piped = errno==ESPIPE; new_options = 0; end_options = 0; fromend = 1; bylines = -1; follow = 0; bkwds = 0; n = -1; for (argind = 1; argind < argc; argind++) { arg = argv[argind]; if (*arg == '-') { arg++; switch(*arg) { case '-': if (!isspace(*(arg+1)) && *(arg+1) != '\0') goto errcom; argind++; end_options = 1; break; case 'f': follow = 1; new_options = 1; break; case 'c': case 'n': if (bylines != -1) goto errcom; if (*arg == 'c') bylines = 0; if (*arg == 'n') bylines = 1; arg++; if (isspace(*arg) || *arg == '\0') { if (argc <= argind+1) { if (new_options) goto errcom; bylines = -1; break; } arg = argv[++argind]; } fromend = !(*arg=='+'); if (*arg == '-' || *arg == '+') arg++; if (isdigit(*arg)) { n = 0; while(isdigit(*arg)) n = n*10 + *arg++ - '0'; if (!isspace(*arg) && *arg != '\0') goto errcom; } else { if (new_options) goto errcom; bylines = -1; break; } new_options = 1; break; default: if (new_options) goto errcom; break; } } else break; if (!new_options || end_options) break; } if (!new_options && !end_options) { arg = argv[1]; if(argc<=1 || *arg!='-'&&*arg!='+') { arg = "-10l"; argc++; argv--; } fromend = *arg=='-'; arg++; if (isdigit(*arg)) { n = 0; while(isdigit(*arg)) n = n*10 + *arg++ - '0'; } else n = -1; if (argc > 2 && argv[2][0] == '-' && argv[2][1] == '-' && (isspace(argv[2][2]) || argv[2][2] == '\0')) argind = 3; else argind = 2; } if(argc > argind) { if (argc > argind + 1) goto errcom; close(0); piped = 0; if(open(argv[argind],0)!=0) { pfmt(stderr, MM_ERROR, ":92:Cannot open %s: %s\n", argv[argind], strerror(errno)); exit(3); } } if (!new_options && !end_options) { while(*arg) switch(*arg++) { case 'b': if(n == -1) n = 10; n <<= BSHIFT; if(bylines!=-1 || bkwds==1) goto errcom; bylines=0; break; case 'c': if(bylines!=-1 || bkwds==1) goto errcom; bylines=0; break; case 'f': if(bkwds) goto errcom; follow = 1; break; case 'r': if(follow) goto errcom; if (n==-1) n = LBIN; else if (!fromend) n++; if (bylines==0) goto errcom; bkwds = 1; fromend = 1; bylines = 1; break; case 'l': if(bylines!=-1 && bylines==1) goto errcom; bylines = 1; break; default: goto errcom; } } if (n == -1) n = 10; if(!fromend&&n>0) n--; if(bylines==-1) bylines = 1; if (bkwds) follow=0; if(fromend) goto keep; /*seek from beginning */ if(bylines) { j = 0; while(n-->0) { do { if(j--<=0) { p = bin; j = read(0,p,RDSIZE); if(j--<=0) fexit(); } } while(*p++ != '\n'); } write(1,p,j); } else if(n>0) { /* '-b' */ if(!piped) fstat64(0,&statb); if(piped||(statb.st_mode&S_IFMT)==S_IFCHR) while(n>0) { i = n>512?512:n; i = read(0,bin,i); if(i<=0) fexit(); n -= i; } else lseek64(0,n,0); } copy: while((i=read(0,bin,RDSIZE))>0) write(1,bin,i); fexit(); /*seek from end*/ keep: if(n < 0) { /* so we work correctly with -f0 */ (void)lseek64(0,(off64_t)0,2); fexit(); } if(!piped) { fstat64(0,&statb); di = !bylines&&n di) lseek64(0,-di,2); if (!bylines) goto copy; } partial = 1; for(;;) { i = 0; do { j = read(0,&bin[i],LBIN-i); if(j<=0) goto brka; i += j; } while(i=LBIN ? i+1: i-n+LBIN; k--; } else { if(bkwds && bin[i==0?LBIN-1:i-1]!='\n'){ /* force trailing newline */ bin[i]='\n'; if(++i>=LBIN) {i = 0; partial = 0;} } k = i; j = 0; do { lastnl = k; do { if(--k<0) { if(partial) { if(bkwds) (void)write(1,bin,lastnl+1); goto brkb; } k = LBIN -1; } } while(bin[k]!='\n'&&k!=i); if(bkwds && j>0){ if(k=LBIN) k = 0; } while(bin[k]!='\n'&&k!=i); } if(k 0) write (1, bin, n); } }