]> gcc.gnu.org Git - gcc.git/blame - gcc/ggc-common.c
re PR rtl-optimization/31500 (FAIL: gcc.dg/Warray-bounds.c (internal compiler error))
[gcc.git] / gcc / ggc-common.c
CommitLineData
b49a6a90 1/* Simple garbage collection for the GNU compiler.
83f676b3 2 Copyright (C) 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006
14011ca4 3 Free Software Foundation, Inc.
b49a6a90 4
1322177d 5This file is part of GCC.
b49a6a90 6
1322177d
LB
7GCC is free software; you can redistribute it and/or modify it under
8the terms of the GNU General Public License as published by the Free
9Software Foundation; either version 2, or (at your option) any later
10version.
b49a6a90 11
1322177d
LB
12GCC is distributed in the hope that it will be useful, but WITHOUT ANY
13WARRANTY; without even the implied warranty of MERCHANTABILITY or
14a774a9
RK
14FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
15for more details.
b49a6a90 16
14a774a9 17You should have received a copy of the GNU General Public License
1322177d 18along with GCC; see the file COPYING. If not, write to the Free
366ccddb
KC
19Software Foundation, 51 Franklin Street, Fifth Floor, Boston, MA
2002110-1301, USA. */
b49a6a90
AS
21
22/* Generic garbage collection (GC) functions and data, not specific to
23 any particular GC implementation. */
24
25#include "config.h"
26#include "system.h"
4977bab6 27#include "coretypes.h"
4c160717 28#include "hashtab.h"
1b42a6a9 29#include "ggc.h"
17211ab5 30#include "toplev.h"
9ac121af 31#include "params.h"
18c81520 32#include "hosthooks.h"
4d0c31e6 33#include "hosthooks-def.h"
17211ab5 34
16226f1e
KG
35#ifdef HAVE_SYS_RESOURCE_H
36# include <sys/resource.h>
37#endif
38
17211ab5
GK
39#ifdef HAVE_MMAP_FILE
40# include <sys/mman.h>
8eb6a092
EB
41# ifdef HAVE_MINCORE
42/* This is on Solaris. */
43# include <sys/types.h>
44# endif
45#endif
46
47#ifndef MAP_FAILED
48# define MAP_FAILED ((void *)-1)
17211ab5
GK
49#endif
50
9a0a7d5d 51#ifdef ENABLE_VALGRIND_CHECKING
a207b594
HPN
52# ifdef HAVE_VALGRIND_MEMCHECK_H
53# include <valgrind/memcheck.h>
54# elif defined HAVE_MEMCHECK_H
55# include <memcheck.h>
14011ca4 56# else
a207b594 57# include <valgrind.h>
14011ca4 58# endif
9a0a7d5d
HPN
59#else
60/* Avoid #ifdef:s when we can help it. */
61#define VALGRIND_DISCARD(x)
62#endif
b49a6a90 63
07724022
JH
64/* When set, ggc_collect will do collection. */
65bool ggc_force_collect;
66
3277221c
MM
67/* Statistics about the allocation. */
68static ggc_statistics *ggc_stats;
69
17211ab5
GK
70struct traversal_state;
71
20c1dc5e
AJ
72static int ggc_htab_delete (void **, void *);
73static hashval_t saving_htab_hash (const void *);
74static int saving_htab_eq (const void *, const void *);
75static int call_count (void **, void *);
76static int call_alloc (void **, void *);
77static int compare_ptr_data (const void *, const void *);
78static void relocate_ptrs (void *, void *);
79static void write_pch_globals (const struct ggc_root_tab * const *tab,
80 struct traversal_state *state);
81static double ggc_rlimit_bound (double);
b49a6a90
AS
82
83/* Maintain global roots that are preserved during GC. */
84
4c160717
RK
85/* Process a slot of an htab by deleting it if it has not been marked. */
86
87static int
20c1dc5e 88ggc_htab_delete (void **slot, void *info)
4c160717 89{
e2500fed 90 const struct ggc_cache_tab *r = (const struct ggc_cache_tab *) info;
4c160717
RK
91
92 if (! (*r->marked_p) (*slot))
e2500fed
GK
93 htab_clear_slot (*r->base, slot);
94 else
95 (*r->cb) (*slot);
4c160717
RK
96
97 return 1;
98}
99
cb2ec151
RH
100/* Iterate through all registered roots and mark each element. */
101
b49a6a90 102void
20c1dc5e 103ggc_mark_roots (void)
96df4529 104{
e2500fed
GK
105 const struct ggc_root_tab *const *rt;
106 const struct ggc_root_tab *rti;
107 const struct ggc_cache_tab *const *ct;
108 const struct ggc_cache_tab *cti;
109 size_t i;
589005ff 110
e2500fed
GK
111 for (rt = gt_ggc_deletable_rtab; *rt; rt++)
112 for (rti = *rt; rti->base != NULL; rti++)
113 memset (rti->base, 0, rti->stride);
114
115 for (rt = gt_ggc_rtab; *rt; rt++)
116 for (rti = *rt; rti->base != NULL; rti++)
117 for (i = 0; i < rti->nelt; i++)
118 (*rti->cb)(*(void **)((char *)rti->base + rti->stride * i));
bedda2da 119
17211ab5 120 ggc_mark_stringpool ();
bedda2da 121
4c160717 122 /* Now scan all hash tables that have objects which are to be deleted if
e2500fed
GK
123 they are not already marked. */
124 for (ct = gt_ggc_cache_rtab; *ct; ct++)
125 for (cti = *ct; cti->base != NULL; cti++)
690eed2c 126 if (*cti->base)
17211ab5
GK
127 {
128 ggc_set_mark (*cti->base);
20c1dc5e 129 htab_traverse_noresize (*cti->base, ggc_htab_delete, (void *) cti);
17211ab5
GK
130 ggc_set_mark ((*cti->base)->entries);
131 }
96df4529
AS
132}
133
e2500fed
GK
134/* Allocate a block of memory, then clear it. */
135void *
b9dcdee4 136ggc_alloc_cleared_stat (size_t size MEM_STAT_DECL)
ef8288f7 137{
b9dcdee4 138 void *buf = ggc_alloc_stat (size PASS_MEM_STAT);
e2500fed
GK
139 memset (buf, 0, size);
140 return buf;
ef8288f7
RH
141}
142
e2500fed
GK
143/* Resize a block of memory, possibly re-allocating it. */
144void *
b9dcdee4 145ggc_realloc_stat (void *x, size_t size MEM_STAT_DECL)
ef8288f7 146{
e2500fed
GK
147 void *r;
148 size_t old_size;
ef8288f7 149
e2500fed 150 if (x == NULL)
b9dcdee4 151 return ggc_alloc_stat (size PASS_MEM_STAT);
ef8288f7 152
e2500fed 153 old_size = ggc_get_size (x);
685fe032 154
e2500fed 155 if (size <= old_size)
9a0a7d5d
HPN
156 {
157 /* Mark the unwanted memory as unaccessible. We also need to make
158 the "new" size accessible, since ggc_get_size returns the size of
159 the pool, not the size of the individually allocated object, the
160 size which was previously made accessible. Unfortunately, we
161 don't know that previously allocated size. Without that
162 knowledge we have to lose some initialization-tracking for the
163 old parts of the object. An alternative is to mark the whole
20c1dc5e 164 old_size as reachable, but that would lose tracking of writes
9a0a7d5d
HPN
165 after the end of the object (by small offsets). Discard the
166 handle to avoid handle leak. */
167 VALGRIND_DISCARD (VALGRIND_MAKE_NOACCESS ((char *) x + size,
168 old_size - size));
169 VALGRIND_DISCARD (VALGRIND_MAKE_READABLE (x, size));
170 return x;
171 }
ef8288f7 172
b9dcdee4 173 r = ggc_alloc_stat (size PASS_MEM_STAT);
9a0a7d5d
HPN
174
175 /* Since ggc_get_size returns the size of the pool, not the size of the
176 individually allocated object, we'd access parts of the old object
177 that were marked invalid with the memcpy below. We lose a bit of the
178 initialization-tracking since some of it may be uninitialized. */
179 VALGRIND_DISCARD (VALGRIND_MAKE_READABLE (x, old_size));
180
e2500fed 181 memcpy (r, x, old_size);
9a0a7d5d
HPN
182
183 /* The old object is not supposed to be used anymore. */
685fe032 184 ggc_free (x);
9a0a7d5d 185
e2500fed 186 return r;
ef8288f7
RH
187}
188
e2500fed 189/* Like ggc_alloc_cleared, but performs a multiplication. */
f8a83ee3 190void *
20c1dc5e 191ggc_calloc (size_t s1, size_t s2)
f8a83ee3 192{
e2500fed 193 return ggc_alloc_cleared (s1 * s2);
f8a83ee3
ZW
194}
195
17211ab5 196/* These are for splay_tree_new_ggc. */
20c1dc5e
AJ
197void *
198ggc_splay_alloc (int sz, void *nl)
17211ab5 199{
282899df 200 gcc_assert (!nl);
17211ab5
GK
201 return ggc_alloc (sz);
202}
203
204void
20c1dc5e 205ggc_splay_dont_free (void * x ATTRIBUTE_UNUSED, void *nl)
17211ab5 206{
282899df 207 gcc_assert (!nl);
17211ab5
GK
208}
209
3277221c 210/* Print statistics that are independent of the collector in use. */
fba0bfd4
ZW
211#define SCALE(x) ((unsigned long) ((x) < 1024*10 \
212 ? (x) \
213 : ((x) < 1024*1024*10 \
214 ? (x) / 1024 \
215 : (x) / (1024*1024))))
216#define LABEL(x) ((x) < 1024*10 ? ' ' : ((x) < 1024*1024*10 ? 'k' : 'M'))
3277221c
MM
217
218void
20c1dc5e
AJ
219ggc_print_common_statistics (FILE *stream ATTRIBUTE_UNUSED,
220 ggc_statistics *stats)
3277221c 221{
3277221c
MM
222 /* Set the pointer so that during collection we will actually gather
223 the statistics. */
224 ggc_stats = stats;
225
226 /* Then do one collection to fill in the statistics. */
227 ggc_collect ();
228
17211ab5
GK
229 /* At present, we don't really gather any interesting statistics. */
230
231 /* Don't gather statistics any more. */
232 ggc_stats = NULL;
233}
234\f
235/* Functions for saving and restoring GCable memory to disk. */
236
237static htab_t saving_htab;
238
20c1dc5e 239struct ptr_data
17211ab5
GK
240{
241 void *obj;
242 void *note_ptr_cookie;
243 gt_note_pointers note_ptr_fn;
244 gt_handle_reorder reorder_fn;
245 size_t size;
246 void *new_addr;
08cee789 247 enum gt_types_enum type;
17211ab5
GK
248};
249
250#define POINTER_HASH(x) (hashval_t)((long)x >> 3)
251
252/* Register an object in the hash table. */
253
254int
20c1dc5e 255gt_pch_note_object (void *obj, void *note_ptr_cookie,
08cee789
DJ
256 gt_note_pointers note_ptr_fn,
257 enum gt_types_enum type)
17211ab5
GK
258{
259 struct ptr_data **slot;
20c1dc5e 260
17211ab5
GK
261 if (obj == NULL || obj == (void *) 1)
262 return 0;
263
264 slot = (struct ptr_data **)
265 htab_find_slot_with_hash (saving_htab, obj, POINTER_HASH (obj),
266 INSERT);
267 if (*slot != NULL)
268 {
282899df
NS
269 gcc_assert ((*slot)->note_ptr_fn == note_ptr_fn
270 && (*slot)->note_ptr_cookie == note_ptr_cookie);
17211ab5
GK
271 return 0;
272 }
20c1dc5e 273
17211ab5
GK
274 *slot = xcalloc (sizeof (struct ptr_data), 1);
275 (*slot)->obj = obj;
276 (*slot)->note_ptr_fn = note_ptr_fn;
277 (*slot)->note_ptr_cookie = note_ptr_cookie;
278 if (note_ptr_fn == gt_pch_p_S)
279 (*slot)->size = strlen (obj) + 1;
280 else
281 (*slot)->size = ggc_get_size (obj);
08cee789 282 (*slot)->type = type;
17211ab5
GK
283 return 1;
284}
285
286/* Register an object in the hash table. */
287
288void
20c1dc5e
AJ
289gt_pch_note_reorder (void *obj, void *note_ptr_cookie,
290 gt_handle_reorder reorder_fn)
17211ab5
GK
291{
292 struct ptr_data *data;
20c1dc5e 293
17211ab5
GK
294 if (obj == NULL || obj == (void *) 1)
295 return;
296
297 data = htab_find_with_hash (saving_htab, obj, POINTER_HASH (obj));
282899df 298 gcc_assert (data && data->note_ptr_cookie == note_ptr_cookie);
20c1dc5e 299
17211ab5
GK
300 data->reorder_fn = reorder_fn;
301}
302
303/* Hash and equality functions for saving_htab, callbacks for htab_create. */
304
305static hashval_t
20c1dc5e 306saving_htab_hash (const void *p)
17211ab5
GK
307{
308 return POINTER_HASH (((struct ptr_data *)p)->obj);
309}
310
311static int
20c1dc5e 312saving_htab_eq (const void *p1, const void *p2)
17211ab5
GK
313{
314 return ((struct ptr_data *)p1)->obj == p2;
315}
316
317/* Handy state for the traversal functions. */
318
20c1dc5e 319struct traversal_state
17211ab5
GK
320{
321 FILE *f;
322 struct ggc_pch_data *d;
323 size_t count;
324 struct ptr_data **ptrs;
325 size_t ptrs_i;
326};
327
328/* Callbacks for htab_traverse. */
329
330static int
20c1dc5e 331call_count (void **slot, void *state_p)
17211ab5
GK
332{
333 struct ptr_data *d = (struct ptr_data *)*slot;
334 struct traversal_state *state = (struct traversal_state *)state_p;
20c1dc5e 335
08cee789
DJ
336 ggc_pch_count_object (state->d, d->obj, d->size,
337 d->note_ptr_fn == gt_pch_p_S,
338 d->type);
17211ab5
GK
339 state->count++;
340 return 1;
341}
342
343static int
20c1dc5e 344call_alloc (void **slot, void *state_p)
17211ab5
GK
345{
346 struct ptr_data *d = (struct ptr_data *)*slot;
347 struct traversal_state *state = (struct traversal_state *)state_p;
20c1dc5e 348
08cee789
DJ
349 d->new_addr = ggc_pch_alloc_object (state->d, d->obj, d->size,
350 d->note_ptr_fn == gt_pch_p_S,
351 d->type);
17211ab5
GK
352 state->ptrs[state->ptrs_i++] = d;
353 return 1;
354}
355
356/* Callback for qsort. */
357
358static int
20c1dc5e 359compare_ptr_data (const void *p1_p, const void *p2_p)
17211ab5
GK
360{
361 struct ptr_data *p1 = *(struct ptr_data *const *)p1_p;
362 struct ptr_data *p2 = *(struct ptr_data *const *)p2_p;
363 return (((size_t)p1->new_addr > (size_t)p2->new_addr)
364 - ((size_t)p1->new_addr < (size_t)p2->new_addr));
365}
366
367/* Callbacks for note_ptr_fn. */
368
369static void
20c1dc5e 370relocate_ptrs (void *ptr_p, void *state_p)
17211ab5
GK
371{
372 void **ptr = (void **)ptr_p;
20c1dc5e 373 struct traversal_state *state ATTRIBUTE_UNUSED
17211ab5
GK
374 = (struct traversal_state *)state_p;
375 struct ptr_data *result;
376
377 if (*ptr == NULL || *ptr == (void *)1)
378 return;
20c1dc5e 379
17211ab5 380 result = htab_find_with_hash (saving_htab, *ptr, POINTER_HASH (*ptr));
282899df 381 gcc_assert (result);
17211ab5
GK
382 *ptr = result->new_addr;
383}
384
385/* Write out, after relocation, the pointers in TAB. */
386static void
20c1dc5e
AJ
387write_pch_globals (const struct ggc_root_tab * const *tab,
388 struct traversal_state *state)
17211ab5
GK
389{
390 const struct ggc_root_tab *const *rt;
391 const struct ggc_root_tab *rti;
392 size_t i;
393
394 for (rt = tab; *rt; rt++)
395 for (rti = *rt; rti->base != NULL; rti++)
396 for (i = 0; i < rti->nelt; i++)
397 {
398 void *ptr = *(void **)((char *)rti->base + rti->stride * i);
399 struct ptr_data *new_ptr;
400 if (ptr == NULL || ptr == (void *)1)
401 {
20c1dc5e 402 if (fwrite (&ptr, sizeof (void *), 1, state->f)
17211ab5 403 != 1)
fa6ef813 404 fatal_error ("can't write PCH file: %m");
17211ab5
GK
405 }
406 else
407 {
20c1dc5e 408 new_ptr = htab_find_with_hash (saving_htab, ptr,
17211ab5 409 POINTER_HASH (ptr));
20c1dc5e 410 if (fwrite (&new_ptr->new_addr, sizeof (void *), 1, state->f)
17211ab5 411 != 1)
fa6ef813 412 fatal_error ("can't write PCH file: %m");
17211ab5
GK
413 }
414 }
415}
416
417/* Hold the information we need to mmap the file back in. */
418
20c1dc5e 419struct mmap_info
17211ab5
GK
420{
421 size_t offset;
422 size_t size;
423 void *preferred_base;
424};
425
426/* Write out the state of the compiler to F. */
427
428void
20c1dc5e 429gt_pch_save (FILE *f)
17211ab5
GK
430{
431 const struct ggc_root_tab *const *rt;
432 const struct ggc_root_tab *rti;
433 size_t i;
434 struct traversal_state state;
435 char *this_object = NULL;
436 size_t this_object_size = 0;
437 struct mmap_info mmi;
90aa6719 438 const size_t mmap_offset_alignment = host_hooks.gt_pch_alloc_granularity();
17211ab5
GK
439
440 gt_pch_save_stringpool ();
441
442 saving_htab = htab_create (50000, saving_htab_hash, saving_htab_eq, free);
443
444 for (rt = gt_ggc_rtab; *rt; rt++)
445 for (rti = *rt; rti->base != NULL; rti++)
446 for (i = 0; i < rti->nelt; i++)
447 (*rti->pchw)(*(void **)((char *)rti->base + rti->stride * i));
448
449 for (rt = gt_pch_cache_rtab; *rt; rt++)
450 for (rti = *rt; rti->base != NULL; rti++)
451 for (i = 0; i < rti->nelt; i++)
452 (*rti->pchw)(*(void **)((char *)rti->base + rti->stride * i));
453
454 /* Prepare the objects for writing, determine addresses and such. */
455 state.f = f;
456 state.d = init_ggc_pch();
457 state.count = 0;
458 htab_traverse (saving_htab, call_count, &state);
459
460 mmi.size = ggc_pch_total_size (state.d);
461
18c81520
GK
462 /* Try to arrange things so that no relocation is necessary, but
463 don't try very hard. On most platforms, this will always work,
464 and on the rest it's a lot of work to do better.
465 (The extra work goes in HOST_HOOKS_GT_PCH_GET_ADDRESS and
466 HOST_HOOKS_GT_PCH_USE_ADDRESS.) */
4d0c31e6 467 mmi.preferred_base = host_hooks.gt_pch_get_address (mmi.size, fileno (f));
18c81520 468
17211ab5
GK
469 ggc_pch_this_base (state.d, mmi.preferred_base);
470
5ed6ace5 471 state.ptrs = XNEWVEC (struct ptr_data *, state.count);
17211ab5
GK
472 state.ptrs_i = 0;
473 htab_traverse (saving_htab, call_alloc, &state);
474 qsort (state.ptrs, state.count, sizeof (*state.ptrs), compare_ptr_data);
475
476 /* Write out all the scalar variables. */
477 for (rt = gt_pch_scalar_rtab; *rt; rt++)
478 for (rti = *rt; rti->base != NULL; rti++)
479 if (fwrite (rti->base, rti->stride, 1, f) != 1)
fa6ef813 480 fatal_error ("can't write PCH file: %m");
17211ab5
GK
481
482 /* Write out all the global pointers, after translation. */
483 write_pch_globals (gt_ggc_rtab, &state);
484 write_pch_globals (gt_pch_cache_rtab, &state);
485
90aa6719
DS
486 /* Pad the PCH file so that the mmapped area starts on an allocation
487 granularity (usually page) boundary. */
17211ab5 488 {
70f8b89f
KG
489 long o;
490 o = ftell (state.f) + sizeof (mmi);
491 if (o == -1)
fa6ef813 492 fatal_error ("can't get position in PCH file: %m");
90aa6719
DS
493 mmi.offset = mmap_offset_alignment - o % mmap_offset_alignment;
494 if (mmi.offset == mmap_offset_alignment)
17211ab5
GK
495 mmi.offset = 0;
496 mmi.offset += o;
497 }
498 if (fwrite (&mmi, sizeof (mmi), 1, state.f) != 1)
fa6ef813 499 fatal_error ("can't write PCH file: %m");
17211ab5
GK
500 if (mmi.offset != 0
501 && fseek (state.f, mmi.offset, SEEK_SET) != 0)
fa6ef813 502 fatal_error ("can't write padding to PCH file: %m");
17211ab5 503
08cee789
DJ
504 ggc_pch_prepare_write (state.d, state.f);
505
17211ab5
GK
506 /* Actually write out the objects. */
507 for (i = 0; i < state.count; i++)
3277221c 508 {
17211ab5
GK
509 if (this_object_size < state.ptrs[i]->size)
510 {
511 this_object_size = state.ptrs[i]->size;
512 this_object = xrealloc (this_object, this_object_size);
513 }
514 memcpy (this_object, state.ptrs[i]->obj, state.ptrs[i]->size);
515 if (state.ptrs[i]->reorder_fn != NULL)
20c1dc5e 516 state.ptrs[i]->reorder_fn (state.ptrs[i]->obj,
17211ab5
GK
517 state.ptrs[i]->note_ptr_cookie,
518 relocate_ptrs, &state);
20c1dc5e 519 state.ptrs[i]->note_ptr_fn (state.ptrs[i]->obj,
17211ab5
GK
520 state.ptrs[i]->note_ptr_cookie,
521 relocate_ptrs, &state);
522 ggc_pch_write_object (state.d, state.f, state.ptrs[i]->obj,
4d0c31e6
RH
523 state.ptrs[i]->new_addr, state.ptrs[i]->size,
524 state.ptrs[i]->note_ptr_fn == gt_pch_p_S);
17211ab5
GK
525 if (state.ptrs[i]->note_ptr_fn != gt_pch_p_S)
526 memcpy (state.ptrs[i]->obj, this_object, state.ptrs[i]->size);
3277221c 527 }
17211ab5 528 ggc_pch_finish (state.d, state.f);
d24ecd21 529 gt_pch_fixup_stringpool ();
17211ab5
GK
530
531 free (state.ptrs);
532 htab_delete (saving_htab);
533}
534
535/* Read the state of the compiler back in from F. */
536
537void
20c1dc5e 538gt_pch_restore (FILE *f)
17211ab5
GK
539{
540 const struct ggc_root_tab *const *rt;
541 const struct ggc_root_tab *rti;
542 size_t i;
543 struct mmap_info mmi;
4d0c31e6 544 int result;
17211ab5
GK
545
546 /* Delete any deletable objects. This makes ggc_pch_read much
547 faster, as it can be sure that no GCable objects remain other
548 than the ones just read in. */
549 for (rt = gt_ggc_deletable_rtab; *rt; rt++)
550 for (rti = *rt; rti->base != NULL; rti++)
551 memset (rti->base, 0, rti->stride);
552
553 /* Read in all the scalar variables. */
554 for (rt = gt_pch_scalar_rtab; *rt; rt++)
555 for (rti = *rt; rti->base != NULL; rti++)
556 if (fread (rti->base, rti->stride, 1, f) != 1)
fa6ef813 557 fatal_error ("can't read PCH file: %m");
17211ab5
GK
558
559 /* Read in all the global pointers, in 6 easy loops. */
560 for (rt = gt_ggc_rtab; *rt; rt++)
561 for (rti = *rt; rti->base != NULL; rti++)
562 for (i = 0; i < rti->nelt; i++)
563 if (fread ((char *)rti->base + rti->stride * i,
564 sizeof (void *), 1, f) != 1)
fa6ef813 565 fatal_error ("can't read PCH file: %m");
17211ab5
GK
566
567 for (rt = gt_pch_cache_rtab; *rt; rt++)
568 for (rti = *rt; rti->base != NULL; rti++)
569 for (i = 0; i < rti->nelt; i++)
570 if (fread ((char *)rti->base + rti->stride * i,
571 sizeof (void *), 1, f) != 1)
fa6ef813 572 fatal_error ("can't read PCH file: %m");
17211ab5
GK
573
574 if (fread (&mmi, sizeof (mmi), 1, f) != 1)
fa6ef813 575 fatal_error ("can't read PCH file: %m");
20c1dc5e 576
4d0c31e6
RH
577 result = host_hooks.gt_pch_use_address (mmi.preferred_base, mmi.size,
578 fileno (f), mmi.offset);
579 if (result < 0)
580 fatal_error ("had to relocate PCH");
581 if (result == 0)
18c81520 582 {
4d0c31e6
RH
583 if (fseek (f, mmi.offset, SEEK_SET) != 0
584 || fread (mmi.preferred_base, mmi.size, 1, f) != 1)
585 fatal_error ("can't read PCH file: %m");
586 }
587 else if (fseek (f, mmi.offset + mmi.size, SEEK_SET) != 0)
588 fatal_error ("can't read PCH file: %m");
8eb6a092 589
4d0c31e6 590 ggc_pch_read (f, mmi.preferred_base);
18c81520 591
4d0c31e6
RH
592 gt_pch_restore_stringpool ();
593}
18c81520 594
4d0c31e6
RH
595/* Default version of HOST_HOOKS_GT_PCH_GET_ADDRESS when mmap is not present.
596 Select no address whatsoever, and let gt_pch_save choose what it will with
597 malloc, presumably. */
ee0d75ef 598
4d0c31e6
RH
599void *
600default_gt_pch_get_address (size_t size ATTRIBUTE_UNUSED,
601 int fd ATTRIBUTE_UNUSED)
602{
603 return NULL;
604}
ee0d75ef 605
4d0c31e6
RH
606/* Default version of HOST_HOOKS_GT_PCH_USE_ADDRESS when mmap is not present.
607 Allocate SIZE bytes with malloc. Return 0 if the address we got is the
608 same as base, indicating that the memory has been allocated but needs to
609 be read in from the file. Return -1 if the address differs, to relocation
610 of the PCH file would be required. */
611
612int
613default_gt_pch_use_address (void *base, size_t size, int fd ATTRIBUTE_UNUSED,
614 size_t offset ATTRIBUTE_UNUSED)
615{
616 void *addr = xmalloc (size);
617 return (addr == base) - 1;
618}
ee0d75ef 619
90aa6719
DS
620/* Default version of HOST_HOOKS_GT_PCH_GET_ADDRESS. Return the
621 alignment required for allocating virtual memory. Usually this is the
622 same as pagesize. */
623
624size_t
625default_gt_pch_alloc_granularity (void)
626{
627 return getpagesize();
628}
629
4d0c31e6
RH
630#if HAVE_MMAP_FILE
631/* Default version of HOST_HOOKS_GT_PCH_GET_ADDRESS when mmap is present.
632 We temporarily allocate SIZE bytes, and let the kernel place the data
d1a6adeb 633 wherever it will. If it worked, that's our spot, if not we're likely
4d0c31e6 634 to be in trouble. */
8eb6a092 635
4d0c31e6
RH
636void *
637mmap_gt_pch_get_address (size_t size, int fd)
638{
639 void *ret;
18c81520 640
4d0c31e6
RH
641 ret = mmap (NULL, size, PROT_READ | PROT_WRITE, MAP_PRIVATE, fd, 0);
642 if (ret == (void *) MAP_FAILED)
643 ret = NULL;
644 else
645 munmap (ret, size);
3277221c 646
4d0c31e6
RH
647 return ret;
648}
3277221c 649
4d0c31e6
RH
650/* Default version of HOST_HOOKS_GT_PCH_USE_ADDRESS when mmap is present.
651 Map SIZE bytes of FD+OFFSET at BASE. Return 1 if we succeeded at
652 mapping the data at BASE, -1 if we couldn't.
20c1dc5e 653
4d0c31e6
RH
654 This version assumes that the kernel honors the START operand of mmap
655 even without MAP_FIXED if START through START+SIZE are not currently
656 mapped with something. */
17211ab5 657
4d0c31e6
RH
658int
659mmap_gt_pch_use_address (void *base, size_t size, int fd, size_t offset)
660{
661 void *addr;
17211ab5 662
4d0c31e6
RH
663 /* We're called with size == 0 if we're not planning to load a PCH
664 file at all. This allows the hook to free any static space that
665 we might have allocated at link time. */
666 if (size == 0)
667 return -1;
668
669 addr = mmap (base, size, PROT_READ | PROT_WRITE, MAP_PRIVATE,
670 fd, offset);
671
672 return addr == base ? 1 : -1;
3277221c 673}
4d0c31e6 674#endif /* HAVE_MMAP_FILE */
9ac121af 675
d37e6b50 676/* Modify the bound based on rlimits. */
16226f1e 677static double
20c1dc5e 678ggc_rlimit_bound (double limit)
16226f1e
KG
679{
680#if defined(HAVE_GETRLIMIT)
681 struct rlimit rlim;
d37e6b50
GK
682# if defined (RLIMIT_AS)
683 /* RLIMIT_AS is what POSIX says is the limit on mmap. Presumably
684 any OS which has RLIMIT_AS also has a working mmap that GCC will use. */
685 if (getrlimit (RLIMIT_AS, &rlim) == 0
a2581175 686 && rlim.rlim_cur != (rlim_t) RLIM_INFINITY
16226f1e
KG
687 && rlim.rlim_cur < limit)
688 limit = rlim.rlim_cur;
d37e6b50
GK
689# elif defined (RLIMIT_DATA)
690 /* ... but some older OSs bound mmap based on RLIMIT_DATA, or we
691 might be on an OS that has a broken mmap. (Others don't bound
692 mmap at all, apparently.) */
16226f1e 693 if (getrlimit (RLIMIT_DATA, &rlim) == 0
a2581175 694 && rlim.rlim_cur != (rlim_t) RLIM_INFINITY
d37e6b50
GK
695 && rlim.rlim_cur < limit
696 /* Darwin has this horribly bogus default setting of
697 RLIMIT_DATA, to 6144Kb. No-one notices because RLIMIT_DATA
698 appears to be ignored. Ignore such silliness. If a limit
699 this small was actually effective for mmap, GCC wouldn't even
700 start up. */
701 && rlim.rlim_cur >= 8 * 1024 * 1024)
16226f1e 702 limit = rlim.rlim_cur;
d37e6b50 703# endif /* RLIMIT_AS or RLIMIT_DATA */
16226f1e
KG
704#endif /* HAVE_GETRLIMIT */
705
706 return limit;
707}
708
9ac121af
KG
709/* Heuristic to set a default for GGC_MIN_EXPAND. */
710int
20c1dc5e 711ggc_min_expand_heuristic (void)
9ac121af
KG
712{
713 double min_expand = physmem_total();
16226f1e
KG
714
715 /* Adjust for rlimits. */
716 min_expand = ggc_rlimit_bound (min_expand);
20c1dc5e 717
9ac121af
KG
718 /* The heuristic is a percentage equal to 30% + 70%*(RAM/1GB), yielding
719 a lower bound of 30% and an upper bound of 100% (when RAM >= 1GB). */
720 min_expand /= 1024*1024*1024;
721 min_expand *= 70;
722 min_expand = MIN (min_expand, 70);
723 min_expand += 30;
724
725 return min_expand;
726}
727
728/* Heuristic to set a default for GGC_MIN_HEAPSIZE. */
729int
20c1dc5e 730ggc_min_heapsize_heuristic (void)
9ac121af 731{
d37e6b50
GK
732 double phys_kbytes = physmem_total();
733 double limit_kbytes = ggc_rlimit_bound (phys_kbytes * 2);
16226f1e 734
d37e6b50
GK
735 phys_kbytes /= 1024; /* Convert to Kbytes. */
736 limit_kbytes /= 1024;
20c1dc5e 737
9ac121af
KG
738 /* The heuristic is RAM/8, with a lower bound of 4M and an upper
739 bound of 128M (when RAM >= 1GB). */
d37e6b50
GK
740 phys_kbytes /= 8;
741
742#if defined(HAVE_GETRLIMIT) && defined (RLIMIT_RSS)
743 /* Try not to overrun the RSS limit while doing garbage collection.
744 The RSS limit is only advisory, so no margin is subtracted. */
745 {
746 struct rlimit rlim;
747 if (getrlimit (RLIMIT_RSS, &rlim) == 0
748 && rlim.rlim_cur != (rlim_t) RLIM_INFINITY)
749 phys_kbytes = MIN (phys_kbytes, rlim.rlim_cur / 1024);
750 }
751# endif
752
753 /* Don't blindly run over our data limit; do GC at least when the
ded5f8f4
NF
754 *next* GC would be within 20Mb of the limit or within a quarter of
755 the limit, whichever is larger. If GCC does hit the data limit,
756 compilation will fail, so this tries to be conservative. */
757 limit_kbytes = MAX (0, limit_kbytes - MAX (limit_kbytes / 4, 20 * 1024));
d37e6b50
GK
758 limit_kbytes = (limit_kbytes * 100) / (110 + ggc_min_expand_heuristic());
759 phys_kbytes = MIN (phys_kbytes, limit_kbytes);
760
761 phys_kbytes = MAX (phys_kbytes, 4 * 1024);
762 phys_kbytes = MIN (phys_kbytes, 128 * 1024);
9ac121af 763
d37e6b50 764 return phys_kbytes;
9ac121af
KG
765}
766
767void
20c1dc5e 768init_ggc_heuristics (void)
9ac121af 769{
d85a0aae 770#if !defined ENABLE_GC_CHECKING && !defined ENABLE_GC_ALWAYS_COLLECT
9ac121af
KG
771 set_param_value ("ggc-min-expand", ggc_min_expand_heuristic());
772 set_param_value ("ggc-min-heapsize", ggc_min_heapsize_heuristic());
773#endif
774}
b9dcdee4
JH
775
776#ifdef GATHER_STATISTICS
777
778/* Datastructure used to store per-call-site statistics. */
779struct loc_descriptor
780{
781 const char *file;
782 int line;
783 const char *function;
784 int times;
785 size_t allocated;
786 size_t overhead;
07724022
JH
787 size_t freed;
788 size_t collected;
b9dcdee4
JH
789};
790
791/* Hashtable used for statistics. */
792static htab_t loc_hash;
793
794/* Hash table helpers functions. */
795static hashval_t
796hash_descriptor (const void *p)
797{
798 const struct loc_descriptor *d = p;
799
800 return htab_hash_pointer (d->function) | d->line;
801}
802
803static int
804eq_descriptor (const void *p1, const void *p2)
805{
806 const struct loc_descriptor *d = p1;
807 const struct loc_descriptor *d2 = p2;
808
809 return (d->file == d2->file && d->line == d2->line
810 && d->function == d2->function);
811}
812
07724022
JH
813/* Hashtable converting address of allocated field to loc descriptor. */
814static htab_t ptr_hash;
815struct ptr_hash_entry
816{
817 void *ptr;
818 struct loc_descriptor *loc;
819 size_t size;
820};
821
822/* Hash table helpers functions. */
823static hashval_t
824hash_ptr (const void *p)
825{
826 const struct ptr_hash_entry *d = p;
827
828 return htab_hash_pointer (d->ptr);
829}
830
831static int
832eq_ptr (const void *p1, const void *p2)
833{
834 const struct ptr_hash_entry *p = p1;
835
836 return (p->ptr == p2);
837}
838
b9dcdee4
JH
839/* Return descriptor for given call site, create new one if needed. */
840static struct loc_descriptor *
841loc_descriptor (const char *name, int line, const char *function)
842{
843 struct loc_descriptor loc;
844 struct loc_descriptor **slot;
845
846 loc.file = name;
847 loc.line = line;
848 loc.function = function;
849 if (!loc_hash)
850 loc_hash = htab_create (10, hash_descriptor, eq_descriptor, NULL);
851
852 slot = (struct loc_descriptor **) htab_find_slot (loc_hash, &loc, 1);
853 if (*slot)
854 return *slot;
855 *slot = xcalloc (sizeof (**slot), 1);
856 (*slot)->file = name;
857 (*slot)->line = line;
858 (*slot)->function = function;
859 return *slot;
860}
861
d1a6adeb
KH
862/* Record ALLOCATED and OVERHEAD bytes to descriptor NAME:LINE (FUNCTION). */
863void
07724022 864ggc_record_overhead (size_t allocated, size_t overhead, void *ptr,
d1a6adeb 865 const char *name, int line, const char *function)
b9dcdee4
JH
866{
867 struct loc_descriptor *loc = loc_descriptor (name, line, function);
5ed6ace5 868 struct ptr_hash_entry *p = XNEW (struct ptr_hash_entry);
07724022
JH
869 PTR *slot;
870
871 p->ptr = ptr;
872 p->loc = loc;
873 p->size = allocated + overhead;
874 if (!ptr_hash)
875 ptr_hash = htab_create (10, hash_ptr, eq_ptr, NULL);
876 slot = htab_find_slot_with_hash (ptr_hash, ptr, htab_hash_pointer (ptr), INSERT);
282899df 877 gcc_assert (!*slot);
07724022 878 *slot = p;
b9dcdee4
JH
879
880 loc->times++;
881 loc->allocated+=allocated;
882 loc->overhead+=overhead;
883}
884
07724022
JH
885/* Helper function for prune_overhead_list. See if SLOT is still marked and
886 remove it from hashtable if it is not. */
887static int
888ggc_prune_ptr (void **slot, void *b ATTRIBUTE_UNUSED)
889{
890 struct ptr_hash_entry *p = *slot;
891 if (!ggc_marked_p (p->ptr))
892 {
893 p->loc->collected += p->size;
894 htab_clear_slot (ptr_hash, slot);
895 free (p);
896 }
897 return 1;
898}
899
900/* After live values has been marked, walk all recorded pointers and see if
901 they are still live. */
902void
903ggc_prune_overhead_list (void)
904{
905 htab_traverse (ptr_hash, ggc_prune_ptr, NULL);
906}
907
908/* Notice that the pointer has been freed. */
83f676b3
RS
909void
910ggc_free_overhead (void *ptr)
07724022
JH
911{
912 PTR *slot = htab_find_slot_with_hash (ptr_hash, ptr, htab_hash_pointer (ptr),
913 NO_INSERT);
914 struct ptr_hash_entry *p = *slot;
915 p->loc->freed += p->size;
916 htab_clear_slot (ptr_hash, slot);
917 free (p);
918}
919
b9dcdee4
JH
920/* Helper for qsort; sort descriptors by amount of memory consumed. */
921static int
a5573239 922final_cmp_statistic (const void *loc1, const void *loc2)
b9dcdee4
JH
923{
924 struct loc_descriptor *l1 = *(struct loc_descriptor **) loc1;
925 struct loc_descriptor *l2 = *(struct loc_descriptor **) loc2;
a5573239
JH
926 long diff;
927 diff = ((long)(l1->allocated + l1->overhead - l1->freed) -
85914593 928 (l2->allocated + l2->overhead - l2->freed));
a5573239
JH
929 return diff > 0 ? 1 : diff < 0 ? -1 : 0;
930}
931
932/* Helper for qsort; sort descriptors by amount of memory consumed. */
933static int
934cmp_statistic (const void *loc1, const void *loc2)
935{
936 struct loc_descriptor *l1 = *(struct loc_descriptor **) loc1;
937 struct loc_descriptor *l2 = *(struct loc_descriptor **) loc2;
938 long diff;
939
940 diff = ((long)(l1->allocated + l1->overhead - l1->freed - l1->collected) -
941 (l2->allocated + l2->overhead - l2->freed - l2->collected));
942 if (diff)
943 return diff > 0 ? 1 : diff < 0 ? -1 : 0;
944 diff = ((long)(l1->allocated + l1->overhead - l1->freed) -
945 (l2->allocated + l2->overhead - l2->freed));
946 return diff > 0 ? 1 : diff < 0 ? -1 : 0;
b9dcdee4
JH
947}
948
949/* Collect array of the descriptors from hashtable. */
950struct loc_descriptor **loc_array;
951static int
952add_statistics (void **slot, void *b)
953{
954 int *n = (int *)b;
955 loc_array[*n] = (struct loc_descriptor *) *slot;
956 (*n)++;
957 return 1;
958}
959
960/* Dump per-site memory statistics. */
961#endif
83f676b3 962void
a5573239 963dump_ggc_loc_statistics (bool final ATTRIBUTE_UNUSED)
b9dcdee4
JH
964{
965#ifdef GATHER_STATISTICS
966 int nentries = 0;
967 char s[4096];
07724022 968 size_t collected = 0, freed = 0, allocated = 0, overhead = 0, times = 0;
b9dcdee4
JH
969 int i;
970
07724022
JH
971 ggc_force_collect = true;
972 ggc_collect ();
973
b9dcdee4
JH
974 loc_array = xcalloc (sizeof (*loc_array), loc_hash->n_elements);
975 fprintf (stderr, "-------------------------------------------------------\n");
07724022
JH
976 fprintf (stderr, "\n%-48s %10s %10s %10s %10s %10s\n",
977 "source location", "Garbage", "Freed", "Leak", "Overhead", "Times");
b9dcdee4 978 fprintf (stderr, "-------------------------------------------------------\n");
b9dcdee4 979 htab_traverse (loc_hash, add_statistics, &nentries);
a5573239
JH
980 qsort (loc_array, nentries, sizeof (*loc_array),
981 final ? final_cmp_statistic : cmp_statistic);
b9dcdee4
JH
982 for (i = 0; i < nentries; i++)
983 {
984 struct loc_descriptor *d = loc_array[i];
07724022
JH
985 allocated += d->allocated;
986 times += d->times;
987 freed += d->freed;
988 collected += d->collected;
b9dcdee4
JH
989 overhead += d->overhead;
990 }
991 for (i = 0; i < nentries; i++)
992 {
993 struct loc_descriptor *d = loc_array[i];
994 if (d->allocated)
995 {
996 const char *s1 = d->file;
997 const char *s2;
998 while ((s2 = strstr (s1, "gcc/")))
999 s1 = s2 + 4;
1000 sprintf (s, "%s:%i (%s)", s1, d->line, d->function);
07724022
JH
1001 s[48] = 0;
1002 fprintf (stderr, "%-48s %10li:%4.1f%% %10li:%4.1f%% %10li:%4.1f%% %10li:%4.1f%% %10li\n", s,
1003 (long)d->collected,
1004 (d->collected) * 100.0 / collected,
1005 (long)d->freed,
1006 (d->freed) * 100.0 / freed,
1007 (long)(d->allocated + d->overhead - d->freed - d->collected),
1008 (d->allocated + d->overhead - d->freed - d->collected) * 100.0
1009 / (allocated + overhead - freed - collected),
1010 (long)d->overhead,
1011 d->overhead * 100.0 / overhead,
1012 (long)d->times);
b9dcdee4
JH
1013 }
1014 }
07724022
JH
1015 fprintf (stderr, "%-48s %10ld %10ld %10ld %10ld %10ld\n",
1016 "Total", (long)collected, (long)freed,
1017 (long)(allocated + overhead - freed - collected), (long)overhead,
1018 (long)times);
1019 fprintf (stderr, "%-48s %10s %10s %10s %10s %10s\n",
1020 "source location", "Garbage", "Freed", "Leak", "Overhead", "Times");
b9dcdee4
JH
1021 fprintf (stderr, "-------------------------------------------------------\n");
1022#endif
1023}
This page took 2.183525 seconds and 5 git commands to generate.