From 87bc8e2b3d66d3e87c2d01a7b3107a8ecce2a07f Mon Sep 17 00:00:00 2001 From: Paul Sokolovsky Date: Thu, 15 Jan 2015 10:46:27 +0200 Subject: [PATCH] pyexec: Add event-driven variant pyexec_friendly_repl(). pyexec_friendly_repl_process_char() and friends, useful for ports which integrate into existing cooperative multitasking system. Unlike readline() refactor before, this was implemented in less formal, trial&error process, minor functionality regressions are still known (like soft&hard reset support). So, original loop-based pyexec_friendly_repl() is left intact, specific implementation selectable by config setting. --- py/mpconfig.h | 5 +++ stmhal/pyexec.c | 110 ++++++++++++++++++++++++++++++++++++++++++++++++ stmhal/pyexec.h | 3 ++ 3 files changed, 118 insertions(+) diff --git a/py/mpconfig.h b/py/mpconfig.h index 5718ffacbc..eb39d54c4d 100644 --- a/py/mpconfig.h +++ b/py/mpconfig.h @@ -235,6 +235,11 @@ #define MICROPY_HELPER_REPL (0) #endif +// Whether port requires event-driven REPL functions +#ifndef MICROPY_REPL_EVENT_DRIVEN +#define MICROPY_REPL_EVENT_DRIVEN (0) +#endif + // Whether to include lexer helper function for unix #ifndef MICROPY_HELPER_LEXER_UNIX #define MICROPY_HELPER_LEXER_UNIX (0) diff --git a/stmhal/pyexec.c b/stmhal/pyexec.c index 98cdb4ea7a..f4b8c10fd6 100644 --- a/stmhal/pyexec.c +++ b/stmhal/pyexec.c @@ -195,6 +195,114 @@ raw_repl_reset: } } +#if MICROPY_REPL_EVENT_DRIVEN + +typedef struct _friendly_repl_t { + vstr_t line; + bool cont_line; +} friendly_repl_t; + +friendly_repl_t repl; + +void pyexec_friendly_repl_init(void) { + vstr_init(&repl.line, 32); + repl.cont_line = false; + readline_init(&repl.line); + stdout_tx_str(">>> "); +} + +void pyexec_friendly_repl_reset() { + repl.cont_line = false; + vstr_reset(&repl.line); + readline_init(&repl.line); +} + +int pyexec_friendly_repl_process_char(int c) { + int ret = readline_process_char(c); + + if (!repl.cont_line) { + + if (ret == CHAR_CTRL_A) { + // change to raw REPL + pyexec_mode_kind = PYEXEC_MODE_RAW_REPL; + stdout_tx_str("\r\n"); + vstr_clear(&repl.line); + return PYEXEC_SWITCH_MODE; + } else if (ret == CHAR_CTRL_B) { + // reset friendly REPL + stdout_tx_str("\r\n"); + goto friendly_repl_reset; + } else if (ret == CHAR_CTRL_C) { + // break + stdout_tx_str("\r\n"); + goto input_restart; + } else if (ret == CHAR_CTRL_D) { + // exit for a soft reset + stdout_tx_str("\r\n"); + vstr_clear(&repl.line); + return PYEXEC_FORCED_EXIT; + } else if (vstr_len(&repl.line) == 0) { + //goto input_restart; + } + + if (ret < 0) { + return 0; + } + + if (!mp_repl_continue_with_input(vstr_str(&repl.line))) { + goto exec; + } + + vstr_add_char(&repl.line, '\n'); + repl.cont_line = true; + stdout_tx_str("... "); + readline_note_newline(); + return 0; + + } else { + + if (ret == CHAR_CTRL_C) { + // cancel everything + stdout_tx_str("\r\n"); + repl.cont_line = false; + goto input_restart; + } else if (ret == CHAR_CTRL_D) { + // stop entering compound statement + goto exec; + } + + if (ret < 0) { + return 0; + } + + if (mp_repl_continue_with_input(vstr_str(&repl.line))) { + vstr_add_char(&repl.line, '\n'); + stdout_tx_str("... "); + readline_note_newline(); + return 0; + } + +exec: ; + mp_lexer_t *lex = mp_lexer_new_from_str_len(MP_QSTR__lt_stdin_gt_, vstr_str(&repl.line), vstr_len(&repl.line), 0); + if (lex == NULL) { + printf("MemoryError\n"); + } else { + int ret = parse_compile_execute(lex, MP_PARSE_SINGLE_INPUT, EXEC_FLAG_ALLOW_DEBUGGING | EXEC_FLAG_IS_REPL); + if (ret & PYEXEC_FORCED_EXIT) { + return ret; + } + } + +friendly_repl_reset: // TODO +input_restart: + pyexec_friendly_repl_reset(); + stdout_tx_str(">>> "); + return 0; + } +} + +#else //MICROPY_REPL_EVENT_DRIVEN + int pyexec_friendly_repl(void) { vstr_t line; vstr_init(&line, 32); @@ -280,6 +388,8 @@ friendly_repl_reset: } } +#endif //MICROPY_REPL_EVENT_DRIVEN + int pyexec_file(const char *filename) { mp_lexer_t *lex = mp_lexer_new_from_file(filename); diff --git a/stmhal/pyexec.h b/stmhal/pyexec.h index e360273519..d01d505a78 100644 --- a/stmhal/pyexec.h +++ b/stmhal/pyexec.h @@ -32,9 +32,12 @@ typedef enum { extern pyexec_mode_kind_t pyexec_mode_kind; #define PYEXEC_FORCED_EXIT (0x100) +#define PYEXEC_SWITCH_MODE (0x200) int pyexec_raw_repl(void); int pyexec_friendly_repl(void); int pyexec_file(const char *filename); +void pyexec_friendly_repl_init(void); +int pyexec_friendly_repl_process_char(int c); MP_DECLARE_CONST_FUN_OBJ(pyb_set_repl_info_obj);