handlers can read response before sending req body (fixes #131, #2566)
[lighttpd.git] / src / network_darwin_sendfile.c
blob061bbd96b831800d8ab0d1f567b563df313f982e
1 #include "first.h"
3 #include "network_backends.h"
5 #if defined(USE_DARWIN_SENDFILE)
7 #include "network.h"
8 #include "log.h"
10 #include <sys/types.h>
11 #include <sys/socket.h>
12 #include <sys/uio.h>
14 #include <errno.h>
15 #include <string.h>
17 int network_write_file_chunk_sendfile(server *srv, connection *con, int fd, chunkqueue *cq, off_t *p_max_bytes) {
18 chunk* const c = cq->first;
19 off_t offset, written = 0;
20 off_t toSend;
21 int r;
23 force_assert(NULL != c);
24 force_assert(FILE_CHUNK == c->type);
25 force_assert(c->offset >= 0 && c->offset <= c->file.length);
27 offset = c->file.start + c->offset;
28 toSend = c->file.length - c->offset;
29 if (toSend > *p_max_bytes) toSend = *p_max_bytes;
31 if (0 == toSend) {
32 chunkqueue_remove_finished_chunks(cq);
33 return 0;
36 if (0 != network_open_file_chunk(srv, con, cq)) return -1;
38 /* Darwin sendfile() */
39 written = toSend;
40 if (-1 == (r = sendfile(c->file.fd, fd, offset, &written, NULL, 0))) {
41 switch(errno) {
42 case EAGAIN:
43 case EINTR:
44 /* for EAGAIN/EINTR written still contains the sent bytes */
45 break; /* try again later */
46 case EPIPE:
47 case ENOTCONN:
48 return -2;
49 default:
50 log_error_write(srv, __FILE__, __LINE__, "ssd", "sendfile: ", strerror(errno), errno);
51 return -1;
55 if (written >= 0) {
56 chunkqueue_mark_written(cq, written);
57 *p_max_bytes -= written;
60 return (r >= 0 && written == toSend) ? 0 : -3;
63 #endif /* USE_DARWIN_SENDFILE */