57 #include <arpa/telnet.h> 58 #include <sys/ioctl.h> 59 #include <sys/types.h> 62 #include <netinet/tcp.h> 73 #define ANSI_CLEAR CSI "2J" CSI "1;1H" 75 #define ANSI_RESET CSI "0m" 77 #define ANSI_BOLD CSI "1m" 79 #define ANSI_DIM CSI "2m" 81 #define ANSI_DRED ANSI_DIM CSI "31m" 83 #define ANSI_BRED ANSI_BOLD CSI "31m" 85 #define ANSI_CLEARLINE CSI "2K" 87 #define ANSI_SCROLLDN CSI "1T" 89 #define ANSI_SAVECURSOR CSI "s" 91 #define ANSI_RESTCURSOR CSI "u" 95 #define UNIX_CLI_MAX_DEPTH_TELNET 32 98 #define UNIX_CLI_MAX_TERMINAL_WIDTH 512 100 #define UNIX_CLI_MAX_TERMINAL_HEIGHT 512 102 #define UNIX_CLI_DEFAULT_TERMINAL_HEIGHT 24 104 #define UNIX_CLI_DEFAULT_TERMINAL_WIDTH 80 113 #define _(a) { .line = (u8 *)(a), .length = sizeof(a) - 1 } 116 _(
" _______ _ _ _____ ___ \n"),
117 _(
" __/ __/ _ \\ (_)__ | | / / _ \\/ _ \\\n"),
118 _(
" _/ _// // / / / _ \\ | |/ / ___/ ___/\n"),
119 _(
" /_/ /____(_)_/\\___/ |___/_/ /_/ \n"),
126 _(
ANSI_BRED " __/ __/ _ \\ (_)__ " ANSI_RESET
" | | / / _ \\/ _ \\\n"),
127 _(
ANSI_BRED " _/ _// // / / / _ \\" ANSI_RESET
" | |/ / ___/ ___/\n"),
128 _(
ANSI_BRED " /_/ /____(_)_/\\___/" ANSI_RESET
" |___/_/ /_/ \n"),
342 #define CTL(c) (u8[]){ (c) - '@', 0 } 344 #define _(a,b) { .input = (u8 *)(a), .len = sizeof(a) - 1, .action = (b) } 549 u8 * input,
u32 ilen,
i32 * matched)
559 if (memcmp (input, a->
input, a->
len) == 0)
569 if (memcmp (input, a->
input, ilen) == 0)
588 u8 * buffer,
uword buffer_bytes)
639 for (i = 0; i <
len; i++, str++)
680 if (n < 0 && errno != EAGAIN)
696 else if ((
word) n < (
word) buffer_bytes)
716 u8 * buffer,
uword buffer_bytes)
720 while (end < buffer_bytes)
727 buffer_bytes - start) + start;
739 if (end < buffer_bytes)
749 if (buffer_bytes > 0)
834 prompt =
format (0,
"\r%s-- more -- (%d-%d/%d)%s",
849 char *message,
char *postfix)
853 prompt =
format (0,
"\r%s-- %s --%s%s",
871 sizeof (ANSI_CLEARLINE) - 1);
878 for (i = 0; i < cf->
width - 1; i++)
917 sizeof (ANSI_CLEAR) - 1);
926 for (i = 0; i < cf->
height - 1 &&
935 if (pi && line[pi->
length - 1] !=
'\n')
974 line_index = len_or_index;
998 if (j < l && p[j] ==
'\n')
1006 pi->
line = line_index;
1027 word i, old_line, old_offset;
1043 old_line = pi->
line;
1058 if (pi->
line == old_line &&
1127 if (row < cf->height - 1)
1136 if (line[pi->
length - 1] !=
'\n' && row == cf->
height - 2)
1190 if (strncasecmp(a, (char *)term, (size_t)len) == 0) return 1; \ 1195 _(
"xterm-256color");
1197 _(
"screen-256color");
1222 if (strncasecmp(a, (char *)term, (size_t)len) == 0) return 1; \ 1271 len =
ARRAY_LEN (unix_cli_banner_color);
1279 for (i = 0; i <
len; i++)
1282 banner[i].line, banner[i].length);
1300 uword event_type, *event_data = 0;
1333 clib_warning (
"BUG: unknown event type 0x%wx", event_type);
1416 switch (input_vector[1])
1442 if (input_vector[i - 1] == IAC && input_vector[i] == SE)
1445 switch (input_vector[2])
1448 if (input_vector[3] != 0)
1452 u8 *term = input_vector + 4;
1476 clib_net_to_host_u16 (*((
u16 *) (input_vector + 3)));
1483 clib_net_to_host_u16 (*((
u16 *) (input_vector + 5)));
1504 if (i == UNIX_CLI_MAX_DEPTH_TELNET)
1557 u8 **possible_commands;
1649 while (cf->
cursor > delta
1652 while (cf->
cursor > delta
1661 for (j = delta; j > 0; j--, cf->
cursor--)
1677 for (j = delta; j > 0; j--, cf->
cursor--)
1679 for (j = delta; j > 0; j--, cf->
cursor++)
1913 else if (input ==
'D' -
'@')
1935 sizeof (ANSI_CLEAR) - 1);
1972 if (
vec_len (possible_commands) == 1)
1974 u8 *completed = possible_commands[0];
2003 else if (
vec_len (possible_commands) >= 2)
2005 u8 **possible_command;
2006 uword max_command_len = 0, min_command_len = ~0;
2011 if (
vec_len (*possible_command) > max_command_len)
2012 max_command_len =
vec_len (*possible_command);
2013 if (
vec_len (*possible_command) < min_command_len)
2014 min_command_len =
vec_len (*possible_command);
2022 if (i + max_command_len >= cf->
width)
2029 for (j =
vec_len (*possible_command); j < max_command_len + 2;
2034 i += max_command_len + 2;
2054 for (; i < min_command_len; i++)
2063 common = (*possible_command)[
i];
2065 else if (common != (*possible_command)[i])
2133 if (pi && line[pi->
length - 1] !=
'\n')
2159 if (line[pi->
length - 1] !=
'\n')
2186 sizeof (ANSI_SCROLLDN) - 1);
2188 sizeof (ANSI_SAVECURSOR) - 1);
2191 sizeof (ANSI_CLEARLINE) - 1);
2194 sizeof (ANSI_RESTCURSOR) - 1);
2210 if (pi && line[pi->
length - 1] !=
'\n')
2235 if (pi && line[pi->
length - 1] !=
'\n')
2258 if (pi && line[pi->
length - 1] !=
'\n')
2286 if (pi && line[pi->
length - 1] !=
'\n')
2367 for (offset = 0; offset <=
vec_len (item) - limit; offset++)
2369 for (k = 0; k < limit; k++)
2374 goto found_at_offset;
2409 else if (isprint (input))
2436 for (; j > 0; j--, cf->
cursor--)
2581 lv =
format (lv,
"%U[%d]: %v",
2588 int rv __attribute__ ((unused)) = write (um->
log_fd, lv,
vec_len (lv));
2604 _vec_len (expanded) -= 1;
2735 for (i = 0; i <
vec_len (data); i++)
2741 for (i = 0; i <
vec_len (data); i++)
2747 _vec_len (data) = 0;
2781 if (n < 0 && errno != EAGAIN)
2812 int n, n_read, n_try;
2825 if (n < 0 && errno != EAGAIN)
2828 n_read = n < 0 ? 0 : n;
2877 file_desc =
format (0,
"%s", name);
2879 name = (
char *)
format (0,
"unix-cli-%s", name);
2897 if (this_vlib_main == 0)
2915 .process_log2_n_stack_bytes = 18,
2937 template.file_descriptor = fd;
2939 template.description = file_desc;
2978 (void) setsockopt (client.fd, IPPROTO_TCP, TCP_NODELAY,
2979 (
void *) &one,
sizeof (one));
2998 u8 charmode_option[] = {
2999 IAC, WONT, TELOPT_LINEMODE,
3000 IAC, DONT, TELOPT_LINEMODE,
3001 IAC, WILL, TELOPT_SGA,
3002 IAC, DO, TELOPT_SGA,
3003 IAC, WILL, TELOPT_ECHO,
3004 IAC, DONT, TELOPT_ECHO,
3005 IAC, DO, TELOPT_TTYPE,
3006 IAC, SB, TELOPT_TTYPE, 1, IAC, SE,
3007 IAC, DO, TELOPT_NAWS,
3008 IAC, SB, TELOPT_NAWS, 1, IAC, SE,
3074 if (ioctl (STDIN_FILENO, TIOCGWINSZ, &ws) < 0)
3082 cf->
width = ws.ws_col;
3113 struct sigaction sa;
3124 if ((flags = fcntl (STDIN_FILENO, F_GETFL, 0)) < 0)
3126 (void) fcntl (STDIN_FILENO, F_SETFL, flags | O_NONBLOCK);
3139 if (sigaction (SIGWINCH, &sa, 0) < 0)
3143 if (ioctl (STDIN_FILENO, TIOCGWINSZ, &ws) == 0)
3145 cf->
width = ws.ws_col;
3173 tcgetattr (STDIN_FILENO, &um->
tio_stdin);
3179 tio.c_lflag &= ~(ECHO | ICANON | IEXTEN);
3181 tio.c_iflag &= ~(IXON | IXOFF);
3183 tio.c_cc[VTIME] = 0;
3184 tio.c_cc[VSTOP] = _POSIX_VDISABLE;
3185 tio.c_cc[VSTART] = _POSIX_VDISABLE;
3186 tcsetattr (STDIN_FILENO, TCSAFLUSH, &tio);
3189 term = (
u8 *) getenv (
"TERM");
3192 int len = strlen ((
char *) term);
3210 if (s->config && s->config[0] != 0)
3216 if (strncmp (s->config,
"/run", 4) == 0)
3218 u8 *tmp =
format (0,
"%s", s->config);
3220 while (i && tmp[--i] !=
'/')
3238 template.file_descriptor = s->fd;
3239 template.description =
format (0,
"cli listener %s", s->config);
3265 if (isatty (STDIN_FILENO) && um->
tio_isset)
3266 tcsetattr (STDIN_FILENO, TCSAFLUSH, &um->
tio_stdin);
3281 char *fmt = (prompt[strlen (prompt) - 1] ==
' ') ?
"%s" :
"%s ";
3339 .short_help =
"Exit CLI",
3347 .short_help =
"Exit CLI",
3366 if (!
unformat (input,
"%s", &file_name))
3373 fd = open (file_name, O_RDONLY);
3384 if (fstat (fd, &s) < 0)
3390 if (!(S_ISREG (s.st_mode) || S_ISLNK (s.st_mode)))
3436 .short_help =
"exec <filename>",
3449 int i, n_errors_to_show;
3452 n_errors_to_show = 1 << 30;
3456 if (!
unformat (input,
"%d", &n_errors_to_show))
3460 "expecting integer number of errors to show, got `%U'",
3473 while (n_errors_to_show > 0)
3481 n_errors_to_show -= 1;
3488 if (
vec_len (unix_errors) == 0)
3493 for (i =
vec_len (unix_errors) - 1; i >= 0; i--)
3511 .path =
"show unix errors",
3512 .short_help =
"Show Unix system call error history",
3525 char path[PATH_MAX];
3528 vlib_cli_output (vm,
"%3s %6s %12s %12s %12s %-32s %s",
"FD",
"Thread",
3529 "Read",
"Write",
"Error",
"File Name",
"Description");
3535 s = format (s,
"/proc/self/fd/%d%c", f->file_descriptor, 0);
3536 rv = readlink((char *) s, path, PATH_MAX - 1);
3538 path[rv < 0 ? 0 : rv] = 0;
3540 vlib_cli_output (vm,
"%3d %6d %12d %12d %12d %-32s %v",
3541 f->file_descriptor, f->polling_thread_index,
3542 f->read_events, f->write_events, f->error_events,
3543 path, f->description);
3544 vec_reset_length (s);
3554 .path =
"show unix files",
3555 .short_help =
"Show Unix files in use",
3592 .short_help =
"Show current session command history",
3614 "line-by-line" :
"char-by-char");
3624 " (disabled by history limit)");
3630 || cf->
height ?
"" :
" (disabled by terminal height)",
3633 " (disabled by buffer limit)");
3663 .path =
"show terminal",
3664 .short_help =
"Show current session terminal settings",
3681 vlib_cli_output (vm,
"%-5s %-5s %-20s %s",
"PNI",
"FD",
"Name",
"Flags");
3683 #define fl(x, y) ( (x) ? toupper((y)) : tolower((y)) ) 3686 uf = pool_elt_at_index (fm->file_pool, cf->clib_file_index);
3687 n = vlib_get_node (vm, cf->process_node_index);
3688 vlib_cli_output (vm,
3689 "%-5d %-5d %-20v %c%c%c%c%c\n",
3690 cf->process_node_index,
3691 uf->file_descriptor,
3693 fl (cf->is_interactive,
'i'),
3694 fl (cf->is_socket,
's'),
3695 fl (cf->line_mode,
'l'),
3696 fl (cf->has_epipe,
'p'),
3697 fl (cf->ansi_capable,
'a'));
3743 .path =
"show cli-sessions",
3744 .short_help =
"Show current CLI sessions",
3771 else if (
unformat (line_input,
"off"))
3775 "Pager limit set to %u lines; note, this is global.\n",
3800 .path =
"set terminal pager",
3801 .short_help =
"set terminal pager [on|off] [limit <lines>]",
3828 else if (
unformat (line_input,
"off"))
3850 for (i = 0; i < limit; i++)
3871 .path =
"set terminal history",
3872 .short_help =
"set terminal history [on|off] [limit <lines>]",
3910 .path =
"set terminal ansi",
3911 .short_help =
"set terminal ansi [on|off]",
3917 #define MAX_CLI_WAIT 86400 3931 if (
unformat (line_input,
"%f", &sec))
3938 if (sec <= 0 || sec >
MAX_CLI_WAIT || floor (sec * 1000) / 1000 != sec)
3940 "<sec> must be a positive value and less than 86400 (one day) with no more than msec precision.");
3951 .short_help =
"wait <sec>",
3978 .short_help =
"echo <rest-of-line>",
3992 if (!
unformat (input,
"%s", ¯o_name))
4016 .short_help =
"define <variable-name> <value>",
4029 if (!
unformat (input,
"%s", ¯o_name))
4042 .short_help =
"undefine <variable-name>",
4054 if (
unformat (input,
"noevaluate %=", &evaluate, 0))
4056 else if (
unformat (input,
"noeval %=", &evaluate, 0))
4065 .path =
"show macro",
4066 .short_help =
"show macro [noevaluate]",
#define vec_validate(V, I)
Make sure vector is long enough for given index (no header, unspecified alignment) ...
u32 history_limit
Maximum number of history entries this session will store.
u32 command_number
Current command line counter.
uword output_function_arg
static clib_error_t * unix_cli_init(vlib_main_t *vm)
#define vec_foreach_index(var, v)
Iterate over vector indices.
u8 * format_clib_error(u8 *s, va_list *va)
static void clib_file_del(clib_file_main_t *um, clib_file_t *f)
u8 is_interactive
Whether the session is interactive or not.
u32 pager_start
Line number of top of page.
Action parser found a partial match.
void vlib_unix_cli_set_prompt(char *prompt)
Set the CLI prompt.
#define ANSI_RESTCURSOR
ANSI restore cursor position if previously saved.
static f64 vlib_process_wait_for_event_or_clock(vlib_main_t *vm, f64 dt)
Suspend a cooperative multi-tasking thread Waits for an event, or for the indicated number of seconds...
vl_api_wireguard_peer_flags_t flags
u8 * input_vector
Vector of input saved by Unix input node to be processed by CLI process.
static uword * vlib_process_wait_for_event(vlib_main_t *vm)
static uword vlib_current_process(vlib_main_t *vm)
#define ANSI_SCROLLDN
ANSI scroll screen down one line.
u8 * line
The line to print.
Home key (jump to start of line)
static void unix_vlib_cli_output(uword cli_file_index, u8 *buffer, uword buffer_bytes)
VLIB CLI output function.
unix_cli_timeout_event_type_t
CLI session telnet negotiation timer events.
static void unix_cli_set_session_noninteractive(unix_cli_file_t *cf)
Set a session to be non-interactive.
u8 * cli_prompt
Prompt string for CLI.
u32 cli_pager_buffer_limit
u8 ** command_history
Array of vectors of commands in the history.
clib_memset(h->entries, 0, sizeof(h->entries[0]) *entries)
static f64 vlib_time_now(vlib_main_t *vm)
unix_cli_new_session_t * new_sessions
List of new sessions.
static unix_cli_file_t * unix_cli_file_if_interactive(unix_cli_main_t *cm)
#define vec_add1(V, E)
Add 1 element to end of vector (unspecified alignment).
unix_cli_pager_index_t * pager_index
Index of line fragments in the pager buffer.
u8 * current_command
The command currently pointed at by the history cursor.
clib_error_t * clib_socket_init(clib_socket_t *s)
#define VLIB_MAIN_LOOP_EXIT_CLI
#define vlib_call_config_function(vm, x)
void clib_longjmp(clib_longjmp_t *save, uword return_value)
u32 cursor
Position of the insert cursor on the current input line.
clib_socket_t cli_listen_socket
int vlib_cli_input(vlib_main_t *vm, unformat_input_t *input, vlib_cli_output_function_t *function, uword function_arg)
#define pool_get(P, E)
Allocate an object E from a pool P (unspecified alignment).
static void unix_cli_pager_reset(unix_cli_file_t *f)
Resets the pager buffer and other data.
vlib_main_t ** vlib_mains
unix_cli_parse_action_t action
Action to take when matched.
static uword vlib_process_suspend_time_is_zero(f64 dt)
Returns TRUE if a process suspend time is less than 10us.
#define vec_reset_length(v)
Reset vector length to zero NULL-pointer tolerant.
#define vlib_worker_thread_barrier_sync(X)
static void unix_cli_ansi_cursor(unix_cli_file_t *cf, clib_file_t *uf, u16 x, u16 y)
Uses an ANSI escape sequence to move the cursor.
#define clib_memcpy(d, s, n)
#define vec_add(V, E, N)
Add N elements to end of vector V (no header, unspecified alignment)
#define CLIB_SOCKET_F_IS_SERVER
static void unix_cli_resize_interrupt(int signum)
The system terminal has informed us that the window size has changed.
u8 has_epipe
If EPIPE has been detected, prevent further write-related activity on the descriptor.
Jump cursor to start of right word.
u8 ansi_capable
Can we do ANSI output?
static clib_error_t * echo_cmd(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
clib_macro_main_t macro_main
Macro tables for this session.
static int unix_cli_line_process_one(unix_cli_main_t *cm, unix_main_t *um, unix_cli_file_t *cf, clib_file_t *uf, u8 input, unix_cli_parse_action_t action)
Process actionable input.
static clib_error_t * unix_cli_read_ready(clib_file_t *uf)
Called when a CLI session file descriptor has data to be read.
Jump cursor to start of left word.
#define pool_foreach(VAR, POOL, BODY)
Iterate through pool.
void vlib_worker_thread_node_runtime_update(void)
u8 * output_vector
Vector of output pending write to file descriptor.
format_function_t format_clib_macro_main
#define VLIB_INIT_FUNCTION(x)
static uword vlib_process_get_events(vlib_main_t *vm, uword **data_vector)
Return the first event type which has occurred and a vector of per-event data of that type...
#define CTL(c)
Given a capital ASCII letter character return a NUL terminated string with the control code for that ...
#define vec_new(T, N)
Create new vector of given type and length (unspecified alignment, no header).
u8 is_socket
Whether the session is attached to a socket.
static void unix_cli_pager_redraw(unix_cli_file_t *cf, clib_file_t *uf)
Redraw the currently displayed page of text.
u32 len
Length of input without final NUL.
Mapping of input buffer strings to action values.
vlib_parse_match_t vlib_parse_eval(u8 *input)
#define vec_elt_at_index(v, i)
Get vector value at index i checking that i is in bounds.
#define clib_error_return(e, args...)
Enter pressed (CR, CRLF, LF, etc)
clib_file_main_t file_main
u32 process_node_index
Process node identifier.
#define vec_resize(V, N)
Resize a vector (no header, unspecified alignment) Add N elements to end of given vector V...
Add a CLI session to the new session list.
static clib_error_t * unix_cli_quit(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to quit the terminal session.
u32 * unused_cli_process_node_indices
Vec pool of unused session indices.
Carriage return, newline or enter.
int clib_macro_set_value(clib_macro_main_t *mm, char *name, char *value)
static void unix_cli_process_input(unix_cli_main_t *cm, uword cli_file_index)
Process input to a CLI session.
unix_cli_file_t * cli_file_pool
Vec pool of CLI sessions.
static int unix_cli_line_edit(unix_cli_main_t *cm, unix_main_t *um, clib_file_main_t *fm, unix_cli_file_t *cf)
Process input bytes on a stream to provide line editing and command history in the CLI...
#define CSI
ANSI Control Sequence Introducer.
int search_mode
If non-zero then the CLI is searching in the history array.
#define pool_elt_at_index(p, i)
Returns pointer to element at given index.
static void unix_cli_cli_prompt(unix_cli_file_t *cf, clib_file_t *uf)
Output the CLI prompt.
static u8 unix_cli_terminal_type_noninteractive(u8 *term, uword len)
Identify whether a terminal type is non-interactive.
static void vlib_process_signal_event(vlib_main_t *vm, uword node_index, uword type_opaque, uword data)
clib_error_t * clib_socket_accept(clib_socket_t *server, clib_socket_t *client)
u32 vlib_process_create(vlib_main_t *vm, char *name, vlib_node_function_t *f, u32 log2_n_stack_bytes)
Create a vlib process.
static clib_macro_main_t * get_macro_main(void)
Return the macro main / tables we should use for this session.
#define clib_error_return_unix(e, args...)
#define ANSI_CLEAR
ANSI clear screen.
u32 vlib_register_node(vlib_main_t *vm, vlib_node_registration_t *r)
static clib_error_t * unix_cli_set_terminal_pager(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to set terminal pager settings.
#define pool_put(P, E)
Free an object E in pool P.
static clib_error_t * unix_cli_show_terminal(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to show terminal status.
#define VLIB_CONFIG_FUNCTION(x, n,...)
u8 * input
Input string to match.
void(* file_update)(clib_file_t *file, clib_file_update_type_t update_type)
u32 node_index
Node index.
static clib_error_t * unix_cli_show_cli_sessions(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to display a list of CLI sessions.
static void unix_cli_file_welcome(unix_cli_main_t *cm, unix_cli_file_t *cf)
Emit initial welcome banner and prompt on a connection.
static unix_cli_main_t unix_cli_main
CLI global state.
u8 ** pager_vector
Pager buffer.
#define UNIX_CLI_MAX_TERMINAL_HEIGHT
Maximum terminal height we will accept.
static void unix_vlib_cli_output_cooked(unix_cli_file_t *cf, clib_file_t *uf, u8 *buffer, uword buffer_bytes)
Process a buffer for CRLF handling before outputting it to the CLI.
Search forwards in command history.
unix_error_history_t error_history[128]
#define UNIX_CLI_MAX_TERMINAL_WIDTH
Maximum terminal width we will accept.
static vlib_process_t * vlib_get_current_process(vlib_main_t *vm)
static void unix_vlib_cli_output_cursor_left(unix_cli_file_t *cf, clib_file_t *uf)
Moves the terminal cursor one character to the left, with special handling when it reaches the left e...
sll srl srl sll sra u16x4 i
u32 current_input_file_index
File pool index of current input.
static clib_error_t * unix_wait_cmd(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to wait <sec> seconds.
static unix_cli_parse_action_t unix_cli_match_action(unix_cli_parse_actions_t *a, u8 *input, u32 ilen, i32 *matched)
Search for a byte sequence in the action list.
#define vec_free(V)
Free vector's memory (no header).
void clib_macro_free(clib_macro_main_t *mm)
static unix_cli_parse_actions_t unix_cli_parse_strings[]
Patterns to match on a CLI input stream.
static u32 unix_cli_file_add(unix_cli_main_t *cm, char *name, int fd)
Store a new CLI session.
static void unix_cli_del_pending_output(clib_file_t *uf, unix_cli_file_t *cf, uword n_bytes)
Delete all bytes from the output vector and flag the I/O system that no more bytes are available to b...
#define VLIB_MAIN_LOOP_EXIT_FUNCTION(x)
#define ESC
ANSI escape code.
#define ANSI_CLEARLINE
ANSI clear line cursor is on.
#define clib_warning(format, args...)
static void unix_cli_kill(unix_cli_main_t *cm, uword cli_file_index)
Destroy a CLI session.
#define pool_is_free_index(P, I)
Use free bitmap to query whether given index is free.
Action parser did not find any match.
#define ANSI_SAVECURSOR
ANSI save cursor position.
u8 no_pager
Disable the pager?
#define ANSI_RESET
ANSI reset color settings.
#define VLIB_CLI_COMMAND(x,...)
static clib_error_t * unix_cli_config(vlib_main_t *vm, unformat_input_t *input)
Handle configuration directives in the unix section.
static unix_cli_banner_t unix_cli_banner[]
Plain welcome banner.
void vlib_cli_output(vlib_main_t *vm, char *fmt,...)
#define ANSI_BOLD
ANSI Start bold text.
static clib_error_t * unix_cli_listen_read_ready(clib_file_t *uf)
Telnet listening socket has a new connection.
#define vec_delete(V, N, M)
Delete N elements starting at element M.
static uword clib_file_add(clib_file_main_t *um, clib_file_t *template)
#define UNIX_FLAG_INTERACTIVE
static void unix_cli_add_pending_output(clib_file_t *uf, unix_cli_file_t *cf, u8 *buffer, uword buffer_bytes)
Add bytes to the output vector and then flagg the I/O system that bytes are available to be sent...
u32 new_session_process_node_index
New session process node identifier.
static void unix_cli_pager_add_line(unix_cli_file_t *cf, u8 *line, word len_or_index)
Process and add a line to the pager index.
#define UNIX_CLI_DEFAULT_TERMINAL_HEIGHT
Default terminal height.
uword cf_index
Session index of the new session.
#define vec_append(v1, v2)
Append v2 after v1.
struct _socket_t clib_socket_t
#define UNIX_FILE_DATA_AVAILABLE_TO_WRITE
static void vlib_node_set_state(vlib_main_t *vm, u32 node_index, vlib_node_state_t new_state)
Set node dispatch state.
u8 has_history
This session has command history.
int clib_macro_unset(clib_macro_main_t *mm, char *name)
A file descriptor has data to be read.
static void clib_socket_free(clib_socket_t *s)
static vlib_main_t * vlib_get_main(void)
A CLI session wants to close.
static clib_error_t * unix_cli_error_detected(clib_file_t *uf)
Called when a CLI session file descriptor has an error condition.
static void unix_cli_pager_reindex(unix_cli_file_t *cf)
Reindex entire pager buffer.
u8 started
Has the session started?
struct _vlib_node_registration vlib_node_registration_t
clib_error_t * vlib_unix_recursive_mkdir(char *path)
static void unix_vlib_cli_output_raw(unix_cli_file_t *cf, clib_file_t *uf, u8 *buffer, uword buffer_bytes)
Send a buffer to the CLI stream if possible, enqueue it otherwise.
static unix_cli_banner_t unix_cli_banner_color[]
ANSI color welcome banner.
static unix_cli_parse_actions_t unix_cli_parse_pager[]
Patterns to match when a CLI session is in the pager.
static void unix_cli_pager_prompt_erase(unix_cli_file_t *cf, clib_file_t *uf)
Erase the printed pager prompt.
vl_api_mac_event_action_t action
#define vec_len(v)
Number of elements in vector (rvalue-only, NULL tolerant)
static unix_cli_file_t * unix_cli_file_if_exists(unix_cli_main_t *cm)
static clib_error_t * unix_show_files(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to show various unix error statistics.
End key (jump to end of line)
static void unix_cli_pager_prompt(unix_cli_file_t *cf, clib_file_t *uf)
Output a pager prompt and show number of buffered lines.
clib_macro_main_t macro_main
system default macro table
u8 * search_key
The string being searched for in the history.
Search backwards in command history.
#define clib_unix_warning(format, args...)
static clib_error_t * unix_cli_show_history(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to show session command history.
static u8 unix_cli_terminal_type_ansi(u8 *term, uword len)
Identify whether a terminal type is ANSI capable.
void clib_macro_init(clib_macro_main_t *mm)
static clib_error_t * unix_config(vlib_main_t *vm, unformat_input_t *input)
Clear and redraw the page on the terminal.
struct clib_bihash_value offset
template key/value backing page structure
static vlib_process_t * vlib_get_process_from_node(vlib_main_t *vm, vlib_node_t *node)
static clib_error_t * unix_cli_set_terminal_ansi(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to set terminal ANSI settings.
void vlib_worker_thread_barrier_release(vlib_main_t *vm)
static clib_error_t * undefine_cmd_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
static void unix_cli_pager_message(unix_cli_file_t *cf, clib_file_t *uf, char *message, char *postfix)
Output a pager "skipping" message.
i8 * clib_macro_eval(clib_macro_main_t *mm, i8 *s, i32 complain, u16 level, u16 max_level)
static clib_error_t * unix_cli_exec(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to execute a VPP command script.
static vlib_node_t * vlib_get_node(vlib_main_t *vm, u32 i)
Get vlib node by index.
Erase line to left of cursor.
static clib_error_t * unix_cli_write_ready(clib_file_t *uf)
Called when a CLI session file descriptor can be written to without blocking.
static void unix_cli_file_free(unix_cli_file_t *f)
Release storage used by a CLI session.
#define vec_foreach(var, vec)
Vector iterator.
f64 end
end of the time range
u8 cursor_direction
The current direction of cursor travel.
u8 crlf_mode
Set if the CRLF mode wants CR + LF.
clib_longjmp_t main_loop_exit
vlib_cli_output_function_t * output_function
static uword unix_cli_process(vlib_main_t *vm, vlib_node_runtime_t *rt, vlib_frame_t *f)
Handle system events.
u32 clib_file_index
The file index held by unix.c.
#define CLIB_SOCKET_F_ALLOW_GROUP_WRITE
i32 excursion
How far from the end of the history array the user has browsed.
#define UNIX_CLI_DEFAULT_TERMINAL_WIDTH
Default terminal width.
void vlib_start_process(vlib_main_t *vm, uword process_index)
static uword unix_cli_new_session_process(vlib_main_t *vm, vlib_node_runtime_t *rt, vlib_frame_t *f)
A failsafe manager that ensures CLI sessions issue an initial prompt if TELNET negotiation fails...
u8 line_mode
Line mode or char mode.
u32 length
The length of the line without terminating NUL.
static clib_error_t * define_cmd_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
#define ANSI_BRED
ANSI Start bright red text.
f64 deadline
Deadline after which the new session must have a prompt.
u32 stdin_cli_file_index
The session index of the stdin cli.
#define clib_panic(format, args...)
static clib_error_t * show_macro_cmd_fn(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
u8 ** vlib_cli_get_possible_completions(u8 *str)
static clib_error_t * unix_cli_exit(vlib_main_t *vm)
Called when VPP is shutting down, this restores the system terminal state if previously saved...
static clib_error_t * unix_cli_set_terminal_history(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to set terminal history settings.
Each new session is stored on a list with a deadline after which a prompt is issued, in case the session TELNET negotiation fails to complete.
unix_cli_parse_action_t
CLI actions.
static clib_error_t * unix_show_errors(vlib_main_t *vm, unformat_input_t *input, vlib_cli_command_t *cmd)
CLI command to show various unix error statistics.
static word unix_vlib_findchr(u8 chr, u8 *str, word len)
A bit like strchr with a buffer length limit.
Erase line to right & including cursor.
u32 height
Terminal height.
static i32 unix_cli_process_telnet(unix_main_t *um, unix_cli_file_t *cf, clib_file_t *uf, u8 *input_vector, uword len)
A mostly no-op Telnet state machine.
#define UNIX_CLI_MAX_DEPTH_TELNET
Maximum depth into a byte stream from which to compile a Telnet protocol message. ...
unix_cli_process_event_type_t
CLI session events.