xref: /aosp_15_r20/external/libwebsockets/lib/plat/freertos/freertos-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 #include "private-lib-core.h"
26*1c60b9acSAndroid Build Coastguard Worker 
27*1c60b9acSAndroid Build Coastguard Worker int
lws_plat_service(struct lws_context * context,int timeout_ms)28*1c60b9acSAndroid Build Coastguard Worker lws_plat_service(struct lws_context *context, int timeout_ms)
29*1c60b9acSAndroid Build Coastguard Worker {
30*1c60b9acSAndroid Build Coastguard Worker 	int n = _lws_plat_service_tsi(context, timeout_ms, 0);
31*1c60b9acSAndroid Build Coastguard Worker 
32*1c60b9acSAndroid Build Coastguard Worker #if !defined(LWS_AMAZON_RTOS)
33*1c60b9acSAndroid Build Coastguard Worker 	esp_task_wdt_reset();
34*1c60b9acSAndroid Build Coastguard Worker #endif
35*1c60b9acSAndroid Build Coastguard Worker 
36*1c60b9acSAndroid Build Coastguard Worker 	return n;
37*1c60b9acSAndroid Build Coastguard Worker }
38*1c60b9acSAndroid Build Coastguard Worker 
39*1c60b9acSAndroid Build Coastguard Worker 
40*1c60b9acSAndroid Build Coastguard Worker int
_lws_plat_service_tsi(struct lws_context * context,int timeout_ms,int tsi)41*1c60b9acSAndroid Build Coastguard Worker _lws_plat_service_tsi(struct lws_context *context, int timeout_ms, int tsi)
42*1c60b9acSAndroid Build Coastguard Worker {
43*1c60b9acSAndroid Build Coastguard Worker 	volatile struct lws_context_per_thread *vpt;
44*1c60b9acSAndroid Build Coastguard Worker 	struct lws_context_per_thread *pt;
45*1c60b9acSAndroid Build Coastguard Worker 	lws_usec_t timeout_us;
46*1c60b9acSAndroid Build Coastguard Worker 	int n = -1, m, c, a = 0;
47*1c60b9acSAndroid Build Coastguard Worker 
48*1c60b9acSAndroid Build Coastguard Worker 	/* stay dead once we are dead */
49*1c60b9acSAndroid Build Coastguard Worker 
50*1c60b9acSAndroid Build Coastguard Worker 	if (!context)
51*1c60b9acSAndroid Build Coastguard Worker 		return 1;
52*1c60b9acSAndroid Build Coastguard Worker 
53*1c60b9acSAndroid Build Coastguard Worker 	pt = &context->pt[tsi];
54*1c60b9acSAndroid Build Coastguard Worker 	vpt = (volatile struct lws_context_per_thread *)pt;
55*1c60b9acSAndroid Build Coastguard Worker 
56*1c60b9acSAndroid Build Coastguard Worker 	{
57*1c60b9acSAndroid Build Coastguard Worker 		unsigned long m = lws_now_secs();
58*1c60b9acSAndroid Build Coastguard Worker 
59*1c60b9acSAndroid Build Coastguard Worker 		if (m > context->time_last_state_dump) {
60*1c60b9acSAndroid Build Coastguard Worker 			context->time_last_state_dump = m;
61*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_ESP_PLATFORM)
62*1c60b9acSAndroid Build Coastguard Worker 			n = esp_get_free_heap_size();
63*1c60b9acSAndroid Build Coastguard Worker #else
64*1c60b9acSAndroid Build Coastguard Worker 			n = xPortGetFreeHeapSize();
65*1c60b9acSAndroid Build Coastguard Worker #endif
66*1c60b9acSAndroid Build Coastguard Worker 			if ((unsigned int)n != context->last_free_heap) {
67*1c60b9acSAndroid Build Coastguard Worker 				if ((unsigned int)n > context->last_free_heap)
68*1c60b9acSAndroid Build Coastguard Worker 					lwsl_debug(" heap :%ld (+%ld)\n",
69*1c60b9acSAndroid Build Coastguard Worker 						    (unsigned long)n,
70*1c60b9acSAndroid Build Coastguard Worker 						    (unsigned long)(n -
71*1c60b9acSAndroid Build Coastguard Worker 						      context->last_free_heap));
72*1c60b9acSAndroid Build Coastguard Worker 				else
73*1c60b9acSAndroid Build Coastguard Worker 					lwsl_debug(" heap :%ld (-%ld)\n",
74*1c60b9acSAndroid Build Coastguard Worker 						    (unsigned long)n,
75*1c60b9acSAndroid Build Coastguard Worker 						    (unsigned long)(
76*1c60b9acSAndroid Build Coastguard Worker 						      context->last_free_heap -
77*1c60b9acSAndroid Build Coastguard Worker 						      n));
78*1c60b9acSAndroid Build Coastguard Worker 				context->last_free_heap = n;
79*1c60b9acSAndroid Build Coastguard Worker 			}
80*1c60b9acSAndroid Build Coastguard Worker 		}
81*1c60b9acSAndroid Build Coastguard Worker 	}
82*1c60b9acSAndroid Build Coastguard Worker 
83*1c60b9acSAndroid Build Coastguard Worker 	if (timeout_ms < 0)
84*1c60b9acSAndroid Build Coastguard Worker 		timeout_ms = 0;
85*1c60b9acSAndroid Build Coastguard Worker 	else
86*1c60b9acSAndroid Build Coastguard Worker 		/* force a default timeout of 23 days */
87*1c60b9acSAndroid Build Coastguard Worker 		timeout_ms = 2000000000;
88*1c60b9acSAndroid Build Coastguard Worker 	timeout_us = ((lws_usec_t)timeout_ms) * LWS_US_PER_MS;
89*1c60b9acSAndroid Build Coastguard Worker 
90*1c60b9acSAndroid Build Coastguard Worker 	if (!pt->service_tid_detected && context->vhost_list) {
91*1c60b9acSAndroid Build Coastguard Worker 		lws_fakewsi_def_plwsa(pt);
92*1c60b9acSAndroid Build Coastguard Worker 
93*1c60b9acSAndroid Build Coastguard Worker 		lws_fakewsi_prep_plwsa_ctx(context);
94*1c60b9acSAndroid Build Coastguard Worker 
95*1c60b9acSAndroid Build Coastguard Worker 		pt->service_tid = context->vhost_list->protocols[0].callback(
96*1c60b9acSAndroid Build Coastguard Worker 			(struct lws *)plwsa, LWS_CALLBACK_GET_THREAD_ID,
97*1c60b9acSAndroid Build Coastguard Worker 			NULL, NULL, 0);
98*1c60b9acSAndroid Build Coastguard Worker 		pt->service_tid_detected = 1;
99*1c60b9acSAndroid Build Coastguard Worker 	}
100*1c60b9acSAndroid Build Coastguard Worker 
101*1c60b9acSAndroid Build Coastguard Worker 	/*
102*1c60b9acSAndroid Build Coastguard Worker 	 * is there anybody with pending stuff that needs service forcing?
103*1c60b9acSAndroid Build Coastguard Worker 	 */
104*1c60b9acSAndroid Build Coastguard Worker #if !defined(LWS_AMAZON_RTOS)
105*1c60b9acSAndroid Build Coastguard Worker again:
106*1c60b9acSAndroid Build Coastguard Worker #endif
107*1c60b9acSAndroid Build Coastguard Worker 	n = 0;
108*1c60b9acSAndroid Build Coastguard Worker 	if (lws_service_adjust_timeout(context, 1, tsi)) {
109*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_AMAZON_RTOS)
110*1c60b9acSAndroid Build Coastguard Worker again:
111*1c60b9acSAndroid Build Coastguard Worker #endif /* LWS_AMAZON_RTOS */
112*1c60b9acSAndroid Build Coastguard Worker 
113*1c60b9acSAndroid Build Coastguard Worker 		a = 0;
114*1c60b9acSAndroid Build Coastguard Worker 		if (timeout_us) {
115*1c60b9acSAndroid Build Coastguard Worker 			lws_usec_t us;
116*1c60b9acSAndroid Build Coastguard Worker 
117*1c60b9acSAndroid Build Coastguard Worker 			lws_pt_lock(pt, __func__);
118*1c60b9acSAndroid Build Coastguard Worker 			/* don't stay in poll wait longer than next hr timeout */
119*1c60b9acSAndroid Build Coastguard Worker 			us = __lws_sul_service_ripe(pt->pt_sul_owner,
120*1c60b9acSAndroid Build Coastguard Worker 						    LWS_COUNT_PT_SUL_OWNERS,
121*1c60b9acSAndroid Build Coastguard Worker 						    lws_now_usecs());
122*1c60b9acSAndroid Build Coastguard Worker 			if (us && us < timeout_us)
123*1c60b9acSAndroid Build Coastguard Worker 				timeout_us = us;
124*1c60b9acSAndroid Build Coastguard Worker 
125*1c60b9acSAndroid Build Coastguard Worker 			lws_pt_unlock(pt);
126*1c60b9acSAndroid Build Coastguard Worker 		}
127*1c60b9acSAndroid Build Coastguard Worker 
128*1c60b9acSAndroid Build Coastguard Worker 	//	n = poll(pt->fds, pt->fds_count, timeout_ms);
129*1c60b9acSAndroid Build Coastguard Worker 		{
130*1c60b9acSAndroid Build Coastguard Worker 			fd_set readfds, writefds, errfds;
131*1c60b9acSAndroid Build Coastguard Worker 			struct timeval tv = { timeout_us / LWS_US_PER_SEC,
132*1c60b9acSAndroid Build Coastguard Worker 					      timeout_us % LWS_US_PER_SEC }, *ptv = &tv;
133*1c60b9acSAndroid Build Coastguard Worker 			int max_fd = 0;
134*1c60b9acSAndroid Build Coastguard Worker 			FD_ZERO(&readfds);
135*1c60b9acSAndroid Build Coastguard Worker 			FD_ZERO(&writefds);
136*1c60b9acSAndroid Build Coastguard Worker 			FD_ZERO(&errfds);
137*1c60b9acSAndroid Build Coastguard Worker 
138*1c60b9acSAndroid Build Coastguard Worker 			for (n = 0; n < (int)pt->fds_count; n++) {
139*1c60b9acSAndroid Build Coastguard Worker 				pt->fds[n].revents = 0;
140*1c60b9acSAndroid Build Coastguard Worker 				if (pt->fds[n].fd >= max_fd)
141*1c60b9acSAndroid Build Coastguard Worker 					max_fd = pt->fds[n].fd;
142*1c60b9acSAndroid Build Coastguard Worker 				if (pt->fds[n].events & LWS_POLLIN)
143*1c60b9acSAndroid Build Coastguard Worker 					FD_SET(pt->fds[n].fd, &readfds);
144*1c60b9acSAndroid Build Coastguard Worker 				if (pt->fds[n].events & LWS_POLLOUT)
145*1c60b9acSAndroid Build Coastguard Worker 					FD_SET(pt->fds[n].fd, &writefds);
146*1c60b9acSAndroid Build Coastguard Worker 				FD_SET(pt->fds[n].fd, &errfds);
147*1c60b9acSAndroid Build Coastguard Worker 			}
148*1c60b9acSAndroid Build Coastguard Worker 
149*1c60b9acSAndroid Build Coastguard Worker 			vpt->inside_poll = 1;
150*1c60b9acSAndroid Build Coastguard Worker 			lws_memory_barrier();
151*1c60b9acSAndroid Build Coastguard Worker 			n = select(max_fd + 1, &readfds, &writefds, &errfds, ptv);
152*1c60b9acSAndroid Build Coastguard Worker 			vpt->inside_poll = 0;
153*1c60b9acSAndroid Build Coastguard Worker 			lws_memory_barrier();
154*1c60b9acSAndroid Build Coastguard Worker 			n = 0;
155*1c60b9acSAndroid Build Coastguard Worker 
156*1c60b9acSAndroid Build Coastguard Worker 			for (m = 0; m < (int)pt->fds_count; m++) {
157*1c60b9acSAndroid Build Coastguard Worker 				c = 0;
158*1c60b9acSAndroid Build Coastguard Worker 				if (FD_ISSET(pt->fds[m].fd, &readfds)) {
159*1c60b9acSAndroid Build Coastguard Worker 					pt->fds[m].revents |= LWS_POLLIN;
160*1c60b9acSAndroid Build Coastguard Worker 					c = 1;
161*1c60b9acSAndroid Build Coastguard Worker 				}
162*1c60b9acSAndroid Build Coastguard Worker 				if (FD_ISSET(pt->fds[m].fd, &writefds)) {
163*1c60b9acSAndroid Build Coastguard Worker 					pt->fds[m].revents |= LWS_POLLOUT;
164*1c60b9acSAndroid Build Coastguard Worker 					c = 1;
165*1c60b9acSAndroid Build Coastguard Worker 				}
166*1c60b9acSAndroid Build Coastguard Worker 				if (FD_ISSET(pt->fds[m].fd, &errfds)) {
167*1c60b9acSAndroid Build Coastguard Worker 					// lwsl_notice("errfds %d\n", pt->fds[m].fd);
168*1c60b9acSAndroid Build Coastguard Worker 					pt->fds[m].revents |= LWS_POLLHUP;
169*1c60b9acSAndroid Build Coastguard Worker 					c = 1;
170*1c60b9acSAndroid Build Coastguard Worker 				}
171*1c60b9acSAndroid Build Coastguard Worker 
172*1c60b9acSAndroid Build Coastguard Worker 				if (c)
173*1c60b9acSAndroid Build Coastguard Worker 					n++;
174*1c60b9acSAndroid Build Coastguard Worker 			}
175*1c60b9acSAndroid Build Coastguard Worker 		}
176*1c60b9acSAndroid Build Coastguard Worker 
177*1c60b9acSAndroid Build Coastguard Worker 		m = 0;
178*1c60b9acSAndroid Build Coastguard Worker 
179*1c60b9acSAndroid Build Coastguard Worker 	#if defined(LWS_ROLE_WS) && !defined(LWS_WITHOUT_EXTENSIONS)
180*1c60b9acSAndroid Build Coastguard Worker 		m |= !!pt->ws.rx_draining_ext_list;
181*1c60b9acSAndroid Build Coastguard Worker 	#endif
182*1c60b9acSAndroid Build Coastguard Worker 
183*1c60b9acSAndroid Build Coastguard Worker #if defined(LWS_WITH_TLS)
184*1c60b9acSAndroid Build Coastguard Worker 		if (pt->context->tls_ops &&
185*1c60b9acSAndroid Build Coastguard Worker 		    pt->context->tls_ops->fake_POLLIN_for_buffered)
186*1c60b9acSAndroid Build Coastguard Worker 			m |= pt->context->tls_ops->fake_POLLIN_for_buffered(pt);
187*1c60b9acSAndroid Build Coastguard Worker #endif
188*1c60b9acSAndroid Build Coastguard Worker 		if (!m && !n)
189*1c60b9acSAndroid Build Coastguard Worker 			return 0;
190*1c60b9acSAndroid Build Coastguard Worker 	} else
191*1c60b9acSAndroid Build Coastguard Worker 		a = 1;
192*1c60b9acSAndroid Build Coastguard Worker 
193*1c60b9acSAndroid Build Coastguard Worker 	m = lws_service_flag_pending(context, tsi);
194*1c60b9acSAndroid Build Coastguard Worker 	c = m ? -1 : n;
195*1c60b9acSAndroid Build Coastguard Worker 
196*1c60b9acSAndroid Build Coastguard Worker 	/* any socket with events to service? */
197*1c60b9acSAndroid Build Coastguard Worker 	for (n = 0; n < (int)pt->fds_count && c; n++) {
198*1c60b9acSAndroid Build Coastguard Worker 		if (!pt->fds[n].revents)
199*1c60b9acSAndroid Build Coastguard Worker 			continue;
200*1c60b9acSAndroid Build Coastguard Worker 
201*1c60b9acSAndroid Build Coastguard Worker 		c--;
202*1c60b9acSAndroid Build Coastguard Worker 
203*1c60b9acSAndroid Build Coastguard Worker 		m = lws_service_fd_tsi(context, &pt->fds[n], tsi);
204*1c60b9acSAndroid Build Coastguard Worker 		if (m < 0)
205*1c60b9acSAndroid Build Coastguard Worker 			return -1;
206*1c60b9acSAndroid Build Coastguard Worker 		/* if something closed, retry this slot */
207*1c60b9acSAndroid Build Coastguard Worker 		if (m)
208*1c60b9acSAndroid Build Coastguard Worker 			n--;
209*1c60b9acSAndroid Build Coastguard Worker 	}
210*1c60b9acSAndroid Build Coastguard Worker 
211*1c60b9acSAndroid Build Coastguard Worker 	if (a)
212*1c60b9acSAndroid Build Coastguard Worker 		goto again;
213*1c60b9acSAndroid Build Coastguard Worker 
214*1c60b9acSAndroid Build Coastguard Worker 	return 0;
215*1c60b9acSAndroid Build Coastguard Worker }
216