blob: a741f090fe00b55d5a028433ba1b33e1f7a791f5 [file] [log] [blame]
Igor Sysoev0c331d92002-08-15 17:20:26 +00001
2#include <ngx_config.h>
Igor Sysoev016b8522002-08-29 16:59:54 +00003#include <ngx_core.h>
Igor Sysoev0c331d92002-08-15 17:20:26 +00004#include <ngx_http.h>
5
6int ngx_read_http_request_line(ngx_http_request_t *r)
7{
Igor Sysoev481b6432002-12-04 16:29:40 +00008 char ch;
9 char *p;
Igor Sysoev0c331d92002-08-15 17:20:26 +000010 enum {
Igor Sysoev016b8522002-08-29 16:59:54 +000011 sw_start = 0,
12 sw_space_after_method,
13 sw_spaces_before_uri,
14 sw_after_slash_in_uri,
15 sw_check_uri,
16 sw_uri,
17 sw_http_09,
18 sw_http_version,
19 sw_first_major_digit,
20 sw_major_digit,
21 sw_first_minor_digit,
22 sw_minor_digit,
23 sw_almost_done,
24 sw_done
Igor Sysoev481b6432002-12-04 16:29:40 +000025 } state;
26
27 state = r->state;
28 p = r->header_in->pos.mem;
Igor Sysoev0c331d92002-08-15 17:20:26 +000029
Igor Sysoev016b8522002-08-29 16:59:54 +000030 while (p < r->header_in->last.mem && state < sw_done) {
Igor Sysoev0c331d92002-08-15 17:20:26 +000031 ch = *p++;
32
33/*
34printf("\nstate: %d, pos: %x, end: %x, char: '%c' buf: %s",
Igor Sysoev016b8522002-08-29 16:59:54 +000035 state, p, r->header_in->last, ch, p);
Igor Sysoev0c331d92002-08-15 17:20:26 +000036*/
37
Igor Sysoev2ba1ee02002-10-04 17:58:04 +000038 /* GCC 2.95.2 and VC 6.0 compile this switch as jump table */
Igor Sysoev0c331d92002-08-15 17:20:26 +000039
40 switch (state) {
41
42 /* HTTP methods: GET, HEAD, POST */
Igor Sysoev016b8522002-08-29 16:59:54 +000043 case sw_start:
Igor Sysoev0c331d92002-08-15 17:20:26 +000044 switch (ch) {
45 case 'G':
Igor Sysoev4e9393a2003-01-09 05:36:00 +000046 if (p + 1 >= r->header_in->last.mem) {
Igor Sysoev016b8522002-08-29 16:59:54 +000047 return NGX_AGAIN;
Igor Sysoev4e9393a2003-01-09 05:36:00 +000048 }
Igor Sysoev0c331d92002-08-15 17:20:26 +000049
Igor Sysoev4e9393a2003-01-09 05:36:00 +000050 if (*p != 'E' || *(p + 1) != 'T') {
Igor Sysoev1af7c822002-09-13 14:47:42 +000051 return NGX_HTTP_PARSE_INVALID_METHOD;
Igor Sysoev4e9393a2003-01-09 05:36:00 +000052 }
Igor Sysoev0c331d92002-08-15 17:20:26 +000053
54 r->method = NGX_HTTP_GET;
55 p += 2;
56 break;
57
58 case 'H':
Igor Sysoev4e9393a2003-01-09 05:36:00 +000059 if (p + 2 >= r->header_in->last.mem) {
Igor Sysoev016b8522002-08-29 16:59:54 +000060 return NGX_AGAIN;
Igor Sysoev4e9393a2003-01-09 05:36:00 +000061 }
Igor Sysoev0c331d92002-08-15 17:20:26 +000062
Igor Sysoev4e9393a2003-01-09 05:36:00 +000063 if (*p != 'E' || *(p + 1) != 'A' || *(p + 2) != 'D') {
Igor Sysoev1af7c822002-09-13 14:47:42 +000064 return NGX_HTTP_PARSE_INVALID_METHOD;
Igor Sysoev4e9393a2003-01-09 05:36:00 +000065 }
Igor Sysoev0c331d92002-08-15 17:20:26 +000066
67 r->method = NGX_HTTP_HEAD;
68 p += 3;
69 break;
70
71 case 'P':
Igor Sysoev4e9393a2003-01-09 05:36:00 +000072 if (p + 2 >= r->header_in->last.mem) {
Igor Sysoev016b8522002-08-29 16:59:54 +000073 return NGX_AGAIN;
Igor Sysoev4e9393a2003-01-09 05:36:00 +000074 }
Igor Sysoev0c331d92002-08-15 17:20:26 +000075
Igor Sysoev4e9393a2003-01-09 05:36:00 +000076 if (*p != 'O' || *(p + 1) != 'S' || *(p + 2) != 'T') {
Igor Sysoev1af7c822002-09-13 14:47:42 +000077 return NGX_HTTP_PARSE_INVALID_METHOD;
Igor Sysoev4e9393a2003-01-09 05:36:00 +000078 }
Igor Sysoev0c331d92002-08-15 17:20:26 +000079
80 r->method = NGX_HTTP_POST;
81 p += 3;
82 break;
83
84 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +000085 return NGX_HTTP_PARSE_INVALID_METHOD;
Igor Sysoev0c331d92002-08-15 17:20:26 +000086 }
87
Igor Sysoev016b8522002-08-29 16:59:54 +000088 state = sw_space_after_method;
Igor Sysoev0c331d92002-08-15 17:20:26 +000089 break;
90
91 /* single space after method */
Igor Sysoev016b8522002-08-29 16:59:54 +000092 case sw_space_after_method:
Igor Sysoev0c331d92002-08-15 17:20:26 +000093 switch (ch) {
94 case ' ':
Igor Sysoev016b8522002-08-29 16:59:54 +000095 state = sw_spaces_before_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +000096 break;
97 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +000098 return NGX_HTTP_PARSE_INVALID_METHOD;
Igor Sysoev0c331d92002-08-15 17:20:26 +000099 }
100 break;
101
102 /* space* before URI */
Igor Sysoev016b8522002-08-29 16:59:54 +0000103 case sw_spaces_before_uri:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000104 switch (ch) {
105 case '/':
106 r->uri_start = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000107 state = sw_after_slash_in_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000108 break;
109 case ' ':
110 break;
111 default:
112 r->unusual_uri = 1;
113 r->uri_start = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000114 state = sw_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000115 break;
116 }
117 break;
118
Igor Sysoev42feecb2002-12-15 06:25:09 +0000119 /* check "/." or "//" */
Igor Sysoev016b8522002-08-29 16:59:54 +0000120 case sw_after_slash_in_uri:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000121 switch (ch) {
122 case CR:
123 r->uri_end = p - 1;
124 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000125 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000126 break;
127 case LF:
128 r->uri_end = p - 1;
129 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000130 state = sw_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000131 break;
132 case ' ':
133 r->uri_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000134 state = sw_http_09;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000135 break;
136 case '.':
137 r->complex_uri = 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000138 state = sw_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000139 break;
140 case '/':
Igor Sysoev42feecb2002-12-15 06:25:09 +0000141#if (WIN32)
Igor Sysoev0c331d92002-08-15 17:20:26 +0000142 r->complex_uri = 1;
Igor Sysoev42feecb2002-12-15 06:25:09 +0000143#endif
Igor Sysoev0c331d92002-08-15 17:20:26 +0000144 break;
145 case '?':
146 r->args_start = p;
Igor Sysoev016b8522002-08-29 16:59:54 +0000147 state = sw_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000148 break;
149 default:
Igor Sysoev016b8522002-08-29 16:59:54 +0000150 state = sw_check_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000151 break;
152 }
153 break;
154
155 /* check slash in URI */
Igor Sysoev016b8522002-08-29 16:59:54 +0000156 case sw_check_uri:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000157 switch (ch) {
158 case CR:
159 r->uri_end = p - 1;
160 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000161 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000162 break;
163 case LF:
164 r->uri_end = p - 1;
165 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000166 state = sw_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000167 break;
168 case ' ':
169 r->uri_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000170 state = sw_http_09;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000171 break;
172 case '.':
173 r->uri_ext = p;
174 break;
175 case '/':
176 r->uri_ext = NULL;
Igor Sysoev016b8522002-08-29 16:59:54 +0000177 state = sw_after_slash_in_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000178 break;
179 case '?':
180 r->args_start = p;
Igor Sysoev016b8522002-08-29 16:59:54 +0000181 state = sw_uri;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000182 break;
183 }
184 break;
185
186 /* URI */
Igor Sysoev016b8522002-08-29 16:59:54 +0000187 case sw_uri:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000188 switch (ch) {
189 case CR:
190 r->uri_end = p - 1;
191 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000192 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000193 break;
194 case LF:
195 r->uri_end = p - 1;
196 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000197 state = sw_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000198 break;
199 case ' ':
200 r->uri_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000201 state = sw_http_09;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000202 break;
203 }
204 break;
205
206 /* space+ after URI */
Igor Sysoev016b8522002-08-29 16:59:54 +0000207 case sw_http_09:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000208 switch (ch) {
209 case ' ':
210 break;
211 case CR:
212 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000213 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000214 break;
215 case LF:
216 r->http_minor = 9;
Igor Sysoev016b8522002-08-29 16:59:54 +0000217 state = sw_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000218 break;
219 case 'H':
Igor Sysoev016b8522002-08-29 16:59:54 +0000220 state = sw_http_version;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000221 break;
222 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +0000223 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000224 }
225 break;
226
Igor Sysoev481b6432002-12-04 16:29:40 +0000227 /* "TTP/" */
Igor Sysoev016b8522002-08-29 16:59:54 +0000228 case sw_http_version:
229 if (p + 2 >= r->header_in->last.mem) {
230 r->state = sw_http_version;
231 r->header_in->pos.mem = p - 1;
232 return NGX_AGAIN;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000233 }
234
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000235 if (ch != 'T' || *p != 'T' || *(p + 1) != 'P' || *(p + 2) != '/') {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000236 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000237 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000238
239 p += 3;
Igor Sysoev016b8522002-08-29 16:59:54 +0000240 state = sw_first_major_digit;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000241 break;
242
243 /* first digit of major HTTP version */
Igor Sysoev016b8522002-08-29 16:59:54 +0000244 case sw_first_major_digit:
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000245 if (ch < '1' || ch > '9') {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000246 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000247 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000248
249 r->http_major = ch - '0';
Igor Sysoev016b8522002-08-29 16:59:54 +0000250 state = sw_major_digit;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000251 break;
252
253 /* major HTTP version or dot */
Igor Sysoev016b8522002-08-29 16:59:54 +0000254 case sw_major_digit:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000255 if (ch == '.') {
Igor Sysoev016b8522002-08-29 16:59:54 +0000256 state = sw_first_minor_digit;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000257 break;
258 }
259
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000260 if (ch < '0' || ch > '9') {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000261 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000262 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000263
264 r->http_major = r->http_major * 10 + ch - '0';
265 break;
266
267 /* first digit of minor HTTP version */
Igor Sysoev016b8522002-08-29 16:59:54 +0000268 case sw_first_minor_digit:
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000269 if (ch < '0' || ch > '9') {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000270 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000271 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000272
273 r->http_minor = ch - '0';
Igor Sysoev016b8522002-08-29 16:59:54 +0000274 state = sw_minor_digit;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000275 break;
276
277 /* minor HTTP version or end of request line */
Igor Sysoev016b8522002-08-29 16:59:54 +0000278 case sw_minor_digit:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000279 if (ch == CR) {
Igor Sysoev016b8522002-08-29 16:59:54 +0000280 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000281 break;
282 }
283
284 if (ch == LF) {
Igor Sysoev016b8522002-08-29 16:59:54 +0000285 state = sw_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000286 break;
287 }
288
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000289 if (ch < '0' || ch > '9') {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000290 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000291 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000292
293 r->http_minor = r->http_minor * 10 + ch - '0';
294 break;
295
296 /* end of request line */
Igor Sysoev016b8522002-08-29 16:59:54 +0000297 case sw_almost_done:
Igor Sysoev2ba1ee02002-10-04 17:58:04 +0000298 r->request_end = p - 2;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000299 switch (ch) {
300 case LF:
Igor Sysoev016b8522002-08-29 16:59:54 +0000301 state = sw_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000302 break;
303 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +0000304 return NGX_HTTP_PARSE_INVALID_REQUEST;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000305 }
306 break;
307 }
308 }
309
Igor Sysoev016b8522002-08-29 16:59:54 +0000310 r->header_in->pos.mem = p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000311
Igor Sysoev016b8522002-08-29 16:59:54 +0000312 if (state == sw_done) {
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000313 if (r->request_end == NULL) {
Igor Sysoev2ba1ee02002-10-04 17:58:04 +0000314 r->request_end = p - 1;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000315 }
316
Igor Sysoev0c331d92002-08-15 17:20:26 +0000317 r->http_version = r->http_major * 1000 + r->http_minor;
Igor Sysoev016b8522002-08-29 16:59:54 +0000318 r->state = sw_start;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000319
320 if (r->http_version == 9 && r->method == NGX_HTTP_HEAD) {
Igor Sysoev1af7c822002-09-13 14:47:42 +0000321 return NGX_HTTP_PARSE_INVALID_HEAD;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000322 } else {
Igor Sysoev016b8522002-08-29 16:59:54 +0000323 return NGX_OK;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000324 }
325
Igor Sysoev0c331d92002-08-15 17:20:26 +0000326 } else {
327 r->state = state;
Igor Sysoev016b8522002-08-29 16:59:54 +0000328 return NGX_AGAIN;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000329 }
330}
331
Igor Sysoevb0869052002-12-10 18:05:12 +0000332int ngx_read_http_header_line(ngx_http_request_t *r, ngx_hunk_t *h)
Igor Sysoev0c331d92002-08-15 17:20:26 +0000333{
Igor Sysoev481b6432002-12-04 16:29:40 +0000334 char c, ch;
335 char *p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000336 enum {
Igor Sysoev016b8522002-08-29 16:59:54 +0000337 sw_start = 0,
338 sw_name,
339 sw_space_before_value,
340 sw_value,
341 sw_space_after_value,
342 sw_almost_done,
343 sw_header_almost_done,
344 sw_done,
345 sw_header_done
Igor Sysoev481b6432002-12-04 16:29:40 +0000346 } state;
347
348 state = r->state;
Igor Sysoevb0869052002-12-10 18:05:12 +0000349 p = h->pos.mem;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000350
Igor Sysoevb0869052002-12-10 18:05:12 +0000351 while (p < h->last.mem && state < sw_done) {
Igor Sysoev0c331d92002-08-15 17:20:26 +0000352 ch = *p++;
353
354/*
355printf("\nstate: %d, pos: %x, end: %x, char: '%c' buf: %s",
Igor Sysoevb0869052002-12-10 18:05:12 +0000356 state, p, h->last.mem, ch, p);
Igor Sysoev0c331d92002-08-15 17:20:26 +0000357*/
358
359 switch (state) {
360
361 /* first char */
Igor Sysoev016b8522002-08-29 16:59:54 +0000362 case sw_start:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000363 switch (ch) {
364 case CR:
365 r->header_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000366 state = sw_header_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000367 break;
368 case LF:
369 r->header_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000370 state = sw_header_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000371 break;
372 default:
Igor Sysoev016b8522002-08-29 16:59:54 +0000373 state = sw_name;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000374 r->header_name_start = p - 1;
375
376 c = ch | 0x20;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000377 if (c >= 'a' && c <= 'z') {
Igor Sysoev0c331d92002-08-15 17:20:26 +0000378 break;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000379 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000380
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000381 if (ch == '-') {
Igor Sysoev0c331d92002-08-15 17:20:26 +0000382 break;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000383 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000384
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000385 if (ch >= '0' && ch <= '9') {
Igor Sysoev0c331d92002-08-15 17:20:26 +0000386 break;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000387 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000388
Igor Sysoev1af7c822002-09-13 14:47:42 +0000389 return NGX_HTTP_PARSE_INVALID_HEADER;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000390
391 }
392 break;
393
394 /* header name */
Igor Sysoev016b8522002-08-29 16:59:54 +0000395 case sw_name:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000396 c = ch | 0x20;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000397 if (c >= 'a' && c <= 'z') {
Igor Sysoev0c331d92002-08-15 17:20:26 +0000398 break;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000399 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000400
401 if (ch == ':') {
402 r->header_name_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000403 state = sw_space_before_value;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000404 break;
405 }
406
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000407 if (ch == '-') {
Igor Sysoev0c331d92002-08-15 17:20:26 +0000408 break;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000409 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000410
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000411 if (ch >= '0' && ch <= '9') {
Igor Sysoev0c331d92002-08-15 17:20:26 +0000412 break;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000413 }
Igor Sysoev0c331d92002-08-15 17:20:26 +0000414
Igor Sysoev1af7c822002-09-13 14:47:42 +0000415 return NGX_HTTP_PARSE_INVALID_HEADER;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000416
417 /* space* before header value */
Igor Sysoev016b8522002-08-29 16:59:54 +0000418 case sw_space_before_value:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000419 switch (ch) {
420 case ' ':
421 break;
422 case CR:
423 r->header_start = r->header_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000424 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000425 break;
426 case LF:
427 r->header_start = r->header_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000428 state = sw_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000429 break;
430 default:
431 r->header_start = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000432 state = sw_value;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000433 break;
434 }
435 break;
436
437 /* header value */
Igor Sysoev016b8522002-08-29 16:59:54 +0000438 case sw_value:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000439 switch (ch) {
440 case ' ':
441 r->header_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000442 state = sw_space_after_value;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000443 break;
444 case CR:
445 r->header_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000446 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000447 break;
448 case LF:
449 r->header_end = p - 1;
Igor Sysoev016b8522002-08-29 16:59:54 +0000450 state = sw_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000451 break;
452 }
453 break;
454
455 /* space* before end of header line */
Igor Sysoev016b8522002-08-29 16:59:54 +0000456 case sw_space_after_value:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000457 switch (ch) {
458 case ' ':
459 break;
460 case CR:
Igor Sysoev016b8522002-08-29 16:59:54 +0000461 state = sw_almost_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000462 break;
463 case LF:
Igor Sysoev016b8522002-08-29 16:59:54 +0000464 state = sw_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000465 break;
466 default:
Igor Sysoev016b8522002-08-29 16:59:54 +0000467 state = sw_value;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000468 break;
469 }
470 break;
471
472 /* end of header line */
Igor Sysoev016b8522002-08-29 16:59:54 +0000473 case sw_almost_done:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000474 switch (ch) {
475 case LF:
Igor Sysoev016b8522002-08-29 16:59:54 +0000476 state = sw_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000477 break;
478 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +0000479 return NGX_HTTP_PARSE_INVALID_HEADER;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000480 }
481 break;
482
483 /* end of header */
Igor Sysoev016b8522002-08-29 16:59:54 +0000484 case sw_header_almost_done:
Igor Sysoev0c331d92002-08-15 17:20:26 +0000485 switch (ch) {
486 case LF:
Igor Sysoev016b8522002-08-29 16:59:54 +0000487 state = sw_header_done;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000488 break;
489 default:
Igor Sysoev1af7c822002-09-13 14:47:42 +0000490 return NGX_HTTP_PARSE_INVALID_HEADER;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000491 }
492 break;
493 }
494 }
495
Igor Sysoevb0869052002-12-10 18:05:12 +0000496 h->pos.mem = p;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000497
Igor Sysoev016b8522002-08-29 16:59:54 +0000498 if (state == sw_done) {
499 r->state = sw_start;
500 return NGX_OK;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000501
Igor Sysoev016b8522002-08-29 16:59:54 +0000502 } else if (state == sw_header_done) {
503 r->state = sw_start;
Igor Sysoev1af7c822002-09-13 14:47:42 +0000504 return NGX_HTTP_PARSE_HEADER_DONE;
Igor Sysoev4e9393a2003-01-09 05:36:00 +0000505
Igor Sysoev0c331d92002-08-15 17:20:26 +0000506 } else {
507 r->state = state;
Igor Sysoev016b8522002-08-29 16:59:54 +0000508 return NGX_AGAIN;
Igor Sysoev0c331d92002-08-15 17:20:26 +0000509 }
510}