py/gc: Use size_t instead of mp_uint_t to count things related to heap.
size_t is the correct type to use to count things related to the size of the address space. Using size_t (instead of mp_uint_t) is important for the efficiency of ports that configure mp_uint_t to larger than the machine word size.
This commit is contained in:
parent
f7782f8082
commit
d977d268e8
56
py/gc.c
56
py/gc.c
@ -105,7 +105,7 @@ void gc_init(void *start, void *end) {
|
|||||||
// F = A * BLOCKS_PER_ATB / BLOCKS_PER_FTB
|
// F = A * BLOCKS_PER_ATB / BLOCKS_PER_FTB
|
||||||
// P = A * BLOCKS_PER_ATB * BYTES_PER_BLOCK
|
// P = A * BLOCKS_PER_ATB * BYTES_PER_BLOCK
|
||||||
// => T = A * (1 + BLOCKS_PER_ATB / BLOCKS_PER_FTB + BLOCKS_PER_ATB * BYTES_PER_BLOCK)
|
// => T = A * (1 + BLOCKS_PER_ATB / BLOCKS_PER_FTB + BLOCKS_PER_ATB * BYTES_PER_BLOCK)
|
||||||
mp_uint_t total_byte_len = (byte*)end - (byte*)start;
|
size_t total_byte_len = (byte*)end - (byte*)start;
|
||||||
#if MICROPY_ENABLE_FINALISER
|
#if MICROPY_ENABLE_FINALISER
|
||||||
MP_STATE_MEM(gc_alloc_table_byte_len) = total_byte_len * BITS_PER_BYTE / (BITS_PER_BYTE + BITS_PER_BYTE * BLOCKS_PER_ATB / BLOCKS_PER_FTB + BITS_PER_BYTE * BLOCKS_PER_ATB * BYTES_PER_BLOCK);
|
MP_STATE_MEM(gc_alloc_table_byte_len) = total_byte_len * BITS_PER_BYTE / (BITS_PER_BYTE + BITS_PER_BYTE * BLOCKS_PER_ATB / BLOCKS_PER_FTB + BITS_PER_BYTE * BLOCKS_PER_ATB * BYTES_PER_BLOCK);
|
||||||
#else
|
#else
|
||||||
@ -115,11 +115,11 @@ void gc_init(void *start, void *end) {
|
|||||||
MP_STATE_MEM(gc_alloc_table_start) = (byte*)start;
|
MP_STATE_MEM(gc_alloc_table_start) = (byte*)start;
|
||||||
|
|
||||||
#if MICROPY_ENABLE_FINALISER
|
#if MICROPY_ENABLE_FINALISER
|
||||||
mp_uint_t gc_finaliser_table_byte_len = (MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB + BLOCKS_PER_FTB - 1) / BLOCKS_PER_FTB;
|
size_t gc_finaliser_table_byte_len = (MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB + BLOCKS_PER_FTB - 1) / BLOCKS_PER_FTB;
|
||||||
MP_STATE_MEM(gc_finaliser_table_start) = MP_STATE_MEM(gc_alloc_table_start) + MP_STATE_MEM(gc_alloc_table_byte_len);
|
MP_STATE_MEM(gc_finaliser_table_start) = MP_STATE_MEM(gc_alloc_table_start) + MP_STATE_MEM(gc_alloc_table_byte_len);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
mp_uint_t gc_pool_block_len = MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB;
|
size_t gc_pool_block_len = MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB;
|
||||||
MP_STATE_MEM(gc_pool_start) = (byte*)end - gc_pool_block_len * BYTES_PER_BLOCK;
|
MP_STATE_MEM(gc_pool_start) = (byte*)end - gc_pool_block_len * BYTES_PER_BLOCK;
|
||||||
MP_STATE_MEM(gc_pool_end) = end;
|
MP_STATE_MEM(gc_pool_end) = end;
|
||||||
|
|
||||||
@ -175,7 +175,7 @@ bool gc_is_locked(void) {
|
|||||||
#define VERIFY_MARK_AND_PUSH(ptr) \
|
#define VERIFY_MARK_AND_PUSH(ptr) \
|
||||||
do { \
|
do { \
|
||||||
if (VERIFY_PTR(ptr)) { \
|
if (VERIFY_PTR(ptr)) { \
|
||||||
mp_uint_t _block = BLOCK_FROM_PTR(ptr); \
|
size_t _block = BLOCK_FROM_PTR(ptr); \
|
||||||
if (ATB_GET_KIND(_block) == AT_HEAD) { \
|
if (ATB_GET_KIND(_block) == AT_HEAD) { \
|
||||||
/* an unmarked head, mark it, and push it on gc stack */ \
|
/* an unmarked head, mark it, and push it on gc stack */ \
|
||||||
ATB_HEAD_TO_MARK(_block); \
|
ATB_HEAD_TO_MARK(_block); \
|
||||||
@ -194,7 +194,7 @@ STATIC void gc_drain_stack(void) {
|
|||||||
size_t block = *--MP_STATE_MEM(gc_sp);
|
size_t block = *--MP_STATE_MEM(gc_sp);
|
||||||
|
|
||||||
// work out number of consecutive blocks in the chain starting with this one
|
// work out number of consecutive blocks in the chain starting with this one
|
||||||
mp_uint_t n_blocks = 0;
|
size_t n_blocks = 0;
|
||||||
do {
|
do {
|
||||||
n_blocks += 1;
|
n_blocks += 1;
|
||||||
} while (ATB_GET_KIND(block + n_blocks) == AT_TAIL);
|
} while (ATB_GET_KIND(block + n_blocks) == AT_TAIL);
|
||||||
@ -214,7 +214,7 @@ STATIC void gc_deal_with_stack_overflow(void) {
|
|||||||
MP_STATE_MEM(gc_sp) = MP_STATE_MEM(gc_stack);
|
MP_STATE_MEM(gc_sp) = MP_STATE_MEM(gc_stack);
|
||||||
|
|
||||||
// scan entire memory looking for blocks which have been marked but not their children
|
// scan entire memory looking for blocks which have been marked but not their children
|
||||||
for (mp_uint_t block = 0; block < MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB; block++) {
|
for (size_t block = 0; block < MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB; block++) {
|
||||||
// trace (again) if mark bit set
|
// trace (again) if mark bit set
|
||||||
if (ATB_GET_KIND(block) == AT_MARK) {
|
if (ATB_GET_KIND(block) == AT_MARK) {
|
||||||
*MP_STATE_MEM(gc_sp)++ = block;
|
*MP_STATE_MEM(gc_sp)++ = block;
|
||||||
@ -303,8 +303,8 @@ void gc_info(gc_info_t *info) {
|
|||||||
info->num_1block = 0;
|
info->num_1block = 0;
|
||||||
info->num_2block = 0;
|
info->num_2block = 0;
|
||||||
info->max_block = 0;
|
info->max_block = 0;
|
||||||
for (mp_uint_t block = 0, len = 0; block < MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB; block++) {
|
for (size_t block = 0, len = 0; block < MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB; block++) {
|
||||||
mp_uint_t kind = ATB_GET_KIND(block);
|
size_t kind = ATB_GET_KIND(block);
|
||||||
if (kind == AT_FREE || kind == AT_HEAD) {
|
if (kind == AT_FREE || kind == AT_HEAD) {
|
||||||
if (len == 1) {
|
if (len == 1) {
|
||||||
info->num_1block += 1;
|
info->num_1block += 1;
|
||||||
@ -342,7 +342,7 @@ void gc_info(gc_info_t *info) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void *gc_alloc(size_t n_bytes, bool has_finaliser) {
|
void *gc_alloc(size_t n_bytes, bool has_finaliser) {
|
||||||
mp_uint_t n_blocks = ((n_bytes + BYTES_PER_BLOCK - 1) & (~(BYTES_PER_BLOCK - 1))) / BYTES_PER_BLOCK;
|
size_t n_blocks = ((n_bytes + BYTES_PER_BLOCK - 1) & (~(BYTES_PER_BLOCK - 1))) / BYTES_PER_BLOCK;
|
||||||
DEBUG_printf("gc_alloc(" UINT_FMT " bytes -> " UINT_FMT " blocks)\n", n_bytes, n_blocks);
|
DEBUG_printf("gc_alloc(" UINT_FMT " bytes -> " UINT_FMT " blocks)\n", n_bytes, n_blocks);
|
||||||
|
|
||||||
// check if GC is locked
|
// check if GC is locked
|
||||||
@ -355,10 +355,10 @@ void *gc_alloc(size_t n_bytes, bool has_finaliser) {
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
mp_uint_t i;
|
size_t i;
|
||||||
mp_uint_t end_block;
|
size_t end_block;
|
||||||
mp_uint_t start_block;
|
size_t start_block;
|
||||||
mp_uint_t n_free = 0;
|
size_t n_free = 0;
|
||||||
int collected = !MP_STATE_MEM(gc_auto_collect_enabled);
|
int collected = !MP_STATE_MEM(gc_auto_collect_enabled);
|
||||||
for (;;) {
|
for (;;) {
|
||||||
|
|
||||||
@ -400,7 +400,7 @@ found:
|
|||||||
|
|
||||||
// mark rest of blocks as used tail
|
// mark rest of blocks as used tail
|
||||||
// TODO for a run of many blocks can make this more efficient
|
// TODO for a run of many blocks can make this more efficient
|
||||||
for (mp_uint_t bl = start_block + 1; bl <= end_block; bl++) {
|
for (size_t bl = start_block + 1; bl <= end_block; bl++) {
|
||||||
ATB_FREE_TO_TAIL(bl);
|
ATB_FREE_TO_TAIL(bl);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -454,7 +454,7 @@ void gc_free(void *ptr) {
|
|||||||
DEBUG_printf("gc_free(%p)\n", ptr);
|
DEBUG_printf("gc_free(%p)\n", ptr);
|
||||||
|
|
||||||
if (VERIFY_PTR(ptr)) {
|
if (VERIFY_PTR(ptr)) {
|
||||||
mp_uint_t block = BLOCK_FROM_PTR(ptr);
|
size_t block = BLOCK_FROM_PTR(ptr);
|
||||||
if (ATB_GET_KIND(block) == AT_HEAD) {
|
if (ATB_GET_KIND(block) == AT_HEAD) {
|
||||||
#if MICROPY_ENABLE_FINALISER
|
#if MICROPY_ENABLE_FINALISER
|
||||||
FTB_CLEAR(block);
|
FTB_CLEAR(block);
|
||||||
@ -483,10 +483,10 @@ void gc_free(void *ptr) {
|
|||||||
|
|
||||||
size_t gc_nbytes(const void *ptr) {
|
size_t gc_nbytes(const void *ptr) {
|
||||||
if (VERIFY_PTR(ptr)) {
|
if (VERIFY_PTR(ptr)) {
|
||||||
mp_uint_t block = BLOCK_FROM_PTR(ptr);
|
size_t block = BLOCK_FROM_PTR(ptr);
|
||||||
if (ATB_GET_KIND(block) == AT_HEAD) {
|
if (ATB_GET_KIND(block) == AT_HEAD) {
|
||||||
// work out number of consecutive blocks in the chain starting with this on
|
// work out number of consecutive blocks in the chain starting with this on
|
||||||
mp_uint_t n_blocks = 0;
|
size_t n_blocks = 0;
|
||||||
do {
|
do {
|
||||||
n_blocks += 1;
|
n_blocks += 1;
|
||||||
} while (ATB_GET_KIND(block + n_blocks) == AT_TAIL);
|
} while (ATB_GET_KIND(block + n_blocks) == AT_TAIL);
|
||||||
@ -551,7 +551,7 @@ void *gc_realloc(void *ptr_in, size_t n_bytes, bool allow_move) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// get first block
|
// get first block
|
||||||
mp_uint_t block = BLOCK_FROM_PTR(ptr);
|
size_t block = BLOCK_FROM_PTR(ptr);
|
||||||
|
|
||||||
// sanity check the ptr is pointing to the head of a block
|
// sanity check the ptr is pointing to the head of a block
|
||||||
if (ATB_GET_KIND(block) != AT_HEAD) {
|
if (ATB_GET_KIND(block) != AT_HEAD) {
|
||||||
@ -559,7 +559,7 @@ void *gc_realloc(void *ptr_in, size_t n_bytes, bool allow_move) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// compute number of new blocks that are requested
|
// compute number of new blocks that are requested
|
||||||
mp_uint_t new_blocks = (n_bytes + BYTES_PER_BLOCK - 1) / BYTES_PER_BLOCK;
|
size_t new_blocks = (n_bytes + BYTES_PER_BLOCK - 1) / BYTES_PER_BLOCK;
|
||||||
|
|
||||||
// Get the total number of consecutive blocks that are already allocated to
|
// Get the total number of consecutive blocks that are already allocated to
|
||||||
// this chunk of memory, and then count the number of free blocks following
|
// this chunk of memory, and then count the number of free blocks following
|
||||||
@ -567,10 +567,10 @@ void *gc_realloc(void *ptr_in, size_t n_bytes, bool allow_move) {
|
|||||||
// free blocks to satisfy the realloc. Note that we need to compute the
|
// free blocks to satisfy the realloc. Note that we need to compute the
|
||||||
// total size of the existing memory chunk so we can correctly and
|
// total size of the existing memory chunk so we can correctly and
|
||||||
// efficiently shrink it (see below for shrinking code).
|
// efficiently shrink it (see below for shrinking code).
|
||||||
mp_uint_t n_free = 0;
|
size_t n_free = 0;
|
||||||
mp_uint_t n_blocks = 1; // counting HEAD block
|
size_t n_blocks = 1; // counting HEAD block
|
||||||
mp_uint_t max_block = MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB;
|
size_t max_block = MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB;
|
||||||
for (mp_uint_t bl = block + n_blocks; bl < max_block; bl++) {
|
for (size_t bl = block + n_blocks; bl < max_block; bl++) {
|
||||||
byte block_type = ATB_GET_KIND(bl);
|
byte block_type = ATB_GET_KIND(bl);
|
||||||
if (block_type == AT_TAIL) {
|
if (block_type == AT_TAIL) {
|
||||||
n_blocks++;
|
n_blocks++;
|
||||||
@ -595,7 +595,7 @@ void *gc_realloc(void *ptr_in, size_t n_bytes, bool allow_move) {
|
|||||||
// check if we can shrink the allocated area
|
// check if we can shrink the allocated area
|
||||||
if (new_blocks < n_blocks) {
|
if (new_blocks < n_blocks) {
|
||||||
// free unneeded tail blocks
|
// free unneeded tail blocks
|
||||||
for (mp_uint_t bl = block + new_blocks, count = n_blocks - new_blocks; count > 0; bl++, count--) {
|
for (size_t bl = block + new_blocks, count = n_blocks - new_blocks; count > 0; bl++, count--) {
|
||||||
ATB_ANY_TO_FREE(bl);
|
ATB_ANY_TO_FREE(bl);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -614,7 +614,7 @@ void *gc_realloc(void *ptr_in, size_t n_bytes, bool allow_move) {
|
|||||||
// check if we can expand in place
|
// check if we can expand in place
|
||||||
if (new_blocks <= n_blocks + n_free) {
|
if (new_blocks <= n_blocks + n_free) {
|
||||||
// mark few more blocks as used tail
|
// mark few more blocks as used tail
|
||||||
for (mp_uint_t bl = block + n_blocks; bl < block + new_blocks; bl++) {
|
for (size_t bl = block + n_blocks; bl < block + new_blocks; bl++) {
|
||||||
assert(ATB_GET_KIND(bl) == AT_FREE);
|
assert(ATB_GET_KIND(bl) == AT_FREE);
|
||||||
ATB_FREE_TO_TAIL(bl);
|
ATB_FREE_TO_TAIL(bl);
|
||||||
}
|
}
|
||||||
@ -665,18 +665,18 @@ void gc_dump_info(void) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void gc_dump_alloc_table(void) {
|
void gc_dump_alloc_table(void) {
|
||||||
static const mp_uint_t DUMP_BYTES_PER_LINE = 64;
|
static const size_t DUMP_BYTES_PER_LINE = 64;
|
||||||
#if !EXTENSIVE_HEAP_PROFILING
|
#if !EXTENSIVE_HEAP_PROFILING
|
||||||
// When comparing heap output we don't want to print the starting
|
// When comparing heap output we don't want to print the starting
|
||||||
// pointer of the heap because it changes from run to run.
|
// pointer of the heap because it changes from run to run.
|
||||||
mp_printf(&mp_plat_print, "GC memory layout; from %p:", MP_STATE_MEM(gc_pool_start));
|
mp_printf(&mp_plat_print, "GC memory layout; from %p:", MP_STATE_MEM(gc_pool_start));
|
||||||
#endif
|
#endif
|
||||||
for (mp_uint_t bl = 0; bl < MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB; bl++) {
|
for (size_t bl = 0; bl < MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB; bl++) {
|
||||||
if (bl % DUMP_BYTES_PER_LINE == 0) {
|
if (bl % DUMP_BYTES_PER_LINE == 0) {
|
||||||
// a new line of blocks
|
// a new line of blocks
|
||||||
{
|
{
|
||||||
// check if this line contains only free blocks
|
// check if this line contains only free blocks
|
||||||
mp_uint_t bl2 = bl;
|
size_t bl2 = bl;
|
||||||
while (bl2 < MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB && ATB_GET_KIND(bl2) == AT_FREE) {
|
while (bl2 < MP_STATE_MEM(gc_alloc_table_byte_len) * BLOCKS_PER_ATB && ATB_GET_KIND(bl2) == AT_FREE) {
|
||||||
bl2++;
|
bl2++;
|
||||||
}
|
}
|
||||||
|
10
py/mpstate.h
10
py/mpstate.h
@ -48,7 +48,7 @@ typedef struct _mp_state_mem_t {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
byte *gc_alloc_table_start;
|
byte *gc_alloc_table_start;
|
||||||
mp_uint_t gc_alloc_table_byte_len;
|
size_t gc_alloc_table_byte_len;
|
||||||
#if MICROPY_ENABLE_FINALISER
|
#if MICROPY_ENABLE_FINALISER
|
||||||
byte *gc_finaliser_table_start;
|
byte *gc_finaliser_table_start;
|
||||||
#endif
|
#endif
|
||||||
@ -56,8 +56,8 @@ typedef struct _mp_state_mem_t {
|
|||||||
byte *gc_pool_end;
|
byte *gc_pool_end;
|
||||||
|
|
||||||
int gc_stack_overflow;
|
int gc_stack_overflow;
|
||||||
mp_uint_t gc_stack[MICROPY_ALLOC_GC_STACK_SIZE];
|
size_t gc_stack[MICROPY_ALLOC_GC_STACK_SIZE];
|
||||||
mp_uint_t *gc_sp;
|
size_t *gc_sp;
|
||||||
uint16_t gc_lock_depth;
|
uint16_t gc_lock_depth;
|
||||||
|
|
||||||
// This variable controls auto garbage collection. If set to 0 then the
|
// This variable controls auto garbage collection. If set to 0 then the
|
||||||
@ -65,10 +65,10 @@ typedef struct _mp_state_mem_t {
|
|||||||
// you can still allocate/free memory and also explicitly call gc_collect.
|
// you can still allocate/free memory and also explicitly call gc_collect.
|
||||||
uint16_t gc_auto_collect_enabled;
|
uint16_t gc_auto_collect_enabled;
|
||||||
|
|
||||||
mp_uint_t gc_last_free_atb_index;
|
size_t gc_last_free_atb_index;
|
||||||
|
|
||||||
#if MICROPY_PY_GC_COLLECT_RETVAL
|
#if MICROPY_PY_GC_COLLECT_RETVAL
|
||||||
mp_uint_t gc_collected;
|
size_t gc_collected;
|
||||||
#endif
|
#endif
|
||||||
} mp_state_mem_t;
|
} mp_state_mem_t;
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user