xref: /aosp_15_r20/external/libwebsockets/lib/plat/unix/unix-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 - 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 #if !defined(_GNU_SOURCE)
26*1c60b9acSAndroid Build Coastguard Worker #define _GNU_SOURCE
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 int
lws_poll_listen_fd(struct lws_pollfd * fd)31*1c60b9acSAndroid Build Coastguard Worker lws_poll_listen_fd(struct lws_pollfd *fd)
32*1c60b9acSAndroid Build Coastguard Worker {
33*1c60b9acSAndroid Build Coastguard Worker 	return poll(fd, 1, 0);
34*1c60b9acSAndroid Build Coastguard Worker }
35*1c60b9acSAndroid Build Coastguard Worker 
36*1c60b9acSAndroid Build Coastguard Worker int
_lws_plat_service_forced_tsi(struct lws_context * context,int tsi)37*1c60b9acSAndroid Build Coastguard Worker _lws_plat_service_forced_tsi(struct lws_context *context, int tsi)
38*1c60b9acSAndroid Build Coastguard Worker {
39*1c60b9acSAndroid Build Coastguard Worker 	struct lws_context_per_thread *pt = &context->pt[tsi];
40*1c60b9acSAndroid Build Coastguard Worker 	int m, n, r;
41*1c60b9acSAndroid Build Coastguard Worker 
42*1c60b9acSAndroid Build Coastguard Worker 	r = lws_service_flag_pending(context, tsi);
43*1c60b9acSAndroid Build Coastguard Worker 
44*1c60b9acSAndroid Build Coastguard Worker 	/* any socket with events to service? */
45*1c60b9acSAndroid Build Coastguard Worker 	for (n = 0; n < (int)pt->fds_count; n++) {
46*1c60b9acSAndroid Build Coastguard Worker 		lws_sockfd_type fd = pt->fds[n].fd;
47*1c60b9acSAndroid Build Coastguard Worker 
48*1c60b9acSAndroid Build Coastguard Worker 		if (!pt->fds[n].revents)
49*1c60b9acSAndroid Build Coastguard Worker 			continue;
50*1c60b9acSAndroid Build Coastguard Worker 
51*1c60b9acSAndroid Build Coastguard Worker 		m = lws_service_fd_tsi(context, &pt->fds[n], tsi);
52*1c60b9acSAndroid Build Coastguard Worker 		if (m < 0) {
53*1c60b9acSAndroid Build Coastguard Worker 			lwsl_err("%s: lws_service_fd_tsi returned %d\n",
54*1c60b9acSAndroid Build Coastguard Worker 				 __func__, m);
55*1c60b9acSAndroid Build Coastguard Worker 			return -1;
56*1c60b9acSAndroid Build Coastguard Worker 		}
57*1c60b9acSAndroid Build Coastguard Worker 
58*1c60b9acSAndroid Build Coastguard Worker 		/* if something closed, retry this slot since may have been
59*1c60b9acSAndroid Build Coastguard Worker 		 * swapped with end fd */
60*1c60b9acSAndroid Build Coastguard Worker 		if (m && pt->fds[n].fd != fd)
61*1c60b9acSAndroid Build Coastguard Worker 			n--;
62*1c60b9acSAndroid Build Coastguard Worker 	}
63*1c60b9acSAndroid Build Coastguard Worker 
64*1c60b9acSAndroid Build Coastguard Worker 	lws_service_do_ripe_rxflow(pt);
65*1c60b9acSAndroid Build Coastguard Worker 
66*1c60b9acSAndroid Build Coastguard Worker 	return r;
67*1c60b9acSAndroid Build Coastguard Worker }
68*1c60b9acSAndroid Build Coastguard Worker 
69*1c60b9acSAndroid Build Coastguard Worker #define LWS_POLL_WAIT_LIMIT 2000000000
70*1c60b9acSAndroid Build Coastguard Worker 
71*1c60b9acSAndroid Build Coastguard Worker int
_lws_plat_service_tsi(struct lws_context * context,int timeout_ms,int tsi)72*1c60b9acSAndroid Build Coastguard Worker _lws_plat_service_tsi(struct lws_context *context, int timeout_ms, int tsi)
73*1c60b9acSAndroid Build Coastguard Worker {
74*1c60b9acSAndroid Build Coastguard Worker 	volatile struct lws_foreign_thread_pollfd *ftp, *next;
75*1c60b9acSAndroid Build Coastguard Worker 	volatile struct lws_context_per_thread *vpt;
76*1c60b9acSAndroid Build Coastguard Worker 	struct lws_context_per_thread *pt;
77*1c60b9acSAndroid Build Coastguard Worker 	lws_usec_t timeout_us, us;
78*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SYS_METRICS)
79*1c60b9acSAndroid Build Coastguard Worker 	lws_usec_t a, b;
80*1c60b9acSAndroid Build Coastguard Worker #endif
81*1c60b9acSAndroid Build Coastguard Worker 	int n;
82*1c60b9acSAndroid Build Coastguard Worker #if (defined(LWS_ROLE_WS) && !defined(LWS_WITHOUT_EXTENSIONS)) || defined(LWS_WITH_TLS)
83*1c60b9acSAndroid Build Coastguard Worker 	int m;
84*1c60b9acSAndroid Build Coastguard Worker #endif
85*1c60b9acSAndroid Build Coastguard Worker 
86*1c60b9acSAndroid Build Coastguard Worker 	/* stay dead once we are dead */
87*1c60b9acSAndroid Build Coastguard Worker 
88*1c60b9acSAndroid Build Coastguard Worker 	if (!context)
89*1c60b9acSAndroid Build Coastguard Worker 		return 1;
90*1c60b9acSAndroid Build Coastguard Worker 
91*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SYS_METRICS)
92*1c60b9acSAndroid Build Coastguard Worker 	b =
93*1c60b9acSAndroid Build Coastguard Worker #endif
94*1c60b9acSAndroid Build Coastguard Worker 			us = lws_now_usecs();
95*1c60b9acSAndroid Build Coastguard Worker 
96*1c60b9acSAndroid Build Coastguard Worker 	pt = &context->pt[tsi];
97*1c60b9acSAndroid Build Coastguard Worker 	vpt = (volatile struct lws_context_per_thread *)pt;
98*1c60b9acSAndroid Build Coastguard Worker 
99*1c60b9acSAndroid Build Coastguard Worker 	if (timeout_ms < 0)
100*1c60b9acSAndroid Build Coastguard Worker 		timeout_ms = 0;
101*1c60b9acSAndroid Build Coastguard Worker 	else
102*1c60b9acSAndroid Build Coastguard Worker 		/* force a default timeout of 23 days */
103*1c60b9acSAndroid Build Coastguard Worker 		timeout_ms = LWS_POLL_WAIT_LIMIT;
104*1c60b9acSAndroid Build Coastguard Worker 	timeout_us = ((lws_usec_t)timeout_ms) * LWS_US_PER_MS;
105*1c60b9acSAndroid Build Coastguard Worker 
106*1c60b9acSAndroid Build Coastguard Worker 	if (context->event_loop_ops->run_pt)
107*1c60b9acSAndroid Build Coastguard Worker 		context->event_loop_ops->run_pt(context, tsi);
108*1c60b9acSAndroid Build Coastguard Worker 
109*1c60b9acSAndroid Build Coastguard Worker 	if (!pt->service_tid_detected && context->vhost_list) {
110*1c60b9acSAndroid Build Coastguard Worker 		lws_fakewsi_def_plwsa(pt);
111*1c60b9acSAndroid Build Coastguard Worker 
112*1c60b9acSAndroid Build Coastguard Worker 		lws_fakewsi_prep_plwsa_ctx(context);
113*1c60b9acSAndroid Build Coastguard Worker 
114*1c60b9acSAndroid Build Coastguard Worker 		pt->service_tid = context->vhost_list->protocols[0].callback(
115*1c60b9acSAndroid Build Coastguard Worker 					(struct lws *)plwsa,
116*1c60b9acSAndroid Build Coastguard Worker 					LWS_CALLBACK_GET_THREAD_ID,
117*1c60b9acSAndroid Build Coastguard Worker 					NULL, NULL, 0);
118*1c60b9acSAndroid Build Coastguard Worker 		pt->service_tid_detected = 1;
119*1c60b9acSAndroid Build Coastguard Worker 	}
120*1c60b9acSAndroid Build Coastguard Worker 
121*1c60b9acSAndroid Build Coastguard Worker 	lws_pt_lock(pt, __func__);
122*1c60b9acSAndroid Build Coastguard Worker 	/*
123*1c60b9acSAndroid Build Coastguard Worker 	 * service ripe scheduled events, and limit wait to next expected one
124*1c60b9acSAndroid Build Coastguard Worker 	 */
125*1c60b9acSAndroid Build Coastguard Worker 	us = __lws_sul_service_ripe(pt->pt_sul_owner, LWS_COUNT_PT_SUL_OWNERS, us);
126*1c60b9acSAndroid Build Coastguard Worker 	if (us && us < timeout_us)
127*1c60b9acSAndroid Build Coastguard Worker 		/*
128*1c60b9acSAndroid Build Coastguard Worker 		 * If something wants zero wait, that's OK, but if the next sul
129*1c60b9acSAndroid Build Coastguard Worker 		 * coming ripe is an interval less than our wait resolution,
130*1c60b9acSAndroid Build Coastguard Worker 		 * bump it to be the wait resolution.
131*1c60b9acSAndroid Build Coastguard Worker 		 */
132*1c60b9acSAndroid Build Coastguard Worker 		timeout_us = us < context->us_wait_resolution ?
133*1c60b9acSAndroid Build Coastguard Worker 					context->us_wait_resolution : us;
134*1c60b9acSAndroid Build Coastguard Worker 
135*1c60b9acSAndroid Build Coastguard Worker 	lws_pt_unlock(pt);
136*1c60b9acSAndroid Build Coastguard Worker 
137*1c60b9acSAndroid Build Coastguard Worker 	/*
138*1c60b9acSAndroid Build Coastguard Worker 	 * is there anybody with pending stuff that needs service forcing?
139*1c60b9acSAndroid Build Coastguard Worker 	 */
140*1c60b9acSAndroid Build Coastguard Worker 	if (!lws_service_adjust_timeout(context, 1, tsi))
141*1c60b9acSAndroid Build Coastguard Worker 		timeout_us = 0;
142*1c60b9acSAndroid Build Coastguard Worker 
143*1c60b9acSAndroid Build Coastguard Worker 	/* ensure we don't wrap at 2^31 with poll()'s signed int ms */
144*1c60b9acSAndroid Build Coastguard Worker 
145*1c60b9acSAndroid Build Coastguard Worker 	timeout_us /= LWS_US_PER_MS; /* ms now */
146*1c60b9acSAndroid Build Coastguard Worker 
147*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SYS_METRICS)
148*1c60b9acSAndroid Build Coastguard Worker 	a = lws_now_usecs() - b;
149*1c60b9acSAndroid Build Coastguard Worker #endif
150*1c60b9acSAndroid Build Coastguard Worker 	vpt->inside_poll = 1;
151*1c60b9acSAndroid Build Coastguard Worker 	lws_memory_barrier();
152*1c60b9acSAndroid Build Coastguard Worker 	n = poll(pt->fds, pt->fds_count, (int)timeout_us /* ms now */ );
153*1c60b9acSAndroid Build Coastguard Worker 	vpt->inside_poll = 0;
154*1c60b9acSAndroid Build Coastguard Worker 	lws_memory_barrier();
155*1c60b9acSAndroid Build Coastguard Worker 
156*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SYS_METRICS)
157*1c60b9acSAndroid Build Coastguard Worker 	b = lws_now_usecs();
158*1c60b9acSAndroid Build Coastguard Worker #endif
159*1c60b9acSAndroid Build Coastguard Worker 	/* Collision will be rare and brief.  Spin until it completes */
160*1c60b9acSAndroid Build Coastguard Worker 	while (vpt->foreign_spinlock)
161*1c60b9acSAndroid Build Coastguard Worker 		;
162*1c60b9acSAndroid Build Coastguard Worker 
163*1c60b9acSAndroid Build Coastguard Worker 	/*
164*1c60b9acSAndroid Build Coastguard Worker 	 * At this point we are not inside a foreign thread pollfd
165*1c60b9acSAndroid Build Coastguard Worker 	 * change, and we have marked ourselves as outside the poll()
166*1c60b9acSAndroid Build Coastguard Worker 	 * wait.  So we are the only guys that can modify the
167*1c60b9acSAndroid Build Coastguard Worker 	 * lws_foreign_thread_pollfd list on the pt.  Drain the list
168*1c60b9acSAndroid Build Coastguard Worker 	 * and apply the changes to the affected pollfds in the correct
169*1c60b9acSAndroid Build Coastguard Worker 	 * order.
170*1c60b9acSAndroid Build Coastguard Worker 	 */
171*1c60b9acSAndroid Build Coastguard Worker 
172*1c60b9acSAndroid Build Coastguard Worker 	lws_pt_lock(pt, __func__);
173*1c60b9acSAndroid Build Coastguard Worker 
174*1c60b9acSAndroid Build Coastguard Worker 	ftp = vpt->foreign_pfd_list;
175*1c60b9acSAndroid Build Coastguard Worker 	//lwsl_notice("cleared list %p\n", ftp);
176*1c60b9acSAndroid Build Coastguard Worker 	while (ftp) {
177*1c60b9acSAndroid Build Coastguard Worker 		struct lws *wsi;
178*1c60b9acSAndroid Build Coastguard Worker 		struct lws_pollfd *pfd;
179*1c60b9acSAndroid Build Coastguard Worker 
180*1c60b9acSAndroid Build Coastguard Worker 		next = ftp->next;
181*1c60b9acSAndroid Build Coastguard Worker 		pfd = &vpt->fds[ftp->fd_index];
182*1c60b9acSAndroid Build Coastguard Worker 		if (lws_socket_is_valid(pfd->fd)) {
183*1c60b9acSAndroid Build Coastguard Worker 			wsi = wsi_from_fd(context, pfd->fd);
184*1c60b9acSAndroid Build Coastguard Worker 			if (wsi)
185*1c60b9acSAndroid Build Coastguard Worker 				__lws_change_pollfd(wsi, ftp->_and,
186*1c60b9acSAndroid Build Coastguard Worker 						    ftp->_or);
187*1c60b9acSAndroid Build Coastguard Worker 		}
188*1c60b9acSAndroid Build Coastguard Worker 		lws_free((void *)ftp);
189*1c60b9acSAndroid Build Coastguard Worker 		ftp = next;
190*1c60b9acSAndroid Build Coastguard Worker 	}
191*1c60b9acSAndroid Build Coastguard Worker 	vpt->foreign_pfd_list = NULL;
192*1c60b9acSAndroid Build Coastguard Worker 	lws_memory_barrier();
193*1c60b9acSAndroid Build Coastguard Worker 
194*1c60b9acSAndroid Build Coastguard Worker 	lws_pt_unlock(pt);
195*1c60b9acSAndroid Build Coastguard Worker 
196*1c60b9acSAndroid Build Coastguard Worker #if (defined(LWS_ROLE_WS) && !defined(LWS_WITHOUT_EXTENSIONS)) || defined(LWS_WITH_TLS)
197*1c60b9acSAndroid Build Coastguard Worker 	m = 0;
198*1c60b9acSAndroid Build Coastguard Worker #endif
199*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_ROLE_WS) && !defined(LWS_WITHOUT_EXTENSIONS)
200*1c60b9acSAndroid Build Coastguard Worker 	m |= !!pt->ws.rx_draining_ext_list;
201*1c60b9acSAndroid Build Coastguard Worker #endif
202*1c60b9acSAndroid Build Coastguard Worker 
203*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_TLS)
204*1c60b9acSAndroid Build Coastguard Worker 	if (pt->context->tls_ops &&
205*1c60b9acSAndroid Build Coastguard Worker 	    pt->context->tls_ops->fake_POLLIN_for_buffered)
206*1c60b9acSAndroid Build Coastguard Worker 		m |= pt->context->tls_ops->fake_POLLIN_for_buffered(pt);
207*1c60b9acSAndroid Build Coastguard Worker #endif
208*1c60b9acSAndroid Build Coastguard Worker 
209*1c60b9acSAndroid Build Coastguard Worker 	if (
210*1c60b9acSAndroid Build Coastguard Worker #if (defined(LWS_ROLE_WS) && !defined(LWS_WITHOUT_EXTENSIONS)) || defined(LWS_WITH_TLS)
211*1c60b9acSAndroid Build Coastguard Worker 		!m &&
212*1c60b9acSAndroid Build Coastguard Worker #endif
213*1c60b9acSAndroid Build Coastguard Worker 		!n) /* nothing to do */
214*1c60b9acSAndroid Build Coastguard Worker 		lws_service_do_ripe_rxflow(pt);
215*1c60b9acSAndroid Build Coastguard Worker 	else
216*1c60b9acSAndroid Build Coastguard Worker 		if (_lws_plat_service_forced_tsi(context, tsi) < 0)
217*1c60b9acSAndroid Build Coastguard Worker 			return -1;
218*1c60b9acSAndroid Build Coastguard Worker 
219*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_SYS_METRICS)
220*1c60b9acSAndroid Build Coastguard Worker 	lws_metric_event(context->mt_service, METRES_GO,
221*1c60b9acSAndroid Build Coastguard Worker 			 (u_mt_t) (a + (lws_now_usecs() - b)));
222*1c60b9acSAndroid Build Coastguard Worker #endif
223*1c60b9acSAndroid Build Coastguard Worker 
224*1c60b9acSAndroid Build Coastguard Worker 	if (pt->destroy_self) {
225*1c60b9acSAndroid Build Coastguard Worker 		lws_context_destroy(pt->context);
226*1c60b9acSAndroid Build Coastguard Worker 		return -1;
227*1c60b9acSAndroid Build Coastguard Worker 	}
228*1c60b9acSAndroid Build Coastguard Worker 
229*1c60b9acSAndroid Build Coastguard Worker 	return 0;
230*1c60b9acSAndroid Build Coastguard Worker }
231*1c60b9acSAndroid Build Coastguard Worker 
232*1c60b9acSAndroid Build Coastguard Worker int
lws_plat_service(struct lws_context * context,int timeout_ms)233*1c60b9acSAndroid Build Coastguard Worker lws_plat_service(struct lws_context *context, int timeout_ms)
234*1c60b9acSAndroid Build Coastguard Worker {
235*1c60b9acSAndroid Build Coastguard Worker 	return _lws_plat_service_tsi(context, timeout_ms, 0);
236*1c60b9acSAndroid Build Coastguard Worker }
237