Initial version.
This commit is contained in:
240
src/nxt_linux_sendfile.c
Normal file
240
src/nxt_linux_sendfile.c
Normal file
@@ -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;
|
||||
}
|
||||
Reference in New Issue
Block a user