xref: /aosp_15_r20/external/musl/src/regex/tre-mem.c (revision c9945492fdd68bbe62686c5b452b4dc1be3f8453)
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