xref: /netbsd-src/usr.bin/progress/progress.c (revision e1d1003b4c257cc6fa7239b1b93d99fb1985449c)
1*e1d1003bSgson /*	$NetBSD: progress.c,v 1.25 2021/08/17 07:18:43 gson Exp $ */
298f6cccbSjhawk 
398f6cccbSjhawk /*-
498f6cccbSjhawk  * Copyright (c) 2003 The NetBSD Foundation, Inc.
598f6cccbSjhawk  * All rights reserved.
698f6cccbSjhawk  *
798f6cccbSjhawk  * This code is derived from software contributed to The NetBSD Foundation
898f6cccbSjhawk  * by John Hawkinson.
998f6cccbSjhawk  *
1098f6cccbSjhawk  * Redistribution and use in source and binary forms, with or without
1198f6cccbSjhawk  * modification, are permitted provided that the following conditions
1298f6cccbSjhawk  * are met:
1398f6cccbSjhawk  * 1. Redistributions of source code must retain the above copyright
1498f6cccbSjhawk  *    notice, this list of conditions and the following disclaimer.
1598f6cccbSjhawk  * 2. Redistributions in binary form must reproduce the above copyright
1698f6cccbSjhawk  *    notice, this list of conditions and the following disclaimer in the
1798f6cccbSjhawk  *    documentation and/or other materials provided with the distribution.
1898f6cccbSjhawk  *
1998f6cccbSjhawk  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
2098f6cccbSjhawk  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
2198f6cccbSjhawk  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
2298f6cccbSjhawk  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
2398f6cccbSjhawk  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
2498f6cccbSjhawk  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
2598f6cccbSjhawk  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
2698f6cccbSjhawk  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
2798f6cccbSjhawk  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
2898f6cccbSjhawk  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
2998f6cccbSjhawk  * POSSIBILITY OF SUCH DAMAGE.
3098f6cccbSjhawk  */
3198f6cccbSjhawk 
3298f6cccbSjhawk #include <sys/cdefs.h>
3398f6cccbSjhawk #ifndef lint
34*e1d1003bSgson __RCSID("$NetBSD: progress.c,v 1.25 2021/08/17 07:18:43 gson Exp $");
3598f6cccbSjhawk #endif				/* not lint */
3698f6cccbSjhawk 
3798f6cccbSjhawk #include <sys/types.h>
3898f6cccbSjhawk #include <sys/ioctl.h>
399a313cc5Sriastradh #include <sys/stat.h>
4098f6cccbSjhawk #include <sys/wait.h>
4198f6cccbSjhawk 
4298f6cccbSjhawk #include <err.h>
4398f6cccbSjhawk #include <errno.h>
4498f6cccbSjhawk #include <fcntl.h>
4579b63fa6Shubertf #include <inttypes.h>
4698f6cccbSjhawk #include <limits.h>
479a313cc5Sriastradh #include <signal.h>
4898f6cccbSjhawk #include <stdio.h>
4998f6cccbSjhawk #include <stdlib.h>
5098f6cccbSjhawk #include <string.h>
5198f6cccbSjhawk #include <unistd.h>
5298f6cccbSjhawk 
5398f6cccbSjhawk #define GLOBAL			/* force GLOBAL decls in progressbar.h to be
5498f6cccbSjhawk 				 * declared */
5598f6cccbSjhawk #include "progressbar.h"
5698f6cccbSjhawk 
5789a35a0eSdholland static void broken_pipe(int unused);
586818646aSjoerg __dead static void usage(void);
5998f6cccbSjhawk 
6098f6cccbSjhawk static void
broken_pipe(int unused)6189a35a0eSdholland broken_pipe(int unused)
6289a35a0eSdholland {
6389a35a0eSdholland 	signal(SIGPIPE, SIG_DFL);
6489a35a0eSdholland 	progressmeter(1);
6589a35a0eSdholland 	kill(getpid(), SIGPIPE);
6689a35a0eSdholland }
6789a35a0eSdholland 
6889a35a0eSdholland static void
usage(void)6998f6cccbSjhawk usage(void)
7098f6cccbSjhawk {
7198f6cccbSjhawk 	fprintf(stderr,
72ac798ebcSbriggs 	    "usage: %s [-ez] [-b buffersize] [-f file] [-l length]\n"
73ac798ebcSbriggs 	    "       %*.s [-p prefix] cmd [args...]\n",
74ac798ebcSbriggs 	    getprogname(), (int) strlen(getprogname()), "");
751a7e106bShubertf 	exit(EXIT_FAILURE);
7698f6cccbSjhawk }
7798f6cccbSjhawk 
7898f6cccbSjhawk int
main(int argc,char * argv[])7998f6cccbSjhawk main(int argc, char *argv[])
8098f6cccbSjhawk {
81ac798ebcSbriggs 	char *fb_buf;
8298f6cccbSjhawk 	char *infile = NULL;
838d3b64e3Sdsl 	pid_t pid = 0, gzippid = 0, deadpid;
848d3b64e3Sdsl 	int ch, fd, outpipe[2];
858d3b64e3Sdsl 	int ws, gzipstat, cmdstat;
86ddd39082Sgarbled 	int eflag = 0, lflag = 0, zflag = 0;
8798f6cccbSjhawk 	ssize_t nr, nw, off;
88ac798ebcSbriggs 	size_t buffersize;
8998f6cccbSjhawk 	struct stat statb;
9018350cdfSchristos 	struct ttysize ts;
9198f6cccbSjhawk 
9298f6cccbSjhawk 	setprogname(argv[0]);
9398f6cccbSjhawk 
9498f6cccbSjhawk 	/* defaults: Read from stdin, 0 filesize (no completion estimate) */
9598f6cccbSjhawk 	fd = STDIN_FILENO;
9698f6cccbSjhawk 	filesize = 0;
97ac798ebcSbriggs 	buffersize = 64 * 1024;
98849866f9Shubertf 	prefix = NULL;
9998f6cccbSjhawk 
100ac798ebcSbriggs 	while ((ch = getopt(argc, argv, "b:ef:l:p:z")) != -1)
10198f6cccbSjhawk 		switch (ch) {
102ac798ebcSbriggs 		case 'b':
103ac798ebcSbriggs 			buffersize = (size_t) strsuftoll("buffer size", optarg,
10440771355Stron 			    0, SSIZE_MAX);
105ac798ebcSbriggs 			break;
106ddd39082Sgarbled 		case 'e':
107ddd39082Sgarbled 			eflag++;
108ddd39082Sgarbled 			break;
10998f6cccbSjhawk 		case 'f':
11098f6cccbSjhawk 			infile = optarg;
11198f6cccbSjhawk 			break;
11298f6cccbSjhawk 		case 'l':
11398f6cccbSjhawk 			lflag++;
114f9f551dbSlukem 			filesize = strsuftoll("input size", optarg, 0,
115f9f551dbSlukem 			    LLONG_MAX);
11698f6cccbSjhawk 			break;
117849866f9Shubertf 		case 'p':
118849866f9Shubertf 			prefix = optarg;
119849866f9Shubertf 			break;
12098f6cccbSjhawk 		case 'z':
12198f6cccbSjhawk 			zflag++;
12298f6cccbSjhawk 			break;
12398f6cccbSjhawk 		case '?':
1241a7e106bShubertf 		default:
12598f6cccbSjhawk 			usage();
12698f6cccbSjhawk 			/* NOTREACHED */
12798f6cccbSjhawk 		}
12898f6cccbSjhawk 	argc -= optind;
12998f6cccbSjhawk 	argv += optind;
13098f6cccbSjhawk 
13198f6cccbSjhawk 	if (argc < 1)
13298f6cccbSjhawk 		usage();
1331a7e106bShubertf 
13498f6cccbSjhawk 	if (infile && (fd = open(infile, O_RDONLY, 0)) < 0)
13598f6cccbSjhawk 		err(1, "%s", infile);
13698f6cccbSjhawk 
13798f6cccbSjhawk 	/* stat() to get the filesize unless overridden, or -z */
138ce8f04b1Shubertf 	if (!zflag && !lflag && (fstat(fd, &statb) == 0)) {
139ce8f04b1Shubertf 		if (S_ISFIFO(statb.st_mode)) {
140ce8f04b1Shubertf 			/* stat(2) on pipe may return only the
141ce8f04b1Shubertf 			 * first few bytes with more coming.
142ce8f04b1Shubertf 			 * Don't trust!
143ce8f04b1Shubertf 			 */
144ce8f04b1Shubertf 		} else {
14598f6cccbSjhawk 			filesize = statb.st_size;
146ce8f04b1Shubertf 		}
147ce8f04b1Shubertf 	}
14898f6cccbSjhawk 
14998f6cccbSjhawk 	/* gzip -l the file if we have the name and -z is given */
15098f6cccbSjhawk 	if (zflag && !lflag && infile != NULL) {
15198f6cccbSjhawk 		FILE *gzipsizepipe;
1528d3b64e3Sdsl 		char buf[256], *cp, *cmd;
15398f6cccbSjhawk 
15498f6cccbSjhawk 		/*
15598f6cccbSjhawk 		 * Read second word of last line of gzip -l output. Looks like:
15698f6cccbSjhawk 		 * % gzip -l ../etc.tgz
15798f6cccbSjhawk 		 *   compressed uncompressed  ratio uncompressed_name
15898f6cccbSjhawk 		 * 	 119737       696320  82.8% ../etc.tar
15998f6cccbSjhawk 		 */
16098f6cccbSjhawk 
16198f6cccbSjhawk 		asprintf(&cmd, "gzip -l %s", infile);
16298f6cccbSjhawk 		if ((gzipsizepipe = popen(cmd, "r")) == NULL)
16398f6cccbSjhawk 			err(1, "reading compressed file length");
16498f6cccbSjhawk 		for (; fgets(buf, 256, gzipsizepipe) != NULL;)
16598f6cccbSjhawk 		    continue;
1668d3b64e3Sdsl 		strtoimax(buf, &cp, 10);
1678d3b64e3Sdsl 		filesize = strtoimax(cp, NULL, 10);
16898f6cccbSjhawk 		if (pclose(gzipsizepipe) < 0)
16998f6cccbSjhawk 			err(1, "closing compressed file length pipe");
17098f6cccbSjhawk 		free(cmd);
17198f6cccbSjhawk 	}
17298f6cccbSjhawk 	/* Pipe input through gzip -dc if -z is given */
17398f6cccbSjhawk 	if (zflag) {
17498f6cccbSjhawk 		int gzippipe[2];
17598f6cccbSjhawk 
17698f6cccbSjhawk 		if (pipe(gzippipe) < 0)
17798f6cccbSjhawk 			err(1, "gzip pipe");
17898f6cccbSjhawk 		gzippid = fork();
17998f6cccbSjhawk 		if (gzippid < 0)
18098f6cccbSjhawk 			err(1, "fork for gzip");
18198f6cccbSjhawk 
18298f6cccbSjhawk 		if (gzippid) {
18398f6cccbSjhawk 			/* parent */
18498f6cccbSjhawk 			dup2(gzippipe[0], fd);
18598f6cccbSjhawk 			close(gzippipe[0]);
18698f6cccbSjhawk 			close(gzippipe[1]);
18798f6cccbSjhawk 		} else {
18898f6cccbSjhawk 			dup2(gzippipe[1], STDOUT_FILENO);
18998f6cccbSjhawk 			dup2(fd, STDIN_FILENO);
19098f6cccbSjhawk 			close(gzippipe[0]);
19198f6cccbSjhawk 			close(gzippipe[1]);
19298f6cccbSjhawk 			if (execlp("gzip", "gzip", "-dc", NULL))
19398f6cccbSjhawk 				err(1, "exec()ing gzip");
19498f6cccbSjhawk 		}
19598f6cccbSjhawk 	}
19698f6cccbSjhawk 
19798f6cccbSjhawk 	/* Initialize progressbar.c's global state */
19898f6cccbSjhawk 	bytes = 0;
19998f6cccbSjhawk 	progress = 1;
200ddd39082Sgarbled 	ttyout = eflag ? stderr : stdout;
20118350cdfSchristos 
2021a7e106bShubertf 	if (ioctl(fileno(ttyout), TIOCGSIZE, &ts) == -1)
20398f6cccbSjhawk 		ttywidth = 80;
2041a7e106bShubertf 	else
20518350cdfSchristos 		ttywidth = ts.ts_cols;
20698f6cccbSjhawk 
207ac798ebcSbriggs 	fb_buf = malloc(buffersize);
208ac798ebcSbriggs 	if (fb_buf == NULL)
209ac798ebcSbriggs 		err(1, "malloc for buffersize");
210ac798ebcSbriggs 
21198f6cccbSjhawk 	if (pipe(outpipe) < 0)
21298f6cccbSjhawk 		err(1, "output pipe");
21398f6cccbSjhawk 	pid = fork();
21498f6cccbSjhawk 	if (pid < 0)
21598f6cccbSjhawk 		err(1, "fork for output pipe");
21698f6cccbSjhawk 
21798f6cccbSjhawk 	if (pid == 0) {
21898f6cccbSjhawk 		/* child */
21998f6cccbSjhawk 		dup2(outpipe[0], STDIN_FILENO);
22098f6cccbSjhawk 		close(outpipe[0]);
22198f6cccbSjhawk 		close(outpipe[1]);
22298f6cccbSjhawk 		execvp(argv[0], argv);
22398f6cccbSjhawk 		err(1, "could not exec %s", argv[0]);
22498f6cccbSjhawk 	}
22598f6cccbSjhawk 	close(outpipe[0]);
22698f6cccbSjhawk 
22789a35a0eSdholland 	signal(SIGPIPE, broken_pipe);
22898f6cccbSjhawk 	progressmeter(-1);
22989a35a0eSdholland 
2307a9652e2Sgson 	while (1) {
2317a9652e2Sgson 		do {
2327a9652e2Sgson 			nr = read(fd, fb_buf, buffersize);
2337a9652e2Sgson 		} while (nr < 0 && errno == EINTR);
234*e1d1003bSgson 		if (nr < 0) {
235*e1d1003bSgson 			progressmeter(1);
236*e1d1003bSgson 			err(1, "reading input");
237*e1d1003bSgson 		}
238*e1d1003bSgson 		if (nr == 0)
2397a9652e2Sgson 			break;
24098f6cccbSjhawk 		for (off = 0; nr; nr -= nw, off += nw, bytes += nw)
24123066b7fSenami 			if ((nw = write(outpipe[1], fb_buf + off,
24289a35a0eSdholland 			    (size_t) nr)) < 0) {
2430de2ffcfSlukem 				if (errno == EINTR) {
2440de2ffcfSlukem 					nw = 0;
2450de2ffcfSlukem 					continue;
2460de2ffcfSlukem 				}
24789a35a0eSdholland 				progressmeter(1);
24824b68838Sagc 				err(1, "writing %u bytes to output pipe",
24924b68838Sagc 							(unsigned) nr);
25089a35a0eSdholland 			}
2517a9652e2Sgson 	}
25298f6cccbSjhawk 	close(outpipe[1]);
25398f6cccbSjhawk 
2548d3b64e3Sdsl 	gzipstat = 0;
2558d3b64e3Sdsl 	cmdstat = 0;
2560328c523Sross 	while (pid || gzippid) {
2578d3b64e3Sdsl 		deadpid = wait(&ws);
2588d3b64e3Sdsl 		/*
2598d3b64e3Sdsl 		 * We need to exit with an error if the command (or gzip)
2608d3b64e3Sdsl 		 * exited abnormally.
2618d3b64e3Sdsl 		 * Unfortunately we can't generate a true 'exited by signal'
2628d3b64e3Sdsl 		 * error without sending the signal to ourselves :-(
2638d3b64e3Sdsl 		 */
2648d3b64e3Sdsl 		ws = WIFSIGNALED(ws) ? WTERMSIG(ws) : WEXITSTATUS(ws);
2650328c523Sross 
266a016b984Sgson 		if (deadpid == -1 && errno == EINTR)
2678d3b64e3Sdsl 			continue;
2688d3b64e3Sdsl 		if (deadpid == pid) {
2690328c523Sross 			pid = 0;
2708d3b64e3Sdsl 			cmdstat = ws;
2718d3b64e3Sdsl 			continue;
2728d3b64e3Sdsl 		}
2738d3b64e3Sdsl 		if (deadpid == gzippid) {
2740328c523Sross 			gzippid = 0;
2758d3b64e3Sdsl 			gzipstat = ws;
276182f33adSross 			continue;
2778d3b64e3Sdsl 		}
2788d3b64e3Sdsl 		break;
2790328c523Sross 	}
28098f6cccbSjhawk 
28198f6cccbSjhawk 	progressmeter(1);
28289a35a0eSdholland 	signal(SIGPIPE, SIG_DFL);
283ac798ebcSbriggs 
284ac798ebcSbriggs 	free(fb_buf);
285ac798ebcSbriggs 
2868d3b64e3Sdsl 	exit(cmdstat ? cmdstat : gzipstat);
28798f6cccbSjhawk }
288