/* // Copyright (c) 2010-2017 Intel Corporation // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. */ #include <string.h> #include <errno.h> #include <stdlib.h> #include <fcntl.h> #include <unistd.h> #include "log.h" #include "input.h" #include "display.h" #include "run.h" #include "cmd_parser.h" #include "input_curses.h" #include "histedit.h" #include "libedit_autoconf.h" static EditLine *el; static History *hist; static struct input input_curses; static int tabbed; static void show_history(struct input *input) { HistEvent event; history(hist, &event, H_LAST); do { plog_info("%s", event.str); /* event.str contains newline */ } while (history(hist, &event, H_PREV) != -1); } static int complete(__attribute__((unused)) int ch) { const LineInfo *li; size_t len; size_t n_match = 0; char complete_cmd[128] = {0}; int complete_cmd_partial = 0; li = el_line(el); for (size_t i = 0; i < cmd_parser_n_cmd(); ++i) { len = li->lastchar - li->buffer; if (strncmp(cmd_parser_cmd(i), li->buffer, len) == 0) { if (n_match) { size_t cur_len = strlen(complete_cmd); for (size_t j = 0; j < cur_len; ++j) { if (complete_cmd[j] != cmd_parser_cmd(i)[j]) { complete_cmd[j] = 0; complete_cmd_partial = 1; break; } } } else { strcpy(complete_cmd, cmd_parser_cmd(i)); } n_match++; } } /* Complete only if there are more characters known than currently entered. */ if (n_match && len < strlen(complete_cmd)) { el_deletestr(el, li->cursor - li->buffer); el_insertstr(el, complete_cmd); if (!complete_cmd_partial) el_insertstr(el, " "); return CC_REDISPLAY; } else if (tabbed) { int printed = 0; for (size_t i = 0; i < cmd_parser_n_cmd(); ++i) { len = li->lastchar - li->buffer; if (strncmp(cmd_parser_cmd(i), li->buffer, len) == 0) { plog_info("%-23s", cmd_parser_cmd(i)); printed++; } if (printed == 4) { printed = 0; plog_info("\n"); } } if (printed) plog_info("\n"); } else { tabbed = 1; } return CC_REDISPLAY; } /* Returns non-zero if stdin is readable */ static int peek_stdin(void) { int tmp; fd_set in_fd; struct timeval tv; tv.tv_sec = 0; tv.tv_usec = 10000; FD_ZERO(&in_fd); FD_SET(fileno(stdin), &in_fd); tmp = select(fileno(stdin) + 1, &in_fd, NULL, NULL, &tv); return FD_ISSET(fileno(stdin), &in_fd); } #ifdef HAVE_LIBEDIT_EL_RFUNC_T static int do_get_char(EditLine *e, wchar_t *c) #else static int get_char(EditLine *e, char *c) #endif { *c = display_getch(); /* If no characters have been entered, number keys switch the screen and '0' resets stats. This is provided as a fall-back in case F-keys do not function. The keys are intercepted before returning control to libedit. */ if (*c >= '0' && *c <= '9') { const LineInfo *li; li = el_line(e); if (li->lastchar == li->buffer) { if (*c >= '1') { display_screen(*c - '0' - 1); return 0; } else { cmd_parser_parse("reset stats", &input_curses); return 0; } } } if (*c == '=') { toggle_display_screen(); return 0; } /* Escape by itself is the first character used for more complex escape sequences like F-keys. libedit can't be used to detect both ESC as a unitary key and more complex sequences starting ESC at the same time. */ if (*c == 27 && !peek_stdin()) { quit(); return 0; } else if (*c != 9) { tabbed = 0; } return 1; } #ifdef HAVE_LIBEDIT_EL_RFUNC_T static el_rfunc_t get_char = &do_get_char; #endif static void proc_keyboard(struct input *input) { const char *line; const LineInfo *li; HistEvent hist_event; int len; line = el_gets(el, &len); li = el_line(el); if (len == 0 || line == NULL) { display_cmd("", 0, 0); return; } else if (len > 0) { if (len == 1 && line[0] == '\n') { display_print_page(); el_set(el, EL_UNBUFFERED, 0); el_set(el, EL_UNBUFFERED, 1); return; } if (line[len-1] == '\n') { if (hist) { history(hist, &hist_event, H_ENTER, line); } char *line2 = strndup(line, len); line2[len - 1] = 0; /* replace \n */ cmd_parser_parse(line2, input); free(line2); el_set(el, EL_UNBUFFERED, 0); el_set(el, EL_UNBUFFERED, 1); display_cmd("", 0, 0); return; } if (line[len-1] == 4) { return; /* should quit*/ } } else { if (errno) { return; } display_cmd("", 0, 0); return; } display_cmd(line, len, li->cursor - li->buffer); } static int key_f1(__attribute__((unused)) int ch) {display_screen(0); return CC_REDISPLAY;} static int key_f2(__attribute__((unused)) int ch) {display_screen(1); return CC_REDISPLAY;} static int key_f3(__attribute__((unused)) int ch) {display_screen(2); return CC_REDISPLAY;} static int key_f4(__attribute__((unused)) int ch) {display_screen(3); return CC_REDISPLAY;} static int key_f5(__attribute__((unused)) int ch) {display_screen(4); return CC_REDISPLAY;} static int key_f6(__attribute__((unused)) int ch) {display_screen(5); return CC_REDISPLAY;} static int key_f7(__attribute__((unused)) int ch) {display_screen(6); return CC_REDISPLAY;} static int key_f8(__attribute__((unused)) int ch) {display_screen(7); return CC_REDISPLAY;} static int key_f9(__attribute__((unused)) int ch) {display_screen(8); return CC_REDISPLAY;} static int key_f10(__attribute__((unused)) int ch) {display_screen(9); return CC_REDISPLAY;} static int key_f11(__attribute__((unused)) int ch) {display_screen(10); return CC_REDISPLAY;} static int key_f12(__attribute__((unused)) int ch) {display_screen(11); return CC_REDISPLAY;} static int key_page_up(__attribute__((unused)) int ch) {display_page_up(); return CC_REDISPLAY;} static int key_page_down(__attribute__((unused)) int ch) {display_page_down(); return CC_REDISPLAY;} static void setup_el(void) { int pty; FILE *dev_pty; HistEvent hist_event; /* Open a pseudo-terminal for use in libedit. This is required since the library checks if it is using a tty. If the file descriptor does not represent a tty, the library disables editing. */ pty = posix_openpt(O_RDWR); /* TODO: On error (posix_openpt() < 0), fall-back to non-libedit implementation. */ grantpt(pty); unlockpt(pty); dev_pty = fdopen(pty, "wr"); el = el_init("", dev_pty, dev_pty, dev_pty); el_set(el, EL_EDITOR, "emacs"); el_set(el, EL_ADDFN, "complete", "Command completion", complete); el_set(el, EL_ADDFN, "key_f1", "Switch to screen 1", key_f1); el_set(el, EL_ADDFN, "key_f2", "Switch to screen 2", key_f2); el_set(el, EL_ADDFN, "key_f3", "Switch to screen 3", key_f3); el_set(el, EL_ADDFN, "key_f4", "Switch to screen 4", key_f4); el_set(el, EL_ADDFN, "key_f5", "Switch to screen 5", key_f5); el_set(el, EL_ADDFN, "key_f6", "Switch to screen 6", key_f6); el_set(el, EL_ADDFN, "key_f7", "Switch to screen 7", key_f7); el_set(el, EL_ADDFN, "key_f8", "Switch to screen 8", key_f8); el_set(el, EL_ADDFN, "key_f9", "Switch to screen 9", key_f5); el_set(el, EL_ADDFN, "key_f10", "Switch to screen 10", key_f6); el_set(el, EL_ADDFN, "key_f11", "Switch to screen 11", key_f7); el_set(el, EL_ADDFN, "key_f12", "Switch to screen 12", key_f8); el_set(el, EL_ADDFN, "key_page_up", "Page up", key_page_up); el_set(el, EL_ADDFN, "key_page_down", "Page down", key_page_down); el_set(el, EL_BIND, "^I", "complete", NULL); el_set(el, EL_BIND, "^r", "em-inc-search-prev", NULL); el_set(el, EL_BIND, "^[[11~", "key_f1", NULL); el_set(el, EL_BIND, "^[[12~", "key_f2", NULL); el_set(el, EL_BIND, "^[[13~", "key_f3", NULL); el_set(el, EL_BIND, "^[[14~", "key_f4", NULL); el_set(el, EL_BIND, "^[[15~", "key_f5", NULL); el_set(el, EL_BIND, "^[[17~", "key_f6", NULL); el_set(el, EL_BIND, "^[[18~", "key_f7", NULL); el_set(el, EL_BIND, "^[[19~", "key_f8", NULL); el_set(el, EL_BIND, "^[[20~", "key_f9", NULL); el_set(el, EL_BIND, "^[[21~", "key_f10", NULL); el_set(el, EL_BIND, "^[[23~", "key_f11", NULL); el_set(el, EL_BIND, "^[[24~", "key_f12", NULL); el_set(el, EL_BIND, "^[OP", "key_f1", NULL); el_set(el, EL_BIND, "^[OQ", "key_f2", NULL); el_set(el, EL_BIND, "^[OR", "key_f3", NULL); el_set(el, EL_BIND, "^[OS", "key_f4", NULL); el_set(el, EL_BIND, "^[[5~", "key_page_up", NULL); el_set(el, EL_BIND, "^[[6~", "key_page_down", NULL); hist = history_init(); if (hist) { history(hist, &hist_event, H_SETSIZE, 1000); el_set(el, EL_HIST, history, hist); } el_set(el, EL_UNBUFFERED, 1); el_set(el, EL_GETCFN, get_char); } void reg_input_curses(void) { setup_el(); input_curses.fd = fileno(stdin); input_curses.proc_input = proc_keyboard; input_curses.history = show_history; reg_input(&input_curses); } void unreg_input_curses(void) { history_end(hist); el_end(el); unreg_input(&input_curses); }