xref: /aosp_15_r20/external/libwebsockets/lib/roles/h1/ops-h1.c (revision 1c60b9aca93fdbc9b5f19b2d2194c91294b22281)
1*1c60b9acSAndroid Build Coastguard Worker /*
2*1c60b9acSAndroid Build Coastguard Worker  * libwebsockets - small server side websockets and web server implementation
3*1c60b9acSAndroid Build Coastguard Worker  *
4*1c60b9acSAndroid Build Coastguard Worker  * Copyright (C) 2010 - 2019 Andy Green <[email protected]>
5*1c60b9acSAndroid Build Coastguard Worker  *
6*1c60b9acSAndroid Build Coastguard Worker  * Permission is hereby granted, free of charge, to any person obtaining a copy
7*1c60b9acSAndroid Build Coastguard Worker  * of this software and associated documentation files (the "Software"), to
8*1c60b9acSAndroid Build Coastguard Worker  * deal in the Software without restriction, including without limitation the
9*1c60b9acSAndroid Build Coastguard Worker  * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
10*1c60b9acSAndroid Build Coastguard Worker  * sell copies of the Software, and to permit persons to whom the Software is
11*1c60b9acSAndroid Build Coastguard Worker  * furnished to do so, subject to the following conditions:
12*1c60b9acSAndroid Build Coastguard Worker  *
13*1c60b9acSAndroid Build Coastguard Worker  * The above copyright notice and this permission notice shall be included in
14*1c60b9acSAndroid Build Coastguard Worker  * all copies or substantial portions of the Software.
15*1c60b9acSAndroid Build Coastguard Worker  *
16*1c60b9acSAndroid Build Coastguard Worker  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17*1c60b9acSAndroid Build Coastguard Worker  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18*1c60b9acSAndroid Build Coastguard Worker  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19*1c60b9acSAndroid Build Coastguard Worker  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20*1c60b9acSAndroid Build Coastguard Worker  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21*1c60b9acSAndroid Build Coastguard Worker  * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
22*1c60b9acSAndroid Build Coastguard Worker  * IN THE SOFTWARE.
23*1c60b9acSAndroid Build Coastguard Worker  */
24*1c60b9acSAndroid Build Coastguard Worker 
25*1c60b9acSAndroid Build Coastguard Worker #include <private-lib-core.h>
26*1c60b9acSAndroid Build Coastguard Worker 
27*1c60b9acSAndroid Build Coastguard Worker #ifndef min
28*1c60b9acSAndroid Build Coastguard Worker #define min(a, b) ((a) < (b) ? (a) : (b))
29*1c60b9acSAndroid Build Coastguard Worker #endif
30*1c60b9acSAndroid Build Coastguard Worker 
31*1c60b9acSAndroid Build Coastguard Worker 
32*1c60b9acSAndroid Build Coastguard Worker /*
33*1c60b9acSAndroid Build Coastguard Worker  * We have to take care about parsing because the headers may be split
34*1c60b9acSAndroid Build Coastguard Worker  * into multiple fragments.  They may contain unknown headers with arbitrary
35*1c60b9acSAndroid Build Coastguard Worker  * argument lengths.  So, we parse using a single-character at a time state
36*1c60b9acSAndroid Build Coastguard Worker  * machine that is completely independent of packet size.
37*1c60b9acSAndroid Build Coastguard Worker  *
38*1c60b9acSAndroid Build Coastguard Worker  * Returns <0 for error or length of chars consumed from buf (up to len)
39*1c60b9acSAndroid Build Coastguard Worker  */
40*1c60b9acSAndroid Build Coastguard Worker 
41*1c60b9acSAndroid Build Coastguard Worker int
lws_read_h1(struct lws * wsi,unsigned char * buf,lws_filepos_t len)42*1c60b9acSAndroid Build Coastguard Worker lws_read_h1(struct lws *wsi, unsigned char *buf, lws_filepos_t len)
43*1c60b9acSAndroid Build Coastguard Worker {
44*1c60b9acSAndroid Build Coastguard Worker 	unsigned char *last_char, *oldbuf = buf;
45*1c60b9acSAndroid Build Coastguard Worker 	lws_filepos_t body_chunk_len;
46*1c60b9acSAndroid Build Coastguard Worker 	size_t n;
47*1c60b9acSAndroid Build Coastguard Worker 
48*1c60b9acSAndroid Build Coastguard Worker 	lwsl_debug("%s: h1 path: wsi state 0x%x\n", __func__, lwsi_state(wsi));
49*1c60b9acSAndroid Build Coastguard Worker 
50*1c60b9acSAndroid Build Coastguard Worker 	switch (lwsi_state(wsi)) {
51*1c60b9acSAndroid Build Coastguard Worker 
52*1c60b9acSAndroid Build Coastguard Worker 	case LRS_ISSUING_FILE:
53*1c60b9acSAndroid Build Coastguard Worker 		return 0;
54*1c60b9acSAndroid Build Coastguard Worker 
55*1c60b9acSAndroid Build Coastguard Worker 	case LRS_ESTABLISHED:
56*1c60b9acSAndroid Build Coastguard Worker 
57*1c60b9acSAndroid Build Coastguard Worker 		if (lwsi_role_ws(wsi))
58*1c60b9acSAndroid Build Coastguard Worker 			goto ws_mode;
59*1c60b9acSAndroid Build Coastguard Worker 
60*1c60b9acSAndroid Build Coastguard Worker 		if (lwsi_role_client(wsi))
61*1c60b9acSAndroid Build Coastguard Worker 			break;
62*1c60b9acSAndroid Build Coastguard Worker 
63*1c60b9acSAndroid Build Coastguard Worker 		wsi->hdr_parsing_completed = 0;
64*1c60b9acSAndroid Build Coastguard Worker 
65*1c60b9acSAndroid Build Coastguard Worker 		/* fallthru */
66*1c60b9acSAndroid Build Coastguard Worker 
67*1c60b9acSAndroid Build Coastguard Worker 	case LRS_HEADERS:
68*1c60b9acSAndroid Build Coastguard Worker 		if (!wsi->http.ah) {
69*1c60b9acSAndroid Build Coastguard Worker 			lwsl_err("%s: LRS_HEADERS: NULL ah\n", __func__);
70*1c60b9acSAndroid Build Coastguard Worker 			assert(0);
71*1c60b9acSAndroid Build Coastguard Worker 		}
72*1c60b9acSAndroid Build Coastguard Worker 		lwsl_parser("issuing %d bytes to parser\n", (int)len);
73*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_ROLE_WS) && defined(LWS_WITH_CLIENT)
74*1c60b9acSAndroid Build Coastguard Worker 		if (lws_ws_handshake_client(wsi, &buf, (size_t)len))
75*1c60b9acSAndroid Build Coastguard Worker 			goto bail;
76*1c60b9acSAndroid Build Coastguard Worker #endif
77*1c60b9acSAndroid Build Coastguard Worker 		last_char = buf;
78*1c60b9acSAndroid Build Coastguard Worker 		if (lws_handshake_server(wsi, &buf, (size_t)len))
79*1c60b9acSAndroid Build Coastguard Worker 			/* Handshake indicates this session is done. */
80*1c60b9acSAndroid Build Coastguard Worker 			goto bail;
81*1c60b9acSAndroid Build Coastguard Worker 
82*1c60b9acSAndroid Build Coastguard Worker 		/* we might have transitioned to RAW */
83*1c60b9acSAndroid Build Coastguard Worker 		if (wsi->role_ops == &role_ops_raw_skt
84*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_ROLE_RAW_FILE)
85*1c60b9acSAndroid Build Coastguard Worker 				||
86*1c60b9acSAndroid Build Coastguard Worker 		    wsi->role_ops == &role_ops_raw_file
87*1c60b9acSAndroid Build Coastguard Worker #endif
88*1c60b9acSAndroid Build Coastguard Worker 		    )
89*1c60b9acSAndroid Build Coastguard Worker 			 /* we gave the read buffer to RAW handler already */
90*1c60b9acSAndroid Build Coastguard Worker 			goto read_ok;
91*1c60b9acSAndroid Build Coastguard Worker 
92*1c60b9acSAndroid Build Coastguard Worker 		/*
93*1c60b9acSAndroid Build Coastguard Worker 		 * It's possible that we've exhausted our data already, or
94*1c60b9acSAndroid Build Coastguard Worker 		 * rx flow control has stopped us dealing with this early,
95*1c60b9acSAndroid Build Coastguard Worker 		 * but lws_handshake_server doesn't update len for us.
96*1c60b9acSAndroid Build Coastguard Worker 		 * Figure out how much was read, so that we can proceed
97*1c60b9acSAndroid Build Coastguard Worker 		 * appropriately:
98*1c60b9acSAndroid Build Coastguard Worker 		 */
99*1c60b9acSAndroid Build Coastguard Worker 		len -= (unsigned int)lws_ptr_diff(buf, last_char);
100*1c60b9acSAndroid Build Coastguard Worker 
101*1c60b9acSAndroid Build Coastguard Worker 		if (!wsi->hdr_parsing_completed)
102*1c60b9acSAndroid Build Coastguard Worker 			/* More header content on the way */
103*1c60b9acSAndroid Build Coastguard Worker 			goto read_ok;
104*1c60b9acSAndroid Build Coastguard Worker 
105*1c60b9acSAndroid Build Coastguard Worker 		switch (lwsi_state(wsi)) {
106*1c60b9acSAndroid Build Coastguard Worker 			case LRS_ESTABLISHED:
107*1c60b9acSAndroid Build Coastguard Worker 			case LRS_HEADERS:
108*1c60b9acSAndroid Build Coastguard Worker 				goto read_ok;
109*1c60b9acSAndroid Build Coastguard Worker 			case LRS_ISSUING_FILE:
110*1c60b9acSAndroid Build Coastguard Worker 				goto read_ok;
111*1c60b9acSAndroid Build Coastguard Worker 			case LRS_DISCARD_BODY:
112*1c60b9acSAndroid Build Coastguard Worker 			case LRS_BODY:
113*1c60b9acSAndroid Build Coastguard Worker 				wsi->http.rx_content_remain =
114*1c60b9acSAndroid Build Coastguard Worker 						wsi->http.rx_content_length;
115*1c60b9acSAndroid Build Coastguard Worker 				if (wsi->http.rx_content_remain)
116*1c60b9acSAndroid Build Coastguard Worker 					goto http_postbody;
117*1c60b9acSAndroid Build Coastguard Worker 
118*1c60b9acSAndroid Build Coastguard Worker 				/* there is no POST content */
119*1c60b9acSAndroid Build Coastguard Worker 				goto postbody_completion;
120*1c60b9acSAndroid Build Coastguard Worker 			default:
121*1c60b9acSAndroid Build Coastguard Worker 				break;
122*1c60b9acSAndroid Build Coastguard Worker 		}
123*1c60b9acSAndroid Build Coastguard Worker 		break;
124*1c60b9acSAndroid Build Coastguard Worker 
125*1c60b9acSAndroid Build Coastguard Worker 	case LRS_DISCARD_BODY:
126*1c60b9acSAndroid Build Coastguard Worker 	case LRS_BODY:
127*1c60b9acSAndroid Build Coastguard Worker http_postbody:
128*1c60b9acSAndroid Build Coastguard Worker 		lwsl_info("%s: http post body: cl set %d, remain %d, len %d\n", __func__,
129*1c60b9acSAndroid Build Coastguard Worker 			    (int)wsi->http.content_length_given,
130*1c60b9acSAndroid Build Coastguard Worker 			    (int)wsi->http.rx_content_remain, (int)len);
131*1c60b9acSAndroid Build Coastguard Worker 
132*1c60b9acSAndroid Build Coastguard Worker 		if (wsi->http.content_length_given && !wsi->http.rx_content_remain)
133*1c60b9acSAndroid Build Coastguard Worker 			goto postbody_completion;
134*1c60b9acSAndroid Build Coastguard Worker 
135*1c60b9acSAndroid Build Coastguard Worker 		while (len && (!wsi->http.content_length_given || wsi->http.rx_content_remain)) {
136*1c60b9acSAndroid Build Coastguard Worker 			/* Copy as much as possible, up to the limit of:
137*1c60b9acSAndroid Build Coastguard Worker 			 * what we have in the read buffer (len)
138*1c60b9acSAndroid Build Coastguard Worker 			 * remaining portion of the POST body (content_remain)
139*1c60b9acSAndroid Build Coastguard Worker 			 */
140*1c60b9acSAndroid Build Coastguard Worker 			if (wsi->http.content_length_given)
141*1c60b9acSAndroid Build Coastguard Worker 				body_chunk_len = min(wsi->http.rx_content_remain, len);
142*1c60b9acSAndroid Build Coastguard Worker 			else
143*1c60b9acSAndroid Build Coastguard Worker 				body_chunk_len = len;
144*1c60b9acSAndroid Build Coastguard Worker 			wsi->http.rx_content_remain -= body_chunk_len;
145*1c60b9acSAndroid Build Coastguard Worker 			// len -= body_chunk_len;
146*1c60b9acSAndroid Build Coastguard Worker #ifdef LWS_WITH_CGI
147*1c60b9acSAndroid Build Coastguard Worker 			if (wsi->http.cgi) {
148*1c60b9acSAndroid Build Coastguard Worker 				struct lws_cgi_args args;
149*1c60b9acSAndroid Build Coastguard Worker 
150*1c60b9acSAndroid Build Coastguard Worker 				args.ch = LWS_STDIN;
151*1c60b9acSAndroid Build Coastguard Worker 				args.stdwsi = &wsi->http.cgi->lsp->stdwsi[0];
152*1c60b9acSAndroid Build Coastguard Worker 				args.data = buf;
153*1c60b9acSAndroid Build Coastguard Worker 				args.len = (int)(unsigned int)body_chunk_len;
154*1c60b9acSAndroid Build Coastguard Worker 
155*1c60b9acSAndroid Build Coastguard Worker 				/* returns how much used */
156*1c60b9acSAndroid Build Coastguard Worker 				n = (unsigned int)user_callback_handle_rxflow(
157*1c60b9acSAndroid Build Coastguard Worker 					wsi->a.protocol->callback,
158*1c60b9acSAndroid Build Coastguard Worker 					wsi, LWS_CALLBACK_CGI_STDIN_DATA,
159*1c60b9acSAndroid Build Coastguard Worker 					wsi->user_space,
160*1c60b9acSAndroid Build Coastguard Worker 					(void *)&args, 0);
161*1c60b9acSAndroid Build Coastguard Worker 				if ((int)n < 0)
162*1c60b9acSAndroid Build Coastguard Worker 					goto bail;
163*1c60b9acSAndroid Build Coastguard Worker 			} else {
164*1c60b9acSAndroid Build Coastguard Worker #endif
165*1c60b9acSAndroid Build Coastguard Worker 				if (lwsi_state(wsi) != LRS_DISCARD_BODY) {
166*1c60b9acSAndroid Build Coastguard Worker 					lwsl_info("%s: HTTP_BODY %d\n", __func__, (int)body_chunk_len);
167*1c60b9acSAndroid Build Coastguard Worker 					n = (unsigned int)wsi->a.protocol->callback(wsi,
168*1c60b9acSAndroid Build Coastguard Worker 						LWS_CALLBACK_HTTP_BODY, wsi->user_space,
169*1c60b9acSAndroid Build Coastguard Worker 						buf, (size_t)body_chunk_len);
170*1c60b9acSAndroid Build Coastguard Worker 					if (n)
171*1c60b9acSAndroid Build Coastguard Worker 						goto bail;
172*1c60b9acSAndroid Build Coastguard Worker 				}
173*1c60b9acSAndroid Build Coastguard Worker 				n = (size_t)body_chunk_len;
174*1c60b9acSAndroid Build Coastguard Worker #ifdef LWS_WITH_CGI
175*1c60b9acSAndroid Build Coastguard Worker 			}
176*1c60b9acSAndroid Build Coastguard Worker #endif
177*1c60b9acSAndroid Build Coastguard Worker 			lwsl_info("%s: advancing buf by %d\n", __func__, (int)n);
178*1c60b9acSAndroid Build Coastguard Worker 			buf += n;
179*1c60b9acSAndroid Build Coastguard Worker 
180*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_ROLE_H2)
181*1c60b9acSAndroid Build Coastguard Worker 			if (lwsi_role_h2(wsi) && !wsi->http.content_length_given) {
182*1c60b9acSAndroid Build Coastguard Worker 				struct lws *w = lws_get_network_wsi(wsi);
183*1c60b9acSAndroid Build Coastguard Worker 
184*1c60b9acSAndroid Build Coastguard Worker 				if (w)
185*1c60b9acSAndroid Build Coastguard Worker 					lwsl_info("%s: h2: nwsi h2 flags %d\n", __func__,
186*1c60b9acSAndroid Build Coastguard Worker 						w->h2.h2n ? w->h2.h2n->flags: -1);
187*1c60b9acSAndroid Build Coastguard Worker 
188*1c60b9acSAndroid Build Coastguard Worker 				if (w && w->h2.h2n && !(w->h2.h2n->flags & 1)) {
189*1c60b9acSAndroid Build Coastguard Worker 					lwsl_info("%s: h2, no cl, not END_STREAM, continuing\n", __func__);
190*1c60b9acSAndroid Build Coastguard Worker 					lws_set_timeout(wsi,
191*1c60b9acSAndroid Build Coastguard Worker 						PENDING_TIMEOUT_HTTP_CONTENT,
192*1c60b9acSAndroid Build Coastguard Worker 						(int)wsi->a.context->timeout_secs);
193*1c60b9acSAndroid Build Coastguard Worker 					break;
194*1c60b9acSAndroid Build Coastguard Worker 				}
195*1c60b9acSAndroid Build Coastguard Worker 				goto postbody_completion;
196*1c60b9acSAndroid Build Coastguard Worker 			}
197*1c60b9acSAndroid Build Coastguard Worker #endif
198*1c60b9acSAndroid Build Coastguard Worker 
199*1c60b9acSAndroid Build Coastguard Worker 			if (wsi->http.rx_content_remain)  {
200*1c60b9acSAndroid Build Coastguard Worker 				lws_set_timeout(wsi,
201*1c60b9acSAndroid Build Coastguard Worker 						PENDING_TIMEOUT_HTTP_CONTENT,
202*1c60b9acSAndroid Build Coastguard Worker 						(int)wsi->a.context->timeout_secs);
203*1c60b9acSAndroid Build Coastguard Worker 				break;
204*1c60b9acSAndroid Build Coastguard Worker 			}
205*1c60b9acSAndroid Build Coastguard Worker 			/* he sent all the content in time */
206*1c60b9acSAndroid Build Coastguard Worker postbody_completion:
207*1c60b9acSAndroid Build Coastguard Worker #ifdef LWS_WITH_CGI
208*1c60b9acSAndroid Build Coastguard Worker 			/*
209*1c60b9acSAndroid Build Coastguard Worker 			 * If we're running a cgi, we can't let him off the
210*1c60b9acSAndroid Build Coastguard Worker 			 * hook just because he sent his POST data
211*1c60b9acSAndroid Build Coastguard Worker 			 */
212*1c60b9acSAndroid Build Coastguard Worker 			if (wsi->http.cgi)
213*1c60b9acSAndroid Build Coastguard Worker 				lws_set_timeout(wsi, PENDING_TIMEOUT_CGI,
214*1c60b9acSAndroid Build Coastguard Worker 						(int)wsi->a.context->timeout_secs);
215*1c60b9acSAndroid Build Coastguard Worker 			else
216*1c60b9acSAndroid Build Coastguard Worker #endif
217*1c60b9acSAndroid Build Coastguard Worker 			lws_set_timeout(wsi, NO_PENDING_TIMEOUT, 0);
218*1c60b9acSAndroid Build Coastguard Worker #ifdef LWS_WITH_CGI
219*1c60b9acSAndroid Build Coastguard Worker 			if (!wsi->http.cgi)
220*1c60b9acSAndroid Build Coastguard Worker #endif
221*1c60b9acSAndroid Build Coastguard Worker 			{
222*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SERVER)
223*1c60b9acSAndroid Build Coastguard Worker 				if (lwsi_state(wsi) == LRS_DISCARD_BODY) {
224*1c60b9acSAndroid Build Coastguard Worker 					/*
225*1c60b9acSAndroid Build Coastguard Worker 					 * repeat the transaction completed
226*1c60b9acSAndroid Build Coastguard Worker 					 * that got us into this state, having
227*1c60b9acSAndroid Build Coastguard Worker 					 * consumed the pending body now
228*1c60b9acSAndroid Build Coastguard Worker 					 */
229*1c60b9acSAndroid Build Coastguard Worker 
230*1c60b9acSAndroid Build Coastguard Worker 					if (lws_http_transaction_completed(wsi))
231*1c60b9acSAndroid Build Coastguard Worker 						goto bail;
232*1c60b9acSAndroid Build Coastguard Worker 					break;
233*1c60b9acSAndroid Build Coastguard Worker 				}
234*1c60b9acSAndroid Build Coastguard Worker #endif
235*1c60b9acSAndroid Build Coastguard Worker 				lwsl_info("HTTP_BODY_COMPLETION: %s (%s)\n",
236*1c60b9acSAndroid Build Coastguard Worker 					  lws_wsi_tag(wsi), wsi->a.protocol->name);
237*1c60b9acSAndroid Build Coastguard Worker 
238*1c60b9acSAndroid Build Coastguard Worker 				n = (unsigned int)wsi->a.protocol->callback(wsi,
239*1c60b9acSAndroid Build Coastguard Worker 					LWS_CALLBACK_HTTP_BODY_COMPLETION,
240*1c60b9acSAndroid Build Coastguard Worker 					wsi->user_space, NULL, 0);
241*1c60b9acSAndroid Build Coastguard Worker 				if (n) {
242*1c60b9acSAndroid Build Coastguard Worker 					lwsl_info("%s: bailing after BODY_COMPLETION\n", __func__);
243*1c60b9acSAndroid Build Coastguard Worker 					goto bail;
244*1c60b9acSAndroid Build Coastguard Worker 				}
245*1c60b9acSAndroid Build Coastguard Worker 
246*1c60b9acSAndroid Build Coastguard Worker 				if (wsi->mux_substream)
247*1c60b9acSAndroid Build Coastguard Worker 					lwsi_set_state(wsi, LRS_ESTABLISHED);
248*1c60b9acSAndroid Build Coastguard Worker 			}
249*1c60b9acSAndroid Build Coastguard Worker 
250*1c60b9acSAndroid Build Coastguard Worker 			break;
251*1c60b9acSAndroid Build Coastguard Worker 		}
252*1c60b9acSAndroid Build Coastguard Worker 		break;
253*1c60b9acSAndroid Build Coastguard Worker 
254*1c60b9acSAndroid Build Coastguard Worker 	case LRS_RETURNED_CLOSE:
255*1c60b9acSAndroid Build Coastguard Worker 	case LRS_AWAITING_CLOSE_ACK:
256*1c60b9acSAndroid Build Coastguard Worker 	case LRS_WAITING_TO_SEND_CLOSE:
257*1c60b9acSAndroid Build Coastguard Worker 	case LRS_SHUTDOWN:
258*1c60b9acSAndroid Build Coastguard Worker 
259*1c60b9acSAndroid Build Coastguard Worker ws_mode:
260*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_CLIENT) && defined(LWS_ROLE_WS)
261*1c60b9acSAndroid Build Coastguard Worker 		// lwsl_notice("%s: ws_mode\n", __func__);
262*1c60b9acSAndroid Build Coastguard Worker 		if (lws_ws_handshake_client(wsi, &buf, (size_t)len))
263*1c60b9acSAndroid Build Coastguard Worker 			goto bail;
264*1c60b9acSAndroid Build Coastguard Worker #endif
265*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_ROLE_WS)
266*1c60b9acSAndroid Build Coastguard Worker 		if (lwsi_role_ws(wsi) && lwsi_role_server(wsi) &&
267*1c60b9acSAndroid Build Coastguard Worker 			/*
268*1c60b9acSAndroid Build Coastguard Worker 			 * for h2 we are on the swsi
269*1c60b9acSAndroid Build Coastguard Worker 			 */
270*1c60b9acSAndroid Build Coastguard Worker 		    lws_parse_ws(wsi, &buf, (size_t)len) < 0) {
271*1c60b9acSAndroid Build Coastguard Worker 			lwsl_info("%s: lws_parse_ws bailed\n", __func__);
272*1c60b9acSAndroid Build Coastguard Worker 			goto bail;
273*1c60b9acSAndroid Build Coastguard Worker 		}
274*1c60b9acSAndroid Build Coastguard Worker #endif
275*1c60b9acSAndroid Build Coastguard Worker 		// lwsl_notice("%s: ws_mode: buf moved on by %d\n", __func__,
276*1c60b9acSAndroid Build Coastguard Worker 		//	       lws_ptr_diff(buf, oldbuf));
277*1c60b9acSAndroid Build Coastguard Worker 		break;
278*1c60b9acSAndroid Build Coastguard Worker 
279*1c60b9acSAndroid Build Coastguard Worker 	case LRS_DEFERRING_ACTION:
280*1c60b9acSAndroid Build Coastguard Worker 		lwsl_notice("%s: LRS_DEFERRING_ACTION\n", __func__);
281*1c60b9acSAndroid Build Coastguard Worker 		break;
282*1c60b9acSAndroid Build Coastguard Worker 
283*1c60b9acSAndroid Build Coastguard Worker 	case LRS_SSL_ACK_PENDING:
284*1c60b9acSAndroid Build Coastguard Worker 		break;
285*1c60b9acSAndroid Build Coastguard Worker 
286*1c60b9acSAndroid Build Coastguard Worker 	case LRS_FLUSHING_BEFORE_CLOSE:
287*1c60b9acSAndroid Build Coastguard Worker 		break;
288*1c60b9acSAndroid Build Coastguard Worker 
289*1c60b9acSAndroid Build Coastguard Worker 	case LRS_DEAD_SOCKET:
290*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: Unhandled state LRS_DEAD_SOCKET\n", __func__);
291*1c60b9acSAndroid Build Coastguard Worker 		goto bail;
292*1c60b9acSAndroid Build Coastguard Worker 		// assert(0);
293*1c60b9acSAndroid Build Coastguard Worker 		/* fallthru */
294*1c60b9acSAndroid Build Coastguard Worker 
295*1c60b9acSAndroid Build Coastguard Worker 	default:
296*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: Unhandled state %d\n", __func__, lwsi_state(wsi));
297*1c60b9acSAndroid Build Coastguard Worker 		assert(0);
298*1c60b9acSAndroid Build Coastguard Worker 		goto bail;
299*1c60b9acSAndroid Build Coastguard Worker 	}
300*1c60b9acSAndroid Build Coastguard Worker 
301*1c60b9acSAndroid Build Coastguard Worker read_ok:
302*1c60b9acSAndroid Build Coastguard Worker 	/* Nothing more to do for now */
303*1c60b9acSAndroid Build Coastguard Worker //	lwsl_info("%s: %p: read_ok, used %ld (len %d, state %d)\n", __func__,
304*1c60b9acSAndroid Build Coastguard Worker //		  wsi, (long)(buf - oldbuf), (int)len, wsi->state);
305*1c60b9acSAndroid Build Coastguard Worker 
306*1c60b9acSAndroid Build Coastguard Worker 	return lws_ptr_diff(buf, oldbuf);
307*1c60b9acSAndroid Build Coastguard Worker 
308*1c60b9acSAndroid Build Coastguard Worker bail:
309*1c60b9acSAndroid Build Coastguard Worker 	/*
310*1c60b9acSAndroid Build Coastguard Worker 	 * h2 / h2-ws calls us recursively in
311*1c60b9acSAndroid Build Coastguard Worker 	 *
312*1c60b9acSAndroid Build Coastguard Worker 	 * lws_read_h1()->
313*1c60b9acSAndroid Build Coastguard Worker 	 *   lws_h2_parser()->
314*1c60b9acSAndroid Build Coastguard Worker 	 *     lws_read_h1()
315*1c60b9acSAndroid Build Coastguard Worker 	 *
316*1c60b9acSAndroid Build Coastguard Worker 	 * pattern, having stripped the h2 framing in the middle.
317*1c60b9acSAndroid Build Coastguard Worker 	 *
318*1c60b9acSAndroid Build Coastguard Worker 	 * When taking down the whole connection, make sure that only the
319*1c60b9acSAndroid Build Coastguard Worker 	 * outer lws_read() does the wsi close.
320*1c60b9acSAndroid Build Coastguard Worker 	 */
321*1c60b9acSAndroid Build Coastguard Worker 	if (!wsi->outer_will_close)
322*1c60b9acSAndroid Build Coastguard Worker 		lws_close_free_wsi(wsi, LWS_CLOSE_STATUS_NOSTATUS,
323*1c60b9acSAndroid Build Coastguard Worker 				   "lws_read_h1 bail");
324*1c60b9acSAndroid Build Coastguard Worker 
325*1c60b9acSAndroid Build Coastguard Worker 	return -1;
326*1c60b9acSAndroid Build Coastguard Worker }
327*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SERVER)
328*1c60b9acSAndroid Build Coastguard Worker static int
lws_h1_server_socket_service(struct lws * wsi,struct lws_pollfd * pollfd)329*1c60b9acSAndroid Build Coastguard Worker lws_h1_server_socket_service(struct lws *wsi, struct lws_pollfd *pollfd)
330*1c60b9acSAndroid Build Coastguard Worker {
331*1c60b9acSAndroid Build Coastguard Worker 	struct lws_context_per_thread *pt = &wsi->a.context->pt[(int)wsi->tsi];
332*1c60b9acSAndroid Build Coastguard Worker 	struct lws_tokens ebuf;
333*1c60b9acSAndroid Build Coastguard Worker 	int n, buffered;
334*1c60b9acSAndroid Build Coastguard Worker 
335*1c60b9acSAndroid Build Coastguard Worker 	if (lwsi_state(wsi) == LRS_DEFERRING_ACTION)
336*1c60b9acSAndroid Build Coastguard Worker 		goto try_pollout;
337*1c60b9acSAndroid Build Coastguard Worker 
338*1c60b9acSAndroid Build Coastguard Worker 	/* any incoming data ready? */
339*1c60b9acSAndroid Build Coastguard Worker 
340*1c60b9acSAndroid Build Coastguard Worker 	if (!(pollfd->revents & pollfd->events & LWS_POLLIN))
341*1c60b9acSAndroid Build Coastguard Worker 		goto try_pollout;
342*1c60b9acSAndroid Build Coastguard Worker 
343*1c60b9acSAndroid Build Coastguard Worker 	/*
344*1c60b9acSAndroid Build Coastguard Worker 	 * If we previously just did POLLIN when IN and OUT were signaled
345*1c60b9acSAndroid Build Coastguard Worker 	 * (because POLLIN processing may have used up the POLLOUT), don't let
346*1c60b9acSAndroid Build Coastguard Worker 	 * that happen twice in a row... next time we see the situation favour
347*1c60b9acSAndroid Build Coastguard Worker 	 * POLLOUT
348*1c60b9acSAndroid Build Coastguard Worker 	 */
349*1c60b9acSAndroid Build Coastguard Worker 
350*1c60b9acSAndroid Build Coastguard Worker 	if (wsi->favoured_pollin &&
351*1c60b9acSAndroid Build Coastguard Worker 	    (pollfd->revents & pollfd->events & LWS_POLLOUT)) {
352*1c60b9acSAndroid Build Coastguard Worker 		// lwsl_notice("favouring pollout\n");
353*1c60b9acSAndroid Build Coastguard Worker 		wsi->favoured_pollin = 0;
354*1c60b9acSAndroid Build Coastguard Worker 		goto try_pollout;
355*1c60b9acSAndroid Build Coastguard Worker 	}
356*1c60b9acSAndroid Build Coastguard Worker 
357*1c60b9acSAndroid Build Coastguard Worker 	/*
358*1c60b9acSAndroid Build Coastguard Worker 	 * We haven't processed that the tunnel is set up yet, so
359*1c60b9acSAndroid Build Coastguard Worker 	 * defer reading
360*1c60b9acSAndroid Build Coastguard Worker 	 */
361*1c60b9acSAndroid Build Coastguard Worker 
362*1c60b9acSAndroid Build Coastguard Worker 	if (lwsi_state(wsi) == LRS_SSL_ACK_PENDING)
363*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_HANDLED;
364*1c60b9acSAndroid Build Coastguard Worker 
365*1c60b9acSAndroid Build Coastguard Worker 	/* these states imply we MUST have an ah attached */
366*1c60b9acSAndroid Build Coastguard Worker 
367*1c60b9acSAndroid Build Coastguard Worker 	if ((lwsi_state(wsi) == LRS_ESTABLISHED ||
368*1c60b9acSAndroid Build Coastguard Worker 	     lwsi_state(wsi) == LRS_ISSUING_FILE ||
369*1c60b9acSAndroid Build Coastguard Worker 	     lwsi_state(wsi) == LRS_HEADERS ||
370*1c60b9acSAndroid Build Coastguard Worker 	     lwsi_state(wsi) == LRS_DOING_TRANSACTION || /* at least, SSE */
371*1c60b9acSAndroid Build Coastguard Worker 	     lwsi_state(wsi) == LRS_DISCARD_BODY ||
372*1c60b9acSAndroid Build Coastguard Worker 	     lwsi_state(wsi) == LRS_BODY)) {
373*1c60b9acSAndroid Build Coastguard Worker 
374*1c60b9acSAndroid Build Coastguard Worker 		if (!wsi->http.ah && lws_header_table_attach(wsi, 0)) {
375*1c60b9acSAndroid Build Coastguard Worker 			lwsl_info("%s: %s: ah not available\n", __func__,
376*1c60b9acSAndroid Build Coastguard Worker 				  lws_wsi_tag(wsi));
377*1c60b9acSAndroid Build Coastguard Worker 			goto try_pollout;
378*1c60b9acSAndroid Build Coastguard Worker 		}
379*1c60b9acSAndroid Build Coastguard Worker 
380*1c60b9acSAndroid Build Coastguard Worker 		/*
381*1c60b9acSAndroid Build Coastguard Worker 		 * We got here because there was specifically POLLIN...
382*1c60b9acSAndroid Build Coastguard Worker 		 * regardless of our buflist state, we need to get it,
383*1c60b9acSAndroid Build Coastguard Worker 		 * and either use it, or append to the buflist and use
384*1c60b9acSAndroid Build Coastguard Worker 		 * buflist head material.
385*1c60b9acSAndroid Build Coastguard Worker 		 *
386*1c60b9acSAndroid Build Coastguard Worker 		 * We will not notice a connection close until the buflist is
387*1c60b9acSAndroid Build Coastguard Worker 		 * exhausted and we tried to do a read of some kind.
388*1c60b9acSAndroid Build Coastguard Worker 		 */
389*1c60b9acSAndroid Build Coastguard Worker 
390*1c60b9acSAndroid Build Coastguard Worker 		ebuf.token = NULL;
391*1c60b9acSAndroid Build Coastguard Worker 		ebuf.len = 0;
392*1c60b9acSAndroid Build Coastguard Worker 		buffered = lws_buflist_aware_read(pt, wsi, &ebuf, 0, __func__);
393*1c60b9acSAndroid Build Coastguard Worker 		switch (ebuf.len) {
394*1c60b9acSAndroid Build Coastguard Worker 		case 0:
395*1c60b9acSAndroid Build Coastguard Worker 			lwsl_info("%s: read 0 len a\n", __func__);
396*1c60b9acSAndroid Build Coastguard Worker 			wsi->seen_zero_length_recv = 1;
397*1c60b9acSAndroid Build Coastguard Worker 			if (lws_change_pollfd(wsi, LWS_POLLIN, 0))
398*1c60b9acSAndroid Build Coastguard Worker 				goto fail;
399*1c60b9acSAndroid Build Coastguard Worker #if !defined(LWS_WITHOUT_EXTENSIONS)
400*1c60b9acSAndroid Build Coastguard Worker 			/*
401*1c60b9acSAndroid Build Coastguard Worker 			 * autobahn requires us to win the race between close
402*1c60b9acSAndroid Build Coastguard Worker 			 * and draining the extensions
403*1c60b9acSAndroid Build Coastguard Worker 			 */
404*1c60b9acSAndroid Build Coastguard Worker 			if (wsi->ws &&
405*1c60b9acSAndroid Build Coastguard Worker 			    (wsi->ws->rx_draining_ext ||
406*1c60b9acSAndroid Build Coastguard Worker 			     wsi->ws->tx_draining_ext))
407*1c60b9acSAndroid Build Coastguard Worker 				goto try_pollout;
408*1c60b9acSAndroid Build Coastguard Worker #endif
409*1c60b9acSAndroid Build Coastguard Worker 			/*
410*1c60b9acSAndroid Build Coastguard Worker 			 * normally, we respond to close with logically closing
411*1c60b9acSAndroid Build Coastguard Worker 			 * our side immediately
412*1c60b9acSAndroid Build Coastguard Worker 			 */
413*1c60b9acSAndroid Build Coastguard Worker 			goto fail;
414*1c60b9acSAndroid Build Coastguard Worker 
415*1c60b9acSAndroid Build Coastguard Worker 		case LWS_SSL_CAPABLE_ERROR:
416*1c60b9acSAndroid Build Coastguard Worker 			goto fail;
417*1c60b9acSAndroid Build Coastguard Worker 		case LWS_SSL_CAPABLE_MORE_SERVICE:
418*1c60b9acSAndroid Build Coastguard Worker 			goto try_pollout;
419*1c60b9acSAndroid Build Coastguard Worker 		}
420*1c60b9acSAndroid Build Coastguard Worker 
421*1c60b9acSAndroid Build Coastguard Worker 		/* just ignore incoming if waiting for close */
422*1c60b9acSAndroid Build Coastguard Worker 		if (lwsi_state(wsi) == LRS_FLUSHING_BEFORE_CLOSE) {
423*1c60b9acSAndroid Build Coastguard Worker 			lwsl_notice("%s: just ignoring\n", __func__);
424*1c60b9acSAndroid Build Coastguard Worker 			goto try_pollout;
425*1c60b9acSAndroid Build Coastguard Worker 		}
426*1c60b9acSAndroid Build Coastguard Worker 
427*1c60b9acSAndroid Build Coastguard Worker 		if (lwsi_state(wsi) == LRS_ISSUING_FILE) {
428*1c60b9acSAndroid Build Coastguard Worker 			// lwsl_notice("stashing: wsi %p: bd %d\n", wsi, buffered);
429*1c60b9acSAndroid Build Coastguard Worker 			if (lws_buflist_aware_finished_consuming(wsi, &ebuf, 0,
430*1c60b9acSAndroid Build Coastguard Worker 							buffered, __func__))
431*1c60b9acSAndroid Build Coastguard Worker 				return LWS_HPI_RET_PLEASE_CLOSE_ME;
432*1c60b9acSAndroid Build Coastguard Worker 
433*1c60b9acSAndroid Build Coastguard Worker 			goto try_pollout;
434*1c60b9acSAndroid Build Coastguard Worker 		}
435*1c60b9acSAndroid Build Coastguard Worker 
436*1c60b9acSAndroid Build Coastguard Worker 		/*
437*1c60b9acSAndroid Build Coastguard Worker 		 * Otherwise give it to whoever wants it according to the
438*1c60b9acSAndroid Build Coastguard Worker 		 * connection state
439*1c60b9acSAndroid Build Coastguard Worker 		 */
440*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_ROLE_H2)
441*1c60b9acSAndroid Build Coastguard Worker 		if (lwsi_role_h2(wsi) && lwsi_state(wsi) != LRS_BODY)
442*1c60b9acSAndroid Build Coastguard Worker 			n = lws_read_h2(wsi, ebuf.token, (unsigned int)ebuf.len);
443*1c60b9acSAndroid Build Coastguard Worker 		else
444*1c60b9acSAndroid Build Coastguard Worker #endif
445*1c60b9acSAndroid Build Coastguard Worker 			n = lws_read_h1(wsi, ebuf.token, (unsigned int)ebuf.len);
446*1c60b9acSAndroid Build Coastguard Worker 		if (n < 0) /* we closed wsi */
447*1c60b9acSAndroid Build Coastguard Worker 			return LWS_HPI_RET_WSI_ALREADY_DIED;
448*1c60b9acSAndroid Build Coastguard Worker 
449*1c60b9acSAndroid Build Coastguard Worker 		// lwsl_notice("%s: consumed %d\n", __func__, n);
450*1c60b9acSAndroid Build Coastguard Worker 
451*1c60b9acSAndroid Build Coastguard Worker 		if (lws_buflist_aware_finished_consuming(wsi, &ebuf, n,
452*1c60b9acSAndroid Build Coastguard Worker 							 buffered, __func__))
453*1c60b9acSAndroid Build Coastguard Worker 			return LWS_HPI_RET_PLEASE_CLOSE_ME;
454*1c60b9acSAndroid Build Coastguard Worker 
455*1c60b9acSAndroid Build Coastguard Worker 		/*
456*1c60b9acSAndroid Build Coastguard Worker 		 * during the parsing our role changed to something non-http,
457*1c60b9acSAndroid Build Coastguard Worker 		 * so the ah has no further meaning
458*1c60b9acSAndroid Build Coastguard Worker 		 */
459*1c60b9acSAndroid Build Coastguard Worker 
460*1c60b9acSAndroid Build Coastguard Worker 		if (wsi->http.ah &&
461*1c60b9acSAndroid Build Coastguard Worker 		    !lwsi_role_h1(wsi) &&
462*1c60b9acSAndroid Build Coastguard Worker 		    !lwsi_role_h2(wsi) &&
463*1c60b9acSAndroid Build Coastguard Worker 		    !lwsi_role_cgi(wsi))
464*1c60b9acSAndroid Build Coastguard Worker 			lws_header_table_detach(wsi, 0);
465*1c60b9acSAndroid Build Coastguard Worker 
466*1c60b9acSAndroid Build Coastguard Worker 		/*
467*1c60b9acSAndroid Build Coastguard Worker 		 * He may have used up the writability above, if we will defer
468*1c60b9acSAndroid Build Coastguard Worker 		 * POLLOUT processing in favour of POLLIN, note it
469*1c60b9acSAndroid Build Coastguard Worker 		 */
470*1c60b9acSAndroid Build Coastguard Worker 
471*1c60b9acSAndroid Build Coastguard Worker 		if (pollfd->revents & LWS_POLLOUT)
472*1c60b9acSAndroid Build Coastguard Worker 			wsi->favoured_pollin = 1;
473*1c60b9acSAndroid Build Coastguard Worker 
474*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_HANDLED;
475*1c60b9acSAndroid Build Coastguard Worker 	}
476*1c60b9acSAndroid Build Coastguard Worker 
477*1c60b9acSAndroid Build Coastguard Worker 	/*
478*1c60b9acSAndroid Build Coastguard Worker 	 * He may have used up the writability above, if we will defer POLLOUT
479*1c60b9acSAndroid Build Coastguard Worker 	 * processing in favour of POLLIN, note it
480*1c60b9acSAndroid Build Coastguard Worker 	 */
481*1c60b9acSAndroid Build Coastguard Worker 
482*1c60b9acSAndroid Build Coastguard Worker 	if (pollfd->revents & LWS_POLLOUT)
483*1c60b9acSAndroid Build Coastguard Worker 		wsi->favoured_pollin = 1;
484*1c60b9acSAndroid Build Coastguard Worker 
485*1c60b9acSAndroid Build Coastguard Worker try_pollout:
486*1c60b9acSAndroid Build Coastguard Worker 
487*1c60b9acSAndroid Build Coastguard Worker 	/* this handles POLLOUT for http serving fragments */
488*1c60b9acSAndroid Build Coastguard Worker 
489*1c60b9acSAndroid Build Coastguard Worker 	if (!(pollfd->revents & LWS_POLLOUT))
490*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_HANDLED;
491*1c60b9acSAndroid Build Coastguard Worker 
492*1c60b9acSAndroid Build Coastguard Worker 	/* one shot */
493*1c60b9acSAndroid Build Coastguard Worker 	if (lws_change_pollfd(wsi, LWS_POLLOUT, 0)) {
494*1c60b9acSAndroid Build Coastguard Worker 		lwsl_notice("%s a\n", __func__);
495*1c60b9acSAndroid Build Coastguard Worker 		goto fail;
496*1c60b9acSAndroid Build Coastguard Worker 	}
497*1c60b9acSAndroid Build Coastguard Worker 
498*1c60b9acSAndroid Build Coastguard Worker 	/* clear back-to-back write detection */
499*1c60b9acSAndroid Build Coastguard Worker 	wsi->could_have_pending = 0;
500*1c60b9acSAndroid Build Coastguard Worker 
501*1c60b9acSAndroid Build Coastguard Worker 	if (lwsi_state(wsi) == LRS_DEFERRING_ACTION) {
502*1c60b9acSAndroid Build Coastguard Worker 		lwsl_debug("%s: LRS_DEFERRING_ACTION now writable\n", __func__);
503*1c60b9acSAndroid Build Coastguard Worker 
504*1c60b9acSAndroid Build Coastguard Worker 		lwsi_set_state(wsi, LRS_ESTABLISHED);
505*1c60b9acSAndroid Build Coastguard Worker 		if (lws_change_pollfd(wsi, LWS_POLLOUT, 0)) {
506*1c60b9acSAndroid Build Coastguard Worker 			lwsl_info("failed at set pollfd\n");
507*1c60b9acSAndroid Build Coastguard Worker 			goto fail;
508*1c60b9acSAndroid Build Coastguard Worker 		}
509*1c60b9acSAndroid Build Coastguard Worker 	}
510*1c60b9acSAndroid Build Coastguard Worker 
511*1c60b9acSAndroid Build Coastguard Worker 	if (!wsi->hdr_parsing_completed)
512*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_HANDLED;
513*1c60b9acSAndroid Build Coastguard Worker 
514*1c60b9acSAndroid Build Coastguard Worker 	if (lwsi_state(wsi) != LRS_ISSUING_FILE) {
515*1c60b9acSAndroid Build Coastguard Worker 
516*1c60b9acSAndroid Build Coastguard Worker 		if (lws_has_buffered_out(wsi)) {
517*1c60b9acSAndroid Build Coastguard Worker 			//lwsl_notice("%s: completing partial\n", __func__);
518*1c60b9acSAndroid Build Coastguard Worker 			if (lws_issue_raw(wsi, NULL, 0) < 0) {
519*1c60b9acSAndroid Build Coastguard Worker 				lwsl_info("%s signalling to close\n", __func__);
520*1c60b9acSAndroid Build Coastguard Worker 				goto fail;
521*1c60b9acSAndroid Build Coastguard Worker 			}
522*1c60b9acSAndroid Build Coastguard Worker 			return LWS_HPI_RET_HANDLED;
523*1c60b9acSAndroid Build Coastguard Worker 		}
524*1c60b9acSAndroid Build Coastguard Worker 
525*1c60b9acSAndroid Build Coastguard Worker 		n = user_callback_handle_rxflow(wsi->a.protocol->callback, wsi,
526*1c60b9acSAndroid Build Coastguard Worker 						LWS_CALLBACK_HTTP_WRITEABLE,
527*1c60b9acSAndroid Build Coastguard Worker 						wsi->user_space, NULL, 0);
528*1c60b9acSAndroid Build Coastguard Worker 		if (n < 0) {
529*1c60b9acSAndroid Build Coastguard Worker 			lwsl_info("writeable_fail\n");
530*1c60b9acSAndroid Build Coastguard Worker 			goto fail;
531*1c60b9acSAndroid Build Coastguard Worker 		}
532*1c60b9acSAndroid Build Coastguard Worker 
533*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_HANDLED;
534*1c60b9acSAndroid Build Coastguard Worker 	}
535*1c60b9acSAndroid Build Coastguard Worker 
536*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_FILE_OPS)
537*1c60b9acSAndroid Build Coastguard Worker 
538*1c60b9acSAndroid Build Coastguard Worker 	/* >0 == completion, <0 == error
539*1c60b9acSAndroid Build Coastguard Worker 	 *
540*1c60b9acSAndroid Build Coastguard Worker 	 * We'll get a LWS_CALLBACK_HTTP_FILE_COMPLETION callback when
541*1c60b9acSAndroid Build Coastguard Worker 	 * it's done.  That's the case even if we just completed the
542*1c60b9acSAndroid Build Coastguard Worker 	 * send, so wait for that.
543*1c60b9acSAndroid Build Coastguard Worker 	 */
544*1c60b9acSAndroid Build Coastguard Worker 	n = lws_serve_http_file_fragment(wsi);
545*1c60b9acSAndroid Build Coastguard Worker 	if (n < 0)
546*1c60b9acSAndroid Build Coastguard Worker 		goto fail;
547*1c60b9acSAndroid Build Coastguard Worker #endif
548*1c60b9acSAndroid Build Coastguard Worker 
549*1c60b9acSAndroid Build Coastguard Worker 	return LWS_HPI_RET_HANDLED;
550*1c60b9acSAndroid Build Coastguard Worker 
551*1c60b9acSAndroid Build Coastguard Worker 
552*1c60b9acSAndroid Build Coastguard Worker fail:
553*1c60b9acSAndroid Build Coastguard Worker 	lws_close_free_wsi(wsi, LWS_CLOSE_STATUS_NOSTATUS,
554*1c60b9acSAndroid Build Coastguard Worker 			   "server socket svc fail");
555*1c60b9acSAndroid Build Coastguard Worker 
556*1c60b9acSAndroid Build Coastguard Worker 	return LWS_HPI_RET_WSI_ALREADY_DIED;
557*1c60b9acSAndroid Build Coastguard Worker }
558*1c60b9acSAndroid Build Coastguard Worker #endif
559*1c60b9acSAndroid Build Coastguard Worker 
560*1c60b9acSAndroid Build Coastguard Worker static int
rops_handle_POLLIN_h1(struct lws_context_per_thread * pt,struct lws * wsi,struct lws_pollfd * pollfd)561*1c60b9acSAndroid Build Coastguard Worker rops_handle_POLLIN_h1(struct lws_context_per_thread *pt, struct lws *wsi,
562*1c60b9acSAndroid Build Coastguard Worker 		       struct lws_pollfd *pollfd)
563*1c60b9acSAndroid Build Coastguard Worker {
564*1c60b9acSAndroid Build Coastguard Worker 	if (lwsi_state(wsi) == LRS_IDLING) {
565*1c60b9acSAndroid Build Coastguard Worker 		uint8_t buf[1];
566*1c60b9acSAndroid Build Coastguard Worker 		int rlen;
567*1c60b9acSAndroid Build Coastguard Worker 
568*1c60b9acSAndroid Build Coastguard Worker 		/*
569*1c60b9acSAndroid Build Coastguard Worker 		 * h1 staggered spins here in IDLING if we don't close it.
570*1c60b9acSAndroid Build Coastguard Worker 		 * It shows POLLIN but the tls connection returns ERROR if
571*1c60b9acSAndroid Build Coastguard Worker 		 * you try to read it.
572*1c60b9acSAndroid Build Coastguard Worker 		 */
573*1c60b9acSAndroid Build Coastguard Worker 
574*1c60b9acSAndroid Build Coastguard Worker 		// lwsl_notice("%s: %p: wsistate 0x%x %s, revents 0x%x\n",
575*1c60b9acSAndroid Build Coastguard Worker 		//	    __func__, wsi, wsi->wsistate, wsi->role_ops->name,
576*1c60b9acSAndroid Build Coastguard Worker 		//	    pollfd->revents);
577*1c60b9acSAndroid Build Coastguard Worker 
578*1c60b9acSAndroid Build Coastguard Worker 		rlen = lws_ssl_capable_read(wsi, buf, sizeof(buf));
579*1c60b9acSAndroid Build Coastguard Worker 		if (rlen == LWS_SSL_CAPABLE_ERROR)
580*1c60b9acSAndroid Build Coastguard Worker 			return LWS_HPI_RET_PLEASE_CLOSE_ME;
581*1c60b9acSAndroid Build Coastguard Worker 	}
582*1c60b9acSAndroid Build Coastguard Worker 
583*1c60b9acSAndroid Build Coastguard Worker #ifdef LWS_WITH_CGI
584*1c60b9acSAndroid Build Coastguard Worker 	if (wsi->http.cgi && (pollfd->revents & LWS_POLLOUT)) {
585*1c60b9acSAndroid Build Coastguard Worker 		if (lws_handle_POLLOUT_event(wsi, pollfd))
586*1c60b9acSAndroid Build Coastguard Worker 			return LWS_HPI_RET_PLEASE_CLOSE_ME;
587*1c60b9acSAndroid Build Coastguard Worker 
588*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_HANDLED;
589*1c60b9acSAndroid Build Coastguard Worker 	}
590*1c60b9acSAndroid Build Coastguard Worker #endif
591*1c60b9acSAndroid Build Coastguard Worker 
592*1c60b9acSAndroid Build Coastguard Worker 	/* Priority 2: pre- compression transform */
593*1c60b9acSAndroid Build Coastguard Worker 
594*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_HTTP_STREAM_COMPRESSION)
595*1c60b9acSAndroid Build Coastguard Worker 	if (wsi->http.comp_ctx.buflist_comp ||
596*1c60b9acSAndroid Build Coastguard Worker 	    wsi->http.comp_ctx.may_have_more) {
597*1c60b9acSAndroid Build Coastguard Worker 		enum lws_write_protocol wp = LWS_WRITE_HTTP;
598*1c60b9acSAndroid Build Coastguard Worker 
599*1c60b9acSAndroid Build Coastguard Worker 		lwsl_info("%s: completing comp partial (buflist_comp %p, may %d)\n",
600*1c60b9acSAndroid Build Coastguard Worker 				__func__, wsi->http.comp_ctx.buflist_comp,
601*1c60b9acSAndroid Build Coastguard Worker 				wsi->http.comp_ctx.may_have_more
602*1c60b9acSAndroid Build Coastguard Worker 				);
603*1c60b9acSAndroid Build Coastguard Worker 
604*1c60b9acSAndroid Build Coastguard Worker 		if (lws_rops_fidx(wsi->role_ops, LWS_ROPS_write_role_protocol) &&
605*1c60b9acSAndroid Build Coastguard Worker 		    lws_rops_func_fidx(wsi->role_ops, LWS_ROPS_write_role_protocol).
606*1c60b9acSAndroid Build Coastguard Worker 					write_role_protocol(wsi, NULL, 0, &wp) < 0) {
607*1c60b9acSAndroid Build Coastguard Worker 			lwsl_info("%s signalling to close\n", __func__);
608*1c60b9acSAndroid Build Coastguard Worker 			return LWS_HPI_RET_PLEASE_CLOSE_ME;
609*1c60b9acSAndroid Build Coastguard Worker 		}
610*1c60b9acSAndroid Build Coastguard Worker 		lws_callback_on_writable(wsi);
611*1c60b9acSAndroid Build Coastguard Worker 
612*1c60b9acSAndroid Build Coastguard Worker 		if (!wsi->http.comp_ctx.buflist_comp &&
613*1c60b9acSAndroid Build Coastguard Worker 		    !wsi->http.comp_ctx.may_have_more &&
614*1c60b9acSAndroid Build Coastguard Worker 		    wsi->http.deferred_transaction_completed) {
615*1c60b9acSAndroid Build Coastguard Worker 			wsi->http.deferred_transaction_completed = 0;
616*1c60b9acSAndroid Build Coastguard Worker 			if (lws_http_transaction_completed(wsi))
617*1c60b9acSAndroid Build Coastguard Worker 				return LWS_HPI_RET_PLEASE_CLOSE_ME;
618*1c60b9acSAndroid Build Coastguard Worker 		}
619*1c60b9acSAndroid Build Coastguard Worker 
620*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_HANDLED;
621*1c60b9acSAndroid Build Coastguard Worker 	}
622*1c60b9acSAndroid Build Coastguard Worker #endif
623*1c60b9acSAndroid Build Coastguard Worker 
624*1c60b9acSAndroid Build Coastguard Worker         if (lws_is_flowcontrolled(wsi))
625*1c60b9acSAndroid Build Coastguard Worker                 /* We cannot deal with any kind of new RX because we are
626*1c60b9acSAndroid Build Coastguard Worker                  * RX-flowcontrolled.
627*1c60b9acSAndroid Build Coastguard Worker                  */
628*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_HANDLED;
629*1c60b9acSAndroid Build Coastguard Worker 
630*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SERVER)
631*1c60b9acSAndroid Build Coastguard Worker 	if (!lwsi_role_client(wsi)) {
632*1c60b9acSAndroid Build Coastguard Worker 		int n;
633*1c60b9acSAndroid Build Coastguard Worker 
634*1c60b9acSAndroid Build Coastguard Worker 		lwsl_debug("%s: %s: wsistate 0x%x\n", __func__, lws_wsi_tag(wsi),
635*1c60b9acSAndroid Build Coastguard Worker 			   (unsigned int)wsi->wsistate);
636*1c60b9acSAndroid Build Coastguard Worker 
637*1c60b9acSAndroid Build Coastguard Worker 		if (pollfd->revents & LWS_POLLHUP &&
638*1c60b9acSAndroid Build Coastguard Worker 		    !lws_buflist_total_len(&wsi->buflist))
639*1c60b9acSAndroid Build Coastguard Worker 			return LWS_HPI_RET_PLEASE_CLOSE_ME;
640*1c60b9acSAndroid Build Coastguard Worker 
641*1c60b9acSAndroid Build Coastguard Worker 		n = lws_h1_server_socket_service(wsi, pollfd);
642*1c60b9acSAndroid Build Coastguard Worker 		if (n != LWS_HPI_RET_HANDLED)
643*1c60b9acSAndroid Build Coastguard Worker 			return n;
644*1c60b9acSAndroid Build Coastguard Worker 		if (lwsi_state(wsi) != LRS_SSL_INIT)
645*1c60b9acSAndroid Build Coastguard Worker 			if (lws_server_socket_service_ssl(wsi,
646*1c60b9acSAndroid Build Coastguard Worker 							  LWS_SOCK_INVALID,
647*1c60b9acSAndroid Build Coastguard Worker 					!!(pollfd->revents & LWS_POLLIN)))
648*1c60b9acSAndroid Build Coastguard Worker 				return LWS_HPI_RET_PLEASE_CLOSE_ME;
649*1c60b9acSAndroid Build Coastguard Worker 
650*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_HANDLED;
651*1c60b9acSAndroid Build Coastguard Worker 	}
652*1c60b9acSAndroid Build Coastguard Worker #endif
653*1c60b9acSAndroid Build Coastguard Worker 
654*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_CLIENT)
655*1c60b9acSAndroid Build Coastguard Worker 	if ((pollfd->revents & LWS_POLLIN) &&
656*1c60b9acSAndroid Build Coastguard Worker 	     wsi->hdr_parsing_completed && !wsi->told_user_closed) {
657*1c60b9acSAndroid Build Coastguard Worker 
658*1c60b9acSAndroid Build Coastguard Worker 		/*
659*1c60b9acSAndroid Build Coastguard Worker 		 * In SSL mode we get POLLIN notification about
660*1c60b9acSAndroid Build Coastguard Worker 		 * encrypted data in.
661*1c60b9acSAndroid Build Coastguard Worker 		 *
662*1c60b9acSAndroid Build Coastguard Worker 		 * But that is not necessarily related to decrypted
663*1c60b9acSAndroid Build Coastguard Worker 		 * data out becoming available; in may need to perform
664*1c60b9acSAndroid Build Coastguard Worker 		 * other in or out before that happens.
665*1c60b9acSAndroid Build Coastguard Worker 		 *
666*1c60b9acSAndroid Build Coastguard Worker 		 * simply mark ourselves as having readable data
667*1c60b9acSAndroid Build Coastguard Worker 		 * and turn off our POLLIN
668*1c60b9acSAndroid Build Coastguard Worker 		 */
669*1c60b9acSAndroid Build Coastguard Worker 		wsi->client_rx_avail = 1;
670*1c60b9acSAndroid Build Coastguard Worker 		if (lws_change_pollfd(wsi, LWS_POLLIN, 0))
671*1c60b9acSAndroid Build Coastguard Worker 			return LWS_HPI_RET_PLEASE_CLOSE_ME;
672*1c60b9acSAndroid Build Coastguard Worker 
673*1c60b9acSAndroid Build Coastguard Worker 		//lwsl_notice("calling back %s\n", wsi->a.protocol->name);
674*1c60b9acSAndroid Build Coastguard Worker 
675*1c60b9acSAndroid Build Coastguard Worker 		/* let user code know, he'll usually ask for writeable
676*1c60b9acSAndroid Build Coastguard Worker 		 * callback and drain / re-enable it there
677*1c60b9acSAndroid Build Coastguard Worker 		 */
678*1c60b9acSAndroid Build Coastguard Worker 		if (user_callback_handle_rxflow(wsi->a.protocol->callback, wsi,
679*1c60b9acSAndroid Build Coastguard Worker 					       LWS_CALLBACK_RECEIVE_CLIENT_HTTP,
680*1c60b9acSAndroid Build Coastguard Worker 						wsi->user_space, NULL, 0)) {
681*1c60b9acSAndroid Build Coastguard Worker 			lwsl_info("RECEIVE_CLIENT_HTTP closed it\n");
682*1c60b9acSAndroid Build Coastguard Worker 			return LWS_HPI_RET_PLEASE_CLOSE_ME;
683*1c60b9acSAndroid Build Coastguard Worker 		}
684*1c60b9acSAndroid Build Coastguard Worker 
685*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_HANDLED;
686*1c60b9acSAndroid Build Coastguard Worker 	}
687*1c60b9acSAndroid Build Coastguard Worker #endif
688*1c60b9acSAndroid Build Coastguard Worker 
689*1c60b9acSAndroid Build Coastguard Worker //	if (lwsi_state(wsi) == LRS_ESTABLISHED)
690*1c60b9acSAndroid Build Coastguard Worker //		return LWS_HPI_RET_HANDLED;
691*1c60b9acSAndroid Build Coastguard Worker 
692*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_CLIENT)
693*1c60b9acSAndroid Build Coastguard Worker 	if ((pollfd->revents & LWS_POLLOUT) &&
694*1c60b9acSAndroid Build Coastguard Worker 	    lws_handle_POLLOUT_event(wsi, pollfd)) {
695*1c60b9acSAndroid Build Coastguard Worker 		lwsl_debug("POLLOUT event closed it\n");
696*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_PLEASE_CLOSE_ME;
697*1c60b9acSAndroid Build Coastguard Worker 	}
698*1c60b9acSAndroid Build Coastguard Worker 
699*1c60b9acSAndroid Build Coastguard Worker 	if (lws_http_client_socket_service(wsi, pollfd))
700*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_WSI_ALREADY_DIED;
701*1c60b9acSAndroid Build Coastguard Worker #endif
702*1c60b9acSAndroid Build Coastguard Worker 
703*1c60b9acSAndroid Build Coastguard Worker 	if (lwsi_state(wsi) == LRS_WAITING_CONNECT &&
704*1c60b9acSAndroid Build Coastguard Worker 	    (pollfd->revents & LWS_POLLHUP))
705*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HPI_RET_PLEASE_CLOSE_ME;
706*1c60b9acSAndroid Build Coastguard Worker 
707*1c60b9acSAndroid Build Coastguard Worker 	return LWS_HPI_RET_HANDLED;
708*1c60b9acSAndroid Build Coastguard Worker }
709*1c60b9acSAndroid Build Coastguard Worker 
710*1c60b9acSAndroid Build Coastguard Worker static int
rops_handle_POLLOUT_h1(struct lws * wsi)711*1c60b9acSAndroid Build Coastguard Worker rops_handle_POLLOUT_h1(struct lws *wsi)
712*1c60b9acSAndroid Build Coastguard Worker {
713*1c60b9acSAndroid Build Coastguard Worker 
714*1c60b9acSAndroid Build Coastguard Worker 
715*1c60b9acSAndroid Build Coastguard Worker 	if (lwsi_state(wsi) == LRS_ISSUE_HTTP_BODY ||
716*1c60b9acSAndroid Build Coastguard Worker 	    lwsi_state(wsi) == LRS_WAITING_SERVER_REPLY) {
717*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_HTTP_PROXY)
718*1c60b9acSAndroid Build Coastguard Worker 		if (wsi->http.proxy_clientside) {
719*1c60b9acSAndroid Build Coastguard Worker 			unsigned char *buf, prebuf[LWS_PRE + 1024];
720*1c60b9acSAndroid Build Coastguard Worker 			size_t len = lws_buflist_next_segment_len(
721*1c60b9acSAndroid Build Coastguard Worker 					&wsi->parent->http.buflist_post_body, &buf);
722*1c60b9acSAndroid Build Coastguard Worker 			int n;
723*1c60b9acSAndroid Build Coastguard Worker 
724*1c60b9acSAndroid Build Coastguard Worker 			if (len > sizeof(prebuf) - LWS_PRE)
725*1c60b9acSAndroid Build Coastguard Worker 				len = sizeof(prebuf) - LWS_PRE;
726*1c60b9acSAndroid Build Coastguard Worker 
727*1c60b9acSAndroid Build Coastguard Worker 			if (len) {
728*1c60b9acSAndroid Build Coastguard Worker 				memcpy(prebuf + LWS_PRE, buf, len);
729*1c60b9acSAndroid Build Coastguard Worker 
730*1c60b9acSAndroid Build Coastguard Worker 				lwsl_debug("%s: %s: proxying body %d %d %d %d %d\n",
731*1c60b9acSAndroid Build Coastguard Worker 						__func__, lws_wsi_tag(wsi), (int)len,
732*1c60b9acSAndroid Build Coastguard Worker 						(int)wsi->http.tx_content_length,
733*1c60b9acSAndroid Build Coastguard Worker 						(int)wsi->http.tx_content_remain,
734*1c60b9acSAndroid Build Coastguard Worker 						(int)wsi->http.rx_content_length,
735*1c60b9acSAndroid Build Coastguard Worker 						(int)wsi->http.rx_content_remain
736*1c60b9acSAndroid Build Coastguard Worker 						);
737*1c60b9acSAndroid Build Coastguard Worker 
738*1c60b9acSAndroid Build Coastguard Worker 				n = lws_write(wsi, prebuf + LWS_PRE, len, LWS_WRITE_HTTP);
739*1c60b9acSAndroid Build Coastguard Worker 				if (n < 0) {
740*1c60b9acSAndroid Build Coastguard Worker 					lwsl_err("%s: PROXY_BODY: write %d failed\n",
741*1c60b9acSAndroid Build Coastguard Worker 						 __func__, (int)len);
742*1c60b9acSAndroid Build Coastguard Worker 					return LWS_HP_RET_BAIL_DIE;
743*1c60b9acSAndroid Build Coastguard Worker 				}
744*1c60b9acSAndroid Build Coastguard Worker 
745*1c60b9acSAndroid Build Coastguard Worker 				lws_buflist_use_segment(&wsi->parent->http.buflist_post_body, len);
746*1c60b9acSAndroid Build Coastguard Worker 
747*1c60b9acSAndroid Build Coastguard Worker 			}
748*1c60b9acSAndroid Build Coastguard Worker 
749*1c60b9acSAndroid Build Coastguard Worker 			if (wsi->parent->http.buflist_post_body) {
750*1c60b9acSAndroid Build Coastguard Worker 				lws_callback_on_writable(wsi);
751*1c60b9acSAndroid Build Coastguard Worker 				return LWS_HP_RET_DROP_POLLOUT;
752*1c60b9acSAndroid Build Coastguard Worker 			}
753*1c60b9acSAndroid Build Coastguard Worker 
754*1c60b9acSAndroid Build Coastguard Worker 			lwsl_wsi_err(wsi, "nothing to send");
755*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_ROLE_H1) || defined(LWS_ROLE_H2)
756*1c60b9acSAndroid Build Coastguard Worker 			/* prepare ourselves to do the parsing */
757*1c60b9acSAndroid Build Coastguard Worker 			wsi->http.ah->parser_state = WSI_TOKEN_NAME_PART;
758*1c60b9acSAndroid Build Coastguard Worker 			wsi->http.ah->lextable_pos = 0;
759*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_CUSTOM_HEADERS)
760*1c60b9acSAndroid Build Coastguard Worker 			wsi->http.ah->unk_pos = 0;
761*1c60b9acSAndroid Build Coastguard Worker #endif
762*1c60b9acSAndroid Build Coastguard Worker #endif
763*1c60b9acSAndroid Build Coastguard Worker 			lwsi_set_state(wsi, LRS_WAITING_SERVER_REPLY);
764*1c60b9acSAndroid Build Coastguard Worker 			lws_set_timeout(wsi, PENDING_TIMEOUT_AWAITING_SERVER_RESPONSE,
765*1c60b9acSAndroid Build Coastguard Worker 					(int)wsi->a.context->timeout_secs);
766*1c60b9acSAndroid Build Coastguard Worker 
767*1c60b9acSAndroid Build Coastguard Worker 			return LWS_HP_RET_DROP_POLLOUT;
768*1c60b9acSAndroid Build Coastguard Worker 		}
769*1c60b9acSAndroid Build Coastguard Worker #endif
770*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HP_RET_USER_SERVICE;
771*1c60b9acSAndroid Build Coastguard Worker 	}
772*1c60b9acSAndroid Build Coastguard Worker 
773*1c60b9acSAndroid Build Coastguard Worker 	if (lwsi_role_client(wsi))
774*1c60b9acSAndroid Build Coastguard Worker 		return LWS_HP_RET_USER_SERVICE;
775*1c60b9acSAndroid Build Coastguard Worker 
776*1c60b9acSAndroid Build Coastguard Worker 	return LWS_HP_RET_BAIL_OK;
777*1c60b9acSAndroid Build Coastguard Worker }
778*1c60b9acSAndroid Build Coastguard Worker 
779*1c60b9acSAndroid Build Coastguard Worker static int
rops_write_role_protocol_h1(struct lws * wsi,unsigned char * buf,size_t len,enum lws_write_protocol * wp)780*1c60b9acSAndroid Build Coastguard Worker rops_write_role_protocol_h1(struct lws *wsi, unsigned char *buf, size_t len,
781*1c60b9acSAndroid Build Coastguard Worker 			    enum lws_write_protocol *wp)
782*1c60b9acSAndroid Build Coastguard Worker {
783*1c60b9acSAndroid Build Coastguard Worker 	size_t olen = len;
784*1c60b9acSAndroid Build Coastguard Worker 	int n;
785*1c60b9acSAndroid Build Coastguard Worker 
786*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_HTTP_STREAM_COMPRESSION)
787*1c60b9acSAndroid Build Coastguard Worker 	if (wsi->http.lcs && (((*wp) & 0x1f) == LWS_WRITE_HTTP_FINAL ||
788*1c60b9acSAndroid Build Coastguard Worker 			      ((*wp) & 0x1f) == LWS_WRITE_HTTP)) {
789*1c60b9acSAndroid Build Coastguard Worker 		unsigned char mtubuf[1500 + LWS_PRE +
790*1c60b9acSAndroid Build Coastguard Worker 				     LWS_HTTP_CHUNK_HDR_MAX_SIZE +
791*1c60b9acSAndroid Build Coastguard Worker 				     LWS_HTTP_CHUNK_TRL_MAX_SIZE],
792*1c60b9acSAndroid Build Coastguard Worker 			      *out = mtubuf + LWS_PRE +
793*1c60b9acSAndroid Build Coastguard Worker 				     LWS_HTTP_CHUNK_HDR_MAX_SIZE;
794*1c60b9acSAndroid Build Coastguard Worker 		size_t o = sizeof(mtubuf) - LWS_PRE -
795*1c60b9acSAndroid Build Coastguard Worker 			   LWS_HTTP_CHUNK_HDR_MAX_SIZE -
796*1c60b9acSAndroid Build Coastguard Worker 			   LWS_HTTP_CHUNK_TRL_MAX_SIZE;
797*1c60b9acSAndroid Build Coastguard Worker 
798*1c60b9acSAndroid Build Coastguard Worker 		n = lws_http_compression_transform(wsi, buf, len, wp, &out, &o);
799*1c60b9acSAndroid Build Coastguard Worker 		if (n)
800*1c60b9acSAndroid Build Coastguard Worker 			return n;
801*1c60b9acSAndroid Build Coastguard Worker 
802*1c60b9acSAndroid Build Coastguard Worker 		lwsl_info("%s: %s: transformed %d bytes to %d "
803*1c60b9acSAndroid Build Coastguard Worker 			   "(wp 0x%x, more %d)\n", __func__,
804*1c60b9acSAndroid Build Coastguard Worker 			   lws_wsi_tag(wsi), (int)len,
805*1c60b9acSAndroid Build Coastguard Worker 			   (int)o, (int)*wp, wsi->http.comp_ctx.may_have_more);
806*1c60b9acSAndroid Build Coastguard Worker 
807*1c60b9acSAndroid Build Coastguard Worker 		if (!o)
808*1c60b9acSAndroid Build Coastguard Worker 			return (int)olen;
809*1c60b9acSAndroid Build Coastguard Worker 
810*1c60b9acSAndroid Build Coastguard Worker 		if (wsi->http.comp_ctx.chunking) {
811*1c60b9acSAndroid Build Coastguard Worker 			char c[LWS_HTTP_CHUNK_HDR_MAX_SIZE + 2];
812*1c60b9acSAndroid Build Coastguard Worker 			/*
813*1c60b9acSAndroid Build Coastguard Worker 			 * this only needs dealing with on http/1.1 to allow
814*1c60b9acSAndroid Build Coastguard Worker 			 * pipelining
815*1c60b9acSAndroid Build Coastguard Worker 			 */
816*1c60b9acSAndroid Build Coastguard Worker 			n = lws_snprintf(c, sizeof(c), "%X\x0d\x0a", (int)o);
817*1c60b9acSAndroid Build Coastguard Worker 			lwsl_info("%s: chunk (%d) %s", __func__, (int)o, c);
818*1c60b9acSAndroid Build Coastguard Worker 			out -= n;
819*1c60b9acSAndroid Build Coastguard Worker 			o += (unsigned int)n;
820*1c60b9acSAndroid Build Coastguard Worker 			memcpy(out, c, (unsigned int)n);
821*1c60b9acSAndroid Build Coastguard Worker 			out[o++] = '\x0d';
822*1c60b9acSAndroid Build Coastguard Worker 			out[o++] = '\x0a';
823*1c60b9acSAndroid Build Coastguard Worker 
824*1c60b9acSAndroid Build Coastguard Worker 			if (((*wp) & 0x1f) == LWS_WRITE_HTTP_FINAL) {
825*1c60b9acSAndroid Build Coastguard Worker 				lwsl_info("%s: final chunk\n", __func__);
826*1c60b9acSAndroid Build Coastguard Worker 				out[o++] = '0';
827*1c60b9acSAndroid Build Coastguard Worker 				out[o++] = '\x0d';
828*1c60b9acSAndroid Build Coastguard Worker 				out[o++] = '\x0a';
829*1c60b9acSAndroid Build Coastguard Worker 				out[o++] = '\x0d';
830*1c60b9acSAndroid Build Coastguard Worker 				out[o++] = '\x0a';
831*1c60b9acSAndroid Build Coastguard Worker 			}
832*1c60b9acSAndroid Build Coastguard Worker 		}
833*1c60b9acSAndroid Build Coastguard Worker 
834*1c60b9acSAndroid Build Coastguard Worker 		buf = out;
835*1c60b9acSAndroid Build Coastguard Worker 		len = o;
836*1c60b9acSAndroid Build Coastguard Worker 	}
837*1c60b9acSAndroid Build Coastguard Worker #endif
838*1c60b9acSAndroid Build Coastguard Worker 
839*1c60b9acSAndroid Build Coastguard Worker 	n = lws_issue_raw(wsi, (unsigned char *)buf, len);
840*1c60b9acSAndroid Build Coastguard Worker 	if (n < 0)
841*1c60b9acSAndroid Build Coastguard Worker 		return n;
842*1c60b9acSAndroid Build Coastguard Worker 
843*1c60b9acSAndroid Build Coastguard Worker 	/* hide there may have been compression */
844*1c60b9acSAndroid Build Coastguard Worker 
845*1c60b9acSAndroid Build Coastguard Worker 	return (int)olen;
846*1c60b9acSAndroid Build Coastguard Worker }
847*1c60b9acSAndroid Build Coastguard Worker 
848*1c60b9acSAndroid Build Coastguard Worker static int
rops_alpn_negotiated_h1(struct lws * wsi,const char * alpn)849*1c60b9acSAndroid Build Coastguard Worker rops_alpn_negotiated_h1(struct lws *wsi, const char *alpn)
850*1c60b9acSAndroid Build Coastguard Worker {
851*1c60b9acSAndroid Build Coastguard Worker 	lwsl_debug("%s: client %d\n", __func__, lwsi_role_client(wsi));
852*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_CLIENT)
853*1c60b9acSAndroid Build Coastguard Worker 	if (lwsi_role_client(wsi)) {
854*1c60b9acSAndroid Build Coastguard Worker 		/*
855*1c60b9acSAndroid Build Coastguard Worker 		 * If alpn asserts it is http/1.1, server support for KA is
856*1c60b9acSAndroid Build Coastguard Worker 		 * mandatory.
857*1c60b9acSAndroid Build Coastguard Worker 		 *
858*1c60b9acSAndroid Build Coastguard Worker 		 * Knowing this lets us proceed with sending pipelined headers
859*1c60b9acSAndroid Build Coastguard Worker 		 * before we received the first response headers.
860*1c60b9acSAndroid Build Coastguard Worker 		 */
861*1c60b9acSAndroid Build Coastguard Worker 		wsi->keepalive_active = 1;
862*1c60b9acSAndroid Build Coastguard Worker 	}
863*1c60b9acSAndroid Build Coastguard Worker #endif
864*1c60b9acSAndroid Build Coastguard Worker 
865*1c60b9acSAndroid Build Coastguard Worker 	return 0;
866*1c60b9acSAndroid Build Coastguard Worker }
867*1c60b9acSAndroid Build Coastguard Worker 
868*1c60b9acSAndroid Build Coastguard Worker static int
rops_destroy_role_h1(struct lws * wsi)869*1c60b9acSAndroid Build Coastguard Worker rops_destroy_role_h1(struct lws *wsi)
870*1c60b9acSAndroid Build Coastguard Worker {
871*1c60b9acSAndroid Build Coastguard Worker 	struct lws_context_per_thread *pt = &wsi->a.context->pt[(int)wsi->tsi];
872*1c60b9acSAndroid Build Coastguard Worker 	struct allocated_headers *ah;
873*1c60b9acSAndroid Build Coastguard Worker 
874*1c60b9acSAndroid Build Coastguard Worker 	/* we may not have an ah, but may be on the waiting list... */
875*1c60b9acSAndroid Build Coastguard Worker 	lwsl_info("%s: ah det due to close\n", __func__);
876*1c60b9acSAndroid Build Coastguard Worker 	__lws_header_table_detach(wsi, 0);
877*1c60b9acSAndroid Build Coastguard Worker 
878*1c60b9acSAndroid Build Coastguard Worker 	 ah = pt->http.ah_list;
879*1c60b9acSAndroid Build Coastguard Worker 
880*1c60b9acSAndroid Build Coastguard Worker 	while (ah) {
881*1c60b9acSAndroid Build Coastguard Worker 		if (ah->in_use && ah->wsi == wsi) {
882*1c60b9acSAndroid Build Coastguard Worker 			lwsl_err("%s: ah leak: wsi %s\n", __func__,
883*1c60b9acSAndroid Build Coastguard Worker 					lws_wsi_tag(wsi));
884*1c60b9acSAndroid Build Coastguard Worker 			ah->in_use = 0;
885*1c60b9acSAndroid Build Coastguard Worker 			ah->wsi = NULL;
886*1c60b9acSAndroid Build Coastguard Worker 			pt->http.ah_count_in_use--;
887*1c60b9acSAndroid Build Coastguard Worker 			break;
888*1c60b9acSAndroid Build Coastguard Worker 		}
889*1c60b9acSAndroid Build Coastguard Worker 		ah = ah->next;
890*1c60b9acSAndroid Build Coastguard Worker 	}
891*1c60b9acSAndroid Build Coastguard Worker 
892*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_HTTP_STREAM_COMPRESSION)
893*1c60b9acSAndroid Build Coastguard Worker 	lws_http_compression_destroy(wsi);
894*1c60b9acSAndroid Build Coastguard Worker #endif
895*1c60b9acSAndroid Build Coastguard Worker 
896*1c60b9acSAndroid Build Coastguard Worker #ifdef LWS_ROLE_WS
897*1c60b9acSAndroid Build Coastguard Worker 	lws_free_set_NULL(wsi->ws);
898*1c60b9acSAndroid Build Coastguard Worker #endif
899*1c60b9acSAndroid Build Coastguard Worker 	return 0;
900*1c60b9acSAndroid Build Coastguard Worker }
901*1c60b9acSAndroid Build Coastguard Worker 
902*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SERVER)
903*1c60b9acSAndroid Build Coastguard Worker 
904*1c60b9acSAndroid Build Coastguard Worker static int
rops_adoption_bind_h1(struct lws * wsi,int type,const char * vh_prot_name)905*1c60b9acSAndroid Build Coastguard Worker rops_adoption_bind_h1(struct lws *wsi, int type, const char *vh_prot_name)
906*1c60b9acSAndroid Build Coastguard Worker {
907*1c60b9acSAndroid Build Coastguard Worker 	if (!(type & LWS_ADOPT_HTTP))
908*1c60b9acSAndroid Build Coastguard Worker 		return 0; /* no match */
909*1c60b9acSAndroid Build Coastguard Worker 
910*1c60b9acSAndroid Build Coastguard Worker 	if (type & _LWS_ADOPT_FINISH && !lwsi_role_http(wsi))
911*1c60b9acSAndroid Build Coastguard Worker 		return 0;
912*1c60b9acSAndroid Build Coastguard Worker 
913*1c60b9acSAndroid Build Coastguard Worker 	if (type & _LWS_ADOPT_FINISH) {
914*1c60b9acSAndroid Build Coastguard Worker 		if (!lws_header_table_attach(wsi, 0))
915*1c60b9acSAndroid Build Coastguard Worker 			lwsl_debug("Attached ah immediately\n");
916*1c60b9acSAndroid Build Coastguard Worker 		else
917*1c60b9acSAndroid Build Coastguard Worker 			lwsl_info("%s: waiting for ah\n", __func__);
918*1c60b9acSAndroid Build Coastguard Worker 
919*1c60b9acSAndroid Build Coastguard Worker 		return 1;
920*1c60b9acSAndroid Build Coastguard Worker 	}
921*1c60b9acSAndroid Build Coastguard Worker 
922*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SERVER) && defined(LWS_WITH_SECURE_STREAMS)
923*1c60b9acSAndroid Build Coastguard Worker 	if (wsi->a.vhost->ss_handle &&
924*1c60b9acSAndroid Build Coastguard Worker 	    wsi->a.vhost->ss_handle->policy->protocol == LWSSSP_RAW) {
925*1c60b9acSAndroid Build Coastguard Worker 		lws_role_transition(wsi, LWSIFR_SERVER, (type & LWS_ADOPT_ALLOW_SSL) ?
926*1c60b9acSAndroid Build Coastguard Worker 				LRS_SSL_INIT : LRS_ESTABLISHED, &role_ops_raw_skt);
927*1c60b9acSAndroid Build Coastguard Worker 		return 1;
928*1c60b9acSAndroid Build Coastguard Worker 	}
929*1c60b9acSAndroid Build Coastguard Worker #endif
930*1c60b9acSAndroid Build Coastguard Worker 
931*1c60b9acSAndroid Build Coastguard Worker 	/* If Non-TLS and HTTP2 prior knowledge is enabled, skip to clear text HTTP2 */
932*1c60b9acSAndroid Build Coastguard Worker 
933*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_HTTP2)
934*1c60b9acSAndroid Build Coastguard Worker 	if ((!(type & LWS_ADOPT_ALLOW_SSL)) && (wsi->a.vhost->options & LWS_SERVER_OPTION_H2_PRIOR_KNOWLEDGE)) {
935*1c60b9acSAndroid Build Coastguard Worker 		lwsl_info("http/2 prior knowledge\n");
936*1c60b9acSAndroid Build Coastguard Worker 		lws_metrics_tag_wsi_add(wsi, "upg", "h2_prior");
937*1c60b9acSAndroid Build Coastguard Worker 		lws_role_call_alpn_negotiated(wsi, "h2");
938*1c60b9acSAndroid Build Coastguard Worker 	}
939*1c60b9acSAndroid Build Coastguard Worker 	else
940*1c60b9acSAndroid Build Coastguard Worker #endif
941*1c60b9acSAndroid Build Coastguard Worker 		lws_role_transition(wsi, LWSIFR_SERVER, (type & LWS_ADOPT_ALLOW_SSL) ?
942*1c60b9acSAndroid Build Coastguard Worker 				LRS_SSL_INIT : LRS_HEADERS, &role_ops_h1);
943*1c60b9acSAndroid Build Coastguard Worker 
944*1c60b9acSAndroid Build Coastguard Worker 	/*
945*1c60b9acSAndroid Build Coastguard Worker 	 * Otherwise, we have to bind to h1 as a default even when we're actually going to
946*1c60b9acSAndroid Build Coastguard Worker 	 * replace it as an h2 bind later.  So don't take this seriously if the
947*1c60b9acSAndroid Build Coastguard Worker 	 * default is disabled (ws upgrade caees properly about it)
948*1c60b9acSAndroid Build Coastguard Worker 	 */
949*1c60b9acSAndroid Build Coastguard Worker 
950*1c60b9acSAndroid Build Coastguard Worker 	if (!vh_prot_name && wsi->a.vhost->default_protocol_index <
951*1c60b9acSAndroid Build Coastguard Worker 			     wsi->a.vhost->count_protocols)
952*1c60b9acSAndroid Build Coastguard Worker 		wsi->a.protocol = &wsi->a.vhost->protocols[
953*1c60b9acSAndroid Build Coastguard Worker 				wsi->a.vhost->default_protocol_index];
954*1c60b9acSAndroid Build Coastguard Worker 	else
955*1c60b9acSAndroid Build Coastguard Worker 		wsi->a.protocol = &wsi->a.vhost->protocols[0];
956*1c60b9acSAndroid Build Coastguard Worker 
957*1c60b9acSAndroid Build Coastguard Worker 	/* the transport is accepted... give him time to negotiate */
958*1c60b9acSAndroid Build Coastguard Worker 	lws_set_timeout(wsi, PENDING_TIMEOUT_ESTABLISH_WITH_SERVER,
959*1c60b9acSAndroid Build Coastguard Worker 			(int)wsi->a.context->timeout_secs);
960*1c60b9acSAndroid Build Coastguard Worker 
961*1c60b9acSAndroid Build Coastguard Worker 	return 1; /* bound */
962*1c60b9acSAndroid Build Coastguard Worker }
963*1c60b9acSAndroid Build Coastguard Worker 
964*1c60b9acSAndroid Build Coastguard Worker #endif
965*1c60b9acSAndroid Build Coastguard Worker 
966*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_CLIENT)
967*1c60b9acSAndroid Build Coastguard Worker 
968*1c60b9acSAndroid Build Coastguard Worker static const char * const http_methods[] = {
969*1c60b9acSAndroid Build Coastguard Worker 	"GET", "POST", "OPTIONS", "HEAD", "PUT", "PATCH", "DELETE", "CONNECT"
970*1c60b9acSAndroid Build Coastguard Worker };
971*1c60b9acSAndroid Build Coastguard Worker 
972*1c60b9acSAndroid Build Coastguard Worker static int
rops_client_bind_h1(struct lws * wsi,const struct lws_client_connect_info * i)973*1c60b9acSAndroid Build Coastguard Worker rops_client_bind_h1(struct lws *wsi, const struct lws_client_connect_info *i)
974*1c60b9acSAndroid Build Coastguard Worker {
975*1c60b9acSAndroid Build Coastguard Worker 	int n;
976*1c60b9acSAndroid Build Coastguard Worker 
977*1c60b9acSAndroid Build Coastguard Worker 	if (!i) {
978*1c60b9acSAndroid Build Coastguard Worker 		/* we are finalizing an already-selected role */
979*1c60b9acSAndroid Build Coastguard Worker 
980*1c60b9acSAndroid Build Coastguard Worker 		/*
981*1c60b9acSAndroid Build Coastguard Worker 		 * If we stay in http, assuming there wasn't already-set
982*1c60b9acSAndroid Build Coastguard Worker 		 * external user_space, since we know our initial protocol
983*1c60b9acSAndroid Build Coastguard Worker 		 * we can assign the user space now, otherwise do it after the
984*1c60b9acSAndroid Build Coastguard Worker 		 * ws subprotocol negotiated
985*1c60b9acSAndroid Build Coastguard Worker 		 */
986*1c60b9acSAndroid Build Coastguard Worker 		if (!wsi->user_space && wsi->stash->cis[CIS_METHOD])
987*1c60b9acSAndroid Build Coastguard Worker 			if (lws_ensure_user_space(wsi))
988*1c60b9acSAndroid Build Coastguard Worker 				return 1;
989*1c60b9acSAndroid Build Coastguard Worker 
990*1c60b9acSAndroid Build Coastguard Worker 		 /*
991*1c60b9acSAndroid Build Coastguard Worker 		  * For ws, default to http/1.1 only.  If i->alpn had been set
992*1c60b9acSAndroid Build Coastguard Worker 		  * though, defer to whatever he has set in there (eg, "h2").
993*1c60b9acSAndroid Build Coastguard Worker 		  *
994*1c60b9acSAndroid Build Coastguard Worker 		  * The problem is he has to commit to h2 before he can find
995*1c60b9acSAndroid Build Coastguard Worker 		  * out if the server has the SETTINGS for ws-over-h2 enabled;
996*1c60b9acSAndroid Build Coastguard Worker 		  * if not then ws is not possible on that connection.  So we
997*1c60b9acSAndroid Build Coastguard Worker 		  * only try h2 if he assertively said to use h2 alpn, otherwise
998*1c60b9acSAndroid Build Coastguard Worker 		  * ws implies alpn restriction to h1.
999*1c60b9acSAndroid Build Coastguard Worker 		  */
1000*1c60b9acSAndroid Build Coastguard Worker 		if (!wsi->stash->cis[CIS_METHOD] && !wsi->stash->cis[CIS_ALPN])
1001*1c60b9acSAndroid Build Coastguard Worker 			wsi->stash->cis[CIS_ALPN] = "http/1.1";
1002*1c60b9acSAndroid Build Coastguard Worker 
1003*1c60b9acSAndroid Build Coastguard Worker 		/* if we went on the ah waiting list, it's ok, we can wait.
1004*1c60b9acSAndroid Build Coastguard Worker 		 *
1005*1c60b9acSAndroid Build Coastguard Worker 		 * When we do get the ah, now or later, he will end up at
1006*1c60b9acSAndroid Build Coastguard Worker 		 * lws_http_client_connect_via_info2().
1007*1c60b9acSAndroid Build Coastguard Worker 		 */
1008*1c60b9acSAndroid Build Coastguard Worker 		if (lws_header_table_attach(wsi, 0)
1009*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_CLIENT)
1010*1c60b9acSAndroid Build Coastguard Worker 				< 0)
1011*1c60b9acSAndroid Build Coastguard Worker 			/*
1012*1c60b9acSAndroid Build Coastguard Worker 			 * if we failed here, the connection is already closed
1013*1c60b9acSAndroid Build Coastguard Worker 			 * and freed.
1014*1c60b9acSAndroid Build Coastguard Worker 			 */
1015*1c60b9acSAndroid Build Coastguard Worker 			return -1;
1016*1c60b9acSAndroid Build Coastguard Worker #else
1017*1c60b9acSAndroid Build Coastguard Worker 			)
1018*1c60b9acSAndroid Build Coastguard Worker 				return 0;
1019*1c60b9acSAndroid Build Coastguard Worker #endif
1020*1c60b9acSAndroid Build Coastguard Worker 
1021*1c60b9acSAndroid Build Coastguard Worker 		return 0;
1022*1c60b9acSAndroid Build Coastguard Worker 	}
1023*1c60b9acSAndroid Build Coastguard Worker 
1024*1c60b9acSAndroid Build Coastguard Worker 	/*
1025*1c60b9acSAndroid Build Coastguard Worker 	 * Clients that want to be h1, h2, or ws all start out as h1
1026*1c60b9acSAndroid Build Coastguard Worker 	 * (we don't yet know if the server supports h2 or ws), unless their
1027*1c60b9acSAndroid Build Coastguard Worker 	 * alpn is only "h2"
1028*1c60b9acSAndroid Build Coastguard Worker 	 */
1029*1c60b9acSAndroid Build Coastguard Worker 
1030*1c60b9acSAndroid Build Coastguard Worker //	if (i->alpn && !strcmp(i->alpn, "h2"))
1031*1c60b9acSAndroid Build Coastguard Worker //		return 0; /* we are h1, he only wants h2 */
1032*1c60b9acSAndroid Build Coastguard Worker 
1033*1c60b9acSAndroid Build Coastguard Worker 	if (!i->method) { /* websockets */
1034*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_ROLE_WS)
1035*1c60b9acSAndroid Build Coastguard Worker 		if (lws_create_client_ws_object(i, wsi))
1036*1c60b9acSAndroid Build Coastguard Worker 			goto fail_wsi;
1037*1c60b9acSAndroid Build Coastguard Worker 
1038*1c60b9acSAndroid Build Coastguard Worker 		goto bind_h1;
1039*1c60b9acSAndroid Build Coastguard Worker #else
1040*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: ws role not configured\n", __func__);
1041*1c60b9acSAndroid Build Coastguard Worker 
1042*1c60b9acSAndroid Build Coastguard Worker 		goto fail_wsi;
1043*1c60b9acSAndroid Build Coastguard Worker #endif
1044*1c60b9acSAndroid Build Coastguard Worker 	}
1045*1c60b9acSAndroid Build Coastguard Worker 
1046*1c60b9acSAndroid Build Coastguard Worker 	/* if a recognized http method, bind to it */
1047*1c60b9acSAndroid Build Coastguard Worker 
1048*1c60b9acSAndroid Build Coastguard Worker 	for (n = 0; n < (int)LWS_ARRAY_SIZE(http_methods); n++)
1049*1c60b9acSAndroid Build Coastguard Worker 		if (!strcmp(i->method, http_methods[n]))
1050*1c60b9acSAndroid Build Coastguard Worker 			goto bind_h1;
1051*1c60b9acSAndroid Build Coastguard Worker 
1052*1c60b9acSAndroid Build Coastguard Worker 	/* other roles may bind to it */
1053*1c60b9acSAndroid Build Coastguard Worker 
1054*1c60b9acSAndroid Build Coastguard Worker 	return 0; /* no match */
1055*1c60b9acSAndroid Build Coastguard Worker 
1056*1c60b9acSAndroid Build Coastguard Worker bind_h1:
1057*1c60b9acSAndroid Build Coastguard Worker 	/* assert the mode and union status (hdr) clearly */
1058*1c60b9acSAndroid Build Coastguard Worker 	lws_role_transition(wsi, LWSIFR_CLIENT, LRS_UNCONNECTED, &role_ops_h1);
1059*1c60b9acSAndroid Build Coastguard Worker 
1060*1c60b9acSAndroid Build Coastguard Worker 	return 1; /* matched */
1061*1c60b9acSAndroid Build Coastguard Worker 
1062*1c60b9acSAndroid Build Coastguard Worker fail_wsi:
1063*1c60b9acSAndroid Build Coastguard Worker 	return -1;
1064*1c60b9acSAndroid Build Coastguard Worker }
1065*1c60b9acSAndroid Build Coastguard Worker #endif
1066*1c60b9acSAndroid Build Coastguard Worker 
1067*1c60b9acSAndroid Build Coastguard Worker static int
rops_close_kill_connection_h1(struct lws * wsi,enum lws_close_status reason)1068*1c60b9acSAndroid Build Coastguard Worker rops_close_kill_connection_h1(struct lws *wsi, enum lws_close_status reason)
1069*1c60b9acSAndroid Build Coastguard Worker {
1070*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_HTTP_PROXY)
1071*1c60b9acSAndroid Build Coastguard Worker 	if (!wsi->http.proxy_clientside)
1072*1c60b9acSAndroid Build Coastguard Worker 		return 0;
1073*1c60b9acSAndroid Build Coastguard Worker 
1074*1c60b9acSAndroid Build Coastguard Worker 	wsi->http.proxy_clientside = 0;
1075*1c60b9acSAndroid Build Coastguard Worker 
1076*1c60b9acSAndroid Build Coastguard Worker 	if (user_callback_handle_rxflow(wsi->a.protocol->callback, wsi,
1077*1c60b9acSAndroid Build Coastguard Worker 					LWS_CALLBACK_COMPLETED_CLIENT_HTTP,
1078*1c60b9acSAndroid Build Coastguard Worker 					wsi->user_space, NULL, 0))
1079*1c60b9acSAndroid Build Coastguard Worker 		return 0;
1080*1c60b9acSAndroid Build Coastguard Worker #endif
1081*1c60b9acSAndroid Build Coastguard Worker 	return 0;
1082*1c60b9acSAndroid Build Coastguard Worker }
1083*1c60b9acSAndroid Build Coastguard Worker 
1084*1c60b9acSAndroid Build Coastguard Worker int
rops_pt_init_destroy_h1(struct lws_context * context,const struct lws_context_creation_info * info,struct lws_context_per_thread * pt,int destroy)1085*1c60b9acSAndroid Build Coastguard Worker rops_pt_init_destroy_h1(struct lws_context *context,
1086*1c60b9acSAndroid Build Coastguard Worker 		    const struct lws_context_creation_info *info,
1087*1c60b9acSAndroid Build Coastguard Worker 		    struct lws_context_per_thread *pt, int destroy)
1088*1c60b9acSAndroid Build Coastguard Worker {
1089*1c60b9acSAndroid Build Coastguard Worker 	/*
1090*1c60b9acSAndroid Build Coastguard Worker 	 * We only want to do this once... we will do it if no h2 support
1091*1c60b9acSAndroid Build Coastguard Worker 	 * otherwise let h2 ops do it.
1092*1c60b9acSAndroid Build Coastguard Worker 	 */
1093*1c60b9acSAndroid Build Coastguard Worker #if !defined(LWS_ROLE_H2) && defined(LWS_WITH_SERVER)
1094*1c60b9acSAndroid Build Coastguard Worker 	if (!destroy) {
1095*1c60b9acSAndroid Build Coastguard Worker 
1096*1c60b9acSAndroid Build Coastguard Worker 		pt->sul_ah_lifecheck.cb = lws_sul_http_ah_lifecheck;
1097*1c60b9acSAndroid Build Coastguard Worker 
1098*1c60b9acSAndroid Build Coastguard Worker 		__lws_sul_insert_us(&pt->pt_sul_owner[LWSSULLI_MISS_IF_SUSPENDED],
1099*1c60b9acSAndroid Build Coastguard Worker 				 &pt->sul_ah_lifecheck, 30 * LWS_US_PER_SEC);
1100*1c60b9acSAndroid Build Coastguard Worker 	} else
1101*1c60b9acSAndroid Build Coastguard Worker 		lws_dll2_remove(&pt->sul_ah_lifecheck.list);
1102*1c60b9acSAndroid Build Coastguard Worker #endif
1103*1c60b9acSAndroid Build Coastguard Worker 
1104*1c60b9acSAndroid Build Coastguard Worker 	return 0;
1105*1c60b9acSAndroid Build Coastguard Worker }
1106*1c60b9acSAndroid Build Coastguard Worker 
1107*1c60b9acSAndroid Build Coastguard Worker static const lws_rops_t rops_table_h1[] = {
1108*1c60b9acSAndroid Build Coastguard Worker 	/*  1 */ { .pt_init_destroy	  = rops_pt_init_destroy_h1 },
1109*1c60b9acSAndroid Build Coastguard Worker 	/*  2 */ { .handle_POLLIN	  = rops_handle_POLLIN_h1 },
1110*1c60b9acSAndroid Build Coastguard Worker 	/*  3 */ { .handle_POLLOUT	  = rops_handle_POLLOUT_h1 },
1111*1c60b9acSAndroid Build Coastguard Worker 	/*  4 */ { .write_role_protocol	  = rops_write_role_protocol_h1 },
1112*1c60b9acSAndroid Build Coastguard Worker 	/*  5 */ { .alpn_negotiated	  = rops_alpn_negotiated_h1 },
1113*1c60b9acSAndroid Build Coastguard Worker 	/*  6 */ { .close_kill_connection = rops_close_kill_connection_h1 },
1114*1c60b9acSAndroid Build Coastguard Worker 	/*  7 */ { .destroy_role	  = rops_destroy_role_h1 },
1115*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SERVER)
1116*1c60b9acSAndroid Build Coastguard Worker 	/*  8 */ { .adoption_bind	  = rops_adoption_bind_h1 },
1117*1c60b9acSAndroid Build Coastguard Worker #endif
1118*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_CLIENT)
1119*1c60b9acSAndroid Build Coastguard Worker 	/*  8 if client and no server */
1120*1c60b9acSAndroid Build Coastguard Worker 	/*  9 */ { .client_bind		  = rops_client_bind_h1 },
1121*1c60b9acSAndroid Build Coastguard Worker #endif
1122*1c60b9acSAndroid Build Coastguard Worker };
1123*1c60b9acSAndroid Build Coastguard Worker 
1124*1c60b9acSAndroid Build Coastguard Worker const struct lws_role_ops role_ops_h1 = {
1125*1c60b9acSAndroid Build Coastguard Worker 	/* role name */			"h1",
1126*1c60b9acSAndroid Build Coastguard Worker 	/* alpn id */			"http/1.1",
1127*1c60b9acSAndroid Build Coastguard Worker 	/* rops_table */		rops_table_h1,
1128*1c60b9acSAndroid Build Coastguard Worker 	/* rops_idx */			{
1129*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_check_upgrades */
1130*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_pt_init_destroy */		0x01,
1131*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_init_vhost */
1132*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_destroy_vhost */			0x00,
1133*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_service_flag_pending */
1134*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_handle_POLLIN */			0x02,
1135*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_handle_POLLOUT */
1136*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_perform_user_POLLOUT */		0x30,
1137*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_callback_on_writable */
1138*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_tx_credit */			0x00,
1139*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_write_role_protocol */
1140*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_encapsulation_parent */		0x40,
1141*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_alpn_negotiated */
1142*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_close_via_role_protocol */	0x50,
1143*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_close_role */
1144*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_close_kill_connection */		0x06,
1145*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_destroy_role */
1146*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SERVER)
1147*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_adoption_bind */			0x78,
1148*1c60b9acSAndroid Build Coastguard Worker #else
1149*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_adoption_bind */			0x70,
1150*1c60b9acSAndroid Build Coastguard Worker #endif
1151*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_client_bind */
1152*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_CLIENT)
1153*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SERVER)
1154*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_issue_keepalive */		0x90,
1155*1c60b9acSAndroid Build Coastguard Worker #else
1156*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_issue_keepalive */		0x80,
1157*1c60b9acSAndroid Build Coastguard Worker #endif
1158*1c60b9acSAndroid Build Coastguard Worker #else
1159*1c60b9acSAndroid Build Coastguard Worker 	  /* LWS_ROPS_issue_keepalive */		0x00,
1160*1c60b9acSAndroid Build Coastguard Worker #endif
1161*1c60b9acSAndroid Build Coastguard Worker 					},
1162*1c60b9acSAndroid Build Coastguard Worker 	/* adoption_cb clnt, srv */	{ LWS_CALLBACK_SERVER_NEW_CLIENT_INSTANTIATED,
1163*1c60b9acSAndroid Build Coastguard Worker 					  LWS_CALLBACK_SERVER_NEW_CLIENT_INSTANTIATED },
1164*1c60b9acSAndroid Build Coastguard Worker 	/* rx_cb clnt, srv */		{ LWS_CALLBACK_RECEIVE_CLIENT_HTTP,
1165*1c60b9acSAndroid Build Coastguard Worker 					  0 /* may be POST, etc */ },
1166*1c60b9acSAndroid Build Coastguard Worker 	/* writeable cb clnt, srv */	{ LWS_CALLBACK_CLIENT_HTTP_WRITEABLE,
1167*1c60b9acSAndroid Build Coastguard Worker 					  LWS_CALLBACK_HTTP_WRITEABLE },
1168*1c60b9acSAndroid Build Coastguard Worker 	/* close cb clnt, srv */	{ LWS_CALLBACK_CLOSED_CLIENT_HTTP,
1169*1c60b9acSAndroid Build Coastguard Worker 					  LWS_CALLBACK_CLOSED_HTTP },
1170*1c60b9acSAndroid Build Coastguard Worker 	/* protocol_bind cb c, srv */	{ LWS_CALLBACK_CLIENT_HTTP_BIND_PROTOCOL,
1171*1c60b9acSAndroid Build Coastguard Worker 					  LWS_CALLBACK_HTTP_BIND_PROTOCOL },
1172*1c60b9acSAndroid Build Coastguard Worker 	/* protocol_unbind cb c, srv */	{ LWS_CALLBACK_CLIENT_HTTP_DROP_PROTOCOL,
1173*1c60b9acSAndroid Build Coastguard Worker 					  LWS_CALLBACK_HTTP_DROP_PROTOCOL },
1174*1c60b9acSAndroid Build Coastguard Worker 	/* file_handle */		0,
1175*1c60b9acSAndroid Build Coastguard Worker };
1176