1 /****************************************************************************
2 * fs/vfs/fs_sendfile.c
3 *
4 * Copyright (c) 2023 Huawei Device Co., Ltd. All rights reserved.
5 * Based on NuttX originally from nuttx source (nuttx/fs/ and nuttx/drivers/)
6 *
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
10 *
11 * http://www.apache.org/licenses/LICENSE-2.0
12 *
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
18 *
19 ****************************************************************************/
20
21 /****************************************************************************
22 * Included Files
23 ****************************************************************************/
24
25 #include "vfs_config.h"
26
27 #include <fs/file.h>
28 #include <stdbool.h>
29 #include <stdlib.h>
30 #include <unistd.h>
31 #include <errno.h>
32
33
34 #ifndef CONFIG_LIB_SENDFILE_BUFSIZE
35 # define CONFIG_LIB_SENDFILE_BUFSIZE 512
36 #endif
37
38 /****************************************************************************
39 * Public Functions
40 ****************************************************************************/
41
42 /****************************************************************************
43 * Name: sendfile
44 *
45 * Description:
46 * sendfile() copies data between one file descriptor and another.
47 * Used with file descriptors it basically just wraps a sequence of
48 * reads() and writes() to perform a copy.
49 *
50 * If the destination descriptor is a socket, it gives a better
51 * performance than simple reds() and writes(). The data is read directly
52 * into the net buffer and the whole tcp window is filled if possible.
53 *
54 * NOTE: This interface is *not* specified in POSIX.1-2001, or other
55 * standards. The implementation here is very similar to the Linux
56 * sendfile interface. Other UNIX systems implement sendfile() with
57 * different semantics and prototypes. sendfile() should not be used
58 * in portable programs.
59 *
60 * Input Parameters:
61 * infd - A file (or socket) descriptor opened for reading
62 * outfd - A descriptor opened for writing.
63 * offset - If 'offset' is not NULL, then it points to a variable
64 * holding the file offset from which sendfile() will start
65 * reading data from 'infd'. When sendfile() returns, this
66 * variable will be set to the offset of the byte following
67 * the last byte that was read. If 'offset' is not NULL,
68 * then sendfile() does not modify the current file offset of
69 * 'infd'; otherwise the current file offset is adjusted to
70 * reflect the number of bytes read from 'infd.'
71 *
72 * If 'offset' is NULL, then data will be read from 'infd'
73 * starting at the current file offset, and the file offset
74 * will be updated by the call.
75 * count - The number of bytes to copy between the file descriptors.
76 *
77 * Returned Value:
78 * If the transfer was successful, the number of bytes written to outfd is
79 * returned. On error, -1 is returned, and errno is set appropriately.
80 * There error values are those returned by read() or write() plus:
81 *
82 * EINVAL - Bad input parameters.
83 * ENOMEM - Could not allocated an I/O buffer
84 *
85 ****************************************************************************/
86
sendfile(int outfd,int infd,off_t * offset,size_t count)87 ssize_t sendfile(int outfd, int infd, off_t *offset, size_t count)
88 {
89 uint8_t *iobuffer;
90 uint8_t *wrbuffer;
91 off_t startpos = 0;
92 ssize_t nbytesread;
93 ssize_t nbyteswritten;
94 size_t ntransferred;
95 bool endxfr;
96
97 /* Get the current file position. */
98
99 if (offset)
100 {
101 /* Use lseek to get the current file position */
102
103 startpos = lseek(infd, 0, SEEK_CUR);
104 if (startpos == (off_t)-1)
105 {
106 return VFS_ERROR;
107 }
108
109 /* Use lseek again to set the new file position */
110
111 if (lseek(infd, *offset, SEEK_SET) == (off_t)-1)
112 {
113 return VFS_ERROR;
114 }
115 }
116
117 /* Allocate an I/O buffer */
118
119 iobuffer = (void *)malloc(CONFIG_LIB_SENDFILE_BUFSIZE);
120 if (!iobuffer)
121 {
122 set_errno(ENOMEM);
123 return VFS_ERROR;
124 }
125
126 /* Now transfer 'count' bytes from the infd to the outfd */
127
128 for (ntransferred = 0, endxfr = false; ntransferred < count && !endxfr; )
129 {
130 /* Loop until the read side of the transfer comes to some conclusion */
131
132 do
133 {
134 /* Read a buffer of data from the infd */
135
136 nbytesread = read(infd, iobuffer, CONFIG_LIB_SENDFILE_BUFSIZE);
137
138 /* Check for end of file */
139
140 if (nbytesread == 0)
141 {
142 /* End of file. Break out and return current number of bytes
143 * transferred.
144 */
145
146 endxfr = true;
147 break;
148 }
149
150 /* Check for a read ERROR. EINTR is a special case. This function
151 * should break out and return an error if EINTR is returned and
152 * no data has been transferred. But what should it do if some
153 * data has been transferred? I suppose just continue?
154 */
155
156 else if (nbytesread < 0)
157 {
158 int errcode = get_errno();
159
160 /* EINTR is not an error (but will still stop the copy) */
161
162 if (errcode != EINTR || ntransferred == 0)
163 {
164 /* Read error. Break out and return the error condition. */
165
166 set_errno(errcode);
167 ntransferred = VFS_ERROR;
168 endxfr = true;
169 break;
170 }
171 }
172 }
173 while (nbytesread < 0);
174
175 /* Was anything read? */
176
177 if (!endxfr)
178 {
179 /* Yes.. Loop until the read side of the transfer comes to some
180 * conclusion.
181 */
182
183 wrbuffer = iobuffer;
184 do
185 {
186 /* Write the buffer of data to the outfd */
187
188 nbyteswritten = write(outfd, wrbuffer, nbytesread);
189
190 /* Check for a complete (or parial) write. write() should not
191 * return zero.
192 */
193
194 if (nbyteswritten >= 0)
195 {
196 /* Advance the buffer pointer and decrement the number of bytes
197 * remaining in the iobuffer. Typically, nbytesread will now
198 * be zero.
199 */
200
201 wrbuffer += nbyteswritten;
202 nbytesread -= nbyteswritten;
203
204 /* Increment the total number of bytes successfully transferred. */
205
206 ntransferred += nbyteswritten;
207 }
208
209 /* Otherwise an error occurred */
210
211 else
212 {
213 int errcode = get_errno();
214
215 /* Check for a read ERROR. EINTR is a special case. This
216 * function should break out and return an error if EINTR
217 * is returned and no data has been transferred. But what
218 * should it do if some data has been transferred? I
219 * suppose just continue?
220 */
221
222 if (errcode != EINTR || ntransferred == 0)
223 {
224 /* Write error. Break out and return the error
225 * condition.
226 */
227
228 set_errno(errcode);
229 ntransferred = VFS_ERROR;
230 endxfr = true;
231 break;
232 }
233 }
234 }
235 while (nbytesread > 0);
236 }
237 }
238
239 /* Release the I/O buffer */
240
241 free(iobuffer);
242
243 /* Return the current file position */
244
245 if (offset)
246 {
247 /* Use lseek to get the current file position */
248
249 off_t curpos = lseek(infd, 0, SEEK_CUR);
250 if (curpos == (off_t)-1)
251 {
252 return VFS_ERROR;
253 }
254
255 /* Return the current file position */
256
257 *offset = curpos;
258
259 /* Use lseek again to restore the original file position */
260
261 if (lseek(infd, startpos, SEEK_SET) == (off_t)-1)
262 {
263 return VFS_ERROR;
264 }
265 }
266
267 /* Finally return the number of bytes actually transferred (or VFS_ERROR
268 * if any failure occurred).
269 */
270
271 return ntransferred;
272 }
273
274