annotate src/os/unix/ngx_udp_sendmsg_chain.c @ 6949:ff0c8e11edbc

Simplified and improved sendfile() code on Linux. The ngx_linux_sendfile() function is now used for both normal sendfile() and sendfile in threads. The ngx_linux_sendfile_thread() function was modified to use the same interface as ngx_linux_sendfile(), and is simply called from ngx_linux_sendfile() when threads are enabled. Special return code NGX_DONE is used to indicate that a thread task was posted and no further actions are needed. If number of bytes sent is less that what we were sending, we now always retry sending. This is needed for sendfile() in threads as the number of bytes we are sending might have been changed since the thread task was posted. And this is also needed for Linux 4.3+, as sendfile() might be interrupted at any time and provides no indication if it was interrupted or not (ticket #1174).
author Maxim Dounin <mdounin@mdounin.ru>
date Tue, 28 Mar 2017 18:15:39 +0300
parents 56fc55e32f23
children dbb0c854e308
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
6692
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
1
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
2 /*
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
3 * Copyright (C) Igor Sysoev
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
4 * Copyright (C) Nginx, Inc.
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
5 */
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
6
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
7
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
8 #include <ngx_config.h>
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
9 #include <ngx_core.h>
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
10 #include <ngx_event.h>
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
11
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
12
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
13 static ngx_chain_t *ngx_udp_output_chain_to_iovec(ngx_iovec_t *vec,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
14 ngx_chain_t *in, ngx_log_t *log);
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
15 static ssize_t ngx_sendmsg(ngx_connection_t *c, ngx_iovec_t *vec);
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
16
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
17
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
18 ngx_chain_t *
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
19 ngx_udp_unix_sendmsg_chain(ngx_connection_t *c, ngx_chain_t *in, off_t limit)
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
20 {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
21 ssize_t n;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
22 off_t send;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
23 ngx_chain_t *cl;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
24 ngx_event_t *wev;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
25 ngx_iovec_t vec;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
26 struct iovec iovs[NGX_IOVS_PREALLOCATE];
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
27
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
28 wev = c->write;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
29
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
30 if (!wev->ready) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
31 return in;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
32 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
33
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
34 #if (NGX_HAVE_KQUEUE)
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
35
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
36 if ((ngx_event_flags & NGX_USE_KQUEUE_EVENT) && wev->pending_eof) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
37 (void) ngx_connection_error(c, wev->kq_errno,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
38 "kevent() reported about an closed connection");
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
39 wev->error = 1;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
40 return NGX_CHAIN_ERROR;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
41 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
42
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
43 #endif
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
44
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
45 /* the maximum limit size is the maximum size_t value - the page size */
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
46
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
47 if (limit == 0 || limit > (off_t) (NGX_MAX_SIZE_T_VALUE - ngx_pagesize)) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
48 limit = NGX_MAX_SIZE_T_VALUE - ngx_pagesize;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
49 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
50
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
51 send = 0;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
52
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
53 vec.iovs = iovs;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
54 vec.nalloc = NGX_IOVS_PREALLOCATE;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
55
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
56 for ( ;; ) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
57
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
58 /* create the iovec and coalesce the neighbouring bufs */
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
59
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
60 cl = ngx_udp_output_chain_to_iovec(&vec, in, c->log);
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
61
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
62 if (cl == NGX_CHAIN_ERROR) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
63 return NGX_CHAIN_ERROR;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
64 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
65
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
66 if (cl && cl->buf->in_file) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
67 ngx_log_error(NGX_LOG_ALERT, c->log, 0,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
68 "file buf in sendmsg "
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
69 "t:%d r:%d f:%d %p %p-%p %p %O-%O",
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
70 cl->buf->temporary,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
71 cl->buf->recycled,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
72 cl->buf->in_file,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
73 cl->buf->start,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
74 cl->buf->pos,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
75 cl->buf->last,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
76 cl->buf->file,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
77 cl->buf->file_pos,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
78 cl->buf->file_last);
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
79
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
80 ngx_debug_point();
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
81
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
82 return NGX_CHAIN_ERROR;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
83 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
84
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
85 if (cl == in) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
86 return in;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
87 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
88
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
89 send += vec.size;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
90
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
91 n = ngx_sendmsg(c, &vec);
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
92
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
93 if (n == NGX_ERROR) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
94 return NGX_CHAIN_ERROR;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
95 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
96
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
97 if (n == NGX_AGAIN) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
98 wev->ready = 0;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
99 return in;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
100 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
101
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
102 c->sent += n;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
103
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
104 in = ngx_chain_update_sent(in, n);
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
105
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
106 if (send >= limit || in == NULL) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
107 return in;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
108 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
109 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
110 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
111
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
112
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
113 static ngx_chain_t *
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
114 ngx_udp_output_chain_to_iovec(ngx_iovec_t *vec, ngx_chain_t *in, ngx_log_t *log)
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
115 {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
116 size_t total, size;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
117 u_char *prev;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
118 ngx_uint_t n, flush;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
119 ngx_chain_t *cl;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
120 struct iovec *iov;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
121
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
122 cl = in;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
123 iov = NULL;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
124 prev = NULL;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
125 total = 0;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
126 n = 0;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
127 flush = 0;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
128
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
129 for ( /* void */ ; in && !flush; in = in->next) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
130
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
131 if (in->buf->flush || in->buf->last_buf) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
132 flush = 1;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
133 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
134
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
135 if (ngx_buf_special(in->buf)) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
136 continue;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
137 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
138
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
139 if (in->buf->in_file) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
140 break;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
141 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
142
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
143 if (!ngx_buf_in_memory(in->buf)) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
144 ngx_log_error(NGX_LOG_ALERT, log, 0,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
145 "bad buf in output chain "
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
146 "t:%d r:%d f:%d %p %p-%p %p %O-%O",
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
147 in->buf->temporary,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
148 in->buf->recycled,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
149 in->buf->in_file,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
150 in->buf->start,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
151 in->buf->pos,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
152 in->buf->last,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
153 in->buf->file,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
154 in->buf->file_pos,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
155 in->buf->file_last);
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
156
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
157 ngx_debug_point();
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
158
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
159 return NGX_CHAIN_ERROR;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
160 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
161
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
162 size = in->buf->last - in->buf->pos;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
163
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
164 if (prev == in->buf->pos) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
165 iov->iov_len += size;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
166
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
167 } else {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
168 if (n == vec->nalloc) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
169 ngx_log_error(NGX_LOG_ALERT, log, 0,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
170 "too many parts in a datagram");
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
171 return NGX_CHAIN_ERROR;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
172 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
173
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
174 iov = &vec->iovs[n++];
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
175
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
176 iov->iov_base = (void *) in->buf->pos;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
177 iov->iov_len = size;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
178 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
179
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
180 prev = in->buf->pos + size;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
181 total += size;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
182 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
183
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
184 if (!flush) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
185 #if (NGX_SUPPRESS_WARN)
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
186 vec->size = 0;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
187 vec->count = 0;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
188 #endif
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
189 return cl;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
190 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
191
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
192 vec->count = n;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
193 vec->size = total;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
194
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
195 return in;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
196 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
197
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
198
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
199 static ssize_t
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
200 ngx_sendmsg(ngx_connection_t *c, ngx_iovec_t *vec)
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
201 {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
202 ssize_t n;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
203 ngx_err_t err;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
204 struct msghdr msg;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
205
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
206 ngx_memzero(&msg, sizeof(struct msghdr));
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
207
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
208 if (c->socklen) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
209 msg.msg_name = c->sockaddr;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
210 msg.msg_namelen = c->socklen;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
211 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
212
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
213 msg.msg_iov = vec->iovs;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
214 msg.msg_iovlen = vec->count;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
215
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
216 eintr:
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
217
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
218 n = sendmsg(c->fd, &msg, 0);
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
219
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
220 ngx_log_debug2(NGX_LOG_DEBUG_EVENT, c->log, 0,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
221 "sendmsg: %z of %uz", n, vec->size);
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
222
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
223 if (n == -1) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
224 err = ngx_errno;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
225
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
226 switch (err) {
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
227 case NGX_EAGAIN:
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
228 ngx_log_debug0(NGX_LOG_DEBUG_EVENT, c->log, err,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
229 "sendmsg() not ready");
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
230 return NGX_AGAIN;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
231
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
232 case NGX_EINTR:
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
233 ngx_log_debug0(NGX_LOG_DEBUG_EVENT, c->log, err,
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
234 "sendmsg() was interrupted");
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
235 goto eintr;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
236
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
237 default:
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
238 c->write->error = 1;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
239 ngx_connection_error(c, err, "sendmsg() failed");
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
240 return NGX_ERROR;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
241 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
242 }
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
243
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
244 return n;
56fc55e32f23 Stream: filters.
Roman Arutyunyan <arut@nginx.com>
parents:
diff changeset
245 }