xref: /aosp_15_r20/external/e2fsprogs/misc/logsave.c (revision 6a54128f25917bfc36a8a6e9d722c04a0b4641b6)
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