xref: /aosp_15_r20/external/libwebsockets/lib/misc/lwsac/lwsac.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 - 2020 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 #include "private-lib-misc-lwsac.h"
27*1c60b9acSAndroid Build Coastguard Worker 
28*1c60b9acSAndroid Build Coastguard Worker void
lws_list_ptr_insert(lws_list_ptr * head,lws_list_ptr * add,lws_list_ptr_sort_func_t sort_func)29*1c60b9acSAndroid Build Coastguard Worker lws_list_ptr_insert(lws_list_ptr *head, lws_list_ptr *add,
30*1c60b9acSAndroid Build Coastguard Worker 		    lws_list_ptr_sort_func_t sort_func)
31*1c60b9acSAndroid Build Coastguard Worker {
32*1c60b9acSAndroid Build Coastguard Worker 	while (sort_func && *head) {
33*1c60b9acSAndroid Build Coastguard Worker 		if (sort_func(add, *head) <= 0)
34*1c60b9acSAndroid Build Coastguard Worker 			break;
35*1c60b9acSAndroid Build Coastguard Worker 
36*1c60b9acSAndroid Build Coastguard Worker 		head = *head;
37*1c60b9acSAndroid Build Coastguard Worker 	}
38*1c60b9acSAndroid Build Coastguard Worker 
39*1c60b9acSAndroid Build Coastguard Worker 	*add = *head;
40*1c60b9acSAndroid Build Coastguard Worker 	*head = add;
41*1c60b9acSAndroid Build Coastguard Worker }
42*1c60b9acSAndroid Build Coastguard Worker 
43*1c60b9acSAndroid Build Coastguard Worker size_t
lwsac_align(size_t length)44*1c60b9acSAndroid Build Coastguard Worker lwsac_align(size_t length)
45*1c60b9acSAndroid Build Coastguard Worker {
46*1c60b9acSAndroid Build Coastguard Worker 	size_t align = sizeof(int *);
47*1c60b9acSAndroid Build Coastguard Worker 
48*1c60b9acSAndroid Build Coastguard Worker 	if (length & (align - 1))
49*1c60b9acSAndroid Build Coastguard Worker 		length += align - (length & (align - 1));
50*1c60b9acSAndroid Build Coastguard Worker 
51*1c60b9acSAndroid Build Coastguard Worker 	return length;
52*1c60b9acSAndroid Build Coastguard Worker }
53*1c60b9acSAndroid Build Coastguard Worker 
54*1c60b9acSAndroid Build Coastguard Worker size_t
lwsac_sizeof(int first)55*1c60b9acSAndroid Build Coastguard Worker lwsac_sizeof(int first)
56*1c60b9acSAndroid Build Coastguard Worker {
57*1c60b9acSAndroid Build Coastguard Worker 	return sizeof(struct lwsac) + (first ? sizeof(struct lwsac_head) : 0);
58*1c60b9acSAndroid Build Coastguard Worker }
59*1c60b9acSAndroid Build Coastguard Worker 
60*1c60b9acSAndroid Build Coastguard Worker size_t
lwsac_get_tail_pos(struct lwsac * lac)61*1c60b9acSAndroid Build Coastguard Worker lwsac_get_tail_pos(struct lwsac *lac)
62*1c60b9acSAndroid Build Coastguard Worker {
63*1c60b9acSAndroid Build Coastguard Worker 	return lac->ofs;
64*1c60b9acSAndroid Build Coastguard Worker }
65*1c60b9acSAndroid Build Coastguard Worker 
66*1c60b9acSAndroid Build Coastguard Worker struct lwsac *
lwsac_get_next(struct lwsac * lac)67*1c60b9acSAndroid Build Coastguard Worker lwsac_get_next(struct lwsac *lac)
68*1c60b9acSAndroid Build Coastguard Worker {
69*1c60b9acSAndroid Build Coastguard Worker 	return lac->next;
70*1c60b9acSAndroid Build Coastguard Worker }
71*1c60b9acSAndroid Build Coastguard Worker 
72*1c60b9acSAndroid Build Coastguard Worker int
lwsac_extend(struct lwsac * head,size_t amount)73*1c60b9acSAndroid Build Coastguard Worker lwsac_extend(struct lwsac *head, size_t amount)
74*1c60b9acSAndroid Build Coastguard Worker {
75*1c60b9acSAndroid Build Coastguard Worker 	struct lwsac_head *lachead;
76*1c60b9acSAndroid Build Coastguard Worker 	struct lwsac *bf;
77*1c60b9acSAndroid Build Coastguard Worker 
78*1c60b9acSAndroid Build Coastguard Worker 	assert(head);
79*1c60b9acSAndroid Build Coastguard Worker 	lachead = (struct lwsac_head *)&head[1];
80*1c60b9acSAndroid Build Coastguard Worker 
81*1c60b9acSAndroid Build Coastguard Worker 	bf = lachead->curr;
82*1c60b9acSAndroid Build Coastguard Worker 	assert(bf);
83*1c60b9acSAndroid Build Coastguard Worker 
84*1c60b9acSAndroid Build Coastguard Worker 	if (bf->alloc_size - bf->ofs < lwsac_align(amount))
85*1c60b9acSAndroid Build Coastguard Worker 		return 1;
86*1c60b9acSAndroid Build Coastguard Worker 
87*1c60b9acSAndroid Build Coastguard Worker 	/* memset so constant folding never sees uninitialized data */
88*1c60b9acSAndroid Build Coastguard Worker 
89*1c60b9acSAndroid Build Coastguard Worker 	memset(((uint8_t *)bf) + bf->ofs, 0, lwsac_align(amount));
90*1c60b9acSAndroid Build Coastguard Worker 	bf->ofs += lwsac_align(amount);
91*1c60b9acSAndroid Build Coastguard Worker 
92*1c60b9acSAndroid Build Coastguard Worker 	return 0;
93*1c60b9acSAndroid Build Coastguard Worker }
94*1c60b9acSAndroid Build Coastguard Worker 
95*1c60b9acSAndroid Build Coastguard Worker static void *
_lwsac_use(struct lwsac ** head,size_t ensure,size_t chunk_size,char backfill)96*1c60b9acSAndroid Build Coastguard Worker _lwsac_use(struct lwsac **head, size_t ensure, size_t chunk_size, char backfill)
97*1c60b9acSAndroid Build Coastguard Worker {
98*1c60b9acSAndroid Build Coastguard Worker 	struct lwsac_head *lachead = NULL;
99*1c60b9acSAndroid Build Coastguard Worker 	size_t ofs, alloc, al, hp;
100*1c60b9acSAndroid Build Coastguard Worker 	struct lwsac *bf = *head;
101*1c60b9acSAndroid Build Coastguard Worker 
102*1c60b9acSAndroid Build Coastguard Worker 	if (bf)
103*1c60b9acSAndroid Build Coastguard Worker 		lachead = (struct lwsac_head *)&bf[1];
104*1c60b9acSAndroid Build Coastguard Worker 
105*1c60b9acSAndroid Build Coastguard Worker 	al = lwsac_align(ensure);
106*1c60b9acSAndroid Build Coastguard Worker 
107*1c60b9acSAndroid Build Coastguard Worker 	/* backfill into earlier chunks if that is allowed */
108*1c60b9acSAndroid Build Coastguard Worker 
109*1c60b9acSAndroid Build Coastguard Worker 	if (backfill)
110*1c60b9acSAndroid Build Coastguard Worker 		/*
111*1c60b9acSAndroid Build Coastguard Worker 		 * check if anything can take it, from the start
112*1c60b9acSAndroid Build Coastguard Worker 		 */
113*1c60b9acSAndroid Build Coastguard Worker 		while (bf) {
114*1c60b9acSAndroid Build Coastguard Worker 			if (bf->alloc_size - bf->ofs >= ensure)
115*1c60b9acSAndroid Build Coastguard Worker 				goto do_use;
116*1c60b9acSAndroid Build Coastguard Worker 
117*1c60b9acSAndroid Build Coastguard Worker 			bf = bf->next;
118*1c60b9acSAndroid Build Coastguard Worker 		}
119*1c60b9acSAndroid Build Coastguard Worker 	else {
120*1c60b9acSAndroid Build Coastguard Worker 		/*
121*1c60b9acSAndroid Build Coastguard Worker 		 * If there's a current chunk, just check if he can take it
122*1c60b9acSAndroid Build Coastguard Worker 		 */
123*1c60b9acSAndroid Build Coastguard Worker 		if (lachead && lachead->curr) {
124*1c60b9acSAndroid Build Coastguard Worker 			bf = lachead->curr;
125*1c60b9acSAndroid Build Coastguard Worker 			if (bf->alloc_size - bf->ofs >= ensure)
126*1c60b9acSAndroid Build Coastguard Worker 				goto do_use;
127*1c60b9acSAndroid Build Coastguard Worker 		}
128*1c60b9acSAndroid Build Coastguard Worker 	}
129*1c60b9acSAndroid Build Coastguard Worker 
130*1c60b9acSAndroid Build Coastguard Worker 	/* nothing can currently take it... so we must allocate */
131*1c60b9acSAndroid Build Coastguard Worker 
132*1c60b9acSAndroid Build Coastguard Worker 	hp = sizeof(*bf); /* always need the normal header part... */
133*1c60b9acSAndroid Build Coastguard Worker 	if (!*head)
134*1c60b9acSAndroid Build Coastguard Worker 		hp += sizeof(struct lwsac_head);
135*1c60b9acSAndroid Build Coastguard Worker 
136*1c60b9acSAndroid Build Coastguard Worker 	if (!chunk_size)
137*1c60b9acSAndroid Build Coastguard Worker 		alloc = LWSAC_CHUNK_SIZE + hp;
138*1c60b9acSAndroid Build Coastguard Worker 	else
139*1c60b9acSAndroid Build Coastguard Worker 		alloc = chunk_size + hp;
140*1c60b9acSAndroid Build Coastguard Worker 
141*1c60b9acSAndroid Build Coastguard Worker 	/*
142*1c60b9acSAndroid Build Coastguard Worker 	 * If we get asked for something outside our expectation,
143*1c60b9acSAndroid Build Coastguard Worker 	 * increase the allocation to meet it
144*1c60b9acSAndroid Build Coastguard Worker 	 */
145*1c60b9acSAndroid Build Coastguard Worker 
146*1c60b9acSAndroid Build Coastguard Worker 	if (al >= alloc - hp)
147*1c60b9acSAndroid Build Coastguard Worker 		alloc = al + hp;
148*1c60b9acSAndroid Build Coastguard Worker 
149*1c60b9acSAndroid Build Coastguard Worker 	lwsl_debug("%s: alloc %d for %d\n", __func__, (int)alloc, (int)ensure);
150*1c60b9acSAndroid Build Coastguard Worker 	bf = malloc(alloc);
151*1c60b9acSAndroid Build Coastguard Worker 	if (!bf) {
152*1c60b9acSAndroid Build Coastguard Worker 		lwsl_err("%s: OOM trying to alloc %llud\n", __func__,
153*1c60b9acSAndroid Build Coastguard Worker 				(unsigned long long)alloc);
154*1c60b9acSAndroid Build Coastguard Worker 		return NULL;
155*1c60b9acSAndroid Build Coastguard Worker 	}
156*1c60b9acSAndroid Build Coastguard Worker 
157*1c60b9acSAndroid Build Coastguard Worker 	/*
158*1c60b9acSAndroid Build Coastguard Worker 	 * belabouring the point... ofs is aligned to the platform's
159*1c60b9acSAndroid Build Coastguard Worker 	 * generic struct alignment at the start then
160*1c60b9acSAndroid Build Coastguard Worker 	 */
161*1c60b9acSAndroid Build Coastguard Worker 	bf->ofs = sizeof(*bf);
162*1c60b9acSAndroid Build Coastguard Worker 
163*1c60b9acSAndroid Build Coastguard Worker 	if (!*head) {
164*1c60b9acSAndroid Build Coastguard Worker 		/*
165*1c60b9acSAndroid Build Coastguard Worker 		 * We are the first, head, entry...
166*1c60b9acSAndroid Build Coastguard Worker 		 */
167*1c60b9acSAndroid Build Coastguard Worker 		*head = bf;
168*1c60b9acSAndroid Build Coastguard Worker 		/*
169*1c60b9acSAndroid Build Coastguard Worker 		 * ... allocate for the special head block
170*1c60b9acSAndroid Build Coastguard Worker 		 */
171*1c60b9acSAndroid Build Coastguard Worker 		bf->ofs += sizeof(*lachead);
172*1c60b9acSAndroid Build Coastguard Worker 		lachead = (struct lwsac_head *)&bf[1];
173*1c60b9acSAndroid Build Coastguard Worker 		memset(lachead, 0, sizeof(*lachead));
174*1c60b9acSAndroid Build Coastguard Worker 	} else
175*1c60b9acSAndroid Build Coastguard Worker 		if (lachead->curr)
176*1c60b9acSAndroid Build Coastguard Worker 			lachead->curr->next = bf;
177*1c60b9acSAndroid Build Coastguard Worker 
178*1c60b9acSAndroid Build Coastguard Worker 	lachead->curr = bf;
179*1c60b9acSAndroid Build Coastguard Worker 	bf->head = *head;
180*1c60b9acSAndroid Build Coastguard Worker 	bf->next = NULL;
181*1c60b9acSAndroid Build Coastguard Worker 	bf->alloc_size = alloc;
182*1c60b9acSAndroid Build Coastguard Worker 
183*1c60b9acSAndroid Build Coastguard Worker 	lachead->total_alloc_size += alloc;
184*1c60b9acSAndroid Build Coastguard Worker 	lachead->total_blocks++;
185*1c60b9acSAndroid Build Coastguard Worker 
186*1c60b9acSAndroid Build Coastguard Worker do_use:
187*1c60b9acSAndroid Build Coastguard Worker 
188*1c60b9acSAndroid Build Coastguard Worker 	ofs = bf->ofs;
189*1c60b9acSAndroid Build Coastguard Worker 
190*1c60b9acSAndroid Build Coastguard Worker 	if (al > ensure)
191*1c60b9acSAndroid Build Coastguard Worker 		/* zero down the alignment padding part */
192*1c60b9acSAndroid Build Coastguard Worker 		memset((char *)bf + ofs + ensure, 0, al - ensure);
193*1c60b9acSAndroid Build Coastguard Worker 
194*1c60b9acSAndroid Build Coastguard Worker 	bf->ofs += al;
195*1c60b9acSAndroid Build Coastguard Worker 	if (bf->ofs >= bf->alloc_size)
196*1c60b9acSAndroid Build Coastguard Worker 		bf->ofs = bf->alloc_size;
197*1c60b9acSAndroid Build Coastguard Worker 
198*1c60b9acSAndroid Build Coastguard Worker 	return (char *)bf + ofs;
199*1c60b9acSAndroid Build Coastguard Worker }
200*1c60b9acSAndroid Build Coastguard Worker 
201*1c60b9acSAndroid Build Coastguard Worker void *
lwsac_use(struct lwsac ** head,size_t ensure,size_t chunk_size)202*1c60b9acSAndroid Build Coastguard Worker lwsac_use(struct lwsac **head, size_t ensure, size_t chunk_size)
203*1c60b9acSAndroid Build Coastguard Worker {
204*1c60b9acSAndroid Build Coastguard Worker 	return _lwsac_use(head, ensure, chunk_size, 0);
205*1c60b9acSAndroid Build Coastguard Worker }
206*1c60b9acSAndroid Build Coastguard Worker 
207*1c60b9acSAndroid Build Coastguard Worker void *
lwsac_use_backfill(struct lwsac ** head,size_t ensure,size_t chunk_size)208*1c60b9acSAndroid Build Coastguard Worker lwsac_use_backfill(struct lwsac **head, size_t ensure, size_t chunk_size)
209*1c60b9acSAndroid Build Coastguard Worker {
210*1c60b9acSAndroid Build Coastguard Worker 	return _lwsac_use(head, ensure, chunk_size, 1);
211*1c60b9acSAndroid Build Coastguard Worker }
212*1c60b9acSAndroid Build Coastguard Worker 
213*1c60b9acSAndroid Build Coastguard Worker uint8_t *
lwsac_scan_extant(struct lwsac * head,uint8_t * find,size_t len,int nul)214*1c60b9acSAndroid Build Coastguard Worker lwsac_scan_extant(struct lwsac *head, uint8_t *find, size_t len, int nul)
215*1c60b9acSAndroid Build Coastguard Worker {
216*1c60b9acSAndroid Build Coastguard Worker 	while (head) {
217*1c60b9acSAndroid Build Coastguard Worker 		uint8_t *pos = (uint8_t *)&head[1],
218*1c60b9acSAndroid Build Coastguard Worker 			*end = ((uint8_t *)head) + head->ofs - len;
219*1c60b9acSAndroid Build Coastguard Worker 
220*1c60b9acSAndroid Build Coastguard Worker 		if (head->ofs - sizeof(*head) >= len)
221*1c60b9acSAndroid Build Coastguard Worker 			while (pos < end) {
222*1c60b9acSAndroid Build Coastguard Worker 				if (*pos == *find && (!nul || !pos[len]) &&
223*1c60b9acSAndroid Build Coastguard Worker 				    pos[len - 1] == find[len - 1] &&
224*1c60b9acSAndroid Build Coastguard Worker 				    !memcmp(pos, find, len))
225*1c60b9acSAndroid Build Coastguard Worker 					/* found the blob */
226*1c60b9acSAndroid Build Coastguard Worker 					return pos;
227*1c60b9acSAndroid Build Coastguard Worker 				pos++;
228*1c60b9acSAndroid Build Coastguard Worker 			}
229*1c60b9acSAndroid Build Coastguard Worker 
230*1c60b9acSAndroid Build Coastguard Worker 		head = head->next;
231*1c60b9acSAndroid Build Coastguard Worker 	}
232*1c60b9acSAndroid Build Coastguard Worker 
233*1c60b9acSAndroid Build Coastguard Worker 	return NULL;
234*1c60b9acSAndroid Build Coastguard Worker }
235*1c60b9acSAndroid Build Coastguard Worker 
236*1c60b9acSAndroid Build Coastguard Worker uint64_t
lwsac_total_overhead(struct lwsac * head)237*1c60b9acSAndroid Build Coastguard Worker lwsac_total_overhead(struct lwsac *head)
238*1c60b9acSAndroid Build Coastguard Worker {
239*1c60b9acSAndroid Build Coastguard Worker 	uint64_t overhead = 0;
240*1c60b9acSAndroid Build Coastguard Worker 
241*1c60b9acSAndroid Build Coastguard Worker 	while (head) {
242*1c60b9acSAndroid Build Coastguard Worker 		overhead += (head->alloc_size - head->ofs) + sizeof(*head);
243*1c60b9acSAndroid Build Coastguard Worker 
244*1c60b9acSAndroid Build Coastguard Worker 		head = head->next;
245*1c60b9acSAndroid Build Coastguard Worker 	}
246*1c60b9acSAndroid Build Coastguard Worker 
247*1c60b9acSAndroid Build Coastguard Worker 	return overhead;
248*1c60b9acSAndroid Build Coastguard Worker }
249*1c60b9acSAndroid Build Coastguard Worker 
250*1c60b9acSAndroid Build Coastguard Worker void *
lwsac_use_zero(struct lwsac ** head,size_t ensure,size_t chunk_size)251*1c60b9acSAndroid Build Coastguard Worker lwsac_use_zero(struct lwsac **head, size_t ensure, size_t chunk_size)
252*1c60b9acSAndroid Build Coastguard Worker {
253*1c60b9acSAndroid Build Coastguard Worker 	void *p = lwsac_use(head, ensure, chunk_size);
254*1c60b9acSAndroid Build Coastguard Worker 
255*1c60b9acSAndroid Build Coastguard Worker 	if (p)
256*1c60b9acSAndroid Build Coastguard Worker 		memset(p, 0, ensure);
257*1c60b9acSAndroid Build Coastguard Worker 
258*1c60b9acSAndroid Build Coastguard Worker 	return p;
259*1c60b9acSAndroid Build Coastguard Worker }
260*1c60b9acSAndroid Build Coastguard Worker 
261*1c60b9acSAndroid Build Coastguard Worker void
lwsac_free(struct lwsac ** head)262*1c60b9acSAndroid Build Coastguard Worker lwsac_free(struct lwsac **head)
263*1c60b9acSAndroid Build Coastguard Worker {
264*1c60b9acSAndroid Build Coastguard Worker 	struct lwsac *it = *head;
265*1c60b9acSAndroid Build Coastguard Worker 
266*1c60b9acSAndroid Build Coastguard Worker 	*head = NULL;
267*1c60b9acSAndroid Build Coastguard Worker 	lwsl_debug("%s: head %p\n", __func__, *head);
268*1c60b9acSAndroid Build Coastguard Worker 
269*1c60b9acSAndroid Build Coastguard Worker 	while (it) {
270*1c60b9acSAndroid Build Coastguard Worker 		struct lwsac *tmp = it->next;
271*1c60b9acSAndroid Build Coastguard Worker 
272*1c60b9acSAndroid Build Coastguard Worker 		free(it);
273*1c60b9acSAndroid Build Coastguard Worker 		it = tmp;
274*1c60b9acSAndroid Build Coastguard Worker 	}
275*1c60b9acSAndroid Build Coastguard Worker }
276*1c60b9acSAndroid Build Coastguard Worker 
277*1c60b9acSAndroid Build Coastguard Worker void
lwsac_info(struct lwsac * head)278*1c60b9acSAndroid Build Coastguard Worker lwsac_info(struct lwsac *head)
279*1c60b9acSAndroid Build Coastguard Worker {
280*1c60b9acSAndroid Build Coastguard Worker #if _LWS_ENABLED_LOGS & LLL_DEBUG
281*1c60b9acSAndroid Build Coastguard Worker 	struct lwsac_head *lachead;
282*1c60b9acSAndroid Build Coastguard Worker 
283*1c60b9acSAndroid Build Coastguard Worker 	if (!head) {
284*1c60b9acSAndroid Build Coastguard Worker 		lwsl_debug("%s: empty\n", __func__);
285*1c60b9acSAndroid Build Coastguard Worker 		return;
286*1c60b9acSAndroid Build Coastguard Worker 	}
287*1c60b9acSAndroid Build Coastguard Worker 
288*1c60b9acSAndroid Build Coastguard Worker 	lachead = (struct lwsac_head *)&head[1];
289*1c60b9acSAndroid Build Coastguard Worker 
290*1c60b9acSAndroid Build Coastguard Worker 	lwsl_debug("%s: lac %p: %dKiB in %d blocks\n", __func__, head,
291*1c60b9acSAndroid Build Coastguard Worker 		   (int)(lachead->total_alloc_size >> 10), lachead->total_blocks);
292*1c60b9acSAndroid Build Coastguard Worker #endif
293*1c60b9acSAndroid Build Coastguard Worker }
294*1c60b9acSAndroid Build Coastguard Worker 
295*1c60b9acSAndroid Build Coastguard Worker uint64_t
lwsac_total_alloc(struct lwsac * head)296*1c60b9acSAndroid Build Coastguard Worker lwsac_total_alloc(struct lwsac *head)
297*1c60b9acSAndroid Build Coastguard Worker {
298*1c60b9acSAndroid Build Coastguard Worker 	struct lwsac_head *lachead;
299*1c60b9acSAndroid Build Coastguard Worker 
300*1c60b9acSAndroid Build Coastguard Worker 	if (!head)
301*1c60b9acSAndroid Build Coastguard Worker 		return 0;
302*1c60b9acSAndroid Build Coastguard Worker 
303*1c60b9acSAndroid Build Coastguard Worker 	lachead = (struct lwsac_head *)&head[1];
304*1c60b9acSAndroid Build Coastguard Worker 	return lachead->total_alloc_size;
305*1c60b9acSAndroid Build Coastguard Worker }
306*1c60b9acSAndroid Build Coastguard Worker 
307*1c60b9acSAndroid Build Coastguard Worker void
lwsac_reference(struct lwsac * head)308*1c60b9acSAndroid Build Coastguard Worker lwsac_reference(struct lwsac *head)
309*1c60b9acSAndroid Build Coastguard Worker {
310*1c60b9acSAndroid Build Coastguard Worker 	struct lwsac_head *lachead = (struct lwsac_head *)&head[1];
311*1c60b9acSAndroid Build Coastguard Worker 
312*1c60b9acSAndroid Build Coastguard Worker 	lachead->refcount++;
313*1c60b9acSAndroid Build Coastguard Worker 	lwsl_debug("%s: head %p: (det %d) refcount -> %d\n",
314*1c60b9acSAndroid Build Coastguard Worker 		    __func__, head, lachead->detached, lachead->refcount);
315*1c60b9acSAndroid Build Coastguard Worker }
316*1c60b9acSAndroid Build Coastguard Worker 
317*1c60b9acSAndroid Build Coastguard Worker void
lwsac_unreference(struct lwsac ** head)318*1c60b9acSAndroid Build Coastguard Worker lwsac_unreference(struct lwsac **head)
319*1c60b9acSAndroid Build Coastguard Worker {
320*1c60b9acSAndroid Build Coastguard Worker 	struct lwsac_head *lachead;
321*1c60b9acSAndroid Build Coastguard Worker 
322*1c60b9acSAndroid Build Coastguard Worker 	if (!(*head))
323*1c60b9acSAndroid Build Coastguard Worker 		return;
324*1c60b9acSAndroid Build Coastguard Worker 
325*1c60b9acSAndroid Build Coastguard Worker 	lachead = (struct lwsac_head *)&(*head)[1];
326*1c60b9acSAndroid Build Coastguard Worker 
327*1c60b9acSAndroid Build Coastguard Worker 	if (!lachead->refcount)
328*1c60b9acSAndroid Build Coastguard Worker 		lwsl_warn("%s: refcount going below zero\n", __func__);
329*1c60b9acSAndroid Build Coastguard Worker 
330*1c60b9acSAndroid Build Coastguard Worker 	lachead->refcount--;
331*1c60b9acSAndroid Build Coastguard Worker 
332*1c60b9acSAndroid Build Coastguard Worker 	lwsl_debug("%s: head %p: (det %d) refcount -> %d\n",
333*1c60b9acSAndroid Build Coastguard Worker 		    __func__, *head, lachead->detached, lachead->refcount);
334*1c60b9acSAndroid Build Coastguard Worker 
335*1c60b9acSAndroid Build Coastguard Worker 	if (lachead->detached && !lachead->refcount) {
336*1c60b9acSAndroid Build Coastguard Worker 		lwsl_debug("%s: head %p: FREED\n", __func__, *head);
337*1c60b9acSAndroid Build Coastguard Worker 		lwsac_free(head);
338*1c60b9acSAndroid Build Coastguard Worker 	}
339*1c60b9acSAndroid Build Coastguard Worker }
340*1c60b9acSAndroid Build Coastguard Worker 
341*1c60b9acSAndroid Build Coastguard Worker void
lwsac_detach(struct lwsac ** head)342*1c60b9acSAndroid Build Coastguard Worker lwsac_detach(struct lwsac **head)
343*1c60b9acSAndroid Build Coastguard Worker {
344*1c60b9acSAndroid Build Coastguard Worker 	struct lwsac_head *lachead;
345*1c60b9acSAndroid Build Coastguard Worker 
346*1c60b9acSAndroid Build Coastguard Worker 	if (!(*head))
347*1c60b9acSAndroid Build Coastguard Worker 		return;
348*1c60b9acSAndroid Build Coastguard Worker 
349*1c60b9acSAndroid Build Coastguard Worker 	lachead = (struct lwsac_head *)&(*head)[1];
350*1c60b9acSAndroid Build Coastguard Worker 
351*1c60b9acSAndroid Build Coastguard Worker 	lachead->detached = 1;
352*1c60b9acSAndroid Build Coastguard Worker 	if (!lachead->refcount) {
353*1c60b9acSAndroid Build Coastguard Worker 		lwsl_debug("%s: head %p: FREED\n", __func__, *head);
354*1c60b9acSAndroid Build Coastguard Worker 		lwsac_free(head);
355*1c60b9acSAndroid Build Coastguard Worker 	} else
356*1c60b9acSAndroid Build Coastguard Worker 		lwsl_debug("%s: head %p: refcount %d: Marked as detached\n",
357*1c60b9acSAndroid Build Coastguard Worker 			    __func__, *head, lachead->refcount);
358*1c60b9acSAndroid Build Coastguard Worker }
359