40ce7fa6dd
2018-11-05 Martin Liska <mliska@suse.cz> * alloc-pool.h (struct pool_usage): Use SIZE_AMOUNT. * bitmap.h (struct bitmap_usage): Likewise. * ggc-common.c (SCALE): Remove. (LABEL): Likewise. (struct ggc_usage): Use SIZE_AMOUNT. And update compare method. * ggc-page.c (SCALE): Remove. (STAT_LABEL): Remove. (ggc_print_statistics): Use SIZE_AMOUNT. * gimple.h (SCALE): Remove. (LABEL): Likewise. * input.c (ONE_K): Remove. (ONE_M): Likewise. (SCALE): Likewise. (STAT_LABEL): Likewise. (FORMAT_AMOUNT): Likewise. (dump_line_table_statistics): Use SIZE_AMOUNT. * mem-stats.h (struct mem_usage): Likewise. * rtl.c (dump_rtx_statistics): Likewise. (rtx_alloc_counts): Change type to size_t. (rtx_alloc_sizes): Likewise. (rtx_count_cmp): New. (dump_rtx_statistics): Sort first based on counts. * tree.c (tree_nodes_cmp): New. (tree_codes_cmp): New. (dump_tree_statistics): Sort first based on counts. * system.h (ONE_K): New. (ONE_M): Likewise. (SIZE_SCALE): Likewise. (SIZE_LABEL): Likewise. (SIZE_AMOUNT): Likewise. * tree-cfg.c (dump_cfg_stats): Use SIZE_AMOUNT. * tree-dfa.c (dump_dfa_stats): Likewise. * tree-phinodes.c (phinodes_print_statistics): Likewise. * tree-ssanames.c (ssanames_print_statistics): Likewise. * tree.c (dump_tree_statistics): Likewise. * vec.c (struct vec_usage): Likewise. * trans-mem.c (tm_mangle): Enlarge buffer in order to not trigger a -Werror=format-overflow with --enable-gather-detailed-stats. From-SVN: r265800
2647 lines
73 KiB
C
2647 lines
73 KiB
C
/* "Bag-of-pages" garbage collector for the GNU compiler.
|
||
Copyright (C) 1999-2018 Free Software Foundation, Inc.
|
||
|
||
This file is part of GCC.
|
||
|
||
GCC is free software; you can redistribute it and/or modify it under
|
||
the terms of the GNU General Public License as published by the Free
|
||
Software Foundation; either version 3, or (at your option) any later
|
||
version.
|
||
|
||
GCC is distributed in the hope that it will be useful, but WITHOUT ANY
|
||
WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||
FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||
for more details.
|
||
|
||
You should have received a copy of the GNU General Public License
|
||
along with GCC; see the file COPYING3. If not see
|
||
<http://www.gnu.org/licenses/>. */
|
||
|
||
#include "config.h"
|
||
#include "system.h"
|
||
#include "coretypes.h"
|
||
#include "backend.h"
|
||
#include "alias.h"
|
||
#include "tree.h"
|
||
#include "rtl.h"
|
||
#include "memmodel.h"
|
||
#include "tm_p.h"
|
||
#include "diagnostic-core.h"
|
||
#include "flags.h"
|
||
#include "ggc-internal.h"
|
||
#include "timevar.h"
|
||
#include "params.h"
|
||
#include "cgraph.h"
|
||
#include "cfgloop.h"
|
||
#include "plugin.h"
|
||
|
||
/* Prefer MAP_ANON(YMOUS) to /dev/zero, since we don't need to keep a
|
||
file open. Prefer either to valloc. */
|
||
#ifdef HAVE_MMAP_ANON
|
||
# undef HAVE_MMAP_DEV_ZERO
|
||
# define USING_MMAP
|
||
#endif
|
||
|
||
#ifdef HAVE_MMAP_DEV_ZERO
|
||
# define USING_MMAP
|
||
#endif
|
||
|
||
#ifndef USING_MMAP
|
||
#define USING_MALLOC_PAGE_GROUPS
|
||
#endif
|
||
|
||
#if defined(HAVE_MADVISE) && HAVE_DECL_MADVISE && defined(MADV_DONTNEED) \
|
||
&& defined(USING_MMAP)
|
||
# define USING_MADVISE
|
||
#endif
|
||
|
||
/* Strategy:
|
||
|
||
This garbage-collecting allocator allocates objects on one of a set
|
||
of pages. Each page can allocate objects of a single size only;
|
||
available sizes are powers of two starting at four bytes. The size
|
||
of an allocation request is rounded up to the next power of two
|
||
(`order'), and satisfied from the appropriate page.
|
||
|
||
Each page is recorded in a page-entry, which also maintains an
|
||
in-use bitmap of object positions on the page. This allows the
|
||
allocation state of a particular object to be flipped without
|
||
touching the page itself.
|
||
|
||
Each page-entry also has a context depth, which is used to track
|
||
pushing and popping of allocation contexts. Only objects allocated
|
||
in the current (highest-numbered) context may be collected.
|
||
|
||
Page entries are arranged in an array of singly-linked lists. The
|
||
array is indexed by the allocation size, in bits, of the pages on
|
||
it; i.e. all pages on a list allocate objects of the same size.
|
||
Pages are ordered on the list such that all non-full pages precede
|
||
all full pages, with non-full pages arranged in order of decreasing
|
||
context depth.
|
||
|
||
Empty pages (of all orders) are kept on a single page cache list,
|
||
and are considered first when new pages are required; they are
|
||
deallocated at the start of the next collection if they haven't
|
||
been recycled by then. */
|
||
|
||
/* Define GGC_DEBUG_LEVEL to print debugging information.
|
||
0: No debugging output.
|
||
1: GC statistics only.
|
||
2: Page-entry allocations/deallocations as well.
|
||
3: Object allocations as well.
|
||
4: Object marks as well. */
|
||
#define GGC_DEBUG_LEVEL (0)
|
||
|
||
/* A two-level tree is used to look up the page-entry for a given
|
||
pointer. Two chunks of the pointer's bits are extracted to index
|
||
the first and second levels of the tree, as follows:
|
||
|
||
HOST_PAGE_SIZE_BITS
|
||
32 | |
|
||
msb +----------------+----+------+------+ lsb
|
||
| | |
|
||
PAGE_L1_BITS |
|
||
| |
|
||
PAGE_L2_BITS
|
||
|
||
The bottommost HOST_PAGE_SIZE_BITS are ignored, since page-entry
|
||
pages are aligned on system page boundaries. The next most
|
||
significant PAGE_L2_BITS and PAGE_L1_BITS are the second and first
|
||
index values in the lookup table, respectively.
|
||
|
||
For 32-bit architectures and the settings below, there are no
|
||
leftover bits. For architectures with wider pointers, the lookup
|
||
tree points to a list of pages, which must be scanned to find the
|
||
correct one. */
|
||
|
||
#define PAGE_L1_BITS (8)
|
||
#define PAGE_L2_BITS (32 - PAGE_L1_BITS - G.lg_pagesize)
|
||
#define PAGE_L1_SIZE ((uintptr_t) 1 << PAGE_L1_BITS)
|
||
#define PAGE_L2_SIZE ((uintptr_t) 1 << PAGE_L2_BITS)
|
||
|
||
#define LOOKUP_L1(p) \
|
||
(((uintptr_t) (p) >> (32 - PAGE_L1_BITS)) & ((1 << PAGE_L1_BITS) - 1))
|
||
|
||
#define LOOKUP_L2(p) \
|
||
(((uintptr_t) (p) >> G.lg_pagesize) & ((1 << PAGE_L2_BITS) - 1))
|
||
|
||
/* The number of objects per allocation page, for objects on a page of
|
||
the indicated ORDER. */
|
||
#define OBJECTS_PER_PAGE(ORDER) objects_per_page_table[ORDER]
|
||
|
||
/* The number of objects in P. */
|
||
#define OBJECTS_IN_PAGE(P) ((P)->bytes / OBJECT_SIZE ((P)->order))
|
||
|
||
/* The size of an object on a page of the indicated ORDER. */
|
||
#define OBJECT_SIZE(ORDER) object_size_table[ORDER]
|
||
|
||
/* For speed, we avoid doing a general integer divide to locate the
|
||
offset in the allocation bitmap, by precalculating numbers M, S
|
||
such that (O * M) >> S == O / Z (modulo 2^32), for any offset O
|
||
within the page which is evenly divisible by the object size Z. */
|
||
#define DIV_MULT(ORDER) inverse_table[ORDER].mult
|
||
#define DIV_SHIFT(ORDER) inverse_table[ORDER].shift
|
||
#define OFFSET_TO_BIT(OFFSET, ORDER) \
|
||
(((OFFSET) * DIV_MULT (ORDER)) >> DIV_SHIFT (ORDER))
|
||
|
||
/* We use this structure to determine the alignment required for
|
||
allocations. For power-of-two sized allocations, that's not a
|
||
problem, but it does matter for odd-sized allocations.
|
||
We do not care about alignment for floating-point types. */
|
||
|
||
struct max_alignment {
|
||
char c;
|
||
union {
|
||
int64_t i;
|
||
void *p;
|
||
} u;
|
||
};
|
||
|
||
/* The biggest alignment required. */
|
||
|
||
#define MAX_ALIGNMENT (offsetof (struct max_alignment, u))
|
||
|
||
|
||
/* The number of extra orders, not corresponding to power-of-two sized
|
||
objects. */
|
||
|
||
#define NUM_EXTRA_ORDERS ARRAY_SIZE (extra_order_size_table)
|
||
|
||
#define RTL_SIZE(NSLOTS) \
|
||
(RTX_HDR_SIZE + (NSLOTS) * sizeof (rtunion))
|
||
|
||
#define TREE_EXP_SIZE(OPS) \
|
||
(sizeof (struct tree_exp) + ((OPS) - 1) * sizeof (tree))
|
||
|
||
/* The Ith entry is the maximum size of an object to be stored in the
|
||
Ith extra order. Adding a new entry to this array is the *only*
|
||
thing you need to do to add a new special allocation size. */
|
||
|
||
static const size_t extra_order_size_table[] = {
|
||
/* Extra orders for small non-power-of-two multiples of MAX_ALIGNMENT.
|
||
There are a lot of structures with these sizes and explicitly
|
||
listing them risks orders being dropped because they changed size. */
|
||
MAX_ALIGNMENT * 3,
|
||
MAX_ALIGNMENT * 5,
|
||
MAX_ALIGNMENT * 6,
|
||
MAX_ALIGNMENT * 7,
|
||
MAX_ALIGNMENT * 9,
|
||
MAX_ALIGNMENT * 10,
|
||
MAX_ALIGNMENT * 11,
|
||
MAX_ALIGNMENT * 12,
|
||
MAX_ALIGNMENT * 13,
|
||
MAX_ALIGNMENT * 14,
|
||
MAX_ALIGNMENT * 15,
|
||
sizeof (struct tree_decl_non_common),
|
||
sizeof (struct tree_field_decl),
|
||
sizeof (struct tree_parm_decl),
|
||
sizeof (struct tree_var_decl),
|
||
sizeof (struct tree_type_non_common),
|
||
sizeof (struct function),
|
||
sizeof (struct basic_block_def),
|
||
sizeof (struct cgraph_node),
|
||
sizeof (struct loop),
|
||
};
|
||
|
||
/* The total number of orders. */
|
||
|
||
#define NUM_ORDERS (HOST_BITS_PER_PTR + NUM_EXTRA_ORDERS)
|
||
|
||
/* Compute the smallest nonnegative number which when added to X gives
|
||
a multiple of F. */
|
||
|
||
#define ROUND_UP_VALUE(x, f) ((f) - 1 - ((f) - 1 + (x)) % (f))
|
||
|
||
/* Round X to next multiple of the page size */
|
||
|
||
#define PAGE_ALIGN(x) ROUND_UP ((x), G.pagesize)
|
||
|
||
/* The Ith entry is the number of objects on a page or order I. */
|
||
|
||
static unsigned objects_per_page_table[NUM_ORDERS];
|
||
|
||
/* The Ith entry is the size of an object on a page of order I. */
|
||
|
||
static size_t object_size_table[NUM_ORDERS];
|
||
|
||
/* The Ith entry is a pair of numbers (mult, shift) such that
|
||
((k * mult) >> shift) mod 2^32 == (k / OBJECT_SIZE(I)) mod 2^32,
|
||
for all k evenly divisible by OBJECT_SIZE(I). */
|
||
|
||
static struct
|
||
{
|
||
size_t mult;
|
||
unsigned int shift;
|
||
}
|
||
inverse_table[NUM_ORDERS];
|
||
|
||
/* A page_entry records the status of an allocation page. This
|
||
structure is dynamically sized to fit the bitmap in_use_p. */
|
||
struct page_entry
|
||
{
|
||
/* The next page-entry with objects of the same size, or NULL if
|
||
this is the last page-entry. */
|
||
struct page_entry *next;
|
||
|
||
/* The previous page-entry with objects of the same size, or NULL if
|
||
this is the first page-entry. The PREV pointer exists solely to
|
||
keep the cost of ggc_free manageable. */
|
||
struct page_entry *prev;
|
||
|
||
/* The number of bytes allocated. (This will always be a multiple
|
||
of the host system page size.) */
|
||
size_t bytes;
|
||
|
||
/* The address at which the memory is allocated. */
|
||
char *page;
|
||
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
/* Back pointer to the page group this page came from. */
|
||
struct page_group *group;
|
||
#endif
|
||
|
||
/* This is the index in the by_depth varray where this page table
|
||
can be found. */
|
||
unsigned long index_by_depth;
|
||
|
||
/* Context depth of this page. */
|
||
unsigned short context_depth;
|
||
|
||
/* The number of free objects remaining on this page. */
|
||
unsigned short num_free_objects;
|
||
|
||
/* A likely candidate for the bit position of a free object for the
|
||
next allocation from this page. */
|
||
unsigned short next_bit_hint;
|
||
|
||
/* The lg of size of objects allocated from this page. */
|
||
unsigned char order;
|
||
|
||
/* Discarded page? */
|
||
bool discarded;
|
||
|
||
/* A bit vector indicating whether or not objects are in use. The
|
||
Nth bit is one if the Nth object on this page is allocated. This
|
||
array is dynamically sized. */
|
||
unsigned long in_use_p[1];
|
||
};
|
||
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
/* A page_group describes a large allocation from malloc, from which
|
||
we parcel out aligned pages. */
|
||
struct page_group
|
||
{
|
||
/* A linked list of all extant page groups. */
|
||
struct page_group *next;
|
||
|
||
/* The address we received from malloc. */
|
||
char *allocation;
|
||
|
||
/* The size of the block. */
|
||
size_t alloc_size;
|
||
|
||
/* A bitmask of pages in use. */
|
||
unsigned int in_use;
|
||
};
|
||
#endif
|
||
|
||
#if HOST_BITS_PER_PTR <= 32
|
||
|
||
/* On 32-bit hosts, we use a two level page table, as pictured above. */
|
||
typedef page_entry **page_table[PAGE_L1_SIZE];
|
||
|
||
#else
|
||
|
||
/* On 64-bit hosts, we use the same two level page tables plus a linked
|
||
list that disambiguates the top 32-bits. There will almost always be
|
||
exactly one entry in the list. */
|
||
typedef struct page_table_chain
|
||
{
|
||
struct page_table_chain *next;
|
||
size_t high_bits;
|
||
page_entry **table[PAGE_L1_SIZE];
|
||
} *page_table;
|
||
|
||
#endif
|
||
|
||
class finalizer
|
||
{
|
||
public:
|
||
finalizer (void *addr, void (*f)(void *)) : m_addr (addr), m_function (f) {}
|
||
|
||
void *addr () const { return m_addr; }
|
||
|
||
void call () const { m_function (m_addr); }
|
||
|
||
private:
|
||
void *m_addr;
|
||
void (*m_function)(void *);
|
||
};
|
||
|
||
class vec_finalizer
|
||
{
|
||
public:
|
||
vec_finalizer (uintptr_t addr, void (*f)(void *), size_t s, size_t n) :
|
||
m_addr (addr), m_function (f), m_object_size (s), m_n_objects (n) {}
|
||
|
||
void call () const
|
||
{
|
||
for (size_t i = 0; i < m_n_objects; i++)
|
||
m_function (reinterpret_cast<void *> (m_addr + (i * m_object_size)));
|
||
}
|
||
|
||
void *addr () const { return reinterpret_cast<void *> (m_addr); }
|
||
|
||
private:
|
||
uintptr_t m_addr;
|
||
void (*m_function)(void *);
|
||
size_t m_object_size;
|
||
size_t m_n_objects;
|
||
};
|
||
|
||
#ifdef ENABLE_GC_ALWAYS_COLLECT
|
||
/* List of free objects to be verified as actually free on the
|
||
next collection. */
|
||
struct free_object
|
||
{
|
||
void *object;
|
||
struct free_object *next;
|
||
};
|
||
#endif
|
||
|
||
/* The rest of the global variables. */
|
||
static struct ggc_globals
|
||
{
|
||
/* The Nth element in this array is a page with objects of size 2^N.
|
||
If there are any pages with free objects, they will be at the
|
||
head of the list. NULL if there are no page-entries for this
|
||
object size. */
|
||
page_entry *pages[NUM_ORDERS];
|
||
|
||
/* The Nth element in this array is the last page with objects of
|
||
size 2^N. NULL if there are no page-entries for this object
|
||
size. */
|
||
page_entry *page_tails[NUM_ORDERS];
|
||
|
||
/* Lookup table for associating allocation pages with object addresses. */
|
||
page_table lookup;
|
||
|
||
/* The system's page size. */
|
||
size_t pagesize;
|
||
size_t lg_pagesize;
|
||
|
||
/* Bytes currently allocated. */
|
||
size_t allocated;
|
||
|
||
/* Bytes currently allocated at the end of the last collection. */
|
||
size_t allocated_last_gc;
|
||
|
||
/* Total amount of memory mapped. */
|
||
size_t bytes_mapped;
|
||
|
||
/* Bit N set if any allocations have been done at context depth N. */
|
||
unsigned long context_depth_allocations;
|
||
|
||
/* Bit N set if any collections have been done at context depth N. */
|
||
unsigned long context_depth_collections;
|
||
|
||
/* The current depth in the context stack. */
|
||
unsigned short context_depth;
|
||
|
||
/* A file descriptor open to /dev/zero for reading. */
|
||
#if defined (HAVE_MMAP_DEV_ZERO)
|
||
int dev_zero_fd;
|
||
#endif
|
||
|
||
/* A cache of free system pages. */
|
||
page_entry *free_pages;
|
||
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
page_group *page_groups;
|
||
#endif
|
||
|
||
/* The file descriptor for debugging output. */
|
||
FILE *debug_file;
|
||
|
||
/* Current number of elements in use in depth below. */
|
||
unsigned int depth_in_use;
|
||
|
||
/* Maximum number of elements that can be used before resizing. */
|
||
unsigned int depth_max;
|
||
|
||
/* Each element of this array is an index in by_depth where the given
|
||
depth starts. This structure is indexed by that given depth we
|
||
are interested in. */
|
||
unsigned int *depth;
|
||
|
||
/* Current number of elements in use in by_depth below. */
|
||
unsigned int by_depth_in_use;
|
||
|
||
/* Maximum number of elements that can be used before resizing. */
|
||
unsigned int by_depth_max;
|
||
|
||
/* Each element of this array is a pointer to a page_entry, all
|
||
page_entries can be found in here by increasing depth.
|
||
index_by_depth in the page_entry is the index into this data
|
||
structure where that page_entry can be found. This is used to
|
||
speed up finding all page_entries at a particular depth. */
|
||
page_entry **by_depth;
|
||
|
||
/* Each element is a pointer to the saved in_use_p bits, if any,
|
||
zero otherwise. We allocate them all together, to enable a
|
||
better runtime data access pattern. */
|
||
unsigned long **save_in_use;
|
||
|
||
/* Finalizers for single objects. The first index is collection_depth. */
|
||
vec<vec<finalizer> > finalizers;
|
||
|
||
/* Finalizers for vectors of objects. */
|
||
vec<vec<vec_finalizer> > vec_finalizers;
|
||
|
||
#ifdef ENABLE_GC_ALWAYS_COLLECT
|
||
/* List of free objects to be verified as actually free on the
|
||
next collection. */
|
||
struct free_object *free_object_list;
|
||
#endif
|
||
|
||
struct
|
||
{
|
||
/* Total GC-allocated memory. */
|
||
unsigned long long total_allocated;
|
||
/* Total overhead for GC-allocated memory. */
|
||
unsigned long long total_overhead;
|
||
|
||
/* Total allocations and overhead for sizes less than 32, 64 and 128.
|
||
These sizes are interesting because they are typical cache line
|
||
sizes. */
|
||
|
||
unsigned long long total_allocated_under32;
|
||
unsigned long long total_overhead_under32;
|
||
|
||
unsigned long long total_allocated_under64;
|
||
unsigned long long total_overhead_under64;
|
||
|
||
unsigned long long total_allocated_under128;
|
||
unsigned long long total_overhead_under128;
|
||
|
||
/* The allocations for each of the allocation orders. */
|
||
unsigned long long total_allocated_per_order[NUM_ORDERS];
|
||
|
||
/* The overhead for each of the allocation orders. */
|
||
unsigned long long total_overhead_per_order[NUM_ORDERS];
|
||
} stats;
|
||
} G;
|
||
|
||
/* True if a gc is currently taking place. */
|
||
|
||
static bool in_gc = false;
|
||
|
||
/* The size in bytes required to maintain a bitmap for the objects
|
||
on a page-entry. */
|
||
#define BITMAP_SIZE(Num_objects) \
|
||
(CEIL ((Num_objects), HOST_BITS_PER_LONG) * sizeof (long))
|
||
|
||
/* Allocate pages in chunks of this size, to throttle calls to memory
|
||
allocation routines. The first page is used, the rest go onto the
|
||
free list. This cannot be larger than HOST_BITS_PER_INT for the
|
||
in_use bitmask for page_group. Hosts that need a different value
|
||
can override this by defining GGC_QUIRE_SIZE explicitly. */
|
||
#ifndef GGC_QUIRE_SIZE
|
||
# ifdef USING_MMAP
|
||
# define GGC_QUIRE_SIZE 512 /* 2MB for 4K pages */
|
||
# else
|
||
# define GGC_QUIRE_SIZE 16
|
||
# endif
|
||
#endif
|
||
|
||
/* Initial guess as to how many page table entries we might need. */
|
||
#define INITIAL_PTE_COUNT 128
|
||
|
||
static page_entry *lookup_page_table_entry (const void *);
|
||
static void set_page_table_entry (void *, page_entry *);
|
||
#ifdef USING_MMAP
|
||
static char *alloc_anon (char *, size_t, bool check);
|
||
#endif
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
static size_t page_group_index (char *, char *);
|
||
static void set_page_group_in_use (page_group *, char *);
|
||
static void clear_page_group_in_use (page_group *, char *);
|
||
#endif
|
||
static struct page_entry * alloc_page (unsigned);
|
||
static void free_page (struct page_entry *);
|
||
static void release_pages (void);
|
||
static void clear_marks (void);
|
||
static void sweep_pages (void);
|
||
static void ggc_recalculate_in_use_p (page_entry *);
|
||
static void compute_inverse (unsigned);
|
||
static inline void adjust_depth (void);
|
||
static void move_ptes_to_front (int, int);
|
||
|
||
void debug_print_page_list (int);
|
||
static void push_depth (unsigned int);
|
||
static void push_by_depth (page_entry *, unsigned long *);
|
||
|
||
/* Push an entry onto G.depth. */
|
||
|
||
inline static void
|
||
push_depth (unsigned int i)
|
||
{
|
||
if (G.depth_in_use >= G.depth_max)
|
||
{
|
||
G.depth_max *= 2;
|
||
G.depth = XRESIZEVEC (unsigned int, G.depth, G.depth_max);
|
||
}
|
||
G.depth[G.depth_in_use++] = i;
|
||
}
|
||
|
||
/* Push an entry onto G.by_depth and G.save_in_use. */
|
||
|
||
inline static void
|
||
push_by_depth (page_entry *p, unsigned long *s)
|
||
{
|
||
if (G.by_depth_in_use >= G.by_depth_max)
|
||
{
|
||
G.by_depth_max *= 2;
|
||
G.by_depth = XRESIZEVEC (page_entry *, G.by_depth, G.by_depth_max);
|
||
G.save_in_use = XRESIZEVEC (unsigned long *, G.save_in_use,
|
||
G.by_depth_max);
|
||
}
|
||
G.by_depth[G.by_depth_in_use] = p;
|
||
G.save_in_use[G.by_depth_in_use++] = s;
|
||
}
|
||
|
||
#if (GCC_VERSION < 3001)
|
||
#define prefetch(X) ((void) X)
|
||
#else
|
||
#define prefetch(X) __builtin_prefetch (X)
|
||
#endif
|
||
|
||
#define save_in_use_p_i(__i) \
|
||
(G.save_in_use[__i])
|
||
#define save_in_use_p(__p) \
|
||
(save_in_use_p_i (__p->index_by_depth))
|
||
|
||
/* Traverse the page table and find the entry for a page.
|
||
If the object wasn't allocated in GC return NULL. */
|
||
|
||
static inline page_entry *
|
||
safe_lookup_page_table_entry (const void *p)
|
||
{
|
||
page_entry ***base;
|
||
size_t L1, L2;
|
||
|
||
#if HOST_BITS_PER_PTR <= 32
|
||
base = &G.lookup[0];
|
||
#else
|
||
page_table table = G.lookup;
|
||
uintptr_t high_bits = (uintptr_t) p & ~ (uintptr_t) 0xffffffff;
|
||
while (1)
|
||
{
|
||
if (table == NULL)
|
||
return NULL;
|
||
if (table->high_bits == high_bits)
|
||
break;
|
||
table = table->next;
|
||
}
|
||
base = &table->table[0];
|
||
#endif
|
||
|
||
/* Extract the level 1 and 2 indices. */
|
||
L1 = LOOKUP_L1 (p);
|
||
L2 = LOOKUP_L2 (p);
|
||
if (! base[L1])
|
||
return NULL;
|
||
|
||
return base[L1][L2];
|
||
}
|
||
|
||
/* Traverse the page table and find the entry for a page.
|
||
Die (probably) if the object wasn't allocated via GC. */
|
||
|
||
static inline page_entry *
|
||
lookup_page_table_entry (const void *p)
|
||
{
|
||
page_entry ***base;
|
||
size_t L1, L2;
|
||
|
||
#if HOST_BITS_PER_PTR <= 32
|
||
base = &G.lookup[0];
|
||
#else
|
||
page_table table = G.lookup;
|
||
uintptr_t high_bits = (uintptr_t) p & ~ (uintptr_t) 0xffffffff;
|
||
while (table->high_bits != high_bits)
|
||
table = table->next;
|
||
base = &table->table[0];
|
||
#endif
|
||
|
||
/* Extract the level 1 and 2 indices. */
|
||
L1 = LOOKUP_L1 (p);
|
||
L2 = LOOKUP_L2 (p);
|
||
|
||
return base[L1][L2];
|
||
}
|
||
|
||
/* Set the page table entry for a page. */
|
||
|
||
static void
|
||
set_page_table_entry (void *p, page_entry *entry)
|
||
{
|
||
page_entry ***base;
|
||
size_t L1, L2;
|
||
|
||
#if HOST_BITS_PER_PTR <= 32
|
||
base = &G.lookup[0];
|
||
#else
|
||
page_table table;
|
||
uintptr_t high_bits = (uintptr_t) p & ~ (uintptr_t) 0xffffffff;
|
||
for (table = G.lookup; table; table = table->next)
|
||
if (table->high_bits == high_bits)
|
||
goto found;
|
||
|
||
/* Not found -- allocate a new table. */
|
||
table = XCNEW (struct page_table_chain);
|
||
table->next = G.lookup;
|
||
table->high_bits = high_bits;
|
||
G.lookup = table;
|
||
found:
|
||
base = &table->table[0];
|
||
#endif
|
||
|
||
/* Extract the level 1 and 2 indices. */
|
||
L1 = LOOKUP_L1 (p);
|
||
L2 = LOOKUP_L2 (p);
|
||
|
||
if (base[L1] == NULL)
|
||
base[L1] = XCNEWVEC (page_entry *, PAGE_L2_SIZE);
|
||
|
||
base[L1][L2] = entry;
|
||
}
|
||
|
||
/* Prints the page-entry for object size ORDER, for debugging. */
|
||
|
||
DEBUG_FUNCTION void
|
||
debug_print_page_list (int order)
|
||
{
|
||
page_entry *p;
|
||
printf ("Head=%p, Tail=%p:\n", (void *) G.pages[order],
|
||
(void *) G.page_tails[order]);
|
||
p = G.pages[order];
|
||
while (p != NULL)
|
||
{
|
||
printf ("%p(%1d|%3d) -> ", (void *) p, p->context_depth,
|
||
p->num_free_objects);
|
||
p = p->next;
|
||
}
|
||
printf ("NULL\n");
|
||
fflush (stdout);
|
||
}
|
||
|
||
#ifdef USING_MMAP
|
||
/* Allocate SIZE bytes of anonymous memory, preferably near PREF,
|
||
(if non-null). The ifdef structure here is intended to cause a
|
||
compile error unless exactly one of the HAVE_* is defined. */
|
||
|
||
static inline char *
|
||
alloc_anon (char *pref ATTRIBUTE_UNUSED, size_t size, bool check)
|
||
{
|
||
#ifdef HAVE_MMAP_ANON
|
||
char *page = (char *) mmap (pref, size, PROT_READ | PROT_WRITE,
|
||
MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
|
||
#endif
|
||
#ifdef HAVE_MMAP_DEV_ZERO
|
||
char *page = (char *) mmap (pref, size, PROT_READ | PROT_WRITE,
|
||
MAP_PRIVATE, G.dev_zero_fd, 0);
|
||
#endif
|
||
|
||
if (page == (char *) MAP_FAILED)
|
||
{
|
||
if (!check)
|
||
return NULL;
|
||
perror ("virtual memory exhausted");
|
||
exit (FATAL_EXIT_CODE);
|
||
}
|
||
|
||
/* Remember that we allocated this memory. */
|
||
G.bytes_mapped += size;
|
||
|
||
/* Pretend we don't have access to the allocated pages. We'll enable
|
||
access to smaller pieces of the area in ggc_internal_alloc. Discard the
|
||
handle to avoid handle leak. */
|
||
VALGRIND_DISCARD (VALGRIND_MAKE_MEM_NOACCESS (page, size));
|
||
|
||
return page;
|
||
}
|
||
#endif
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
/* Compute the index for this page into the page group. */
|
||
|
||
static inline size_t
|
||
page_group_index (char *allocation, char *page)
|
||
{
|
||
return (size_t) (page - allocation) >> G.lg_pagesize;
|
||
}
|
||
|
||
/* Set and clear the in_use bit for this page in the page group. */
|
||
|
||
static inline void
|
||
set_page_group_in_use (page_group *group, char *page)
|
||
{
|
||
group->in_use |= 1 << page_group_index (group->allocation, page);
|
||
}
|
||
|
||
static inline void
|
||
clear_page_group_in_use (page_group *group, char *page)
|
||
{
|
||
group->in_use &= ~(1 << page_group_index (group->allocation, page));
|
||
}
|
||
#endif
|
||
|
||
/* Allocate a new page for allocating objects of size 2^ORDER,
|
||
and return an entry for it. The entry is not added to the
|
||
appropriate page_table list. */
|
||
|
||
static inline struct page_entry *
|
||
alloc_page (unsigned order)
|
||
{
|
||
struct page_entry *entry, *p, **pp;
|
||
char *page;
|
||
size_t num_objects;
|
||
size_t bitmap_size;
|
||
size_t page_entry_size;
|
||
size_t entry_size;
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
page_group *group;
|
||
#endif
|
||
|
||
num_objects = OBJECTS_PER_PAGE (order);
|
||
bitmap_size = BITMAP_SIZE (num_objects + 1);
|
||
page_entry_size = sizeof (page_entry) - sizeof (long) + bitmap_size;
|
||
entry_size = num_objects * OBJECT_SIZE (order);
|
||
if (entry_size < G.pagesize)
|
||
entry_size = G.pagesize;
|
||
entry_size = PAGE_ALIGN (entry_size);
|
||
|
||
entry = NULL;
|
||
page = NULL;
|
||
|
||
/* Check the list of free pages for one we can use. */
|
||
for (pp = &G.free_pages, p = *pp; p; pp = &p->next, p = *pp)
|
||
if (p->bytes == entry_size)
|
||
break;
|
||
|
||
if (p != NULL)
|
||
{
|
||
if (p->discarded)
|
||
G.bytes_mapped += p->bytes;
|
||
p->discarded = false;
|
||
|
||
/* Recycle the allocated memory from this page ... */
|
||
*pp = p->next;
|
||
page = p->page;
|
||
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
group = p->group;
|
||
#endif
|
||
|
||
/* ... and, if possible, the page entry itself. */
|
||
if (p->order == order)
|
||
{
|
||
entry = p;
|
||
memset (entry, 0, page_entry_size);
|
||
}
|
||
else
|
||
free (p);
|
||
}
|
||
#ifdef USING_MMAP
|
||
else if (entry_size == G.pagesize)
|
||
{
|
||
/* We want just one page. Allocate a bunch of them and put the
|
||
extras on the freelist. (Can only do this optimization with
|
||
mmap for backing store.) */
|
||
struct page_entry *e, *f = G.free_pages;
|
||
int i, entries = GGC_QUIRE_SIZE;
|
||
|
||
page = alloc_anon (NULL, G.pagesize * GGC_QUIRE_SIZE, false);
|
||
if (page == NULL)
|
||
{
|
||
page = alloc_anon (NULL, G.pagesize, true);
|
||
entries = 1;
|
||
}
|
||
|
||
/* This loop counts down so that the chain will be in ascending
|
||
memory order. */
|
||
for (i = entries - 1; i >= 1; i--)
|
||
{
|
||
e = XCNEWVAR (struct page_entry, page_entry_size);
|
||
e->order = order;
|
||
e->bytes = G.pagesize;
|
||
e->page = page + (i << G.lg_pagesize);
|
||
e->next = f;
|
||
f = e;
|
||
}
|
||
|
||
G.free_pages = f;
|
||
}
|
||
else
|
||
page = alloc_anon (NULL, entry_size, true);
|
||
#endif
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
else
|
||
{
|
||
/* Allocate a large block of memory and serve out the aligned
|
||
pages therein. This results in much less memory wastage
|
||
than the traditional implementation of valloc. */
|
||
|
||
char *allocation, *a, *enda;
|
||
size_t alloc_size, head_slop, tail_slop;
|
||
int multiple_pages = (entry_size == G.pagesize);
|
||
|
||
if (multiple_pages)
|
||
alloc_size = GGC_QUIRE_SIZE * G.pagesize;
|
||
else
|
||
alloc_size = entry_size + G.pagesize - 1;
|
||
allocation = XNEWVEC (char, alloc_size);
|
||
|
||
page = (char *) (((uintptr_t) allocation + G.pagesize - 1) & -G.pagesize);
|
||
head_slop = page - allocation;
|
||
if (multiple_pages)
|
||
tail_slop = ((size_t) allocation + alloc_size) & (G.pagesize - 1);
|
||
else
|
||
tail_slop = alloc_size - entry_size - head_slop;
|
||
enda = allocation + alloc_size - tail_slop;
|
||
|
||
/* We allocated N pages, which are likely not aligned, leaving
|
||
us with N-1 usable pages. We plan to place the page_group
|
||
structure somewhere in the slop. */
|
||
if (head_slop >= sizeof (page_group))
|
||
group = (page_group *)page - 1;
|
||
else
|
||
{
|
||
/* We magically got an aligned allocation. Too bad, we have
|
||
to waste a page anyway. */
|
||
if (tail_slop == 0)
|
||
{
|
||
enda -= G.pagesize;
|
||
tail_slop += G.pagesize;
|
||
}
|
||
gcc_assert (tail_slop >= sizeof (page_group));
|
||
group = (page_group *)enda;
|
||
tail_slop -= sizeof (page_group);
|
||
}
|
||
|
||
/* Remember that we allocated this memory. */
|
||
group->next = G.page_groups;
|
||
group->allocation = allocation;
|
||
group->alloc_size = alloc_size;
|
||
group->in_use = 0;
|
||
G.page_groups = group;
|
||
G.bytes_mapped += alloc_size;
|
||
|
||
/* If we allocated multiple pages, put the rest on the free list. */
|
||
if (multiple_pages)
|
||
{
|
||
struct page_entry *e, *f = G.free_pages;
|
||
for (a = enda - G.pagesize; a != page; a -= G.pagesize)
|
||
{
|
||
e = XCNEWVAR (struct page_entry, page_entry_size);
|
||
e->order = order;
|
||
e->bytes = G.pagesize;
|
||
e->page = a;
|
||
e->group = group;
|
||
e->next = f;
|
||
f = e;
|
||
}
|
||
G.free_pages = f;
|
||
}
|
||
}
|
||
#endif
|
||
|
||
if (entry == NULL)
|
||
entry = XCNEWVAR (struct page_entry, page_entry_size);
|
||
|
||
entry->bytes = entry_size;
|
||
entry->page = page;
|
||
entry->context_depth = G.context_depth;
|
||
entry->order = order;
|
||
entry->num_free_objects = num_objects;
|
||
entry->next_bit_hint = 1;
|
||
|
||
G.context_depth_allocations |= (unsigned long)1 << G.context_depth;
|
||
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
entry->group = group;
|
||
set_page_group_in_use (group, page);
|
||
#endif
|
||
|
||
/* Set the one-past-the-end in-use bit. This acts as a sentry as we
|
||
increment the hint. */
|
||
entry->in_use_p[num_objects / HOST_BITS_PER_LONG]
|
||
= (unsigned long) 1 << (num_objects % HOST_BITS_PER_LONG);
|
||
|
||
set_page_table_entry (page, entry);
|
||
|
||
if (GGC_DEBUG_LEVEL >= 2)
|
||
fprintf (G.debug_file,
|
||
"Allocating page at %p, object size=%lu, data %p-%p\n",
|
||
(void *) entry, (unsigned long) OBJECT_SIZE (order), page,
|
||
page + entry_size - 1);
|
||
|
||
return entry;
|
||
}
|
||
|
||
/* Adjust the size of G.depth so that no index greater than the one
|
||
used by the top of the G.by_depth is used. */
|
||
|
||
static inline void
|
||
adjust_depth (void)
|
||
{
|
||
page_entry *top;
|
||
|
||
if (G.by_depth_in_use)
|
||
{
|
||
top = G.by_depth[G.by_depth_in_use-1];
|
||
|
||
/* Peel back indices in depth that index into by_depth, so that
|
||
as new elements are added to by_depth, we note the indices
|
||
of those elements, if they are for new context depths. */
|
||
while (G.depth_in_use > (size_t)top->context_depth+1)
|
||
--G.depth_in_use;
|
||
}
|
||
}
|
||
|
||
/* For a page that is no longer needed, put it on the free page list. */
|
||
|
||
static void
|
||
free_page (page_entry *entry)
|
||
{
|
||
if (GGC_DEBUG_LEVEL >= 2)
|
||
fprintf (G.debug_file,
|
||
"Deallocating page at %p, data %p-%p\n", (void *) entry,
|
||
entry->page, entry->page + entry->bytes - 1);
|
||
|
||
/* Mark the page as inaccessible. Discard the handle to avoid handle
|
||
leak. */
|
||
VALGRIND_DISCARD (VALGRIND_MAKE_MEM_NOACCESS (entry->page, entry->bytes));
|
||
|
||
set_page_table_entry (entry->page, NULL);
|
||
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
clear_page_group_in_use (entry->group, entry->page);
|
||
#endif
|
||
|
||
if (G.by_depth_in_use > 1)
|
||
{
|
||
page_entry *top = G.by_depth[G.by_depth_in_use-1];
|
||
int i = entry->index_by_depth;
|
||
|
||
/* We cannot free a page from a context deeper than the current
|
||
one. */
|
||
gcc_assert (entry->context_depth == top->context_depth);
|
||
|
||
/* Put top element into freed slot. */
|
||
G.by_depth[i] = top;
|
||
G.save_in_use[i] = G.save_in_use[G.by_depth_in_use-1];
|
||
top->index_by_depth = i;
|
||
}
|
||
--G.by_depth_in_use;
|
||
|
||
adjust_depth ();
|
||
|
||
entry->next = G.free_pages;
|
||
G.free_pages = entry;
|
||
}
|
||
|
||
/* Release the free page cache to the system. */
|
||
|
||
static void
|
||
release_pages (void)
|
||
{
|
||
#ifdef USING_MADVISE
|
||
page_entry *p, *start_p;
|
||
char *start;
|
||
size_t len;
|
||
size_t mapped_len;
|
||
page_entry *next, *prev, *newprev;
|
||
size_t free_unit = (GGC_QUIRE_SIZE/2) * G.pagesize;
|
||
|
||
/* First free larger continuous areas to the OS.
|
||
This allows other allocators to grab these areas if needed.
|
||
This is only done on larger chunks to avoid fragmentation.
|
||
This does not always work because the free_pages list is only
|
||
approximately sorted. */
|
||
|
||
p = G.free_pages;
|
||
prev = NULL;
|
||
while (p)
|
||
{
|
||
start = p->page;
|
||
start_p = p;
|
||
len = 0;
|
||
mapped_len = 0;
|
||
newprev = prev;
|
||
while (p && p->page == start + len)
|
||
{
|
||
len += p->bytes;
|
||
if (!p->discarded)
|
||
mapped_len += p->bytes;
|
||
newprev = p;
|
||
p = p->next;
|
||
}
|
||
if (len >= free_unit)
|
||
{
|
||
while (start_p != p)
|
||
{
|
||
next = start_p->next;
|
||
free (start_p);
|
||
start_p = next;
|
||
}
|
||
munmap (start, len);
|
||
if (prev)
|
||
prev->next = p;
|
||
else
|
||
G.free_pages = p;
|
||
G.bytes_mapped -= mapped_len;
|
||
continue;
|
||
}
|
||
prev = newprev;
|
||
}
|
||
|
||
/* Now give back the fragmented pages to the OS, but keep the address
|
||
space to reuse it next time. */
|
||
|
||
for (p = G.free_pages; p; )
|
||
{
|
||
if (p->discarded)
|
||
{
|
||
p = p->next;
|
||
continue;
|
||
}
|
||
start = p->page;
|
||
len = p->bytes;
|
||
start_p = p;
|
||
p = p->next;
|
||
while (p && p->page == start + len)
|
||
{
|
||
len += p->bytes;
|
||
p = p->next;
|
||
}
|
||
/* Give the page back to the kernel, but don't free the mapping.
|
||
This avoids fragmentation in the virtual memory map of the
|
||
process. Next time we can reuse it by just touching it. */
|
||
madvise (start, len, MADV_DONTNEED);
|
||
/* Don't count those pages as mapped to not touch the garbage collector
|
||
unnecessarily. */
|
||
G.bytes_mapped -= len;
|
||
while (start_p != p)
|
||
{
|
||
start_p->discarded = true;
|
||
start_p = start_p->next;
|
||
}
|
||
}
|
||
#endif
|
||
#if defined(USING_MMAP) && !defined(USING_MADVISE)
|
||
page_entry *p, *next;
|
||
char *start;
|
||
size_t len;
|
||
|
||
/* Gather up adjacent pages so they are unmapped together. */
|
||
p = G.free_pages;
|
||
|
||
while (p)
|
||
{
|
||
start = p->page;
|
||
next = p->next;
|
||
len = p->bytes;
|
||
free (p);
|
||
p = next;
|
||
|
||
while (p && p->page == start + len)
|
||
{
|
||
next = p->next;
|
||
len += p->bytes;
|
||
free (p);
|
||
p = next;
|
||
}
|
||
|
||
munmap (start, len);
|
||
G.bytes_mapped -= len;
|
||
}
|
||
|
||
G.free_pages = NULL;
|
||
#endif
|
||
#ifdef USING_MALLOC_PAGE_GROUPS
|
||
page_entry **pp, *p;
|
||
page_group **gp, *g;
|
||
|
||
/* Remove all pages from free page groups from the list. */
|
||
pp = &G.free_pages;
|
||
while ((p = *pp) != NULL)
|
||
if (p->group->in_use == 0)
|
||
{
|
||
*pp = p->next;
|
||
free (p);
|
||
}
|
||
else
|
||
pp = &p->next;
|
||
|
||
/* Remove all free page groups, and release the storage. */
|
||
gp = &G.page_groups;
|
||
while ((g = *gp) != NULL)
|
||
if (g->in_use == 0)
|
||
{
|
||
*gp = g->next;
|
||
G.bytes_mapped -= g->alloc_size;
|
||
free (g->allocation);
|
||
}
|
||
else
|
||
gp = &g->next;
|
||
#endif
|
||
}
|
||
|
||
/* This table provides a fast way to determine ceil(log_2(size)) for
|
||
allocation requests. The minimum allocation size is eight bytes. */
|
||
#define NUM_SIZE_LOOKUP 512
|
||
static unsigned char size_lookup[NUM_SIZE_LOOKUP] =
|
||
{
|
||
3, 3, 3, 3, 3, 3, 3, 3, 3, 4, 4, 4, 4, 4, 4, 4,
|
||
4, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5, 5,
|
||
5, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6,
|
||
6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6, 6,
|
||
6, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||
7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7,
|
||
7, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||
8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, 8,
|
||
8, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9,
|
||
9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9, 9
|
||
};
|
||
|
||
/* For a given size of memory requested for allocation, return the
|
||
actual size that is going to be allocated, as well as the size
|
||
order. */
|
||
|
||
static void
|
||
ggc_round_alloc_size_1 (size_t requested_size,
|
||
size_t *size_order,
|
||
size_t *alloced_size)
|
||
{
|
||
size_t order, object_size;
|
||
|
||
if (requested_size < NUM_SIZE_LOOKUP)
|
||
{
|
||
order = size_lookup[requested_size];
|
||
object_size = OBJECT_SIZE (order);
|
||
}
|
||
else
|
||
{
|
||
order = 10;
|
||
while (requested_size > (object_size = OBJECT_SIZE (order)))
|
||
order++;
|
||
}
|
||
|
||
if (size_order)
|
||
*size_order = order;
|
||
if (alloced_size)
|
||
*alloced_size = object_size;
|
||
}
|
||
|
||
/* For a given size of memory requested for allocation, return the
|
||
actual size that is going to be allocated. */
|
||
|
||
size_t
|
||
ggc_round_alloc_size (size_t requested_size)
|
||
{
|
||
size_t size = 0;
|
||
|
||
ggc_round_alloc_size_1 (requested_size, NULL, &size);
|
||
return size;
|
||
}
|
||
|
||
/* Push a finalizer onto the appropriate vec. */
|
||
|
||
static void
|
||
add_finalizer (void *result, void (*f)(void *), size_t s, size_t n)
|
||
{
|
||
if (f == NULL)
|
||
/* No finalizer. */;
|
||
else if (n == 1)
|
||
{
|
||
finalizer fin (result, f);
|
||
G.finalizers[G.context_depth].safe_push (fin);
|
||
}
|
||
else
|
||
{
|
||
vec_finalizer fin (reinterpret_cast<uintptr_t> (result), f, s, n);
|
||
G.vec_finalizers[G.context_depth].safe_push (fin);
|
||
}
|
||
}
|
||
|
||
/* Allocate a chunk of memory of SIZE bytes. Its contents are undefined. */
|
||
|
||
void *
|
||
ggc_internal_alloc (size_t size, void (*f)(void *), size_t s, size_t n
|
||
MEM_STAT_DECL)
|
||
{
|
||
size_t order, word, bit, object_offset, object_size;
|
||
struct page_entry *entry;
|
||
void *result;
|
||
|
||
ggc_round_alloc_size_1 (size, &order, &object_size);
|
||
|
||
/* If there are non-full pages for this size allocation, they are at
|
||
the head of the list. */
|
||
entry = G.pages[order];
|
||
|
||
/* If there is no page for this object size, or all pages in this
|
||
context are full, allocate a new page. */
|
||
if (entry == NULL || entry->num_free_objects == 0)
|
||
{
|
||
struct page_entry *new_entry;
|
||
new_entry = alloc_page (order);
|
||
|
||
new_entry->index_by_depth = G.by_depth_in_use;
|
||
push_by_depth (new_entry, 0);
|
||
|
||
/* We can skip context depths, if we do, make sure we go all the
|
||
way to the new depth. */
|
||
while (new_entry->context_depth >= G.depth_in_use)
|
||
push_depth (G.by_depth_in_use-1);
|
||
|
||
/* If this is the only entry, it's also the tail. If it is not
|
||
the only entry, then we must update the PREV pointer of the
|
||
ENTRY (G.pages[order]) to point to our new page entry. */
|
||
if (entry == NULL)
|
||
G.page_tails[order] = new_entry;
|
||
else
|
||
entry->prev = new_entry;
|
||
|
||
/* Put new pages at the head of the page list. By definition the
|
||
entry at the head of the list always has a NULL pointer. */
|
||
new_entry->next = entry;
|
||
new_entry->prev = NULL;
|
||
entry = new_entry;
|
||
G.pages[order] = new_entry;
|
||
|
||
/* For a new page, we know the word and bit positions (in the
|
||
in_use bitmap) of the first available object -- they're zero. */
|
||
new_entry->next_bit_hint = 1;
|
||
word = 0;
|
||
bit = 0;
|
||
object_offset = 0;
|
||
}
|
||
else
|
||
{
|
||
/* First try to use the hint left from the previous allocation
|
||
to locate a clear bit in the in-use bitmap. We've made sure
|
||
that the one-past-the-end bit is always set, so if the hint
|
||
has run over, this test will fail. */
|
||
unsigned hint = entry->next_bit_hint;
|
||
word = hint / HOST_BITS_PER_LONG;
|
||
bit = hint % HOST_BITS_PER_LONG;
|
||
|
||
/* If the hint didn't work, scan the bitmap from the beginning. */
|
||
if ((entry->in_use_p[word] >> bit) & 1)
|
||
{
|
||
word = bit = 0;
|
||
while (~entry->in_use_p[word] == 0)
|
||
++word;
|
||
|
||
#if GCC_VERSION >= 3004
|
||
bit = __builtin_ctzl (~entry->in_use_p[word]);
|
||
#else
|
||
while ((entry->in_use_p[word] >> bit) & 1)
|
||
++bit;
|
||
#endif
|
||
|
||
hint = word * HOST_BITS_PER_LONG + bit;
|
||
}
|
||
|
||
/* Next time, try the next bit. */
|
||
entry->next_bit_hint = hint + 1;
|
||
|
||
object_offset = hint * object_size;
|
||
}
|
||
|
||
/* Set the in-use bit. */
|
||
entry->in_use_p[word] |= ((unsigned long) 1 << bit);
|
||
|
||
/* Keep a running total of the number of free objects. If this page
|
||
fills up, we may have to move it to the end of the list if the
|
||
next page isn't full. If the next page is full, all subsequent
|
||
pages are full, so there's no need to move it. */
|
||
if (--entry->num_free_objects == 0
|
||
&& entry->next != NULL
|
||
&& entry->next->num_free_objects > 0)
|
||
{
|
||
/* We have a new head for the list. */
|
||
G.pages[order] = entry->next;
|
||
|
||
/* We are moving ENTRY to the end of the page table list.
|
||
The new page at the head of the list will have NULL in
|
||
its PREV field and ENTRY will have NULL in its NEXT field. */
|
||
entry->next->prev = NULL;
|
||
entry->next = NULL;
|
||
|
||
/* Append ENTRY to the tail of the list. */
|
||
entry->prev = G.page_tails[order];
|
||
G.page_tails[order]->next = entry;
|
||
G.page_tails[order] = entry;
|
||
}
|
||
|
||
/* Calculate the object's address. */
|
||
result = entry->page + object_offset;
|
||
if (GATHER_STATISTICS)
|
||
ggc_record_overhead (OBJECT_SIZE (order), OBJECT_SIZE (order) - size,
|
||
result FINAL_PASS_MEM_STAT);
|
||
|
||
#ifdef ENABLE_GC_CHECKING
|
||
/* Keep poisoning-by-writing-0xaf the object, in an attempt to keep the
|
||
exact same semantics in presence of memory bugs, regardless of
|
||
ENABLE_VALGRIND_CHECKING. We override this request below. Drop the
|
||
handle to avoid handle leak. */
|
||
VALGRIND_DISCARD (VALGRIND_MAKE_MEM_UNDEFINED (result, object_size));
|
||
|
||
/* `Poison' the entire allocated object, including any padding at
|
||
the end. */
|
||
memset (result, 0xaf, object_size);
|
||
|
||
/* Make the bytes after the end of the object unaccessible. Discard the
|
||
handle to avoid handle leak. */
|
||
VALGRIND_DISCARD (VALGRIND_MAKE_MEM_NOACCESS ((char *) result + size,
|
||
object_size - size));
|
||
#endif
|
||
|
||
/* Tell Valgrind that the memory is there, but its content isn't
|
||
defined. The bytes at the end of the object are still marked
|
||
unaccessible. */
|
||
VALGRIND_DISCARD (VALGRIND_MAKE_MEM_UNDEFINED (result, size));
|
||
|
||
/* Keep track of how many bytes are being allocated. This
|
||
information is used in deciding when to collect. */
|
||
G.allocated += object_size;
|
||
|
||
/* For timevar statistics. */
|
||
timevar_ggc_mem_total += object_size;
|
||
|
||
if (f)
|
||
add_finalizer (result, f, s, n);
|
||
|
||
if (GATHER_STATISTICS)
|
||
{
|
||
size_t overhead = object_size - size;
|
||
|
||
G.stats.total_overhead += overhead;
|
||
G.stats.total_allocated += object_size;
|
||
G.stats.total_overhead_per_order[order] += overhead;
|
||
G.stats.total_allocated_per_order[order] += object_size;
|
||
|
||
if (size <= 32)
|
||
{
|
||
G.stats.total_overhead_under32 += overhead;
|
||
G.stats.total_allocated_under32 += object_size;
|
||
}
|
||
if (size <= 64)
|
||
{
|
||
G.stats.total_overhead_under64 += overhead;
|
||
G.stats.total_allocated_under64 += object_size;
|
||
}
|
||
if (size <= 128)
|
||
{
|
||
G.stats.total_overhead_under128 += overhead;
|
||
G.stats.total_allocated_under128 += object_size;
|
||
}
|
||
}
|
||
|
||
if (GGC_DEBUG_LEVEL >= 3)
|
||
fprintf (G.debug_file,
|
||
"Allocating object, requested size=%lu, actual=%lu at %p on %p\n",
|
||
(unsigned long) size, (unsigned long) object_size, result,
|
||
(void *) entry);
|
||
|
||
return result;
|
||
}
|
||
|
||
/* Mark function for strings. */
|
||
|
||
void
|
||
gt_ggc_m_S (const void *p)
|
||
{
|
||
page_entry *entry;
|
||
unsigned bit, word;
|
||
unsigned long mask;
|
||
unsigned long offset;
|
||
|
||
if (!p)
|
||
return;
|
||
|
||
/* Look up the page on which the object is alloced. If it was not
|
||
GC allocated, gracefully bail out. */
|
||
entry = safe_lookup_page_table_entry (p);
|
||
if (!entry)
|
||
return;
|
||
|
||
/* Calculate the index of the object on the page; this is its bit
|
||
position in the in_use_p bitmap. Note that because a char* might
|
||
point to the middle of an object, we need special code here to
|
||
make sure P points to the start of an object. */
|
||
offset = ((const char *) p - entry->page) % object_size_table[entry->order];
|
||
if (offset)
|
||
{
|
||
/* Here we've seen a char* which does not point to the beginning
|
||
of an allocated object. We assume it points to the middle of
|
||
a STRING_CST. */
|
||
gcc_assert (offset == offsetof (struct tree_string, str));
|
||
p = ((const char *) p) - offset;
|
||
gt_ggc_mx_lang_tree_node (CONST_CAST (void *, p));
|
||
return;
|
||
}
|
||
|
||
bit = OFFSET_TO_BIT (((const char *) p) - entry->page, entry->order);
|
||
word = bit / HOST_BITS_PER_LONG;
|
||
mask = (unsigned long) 1 << (bit % HOST_BITS_PER_LONG);
|
||
|
||
/* If the bit was previously set, skip it. */
|
||
if (entry->in_use_p[word] & mask)
|
||
return;
|
||
|
||
/* Otherwise set it, and decrement the free object count. */
|
||
entry->in_use_p[word] |= mask;
|
||
entry->num_free_objects -= 1;
|
||
|
||
if (GGC_DEBUG_LEVEL >= 4)
|
||
fprintf (G.debug_file, "Marking %p\n", p);
|
||
|
||
return;
|
||
}
|
||
|
||
|
||
/* User-callable entry points for marking string X. */
|
||
|
||
void
|
||
gt_ggc_mx (const char *& x)
|
||
{
|
||
gt_ggc_m_S (x);
|
||
}
|
||
|
||
void
|
||
gt_ggc_mx (unsigned char *& x)
|
||
{
|
||
gt_ggc_m_S (x);
|
||
}
|
||
|
||
void
|
||
gt_ggc_mx (unsigned char& x ATTRIBUTE_UNUSED)
|
||
{
|
||
}
|
||
|
||
/* If P is not marked, marks it and return false. Otherwise return true.
|
||
P must have been allocated by the GC allocator; it mustn't point to
|
||
static objects, stack variables, or memory allocated with malloc. */
|
||
|
||
int
|
||
ggc_set_mark (const void *p)
|
||
{
|
||
page_entry *entry;
|
||
unsigned bit, word;
|
||
unsigned long mask;
|
||
|
||
/* Look up the page on which the object is alloced. If the object
|
||
wasn't allocated by the collector, we'll probably die. */
|
||
entry = lookup_page_table_entry (p);
|
||
gcc_assert (entry);
|
||
|
||
/* Calculate the index of the object on the page; this is its bit
|
||
position in the in_use_p bitmap. */
|
||
bit = OFFSET_TO_BIT (((const char *) p) - entry->page, entry->order);
|
||
word = bit / HOST_BITS_PER_LONG;
|
||
mask = (unsigned long) 1 << (bit % HOST_BITS_PER_LONG);
|
||
|
||
/* If the bit was previously set, skip it. */
|
||
if (entry->in_use_p[word] & mask)
|
||
return 1;
|
||
|
||
/* Otherwise set it, and decrement the free object count. */
|
||
entry->in_use_p[word] |= mask;
|
||
entry->num_free_objects -= 1;
|
||
|
||
if (GGC_DEBUG_LEVEL >= 4)
|
||
fprintf (G.debug_file, "Marking %p\n", p);
|
||
|
||
return 0;
|
||
}
|
||
|
||
/* Return 1 if P has been marked, zero otherwise.
|
||
P must have been allocated by the GC allocator; it mustn't point to
|
||
static objects, stack variables, or memory allocated with malloc. */
|
||
|
||
int
|
||
ggc_marked_p (const void *p)
|
||
{
|
||
page_entry *entry;
|
||
unsigned bit, word;
|
||
unsigned long mask;
|
||
|
||
/* Look up the page on which the object is alloced. If the object
|
||
wasn't allocated by the collector, we'll probably die. */
|
||
entry = lookup_page_table_entry (p);
|
||
gcc_assert (entry);
|
||
|
||
/* Calculate the index of the object on the page; this is its bit
|
||
position in the in_use_p bitmap. */
|
||
bit = OFFSET_TO_BIT (((const char *) p) - entry->page, entry->order);
|
||
word = bit / HOST_BITS_PER_LONG;
|
||
mask = (unsigned long) 1 << (bit % HOST_BITS_PER_LONG);
|
||
|
||
return (entry->in_use_p[word] & mask) != 0;
|
||
}
|
||
|
||
/* Return the size of the gc-able object P. */
|
||
|
||
size_t
|
||
ggc_get_size (const void *p)
|
||
{
|
||
page_entry *pe = lookup_page_table_entry (p);
|
||
return OBJECT_SIZE (pe->order);
|
||
}
|
||
|
||
/* Release the memory for object P. */
|
||
|
||
void
|
||
ggc_free (void *p)
|
||
{
|
||
if (in_gc)
|
||
return;
|
||
|
||
page_entry *pe = lookup_page_table_entry (p);
|
||
size_t order = pe->order;
|
||
size_t size = OBJECT_SIZE (order);
|
||
|
||
if (GATHER_STATISTICS)
|
||
ggc_free_overhead (p);
|
||
|
||
if (GGC_DEBUG_LEVEL >= 3)
|
||
fprintf (G.debug_file,
|
||
"Freeing object, actual size=%lu, at %p on %p\n",
|
||
(unsigned long) size, p, (void *) pe);
|
||
|
||
#ifdef ENABLE_GC_CHECKING
|
||
/* Poison the data, to indicate the data is garbage. */
|
||
VALGRIND_DISCARD (VALGRIND_MAKE_MEM_UNDEFINED (p, size));
|
||
memset (p, 0xa5, size);
|
||
#endif
|
||
/* Let valgrind know the object is free. */
|
||
VALGRIND_DISCARD (VALGRIND_MAKE_MEM_NOACCESS (p, size));
|
||
|
||
#ifdef ENABLE_GC_ALWAYS_COLLECT
|
||
/* In the completely-anal-checking mode, we do *not* immediately free
|
||
the data, but instead verify that the data is *actually* not
|
||
reachable the next time we collect. */
|
||
{
|
||
struct free_object *fo = XNEW (struct free_object);
|
||
fo->object = p;
|
||
fo->next = G.free_object_list;
|
||
G.free_object_list = fo;
|
||
}
|
||
#else
|
||
{
|
||
unsigned int bit_offset, word, bit;
|
||
|
||
G.allocated -= size;
|
||
|
||
/* Mark the object not-in-use. */
|
||
bit_offset = OFFSET_TO_BIT (((const char *) p) - pe->page, order);
|
||
word = bit_offset / HOST_BITS_PER_LONG;
|
||
bit = bit_offset % HOST_BITS_PER_LONG;
|
||
pe->in_use_p[word] &= ~(1UL << bit);
|
||
|
||
if (pe->num_free_objects++ == 0)
|
||
{
|
||
page_entry *p, *q;
|
||
|
||
/* If the page is completely full, then it's supposed to
|
||
be after all pages that aren't. Since we've freed one
|
||
object from a page that was full, we need to move the
|
||
page to the head of the list.
|
||
|
||
PE is the node we want to move. Q is the previous node
|
||
and P is the next node in the list. */
|
||
q = pe->prev;
|
||
if (q && q->num_free_objects == 0)
|
||
{
|
||
p = pe->next;
|
||
|
||
q->next = p;
|
||
|
||
/* If PE was at the end of the list, then Q becomes the
|
||
new end of the list. If PE was not the end of the
|
||
list, then we need to update the PREV field for P. */
|
||
if (!p)
|
||
G.page_tails[order] = q;
|
||
else
|
||
p->prev = q;
|
||
|
||
/* Move PE to the head of the list. */
|
||
pe->next = G.pages[order];
|
||
pe->prev = NULL;
|
||
G.pages[order]->prev = pe;
|
||
G.pages[order] = pe;
|
||
}
|
||
|
||
/* Reset the hint bit to point to the only free object. */
|
||
pe->next_bit_hint = bit_offset;
|
||
}
|
||
}
|
||
#endif
|
||
}
|
||
|
||
/* Subroutine of init_ggc which computes the pair of numbers used to
|
||
perform division by OBJECT_SIZE (order) and fills in inverse_table[].
|
||
|
||
This algorithm is taken from Granlund and Montgomery's paper
|
||
"Division by Invariant Integers using Multiplication"
|
||
(Proc. SIGPLAN PLDI, 1994), section 9 (Exact division by
|
||
constants). */
|
||
|
||
static void
|
||
compute_inverse (unsigned order)
|
||
{
|
||
size_t size, inv;
|
||
unsigned int e;
|
||
|
||
size = OBJECT_SIZE (order);
|
||
e = 0;
|
||
while (size % 2 == 0)
|
||
{
|
||
e++;
|
||
size >>= 1;
|
||
}
|
||
|
||
inv = size;
|
||
while (inv * size != 1)
|
||
inv = inv * (2 - inv*size);
|
||
|
||
DIV_MULT (order) = inv;
|
||
DIV_SHIFT (order) = e;
|
||
}
|
||
|
||
/* Initialize the ggc-mmap allocator. */
|
||
void
|
||
init_ggc (void)
|
||
{
|
||
static bool init_p = false;
|
||
unsigned order;
|
||
|
||
if (init_p)
|
||
return;
|
||
init_p = true;
|
||
|
||
G.pagesize = getpagesize ();
|
||
G.lg_pagesize = exact_log2 (G.pagesize);
|
||
|
||
#ifdef HAVE_MMAP_DEV_ZERO
|
||
G.dev_zero_fd = open ("/dev/zero", O_RDONLY);
|
||
if (G.dev_zero_fd == -1)
|
||
internal_error ("open /dev/zero: %m");
|
||
#endif
|
||
|
||
#if 0
|
||
G.debug_file = fopen ("ggc-mmap.debug", "w");
|
||
#else
|
||
G.debug_file = stdout;
|
||
#endif
|
||
|
||
#ifdef USING_MMAP
|
||
/* StunOS has an amazing off-by-one error for the first mmap allocation
|
||
after fiddling with RLIMIT_STACK. The result, as hard as it is to
|
||
believe, is an unaligned page allocation, which would cause us to
|
||
hork badly if we tried to use it. */
|
||
{
|
||
char *p = alloc_anon (NULL, G.pagesize, true);
|
||
struct page_entry *e;
|
||
if ((uintptr_t)p & (G.pagesize - 1))
|
||
{
|
||
/* How losing. Discard this one and try another. If we still
|
||
can't get something useful, give up. */
|
||
|
||
p = alloc_anon (NULL, G.pagesize, true);
|
||
gcc_assert (!((uintptr_t)p & (G.pagesize - 1)));
|
||
}
|
||
|
||
/* We have a good page, might as well hold onto it... */
|
||
e = XCNEW (struct page_entry);
|
||
e->bytes = G.pagesize;
|
||
e->page = p;
|
||
e->next = G.free_pages;
|
||
G.free_pages = e;
|
||
}
|
||
#endif
|
||
|
||
/* Initialize the object size table. */
|
||
for (order = 0; order < HOST_BITS_PER_PTR; ++order)
|
||
object_size_table[order] = (size_t) 1 << order;
|
||
for (order = HOST_BITS_PER_PTR; order < NUM_ORDERS; ++order)
|
||
{
|
||
size_t s = extra_order_size_table[order - HOST_BITS_PER_PTR];
|
||
|
||
/* If S is not a multiple of the MAX_ALIGNMENT, then round it up
|
||
so that we're sure of getting aligned memory. */
|
||
s = ROUND_UP (s, MAX_ALIGNMENT);
|
||
object_size_table[order] = s;
|
||
}
|
||
|
||
/* Initialize the objects-per-page and inverse tables. */
|
||
for (order = 0; order < NUM_ORDERS; ++order)
|
||
{
|
||
objects_per_page_table[order] = G.pagesize / OBJECT_SIZE (order);
|
||
if (objects_per_page_table[order] == 0)
|
||
objects_per_page_table[order] = 1;
|
||
compute_inverse (order);
|
||
}
|
||
|
||
/* Reset the size_lookup array to put appropriately sized objects in
|
||
the special orders. All objects bigger than the previous power
|
||
of two, but no greater than the special size, should go in the
|
||
new order. */
|
||
for (order = HOST_BITS_PER_PTR; order < NUM_ORDERS; ++order)
|
||
{
|
||
int o;
|
||
int i;
|
||
|
||
i = OBJECT_SIZE (order);
|
||
if (i >= NUM_SIZE_LOOKUP)
|
||
continue;
|
||
|
||
for (o = size_lookup[i]; o == size_lookup [i]; --i)
|
||
size_lookup[i] = order;
|
||
}
|
||
|
||
G.depth_in_use = 0;
|
||
G.depth_max = 10;
|
||
G.depth = XNEWVEC (unsigned int, G.depth_max);
|
||
|
||
G.by_depth_in_use = 0;
|
||
G.by_depth_max = INITIAL_PTE_COUNT;
|
||
G.by_depth = XNEWVEC (page_entry *, G.by_depth_max);
|
||
G.save_in_use = XNEWVEC (unsigned long *, G.by_depth_max);
|
||
|
||
/* Allocate space for the depth 0 finalizers. */
|
||
G.finalizers.safe_push (vNULL);
|
||
G.vec_finalizers.safe_push (vNULL);
|
||
gcc_assert (G.finalizers.length() == 1);
|
||
}
|
||
|
||
/* Merge the SAVE_IN_USE_P and IN_USE_P arrays in P so that IN_USE_P
|
||
reflects reality. Recalculate NUM_FREE_OBJECTS as well. */
|
||
|
||
static void
|
||
ggc_recalculate_in_use_p (page_entry *p)
|
||
{
|
||
unsigned int i;
|
||
size_t num_objects;
|
||
|
||
/* Because the past-the-end bit in in_use_p is always set, we
|
||
pretend there is one additional object. */
|
||
num_objects = OBJECTS_IN_PAGE (p) + 1;
|
||
|
||
/* Reset the free object count. */
|
||
p->num_free_objects = num_objects;
|
||
|
||
/* Combine the IN_USE_P and SAVE_IN_USE_P arrays. */
|
||
for (i = 0;
|
||
i < CEIL (BITMAP_SIZE (num_objects),
|
||
sizeof (*p->in_use_p));
|
||
++i)
|
||
{
|
||
unsigned long j;
|
||
|
||
/* Something is in use if it is marked, or if it was in use in a
|
||
context further down the context stack. */
|
||
p->in_use_p[i] |= save_in_use_p (p)[i];
|
||
|
||
/* Decrement the free object count for every object allocated. */
|
||
for (j = p->in_use_p[i]; j; j >>= 1)
|
||
p->num_free_objects -= (j & 1);
|
||
}
|
||
|
||
gcc_assert (p->num_free_objects < num_objects);
|
||
}
|
||
|
||
/* Unmark all objects. */
|
||
|
||
static void
|
||
clear_marks (void)
|
||
{
|
||
unsigned order;
|
||
|
||
for (order = 2; order < NUM_ORDERS; order++)
|
||
{
|
||
page_entry *p;
|
||
|
||
for (p = G.pages[order]; p != NULL; p = p->next)
|
||
{
|
||
size_t num_objects = OBJECTS_IN_PAGE (p);
|
||
size_t bitmap_size = BITMAP_SIZE (num_objects + 1);
|
||
|
||
/* The data should be page-aligned. */
|
||
gcc_assert (!((uintptr_t) p->page & (G.pagesize - 1)));
|
||
|
||
/* Pages that aren't in the topmost context are not collected;
|
||
nevertheless, we need their in-use bit vectors to store GC
|
||
marks. So, back them up first. */
|
||
if (p->context_depth < G.context_depth)
|
||
{
|
||
if (! save_in_use_p (p))
|
||
save_in_use_p (p) = XNEWVAR (unsigned long, bitmap_size);
|
||
memcpy (save_in_use_p (p), p->in_use_p, bitmap_size);
|
||
}
|
||
|
||
/* Reset reset the number of free objects and clear the
|
||
in-use bits. These will be adjusted by mark_obj. */
|
||
p->num_free_objects = num_objects;
|
||
memset (p->in_use_p, 0, bitmap_size);
|
||
|
||
/* Make sure the one-past-the-end bit is always set. */
|
||
p->in_use_p[num_objects / HOST_BITS_PER_LONG]
|
||
= ((unsigned long) 1 << (num_objects % HOST_BITS_PER_LONG));
|
||
}
|
||
}
|
||
}
|
||
|
||
/* Check if any blocks with a registered finalizer have become unmarked. If so
|
||
run the finalizer and unregister it because the block is about to be freed.
|
||
Note that no garantee is made about what order finalizers will run in so
|
||
touching other objects in gc memory is extremely unwise. */
|
||
|
||
static void
|
||
ggc_handle_finalizers ()
|
||
{
|
||
unsigned dlen = G.finalizers.length();
|
||
for (unsigned d = G.context_depth; d < dlen; ++d)
|
||
{
|
||
vec<finalizer> &v = G.finalizers[d];
|
||
unsigned length = v.length ();
|
||
for (unsigned int i = 0; i < length;)
|
||
{
|
||
finalizer &f = v[i];
|
||
if (!ggc_marked_p (f.addr ()))
|
||
{
|
||
f.call ();
|
||
v.unordered_remove (i);
|
||
length--;
|
||
}
|
||
else
|
||
i++;
|
||
}
|
||
}
|
||
|
||
gcc_assert (dlen == G.vec_finalizers.length());
|
||
for (unsigned d = G.context_depth; d < dlen; ++d)
|
||
{
|
||
vec<vec_finalizer> &vv = G.vec_finalizers[d];
|
||
unsigned length = vv.length ();
|
||
for (unsigned int i = 0; i < length;)
|
||
{
|
||
vec_finalizer &f = vv[i];
|
||
if (!ggc_marked_p (f.addr ()))
|
||
{
|
||
f.call ();
|
||
vv.unordered_remove (i);
|
||
length--;
|
||
}
|
||
else
|
||
i++;
|
||
}
|
||
}
|
||
}
|
||
|
||
/* Free all empty pages. Partially empty pages need no attention
|
||
because the `mark' bit doubles as an `unused' bit. */
|
||
|
||
static void
|
||
sweep_pages (void)
|
||
{
|
||
unsigned order;
|
||
|
||
for (order = 2; order < NUM_ORDERS; order++)
|
||
{
|
||
/* The last page-entry to consider, regardless of entries
|
||
placed at the end of the list. */
|
||
page_entry * const last = G.page_tails[order];
|
||
|
||
size_t num_objects;
|
||
size_t live_objects;
|
||
page_entry *p, *previous;
|
||
int done;
|
||
|
||
p = G.pages[order];
|
||
if (p == NULL)
|
||
continue;
|
||
|
||
previous = NULL;
|
||
do
|
||
{
|
||
page_entry *next = p->next;
|
||
|
||
/* Loop until all entries have been examined. */
|
||
done = (p == last);
|
||
|
||
num_objects = OBJECTS_IN_PAGE (p);
|
||
|
||
/* Add all live objects on this page to the count of
|
||
allocated memory. */
|
||
live_objects = num_objects - p->num_free_objects;
|
||
|
||
G.allocated += OBJECT_SIZE (order) * live_objects;
|
||
|
||
/* Only objects on pages in the topmost context should get
|
||
collected. */
|
||
if (p->context_depth < G.context_depth)
|
||
;
|
||
|
||
/* Remove the page if it's empty. */
|
||
else if (live_objects == 0)
|
||
{
|
||
/* If P was the first page in the list, then NEXT
|
||
becomes the new first page in the list, otherwise
|
||
splice P out of the forward pointers. */
|
||
if (! previous)
|
||
G.pages[order] = next;
|
||
else
|
||
previous->next = next;
|
||
|
||
/* Splice P out of the back pointers too. */
|
||
if (next)
|
||
next->prev = previous;
|
||
|
||
/* Are we removing the last element? */
|
||
if (p == G.page_tails[order])
|
||
G.page_tails[order] = previous;
|
||
free_page (p);
|
||
p = previous;
|
||
}
|
||
|
||
/* If the page is full, move it to the end. */
|
||
else if (p->num_free_objects == 0)
|
||
{
|
||
/* Don't move it if it's already at the end. */
|
||
if (p != G.page_tails[order])
|
||
{
|
||
/* Move p to the end of the list. */
|
||
p->next = NULL;
|
||
p->prev = G.page_tails[order];
|
||
G.page_tails[order]->next = p;
|
||
|
||
/* Update the tail pointer... */
|
||
G.page_tails[order] = p;
|
||
|
||
/* ... and the head pointer, if necessary. */
|
||
if (! previous)
|
||
G.pages[order] = next;
|
||
else
|
||
previous->next = next;
|
||
|
||
/* And update the backpointer in NEXT if necessary. */
|
||
if (next)
|
||
next->prev = previous;
|
||
|
||
p = previous;
|
||
}
|
||
}
|
||
|
||
/* If we've fallen through to here, it's a page in the
|
||
topmost context that is neither full nor empty. Such a
|
||
page must precede pages at lesser context depth in the
|
||
list, so move it to the head. */
|
||
else if (p != G.pages[order])
|
||
{
|
||
previous->next = p->next;
|
||
|
||
/* Update the backchain in the next node if it exists. */
|
||
if (p->next)
|
||
p->next->prev = previous;
|
||
|
||
/* Move P to the head of the list. */
|
||
p->next = G.pages[order];
|
||
p->prev = NULL;
|
||
G.pages[order]->prev = p;
|
||
|
||
/* Update the head pointer. */
|
||
G.pages[order] = p;
|
||
|
||
/* Are we moving the last element? */
|
||
if (G.page_tails[order] == p)
|
||
G.page_tails[order] = previous;
|
||
p = previous;
|
||
}
|
||
|
||
previous = p;
|
||
p = next;
|
||
}
|
||
while (! done);
|
||
|
||
/* Now, restore the in_use_p vectors for any pages from contexts
|
||
other than the current one. */
|
||
for (p = G.pages[order]; p; p = p->next)
|
||
if (p->context_depth != G.context_depth)
|
||
ggc_recalculate_in_use_p (p);
|
||
}
|
||
}
|
||
|
||
#ifdef ENABLE_GC_CHECKING
|
||
/* Clobber all free objects. */
|
||
|
||
static void
|
||
poison_pages (void)
|
||
{
|
||
unsigned order;
|
||
|
||
for (order = 2; order < NUM_ORDERS; order++)
|
||
{
|
||
size_t size = OBJECT_SIZE (order);
|
||
page_entry *p;
|
||
|
||
for (p = G.pages[order]; p != NULL; p = p->next)
|
||
{
|
||
size_t num_objects;
|
||
size_t i;
|
||
|
||
if (p->context_depth != G.context_depth)
|
||
/* Since we don't do any collection for pages in pushed
|
||
contexts, there's no need to do any poisoning. And
|
||
besides, the IN_USE_P array isn't valid until we pop
|
||
contexts. */
|
||
continue;
|
||
|
||
num_objects = OBJECTS_IN_PAGE (p);
|
||
for (i = 0; i < num_objects; i++)
|
||
{
|
||
size_t word, bit;
|
||
word = i / HOST_BITS_PER_LONG;
|
||
bit = i % HOST_BITS_PER_LONG;
|
||
if (((p->in_use_p[word] >> bit) & 1) == 0)
|
||
{
|
||
char *object = p->page + i * size;
|
||
|
||
/* Keep poison-by-write when we expect to use Valgrind,
|
||
so the exact same memory semantics is kept, in case
|
||
there are memory errors. We override this request
|
||
below. */
|
||
VALGRIND_DISCARD (VALGRIND_MAKE_MEM_UNDEFINED (object,
|
||
size));
|
||
memset (object, 0xa5, size);
|
||
|
||
/* Drop the handle to avoid handle leak. */
|
||
VALGRIND_DISCARD (VALGRIND_MAKE_MEM_NOACCESS (object, size));
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
#else
|
||
#define poison_pages()
|
||
#endif
|
||
|
||
#ifdef ENABLE_GC_ALWAYS_COLLECT
|
||
/* Validate that the reportedly free objects actually are. */
|
||
|
||
static void
|
||
validate_free_objects (void)
|
||
{
|
||
struct free_object *f, *next, *still_free = NULL;
|
||
|
||
for (f = G.free_object_list; f ; f = next)
|
||
{
|
||
page_entry *pe = lookup_page_table_entry (f->object);
|
||
size_t bit, word;
|
||
|
||
bit = OFFSET_TO_BIT ((char *)f->object - pe->page, pe->order);
|
||
word = bit / HOST_BITS_PER_LONG;
|
||
bit = bit % HOST_BITS_PER_LONG;
|
||
next = f->next;
|
||
|
||
/* Make certain it isn't visible from any root. Notice that we
|
||
do this check before sweep_pages merges save_in_use_p. */
|
||
gcc_assert (!(pe->in_use_p[word] & (1UL << bit)));
|
||
|
||
/* If the object comes from an outer context, then retain the
|
||
free_object entry, so that we can verify that the address
|
||
isn't live on the stack in some outer context. */
|
||
if (pe->context_depth != G.context_depth)
|
||
{
|
||
f->next = still_free;
|
||
still_free = f;
|
||
}
|
||
else
|
||
free (f);
|
||
}
|
||
|
||
G.free_object_list = still_free;
|
||
}
|
||
#else
|
||
#define validate_free_objects()
|
||
#endif
|
||
|
||
/* Top level mark-and-sweep routine. */
|
||
|
||
void
|
||
ggc_collect (void)
|
||
{
|
||
/* Avoid frequent unnecessary work by skipping collection if the
|
||
total allocations haven't expanded much since the last
|
||
collection. */
|
||
float allocated_last_gc =
|
||
MAX (G.allocated_last_gc, (size_t)PARAM_VALUE (GGC_MIN_HEAPSIZE) * 1024);
|
||
|
||
float min_expand = allocated_last_gc * PARAM_VALUE (GGC_MIN_EXPAND) / 100;
|
||
if (G.allocated < allocated_last_gc + min_expand && !ggc_force_collect)
|
||
return;
|
||
|
||
timevar_push (TV_GC);
|
||
if (!quiet_flag)
|
||
fprintf (stderr, " {GC %luk -> ", (unsigned long) G.allocated / 1024);
|
||
if (GGC_DEBUG_LEVEL >= 2)
|
||
fprintf (G.debug_file, "BEGIN COLLECTING\n");
|
||
|
||
/* Zero the total allocated bytes. This will be recalculated in the
|
||
sweep phase. */
|
||
G.allocated = 0;
|
||
|
||
/* Release the pages we freed the last time we collected, but didn't
|
||
reuse in the interim. */
|
||
release_pages ();
|
||
|
||
/* Indicate that we've seen collections at this context depth. */
|
||
G.context_depth_collections = ((unsigned long)1 << (G.context_depth + 1)) - 1;
|
||
|
||
invoke_plugin_callbacks (PLUGIN_GGC_START, NULL);
|
||
|
||
in_gc = true;
|
||
clear_marks ();
|
||
ggc_mark_roots ();
|
||
ggc_handle_finalizers ();
|
||
|
||
if (GATHER_STATISTICS)
|
||
ggc_prune_overhead_list ();
|
||
|
||
poison_pages ();
|
||
validate_free_objects ();
|
||
sweep_pages ();
|
||
|
||
in_gc = false;
|
||
G.allocated_last_gc = G.allocated;
|
||
|
||
invoke_plugin_callbacks (PLUGIN_GGC_END, NULL);
|
||
|
||
timevar_pop (TV_GC);
|
||
|
||
if (!quiet_flag)
|
||
fprintf (stderr, "%luk}", (unsigned long) G.allocated / 1024);
|
||
if (GGC_DEBUG_LEVEL >= 2)
|
||
fprintf (G.debug_file, "END COLLECTING\n");
|
||
}
|
||
|
||
/* Assume that all GGC memory is reachable and grow the limits for next collection.
|
||
With checking, trigger GGC so -Q compilation outputs how much of memory really is
|
||
reachable. */
|
||
|
||
void
|
||
ggc_grow (void)
|
||
{
|
||
if (!flag_checking)
|
||
G.allocated_last_gc = MAX (G.allocated_last_gc,
|
||
G.allocated);
|
||
else
|
||
ggc_collect ();
|
||
if (!quiet_flag)
|
||
fprintf (stderr, " {GC start %luk} ", (unsigned long) G.allocated / 1024);
|
||
}
|
||
|
||
void
|
||
ggc_print_statistics (void)
|
||
{
|
||
struct ggc_statistics stats;
|
||
unsigned int i;
|
||
size_t total_overhead = 0;
|
||
|
||
/* Clear the statistics. */
|
||
memset (&stats, 0, sizeof (stats));
|
||
|
||
/* Make sure collection will really occur. */
|
||
G.allocated_last_gc = 0;
|
||
|
||
/* Collect and print the statistics common across collectors. */
|
||
ggc_print_common_statistics (stderr, &stats);
|
||
|
||
/* Release free pages so that we will not count the bytes allocated
|
||
there as part of the total allocated memory. */
|
||
release_pages ();
|
||
|
||
/* Collect some information about the various sizes of
|
||
allocation. */
|
||
fprintf (stderr,
|
||
"Memory still allocated at the end of the compilation process\n");
|
||
fprintf (stderr, "%-8s %10s %10s %10s\n",
|
||
"Size", "Allocated", "Used", "Overhead");
|
||
for (i = 0; i < NUM_ORDERS; ++i)
|
||
{
|
||
page_entry *p;
|
||
size_t allocated;
|
||
size_t in_use;
|
||
size_t overhead;
|
||
|
||
/* Skip empty entries. */
|
||
if (!G.pages[i])
|
||
continue;
|
||
|
||
overhead = allocated = in_use = 0;
|
||
|
||
/* Figure out the total number of bytes allocated for objects of
|
||
this size, and how many of them are actually in use. Also figure
|
||
out how much memory the page table is using. */
|
||
for (p = G.pages[i]; p; p = p->next)
|
||
{
|
||
allocated += p->bytes;
|
||
in_use +=
|
||
(OBJECTS_IN_PAGE (p) - p->num_free_objects) * OBJECT_SIZE (i);
|
||
|
||
overhead += (sizeof (page_entry) - sizeof (long)
|
||
+ BITMAP_SIZE (OBJECTS_IN_PAGE (p) + 1));
|
||
}
|
||
fprintf (stderr, "%-8zu %10zu%c %10zu%c %10zu%c\n",
|
||
OBJECT_SIZE (i),
|
||
SIZE_AMOUNT (allocated),
|
||
SIZE_AMOUNT (in_use),
|
||
SIZE_AMOUNT (overhead));
|
||
total_overhead += overhead;
|
||
}
|
||
fprintf (stderr, "%-8s %10zu%c %10zu%c %10zu%c\n",
|
||
"Total",
|
||
SIZE_AMOUNT (G.bytes_mapped),
|
||
SIZE_AMOUNT (G.allocated),
|
||
SIZE_AMOUNT (total_overhead));
|
||
|
||
if (GATHER_STATISTICS)
|
||
{
|
||
fprintf (stderr, "\nTotal allocations and overheads during "
|
||
"the compilation process\n");
|
||
|
||
fprintf (stderr, "Total Overhead: %9"
|
||
HOST_LONG_LONG_FORMAT "d%c\n",
|
||
SIZE_AMOUNT (G.stats.total_overhead));
|
||
fprintf (stderr, "Total Allocated: %9"
|
||
HOST_LONG_LONG_FORMAT "d%c\n",
|
||
SIZE_AMOUNT (G.stats.total_allocated));
|
||
|
||
fprintf (stderr, "Total Overhead under 32B: %9"
|
||
HOST_LONG_LONG_FORMAT "d%c\n",
|
||
SIZE_AMOUNT (G.stats.total_overhead_under32));
|
||
fprintf (stderr, "Total Allocated under 32B: %9"
|
||
HOST_LONG_LONG_FORMAT "d%c\n",
|
||
SIZE_AMOUNT (G.stats.total_allocated_under32));
|
||
fprintf (stderr, "Total Overhead under 64B: %9"
|
||
HOST_LONG_LONG_FORMAT "d%c\n",
|
||
SIZE_AMOUNT (G.stats.total_overhead_under64));
|
||
fprintf (stderr, "Total Allocated under 64B: %9"
|
||
HOST_LONG_LONG_FORMAT "d%c\n",
|
||
SIZE_AMOUNT (G.stats.total_allocated_under64));
|
||
fprintf (stderr, "Total Overhead under 128B: %9"
|
||
HOST_LONG_LONG_FORMAT "d%c\n",
|
||
SIZE_AMOUNT (G.stats.total_overhead_under128));
|
||
fprintf (stderr, "Total Allocated under 128B: %9"
|
||
HOST_LONG_LONG_FORMAT "d%c\n",
|
||
SIZE_AMOUNT (G.stats.total_allocated_under128));
|
||
|
||
for (i = 0; i < NUM_ORDERS; i++)
|
||
if (G.stats.total_allocated_per_order[i])
|
||
{
|
||
fprintf (stderr, "Total Overhead page size %9zu: %9"
|
||
HOST_LONG_LONG_FORMAT "d%c\n",
|
||
OBJECT_SIZE (i),
|
||
SIZE_AMOUNT (G.stats.total_overhead_per_order[i]));
|
||
fprintf (stderr, "Total Allocated page size %9zu: %9"
|
||
HOST_LONG_LONG_FORMAT "d%c\n",
|
||
OBJECT_SIZE (i),
|
||
SIZE_AMOUNT (G.stats.total_allocated_per_order[i]));
|
||
}
|
||
}
|
||
}
|
||
|
||
struct ggc_pch_ondisk
|
||
{
|
||
unsigned totals[NUM_ORDERS];
|
||
};
|
||
|
||
struct ggc_pch_data
|
||
{
|
||
struct ggc_pch_ondisk d;
|
||
uintptr_t base[NUM_ORDERS];
|
||
size_t written[NUM_ORDERS];
|
||
};
|
||
|
||
struct ggc_pch_data *
|
||
init_ggc_pch (void)
|
||
{
|
||
return XCNEW (struct ggc_pch_data);
|
||
}
|
||
|
||
void
|
||
ggc_pch_count_object (struct ggc_pch_data *d, void *x ATTRIBUTE_UNUSED,
|
||
size_t size, bool is_string ATTRIBUTE_UNUSED)
|
||
{
|
||
unsigned order;
|
||
|
||
if (size < NUM_SIZE_LOOKUP)
|
||
order = size_lookup[size];
|
||
else
|
||
{
|
||
order = 10;
|
||
while (size > OBJECT_SIZE (order))
|
||
order++;
|
||
}
|
||
|
||
d->d.totals[order]++;
|
||
}
|
||
|
||
size_t
|
||
ggc_pch_total_size (struct ggc_pch_data *d)
|
||
{
|
||
size_t a = 0;
|
||
unsigned i;
|
||
|
||
for (i = 0; i < NUM_ORDERS; i++)
|
||
a += PAGE_ALIGN (d->d.totals[i] * OBJECT_SIZE (i));
|
||
return a;
|
||
}
|
||
|
||
void
|
||
ggc_pch_this_base (struct ggc_pch_data *d, void *base)
|
||
{
|
||
uintptr_t a = (uintptr_t) base;
|
||
unsigned i;
|
||
|
||
for (i = 0; i < NUM_ORDERS; i++)
|
||
{
|
||
d->base[i] = a;
|
||
a += PAGE_ALIGN (d->d.totals[i] * OBJECT_SIZE (i));
|
||
}
|
||
}
|
||
|
||
|
||
char *
|
||
ggc_pch_alloc_object (struct ggc_pch_data *d, void *x ATTRIBUTE_UNUSED,
|
||
size_t size, bool is_string ATTRIBUTE_UNUSED)
|
||
{
|
||
unsigned order;
|
||
char *result;
|
||
|
||
if (size < NUM_SIZE_LOOKUP)
|
||
order = size_lookup[size];
|
||
else
|
||
{
|
||
order = 10;
|
||
while (size > OBJECT_SIZE (order))
|
||
order++;
|
||
}
|
||
|
||
result = (char *) d->base[order];
|
||
d->base[order] += OBJECT_SIZE (order);
|
||
return result;
|
||
}
|
||
|
||
void
|
||
ggc_pch_prepare_write (struct ggc_pch_data *d ATTRIBUTE_UNUSED,
|
||
FILE *f ATTRIBUTE_UNUSED)
|
||
{
|
||
/* Nothing to do. */
|
||
}
|
||
|
||
void
|
||
ggc_pch_write_object (struct ggc_pch_data *d,
|
||
FILE *f, void *x, void *newx ATTRIBUTE_UNUSED,
|
||
size_t size, bool is_string ATTRIBUTE_UNUSED)
|
||
{
|
||
unsigned order;
|
||
static const char emptyBytes[256] = { 0 };
|
||
|
||
if (size < NUM_SIZE_LOOKUP)
|
||
order = size_lookup[size];
|
||
else
|
||
{
|
||
order = 10;
|
||
while (size > OBJECT_SIZE (order))
|
||
order++;
|
||
}
|
||
|
||
if (fwrite (x, size, 1, f) != 1)
|
||
fatal_error (input_location, "can%'t write PCH file: %m");
|
||
|
||
/* If SIZE is not the same as OBJECT_SIZE(order), then we need to pad the
|
||
object out to OBJECT_SIZE(order). This happens for strings. */
|
||
|
||
if (size != OBJECT_SIZE (order))
|
||
{
|
||
unsigned padding = OBJECT_SIZE (order) - size;
|
||
|
||
/* To speed small writes, we use a nulled-out array that's larger
|
||
than most padding requests as the source for our null bytes. This
|
||
permits us to do the padding with fwrite() rather than fseek(), and
|
||
limits the chance the OS may try to flush any outstanding writes. */
|
||
if (padding <= sizeof (emptyBytes))
|
||
{
|
||
if (fwrite (emptyBytes, 1, padding, f) != padding)
|
||
fatal_error (input_location, "can%'t write PCH file");
|
||
}
|
||
else
|
||
{
|
||
/* Larger than our buffer? Just default to fseek. */
|
||
if (fseek (f, padding, SEEK_CUR) != 0)
|
||
fatal_error (input_location, "can%'t write PCH file");
|
||
}
|
||
}
|
||
|
||
d->written[order]++;
|
||
if (d->written[order] == d->d.totals[order]
|
||
&& fseek (f, ROUND_UP_VALUE (d->d.totals[order] * OBJECT_SIZE (order),
|
||
G.pagesize),
|
||
SEEK_CUR) != 0)
|
||
fatal_error (input_location, "can%'t write PCH file: %m");
|
||
}
|
||
|
||
void
|
||
ggc_pch_finish (struct ggc_pch_data *d, FILE *f)
|
||
{
|
||
if (fwrite (&d->d, sizeof (d->d), 1, f) != 1)
|
||
fatal_error (input_location, "can%'t write PCH file: %m");
|
||
free (d);
|
||
}
|
||
|
||
/* Move the PCH PTE entries just added to the end of by_depth, to the
|
||
front. */
|
||
|
||
static void
|
||
move_ptes_to_front (int count_old_page_tables, int count_new_page_tables)
|
||
{
|
||
/* First, we swap the new entries to the front of the varrays. */
|
||
page_entry **new_by_depth;
|
||
unsigned long **new_save_in_use;
|
||
|
||
new_by_depth = XNEWVEC (page_entry *, G.by_depth_max);
|
||
new_save_in_use = XNEWVEC (unsigned long *, G.by_depth_max);
|
||
|
||
memcpy (&new_by_depth[0],
|
||
&G.by_depth[count_old_page_tables],
|
||
count_new_page_tables * sizeof (void *));
|
||
memcpy (&new_by_depth[count_new_page_tables],
|
||
&G.by_depth[0],
|
||
count_old_page_tables * sizeof (void *));
|
||
memcpy (&new_save_in_use[0],
|
||
&G.save_in_use[count_old_page_tables],
|
||
count_new_page_tables * sizeof (void *));
|
||
memcpy (&new_save_in_use[count_new_page_tables],
|
||
&G.save_in_use[0],
|
||
count_old_page_tables * sizeof (void *));
|
||
|
||
free (G.by_depth);
|
||
free (G.save_in_use);
|
||
|
||
G.by_depth = new_by_depth;
|
||
G.save_in_use = new_save_in_use;
|
||
|
||
/* Now update all the index_by_depth fields. */
|
||
for (unsigned i = G.by_depth_in_use; i--;)
|
||
{
|
||
page_entry *p = G.by_depth[i];
|
||
p->index_by_depth = i;
|
||
}
|
||
|
||
/* And last, we update the depth pointers in G.depth. The first
|
||
entry is already 0, and context 0 entries always start at index
|
||
0, so there is nothing to update in the first slot. We need a
|
||
second slot, only if we have old ptes, and if we do, they start
|
||
at index count_new_page_tables. */
|
||
if (count_old_page_tables)
|
||
push_depth (count_new_page_tables);
|
||
}
|
||
|
||
void
|
||
ggc_pch_read (FILE *f, void *addr)
|
||
{
|
||
struct ggc_pch_ondisk d;
|
||
unsigned i;
|
||
char *offs = (char *) addr;
|
||
unsigned long count_old_page_tables;
|
||
unsigned long count_new_page_tables;
|
||
|
||
count_old_page_tables = G.by_depth_in_use;
|
||
|
||
/* We've just read in a PCH file. So, every object that used to be
|
||
allocated is now free. */
|
||
clear_marks ();
|
||
#ifdef ENABLE_GC_CHECKING
|
||
poison_pages ();
|
||
#endif
|
||
/* Since we free all the allocated objects, the free list becomes
|
||
useless. Validate it now, which will also clear it. */
|
||
validate_free_objects ();
|
||
|
||
/* No object read from a PCH file should ever be freed. So, set the
|
||
context depth to 1, and set the depth of all the currently-allocated
|
||
pages to be 1 too. PCH pages will have depth 0. */
|
||
gcc_assert (!G.context_depth);
|
||
G.context_depth = 1;
|
||
/* Allocate space for the depth 1 finalizers. */
|
||
G.finalizers.safe_push (vNULL);
|
||
G.vec_finalizers.safe_push (vNULL);
|
||
gcc_assert (G.finalizers.length() == 2);
|
||
for (i = 0; i < NUM_ORDERS; i++)
|
||
{
|
||
page_entry *p;
|
||
for (p = G.pages[i]; p != NULL; p = p->next)
|
||
p->context_depth = G.context_depth;
|
||
}
|
||
|
||
/* Allocate the appropriate page-table entries for the pages read from
|
||
the PCH file. */
|
||
if (fread (&d, sizeof (d), 1, f) != 1)
|
||
fatal_error (input_location, "can%'t read PCH file: %m");
|
||
|
||
for (i = 0; i < NUM_ORDERS; i++)
|
||
{
|
||
struct page_entry *entry;
|
||
char *pte;
|
||
size_t bytes;
|
||
size_t num_objs;
|
||
size_t j;
|
||
|
||
if (d.totals[i] == 0)
|
||
continue;
|
||
|
||
bytes = PAGE_ALIGN (d.totals[i] * OBJECT_SIZE (i));
|
||
num_objs = bytes / OBJECT_SIZE (i);
|
||
entry = XCNEWVAR (struct page_entry, (sizeof (struct page_entry)
|
||
- sizeof (long)
|
||
+ BITMAP_SIZE (num_objs + 1)));
|
||
entry->bytes = bytes;
|
||
entry->page = offs;
|
||
entry->context_depth = 0;
|
||
offs += bytes;
|
||
entry->num_free_objects = 0;
|
||
entry->order = i;
|
||
|
||
for (j = 0;
|
||
j + HOST_BITS_PER_LONG <= num_objs + 1;
|
||
j += HOST_BITS_PER_LONG)
|
||
entry->in_use_p[j / HOST_BITS_PER_LONG] = -1;
|
||
for (; j < num_objs + 1; j++)
|
||
entry->in_use_p[j / HOST_BITS_PER_LONG]
|
||
|= 1L << (j % HOST_BITS_PER_LONG);
|
||
|
||
for (pte = entry->page;
|
||
pte < entry->page + entry->bytes;
|
||
pte += G.pagesize)
|
||
set_page_table_entry (pte, entry);
|
||
|
||
if (G.page_tails[i] != NULL)
|
||
G.page_tails[i]->next = entry;
|
||
else
|
||
G.pages[i] = entry;
|
||
G.page_tails[i] = entry;
|
||
|
||
/* We start off by just adding all the new information to the
|
||
end of the varrays, later, we will move the new information
|
||
to the front of the varrays, as the PCH page tables are at
|
||
context 0. */
|
||
push_by_depth (entry, 0);
|
||
}
|
||
|
||
/* Now, we update the various data structures that speed page table
|
||
handling. */
|
||
count_new_page_tables = G.by_depth_in_use - count_old_page_tables;
|
||
|
||
move_ptes_to_front (count_old_page_tables, count_new_page_tables);
|
||
|
||
/* Update the statistics. */
|
||
G.allocated = G.allocated_last_gc = offs - (char *)addr;
|
||
}
|