xref: /netbsd-src/usr.bin/utoppya/utoppya.c (revision ff532697041141c112fb967adfa4baa7286a0805)
1*ff532697Schristos /*	$NetBSD: utoppya.c,v 1.6 2015/06/16 22:54:11 christos Exp $	*/
234b4a96dSscw 
334b4a96dSscw /*-
434b4a96dSscw  * Copyright (c) 2006 The NetBSD Foundation, Inc.
534b4a96dSscw  * All rights reserved.
634b4a96dSscw  *
734b4a96dSscw  * This code is derived from software contributed to The NetBSD Foundation
834b4a96dSscw  * by Steve C. Woodford.
934b4a96dSscw  *
1034b4a96dSscw  * Redistribution and use in source and binary forms, with or without
1134b4a96dSscw  * modification, are permitted provided that the following conditions
1234b4a96dSscw  * are met:
1334b4a96dSscw  * 1. Redistributions of source code must retain the above copyright
1434b4a96dSscw  *    notice, this list of conditions and the following disclaimer.
1534b4a96dSscw  * 2. Redistributions in binary form must reproduce the above copyright
1634b4a96dSscw  *    notice, this list of conditions and the following disclaimer in the
1734b4a96dSscw  *    documentation and/or other materials provided with the distribution.
1834b4a96dSscw  *
1934b4a96dSscw  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
2034b4a96dSscw  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
2134b4a96dSscw  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
2234b4a96dSscw  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
2334b4a96dSscw  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
2434b4a96dSscw  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
2534b4a96dSscw  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
2634b4a96dSscw  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
2734b4a96dSscw  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
2834b4a96dSscw  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
2934b4a96dSscw  * POSSIBILITY OF SUCH DAMAGE.
3034b4a96dSscw  */
3134b4a96dSscw 
3234b4a96dSscw #include <sys/types.h>
3334b4a96dSscw #include <sys/stat.h>
3434b4a96dSscw #include <sys/ioctl.h>
3534b4a96dSscw 
3634b4a96dSscw #include <err.h>
3734b4a96dSscw #include <errno.h>
3834b4a96dSscw #include <fcntl.h>
3934b4a96dSscw #include <libgen.h>
4034b4a96dSscw #include <sysexits.h>
4134b4a96dSscw #include <stdio.h>
4234b4a96dSscw #include <stdlib.h>
4334b4a96dSscw #include <strings.h>
4434b4a96dSscw #include <unistd.h>
4534b4a96dSscw #include <time.h>
46d6d78510Smartin #include <inttypes.h>
4734b4a96dSscw 
4834b4a96dSscw #include <dev/usb/utoppy.h>
4934b4a96dSscw 
5034b4a96dSscw #define	GLOBAL
5134b4a96dSscw #include "progressbar.h"
5234b4a96dSscw 
5334b4a96dSscw #define	_PATH_DEV_UTOPPY	"/dev/utoppy0"
5434b4a96dSscw 
5534b4a96dSscw /*
5634b4a96dSscw  * This looks weird for a reason. The toppy protocol allows for data to be
5734b4a96dSscw  * transferred in 65535-byte chunks only. Anything more than this has to be
5834b4a96dSscw  * split within the driver. The following value leaves enough space for the
5934b4a96dSscw  * packet header plus some alignmnent slop.
6034b4a96dSscw  */
6134b4a96dSscw #define	TOPPY_IO_SIZE	0xffec
6234b4a96dSscw 
6334b4a96dSscw static int toppy_fd;
6434b4a96dSscw 
6534b4a96dSscw static void cmd_df(int, char **);
6634b4a96dSscw static void cmd_ls(int, char **);
6734b4a96dSscw static void cmd_rm(int, char **);
6834b4a96dSscw static void cmd_mkdir(int, char **);
6934b4a96dSscw static void cmd_rename(int, char **);
7034b4a96dSscw static void cmd_get(int, char **);
7134b4a96dSscw static void cmd_put(int, char **);
7234b4a96dSscw 
737654d36aSjoerg static const struct toppy_command {
7434b4a96dSscw 	const char *tc_cmd;
7534b4a96dSscw 	void (*tc_handler)(int, char **);
7634b4a96dSscw } toppy_commands[] = {
7734b4a96dSscw 	{"df",		cmd_df},
7834b4a96dSscw 	{"ls",		cmd_ls},
7934b4a96dSscw 	{"get",		cmd_get},
8034b4a96dSscw 	{"mkdir",	cmd_mkdir},
8134b4a96dSscw 	{"put",		cmd_put},
8234b4a96dSscw 	{"rename",	cmd_rename},
8334b4a96dSscw 	{"rm",		cmd_rm},
8434b4a96dSscw 	{NULL,		NULL}
8534b4a96dSscw };
8634b4a96dSscw 
877654d36aSjoerg __dead static void
usage(void)8834b4a96dSscw usage(void)
8934b4a96dSscw {
9034b4a96dSscw 
9134b4a96dSscw 	fprintf(stderr, "usage: %s [-f <path>] <cmd> ...\n",
9234b4a96dSscw 	    getprogname());
9334b4a96dSscw 
9434b4a96dSscw 	exit(EX_USAGE);
9534b4a96dSscw }
9634b4a96dSscw 
9734b4a96dSscw int
main(int argc,char * argv[])9834b4a96dSscw main(int argc, char *argv[])
9934b4a96dSscw {
1007654d36aSjoerg 	const struct toppy_command *tc;
10134b4a96dSscw 	const char *devpath;
10234b4a96dSscw 	int ch;
10334b4a96dSscw 
10434b4a96dSscw 	setprogname(argv[0]);
10534b4a96dSscw 	devpath = _PATH_DEV_UTOPPY;
10634b4a96dSscw 
10734b4a96dSscw 	while ((ch = getopt(argc, argv, "f:")) != -1) {
10834b4a96dSscw 		switch (ch) {
10934b4a96dSscw 		case 'f':
11034b4a96dSscw 			devpath = optarg;
11134b4a96dSscw 			break;
11234b4a96dSscw 
11334b4a96dSscw 		default:
11434b4a96dSscw 			usage();
11534b4a96dSscw 		}
11634b4a96dSscw 	}
11734b4a96dSscw 	argc -= optind;
11834b4a96dSscw 	argv += optind;
11934b4a96dSscw 
12034b4a96dSscw 	if (argc == 0)
12134b4a96dSscw 		usage();
12234b4a96dSscw 
12334b4a96dSscw 	for (tc = toppy_commands; tc->tc_cmd != NULL; tc++)
12434b4a96dSscw 		if (strcasecmp(argv[0], tc->tc_cmd) == 0)
12534b4a96dSscw 			break;
12634b4a96dSscw 
12734b4a96dSscw 	if (tc->tc_cmd == NULL)
12834b4a96dSscw 		errx(EX_USAGE, "'%s' is not a valid command", argv[0]);
12934b4a96dSscw 
13034b4a96dSscw 	if ((toppy_fd = open(devpath, O_RDWR)) < 0)
13134b4a96dSscw 		err(EX_OSERR, "open(%s)", devpath);
13234b4a96dSscw 
13334b4a96dSscw 	(*tc->tc_handler)(argc, argv);
13434b4a96dSscw 
13534b4a96dSscw 	close(toppy_fd);
13634b4a96dSscw 
13734b4a96dSscw 	return (0);
13834b4a96dSscw }
13934b4a96dSscw 
14034b4a96dSscw static int
find_toppy_dirent(const char * path,struct utoppy_dirent * udp)14134b4a96dSscw find_toppy_dirent(const char *path, struct utoppy_dirent *udp)
14234b4a96dSscw {
14334b4a96dSscw 	struct utoppy_dirent ud;
14434b4a96dSscw 	char *d, *b, dir[FILENAME_MAX];
14534b4a96dSscw 	ssize_t l;
14634b4a96dSscw 
14734b4a96dSscw 	strncpy(dir, path, sizeof(dir));
14834b4a96dSscw 	b = basename(dir);
14934b4a96dSscw 	d = dirname(dir);
15034b4a96dSscw 	if (strcmp(b, "/") == 0 || strcmp(b, ".") == 0 || strcmp(d, ".") == 0)
15134b4a96dSscw 		errx(EX_USAGE, "'%s' is not a valid Toppy pathname", path);
15234b4a96dSscw 
15334b4a96dSscw 	if (ioctl(toppy_fd, UTOPPYIOREADDIR, &d) < 0)
15434b4a96dSscw 		err(EX_OSERR, "ioctl(UTOPPYIOREADDIR, %s)", d);
15534b4a96dSscw 
15634b4a96dSscw 	if (udp == NULL)
15734b4a96dSscw 		udp = &ud;
15834b4a96dSscw 
15934b4a96dSscw 	while ((l = read(toppy_fd, udp, sizeof(*udp))) == sizeof(*udp)) {
16034b4a96dSscw 		if (strcmp(b, udp->ud_path) == 0)
16134b4a96dSscw 			break;
16234b4a96dSscw 	}
16334b4a96dSscw 
16434b4a96dSscw 	if (l < 0)
16534b4a96dSscw 		err(EX_OSERR, "read(TOPPYDIR, %s)", d);
16634b4a96dSscw 
16734b4a96dSscw 	if (l == 0)
16834b4a96dSscw 		return (0);
16934b4a96dSscw 
17034b4a96dSscw 	while (read(toppy_fd, &ud, sizeof(ud)) > 0)
17134b4a96dSscw 		;
17234b4a96dSscw 
17334b4a96dSscw 	return (1);
17434b4a96dSscw }
17534b4a96dSscw 
17634b4a96dSscw static void
cmd_df(int argc,char ** argv)17734b4a96dSscw cmd_df(int argc, char **argv)
17834b4a96dSscw {
17934b4a96dSscw 	struct utoppy_stats us;
18034b4a96dSscw 
18134b4a96dSscw 	if (ioctl(toppy_fd, UTOPPYIOSTATS, &us) < 0)
18234b4a96dSscw 		err(EX_OSERR, "ioctl(UTOPPYIOSTATS)");
18334b4a96dSscw 
184d6d78510Smartin 	printf("Hard Disk Size: %" PRId64 " MB\n", us.us_hdd_size / (1024 * 1024));
185d6d78510Smartin 	printf("Hard Disk Free: %" PRId64 " MB\n", us.us_hdd_free / (1024 * 1024));
18634b4a96dSscw }
18734b4a96dSscw 
18834b4a96dSscw static void
cmd_ls(int argc,char ** argv)18934b4a96dSscw cmd_ls(int argc, char **argv)
19034b4a96dSscw {
19134b4a96dSscw 	struct utoppy_dirent ud;
19234b4a96dSscw 	struct tm *tm;
19334b4a96dSscw 	char *dir, *ext, dirbuf[2], ex, ft, tmbuf[32];
19434b4a96dSscw 	ssize_t l;
19534b4a96dSscw 
19634b4a96dSscw 	if (argc == 1) {
19734b4a96dSscw 		dirbuf[0] = '/';
19834b4a96dSscw 		dirbuf[1] = '\0';
19934b4a96dSscw 		dir = dirbuf;
20034b4a96dSscw 	} else
20134b4a96dSscw 	if (argc == 2)
20234b4a96dSscw 		dir = argv[1];
20334b4a96dSscw 	else
20434b4a96dSscw 		errx(EX_USAGE, "usage: ls [toppy-pathname]");
20534b4a96dSscw 
20634b4a96dSscw 	if (ioctl(toppy_fd, UTOPPYIOREADDIR, &dir) < 0)
20734b4a96dSscw 		err(EX_OSERR, "ioctl(UTOPPYIOREADDIR, %s)", dir);
20834b4a96dSscw 
20934b4a96dSscw 	while ((l = read(toppy_fd, &ud, sizeof(ud))) == sizeof(ud)) {
21034b4a96dSscw 		switch (ud.ud_type) {
21134b4a96dSscw 		default:
21234b4a96dSscw 			ft = '?';
21334b4a96dSscw 			break;
21434b4a96dSscw 
21534b4a96dSscw 		case UTOPPY_DIRENT_DIRECTORY:
21634b4a96dSscw 			ft = 'd';
21734b4a96dSscw 			break;
21834b4a96dSscw 
21934b4a96dSscw 		case UTOPPY_DIRENT_FILE:
22034b4a96dSscw 			ft = '-';
22134b4a96dSscw 			break;
22234b4a96dSscw 		}
22334b4a96dSscw 
22434b4a96dSscw 		if ((ext = strrchr(ud.ud_path, '.')) != NULL &&
22534b4a96dSscw 		    strcasecmp(ext, ".tap") == 0)
22634b4a96dSscw 			ex = 'x';
22734b4a96dSscw 		else
22834b4a96dSscw 			ex = '-';
22934b4a96dSscw 
23034b4a96dSscw 		tm = localtime(&ud.ud_mtime);
23134b4a96dSscw 		strftime(tmbuf, sizeof(tmbuf), "%b %e %G %R", tm);
23234b4a96dSscw 
233d6d78510Smartin 		printf("%crw%c %11lld %s %s\n", ft, ex, (long long)ud.ud_size,
234d6d78510Smartin 		    tmbuf, ud.ud_path);
23534b4a96dSscw 	}
23634b4a96dSscw 
23734b4a96dSscw 	if (l < 0)
23834b4a96dSscw 		err(EX_OSERR, "read(utoppy_dirent)");
23934b4a96dSscw }
24034b4a96dSscw 
24134b4a96dSscw static void
cmd_rm(int argc,char ** argv)24234b4a96dSscw cmd_rm(int argc, char **argv)
24334b4a96dSscw {
24434b4a96dSscw 	char *path;
24534b4a96dSscw 
24634b4a96dSscw 	if (argc != 2)
24734b4a96dSscw 		errx(EX_USAGE, "usage: rm <toppy-pathname>");
24834b4a96dSscw 
24934b4a96dSscw 	path = argv[1];
25034b4a96dSscw 
25134b4a96dSscw 	if (ioctl(toppy_fd, UTOPPYIODELETE, &path) < 0)
25234b4a96dSscw 		err(EX_OSERR, "ioctl(UTOPPYIODELETE, %s)", path);
25334b4a96dSscw }
25434b4a96dSscw 
25534b4a96dSscw static void
cmd_mkdir(int argc,char ** argv)25634b4a96dSscw cmd_mkdir(int argc, char **argv)
25734b4a96dSscw {
25834b4a96dSscw 	char *path;
25934b4a96dSscw 
26034b4a96dSscw 	if (argc != 2)
26134b4a96dSscw 		errx(EX_USAGE, "usage: mkdir <toppy-pathname>");
26234b4a96dSscw 
26334b4a96dSscw 	path = argv[1];
26434b4a96dSscw 
26534b4a96dSscw 	if (find_toppy_dirent(path, NULL))
26634b4a96dSscw 		errx(EX_DATAERR, "'%s' already exists", path);
26734b4a96dSscw 
26834b4a96dSscw 	if (ioctl(toppy_fd, UTOPPYIOMKDIR, &path) < 0)
26934b4a96dSscw 		err(EX_OSERR, "ioctl(UTOPPYIOMKDIR, %s)", path);
27034b4a96dSscw }
27134b4a96dSscw 
27234b4a96dSscw static void
cmd_rename(int argc,char ** argv)27334b4a96dSscw cmd_rename(int argc, char **argv)
27434b4a96dSscw {
27534b4a96dSscw 	struct utoppy_dirent ud;
27634b4a96dSscw 	struct utoppy_rename ur;
27734b4a96dSscw 	char *oldpath, *newpath, *o, *n;
27834b4a96dSscw 
27934b4a96dSscw 	if (argc != 3)
28034b4a96dSscw 		errx(EX_USAGE, "usage: rename <from> <to>");
28134b4a96dSscw 
28234b4a96dSscw 	o = oldpath = argv[1];
28334b4a96dSscw 	n = newpath = argv[2];
28434b4a96dSscw 
28534b4a96dSscw 	for (o = oldpath; *o != '\0'; o++)
28634b4a96dSscw 		if (*o == '\\')
28734b4a96dSscw 			*o = '/';
28834b4a96dSscw 	for (n = newpath; *n != '\0'; n++)
28934b4a96dSscw 		if (*n == '\\')
29034b4a96dSscw 			*n = '/';
29134b4a96dSscw 
29234b4a96dSscw 	for (o = oldpath; *o && *o == '\\'; o++)
29334b4a96dSscw 		;
29434b4a96dSscw 	for (n = newpath; *n && *n == '\\'; n++)
29534b4a96dSscw 		;
29634b4a96dSscw 
29734b4a96dSscw 	if (strcmp(n, o) == 0)
298*ff532697Schristos 		errx(EX_DATAERR, "'%s' and '%s' refer to the same file",
29934b4a96dSscw 		    oldpath, newpath);
30034b4a96dSscw 
30134b4a96dSscw 	if (find_toppy_dirent(oldpath, &ud) == 0)
30234b4a96dSscw 		errx(EX_DATAERR, "'%s' does not exist on the Toppy", oldpath);
30334b4a96dSscw 
30434b4a96dSscw 	if (ud.ud_type != UTOPPY_DIRENT_FILE)
30534b4a96dSscw 		errx(EX_DATAERR, "%s: not a regular file", oldpath);
30634b4a96dSscw 
30734b4a96dSscw 	if (find_toppy_dirent(newpath, &ud))
30834b4a96dSscw 		errx(EX_DATAERR, "'%s' already exists", newpath);
30934b4a96dSscw 
31034b4a96dSscw 	ur.ur_old_path = o;
31134b4a96dSscw 	ur.ur_new_path = n;
31234b4a96dSscw 
31334b4a96dSscw 	if (ioctl(toppy_fd, UTOPPYIORENAME, &ur) < 0)
31434b4a96dSscw 		err(EX_OSERR, "ioctl(UTOPPYIORENAME, %s, %s)", oldpath,
31534b4a96dSscw 		    newpath);
31634b4a96dSscw }
31734b4a96dSscw 
31834b4a96dSscw 
31934b4a96dSscw static void
init_progress(FILE * to,char * f,off_t fsize,off_t restart)32034b4a96dSscw init_progress(FILE *to, char *f, off_t fsize, off_t restart)
32134b4a96dSscw {
32234b4a96dSscw 	struct ttysize ts;
32334b4a96dSscw 
32434b4a96dSscw 	if (ioctl(fileno(to), TIOCGSIZE, &ts) == -1)
32534b4a96dSscw 		ttywidth = 80;
32634b4a96dSscw 	else
32734b4a96dSscw 		ttywidth = ts.ts_cols;
32834b4a96dSscw 
32934b4a96dSscw 	ttyout = to;
33034b4a96dSscw 	progress = 1;
33134b4a96dSscw 	bytes = 0;
33234b4a96dSscw 	filesize = fsize;
33334b4a96dSscw 	restart_point = restart;
33434b4a96dSscw 	prefix = f;
33534b4a96dSscw }
33634b4a96dSscw 
33734b4a96dSscw static void
cmd_get(int argc,char ** argv)33834b4a96dSscw cmd_get(int argc, char **argv)
33934b4a96dSscw {
34034b4a96dSscw 	struct utoppy_readfile ur;
34134b4a96dSscw 	struct utoppy_dirent ud;
34234b4a96dSscw 	struct stat st;
34334b4a96dSscw 	char *dst, dstbuf[FILENAME_MAX];
34434b4a96dSscw 	uint8_t *buf;
34534b4a96dSscw 	ssize_t l;
34634b4a96dSscw 	size_t rv;
34734b4a96dSscw 	int ch, turbo_mode = 0, reget = 0, progbar = 0;
34834b4a96dSscw 	FILE *ofp, *to;
34934b4a96dSscw 
35034b4a96dSscw 	optind = 1;
35134b4a96dSscw 	optreset = 1;
35234b4a96dSscw 
35334b4a96dSscw 	while ((ch = getopt(argc, argv, "prt")) != -1) {
35434b4a96dSscw 		switch (ch) {
35534b4a96dSscw 		case 'p':
35634b4a96dSscw 			progbar = 1;
35734b4a96dSscw 			break;
35834b4a96dSscw 		case 'r':
35934b4a96dSscw 			reget = 1;
36034b4a96dSscw 			break;
36134b4a96dSscw 		case 't':
36234b4a96dSscw 			turbo_mode = 1;
36334b4a96dSscw 			break;
36434b4a96dSscw 		default:
36534b4a96dSscw  get_usage:
36634b4a96dSscw 			errx(EX_USAGE, "usage: get [-prt] <toppy-pathname> "
36734b4a96dSscw 			    "[file | directory]");
36834b4a96dSscw 		}
36934b4a96dSscw 	}
37034b4a96dSscw 	argc -= optind;
37134b4a96dSscw 	argv += optind;
37234b4a96dSscw 
37334b4a96dSscw 	if (argc == 1)
37434b4a96dSscw 		dst = basename(argv[0]);
37534b4a96dSscw 	else
37634b4a96dSscw 	if (argc == 2) {
37734b4a96dSscw 		dst = argv[1];
37834b4a96dSscw 		if (stat(dst, &st) == 0 && S_ISDIR(st.st_mode)) {
37934b4a96dSscw 			snprintf(dstbuf, sizeof(dstbuf), "%s/%s", dst,
38034b4a96dSscw 			    basename(argv[0]));
38134b4a96dSscw 			dst = dstbuf;
38234b4a96dSscw 		}
38334b4a96dSscw 	} else
38434b4a96dSscw 		goto get_usage;
38534b4a96dSscw 
38634b4a96dSscw 	ur.ur_path = argv[0];
38734b4a96dSscw 	ur.ur_offset = 0;
38834b4a96dSscw 
38934b4a96dSscw 	if ((buf = malloc(TOPPY_IO_SIZE)) == NULL)
39034b4a96dSscw 		err(EX_OSERR, "malloc(TOPPY_IO_SIZE)");
39134b4a96dSscw 
39234b4a96dSscw 	if (strcmp(dst, "-") == 0) {
39334b4a96dSscw 		ofp = stdout;
39434b4a96dSscw 		to = stderr;
39534b4a96dSscw 		if (reget)
39634b4a96dSscw 			warnx("Ignoring -r option in combination with stdout");
39734b4a96dSscw 	} else {
39834b4a96dSscw 		to = stdout;
39934b4a96dSscw 
40034b4a96dSscw 		if (reget) {
40134b4a96dSscw 			if (stat(dst, &st) < 0) {
40234b4a96dSscw 				if (errno != ENOENT)
40334b4a96dSscw 					err(EX_OSERR, "stat(%s)", dst);
40434b4a96dSscw 			} else
40534b4a96dSscw 			if (!S_ISREG(st.st_mode))
40634b4a96dSscw 				errx(EX_DATAERR, "-r only works with regular "
40734b4a96dSscw 				    "files");
40834b4a96dSscw 			else
40934b4a96dSscw 				ur.ur_offset = st.st_size;
41034b4a96dSscw 		}
41134b4a96dSscw 
41234b4a96dSscw 		if ((ofp = fopen(dst, reget ? "a" : "w")) == NULL)
41334b4a96dSscw 			err(EX_OSERR, "fopen(%s)", dst);
41434b4a96dSscw 	}
41534b4a96dSscw 
41634b4a96dSscw 	if (progbar) {
41734b4a96dSscw 		if (find_toppy_dirent(ur.ur_path, &ud) == 0)
41834b4a96dSscw 			ud.ud_size = 0;
41934b4a96dSscw 		init_progress(to, dst, ud.ud_size, ur.ur_offset);
42034b4a96dSscw 	}
42134b4a96dSscw 
42234b4a96dSscw 	if (ioctl(toppy_fd, UTOPPYIOTURBO, &turbo_mode) < 0)
42334b4a96dSscw 		err(EX_OSERR, "ioctl(UTOPPYIOTURBO, %d)", turbo_mode);
42434b4a96dSscw 
42534b4a96dSscw 	if (ioctl(toppy_fd, UTOPPYIOREADFILE, &ur) < 0)
42634b4a96dSscw 		err(EX_OSERR, "ioctl(UTOPPYIOREADFILE, %s)", ur.ur_path);
42734b4a96dSscw 
42834b4a96dSscw 	if (progbar)
42934b4a96dSscw 		progressmeter(-1);
43034b4a96dSscw 
43134b4a96dSscw 	for (;;) {
43234b4a96dSscw 		while ((l = read(toppy_fd, buf, TOPPY_IO_SIZE)) < 0 &&
43334b4a96dSscw 		    errno == EINTR)
43434b4a96dSscw 			;
43534b4a96dSscw 
43634b4a96dSscw 		if (l <= 0)
43734b4a96dSscw 			break;
43834b4a96dSscw 
43934b4a96dSscw 		rv = fwrite(buf, 1, l, ofp);
44034b4a96dSscw 
44123ed3887Slukem 		if (rv != (size_t)l) {
44234b4a96dSscw 			if (ofp != stdout)
44334b4a96dSscw 				fclose(ofp);
44434b4a96dSscw 			progressmeter(1);
44534b4a96dSscw 			err(EX_OSERR, "fwrite(%s)", dst);
44634b4a96dSscw 		}
44734b4a96dSscw 		bytes += l;
44834b4a96dSscw 	}
44934b4a96dSscw 
45034b4a96dSscw 	if (progbar)
45134b4a96dSscw 		progressmeter(1);
45234b4a96dSscw 
45334b4a96dSscw 	if (ofp != stdout)
45434b4a96dSscw 		fclose(ofp);
45534b4a96dSscw 
45634b4a96dSscw 	if (l < 0)
45734b4a96dSscw 		err(EX_OSERR, "read(TOPPY: ur.ur_path)");
45834b4a96dSscw 
45934b4a96dSscw 	free(buf);
46034b4a96dSscw }
46134b4a96dSscw 
46234b4a96dSscw static void
cmd_put(int argc,char ** argv)46334b4a96dSscw cmd_put(int argc, char **argv)
46434b4a96dSscw {
46534b4a96dSscw 	struct utoppy_writefile uw;
46634b4a96dSscw 	struct utoppy_dirent ud;
46734b4a96dSscw 	struct stat st;
46834b4a96dSscw 	char dstbuf[FILENAME_MAX];
46934b4a96dSscw 	char *src;
47034b4a96dSscw 	void *buf;
47134b4a96dSscw 	ssize_t rv;
47234b4a96dSscw 	size_t l;
47334b4a96dSscw 	int ch, turbo_mode = 0, reput = 0, progbar = 0;
47434b4a96dSscw 	FILE *ifp;
47534b4a96dSscw 
47634b4a96dSscw 	optind = 1;
47734b4a96dSscw 	optreset = 1;
47834b4a96dSscw 
47934b4a96dSscw 	while ((ch = getopt(argc, argv, "prt")) != -1) {
48034b4a96dSscw 		switch (ch) {
48134b4a96dSscw 		case 'p':
48234b4a96dSscw 			progbar = 1;
48334b4a96dSscw 			break;
48434b4a96dSscw 		case 'r':
48534b4a96dSscw 			reput = 1;
48634b4a96dSscw 			break;
48734b4a96dSscw 		case 't':
48834b4a96dSscw 			turbo_mode = 1;
48934b4a96dSscw 			break;
49034b4a96dSscw 		default:
49134b4a96dSscw  put_usage:
49234b4a96dSscw 			errx(EX_USAGE, "usage: put [-prt] <local-pathname> "
49334b4a96dSscw 			    "<toppy-pathname>");
49434b4a96dSscw 		}
49534b4a96dSscw 	}
49634b4a96dSscw 	argc -= optind;
49734b4a96dSscw 	argv += optind;
49834b4a96dSscw 
49934b4a96dSscw 	if (argc != 2)
50034b4a96dSscw 		goto put_usage;
50134b4a96dSscw 
50234b4a96dSscw 	src = argv[0];
50334b4a96dSscw 	uw.uw_path = argv[1];
50434b4a96dSscw 
50534b4a96dSscw 	if (stat(src, &st) < 0)
50634b4a96dSscw 		err(EX_OSERR, "%s", src);
50734b4a96dSscw 
50834b4a96dSscw 	if (!S_ISREG(st.st_mode))
50934b4a96dSscw 		errx(EX_DATAERR, "'%s' is not a regular file", src);
51034b4a96dSscw 
51134b4a96dSscw 	uw.uw_size = st.st_size;
51234b4a96dSscw 	uw.uw_mtime = st.st_mtime;
51334b4a96dSscw 	uw.uw_offset = 0;
51434b4a96dSscw 
51534b4a96dSscw 	if (find_toppy_dirent(uw.uw_path, &ud)) {
51634b4a96dSscw 		if (ud.ud_type == UTOPPY_DIRENT_DIRECTORY) {
51734b4a96dSscw 			snprintf(dstbuf, sizeof(dstbuf), "%s/%s", uw.uw_path,
51834b4a96dSscw 			    basename(src));
51934b4a96dSscw 			uw.uw_path = dstbuf;
52034b4a96dSscw 		} else
52134b4a96dSscw 		if (ud.ud_type != UTOPPY_DIRENT_FILE)
52234b4a96dSscw 			errx(EX_DATAERR, "'%s' is not a regular file.",
52334b4a96dSscw 			    uw.uw_path);
52434b4a96dSscw 		else
52534b4a96dSscw 		if (reput) {
52634b4a96dSscw 			if (ud.ud_size > uw.uw_size)
52734b4a96dSscw 				errx(EX_DATAERR, "'%s' is already larger than "
52834b4a96dSscw 				    "'%s'", uw.uw_path, src);
52934b4a96dSscw 
53034b4a96dSscw 			uw.uw_size -= ud.ud_size;
53134b4a96dSscw 			uw.uw_offset = ud.ud_size;
53234b4a96dSscw 		}
53334b4a96dSscw 	}
53434b4a96dSscw 
53534b4a96dSscw 	if ((buf = malloc(TOPPY_IO_SIZE)) == NULL)
53634b4a96dSscw 		err(EX_OSERR, "malloc(TOPPY_IO_SIZE)");
53734b4a96dSscw 
53834b4a96dSscw 	if ((ifp = fopen(src, "r")) == NULL)
53934b4a96dSscw 		err(EX_OSERR, "fopen(%s)", src);
54034b4a96dSscw 
54134b4a96dSscw 	if (ioctl(toppy_fd, UTOPPYIOTURBO, &turbo_mode) < 0)
54234b4a96dSscw 		err(EX_OSERR, "ioctl(UTOPPYIOTURBO, %d)", turbo_mode);
54334b4a96dSscw 
54434b4a96dSscw 	if (ioctl(toppy_fd, UTOPPYIOWRITEFILE, &uw) < 0)
54534b4a96dSscw 		err(EX_OSERR, "ioctl(UTOPPYIOWRITEFILE, %s)", uw.uw_path);
54634b4a96dSscw 
54734b4a96dSscw 	if (progbar)
54834b4a96dSscw 		init_progress(stdout, src, st.st_size, uw.uw_offset);
54934b4a96dSscw 
55034b4a96dSscw 	if (progbar)
55134b4a96dSscw 		progressmeter(-1);
55234b4a96dSscw 
55334b4a96dSscw 	while ((l = fread(buf, 1, TOPPY_IO_SIZE, ifp)) > 0) {
55434b4a96dSscw 		rv = write(toppy_fd, buf, l);
55523ed3887Slukem 		if ((size_t)rv != l) {
55634b4a96dSscw 			fclose(ifp);
55734b4a96dSscw 			if (progbar)
55834b4a96dSscw 				progressmeter(1);
55934b4a96dSscw 			err(EX_OSERR, "write(TOPPY: %s)", uw.uw_path);
56034b4a96dSscw 		}
56134b4a96dSscw 		bytes += l;
56234b4a96dSscw 	}
56334b4a96dSscw 
56434b4a96dSscw 	if (progbar)
56534b4a96dSscw 		progressmeter(1);
56634b4a96dSscw 
56734b4a96dSscw 	if (ferror(ifp))
56834b4a96dSscw 		err(EX_OSERR, "fread(%s)", src);
56934b4a96dSscw 
57034b4a96dSscw 	fclose(ifp);
57134b4a96dSscw 	free(buf);
57234b4a96dSscw }
573