oops - minor formatting tidy ups to previous delta
[official-gcc.git] / gcc / ggc-common.c
blobb5dad6bbd7689e56c803a0a55eda5b3ca593d5ab
1 /* Simple garbage collection for the GNU compiler.
2 Copyright (C) 1999, 2000, 2001, 2002 Free Software Foundation, Inc.
4 This file is part of GCC.
6 GCC is free software; you can redistribute it and/or modify it under
7 the terms of the GNU General Public License as published by the Free
8 Software Foundation; either version 2, or (at your option) any later
9 version.
11 GCC is distributed in the hope that it will be useful, but WITHOUT ANY
12 WARRANTY; without even the implied warranty of MERCHANTABILITY or
13 FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14 for more details.
16 You should have received a copy of the GNU General Public License
17 along with GCC; see the file COPYING. If not, write to the Free
18 Software Foundation, 59 Temple Place - Suite 330, Boston, MA
19 02111-1307, USA. */
21 /* Generic garbage collection (GC) functions and data, not specific to
22 any particular GC implementation. */
24 #include "config.h"
25 #include "system.h"
26 #include "rtl.h"
27 #include "tree.h"
28 #include "tm_p.h"
29 #include "hashtab.h"
30 #include "varray.h"
31 #include "ggc.h"
32 #include "langhooks.h"
34 /* Statistics about the allocation. */
35 static ggc_statistics *ggc_stats;
37 static void ggc_mark_rtx_children_1 PARAMS ((rtx));
38 static int ggc_htab_delete PARAMS ((void **, void *));
40 /* Maintain global roots that are preserved during GC. */
42 /* Global roots that are preserved during calls to gc. */
44 struct ggc_root
46 struct ggc_root *next;
47 void *base;
48 int nelt;
49 int size;
50 void (*cb) PARAMS ((void *));
53 static struct ggc_root *roots;
55 /* Add BASE as a new garbage collection root. It is an array of
56 length NELT with each element SIZE bytes long. CB is a
57 function that will be called with a pointer to each element
58 of the array; it is the intention that CB call the appropriate
59 routine to mark gc-able memory for that element. */
61 void
62 ggc_add_root (base, nelt, size, cb)
63 void *base;
64 int nelt, size;
65 void (*cb) PARAMS ((void *));
67 struct ggc_root *x = (struct ggc_root *) xmalloc (sizeof (*x));
69 x->next = roots;
70 x->base = base;
71 x->nelt = nelt;
72 x->size = size;
73 x->cb = cb;
75 roots = x;
78 /* Process a slot of an htab by deleting it if it has not been marked. */
80 static int
81 ggc_htab_delete (slot, info)
82 void **slot;
83 void *info;
85 const struct ggc_cache_tab *r = (const struct ggc_cache_tab *) info;
87 if (! (*r->marked_p) (*slot))
88 htab_clear_slot (*r->base, slot);
89 else
90 (*r->cb) (*slot);
92 return 1;
95 /* Iterate through all registered roots and mark each element. */
97 void
98 ggc_mark_roots ()
100 struct ggc_root *x;
101 const struct ggc_root_tab *const *rt;
102 const struct ggc_root_tab *rti;
103 const struct ggc_cache_tab *const *ct;
104 const struct ggc_cache_tab *cti;
105 size_t i;
107 for (rt = gt_ggc_deletable_rtab; *rt; rt++)
108 for (rti = *rt; rti->base != NULL; rti++)
109 memset (rti->base, 0, rti->stride);
111 for (rt = gt_ggc_rtab; *rt; rt++)
112 for (rti = *rt; rti->base != NULL; rti++)
113 for (i = 0; i < rti->nelt; i++)
114 (*rti->cb)(*(void **)((char *)rti->base + rti->stride * i));
116 for (x = roots; x != NULL; x = x->next)
118 char *elt = x->base;
119 int s = x->size, n = x->nelt;
120 void (*cb) PARAMS ((void *)) = x->cb;
121 int i;
123 for (i = 0; i < n; ++i, elt += s)
124 (*cb)(elt);
127 /* Now scan all hash tables that have objects which are to be deleted if
128 they are not already marked. */
129 for (ct = gt_ggc_cache_rtab; *ct; ct++)
130 for (cti = *ct; cti->base != NULL; cti++)
131 htab_traverse (*cti->base, ggc_htab_delete, (PTR) cti);
134 /* R had not been previously marked, but has now been marked via
135 ggc_set_mark. Now recurse and process the children. */
137 void
138 ggc_mark_rtx_children (r)
139 rtx r;
141 rtx i, last;
143 /* Special case the instruction chain. This is a data structure whose
144 chain length is potentially unbounded, and which contain references
145 within the chain (e.g. label_ref and insn_list). If do nothing here,
146 we risk blowing the stack recursing through a long chain of insns.
148 Combat this by marking all of the instructions in the chain before
149 marking the contents of those instructions. */
151 switch (GET_CODE (r))
153 case INSN:
154 case JUMP_INSN:
155 case CALL_INSN:
156 case NOTE:
157 case CODE_LABEL:
158 case BARRIER:
159 for (i = NEXT_INSN (r); ; i = NEXT_INSN (i))
160 if (! ggc_test_and_set_mark (i))
161 break;
162 last = i;
164 for (i = NEXT_INSN (r); i != last; i = NEXT_INSN (i))
165 ggc_mark_rtx_children_1 (i);
167 default:
168 break;
171 ggc_mark_rtx_children_1 (r);
174 static void
175 ggc_mark_rtx_children_1 (r)
176 rtx r;
178 const char *fmt;
179 int i;
180 rtx next_rtx;
184 enum rtx_code code = GET_CODE (r);
185 /* This gets set to a child rtx to eliminate tail recursion. */
186 next_rtx = NULL;
188 /* Collect statistics, if appropriate. */
189 if (ggc_stats)
191 ++ggc_stats->num_rtxs[(int) code];
192 ggc_stats->size_rtxs[(int) code] += ggc_get_size (r);
195 /* ??? If (some of) these are really pass-dependent info, do we
196 have any right poking our noses in? */
197 switch (code)
199 case MEM:
200 gt_ggc_m_mem_attrs (MEM_ATTRS (r));
201 break;
202 case JUMP_INSN:
203 ggc_mark_rtx (JUMP_LABEL (r));
204 break;
205 case CODE_LABEL:
206 ggc_mark_rtx (LABEL_REFS (r));
207 break;
208 case LABEL_REF:
209 ggc_mark_rtx (LABEL_NEXTREF (r));
210 ggc_mark_rtx (CONTAINING_INSN (r));
211 break;
212 case ADDRESSOF:
213 ggc_mark_tree (ADDRESSOF_DECL (r));
214 break;
215 case NOTE:
216 switch (NOTE_LINE_NUMBER (r))
218 case NOTE_INSN_EXPECTED_VALUE:
219 ggc_mark_rtx (NOTE_EXPECTED_VALUE (r));
220 break;
222 case NOTE_INSN_BLOCK_BEG:
223 case NOTE_INSN_BLOCK_END:
224 ggc_mark_tree (NOTE_BLOCK (r));
225 break;
227 default:
228 break;
230 break;
232 default:
233 break;
236 for (fmt = GET_RTX_FORMAT (GET_CODE (r)), i = 0; *fmt ; ++fmt, ++i)
238 rtx exp;
239 switch (*fmt)
241 case 'e': case 'u':
242 exp = XEXP (r, i);
243 if (ggc_test_and_set_mark (exp))
245 if (next_rtx == NULL)
246 next_rtx = exp;
247 else
248 ggc_mark_rtx_children (exp);
250 break;
251 case 'V': case 'E':
252 gt_ggc_m_rtvec_def (XVEC (r, i));
253 break;
257 while ((r = next_rtx) != NULL);
260 /* Various adaptor functions. */
261 void
262 gt_ggc_mx_rtx_def (x)
263 void *x;
265 ggc_mark_rtx((rtx)x);
268 /* Allocate a block of memory, then clear it. */
269 void *
270 ggc_alloc_cleared (size)
271 size_t size;
273 void *buf = ggc_alloc (size);
274 memset (buf, 0, size);
275 return buf;
278 /* Resize a block of memory, possibly re-allocating it. */
279 void *
280 ggc_realloc (x, size)
281 void *x;
282 size_t size;
284 void *r;
285 size_t old_size;
287 if (x == NULL)
288 return ggc_alloc (size);
290 old_size = ggc_get_size (x);
291 if (size <= old_size)
292 return x;
294 r = ggc_alloc (size);
295 memcpy (r, x, old_size);
296 return r;
299 /* Like ggc_alloc_cleared, but performs a multiplication. */
300 void *
301 ggc_calloc (s1, s2)
302 size_t s1, s2;
304 return ggc_alloc_cleared (s1 * s2);
307 /* Print statistics that are independent of the collector in use. */
308 #define SCALE(x) ((unsigned long) ((x) < 1024*10 \
309 ? (x) \
310 : ((x) < 1024*1024*10 \
311 ? (x) / 1024 \
312 : (x) / (1024*1024))))
313 #define LABEL(x) ((x) < 1024*10 ? ' ' : ((x) < 1024*1024*10 ? 'k' : 'M'))
315 void
316 ggc_print_common_statistics (stream, stats)
317 FILE *stream;
318 ggc_statistics *stats;
320 int code;
322 /* Set the pointer so that during collection we will actually gather
323 the statistics. */
324 ggc_stats = stats;
326 /* Then do one collection to fill in the statistics. */
327 ggc_collect ();
329 /* Total the statistics. */
330 for (code = 0; code < MAX_TREE_CODES; ++code)
332 stats->total_num_trees += stats->num_trees[code];
333 stats->total_size_trees += stats->size_trees[code];
335 for (code = 0; code < NUM_RTX_CODE; ++code)
337 stats->total_num_rtxs += stats->num_rtxs[code];
338 stats->total_size_rtxs += stats->size_rtxs[code];
341 /* Print the statistics for trees. */
342 fprintf (stream, "\n%-17s%10s %16s %10s\n", "Tree",
343 "Number", "Bytes", "% Total");
344 for (code = 0; code < MAX_TREE_CODES; ++code)
345 if (ggc_stats->num_trees[code])
347 fprintf (stream, "%-17s%10u%16ld%c %10.3f\n",
348 tree_code_name[code],
349 ggc_stats->num_trees[code],
350 SCALE (ggc_stats->size_trees[code]),
351 LABEL (ggc_stats->size_trees[code]),
352 (100 * ((double) ggc_stats->size_trees[code])
353 / ggc_stats->total_size_trees));
355 fprintf (stream,
356 "%-17s%10u%16ld%c\n", "Total",
357 ggc_stats->total_num_trees,
358 SCALE (ggc_stats->total_size_trees),
359 LABEL (ggc_stats->total_size_trees));
361 /* Print the statistics for RTL. */
362 fprintf (stream, "\n%-17s%10s %16s %10s\n", "RTX",
363 "Number", "Bytes", "% Total");
364 for (code = 0; code < NUM_RTX_CODE; ++code)
365 if (ggc_stats->num_rtxs[code])
367 fprintf (stream, "%-17s%10u%16ld%c %10.3f\n",
368 rtx_name[code],
369 ggc_stats->num_rtxs[code],
370 SCALE (ggc_stats->size_rtxs[code]),
371 LABEL (ggc_stats->size_rtxs[code]),
372 (100 * ((double) ggc_stats->size_rtxs[code])
373 / ggc_stats->total_size_rtxs));
375 fprintf (stream,
376 "%-17s%10u%16ld%c\n", "Total",
377 ggc_stats->total_num_rtxs,
378 SCALE (ggc_stats->total_size_rtxs),
379 LABEL (ggc_stats->total_size_rtxs));
381 /* Don't gather statistics any more. */
382 ggc_stats = NULL;