1*c9945492SAndroid Build Coastguard Worker /*
2*c9945492SAndroid Build Coastguard Worker tre-mem.c - TRE memory allocator
3*c9945492SAndroid Build Coastguard Worker
4*c9945492SAndroid Build Coastguard Worker Copyright (c) 2001-2009 Ville Laurikari <[email protected]>
5*c9945492SAndroid Build Coastguard Worker All rights reserved.
6*c9945492SAndroid Build Coastguard Worker
7*c9945492SAndroid Build Coastguard Worker Redistribution and use in source and binary forms, with or without
8*c9945492SAndroid Build Coastguard Worker modification, are permitted provided that the following conditions
9*c9945492SAndroid Build Coastguard Worker are met:
10*c9945492SAndroid Build Coastguard Worker
11*c9945492SAndroid Build Coastguard Worker 1. Redistributions of source code must retain the above copyright
12*c9945492SAndroid Build Coastguard Worker notice, this list of conditions and the following disclaimer.
13*c9945492SAndroid Build Coastguard Worker
14*c9945492SAndroid Build Coastguard Worker 2. Redistributions in binary form must reproduce the above copyright
15*c9945492SAndroid Build Coastguard Worker notice, this list of conditions and the following disclaimer in the
16*c9945492SAndroid Build Coastguard Worker documentation and/or other materials provided with the distribution.
17*c9945492SAndroid Build Coastguard Worker
18*c9945492SAndroid Build Coastguard Worker THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER AND CONTRIBUTORS
19*c9945492SAndroid Build Coastguard Worker ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20*c9945492SAndroid Build Coastguard Worker LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21*c9945492SAndroid Build Coastguard Worker A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22*c9945492SAndroid Build Coastguard Worker HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23*c9945492SAndroid Build Coastguard Worker SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24*c9945492SAndroid Build Coastguard Worker LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25*c9945492SAndroid Build Coastguard Worker DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26*c9945492SAndroid Build Coastguard Worker THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27*c9945492SAndroid Build Coastguard Worker (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28*c9945492SAndroid Build Coastguard Worker OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29*c9945492SAndroid Build Coastguard Worker
30*c9945492SAndroid Build Coastguard Worker */
31*c9945492SAndroid Build Coastguard Worker
32*c9945492SAndroid Build Coastguard Worker /*
33*c9945492SAndroid Build Coastguard Worker This memory allocator is for allocating small memory blocks efficiently
34*c9945492SAndroid Build Coastguard Worker in terms of memory overhead and execution speed. The allocated blocks
35*c9945492SAndroid Build Coastguard Worker cannot be freed individually, only all at once. There can be multiple
36*c9945492SAndroid Build Coastguard Worker allocators, though.
37*c9945492SAndroid Build Coastguard Worker */
38*c9945492SAndroid Build Coastguard Worker
39*c9945492SAndroid Build Coastguard Worker #include <stdlib.h>
40*c9945492SAndroid Build Coastguard Worker #include <string.h>
41*c9945492SAndroid Build Coastguard Worker
42*c9945492SAndroid Build Coastguard Worker #include "tre.h"
43*c9945492SAndroid Build Coastguard Worker
44*c9945492SAndroid Build Coastguard Worker /*
45*c9945492SAndroid Build Coastguard Worker This memory allocator is for allocating small memory blocks efficiently
46*c9945492SAndroid Build Coastguard Worker in terms of memory overhead and execution speed. The allocated blocks
47*c9945492SAndroid Build Coastguard Worker cannot be freed individually, only all at once. There can be multiple
48*c9945492SAndroid Build Coastguard Worker allocators, though.
49*c9945492SAndroid Build Coastguard Worker */
50*c9945492SAndroid Build Coastguard Worker
51*c9945492SAndroid Build Coastguard Worker /* Returns a new memory allocator or NULL if out of memory. */
52*c9945492SAndroid Build Coastguard Worker tre_mem_t
tre_mem_new_impl(int provided,void * provided_block)53*c9945492SAndroid Build Coastguard Worker tre_mem_new_impl(int provided, void *provided_block)
54*c9945492SAndroid Build Coastguard Worker {
55*c9945492SAndroid Build Coastguard Worker tre_mem_t mem;
56*c9945492SAndroid Build Coastguard Worker if (provided)
57*c9945492SAndroid Build Coastguard Worker {
58*c9945492SAndroid Build Coastguard Worker mem = provided_block;
59*c9945492SAndroid Build Coastguard Worker memset(mem, 0, sizeof(*mem));
60*c9945492SAndroid Build Coastguard Worker }
61*c9945492SAndroid Build Coastguard Worker else
62*c9945492SAndroid Build Coastguard Worker mem = xcalloc(1, sizeof(*mem));
63*c9945492SAndroid Build Coastguard Worker if (mem == NULL)
64*c9945492SAndroid Build Coastguard Worker return NULL;
65*c9945492SAndroid Build Coastguard Worker return mem;
66*c9945492SAndroid Build Coastguard Worker }
67*c9945492SAndroid Build Coastguard Worker
68*c9945492SAndroid Build Coastguard Worker
69*c9945492SAndroid Build Coastguard Worker /* Frees the memory allocator and all memory allocated with it. */
70*c9945492SAndroid Build Coastguard Worker void
tre_mem_destroy(tre_mem_t mem)71*c9945492SAndroid Build Coastguard Worker tre_mem_destroy(tre_mem_t mem)
72*c9945492SAndroid Build Coastguard Worker {
73*c9945492SAndroid Build Coastguard Worker tre_list_t *tmp, *l = mem->blocks;
74*c9945492SAndroid Build Coastguard Worker
75*c9945492SAndroid Build Coastguard Worker while (l != NULL)
76*c9945492SAndroid Build Coastguard Worker {
77*c9945492SAndroid Build Coastguard Worker xfree(l->data);
78*c9945492SAndroid Build Coastguard Worker tmp = l->next;
79*c9945492SAndroid Build Coastguard Worker xfree(l);
80*c9945492SAndroid Build Coastguard Worker l = tmp;
81*c9945492SAndroid Build Coastguard Worker }
82*c9945492SAndroid Build Coastguard Worker xfree(mem);
83*c9945492SAndroid Build Coastguard Worker }
84*c9945492SAndroid Build Coastguard Worker
85*c9945492SAndroid Build Coastguard Worker
86*c9945492SAndroid Build Coastguard Worker /* Allocates a block of `size' bytes from `mem'. Returns a pointer to the
87*c9945492SAndroid Build Coastguard Worker allocated block or NULL if an underlying malloc() failed. */
88*c9945492SAndroid Build Coastguard Worker void *
tre_mem_alloc_impl(tre_mem_t mem,int provided,void * provided_block,int zero,size_t size)89*c9945492SAndroid Build Coastguard Worker tre_mem_alloc_impl(tre_mem_t mem, int provided, void *provided_block,
90*c9945492SAndroid Build Coastguard Worker int zero, size_t size)
91*c9945492SAndroid Build Coastguard Worker {
92*c9945492SAndroid Build Coastguard Worker void *ptr;
93*c9945492SAndroid Build Coastguard Worker
94*c9945492SAndroid Build Coastguard Worker if (mem->failed)
95*c9945492SAndroid Build Coastguard Worker {
96*c9945492SAndroid Build Coastguard Worker return NULL;
97*c9945492SAndroid Build Coastguard Worker }
98*c9945492SAndroid Build Coastguard Worker
99*c9945492SAndroid Build Coastguard Worker if (mem->n < size)
100*c9945492SAndroid Build Coastguard Worker {
101*c9945492SAndroid Build Coastguard Worker /* We need more memory than is available in the current block.
102*c9945492SAndroid Build Coastguard Worker Allocate a new block. */
103*c9945492SAndroid Build Coastguard Worker tre_list_t *l;
104*c9945492SAndroid Build Coastguard Worker if (provided)
105*c9945492SAndroid Build Coastguard Worker {
106*c9945492SAndroid Build Coastguard Worker if (provided_block == NULL)
107*c9945492SAndroid Build Coastguard Worker {
108*c9945492SAndroid Build Coastguard Worker mem->failed = 1;
109*c9945492SAndroid Build Coastguard Worker return NULL;
110*c9945492SAndroid Build Coastguard Worker }
111*c9945492SAndroid Build Coastguard Worker mem->ptr = provided_block;
112*c9945492SAndroid Build Coastguard Worker mem->n = TRE_MEM_BLOCK_SIZE;
113*c9945492SAndroid Build Coastguard Worker }
114*c9945492SAndroid Build Coastguard Worker else
115*c9945492SAndroid Build Coastguard Worker {
116*c9945492SAndroid Build Coastguard Worker int block_size;
117*c9945492SAndroid Build Coastguard Worker if (size * 8 > TRE_MEM_BLOCK_SIZE)
118*c9945492SAndroid Build Coastguard Worker block_size = size * 8;
119*c9945492SAndroid Build Coastguard Worker else
120*c9945492SAndroid Build Coastguard Worker block_size = TRE_MEM_BLOCK_SIZE;
121*c9945492SAndroid Build Coastguard Worker l = xmalloc(sizeof(*l));
122*c9945492SAndroid Build Coastguard Worker if (l == NULL)
123*c9945492SAndroid Build Coastguard Worker {
124*c9945492SAndroid Build Coastguard Worker mem->failed = 1;
125*c9945492SAndroid Build Coastguard Worker return NULL;
126*c9945492SAndroid Build Coastguard Worker }
127*c9945492SAndroid Build Coastguard Worker l->data = xmalloc(block_size);
128*c9945492SAndroid Build Coastguard Worker if (l->data == NULL)
129*c9945492SAndroid Build Coastguard Worker {
130*c9945492SAndroid Build Coastguard Worker xfree(l);
131*c9945492SAndroid Build Coastguard Worker mem->failed = 1;
132*c9945492SAndroid Build Coastguard Worker return NULL;
133*c9945492SAndroid Build Coastguard Worker }
134*c9945492SAndroid Build Coastguard Worker l->next = NULL;
135*c9945492SAndroid Build Coastguard Worker if (mem->current != NULL)
136*c9945492SAndroid Build Coastguard Worker mem->current->next = l;
137*c9945492SAndroid Build Coastguard Worker if (mem->blocks == NULL)
138*c9945492SAndroid Build Coastguard Worker mem->blocks = l;
139*c9945492SAndroid Build Coastguard Worker mem->current = l;
140*c9945492SAndroid Build Coastguard Worker mem->ptr = l->data;
141*c9945492SAndroid Build Coastguard Worker mem->n = block_size;
142*c9945492SAndroid Build Coastguard Worker }
143*c9945492SAndroid Build Coastguard Worker }
144*c9945492SAndroid Build Coastguard Worker
145*c9945492SAndroid Build Coastguard Worker /* Make sure the next pointer will be aligned. */
146*c9945492SAndroid Build Coastguard Worker size += ALIGN(mem->ptr + size, long);
147*c9945492SAndroid Build Coastguard Worker
148*c9945492SAndroid Build Coastguard Worker /* Allocate from current block. */
149*c9945492SAndroid Build Coastguard Worker ptr = mem->ptr;
150*c9945492SAndroid Build Coastguard Worker mem->ptr += size;
151*c9945492SAndroid Build Coastguard Worker mem->n -= size;
152*c9945492SAndroid Build Coastguard Worker
153*c9945492SAndroid Build Coastguard Worker /* Set to zero if needed. */
154*c9945492SAndroid Build Coastguard Worker if (zero)
155*c9945492SAndroid Build Coastguard Worker memset(ptr, 0, size);
156*c9945492SAndroid Build Coastguard Worker
157*c9945492SAndroid Build Coastguard Worker return ptr;
158*c9945492SAndroid Build Coastguard Worker }
159