Fix bug 424: doing full_read breaks things like cat which should return a
[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
33         while (!size || total < size)
34         {
35                 ssize_t wrote, xread = (size && size < BUFSIZ) ? size : BUFSIZ;
36                 xread = safe_read(src_fd, buffer, xread);
37                 if (xread > 0) {
38                         /* A -1 dst_fd means we need to fake it... */
39                         wrote = (dst_fd < 0) ? xread : bb_full_write(dst_fd, buffer, xread);
40                         if (wrote < xread) {
41                                 bb_perror_msg(bb_msg_write_error);
42                                 break;
43                         }
44                         total += wrote;
45                         size -= wrote;
46                 } else if (xread < 0) {
47                         bb_perror_msg(bb_msg_read_error);
48                         break;
49                 } else if (xread == 0) {
50                         /* All done. */
51                         status = 0;
52                         break;
53                 }
54         }
55                 
56 out:
57         RELEASE_CONFIG_BUFFER(buffer);
58
59         return status ? status : total;
60 }
61
62
63 extern int bb_copyfd_size(int fd1, int fd2, const off_t size)
64 {
65         if (size) {
66                 return(bb_full_fd_action(fd1, fd2, size));
67         }
68         return(0);
69 }
70
71 extern int bb_copyfd_eof(int fd1, int fd2)
72 {
73         return(bb_full_fd_action(fd1, fd2, 0));
74 }