diff options
Diffstat (limited to '')
-rw-r--r-- | src/nxt_linux_sendfile.c | 240 |
1 files changed, 240 insertions, 0 deletions
diff --git a/src/nxt_linux_sendfile.c b/src/nxt_linux_sendfile.c new file mode 100644 index 00000000..9c8f563c --- /dev/null +++ b/src/nxt_linux_sendfile.c @@ -0,0 +1,240 @@ + +/* + * Copyright (C) Igor Sysoev + * Copyright (C) NGINX, Inc. + */ + +#include <nxt_main.h> + + +/* + * sendfile() has been introduced in Linux 2.2. + * It supported 32-bit offsets only. + * + * Linux 2.4.21 has introduced sendfile64(). However, even on 64-bit + * platforms it returns EINVAL if the count argument is more than 2G-1 bytes. + * In Linux 2.6.17 sendfile() has been internally changed to splice() + * and this limitation has gone. + */ + +#ifdef NXT_TEST_BUILD_LINUX_SENDFILE + +#define MSG_NOSIGNAL 0x4000 +#define MSG_MORE 0x8000 + +ssize_t nxt_linux_event_conn_io_sendfile(nxt_event_conn_t *c, nxt_buf_t *b, + size_t limit); + +static ssize_t nxt_sys_sendfile(int out_fd, int in_fd, off_t *offset, + size_t count) +{ + return -1; +} + +#else +#define nxt_sys_sendfile sendfile +#endif + + +static ssize_t nxt_linux_send(nxt_event_conn_t *c, void *buf, size_t size, + nxt_uint_t flags); +static ssize_t nxt_linux_sendmsg(nxt_event_conn_t *c, + nxt_sendbuf_coalesce_t *sb, nxt_uint_t niov, nxt_uint_t flags); + + +ssize_t +nxt_linux_event_conn_io_sendfile(nxt_event_conn_t *c, nxt_buf_t *b, + size_t limit) +{ + size_t size; + ssize_t n; + nxt_buf_t *fb; + nxt_err_t err; + nxt_off_t offset; + nxt_uint_t niov, flags; + struct iovec iov[NXT_IOBUF_MAX]; + nxt_sendbuf_coalesce_t sb; + + sb.buf = b; + sb.iobuf = iov; + sb.nmax = NXT_IOBUF_MAX; + sb.sync = 0; + sb.size = 0; + sb.limit = limit; + + niov = nxt_sendbuf_mem_coalesce(&sb); + + if (niov == 0 && sb.sync) { + return 0; + } + + fb = (sb.buf != NULL && nxt_buf_is_file(sb.buf)) ? sb.buf : NULL; + + if (niov != 0) { + + flags = MSG_NOSIGNAL; + + if (fb != NULL) { + /* + * The Linux-specific MSG_MORE flag is cheaper + * than additional setsockopt(TCP_CORK) syscall. + */ + flags |= MSG_MORE; + } + + if (niov == 1) { + /* + * Disposal of surplus kernel msghdr + * and iovec copy-in operations. + */ + return nxt_linux_send(c, iov->iov_base, iov->iov_len, flags); + } + + return nxt_linux_sendmsg(c, &sb, niov, flags); + } + + size = nxt_sendbuf_file_coalesce(&sb); + + nxt_log_debug(c->socket.log, "sendfile(%d, %FD, @%O, %uz)", + c->socket.fd, fb->file->fd, fb->file_pos, size); + + offset = fb->file_pos; + + n = nxt_sys_sendfile(c->socket.fd, fb->file->fd, &offset, size); + + err = (n == -1) ? nxt_errno : 0; + + nxt_log_debug(c->socket.log, "sendfile(): %d", n); + + if (n == -1) { + switch (err) { + + case NXT_EAGAIN: + c->socket.write_ready = 0; + break; + + case NXT_EINTR: + break; + + default: + c->socket.error = err; + nxt_log_error(nxt_socket_error_level(err, c->socket.log_error), + c->socket.log, + "sendfile(%d, %FD, %O, %uz) failed %E \"%FN\"", + c->socket.fd, fb->file->fd, fb->file_pos, size, + err, fb->file->name); + + return NXT_ERROR; + } + + nxt_log_debug(c->socket.log, "sendfile() %E", err); + + return 0; + } + + if (n < (ssize_t) size) { + c->socket.write_ready = 0; + } + + return n; +} + + +static ssize_t +nxt_linux_send(nxt_event_conn_t *c, void *buf, size_t size, nxt_uint_t flags) +{ + ssize_t n; + nxt_err_t err; + + n = send(c->socket.fd, buf, size, flags); + + err = (n == -1) ? nxt_errno : 0; + + nxt_log_debug(c->socket.log, "send(%d, %p, %uz, 0x%uXi): %z", + c->socket.fd, buf, size, flags, n); + + if (n == -1) { + switch (err) { + + case NXT_EAGAIN: + c->socket.write_ready = 0; + break; + + case NXT_EINTR: + break; + + default: + c->socket.error = err; + nxt_log_error(nxt_socket_error_level(err, c->socket.log_error), + c->socket.log, "send(%d, %p, %uz, 0x%uXi) failed %E", + c->socket.fd, buf, size, flags, err); + + return NXT_ERROR; + } + + nxt_log_debug(c->socket.log, "send() %E", err); + + return 0; + } + + if (n < (ssize_t) size) { + c->socket.write_ready = 0; + } + + return n; +} + + +static ssize_t +nxt_linux_sendmsg(nxt_event_conn_t *c, nxt_sendbuf_coalesce_t *sb, + nxt_uint_t niov, nxt_uint_t flags) +{ + ssize_t n; + nxt_err_t err; + struct msghdr msg; + + msg.msg_name = NULL; + msg.msg_namelen = 0; + msg.msg_iov = sb->iobuf; + msg.msg_iovlen = niov; + msg.msg_control = NULL; + msg.msg_controllen = 0; + msg.msg_flags = 0; + + n = sendmsg(c->socket.fd, &msg, flags); + + err = (n == -1) ? nxt_errno : 0; + + nxt_log_debug(c->socket.log, "sendmsg(%d, %ui, 0x%uXi): %d", + c->socket.fd, niov, flags, n); + + if (n == -1) { + switch (err) { + + case NXT_EAGAIN: + c->socket.write_ready = 0; + break; + + case NXT_EINTR: + break; + + default: + c->socket.error = err; + nxt_log_error(nxt_socket_error_level(err, c->socket.log_error), + c->socket.log, "sendmsg(%d, %ui, 0x%uXi) failed %E", + c->socket.fd, niov, flags, err); + + return NXT_ERROR; + } + + nxt_log_debug(c->socket.log, "sendmsg() %E", err); + + return 0; + } + + if (n < (ssize_t) sb->size) { + c->socket.write_ready = 0; + } + + return n; +} |