py/runtime: mp_stack_ctrl_init() should be called immediately on startup.
Calling it from mp_init() is too late for some ports (like Unix), and leads to incomplete stack frame being captured, with following GC issues. So, now each port should call mp_stack_ctrl_init() on its own, ASAP after startup, and taking special precautions so it really was called before stack variables get allocated (because if such variable with a pointer is missed, it may lead to over-collecting (typical symptom is segfaulting)).
This commit is contained in:
parent
850212203a
commit
d3b1f0b627
@ -61,7 +61,6 @@ const mp_obj_module_t mp_module___main__ = {
|
|||||||
|
|
||||||
void mp_init(void) {
|
void mp_init(void) {
|
||||||
qstr_init();
|
qstr_init();
|
||||||
mp_stack_ctrl_init();
|
|
||||||
|
|
||||||
// no pending exceptions to start with
|
// no pending exceptions to start with
|
||||||
MP_STATE_VM(mp_pending_exception) = MP_OBJ_NULL;
|
MP_STATE_VM(mp_pending_exception) = MP_OBJ_NULL;
|
||||||
|
@ -32,6 +32,7 @@ void do_str(const char *src, mp_parse_input_kind_t input_kind) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char **argv) {
|
int main(int argc, char **argv) {
|
||||||
|
mp_stack_ctrl_init();
|
||||||
mp_stack_set_limit(10240);
|
mp_stack_set_limit(10240);
|
||||||
void *heap = malloc(16 * 1024);
|
void *heap = malloc(16 * 1024);
|
||||||
gc_init(heap, (char*)heap + 16 * 1024);
|
gc_init(heap, (char*)heap + 16 * 1024);
|
||||||
|
@ -49,6 +49,7 @@ end:
|
|||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
const char a[] = {"sim"};
|
const char a[] = {"sim"};
|
||||||
|
mp_stack_ctrl_init();
|
||||||
mp_stack_set_limit(10240);
|
mp_stack_set_limit(10240);
|
||||||
void *heap = malloc(256 * 1024);
|
void *heap = malloc(256 * 1024);
|
||||||
gc_init(heap, (char*)heap + 256 * 1024);
|
gc_init(heap, (char*)heap + 256 * 1024);
|
||||||
|
@ -335,6 +335,7 @@ int main(void) {
|
|||||||
|
|
||||||
// Stack limit should be less than real stack size, so we have a chance
|
// Stack limit should be less than real stack size, so we have a chance
|
||||||
// to recover from limit hit. (Limit is measured in bytes.)
|
// to recover from limit hit. (Limit is measured in bytes.)
|
||||||
|
mp_stack_ctrl_init();
|
||||||
mp_stack_set_limit((char*)&_ram_end - (char*)&_heap_end - 1024);
|
mp_stack_set_limit((char*)&_ram_end - (char*)&_heap_end - 1024);
|
||||||
|
|
||||||
/* STM32F4xx HAL library initialization:
|
/* STM32F4xx HAL library initialization:
|
||||||
|
@ -253,6 +253,7 @@ int main(void) {
|
|||||||
#define SCB_CCR_STKALIGN (1 << 9)
|
#define SCB_CCR_STKALIGN (1 << 9)
|
||||||
SCB_CCR |= SCB_CCR_STKALIGN;
|
SCB_CCR |= SCB_CCR_STKALIGN;
|
||||||
|
|
||||||
|
mp_stack_ctrl_init();
|
||||||
mp_stack_set_limit(10240);
|
mp_stack_set_limit(10240);
|
||||||
|
|
||||||
pinMode(LED_BUILTIN, OUTPUT);
|
pinMode(LED_BUILTIN, OUTPUT);
|
||||||
|
12
unix/main.c
12
unix/main.c
@ -376,7 +376,19 @@ STATIC void set_sys_argv(char *argv[], int argc, int start_arg) {
|
|||||||
#define PATHLIST_SEP_CHAR ':'
|
#define PATHLIST_SEP_CHAR ':'
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
MP_NOINLINE int main_(int argc, char **argv);
|
||||||
|
|
||||||
int main(int argc, char **argv) {
|
int main(int argc, char **argv) {
|
||||||
|
// We should capture stack top ASAP after start, and it should be
|
||||||
|
// captured guaranteedly before any other stack variables are allocated.
|
||||||
|
// For this, actual main (renamed main_) should not be inlined into
|
||||||
|
// this function. main_() itself may have other functions inlined (with
|
||||||
|
// their own stack variables), that's why we need this main/main_ split.
|
||||||
|
mp_stack_ctrl_init();
|
||||||
|
return main_(argc, argv);
|
||||||
|
}
|
||||||
|
|
||||||
|
MP_NOINLINE int main_(int argc, char **argv) {
|
||||||
mp_stack_set_limit(40000 * (BYTES_PER_WORD / 4));
|
mp_stack_set_limit(40000 * (BYTES_PER_WORD / 4));
|
||||||
|
|
||||||
pre_process_options(argc, argv);
|
pre_process_options(argc, argv);
|
||||||
|
@ -214,6 +214,10 @@ void mp_unix_mark_exec(void);
|
|||||||
#define MP_PLAT_ALLOC_EXEC(min_size, ptr, size) mp_unix_alloc_exec(min_size, ptr, size)
|
#define MP_PLAT_ALLOC_EXEC(min_size, ptr, size) mp_unix_alloc_exec(min_size, ptr, size)
|
||||||
#define MP_PLAT_FREE_EXEC(ptr, size) mp_unix_free_exec(ptr, size)
|
#define MP_PLAT_FREE_EXEC(ptr, size) mp_unix_free_exec(ptr, size)
|
||||||
|
|
||||||
|
#ifndef MP_NOINLINE
|
||||||
|
#define MP_NOINLINE __attribute__((noinline))
|
||||||
|
#endif
|
||||||
|
|
||||||
#if MICROPY_PY_OS_DUPTERM
|
#if MICROPY_PY_OS_DUPTERM
|
||||||
#define MP_PLAT_PRINT_STRN(str, len) mp_hal_stdout_tx_strn_cooked(str, len)
|
#define MP_PLAT_PRINT_STRN(str, len) mp_hal_stdout_tx_strn_cooked(str, len)
|
||||||
#else
|
#else
|
||||||
|
@ -178,6 +178,10 @@ extern const struct _mp_obj_module_t mp_module_time;
|
|||||||
#include "init.h"
|
#include "init.h"
|
||||||
#include "sleep.h"
|
#include "sleep.h"
|
||||||
|
|
||||||
|
#ifdef __GNUC__
|
||||||
|
#define MP_NOINLINE __attribute__((noinline))
|
||||||
|
#endif
|
||||||
|
|
||||||
// MSVC specifics
|
// MSVC specifics
|
||||||
#ifdef _MSC_VER
|
#ifdef _MSC_VER
|
||||||
|
|
||||||
@ -191,6 +195,7 @@ extern const struct _mp_obj_module_t mp_module_time;
|
|||||||
// CL specific overrides from mpconfig
|
// CL specific overrides from mpconfig
|
||||||
|
|
||||||
#define NORETURN __declspec(noreturn)
|
#define NORETURN __declspec(noreturn)
|
||||||
|
#define MP_NOINLINE __declspec(noinline)
|
||||||
#define MP_LIKELY(x) (x)
|
#define MP_LIKELY(x) (x)
|
||||||
#define MP_UNLIKELY(x) (x)
|
#define MP_UNLIKELY(x) (x)
|
||||||
#define MICROPY_PORT_CONSTANTS { "dummy", 0 } //can't have zero-sized array
|
#define MICROPY_PORT_CONSTANTS { "dummy", 0 } //can't have zero-sized array
|
||||||
|
Loading…
x
Reference in New Issue
Block a user