2014-03-12 18:06:26 -07:00
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
#include <stm32f4xx_hal.h>
|
|
|
|
|
|
|
|
#include "misc.h"
|
|
|
|
#include "mpconfig.h"
|
|
|
|
#include "qstr.h"
|
|
|
|
#include "obj.h"
|
|
|
|
#include "gc.h"
|
|
|
|
#include "gccollect.h"
|
|
|
|
|
|
|
|
machine_uint_t gc_helper_get_regs_and_sp(machine_uint_t *regs);
|
|
|
|
|
|
|
|
// obsolete
|
|
|
|
// void gc_helper_get_regs_and_clean_stack(machine_uint_t *regs, machine_uint_t heap_end);
|
|
|
|
|
|
|
|
void gc_collect(void) {
|
|
|
|
// get current time, in case we want to time the GC
|
|
|
|
uint32_t start = HAL_GetTick();
|
|
|
|
|
|
|
|
// start the GC
|
|
|
|
gc_collect_start();
|
|
|
|
|
2014-04-12 18:50:20 +01:00
|
|
|
// We need to scan everything in RAM that can hold a pointer.
|
|
|
|
// The data segment is used, but should not contain pointers, so we just scan the bss.
|
|
|
|
gc_collect_root((void**)&_sbss, ((uint32_t)&_ebss - (uint32_t)&_sbss) / sizeof(uint32_t));
|
2014-03-12 18:06:26 -07:00
|
|
|
|
|
|
|
// get the registers and the sp
|
|
|
|
machine_uint_t regs[10];
|
|
|
|
machine_uint_t sp = gc_helper_get_regs_and_sp(regs);
|
|
|
|
|
|
|
|
// trace the stack, including the registers (since they live on the stack in this function)
|
|
|
|
gc_collect_root((void**)sp, ((uint32_t)&_ram_end - sp) / sizeof(uint32_t));
|
|
|
|
|
|
|
|
// end the GC
|
|
|
|
gc_collect_end();
|
|
|
|
|
|
|
|
if (0) {
|
|
|
|
// print GC info
|
|
|
|
uint32_t ticks = HAL_GetTick() - start; // TODO implement a function that does this properly
|
|
|
|
gc_info_t info;
|
|
|
|
gc_info(&info);
|
|
|
|
printf("GC@%lu %lums\n", start, ticks);
|
|
|
|
printf(" %lu total\n", info.total);
|
|
|
|
printf(" %lu : %lu\n", info.used, info.free);
|
|
|
|
printf(" 1=%lu 2=%lu m=%lu\n", info.num_1block, info.num_2block, info.max_block);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static mp_obj_t pyb_gc(void) {
|
|
|
|
gc_collect();
|
|
|
|
return mp_const_none;
|
|
|
|
}
|
|
|
|
|
|
|
|
MP_DEFINE_CONST_FUN_OBJ_0(pyb_gc_obj, pyb_gc);
|