command history in shell

This commit is contained in:
Benji Dial 2021-03-02 20:40:10 -05:00
parent 1d69a46f5d
commit e9a11deef0
3 changed files with 129 additions and 5 deletions

View file

@ -14,9 +14,11 @@ void main(const char *arg) {
term_add_sz("Portland Highway\nType \"help\" for help.\n"); term_add_sz("Portland Highway\nType \"help\" for help.\n");
term_paint(); term_paint();
struct history *h = new_history(200);
char cmd_buf[128]; char cmd_buf[128];
while (1) { while (1) {
read_line(cmd_buf, 127, "> "); read_line(cmd_buf, 127, "> ", h);
run_line(cmd_buf); run_line(cmd_buf);
} }
} }

View file

@ -3,8 +3,14 @@
#include <stdint.h> #include <stdint.h>
struct history;
//returns zero if memory allocation failed or if max_entries was zero or one
struct history *new_history(uint32_t max_entries);
//returns length of string without null terminator //returns length of string without null terminator
//max_length doesn't include null terminator //max_length doesn't include null terminator
uint32_t read_line(char *sz, uint32_t max_length, const char *prompt); //pass null pointer in hist for no history
uint32_t read_line(char *sz, uint32_t max_length, const char *prompt, struct history *hist);
#endif #endif

View file

@ -1,13 +1,82 @@
#include <libterm/terminal.h> #include <libterm/terminal.h>
#include <knob/block.h> #include <knob/block.h>
#include <knob/heap.h>
#include <knob/key.h> #include <knob/key.h>
#include <stdint.h> #include <stdint.h>
//returns length of string without null terminator struct he {
//max_length doesn't include null terminator struct he *prev;
uint32_t read_line(char *sz, uint32_t max_length, const char *prompt) { struct he *next;
const char *data;
};
struct history {
uint32_t entries_left;
struct he *newest_entry;
struct he *oldest_entry;
};
static void move_entry_to_front(struct history *h, struct he *e) {
if (!e || !h || (e == h->newest_entry))
return;
if (e->prev)
e->prev->next = e->next;
if (e->next)
e->next->prev = e->prev;
if (e == h->oldest_entry)
h->oldest_entry = e->next;
e->prev = h->newest_entry;
e->next = 0;
if (h->newest_entry)
h->newest_entry->next = e;
h->newest_entry = e;
}
static void add_to_history(struct history *h, const char *from) {
const uint32_t from_size = strlen(from) + 1;
char *const copy = get_block(from_size);
blockcpy(copy, from, from_size);
struct he *entry;
if (h->entries_left) {
--h->entries_left;
entry = get_block(sizeof(struct he));
}
else {
entry = h->oldest_entry;
h->oldest_entry = entry->next;
entry->next->prev = 0;
free_block(entry->data);
}
entry->data = copy;
entry->prev = h->newest_entry;
entry->next = 0;
if (h->newest_entry)
h->newest_entry->next = entry;
else
h->oldest_entry = entry;
h->newest_entry = entry;
}
struct history *new_history(uint32_t max_entries) {
if (max_entries < 2)
return 0;
struct history *h = get_block(sizeof(struct history));
if (!h)
return 0;
h->newest_entry = 0;
h->oldest_entry = 0;
h->entries_left = max_entries;
return h;
}
uint32_t read_line(char *sz, uint32_t max_length, const char *prompt, struct history *hist) {
term_add_sz(prompt); term_add_sz(prompt);
term_add_char(' '); term_add_char(' ');
term_paint(); term_paint();
@ -15,6 +84,9 @@ uint32_t read_line(char *sz, uint32_t max_length, const char *prompt) {
uint32_t l = 0; uint32_t l = 0;
uint32_t c = 0; uint32_t c = 0;
struct he *h_on = 0;
bool he_modified = false;
while (1) { while (1) {
struct key_packet kp = term_get_key_blocking(); struct key_packet kp = term_get_key_blocking();
switch (kp.key_id) { switch (kp.key_id) {
@ -54,6 +126,7 @@ uint32_t read_line(char *sz, uint32_t max_length, const char *prompt) {
case KEY_BSPACE: case KEY_BSPACE:
if (!c) if (!c)
continue; continue;
he_modified = true;
--c; --c;
--l; --l;
for (uint32_t i = c; i < l; ++i) for (uint32_t i = c; i < l; ++i)
@ -71,16 +144,59 @@ uint32_t read_line(char *sz, uint32_t max_length, const char *prompt) {
term_cursor_right(); term_cursor_right();
} }
sz[l] = '\0'; sz[l] = '\0';
if (hist) {
if (he_modified)
add_to_history(hist, sz);
else if (h_on)
move_entry_to_front(hist, h_on);
}
term_add_char('\n'); term_add_char('\n');
term_paint(); term_paint();
_yield_task(); _yield_task();
return l; return l;
case KEY_UP_ARROW:
if (!hist || !hist->newest_entry || (h_on == hist->oldest_entry))
continue;
he_modified = false;
h_on = h_on ? h_on->prev : hist->newest_entry;
for (uint32_t i = 0; i < c; ++i)
term_cursor_left();
for (uint32_t i = 0; i < l; ++i)
term_add_char(' ');
for (uint32_t i = 0; i < l; ++i)
term_cursor_left();
l = strcpy(sz, h_on->data);
term_add_sn_no_ww(sz, l);
c = l;
term_paint();
continue;
case KEY_DOWN_ARROW:
if (!h_on)
continue;
he_modified = false;
h_on = h_on->next;
for (uint32_t i = 0; i < c; ++i)
term_cursor_left();
for (uint32_t i = 0; i < l; ++i)
term_add_char(' ');
for (uint32_t i = 0; i < l; ++i)
term_cursor_left();
if (h_on) {
l = strcpy(sz, h_on->data);
term_add_sn_no_ww(sz, l);
}
else
l = 0;
c = l;
term_paint();
continue;
default: default:
if (l == max_length) if (l == max_length)
continue; continue;
char ch = key_to_char(kp); char ch = key_to_char(kp);
if (!ch) if (!ch)
continue; continue;
he_modified = true;
if (c == l) { if (c == l) {
++l; ++l;
term_add_char(sz[c++] = ch); term_add_char(sz[c++] = ch);