Home | History | Annotate | Line # | Download | only in utoppya
utoppya.c revision 1.2
      1  1.2  martin /*	$NetBSD: utoppya.c,v 1.2 2006/04/03 13:30:24 martin Exp $	*/
      2  1.1     scw 
      3  1.1     scw /*-
      4  1.1     scw  * Copyright (c) 2006 The NetBSD Foundation, Inc.
      5  1.1     scw  * All rights reserved.
      6  1.1     scw  *
      7  1.1     scw  * This code is derived from software contributed to The NetBSD Foundation
      8  1.1     scw  * by Steve C. Woodford.
      9  1.1     scw  *
     10  1.1     scw  * Redistribution and use in source and binary forms, with or without
     11  1.1     scw  * modification, are permitted provided that the following conditions
     12  1.1     scw  * are met:
     13  1.1     scw  * 1. Redistributions of source code must retain the above copyright
     14  1.1     scw  *    notice, this list of conditions and the following disclaimer.
     15  1.1     scw  * 2. Redistributions in binary form must reproduce the above copyright
     16  1.1     scw  *    notice, this list of conditions and the following disclaimer in the
     17  1.1     scw  *    documentation and/or other materials provided with the distribution.
     18  1.1     scw  * 3. All advertising materials mentioning features or use of this software
     19  1.1     scw  *    must display the following acknowledgement:
     20  1.1     scw  *        This product includes software developed by the NetBSD
     21  1.1     scw  *        Foundation, Inc. and its contributors.
     22  1.1     scw  * 4. Neither the name of The NetBSD Foundation nor the names of its
     23  1.1     scw  *    contributors may be used to endorse or promote products derived
     24  1.1     scw  *    from this software without specific prior written permission.
     25  1.1     scw  *
     26  1.1     scw  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
     27  1.1     scw  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
     28  1.1     scw  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
     29  1.1     scw  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
     30  1.1     scw  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
     31  1.1     scw  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
     32  1.1     scw  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
     33  1.1     scw  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
     34  1.1     scw  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
     35  1.1     scw  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
     36  1.1     scw  * POSSIBILITY OF SUCH DAMAGE.
     37  1.1     scw  */
     38  1.1     scw 
     39  1.1     scw #include <sys/types.h>
     40  1.1     scw #include <sys/stat.h>
     41  1.1     scw #include <sys/ioctl.h>
     42  1.1     scw 
     43  1.1     scw #include <err.h>
     44  1.1     scw #include <errno.h>
     45  1.1     scw #include <fcntl.h>
     46  1.1     scw #include <libgen.h>
     47  1.1     scw #include <sysexits.h>
     48  1.1     scw #include <stdio.h>
     49  1.1     scw #include <stdlib.h>
     50  1.1     scw #include <strings.h>
     51  1.1     scw #include <unistd.h>
     52  1.1     scw #include <time.h>
     53  1.2  martin #include <inttypes.h>
     54  1.1     scw 
     55  1.1     scw #include <dev/usb/utoppy.h>
     56  1.1     scw 
     57  1.1     scw #define	GLOBAL
     58  1.1     scw #include "progressbar.h"
     59  1.1     scw 
     60  1.1     scw #define	_PATH_DEV_UTOPPY	"/dev/utoppy0"
     61  1.1     scw 
     62  1.1     scw /*
     63  1.1     scw  * This looks weird for a reason. The toppy protocol allows for data to be
     64  1.1     scw  * transferred in 65535-byte chunks only. Anything more than this has to be
     65  1.1     scw  * split within the driver. The following value leaves enough space for the
     66  1.1     scw  * packet header plus some alignmnent slop.
     67  1.1     scw  */
     68  1.1     scw #define	TOPPY_IO_SIZE	0xffec
     69  1.1     scw 
     70  1.1     scw static int toppy_fd;
     71  1.1     scw 
     72  1.1     scw static void cmd_df(int, char **);
     73  1.1     scw static void cmd_ls(int, char **);
     74  1.1     scw static void cmd_rm(int, char **);
     75  1.1     scw static void cmd_mkdir(int, char **);
     76  1.1     scw static void cmd_rename(int, char **);
     77  1.1     scw static void cmd_get(int, char **);
     78  1.1     scw static void cmd_put(int, char **);
     79  1.1     scw 
     80  1.1     scw static struct toppy_command {
     81  1.1     scw 	const char *tc_cmd;
     82  1.1     scw 	void (*tc_handler)(int, char **);
     83  1.1     scw } toppy_commands[] = {
     84  1.1     scw 	{"df",		cmd_df},
     85  1.1     scw 	{"ls",		cmd_ls},
     86  1.1     scw 	{"get",		cmd_get},
     87  1.1     scw 	{"mkdir",	cmd_mkdir},
     88  1.1     scw 	{"put",		cmd_put},
     89  1.1     scw 	{"rename",	cmd_rename},
     90  1.1     scw 	{"rm",		cmd_rm},
     91  1.1     scw 	{NULL,		NULL}
     92  1.1     scw };
     93  1.1     scw 
     94  1.1     scw static void
     95  1.1     scw usage(void)
     96  1.1     scw {
     97  1.1     scw 
     98  1.1     scw 	fprintf(stderr, "usage: %s [-f <path>] <cmd> ...\n",
     99  1.1     scw 	    getprogname());
    100  1.1     scw 
    101  1.1     scw 	exit(EX_USAGE);
    102  1.1     scw }
    103  1.1     scw 
    104  1.1     scw int
    105  1.1     scw main(int argc, char *argv[])
    106  1.1     scw {
    107  1.1     scw 	struct toppy_command *tc;
    108  1.1     scw 	const char *devpath;
    109  1.1     scw 	int ch;
    110  1.1     scw 
    111  1.1     scw 	setprogname(argv[0]);
    112  1.1     scw 	devpath = _PATH_DEV_UTOPPY;
    113  1.1     scw 
    114  1.1     scw 	while ((ch = getopt(argc, argv, "f:")) != -1) {
    115  1.1     scw 		switch (ch) {
    116  1.1     scw 		case 'f':
    117  1.1     scw 			devpath = optarg;
    118  1.1     scw 			break;
    119  1.1     scw 
    120  1.1     scw 		default:
    121  1.1     scw 			usage();
    122  1.1     scw 		}
    123  1.1     scw 	}
    124  1.1     scw 	argc -= optind;
    125  1.1     scw 	argv += optind;
    126  1.1     scw 
    127  1.1     scw 	if (argc == 0)
    128  1.1     scw 		usage();
    129  1.1     scw 
    130  1.1     scw 	for (tc = toppy_commands; tc->tc_cmd != NULL; tc++)
    131  1.1     scw 		if (strcasecmp(argv[0], tc->tc_cmd) == 0)
    132  1.1     scw 			break;
    133  1.1     scw 
    134  1.1     scw 	if (tc->tc_cmd == NULL)
    135  1.1     scw 		errx(EX_USAGE, "'%s' is not a valid command", argv[0]);
    136  1.1     scw 
    137  1.1     scw 	if ((toppy_fd = open(devpath, O_RDWR)) < 0)
    138  1.1     scw 		err(EX_OSERR, "open(%s)", devpath);
    139  1.1     scw 
    140  1.1     scw 	(*tc->tc_handler)(argc, argv);
    141  1.1     scw 
    142  1.1     scw 	close(toppy_fd);
    143  1.1     scw 
    144  1.1     scw 	return (0);
    145  1.1     scw }
    146  1.1     scw 
    147  1.1     scw static int
    148  1.1     scw find_toppy_dirent(const char *path, struct utoppy_dirent *udp)
    149  1.1     scw {
    150  1.1     scw 	struct utoppy_dirent ud;
    151  1.1     scw 	char *d, *b, dir[FILENAME_MAX];
    152  1.1     scw 	ssize_t l;
    153  1.1     scw 
    154  1.1     scw 	strncpy(dir, path, sizeof(dir));
    155  1.1     scw 	b = basename(dir);
    156  1.1     scw 	d = dirname(dir);
    157  1.1     scw 	if (strcmp(b, "/") == 0 || strcmp(b, ".") == 0 || strcmp(d, ".") == 0)
    158  1.1     scw 		errx(EX_USAGE, "'%s' is not a valid Toppy pathname", path);
    159  1.1     scw 
    160  1.1     scw 	if (ioctl(toppy_fd, UTOPPYIOREADDIR, &d) < 0)
    161  1.1     scw 		err(EX_OSERR, "ioctl(UTOPPYIOREADDIR, %s)", d);
    162  1.1     scw 
    163  1.1     scw 	if (udp == NULL)
    164  1.1     scw 		udp = &ud;
    165  1.1     scw 
    166  1.1     scw 	while ((l = read(toppy_fd, udp, sizeof(*udp))) == sizeof(*udp)) {
    167  1.1     scw 		if (strcmp(b, udp->ud_path) == 0)
    168  1.1     scw 			break;
    169  1.1     scw 	}
    170  1.1     scw 
    171  1.1     scw 	if (l < 0)
    172  1.1     scw 		err(EX_OSERR, "read(TOPPYDIR, %s)", d);
    173  1.1     scw 
    174  1.1     scw 	if (l == 0)
    175  1.1     scw 		return (0);
    176  1.1     scw 
    177  1.1     scw 	while (read(toppy_fd, &ud, sizeof(ud)) > 0)
    178  1.1     scw 		;
    179  1.1     scw 
    180  1.1     scw 	return (1);
    181  1.1     scw }
    182  1.1     scw 
    183  1.1     scw static void
    184  1.1     scw cmd_df(int argc, char **argv)
    185  1.1     scw {
    186  1.1     scw 	struct utoppy_stats us;
    187  1.1     scw 
    188  1.1     scw 	if (ioctl(toppy_fd, UTOPPYIOSTATS, &us) < 0)
    189  1.1     scw 		err(EX_OSERR, "ioctl(UTOPPYIOSTATS)");
    190  1.1     scw 
    191  1.2  martin 	printf("Hard Disk Size: %" PRId64 " MB\n", us.us_hdd_size / (1024 * 1024));
    192  1.2  martin 	printf("Hard Disk Free: %" PRId64 " MB\n", us.us_hdd_free / (1024 * 1024));
    193  1.1     scw }
    194  1.1     scw 
    195  1.1     scw static void
    196  1.1     scw cmd_ls(int argc, char **argv)
    197  1.1     scw {
    198  1.1     scw 	struct utoppy_dirent ud;
    199  1.1     scw 	struct tm *tm;
    200  1.1     scw 	char *dir, *ext, dirbuf[2], ex, ft, tmbuf[32];
    201  1.1     scw 	ssize_t l;
    202  1.1     scw 
    203  1.1     scw 	if (argc == 1) {
    204  1.1     scw 		dirbuf[0] = '/';
    205  1.1     scw 		dirbuf[1] = '\0';
    206  1.1     scw 		dir = dirbuf;
    207  1.1     scw 	} else
    208  1.1     scw 	if (argc == 2)
    209  1.1     scw 		dir = argv[1];
    210  1.1     scw 	else
    211  1.1     scw 		errx(EX_USAGE, "usage: ls [toppy-pathname]");
    212  1.1     scw 
    213  1.1     scw 	if (ioctl(toppy_fd, UTOPPYIOREADDIR, &dir) < 0)
    214  1.1     scw 		err(EX_OSERR, "ioctl(UTOPPYIOREADDIR, %s)", dir);
    215  1.1     scw 
    216  1.1     scw 	while ((l = read(toppy_fd, &ud, sizeof(ud))) == sizeof(ud)) {
    217  1.1     scw 		switch (ud.ud_type) {
    218  1.1     scw 		default:
    219  1.1     scw 			ft = '?';
    220  1.1     scw 			break;
    221  1.1     scw 
    222  1.1     scw 		case UTOPPY_DIRENT_DIRECTORY:
    223  1.1     scw 			ft = 'd';
    224  1.1     scw 			break;
    225  1.1     scw 
    226  1.1     scw 		case UTOPPY_DIRENT_FILE:
    227  1.1     scw 			ft = '-';
    228  1.1     scw 			break;
    229  1.1     scw 		}
    230  1.1     scw 
    231  1.1     scw 		if ((ext = strrchr(ud.ud_path, '.')) != NULL &&
    232  1.1     scw 		    strcasecmp(ext, ".tap") == 0)
    233  1.1     scw 			ex = 'x';
    234  1.1     scw 		else
    235  1.1     scw 			ex = '-';
    236  1.1     scw 
    237  1.1     scw 		tm = localtime(&ud.ud_mtime);
    238  1.1     scw 		strftime(tmbuf, sizeof(tmbuf), "%b %e %G %R", tm);
    239  1.1     scw 
    240  1.2  martin 		printf("%crw%c %11lld %s %s\n", ft, ex, (long long)ud.ud_size,
    241  1.2  martin 		    tmbuf, ud.ud_path);
    242  1.1     scw 	}
    243  1.1     scw 
    244  1.1     scw 	if (l < 0)
    245  1.1     scw 		err(EX_OSERR, "read(utoppy_dirent)");
    246  1.1     scw }
    247  1.1     scw 
    248  1.1     scw static void
    249  1.1     scw cmd_rm(int argc, char **argv)
    250  1.1     scw {
    251  1.1     scw 	char *path;
    252  1.1     scw 
    253  1.1     scw 	if (argc != 2)
    254  1.1     scw 		errx(EX_USAGE, "usage: rm <toppy-pathname>");
    255  1.1     scw 
    256  1.1     scw 	path = argv[1];
    257  1.1     scw 
    258  1.1     scw 	if (ioctl(toppy_fd, UTOPPYIODELETE, &path) < 0)
    259  1.1     scw 		err(EX_OSERR, "ioctl(UTOPPYIODELETE, %s)", path);
    260  1.1     scw }
    261  1.1     scw 
    262  1.1     scw static void
    263  1.1     scw cmd_mkdir(int argc, char **argv)
    264  1.1     scw {
    265  1.1     scw 	char *path;
    266  1.1     scw 
    267  1.1     scw 	if (argc != 2)
    268  1.1     scw 		errx(EX_USAGE, "usage: mkdir <toppy-pathname>");
    269  1.1     scw 
    270  1.1     scw 	path = argv[1];
    271  1.1     scw 
    272  1.1     scw 	if (find_toppy_dirent(path, NULL))
    273  1.1     scw 		errx(EX_DATAERR, "'%s' already exists", path);
    274  1.1     scw 
    275  1.1     scw 	if (ioctl(toppy_fd, UTOPPYIOMKDIR, &path) < 0)
    276  1.1     scw 		err(EX_OSERR, "ioctl(UTOPPYIOMKDIR, %s)", path);
    277  1.1     scw }
    278  1.1     scw 
    279  1.1     scw static void
    280  1.1     scw cmd_rename(int argc, char **argv)
    281  1.1     scw {
    282  1.1     scw 	struct utoppy_dirent ud;
    283  1.1     scw 	struct utoppy_rename ur;
    284  1.1     scw 	char *oldpath, *newpath, *o, *n;
    285  1.1     scw 
    286  1.1     scw 	if (argc != 3)
    287  1.1     scw 		errx(EX_USAGE, "usage: rename <from> <to>");
    288  1.1     scw 
    289  1.1     scw 	o = oldpath = argv[1];
    290  1.1     scw 	n = newpath = argv[2];
    291  1.1     scw 
    292  1.1     scw 	for (o = oldpath; *o != '\0'; o++)
    293  1.1     scw 		if (*o == '\\')
    294  1.1     scw 			*o = '/';
    295  1.1     scw 	for (n = newpath; *n != '\0'; n++)
    296  1.1     scw 		if (*n == '\\')
    297  1.1     scw 			*n = '/';
    298  1.1     scw 
    299  1.1     scw 	for (o = oldpath; *o && *o == '\\'; o++)
    300  1.1     scw 		;
    301  1.1     scw 	for (n = newpath; *n && *n == '\\'; n++)
    302  1.1     scw 		;
    303  1.1     scw 
    304  1.1     scw 	if (strcmp(n, o) == 0)
    305  1.1     scw 		errx(EX_DATAERR, "'%s' and '%s' refer to the same file\n",
    306  1.1     scw 		    oldpath, newpath);
    307  1.1     scw 
    308  1.1     scw 	if (find_toppy_dirent(oldpath, &ud) == 0)
    309  1.1     scw 		errx(EX_DATAERR, "'%s' does not exist on the Toppy", oldpath);
    310  1.1     scw 
    311  1.1     scw 	if (ud.ud_type != UTOPPY_DIRENT_FILE)
    312  1.1     scw 		errx(EX_DATAERR, "%s: not a regular file", oldpath);
    313  1.1     scw 
    314  1.1     scw 	if (find_toppy_dirent(newpath, &ud))
    315  1.1     scw 		errx(EX_DATAERR, "'%s' already exists", newpath);
    316  1.1     scw 
    317  1.1     scw 	ur.ur_old_path = o;
    318  1.1     scw 	ur.ur_new_path = n;
    319  1.1     scw 
    320  1.1     scw 	if (ioctl(toppy_fd, UTOPPYIORENAME, &ur) < 0)
    321  1.1     scw 		err(EX_OSERR, "ioctl(UTOPPYIORENAME, %s, %s)", oldpath,
    322  1.1     scw 		    newpath);
    323  1.1     scw }
    324  1.1     scw 
    325  1.1     scw 
    326  1.1     scw static void
    327  1.1     scw init_progress(FILE *to, char *f, off_t fsize, off_t restart)
    328  1.1     scw {
    329  1.1     scw 	struct ttysize ts;
    330  1.1     scw 
    331  1.1     scw 	if (ioctl(fileno(to), TIOCGSIZE, &ts) == -1)
    332  1.1     scw 		ttywidth = 80;
    333  1.1     scw 	else
    334  1.1     scw 		ttywidth = ts.ts_cols;
    335  1.1     scw 
    336  1.1     scw 	ttyout = to;
    337  1.1     scw 	progress = 1;
    338  1.1     scw 	bytes = 0;
    339  1.1     scw 	filesize = fsize;
    340  1.1     scw 	restart_point = restart;
    341  1.1     scw 	prefix = f;
    342  1.1     scw }
    343  1.1     scw 
    344  1.1     scw static void
    345  1.1     scw cmd_get(int argc, char **argv)
    346  1.1     scw {
    347  1.1     scw 	struct utoppy_readfile ur;
    348  1.1     scw 	struct utoppy_dirent ud;
    349  1.1     scw 	struct stat st;
    350  1.1     scw 	char *dst, dstbuf[FILENAME_MAX];
    351  1.1     scw 	uint8_t *buf;
    352  1.1     scw 	ssize_t l;
    353  1.1     scw 	size_t rv;
    354  1.1     scw 	int ch, turbo_mode = 0, reget = 0, progbar = 0;
    355  1.1     scw 	FILE *ofp, *to;
    356  1.1     scw 
    357  1.1     scw 	optind = 1;
    358  1.1     scw 	optreset = 1;
    359  1.1     scw 
    360  1.1     scw 	while ((ch = getopt(argc, argv, "prt")) != -1) {
    361  1.1     scw 		switch (ch) {
    362  1.1     scw 		case 'p':
    363  1.1     scw 			progbar = 1;
    364  1.1     scw 			break;
    365  1.1     scw 		case 'r':
    366  1.1     scw 			reget = 1;
    367  1.1     scw 			break;
    368  1.1     scw 		case 't':
    369  1.1     scw 			turbo_mode = 1;
    370  1.1     scw 			break;
    371  1.1     scw 		default:
    372  1.1     scw  get_usage:
    373  1.1     scw 			errx(EX_USAGE, "usage: get [-prt] <toppy-pathname> "
    374  1.1     scw 			    "[file | directory]");
    375  1.1     scw 		}
    376  1.1     scw 	}
    377  1.1     scw 	argc -= optind;
    378  1.1     scw 	argv += optind;
    379  1.1     scw 
    380  1.1     scw 	if (argc == 1)
    381  1.1     scw 		dst = basename(argv[0]);
    382  1.1     scw 	else
    383  1.1     scw 	if (argc == 2) {
    384  1.1     scw 		dst = argv[1];
    385  1.1     scw 		if (stat(dst, &st) == 0 && S_ISDIR(st.st_mode)) {
    386  1.1     scw 			snprintf(dstbuf, sizeof(dstbuf), "%s/%s", dst,
    387  1.1     scw 			    basename(argv[0]));
    388  1.1     scw 			dst = dstbuf;
    389  1.1     scw 		}
    390  1.1     scw 	} else
    391  1.1     scw 		goto get_usage;
    392  1.1     scw 
    393  1.1     scw 	ur.ur_path = argv[0];
    394  1.1     scw 	ur.ur_offset = 0;
    395  1.1     scw 
    396  1.1     scw 	if ((buf = malloc(TOPPY_IO_SIZE)) == NULL)
    397  1.1     scw 		err(EX_OSERR, "malloc(TOPPY_IO_SIZE)");
    398  1.1     scw 
    399  1.1     scw 	if (strcmp(dst, "-") == 0) {
    400  1.1     scw 		ofp = stdout;
    401  1.1     scw 		to = stderr;
    402  1.1     scw 		if (reget)
    403  1.1     scw 			warnx("Ignoring -r option in combination with stdout");
    404  1.1     scw 	} else {
    405  1.1     scw 		to = stdout;
    406  1.1     scw 
    407  1.1     scw 		if (reget) {
    408  1.1     scw 			if (stat(dst, &st) < 0) {
    409  1.1     scw 				if (errno != ENOENT)
    410  1.1     scw 					err(EX_OSERR, "stat(%s)", dst);
    411  1.1     scw 			} else
    412  1.1     scw 			if (!S_ISREG(st.st_mode))
    413  1.1     scw 				errx(EX_DATAERR, "-r only works with regular "
    414  1.1     scw 				    "files");
    415  1.1     scw 			else
    416  1.1     scw 				ur.ur_offset = st.st_size;
    417  1.1     scw 		}
    418  1.1     scw 
    419  1.1     scw 		if ((ofp = fopen(dst, reget ? "a" : "w")) == NULL)
    420  1.1     scw 			err(EX_OSERR, "fopen(%s)", dst);
    421  1.1     scw 	}
    422  1.1     scw 
    423  1.1     scw 	if (progbar) {
    424  1.1     scw 		if (find_toppy_dirent(ur.ur_path, &ud) == 0)
    425  1.1     scw 			ud.ud_size = 0;
    426  1.1     scw 		init_progress(to, dst, ud.ud_size, ur.ur_offset);
    427  1.1     scw 	}
    428  1.1     scw 
    429  1.1     scw 	if (ioctl(toppy_fd, UTOPPYIOTURBO, &turbo_mode) < 0)
    430  1.1     scw 		err(EX_OSERR, "ioctl(UTOPPYIOTURBO, %d)", turbo_mode);
    431  1.1     scw 
    432  1.1     scw 	if (ioctl(toppy_fd, UTOPPYIOREADFILE, &ur) < 0)
    433  1.1     scw 		err(EX_OSERR, "ioctl(UTOPPYIOREADFILE, %s)", ur.ur_path);
    434  1.1     scw 
    435  1.1     scw 	if (progbar)
    436  1.1     scw 		progressmeter(-1);
    437  1.1     scw 
    438  1.1     scw 	for (;;) {
    439  1.1     scw 		while ((l = read(toppy_fd, buf, TOPPY_IO_SIZE)) < 0 &&
    440  1.1     scw 		    errno == EINTR)
    441  1.1     scw 			;
    442  1.1     scw 
    443  1.1     scw 		if (l <= 0)
    444  1.1     scw 			break;
    445  1.1     scw 
    446  1.1     scw 		rv = fwrite(buf, 1, l, ofp);
    447  1.1     scw 
    448  1.1     scw 		if (rv != l) {
    449  1.1     scw 			if (ofp != stdout)
    450  1.1     scw 				fclose(ofp);
    451  1.1     scw 			progressmeter(1);
    452  1.1     scw 			err(EX_OSERR, "fwrite(%s)", dst);
    453  1.1     scw 		}
    454  1.1     scw 		bytes += l;
    455  1.1     scw 	}
    456  1.1     scw 
    457  1.1     scw 	if (progbar)
    458  1.1     scw 		progressmeter(1);
    459  1.1     scw 
    460  1.1     scw 	if (ofp != stdout)
    461  1.1     scw 		fclose(ofp);
    462  1.1     scw 
    463  1.1     scw 	if (l < 0)
    464  1.1     scw 		err(EX_OSERR, "read(TOPPY: ur.ur_path)");
    465  1.1     scw 
    466  1.1     scw 	free(buf);
    467  1.1     scw }
    468  1.1     scw 
    469  1.1     scw static void
    470  1.1     scw cmd_put(int argc, char **argv)
    471  1.1     scw {
    472  1.1     scw 	struct utoppy_writefile uw;
    473  1.1     scw 	struct utoppy_dirent ud;
    474  1.1     scw 	struct stat st;
    475  1.1     scw 	char dstbuf[FILENAME_MAX];
    476  1.1     scw 	char *src;
    477  1.1     scw 	void *buf;
    478  1.1     scw 	ssize_t rv;
    479  1.1     scw 	size_t l;
    480  1.1     scw 	int ch, turbo_mode = 0, reput = 0, progbar = 0;
    481  1.1     scw 	FILE *ifp;
    482  1.1     scw 
    483  1.1     scw 	optind = 1;
    484  1.1     scw 	optreset = 1;
    485  1.1     scw 
    486  1.1     scw 	while ((ch = getopt(argc, argv, "prt")) != -1) {
    487  1.1     scw 		switch (ch) {
    488  1.1     scw 		case 'p':
    489  1.1     scw 			progbar = 1;
    490  1.1     scw 			break;
    491  1.1     scw 		case 'r':
    492  1.1     scw 			reput = 1;
    493  1.1     scw 			break;
    494  1.1     scw 		case 't':
    495  1.1     scw 			turbo_mode = 1;
    496  1.1     scw 			break;
    497  1.1     scw 		default:
    498  1.1     scw  put_usage:
    499  1.1     scw 			errx(EX_USAGE, "usage: put [-prt] <local-pathname> "
    500  1.1     scw 			    "<toppy-pathname>");
    501  1.1     scw 		}
    502  1.1     scw 	}
    503  1.1     scw 	argc -= optind;
    504  1.1     scw 	argv += optind;
    505  1.1     scw 
    506  1.1     scw 	if (argc != 2)
    507  1.1     scw 		goto put_usage;
    508  1.1     scw 
    509  1.1     scw 	src = argv[0];
    510  1.1     scw 	uw.uw_path = argv[1];
    511  1.1     scw 
    512  1.1     scw 	if (stat(src, &st) < 0)
    513  1.1     scw 		err(EX_OSERR, "%s", src);
    514  1.1     scw 
    515  1.1     scw 	if (!S_ISREG(st.st_mode))
    516  1.1     scw 		errx(EX_DATAERR, "'%s' is not a regular file", src);
    517  1.1     scw 
    518  1.1     scw 	uw.uw_size = st.st_size;
    519  1.1     scw 	uw.uw_mtime = st.st_mtime;
    520  1.1     scw 	uw.uw_offset = 0;
    521  1.1     scw 
    522  1.1     scw 	if (find_toppy_dirent(uw.uw_path, &ud)) {
    523  1.1     scw 		if (ud.ud_type == UTOPPY_DIRENT_DIRECTORY) {
    524  1.1     scw 			snprintf(dstbuf, sizeof(dstbuf), "%s/%s", uw.uw_path,
    525  1.1     scw 			    basename(src));
    526  1.1     scw 			uw.uw_path = dstbuf;
    527  1.1     scw 		} else
    528  1.1     scw 		if (ud.ud_type != UTOPPY_DIRENT_FILE)
    529  1.1     scw 			errx(EX_DATAERR, "'%s' is not a regular file.",
    530  1.1     scw 			    uw.uw_path);
    531  1.1     scw 		else
    532  1.1     scw 		if (reput) {
    533  1.1     scw 			if (ud.ud_size > uw.uw_size)
    534  1.1     scw 				errx(EX_DATAERR, "'%s' is already larger than "
    535  1.1     scw 				    "'%s'", uw.uw_path, src);
    536  1.1     scw 
    537  1.1     scw 			uw.uw_size -= ud.ud_size;
    538  1.1     scw 			uw.uw_offset = ud.ud_size;
    539  1.1     scw 		}
    540  1.1     scw 	}
    541  1.1     scw 
    542  1.1     scw 	if ((buf = malloc(TOPPY_IO_SIZE)) == NULL)
    543  1.1     scw 		err(EX_OSERR, "malloc(TOPPY_IO_SIZE)");
    544  1.1     scw 
    545  1.1     scw 	if ((ifp = fopen(src, "r")) == NULL)
    546  1.1     scw 		err(EX_OSERR, "fopen(%s)", src);
    547  1.1     scw 
    548  1.1     scw 	if (ioctl(toppy_fd, UTOPPYIOTURBO, &turbo_mode) < 0)
    549  1.1     scw 		err(EX_OSERR, "ioctl(UTOPPYIOTURBO, %d)", turbo_mode);
    550  1.1     scw 
    551  1.1     scw 	if (ioctl(toppy_fd, UTOPPYIOWRITEFILE, &uw) < 0)
    552  1.1     scw 		err(EX_OSERR, "ioctl(UTOPPYIOWRITEFILE, %s)", uw.uw_path);
    553  1.1     scw 
    554  1.1     scw 	if (progbar)
    555  1.1     scw 		init_progress(stdout, src, st.st_size, uw.uw_offset);
    556  1.1     scw 
    557  1.1     scw 	if (progbar)
    558  1.1     scw 		progressmeter(-1);
    559  1.1     scw 
    560  1.1     scw 	while ((l = fread(buf, 1, TOPPY_IO_SIZE, ifp)) > 0) {
    561  1.1     scw 		rv = write(toppy_fd, buf, l);
    562  1.1     scw 		if (rv != l) {
    563  1.1     scw 			fclose(ifp);
    564  1.1     scw 			if (progbar)
    565  1.1     scw 				progressmeter(1);
    566  1.1     scw 			err(EX_OSERR, "write(TOPPY: %s)", uw.uw_path);
    567  1.1     scw 		}
    568  1.1     scw 		bytes += l;
    569  1.1     scw 	}
    570  1.1     scw 
    571  1.1     scw 	if (progbar)
    572  1.1     scw 		progressmeter(1);
    573  1.1     scw 
    574  1.1     scw 	if (ferror(ifp))
    575  1.1     scw 		err(EX_OSERR, "fread(%s)", src);
    576  1.1     scw 
    577  1.1     scw 	fclose(ifp);
    578  1.1     scw 	free(buf);
    579  1.1     scw }
    580