xref: /aosp_15_r20/external/libwebsockets/lib/plat/windows/windows-service.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 - 2021 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 #ifndef _WINSOCK_DEPRECATED_NO_WARNINGS
26*1c60b9acSAndroid Build Coastguard Worker #define _WINSOCK_DEPRECATED_NO_WARNINGS
27*1c60b9acSAndroid Build Coastguard Worker #endif
28*1c60b9acSAndroid Build Coastguard Worker #include "private-lib-core.h"
29*1c60b9acSAndroid Build Coastguard Worker 
30*1c60b9acSAndroid Build Coastguard Worker 
31*1c60b9acSAndroid Build Coastguard Worker int
_lws_plat_service_forced_tsi(struct lws_context * context,int tsi)32*1c60b9acSAndroid Build Coastguard Worker _lws_plat_service_forced_tsi(struct lws_context *context, int tsi)
33*1c60b9acSAndroid Build Coastguard Worker {
34*1c60b9acSAndroid Build Coastguard Worker 	struct lws_context_per_thread *pt = &context->pt[tsi];
35*1c60b9acSAndroid Build Coastguard Worker 	int m, n, r;
36*1c60b9acSAndroid Build Coastguard Worker 
37*1c60b9acSAndroid Build Coastguard Worker 	r = lws_service_flag_pending(context, tsi);
38*1c60b9acSAndroid Build Coastguard Worker 
39*1c60b9acSAndroid Build Coastguard Worker 	/* any socket with events to service? */
40*1c60b9acSAndroid Build Coastguard Worker 	for (n = 0; n < (int)pt->fds_count; n++) {
41*1c60b9acSAndroid Build Coastguard Worker 		if (!pt->fds[n].revents)
42*1c60b9acSAndroid Build Coastguard Worker 			continue;
43*1c60b9acSAndroid Build Coastguard Worker 
44*1c60b9acSAndroid Build Coastguard Worker 		m = lws_service_fd_tsi(context, &pt->fds[n], tsi);
45*1c60b9acSAndroid Build Coastguard Worker 		if (m < 0)
46*1c60b9acSAndroid Build Coastguard Worker 			return -1;
47*1c60b9acSAndroid Build Coastguard Worker 		/* if something closed, retry this slot */
48*1c60b9acSAndroid Build Coastguard Worker 		if (m)
49*1c60b9acSAndroid Build Coastguard Worker 			n--;
50*1c60b9acSAndroid Build Coastguard Worker 	}
51*1c60b9acSAndroid Build Coastguard Worker 
52*1c60b9acSAndroid Build Coastguard Worker 	lws_service_do_ripe_rxflow(pt);
53*1c60b9acSAndroid Build Coastguard Worker 
54*1c60b9acSAndroid Build Coastguard Worker 	return r;
55*1c60b9acSAndroid Build Coastguard Worker }
56*1c60b9acSAndroid Build Coastguard Worker 
57*1c60b9acSAndroid Build Coastguard Worker extern void lws_client_conn_wait_timeout(lws_sorted_usec_list_t *sul);
58*1c60b9acSAndroid Build Coastguard Worker 
59*1c60b9acSAndroid Build Coastguard Worker int
_lws_plat_service_tsi(struct lws_context * context,int timeout_ms,int tsi)60*1c60b9acSAndroid Build Coastguard Worker _lws_plat_service_tsi(struct lws_context *context, int timeout_ms, int tsi)
61*1c60b9acSAndroid Build Coastguard Worker {
62*1c60b9acSAndroid Build Coastguard Worker 	struct lws_context_per_thread *pt;
63*1c60b9acSAndroid Build Coastguard Worker 	struct lws_pollfd *pfd;
64*1c60b9acSAndroid Build Coastguard Worker 	lws_usec_t timeout_us;
65*1c60b9acSAndroid Build Coastguard Worker 	struct lws *wsi;
66*1c60b9acSAndroid Build Coastguard Worker 	unsigned int i;
67*1c60b9acSAndroid Build Coastguard Worker 	int n;
68*1c60b9acSAndroid Build Coastguard Worker 
69*1c60b9acSAndroid Build Coastguard Worker 	/* stay dead once we are dead */
70*1c60b9acSAndroid Build Coastguard Worker 	if (context == NULL)
71*1c60b9acSAndroid Build Coastguard Worker 		return 1;
72*1c60b9acSAndroid Build Coastguard Worker 
73*1c60b9acSAndroid Build Coastguard Worker 	pt = &context->pt[tsi];
74*1c60b9acSAndroid Build Coastguard Worker 
75*1c60b9acSAndroid Build Coastguard Worker 	if (!pt->service_tid_detected && context->vhost_list) {
76*1c60b9acSAndroid Build Coastguard Worker 		lws_fakewsi_def_plwsa(pt);
77*1c60b9acSAndroid Build Coastguard Worker 
78*1c60b9acSAndroid Build Coastguard Worker 		lws_fakewsi_prep_plwsa_ctx(context);
79*1c60b9acSAndroid Build Coastguard Worker 
80*1c60b9acSAndroid Build Coastguard Worker 		pt->service_tid = context->vhost_list->
81*1c60b9acSAndroid Build Coastguard Worker 			protocols[0].callback((struct lws *)plwsa,
82*1c60b9acSAndroid Build Coastguard Worker 					LWS_CALLBACK_GET_THREAD_ID,
83*1c60b9acSAndroid Build Coastguard Worker 						  NULL, NULL, 0);
84*1c60b9acSAndroid Build Coastguard Worker 		pt->service_tid_detected = 1;
85*1c60b9acSAndroid Build Coastguard Worker 	}
86*1c60b9acSAndroid Build Coastguard Worker 
87*1c60b9acSAndroid Build Coastguard Worker 	if (timeout_ms < 0)
88*1c60b9acSAndroid Build Coastguard Worker 		timeout_ms = 0;
89*1c60b9acSAndroid Build Coastguard Worker 	else
90*1c60b9acSAndroid Build Coastguard Worker 		/* force a default timeout of 23 days */
91*1c60b9acSAndroid Build Coastguard Worker 		timeout_ms = 2000000000;
92*1c60b9acSAndroid Build Coastguard Worker 	timeout_us = ((lws_usec_t)timeout_ms) * LWS_US_PER_MS;
93*1c60b9acSAndroid Build Coastguard Worker 
94*1c60b9acSAndroid Build Coastguard Worker 	if (context->event_loop_ops->run_pt)
95*1c60b9acSAndroid Build Coastguard Worker 		context->event_loop_ops->run_pt(context, tsi);
96*1c60b9acSAndroid Build Coastguard Worker 
97*1c60b9acSAndroid Build Coastguard Worker 	for (i = 0; i < pt->fds_count; ++i) {
98*1c60b9acSAndroid Build Coastguard Worker 		pfd = &pt->fds[i];
99*1c60b9acSAndroid Build Coastguard Worker 
100*1c60b9acSAndroid Build Coastguard Worker 		if (!(pfd->events & LWS_POLLOUT))
101*1c60b9acSAndroid Build Coastguard Worker 			continue;
102*1c60b9acSAndroid Build Coastguard Worker 
103*1c60b9acSAndroid Build Coastguard Worker 		wsi = wsi_from_fd(context, pfd->fd);
104*1c60b9acSAndroid Build Coastguard Worker 		if (!wsi || wsi->listener)
105*1c60b9acSAndroid Build Coastguard Worker 			continue;
106*1c60b9acSAndroid Build Coastguard Worker 		if (wsi->sock_send_blocking)
107*1c60b9acSAndroid Build Coastguard Worker 			continue;
108*1c60b9acSAndroid Build Coastguard Worker 		pfd->revents = LWS_POLLOUT;
109*1c60b9acSAndroid Build Coastguard Worker 		n = lws_service_fd(context, pfd);
110*1c60b9acSAndroid Build Coastguard Worker 		if (n < 0)
111*1c60b9acSAndroid Build Coastguard Worker 			return -1;
112*1c60b9acSAndroid Build Coastguard Worker 
113*1c60b9acSAndroid Build Coastguard Worker 		/*
114*1c60b9acSAndroid Build Coastguard Worker 		 * Force WSAWaitForMultipleEvents() to check events
115*1c60b9acSAndroid Build Coastguard Worker 		 * and then return immediately.
116*1c60b9acSAndroid Build Coastguard Worker 		 */
117*1c60b9acSAndroid Build Coastguard Worker 		timeout_us = 0;
118*1c60b9acSAndroid Build Coastguard Worker 
119*1c60b9acSAndroid Build Coastguard Worker 		/* if something closed, retry this slot */
120*1c60b9acSAndroid Build Coastguard Worker 		if (n)
121*1c60b9acSAndroid Build Coastguard Worker 			i--;
122*1c60b9acSAndroid Build Coastguard Worker 	}
123*1c60b9acSAndroid Build Coastguard Worker 
124*1c60b9acSAndroid Build Coastguard Worker 	/*
125*1c60b9acSAndroid Build Coastguard Worker 	 * service pending callbacks and get maximum wait time
126*1c60b9acSAndroid Build Coastguard Worker 	 */
127*1c60b9acSAndroid Build Coastguard Worker 	{
128*1c60b9acSAndroid Build Coastguard Worker 		lws_usec_t us;
129*1c60b9acSAndroid Build Coastguard Worker 
130*1c60b9acSAndroid Build Coastguard Worker 		lws_pt_lock(pt, __func__);
131*1c60b9acSAndroid Build Coastguard Worker 		/* don't stay in poll wait longer than next hr timeout */
132*1c60b9acSAndroid Build Coastguard Worker 		us = __lws_sul_service_ripe(pt->pt_sul_owner,
133*1c60b9acSAndroid Build Coastguard Worker 					    LWS_COUNT_PT_SUL_OWNERS,
134*1c60b9acSAndroid Build Coastguard Worker 					    lws_now_usecs());
135*1c60b9acSAndroid Build Coastguard Worker 		if (us && us < timeout_us)
136*1c60b9acSAndroid Build Coastguard Worker 			/*
137*1c60b9acSAndroid Build Coastguard Worker 			 * If something wants zero wait, that's OK, but if the next sul
138*1c60b9acSAndroid Build Coastguard Worker 			 * coming ripe is an interval less than our wait resolution,
139*1c60b9acSAndroid Build Coastguard Worker 			 * bump it to be the wait resolution.
140*1c60b9acSAndroid Build Coastguard Worker 			 */
141*1c60b9acSAndroid Build Coastguard Worker 			timeout_us = us < context->us_wait_resolution ?
142*1c60b9acSAndroid Build Coastguard Worker 					context->us_wait_resolution : us;
143*1c60b9acSAndroid Build Coastguard Worker 
144*1c60b9acSAndroid Build Coastguard Worker 		lws_pt_unlock(pt);
145*1c60b9acSAndroid Build Coastguard Worker 	}
146*1c60b9acSAndroid Build Coastguard Worker 
147*1c60b9acSAndroid Build Coastguard Worker 	if (_lws_plat_service_forced_tsi(context, tsi))
148*1c60b9acSAndroid Build Coastguard Worker 		timeout_us = 0;
149*1c60b9acSAndroid Build Coastguard Worker 
150*1c60b9acSAndroid Build Coastguard Worker 	/*
151*1c60b9acSAndroid Build Coastguard Worker 	 * is there anybody with pending stuff that needs service forcing?
152*1c60b9acSAndroid Build Coastguard Worker 	 */
153*1c60b9acSAndroid Build Coastguard Worker 
154*1c60b9acSAndroid Build Coastguard Worker 	if (!lws_service_adjust_timeout(context, 1, tsi))
155*1c60b9acSAndroid Build Coastguard Worker 		timeout_us = 0;
156*1c60b9acSAndroid Build Coastguard Worker 
157*1c60b9acSAndroid Build Coastguard Worker //	lwsl_notice("%s: in %dms, count %d\n", __func__, (int)(timeout_us / 1000), pt->fds_count);
158*1c60b9acSAndroid Build Coastguard Worker //	for (n = 0; n < (int)pt->fds_count; n++)
159*1c60b9acSAndroid Build Coastguard Worker //		lwsl_notice("%s: fd %d ev 0x%x POLLIN %d, POLLOUT %d\n", __func__, (int)pt->fds[n].fd, (int)pt->fds[n].events, POLLIN, POLLOUT);
160*1c60b9acSAndroid Build Coastguard Worker 	int d = WSAPoll((WSAPOLLFD *)&pt->fds[0], pt->fds_count, (int)(timeout_us / LWS_US_PER_MS));
161*1c60b9acSAndroid Build Coastguard Worker 	if (d < 0) {
162*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: WSAPoll failed: count %d, err %d: %d\n", __func__, pt->fds_count, d, WSAGetLastError());
163*1c60b9acSAndroid Build Coastguard Worker 		return 0;
164*1c60b9acSAndroid Build Coastguard Worker 	}
165*1c60b9acSAndroid Build Coastguard Worker //	lwsl_notice("%s: out\n", __func__);
166*1c60b9acSAndroid Build Coastguard Worker 
167*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_TLS)
168*1c60b9acSAndroid Build Coastguard Worker 	if (pt->context->tls_ops &&
169*1c60b9acSAndroid Build Coastguard Worker 	    pt->context->tls_ops->fake_POLLIN_for_buffered)
170*1c60b9acSAndroid Build Coastguard Worker 		pt->context->tls_ops->fake_POLLIN_for_buffered(pt);
171*1c60b9acSAndroid Build Coastguard Worker #endif
172*1c60b9acSAndroid Build Coastguard Worker 
173*1c60b9acSAndroid Build Coastguard Worker 	for (n = 0; n < (int)pt->fds_count; n++)
174*1c60b9acSAndroid Build Coastguard Worker 		if (pt->fds[n].fd != LWS_SOCK_INVALID && pt->fds[n].revents) {
175*1c60b9acSAndroid Build Coastguard Worker //			lwsl_notice("%s: idx %d, revents 0x%x\n", __func__, n, pt->fds[n].revents);
176*1c60b9acSAndroid Build Coastguard Worker 			lws_service_fd_tsi(context, &pt->fds[n], tsi);
177*1c60b9acSAndroid Build Coastguard Worker 		}
178*1c60b9acSAndroid Build Coastguard Worker 
179*1c60b9acSAndroid Build Coastguard Worker 	return 0;
180*1c60b9acSAndroid Build Coastguard Worker }
181*1c60b9acSAndroid Build Coastguard Worker 
182*1c60b9acSAndroid Build Coastguard Worker int
lws_plat_service(struct lws_context * context,int timeout_ms)183*1c60b9acSAndroid Build Coastguard Worker lws_plat_service(struct lws_context *context, int timeout_ms)
184*1c60b9acSAndroid Build Coastguard Worker {
185*1c60b9acSAndroid Build Coastguard Worker 	return _lws_plat_service_tsi(context, timeout_ms, 0);
186*1c60b9acSAndroid Build Coastguard Worker }
187