1*c0909341SAndroid Build Coastguard Worker /*
2*c0909341SAndroid Build Coastguard Worker * Copyright © 2020, VideoLAN and dav1d authors
3*c0909341SAndroid Build Coastguard Worker * Copyright © 2020, Two Orioles, LLC
4*c0909341SAndroid Build Coastguard Worker * All rights reserved.
5*c0909341SAndroid Build Coastguard Worker *
6*c0909341SAndroid Build Coastguard Worker * Redistribution and use in source and binary forms, with or without
7*c0909341SAndroid Build Coastguard Worker * modification, are permitted provided that the following conditions are met:
8*c0909341SAndroid Build Coastguard Worker *
9*c0909341SAndroid Build Coastguard Worker * 1. Redistributions of source code must retain the above copyright notice, this
10*c0909341SAndroid Build Coastguard Worker * list of conditions and the following disclaimer.
11*c0909341SAndroid Build Coastguard Worker *
12*c0909341SAndroid Build Coastguard Worker * 2. Redistributions in binary form must reproduce the above copyright notice,
13*c0909341SAndroid Build Coastguard Worker * this list of conditions and the following disclaimer in the documentation
14*c0909341SAndroid Build Coastguard Worker * and/or other materials provided with the distribution.
15*c0909341SAndroid Build Coastguard Worker *
16*c0909341SAndroid Build Coastguard Worker * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
17*c0909341SAndroid Build Coastguard Worker * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18*c0909341SAndroid Build Coastguard Worker * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
19*c0909341SAndroid Build Coastguard Worker * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
20*c0909341SAndroid Build Coastguard Worker * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
21*c0909341SAndroid Build Coastguard Worker * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
22*c0909341SAndroid Build Coastguard Worker * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
23*c0909341SAndroid Build Coastguard Worker * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
24*c0909341SAndroid Build Coastguard Worker * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25*c0909341SAndroid Build Coastguard Worker * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26*c0909341SAndroid Build Coastguard Worker */
27*c0909341SAndroid Build Coastguard Worker
28*c0909341SAndroid Build Coastguard Worker #include "config.h"
29*c0909341SAndroid Build Coastguard Worker
30*c0909341SAndroid Build Coastguard Worker #include <stdint.h>
31*c0909341SAndroid Build Coastguard Worker
32*c0909341SAndroid Build Coastguard Worker #include "src/internal.h"
33*c0909341SAndroid Build Coastguard Worker
34*c0909341SAndroid Build Coastguard Worker #if TRACK_HEAP_ALLOCATIONS
35*c0909341SAndroid Build Coastguard Worker #include <stdio.h>
36*c0909341SAndroid Build Coastguard Worker
37*c0909341SAndroid Build Coastguard Worker #include "src/log.h"
38*c0909341SAndroid Build Coastguard Worker
39*c0909341SAndroid Build Coastguard Worker #define DEFAULT_ALIGN 16
40*c0909341SAndroid Build Coastguard Worker
41*c0909341SAndroid Build Coastguard Worker typedef struct {
42*c0909341SAndroid Build Coastguard Worker size_t sz;
43*c0909341SAndroid Build Coastguard Worker unsigned align;
44*c0909341SAndroid Build Coastguard Worker enum AllocationType type;
45*c0909341SAndroid Build Coastguard Worker } Dav1dAllocationData;
46*c0909341SAndroid Build Coastguard Worker
47*c0909341SAndroid Build Coastguard Worker typedef struct {
48*c0909341SAndroid Build Coastguard Worker size_t curr_sz;
49*c0909341SAndroid Build Coastguard Worker size_t peak_sz;
50*c0909341SAndroid Build Coastguard Worker unsigned num_allocs;
51*c0909341SAndroid Build Coastguard Worker unsigned num_reuses;
52*c0909341SAndroid Build Coastguard Worker } AllocStats;
53*c0909341SAndroid Build Coastguard Worker
54*c0909341SAndroid Build Coastguard Worker static AllocStats tracked_allocs[N_ALLOC_TYPES];
55*c0909341SAndroid Build Coastguard Worker static size_t curr_total_sz;
56*c0909341SAndroid Build Coastguard Worker static size_t peak_total_sz;
57*c0909341SAndroid Build Coastguard Worker static pthread_mutex_t track_alloc_mutex = PTHREAD_MUTEX_INITIALIZER;
58*c0909341SAndroid Build Coastguard Worker
track_alloc(const enum AllocationType type,char * ptr,const size_t sz,const size_t align)59*c0909341SAndroid Build Coastguard Worker static void *track_alloc(const enum AllocationType type, char *ptr,
60*c0909341SAndroid Build Coastguard Worker const size_t sz, const size_t align)
61*c0909341SAndroid Build Coastguard Worker {
62*c0909341SAndroid Build Coastguard Worker assert(align >= sizeof(Dav1dAllocationData));
63*c0909341SAndroid Build Coastguard Worker if (ptr) {
64*c0909341SAndroid Build Coastguard Worker ptr += align;
65*c0909341SAndroid Build Coastguard Worker Dav1dAllocationData *const d = &((Dav1dAllocationData*)ptr)[-1];
66*c0909341SAndroid Build Coastguard Worker AllocStats *const s = &tracked_allocs[type];
67*c0909341SAndroid Build Coastguard Worker
68*c0909341SAndroid Build Coastguard Worker d->sz = sz;
69*c0909341SAndroid Build Coastguard Worker d->align = (unsigned)align;
70*c0909341SAndroid Build Coastguard Worker d->type = type;
71*c0909341SAndroid Build Coastguard Worker
72*c0909341SAndroid Build Coastguard Worker pthread_mutex_lock(&track_alloc_mutex);
73*c0909341SAndroid Build Coastguard Worker s->num_allocs++;
74*c0909341SAndroid Build Coastguard Worker s->curr_sz += sz;
75*c0909341SAndroid Build Coastguard Worker if (s->curr_sz > s->peak_sz)
76*c0909341SAndroid Build Coastguard Worker s->peak_sz = s->curr_sz;
77*c0909341SAndroid Build Coastguard Worker
78*c0909341SAndroid Build Coastguard Worker curr_total_sz += sz;
79*c0909341SAndroid Build Coastguard Worker if (curr_total_sz > peak_total_sz)
80*c0909341SAndroid Build Coastguard Worker peak_total_sz = curr_total_sz;
81*c0909341SAndroid Build Coastguard Worker pthread_mutex_unlock(&track_alloc_mutex);
82*c0909341SAndroid Build Coastguard Worker }
83*c0909341SAndroid Build Coastguard Worker return ptr;
84*c0909341SAndroid Build Coastguard Worker }
85*c0909341SAndroid Build Coastguard Worker
track_free(char * const ptr)86*c0909341SAndroid Build Coastguard Worker static void *track_free(char *const ptr) {
87*c0909341SAndroid Build Coastguard Worker const Dav1dAllocationData *const d = &((Dav1dAllocationData*)ptr)[-1];
88*c0909341SAndroid Build Coastguard Worker const size_t sz = d->sz;
89*c0909341SAndroid Build Coastguard Worker
90*c0909341SAndroid Build Coastguard Worker pthread_mutex_lock(&track_alloc_mutex);
91*c0909341SAndroid Build Coastguard Worker tracked_allocs[d->type].curr_sz -= sz;
92*c0909341SAndroid Build Coastguard Worker curr_total_sz -= sz;
93*c0909341SAndroid Build Coastguard Worker pthread_mutex_unlock(&track_alloc_mutex);
94*c0909341SAndroid Build Coastguard Worker
95*c0909341SAndroid Build Coastguard Worker return ptr - d->align;
96*c0909341SAndroid Build Coastguard Worker }
97*c0909341SAndroid Build Coastguard Worker
dav1d_track_reuse(const enum AllocationType type)98*c0909341SAndroid Build Coastguard Worker static void dav1d_track_reuse(const enum AllocationType type) {
99*c0909341SAndroid Build Coastguard Worker pthread_mutex_lock(&track_alloc_mutex);
100*c0909341SAndroid Build Coastguard Worker tracked_allocs[type].num_reuses++;
101*c0909341SAndroid Build Coastguard Worker pthread_mutex_unlock(&track_alloc_mutex);
102*c0909341SAndroid Build Coastguard Worker }
103*c0909341SAndroid Build Coastguard Worker
dav1d_malloc(const enum AllocationType type,const size_t sz)104*c0909341SAndroid Build Coastguard Worker void *dav1d_malloc(const enum AllocationType type, const size_t sz) {
105*c0909341SAndroid Build Coastguard Worker void *const ptr = malloc(sz + DEFAULT_ALIGN);
106*c0909341SAndroid Build Coastguard Worker return track_alloc(type, ptr, sz, DEFAULT_ALIGN);
107*c0909341SAndroid Build Coastguard Worker }
108*c0909341SAndroid Build Coastguard Worker
dav1d_alloc_aligned(const enum AllocationType type,const size_t sz,const size_t align)109*c0909341SAndroid Build Coastguard Worker void *dav1d_alloc_aligned(const enum AllocationType type,
110*c0909341SAndroid Build Coastguard Worker const size_t sz, const size_t align)
111*c0909341SAndroid Build Coastguard Worker {
112*c0909341SAndroid Build Coastguard Worker void *const ptr = dav1d_alloc_aligned_internal(align, sz + align);
113*c0909341SAndroid Build Coastguard Worker return track_alloc(type, ptr, sz, align);
114*c0909341SAndroid Build Coastguard Worker }
115*c0909341SAndroid Build Coastguard Worker
dav1d_realloc(const enum AllocationType type,void * ptr,const size_t sz)116*c0909341SAndroid Build Coastguard Worker void *dav1d_realloc(const enum AllocationType type,
117*c0909341SAndroid Build Coastguard Worker void *ptr, const size_t sz)
118*c0909341SAndroid Build Coastguard Worker {
119*c0909341SAndroid Build Coastguard Worker if (!ptr)
120*c0909341SAndroid Build Coastguard Worker return dav1d_malloc(type, sz);
121*c0909341SAndroid Build Coastguard Worker ptr = realloc((char*)ptr - DEFAULT_ALIGN, sz + DEFAULT_ALIGN);
122*c0909341SAndroid Build Coastguard Worker if (ptr)
123*c0909341SAndroid Build Coastguard Worker ptr = track_free((char*)ptr + DEFAULT_ALIGN);
124*c0909341SAndroid Build Coastguard Worker return track_alloc(type, ptr, sz, DEFAULT_ALIGN);
125*c0909341SAndroid Build Coastguard Worker }
126*c0909341SAndroid Build Coastguard Worker
dav1d_free(void * ptr)127*c0909341SAndroid Build Coastguard Worker void dav1d_free(void *ptr) {
128*c0909341SAndroid Build Coastguard Worker if (ptr)
129*c0909341SAndroid Build Coastguard Worker free(track_free(ptr));
130*c0909341SAndroid Build Coastguard Worker }
131*c0909341SAndroid Build Coastguard Worker
dav1d_free_aligned(void * ptr)132*c0909341SAndroid Build Coastguard Worker void dav1d_free_aligned(void *ptr) {
133*c0909341SAndroid Build Coastguard Worker if (ptr) {
134*c0909341SAndroid Build Coastguard Worker dav1d_free_aligned_internal(track_free(ptr));
135*c0909341SAndroid Build Coastguard Worker }
136*c0909341SAndroid Build Coastguard Worker }
137*c0909341SAndroid Build Coastguard Worker
cmp_stats(const void * const a,const void * const b)138*c0909341SAndroid Build Coastguard Worker static COLD int cmp_stats(const void *const a, const void *const b) {
139*c0909341SAndroid Build Coastguard Worker const size_t a_sz = ((const AllocStats*)a)->peak_sz;
140*c0909341SAndroid Build Coastguard Worker const size_t b_sz = ((const AllocStats*)b)->peak_sz;
141*c0909341SAndroid Build Coastguard Worker return a_sz < b_sz ? -1 : a_sz > b_sz;
142*c0909341SAndroid Build Coastguard Worker }
143*c0909341SAndroid Build Coastguard Worker
144*c0909341SAndroid Build Coastguard Worker /* Insert spaces as thousands separators for better readability */
format_tsep(char * const s,const size_t n,const size_t value)145*c0909341SAndroid Build Coastguard Worker static COLD int format_tsep(char *const s, const size_t n, const size_t value) {
146*c0909341SAndroid Build Coastguard Worker if (value < 1000)
147*c0909341SAndroid Build Coastguard Worker return snprintf(s, n, "%u", (unsigned)value);
148*c0909341SAndroid Build Coastguard Worker
149*c0909341SAndroid Build Coastguard Worker const int len = format_tsep(s, n, value / 1000);
150*c0909341SAndroid Build Coastguard Worker assert((size_t)len < n);
151*c0909341SAndroid Build Coastguard Worker return len + snprintf(s + len, n - len, " %03u", (unsigned)(value % 1000));
152*c0909341SAndroid Build Coastguard Worker }
153*c0909341SAndroid Build Coastguard Worker
dav1d_log_alloc_stats(Dav1dContext * const c)154*c0909341SAndroid Build Coastguard Worker COLD void dav1d_log_alloc_stats(Dav1dContext *const c) {
155*c0909341SAndroid Build Coastguard Worker static const char *const type_names[N_ALLOC_TYPES] = {
156*c0909341SAndroid Build Coastguard Worker [ALLOC_BLOCK ] = "Block data",
157*c0909341SAndroid Build Coastguard Worker [ALLOC_CDEF ] = "CDEF line buffers",
158*c0909341SAndroid Build Coastguard Worker [ALLOC_CDF ] = "CDF contexts",
159*c0909341SAndroid Build Coastguard Worker [ALLOC_COEF ] = "Coefficient data",
160*c0909341SAndroid Build Coastguard Worker [ALLOC_COMMON_CTX] = "Common context data",
161*c0909341SAndroid Build Coastguard Worker [ALLOC_DAV1DDATA ] = "Dav1dData",
162*c0909341SAndroid Build Coastguard Worker [ALLOC_IPRED ] = "Intra pred edges",
163*c0909341SAndroid Build Coastguard Worker [ALLOC_LF ] = "Loopfilter data",
164*c0909341SAndroid Build Coastguard Worker [ALLOC_LR ] = "Looprestoration data",
165*c0909341SAndroid Build Coastguard Worker [ALLOC_OBU_HDR ] = "OBU headers",
166*c0909341SAndroid Build Coastguard Worker [ALLOC_OBU_META ] = "OBU metadata",
167*c0909341SAndroid Build Coastguard Worker [ALLOC_PAL ] = "Palette data",
168*c0909341SAndroid Build Coastguard Worker [ALLOC_PIC ] = "Picture buffers",
169*c0909341SAndroid Build Coastguard Worker [ALLOC_PIC_CTX ] = "Picture context data",
170*c0909341SAndroid Build Coastguard Worker [ALLOC_REFMVS ] = "Reference mv data",
171*c0909341SAndroid Build Coastguard Worker [ALLOC_SEGMAP ] = "Segmentation maps",
172*c0909341SAndroid Build Coastguard Worker [ALLOC_THREAD_CTX] = "Thread context data",
173*c0909341SAndroid Build Coastguard Worker [ALLOC_TILE ] = "Tile data",
174*c0909341SAndroid Build Coastguard Worker };
175*c0909341SAndroid Build Coastguard Worker
176*c0909341SAndroid Build Coastguard Worker struct {
177*c0909341SAndroid Build Coastguard Worker AllocStats stats;
178*c0909341SAndroid Build Coastguard Worker enum AllocationType type;
179*c0909341SAndroid Build Coastguard Worker } data[N_ALLOC_TYPES];
180*c0909341SAndroid Build Coastguard Worker unsigned total_allocs = 0;
181*c0909341SAndroid Build Coastguard Worker unsigned total_reuses = 0;
182*c0909341SAndroid Build Coastguard Worker
183*c0909341SAndroid Build Coastguard Worker pthread_mutex_lock(&track_alloc_mutex);
184*c0909341SAndroid Build Coastguard Worker for (int i = 0; i < N_ALLOC_TYPES; i++) {
185*c0909341SAndroid Build Coastguard Worker AllocStats *const s = &data[i].stats;
186*c0909341SAndroid Build Coastguard Worker *s = tracked_allocs[i];
187*c0909341SAndroid Build Coastguard Worker data[i].type = i;
188*c0909341SAndroid Build Coastguard Worker total_allocs += s->num_allocs;
189*c0909341SAndroid Build Coastguard Worker total_reuses += s->num_reuses;
190*c0909341SAndroid Build Coastguard Worker }
191*c0909341SAndroid Build Coastguard Worker size_t total_sz = peak_total_sz;
192*c0909341SAndroid Build Coastguard Worker pthread_mutex_unlock(&track_alloc_mutex);
193*c0909341SAndroid Build Coastguard Worker
194*c0909341SAndroid Build Coastguard Worker /* Sort types by memory usage */
195*c0909341SAndroid Build Coastguard Worker qsort(&data, N_ALLOC_TYPES, sizeof(*data), cmp_stats);
196*c0909341SAndroid Build Coastguard Worker
197*c0909341SAndroid Build Coastguard Worker const double inv_total_share = 100.0 / total_sz;
198*c0909341SAndroid Build Coastguard Worker char total_sz_buf[32];
199*c0909341SAndroid Build Coastguard Worker const int sz_len = 4 + format_tsep(total_sz_buf, sizeof(total_sz_buf), total_sz);
200*c0909341SAndroid Build Coastguard Worker
201*c0909341SAndroid Build Coastguard Worker dav1d_log(c, "\n Type Allocs Reuses Share Peak size\n"
202*c0909341SAndroid Build Coastguard Worker "---------------------------------------------------------------------\n");
203*c0909341SAndroid Build Coastguard Worker for (int i = N_ALLOC_TYPES - 1; i >= 0; i--) {
204*c0909341SAndroid Build Coastguard Worker const AllocStats *const s = &data[i].stats;
205*c0909341SAndroid Build Coastguard Worker if (s->num_allocs) {
206*c0909341SAndroid Build Coastguard Worker const double share = s->peak_sz * inv_total_share;
207*c0909341SAndroid Build Coastguard Worker char sz_buf[32];
208*c0909341SAndroid Build Coastguard Worker format_tsep(sz_buf, sizeof(sz_buf), s->peak_sz);
209*c0909341SAndroid Build Coastguard Worker dav1d_log(c, " %-20s%10u%10u%8.1f%%%*s\n", type_names[data[i].type],
210*c0909341SAndroid Build Coastguard Worker s->num_allocs, s->num_reuses, share, sz_len, sz_buf);
211*c0909341SAndroid Build Coastguard Worker }
212*c0909341SAndroid Build Coastguard Worker }
213*c0909341SAndroid Build Coastguard Worker dav1d_log(c, "---------------------------------------------------------------------\n"
214*c0909341SAndroid Build Coastguard Worker "%31u%10u %s\n",
215*c0909341SAndroid Build Coastguard Worker total_allocs, total_reuses, total_sz_buf);
216*c0909341SAndroid Build Coastguard Worker }
217*c0909341SAndroid Build Coastguard Worker #endif /* TRACK_HEAP_ALLOCATIONS */
218*c0909341SAndroid Build Coastguard Worker
mem_pool_destroy(Dav1dMemPool * const pool)219*c0909341SAndroid Build Coastguard Worker static COLD void mem_pool_destroy(Dav1dMemPool *const pool) {
220*c0909341SAndroid Build Coastguard Worker pthread_mutex_destroy(&pool->lock);
221*c0909341SAndroid Build Coastguard Worker dav1d_free(pool);
222*c0909341SAndroid Build Coastguard Worker }
223*c0909341SAndroid Build Coastguard Worker
dav1d_mem_pool_push(Dav1dMemPool * const pool,Dav1dMemPoolBuffer * const buf)224*c0909341SAndroid Build Coastguard Worker void dav1d_mem_pool_push(Dav1dMemPool *const pool, Dav1dMemPoolBuffer *const buf) {
225*c0909341SAndroid Build Coastguard Worker pthread_mutex_lock(&pool->lock);
226*c0909341SAndroid Build Coastguard Worker const int ref_cnt = --pool->ref_cnt;
227*c0909341SAndroid Build Coastguard Worker if (!pool->end) {
228*c0909341SAndroid Build Coastguard Worker buf->next = pool->buf;
229*c0909341SAndroid Build Coastguard Worker pool->buf = buf;
230*c0909341SAndroid Build Coastguard Worker pthread_mutex_unlock(&pool->lock);
231*c0909341SAndroid Build Coastguard Worker assert(ref_cnt > 0);
232*c0909341SAndroid Build Coastguard Worker } else {
233*c0909341SAndroid Build Coastguard Worker pthread_mutex_unlock(&pool->lock);
234*c0909341SAndroid Build Coastguard Worker dav1d_free_aligned(buf->data);
235*c0909341SAndroid Build Coastguard Worker if (!ref_cnt) mem_pool_destroy(pool);
236*c0909341SAndroid Build Coastguard Worker }
237*c0909341SAndroid Build Coastguard Worker }
238*c0909341SAndroid Build Coastguard Worker
dav1d_mem_pool_pop(Dav1dMemPool * const pool,const size_t size)239*c0909341SAndroid Build Coastguard Worker Dav1dMemPoolBuffer *dav1d_mem_pool_pop(Dav1dMemPool *const pool, const size_t size) {
240*c0909341SAndroid Build Coastguard Worker assert(!(size & (sizeof(void*) - 1)));
241*c0909341SAndroid Build Coastguard Worker pthread_mutex_lock(&pool->lock);
242*c0909341SAndroid Build Coastguard Worker Dav1dMemPoolBuffer *buf = pool->buf;
243*c0909341SAndroid Build Coastguard Worker pool->ref_cnt++;
244*c0909341SAndroid Build Coastguard Worker uint8_t *data;
245*c0909341SAndroid Build Coastguard Worker if (buf) {
246*c0909341SAndroid Build Coastguard Worker pool->buf = buf->next;
247*c0909341SAndroid Build Coastguard Worker pthread_mutex_unlock(&pool->lock);
248*c0909341SAndroid Build Coastguard Worker data = buf->data;
249*c0909341SAndroid Build Coastguard Worker if ((uintptr_t)buf - (uintptr_t)data != size) {
250*c0909341SAndroid Build Coastguard Worker /* Reallocate if the size has changed */
251*c0909341SAndroid Build Coastguard Worker dav1d_free_aligned(data);
252*c0909341SAndroid Build Coastguard Worker goto alloc;
253*c0909341SAndroid Build Coastguard Worker }
254*c0909341SAndroid Build Coastguard Worker #if TRACK_HEAP_ALLOCATIONS
255*c0909341SAndroid Build Coastguard Worker dav1d_track_reuse(pool->type);
256*c0909341SAndroid Build Coastguard Worker #endif
257*c0909341SAndroid Build Coastguard Worker } else {
258*c0909341SAndroid Build Coastguard Worker pthread_mutex_unlock(&pool->lock);
259*c0909341SAndroid Build Coastguard Worker alloc:
260*c0909341SAndroid Build Coastguard Worker data = dav1d_alloc_aligned(pool->type,
261*c0909341SAndroid Build Coastguard Worker size + sizeof(Dav1dMemPoolBuffer), 64);
262*c0909341SAndroid Build Coastguard Worker if (!data) {
263*c0909341SAndroid Build Coastguard Worker pthread_mutex_lock(&pool->lock);
264*c0909341SAndroid Build Coastguard Worker const int ref_cnt = --pool->ref_cnt;
265*c0909341SAndroid Build Coastguard Worker pthread_mutex_unlock(&pool->lock);
266*c0909341SAndroid Build Coastguard Worker if (!ref_cnt) mem_pool_destroy(pool);
267*c0909341SAndroid Build Coastguard Worker return NULL;
268*c0909341SAndroid Build Coastguard Worker }
269*c0909341SAndroid Build Coastguard Worker buf = (Dav1dMemPoolBuffer*)(data + size);
270*c0909341SAndroid Build Coastguard Worker buf->data = data;
271*c0909341SAndroid Build Coastguard Worker }
272*c0909341SAndroid Build Coastguard Worker
273*c0909341SAndroid Build Coastguard Worker return buf;
274*c0909341SAndroid Build Coastguard Worker }
275*c0909341SAndroid Build Coastguard Worker
dav1d_mem_pool_init(const enum AllocationType type,Dav1dMemPool ** const ppool)276*c0909341SAndroid Build Coastguard Worker COLD int dav1d_mem_pool_init(const enum AllocationType type,
277*c0909341SAndroid Build Coastguard Worker Dav1dMemPool **const ppool)
278*c0909341SAndroid Build Coastguard Worker {
279*c0909341SAndroid Build Coastguard Worker Dav1dMemPool *const pool = dav1d_malloc(ALLOC_COMMON_CTX,
280*c0909341SAndroid Build Coastguard Worker sizeof(Dav1dMemPool));
281*c0909341SAndroid Build Coastguard Worker if (pool) {
282*c0909341SAndroid Build Coastguard Worker if (!pthread_mutex_init(&pool->lock, NULL)) {
283*c0909341SAndroid Build Coastguard Worker pool->buf = NULL;
284*c0909341SAndroid Build Coastguard Worker pool->ref_cnt = 1;
285*c0909341SAndroid Build Coastguard Worker pool->end = 0;
286*c0909341SAndroid Build Coastguard Worker #if TRACK_HEAP_ALLOCATIONS
287*c0909341SAndroid Build Coastguard Worker pool->type = type;
288*c0909341SAndroid Build Coastguard Worker #endif
289*c0909341SAndroid Build Coastguard Worker *ppool = pool;
290*c0909341SAndroid Build Coastguard Worker return 0;
291*c0909341SAndroid Build Coastguard Worker }
292*c0909341SAndroid Build Coastguard Worker dav1d_free(pool);
293*c0909341SAndroid Build Coastguard Worker }
294*c0909341SAndroid Build Coastguard Worker *ppool = NULL;
295*c0909341SAndroid Build Coastguard Worker return DAV1D_ERR(ENOMEM);
296*c0909341SAndroid Build Coastguard Worker }
297*c0909341SAndroid Build Coastguard Worker
dav1d_mem_pool_end(Dav1dMemPool * const pool)298*c0909341SAndroid Build Coastguard Worker COLD void dav1d_mem_pool_end(Dav1dMemPool *const pool) {
299*c0909341SAndroid Build Coastguard Worker if (pool) {
300*c0909341SAndroid Build Coastguard Worker pthread_mutex_lock(&pool->lock);
301*c0909341SAndroid Build Coastguard Worker Dav1dMemPoolBuffer *buf = pool->buf;
302*c0909341SAndroid Build Coastguard Worker const int ref_cnt = --pool->ref_cnt;
303*c0909341SAndroid Build Coastguard Worker pool->buf = NULL;
304*c0909341SAndroid Build Coastguard Worker pool->end = 1;
305*c0909341SAndroid Build Coastguard Worker pthread_mutex_unlock(&pool->lock);
306*c0909341SAndroid Build Coastguard Worker
307*c0909341SAndroid Build Coastguard Worker while (buf) {
308*c0909341SAndroid Build Coastguard Worker void *const data = buf->data;
309*c0909341SAndroid Build Coastguard Worker buf = buf->next;
310*c0909341SAndroid Build Coastguard Worker dav1d_free_aligned(data);
311*c0909341SAndroid Build Coastguard Worker }
312*c0909341SAndroid Build Coastguard Worker if (!ref_cnt) mem_pool_destroy(pool);
313*c0909341SAndroid Build Coastguard Worker }
314*c0909341SAndroid Build Coastguard Worker }
315