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