blob: 2a43498adcfffb0f03f64998e46287abd993a314 [file] [log] [blame]
Igor Sysoev239baac2003-06-11 15:28:34 +00001
2#include <ngx_config.h>
3#include <ngx_core.h>
4#include <ngx_event.h>
5
6
7ssize_t ngx_wsarecv(ngx_connection_t *c, char *buf, size_t size)
8{
Igor Sysoev7f9d8942003-11-14 07:20:34 +00009 int rc;
10 u_int flags;
11 size_t bytes;
12 WSABUF wsabuf[1];
13 ngx_err_t err;
14 ngx_event_t *rev;
15
16 wsabuf[0].buf = buf;
17 wsabuf[0].len = size;
18 flags = 0;
19 bytes = 0;
20
21 rc = WSARecv(c->fd, wsabuf, 1, &bytes, &flags, NULL, NULL);
22
23 ngx_log_debug(c->log, "WSARecv: %d:%d" _ rc _ bytes);
24
25 rev = c->read;
26
27 if (rc == -1) {
28 rev->ready = 0;
29 err = ngx_socket_errno;
30
31 if (err == WSAEWOULDBLOCK) {
32 ngx_log_error(NGX_LOG_INFO, c->log, err, "WSARecv() EAGAIN");
33 return NGX_AGAIN;
34 }
35
36 rev->error = 1;
37 ngx_log_error(NGX_LOG_CRIT, c->log, err, "WSARecv() failed");
38 return NGX_ERROR;
39 }
40
41 if (bytes < size) {
42 rev->ready = 0;
43 }
44
45 if (bytes == 0) {
46 rev->eof = 1;
47 }
48
49 return bytes;
50}
51
52
53ssize_t ngx_overlapped_wsarecv(ngx_connection_t *c, char *buf, size_t size)
54{
55 int rc;
56 u_int flags;
57 size_t bytes;
58 WSABUF wsabuf[1];
59 ngx_err_t err;
60 ngx_event_t *rev;
61 LPWSAOVERLAPPED ovlp;
62
63 rev = c->read;
64
65 if (!rev->ready) {
66 ngx_log_error(NGX_LOG_ALERT, rev->log, 0, "SECOND WSA POST");
67 return NGX_AGAIN;
68 }
69
70 ngx_log_debug(c->log, "rev->complete: %d" _ rev->complete);
71
72 if (rev->complete) {
73 rev->complete = 0;
74
75 if (ngx_event_flags & NGX_USE_IOCP_EVENT) {
76 if (rev->ovlp.error) {
77 ngx_log_error(NGX_LOG_ERR, c->log, rev->ovlp.error,
78 "WSARecv() failed");
79 return NGX_ERROR;
80 }
81
82 return rev->available;
83 }
84
85 if (WSAGetOverlappedResult(c->fd, (LPWSAOVERLAPPED) &rev->ovlp,
86 &bytes, 0, NULL) == 0) {
87 ngx_log_error(NGX_LOG_CRIT, c->log, ngx_socket_errno,
88 "WSARecv() or WSAGetOverlappedResult() failed");
89
90 return NGX_ERROR;
91 }
92
93 return bytes;
94 }
95
96 ovlp = (LPWSAOVERLAPPED) &rev->ovlp;
97 ngx_memzero(ovlp, sizeof(WSAOVERLAPPED));
98 wsabuf[0].buf = buf;
99 wsabuf[0].len = size;
100 flags = 0;
101 bytes = 0;
102
103 rc = WSARecv(c->fd, wsabuf, 1, &bytes, &flags, ovlp, NULL);
104
105 rev->complete = 0;
106
107 ngx_log_debug(c->log, "WSARecv: %d:%d" _ rc _ bytes);
108
109 if (rc == -1) {
110 err = ngx_socket_errno;
111 if (err == WSA_IO_PENDING) {
112 rev->active = 1;
113 return NGX_AGAIN;
114 }
115
116 rev->error = 1;
117 ngx_log_error(NGX_LOG_CRIT, c->log, err, "WSARecv() failed");
118 return NGX_ERROR;
119 }
120
121 if (ngx_event_flags & NGX_USE_IOCP_EVENT) {
122
123 /*
124 * if a socket was bound with I/O completion port
125 * then GetQueuedCompletionStatus() would anyway return its status
126 * despite that WSARecv() was already complete
127 */
128
129 rev->active = 1;
130 return NGX_AGAIN;
131 }
132
133 if (bytes == 0) {
134 rev->eof = 1;
135 rev->ready = 0;
136
137 } else {
138 rev->ready = 1;
139 }
140
141 rev->active = 0;
142
143 return bytes;
144}
145
146#if 0
147
148/* DELELTE IT WHEN ABOVE FUNC WOULD BE TESTED */
149
150ssize_t ngx_wsarecv(ngx_connection_t *c, char *buf, size_t size)
151{
Igor Sysoev239baac2003-06-11 15:28:34 +0000152 int rc;
153 u_int flags;
154 size_t bytes;
155 WSABUF wsabuf[1];
156 ngx_err_t err;
157 ngx_event_t *rev;
158 LPWSAOVERLAPPED ovlp;
159
160 rev = c->read;
161 bytes = 0;
162
Igor Sysoev0a280a32003-10-12 16:49:16 +0000163 if ((ngx_event_flags & NGX_USE_AIO_EVENT) && rev->ready) {
Igor Sysoev239baac2003-06-11 15:28:34 +0000164 rev->ready = 0;
165
166 /* the overlapped WSARecv() completed */
167
Igor Sysoev0a280a32003-10-12 16:49:16 +0000168 if (ngx_event_flags & NGX_USE_IOCP_EVENT) {
Igor Sysoev239baac2003-06-11 15:28:34 +0000169 if (rev->ovlp.error) {
170 ngx_log_error(NGX_LOG_ERR, c->log, rev->ovlp.error,
171 "WSARecv() failed");
172 return NGX_ERROR;
173 }
174
175 return rev->available;
176 }
177
178 if (WSAGetOverlappedResult(c->fd, (LPWSAOVERLAPPED) &rev->ovlp,
179 &bytes, 0, NULL) == 0) {
180 err = ngx_socket_errno;
181 ngx_log_error(NGX_LOG_CRIT, c->log, err,
182 "WSARecv() or WSAGetOverlappedResult() failed");
183
184 return NGX_ERROR;
185 }
186
187 return bytes;
188 }
189
Igor Sysoev0a280a32003-10-12 16:49:16 +0000190 if (ngx_event_flags & NGX_USE_AIO_EVENT) {
Igor Sysoev239baac2003-06-11 15:28:34 +0000191 ovlp = (LPWSAOVERLAPPED) &c->read->ovlp;
192 ngx_memzero(ovlp, sizeof(WSAOVERLAPPED));
193
194 } else {
195 ovlp = NULL;
196 }
197
198 wsabuf[0].buf = buf;
199 wsabuf[0].len = size;
200 flags = 0;
201
202 rc = WSARecv(c->fd, wsabuf, 1, &bytes, &flags, ovlp, NULL);
203
204 ngx_log_debug(c->log, "WSARecv: %d:%d" _ rc _ bytes);
205
206 if (rc == -1) {
207 err = ngx_socket_errno;
208 if (err == WSA_IO_PENDING) {
209 return NGX_AGAIN;
210
211 } else if (err == WSAEWOULDBLOCK) {
212 ngx_log_error(NGX_LOG_INFO, c->log, err, "WSARecv() EAGAIN");
213 return NGX_AGAIN;
214
215 } else {
216 ngx_log_error(NGX_LOG_CRIT, c->log, err, "WSARecv() failed");
217 return NGX_ERROR;
218 }
219 }
220
Igor Sysoev0a280a32003-10-12 16:49:16 +0000221 if (ngx_event_flags & NGX_USE_IOCP_EVENT) {
Igor Sysoev239baac2003-06-11 15:28:34 +0000222
223 /*
224 * If a socket was bound with I/O completion port
225 * then GetQueuedCompletionStatus() would anyway return its status
226 * despite that WSARecv() was already completed.
227 */
228
229 return NGX_AGAIN;
230 }
231
232 return bytes;
233}
Igor Sysoev7f9d8942003-11-14 07:20:34 +0000234
235#endif