chiark / gitweb /
8e61b5d882d6ac93fe52be18fa6ddf2c490771c9
[dpkg] / lib / dpkg / fdio.c
1 /*
2  * libdpkg - Debian packaging suite library routines
3  * fdio.c - safe file descriptor based input/output
4  *
5  * Copyright © 2009-2010 Guillem Jover <guillem@debian.org>
6  *
7  * This is free software; you can redistribute it and/or modify
8  * it under the terms of the GNU General Public License as published by
9  * the Free Software Foundation; either version 2 of the License, or
10  * (at your option) any later version.
11  *
12  * This is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15  * GNU General Public License for more details.
16  *
17  * You should have received a copy of the GNU General Public License
18  * along with this program.  If not, see <https://www.gnu.org/licenses/>.
19  */
20
21 #include <config.h>
22 #include <compat.h>
23
24 #include <errno.h>
25 #include <fcntl.h>
26 #include <unistd.h>
27
28 #include <dpkg/fdio.h>
29
30 ssize_t
31 fd_read(int fd, void *buf, size_t len)
32 {
33         ssize_t total = 0;
34         char *ptr = buf;
35
36         while (len > 0) {
37                 ssize_t n;
38
39                 n = read(fd, ptr + total, len);
40                 if (n == -1) {
41                         if (errno == EINTR || errno == EAGAIN)
42                                 continue;
43                         return total ? -total : n;
44                 }
45                 if (n == 0)
46                         break;
47
48                 total += n;
49                 len -= n;
50         }
51
52         return total;
53 }
54
55 ssize_t
56 fd_write(int fd, const void *buf, size_t len)
57 {
58         ssize_t total = 0;
59         const char *ptr = buf;
60
61         while (len > 0) {
62                 ssize_t n;
63
64                 n = write(fd, ptr + total, len);
65                 if (n == -1) {
66                         if (errno == EINTR || errno == EAGAIN)
67                                 continue;
68                         return total ? -total : n;
69                 }
70                 if (n == 0)
71                         break;
72
73                 total += n;
74                 len -= n;
75         }
76
77         return total;
78 }
79
80 #ifdef USE_DISK_PREALLOCATE
81 #ifdef HAVE_F_PREALLOCATE
82 static void
83 fd_preallocate_setup(fstore_t *fs, int flags, off_t offset, off_t len)
84 {
85         fs->fst_flags = flags;
86         fs->fst_posmode = F_PEOFPOSMODE;
87         fs->fst_offset = offset;
88         fs->fst_length = len;
89         fs->fst_bytesalloc = 0;
90 }
91 #endif
92
93 /**
94  * Request the kernel to allocate the specified size for a file descriptor.
95  *
96  * We only want to send a hint that we will be using the requested size. But
97  * we do not want to unnecessarily write the file contents. That is why we
98  * are not using posix_fallocate(3) directly if possible, and not at all
99  * on glibc based systems (except on GNU/kFreeBSD).
100  */
101 int
102 fd_allocate_size(int fd, off_t offset, off_t len)
103 {
104         int rc;
105
106         /* Do not preallocate on very small files as that degrades performance
107          * on some filesystems. */
108         if (len < (4 * 4096) - 1)
109                 return 0;
110
111 #if defined(HAVE_F_PREALLOCATE)
112         /* On Mac OS X. */
113         fstore_t fs;
114
115         fd_preallocate_setup(&fs, F_ALLOCATECONTIG, offset, len);
116         rc = fcntl(fd, F_PREALLOCATE, &fs);
117         if (rc < 0 && errno == ENOSPC) {
118                 /* If we cannot get a contiguous allocation, then try
119                  * non-contiguous. */
120                 fd_preallocate_setup(&fs, F_ALLOCATEALL, offset, len);
121                 rc = fcntl(fd, F_PREALLOCATE, &fs);
122         }
123 #elif defined(HAVE_F_ALLOCSP64)
124         /* On Solaris. */
125         struct flock64 fl;
126
127         fl.l_whence = SEEK_SET;
128         fl.l_start = offset;
129         fl.l_len = len;
130
131         rc = fcntl(fd, F_ALLOCSP64, &fl);
132 #elif defined(HAVE_FALLOCATE)
133         /* On Linux. */
134         do {
135                 rc = fallocate(fd, 0, offset, len);
136         } while (rc < 0 && errno == EINTR);
137 #elif defined(HAVE_POSIX_FALLOCATE) && \
138       ((defined(__GLIBC__) && defined(__FreeBSD_kernel__)) || \
139        !defined(__GLIBC__))
140         /*
141          * On BSDs, newer GNU/kFreeBSD and other non-glibc based systems
142          * we can use posix_fallocate(2) which should be a simple syscall
143          * wrapper. But not on other glibc systems, as there the function
144          * will try to allocate the size by writing a '\0' to each block
145          * if the syscall is not implemented or not supported by the
146          * kernel or the filesystem, which we do not want.
147          */
148         rc = posix_fallocate(fd, offset, len);
149 #else
150         errno = ENOSYS;
151         rc = -1;
152 #endif
153
154         return rc;
155 }
156 #else
157 int
158 fd_allocate_size(int fd, off_t offset, off_t len)
159 {
160         errno = ENOSYS;
161         return -1;
162 }
163 #endif