0
|
1
|
|
2 /*
|
|
3 * Copyright (C) Igor Sysoev
|
|
4 */
|
|
5
|
|
6
|
|
7 #include <ngx_config.h>
|
|
8 #include <ngx_core.h>
|
|
9 #include <ngx_event.h>
|
|
10 #include <ngx_aio.h>
|
|
11
|
|
12
|
|
13 ngx_chain_t *ngx_aio_write_chain(ngx_connection_t *c, ngx_chain_t *in,
|
|
14 off_t limit)
|
|
15 {
|
|
16 u_char *buf, *prev;
|
|
17 off_t send, sent;
|
|
18 size_t len;
|
10
|
19 ssize_t n, size;
|
0
|
20 ngx_err_t err;
|
|
21 ngx_chain_t *cl;
|
|
22
|
|
23 send = 0;
|
|
24 sent = 0;
|
|
25 cl = in;
|
|
26
|
|
27 while (cl) {
|
|
28
|
|
29 if (cl->buf->pos == cl->buf->last) {
|
|
30 cl = cl->next;
|
|
31 continue;
|
|
32 }
|
|
33
|
|
34 /* we can post the single aio operation only */
|
|
35
|
|
36 if (!c->write->ready) {
|
|
37 return cl;
|
|
38 }
|
|
39
|
|
40 buf = cl->buf->pos;
|
|
41 prev = buf;
|
|
42 len = 0;
|
|
43
|
|
44 /* coalesce the neighbouring bufs */
|
|
45
|
|
46 while (cl && prev == cl->buf->pos && send < limit) {
|
|
47 if (ngx_buf_special(cl->buf)) {
|
|
48 continue;
|
|
49 }
|
|
50
|
|
51 size = cl->buf->last - cl->buf->pos;
|
|
52
|
|
53 if (send + size > limit) {
|
|
54 size = limit - send;
|
|
55 }
|
|
56
|
|
57 len += size;
|
|
58 prev = cl->buf->pos + size;
|
|
59 send += size;
|
|
60 cl = cl->next;
|
|
61 }
|
|
62
|
|
63 n = ngx_aio_write(c, buf, len);
|
|
64
|
10
|
65 ngx_log_debug1(NGX_LOG_DEBUG_EVENT, c->log, 0, "aio_write: %z", n);
|
0
|
66
|
|
67 if (n == NGX_ERROR) {
|
|
68 return NGX_CHAIN_ERROR;
|
|
69 }
|
|
70
|
|
71 if (n > 0) {
|
|
72 sent += n;
|
|
73 c->sent += n;
|
|
74 }
|
|
75
|
|
76 ngx_log_debug1(NGX_LOG_DEBUG_EVENT, c->log, 0,
|
10
|
77 "aio_write sent: %O", c->sent);
|
0
|
78
|
|
79 for (cl = in; cl; cl = cl->next) {
|
|
80
|
|
81 if (sent >= cl->buf->last - cl->buf->pos) {
|
|
82 sent -= cl->buf->last - cl->buf->pos;
|
|
83 cl->buf->pos = cl->buf->last;
|
|
84
|
|
85 continue;
|
|
86 }
|
|
87
|
|
88 cl->buf->pos += sent;
|
|
89
|
|
90 break;
|
|
91 }
|
|
92 }
|
|
93
|
|
94 return cl;
|
|
95 }
|