240 lines
5.3 KiB
C
240 lines
5.3 KiB
C
|
|
/*
|
|
* 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(c->socket.task, &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_debug(c->socket.task, "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_debug(c->socket.task, "sendfile(): %z", 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(c->socket.task, nxt_socket_error_level(err),
|
|
"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_debug(c->socket.task, "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_debug(c->socket.task, "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(c->socket.task, nxt_socket_error_level(err),
|
|
"send(%d, %p, %uz, 0x%uXi) failed %E",
|
|
c->socket.fd, buf, size, flags, err);
|
|
|
|
return NXT_ERROR;
|
|
}
|
|
|
|
nxt_debug(c->socket.task, "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_debug(c->socket.task, "sendmsg(%d, %ui, 0x%uXi): %z",
|
|
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(c->socket.task, nxt_socket_error_level(err),
|
|
"sendmsg(%d, %ui, 0x%uXi) failed %E",
|
|
c->socket.fd, niov, flags, err);
|
|
|
|
return NXT_ERROR;
|
|
}
|
|
|
|
nxt_debug(c->socket.task, "sendmsg() %E", err);
|
|
|
|
return 0;
|
|
}
|
|
|
|
if (n < (ssize_t) sb->size) {
|
|
c->socket.write_ready = 0;
|
|
}
|
|
|
|
return n;
|
|
}
|