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.
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.
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]
23 * Copyright (c) 1988, 2010, Oracle and/or its affiliates. All rights reserved.
26 /* Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T */
27 /* All Rights Reserved */
37 #include <sys/types.h>
44 static const char *msgtab
[] =
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 */
61 "line out of range", /* 15 */
62 "bad number", /* 16 */
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 */
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 */
82 "illegal or missing delimiter", /* 36 */
84 "replacement string too long", /* 38 */
85 "illegal move destination", /* 39 */
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 */
106 "the x command has become X (upper case)", /* 60 */
107 "Warning: 'w' may destroy input file "
108 "(due to `illegal char' read earlier)",
110 "Caution: 'q' may lose data in buffer;"
111 " 'w' may destroy input file",
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 */
128 #include <sys/types.h>
129 #include <sys/stat.h>
130 #include <sys/statvfs.h>
136 #include <wchar.h> /* I18N */
137 #include <wctype.h> /* I18N */
138 #include <widec.h> /* I18N */
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)
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
154 /* size of substitution replacement pattern buffer */
155 #define RHSIZE (LINE_MAX*2)
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 */
173 static struct Fspec fss
;
177 static char line
[70];
178 static char *linp
= line
;
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
;
191 static char rhsbuf
[RHSIZE
];
196 typedef struct lin
*LINE
;
201 static LINE fendcore
;
204 static LINE savdol
, savdot
;
207 static char genbuf
[LBSIZE
];
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. */
219 static void (*oldhup
)(), (*oldintr
)();
220 static void (*oldquit
)(), (*oldpipe
)();
221 static void quit(int) __NORETURN
;
222 static int vflag
= 1;
227 /* Flag for determining if file being read is encrypted */
229 static int xcode
= -1;
230 static char crbuf
[LBSIZE
];
234 static int tpermflag
;
237 static int tfile
= -1;
241 static char ibuff
[LBSIZE
];
242 static int iblock
= -1;
243 static char obuff
[LBSIZE
];
244 static int oblock
= -1;
247 static long savnames
[26], names
[26];
252 static int fflg
, shflg
;
253 static char prompt
[16] = "*";
262 static int flag28
= 0; /* Prevents write after a partial read */
263 static int save28
= 0; /* Flag whether buffer empty at start of read */
265 static char *name
= "SHELL";
266 static char *rshell
= "/usr/lib/rsh";
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
);
340 static wchar_t get_wchr(void);
342 static struct stat Fl
, Tf
;
344 static struct statvfs U
;
345 static int Short
= 0;
346 static mode_t oldmask
; /* No umask while writing */
348 static jmp_buf savej
;
351 int nulls
; /* Null count */
355 static int errcnt
= 0;
365 main(int argc
, char **argv
)
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 */
374 (void) textdomain(TEXT_DOMAIN
);
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
);
385 while (--p1
>= *argv
)
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)
395 home
= getenv("HOME");
397 while ((c
= getopt(argc
, argv
, "sp:qxC")) != EOF
) {
405 strncpy(prompt
, optarg
, sizeof (prompt
)-1);
410 signal(SIGQUIT
, SIG_DFL
);
425 (void) fprintf(stderr
, gettext(
426 "Usage: ed [- | -s] [-p string] [-x] [-C] [file]\n"
427 " red [- | -s] [-p string] [-x] [-C] [file]\n"));
431 if (argv
[optind
] && strcmp(argv
[optind
], "-") == 0 &&
432 strcmp(argv
[optind
-1], "--") != 0) {
439 argc
= argc
- optind
;
440 argv
= &argv
[optind
];
446 kflag
= run_setkey(&perm
[0], getkey(msgtab
[66]));
448 puts(gettext(msgtab
[64]));
458 if (strlen(p1
) >= (size_t)FNSIZE
) {
459 puts(gettext("file name too long"));
465 while (*p2
++ = *p1
++)
469 } else /* editing with no file so set savtime to 0 */
472 if ((tfname
= tempnam("", "ea")) == NULL
) {
473 puts(gettext(msgtab
[69]));
477 fendcore
= (LINE
)sbrk(0);
479 if (oldintr
!= SIG_IGN
)
480 signal(SIGINT
, onintr
);
481 if (oldhup
!= SIG_IGN
)
482 signal(SIGHUP
, onhup
);
504 if (shflg
&& globp
== 0)
505 write(1, gettext(prompt
), strlen(gettext(prompt
)));
508 if ((c
= getchr()) == ',') {
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) {
521 } else if (c
== ';') {
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) {
538 if ((a1
= address()) == 0) {
543 if ((c
= getchr()) == ';') {
556 if (!globflg
) save();
557 append(gettty
, addr2
);
565 append(gettty
, addr1
-1);
567 /* XPG4 - If no new lines are inserted, then the current */
568 /* line becomes the line after the lines deleted. */
570 if (((linebuf
[0] != '.') || (dot
== (addr1
-1))) &&
586 if (vflag
&& fchange
) {
599 if (!ncflg
) /* there is a filename */
637 if (!globflg
) save();
638 append(gettty
, addr2
-1);
651 if (!globflg
) save();
656 if ((c
= getchr()) < 'a' || c
> 'z')
661 names
[c
-'a'] = addr2
->cur
& ~01;
695 puts(getaline((a1
++)->cur
));
696 } while (a1
<= addr2
);
715 save28
= (dol
!= fendcore
);
716 if (crflag
== 2 || crflag
== -2)
717 crflag
= -1; /* restore crflag for next file */
719 if ((io
= eopen(file
, O_RDONLY
)) < 0) {
721 /* if first entering editor and file does not exist */
722 /* set saved access time to 0 */
726 if (c
== 'e' && vflag
== 0)
729 if (errno
== ENOENT
) {
735 /* get last mod time of file */
736 /* eflg - entered editor with ed or e */
747 if (!globflg
&& (c
== 'r')) save();
748 append(getfile
, addr2
);
757 if (!globflg
) save();
758 substitute(globp
!= 0);
791 /* on NULL-RE condition do not generate error */
793 if ((linebuf
[0] != '.') && (zero
!= dol
) &&
794 (addr1
<= zero
|| addr2
> dol
))
798 io
= eopen(file
, O_WRONLY
);
799 n
= 1; /* set n so newtime will not execute */
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)
815 * Must determine if file is
823 * Determine if there are enough free blocks on system
825 if (!Short
&& statvfs(file
, &U
) == 0 &&
826 U
.f_bfree
< ((Tf
.st_size
/U
.f_frsize
) + 100)) {
832 p1
= savedfile
; /* The current filename */
835 if (c
== 'w' && Fl
.st_nlink
== 1 && ISREG(lFl
)) {
836 if (close(open(file
, O_WRONLY
)) < 0)
842 * If funlink equals one it means that
843 * funny points to a valid file which must
844 * be unlinked when interrupted.
848 if ((io
= creat(funny
, FMODE(Fl
))) >= 0) {
849 chown(funny
, Fl
.st_uid
, Fl
.st_gid
);
850 chmod(funny
, FMODE(Fl
));
854 if (rename(funny
, file
))
857 /* if filenames are the same */
860 /* check if entire buffer was written */
862 if (((addr1
== zero
) ||
863 (addr1
== (zero
+ 1))) &&
868 if (fchange
== 1 && m
!= 0)
873 n
= 1; /* set n so newtime will not execute */
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)
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
;
895 * C is same as X, but always assume input files are
907 (void) crypt_close(perm
);
909 if ((kflag
= run_setkey(&perm
[0], getkey(msgtab
[66]))) == -1) {
922 count
= (addr2
-zero
)&077777;
955 int n
, relerr
, retval
;
961 if ('0' <= c
&& c
<= '9') {
966 } while ((c
= getchr()) >= '0' && c
<= '9');
1028 if ((c
= getchr()) < 'a' || c
> 'z')
1030 for (a1
= zero
; a1
<= dol
; a1
++)
1031 if (names
[c
-'a'] == (a1
->cur
& ~01))
1041 /* on NULL-RE condition do not generate error */
1043 if ((linebuf
[0] != '.') && (a1
< zero
|| a1
> dol
))
1056 addr1
= addr2
= dot
;
1083 /* on NULL-RE condition do not generate error */
1085 if ((linebuf
[0] != '.') && (addr1
<= zero
|| addr2
> dol
))
1092 /* for the c and i commands 0 equal to 1 address */
1093 if (addr1
== zero
) {
1096 if (addr2
== zero
) {
1107 if ((c
= getchr()) == '\n')
1109 if (c
== 'p' || c
== 'l' || c
== 'n') {
1111 if (c
== 'l') listf
++;
1112 if (c
== 'n') listn
++;
1113 if ((c
= getchr()) == '\n')
1128 if (c
== '\n' || c
== EOF
) {
1130 if (*p1
== 0 && comm
!= 'f')
1132 /* ncflg set means do not get mod time of file */
1133 /* since no filename followed f */
1137 while (*p2
++ = *p1
++)
1144 while ((c
= getchr()) == ' ')
1147 ++Xqt
, c
= getchr();
1155 if (c
== EOF
|| (c
== ' ' && !Xqt
))
1157 } while ((c
= getchr()) != '\n');
1166 if (savedfile
[0] == 0 || comm
== 'e' || comm
== 'f') {
1169 while (*p1
++ = *p2
++)
1200 puts(gettext(" nulls replaced by '\\0'"));
1210 signal(SIGINT
, onintr
);
1214 if (funlink
) unlink(funny
); /* remove tmp file */
1215 /* if interrupted a read, only part of file may be in buffer */
1217 sprintf(tstring
, "\007read may be incomplete - beware!\007");
1218 puts(gettext(tstring
));
1227 signal(SIGINT
, SIG_IGN
);
1228 signal(SIGHUP
, SIG_IGN
);
1230 * if there are lines in file and file was not written
1231 * since last update, save in ed.hup, or $HOME/ed.hup
1233 if (dol
> zero
&& fchange
== 1) {
1236 io
= creat("ed.hup",
1237 S_IRUSR
|S_IWUSR
|S_IRGRP
|S_IWGRP
|S_IROTH
|S_IWOTH
);
1238 if (io
< 0 && home
) {
1241 fn
= (char *)calloc(strlen(home
) + 8, sizeof (char));
1244 strcat(fn
, "/ed.hup");
1245 io
= creat(fn
, S_IRUSR
|S_IWUSR
|S_IRGRP
|S_IWGRP
1263 if (code
== 28 && save28
== 0) {
1277 #ifdef NULLS /* Not really nulls, but close enough */
1278 /* This is a bug because of buffering */
1279 if (code
== 28) /* illegal char. */
1282 /* Cant open file or file does not exist */
1283 if ((code
== 3) || (code
== 68)) {
1297 lseek(0, (long)0, 2);
1303 while ((c
= getchr()) != '\n' && c
!= EOF
)
1313 return (0); /* Non-fatal error. */
1322 if (lastc
= peekc
) {
1327 if ((lastc
= (unsigned char)*globp
++) != 0)
1332 if (read(0, &c
, 1) <= 0)
1333 return (lastc
= EOF
);
1334 lastc
= (unsigned char)c
;
1347 while ((c
= getchr()) != '\n') {
1357 if (p
> &linebuf
[LBSIZE
-1])
1361 if (linebuf
[0] == '.' && linebuf
[1] == 0)
1365 * POSIX.2/XPG4 explicitly says no to this:
1367 * in Solaris backslash followed by special character "." is
1368 * special character "." itself; (so terminating input mode can be
1371 * however, POSIX2/XPG4 says, input mode is terminated by
1372 * entering line consisting of only 2 characters: ".\n"
1374 * if (linebuf[0]=='\\' && linebuf[1]=='.' && linebuf[2]==0) {
1392 if ((ninbuf
= read(io
, genbuf
, LBSIZE
)-1) < 0)
1394 puts(gettext("'\\n' appended"));
1400 if (isencrypt(genbuf
, ninbuf
+ 1))
1407 if (run_crypt(count
, genbuf
, ninbuf
+1, perm
) == -1)
1410 if (lp
>= &linebuf
[LBSIZE
]) {
1414 if ((*lp
++ = c
= *fp
++) == 0) {
1425 } while (c
!= '\n');
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 = ")));
1432 count
= (++dot
-zero
)&077777;
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 = ")));
1458 count
= (a1
-zero
-1)&077777;
1467 if (run_crypt(count
-n
, genbuf
, n
, perm
) == -1)
1469 if (write(io
, genbuf
, n
) != n
)
1474 if (dol
->cur
== 0L)break; /* Allow write of null file */
1476 if ((*fp
++ = *lp
++) == 0) {
1481 } while (a1
<= addr2
);
1484 if (run_crypt(count
-n
, genbuf
, n
, perm
) == -1)
1486 if (write(io
, genbuf
, n
) != n
)
1491 append(int (*f
)(void), LINE a
)
1498 while ((*f
)() == 0) {
1499 if (dol
>= endcore
) {
1500 if ((int)sbrk(512 * sizeof (struct lin
)) == -1) {
1512 (--a2
)->cur
= (--a1
)->cur
;
1523 static char savcmd
[LBSIZE
]; /* last command */
1524 char curcmd
[LBSIZE
]; /* current command */
1525 char *psavcmd
, *pcurcmd
, *psavedfile
;
1526 int endflg
= 1, shflg
= 0;
1534 /* read command til end */
1537 * a '!' found in beginning of command is replaced with the saved
1538 * command. a '%' found in command is replaced with the current
1548 while (*pcurcmd
++ = *psavcmd
++)
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
;
1565 if (savedfile
[0] == 0)
1568 psavedfile
= savedfile
;
1569 while (pcurcmd
< curcmd
+ LBSIZE
&&
1570 (*pcurcmd
++ = *psavedfile
++))
1575 } else if (c
== '\\') {
1579 if ((len
= wctomb(pcurcmd
, c
)) <= 0) {
1580 *pcurcmd
= (unsigned char)c
;
1586 /* end of command hit */
1593 strcpy(savcmd
, curcmd
);
1595 if ((pid
= fork()) == 0) {
1596 signal(SIGHUP
, oldhup
);
1597 signal(SIGQUIT
, oldquit
);
1599 execlp(_PATH_BSHELL
, "sh", "-c", curcmd
, NULL
);
1602 savint
= signal(SIGINT
, SIG_IGN
);
1603 while ((rpid
= wait(&retcode
)) != pid
&& rpid
!= (pid_t
)-1)
1605 signal(SIGINT
, savint
);
1606 if (vflag
) puts("!");
1612 if (vflag
&& fchange
) {
1620 * For case where user reads in BOTH a good
1641 rdelete(addr1
, addr2
);
1645 rdelete(LINE ad1
, LINE ad2
)
1654 (a1
++)->cur
= (a2
++)->cur
;
1669 for (a1
= zero
+1; (a1
->cur
&01) == 0; a1
++)
1672 for (a2
= a1
+ 1; a2
<= a3
; ) {
1677 (a1
++)->cur
= (a2
++)->cur
;
1692 bp
= getblock(tl
, READ
);
1695 while (*lp
++ = *bp
++)
1697 bp
= getblock(tl
+= 0400, READ
);
1713 bp
= getblock(tl
, WRITE
);
1716 while (*bp
= *lp
++) {
1717 if (*bp
++ == '\n') {
1723 bp
= getblock(tl
+= 0400, WRITE
);
1728 tline
+= (((lp
-linebuf
)+03)>>1)&077776;
1733 getblock(long atl
, long iof
)
1740 off
= (atl
<<1)&0774;
1742 /* bno is limited to 16 bits */
1748 if (bno
== iblock
) {
1757 if (run_crypt(0L, ibuff
, 512, tperm
) == -1)
1759 blkio(iblock
, ibuff
, write
);
1763 blkio(bno
, ibuff
, read
);
1765 if (run_crypt(0L, ibuff
, 512, tperm
) == -1)
1776 if (run_crypt(0L, crbuf
, 512, tperm
) == -1)
1778 blkio(oblock
, crbuf
, write
);
1780 blkio(oblock
, obuff
, write
);
1787 blkio(int b
, char *buf
, ssize_t (*iofcn
)())
1789 lseek(tfile
, (long)b
<<9, 0);
1790 if ((*iofcn
)(tfile
, buf
, 512) != 512) {
1792 (void) error(32); /* Bypass this if writing null file */
1803 (void) close(tfile
);
1804 (void) unlink(tfname
);
1808 for (markp
= names
; markp
< &names
[26]; )
1818 if ((tfile
= open(tfname
, O_CREAT
|O_EXCL
|O_RDWR
,
1819 S_IRUSR
|S_IWUSR
)) < 0) {
1820 puts(gettext(msgtab
[70]));
1828 (void) crypt_close(tperm
);
1830 if (makekey(tperm
)) {
1832 puts(gettext(msgtab
[65]));
1835 brk((char *)fendcore
);
1836 dot
= zero
= dol
= savdot
= savdol
= fendcore
;
1837 flag28
= save28
= 0;
1838 endcore
= fendcore
- sizeof (struct lin
);
1846 char multic
[MB_LEN_MAX
];
1849 char globuf
[LBSIZE
];
1857 if ((n
= _mbftowc(multic
, &l
, getchr
, &peekc
)) <= 0)
1864 while ((c
= get_wchr()) != '\n') {
1868 /* '\\' has special meaning only if preceding a '\n' */
1874 if ((gp
+ (unsigned int)MB_CUR_MAX
) >= &globuf
[LBSIZE
-1])
1876 if ((len
= wctomb(gp
, c
)) <= 0) {
1877 *gp
= (unsigned char)c
;
1886 for (a1
= zero
; a1
<= dol
; a1
++) {
1888 if (a1
>= addr1
&& a1
<= addr2
&& execute(0, a1
) == k
)
1892 * Special case: g/.../d (avoid n^2 algorithm)
1894 if (globuf
[0] == 'd' && globuf
[1] == '\n' && globuf
[2] == '\0') {
1898 for (a1
= zero
; a1
<= dol
; a1
++) {
1920 for (a1
= addr1
; a1
<= addr2
; a1
++) {
1921 lp
= getaline(a1
->cur
);
1923 if (gp
++ > &genbuf
[LBSIZE
-1])
1928 while (*lp
++ = *gp
++)
1930 addr1
->cur
= putline();
1932 rdelete(addr1
+1, addr2
);
1937 substitute(int inglob
)
1942 int ingsav
; /* For saving arg. */
1947 for (a1
= addr1
; a1
<= addr2
; a1
++) {
1948 if (execute(0, a1
) == 0)
1956 if (execute(1, (LINE
)0) == 0)
1961 if (ocerr1
== 0)continue; /* Don't put out-not changed. */
1962 subnewa
= putline();
1965 for (markp
= names
; markp
< &names
[26]; markp
++)
1966 if (*markp
== a1
->cur
)
1976 return; /* Was in global-no error msg allowed. */
1978 (void) error(35); /* Not in global, but not found. */
1980 (void) error(35); /* RE found, but occurrence match failed. */
1989 char multic
[MB_LEN_MAX
];
1991 static char remem
[RHSIZE
];
1992 static int remflg
= -1;
1995 if ((n
= _mbftowc(multic
, &seof
, getchr
, &peekc
)) <= 0)
1997 if (seof
== '\n' || seof
== ' ')
2003 if ((n
= _mbftowc(multic
, &cl
, getchr
, &peekc
)) <= 0)
2007 if (p
>= &rhsbuf
[RHSIZE
])
2009 if ((n
= _mbftowc(multic
, &cl
, getchr
, &peekc
)) <= 0)
2011 } else if (cl
== '\n') {
2016 if (!(globp
&& globp
[0])) {
2021 } else if (cl
== seof
)
2023 if (p
+ n
> &rhsbuf
[RHSIZE
])
2025 (void) strncpy(p
, multic
, n
);
2029 if (rhsbuf
[0] == '%' && rhsbuf
[1] == 0)
2031 * If there isn't a remembered string, it is an error;
2032 * otherwise the right hand side is the previous right
2039 strcpy(rhsbuf
, remem
);
2041 strcpy(remem
, rhsbuf
);
2045 peekc
= getchr(); /* Gets char after third delimiter. */
2047 c
= LBSIZE
; peekc
= 0;
2049 if (peekc
>= '1' && peekc
<= '9') {
2051 peekc
= 0; /* Allows getchr() to get next char. */
2054 if (i
< '0' || i
> '9')
2058 (void) error(20); /* "Illegal suffix" */
2060 peekc
= i
; /* Effectively an unget. */
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.
2078 if ((p2
= linebp
) == 0)
2080 while (*p1
++ = *p2
++)
2092 if (gsubf
> 0 && gsubf
< LBSIZE
) {
2094 if (gsubf
!= numpass
)
2106 sp
= place(sp
, loc1
, loc2
);
2108 } else if (c
== '\\') {
2110 if (c
>= '1' && c
< nbra
+ '1') {
2111 sp
= place(sp
, braslist
[c
-'1'],
2117 if (sp
>= &genbuf
[LBSIZE
])
2121 loc2
= sp
- genbuf
+ linebuf
;
2122 while (*sp
++ = *lp
++)
2123 if (sp
>= &genbuf
[LBSIZE
])
2127 while (*lp
++ = *sp
++)
2132 place(char *sp
, char *l1
, char *l2
)
2137 if (sp
>= &genbuf
[LBSIZE
])
2144 comple(wchar_t seof
)
2150 char multic
[MB_LEN_MAX
];
2153 if ((n
= _mbftowc(multic
, &c
, getchr
, &peekc
)) < 0)
2155 if (n
== 0 || c
== '\n') {
2161 if (c
== seof
&& !cclass
)
2163 if (cclass
&& c
== ']') {
2165 if (cp
> &genbuf
[LBSIZE
-1])
2170 if (c
== '[' && !cclass
) {
2172 if (cp
> &genbuf
[LBSIZE
-1])
2175 if ((n
= _mbftowc(multic
, &c
, getchr
, &peekc
)) < 0)
2177 if (n
== 0 || c
== '\n')
2180 if (c
== '\\' && !cclass
) {
2181 if (cp
> &genbuf
[LBSIZE
-1])
2184 if ((n
= _mbftowc(multic
, &c
, getchr
, &peekc
)) < 0)
2186 if (n
== 0 || c
== '\n')
2189 if (cp
+ n
> &genbuf
[LBSIZE
-1])
2191 (void) strncpy(cp
, multic
, n
);
2195 if (n
!= 0 && c
== '\n')
2197 if (n
== 0 || c
== '\n')
2201 * NULL RE: do not compile a null regular expression; but process
2202 * input with last regular expression encountered
2205 if (genbuf
[0] != '\0') {
2207 expbuf
= compile(genbuf
, (char *)0, (char *)0);
2220 if ((adt
= address()) == 0)
2223 if (!globflg
) save();
2226 append(getcopy
, ad1
++);
2230 for (ad1
= addr1
; ad1
<= ad2
; )
2231 (ad1
++)->cur
&= ~01;
2236 dot
= adt
+ (ad2
-ad1
);
2242 } else if (adt
>= ad2
) {
2253 reverse(LINE a1
, LINE a2
)
2272 (void) getaline((addr1
++)->cur
);
2278 * Handles error code returned from comple() routine: regular expression
2279 * compile and match routines
2291 execute(int gf
, LINE addr
)
2296 for (c
= 0; c
< nbra
; c
++) {
2305 p1
= getaline(addr
->cur
);
2308 return (step(p1
, expbuf
));
2317 r
= (int)(count
%10);
2326 puts(const char *sp
)
2331 if (fss
.Ffill
&& (listf
== 0)) {
2333 /* deliberate attempt to remove constness of sp because */
2334 /* it needs to be expanded */
2336 if ((i
= expnd((char *)sp
, funny
, &sz
, &fss
)) == -1) {
2337 write(1, funny
, fss
.Flim
& 0377);
2339 write(1, gettext("too long"),
2340 strlen(gettext("too long")));
2343 write(1, funny
, sz
);
2346 write(1, gettext("tab count\n"),
2347 strlen(gettext("tab count\n")));
2352 n
= mbtowc(&c
, sp
, MB_LEN_MAX
);
2357 putchr((unsigned char)*sp
++);
2363 putchr((unsigned char)*sp
++);
2368 putchr('$'); /* end of line is marked with a $ */
2371 /* xpg6 - ensure that the end of line $ is not preceeded with a "\" */
2372 /* by doing a putchr() with listf=0, thereby avoiding the $ case */
2373 /* statement in putchr() */
2375 putchr('$'); /* end of line is marked with a $ */
2387 char buf
[MB_LEN_MAX
], *p
;
2397 if ((len
= wctomb(p
, c
)) <= 0) {
2398 *p
= (unsigned char)c
;
2402 if (col
+ 4 >= 72) {
2407 (void) sprintf(lp
, "\\%03o",
2408 *(unsigned char *)p
++);
2413 if ((len
= wcwidth(c
)) <= 0)
2415 if (col
+ len
>= 72) {
2421 if ((len
= wctomb(lp
, c
)) <= 0) {
2422 *lp
= (unsigned char)c
;
2428 if ((len
= wctomb(lp
, c
)) <= 0) {
2429 *lp
= (unsigned char)c
;
2434 if (c
== '\n' || lp
>= &line
[64]) {
2437 write(1, line
, len
);
2445 putchr(unsigned char c
)
2451 if (listf
&& c
!= '\n') {
2489 /* if $ characters are within the line preceed with \ */
2501 (void) sprintf(lp
, "\\%03o", c
);
2509 * long lines are folded w/ pt of folding indicated by writing
2510 * backslash/newline character
2513 if (col
+ 1 >= 72) {
2520 if (c
== '\n' || lp
>= &line
[64]) {
2523 (void) write(1, line
, len
);
2531 getkey(const char *prompt
)
2536 static char key
[KSIZE
+1];
2540 sig
= signal(SIGINT
, SIG_IGN
);
2541 ioctl(0, TCGETA
, &b
);
2543 b
.c_lflag
&= ~(ECHO
| ECHOE
| ECHOK
| ECHONL
);
2544 ioctl(0, TCSETAW
, &b
);
2545 write(1, gettext(prompt
), strlen(gettext(prompt
)));
2547 while (((c
= getchr()) != EOF
) && (c
!= '\n')) {
2548 if (p
< &key
[KSIZE
])
2554 ioctl(0, TCSETAW
, &b
);
2555 signal(SIGINT
, sig
);
2569 char globuf
[LBSIZE
];
2570 char multic
[MB_LEN_MAX
];
2580 if ((n
= _mbftowc(multic
, &cl
, getchr
, &peekc
)) <= 0)
2586 for (a1
= zero
; a1
<= dol
; a1
++) {
2588 if (a1
>= addr1
&& a1
<= addr2
&& execute(0, a1
) == k
)
2594 * preserve the p, l, and n suffix commands of the G and V
2595 * commands during the interactive section and restore
2596 * on completion of the G and V command.
2604 for (a1
= zero
; a1
<= dol
; a1
++) {
2608 puts(getaline(a1
->cur
));
2609 if ((c
= get_wchr()) == EOF
)
2611 if (c
== 'a' || c
== 'i' || c
== 'c')
2619 if ((len
= wctomb(gp
, c
)) <= 0) {
2620 *gp
= (unsigned char)c
;
2624 while ((c
= get_wchr()) != '\n') {
2626 /* '\\' has special meaning only if preceding a '\n' */
2632 if ((gp
+ (unsigned int)MB_CUR_MAX
) >=
2636 if ((len
= wctomb(gp
, c
)) <= 0) {
2637 *gp
= (unsigned char)c
;
2645 } else if ((c
= get_wchr()) != '\n')
2665 eopen(char *string
, int rw
)
2667 #define w_or_r(a, b) (rw ? a : b)
2671 int chcount
; /* # of char read. */
2673 if (rflg
) { /* restricted shell */
2680 if ((io
= open(string
, rw
)) >= 0) {
2682 chcount
= read(io
, crbuf
, LBSIZE
);
2684 if (isencrypt(crbuf
, chcount
))
2690 if (run_crypt(0L, crbuf
, chcount
, perm
) == -1)
2692 if (fspec(crbuf
, &fss
, 0) < 0) {
2704 xerr
: (void) error(0);
2705 if ((i
= fork()) == 0) {
2706 signal(SIGHUP
, oldhup
);
2707 signal(SIGQUIT
, oldquit
);
2708 signal(SIGPIPE
, oldpipe
);
2709 signal(SIGINT
, (void (*)()) 0);
2710 close(w_or_r(pf
[1], pf
[0]));
2711 close(w_or_r(0, 1));
2712 dup(w_or_r(pf
[0], pf
[1]));
2713 close(w_or_r(pf
[0], pf
[1]));
2714 execlp(_PATH_BSHELL
, "sh", "-c", string
, (char *)0);
2719 close(w_or_r(pf
[0], pf
[1]));
2720 return (w_or_r(pf
[1], pf
[0]));
2729 Xqt
= 0, wait((int *)0);
2741 * Go to end of file name
2745 while (*--p
== '/') /* delete trailing slashes */
2748 * go back to beginning of file
2752 * Copy file name to funny setting p2 at
2759 * Set p1 to point to basename of tfname.
2761 p1
= strrchr(tfname
, '/');
2762 if (strlen(tfname
) > (size_t)6)
2763 p1
= &tfname
[strlen(tfname
)-6];
2765 *p2
= '\007'; /* add unprintable char for funny a unique name */
2767 * Copy tfname to file.
2769 while (*++p2
= *p1
++)
2775 getime(void) /* get modified time of file and save */
2777 if (stat(file
, &Fl
) < 0)
2780 savtime
= Fl
.st_mtime
;
2785 chktime(void) /* check saved mod time against current mod time */
2787 if (savtime
!= 0 && Fl
.st_mtime
!= 0) {
2788 if (savtime
!= Fl
.st_mtime
)
2795 newtime(void) /* get new mod time and save */
2798 savtime
= Fl
.st_mtime
;
2803 * restricted - check for '/' in name and delete trailing '/'
2812 if (*p
++ == '/'&& rflg
) {
2816 /* delete trailing '/' */
2826 * Searches thru beginning of file looking for a string of the form
2829 * where "values" are
2832 * s<num> sets the Flim to <num>
2833 * t??? sets tab stop stuff
2840 fspec(char line
[], struct Fspec
*f
, int up
)
2848 havespec
= fsprtn
= 0;
2849 for (fsp
= line
; *fsp
&& *fsp
!= '\n'; fsp
+= len
) {
2850 if ((len
= mblen(fsp
, MB_CUR_MAX
)) <= 0)
2854 case '<': if (havespec
)
2856 if (*(fsp
+1) == ':') {
2859 if (!ioctl(1, TCGETA
, &arg
) &&
2860 ((arg
.c_oflag
& TAB3
) ==
2869 case 's': if (havespec
&& (n
= numb()) >= 0)
2873 case 't': if (havespec
) targ(f
);
2878 case 'm': if (havespec
) n
= numb();
2882 case ':': if (!havespec
) continue;
2883 if (*(fsp
+1) != '>') fsprtn
= -1;
2886 default: if (!havespec
) continue;
2900 while (*++fsp
>= '0' && *fsp
<= '9')
2901 n
= 10*n
+ *fsp
-'0';
2908 targ(struct Fspec
*f
)
2911 if (*++fsp
== '-') {
2912 if (*(fsp
+ 1) >= '0' && *(fsp
+1) <= '9') tincr(numb(), f
);
2916 if (*fsp
>= '0' && *fsp
<= '9') {
2926 tincr(int n
, struct Fspec
*f
)
2931 for (i
= 0; i
< 20; i
++)
2932 f
->Ftabs
[i
] = l
+= n
;
2938 tstd(struct Fspec
*f
)
2943 if (*(fsp
+1) >= '0' && *(fsp
+1) <= '9') {
2949 fsprtn
= stdtab(std
, f
->Ftabs
);
2954 tlist(struct Fspec
*f
)
2962 if ((n
= numb()) <= last
|| i
>= 20) {
2966 f
->Ftabs
[i
++] = last
= n
;
2967 } while (*++fsp
== ',');
2975 expnd(char line
[], char buf
[], int *sz
, struct Fspec
*f
)
2985 while (*++l
&& *l
!= '\n' && b
< 511) {
2987 while (*t
&& b
>= *t
)
3001 if (*l
!= '\0' && *l
!= '\n') {
3006 if (f
->Flim
&& (b
-1 > (int)f
->Flim
))
3013 clear(struct Fspec
*f
)
3015 f
->Ftabs
[0] = f
->Fdel
= f
->Fmov
= f
->Ffill
= 0;
3021 lenchk(char line
[], struct Fspec
*f
)
3030 while (*++l
&& *l
!= '\n' && b
< 511) {
3032 while (*t
&& b
>= *t
)
3041 if ((*l
!= '\0' && *l
!= '\n') || (f
->Flim
&& (b
-1 > (int)f
->Flim
)))
3049 * stdtabs: standard tabs table
3050 * format: option code letter(s), null, tabs, null
3053 static char stdtabs
[] = {
3054 'a', 0, 1, 10, 16, 36, 72, 0, /* IBM 370 Assembler */
3055 'a', '2', 0, 1, 10, 16, 40, 72, 0, /* IBM Assembler alternative */
3056 'c', 0, 1, 8, 12, 16, 20, 55, 0, /* COBOL, normal */
3057 'c', '2', 0, 1, 6, 10, 14, 49, 0, /* COBOL, crunched */
3058 'c', '3', 0, 1, 6, 10, 14, 18, 22, 26, 30, 34, 38, 42, 46, 50,
3060 'f', 0, 1, 7, 11, 15, 19, 23, 0, /* FORTRAN */
3061 'p', 0, 1, 5, 9, 13, 17, 21, 25, 29, 33, 37, 41, 45, 49, 53, 57, 61, 0,
3063 's', 0, 1, 10, 55, 0, /* SNOBOL */
3064 'u', 0, 1, 12, 20, 44, 0, /* UNIVAC ASM */
3069 * stdtab: return tab list for any "canned" tab option.
3070 * entry: option points to null-terminated option string
3071 * tabvect points to vector to be filled in
3072 * exit: return (0) if legal, tabvect filled, ending with zero
3073 * return (-1) if unknown option
3078 stdtab(char option
[], char tabvect
[NTABS
])
3084 if (strequal(&scan
, option
)) {
3085 strcopy(scan
, tabvect
);
3088 while (*scan
++) /* skip over tab specs */
3092 /* later: look up code in /etc/something */
3093 return (tabvect
[0] ? 0 : -1);
3098 * strequal: checks strings for equality
3099 * entry: scan1 points to scan pointer, str points to string
3100 * exit: return (1) if equal, return (0) if not
3101 * *scan1 is advanced to next nonzero byte after null
3106 strequal(char **scan1
, char *str
)
3110 while ((c
= *scan
++) == *str
&& c
)
3113 if (c
== 0 && *str
== 0)
3123 /* strcopy: copy source to destination */
3127 strcopy(char *source
, char *dest
)
3129 while (*dest
++ = *source
++)
3134 /* This is called before a buffer modifying command so that the */
3135 /* current array of line ptrs is saved in sav and dot and dol are saved */
3146 for (j
= 0; j
<= 25; j
++)
3147 savnames
[j
] = names
[j
];
3149 for (i
= zero
+ 1; i
<= dol
; i
++)
3155 /* The undo command calls this to restore the previous ptr array sav */
3156 /* and swap with cur - dot and dol are swapped also. This allows user to */
3165 LINE i
, tmpdot
, tmpdol
;
3167 tmpdot
= dot
; dot
= savdot
; savdot
= tmpdot
;
3168 tmpdol
= dol
; dol
= savdol
; savdol
= tmpdol
;
3169 /* swap arrays using the greater of dol or savdol as upper limit */
3170 for (i
= zero
+ 1; i
<= ((dol
> savdol
) ? dol
: savdol
); i
++) {
3176 * If the current text lines are swapped with the
3177 * text lines in the save buffer, then swap the current
3178 * marks with those in the save area.
3181 for (j
= 0; j
<= 25; j
++) {
3183 names
[j
] = savnames
[j
];
3192 char multi
[MB_LEN_MAX
];
3194 if (_mbftowc(multi
, &wc
, getchr
, &peekc
) <= 0)