2012-01-25 10:19:52 +04:00
|
|
|
/* vim: tabstop=4 shiftwidth=4 noexpandtab
|
2011-12-07 05:46:35 +04:00
|
|
|
* E-Shell
|
|
|
|
*
|
2012-07-07 08:08:28 +04:00
|
|
|
* This is the "experimental shell". It provides
|
|
|
|
* a somewhat unix-like shell environment, but does
|
|
|
|
* not include a parser any advanced functionality.
|
|
|
|
* It simply cuts its input into arguments and executes
|
|
|
|
* programs.
|
2011-12-07 05:46:35 +04:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <syscall.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <unistd.h>
|
2012-01-25 10:19:52 +04:00
|
|
|
#include <time.h>
|
2012-10-07 10:48:46 +04:00
|
|
|
#include <dirent.h>
|
2011-12-07 05:46:35 +04:00
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
#include "lib/list.h"
|
|
|
|
|
|
|
|
#define SHELL_COMMANDS 512
|
|
|
|
typedef uint32_t(*shell_command_t) (int argc, char ** argv);
|
|
|
|
char * shell_commands[SHELL_COMMANDS];
|
|
|
|
shell_command_t shell_pointers[SHELL_COMMANDS];
|
|
|
|
uint32_t shell_commands_len = 0;
|
|
|
|
|
|
|
|
#define SHELL_HISTORY_ENTRIES 128
|
|
|
|
char * shell_history[SHELL_HISTORY_ENTRIES];
|
|
|
|
size_t shell_history_count = 0;
|
|
|
|
size_t shell_history_offset = 0;
|
|
|
|
|
|
|
|
size_t shell_scroll = 0;
|
|
|
|
char shell_temp[1024];
|
|
|
|
|
|
|
|
int pid;
|
|
|
|
|
|
|
|
void shell_history_insert(char * str) {
|
|
|
|
if (shell_history_count) {
|
|
|
|
if (!strcmp(str, shell_history[shell_history_count-1])) {
|
|
|
|
free(str);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (shell_history_count == SHELL_HISTORY_ENTRIES) {
|
|
|
|
free(shell_history[shell_history_offset]);
|
|
|
|
shell_history[shell_history_offset] = str;
|
|
|
|
shell_history_offset = (shell_history_offset + 1) % SHELL_HISTORY_ENTRIES;
|
|
|
|
} else {
|
|
|
|
shell_history[shell_history_count] = str;
|
|
|
|
shell_history_count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
char * shell_history_get(size_t item) {
|
|
|
|
return shell_history[(item + shell_history_offset) % SHELL_HISTORY_ENTRIES];
|
|
|
|
}
|
|
|
|
|
|
|
|
char * shell_history_prev(size_t item) {
|
|
|
|
return shell_history_get(shell_history_count - item);
|
|
|
|
}
|
|
|
|
|
|
|
|
void shell_install_command(char * name, shell_command_t func) {
|
|
|
|
if (shell_commands_len == SHELL_COMMANDS) {
|
|
|
|
fprintf(stderr, "Ran out of space for static shell commands. The maximum number of commands is %d\n", SHELL_COMMANDS);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
shell_commands[shell_commands_len] = name;
|
|
|
|
shell_pointers[shell_commands_len] = func;
|
|
|
|
shell_commands_len++;
|
|
|
|
}
|
|
|
|
|
|
|
|
shell_command_t shell_find(char * str) {
|
|
|
|
for (uint32_t i = 0; i < shell_commands_len; ++i) {
|
|
|
|
if (!strcmp(str, shell_commands[i])) {
|
|
|
|
return shell_pointers[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_unbuffered() {
|
|
|
|
printf("\033[1560z");
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
void set_buffered() {
|
|
|
|
printf("\033[1561z");
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
|
|
|
|
void install_commands();
|
|
|
|
|
|
|
|
#define KBD_NORMAL 0
|
|
|
|
#define KBD_ESC_A 1
|
|
|
|
#define KBD_ESC_B 2
|
|
|
|
#define KBD_FUNC 3
|
|
|
|
|
|
|
|
int kbd_state = 0;
|
|
|
|
|
|
|
|
#define KEY_NONE 0
|
|
|
|
#define KEY_BACKSPACE 8
|
2012-10-07 10:48:46 +04:00
|
|
|
#define KEY_CTRL_C 3
|
|
|
|
#define KEY_CTRL_R 18
|
2012-10-07 10:01:39 +04:00
|
|
|
#define KEY_ESCAPE 27
|
|
|
|
#define KEY_NORMAL_MAX 256
|
|
|
|
#define KEY_ARROW_UP 257
|
|
|
|
#define KEY_ARROW_DOWN 258
|
|
|
|
#define KEY_ARROW_RIGHT 259
|
|
|
|
#define KEY_ARROW_LEFT 260
|
|
|
|
#define KEY_BAD_STATE -1
|
|
|
|
|
|
|
|
uint32_t kbd_key(uint8_t c) {
|
|
|
|
switch (kbd_state) {
|
|
|
|
case KBD_NORMAL:
|
|
|
|
switch (c) {
|
|
|
|
case 0x1b:
|
|
|
|
kbd_state = KBD_ESC_A;
|
|
|
|
return KEY_NONE;
|
|
|
|
default:
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
case KBD_ESC_A:
|
|
|
|
switch (c) {
|
|
|
|
case 0x5b:
|
|
|
|
kbd_state = KBD_ESC_B;
|
|
|
|
return KEY_NONE;
|
|
|
|
default:
|
|
|
|
kbd_state = KBD_NORMAL;
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
case KBD_ESC_B:
|
|
|
|
switch (c) {
|
|
|
|
case 0x41:
|
|
|
|
kbd_state = KBD_NORMAL;
|
|
|
|
return KEY_ARROW_UP;
|
|
|
|
case 0x42:
|
|
|
|
kbd_state = KBD_NORMAL;
|
|
|
|
return KEY_ARROW_DOWN;
|
|
|
|
case 0x43:
|
|
|
|
kbd_state = KBD_NORMAL;
|
|
|
|
return KEY_ARROW_RIGHT;
|
|
|
|
case 0x44:
|
|
|
|
kbd_state = KBD_NORMAL;
|
|
|
|
return KEY_ARROW_LEFT;
|
|
|
|
default:
|
|
|
|
kbd_state = KBD_NORMAL;
|
|
|
|
return c;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
fprintf(stderr, "Keyboard in unknown state? Not sure what to do. kbd_state = %d\n", kbd_state);
|
|
|
|
return KEY_BAD_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return KEY_BAD_STATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* Maximum command length */
|
2012-01-28 04:04:39 +04:00
|
|
|
#define LINE_LEN 4096
|
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
/* Current working directory */
|
2012-01-25 10:19:52 +04:00
|
|
|
char cwd[1024] = {'/',0};
|
2012-10-07 10:01:39 +04:00
|
|
|
|
|
|
|
/* Timing type */
|
2012-01-25 10:19:52 +04:00
|
|
|
struct timeval {
|
|
|
|
unsigned int tv_sec;
|
|
|
|
unsigned int tv_usec;
|
|
|
|
};
|
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
/* Username */
|
2012-01-28 04:04:39 +04:00
|
|
|
char username[1024];
|
2012-10-07 10:01:39 +04:00
|
|
|
|
|
|
|
/* Hostname for prompt */
|
2012-01-30 22:10:53 +04:00
|
|
|
char _hostname[256];
|
2012-01-28 04:04:39 +04:00
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
/* function to update the cached username */
|
2012-01-28 04:04:39 +04:00
|
|
|
void getusername() {
|
|
|
|
FILE * passwd = fopen("/etc/passwd", "r");
|
|
|
|
char line[LINE_LEN];
|
|
|
|
|
|
|
|
int uid = syscall_getuid();
|
|
|
|
|
|
|
|
while (fgets(line, LINE_LEN, passwd) != NULL) {
|
|
|
|
|
|
|
|
line[strlen(line)-1] = '\0';
|
|
|
|
|
|
|
|
char *p, *tokens[10], *last;
|
|
|
|
int i = 0;
|
|
|
|
for ((p = strtok_r(line, ":", &last)); p;
|
|
|
|
(p = strtok_r(NULL, ":", &last)), i++) {
|
|
|
|
if (i < 511) tokens[i] = p;
|
|
|
|
}
|
|
|
|
tokens[i] = NULL;
|
|
|
|
|
|
|
|
if (atoi(tokens[2]) == uid) {
|
|
|
|
memcpy(username, tokens[0], strlen(tokens[0]) + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fclose(passwd);
|
|
|
|
}
|
2012-01-25 10:19:52 +04:00
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
/* function to update the cached hostname */
|
2012-01-30 22:10:53 +04:00
|
|
|
void gethostname() {
|
|
|
|
char buffer[256];
|
|
|
|
size_t len = syscall_gethostname(buffer);
|
|
|
|
memcpy(_hostname, buffer, len);
|
|
|
|
}
|
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
/* Draw the user prompt */
|
2012-01-26 00:57:27 +04:00
|
|
|
void draw_prompt(int ret) {
|
2012-10-07 10:01:39 +04:00
|
|
|
/* Get the time */
|
2012-01-25 10:19:52 +04:00
|
|
|
struct tm * timeinfo;
|
|
|
|
struct timeval now;
|
|
|
|
syscall_gettimeofday(&now, NULL); //time(NULL);
|
|
|
|
timeinfo = localtime((time_t *)&now.tv_sec);
|
2012-10-07 10:01:39 +04:00
|
|
|
|
|
|
|
/* Format the date and time for prompt display */
|
2012-01-25 10:19:52 +04:00
|
|
|
char date_buffer[80];
|
|
|
|
strftime(date_buffer, 80, "%m/%d", timeinfo);
|
|
|
|
char time_buffer[80];
|
|
|
|
strftime(time_buffer, 80, "%H:%M:%S", timeinfo);
|
2012-10-07 10:01:39 +04:00
|
|
|
|
|
|
|
/* Print the prompt. */
|
2012-01-26 00:57:27 +04:00
|
|
|
printf("\033[1m[\033[1;33m%s \033[1;32m%s \033[1;31m%s \033[1;34m%s\033[0m ",
|
2012-01-30 22:10:53 +04:00
|
|
|
username, _hostname, date_buffer, time_buffer);
|
2012-01-26 00:57:27 +04:00
|
|
|
if (ret != 0) {
|
|
|
|
printf("\033[1;31m%d ", ret);
|
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
/* Print the working directory in there, too */
|
2012-01-27 23:10:58 +04:00
|
|
|
syscall_getcwd(cwd, 1024);
|
2012-01-26 00:57:27 +04:00
|
|
|
printf("\033[0m%s\033[1m]\033[0m\n\033[1;32m$\033[0m ", cwd);
|
2012-03-17 02:09:00 +04:00
|
|
|
printf("\033]1;%s@%s:%s\007", username, _hostname, cwd);
|
2012-01-25 10:19:52 +04:00
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
|
2012-02-08 12:40:44 +04:00
|
|
|
uint32_t child = 0;
|
|
|
|
|
|
|
|
void sig_int(int sig) {
|
2012-10-07 10:01:39 +04:00
|
|
|
/* Interrupt handler */
|
2012-02-08 12:40:44 +04:00
|
|
|
if (child) {
|
|
|
|
syscall_send_signal(child, sig);
|
|
|
|
} else {
|
2012-10-07 10:01:39 +04:00
|
|
|
fprintf(stderr, "XXX: Clear the buffer and re-render the prompt\n");
|
2012-02-08 12:40:44 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
typedef struct {
|
|
|
|
char * buffer;
|
|
|
|
int collected;
|
|
|
|
int requested;
|
|
|
|
int newline;
|
|
|
|
int cancel;
|
|
|
|
int offset;
|
2012-10-07 10:48:46 +04:00
|
|
|
int tabbed;
|
2012-10-07 10:01:39 +04:00
|
|
|
} rline_context_t;
|
2012-01-27 08:46:18 +04:00
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
typedef void (*rline_callback_t)(rline_context_t * context);
|
2012-02-08 12:40:44 +04:00
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
typedef struct {
|
|
|
|
rline_callback_t tab_complete;
|
|
|
|
rline_callback_t redraw_prompt;
|
|
|
|
rline_callback_t special_key;
|
|
|
|
rline_callback_t key_up;
|
|
|
|
rline_callback_t key_down;
|
|
|
|
rline_callback_t key_left;
|
|
|
|
rline_callback_t key_right;
|
|
|
|
} rline_callbacks_t;
|
2012-01-28 04:04:39 +04:00
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
void rline_redraw(rline_context_t * context) {
|
|
|
|
printf("%s", context->buffer);
|
|
|
|
for (int i = context->offset; i < context->collected; ++i) {
|
|
|
|
printf("\033[D");
|
2012-01-27 08:46:18 +04:00
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
fflush(stdout);
|
|
|
|
}
|
2012-01-27 08:46:18 +04:00
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
size_t rline(char * buffer, size_t buf_size, rline_callbacks_t * callbacks) {
|
|
|
|
/* Initialize context */
|
|
|
|
rline_context_t context = {
|
|
|
|
buffer,
|
|
|
|
0,
|
|
|
|
buf_size,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
2012-10-07 10:48:46 +04:00
|
|
|
0,
|
2012-10-07 10:01:39 +04:00
|
|
|
};
|
2012-01-25 10:19:52 +04:00
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
/* Read keys */
|
|
|
|
while ((context.collected < context.requested) && (!context.newline)) {
|
|
|
|
uint32_t key_sym = kbd_key(fgetc(stdin));
|
|
|
|
if (key_sym == KEY_NONE) continue;
|
|
|
|
switch (key_sym) {
|
2012-10-07 10:48:46 +04:00
|
|
|
case KEY_CTRL_C:
|
|
|
|
printf("^C\n");
|
|
|
|
context.buffer[0] = '\0';
|
|
|
|
return 0;
|
2012-10-07 10:01:39 +04:00
|
|
|
case KEY_ARROW_UP:
|
|
|
|
if (callbacks->key_up) {
|
|
|
|
callbacks->key_up(&context);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case KEY_ARROW_DOWN:
|
|
|
|
if (callbacks->key_down) {
|
|
|
|
callbacks->key_down(&context);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case KEY_ARROW_RIGHT:
|
|
|
|
if (callbacks->key_right) {
|
|
|
|
callbacks->key_right(&context);
|
|
|
|
} else {
|
|
|
|
if (context.offset < context.collected) {
|
|
|
|
printf("\033[C");
|
|
|
|
fflush(stdout);
|
|
|
|
context.offset++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case KEY_ARROW_LEFT:
|
|
|
|
if (callbacks->key_left) {
|
|
|
|
callbacks->key_left(&context);
|
|
|
|
} else {
|
|
|
|
if (context.offset > 0) {
|
|
|
|
printf("\033[D");
|
|
|
|
fflush(stdout);
|
|
|
|
context.offset--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case KEY_BACKSPACE:
|
|
|
|
if (context.collected) {
|
|
|
|
if (!context.offset) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
printf("\010 \010");
|
|
|
|
if (context.offset != context.collected) {
|
|
|
|
int remaining = context.collected - context.offset;
|
|
|
|
for (int i = 0; i < remaining; ++i) {
|
|
|
|
printf("%c", context.buffer[context.offset + i]);
|
|
|
|
context.buffer[context.offset + i - 1] = context.buffer[context.offset + i];
|
|
|
|
}
|
|
|
|
printf(" ");
|
|
|
|
for (int i = 0; i < remaining + 1; ++i) {
|
|
|
|
printf("\033[D");
|
|
|
|
}
|
|
|
|
context.offset--;
|
|
|
|
context.collected--;
|
|
|
|
} else {
|
|
|
|
context.buffer[--context.collected] = '\0';
|
|
|
|
context.offset--;
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case 0x0C: /* ^L: Clear Screen, redraw prompt and buffer */
|
|
|
|
printf("\033[H\033[2J");
|
|
|
|
if (callbacks->redraw_prompt) {
|
|
|
|
callbacks->redraw_prompt(&context);
|
|
|
|
}
|
|
|
|
rline_redraw(&context);
|
|
|
|
continue;
|
|
|
|
case '\t':
|
|
|
|
if (callbacks->tab_complete) {
|
|
|
|
callbacks->tab_complete(&context);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
case '\n':
|
|
|
|
while (context.offset < context.collected) {
|
|
|
|
printf("\033[C");
|
|
|
|
context.offset++;
|
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
fflush(stdout);
|
|
|
|
context.newline = 1;
|
|
|
|
continue;
|
2011-12-07 05:46:35 +04:00
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
if (context.offset != context.collected) {
|
|
|
|
for (int i = context.collected; i > context.offset; --i) {
|
|
|
|
context.buffer[i] = context.buffer[i-1];
|
|
|
|
}
|
|
|
|
if (context.collected < context.requested) {
|
|
|
|
context.buffer[context.offset] = (char)key_sym;
|
|
|
|
context.buffer[++context.collected] = '\0';
|
|
|
|
context.offset++;
|
|
|
|
}
|
|
|
|
for (int i = context.offset - 1; i < context.collected; ++i) {
|
|
|
|
printf("%c", context.buffer[i]);
|
|
|
|
}
|
|
|
|
for (int i = context.offset; i < context.collected; ++i) {
|
|
|
|
printf("\033[D");
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
|
|
|
} else {
|
|
|
|
printf("%c", (char)key_sym);
|
|
|
|
if (context.collected < context.requested) {
|
|
|
|
context.buffer[context.collected] = (char)key_sym;
|
|
|
|
context.buffer[++context.collected] = '\0';
|
|
|
|
context.offset++;
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
2011-12-07 05:46:35 +04:00
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Cap that with a null */
|
|
|
|
context.buffer[context.collected] = '\0';
|
|
|
|
return context.collected;
|
|
|
|
}
|
|
|
|
|
|
|
|
void redraw_prompt_func(rline_context_t * context) {
|
|
|
|
draw_prompt(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
void tab_complete_func(rline_context_t * context) {
|
|
|
|
char buf[1024];
|
|
|
|
char * pch;
|
|
|
|
char * cmd;
|
|
|
|
char * save;
|
|
|
|
|
|
|
|
memcpy(buf, context->buffer, 1024);
|
|
|
|
|
|
|
|
pch = strtok_r(buf, " ", &save);
|
|
|
|
cmd = pch;
|
|
|
|
|
|
|
|
char * argv[1024];
|
|
|
|
int argc = 0;
|
|
|
|
|
|
|
|
if (!cmd) {
|
|
|
|
argv[0] = "";
|
|
|
|
argc = 1;
|
|
|
|
} else {
|
|
|
|
while (pch != NULL) {
|
2012-10-07 10:48:46 +04:00
|
|
|
argv[argc] = (char *)pch;
|
2012-10-07 10:01:39 +04:00
|
|
|
++argc;
|
|
|
|
pch = strtok_r(NULL, " ", &save);
|
2011-12-07 05:46:35 +04:00
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
argv[argc] = NULL;
|
|
|
|
|
|
|
|
if (argc < 2) {
|
|
|
|
if (context->buffer[strlen(context->buffer) - 1] == ' ' || argc == 0) {
|
2012-10-07 10:48:46 +04:00
|
|
|
if (!context->tabbed) {
|
|
|
|
context->tabbed = 1;
|
|
|
|
return;
|
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
fprintf(stderr, "\n");
|
|
|
|
for (int i = 0; i < shell_commands_len; ++i) {
|
|
|
|
fprintf(stderr, "%s", shell_commands[i]);
|
|
|
|
if (i < shell_commands_len - 1) {
|
|
|
|
fprintf(stderr, ", ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
redraw_prompt_func(context);
|
|
|
|
rline_redraw(context);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
int count = 0, j = 0;
|
|
|
|
char * match = NULL;
|
|
|
|
for (int i = 0; i < shell_commands_len; ++i) {
|
|
|
|
if (strstr(shell_commands[i], argv[0]) == shell_commands[i]) {
|
2012-10-07 10:48:46 +04:00
|
|
|
//fprintf(stderr, "%s matches %s\n", argv[0], shell_commands[i]);
|
2012-10-07 10:01:39 +04:00
|
|
|
count++;
|
|
|
|
match = shell_commands[i];
|
2012-01-27 23:10:58 +04:00
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
}
|
|
|
|
if (count == 1) {
|
|
|
|
for (int j = 0; j < strlen(context->buffer); ++j) {
|
|
|
|
printf("\010 \010");
|
|
|
|
}
|
|
|
|
printf(match);
|
|
|
|
fflush(stdout);
|
|
|
|
memcpy(context->buffer, match, strlen(match) + 1);
|
|
|
|
context->collected = strlen(context->buffer);
|
|
|
|
context->offset = context->collected;
|
|
|
|
return;
|
2012-01-27 23:10:58 +04:00
|
|
|
} else {
|
2012-10-07 10:48:46 +04:00
|
|
|
if (!context->tabbed) {
|
|
|
|
context->tabbed = 1;
|
|
|
|
return;
|
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
fprintf(stderr, "\n");
|
|
|
|
for (int i = 0; i < shell_commands_len; ++i) {
|
|
|
|
if (strstr(shell_commands[i], argv[0]) == shell_commands[i]) {
|
|
|
|
fprintf(stderr, "%s", shell_commands[i]);
|
|
|
|
++j;
|
|
|
|
if (j < count) {
|
|
|
|
fprintf(stderr, ", ");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
redraw_prompt_func(context);
|
|
|
|
rline_redraw(context);
|
|
|
|
return;
|
2012-01-27 23:10:58 +04:00
|
|
|
}
|
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
} else {
|
2012-10-07 10:48:46 +04:00
|
|
|
/* XXX Should complete to file names here */
|
2012-10-07 10:01:39 +04:00
|
|
|
fprintf(stderr, "%d\n", argc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void history_previous(rline_context_t * context) {
|
|
|
|
if (shell_scroll == 0) {
|
|
|
|
memcpy(shell_temp, context->buffer, strlen(context->buffer) + 1);
|
|
|
|
}
|
|
|
|
if (shell_scroll < shell_history_count) {
|
|
|
|
shell_scroll++;
|
|
|
|
for (size_t i = 0; i < strlen(context->buffer); ++i) {
|
|
|
|
printf("\010 \010");
|
2012-03-09 09:36:40 +04:00
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
char * h = shell_history_prev(shell_scroll);
|
|
|
|
memcpy(context->buffer, h, strlen(h) + 1);
|
|
|
|
printf("%s", h);
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
context->collected = strlen(context->buffer);
|
|
|
|
context->offset = context->collected;
|
|
|
|
}
|
2011-12-08 01:08:40 +04:00
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
void history_next(rline_context_t * context) {
|
|
|
|
if (shell_scroll > 1) {
|
|
|
|
shell_scroll--;
|
|
|
|
for (size_t i = 0; i < strlen(context->buffer); ++i) {
|
|
|
|
printf("\010 \010");
|
|
|
|
}
|
|
|
|
char * h = shell_history_prev(shell_scroll);
|
|
|
|
memcpy(context->buffer, h, strlen(h) + 1);
|
|
|
|
printf("%s", h);
|
|
|
|
fflush(stdout);
|
|
|
|
} else if (shell_scroll == 1) {
|
|
|
|
for (size_t i = 0; i < strlen(context->buffer); ++i) {
|
|
|
|
printf("\010 \010");
|
|
|
|
}
|
|
|
|
shell_scroll = 0;
|
|
|
|
memcpy(context->buffer, shell_temp, strlen(shell_temp) + 1);
|
|
|
|
printf("%s", context->buffer);
|
|
|
|
fflush(stdout);
|
|
|
|
}
|
|
|
|
context->collected = strlen(context->buffer);
|
|
|
|
context->offset = context->collected;
|
|
|
|
}
|
|
|
|
|
|
|
|
int shell_exec(char * buffer, size_t buffer_size) {
|
|
|
|
|
|
|
|
char * pch;
|
|
|
|
char * cmd;
|
|
|
|
char * save;
|
|
|
|
|
|
|
|
/* Read previous history entries */
|
|
|
|
if (buffer[0] == '!') {
|
|
|
|
uint32_t x = atoi((char *)((uintptr_t)buffer + 1));
|
|
|
|
if (x <= shell_history_count) {
|
|
|
|
buffer = shell_history_get(x - 1);
|
|
|
|
buffer_size = strlen(buffer);
|
|
|
|
} else {
|
|
|
|
fprintf(stderr, "esh: !%d: event not found\n", x);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pch = strtok_r(buffer," ", &save);
|
|
|
|
cmd = pch;
|
|
|
|
|
|
|
|
if (!cmd) {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
char * history = malloc(sizeof(char) * (buffer_size + 1));
|
|
|
|
memcpy(history, buffer, (buffer_size + 1));
|
|
|
|
shell_history_insert(history);
|
|
|
|
|
|
|
|
char * argv[1024];
|
|
|
|
int tokenid = 0;
|
|
|
|
|
|
|
|
while (pch) {
|
|
|
|
argv[tokenid] = (char *)pch;
|
|
|
|
++tokenid;
|
|
|
|
pch = strtok_r(NULL, " ", &save);
|
|
|
|
}
|
|
|
|
|
|
|
|
argv[tokenid] = NULL;
|
|
|
|
shell_command_t func = shell_find(argv[0]);
|
|
|
|
|
|
|
|
if (func) {
|
|
|
|
func(tokenid, argv);
|
|
|
|
} else {
|
|
|
|
FILE * file = NULL; //fopen(argv[0], "r");
|
|
|
|
if (!strstr(argv[0],"/")) {
|
|
|
|
cmd = malloc(sizeof(char) * (strlen(argv[0]) + strlen("/bin/") + 1));
|
|
|
|
sprintf(cmd, "%s%s", "/bin/", argv[0]);
|
2012-01-27 23:10:58 +04:00
|
|
|
file = fopen(cmd,"r");
|
|
|
|
if (!file) {
|
2012-10-07 10:01:39 +04:00
|
|
|
printf("Command not found: %s\n", argv[0]);
|
2011-12-08 01:08:40 +04:00
|
|
|
free(cmd);
|
2012-10-07 10:01:39 +04:00
|
|
|
return 1;
|
2011-12-08 01:08:40 +04:00
|
|
|
}
|
2012-01-27 23:10:58 +04:00
|
|
|
fclose(file);
|
2011-12-08 01:08:40 +04:00
|
|
|
} else {
|
2012-10-07 10:01:39 +04:00
|
|
|
file = fopen(argv[0], "r");
|
2012-01-27 23:10:58 +04:00
|
|
|
if (!file) {
|
2012-10-07 10:01:39 +04:00
|
|
|
printf("Command not found: %s\n", argv[0]);
|
2012-01-27 23:10:58 +04:00
|
|
|
free(cmd);
|
2012-10-07 10:01:39 +04:00
|
|
|
return 1;
|
2012-01-27 23:10:58 +04:00
|
|
|
}
|
2011-12-08 01:08:40 +04:00
|
|
|
fclose(file);
|
|
|
|
}
|
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
int nowait = (!strcmp(argv[tokenid-1],"&"));
|
|
|
|
if (nowait) {
|
|
|
|
argv[tokenid-1] = NULL;
|
|
|
|
}
|
|
|
|
|
2011-12-08 01:08:40 +04:00
|
|
|
|
2011-12-07 05:46:35 +04:00
|
|
|
uint32_t f = fork();
|
|
|
|
if (getpid() != pid) {
|
2012-10-07 10:01:39 +04:00
|
|
|
int i = execve(cmd, argv, NULL);
|
2011-12-07 06:36:40 +04:00
|
|
|
return i;
|
2011-12-07 05:46:35 +04:00
|
|
|
} else {
|
2012-10-07 10:01:39 +04:00
|
|
|
int ret_code = 0;
|
2011-12-08 01:08:40 +04:00
|
|
|
if (!nowait) {
|
2012-02-08 12:40:44 +04:00
|
|
|
child = f;
|
2012-10-07 10:01:39 +04:00
|
|
|
ret_code = syscall_wait(f);
|
2012-02-09 01:56:27 +04:00
|
|
|
child = 0;
|
2011-12-07 06:36:40 +04:00
|
|
|
}
|
2011-12-08 01:08:40 +04:00
|
|
|
free(cmd);
|
2012-10-07 10:01:39 +04:00
|
|
|
return ret_code;
|
2011-12-07 05:46:35 +04:00
|
|
|
}
|
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
}
|
|
|
|
|
2012-10-07 10:48:46 +04:00
|
|
|
void add_path_contents() {
|
|
|
|
DIR * dirp = opendir("/bin");
|
|
|
|
|
|
|
|
struct dirent * ent = readdir(dirp);
|
|
|
|
while (ent != NULL) {
|
|
|
|
if (ent->d_name[0] != '.') {
|
|
|
|
char * s = malloc(sizeof(char) * (strlen(ent->d_name) + 1));
|
|
|
|
memcpy(s, ent->d_name, strlen(ent->d_name) + 1);
|
|
|
|
shell_install_command(s, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
ent = readdir(dirp);
|
|
|
|
}
|
|
|
|
closedir(dirp);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
struct command {
|
|
|
|
char * string;
|
|
|
|
void * func;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int comp_shell_commands(const void *p1, const void *p2) {
|
|
|
|
return strcmp(((struct command *)p1)->string, ((struct command *)p2)->string);
|
|
|
|
}
|
|
|
|
|
|
|
|
void sort_commands() {
|
|
|
|
struct command commands[SHELL_COMMANDS];
|
|
|
|
for (int i = 0; i < shell_commands_len; ++i) {
|
|
|
|
commands[i].string = shell_commands[i];
|
|
|
|
commands[i].func = shell_pointers[i];
|
|
|
|
}
|
|
|
|
qsort(&commands, shell_commands_len, sizeof(struct command), comp_shell_commands);
|
|
|
|
for (int i = 0; i < shell_commands_len; ++i) {
|
|
|
|
shell_commands[i] = commands[i].string;
|
|
|
|
shell_pointers[i] = commands[i].func;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-07 10:01:39 +04:00
|
|
|
int main(int argc, char ** argv) {
|
|
|
|
int nowait = 0;
|
|
|
|
int free_cmd = 0;
|
|
|
|
int last_ret = 0;
|
|
|
|
|
|
|
|
pid = getpid();
|
|
|
|
|
|
|
|
syscall_signal(2, sig_int);
|
|
|
|
|
|
|
|
getusername();
|
|
|
|
gethostname();
|
|
|
|
|
|
|
|
FILE * motd = fopen("/etc/motd", "r");
|
|
|
|
if (motd) {
|
|
|
|
size_t s = 0;
|
|
|
|
fseek(motd, 0, SEEK_END);
|
|
|
|
s = ftell(motd);
|
|
|
|
fseek(motd, 0, SEEK_SET);
|
|
|
|
char * m = malloc(sizeof(char) * s);
|
|
|
|
fread(m, s, 1, motd);
|
|
|
|
fwrite(m, s, 1, stdout);
|
|
|
|
fprintf(stdout, "\n");
|
|
|
|
fflush(stdout);
|
|
|
|
free(m);
|
|
|
|
}
|
|
|
|
|
|
|
|
install_commands();
|
2012-10-07 10:48:46 +04:00
|
|
|
add_path_contents();
|
|
|
|
sort_commands();
|
2012-10-07 10:01:39 +04:00
|
|
|
while (1) {
|
|
|
|
draw_prompt(last_ret);
|
|
|
|
char buffer[LINE_LEN] = {0};
|
|
|
|
int buffer_size;
|
|
|
|
rline_callbacks_t callbacks = {
|
|
|
|
tab_complete_func, redraw_prompt_func, NULL,
|
|
|
|
history_previous, history_next,
|
|
|
|
NULL, NULL
|
|
|
|
};
|
|
|
|
set_unbuffered();
|
|
|
|
buffer_size = rline((char *)&buffer, LINE_LEN, &callbacks);
|
|
|
|
set_buffered();
|
|
|
|
|
|
|
|
last_ret = shell_exec(buffer, buffer_size);
|
|
|
|
shell_scroll = 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-12-07 05:46:35 +04:00
|
|
|
exit:
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2012-10-07 10:01:39 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
* cd [path]
|
|
|
|
*/
|
|
|
|
uint32_t shell_cmd_cd(int argc, char * argv[]) {
|
|
|
|
if (argc > 1) {
|
|
|
|
if (chdir(argv[1])) {
|
|
|
|
fprintf(stderr, "%s: could not cd '%s': no such file or directory\n", argv[0], argv[1]);
|
|
|
|
return 1;
|
|
|
|
} /* else success */
|
|
|
|
} else /* argc < 2 */ {
|
|
|
|
char home_path[512];
|
|
|
|
sprintf(home_path, "/home/%s", username);
|
|
|
|
if (chdir(home_path)) {
|
|
|
|
fprintf(stderr, "%s: could not cd %s': no such file or directory\n", argv[0], argv[1]);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* history
|
|
|
|
*/
|
|
|
|
uint32_t shell_cmd_history(int argc, char * argv[]) {
|
|
|
|
for (size_t i = 0; i < shell_history_count; ++i) {
|
|
|
|
printf("%d\t%s\n", i + 1, shell_history_get(i));
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void install_commands() {
|
|
|
|
shell_install_command("cd", shell_cmd_cd);
|
|
|
|
shell_install_command("history", shell_cmd_history);
|
|
|
|
}
|