|
|
|
/*
|
|
|
|
* This file is part of the Micro Python project, http://micropython.org/
|
|
|
|
*
|
|
|
|
* The MIT License (MIT)
|
|
|
|
*
|
|
|
|
* Copyright (c) 2013, 2014 Damien P. George
|
|
|
|
*
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include "py/mpstate.h"
|
|
|
|
#include "input.h"
|
|
|
|
|
|
|
|
#if MICROPY_USE_READLINE == 1
|
|
|
|
#include MICROPY_HAL_H
|
|
|
|
#include "lib/mp-readline/readline.h"
|
|
|
|
#elif MICROPY_USE_READLINE == 2
|
|
|
|
#include <readline/readline.h>
|
|
|
|
#include <readline/history.h>
|
|
|
|
#include <readline/tilde.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
char *prompt(char *p) {
|
|
|
|
#if MICROPY_USE_READLINE == 1
|
|
|
|
// MicroPython supplied readline
|
|
|
|
vstr_t vstr;
|
|
|
|
vstr_init(&vstr, 16);
|
|
|
|
mp_hal_stdio_mode_raw();
|
|
|
|
int ret = readline(&vstr, p);
|
|
|
|
mp_hal_stdio_mode_orig();
|
|
|
|
if (ret != 0) {
|
|
|
|
vstr_clear(&vstr);
|
|
|
|
if (ret == CHAR_CTRL_D) {
|
|
|
|
// EOF
|
|
|
|
return NULL;
|
|
|
|
} else {
|
|
|
|
printf("\n");
|
|
|
|
char *line = malloc(1);
|
|
|
|
line[0] = '\0';
|
|
|
|
return line;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
vstr_null_terminated_str(&vstr);
|
|
|
|
char *line = malloc(vstr.len + 1);
|
|
|
|
memcpy(line, vstr.buf, vstr.len + 1);
|
|
|
|
vstr_clear(&vstr);
|
|
|
|
#elif MICROPY_USE_READLINE == 2
|
|
|
|
// GNU readline
|
|
|
|
char *line = readline(p);
|
|
|
|
if (line) {
|
|
|
|
add_history(line);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
// simple read string
|
|
|
|
static char buf[256];
|
|
|
|
fputs(p, stdout);
|
|
|
|
char *s = fgets(buf, sizeof(buf), stdin);
|
|
|
|
if (!s) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
int l = strlen(buf);
|
|
|
|
if (buf[l - 1] == '\n') {
|
|
|
|
buf[l - 1] = 0;
|
|
|
|
} else {
|
|
|
|
l++;
|
|
|
|
}
|
|
|
|
char *line = malloc(l);
|
|
|
|
memcpy(line, buf, l);
|
|
|
|
#endif
|
|
|
|
return line;
|
|
|
|
}
|
|
|
|
|
|
|
|
void prompt_read_history(void) {
|
|
|
|
#if MICROPY_USE_READLINE_HISTORY
|
|
|
|
#if MICROPY_USE_READLINE == 1
|
|
|
|
readline_init0(); // will clear history pointers
|
|
|
|
char *home = getenv("HOME");
|
|
|
|
if (home != NULL) {
|
|
|
|
vstr_t vstr;
|
|
|
|
vstr_init(&vstr, 50);
|
|
|
|
vstr_printf(&vstr, "%s/.micropython.history", home);
|
|
|
|
FILE *fp = fopen(vstr_null_terminated_str(&vstr), "r");
|
|
|
|
if (fp != NULL) {
|
|
|
|
vstr_reset(&vstr);
|
|
|
|
for (;;) {
|
|
|
|
int c = fgetc(fp);
|
|
|
|
if (c == EOF || c == '\n') {
|
|
|
|
readline_push_history(vstr_null_terminated_str(&vstr));
|
|
|
|
if (c == EOF) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
vstr_reset(&vstr);
|
|
|
|
} else {
|
|
|
|
vstr_add_byte(&vstr, c);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
vstr_clear(&vstr);
|
|
|
|
}
|
|
|
|
#elif MICROPY_USE_READLINE == 2
|
|
|
|
read_history(tilde_expand("~/.micropython.history"));
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void prompt_write_history(void) {
|
|
|
|
#if MICROPY_USE_READLINE_HISTORY
|
|
|
|
#if MICROPY_USE_READLINE == 1
|
|
|
|
char *home = getenv("HOME");
|
|
|
|
if (home != NULL) {
|
|
|
|
vstr_t vstr;
|
|
|
|
vstr_init(&vstr, 50);
|
|
|
|
vstr_printf(&vstr, "%s/.micropython.history", home);
|
|
|
|
FILE *fp = fopen(vstr_null_terminated_str(&vstr), "w");
|
|
|
|
if (fp != NULL) {
|
|
|
|
for (int i = MP_ARRAY_SIZE(MP_STATE_PORT(readline_hist)) - 1; i >= 0; i--) {
|
|
|
|
const char *line = MP_STATE_PORT(readline_hist)[i];
|
|
|
|
if (line != NULL) {
|
|
|
|
fwrite(line, 1, strlen(line), fp);
|
|
|
|
fputc('\n', fp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose(fp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#elif MICROPY_USE_READLINE == 2
|
|
|
|
write_history(tilde_expand("~/.micropython.history"));
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
STATIC mp_obj_t mp_builtin_input(uint n_args, const mp_obj_t *args) {
|
|
|
|
if (n_args == 1) {
|
|
|
|
mp_obj_print(args[0], PRINT_STR);
|
|
|
|
}
|
|
|
|
|
|
|
|
char *line = prompt("");
|
|
|
|
if (line == NULL) {
|
|
|
|
nlr_raise(mp_obj_new_exception(&mp_type_EOFError));
|
|
|
|
}
|
|
|
|
mp_obj_t o = mp_obj_new_str(line, strlen(line), false);
|
|
|
|
free(line);
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
MP_DEFINE_CONST_FUN_OBJ_VAR_BETWEEN(mp_builtin_input_obj, 0, 1, mp_builtin_input);
|