1*6a54128fSAndroid Build Coastguard Worker /*
2*6a54128fSAndroid Build Coastguard Worker * logsave.c --- A program which saves the output of a program until
3*6a54128fSAndroid Build Coastguard Worker * /var/log is mounted.
4*6a54128fSAndroid Build Coastguard Worker *
5*6a54128fSAndroid Build Coastguard Worker * Copyright (C) 2003 Theodore Ts'o.
6*6a54128fSAndroid Build Coastguard Worker *
7*6a54128fSAndroid Build Coastguard Worker * %Begin-Header%
8*6a54128fSAndroid Build Coastguard Worker * This file may be redistributed under the terms of the GNU Public
9*6a54128fSAndroid Build Coastguard Worker * License.
10*6a54128fSAndroid Build Coastguard Worker * %End-Header%
11*6a54128fSAndroid Build Coastguard Worker */
12*6a54128fSAndroid Build Coastguard Worker
13*6a54128fSAndroid Build Coastguard Worker #define _XOPEN_SOURCE 600 /* for inclusion of sa_handler in Solaris */
14*6a54128fSAndroid Build Coastguard Worker
15*6a54128fSAndroid Build Coastguard Worker #include "config.h"
16*6a54128fSAndroid Build Coastguard Worker #include <stdio.h>
17*6a54128fSAndroid Build Coastguard Worker #include <stdlib.h>
18*6a54128fSAndroid Build Coastguard Worker #include <unistd.h>
19*6a54128fSAndroid Build Coastguard Worker #include <string.h>
20*6a54128fSAndroid Build Coastguard Worker #include <sys/types.h>
21*6a54128fSAndroid Build Coastguard Worker #include <sys/wait.h>
22*6a54128fSAndroid Build Coastguard Worker #include <fcntl.h>
23*6a54128fSAndroid Build Coastguard Worker #include <time.h>
24*6a54128fSAndroid Build Coastguard Worker #include <errno.h>
25*6a54128fSAndroid Build Coastguard Worker #ifdef HAVE_SIGNAL_H
26*6a54128fSAndroid Build Coastguard Worker #include <signal.h>
27*6a54128fSAndroid Build Coastguard Worker #endif
28*6a54128fSAndroid Build Coastguard Worker #ifdef HAVE_GETOPT_H
29*6a54128fSAndroid Build Coastguard Worker #include <getopt.h>
30*6a54128fSAndroid Build Coastguard Worker #else
31*6a54128fSAndroid Build Coastguard Worker extern char *optarg;
32*6a54128fSAndroid Build Coastguard Worker extern int optind;
33*6a54128fSAndroid Build Coastguard Worker #endif
34*6a54128fSAndroid Build Coastguard Worker
35*6a54128fSAndroid Build Coastguard Worker static int outfd = -1;
36*6a54128fSAndroid Build Coastguard Worker static int outbufsize = 0;
37*6a54128fSAndroid Build Coastguard Worker static void *outbuf = 0;
38*6a54128fSAndroid Build Coastguard Worker static int verbose = 0;
39*6a54128fSAndroid Build Coastguard Worker static int do_skip = 0;
40*6a54128fSAndroid Build Coastguard Worker static int skip_mode = 0;
41*6a54128fSAndroid Build Coastguard Worker static pid_t child_pid = -1;
42*6a54128fSAndroid Build Coastguard Worker
usage(char * progname)43*6a54128fSAndroid Build Coastguard Worker static void usage(char *progname)
44*6a54128fSAndroid Build Coastguard Worker {
45*6a54128fSAndroid Build Coastguard Worker printf("Usage: %s [-asv] logfile program\n", progname);
46*6a54128fSAndroid Build Coastguard Worker exit(1);
47*6a54128fSAndroid Build Coastguard Worker }
48*6a54128fSAndroid Build Coastguard Worker
49*6a54128fSAndroid Build Coastguard Worker #define SEND_LOG 0x01
50*6a54128fSAndroid Build Coastguard Worker #define SEND_CONSOLE 0x02
51*6a54128fSAndroid Build Coastguard Worker #define SEND_BOTH 0x03
52*6a54128fSAndroid Build Coastguard Worker
53*6a54128fSAndroid Build Coastguard Worker /*
54*6a54128fSAndroid Build Coastguard Worker * Helper function that does the right thing if write returns a
55*6a54128fSAndroid Build Coastguard Worker * partial write, or an EAGAIN/EINTR error.
56*6a54128fSAndroid Build Coastguard Worker */
write_all(int fd,const char * buf,size_t count)57*6a54128fSAndroid Build Coastguard Worker static int write_all(int fd, const char *buf, size_t count)
58*6a54128fSAndroid Build Coastguard Worker {
59*6a54128fSAndroid Build Coastguard Worker ssize_t ret;
60*6a54128fSAndroid Build Coastguard Worker int c = 0;
61*6a54128fSAndroid Build Coastguard Worker
62*6a54128fSAndroid Build Coastguard Worker while (count > 0) {
63*6a54128fSAndroid Build Coastguard Worker ret = write(fd, buf, count);
64*6a54128fSAndroid Build Coastguard Worker if (ret < 0) {
65*6a54128fSAndroid Build Coastguard Worker if ((errno == EAGAIN) || (errno == EINTR))
66*6a54128fSAndroid Build Coastguard Worker continue;
67*6a54128fSAndroid Build Coastguard Worker return -1;
68*6a54128fSAndroid Build Coastguard Worker }
69*6a54128fSAndroid Build Coastguard Worker count -= ret;
70*6a54128fSAndroid Build Coastguard Worker buf += ret;
71*6a54128fSAndroid Build Coastguard Worker c += ret;
72*6a54128fSAndroid Build Coastguard Worker }
73*6a54128fSAndroid Build Coastguard Worker return c;
74*6a54128fSAndroid Build Coastguard Worker }
75*6a54128fSAndroid Build Coastguard Worker
send_output(const char * buffer,int c,int flag)76*6a54128fSAndroid Build Coastguard Worker static void send_output(const char *buffer, int c, int flag)
77*6a54128fSAndroid Build Coastguard Worker {
78*6a54128fSAndroid Build Coastguard Worker const char *cp;
79*6a54128fSAndroid Build Coastguard Worker char *n;
80*6a54128fSAndroid Build Coastguard Worker int cnt, d, del;
81*6a54128fSAndroid Build Coastguard Worker
82*6a54128fSAndroid Build Coastguard Worker if (c == 0)
83*6a54128fSAndroid Build Coastguard Worker c = strlen(buffer);
84*6a54128fSAndroid Build Coastguard Worker
85*6a54128fSAndroid Build Coastguard Worker if (flag & SEND_CONSOLE) {
86*6a54128fSAndroid Build Coastguard Worker cnt = c;
87*6a54128fSAndroid Build Coastguard Worker cp = buffer;
88*6a54128fSAndroid Build Coastguard Worker while (cnt) {
89*6a54128fSAndroid Build Coastguard Worker del = 0;
90*6a54128fSAndroid Build Coastguard Worker for (d=0; d < cnt; d++) {
91*6a54128fSAndroid Build Coastguard Worker if (skip_mode &&
92*6a54128fSAndroid Build Coastguard Worker (cp[d] == '\001' || cp[d] == '\002')) {
93*6a54128fSAndroid Build Coastguard Worker del = 1;
94*6a54128fSAndroid Build Coastguard Worker break;
95*6a54128fSAndroid Build Coastguard Worker }
96*6a54128fSAndroid Build Coastguard Worker }
97*6a54128fSAndroid Build Coastguard Worker write_all(1, cp, d);
98*6a54128fSAndroid Build Coastguard Worker if (del)
99*6a54128fSAndroid Build Coastguard Worker d++;
100*6a54128fSAndroid Build Coastguard Worker cnt -= d;
101*6a54128fSAndroid Build Coastguard Worker cp += d;
102*6a54128fSAndroid Build Coastguard Worker }
103*6a54128fSAndroid Build Coastguard Worker }
104*6a54128fSAndroid Build Coastguard Worker if (!(flag & SEND_LOG))
105*6a54128fSAndroid Build Coastguard Worker return;
106*6a54128fSAndroid Build Coastguard Worker if (outfd > 0)
107*6a54128fSAndroid Build Coastguard Worker write_all(outfd, buffer, c);
108*6a54128fSAndroid Build Coastguard Worker else {
109*6a54128fSAndroid Build Coastguard Worker n = realloc(outbuf, outbufsize + c);
110*6a54128fSAndroid Build Coastguard Worker if (n) {
111*6a54128fSAndroid Build Coastguard Worker outbuf = n;
112*6a54128fSAndroid Build Coastguard Worker memcpy(((char *)outbuf)+outbufsize, buffer, c);
113*6a54128fSAndroid Build Coastguard Worker outbufsize += c;
114*6a54128fSAndroid Build Coastguard Worker }
115*6a54128fSAndroid Build Coastguard Worker }
116*6a54128fSAndroid Build Coastguard Worker }
117*6a54128fSAndroid Build Coastguard Worker
do_read(int fd)118*6a54128fSAndroid Build Coastguard Worker static int do_read(int fd)
119*6a54128fSAndroid Build Coastguard Worker {
120*6a54128fSAndroid Build Coastguard Worker int c;
121*6a54128fSAndroid Build Coastguard Worker char buffer[4096], *cp, *sep;
122*6a54128fSAndroid Build Coastguard Worker
123*6a54128fSAndroid Build Coastguard Worker c = read(fd, buffer, sizeof(buffer)-1);
124*6a54128fSAndroid Build Coastguard Worker if (c <= 0)
125*6a54128fSAndroid Build Coastguard Worker return c;
126*6a54128fSAndroid Build Coastguard Worker if (do_skip) {
127*6a54128fSAndroid Build Coastguard Worker send_output(buffer, c, SEND_CONSOLE);
128*6a54128fSAndroid Build Coastguard Worker buffer[c] = 0;
129*6a54128fSAndroid Build Coastguard Worker cp = buffer;
130*6a54128fSAndroid Build Coastguard Worker while (*cp) {
131*6a54128fSAndroid Build Coastguard Worker if (skip_mode) {
132*6a54128fSAndroid Build Coastguard Worker cp = strchr(cp, '\002');
133*6a54128fSAndroid Build Coastguard Worker if (!cp)
134*6a54128fSAndroid Build Coastguard Worker return 0;
135*6a54128fSAndroid Build Coastguard Worker cp++;
136*6a54128fSAndroid Build Coastguard Worker skip_mode = 0;
137*6a54128fSAndroid Build Coastguard Worker continue;
138*6a54128fSAndroid Build Coastguard Worker }
139*6a54128fSAndroid Build Coastguard Worker sep = strchr(cp, '\001');
140*6a54128fSAndroid Build Coastguard Worker if (sep)
141*6a54128fSAndroid Build Coastguard Worker *sep = 0;
142*6a54128fSAndroid Build Coastguard Worker send_output(cp, 0, SEND_LOG);
143*6a54128fSAndroid Build Coastguard Worker if (sep) {
144*6a54128fSAndroid Build Coastguard Worker cp = sep + 1;
145*6a54128fSAndroid Build Coastguard Worker skip_mode = 1;
146*6a54128fSAndroid Build Coastguard Worker } else
147*6a54128fSAndroid Build Coastguard Worker break;
148*6a54128fSAndroid Build Coastguard Worker }
149*6a54128fSAndroid Build Coastguard Worker } else
150*6a54128fSAndroid Build Coastguard Worker send_output(buffer, c, SEND_BOTH);
151*6a54128fSAndroid Build Coastguard Worker return c;
152*6a54128fSAndroid Build Coastguard Worker }
153*6a54128fSAndroid Build Coastguard Worker
signal_term(int sig)154*6a54128fSAndroid Build Coastguard Worker static void signal_term(int sig)
155*6a54128fSAndroid Build Coastguard Worker {
156*6a54128fSAndroid Build Coastguard Worker if (child_pid > 0)
157*6a54128fSAndroid Build Coastguard Worker kill(child_pid, sig);
158*6a54128fSAndroid Build Coastguard Worker }
159*6a54128fSAndroid Build Coastguard Worker
run_program(char ** argv)160*6a54128fSAndroid Build Coastguard Worker static int run_program(char **argv)
161*6a54128fSAndroid Build Coastguard Worker {
162*6a54128fSAndroid Build Coastguard Worker int fds[2];
163*6a54128fSAndroid Build Coastguard Worker int status, rc, pid;
164*6a54128fSAndroid Build Coastguard Worker char buffer[80];
165*6a54128fSAndroid Build Coastguard Worker #ifdef HAVE_SIGNAL_H
166*6a54128fSAndroid Build Coastguard Worker struct sigaction sa;
167*6a54128fSAndroid Build Coastguard Worker #endif
168*6a54128fSAndroid Build Coastguard Worker
169*6a54128fSAndroid Build Coastguard Worker if (pipe(fds) < 0) {
170*6a54128fSAndroid Build Coastguard Worker perror("pipe");
171*6a54128fSAndroid Build Coastguard Worker exit(1);
172*6a54128fSAndroid Build Coastguard Worker }
173*6a54128fSAndroid Build Coastguard Worker
174*6a54128fSAndroid Build Coastguard Worker #ifdef HAVE_SIGNAL_H
175*6a54128fSAndroid Build Coastguard Worker memset(&sa, 0, sizeof(struct sigaction));
176*6a54128fSAndroid Build Coastguard Worker sa.sa_handler = signal_term;
177*6a54128fSAndroid Build Coastguard Worker sigaction(SIGINT, &sa, 0);
178*6a54128fSAndroid Build Coastguard Worker sigaction(SIGTERM, &sa, 0);
179*6a54128fSAndroid Build Coastguard Worker #ifdef SA_RESTART
180*6a54128fSAndroid Build Coastguard Worker sa.sa_flags = SA_RESTART;
181*6a54128fSAndroid Build Coastguard Worker #endif
182*6a54128fSAndroid Build Coastguard Worker #endif
183*6a54128fSAndroid Build Coastguard Worker
184*6a54128fSAndroid Build Coastguard Worker pid = fork();
185*6a54128fSAndroid Build Coastguard Worker if (pid < 0) {
186*6a54128fSAndroid Build Coastguard Worker perror("vfork");
187*6a54128fSAndroid Build Coastguard Worker exit(1);
188*6a54128fSAndroid Build Coastguard Worker }
189*6a54128fSAndroid Build Coastguard Worker if (pid == 0) {
190*6a54128fSAndroid Build Coastguard Worker dup2(fds[1],1); /* fds[1] replaces stdout */
191*6a54128fSAndroid Build Coastguard Worker dup2(fds[1],2); /* fds[1] replaces stderr */
192*6a54128fSAndroid Build Coastguard Worker close(fds[0]); /* don't need this here */
193*6a54128fSAndroid Build Coastguard Worker close(fds[1]);
194*6a54128fSAndroid Build Coastguard Worker
195*6a54128fSAndroid Build Coastguard Worker execvp(argv[0], argv);
196*6a54128fSAndroid Build Coastguard Worker perror(argv[0]);
197*6a54128fSAndroid Build Coastguard Worker exit(1);
198*6a54128fSAndroid Build Coastguard Worker }
199*6a54128fSAndroid Build Coastguard Worker child_pid = pid;
200*6a54128fSAndroid Build Coastguard Worker close(fds[1]);
201*6a54128fSAndroid Build Coastguard Worker
202*6a54128fSAndroid Build Coastguard Worker while (!(waitpid(pid, &status, WNOHANG ))) {
203*6a54128fSAndroid Build Coastguard Worker do_read(fds[0]);
204*6a54128fSAndroid Build Coastguard Worker }
205*6a54128fSAndroid Build Coastguard Worker child_pid = -1;
206*6a54128fSAndroid Build Coastguard Worker do_read(fds[0]);
207*6a54128fSAndroid Build Coastguard Worker close(fds[0]);
208*6a54128fSAndroid Build Coastguard Worker
209*6a54128fSAndroid Build Coastguard Worker if ( WIFEXITED(status) ) {
210*6a54128fSAndroid Build Coastguard Worker rc = WEXITSTATUS(status);
211*6a54128fSAndroid Build Coastguard Worker if (rc) {
212*6a54128fSAndroid Build Coastguard Worker send_output(argv[0], 0, SEND_BOTH);
213*6a54128fSAndroid Build Coastguard Worker sprintf(buffer, " exited with status code %d\n", rc);
214*6a54128fSAndroid Build Coastguard Worker send_output(buffer, 0, SEND_BOTH);
215*6a54128fSAndroid Build Coastguard Worker }
216*6a54128fSAndroid Build Coastguard Worker } else {
217*6a54128fSAndroid Build Coastguard Worker if (WIFSIGNALED(status)) {
218*6a54128fSAndroid Build Coastguard Worker send_output(argv[0], 0, SEND_BOTH);
219*6a54128fSAndroid Build Coastguard Worker sprintf(buffer, "died with signal %d\n",
220*6a54128fSAndroid Build Coastguard Worker WTERMSIG(status));
221*6a54128fSAndroid Build Coastguard Worker send_output(buffer, 0, SEND_BOTH);
222*6a54128fSAndroid Build Coastguard Worker return 1;
223*6a54128fSAndroid Build Coastguard Worker }
224*6a54128fSAndroid Build Coastguard Worker rc = 0;
225*6a54128fSAndroid Build Coastguard Worker }
226*6a54128fSAndroid Build Coastguard Worker return rc;
227*6a54128fSAndroid Build Coastguard Worker }
228*6a54128fSAndroid Build Coastguard Worker
copy_from_stdin(void)229*6a54128fSAndroid Build Coastguard Worker static int copy_from_stdin(void)
230*6a54128fSAndroid Build Coastguard Worker {
231*6a54128fSAndroid Build Coastguard Worker int c, bad_read = 0;
232*6a54128fSAndroid Build Coastguard Worker
233*6a54128fSAndroid Build Coastguard Worker while (1) {
234*6a54128fSAndroid Build Coastguard Worker c = do_read(0);
235*6a54128fSAndroid Build Coastguard Worker if ((c == 0 ) ||
236*6a54128fSAndroid Build Coastguard Worker ((c < 0) && ((errno == EAGAIN) || (errno == EINTR)))) {
237*6a54128fSAndroid Build Coastguard Worker if (bad_read++ > 3)
238*6a54128fSAndroid Build Coastguard Worker break;
239*6a54128fSAndroid Build Coastguard Worker continue;
240*6a54128fSAndroid Build Coastguard Worker }
241*6a54128fSAndroid Build Coastguard Worker if (c < 0) {
242*6a54128fSAndroid Build Coastguard Worker perror("read");
243*6a54128fSAndroid Build Coastguard Worker exit(1);
244*6a54128fSAndroid Build Coastguard Worker }
245*6a54128fSAndroid Build Coastguard Worker bad_read = 0;
246*6a54128fSAndroid Build Coastguard Worker }
247*6a54128fSAndroid Build Coastguard Worker return 0;
248*6a54128fSAndroid Build Coastguard Worker }
249*6a54128fSAndroid Build Coastguard Worker
250*6a54128fSAndroid Build Coastguard Worker
251*6a54128fSAndroid Build Coastguard Worker
main(int argc,char ** argv)252*6a54128fSAndroid Build Coastguard Worker int main(int argc, char **argv)
253*6a54128fSAndroid Build Coastguard Worker {
254*6a54128fSAndroid Build Coastguard Worker int c, pid, rc;
255*6a54128fSAndroid Build Coastguard Worker char *outfn, **cpp;
256*6a54128fSAndroid Build Coastguard Worker int openflags = O_CREAT|O_WRONLY|O_TRUNC;
257*6a54128fSAndroid Build Coastguard Worker int send_flag = SEND_LOG;
258*6a54128fSAndroid Build Coastguard Worker int do_stdin;
259*6a54128fSAndroid Build Coastguard Worker time_t t;
260*6a54128fSAndroid Build Coastguard Worker
261*6a54128fSAndroid Build Coastguard Worker while ((c = getopt(argc, argv, "+asv")) != EOF) {
262*6a54128fSAndroid Build Coastguard Worker switch (c) {
263*6a54128fSAndroid Build Coastguard Worker case 'a':
264*6a54128fSAndroid Build Coastguard Worker openflags &= ~O_TRUNC;
265*6a54128fSAndroid Build Coastguard Worker openflags |= O_APPEND;
266*6a54128fSAndroid Build Coastguard Worker break;
267*6a54128fSAndroid Build Coastguard Worker case 's':
268*6a54128fSAndroid Build Coastguard Worker do_skip = 1;
269*6a54128fSAndroid Build Coastguard Worker break;
270*6a54128fSAndroid Build Coastguard Worker case 'v':
271*6a54128fSAndroid Build Coastguard Worker verbose++;
272*6a54128fSAndroid Build Coastguard Worker send_flag |= SEND_CONSOLE;
273*6a54128fSAndroid Build Coastguard Worker break;
274*6a54128fSAndroid Build Coastguard Worker }
275*6a54128fSAndroid Build Coastguard Worker }
276*6a54128fSAndroid Build Coastguard Worker if (optind == argc || optind+1 == argc)
277*6a54128fSAndroid Build Coastguard Worker usage(argv[0]);
278*6a54128fSAndroid Build Coastguard Worker outfn = argv[optind];
279*6a54128fSAndroid Build Coastguard Worker optind++;
280*6a54128fSAndroid Build Coastguard Worker argv += optind;
281*6a54128fSAndroid Build Coastguard Worker argc -= optind;
282*6a54128fSAndroid Build Coastguard Worker
283*6a54128fSAndroid Build Coastguard Worker outfd = open(outfn, openflags, 0644);
284*6a54128fSAndroid Build Coastguard Worker do_stdin = !strcmp(argv[0], "-");
285*6a54128fSAndroid Build Coastguard Worker
286*6a54128fSAndroid Build Coastguard Worker send_output("Log of ", 0, send_flag);
287*6a54128fSAndroid Build Coastguard Worker if (do_stdin)
288*6a54128fSAndroid Build Coastguard Worker send_output("stdin", 0, send_flag);
289*6a54128fSAndroid Build Coastguard Worker else {
290*6a54128fSAndroid Build Coastguard Worker for (cpp = argv; *cpp; cpp++) {
291*6a54128fSAndroid Build Coastguard Worker send_output(*cpp, 0, send_flag);
292*6a54128fSAndroid Build Coastguard Worker send_output(" ", 0, send_flag);
293*6a54128fSAndroid Build Coastguard Worker }
294*6a54128fSAndroid Build Coastguard Worker }
295*6a54128fSAndroid Build Coastguard Worker send_output("\n", 0, send_flag);
296*6a54128fSAndroid Build Coastguard Worker t = time(0);
297*6a54128fSAndroid Build Coastguard Worker send_output(ctime(&t), 0, send_flag);
298*6a54128fSAndroid Build Coastguard Worker send_output("\n", 0, send_flag);
299*6a54128fSAndroid Build Coastguard Worker
300*6a54128fSAndroid Build Coastguard Worker if (do_stdin)
301*6a54128fSAndroid Build Coastguard Worker rc = copy_from_stdin();
302*6a54128fSAndroid Build Coastguard Worker else
303*6a54128fSAndroid Build Coastguard Worker rc = run_program(argv);
304*6a54128fSAndroid Build Coastguard Worker
305*6a54128fSAndroid Build Coastguard Worker send_output("\n", 0, send_flag);
306*6a54128fSAndroid Build Coastguard Worker t = time(0);
307*6a54128fSAndroid Build Coastguard Worker send_output(ctime(&t), 0, send_flag);
308*6a54128fSAndroid Build Coastguard Worker send_output("----------------\n", 0, send_flag);
309*6a54128fSAndroid Build Coastguard Worker
310*6a54128fSAndroid Build Coastguard Worker if (outbuf) {
311*6a54128fSAndroid Build Coastguard Worker pid = fork();
312*6a54128fSAndroid Build Coastguard Worker if (pid < 0) {
313*6a54128fSAndroid Build Coastguard Worker perror("fork");
314*6a54128fSAndroid Build Coastguard Worker exit(1);
315*6a54128fSAndroid Build Coastguard Worker }
316*6a54128fSAndroid Build Coastguard Worker if (pid) {
317*6a54128fSAndroid Build Coastguard Worker if (verbose)
318*6a54128fSAndroid Build Coastguard Worker printf("Backgrounding to save %s later\n",
319*6a54128fSAndroid Build Coastguard Worker outfn);
320*6a54128fSAndroid Build Coastguard Worker exit(rc);
321*6a54128fSAndroid Build Coastguard Worker }
322*6a54128fSAndroid Build Coastguard Worker setsid(); /* To avoid getting killed by init */
323*6a54128fSAndroid Build Coastguard Worker while (outfd < 0) {
324*6a54128fSAndroid Build Coastguard Worker outfd = open(outfn, openflags, 0644);
325*6a54128fSAndroid Build Coastguard Worker sleep(1);
326*6a54128fSAndroid Build Coastguard Worker }
327*6a54128fSAndroid Build Coastguard Worker write_all(outfd, outbuf, outbufsize);
328*6a54128fSAndroid Build Coastguard Worker free(outbuf);
329*6a54128fSAndroid Build Coastguard Worker }
330*6a54128fSAndroid Build Coastguard Worker if (outfd >= 0)
331*6a54128fSAndroid Build Coastguard Worker close(outfd);
332*6a54128fSAndroid Build Coastguard Worker
333*6a54128fSAndroid Build Coastguard Worker exit(rc);
334*6a54128fSAndroid Build Coastguard Worker }
335