Coverage Report

Created: 2026-03-10 08:46

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/binutils-gdb/libiberty/objalloc.c
Line
Count
Source
1
/* objalloc.c -- routines to allocate memory for objects
2
   Copyright (C) 1997-2026 Free Software Foundation, Inc.
3
   Written by Ian Lance Taylor, Cygnus Solutions.
4
5
This program is free software; you can redistribute it and/or modify it
6
under the terms of the GNU General Public License as published by the
7
Free Software Foundation; either version 2, or (at your option) any
8
later version.
9
10
This program is distributed in the hope that it will be useful,
11
but WITHOUT ANY WARRANTY; without even the implied warranty of
12
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13
GNU General Public License for more details.
14
15
You should have received a copy of the GNU General Public License
16
along with this program; if not, write to the Free Software
17
Foundation, 51 Franklin Street - Fifth Floor,
18
Boston, MA 02110-1301, USA.  */
19
20
#include "config.h"
21
#include "ansidecl.h"
22
23
#include "objalloc.h"
24
25
/* Get a definition for NULL.  */
26
#include <stdio.h>
27
28
#if VMS
29
#include <stdlib.h>
30
#include <unixlib.h>
31
#else
32
33
/* Get a definition for size_t.  */
34
#include <stddef.h>
35
36
#ifdef HAVE_STDLIB_H
37
#include <stdlib.h>
38
#else
39
/* For systems with larger pointers than ints, this must be declared.  */
40
extern void *malloc (size_t);
41
extern void free (void *);
42
#endif
43
44
#endif
45
46
/* These routines allocate space for an object.  Freeing allocated
47
   space may or may not free all more recently allocated space.
48
49
   We handle large and small allocation requests differently.  If we
50
   don't have enough space in the current block, and the allocation
51
   request is for more than 512 bytes, we simply pass it through to
52
   malloc.  */
53
54
/* The objalloc structure is defined in objalloc.h.  */
55
56
/* This structure appears at the start of each chunk.  */
57
58
struct objalloc_chunk
59
{
60
  /* Next chunk.  */
61
  struct objalloc_chunk *next;
62
  /* If this chunk contains large objects, this is the value of
63
     current_ptr when this chunk was allocated.  If this chunk
64
     contains small objects, this is NULL.  */
65
  char *current_ptr;
66
};
67
68
/* The aligned size of objalloc_chunk.  */
69
70
#define CHUNK_HEADER_SIZE         \
71
93.4M
  ((sizeof (struct objalloc_chunk) + OBJALLOC_ALIGN - 1) \
72
93.4M
   &~ (OBJALLOC_ALIGN - 1))
73
74
/* We ask for this much memory each time we create a chunk which is to
75
   hold small objects.  */
76
77
209M
#define CHUNK_SIZE (4096 - 32)
78
79
/* A request for this amount or more is just passed through to malloc.  */
80
81
11.2M
#define BIG_REQUEST (512)
82
83
/* Create an objalloc structure.  */
84
85
struct objalloc *
86
objalloc_create (void)
87
28.2M
{
88
28.2M
  struct objalloc *ret;
89
28.2M
  struct objalloc_chunk *chunk;
90
91
28.2M
  ret = (struct objalloc *) malloc (sizeof *ret);
92
28.2M
  if (ret == NULL)
93
0
    return NULL;
94
95
28.2M
  ret->chunks = (void *) malloc (CHUNK_SIZE);
96
28.2M
  if (ret->chunks == NULL)
97
0
    {
98
0
      free (ret);
99
0
      return NULL;
100
0
    }
101
102
28.2M
  chunk = (struct objalloc_chunk *) ret->chunks;
103
28.2M
  chunk->next = NULL;
104
28.2M
  chunk->current_ptr = NULL;
105
106
28.2M
  ret->current_ptr = (char *) chunk + CHUNK_HEADER_SIZE;
107
28.2M
  ret->current_space = CHUNK_SIZE - CHUNK_HEADER_SIZE;
108
109
28.2M
  return ret;
110
28.2M
}
111
112
/* Allocate space from an objalloc structure.  */
113
114
void *
115
_objalloc_alloc (struct objalloc *o, unsigned long original_len)
116
11.2M
{
117
11.2M
  unsigned long len = original_len;
118
119
  /* We avoid confusion from zero sized objects by always allocating
120
     at least 1 byte.  */
121
11.2M
  if (len == 0)
122
0
    len = 1;
123
124
11.2M
  len = (len + OBJALLOC_ALIGN - 1) &~ (OBJALLOC_ALIGN - 1);
125
126
  /* Check for overflow in the alignment operation above and the
127
     malloc argument below. */
128
11.2M
  if (len + CHUNK_HEADER_SIZE < original_len)
129
0
    return NULL;
130
131
11.2M
  if (len <= o->current_space)
132
0
    {
133
0
      o->current_ptr += len;
134
0
      o->current_space -= len;
135
0
      return (void *) (o->current_ptr - len);
136
0
    }
137
138
11.2M
  if (len >= BIG_REQUEST)
139
10.5M
    {
140
10.5M
      char *ret;
141
10.5M
      struct objalloc_chunk *chunk;
142
143
10.5M
      ret = (char *) malloc (CHUNK_HEADER_SIZE + len);
144
10.5M
      if (ret == NULL)
145
0
  return NULL;
146
147
10.5M
      chunk = (struct objalloc_chunk *) ret;
148
10.5M
      chunk->next = (struct objalloc_chunk *) o->chunks;
149
10.5M
      chunk->current_ptr = o->current_ptr;
150
151
10.5M
      o->chunks = (void *) chunk;
152
153
10.5M
      return (void *) (ret + CHUNK_HEADER_SIZE);
154
10.5M
    }
155
710k
  else
156
710k
    {
157
710k
      struct objalloc_chunk *chunk;
158
159
710k
      chunk = (struct objalloc_chunk *) malloc (CHUNK_SIZE);
160
710k
      if (chunk == NULL)
161
0
  return NULL;
162
710k
      chunk->next = (struct objalloc_chunk *) o->chunks;
163
710k
      chunk->current_ptr = NULL;
164
165
710k
      o->current_ptr = (char *) chunk + CHUNK_HEADER_SIZE;
166
710k
      o->current_space = CHUNK_SIZE - CHUNK_HEADER_SIZE;
167
168
710k
      o->chunks = (void *) chunk;
169
170
710k
      return objalloc_alloc (o, len);
171
710k
    }
172
11.2M
}
173
174
/* Free an entire objalloc structure.  */
175
176
void
177
objalloc_free (struct objalloc *o)
178
37.3M
{
179
37.3M
  struct objalloc_chunk *l;
180
181
  /* Handle a nullptr as being a no-op. */
182
37.3M
  if (o == NULL)
183
9.07M
    return;
184
185
28.2M
  l = (struct objalloc_chunk *) o->chunks;
186
66.7M
  while (l != NULL)
187
38.5M
    {
188
38.5M
      struct objalloc_chunk *next;
189
190
38.5M
      next = l->next;
191
38.5M
      free (l);
192
38.5M
      l = next;
193
38.5M
    }
194
195
28.2M
  free (o);
196
28.2M
}
197
198
/* Free a block from an objalloc structure.  This also frees all more
199
   recently allocated blocks.  */
200
201
void
202
objalloc_free_block (struct objalloc *o, void *block)
203
75.5M
{
204
75.5M
  struct objalloc_chunk *p, *small;
205
75.5M
  char *b = (char *) block;
206
207
  /* First set P to the chunk which contains the block we are freeing,
208
     and set Q to the last small object chunk we see before P.  */
209
75.5M
  small = NULL;
210
78.9M
  for (p = (struct objalloc_chunk *) o->chunks; p != NULL; p = p->next)
211
78.9M
    {
212
78.9M
      if (p->current_ptr == NULL)
213
75.7M
  {
214
75.7M
    if (b > (char *) p && b < (char *) p + CHUNK_SIZE)
215
75.5M
      break;
216
249k
    small = p;
217
249k
  }
218
3.12M
      else
219
3.12M
  {
220
3.12M
    if (b == (char *) p + CHUNK_HEADER_SIZE)
221
11.1k
      break;
222
3.12M
  }
223
78.9M
    }
224
225
  /* If we can't find the chunk, the caller has made a mistake.  */
226
75.5M
  if (p == NULL)
227
0
    abort ();
228
229
75.5M
  if (p->current_ptr == NULL)
230
75.5M
    {
231
75.5M
      struct objalloc_chunk *q;
232
75.5M
      struct objalloc_chunk *first;
233
234
      /* The block is in a chunk containing small objects.  We can
235
   free every chunk through SMALL, because they have certainly
236
   been allocated more recently.  After SMALL, we will not see
237
   any chunks containing small objects; we can free any big
238
   chunk if the current_ptr is greater than or equal to B.  We
239
   can then reset the new current_ptr to B.  */
240
241
75.5M
      first = NULL;
242
75.5M
      q = (struct objalloc_chunk *) o->chunks;
243
78.8M
      while (q != p)
244
3.34M
  {
245
3.34M
    struct objalloc_chunk *next;
246
247
3.34M
    next = q->next;
248
3.34M
    if (small != NULL)
249
398k
      {
250
398k
        if (small == q)
251
86.9k
    small = NULL;
252
398k
        free (q);
253
398k
      }
254
2.94M
    else if (q->current_ptr > b)
255
548k
      free (q);
256
2.39M
    else if (first == NULL)
257
1.25M
      first = q;
258
259
3.34M
    q = next;
260
3.34M
  }
261
262
75.5M
      if (first == NULL)
263
74.2M
  first = p;
264
75.5M
      o->chunks = (void *) first;
265
266
      /* Now start allocating from this small block again.  */
267
75.5M
      o->current_ptr = b;
268
75.5M
      o->current_space = ((char *) p + CHUNK_SIZE) - b;
269
75.5M
    }
270
11.1k
  else
271
11.1k
    {
272
11.1k
      struct objalloc_chunk *q;
273
11.1k
      char *current_ptr;
274
275
      /* This block is in a large chunk by itself.  We can free
276
         everything on the list up to and including this block.  We
277
         then start allocating from the next chunk containing small
278
         objects, setting current_ptr from the value stored with the
279
         large chunk we are freeing.  */
280
281
11.1k
      current_ptr = p->current_ptr;
282
11.1k
      p = p->next;
283
284
11.1k
      q = (struct objalloc_chunk *) o->chunks;
285
41.4k
      while (q != p)
286
30.2k
  {
287
30.2k
    struct objalloc_chunk *next;
288
289
30.2k
    next = q->next;
290
30.2k
    free (q);
291
30.2k
    q = next;
292
30.2k
  }
293
294
11.1k
      o->chunks = (void *) p;
295
296
15.7k
      while (p->current_ptr != NULL)
297
4.56k
  p = p->next;
298
299
11.1k
      o->current_ptr = current_ptr;
300
11.1k
      o->current_space = ((char *) p + CHUNK_SIZE) - current_ptr;
301
11.1k
    }
302
75.5M
}