1da2e3ebdSchin /***********************************************************************
2da2e3ebdSchin * *
3da2e3ebdSchin * This software is part of the ast package *
4*3e14f97fSRoger A. Faulkner * Copyright (c) 1992-2010 AT&T Intellectual Property *
5da2e3ebdSchin * and is licensed under the *
6da2e3ebdSchin * Common Public License, Version 1.0 *
77c2fbfb3SApril Chin * by AT&T Intellectual Property *
8da2e3ebdSchin * *
9da2e3ebdSchin * A copy of the License is available at *
10da2e3ebdSchin * http://www.opensource.org/licenses/cpl1.0.txt *
11da2e3ebdSchin * (with md5 checksum 059e8cd6165cb4c31e351f2b69388fd9) *
12da2e3ebdSchin * *
13da2e3ebdSchin * Information and Software Systems Research *
14da2e3ebdSchin * AT&T Research *
15da2e3ebdSchin * Florham Park NJ *
16da2e3ebdSchin * *
17da2e3ebdSchin * Glenn Fowler <gsf@research.att.com> *
18da2e3ebdSchin * David Korn <dgk@research.att.com> *
19da2e3ebdSchin * *
20da2e3ebdSchin ***********************************************************************/
21da2e3ebdSchin #pragma prototyped
22da2e3ebdSchin /*
23da2e3ebdSchin * David Korn
24da2e3ebdSchin * AT&T Bell Laboratories
25da2e3ebdSchin *
26da2e3ebdSchin * tee
27da2e3ebdSchin */
28da2e3ebdSchin
29da2e3ebdSchin static const char usage[] =
3034f9b3eeSRoland Mainz "[-?\n@(#)$Id: tee (AT&T Research) 2009-06-19 $\n]"
31da2e3ebdSchin USAGE_LICENSE
32da2e3ebdSchin "[+NAME?tee - duplicate standard input]"
33da2e3ebdSchin "[+DESCRIPTION?\btee\b copies standard input to standard output "
34da2e3ebdSchin "and to zero or more files. The options determine whether "
35da2e3ebdSchin "the specified files are overwritten or appended to. The "
36da2e3ebdSchin "\btee\b utility does not buffer output. If writes to any "
37da2e3ebdSchin "\afile\a fail, writes to other files continue although \btee\b "
38da2e3ebdSchin "will exit with a non-zero exit status.]"
39da2e3ebdSchin "[+?The number of \afile\a operands that can be specified is limited "
40da2e3ebdSchin "by the underlying operating system.]"
41da2e3ebdSchin "[a:append?Append the standard input to the given files rather "
42da2e3ebdSchin "than overwriting them.]"
43da2e3ebdSchin "[i:ignore-interrupts?Ignore SIGINT signal.]"
44da2e3ebdSchin "[l:linebuffer?Set the standard output to be line buffered.]"
45da2e3ebdSchin "\n"
46da2e3ebdSchin "\n[file ...]\n"
47da2e3ebdSchin "\n"
48da2e3ebdSchin "[+EXIT STATUS?]{"
49da2e3ebdSchin "[+0?All files copies successfully.]"
50da2e3ebdSchin "[+>0?An error occurred.]"
51da2e3ebdSchin "}"
52da2e3ebdSchin "[+SEE ALSO?\bcat\b(1), \bsignal\b(3)]"
53da2e3ebdSchin ;
54da2e3ebdSchin
55da2e3ebdSchin #include <cmd.h>
56da2e3ebdSchin #include <ls.h>
57da2e3ebdSchin #include <sig.h>
58da2e3ebdSchin
59da2e3ebdSchin typedef struct Tee_s
60da2e3ebdSchin {
61da2e3ebdSchin Sfdisc_t disc;
6234f9b3eeSRoland Mainz int line;
63da2e3ebdSchin int fd[1];
64da2e3ebdSchin } Tee_t;
65da2e3ebdSchin
66da2e3ebdSchin /*
67da2e3ebdSchin * This discipline writes to each file in the list given in handle
68da2e3ebdSchin */
69da2e3ebdSchin
7034f9b3eeSRoland Mainz static ssize_t
tee_write(Sfio_t * fp,const void * buf,size_t n,Sfdisc_t * handle)7134f9b3eeSRoland Mainz tee_write(Sfio_t* fp, const void* buf, size_t n, Sfdisc_t* handle)
72da2e3ebdSchin {
73da2e3ebdSchin register const char* bp;
74da2e3ebdSchin register const char* ep;
75da2e3ebdSchin register int* hp = ((Tee_t*)handle)->fd;
76da2e3ebdSchin register int fd = sffileno(fp);
77da2e3ebdSchin register ssize_t r;
78da2e3ebdSchin
79da2e3ebdSchin do
80da2e3ebdSchin {
81da2e3ebdSchin bp = (const char*)buf;
82da2e3ebdSchin ep = bp + n;
83da2e3ebdSchin while (bp < ep)
84da2e3ebdSchin {
85da2e3ebdSchin if ((r = write(fd, bp, ep - bp)) <= 0)
8634f9b3eeSRoland Mainz return -1;
87da2e3ebdSchin bp += r;
88da2e3ebdSchin }
89da2e3ebdSchin } while ((fd = *hp++) >= 0);
9034f9b3eeSRoland Mainz return n;
9134f9b3eeSRoland Mainz }
9234f9b3eeSRoland Mainz
9334f9b3eeSRoland Mainz static void
tee_cleanup(register Tee_t * tp)9434f9b3eeSRoland Mainz tee_cleanup(register Tee_t* tp)
9534f9b3eeSRoland Mainz {
9634f9b3eeSRoland Mainz register int* hp;
9734f9b3eeSRoland Mainz register int n;
9834f9b3eeSRoland Mainz
9934f9b3eeSRoland Mainz if (tp)
10034f9b3eeSRoland Mainz {
10134f9b3eeSRoland Mainz sfdisc(sfstdout, NiL);
10234f9b3eeSRoland Mainz if (tp->line >= 0)
10334f9b3eeSRoland Mainz sfset(sfstdout, SF_LINE, tp->line);
10434f9b3eeSRoland Mainz for (hp = tp->fd; (n = *hp) >= 0; hp++)
10534f9b3eeSRoland Mainz close(n);
10634f9b3eeSRoland Mainz }
107da2e3ebdSchin }
108da2e3ebdSchin
109da2e3ebdSchin int
b_tee(int argc,register char ** argv,void * context)110da2e3ebdSchin b_tee(int argc, register char** argv, void* context)
111da2e3ebdSchin {
112da2e3ebdSchin register Tee_t* tp = 0;
113da2e3ebdSchin register int oflag = O_WRONLY|O_TRUNC|O_CREAT|O_BINARY;
114da2e3ebdSchin register int* hp;
115da2e3ebdSchin register char* cp;
116da2e3ebdSchin int line;
117da2e3ebdSchin
11834f9b3eeSRoland Mainz if (argc <= 0)
11934f9b3eeSRoland Mainz {
12034f9b3eeSRoland Mainz if (context && (tp = (Tee_t*)sh_context(context)->data))
12134f9b3eeSRoland Mainz {
12234f9b3eeSRoland Mainz sh_context(context)->data = 0;
12334f9b3eeSRoland Mainz tee_cleanup(tp);
12434f9b3eeSRoland Mainz }
12534f9b3eeSRoland Mainz return 0;
12634f9b3eeSRoland Mainz }
12734f9b3eeSRoland Mainz cmdinit(argc, argv, context, ERROR_CATALOG, ERROR_CALLBACK);
128da2e3ebdSchin line = -1;
12934f9b3eeSRoland Mainz for (;;)
13034f9b3eeSRoland Mainz {
13134f9b3eeSRoland Mainz switch (optget(argv, usage))
132da2e3ebdSchin {
133da2e3ebdSchin case 'a':
134da2e3ebdSchin oflag &= ~O_TRUNC;
135da2e3ebdSchin oflag |= O_APPEND;
13634f9b3eeSRoland Mainz continue;
137da2e3ebdSchin case 'i':
138da2e3ebdSchin signal(SIGINT, SIG_IGN);
13934f9b3eeSRoland Mainz continue;
140da2e3ebdSchin case 'l':
141da2e3ebdSchin line = sfset(sfstdout, 0, 0) & SF_LINE;
142da2e3ebdSchin if ((line == 0) == (opt_info.num == 0))
143da2e3ebdSchin line = -1;
144da2e3ebdSchin else
145da2e3ebdSchin sfset(sfstdout, SF_LINE, !!opt_info.num);
14634f9b3eeSRoland Mainz continue;
147da2e3ebdSchin case ':':
148da2e3ebdSchin error(2, "%s", opt_info.arg);
149da2e3ebdSchin break;
150da2e3ebdSchin case '?':
151da2e3ebdSchin error(ERROR_usage(2), "%s", opt_info.arg);
152da2e3ebdSchin break;
153da2e3ebdSchin }
15434f9b3eeSRoland Mainz break;
15534f9b3eeSRoland Mainz }
156da2e3ebdSchin if (error_info.errors)
157da2e3ebdSchin error(ERROR_usage(2), "%s", optusage(NiL));
158da2e3ebdSchin argv += opt_info.index;
159da2e3ebdSchin argc -= opt_info.index;
1607c2fbfb3SApril Chin #if _ANCIENT_BSD_COMPATIBILITY
161da2e3ebdSchin if (*argv && streq(*argv, "-"))
162da2e3ebdSchin {
163da2e3ebdSchin signal(SIGINT, SIG_IGN);
164da2e3ebdSchin argv++;
165da2e3ebdSchin argc--;
166da2e3ebdSchin }
1677c2fbfb3SApril Chin #endif
168da2e3ebdSchin if (argc > 0)
169da2e3ebdSchin {
17034f9b3eeSRoland Mainz if (tp = (Tee_t*)stakalloc(sizeof(Tee_t) + argc * sizeof(int)))
17134f9b3eeSRoland Mainz {
17234f9b3eeSRoland Mainz memset(&tp->disc, 0, sizeof(tp->disc));
17334f9b3eeSRoland Mainz tp->disc.writef = tee_write;
17434f9b3eeSRoland Mainz if (context)
17534f9b3eeSRoland Mainz sh_context(context)->data = (void*)tp;
17634f9b3eeSRoland Mainz tp->line = line;
177da2e3ebdSchin hp = tp->fd;
178da2e3ebdSchin while (cp = *argv++)
179da2e3ebdSchin {
180da2e3ebdSchin if ((*hp = open(cp, oflag, S_IRUSR|S_IWUSR|S_IRGRP|S_IWGRP|S_IROTH|S_IWOTH)) < 0)
181da2e3ebdSchin error(ERROR_system(0), "%s: cannot create", cp);
18234f9b3eeSRoland Mainz else
18334f9b3eeSRoland Mainz hp++;
184da2e3ebdSchin }
185da2e3ebdSchin if (hp == tp->fd)
186da2e3ebdSchin tp = 0;
187da2e3ebdSchin else
188da2e3ebdSchin {
189da2e3ebdSchin *hp = -1;
190da2e3ebdSchin sfdisc(sfstdout, &tp->disc);
191da2e3ebdSchin }
192da2e3ebdSchin }
19334f9b3eeSRoland Mainz else
19434f9b3eeSRoland Mainz error(ERROR_exit(0), "out of space");
195da2e3ebdSchin }
19634f9b3eeSRoland Mainz if ((sfmove(sfstdin, sfstdout, SF_UNBOUND, -1) < 0 || !sfeof(sfstdin)) && errno != EPIPE)
19734f9b3eeSRoland Mainz error(ERROR_system(0), "read error");
19834f9b3eeSRoland Mainz if (sfsync(sfstdout))
19934f9b3eeSRoland Mainz error(ERROR_system(0), "write error");
20034f9b3eeSRoland Mainz tee_cleanup(tp);
20134f9b3eeSRoland Mainz return error_info.errors;
202da2e3ebdSchin }
203