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