forked from JuliaLang/julia
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrepl-basic.c
148 lines (137 loc) · 3.72 KB
/
repl-basic.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
#include "repl.h"
#include <unistd.h>
char *stdin_buf = NULL;
unsigned long stdin_buf_len = 0;
unsigned long stdin_buf_maxlen = 128;
void init_repl_environment(int argc, char *argv[])
{
stdin_buf = malloc(stdin_buf_maxlen);
}
void jl_prep_terminal(int meta_flag)
{
}
/* Restore the terminal's normal settings and modes. */
void jl_deprep_terminal()
{
}
void jl_input_line_callback(char *input)
{
if (input) {
jl_value_t *ast = jl_parse_input_line(input);
int line_done = !ast || !jl_is_expr(ast) ||
(((jl_expr_t*)ast)->head != jl_continue_sym);
if (line_done) {
jl_deprep_terminal();
int doprint = !ends_with_semicolon(input);
stdin_buf[0] = 0; //also sets input[0] == 0
stdin_buf_len = 0;
handle_input(ast, 0, doprint);
}
}
}
static char *given_prompt=NULL;
static char *prompt_to_use=NULL;
void repl_callback_enable(char *prompt)
{
if (given_prompt == NULL || strcmp(prompt, given_prompt)) {
if (given_prompt) free(given_prompt);
given_prompt = strdup(prompt);
if (prompt_to_use) free(prompt_to_use);
// remove \001 and \002
size_t n = strlen(prompt);
prompt_to_use = malloc(n+1);
size_t o=0;
for(size_t i=0; i < n; i++) {
if (prompt[i] > 2)
prompt_to_use[o++] = prompt[i];
}
prompt_to_use[o] = '\0';
}
jl_write(jl_uv_stdout, prompt_to_use, strlen(prompt_to_use));
jl_prep_terminal(1);
}
static void stdin_buf_pushc(char c)
{
if (stdin_buf_len >= stdin_buf_maxlen) {
stdin_buf_maxlen *= 2;
stdin_buf = realloc(stdin_buf, stdin_buf_maxlen);
if (!stdin_buf) {
// we can safely ignore this error and continue, if that is preferred
perror("realloc");
exit(EXIT_FAILURE);
}
}
stdin_buf[stdin_buf_len] = c;
stdin_buf_len++;
}
static void basic_stdin_callback(void)
{
stdin_buf_pushc('\0');
stdin_buf_len--;
jl_input_line_callback(stdin_buf);
}
void jl_readBuffer(char* base, ssize_t nread)
{
char *start = base;
int esc = 0;
int newline = 0;
while (*start != 0 && nread > 0) {
if (*start < 32 || *start == 127) {
switch (*start) {
case '\n':
case '\r':
//jl_putc('\n', jl_uv_stdout);
stdin_buf_pushc('\n');
newline = 1;
break;
case '\x03':
JL_WRITE(jl_uv_stdout, "^C\n", 3);
jl_clear_input();
break;
case '\x04':
raise(SIGTERM);
break;
case '\x1A':
#ifndef __WIN32__
raise(SIGTSTP);
#endif
break;
case '\e':
esc = 1;
break;
case 127:
case '\b':
if (stdin_buf_len > 0 && stdin_buf[stdin_buf_len-1] != '\n') {
stdin_buf_len--;
JL_WRITE(jl_uv_stdout,"\b \b",3);
}
}
}
else if (esc == 1) {
if (*start == '[') {
esc = 2;
}
else {
esc = 0;
}
}
else if (esc == 2) {
// for now, we just block all 3 character ctrl signals that I know about
// this misses delete ^[[3~ and possibly others?
esc = 0;
}
else {
stdin_buf_pushc(*start);
}
start++;
nread--;
}
if (newline) basic_stdin_callback();
}
void jl_clear_input(void)
{
stdin_buf_len = 0;
stdin_buf[0] = 0;
JL_WRITE(jl_uv_stdout,"\n",1);
repl_callback_enable(prompt_to_use);
}