Mercurial > heirloom-ed
comparison ed.c @ 0:1493bea5ac22 0.1
Initial version of the standalone heirloom-ed
author | markus schnalke <meillo@marmaro.de> |
---|---|
date | Mon, 05 Sep 2011 16:31:35 +0200 |
parents | |
children | a09d0630f05b |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 0:1493bea5ac22 |
---|---|
1 /* | |
2 * Editor | |
3 */ | |
4 | |
5 /* | |
6 * Changes by Gunnar Ritter, Freiburg i. Br., Germany, July 2003. | |
7 */ | |
8 /* from Unix 32V /usr/src/cmd/ed.c */ | |
9 /* | |
10 * Copyright(C) Caldera International Inc. 2001-2002. All rights reserved. | |
11 * | |
12 * Redistribution and use in source and binary forms, with or without | |
13 * modification, are permitted provided that the following conditions | |
14 * are met: | |
15 * Redistributions of source code and documentation must retain the | |
16 * above copyright notice, this list of conditions and the following | |
17 * disclaimer. | |
18 * Redistributions in binary form must reproduce the above copyright | |
19 * notice, this list of conditions and the following disclaimer in the | |
20 * documentation and/or other materials provided with the distribution. | |
21 * All advertising materials mentioning features or use of this software | |
22 * must display the following acknowledgement: | |
23 * This product includes software developed or owned by Caldera | |
24 * International, Inc. | |
25 * Neither the name of Caldera International, Inc. nor the names of | |
26 * other contributors may be used to endorse or promote products | |
27 * derived from this software without specific prior written permission. | |
28 * | |
29 * USE OF THE SOFTWARE PROVIDED FOR UNDER THIS LICENSE BY CALDERA | |
30 * INTERNATIONAL, INC. AND CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR | |
31 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED | |
32 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE | |
33 * ARE DISCLAIMED. IN NO EVENT SHALL CALDERA INTERNATIONAL, INC. BE | |
34 * LIABLE FOR ANY DIRECT, INDIRECT INCIDENTAL, SPECIAL, EXEMPLARY, OR | |
35 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF | |
36 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR | |
37 * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, | |
38 * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE | |
39 * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, | |
40 * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
41 */ | |
42 | |
43 #if __GNUC__ >= 3 && __GNUC_MINOR__ >= 4 || __GNUC__ >= 4 | |
44 #define USED __attribute__ ((used)) | |
45 #elif defined __GNUC__ | |
46 #define USED __attribute__ ((unused)) | |
47 #else | |
48 #define USED | |
49 #endif | |
50 #if defined (SU3) | |
51 static const char sccsid[] USED = "@(#)ed_su3.sl 1.99 (gritter) 7/27/06"; | |
52 #elif defined (SUS) | |
53 static const char sccsid[] USED = "@(#)ed_sus.sl 1.99 (gritter) 7/27/06"; | |
54 #elif defined (S42) | |
55 static const char sccsid[] USED = "@(#)ed_s42.sl 1.99 (gritter) 7/27/06"; | |
56 #else /* !SU3, !SUS, !S42 */ | |
57 static const char sccsid[] USED = "@(#)ed.sl 1.99 (gritter) 7/27/06"; | |
58 #endif /* !SU3, !SUS, !S42 */ | |
59 | |
60 #include <sys/types.h> | |
61 #include <sys/stat.h> | |
62 #include <sys/wait.h> | |
63 #include <sys/stat.h> | |
64 #include <fcntl.h> | |
65 #include <unistd.h> | |
66 #include <time.h> | |
67 #include <string.h> | |
68 #include <stdlib.h> | |
69 #include <signal.h> | |
70 #include "sigset.h" | |
71 #include <termios.h> | |
72 #include <setjmp.h> | |
73 #include <libgen.h> | |
74 #include <inttypes.h> | |
75 #include <locale.h> | |
76 #include <wchar.h> | |
77 #include <ctype.h> | |
78 #include <wctype.h> | |
79 #include <limits.h> | |
80 #include <termios.h> | |
81 static int FNSIZE; | |
82 static int LBSIZE; | |
83 static int RHSIZE; | |
84 #define ESIZE 2048 | |
85 static int GBSIZE; | |
86 #undef EOF | |
87 #define EOF -1 | |
88 #define puts(s) xxputs(s) | |
89 #define getline(t, n) xxgetline(t, n) | |
90 | |
91 #if (LONG_MAX > 017777777777L) | |
92 #define MAXCNT 0777777777777777777777L /* 2^63-1 */ | |
93 #else | |
94 #define MAXCNT 017777777777L /* 2^31-1 */ | |
95 #endif | |
96 #define BLKMSK (MAXCNT>>8) /* was 0377 */ | |
97 | |
98 #define READ 0 | |
99 #define WRITE 1 | |
100 #define EXIST 2 | |
101 | |
102 struct tabulator { | |
103 struct tabulator *t_nxt; /* next list element */ | |
104 const char *t_str; /* tabulator string */ | |
105 int t_tab; /* tab stop position */ | |
106 int t_rep; /* repetitive tab count */ | |
107 }; | |
108 | |
109 static int peekc; | |
110 static int lastc; | |
111 static char *savedfile; | |
112 static char *file; | |
113 static struct stat fstbuf; | |
114 static char *linebuf; | |
115 static char *rhsbuf; | |
116 static char expbuf[ESIZE + 4]; | |
117 static long *zero; | |
118 static long *undzero; | |
119 static long *dot; | |
120 static long *unddot; | |
121 static long *dol; | |
122 static long *unddol; | |
123 static long *addr1; | |
124 static long *addr2; | |
125 static char *genbuf; | |
126 static long count; | |
127 static char *linebp; | |
128 static int ninbuf; | |
129 static int io; | |
130 static int ioeof; | |
131 static int pflag; | |
132 static char *wrtemp; | |
133 static uid_t myuid; | |
134 static void (*oldhup)(int); | |
135 static void (*oldquit)(int); | |
136 static void (*oldpipe)(int); | |
137 static int vflag = 1; | |
138 static int listf; | |
139 static int numbf; | |
140 static char *globp; | |
141 static int tfile = -1; | |
142 static long tline; | |
143 static char tfname[64]; | |
144 static char ibuff[512]; | |
145 static int iblock = -1; | |
146 static char obuff[512]; | |
147 static int oblock = -1; | |
148 static int ichanged; | |
149 static int nleft; | |
150 static long *names; | |
151 static long *undnames; | |
152 static int anymarks; | |
153 static int subnewa; | |
154 static int fchange; | |
155 static int wrapp; | |
156 static unsigned nlall = 128; | |
157 static const char *progname; | |
158 static const char *prompt = "*"; | |
159 static int Pflag; | |
160 static int prhelp; | |
161 static const char *prvmsg; | |
162 static int lastsig; | |
163 static int pipid = -1; | |
164 static int readop; | |
165 static int status; | |
166 static int mb_cur_max; | |
167 static int needsub; | |
168 static int insub; | |
169 static struct tabulator *tabstops; | |
170 static int maxlength; | |
171 static int rspec; | |
172 static int Nflag; | |
173 static int bcount = 22; | |
174 static int ocount = 11; | |
175 | |
176 static jmp_buf savej; | |
177 | |
178 static void usage(char, int); | |
179 static void commands(void); | |
180 static long *address(void); | |
181 static void setdot(void); | |
182 static void setall(void); | |
183 static void setnoaddr(void); | |
184 static void nonzero(void); | |
185 static void newline(void); | |
186 static void filename(int); | |
187 static void exfile(void); | |
188 static void onintr(int); | |
189 static void onhup(int); | |
190 static void onpipe(int); | |
191 static void error(const char *); | |
192 static void error2(const char *, const char *); | |
193 static void errput(const char *, const char *); | |
194 static int getchr(void); | |
195 static int gettty(void); | |
196 static long getnum(void); | |
197 static int getfile(void); | |
198 static void putfile(void); | |
199 static int append(int (*)(void), long *); | |
200 static void callunix(void); | |
201 static char *readcmd(void); | |
202 static void quit(int); | |
203 static void delete(void); | |
204 static void rdelete(long *, long *); | |
205 static void gdelete(void); | |
206 static char *getline(long, int); | |
207 static int putline(void); | |
208 static char *getblock(long, long); | |
209 static void blkio(long, char *, int); | |
210 static void init(void); | |
211 static void global(int, int); | |
212 static void globrd(char **, int); | |
213 static void join(void); | |
214 static void substitute(int); | |
215 static int compsub(void); | |
216 static int getsub(void); | |
217 static int dosub(int); | |
218 static int place(int, const char *, const char *); | |
219 static void move(int); | |
220 static void reverse(long *, long *); | |
221 static int getcopy(void); | |
222 static int execute(int, long *, int); | |
223 static void cmplerr(int); | |
224 static void doprnt(long *, long *); | |
225 static void putd(long); | |
226 static void puts(const char *); | |
227 static void nlputs(const char *); | |
228 static void list(const char *); | |
229 static int lstchr(int); | |
230 static void putstr(const char *); | |
231 static void putchr(int); | |
232 static void checkpoint(void); | |
233 static void undo(void); | |
234 static int maketf(int); | |
235 static int creatf(const char *); | |
236 static int sopen(const char *, int); | |
237 static void sclose(int); | |
238 static void fspec(const char *); | |
239 static const char *ftok(const char **); | |
240 static struct tabulator *tabstring(const char *); | |
241 static void freetabs(void); | |
242 static void expand(const char *); | |
243 static void growlb(const char *); | |
244 static void growrhs(const char *); | |
245 static void growfn(const char *); | |
246 static void help(void); | |
247 | |
248 #define INIT | |
249 #define GETC() getchr() | |
250 #define UNGETC(c) (peekc = c) | |
251 #define PEEKC() (peekc = getchr()) | |
252 #define RETURN(c) return c | |
253 #define ERROR(c) cmplerr(c) | |
254 static wint_t GETWC(char *); | |
255 | |
256 #if defined (SUS) || defined (S42) || defined (SU3) | |
257 | |
258 #include <regex.h> | |
259 | |
260 #define NBRA 9 | |
261 | |
262 static char *braslist[NBRA]; | |
263 static char *braelist[NBRA]; | |
264 static char *loc1, *loc2, *locs; | |
265 static int nbra; | |
266 static int circf; | |
267 static int nodelim; | |
268 | |
269 static char *compile(char *, char *, const char *, int); | |
270 static int step(const char *, const char *); | |
271 | |
272 #else /* !SUS, !S42, !SU3 */ | |
273 | |
274 #include <regexp.h> | |
275 | |
276 #endif /* !SUS, !S42, !SU3 */ | |
277 | |
278 int | |
279 main(int argc, char **argv) | |
280 { | |
281 register int i; | |
282 void (*oldintr)(int); | |
283 | |
284 progname = basename(argv[0]); | |
285 #if defined (SUS) || defined (S42) || defined (SU3) | |
286 setlocale(LC_COLLATE, ""); | |
287 #endif | |
288 setlocale(LC_CTYPE, ""); | |
289 mb_cur_max = MB_CUR_MAX; | |
290 myuid = getuid(); | |
291 oldquit = sigset(SIGQUIT, SIG_IGN); | |
292 oldhup = sigset(SIGHUP, SIG_IGN); | |
293 oldintr = sigset(SIGINT, SIG_IGN); | |
294 if (sigset(SIGTERM, SIG_IGN) != SIG_IGN) | |
295 sigset(SIGTERM, quit); | |
296 oldpipe = sigset(SIGPIPE, onpipe); | |
297 argv++; | |
298 while (argc > 1 && **argv=='-') { | |
299 if ((*argv)[1] == '\0') { | |
300 vflag = 0; | |
301 goto next; | |
302 } else if ((*argv)[1] == '-' && (*argv)[2] == '\0') { | |
303 argv++; | |
304 argc--; | |
305 break; | |
306 } | |
307 letter: switch((*argv)[1]) { | |
308 | |
309 case 's': | |
310 vflag = 0; | |
311 break; | |
312 | |
313 case 'q': | |
314 sigset(SIGQUIT, SIG_DFL); | |
315 vflag = 1; | |
316 break; | |
317 | |
318 case 'p': | |
319 if ((*argv)[2]) | |
320 prompt = &(*argv)[2]; | |
321 else if (argv[1]) { | |
322 prompt = argv[1]; | |
323 argv++; | |
324 argc--; | |
325 } else | |
326 usage((*argv)[1], 1); | |
327 Pflag = 1; | |
328 goto next; | |
329 | |
330 default: | |
331 usage((*argv)[1], 0); | |
332 } | |
333 if ((*argv)[2]) { | |
334 (*argv)++; | |
335 goto letter; | |
336 } | |
337 next: argv++; | |
338 argc--; | |
339 } | |
340 | |
341 growfn("no space"); | |
342 if (argc>1) { | |
343 i = -1; | |
344 do | |
345 if (++i >= FNSIZE) | |
346 growfn("maximum of characters in " | |
347 "file names reached"); | |
348 while (savedfile[i] = (*argv)[i]); | |
349 globp = "e"; | |
350 } | |
351 names = malloc(26*sizeof *names); | |
352 undnames = malloc(26*sizeof *undnames); | |
353 zero = malloc(nlall*sizeof *zero); | |
354 if ((undzero = malloc(nlall*sizeof *undzero)) == NULL) | |
355 puts("no memory for undo"); | |
356 growlb("no space"); | |
357 growrhs("no space"); | |
358 init(); | |
359 if (oldintr != SIG_IGN) | |
360 sigset(SIGINT, onintr); | |
361 if (oldhup != SIG_IGN) | |
362 sigset(SIGHUP, onhup); | |
363 setjmp(savej); | |
364 if (lastsig) { | |
365 sigrelse(lastsig); | |
366 lastsig = 0; | |
367 } | |
368 commands(); | |
369 quit(0); | |
370 /*NOTREACHED*/ | |
371 return 0; | |
372 } | |
373 | |
374 static void | |
375 usage(char c, int misarg) | |
376 { | |
377 if (c) { | |
378 write(2, progname, strlen(progname)); | |
379 if (misarg) | |
380 write(2, ": option requires an argument -- ", 33); | |
381 else | |
382 write(2, ": illegal option -- ", 20); | |
383 write(2, &c, 1); | |
384 write(2, "\n", 1); | |
385 } | |
386 write(2, "usage: ", 7); | |
387 write(2, progname, strlen(progname)); | |
388 write(2, " [- | -s] [-p string] [file]\n", 29); | |
389 exit(2); | |
390 } | |
391 | |
392 static void | |
393 commands(void) | |
394 { | |
395 register long *a1; | |
396 register int c; | |
397 int n; | |
398 | |
399 for (;;) { | |
400 if (pflag) { | |
401 pflag = 0; | |
402 addr1 = addr2 = dot; | |
403 goto print; | |
404 } | |
405 if (Pflag && globp == NULL) | |
406 write(1, prompt, strlen(prompt)); | |
407 addr1 = 0; | |
408 addr2 = 0; | |
409 switch (c = getchr()) { | |
410 case ',': | |
411 case ';': | |
412 addr2 = c == ',' ? zero+1 : dot; | |
413 if (((peekc = getchr()) < '0' || peekc > '9') && | |
414 peekc != ' ' && peekc != '\t' && | |
415 peekc != '+' && peekc != '-' && | |
416 peekc != '^' && peekc != '?' && | |
417 peekc != '/' && peekc != '$' && | |
418 peekc != '.' && peekc != '\'') { | |
419 addr1 = addr2; | |
420 a1 = dol; | |
421 goto loop; | |
422 } | |
423 break; | |
424 default: | |
425 peekc = c; | |
426 } | |
427 do { | |
428 addr1 = addr2; | |
429 if ((a1 = address())==0) { | |
430 c = getchr(); | |
431 break; | |
432 } | |
433 loop: addr2 = a1; | |
434 if ((c=getchr()) == ';') { | |
435 c = ','; | |
436 dot = a1; | |
437 } | |
438 } while (c==','); | |
439 if (addr1==0) | |
440 addr1 = addr2; | |
441 switch(c) { | |
442 | |
443 case 'a': | |
444 setdot(); | |
445 newline(); | |
446 checkpoint(); | |
447 append(gettty, addr2); | |
448 continue; | |
449 | |
450 case 'c': | |
451 #if defined (SU3) | |
452 if (addr1 == zero && addr1+1 <= dol) { | |
453 if (addr1 == addr2) | |
454 addr2++; | |
455 addr1++; | |
456 } | |
457 #endif /* SU3 */ | |
458 delete(); | |
459 append(gettty, addr1-1); | |
460 #if defined (SUS) || defined (SU3) | |
461 if (dot == addr1-1 && addr1 <= dol) | |
462 dot = addr1; | |
463 #endif /* SUS || SU3 */ | |
464 continue; | |
465 | |
466 case 'd': | |
467 delete(); | |
468 continue; | |
469 | |
470 case 'E': | |
471 fchange = 0; | |
472 c = 'e'; | |
473 case 'e': | |
474 setnoaddr(); | |
475 if (vflag && fchange) { | |
476 fchange = 0; | |
477 error("warning: expecting `w'"); | |
478 } | |
479 filename(c); | |
480 init(); | |
481 addr2 = zero; | |
482 goto caseread; | |
483 | |
484 case 'f': | |
485 setnoaddr(); | |
486 filename(c); | |
487 puts(savedfile); | |
488 continue; | |
489 | |
490 case 'g': | |
491 global(1, 0); | |
492 continue; | |
493 | |
494 case 'G': | |
495 global(1, 1); | |
496 continue; | |
497 | |
498 case 'H': | |
499 prhelp = !prhelp; | |
500 /*FALLTHRU*/ | |
501 | |
502 case 'h': | |
503 if ((peekc = getchr()) == 'e') { | |
504 peekc = 0; | |
505 if (getchr() != 'l' || getchr() != 'p' || | |
506 getchr() != '\n') | |
507 error("illegal suffix"); | |
508 setnoaddr(); | |
509 help(); | |
510 continue; | |
511 } | |
512 newline(); | |
513 setnoaddr(); | |
514 if (prvmsg) | |
515 puts(prvmsg); | |
516 continue; | |
517 | |
518 case 'i': | |
519 setdot(); | |
520 #if defined (SU3) | |
521 if (addr1 == zero) { | |
522 if (addr1 == addr2) | |
523 addr2++; | |
524 addr1++; | |
525 if (dol != zero) | |
526 nonzero(); | |
527 } else | |
528 #endif /* SU3 */ | |
529 nonzero(); | |
530 newline(); | |
531 checkpoint(); | |
532 append(gettty, addr2-1); | |
533 if (dot == addr2-1) | |
534 dot++; | |
535 continue; | |
536 | |
537 | |
538 case 'j': | |
539 if (addr2==0) { | |
540 addr1 = dot; | |
541 addr2 = dot+1; | |
542 } | |
543 setdot(); | |
544 newline(); | |
545 nonzero(); | |
546 checkpoint(); | |
547 if (addr1 != addr2) | |
548 join(); | |
549 continue; | |
550 | |
551 case 'k': | |
552 if ((c = getchr()) < 'a' || c > 'z') | |
553 error("mark not lower case"); | |
554 newline(); | |
555 setdot(); | |
556 nonzero(); | |
557 names[c-'a'] = *addr2 & ~01; | |
558 anymarks |= 01; | |
559 continue; | |
560 | |
561 case 'm': | |
562 move(0); | |
563 continue; | |
564 | |
565 case '\n': | |
566 if (addr2==0) | |
567 addr2 = dot+1; | |
568 addr1 = addr2; | |
569 goto print; | |
570 | |
571 case 'n': | |
572 numbf = 1; | |
573 newline(); | |
574 goto print; | |
575 | |
576 case 'N': | |
577 newline(); | |
578 setnoaddr(); | |
579 Nflag = !Nflag; | |
580 continue; | |
581 | |
582 case 'b': | |
583 case 'o': | |
584 n = getnum(); | |
585 newline(); | |
586 setdot(); | |
587 nonzero(); | |
588 if (n >= 0) { | |
589 if (c == 'b') | |
590 bcount = n; | |
591 else | |
592 ocount = n; | |
593 } | |
594 if (c == 'b') { | |
595 a1 = addr2+bcount > dol ? dol : addr2 + bcount; | |
596 doprnt(addr1, a1); | |
597 dot = a1; | |
598 } else { | |
599 a1 = addr2+ocount > dol ? dol : addr2 + ocount; | |
600 doprnt(addr2-ocount<zero+1?zero+1:addr2-ocount, a1); | |
601 dot = addr2; | |
602 } | |
603 continue; | |
604 | |
605 case 'l': | |
606 listf++; | |
607 case 'p': | |
608 newline(); | |
609 print: | |
610 setdot(); | |
611 nonzero(); | |
612 doprnt(addr1, addr2); | |
613 dot = addr2; | |
614 continue; | |
615 | |
616 case 'P': | |
617 setnoaddr(); | |
618 newline(); | |
619 Pflag = !Pflag; | |
620 continue; | |
621 | |
622 case 'Q': | |
623 fchange = 0; | |
624 case 'q': | |
625 setnoaddr(); | |
626 newline(); | |
627 quit(0); | |
628 | |
629 case 'r': | |
630 filename(c); | |
631 caseread: | |
632 if ((io = sopen(file, READ)) < 0) { | |
633 lastc = '\n'; | |
634 error2("cannot open input file", file); | |
635 } | |
636 ioeof = 0; | |
637 setall(); | |
638 ninbuf = 0; | |
639 if (c == 'r') | |
640 checkpoint(); | |
641 n = zero != dol; | |
642 rspec = (c == 'e' || !n) && file[0] != '!'; | |
643 append(getfile, addr2); | |
644 rspec = 0; | |
645 exfile(); | |
646 fchange = n; | |
647 continue; | |
648 | |
649 case 's': | |
650 setdot(); | |
651 nonzero(); | |
652 substitute(globp!=0); | |
653 continue; | |
654 | |
655 case 't': | |
656 move(1); | |
657 continue; | |
658 | |
659 case 'u': | |
660 setdot(); | |
661 newline(); | |
662 if (unddot == NULL) | |
663 error("nothing to undo"); | |
664 undo(); | |
665 continue; | |
666 | |
667 case 'v': | |
668 global(0, 0); | |
669 continue; | |
670 | |
671 case 'V': | |
672 global(0, 1); | |
673 continue; | |
674 | |
675 case 'W': | |
676 wrapp++; | |
677 case 'w': | |
678 write: | |
679 setall(); | |
680 if (zero != dol) | |
681 nonzero(); | |
682 filename(c); | |
683 if(!wrapp || | |
684 ((io = open(file,O_WRONLY|O_APPEND)) == -1) || | |
685 ((lseek(io, 0, SEEK_END)) == -1)) { | |
686 struct stat st; | |
687 if (lstat(file, &st) == 0 && | |
688 (st.st_mode&S_IFMT) == S_IFREG && | |
689 st.st_nlink == 1 && | |
690 (myuid==0 || myuid==st.st_uid)) { | |
691 char *cp, *tp; | |
692 int nio; | |
693 if ((io = sopen(file, EXIST)) < 0) | |
694 error("cannot create output file"); | |
695 if ((wrtemp = malloc(strlen(file)+8)) == NULL) | |
696 error("out of memory"); | |
697 for (cp = file, tp = wrtemp; *cp; cp++) | |
698 *tp++ = *cp; | |
699 while (tp > wrtemp && tp[-1] != '/') | |
700 tp--; | |
701 for (cp = "\7XXXXXX"; *cp; cp++) | |
702 *tp++ = *cp; | |
703 *tp = '\0'; | |
704 if ((nio = mkstemp(wrtemp)) < 0) { | |
705 free(wrtemp); | |
706 wrtemp = NULL; | |
707 ftruncate(io, 0); | |
708 } else { | |
709 close(io); | |
710 io = nio; | |
711 } | |
712 } else { | |
713 if ((io = sopen(file, WRITE)) < 0) | |
714 error("cannot create output file"); | |
715 } | |
716 } | |
717 if (zero != dol) { | |
718 ioeof = 0; | |
719 wrapp = 0; | |
720 putfile(); | |
721 } | |
722 exfile(); | |
723 if (addr1==zero+1 && addr2==dol || addr1==addr2 && dol==zero) | |
724 fchange = 0; | |
725 if (c == 'z') | |
726 quit(0); | |
727 continue; | |
728 | |
729 case 'z': | |
730 if ((peekc=getchr()) != '\n') | |
731 error("illegal suffix"); | |
732 setnoaddr(); | |
733 goto write; | |
734 | |
735 case '=': | |
736 setall(); | |
737 newline(); | |
738 putd((addr2-zero)&MAXCNT); | |
739 putchr('\n'); | |
740 continue; | |
741 | |
742 case '!': | |
743 callunix(); | |
744 continue; | |
745 | |
746 case EOF: | |
747 return; | |
748 | |
749 } | |
750 error("unknown command"); | |
751 } | |
752 } | |
753 | |
754 static long * | |
755 address(void) | |
756 { | |
757 register long *a1; | |
758 register int minus, c; | |
759 int n, relerr; | |
760 | |
761 minus = 0; | |
762 a1 = 0; | |
763 for (;;) { | |
764 c = getchr(); | |
765 if ('0'<=c && c<='9') { | |
766 n = 0; | |
767 do { | |
768 n *= 10; | |
769 n += c - '0'; | |
770 } while ((c = getchr())>='0' && c<='9'); | |
771 peekc = c; | |
772 if (a1==0) | |
773 a1 = zero; | |
774 if (minus<0) | |
775 n = -n; | |
776 a1 += n; | |
777 minus = 0; | |
778 continue; | |
779 } | |
780 relerr = 0; | |
781 if (a1 || minus) | |
782 relerr++; | |
783 switch(c) { | |
784 case ' ': | |
785 case '\t': | |
786 continue; | |
787 | |
788 case '+': | |
789 minus++; | |
790 if (a1==0) | |
791 a1 = dot; | |
792 continue; | |
793 | |
794 case '-': | |
795 case '^': | |
796 minus--; | |
797 if (a1==0) | |
798 a1 = dot; | |
799 continue; | |
800 | |
801 case '?': | |
802 case '/': | |
803 compile(NULL, expbuf, &expbuf[ESIZE], c); | |
804 a1 = dot; | |
805 for (;;) { | |
806 if (c=='/') { | |
807 a1++; | |
808 if (a1 > dol) | |
809 a1 = zero; | |
810 } else { | |
811 a1--; | |
812 if (a1 < zero) | |
813 a1 = dol; | |
814 } | |
815 if (execute(0, a1, 0)) | |
816 break; | |
817 if (a1==dot) | |
818 error("search string not found"); | |
819 } | |
820 break; | |
821 | |
822 case '$': | |
823 a1 = dol; | |
824 break; | |
825 | |
826 case '.': | |
827 a1 = dot; | |
828 break; | |
829 | |
830 case '\'': | |
831 if ((c = getchr()) < 'a' || c > 'z') | |
832 error("mark not lower case"); | |
833 for (a1=zero; a1<=dol; a1++) | |
834 if (names[c-'a'] == (*a1 & ~01)) | |
835 break; | |
836 break; | |
837 | |
838 default: | |
839 peekc = c; | |
840 if (a1==0) | |
841 return(0); | |
842 a1 += minus; | |
843 if (a1<zero || a1>dol) | |
844 error("line out of range"); | |
845 return(a1); | |
846 } | |
847 if (relerr) | |
848 error("bad number"); | |
849 } | |
850 } | |
851 | |
852 static void | |
853 setdot(void) | |
854 { | |
855 if (addr2 == 0) | |
856 addr1 = addr2 = dot; | |
857 if (addr1 > addr2) | |
858 error("bad range"); | |
859 } | |
860 | |
861 static void | |
862 setall(void) | |
863 { | |
864 if (addr2==0) { | |
865 addr1 = zero+1; | |
866 addr2 = dol; | |
867 if (dol==zero) | |
868 addr1 = zero; | |
869 } | |
870 setdot(); | |
871 } | |
872 | |
873 static void | |
874 setnoaddr(void) | |
875 { | |
876 if (addr2) | |
877 error("Illegal address count"); | |
878 } | |
879 | |
880 static void | |
881 nonzero(void) | |
882 { | |
883 if (addr1<=zero || addr2>dol) | |
884 error("line out of range"); | |
885 } | |
886 | |
887 static void | |
888 newline(void) | |
889 { | |
890 register int c; | |
891 | |
892 if ((c = getchr()) == '\n') | |
893 return; | |
894 if (c=='p' || c=='l' || c=='n') { | |
895 pflag++; | |
896 if (c=='l') | |
897 listf++; | |
898 else if (c=='n') | |
899 numbf = 1; | |
900 if (getchr() == '\n') | |
901 return; | |
902 } | |
903 error("illegal suffix"); | |
904 } | |
905 | |
906 static void | |
907 filename(int comm) | |
908 { | |
909 register char *p1, *p2; | |
910 register int c, i; | |
911 | |
912 count = 0; | |
913 c = getchr(); | |
914 if (c=='\n' || c==EOF) { | |
915 p1 = savedfile; | |
916 if (*p1==0 && comm!='f') | |
917 error("illegal or missing filename"); | |
918 p2 = file; | |
919 while (*p2++ = *p1++) | |
920 ; | |
921 return; | |
922 } | |
923 if (c!=' ') | |
924 error("no space after command"); | |
925 while ((c = getchr()) == ' ') | |
926 ; | |
927 if (c=='\n') | |
928 error("illegal or missing filename"); | |
929 i = 0; | |
930 do { | |
931 if (i >= FNSIZE) | |
932 growfn("maximum of characters in file names reached"); | |
933 file[i++] = c; | |
934 if (c==' ' && file[0] != '!' || c==EOF) | |
935 error("illegal or missing filename"); | |
936 } while ((c = getchr()) != '\n'); | |
937 file[i++] = 0; | |
938 if ((savedfile[0]==0 || comm=='e' || comm=='f') && file[0] != '!') { | |
939 p1 = savedfile; | |
940 p2 = file; | |
941 while (*p1++ = *p2++) | |
942 ; | |
943 } | |
944 } | |
945 | |
946 static void | |
947 exfile(void) | |
948 { | |
949 sclose(io); | |
950 io = -1; | |
951 if (wrtemp) { | |
952 extern int rename(const char *, const char *); | |
953 if (rename(wrtemp, file) < 0) | |
954 error("cannot create output file"); | |
955 if (myuid == 0) | |
956 chown(file, fstbuf.st_uid, fstbuf.st_gid); | |
957 chmod(file, fstbuf.st_mode & 07777); | |
958 free(wrtemp); | |
959 wrtemp = NULL; | |
960 } | |
961 if (vflag) { | |
962 putd(count); | |
963 putchr('\n'); | |
964 } | |
965 } | |
966 | |
967 static void | |
968 onintr(int signo) | |
969 { | |
970 lastsig = signo; | |
971 putchr('\n'); | |
972 lastc = '\n'; | |
973 if (readop) { | |
974 puts("\007read may be incomplete - beware!\007"); | |
975 fchange = 0; | |
976 } | |
977 error("interrupt"); | |
978 } | |
979 | |
980 static void | |
981 onhup(int signo) | |
982 { | |
983 if (dol > zero && fchange) { | |
984 addr1 = zero+1; | |
985 addr2 = dol; | |
986 io = creat("ed.hup", 0666); | |
987 if (io < 0) { | |
988 char *home = getenv("HOME"); | |
989 if (home) { | |
990 char *fn = malloc(strlen(home) + 10); | |
991 if (fn) { | |
992 strcpy(fn, home); | |
993 strcat(fn, "/ed.hup"); | |
994 io = creat(fn, 0666); | |
995 } | |
996 } | |
997 } | |
998 if (io >= 0) | |
999 putfile(); | |
1000 } | |
1001 fchange = 0; | |
1002 status = 0200 | signo; | |
1003 quit(0); | |
1004 } | |
1005 | |
1006 static void | |
1007 onpipe(int signo) | |
1008 { | |
1009 lastsig = signo; | |
1010 error("write or open on pipe failed"); | |
1011 } | |
1012 | |
1013 static void | |
1014 error(const char *s) | |
1015 { | |
1016 error2(s, NULL); | |
1017 } | |
1018 | |
1019 static void | |
1020 error2(const char *s, const char *fn) | |
1021 { | |
1022 register int c; | |
1023 | |
1024 wrapp = 0; | |
1025 listf = 0; | |
1026 numbf = 0; | |
1027 errput(s, fn); | |
1028 count = 0; | |
1029 if (lseek(0, 0, SEEK_END) > 0) | |
1030 status = 2; | |
1031 pflag = 0; | |
1032 if (globp) | |
1033 lastc = '\n'; | |
1034 globp = 0; | |
1035 peekc = lastc; | |
1036 if(lastc) | |
1037 while ((c = getchr()) != '\n' && c != EOF) | |
1038 ; | |
1039 if (io > 0) { | |
1040 sclose(io); | |
1041 io = -1; | |
1042 } | |
1043 if (wrtemp) { | |
1044 unlink(wrtemp); | |
1045 free(wrtemp); | |
1046 wrtemp = NULL; | |
1047 } | |
1048 longjmp(savej, 1); | |
1049 } | |
1050 | |
1051 static void | |
1052 errput(const char *s, const char *fn) | |
1053 { | |
1054 prvmsg = s; | |
1055 if (fn) { | |
1056 putchr('?'); | |
1057 puts(fn); | |
1058 } else | |
1059 puts("?"); | |
1060 if (prhelp) | |
1061 puts(s); | |
1062 } | |
1063 | |
1064 static int | |
1065 getchr(void) | |
1066 { | |
1067 char c; | |
1068 if (lastc=peekc) { | |
1069 peekc = 0; | |
1070 return(lastc); | |
1071 } | |
1072 if (globp) { | |
1073 if ((lastc = *globp++) != 0) | |
1074 return(lastc); | |
1075 globp = 0; | |
1076 return(EOF); | |
1077 } | |
1078 if (read(0, &c, 1) <= 0) | |
1079 return(lastc = EOF); | |
1080 lastc = c; | |
1081 return(lastc); | |
1082 } | |
1083 | |
1084 static int | |
1085 gettty(void) | |
1086 { | |
1087 register int c, i; | |
1088 register char *gf; | |
1089 | |
1090 i = 0; | |
1091 gf = globp; | |
1092 while ((c = getchr()) != '\n') { | |
1093 if (c==EOF) { | |
1094 if (gf) | |
1095 peekc = c; | |
1096 return(c); | |
1097 } | |
1098 if (c == 0) | |
1099 continue; | |
1100 if (i >= LBSIZE) | |
1101 growlb("line too long"); | |
1102 linebuf[i++] = c; | |
1103 } | |
1104 if (i >= LBSIZE-2) | |
1105 growlb("line too long"); | |
1106 linebuf[i++] = 0; | |
1107 if (linebuf[0]=='.' && linebuf[1]==0) | |
1108 return(EOF); | |
1109 #if !defined (SUS) && !defined (SU3) | |
1110 if (linebuf[0]=='\\' && linebuf[1]=='.' && linebuf[2]==0) | |
1111 linebuf[0]='.', linebuf[1]=0; | |
1112 #endif | |
1113 return(0); | |
1114 } | |
1115 | |
1116 static long | |
1117 getnum(void) | |
1118 { | |
1119 char scount[20]; | |
1120 int i; | |
1121 | |
1122 i = 0; | |
1123 while ((peekc=getchr()) >= '0' && peekc <= '9' && i < sizeof scount) { | |
1124 scount[i++] = peekc; | |
1125 peekc = 0; | |
1126 } | |
1127 scount[i] = '\0'; | |
1128 return i ? atol(scount) : -1; | |
1129 } | |
1130 | |
1131 static int | |
1132 getfile(void) | |
1133 { | |
1134 register int c, i, j; | |
1135 static int nextj; | |
1136 | |
1137 i = 0; | |
1138 j = nextj; | |
1139 do { | |
1140 if (--ninbuf < 0) { | |
1141 if (ioeof || (ninbuf=read(io, genbuf, LBSIZE)-1) < 0) { | |
1142 if (ioeof == 0 && ninbuf < -1) { | |
1143 puts("input error"); | |
1144 status = 1; | |
1145 } | |
1146 if (i > 0) { | |
1147 puts("'\\n' appended"); | |
1148 c = '\n'; | |
1149 ioeof = 1; | |
1150 goto wrc; | |
1151 } | |
1152 return(EOF); | |
1153 } | |
1154 j = 0; | |
1155 } | |
1156 c = genbuf[j++]&0377; | |
1157 wrc: if (i >= LBSIZE) { | |
1158 lastc = '\n'; | |
1159 growlb("line too long"); | |
1160 } | |
1161 linebuf[i++] = c ? c : '\n'; | |
1162 count++; | |
1163 } while (c != '\n'); | |
1164 linebuf[--i] = 0; | |
1165 nextj = j; | |
1166 if (rspec && dot == zero) | |
1167 fspec(linebuf); | |
1168 if (maxlength && i > maxlength) { | |
1169 putstr("line too long: lno = "); | |
1170 putd((dot - zero+1)&MAXCNT); | |
1171 putchr('\n'); | |
1172 } | |
1173 return(0); | |
1174 } | |
1175 | |
1176 static void | |
1177 putfile(void) | |
1178 { | |
1179 long *a1; | |
1180 int n; | |
1181 register char *fp, *lp; | |
1182 register int nib; | |
1183 | |
1184 nib = 512; | |
1185 fp = genbuf; | |
1186 a1 = addr1; | |
1187 do { | |
1188 lp = getline(*a1++, 0); | |
1189 if (maxlength) { | |
1190 for (n = 0; lp[n]; n++); | |
1191 if (n > maxlength) { | |
1192 putstr("line too long: lno = "); | |
1193 putd((a1-1 - zero)&MAXCNT); | |
1194 putchr('\n'); | |
1195 } | |
1196 } | |
1197 for (;;) { | |
1198 if (--nib < 0) { | |
1199 n = fp-genbuf; | |
1200 if(write(io, genbuf, n) != n) | |
1201 error("write error"); | |
1202 nib = 511; | |
1203 fp = genbuf; | |
1204 } | |
1205 count++; | |
1206 if ((*fp++ = *lp++) == 0) { | |
1207 fp[-1] = '\n'; | |
1208 break; | |
1209 } else if (fp[-1] == '\n') | |
1210 fp[-1] = '\0'; | |
1211 } | |
1212 } while (a1 <= addr2); | |
1213 n = fp-genbuf; | |
1214 if(write(io, genbuf, n) != n) | |
1215 error("write error"); | |
1216 } | |
1217 | |
1218 static int | |
1219 append(int (*f)(void), long *a) | |
1220 { | |
1221 register long *a1, *a2, *rdot; | |
1222 int nline, tl; | |
1223 | |
1224 nline = 0; | |
1225 dot = a; | |
1226 while ((*f)() == 0) { | |
1227 if ((dol-zero)+1 >= nlall) { | |
1228 long *ozero = zero; | |
1229 nlall += 512; | |
1230 if ((zero = realloc(zero, nlall*sizeof *zero))==NULL) { | |
1231 lastc = '\n'; | |
1232 zero = ozero; | |
1233 error("out of memory for append"); | |
1234 } | |
1235 dot += zero - ozero; | |
1236 dol += zero - ozero; | |
1237 addr1 += zero - ozero; | |
1238 addr2 += zero - ozero; | |
1239 if (unddot) { | |
1240 unddot += zero - ozero; | |
1241 unddol += zero - ozero; | |
1242 } | |
1243 if (undzero) { | |
1244 ozero = undzero; | |
1245 if ((undzero = realloc(undzero, | |
1246 nlall*sizeof *undzero)) == 0) { | |
1247 puts("no memory for undo"); | |
1248 free(ozero); | |
1249 } | |
1250 } | |
1251 } | |
1252 tl = putline(); | |
1253 nline++; | |
1254 a1 = ++dol; | |
1255 a2 = a1+1; | |
1256 rdot = ++dot; | |
1257 while (a1 > rdot) | |
1258 *--a2 = *--a1; | |
1259 *rdot = tl; | |
1260 } | |
1261 return(nline); | |
1262 } | |
1263 | |
1264 static void | |
1265 callunix(void) | |
1266 { | |
1267 char *line; | |
1268 void (*savint)(int); | |
1269 pid_t pid, rpid; | |
1270 int retcode; | |
1271 | |
1272 setnoaddr(); | |
1273 line = readcmd(); | |
1274 if ((pid = fork()) == 0) { | |
1275 sigset(SIGHUP, oldhup); | |
1276 sigset(SIGQUIT, oldquit); | |
1277 sigset(SIGPIPE, oldpipe); | |
1278 execl(SHELL, "sh", "-c", line, NULL); | |
1279 _exit(0100); | |
1280 } else if (pid < 0) | |
1281 error("fork failed - try again"); | |
1282 savint = sigset(SIGINT, SIG_IGN); | |
1283 while ((rpid = wait(&retcode)) != pid && rpid != -1) | |
1284 ; | |
1285 sigset(SIGINT, savint); | |
1286 if (vflag) | |
1287 puts("!"); | |
1288 } | |
1289 | |
1290 #define cmadd(c) ((i>=cmsize ? \ | |
1291 ((line=realloc(line,cmsize+=128)) == 0 ? \ | |
1292 (error("line too long"),0) : 0, 0) \ | |
1293 : 0), line[i++]=(c)) | |
1294 | |
1295 static char * | |
1296 readcmd(void) | |
1297 { | |
1298 static char *line, *prev; | |
1299 static int cmsize, pvsize; | |
1300 char *pp; | |
1301 int c, mod = 0, i; | |
1302 | |
1303 i = 0; | |
1304 if ((c = getchr()) == '!') { | |
1305 for (pp = prev; *pp; pp++) | |
1306 line[i++] = *pp; | |
1307 mod = 1; | |
1308 c = getchr(); | |
1309 } | |
1310 while (c != '\n' && c != EOF) { | |
1311 if (c == '\\') { | |
1312 c = getchr(); | |
1313 if (c != '%') | |
1314 cmadd('\\'); | |
1315 cmadd(c); | |
1316 } else if (c == '%') { | |
1317 for (pp = savedfile; *pp; pp++) | |
1318 cmadd(*pp); | |
1319 mod = 1; | |
1320 } else | |
1321 cmadd(c); | |
1322 c = getchr(); | |
1323 } | |
1324 cmadd('\0'); | |
1325 if (pvsize < cmsize && (prev = realloc(prev, pvsize=cmsize)) == 0) | |
1326 error("line too long"); | |
1327 strcpy(prev, line); | |
1328 if (mod) | |
1329 nlputs(line); | |
1330 return line; | |
1331 } | |
1332 | |
1333 static void | |
1334 quit(int signo) | |
1335 { | |
1336 lastsig = signo; | |
1337 if (vflag && fchange) { | |
1338 fchange = 0; | |
1339 error("warning: expecting `w'"); | |
1340 } | |
1341 if (wrtemp) | |
1342 unlink(wrtemp); | |
1343 unlink(tfname); | |
1344 exit(status); | |
1345 } | |
1346 | |
1347 static void | |
1348 delete(void) | |
1349 { | |
1350 setdot(); | |
1351 newline(); | |
1352 nonzero(); | |
1353 checkpoint(); | |
1354 rdelete(addr1, addr2); | |
1355 } | |
1356 | |
1357 static void | |
1358 rdelete(long *ad1, long *ad2) | |
1359 { | |
1360 register long *a1, *a2, *a3; | |
1361 | |
1362 a1 = ad1; | |
1363 a2 = ad2+1; | |
1364 a3 = dol; | |
1365 dol -= a2 - a1; | |
1366 do { | |
1367 *a1++ = *a2++; | |
1368 } while (a2 <= a3); | |
1369 a1 = ad1; | |
1370 if (a1 > dol) | |
1371 a1 = dol; | |
1372 dot = a1; | |
1373 fchange = 1; | |
1374 } | |
1375 | |
1376 static void | |
1377 gdelete(void) | |
1378 { | |
1379 register long *a1, *a2, *a3; | |
1380 | |
1381 a3 = dol; | |
1382 for (a1=zero+1; (*a1&01)==0; a1++) | |
1383 if (a1>=a3) | |
1384 return; | |
1385 for (a2=a1+1; a2<=a3;) { | |
1386 if (*a2&01) { | |
1387 a2++; | |
1388 dot = a1; | |
1389 } else | |
1390 *a1++ = *a2++; | |
1391 } | |
1392 dol = a1-1; | |
1393 if (dot>dol) | |
1394 dot = dol; | |
1395 fchange = 1; | |
1396 } | |
1397 | |
1398 static char * | |
1399 getline(long tl, int nulterm) | |
1400 { | |
1401 register char *bp, *lp; | |
1402 register long nl; | |
1403 | |
1404 lp = linebuf; | |
1405 bp = getblock(tl, READ); | |
1406 nl = nleft; | |
1407 tl &= ~0377; | |
1408 while (*lp++ = *bp++) { | |
1409 if (lp[-1] == '\n' && nulterm) { | |
1410 lp[-1] = '\0'; | |
1411 break; | |
1412 } | |
1413 if (--nl == 0) { | |
1414 bp = getblock(tl+=0400, READ); | |
1415 nl = nleft; | |
1416 } | |
1417 } | |
1418 return(linebuf); | |
1419 } | |
1420 | |
1421 static int | |
1422 putline(void) | |
1423 { | |
1424 register char *bp, *lp; | |
1425 register long nl; | |
1426 long tl; | |
1427 | |
1428 fchange = 1; | |
1429 lp = linebuf; | |
1430 tl = tline; | |
1431 bp = getblock(tl, WRITE); | |
1432 nl = nleft; | |
1433 tl &= ~0377; | |
1434 while (*bp = *lp++) { | |
1435 if (*bp++ == '\n' && insub) { | |
1436 *--bp = 0; | |
1437 linebp = lp; | |
1438 break; | |
1439 } | |
1440 if (--nl == 0) { | |
1441 bp = getblock(tl+=0400, WRITE); | |
1442 nl = nleft; | |
1443 } | |
1444 } | |
1445 nl = tline; | |
1446 tline += (((lp-linebuf)+03)>>1)&(MAXCNT-1); | |
1447 return(nl); | |
1448 } | |
1449 | |
1450 static char * | |
1451 getblock(long atl, long iof) | |
1452 { | |
1453 register long bno, off; | |
1454 | |
1455 bno = (atl>>8)&BLKMSK; | |
1456 off = (atl<<1)&0774; | |
1457 if (bno >= BLKMSK) { | |
1458 lastc = '\n'; | |
1459 error("temp file too big"); | |
1460 } | |
1461 nleft = 512 - off; | |
1462 if (bno==iblock) { | |
1463 ichanged |= iof; | |
1464 return(ibuff+off); | |
1465 } | |
1466 if (bno==oblock) | |
1467 return(obuff+off); | |
1468 if (iof==READ) { | |
1469 if (ichanged) | |
1470 blkio(iblock, ibuff, 1); | |
1471 ichanged = 0; | |
1472 iblock = bno; | |
1473 blkio(bno, ibuff, 0); | |
1474 return(ibuff+off); | |
1475 } | |
1476 if (oblock>=0) | |
1477 blkio(oblock, obuff, 1); | |
1478 oblock = bno; | |
1479 return(obuff+off); | |
1480 } | |
1481 | |
1482 static void | |
1483 blkio(long b, char *buf, int wr) | |
1484 { | |
1485 lseek(tfile, b<<9, SEEK_SET); | |
1486 if ((wr ? write(tfile, buf, 512) : read (tfile, buf, 512)) != 512) { | |
1487 status = 1; | |
1488 error("I/O error on temp file"); | |
1489 } | |
1490 } | |
1491 | |
1492 static void | |
1493 init(void) | |
1494 { | |
1495 register long *markp; | |
1496 | |
1497 tline = 2; | |
1498 for (markp = names; markp < &names[26]; markp++) | |
1499 *markp = 0; | |
1500 for (markp = undnames; markp < &undnames[26]; markp++) | |
1501 *markp = 0; | |
1502 subnewa = 0; | |
1503 anymarks = 0; | |
1504 iblock = -1; | |
1505 oblock = -1; | |
1506 ichanged = 0; | |
1507 tfile = maketf(tfile); | |
1508 dot = dol = zero; | |
1509 unddot = NULL; | |
1510 } | |
1511 | |
1512 static void | |
1513 global(int k, int ia) | |
1514 { | |
1515 register int c; | |
1516 register long *a1; | |
1517 static char *globuf; | |
1518 char mb[MB_LEN_MAX+1]; | |
1519 int spflag = 0; | |
1520 | |
1521 if (globp) | |
1522 error("multiple globals not allowed"); | |
1523 setall(); | |
1524 nonzero(); | |
1525 if ((c=GETWC(mb))=='\n') | |
1526 error("incomplete global expression"); | |
1527 compile(NULL, expbuf, &expbuf[ESIZE], c); | |
1528 if (!ia) { | |
1529 globrd(&globuf, EOF); | |
1530 if (globuf[0] == '\n') | |
1531 globuf[0] = 'p', globuf[1] = '\n', globuf[2] = '\0'; | |
1532 } else { | |
1533 newline(); | |
1534 spflag = pflag; | |
1535 pflag = 0; | |
1536 } | |
1537 checkpoint(); | |
1538 for (a1=zero; a1<=dol; a1++) { | |
1539 *a1 &= ~01; | |
1540 if (a1>=addr1 && a1<=addr2 && execute(0, a1, 0)==k) | |
1541 *a1 |= 01; | |
1542 } | |
1543 /* | |
1544 * Special case: g/.../d (avoid n^2 algorithm) | |
1545 */ | |
1546 if (!ia && globuf[0]=='d' && globuf[1]=='\n' && globuf[2]=='\0') { | |
1547 gdelete(); | |
1548 return; | |
1549 } | |
1550 for (a1=zero; a1<=dol; a1++) { | |
1551 if (*a1 & 01) { | |
1552 *a1 &= ~01; | |
1553 dot = a1; | |
1554 if (ia) { | |
1555 puts(getline(*a1, 0)); | |
1556 if ((c = getchr()) == EOF) | |
1557 error("command expected"); | |
1558 if (c == 'a' || c == 'c' || c == 'i') | |
1559 error("a, i, or c not allowed in G"); | |
1560 else if (c == '&') { | |
1561 if ((c = getchr()) != '\n') | |
1562 error("end of line expected"); | |
1563 if (globuf == 0 || *globuf == 0) | |
1564 error("no remembered command"); | |
1565 } else if (c == '\n') { | |
1566 a1 = zero; | |
1567 continue; | |
1568 } else | |
1569 globrd(&globuf, c); | |
1570 } | |
1571 globp = globuf; | |
1572 commands(); | |
1573 globp = NULL; | |
1574 a1 = zero; | |
1575 } | |
1576 } | |
1577 if (ia) | |
1578 pflag = spflag; | |
1579 } | |
1580 | |
1581 static void | |
1582 globrd(char **globuf, register int c) | |
1583 { | |
1584 register int i; | |
1585 | |
1586 if (*globuf == 0 && (*globuf = malloc(GBSIZE=256)) == 0) | |
1587 error("global too long"); | |
1588 i = 0; | |
1589 if (c != EOF) | |
1590 (*globuf)[i++] = c; | |
1591 while ((c = getchr()) != '\n') { | |
1592 if (c==EOF) | |
1593 error("incomplete global expression"); | |
1594 if (c=='\\') { | |
1595 c = getchr(); | |
1596 if (c!='\n') | |
1597 (*globuf)[i++] = '\\'; | |
1598 } | |
1599 (*globuf)[i++] = c; | |
1600 if (i>=GBSIZE-4 && (*globuf=realloc(*globuf,GBSIZE+=256)) == 0) | |
1601 error("global too long"); | |
1602 } | |
1603 (*globuf)[i++] = '\n'; | |
1604 (*globuf)[i++] = 0; | |
1605 } | |
1606 | |
1607 static void | |
1608 join(void) | |
1609 { | |
1610 register int i, j; | |
1611 register long *a1; | |
1612 | |
1613 j = 0; | |
1614 for (a1=addr1; a1<=addr2; a1++) { | |
1615 i = getline(*a1, 0) - linebuf; | |
1616 while (genbuf[j] = linebuf[i++]) | |
1617 if (j++ >= LBSIZE-2) | |
1618 growlb("line too long"); | |
1619 } | |
1620 i = 0; | |
1621 j = 0; | |
1622 while (linebuf[i++] = genbuf[j++]) | |
1623 ; | |
1624 *addr1 = putline(); | |
1625 if (addr1<addr2) | |
1626 rdelete(addr1+1, addr2); | |
1627 dot = addr1; | |
1628 } | |
1629 | |
1630 static void | |
1631 substitute(int inglob) | |
1632 { | |
1633 register long *markp; | |
1634 register long *a1; | |
1635 intptr_t nl; | |
1636 int gsubf; | |
1637 | |
1638 checkpoint(); | |
1639 gsubf = compsub(); | |
1640 insub = 1; | |
1641 for (a1 = addr1; a1 <= addr2; a1++) { | |
1642 long *ozero; | |
1643 if (execute(0, a1, 1)==0) | |
1644 continue; | |
1645 inglob |= dosub(gsubf < 2); | |
1646 if (gsubf) { | |
1647 int i = 1; | |
1648 | |
1649 while (*loc2) { | |
1650 if (execute(1, NULL, 1)==0) | |
1651 break; | |
1652 inglob |= dosub(gsubf == -1 || ++i == gsubf); | |
1653 } | |
1654 } | |
1655 subnewa = putline(); | |
1656 *a1 &= ~01; | |
1657 if (anymarks) { | |
1658 for (markp = names; markp < &names[26]; markp++) | |
1659 if (*markp == *a1) | |
1660 *markp = subnewa; | |
1661 } | |
1662 *a1 = subnewa; | |
1663 ozero = zero; | |
1664 nl = append(getsub, a1); | |
1665 nl += zero-ozero; | |
1666 a1 += nl; | |
1667 addr2 += nl; | |
1668 } | |
1669 insub = 0; | |
1670 if (inglob==0) | |
1671 error("no match"); | |
1672 } | |
1673 | |
1674 static int | |
1675 compsub(void) | |
1676 { | |
1677 register int seof, c, i; | |
1678 static char *oldrhs; | |
1679 static int orhssz; | |
1680 char mb[MB_LEN_MAX+1]; | |
1681 | |
1682 if ((seof = GETWC(mb)) == '\n' || seof == ' ') | |
1683 error("illegal or missing delimiter"); | |
1684 nodelim = 0; | |
1685 compile(NULL, expbuf, &expbuf[ESIZE], seof); | |
1686 i = 0; | |
1687 for (;;) { | |
1688 c = GETWC(mb); | |
1689 if (c=='\\') { | |
1690 if (i >= RHSIZE-2) | |
1691 growrhs("replacement string too long"); | |
1692 rhsbuf[i++] = c; | |
1693 c = GETWC(mb); | |
1694 } else if (c=='\n') { | |
1695 if (globp && *globp) { | |
1696 if (i >= RHSIZE-2) | |
1697 growrhs("replacement string too long"); | |
1698 rhsbuf[i++] = '\\'; | |
1699 } | |
1700 else if (nodelim) | |
1701 error("illegal or missing delimiter"); | |
1702 else { | |
1703 peekc = c; | |
1704 pflag++; | |
1705 break; | |
1706 } | |
1707 } else if (c==seof) | |
1708 break; | |
1709 for (c = 0; c==0 || mb[c]; c++) { | |
1710 if (i >= RHSIZE-2) | |
1711 growrhs("replacement string too long"); | |
1712 rhsbuf[i++] = mb[c]; | |
1713 } | |
1714 } | |
1715 rhsbuf[i++] = 0; | |
1716 if (rhsbuf[0] == '%' && rhsbuf[1] == 0) { | |
1717 if (orhssz == 0) | |
1718 error("no remembered replacement string"); | |
1719 strcpy(rhsbuf, oldrhs); | |
1720 } else { | |
1721 if (orhssz < RHSIZE && | |
1722 (oldrhs = realloc(oldrhs, orhssz=RHSIZE)) == 0) | |
1723 error("replacement string too long"); | |
1724 strcpy(oldrhs, rhsbuf); | |
1725 } | |
1726 if ((peekc = getchr()) == 'g') { | |
1727 peekc = 0; | |
1728 newline(); | |
1729 return(-1); | |
1730 } else if (peekc >= '0' && peekc <= '9') { | |
1731 c = getnum(); | |
1732 if (c < 1 || c > LBSIZE) | |
1733 error("invalid count"); | |
1734 newline(); | |
1735 return c; | |
1736 } | |
1737 newline(); | |
1738 return(0); | |
1739 } | |
1740 | |
1741 static int | |
1742 getsub(void) | |
1743 { | |
1744 register char *p1, *p2; | |
1745 | |
1746 p1 = linebuf; | |
1747 if ((p2 = linebp) == 0) | |
1748 return(EOF); | |
1749 while (*p1++ = *p2++) | |
1750 ; | |
1751 linebp = 0; | |
1752 return(0); | |
1753 } | |
1754 | |
1755 static int | |
1756 dosub(int really) | |
1757 { | |
1758 register char *lp, *sp; | |
1759 register int i, j, k; | |
1760 int c; | |
1761 | |
1762 if (!really) | |
1763 goto copy; | |
1764 i = 0; | |
1765 j = 0; | |
1766 k = 0; | |
1767 while (&linebuf[i] < loc1) | |
1768 genbuf[j++] = linebuf[i++]; | |
1769 while (c = rhsbuf[k++]&0377) { | |
1770 if (c=='&') { | |
1771 j = place(j, loc1, loc2); | |
1772 continue; | |
1773 } else if (c == '\\') { | |
1774 c = rhsbuf[k++]&0377; | |
1775 if (c >='1' && c < nbra+'1') { | |
1776 j = place(j, braslist[c-'1'], braelist[c-'1']); | |
1777 continue; | |
1778 } | |
1779 } | |
1780 if (j >= LBSIZE) | |
1781 growlb("line too long"); | |
1782 genbuf[j++] = c; | |
1783 } | |
1784 i = loc2 - linebuf; | |
1785 loc2 = j + linebuf; | |
1786 #if defined (SUS) || defined (SU3) || defined (S42) | |
1787 if (loc1 == &linebuf[i]) { | |
1788 int n; | |
1789 wchar_t wc; | |
1790 if (mb_cur_max > 1 && (n = mbtowc(&wc, loc2, mb_cur_max)) > 0) | |
1791 loc2 += n; | |
1792 else | |
1793 loc2++; | |
1794 } | |
1795 #endif /* SUS || SU3 || S42 */ | |
1796 while (genbuf[j++] = linebuf[i++]) | |
1797 if (j >= LBSIZE) | |
1798 growlb("line too long"); | |
1799 if (really) { | |
1800 lp = linebuf; | |
1801 sp = genbuf; | |
1802 } else { | |
1803 copy: sp = linebuf; | |
1804 lp = genbuf; | |
1805 } | |
1806 while (*lp++ = *sp++) | |
1807 ; | |
1808 return really; | |
1809 } | |
1810 | |
1811 static int | |
1812 place(register int j, register const char *l1, register const char *l2) | |
1813 { | |
1814 | |
1815 while (l1 < l2) { | |
1816 genbuf[j++] = *l1++; | |
1817 if (j >= LBSIZE) | |
1818 growlb("line too long"); | |
1819 } | |
1820 return(j); | |
1821 } | |
1822 | |
1823 static void | |
1824 move(int cflag) | |
1825 { | |
1826 register long *adt, *ad1, *ad2; | |
1827 | |
1828 setdot(); | |
1829 nonzero(); | |
1830 if ((adt = address())==0) | |
1831 error("illegal move destination"); | |
1832 newline(); | |
1833 checkpoint(); | |
1834 if (cflag) { | |
1835 long *ozero; | |
1836 intptr_t delta; | |
1837 ad1 = dol; | |
1838 ozero = zero; | |
1839 append(getcopy, ad1++); | |
1840 ad2 = dol; | |
1841 delta = zero - ozero; | |
1842 ad1 += delta; | |
1843 adt += delta; | |
1844 } else { | |
1845 ad2 = addr2; | |
1846 for (ad1 = addr1; ad1 <= ad2;) | |
1847 *ad1++ &= ~01; | |
1848 ad1 = addr1; | |
1849 } | |
1850 ad2++; | |
1851 if (adt<ad1) { | |
1852 dot = adt + (ad2-ad1); | |
1853 if ((++adt)==ad1) | |
1854 return; | |
1855 reverse(adt, ad1); | |
1856 reverse(ad1, ad2); | |
1857 reverse(adt, ad2); | |
1858 } else if (adt >= ad2) { | |
1859 dot = adt++; | |
1860 reverse(ad1, ad2); | |
1861 reverse(ad2, adt); | |
1862 reverse(ad1, adt); | |
1863 } else | |
1864 error("illegal move destination"); | |
1865 fchange = 1; | |
1866 } | |
1867 | |
1868 static void | |
1869 reverse(register long *a1, register long *a2) | |
1870 { | |
1871 register int t; | |
1872 | |
1873 for (;;) { | |
1874 t = *--a2; | |
1875 if (a2 <= a1) | |
1876 return; | |
1877 *a2 = *a1; | |
1878 *a1++ = t; | |
1879 } | |
1880 } | |
1881 | |
1882 static int | |
1883 getcopy(void) | |
1884 { | |
1885 if (addr1 > addr2) | |
1886 return(EOF); | |
1887 getline(*addr1++, 0); | |
1888 return(0); | |
1889 } | |
1890 | |
1891 static int | |
1892 execute(int gf, long *addr, int subst) | |
1893 { | |
1894 register char *p1, *p2, c; | |
1895 | |
1896 for (c=0; c<NBRA; c++) { | |
1897 braslist[c&0377] = 0; | |
1898 braelist[c&0377] = 0; | |
1899 } | |
1900 if (gf) { | |
1901 if (circf) | |
1902 return(0); | |
1903 p1 = linebuf; | |
1904 p2 = genbuf; | |
1905 while (*p1++ = *p2++) | |
1906 ; | |
1907 locs = p1 = loc2; | |
1908 } else { | |
1909 if (addr==zero) | |
1910 return(0); | |
1911 p1 = getline(*addr, 1); | |
1912 locs = 0; | |
1913 } | |
1914 needsub = subst; | |
1915 return step(p1, expbuf); | |
1916 } | |
1917 | |
1918 static void | |
1919 cmplerr(int c) | |
1920 { | |
1921 const char *msg; | |
1922 | |
1923 #if !defined (SUS) && !defined (S42) && !defined (SU3) | |
1924 expbuf[0] = 0; | |
1925 #endif | |
1926 switch (c) { | |
1927 case 11: | |
1928 msg = "Range endpoint too large"; | |
1929 break; | |
1930 case 16: | |
1931 msg = "bad number"; | |
1932 break; | |
1933 case 25: | |
1934 msg = "`\\digit' out of range"; | |
1935 break; | |
1936 case 36: | |
1937 msg = "illegal or missing delimiter"; | |
1938 break; | |
1939 case 41: | |
1940 msg = "no remembered search string"; | |
1941 break; | |
1942 case 42: | |
1943 msg = "'\\( \\)' imbalance"; | |
1944 break; | |
1945 case 43: | |
1946 msg = "Too many `\\(' s"; | |
1947 break; | |
1948 case 44: | |
1949 msg = "more than 2 numbers given"; | |
1950 break; | |
1951 case 45: | |
1952 msg = "'\\}' expected"; | |
1953 break; | |
1954 case 46: | |
1955 msg = "first number exceeds second"; | |
1956 break; | |
1957 case 49: | |
1958 msg = "'[ ]' imbalance"; | |
1959 break; | |
1960 case 50: | |
1961 msg = "regular expression overflow"; | |
1962 break; | |
1963 case 67: | |
1964 msg = "illegal byte sequence"; | |
1965 break; | |
1966 default: | |
1967 msg = "regular expression error"; | |
1968 break; | |
1969 } | |
1970 error(msg); | |
1971 } | |
1972 | |
1973 static void | |
1974 doprnt(long *bot, long *top) | |
1975 { | |
1976 long *a1; | |
1977 | |
1978 a1 = bot; | |
1979 do { | |
1980 if (numbf ^ Nflag) { | |
1981 putd(a1-zero); | |
1982 putchr('\t'); | |
1983 } | |
1984 nlputs(getline(*a1++, 0)); | |
1985 } while (a1 <= top); | |
1986 pflag = 0; | |
1987 listf = 0; | |
1988 numbf = 0; | |
1989 } | |
1990 | |
1991 static void | |
1992 putd(long c) | |
1993 { | |
1994 register int r; | |
1995 | |
1996 r = c%10; | |
1997 c /= 10; | |
1998 if (c) | |
1999 putd(c); | |
2000 putchr(r + '0'); | |
2001 } | |
2002 | |
2003 static void | |
2004 nlputs(register const char *sp) | |
2005 { | |
2006 if (listf) | |
2007 list(sp); | |
2008 else if (tabstops) | |
2009 expand(sp); | |
2010 else | |
2011 puts(sp); | |
2012 } | |
2013 | |
2014 static void | |
2015 puts(register const char *sp) | |
2016 { | |
2017 while (*sp) { | |
2018 if (*sp != '\n') | |
2019 putchr(*sp++ & 0377); | |
2020 else | |
2021 sp++, putchr('\0'); | |
2022 } | |
2023 putchr('\n'); | |
2024 } | |
2025 | |
2026 static void | |
2027 list(const char *lp) | |
2028 { | |
2029 int col, n; | |
2030 wchar_t c; | |
2031 | |
2032 col = numbf ^ Nflag ? 8 : 0; | |
2033 while (*lp) { | |
2034 if (mb_cur_max > 1 && *lp&0200) | |
2035 n = mbtowc(&c, lp, mb_cur_max); | |
2036 else { | |
2037 n = 1; | |
2038 c = *lp&0377; | |
2039 } | |
2040 if (col+1 >= 72) { | |
2041 col = 0; | |
2042 putchr('\\'); | |
2043 putchr('\n'); | |
2044 } | |
2045 if (n<0 || | |
2046 #if defined (SUS) || defined (S42) || defined (SU3) | |
2047 c == '\\' || | |
2048 #endif /* SUS || S42 || SU3 */ | |
2049 !(mb_cur_max>1 ? iswprint(c) : isprint(c))) { | |
2050 if (n<0) | |
2051 n = 1; | |
2052 while (n--) | |
2053 col += lstchr(*lp++&0377); | |
2054 } else if (mb_cur_max>1) { | |
2055 col += wcwidth(c); | |
2056 while (n--) | |
2057 putchr(*lp++&0377); | |
2058 } else { | |
2059 putchr(*lp++&0377); | |
2060 col++; | |
2061 } | |
2062 } | |
2063 #if defined (SUS) || defined (S42) || defined (SU3) | |
2064 putchr('$'); | |
2065 #endif | |
2066 putchr('\n'); | |
2067 } | |
2068 | |
2069 static int | |
2070 lstchr(int c) | |
2071 { | |
2072 int cad = 1, d; | |
2073 | |
2074 #if !defined (SUS) && !defined (S42) && !defined (SU3) | |
2075 if (c=='\t') { | |
2076 c = '>'; | |
2077 goto esc; | |
2078 } | |
2079 if (c=='\b') { | |
2080 c = '<'; | |
2081 esc: | |
2082 putchr('-'); | |
2083 putchr('\b'); | |
2084 putchr(c); | |
2085 } else if (c == '\n') { | |
2086 putchr('\\'); | |
2087 putchr('0'); | |
2088 putchr('0'); | |
2089 putchr('0'); | |
2090 cad = 4; | |
2091 #else /* !SUS, !S42, !SU3 */ | |
2092 if (c == '\n') | |
2093 c = '\0'; | |
2094 if (c == '\\') { | |
2095 putchr('\\'); | |
2096 putchr('\\'); | |
2097 cad = 2; | |
2098 } else if (c == '\a') { | |
2099 putchr('\\'); | |
2100 putchr('a'); | |
2101 cad = 2; | |
2102 } else if (c == '\b') { | |
2103 putchr('\\'); | |
2104 putchr('b'); | |
2105 cad = 2; | |
2106 } else if (c == '\f') { | |
2107 putchr('\\'); | |
2108 putchr('f'); | |
2109 cad = 2; | |
2110 } else if (c == '\r') { | |
2111 putchr('\\'); | |
2112 putchr('r'); | |
2113 cad = 2; | |
2114 } else if (c == '\t') { | |
2115 putchr('\\'); | |
2116 putchr('t'); | |
2117 cad = 2; | |
2118 } else if (c == '\v') { | |
2119 putchr('\\'); | |
2120 putchr('v'); | |
2121 cad = 2; | |
2122 #endif /* !SUS, !S42, !SU3 */ | |
2123 } else { | |
2124 putchr('\\'); | |
2125 putchr(((c&~077)>>6)+'0'); | |
2126 c &= 077; | |
2127 d = c & 07; | |
2128 putchr(c > d ? ((c-d)>>3)+'0' : '0'); | |
2129 putchr(d+'0'); | |
2130 cad = 4; | |
2131 } | |
2132 return cad; | |
2133 } | |
2134 | |
2135 static void | |
2136 putstr(const char *s) | |
2137 { | |
2138 while (*s) | |
2139 putchr(*s++); | |
2140 } | |
2141 | |
2142 static char line[70]; | |
2143 static char *linp = line; | |
2144 | |
2145 static void | |
2146 putchr(int ac) | |
2147 { | |
2148 register char *lp; | |
2149 register int c; | |
2150 | |
2151 lp = linp; | |
2152 c = ac; | |
2153 *lp++ = c; | |
2154 if(c == '\n' || lp >= &line[64]) { | |
2155 linp = line; | |
2156 write(1, line, lp-line); | |
2157 return; | |
2158 } | |
2159 linp = lp; | |
2160 } | |
2161 | |
2162 static void | |
2163 checkpoint(void) | |
2164 { | |
2165 long *a1, *a2; | |
2166 | |
2167 if (undzero && globp == NULL) { | |
2168 for (a1 = zero+1, a2 = undzero+1; a1 <= dol; a1++, a2++) | |
2169 *a2 = *a1; | |
2170 unddot = &undzero[dot-zero]; | |
2171 unddol = &undzero[dol-zero]; | |
2172 for (a1 = names, a2 = undnames; a1 < &names[26]; a1++, a2++) | |
2173 *a2 = *a1; | |
2174 } | |
2175 } | |
2176 | |
2177 #define swap(a, b) (t = a, a = b, b = t) | |
2178 | |
2179 static void | |
2180 undo(void) | |
2181 { | |
2182 long *t; | |
2183 | |
2184 if (undzero == NULL) | |
2185 error("no undo information saved"); | |
2186 swap(zero, undzero); | |
2187 swap(dot, unddot); | |
2188 swap(dol, unddol); | |
2189 swap(names, undnames); | |
2190 } | |
2191 | |
2192 static int | |
2193 maketf(int fd) | |
2194 { | |
2195 char *tmpdir; | |
2196 | |
2197 if (fd == -1) { | |
2198 if ((tmpdir = getenv("TMPDIR")) == NULL || | |
2199 (fd = creatf(tmpdir)) < 0) | |
2200 if ((fd = creatf("/var/tmp")) < 0 && | |
2201 (fd = creatf("/tmp")) < 0) | |
2202 error("cannot create temporary file"); | |
2203 } else | |
2204 ftruncate(fd, 0); /* blkio() will seek to 0 anyway */ | |
2205 return fd; | |
2206 } | |
2207 | |
2208 static int | |
2209 creatf(const char *tmpdir) | |
2210 { | |
2211 if (strlen(tmpdir) >= sizeof tfname - 9) | |
2212 return -1; | |
2213 strcpy(tfname, tmpdir); | |
2214 strcat(tfname, "/eXXXXXX"); | |
2215 return mkstemp(tfname); | |
2216 } | |
2217 | |
2218 static int | |
2219 sopen(const char *fn, int rdwr) | |
2220 { | |
2221 int pf[2], fd = -1; | |
2222 | |
2223 if (fn[0] == '!') { | |
2224 fn++; | |
2225 if (pipe(pf) < 0) | |
2226 error("write or open on pipe failed"); | |
2227 switch (pipid = fork()) { | |
2228 case 0: | |
2229 if (rdwr == READ) | |
2230 dup2(pf[1], 1); | |
2231 else | |
2232 dup2(pf[0], 0); | |
2233 close(pf[0]); | |
2234 close(pf[1]); | |
2235 sigset(SIGHUP, oldhup); | |
2236 sigset(SIGQUIT, oldquit); | |
2237 sigset(SIGPIPE, oldpipe); | |
2238 execl(SHELL, "sh", "-c", fn, NULL); | |
2239 _exit(0100); | |
2240 default: | |
2241 close(pf[rdwr == READ ? 1 : 0]); | |
2242 fd = pf[rdwr == READ ? 0 : 1]; | |
2243 break; | |
2244 case -1: | |
2245 error("fork failed - try again"); | |
2246 } | |
2247 } else if (rdwr == READ) | |
2248 fd = open(fn, O_RDONLY); | |
2249 else if (rdwr == EXIST) | |
2250 fd = open(fn, O_WRONLY); | |
2251 else /*if (rdwr == WRITE)*/ | |
2252 fd = creat(fn, 0666); | |
2253 if (fd >= 0 && rdwr == READ) | |
2254 readop = 1; | |
2255 if (fd >= 0) | |
2256 fstat(fd, &fstbuf); | |
2257 return fd; | |
2258 } | |
2259 | |
2260 static void | |
2261 sclose(int fd) | |
2262 { | |
2263 int status; | |
2264 | |
2265 close(fd); | |
2266 if (pipid >= 0) { | |
2267 while (wait(&status) != pipid); | |
2268 pipid = -1; | |
2269 } | |
2270 readop = 0; | |
2271 } | |
2272 | |
2273 static void | |
2274 fspec(const char *lp) | |
2275 { | |
2276 struct termios ts; | |
2277 const char *cp; | |
2278 | |
2279 freetabs(); | |
2280 maxlength = 0; | |
2281 if (tcgetattr(1, &ts) < 0 | |
2282 #ifdef TAB3 | |
2283 || (ts.c_oflag&TAB3) == 0 | |
2284 #endif | |
2285 ) | |
2286 return; | |
2287 while (lp[0]) { | |
2288 if (lp[0] == '<' && lp[1] == ':') | |
2289 break; | |
2290 lp++; | |
2291 } | |
2292 if (lp[0]) { | |
2293 lp += 2; | |
2294 while ((cp = ftok(&lp)) != NULL) { | |
2295 switch (*cp) { | |
2296 case 't': | |
2297 freetabs(); | |
2298 if ((tabstops = tabstring(&cp[1])) == NULL) | |
2299 goto err; | |
2300 break; | |
2301 case 's': | |
2302 maxlength = atoi(&cp[1]); | |
2303 break; | |
2304 case 'm': | |
2305 case 'd': | |
2306 case 'e': | |
2307 break; | |
2308 case ':': | |
2309 if (cp[1] == '>') { | |
2310 if (tabstops == NULL) | |
2311 if ((tabstops = tabstring("0")) | |
2312 == NULL) | |
2313 goto err; | |
2314 return; | |
2315 } | |
2316 /*FALLTHRU*/ | |
2317 default: | |
2318 err: freetabs(); | |
2319 maxlength = 0; | |
2320 errput("PWB spec problem", NULL); | |
2321 return; | |
2322 } | |
2323 } | |
2324 } | |
2325 } | |
2326 | |
2327 static const char * | |
2328 ftok(const char **lp) | |
2329 { | |
2330 const char *cp; | |
2331 | |
2332 while (**lp && **lp != ':' && (**lp == ' ' || **lp == '\t')) | |
2333 (*lp)++; | |
2334 cp = *lp; | |
2335 while (**lp && **lp != ':' && **lp != ' ' && **lp != '\t') | |
2336 (*lp)++; | |
2337 return cp; | |
2338 } | |
2339 | |
2340 static struct tabulator * | |
2341 repetitive(int repetition) | |
2342 { | |
2343 struct tabulator *tp, *tabspec; | |
2344 int col, i; | |
2345 | |
2346 if ((tp = tabspec = calloc(1, sizeof *tp)) == NULL) | |
2347 return NULL; | |
2348 tp->t_rep = repetition; | |
2349 if (repetition > 0) { | |
2350 for (col = 1+repetition, i = 0; i < 22; col += repetition) { | |
2351 if ((tp->t_nxt = calloc(1, sizeof *tp)) == NULL) | |
2352 return NULL; | |
2353 tp = tp->t_nxt; | |
2354 tp->t_tab = col; | |
2355 } | |
2356 } | |
2357 return tabspec; | |
2358 } | |
2359 | |
2360 #define blank(c) ((c) == ' ' || (c) == '\t') | |
2361 | |
2362 static struct tabulator * | |
2363 tablist(const char *s) | |
2364 { | |
2365 struct tabulator *tp, *tabspec; | |
2366 char *x; | |
2367 int prev = 0, val; | |
2368 | |
2369 if ((tp = tabspec = calloc(1, sizeof *tp)) == NULL) | |
2370 return NULL; | |
2371 for (;;) { | |
2372 while (*s == ',') | |
2373 s++; | |
2374 if (*s == '\0' || blank(*s) || *s == ':') | |
2375 break; | |
2376 val = strtol(s, &x, 10); | |
2377 if (*s == '+') | |
2378 val += prev; | |
2379 prev = val; | |
2380 if (*s == '-' || (*x != ',' && !blank(*x) && *x != ':' && | |
2381 *x != '\0')) | |
2382 return NULL; | |
2383 s = x; | |
2384 if ((tp->t_nxt = calloc(1, sizeof *tp)) == NULL) | |
2385 return NULL; | |
2386 tp = tp->t_nxt; | |
2387 tp->t_tab = val; | |
2388 } | |
2389 return tabspec; | |
2390 } | |
2391 | |
2392 static struct tabulator * | |
2393 tabstring(const char *s) | |
2394 { | |
2395 const struct { | |
2396 const char *c_nam; | |
2397 const char *c_str; | |
2398 } canned[] = { | |
2399 { "a", "1,10,16,36,72" }, | |
2400 { "a2", "1,10,16,40,72" }, | |
2401 { "c", "1,8,12,16,20,55" }, | |
2402 { "c2", "1,6,10,14,49" }, | |
2403 { "c3", "1,6,10,14,18,22,26,30,34,38,42,46,50,54,58,62,67" }, | |
2404 { "f", "1,7,11,15,19,23" }, | |
2405 { "p", "1,5,9,13,17,21,25,29,33,37,41,45,49,53,57,61" }, | |
2406 { "s", "1,10,55" }, | |
2407 { "u", "1,12,20,44" }, | |
2408 { 0, 0 } | |
2409 }; | |
2410 | |
2411 int i, j; | |
2412 | |
2413 if (s[0] == '-') { | |
2414 if (s[1] >= '0' && s[1] <= '9' && ((i = atoi(&s[1])) != 0)) | |
2415 return repetitive(i); | |
2416 for (i = 0; canned[i].c_nam; i++) { | |
2417 for (j = 0; canned[i].c_nam[j]; j++) | |
2418 if (s[j+1] != canned[i].c_nam[j]) | |
2419 break; | |
2420 if ((s[j+1]=='\0' || s[j+1]==':' || blank(s[j+1])) && | |
2421 canned[i].c_nam[j] == '\0') | |
2422 return tablist(canned[i].c_str); | |
2423 } | |
2424 return NULL; | |
2425 } else | |
2426 return tablist(s); | |
2427 } | |
2428 | |
2429 static void | |
2430 freetabs(void) | |
2431 { | |
2432 struct tabulator *tp; | |
2433 | |
2434 tp = tabstops; | |
2435 while (tp) { | |
2436 tabstops = tp->t_nxt; | |
2437 free(tp); | |
2438 tp = tabstops; | |
2439 } | |
2440 } | |
2441 | |
2442 static void | |
2443 expand(const char *s) | |
2444 { | |
2445 struct tabulator *tp = tabstops; | |
2446 int col = 0, n = 1, m, tabcnt = 0, nspc; | |
2447 wchar_t wc; | |
2448 | |
2449 while (*s) { | |
2450 nspc = 0; | |
2451 switch (*s) { | |
2452 case '\n': | |
2453 putchr('\0'); | |
2454 s++; | |
2455 continue; | |
2456 case '\t': | |
2457 if (tp) { | |
2458 if (tp->t_rep) { | |
2459 if (col % tp->t_rep == 0) { | |
2460 nspc++; | |
2461 col++; | |
2462 } | |
2463 while (col % tp->t_rep) { | |
2464 nspc++; | |
2465 col++; | |
2466 } | |
2467 break; | |
2468 } | |
2469 while (tp && (col>tp->t_tab || tp->t_tab == 0)) | |
2470 tp = tp->t_nxt; | |
2471 if (tp && col == tp->t_tab) { | |
2472 nspc++; | |
2473 col++; | |
2474 tp = tp->t_nxt; | |
2475 } | |
2476 if (tp) { | |
2477 while (col < tp->t_tab) { | |
2478 nspc++; | |
2479 col++; | |
2480 } | |
2481 tp = tp->t_nxt; | |
2482 break; | |
2483 } | |
2484 } | |
2485 tabcnt = 1; | |
2486 nspc++; | |
2487 break; | |
2488 default: | |
2489 if (mb_cur_max>1 && (n=mbtowc(&wc, s, mb_cur_max))>0) { | |
2490 if ((m = wcwidth(wc)) > 0) | |
2491 col += m; | |
2492 } else { | |
2493 col++; | |
2494 n = 1; | |
2495 } | |
2496 } | |
2497 if (maxlength && col > maxlength) { | |
2498 putstr("\ntoo long"); | |
2499 break; | |
2500 } | |
2501 if (nspc) { | |
2502 while (nspc--) | |
2503 putchr(' '); | |
2504 s++; | |
2505 } else | |
2506 while (n--) | |
2507 putchr(*s++); | |
2508 } | |
2509 if (tabcnt) | |
2510 putstr("\ntab count"); | |
2511 putchr('\n'); | |
2512 } | |
2513 | |
2514 static wint_t | |
2515 GETWC(char *mb) | |
2516 { | |
2517 int c, n; | |
2518 | |
2519 n = 1; | |
2520 mb[0] = c = GETC(); | |
2521 mb[1] = '\0'; | |
2522 if (mb_cur_max > 1 && c&0200 && c != EOF) { | |
2523 int m; | |
2524 wchar_t wc; | |
2525 | |
2526 while ((m = mbtowc(&wc, mb, mb_cur_max)) < 0 && n<mb_cur_max) { | |
2527 mb[n++] = c = GETC(); | |
2528 mb[n] = '\0'; | |
2529 if (c == '\n' || c == EOF) | |
2530 break; | |
2531 } | |
2532 if (m != n) | |
2533 ERROR(67); | |
2534 return wc; | |
2535 } else | |
2536 return c; | |
2537 } | |
2538 | |
2539 static void | |
2540 growlb(const char *msg) | |
2541 { | |
2542 char *olb = linebuf; | |
2543 int i; | |
2544 | |
2545 LBSIZE += 512; | |
2546 if ((linebuf = realloc(linebuf, LBSIZE)) == NULL || | |
2547 (genbuf = realloc(genbuf, LBSIZE)) == NULL) | |
2548 error(msg); | |
2549 if (linebuf != olb) { | |
2550 loc1 += linebuf - olb; | |
2551 loc2 += linebuf - olb; | |
2552 for (i = 0; i < NBRA; i++) { | |
2553 if (braslist[i]) | |
2554 braslist[i] += linebuf - olb; | |
2555 if (braelist[i]) | |
2556 braelist[i] += linebuf - olb; | |
2557 } | |
2558 } | |
2559 } | |
2560 | |
2561 static void | |
2562 growrhs(const char *msg) | |
2563 { | |
2564 RHSIZE += 256; | |
2565 if ((rhsbuf = realloc(rhsbuf, RHSIZE)) == NULL) | |
2566 error(msg); | |
2567 } | |
2568 | |
2569 static void | |
2570 growfn(const char *msg) | |
2571 { | |
2572 FNSIZE += 64; | |
2573 if ((savedfile = realloc(savedfile, FNSIZE)) == NULL || | |
2574 (file = realloc(file, FNSIZE)) == NULL) | |
2575 error(msg); | |
2576 if (FNSIZE == 64) | |
2577 file[0] = savedfile[0] = 0; | |
2578 } | |
2579 | |
2580 #if defined (SUS) || defined (S42) || defined (SU3) | |
2581 union ptrstore { | |
2582 void *vp; | |
2583 char bp[sizeof (void *)]; | |
2584 }; | |
2585 | |
2586 static void * | |
2587 fetchptr(const char *bp) | |
2588 { | |
2589 union ptrstore u; | |
2590 int i; | |
2591 | |
2592 for (i = 0; i < sizeof (void *); i++) | |
2593 u.bp[i] = bp[i]; | |
2594 return u.vp; | |
2595 } | |
2596 | |
2597 static void | |
2598 storeptr(void *vp, char *bp) | |
2599 { | |
2600 union ptrstore u; | |
2601 int i; | |
2602 | |
2603 u.vp = vp; | |
2604 for (i = 0; i < sizeof (void *); i++) | |
2605 bp[i] = u.bp[i]; | |
2606 } | |
2607 | |
2608 #define add(c) ((i>=LBSIZE ? (growlb("regular expression overflow"),0) : 0), \ | |
2609 genbuf[i++] = (c)) | |
2610 | |
2611 #define copy(s) { \ | |
2612 int m; \ | |
2613 for (m = 0; m==0 || s[m]; m++) \ | |
2614 add(s[m]); \ | |
2615 } | |
2616 | |
2617 static char * | |
2618 compile(char *unused, char *ep, const char *endbuf, int seof) | |
2619 { | |
2620 INIT | |
2621 int c, d, i; | |
2622 regex_t *rp; | |
2623 char *op; | |
2624 char mb[MB_LEN_MAX+1]; | |
2625 | |
2626 op = ep; | |
2627 ep += 2; | |
2628 if ((rp = fetchptr(ep)) == NULL) { | |
2629 if ((rp = calloc(1, sizeof *rp)) == NULL) | |
2630 ERROR(50); | |
2631 storeptr(rp, ep); | |
2632 } | |
2633 ep += sizeof (void *); | |
2634 i = 0; | |
2635 nbra = 0; | |
2636 do { | |
2637 if ((c = GETWC(mb)) == seof) | |
2638 add('\0'); | |
2639 else if (c == '\\') { | |
2640 copy(mb); | |
2641 c = GETWC(mb); | |
2642 if (c == '(') | |
2643 nbra++; | |
2644 goto normchar; | |
2645 } else if (c == '[') { | |
2646 add(c); | |
2647 d = EOF; | |
2648 do { | |
2649 c = GETWC(mb); | |
2650 if (c == EOF || c == '\n') | |
2651 ERROR(49); | |
2652 copy(mb); | |
2653 if (d=='[' && (c==':' || c=='.' || c=='=')) { | |
2654 d = c; | |
2655 do { | |
2656 c = GETWC(mb); | |
2657 if (c == EOF || c == '\n') | |
2658 ERROR(49); | |
2659 copy(mb); | |
2660 } while (c != d || PEEKC() != ']'); | |
2661 c = GETWC(mb); | |
2662 copy(mb); | |
2663 c = EOF; | |
2664 } | |
2665 d = c; | |
2666 } while (c != ']'); | |
2667 } else { | |
2668 if (c == EOF || c == '\n') { | |
2669 if (c == '\n') | |
2670 UNGETC(c); | |
2671 mb[0] = c = '\0'; | |
2672 } | |
2673 if (c == '\0') | |
2674 nodelim = 1; | |
2675 normchar: copy(mb); | |
2676 } | |
2677 } while (genbuf[i-1] != '\0'); | |
2678 if (genbuf[0]) { | |
2679 int reflags = 0; | |
2680 | |
2681 #ifdef REG_ANGLES | |
2682 reflags |= REG_ANGLES; | |
2683 #endif | |
2684 #if defined (SU3) && defined (REG_AVOIDNULL) | |
2685 reflags |= REG_AVOIDNULL; | |
2686 #endif | |
2687 if (op[0]) | |
2688 regfree(rp); | |
2689 op[0] = 0; | |
2690 switch (regcomp(rp, genbuf, reflags)) { | |
2691 case 0: | |
2692 break; | |
2693 case REG_ESUBREG: | |
2694 ERROR(25); | |
2695 /*NOTREACHED*/ | |
2696 case REG_EBRACK: | |
2697 ERROR(49); | |
2698 /*NOTREACHED*/ | |
2699 case REG_EPAREN: | |
2700 ERROR(42); | |
2701 /*NOTREACHED*/ | |
2702 case REG_BADBR: | |
2703 case REG_EBRACE: | |
2704 ERROR(45); | |
2705 /*NOTREACHED*/ | |
2706 case REG_ERANGE: | |
2707 ERROR(11); | |
2708 /*NOTREACHED*/ | |
2709 case REG_ESPACE: | |
2710 ERROR(50); | |
2711 /*NOTREACHED*/ | |
2712 default: | |
2713 ERROR(-1); | |
2714 } | |
2715 op[0] = 1; | |
2716 circf = op[1] = genbuf[0] == '^'; | |
2717 } else if (op[0]) { | |
2718 circf = op[1]; | |
2719 } else | |
2720 ERROR(41); | |
2721 return ep + sizeof (void *); | |
2722 } | |
2723 | |
2724 static int | |
2725 step(const char *lp, const char *ep) | |
2726 { | |
2727 regex_t *rp; | |
2728 regmatch_t bralist[NBRA+1]; | |
2729 int eflag = 0; | |
2730 int res; | |
2731 int i; | |
2732 | |
2733 rp = fetchptr(&ep[2]); | |
2734 if (ep[0] == 0) | |
2735 return 0; | |
2736 if (locs) | |
2737 eflag |= REG_NOTBOL; | |
2738 if ((res = regexec(rp, lp, needsub? NBRA+1 : 0, bralist, eflag)) == 0 && | |
2739 needsub) { | |
2740 loc1 = (char *)lp + bralist[0].rm_so; | |
2741 loc2 = (char *)lp + bralist[0].rm_eo; | |
2742 for (i = 1; i <= NBRA; i++) { | |
2743 if (bralist[i].rm_so != -1) { | |
2744 braslist[i-1] = (char *)lp + bralist[i].rm_so; | |
2745 braelist[i-1] = (char *)lp + bralist[i].rm_eo; | |
2746 } else | |
2747 braslist[i-1] = braelist[i-1] = NULL; | |
2748 } | |
2749 } | |
2750 return res == 0; | |
2751 } | |
2752 #endif /* SUS || S42 || SU3 */ | |
2753 | |
2754 static void | |
2755 help(void) | |
2756 { | |
2757 const char *desc[] = { | |
2758 "(.)a append up to .", | |
2759 "(.)b[n] browse n lines", | |
2760 "(.,.)c change up to .", | |
2761 "(.,.)d delete lines", | |
2762 "e [file] edit file", | |
2763 "E [file] force edit", | |
2764 "f [file] print or set file", | |
2765 "(1,$)g/RE/cmd global cmd", | |
2766 "(1,$)G/RE/ interactive global", | |
2767 "h print last error", | |
2768 "H toggle error messages", | |
2769 "help print this screen", | |
2770 "(.)i insert up to .", | |
2771 "(.,.+1)j join lines", | |
2772 "(.)kx mark line with x", | |
2773 "(.,.)l list lines", | |
2774 "(.,.)ma move lines to a", | |
2775 "(.,.)n number lines", | |
2776 "N revert n and p", | |
2777 "(.)o[n] show n lines of context", | |
2778 "(.,.)p print lines", | |
2779 "P toggle prompt", | |
2780 "q quit", | |
2781 "Q force quit", | |
2782 "($)r read file", | |
2783 "(.,.)s/RE/repl/ search and replace", | |
2784 "(.,.)s/RE/rp/g replace all occurrences", | |
2785 "(.,.)s/RE/rp/n replace n-th occurrence", | |
2786 "(.,.)ta transfer lines to a", | |
2787 "u undo last change", | |
2788 "(1,$)v/RE/cmd reverse global", | |
2789 "(1,$)V/RE/ reverse i/a global", | |
2790 "(1,$)w [file] write file", | |
2791 "(1,$)W [file] append to file", | |
2792 "z write buffer and quit", | |
2793 "($)= print line number", | |
2794 "!command execute shell command", | |
2795 "(.+1)<newline> print one line", | |
2796 "/RE find RE forwards", | |
2797 "?RE find RE backwards", | |
2798 "1 first line", | |
2799 ". current line", | |
2800 "$ last line", | |
2801 ", 1,$", | |
2802 "; .,$", | |
2803 NULL | |
2804 }; | |
2805 char line[100]; | |
2806 int c, half, i, k; | |
2807 | |
2808 half = (sizeof desc / sizeof *desc) / 2; | |
2809 for (i = 0; i < half && desc[i]; i++) { | |
2810 c = 0; | |
2811 for (k = 0; desc[i][k]; k++) | |
2812 line[c++] = desc[i][k]; | |
2813 if (desc[i+half]) { | |
2814 while (c < 40) | |
2815 line[c++] = ' '; | |
2816 for (k = 0; desc[i+half][k]; k++) | |
2817 line[c++] = desc[i+half][k]; | |
2818 } | |
2819 line[c] = 0; | |
2820 puts(line); | |
2821 } | |
2822 } |