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