/**
|
* \file
|
* Copyright 2001-2003 Ximian, Inc
|
* Copyright 2003-2010 Novell, Inc.
|
*
|
* Licensed under the MIT license. See LICENSE file in the project root for full license information.
|
*/
|
#ifndef __MONO_SGEN_CARD_TABLE_INLINES_H__
|
#define __MONO_SGEN_CARD_TABLE_INLINES_H__
|
|
/*WARNING: This function returns the number of cards regardless of overflow in case of overlapping cards.*/
|
mword sgen_card_table_number_of_cards_in_range (mword address, mword size);
|
guint8* sgen_find_next_card (guint8 *card_data, guint8 *end);
|
|
void sgen_card_table_reset_region (mword start, mword end);
|
void* sgen_card_table_align_pointer (void *ptr);
|
void sgen_card_table_mark_range (mword address, mword size);
|
void sgen_cardtable_scan_object (GCObject *obj, mword obj_size, guint8 *cards,
|
ScanCopyContext ctx);
|
|
gboolean sgen_card_table_get_card_data (guint8 *dest, mword address, mword cards);
|
|
guint8* sgen_card_table_alloc_mod_union (char *obj, mword obj_size);
|
void sgen_card_table_free_mod_union (guint8 *mod_union, char *obj, mword obj_size);
|
|
void sgen_card_table_update_mod_union_from_cards (guint8 *dest, guint8 *start_card, size_t num_cards);
|
void sgen_card_table_update_mod_union (guint8 *dest, char *obj, mword obj_size, size_t *out_num_cards);
|
void sgen_card_table_preclean_mod_union (guint8 *cards, guint8 *cards_preclean, size_t num_cards);
|
|
guint8* sgen_get_card_table_configuration (int *shift_bits, gpointer *mask);
|
|
void sgen_card_table_init (SgenRememberedSet *remset);
|
|
/*How many bytes a single card covers*/
|
#define CARD_BITS 9
|
|
/* How many bits of the address space is covered by the card table.
|
* If this value is smaller than the number of address bits, card aliasing is required.
|
*/
|
#define CARD_TABLE_BITS 32
|
|
#define CARD_SIZE_IN_BYTES (1 << CARD_BITS)
|
#define CARD_COUNT_BITS (CARD_TABLE_BITS - CARD_BITS)
|
#define CARD_COUNT_IN_BYTES (1 << CARD_COUNT_BITS)
|
#define CARD_MASK ((1 << CARD_COUNT_BITS) - 1)
|
|
#if SIZEOF_VOID_P * 8 > CARD_TABLE_BITS
|
#define SGEN_HAVE_OVERLAPPING_CARDS 1
|
#endif
|
|
extern guint8 *sgen_cardtable;
|
|
|
#ifdef SGEN_HAVE_OVERLAPPING_CARDS
|
|
static inline guint8*
|
sgen_card_table_get_card_address (mword address)
|
{
|
return sgen_cardtable + ((address >> CARD_BITS) & CARD_MASK);
|
}
|
|
extern guint8 *sgen_shadow_cardtable;
|
|
#define SGEN_SHADOW_CARDTABLE_END (sgen_shadow_cardtable + CARD_COUNT_IN_BYTES)
|
|
static inline guint8*
|
sgen_card_table_get_shadow_card_address (mword address)
|
{
|
return sgen_shadow_cardtable + ((address >> CARD_BITS) & CARD_MASK);
|
}
|
|
static inline gboolean
|
sgen_card_table_card_begin_scanning (mword address)
|
{
|
return *sgen_card_table_get_shadow_card_address (address) != 0;
|
}
|
|
static inline void
|
sgen_card_table_prepare_card_for_scanning (guint8 *card)
|
{
|
}
|
|
#define sgen_card_table_get_card_scan_address sgen_card_table_get_shadow_card_address
|
|
#else
|
|
static inline guint8*
|
sgen_card_table_get_card_address (mword address)
|
{
|
return sgen_cardtable + (address >> CARD_BITS);
|
}
|
|
static inline gboolean
|
sgen_card_table_card_begin_scanning (mword address)
|
{
|
guint8 *card = sgen_card_table_get_card_address (address);
|
gboolean res = *card;
|
*card = 0;
|
return res;
|
}
|
|
static inline void
|
sgen_card_table_prepare_card_for_scanning (guint8 *card)
|
{
|
*card = 0;
|
}
|
|
#define sgen_card_table_get_card_scan_address sgen_card_table_get_card_address
|
|
#endif
|
|
static inline gboolean
|
sgen_card_table_address_is_marked (mword address)
|
{
|
return *sgen_card_table_get_card_address (address) != 0;
|
}
|
|
static inline void
|
sgen_card_table_mark_address (mword address)
|
{
|
*sgen_card_table_get_card_address (address) = 1;
|
}
|
|
static inline size_t
|
sgen_card_table_get_card_offset (char *ptr, char *base)
|
{
|
return (ptr - base) >> CARD_BITS;
|
}
|
|
#endif
|