1 /* 2 * CDDL HEADER START 3 * 4 * The contents of this file are subject to the terms of the 5 * Common Development and Distribution License (the "License"). 6 * You may not use this file except in compliance with the License. 7 * 8 * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 9 * or http://www.opensolaris.org/os/licensing. 10 * See the License for the specific language governing permissions 11 * and limitations under the License. 12 * 13 * When distributing Covered Code, include this CDDL HEADER in each 14 * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 15 * If applicable, add the following below this CDDL HEADER, with the 16 * fields enclosed by brackets "[]" replaced with your own identifying 17 * information: Portions Copyright [yyyy] [name of copyright owner] 18 * 19 * CDDL HEADER END 20 */ 21 22 /* 23 * Copyright (c) 1988, 2010, Oracle and/or its affiliates. All rights reserved. 24 */ 25 26 /* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */ 27 /* All Rights Reserved */ 28 29 /* 30 * Editor 31 */ 32 33 #include <crypt.h> 34 #include <libgen.h> 35 #include <wait.h> 36 #include <string.h> 37 #include <sys/types.h> 38 #include <locale.h> 39 #include <regexpr.h> 40 #include <regex.h> 41 #include <errno.h> 42 #include <paths.h> 43 44 static const char *msgtab[] = 45 { 46 "write or open on pipe failed", /* 0 */ 47 "warning: expecting `w'", /* 1 */ 48 "mark not lower case ascii", /* 2 */ 49 "Cannot open input file", /* 3 */ 50 "PWB spec problem", /* 4 */ 51 "nothing to undo", /* 5 */ 52 "restricted shell", /* 6 */ 53 "cannot create output file", /* 7 */ 54 "filesystem out of space!", /* 8 */ 55 "cannot open file", /* 9 */ 56 "cannot link", /* 10 */ 57 "Range endpoint too large", /* 11 */ 58 "unknown command", /* 12 */ 59 "search string not found", /* 13 */ 60 "-", /* 14 */ 61 "line out of range", /* 15 */ 62 "bad number", /* 16 */ 63 "bad range", /* 17 */ 64 "Illegal address count", /* 18 */ 65 "incomplete global expression", /* 19 */ 66 "illegal suffix", /* 20 */ 67 "illegal or missing filename", /* 21 */ 68 "no space after command", /* 22 */ 69 "fork failed - try again", /* 23 */ 70 "maximum of 64 characters in file names", /* 24 */ 71 "`\\digit' out of range", /* 25 */ 72 "interrupt", /* 26 */ 73 "line too long", /* 27 */ 74 "illegal character in input file", /* 28 */ 75 "write error", /* 29 */ 76 "out of memory for append", /* 30 */ 77 "temp file too big", /* 31 */ 78 "I/O error on temp file", /* 32 */ 79 "multiple globals not allowed", /* 33 */ 80 "global too long", /* 34 */ 81 "no match", /* 35 */ 82 "illegal or missing delimiter", /* 36 */ 83 "-", /* 37 */ 84 "replacement string too long", /* 38 */ 85 "illegal move destination", /* 39 */ 86 "-", /* 40 */ 87 "no remembered search string", /* 41 */ 88 "'\\( \\)' imbalance", /* 42 */ 89 "Too many `\\(' s", /* 43 */ 90 "more than 2 numbers given", /* 44 */ 91 "'\\}' expected", /* 45 */ 92 "first number exceeds second", /* 46 */ 93 "incomplete substitute", /* 47 */ 94 "newline unexpected", /* 48 */ 95 "'[ ]' imbalance", /* 49 */ 96 "regular expression overflow", /* 50 */ 97 "regular expression error", /* 51 */ 98 "command expected", /* 52 */ 99 "a, i, or c not allowed in G", /* 53 */ 100 "end of line expected", /* 54 */ 101 "no remembered replacement string", /* 55 */ 102 "no remembered command", /* 56 */ 103 "illegal redirection", /* 57 */ 104 "possible concurrent update", /* 58 */ 105 "-", /* 59 */ 106 "the x command has become X (upper case)", /* 60 */ 107 "Warning: 'w' may destroy input file " 108 "(due to `illegal char' read earlier)", 109 /* 61 */ 110 "Caution: 'q' may lose data in buffer;" 111 " 'w' may destroy input file", 112 /* 62 */ 113 "Encryption of string failed", /* 63 */ 114 "Encryption facility not available", /* 64 */ 115 "Cannot encrypt temporary file", /* 65 */ 116 "Enter key:", /* 66 */ 117 "Illegal byte sequence", /* 67 */ 118 "File does not exist", /* 68 */ 119 "tempnam failed", /* 69 */ 120 "Cannot open temporary file", /* 70 */ 121 0 122 }; 123 124 #include <stdlib.h> 125 #include <limits.h> 126 #include <stdio.h> 127 #include <signal.h> 128 #include <sys/types.h> 129 #include <sys/stat.h> 130 #include <sys/statvfs.h> 131 #include <unistd.h> 132 #include <termio.h> 133 #include <ctype.h> 134 #include <setjmp.h> 135 #include <fcntl.h> 136 #include <wchar.h> /* I18N */ 137 #include <wctype.h> /* I18N */ 138 #include <widec.h> /* I18N */ 139 140 #define FTYPE(A) (A.st_mode) 141 #define FMODE(A) (A.st_mode) 142 #define IDENTICAL(A, B) (A.st_dev == B.st_dev && A.st_ino == B.st_ino) 143 #define ISBLK(A) ((A.st_mode & S_IFMT) == S_IFBLK) 144 #define ISCHR(A) ((A.st_mode & S_IFMT) == S_IFCHR) 145 #define ISDIR(A) ((A.st_mode & S_IFMT) == S_IFDIR) 146 #define ISFIFO(A) ((A.st_mode & S_IFMT) == S_IFIFO) 147 #define ISREG(A) ((A.st_mode & S_IFMT) == S_IFREG) 148 149 #define PUTM() if (xcode >= 0) puts(gettext(msgtab[xcode])) 150 #define UNGETC(c) (peekc = c) 151 #define FNSIZE PATH_MAX 152 #define LBSIZE LINE_MAX 153 154 /* size of substitution replacement pattern buffer */ 155 #define RHSIZE (LINE_MAX*2) 156 157 #define KSIZE 8 158 159 #define READ 0 160 #define WRITE 1 161 162 extern char *optarg; /* Value of argument */ 163 extern int optind; /* Indicator of argument */ 164 extern int __xpg4; /* defined in xpg4.c; 0 if not xpg4-compiled program */ 165 166 struct Fspec { 167 char Ftabs[22]; 168 char Fdel; 169 unsigned char Flim; 170 char Fmov; 171 char Ffill; 172 }; 173 static struct Fspec fss; 174 175 static char *fsp; 176 static int fsprtn; 177 static char line[70]; 178 static char *linp = line; 179 static int sig; 180 static int Xqt = 0; 181 static int lastc; 182 static char savedfile[FNSIZE]; 183 static char file[FNSIZE]; 184 static char funny[FNSIZE]; 185 static int funlink = 0; 186 static char linebuf[LBSIZE]; 187 static char *tstring = linebuf; 188 189 static char *expbuf; 190 191 static char rhsbuf[RHSIZE]; 192 struct lin { 193 long cur; 194 long sav; 195 }; 196 typedef struct lin *LINE; 197 static LINE zero; 198 static LINE dot; 199 static LINE dol; 200 static LINE endcore; 201 static LINE fendcore; 202 static LINE addr1; 203 static LINE addr2; 204 static LINE savdol, savdot; 205 static int globflg; 206 static int initflg; 207 static char genbuf[LBSIZE]; 208 static long count; 209 static int numpass; /* Number of passes thru dosub(). */ 210 static int gsubf; /* Occurrence value. LBSIZE-1=all. */ 211 static int ocerr1; /* Allows lines NOT changed by dosub() to NOT be put */ 212 /* out. Retains last line changed as current line. */ 213 static int ocerr2; /* Flags if ANY line changed by substitute(). 0=nc. */ 214 static char *nextip; 215 static char *linebp; 216 static int ninbuf; 217 static int peekc; 218 static int io; 219 static void (*oldhup)(), (*oldintr)(); 220 static void (*oldquit)(), (*oldpipe)(); 221 static void quit(int) __NORETURN; 222 static int vflag = 1; 223 static int xflag; 224 static int xtflag; 225 static int kflag; 226 static int crflag; 227 /* Flag for determining if file being read is encrypted */ 228 static int hflag; 229 static int xcode = -1; 230 static char crbuf[LBSIZE]; 231 static int perm[2]; 232 static int tperm[2]; 233 static int permflag; 234 static int tpermflag; 235 static int col; 236 static char *globp; 237 static int tfile = -1; 238 static int tline; 239 static char *tfname; 240 extern char *locs; 241 static char ibuff[LBSIZE]; 242 static int iblock = -1; 243 static char obuff[LBSIZE]; 244 static int oblock = -1; 245 static int ichanged; 246 static int nleft; 247 static long savnames[26], names[26]; 248 static int anymarks; 249 static long subnewa; 250 static int fchange; 251 static int nline; 252 static int fflg, shflg; 253 static char prompt[16] = "*"; 254 static int rflg; 255 static int readflg; 256 static int eflg; 257 static int qflg = 0; 258 static int ncflg; 259 static int listn; 260 static int listf; 261 static int pflag; 262 static int flag28 = 0; /* Prevents write after a partial read */ 263 static int save28 = 0; /* Flag whether buffer empty at start of read */ 264 static long savtime; 265 static char *name = "SHELL"; 266 static char *rshell = "/usr/lib/rsh"; 267 static char *val; 268 static char *home; 269 static int nodelim; 270 271 int makekey(int *); 272 int _mbftowc(char *, wchar_t *, int (*)(), int *); 273 static int error(int code); 274 static void tlist(struct Fspec *); 275 static void tstd(struct Fspec *); 276 static void gdelete(void); 277 static void delete(void); 278 static void exfile(void); 279 static void filename(int comm); 280 static void newline(void); 281 static int gettty(void); 282 static void commands(void); 283 static void undo(void); 284 static void save(void); 285 static void strcopy(char *source, char *dest); 286 static int strequal(char **scan1, char *str); 287 static int stdtab(char *, char *); 288 static int lenchk(char *, struct Fspec *); 289 static void clear(struct Fspec *); 290 static int expnd(char *, char *, int *, struct Fspec *); 291 static void tincr(int, struct Fspec *); 292 static void targ(struct Fspec *); 293 static int numb(void); 294 static int fspec(char *, struct Fspec *, int); 295 static void red(char *); 296 static void newtime(void); 297 static void chktime(void); 298 static void getime(void); 299 static void mkfunny(void); 300 static int eopen(char *, int); 301 static void eclose(int f); 302 static void globaln(int); 303 static char *getkey(const char *); 304 static int execute(int, LINE); 305 static void error1(int); 306 static int getcopy(void); 307 static void move(int); 308 static void dosub(void); 309 static int getsub(void); 310 static int compsub(void); 311 static void substitute(int); 312 static void join(void); 313 static void global(int); 314 static void init(void); 315 static void rdelete(LINE, LINE); 316 static void append(int (*)(void), LINE); 317 static int getfile(void); 318 static void putfile(void); 319 static void onpipe(int); 320 static void onhup(int); 321 static void onintr(int); 322 static void setdot(void); 323 static void setall(void); 324 static void setnoaddr(void); 325 static void nonzero(void); 326 static void setzeroasone(void); 327 static long putline(void); 328 static LINE address(void); 329 static char *getaline(long); 330 static char *getblock(long, long); 331 static char *place(char *, char *, char *); 332 static void comple(wchar_t); 333 static void putchr(unsigned char); 334 static void putwchr(wchar_t); 335 static int getchr(void); 336 static void unixcom(void); 337 static void blkio(int, char *, ssize_t (*)()); 338 static void reverse(LINE, LINE); 339 static void putd(); 340 static wchar_t get_wchr(void); 341 342 static struct stat Fl, Tf; 343 #ifndef RESEARCH 344 static struct statvfs U; 345 static int Short = 0; 346 static mode_t oldmask; /* No umask while writing */ 347 #endif 348 static jmp_buf savej; 349 350 #ifdef NULLS 351 int nulls; /* Null count */ 352 #endif 353 static long ccount; 354 355 static int errcnt = 0; 356 357 358 static void 359 onpipe(int sig) 360 { 361 (int)error(0); 362 } 363 364 int 365 main(int argc, char **argv) 366 { 367 char *p1, *p2; 368 int c; 369 370 (void) setlocale(LC_ALL, ""); 371 #if !defined(TEXT_DOMAIN) /* Should be defined by cc -D */ 372 #define TEXT_DOMAIN "SYS_TEST" /* Use this only if it weren't */ 373 #endif 374 (void) textdomain(TEXT_DOMAIN); 375 376 oldquit = signal(SIGQUIT, SIG_IGN); 377 oldhup = signal(SIGHUP, SIG_IGN); 378 oldintr = signal(SIGINT, SIG_IGN); 379 oldpipe = signal(SIGPIPE, onpipe); 380 if (signal(SIGTERM, SIG_IGN) != SIG_IGN) 381 signal(SIGTERM, quit); 382 p1 = *argv; 383 while (*p1++) 384 ; 385 while (--p1 >= *argv) 386 if (*p1 == '/') 387 break; 388 *argv = p1 + 1; 389 /* if SHELL set in environment and is /usr/lib/rsh, set rflg */ 390 if ((val = getenv(name)) != NULL) 391 if (strcmp(val, rshell) == 0) 392 rflg++; 393 if (**argv == 'r') 394 rflg++; 395 home = getenv("HOME"); 396 while (1) { 397 while ((c = getopt(argc, argv, "sp:qxC")) != EOF) { 398 switch (c) { 399 400 case 's': 401 vflag = 0; 402 break; 403 404 case 'p': 405 strncpy(prompt, optarg, sizeof (prompt)-1); 406 shflg = 1; 407 break; 408 409 case 'q': 410 signal(SIGQUIT, SIG_DFL); 411 vflag = 1; 412 break; 413 414 case 'x': 415 crflag = -1; 416 xflag = 1; 417 break; 418 419 case 'C': 420 crflag = 1; 421 xflag = 1; 422 break; 423 424 case '?': 425 (void) fprintf(stderr, gettext( 426 "Usage: ed [- | -s] [-p string] [-x] [-C] [file]\n" 427 " red [- | -s] [-p string] [-x] [-C] [file]\n")); 428 exit(2); 429 } 430 } 431 if (argv[optind] && strcmp(argv[optind], "-") == 0 && 432 strcmp(argv[optind-1], "--") != 0) { 433 vflag = 0; 434 optind++; 435 continue; 436 } 437 break; 438 } 439 argc = argc - optind; 440 argv = &argv[optind]; 441 442 if (xflag) { 443 if (permflag) 444 crypt_close(perm); 445 permflag = 1; 446 kflag = run_setkey(&perm[0], getkey(msgtab[66])); 447 if (kflag == -1) { 448 puts(gettext(msgtab[64])); 449 xflag = 0; 450 kflag = 0; 451 } 452 if (kflag == 0) 453 crflag = 0; 454 } 455 456 if (argc > 0) { 457 p1 = *argv; 458 if (strlen(p1) >= (size_t)FNSIZE) { 459 puts(gettext("file name too long")); 460 if (kflag) 461 crypt_close(perm); 462 exit(2); 463 } 464 p2 = savedfile; 465 while (*p2++ = *p1++) 466 ; 467 globp = "e"; 468 fflg++; 469 } else /* editing with no file so set savtime to 0 */ 470 savtime = 0; 471 eflg++; 472 if ((tfname = tempnam("", "ea")) == NULL) { 473 puts(gettext(msgtab[69])); 474 exit(2); 475 } 476 477 fendcore = (LINE)sbrk(0); 478 init(); 479 if (oldintr != SIG_IGN) 480 signal(SIGINT, onintr); 481 if (oldhup != SIG_IGN) 482 signal(SIGHUP, onhup); 483 setjmp(savej); 484 commands(); 485 quit(sig); 486 return (0); 487 } 488 489 static void 490 commands(void) 491 { 492 LINE a1; 493 int c; 494 char *p1, *p2; 495 int fsave, m, n; 496 497 for (;;) { 498 nodelim = 0; 499 if (pflag) { 500 pflag = 0; 501 addr1 = addr2 = dot; 502 goto print; 503 } 504 if (shflg && globp == 0) 505 write(1, gettext(prompt), strlen(gettext(prompt))); 506 addr1 = 0; 507 addr2 = 0; 508 if ((c = getchr()) == ',') { 509 addr1 = zero + 1; 510 addr2 = dol; 511 #ifdef XPG6 512 /* XPG4 - it was an error if the second address was */ 513 /* input and the first address was ommitted */ 514 /* Parse second address */ 515 if ((a1 = address()) != 0) { 516 addr2 = a1; 517 } 518 #endif 519 c = getchr(); 520 goto swch; 521 } else if (c == ';') { 522 addr1 = dot; 523 addr2 = dol; 524 #ifdef XPG6 525 /* XPG4 - it was an error if the second address was */ 526 /* input and the first address was ommitted */ 527 /* Parse second address */ 528 if ((a1 = address()) != 0) { 529 addr2 = a1; 530 } 531 #endif 532 c = getchr(); 533 goto swch; 534 } else 535 peekc = c; 536 do { 537 addr1 = addr2; 538 if ((a1 = address()) == 0) { 539 c = getchr(); 540 break; 541 } 542 addr2 = a1; 543 if ((c = getchr()) == ';') { 544 c = ','; 545 dot = a1; 546 } 547 } while (c == ','); 548 if (addr1 == 0) 549 addr1 = addr2; 550 swch: 551 switch (c) { 552 553 case 'a': 554 setdot(); 555 newline(); 556 if (!globflg) save(); 557 append(gettty, addr2); 558 continue; 559 560 case 'c': 561 #ifdef XPG6 562 setzeroasone(); 563 #endif 564 delete(); 565 append(gettty, addr1-1); 566 567 /* XPG4 - If no new lines are inserted, then the current */ 568 /* line becomes the line after the lines deleted. */ 569 570 if (((linebuf[0] != '.') || (dot == (addr1-1))) && 571 (addr2 <= dol)) 572 dot = addr1; 573 continue; 574 575 case 'd': 576 delete(); 577 continue; 578 579 case 'E': 580 fchange = 0; 581 c = 'e'; 582 /* FALLTHROUGH */ 583 case 'e': 584 fflg++; 585 setnoaddr(); 586 if (vflag && fchange) { 587 fchange = 0; 588 (void) error(1); 589 } 590 filename(c); 591 eflg++; 592 init(); 593 addr2 = zero; 594 goto caseread; 595 596 case 'f': 597 setnoaddr(); 598 filename(c); 599 if (!ncflg) /* there is a filename */ 600 getime(); 601 else 602 ncflg--; 603 puts(savedfile); 604 continue; 605 606 case 'g': 607 global(1); 608 continue; 609 case 'G': 610 globaln(1); 611 continue; 612 613 case 'h': 614 newline(); 615 setnoaddr(); 616 PUTM(); 617 continue; 618 619 case 'H': 620 newline(); 621 setnoaddr(); 622 if (!hflag) { 623 hflag = 1; 624 PUTM(); 625 } 626 else 627 hflag = 0; 628 continue; 629 630 case 'i': 631 #ifdef XPG6 632 setzeroasone(); 633 #endif 634 setdot(); 635 nonzero(); 636 newline(); 637 if (!globflg) save(); 638 append(gettty, addr2-1); 639 if (dot == addr2-1) 640 dot += 1; 641 continue; 642 643 case 'j': 644 if (addr2 == 0) { 645 addr1 = dot; 646 addr2 = dot+1; 647 } 648 setdot(); 649 newline(); 650 nonzero(); 651 if (!globflg) save(); 652 join(); 653 continue; 654 655 case 'k': 656 if ((c = getchr()) < 'a' || c > 'z') 657 (void) error(2); 658 newline(); 659 setdot(); 660 nonzero(); 661 names[c-'a'] = addr2->cur & ~01; 662 anymarks |= 01; 663 continue; 664 665 case 'm': 666 move(0); 667 continue; 668 669 case '\n': 670 if (addr2 == 0) 671 addr2 = dot+1; 672 addr1 = addr2; 673 goto print; 674 675 case 'n': 676 listn++; 677 newline(); 678 goto print; 679 680 case 'l': 681 listf++; 682 /* FALLTHROUGH */ 683 case 'p': 684 newline(); 685 print: 686 setdot(); 687 nonzero(); 688 a1 = addr1; 689 do { 690 if (listn) { 691 count = a1 - zero; 692 putd(); 693 putchr('\t'); 694 } 695 puts(getaline((a1++)->cur)); 696 } while (a1 <= addr2); 697 dot = addr2; 698 pflag = 0; 699 listn = 0; 700 listf = 0; 701 continue; 702 703 case 'Q': 704 fchange = 0; 705 /* FALLTHROUGH */ 706 case 'q': 707 setnoaddr(); 708 newline(); 709 quit(sig); 710 711 case 'r': 712 filename(c); 713 caseread: 714 readflg = 1; 715 save28 = (dol != fendcore); 716 if (crflag == 2 || crflag == -2) 717 crflag = -1; /* restore crflag for next file */ 718 errno = 0; 719 if ((io = eopen(file, O_RDONLY)) < 0) { 720 lastc = '\n'; 721 /* if first entering editor and file does not exist */ 722 /* set saved access time to 0 */ 723 if (eflg) { 724 savtime = 0; 725 eflg = 0; 726 if (c == 'e' && vflag == 0) 727 qflg = 1; 728 } 729 if (errno == ENOENT) { 730 (void) error(68); 731 } else { 732 (void) error(3); 733 } 734 } 735 /* get last mod time of file */ 736 /* eflg - entered editor with ed or e */ 737 if (eflg) { 738 eflg = 0; 739 getime(); 740 } 741 setall(); 742 ninbuf = 0; 743 n = zero != dol; 744 #ifdef NULLS 745 nulls = 0; 746 #endif 747 if (!globflg && (c == 'r')) save(); 748 append(getfile, addr2); 749 exfile(); 750 readflg = 0; 751 fchange = n; 752 continue; 753 754 case 's': 755 setdot(); 756 nonzero(); 757 if (!globflg) save(); 758 substitute(globp != 0); 759 continue; 760 761 case 't': 762 move(1); 763 continue; 764 765 case 'u': 766 setdot(); 767 newline(); 768 if (!initflg) 769 undo(); 770 else 771 (void) error(5); 772 fchange = 1; 773 continue; 774 775 case 'v': 776 global(0); 777 continue; 778 case 'V': 779 globaln(0); 780 continue; 781 782 case 'W': 783 case 'w': 784 if (flag28) { 785 flag28 = 0; 786 fchange = 0; 787 (void) error(61); 788 } 789 setall(); 790 791 /* on NULL-RE condition do not generate error */ 792 793 if ((linebuf[0] != '.') && (zero != dol) && 794 (addr1 <= zero || addr2 > dol)) 795 (void) error(15); 796 filename(c); 797 if (Xqt) { 798 io = eopen(file, O_WRONLY); 799 n = 1; /* set n so newtime will not execute */ 800 } else { 801 struct stat lFl; 802 fstat(tfile, &Tf); 803 if (stat(file, &Fl) < 0) { 804 if ((io = creat(file, S_IRUSR|S_IWUSR|S_IRGRP 805 |S_IWGRP|S_IROTH|S_IWOTH)) < 0) 806 (void) error(7); 807 fstat(io, &Fl); 808 Fl.st_mtime = 0; 809 lFl = Fl; 810 close(io); 811 } else { 812 #ifndef RESEARCH 813 oldmask = umask(0); 814 /* 815 * Must determine if file is 816 * a symbolic link 817 */ 818 lstat(file, &lFl); 819 #endif 820 } 821 #ifndef RESEARCH 822 /* 823 * Determine if there are enough free blocks on system 824 */ 825 if (!Short && statvfs(file, &U) == 0 && 826 U.f_bfree < ((Tf.st_size/U.f_frsize) + 100)) { 827 Short = 1; 828 (void) error(8); 829 } 830 Short = 0; 831 #endif 832 p1 = savedfile; /* The current filename */ 833 p2 = file; 834 m = strcmp(p1, p2); 835 if (c == 'w' && Fl.st_nlink == 1 && ISREG(lFl)) { 836 if (close(open(file, O_WRONLY)) < 0) 837 (void) error(9); 838 if (!(n = m)) 839 chktime(); 840 mkfunny(); 841 /* 842 * If funlink equals one it means that 843 * funny points to a valid file which must 844 * be unlinked when interrupted. 845 */ 846 847 funlink = 1; 848 if ((io = creat(funny, FMODE(Fl))) >= 0) { 849 chown(funny, Fl.st_uid, Fl.st_gid); 850 chmod(funny, FMODE(Fl)); 851 putfile(); 852 exfile(); 853 854 if (rename(funny, file)) 855 (void) error(10); 856 funlink = 0; 857 /* if filenames are the same */ 858 if (!n) 859 newtime(); 860 /* check if entire buffer was written */ 861 fsave = fchange; 862 if (((addr1 == zero) || 863 (addr1 == (zero + 1))) && 864 (addr2 == dol)) 865 fchange = 0; 866 else 867 fchange = 1; 868 if (fchange == 1 && m != 0) 869 fchange = fsave; 870 continue; 871 } 872 } else { 873 n = 1; /* set n so newtime will not execute */ 874 } 875 if ((io = open(file, 876 (c == 'w') ? O_WRONLY|O_CREAT|O_TRUNC 877 : O_WRONLY|O_CREAT|O_APPEND, S_IRUSR|S_IWUSR 878 |S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH)) < 0) 879 (void) error(7); 880 } 881 putfile(); 882 exfile(); 883 if (!n) 884 newtime(); 885 fsave = fchange; 886 fchange = (((addr1 == zero) || (addr1 == (zero + 1))) && 887 (addr2 == dol)) ? 0 : 1; 888 /* Leave fchange alone if partial write was to another file */ 889 if (fchange == 1 && m != 0) fchange = fsave; 890 continue; 891 892 case 'C': 893 crflag = 1; 894 /* 895 * C is same as X, but always assume input files are 896 * ciphertext 897 */ 898 goto encrypt; 899 900 case 'X': 901 crflag = -1; 902 encrypt: 903 setnoaddr(); 904 newline(); 905 xflag = 1; 906 if (permflag) 907 (void) crypt_close(perm); 908 permflag = 1; 909 if ((kflag = run_setkey(&perm[0], getkey(msgtab[66]))) == -1) { 910 xflag = 0; 911 kflag = 0; 912 crflag = 0; 913 (void) error(64); 914 } 915 if (kflag == 0) 916 crflag = 0; 917 continue; 918 919 case '=': 920 setall(); 921 newline(); 922 count = (addr2-zero)&077777; 923 putd(); 924 putchr('\n'); 925 continue; 926 927 case '!': 928 unixcom(); 929 continue; 930 931 case EOF: 932 return; 933 934 case 'P': 935 setnoaddr(); 936 newline(); 937 if (shflg) 938 shflg = 0; 939 else 940 shflg++; 941 continue; 942 } 943 if (c == 'x') 944 (void) error(60); 945 else 946 (void) error(12); 947 } 948 } 949 950 LINE 951 address(void) 952 { 953 int minus, c; 954 LINE a1; 955 int n, relerr, retval; 956 957 minus = 0; 958 a1 = 0; 959 for (;;) { 960 c = getchr(); 961 if ('0' <= c && c <= '9') { 962 n = 0; 963 do { 964 n *= 10; 965 n += c - '0'; 966 } while ((c = getchr()) >= '0' && c <= '9'); 967 peekc = c; 968 if (a1 == 0) 969 a1 = zero; 970 if (minus < 0) 971 n = -n; 972 a1 += n; 973 minus = 0; 974 continue; 975 } 976 relerr = 0; 977 if (a1 || minus) 978 relerr++; 979 switch (c) { 980 case ' ': 981 case '\t': 982 continue; 983 984 case '+': 985 minus++; 986 if (a1 == 0) 987 a1 = dot; 988 continue; 989 990 case '-': 991 case '^': 992 minus--; 993 if (a1 == 0) 994 a1 = dot; 995 continue; 996 997 case '?': 998 case '/': 999 comple(c); 1000 a1 = dot; 1001 for (;;) { 1002 if (c == '/') { 1003 a1++; 1004 if (a1 > dol) 1005 a1 = zero; 1006 } else { 1007 a1--; 1008 if (a1 < zero) 1009 a1 = dol; 1010 } 1011 1012 if (execute(0, a1)) 1013 break; 1014 if (a1 == dot) 1015 (void) error(13); 1016 } 1017 break; 1018 1019 case '$': 1020 a1 = dol; 1021 break; 1022 1023 case '.': 1024 a1 = dot; 1025 break; 1026 1027 case '\'': 1028 if ((c = getchr()) < 'a' || c > 'z') 1029 (void) error(2); 1030 for (a1 = zero; a1 <= dol; a1++) 1031 if (names[c-'a'] == (a1->cur & ~01)) 1032 break; 1033 break; 1034 1035 default: 1036 peekc = c; 1037 if (a1 == 0) 1038 return (0); 1039 a1 += minus; 1040 1041 /* on NULL-RE condition do not generate error */ 1042 1043 if ((linebuf[0] != '.') && (a1 < zero || a1 > dol)) 1044 (void) error(15); 1045 return (a1); 1046 } 1047 if (relerr) 1048 (void) error(16); 1049 } 1050 } 1051 1052 static void 1053 setdot(void) 1054 { 1055 if (addr2 == 0) 1056 addr1 = addr2 = dot; 1057 if (addr1 > addr2) 1058 (void) error(17); 1059 } 1060 1061 static void 1062 setall(void) 1063 { 1064 if (addr2 == 0) { 1065 addr1 = zero+1; 1066 addr2 = dol; 1067 if (dol == zero) 1068 addr1 = zero; 1069 } 1070 setdot(); 1071 } 1072 1073 static void 1074 setnoaddr(void) 1075 { 1076 if (addr2) 1077 (void) error(18); 1078 } 1079 1080 static void 1081 nonzero(void) 1082 { 1083 /* on NULL-RE condition do not generate error */ 1084 1085 if ((linebuf[0] != '.') && (addr1 <= zero || addr2 > dol)) 1086 (void) error(15); 1087 } 1088 1089 static void 1090 setzeroasone(void) 1091 { 1092 /* for the c and i commands 0 equal to 1 address */ 1093 if (addr1 == zero) { 1094 addr1 = zero+1; 1095 } 1096 if (addr2 == zero) { 1097 addr2 = zero+1; 1098 } 1099 } 1100 1101 1102 static void 1103 newline(void) 1104 { 1105 int c; 1106 1107 if ((c = getchr()) == '\n') 1108 return; 1109 if (c == 'p' || c == 'l' || c == 'n') { 1110 pflag++; 1111 if (c == 'l') listf++; 1112 if (c == 'n') listn++; 1113 if ((c = getchr()) == '\n') 1114 return; 1115 } 1116 (void) error(20); 1117 } 1118 1119 static void 1120 filename(int comm) 1121 { 1122 char *p1, *p2; 1123 int c; 1124 int i = 0; 1125 1126 count = 0; 1127 c = getchr(); 1128 if (c == '\n' || c == EOF) { 1129 p1 = savedfile; 1130 if (*p1 == 0 && comm != 'f') 1131 (void) error(21); 1132 /* ncflg set means do not get mod time of file */ 1133 /* since no filename followed f */ 1134 if (comm == 'f') 1135 ncflg++; 1136 p2 = file; 1137 while (*p2++ = *p1++) 1138 ; 1139 red(savedfile); 1140 return; 1141 } 1142 if (c != ' ') 1143 (void) error(22); 1144 while ((c = getchr()) == ' ') 1145 ; 1146 if (c == '!') 1147 ++Xqt, c = getchr(); 1148 if (c == '\n') 1149 (void) error(21); 1150 p1 = file; 1151 do { 1152 if (++i >= FNSIZE) 1153 (void) error(24); 1154 *p1++ = c; 1155 if (c == EOF || (c == ' ' && !Xqt)) 1156 (void) error(21); 1157 } while ((c = getchr()) != '\n'); 1158 *p1++ = 0; 1159 if (Xqt) 1160 if (comm == 'f') { 1161 --Xqt; 1162 (void) error(57); 1163 } 1164 else 1165 return; 1166 if (savedfile[0] == 0 || comm == 'e' || comm == 'f') { 1167 p1 = savedfile; 1168 p2 = file; 1169 while (*p1++ = *p2++) 1170 ; 1171 } 1172 red(file); 1173 } 1174 1175 1176 static void 1177 exfile(void) 1178 { 1179 #ifdef NULLS 1180 int c; 1181 #endif 1182 1183 #ifndef RESEARCH 1184 if (oldmask) { 1185 umask(oldmask); 1186 oldmask = 0; 1187 } 1188 #endif 1189 eclose(io); 1190 io = -1; 1191 if (vflag) { 1192 putd(); 1193 putchr('\n'); 1194 #ifdef NULLS 1195 if (nulls) { 1196 c = count; 1197 count = nulls; 1198 nulls = 0; 1199 putd(); 1200 puts(gettext(" nulls replaced by '\\0'")); 1201 count = c; 1202 } 1203 #endif 1204 } 1205 } 1206 1207 static void 1208 onintr(int sig) 1209 { 1210 signal(SIGINT, onintr); 1211 putchr('\n'); 1212 lastc = '\n'; 1213 globflg = 0; 1214 if (funlink) unlink(funny); /* remove tmp file */ 1215 /* if interrupted a read, only part of file may be in buffer */ 1216 if (readflg) { 1217 sprintf(tstring, "\007read may be incomplete - beware!\007"); 1218 puts(gettext(tstring)); 1219 fchange = 0; 1220 } 1221 (void) error(26); 1222 } 1223 1224 static void 1225 onhup(int sig) 1226 { 1227 signal(SIGINT, SIG_IGN); 1228 signal(SIGHUP, SIG_IGN); 1229 /* 1230 * if there are lines in file and file was not written 1231 * since last update, save in ed.hup, or $HOME/ed.hup 1232 */ 1233 if (dol > zero && fchange == 1) { 1234 addr1 = zero+1; 1235 addr2 = dol; 1236 io = creat("ed.hup", 1237 S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH); 1238 if (io < 0 && home) { 1239 char *fn; 1240 1241 fn = (char *)calloc(strlen(home) + 8, sizeof (char)); 1242 if (fn) { 1243 strcpy(fn, home); 1244 strcat(fn, "/ed.hup"); 1245 io = creat(fn, S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP 1246 |S_IROTH|S_IWOTH); 1247 free(fn); 1248 } 1249 } 1250 if (io > 0) 1251 putfile(); 1252 } 1253 fchange = 0; 1254 ++errcnt; 1255 quit(sig); 1256 } 1257 1258 static int 1259 error(int code) 1260 { 1261 int c; 1262 1263 if (code == 28 && save28 == 0) { 1264 fchange = 0; 1265 flag28++; 1266 } 1267 readflg = 0; 1268 ++errcnt; 1269 listf = listn = 0; 1270 pflag = 0; 1271 #ifndef RESEARCH 1272 if (oldmask) { 1273 umask(oldmask); 1274 oldmask = 0; 1275 } 1276 #endif 1277 #ifdef NULLS /* Not really nulls, but close enough */ 1278 /* This is a bug because of buffering */ 1279 if (code == 28) /* illegal char. */ 1280 putd(); 1281 #endif 1282 /* Cant open file or file does not exist */ 1283 if ((code == 3) || (code == 68)) { 1284 if (qflg == 0) { 1285 putchr('?'); 1286 puts(file); 1287 } 1288 else 1289 qflg = 0; 1290 } 1291 else 1292 { 1293 putchr('?'); 1294 putchr('\n'); 1295 } 1296 count = 0; 1297 lseek(0, (long)0, 2); 1298 if (globp) 1299 lastc = '\n'; 1300 globp = 0; 1301 peekc = lastc; 1302 if (lastc) 1303 while ((c = getchr()) != '\n' && c != EOF) 1304 ; 1305 if (io) { 1306 eclose(io); 1307 io = -1; 1308 } 1309 xcode = code; 1310 if (hflag) 1311 PUTM(); 1312 if (code == 4) 1313 return (0); /* Non-fatal error. */ 1314 longjmp(savej, 1); 1315 /* NOTREACHED */ 1316 } 1317 1318 static int 1319 getchr(void) 1320 { 1321 char c; 1322 if (lastc = peekc) { 1323 peekc = 0; 1324 return (lastc); 1325 } 1326 if (globp) { 1327 if ((lastc = (unsigned char)*globp++) != 0) 1328 return (lastc); 1329 globp = 0; 1330 return (EOF); 1331 } 1332 if (read(0, &c, 1) <= 0) 1333 return (lastc = EOF); 1334 lastc = (unsigned char)c; 1335 return (lastc); 1336 } 1337 1338 static int 1339 gettty(void) 1340 { 1341 int c; 1342 char *gf; 1343 char *p; 1344 1345 p = linebuf; 1346 gf = globp; 1347 while ((c = getchr()) != '\n') { 1348 if (c == EOF) { 1349 if (gf) 1350 peekc = c; 1351 return (c); 1352 } 1353 if (c == 0) 1354 continue; 1355 *p++ = c; 1356 1357 if (p > &linebuf[LBSIZE-1]) 1358 (void) error(27); 1359 } 1360 *p++ = 0; 1361 if (linebuf[0] == '.' && linebuf[1] == 0) 1362 return (EOF); 1363 1364 /* 1365 * POSIX.2/XPG4 explicitly says no to this: 1366 * 1367 * in Solaris backslash followed by special character "." is 1368 * special character "." itself; (so terminating input mode can be 1369 * "\.\n"). 1370 * 1371 * however, POSIX2/XPG4 says, input mode is terminated by 1372 * entering line consisting of only 2 characters: ".\n" 1373 * 1374 * if (linebuf[0]=='\\' && linebuf[1]=='.' && linebuf[2]==0) { 1375 * linebuf[0] = '.'; 1376 * linebuf[1] = 0; 1377 * } 1378 */ 1379 return (0); 1380 } 1381 1382 static int 1383 getfile(void) 1384 { 1385 char c; 1386 char *lp, *fp; 1387 1388 lp = linebuf; 1389 fp = nextip; 1390 do { 1391 if (--ninbuf < 0) { 1392 if ((ninbuf = read(io, genbuf, LBSIZE)-1) < 0) 1393 if (lp > linebuf) { 1394 puts(gettext("'\\n' appended")); 1395 *genbuf = '\n'; 1396 } 1397 else 1398 return (EOF); 1399 if (crflag == -1) { 1400 if (isencrypt(genbuf, ninbuf + 1)) 1401 crflag = 2; 1402 else 1403 crflag = -2; 1404 } 1405 fp = genbuf; 1406 if (crflag > 0) 1407 if (run_crypt(count, genbuf, ninbuf+1, perm) == -1) 1408 (void) error(63); 1409 } 1410 if (lp >= &linebuf[LBSIZE]) { 1411 lastc = '\n'; 1412 (void) error(27); 1413 } 1414 if ((*lp++ = c = *fp++) == 0) { 1415 #ifdef NULLS 1416 lp[-1] = '\\'; 1417 *lp++ = '0'; 1418 nulls++; 1419 #else 1420 lp--; 1421 continue; 1422 #endif 1423 } 1424 count++; 1425 } while (c != '\n'); 1426 *--lp = 0; 1427 nextip = fp; 1428 if (fss.Ffill && fss.Flim && lenchk(linebuf, &fss) < 0) { 1429 write(1, gettext("line too long: lno = "), 1430 strlen(gettext("line too long: lno = "))); 1431 ccount = count; 1432 count = (++dot-zero)&077777; 1433 dot--; 1434 putd(); 1435 count = ccount; 1436 putchr('\n'); 1437 } 1438 return (0); 1439 } 1440 1441 static void 1442 putfile(void) 1443 { 1444 int n; 1445 LINE a1; 1446 char *fp, *lp; 1447 int nib; 1448 1449 nib = LBSIZE; 1450 fp = genbuf; 1451 a1 = addr1; 1452 do { 1453 lp = getaline(a1++->cur); 1454 if (fss.Ffill && fss.Flim && lenchk(linebuf, &fss) < 0) { 1455 write(1, gettext("line too long: lno = "), 1456 strlen(gettext("line too long: lno = "))); 1457 ccount = count; 1458 count = (a1-zero-1)&077777; 1459 putd(); 1460 count = ccount; 1461 putchr('\n'); 1462 } 1463 for (;;) { 1464 if (--nib < 0) { 1465 n = fp-genbuf; 1466 if (kflag) 1467 if (run_crypt(count-n, genbuf, n, perm) == -1) 1468 (void) error(63); 1469 if (write(io, genbuf, n) != n) 1470 (void) error(29); 1471 nib = LBSIZE - 1; 1472 fp = genbuf; 1473 } 1474 if (dol->cur == 0L)break; /* Allow write of null file */ 1475 count++; 1476 if ((*fp++ = *lp++) == 0) { 1477 fp[-1] = '\n'; 1478 break; 1479 } 1480 } 1481 } while (a1 <= addr2); 1482 n = fp-genbuf; 1483 if (kflag) 1484 if (run_crypt(count-n, genbuf, n, perm) == -1) 1485 (void) error(63); 1486 if (write(io, genbuf, n) != n) 1487 (void) error(29); 1488 } 1489 1490 static void 1491 append(int (*f)(void), LINE a) 1492 { 1493 LINE a1, a2, rdot; 1494 long tl; 1495 1496 nline = 0; 1497 dot = a; 1498 while ((*f)() == 0) { 1499 if (dol >= endcore) { 1500 if ((int)sbrk(512 * sizeof (struct lin)) == -1) { 1501 lastc = '\n'; 1502 (void) error(30); 1503 } 1504 endcore += 512; 1505 } 1506 tl = putline(); 1507 nline++; 1508 a1 = ++dol; 1509 a2 = a1+1; 1510 rdot = ++dot; 1511 while (a1 > rdot) 1512 (--a2)->cur = (--a1)->cur; 1513 rdot->cur = tl; 1514 } 1515 } 1516 1517 static void 1518 unixcom(void) 1519 { 1520 void (*savint)(); 1521 pid_t pid, rpid; 1522 int retcode; 1523 static char savcmd[LBSIZE]; /* last command */ 1524 char curcmd[LBSIZE]; /* current command */ 1525 char *psavcmd, *pcurcmd, *psavedfile; 1526 int endflg = 1, shflg = 0; 1527 wchar_t c; 1528 int len; 1529 1530 setnoaddr(); 1531 if (rflg) 1532 (void) error(6); 1533 pcurcmd = curcmd; 1534 /* read command til end */ 1535 1536 /* 1537 * a '!' found in beginning of command is replaced with the saved 1538 * command. a '%' found in command is replaced with the current 1539 * filename 1540 */ 1541 1542 c = getchr(); 1543 if (c == '!') { 1544 if (savcmd[0] == 0) 1545 (void) error(56); 1546 else { 1547 psavcmd = savcmd; 1548 while (*pcurcmd++ = *psavcmd++) 1549 ; 1550 --pcurcmd; 1551 shflg = 1; 1552 } 1553 } else 1554 UNGETC(c); /* put c back */ 1555 while (endflg == 1) { 1556 while ((c = get_wchr()) != '\n' && c != '%' && c != '\\') { 1557 if ((len = wctomb(pcurcmd, c)) <= 0) { 1558 *pcurcmd = (unsigned char)c; 1559 len = 1; 1560 } 1561 pcurcmd += len; 1562 } 1563 1564 if (c == '%') { 1565 if (savedfile[0] == 0) 1566 (void) error(21); 1567 else { 1568 psavedfile = savedfile; 1569 while (pcurcmd < curcmd + LBSIZE && 1570 (*pcurcmd++ = *psavedfile++)) 1571 ; 1572 --pcurcmd; 1573 shflg = 1; 1574 } 1575 } else if (c == '\\') { 1576 c = get_wchr(); 1577 if (c != '%') 1578 *pcurcmd++ = '\\'; 1579 if ((len = wctomb(pcurcmd, c)) <= 0) { 1580 *pcurcmd = (unsigned char)c; 1581 len = 1; 1582 } 1583 pcurcmd += len; 1584 } 1585 else 1586 /* end of command hit */ 1587 endflg = 0; 1588 } 1589 *pcurcmd++ = 0; 1590 if (shflg == 1) 1591 puts(curcmd); 1592 /* save command */ 1593 strcpy(savcmd, curcmd); 1594 1595 if ((pid = fork()) == 0) { 1596 signal(SIGHUP, oldhup); 1597 signal(SIGQUIT, oldquit); 1598 close(tfile); 1599 execlp(_PATH_BSHELL, "sh", "-c", curcmd, NULL); 1600 exit(0100); 1601 } 1602 savint = signal(SIGINT, SIG_IGN); 1603 while ((rpid = wait(&retcode)) != pid && rpid != (pid_t)-1) 1604 ; 1605 signal(SIGINT, savint); 1606 if (vflag) puts("!"); 1607 } 1608 1609 static void 1610 quit(int sig) 1611 { 1612 if (vflag && fchange) { 1613 fchange = 0; 1614 if (flag28) { 1615 flag28 = 0; 1616 (void) error(62); 1617 } 1618 1619 /* 1620 * For case where user reads in BOTH a good 1621 * file & a bad file 1622 */ 1623 (void) error(1); 1624 } 1625 unlink(tfname); 1626 if (kflag) 1627 crypt_close(perm); 1628 if (xtflag) 1629 crypt_close(tperm); 1630 exit(errcnt? 2: 0); 1631 } 1632 1633 static void 1634 delete(void) 1635 { 1636 setdot(); 1637 newline(); 1638 nonzero(); 1639 if (!globflg) 1640 save(); 1641 rdelete(addr1, addr2); 1642 } 1643 1644 static void 1645 rdelete(LINE ad1, LINE ad2) 1646 { 1647 LINE a1, a2, a3; 1648 1649 a1 = ad1; 1650 a2 = ad2+1; 1651 a3 = dol; 1652 dol -= a2 - a1; 1653 do { 1654 (a1++)->cur = (a2++)->cur; 1655 } while (a2 <= a3); 1656 a1 = ad1; 1657 if (a1 > dol) 1658 a1 = dol; 1659 dot = a1; 1660 fchange = 1; 1661 } 1662 1663 static void 1664 gdelete(void) 1665 { 1666 LINE a1, a2, a3; 1667 1668 a3 = dol; 1669 for (a1 = zero+1; (a1->cur&01) == 0; a1++) 1670 if (a1 >= a3) 1671 return; 1672 for (a2 = a1 + 1; a2 <= a3; ) { 1673 if (a2->cur & 01) { 1674 a2++; 1675 dot = a1; 1676 } else 1677 (a1++)->cur = (a2++)->cur; 1678 } 1679 dol = a1-1; 1680 if (dot > dol) 1681 dot = dol; 1682 fchange = 1; 1683 } 1684 1685 static char * 1686 getaline(long tl) 1687 { 1688 char *bp, *lp; 1689 int nl; 1690 1691 lp = linebuf; 1692 bp = getblock(tl, READ); 1693 nl = nleft; 1694 tl &= ~0377; 1695 while (*lp++ = *bp++) 1696 if (--nl == 0) { 1697 bp = getblock(tl += 0400, READ); 1698 nl = nleft; 1699 } 1700 return (linebuf); 1701 } 1702 1703 static long 1704 putline(void) 1705 { 1706 char *bp, *lp; 1707 int nl; 1708 long tl; 1709 1710 fchange = 1; 1711 lp = linebuf; 1712 tl = tline; 1713 bp = getblock(tl, WRITE); 1714 nl = nleft; 1715 tl &= ~0377; 1716 while (*bp = *lp++) { 1717 if (*bp++ == '\n') { 1718 *--bp = 0; 1719 linebp = lp; 1720 break; 1721 } 1722 if (--nl == 0) { 1723 bp = getblock(tl += 0400, WRITE); 1724 nl = nleft; 1725 } 1726 } 1727 nl = tline; 1728 tline += (((lp-linebuf)+03)>>1)&077776; 1729 return (nl); 1730 } 1731 1732 static char * 1733 getblock(long atl, long iof) 1734 { 1735 int bno, off; 1736 char *p1, *p2; 1737 int n; 1738 1739 bno = atl >> 8; 1740 off = (atl<<1)&0774; 1741 1742 /* bno is limited to 16 bits */ 1743 if (bno >= 65535) { 1744 lastc = '\n'; 1745 (void) error(31); 1746 } 1747 nleft = 512 - off; 1748 if (bno == iblock) { 1749 ichanged |= iof; 1750 return (ibuff+off); 1751 } 1752 if (bno == oblock) 1753 return (obuff+off); 1754 if (iof == READ) { 1755 if (ichanged) { 1756 if (xtflag) 1757 if (run_crypt(0L, ibuff, 512, tperm) == -1) 1758 (void) error(63); 1759 blkio(iblock, ibuff, write); 1760 } 1761 ichanged = 0; 1762 iblock = bno; 1763 blkio(bno, ibuff, read); 1764 if (xtflag) 1765 if (run_crypt(0L, ibuff, 512, tperm) == -1) 1766 (void) error(63); 1767 return (ibuff+off); 1768 } 1769 if (oblock >= 0) { 1770 if (xtflag) { 1771 p1 = obuff; 1772 p2 = crbuf; 1773 n = 512; 1774 while (n--) 1775 *p2++ = *p1++; 1776 if (run_crypt(0L, crbuf, 512, tperm) == -1) 1777 (void) error(63); 1778 blkio(oblock, crbuf, write); 1779 } else 1780 blkio(oblock, obuff, write); 1781 } 1782 oblock = bno; 1783 return (obuff+off); 1784 } 1785 1786 static void 1787 blkio(int b, char *buf, ssize_t (*iofcn)()) 1788 { 1789 lseek(tfile, (long)b<<9, 0); 1790 if ((*iofcn)(tfile, buf, 512) != 512) { 1791 if (dol != zero) 1792 (void) error(32); /* Bypass this if writing null file */ 1793 } 1794 } 1795 1796 static void 1797 init(void) 1798 { 1799 long *markp; 1800 mode_t omask; 1801 1802 if (tfile != -1) { 1803 (void) close(tfile); 1804 (void) unlink(tfname); 1805 } 1806 1807 tline = 2; 1808 for (markp = names; markp < &names[26]; ) 1809 *markp++ = 0L; 1810 subnewa = 0L; 1811 anymarks = 0; 1812 iblock = -1; 1813 oblock = -1; 1814 ichanged = 0; 1815 initflg = 1; 1816 omask = umask(0); 1817 1818 if ((tfile = open(tfname, O_CREAT|O_EXCL|O_RDWR, 1819 S_IRUSR|S_IWUSR)) < 0) { 1820 puts(gettext(msgtab[70])); 1821 exit(2); 1822 } 1823 1824 umask(omask); 1825 if (xflag) { 1826 xtflag = 1; 1827 if (tpermflag) 1828 (void) crypt_close(tperm); 1829 tpermflag = 1; 1830 if (makekey(tperm)) { 1831 xtflag = 0; 1832 puts(gettext(msgtab[65])); 1833 } 1834 } 1835 brk((char *)fendcore); 1836 dot = zero = dol = savdot = savdol = fendcore; 1837 flag28 = save28 = 0; 1838 endcore = fendcore - sizeof (struct lin); 1839 } 1840 1841 static void 1842 global(int k) 1843 { 1844 char *gp; 1845 wchar_t l; 1846 char multic[MB_LEN_MAX]; 1847 wchar_t c; 1848 LINE a1; 1849 char globuf[LBSIZE]; 1850 int n; 1851 int len; 1852 1853 if (globp) 1854 (void) error(33); 1855 setall(); 1856 nonzero(); 1857 if ((n = _mbftowc(multic, &l, getchr, &peekc)) <= 0) 1858 (void) error(67); 1859 if (l == '\n') 1860 (void) error(19); 1861 save(); 1862 comple(l); 1863 gp = globuf; 1864 while ((c = get_wchr()) != '\n') { 1865 if (c == EOF) 1866 (void) error(19); 1867 1868 /* '\\' has special meaning only if preceding a '\n' */ 1869 if (c == '\\') { 1870 c = get_wchr(); 1871 if (c != '\n') 1872 *gp++ = '\\'; 1873 } 1874 if ((gp + (unsigned int)MB_CUR_MAX) >= &globuf[LBSIZE-1]) 1875 (void) error(34); 1876 if ((len = wctomb(gp, c)) <= 0) { 1877 *gp = (unsigned char)c; 1878 len = 1; 1879 } 1880 gp += len; 1881 } 1882 if (gp == globuf) 1883 *gp++ = 'p'; 1884 *gp++ = '\n'; 1885 *gp++ = 0; 1886 for (a1 = zero; a1 <= dol; a1++) { 1887 a1->cur &= ~01; 1888 if (a1 >= addr1 && a1 <= addr2 && execute(0, a1) == k) 1889 a1->cur |= 01; 1890 } 1891 /* 1892 * Special case: g/.../d (avoid n^2 algorithm) 1893 */ 1894 if (globuf[0] == 'd' && globuf[1] == '\n' && globuf[2] == '\0') { 1895 gdelete(); 1896 return; 1897 } 1898 for (a1 = zero; a1 <= dol; a1++) { 1899 if (a1->cur & 01) { 1900 a1->cur &= ~01; 1901 dot = a1; 1902 globp = globuf; 1903 globflg = 1; 1904 commands(); 1905 globflg = 0; 1906 a1 = zero; 1907 } 1908 } 1909 } 1910 1911 static void 1912 join(void) 1913 { 1914 char *gp, *lp; 1915 LINE a1; 1916 1917 if (addr1 == addr2) 1918 return; 1919 gp = genbuf; 1920 for (a1 = addr1; a1 <= addr2; a1++) { 1921 lp = getaline(a1->cur); 1922 while (*gp = *lp++) 1923 if (gp++ > &genbuf[LBSIZE-1]) 1924 (void) error(27); 1925 } 1926 lp = linebuf; 1927 gp = genbuf; 1928 while (*lp++ = *gp++) 1929 ; 1930 addr1->cur = putline(); 1931 if (addr1 < addr2) 1932 rdelete(addr1+1, addr2); 1933 dot = addr1; 1934 } 1935 1936 static void 1937 substitute(int inglob) 1938 { 1939 int nl; 1940 LINE a1; 1941 long *markp; 1942 int ingsav; /* For saving arg. */ 1943 1944 ingsav = inglob; 1945 ocerr2 = 0; 1946 gsubf = compsub(); 1947 for (a1 = addr1; a1 <= addr2; a1++) { 1948 if (execute(0, a1) == 0) 1949 continue; 1950 numpass = 0; 1951 ocerr1 = 0; 1952 inglob |= 01; 1953 dosub(); 1954 if (gsubf) { 1955 while (*loc2) { 1956 if (execute(1, (LINE)0) == 0) 1957 break; 1958 dosub(); 1959 } 1960 } 1961 if (ocerr1 == 0)continue; /* Don't put out-not changed. */ 1962 subnewa = putline(); 1963 a1->cur &= ~01; 1964 if (anymarks) { 1965 for (markp = names; markp < &names[26]; markp++) 1966 if (*markp == a1->cur) 1967 *markp = subnewa; 1968 } 1969 a1->cur = subnewa; 1970 append(getsub, a1); 1971 nl = nline; 1972 a1 += nl; 1973 addr2 += nl; 1974 } 1975 if (ingsav) 1976 return; /* Was in global-no error msg allowed. */ 1977 if (inglob == 0) 1978 (void) error(35); /* Not in global, but not found. */ 1979 if (ocerr2 == 0) 1980 (void) error(35); /* RE found, but occurrence match failed. */ 1981 } 1982 1983 static int 1984 compsub(void) 1985 { 1986 int c; 1987 wchar_t seof; 1988 char *p; 1989 char multic[MB_LEN_MAX]; 1990 int n; 1991 static char remem[RHSIZE]; 1992 static int remflg = -1; 1993 int i; 1994 1995 if ((n = _mbftowc(multic, &seof, getchr, &peekc)) <= 0) 1996 (void) error(67); 1997 if (seof == '\n' || seof == ' ') 1998 (void) error(36); 1999 comple(seof); 2000 p = rhsbuf; 2001 for (;;) { 2002 wchar_t cl; 2003 if ((n = _mbftowc(multic, &cl, getchr, &peekc)) <= 0) 2004 (void) error(67); 2005 if (cl == '\\') { 2006 *p++ = '\\'; 2007 if (p >= &rhsbuf[RHSIZE]) 2008 (void) error(38); 2009 if ((n = _mbftowc(multic, &cl, getchr, &peekc)) <= 0) 2010 (void) error(67); 2011 } else if (cl == '\n') { 2012 if (nodelim == 1) { 2013 nodelim = 0; 2014 (void) error(36); 2015 } 2016 if (!(globp && globp[0])) { 2017 UNGETC('\n'); 2018 pflag++; 2019 break; 2020 } 2021 } else if (cl == seof) 2022 break; 2023 if (p + n > &rhsbuf[RHSIZE]) 2024 (void) error(38); 2025 (void) strncpy(p, multic, n); 2026 p += n; 2027 } 2028 *p++ = 0; 2029 if (rhsbuf[0] == '%' && rhsbuf[1] == 0) 2030 /* 2031 * If there isn't a remembered string, it is an error; 2032 * otherwise the right hand side is the previous right 2033 * hand side. 2034 */ 2035 2036 if (remflg == -1) 2037 (void) error(55); 2038 else 2039 strcpy(rhsbuf, remem); 2040 else { 2041 strcpy(remem, rhsbuf); 2042 remflg = 0; 2043 } 2044 c = 0; 2045 peekc = getchr(); /* Gets char after third delimiter. */ 2046 if (peekc == 'g') { 2047 c = LBSIZE; peekc = 0; 2048 } 2049 if (peekc >= '1' && peekc <= '9') { 2050 c = peekc-'0'; 2051 peekc = 0; /* Allows getchr() to get next char. */ 2052 while (1) { 2053 i = getchr(); 2054 if (i < '0' || i > '9') 2055 break; 2056 c = c*10 + i-'0'; 2057 if (c > LBSIZE-1) 2058 (void) error(20); /* "Illegal suffix" */ 2059 } 2060 peekc = i; /* Effectively an unget. */ 2061 } 2062 newline(); 2063 return (c); 2064 2065 /* 2066 * Returns occurrence value. 0 & 1 both do first occurrence 2067 * only: c = 0 if ordinary substitute; c = 1 2068 * if use 1 in global sub(s/a/b/1). 0 in global form is illegal. 2069 */ 2070 } 2071 2072 static int 2073 getsub(void) 2074 { 2075 char *p1, *p2; 2076 2077 p1 = linebuf; 2078 if ((p2 = linebp) == 0) 2079 return (EOF); 2080 while (*p1++ = *p2++) 2081 ; 2082 linebp = 0; 2083 return (0); 2084 } 2085 2086 static void 2087 dosub(void) 2088 { 2089 char *lp, *sp, *rp; 2090 int c; 2091 2092 if (gsubf > 0 && gsubf < LBSIZE) { 2093 numpass++; 2094 if (gsubf != numpass) 2095 return; 2096 } 2097 ocerr1++; 2098 ocerr2++; 2099 lp = linebuf; 2100 sp = genbuf; 2101 rp = rhsbuf; 2102 while (lp < loc1) 2103 *sp++ = *lp++; 2104 while (c = *rp++) { 2105 if (c == '&') { 2106 sp = place(sp, loc1, loc2); 2107 continue; 2108 } else if (c == '\\') { 2109 c = *rp++; 2110 if (c >= '1' && c < nbra + '1') { 2111 sp = place(sp, braslist[c-'1'], 2112 braelist[c-'1']); 2113 continue; 2114 } 2115 } 2116 *sp++ = c; 2117 if (sp >= &genbuf[LBSIZE]) 2118 (void) error(27); 2119 } 2120 lp = loc2; 2121 loc2 = sp - genbuf + linebuf; 2122 while (*sp++ = *lp++) 2123 if (sp >= &genbuf[LBSIZE]) 2124 (void) error(27); 2125 lp = linebuf; 2126 sp = genbuf; 2127 while (*lp++ = *sp++) 2128 ; 2129 } 2130 2131 static char * 2132 place(char *sp, char *l1, char *l2) 2133 { 2134 2135 while (l1 < l2) { 2136 *sp++ = *l1++; 2137 if (sp >= &genbuf[LBSIZE]) 2138 (void) error(27); 2139 } 2140 return (sp); 2141 } 2142 2143 static void 2144 comple(wchar_t seof) 2145 { 2146 int cclass = 0; 2147 wchar_t c; 2148 int n; 2149 char *cp = genbuf; 2150 char multic[MB_LEN_MAX]; 2151 2152 while (1) { 2153 if ((n = _mbftowc(multic, &c, getchr, &peekc)) < 0) 2154 error1(67); 2155 if (n == 0 || c == '\n') { 2156 if (cclass) 2157 error1(49); 2158 else 2159 break; 2160 } 2161 if (c == seof && !cclass) 2162 break; 2163 if (cclass && c == ']') { 2164 cclass = 0; 2165 if (cp > &genbuf[LBSIZE-1]) 2166 error1(50); 2167 *cp++ = ']'; 2168 continue; 2169 } 2170 if (c == '[' && !cclass) { 2171 cclass = 1; 2172 if (cp > &genbuf[LBSIZE-1]) 2173 error1(50); 2174 *cp++ = '['; 2175 if ((n = _mbftowc(multic, &c, getchr, &peekc)) < 0) 2176 error1(67); 2177 if (n == 0 || c == '\n') 2178 error1(49); 2179 } 2180 if (c == '\\' && !cclass) { 2181 if (cp > &genbuf[LBSIZE-1]) 2182 error1(50); 2183 *cp++ = '\\'; 2184 if ((n = _mbftowc(multic, &c, getchr, &peekc)) < 0) 2185 error1(67); 2186 if (n == 0 || c == '\n') 2187 error1(36); 2188 } 2189 if (cp + n > &genbuf[LBSIZE-1]) 2190 error1(50); 2191 (void) strncpy(cp, multic, n); 2192 cp += n; 2193 } 2194 *cp = '\0'; 2195 if (n != 0 && c == '\n') 2196 UNGETC('\n'); 2197 if (n == 0 || c == '\n') 2198 nodelim = 1; 2199 2200 /* 2201 * NULL RE: do not compile a null regular expression; but process 2202 * input with last regular expression encountered 2203 */ 2204 2205 if (genbuf[0] != '\0') { 2206 if (expbuf) 2207 free(expbuf); 2208 expbuf = compile(genbuf, (char *)0, (char *)0); 2209 } 2210 if (regerrno) 2211 error1(regerrno); 2212 } 2213 2214 static void 2215 move(int cflag) 2216 { 2217 LINE adt, ad1, ad2; 2218 2219 setdot(); 2220 nonzero(); 2221 if ((adt = address()) == 0) 2222 (void) error(39); 2223 newline(); 2224 if (!globflg) save(); 2225 if (cflag) { 2226 ad1 = dol; 2227 append(getcopy, ad1++); 2228 ad2 = dol; 2229 } else { 2230 ad2 = addr2; 2231 for (ad1 = addr1; ad1 <= ad2; ) 2232 (ad1++)->cur &= ~01; 2233 ad1 = addr1; 2234 } 2235 ad2++; 2236 if (adt < ad1) { 2237 dot = adt + (ad2-ad1); 2238 if ((++adt) == ad1) 2239 return; 2240 reverse(adt, ad1); 2241 reverse(ad1, ad2); 2242 reverse(adt, ad2); 2243 } else if (adt >= ad2) { 2244 dot = adt++; 2245 reverse(ad1, ad2); 2246 reverse(ad2, adt); 2247 reverse(ad1, adt); 2248 } else 2249 (void) error(39); 2250 fchange = 1; 2251 } 2252 2253 static void 2254 reverse(LINE a1, LINE a2) 2255 { 2256 long t; 2257 2258 for (;;) { 2259 t = (--a2)->cur; 2260 if (a2 <= a1) 2261 return; 2262 a2->cur = a1->cur; 2263 (a1++)->cur = t; 2264 } 2265 } 2266 2267 static int 2268 getcopy(void) 2269 { 2270 2271 if (addr1 > addr2) 2272 return (EOF); 2273 (void) getaline((addr1++)->cur); 2274 return (0); 2275 } 2276 2277 2278 /* 2279 * Handles error code returned from comple() routine: regular expression 2280 * compile and match routines 2281 */ 2282 2283 static void 2284 error1(int code) 2285 { 2286 nbra = 0; 2287 (void) error(code); 2288 } 2289 2290 2291 static int 2292 execute(int gf, LINE addr) 2293 { 2294 char *p1; 2295 int c; 2296 2297 for (c = 0; c < nbra; c++) { 2298 braslist[c] = 0; 2299 braelist[c] = 0; 2300 } 2301 if (gf) 2302 locs = p1 = loc2; 2303 else { 2304 if (addr == zero) 2305 return (0); 2306 p1 = getaline(addr->cur); 2307 locs = 0; 2308 } 2309 return (step(p1, expbuf)); 2310 } 2311 2312 2313 static void 2314 putd() 2315 { 2316 int r; 2317 2318 r = (int)(count%10); 2319 count /= 10; 2320 if (count) 2321 putd(); 2322 putchr(r + '0'); 2323 } 2324 2325 2326 int 2327 puts(const char *sp) 2328 { 2329 int n; 2330 wchar_t c; 2331 int sz, i; 2332 if (fss.Ffill && (listf == 0)) { 2333 2334 /* deliberate attempt to remove constness of sp because */ 2335 /* it needs to be expanded */ 2336 2337 if ((i = expnd((char *)sp, funny, &sz, &fss)) == -1) { 2338 write(1, funny, fss.Flim & 0377); 2339 putchr('\n'); 2340 write(1, gettext("too long"), 2341 strlen(gettext("too long"))); 2342 } 2343 else 2344 write(1, funny, sz); 2345 putchr('\n'); 2346 if (i == -2) 2347 write(1, gettext("tab count\n"), 2348 strlen(gettext("tab count\n"))); 2349 return (0); 2350 } 2351 col = 0; 2352 while (*sp) { 2353 n = mbtowc(&c, sp, MB_LEN_MAX); 2354 if (listf) { 2355 if (n < 1) 2356 (void) error(28); 2357 else if (n == 1) 2358 putchr((unsigned char)*sp++); 2359 else { 2360 sp += n; 2361 putwchr(c); 2362 } 2363 } else { 2364 putchr((unsigned char)*sp++); 2365 } 2366 } 2367 #ifndef XPG6 2368 if (listf) 2369 putchr('$'); /* end of line is marked with a $ */ 2370 #else 2371 if (listf) { 2372 /* xpg6 - ensure that the end of line $ is not preceeded with a "\" */ 2373 /* by doing a putchr() with listf=0, thereby avoiding the $ case */ 2374 /* statement in putchr() */ 2375 listf = 0; 2376 putchr('$'); /* end of line is marked with a $ */ 2377 listf++; 2378 } 2379 #endif 2380 putchr('\n'); 2381 return (1); 2382 } 2383 2384 2385 static void 2386 putwchr(wchar_t ac) 2387 { 2388 char buf[MB_LEN_MAX], *p; 2389 char *lp; 2390 wchar_t c; 2391 short len; 2392 2393 lp = linp; 2394 c = ac; 2395 if (listf) { 2396 if (!iswprint(c)) { 2397 p = &buf[0]; 2398 if ((len = wctomb(p, c)) <= 0) { 2399 *p = (unsigned char)c; 2400 len = 1; 2401 }; 2402 while (len--) { 2403 if (col + 4 >= 72) { 2404 col = 0; 2405 *lp++ = '\\'; 2406 *lp++ = '\n'; 2407 } 2408 (void) sprintf(lp, "\\%03o", 2409 *(unsigned char *)p++); 2410 col += 4; 2411 lp += 4; 2412 } 2413 } else { 2414 if ((len = wcwidth(c)) <= 0) 2415 len = 0; 2416 if (col + len >= 72) { 2417 col = 0; 2418 *lp++ = '\\'; 2419 *lp++ = '\n'; 2420 } 2421 col += len; 2422 if ((len = wctomb(lp, c)) <= 0) { 2423 *lp = (unsigned char)c; 2424 len = 1; 2425 } 2426 lp += len; 2427 } 2428 } else { 2429 if ((len = wctomb(lp, c)) <= 0) { 2430 *lp = (unsigned char)c; 2431 len = 1; 2432 } 2433 lp += len; 2434 } 2435 if (c == '\n' || lp >= &line[64]) { 2436 linp = line; 2437 len = lp - line; 2438 write(1, line, len); 2439 return; 2440 } 2441 linp = lp; 2442 } 2443 2444 2445 static void 2446 putchr(unsigned char c) 2447 { 2448 char *lp; 2449 int len; 2450 2451 lp = linp; 2452 if (listf && c != '\n') { 2453 switch (c) { 2454 case '\\' : 2455 *lp++ = '\\'; 2456 *lp++ = '\\'; 2457 col += 2; 2458 break; 2459 case '\007' : 2460 *lp++ = '\\'; 2461 *lp++ = 'a'; 2462 col += 2; 2463 break; 2464 case '\b' : 2465 *lp++ = '\\'; 2466 *lp++ = 'b'; 2467 col += 2; 2468 break; 2469 case '\f' : 2470 *lp++ = '\\'; 2471 *lp++ = 'f'; 2472 col += 2; 2473 break; 2474 case '\r' : 2475 *lp++ = '\\'; 2476 *lp++ = 'r'; 2477 col += 2; 2478 break; 2479 case '\t' : 2480 *lp++ = '\\'; 2481 *lp++ = 't'; 2482 col += 2; 2483 break; 2484 case '\v' : 2485 *lp++ = '\\'; 2486 *lp++ = 'v'; 2487 col += 2; 2488 break; 2489 #ifdef XPG6 2490 /* if $ characters are within the line preceed with \ */ 2491 case '$' : 2492 *lp++ = '\\'; 2493 *lp++ = '$'; 2494 col += 2; 2495 break; 2496 #endif 2497 default: 2498 if (isprint(c)) { 2499 *lp++ = c; 2500 col += 1; 2501 } else { 2502 (void) sprintf(lp, "\\%03o", c); 2503 col += 4; 2504 lp += 4; 2505 } 2506 break; 2507 } 2508 2509 /* 2510 * long lines are folded w/ pt of folding indicated by writing 2511 * backslash/newline character 2512 */ 2513 2514 if (col + 1 >= 72) { 2515 col = 0; 2516 *lp++ = '\\'; 2517 *lp++ = '\n'; 2518 } 2519 } else 2520 *lp++ = c; 2521 if (c == '\n' || lp >= &line[64]) { 2522 linp = line; 2523 len = lp - line; 2524 (void) write(1, line, len); 2525 return; 2526 } 2527 linp = lp; 2528 } 2529 2530 2531 static char * 2532 getkey(const char *prompt) 2533 { 2534 struct termio b; 2535 int save; 2536 void (*sig)(); 2537 static char key[KSIZE+1]; 2538 char *p; 2539 int c; 2540 2541 sig = signal(SIGINT, SIG_IGN); 2542 ioctl(0, TCGETA, &b); 2543 save = b.c_lflag; 2544 b.c_lflag &= ~(ECHO | ECHOE | ECHOK | ECHONL); 2545 ioctl(0, TCSETAW, &b); 2546 write(1, gettext(prompt), strlen(gettext(prompt))); 2547 p = key; 2548 while (((c = getchr()) != EOF) && (c != '\n')) { 2549 if (p < &key[KSIZE]) 2550 *p++ = c; 2551 } 2552 *p = 0; 2553 write(1, "\n", 1); 2554 b.c_lflag = save; 2555 ioctl(0, TCSETAW, &b); 2556 signal(SIGINT, sig); 2557 return (key); 2558 } 2559 2560 2561 static void 2562 globaln(int k) 2563 { 2564 char *gp; 2565 int c; 2566 int n; 2567 wchar_t cl; 2568 LINE a1; 2569 int nfirst; 2570 char globuf[LBSIZE]; 2571 char multic[MB_LEN_MAX]; 2572 int len; 2573 int pflag_save = 0; 2574 int listf_save = 0; 2575 int listn_save = 0; 2576 2577 if (globp) 2578 (void) error(33); 2579 setall(); 2580 nonzero(); 2581 if ((n = _mbftowc(multic, &cl, getchr, &peekc)) <= 0) 2582 (void) error(67); 2583 if (cl == '\n') 2584 (void) error(19); 2585 save(); 2586 comple(cl); 2587 for (a1 = zero; a1 <= dol; a1++) { 2588 a1->cur &= ~01; 2589 if (a1 >= addr1 && a1 <= addr2 && execute(0, a1) == k) 2590 a1->cur |= 01; 2591 } 2592 nfirst = 0; 2593 newline(); 2594 /* 2595 * preserve the p, l, and n suffix commands of the G and V 2596 * commands during the interactive section and restore 2597 * on completion of the G and V command. 2598 */ 2599 pflag_save = pflag; 2600 listf_save = listf; 2601 listn_save = listn; 2602 pflag = 0; 2603 listf = 0; 2604 listn = 0; 2605 for (a1 = zero; a1 <= dol; a1++) { 2606 if (a1->cur & 01) { 2607 a1->cur &= ~01; 2608 dot = a1; 2609 puts(getaline(a1->cur)); 2610 if ((c = get_wchr()) == EOF) 2611 (void) error(52); 2612 if (c == 'a' || c == 'i' || c == 'c') 2613 (void) error(53); 2614 if (c == '\n') { 2615 a1 = zero; 2616 continue; 2617 } 2618 if (c != '&') { 2619 gp = globuf; 2620 if ((len = wctomb(gp, c)) <= 0) { 2621 *gp = (unsigned char)c; 2622 len = 1; 2623 } 2624 gp += len; 2625 while ((c = get_wchr()) != '\n') { 2626 2627 /* '\\' has special meaning only if preceding a '\n' */ 2628 if (c == '\\') { 2629 c = get_wchr(); 2630 if (c != '\n') 2631 *gp++ = '\\'; 2632 } 2633 if ((gp + (unsigned int)MB_CUR_MAX) >= 2634 &globuf[LBSIZE-1]) 2635 (void) error(34); 2636 2637 if ((len = wctomb(gp, c)) <= 0) { 2638 *gp = (unsigned char)c; 2639 len = 1; 2640 } 2641 gp += len; 2642 } 2643 *gp++ = '\n'; 2644 *gp++ = 0; 2645 nfirst = 1; 2646 } else if ((c = get_wchr()) != '\n') 2647 (void) error(54); 2648 globp = globuf; 2649 if (nfirst) { 2650 globflg = 1; 2651 commands(); 2652 globflg = 0; 2653 } else 2654 (void) error(56); 2655 globp = 0; 2656 a1 = zero; 2657 } 2658 } 2659 pflag = pflag_save; 2660 listf = listf_save; 2661 listn = listn_save; 2662 } 2663 2664 2665 static int 2666 eopen(char *string, int rw) 2667 { 2668 #define w_or_r(a, b) (rw ? a : b) 2669 int pf[2]; 2670 pid_t i; 2671 int io; 2672 int chcount; /* # of char read. */ 2673 2674 if (rflg) { /* restricted shell */ 2675 if (Xqt) { 2676 Xqt = 0; 2677 (void) error(6); 2678 } 2679 } 2680 if (!Xqt) { 2681 if ((io = open(string, rw)) >= 0) { 2682 if (fflg) { 2683 chcount = read(io, crbuf, LBSIZE); 2684 if (crflag == -1) { 2685 if (isencrypt(crbuf, chcount)) 2686 crflag = 2; 2687 else 2688 crflag = -2; 2689 } 2690 if (crflag > 0) 2691 if (run_crypt(0L, crbuf, chcount, perm) == -1) 2692 (void) error(63); 2693 if (fspec(crbuf, &fss, 0) < 0) { 2694 fss.Ffill = 0; 2695 fflg = 0; 2696 (void) error(4); 2697 } 2698 lseek(io, 0L, 0); 2699 } 2700 } 2701 fflg = 0; 2702 return (io); 2703 } 2704 if (pipe(pf) < 0) 2705 xerr: (void) error(0); 2706 if ((i = fork()) == 0) { 2707 signal(SIGHUP, oldhup); 2708 signal(SIGQUIT, oldquit); 2709 signal(SIGPIPE, oldpipe); 2710 signal(SIGINT, (void (*)()) 0); 2711 close(w_or_r(pf[1], pf[0])); 2712 close(w_or_r(0, 1)); 2713 dup(w_or_r(pf[0], pf[1])); 2714 close(w_or_r(pf[0], pf[1])); 2715 execlp(_PATH_BSHELL, "sh", "-c", string, (char *)0); 2716 exit(1); 2717 } 2718 if (i == (pid_t)-1) 2719 goto xerr; 2720 close(w_or_r(pf[0], pf[1])); 2721 return (w_or_r(pf[1], pf[0])); 2722 } 2723 2724 2725 static void 2726 eclose(int f) 2727 { 2728 close(f); 2729 if (Xqt) 2730 Xqt = 0, wait((int *)0); 2731 } 2732 2733 2734 static void 2735 mkfunny(void) 2736 { 2737 char *p, *p1, *p2; 2738 2739 p2 = p1 = funny; 2740 p = file; 2741 /* 2742 * Go to end of file name 2743 */ 2744 while (*p) 2745 p++; 2746 while (*--p == '/') /* delete trailing slashes */ 2747 *p = '\0'; 2748 /* 2749 * go back to beginning of file 2750 */ 2751 p = file; 2752 /* 2753 * Copy file name to funny setting p2 at 2754 * basename of file. 2755 */ 2756 while (*p1++ = *p) 2757 if (*p++ == '/') 2758 p2 = p1; 2759 /* 2760 * Set p1 to point to basename of tfname. 2761 */ 2762 p1 = strrchr(tfname, '/'); 2763 if (strlen(tfname) > (size_t)6) 2764 p1 = &tfname[strlen(tfname)-6]; 2765 p1++; 2766 *p2 = '\007'; /* add unprintable char for funny a unique name */ 2767 /* 2768 * Copy tfname to file. 2769 */ 2770 while (*++p2 = *p1++) 2771 ; 2772 } 2773 2774 2775 static void 2776 getime(void) /* get modified time of file and save */ 2777 { 2778 if (stat(file, &Fl) < 0) 2779 savtime = 0; 2780 else 2781 savtime = Fl.st_mtime; 2782 } 2783 2784 2785 static void 2786 chktime(void) /* check saved mod time against current mod time */ 2787 { 2788 if (savtime != 0 && Fl.st_mtime != 0) { 2789 if (savtime != Fl.st_mtime) 2790 (void) error(58); 2791 } 2792 } 2793 2794 2795 static void 2796 newtime(void) /* get new mod time and save */ 2797 { 2798 stat(file, &Fl); 2799 savtime = Fl.st_mtime; 2800 } 2801 2802 2803 /* 2804 * restricted - check for '/' in name and delete trailing '/' 2805 */ 2806 static void 2807 red(char *op) 2808 { 2809 char *p; 2810 2811 p = op; 2812 while (*p) 2813 if (*p++ == '/'&& rflg) { 2814 *op = 0; 2815 (void) error(6); 2816 } 2817 /* delete trailing '/' */ 2818 while (p > op) { 2819 if (*--p == '/') 2820 *p = '\0'; 2821 else break; 2822 } 2823 } 2824 2825 2826 /* 2827 * Searches thru beginning of file looking for a string of the form 2828 * <: values... :> 2829 * 2830 * where "values" are 2831 * 2832 * \b ignored 2833 * s<num> sets the Flim to <num> 2834 * t??? sets tab stop stuff 2835 * d ignored 2836 * m<num> ignored 2837 * e ignored 2838 */ 2839 2840 static int 2841 fspec(char line[], struct Fspec *f, int up) 2842 { 2843 struct termio arg; 2844 int havespec, n; 2845 int len; 2846 2847 if (!up) clear(f); 2848 2849 havespec = fsprtn = 0; 2850 for (fsp = line; *fsp && *fsp != '\n'; fsp += len) { 2851 if ((len = mblen(fsp, MB_CUR_MAX)) <= 0) 2852 len = 1; 2853 switch (*fsp) { 2854 2855 case '<': if (havespec) 2856 return (-1); 2857 if (*(fsp+1) == ':') { 2858 havespec = 1; 2859 clear(f); 2860 if (!ioctl(1, TCGETA, &arg) && 2861 ((arg.c_oflag & TAB3) == 2862 TAB3)) 2863 f->Ffill = 1; 2864 fsp++; 2865 } 2866 continue; 2867 2868 case ' ': continue; 2869 2870 case 's': if (havespec && (n = numb()) >= 0) 2871 f->Flim = n; 2872 continue; 2873 2874 case 't': if (havespec) targ(f); 2875 continue; 2876 2877 case 'd': continue; 2878 2879 case 'm': if (havespec) n = numb(); 2880 continue; 2881 2882 case 'e': continue; 2883 case ':': if (!havespec) continue; 2884 if (*(fsp+1) != '>') fsprtn = -1; 2885 return (fsprtn); 2886 2887 default: if (!havespec) continue; 2888 return (-1); 2889 } 2890 } 2891 return (1); 2892 } 2893 2894 2895 static int 2896 numb(void) 2897 { 2898 int n; 2899 2900 n = 0; 2901 while (*++fsp >= '0' && *fsp <= '9') 2902 n = 10*n + *fsp-'0'; 2903 fsp--; 2904 return (n); 2905 } 2906 2907 2908 static void 2909 targ(struct Fspec *f) 2910 { 2911 2912 if (*++fsp == '-') { 2913 if (*(fsp + 1) >= '0' && *(fsp+1) <= '9') tincr(numb(), f); 2914 else tstd(f); 2915 return; 2916 } 2917 if (*fsp >= '0' && *fsp <= '9') { 2918 tlist(f); 2919 return; 2920 } 2921 fsprtn = -1; 2922 fsp--; 2923 } 2924 2925 2926 static void 2927 tincr(int n, struct Fspec *f) 2928 { 2929 int l, i; 2930 2931 l = 1; 2932 for (i = 0; i < 20; i++) 2933 f->Ftabs[i] = l += n; 2934 f->Ftabs[i] = 0; 2935 } 2936 2937 2938 static void 2939 tstd(struct Fspec *f) 2940 { 2941 char std[3]; 2942 2943 std[0] = *++fsp; 2944 if (*(fsp+1) >= '0' && *(fsp+1) <= '9') { 2945 std[1] = *++fsp; 2946 std[2] = '\0'; 2947 } else { 2948 std[1] = '\0'; 2949 } 2950 fsprtn = stdtab(std, f->Ftabs); 2951 } 2952 2953 2954 static void 2955 tlist(struct Fspec *f) 2956 { 2957 int n, last, i; 2958 2959 fsp--; 2960 last = i = 0; 2961 2962 do { 2963 if ((n = numb()) <= last || i >= 20) { 2964 fsprtn = -1; 2965 return; 2966 } 2967 f->Ftabs[i++] = last = n; 2968 } while (*++fsp == ','); 2969 2970 f->Ftabs[i] = 0; 2971 fsp--; 2972 } 2973 2974 2975 static int 2976 expnd(char line[], char buf[], int *sz, struct Fspec *f) 2977 { 2978 char *l, *t; 2979 int b; 2980 2981 l = line - 1; 2982 b = 1; 2983 t = f->Ftabs; 2984 fsprtn = 0; 2985 2986 while (*++l && *l != '\n' && b < 511) { 2987 if (*l == '\t') { 2988 while (*t && b >= *t) 2989 t++; 2990 if (*t == 0) 2991 fsprtn = -2; 2992 do { 2993 buf[b-1] = ' '; 2994 } while (++b < *t); 2995 } else { 2996 buf[b++ - 1] = *l; 2997 } 2998 } 2999 3000 buf[b] = '\0'; 3001 *sz = b; 3002 if (*l != '\0' && *l != '\n') { 3003 buf[b-1] = '\n'; 3004 return (-1); 3005 } 3006 buf[b-1] = *l; 3007 if (f->Flim && (b-1 > (int)f->Flim)) 3008 return (-1); 3009 return (fsprtn); 3010 } 3011 3012 3013 static void 3014 clear(struct Fspec *f) 3015 { 3016 f->Ftabs[0] = f->Fdel = f->Fmov = f->Ffill = 0; 3017 f->Flim = 0; 3018 } 3019 3020 3021 static int 3022 lenchk(char line[], struct Fspec *f) 3023 { 3024 char *l, *t; 3025 int b; 3026 3027 l = line - 1; 3028 b = 1; 3029 t = f->Ftabs; 3030 3031 while (*++l && *l != '\n' && b < 511) { 3032 if (*l == '\t') { 3033 while (*t && b >= *t) 3034 t++; 3035 while (++b < *t) 3036 ; 3037 } else { 3038 b++; 3039 } 3040 } 3041 3042 if ((*l != '\0' && *l != '\n') || (f->Flim && (b-1 > (int)f->Flim))) 3043 return (-1); 3044 return (0); 3045 } 3046 #define NTABS 21 3047 3048 3049 /* 3050 * stdtabs: standard tabs table 3051 * format: option code letter(s), null, tabs, null 3052 */ 3053 3054 static char stdtabs[] = { 3055 'a', 0, 1, 10, 16, 36, 72, 0, /* IBM 370 Assembler */ 3056 'a', '2', 0, 1, 10, 16, 40, 72, 0, /* IBM Assembler alternative */ 3057 'c', 0, 1, 8, 12, 16, 20, 55, 0, /* COBOL, normal */ 3058 'c', '2', 0, 1, 6, 10, 14, 49, 0, /* COBOL, crunched */ 3059 'c', '3', 0, 1, 6, 10, 14, 18, 22, 26, 30, 34, 38, 42, 46, 50, 3060 54, 58, 62, 67, 0, 3061 'f', 0, 1, 7, 11, 15, 19, 23, 0, /* FORTRAN */ 3062 'p', 0, 1, 5, 9, 13, 17, 21, 25, 29, 33, 37, 41, 45, 49, 53, 57, 61, 0, 3063 /* PL/I */ 3064 's', 0, 1, 10, 55, 0, /* SNOBOL */ 3065 'u', 0, 1, 12, 20, 44, 0, /* UNIVAC ASM */ 3066 0 }; 3067 3068 3069 /* 3070 * stdtab: return tab list for any "canned" tab option. 3071 * entry: option points to null-terminated option string 3072 * tabvect points to vector to be filled in 3073 * exit: return (0) if legal, tabvect filled, ending with zero 3074 * return (-1) if unknown option 3075 */ 3076 3077 3078 static int 3079 stdtab(char option[], char tabvect[NTABS]) 3080 { 3081 char *scan; 3082 tabvect[0] = 0; 3083 scan = stdtabs; 3084 while (*scan) { 3085 if (strequal(&scan, option)) { 3086 strcopy(scan, tabvect); 3087 break; 3088 } else 3089 while (*scan++) /* skip over tab specs */ 3090 ; 3091 } 3092 3093 /* later: look up code in /etc/something */ 3094 return (tabvect[0] ? 0 : -1); 3095 } 3096 3097 3098 /* 3099 * strequal: checks strings for equality 3100 * entry: scan1 points to scan pointer, str points to string 3101 * exit: return (1) if equal, return (0) if not 3102 * *scan1 is advanced to next nonzero byte after null 3103 */ 3104 3105 3106 static int 3107 strequal(char **scan1, char *str) 3108 { 3109 char c, *scan; 3110 scan = *scan1; 3111 while ((c = *scan++) == *str && c) 3112 str++; 3113 *scan1 = scan; 3114 if (c == 0 && *str == 0) 3115 return (1); 3116 if (c) 3117 while (*scan++) 3118 ; 3119 *scan1 = scan; 3120 return (0); 3121 } 3122 3123 3124 /* strcopy: copy source to destination */ 3125 3126 3127 static void 3128 strcopy(char *source, char *dest) 3129 { 3130 while (*dest++ = *source++) 3131 ; 3132 } 3133 3134 3135 /* This is called before a buffer modifying command so that the */ 3136 /* current array of line ptrs is saved in sav and dot and dol are saved */ 3137 3138 3139 static void 3140 save(void) 3141 { 3142 LINE i; 3143 int j; 3144 3145 savdot = dot; 3146 savdol = dol; 3147 for (j = 0; j <= 25; j++) 3148 savnames[j] = names[j]; 3149 3150 for (i = zero + 1; i <= dol; i++) 3151 i->sav = i->cur; 3152 initflg = 0; 3153 } 3154 3155 3156 /* The undo command calls this to restore the previous ptr array sav */ 3157 /* and swap with cur - dot and dol are swapped also. This allows user to */ 3158 /* undo an undo */ 3159 3160 3161 static void 3162 undo(void) 3163 { 3164 int j; 3165 long tmp; 3166 LINE i, tmpdot, tmpdol; 3167 3168 tmpdot = dot; dot = savdot; savdot = tmpdot; 3169 tmpdol = dol; dol = savdol; savdol = tmpdol; 3170 /* swap arrays using the greater of dol or savdol as upper limit */ 3171 for (i = zero + 1; i <= ((dol > savdol) ? dol : savdol); i++) { 3172 tmp = i->cur; 3173 i->cur = i->sav; 3174 i->sav = tmp; 3175 } 3176 /* 3177 * If the current text lines are swapped with the 3178 * text lines in the save buffer, then swap the current 3179 * marks with those in the save area. 3180 */ 3181 3182 for (j = 0; j <= 25; j++) { 3183 tmp = names[j]; 3184 names[j] = savnames[j]; 3185 savnames[j] = tmp; 3186 } 3187 } 3188 3189 static wchar_t 3190 get_wchr(void) 3191 { 3192 wchar_t wc; 3193 char multi[MB_LEN_MAX]; 3194 3195 if (_mbftowc(multi, &wc, getchr, &peekc) <= 0) 3196 wc = getchr(); 3197 return (wc); 3198 }