Bug fix from Kim B. Heino, manifested via:
[oweals/busybox.git] / libbb / copyfd.c
1 /* vi: set sw=4 ts=4: */
2 /*
3  * Utility routines.
4  *
5  * Copyright (C) 1999-2005 by Erik Andersen <andersen@codepoet.org>
6  *
7  * Licensed under GPLv2 or later, see file LICENSE in this tarball for details.
8  */
9
10 #include <errno.h>
11 #include <stdlib.h>
12 #include <string.h>
13 #include <unistd.h>
14
15 #include "busybox.h"
16 #include "libbb.h"
17
18
19 #if BUFSIZ < 4096
20 #undef BUFSIZ
21 #define BUFSIZ 4096
22 #endif
23
24
25 static ssize_t bb_full_fd_action(int src_fd, int dst_fd, size_t size)
26 {
27         int status = -1;
28         size_t total = 0;
29         RESERVE_CONFIG_BUFFER(buffer,BUFSIZ);
30
31         if (src_fd < 0) goto out;
32         while (!size || total < size)
33         {
34                 ssize_t wrote, xread;
35
36                 xread = safe_read(src_fd, buffer,
37                                 (!size || size - total > BUFSIZ) ? BUFSIZ : size - total);
38
39                 if (xread > 0) {
40                         /* A -1 dst_fd means we need to fake it... */
41                         wrote = (dst_fd < 0) ? xread : bb_full_write(dst_fd, buffer, xread);
42                         if (wrote < xread) {
43                                 bb_perror_msg(bb_msg_write_error);
44                                 break;
45                         }
46                         total += wrote;
47                         if (total == size) status = 0;
48                 } else if (xread < 0) {
49                         bb_perror_msg(bb_msg_read_error);
50                         break;
51                 } else if (xread == 0) {
52                         /* All done. */
53                         status = 0;
54                         break;
55                 }
56         }
57
58 out:
59         RELEASE_CONFIG_BUFFER(buffer);
60
61         return status ? status : (ssize_t)total;
62 }
63
64
65 extern int bb_copyfd_size(int fd1, int fd2, const off_t size)
66 {
67         if (size) {
68                 return(bb_full_fd_action(fd1, fd2, size));
69         }
70         return(0);
71 }
72
73 extern int bb_copyfd_eof(int fd1, int fd2)
74 {
75         return(bb_full_fd_action(fd1, fd2, 0));
76 }