2023-11-22 11:58:43 +00:00
/* linenoise.c -- guerrilla line editing library against the idea that a
* line editing lib needs to be 20 , 000 lines of C code .
*
* You can find the latest source code at :
*
* http : //github.com/antirez/linenoise
*
* Does a number of crazy assumptions that happen to be true in 99.9999 % of
* the 2010 UNIX computers around .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
*
2023-11-22 21:44:22 +00:00
* Copyright ( c ) 2023 - 2023 , Danila Gornushko < d3g3v3 at gmail dot com >
* Copyright ( c ) 2010 - 2023 , Salvatore Sanfilippo < antirez at gmail dot com >
2023-11-22 11:58:43 +00:00
* Copyright ( c ) 2010 - 2013 , Pieter Noordhuis < pcnoordhuis at gmail dot com >
*
* All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions are
* met :
*
* * Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer .
*
* * Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* " AS IS " AND ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT
* LIMITED TO , THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED . IN NO EVENT SHALL THE COPYRIGHT
* HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT , INDIRECT , INCIDENTAL ,
* SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT NOT
* LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
*
* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
*
* References :
* - http : //invisible-island.net/xterm/ctlseqs/ctlseqs.html
* - http : //www.3waylabs.com/nw/WWW/products/wizcon/vt220.html
*
* Todo list :
* - Filter bogus Ctrl + < char > combinations .
* - Win32 support
*
* Bloat :
* - History search like Ctrl + r in readline ?
*
* List of escape sequences used by this program , we do everything just
* with three sequences . In order to be so cheap we may have some
* flickering effect with some slow terminal , but the lesser sequences
* the more compatible .
*
* EL ( Erase Line )
* Sequence : ESC [ n K
* Effect : if n is 0 or missing , clear from cursor to end of line
* Effect : if n is 1 , clear from beginning of line to cursor
* Effect : if n is 2 , clear entire line
*
* CUF ( CUrsor Forward )
* Sequence : ESC [ n C
* Effect : moves cursor forward n chars
*
* CUB ( CUrsor Backward )
* Sequence : ESC [ n D
* Effect : moves cursor backward n chars
*
* The following is used to get the terminal width if getting
* the width with the TIOCGWINSZ ioctl fails
*
* DSR ( Device Status Report )
* Sequence : ESC [ 6 n
* Effect : reports the current cusor position as ESC [ n ; m R
* where n is the row and m is the column
*
* When multi line mode is enabled , we also use an additional escape
* sequence . However multi line editing is disabled by default .
*
* CUU ( Cursor Up )
* Sequence : ESC [ n A
* Effect : moves cursor up of n chars .
*
* CUD ( Cursor Down )
* Sequence : ESC [ n B
* Effect : moves cursor down of n chars .
*
* When linenoiseClearScreen ( ) is called , two additional escape sequences
* are used in order to clear the screen and position the cursor at home
* position .
*
* CUP ( Cursor position )
* Sequence : ESC [ H
* Effect : moves the cursor to upper left corner
*
* ED ( Erase display )
* Sequence : ESC [ 2 J
* Effect : clear the whole screen
*
*/
# include <unistd.h>
# include <stdlib.h>
# include <stdio.h>
2023-11-23 20:49:21 +00:00
# include <stdio_ext.h>
2023-11-22 11:58:43 +00:00
# include <errno.h>
# include <string.h>
# include <stdlib.h>
# include <ctype.h>
# include <sys/stat.h>
2023-11-23 20:49:21 +00:00
# include <assert.h>
# include <sys/fcntl.h>
2023-11-22 11:58:43 +00:00
# include <sys/types.h>
# include <unistd.h>
2023-11-22 16:08:14 +00:00
# include "linenoise.h"
2023-11-22 11:58:43 +00:00
# define LINENOISE_DEFAULT_HISTORY_MAX_LEN 100
2023-11-23 21:40:56 +00:00
# define LINENOISE_DEFAULT_MAX_LINE 4096
# define LINENOISE_MINIMAL_MAX_LINE 64
2023-11-23 20:49:21 +00:00
# define LINENOISE_COMMAND_MAX_LEN 32
2023-11-23 21:40:56 +00:00
# define LINENOISE_PASTE_KEY_DELAY 30 /* Delay, in milliseconds, between two characters being pasted from clipboard */
2023-11-22 11:58:43 +00:00
static linenoiseCompletionCallback * completionCallback = NULL ;
static linenoiseHintsCallback * hintsCallback = NULL ;
static linenoiseFreeHintsCallback * freeHintsCallback = NULL ;
2023-11-22 22:35:09 +00:00
static void refreshLineWithCompletion ( struct linenoiseState * ls , linenoiseCompletions * lc , int flags ) ;
static void refreshLineWithFlags ( struct linenoiseState * l , int flags ) ;
2023-11-22 11:58:43 +00:00
2023-11-22 18:04:22 +00:00
static int maskmode = 0 ; /* Show "***" instead of input. For passwords. */
2023-11-23 21:40:56 +00:00
static size_t max_cmdline_length = LINENOISE_DEFAULT_MAX_LINE ;
2023-11-22 11:58:43 +00:00
static int mlmode = 0 ; /* Multi line mode. Default is single line. */
2023-11-23 20:49:21 +00:00
static int dumbmode = 0 ; /* Dumb mode where line editing is disabled. Off by default */
2023-11-22 11:58:43 +00:00
static int history_max_len = LINENOISE_DEFAULT_HISTORY_MAX_LEN ;
static int history_len = 0 ;
static char * * history = NULL ;
2023-11-22 16:08:14 +00:00
2023-11-22 11:58:43 +00:00
enum KEY_ACTION {
KEY_NULL = 0 , /* NULL */
CTRL_A = 1 , /* Ctrl+a */
CTRL_B = 2 , /* Ctrl-b */
CTRL_C = 3 , /* Ctrl-c */
CTRL_D = 4 , /* Ctrl-d */
CTRL_E = 5 , /* Ctrl-e */
CTRL_F = 6 , /* Ctrl-f */
CTRL_H = 8 , /* Ctrl-h */
TAB = 9 , /* Tab */
CTRL_K = 11 , /* Ctrl+k */
CTRL_L = 12 , /* Ctrl+l */
2023-11-22 16:08:14 +00:00
ENTER = 10 , /* Enter */
2023-11-22 11:58:43 +00:00
CTRL_N = 14 , /* Ctrl-n */
CTRL_P = 16 , /* Ctrl-p */
CTRL_T = 20 , /* Ctrl-t */
CTRL_U = 21 , /* Ctrl+u */
CTRL_W = 23 , /* Ctrl+w */
ESC = 27 , /* Escape */
BACKSPACE = 127 /* Backspace */
} ;
int linenoiseHistoryAdd ( const char * line ) ;
2023-11-22 21:44:22 +00:00
# define REFRESH_CLEAN (1<<0) // Clean the old prompt from the screen
# define REFRESH_WRITE (1<<1) // Rewrite the prompt on the screen.
# define REFRESH_ALL (REFRESH_CLEAN|REFRESH_WRITE) // Do both.
2023-11-22 11:58:43 +00:00
static void refreshLine ( struct linenoiseState * l ) ;
2023-11-22 16:08:14 +00:00
/* ======================= Low level terminal handling ====================== */
2023-11-22 14:50:57 +00:00
2023-11-22 18:04:22 +00:00
/* Enable "mask mode". When it is enabled, instead of the input that
* the user is typing , the terminal will just display a corresponding
* number of asterisks , like " **** " . This is useful for passwords and other
* secrets that should not be displayed . */
void linenoiseMaskModeEnable ( void ) {
2023-11-22 17:56:22 +00:00
maskmode = 1 ;
}
2023-11-22 18:04:22 +00:00
/* Disable mask mode. */
void linenoiseMaskModeDisable ( void ) {
2023-11-22 17:56:22 +00:00
maskmode = 0 ;
}
2023-11-22 11:58:43 +00:00
/* Set if to use or not the multi line mode. */
void linenoiseSetMultiLine ( int ml ) {
mlmode = ml ;
}
2023-11-23 20:49:21 +00:00
/* Set if terminal does not recognize escape sequences */
void linenoiseSetDumbMode ( int set ) {
dumbmode = set ;
}
/* Returns whether the current mode is dumbmode or not. */
bool linenoiseIsDumbMode ( void ) {
return dumbmode ;
}
static void flushWrite ( void ) {
if ( __fbufsize ( stdout ) > 0 ) {
fflush ( stdout ) ;
}
fsync ( fileno ( stdout ) ) ;
}
2023-11-22 11:58:43 +00:00
/* Use the ESC [6n escape sequence to query the horizontal cursor position
* and return it . On error - 1 is returned , on success the position of the
* cursor . */
2023-11-23 20:57:39 +00:00
static int getCursorPosition ( void ) {
char buf [ LINENOISE_COMMAND_MAX_LEN ] = { 0 } ;
int cols = 0 ;
int rows = 0 ;
int i = 0 ;
const int out_fd = fileno ( stdout ) ;
const int in_fd = fileno ( stdin ) ;
/* The following ANSI escape sequence is used to get from the TTY the
* cursor position . */
const char get_cursor_cmd [ ] = " \x1b [6n " ;
2023-11-22 17:43:11 +00:00
2023-11-23 20:57:39 +00:00
/* Send the command to the TTY on the other end of the UART.
* Let ' s use unistd ' s write function . Thus , data sent through it are raw
* reducing the overhead compared to using fputs , fprintf , etc . . . */
write ( out_fd , get_cursor_cmd , sizeof ( get_cursor_cmd ) ) ;
/* For USB CDC, it is required to flush the output. */
2023-11-23 20:49:21 +00:00
flushWrite ( ) ;
2023-11-23 20:57:39 +00:00
/* The other end will send its response which format is ESC [ rows ; cols R
* We don ' t know exactly how many bytes we have to read , thus , perform a
* read for each byte .
* Stop right before the last character of the buffer , to be able to NULL
* terminate it . */
2023-11-22 16:08:14 +00:00
while ( i < sizeof ( buf ) - 1 ) {
2023-11-23 20:57:39 +00:00
/* Keep using unistd's functions. Here, using `read` instead of `fgets`
* or ` fgets ` guarantees us that we we can read a byte regardless on
* whether the sender sent end of line character ( s ) ( CR , CRLF , LF ) . */
if ( read ( in_fd , buf + i , 1 ) ! = 1 | | buf [ i ] = = ' R ' ) {
/* If we couldn't read a byte from STDIN or if 'R' was received,
* the transmission is finished . */
break ;
}
/* For some reasons, it is possible that we receive new line character
* after querying the cursor position on some UART . Let ' s ignore them ,
* this will not affect the rest of the program . */
if ( buf [ i ] ! = ' \n ' ) {
i + + ;
}
2023-11-22 11:58:43 +00:00
}
2023-11-23 20:57:39 +00:00
/* NULL-terminate the buffer, this is required by `sscanf`. */
2023-11-22 11:58:43 +00:00
buf [ i ] = ' \0 ' ;
2023-11-23 20:57:39 +00:00
/* Parse the received data to get the position of the cursor. */
if ( buf [ 0 ] ! = ESC | | buf [ 1 ] ! = ' [ ' | | sscanf ( buf + 2 , " %d;%d " , & rows , & cols ) ! = 2 ) {
return - 1 ;
}
2023-11-22 11:58:43 +00:00
return cols ;
}
/* Try to get the number of columns in the current terminal, or assume 80
* if it fails . */
2023-11-23 20:57:39 +00:00
static int getColumns ( void ) {
int start = 0 ;
int cols = 0 ;
int written = 0 ;
char seq [ LINENOISE_COMMAND_MAX_LEN ] = { 0 } ;
const int fd = fileno ( stdout ) ;
/* The following ANSI escape sequence is used to tell the TTY to move
* the cursor to the most - right position . */
const char move_cursor_right [ ] = " \x1b [999C " ;
const size_t cmd_len = sizeof ( move_cursor_right ) ;
/* This one is used to set the cursor position. */
const char set_cursor_pos [ ] = " \x1b [%dD " ;
2023-11-22 11:58:43 +00:00
/* Get the initial position so we can restore it later. */
start = getCursorPosition ( ) ;
2023-11-23 20:57:39 +00:00
if ( start = = - 1 ) {
goto failed ;
}
2023-11-22 11:58:43 +00:00
2023-11-23 20:57:39 +00:00
/* Send the command to go to right margin. Use `write` function instead of
* ` fwrite ` for the same reasons explained in ` getCursorPosition ( ) ` */
if ( write ( fd , move_cursor_right , cmd_len ) ! = cmd_len ) {
goto failed ;
}
2023-11-23 20:49:21 +00:00
flushWrite ( ) ;
2023-11-23 20:57:39 +00:00
/* After sending this command, we can get the new position of the cursor,
* we ' d get the size , in columns , of the opened TTY . */
2023-11-22 11:58:43 +00:00
cols = getCursorPosition ( ) ;
2023-11-23 20:57:39 +00:00
if ( cols = = - 1 ) {
goto failed ;
}
2023-11-22 11:58:43 +00:00
2023-11-23 20:57:39 +00:00
/* Restore the position of the cursor back. */
2023-11-22 11:58:43 +00:00
if ( cols > start ) {
2023-11-23 20:57:39 +00:00
/* Generate the move cursor command. */
written = snprintf ( seq , LINENOISE_COMMAND_MAX_LEN , set_cursor_pos , cols - start ) ;
/* If `written` is equal or bigger than LINENOISE_COMMAND_MAX_LEN, it
* means that the output has been truncated because the size provided
* is too small . */
assert ( written < LINENOISE_COMMAND_MAX_LEN ) ;
/* Send the command with `write`, which is not buffered. */
if ( write ( fd , seq , written ) = = - 1 ) {
2023-11-22 11:58:43 +00:00
/* Can't recover... */
}
2023-11-23 20:49:21 +00:00
flushWrite ( ) ;
2023-11-22 11:58:43 +00:00
}
return cols ;
failed :
return 80 ;
}
/* Clear the screen. Used to handle ctrl+l */
void linenoiseClearScreen ( void ) {
2023-11-22 16:08:14 +00:00
fprintf ( stdout , " \x1b [H \x1b [2J " ) ;
2023-11-23 20:49:21 +00:00
flushWrite ( ) ;
2023-11-22 11:58:43 +00:00
}
/* Beep, used for completion when there is nothing to complete or when all
* the choices were already shown . */
static void linenoiseBeep ( void ) {
2023-11-22 14:54:51 +00:00
fprintf ( stdout , " \ x7 " ) ;
2023-11-23 20:49:21 +00:00
flushWrite ( ) ;
2023-11-22 11:58:43 +00:00
}
/* ============================== Completion ================================ */
/* Free a list of completion option populated by linenoiseAddCompletion(). */
static void freeCompletions ( linenoiseCompletions * lc ) {
size_t i ;
for ( i = 0 ; i < lc - > len ; i + + )
free ( lc - > cvec [ i ] ) ;
if ( lc - > cvec ! = NULL )
free ( lc - > cvec ) ;
}
2023-11-22 22:35:09 +00:00
/* Called by completeLine() and linenoiseShow() to render the current
* edited line with the proposed completion . If the current completion table
* is already available , it is passed as second argument , otherwise the
* function will use the callback to obtain it .
*
* Flags are the same as refreshLine * ( ) , that is REFRESH_ * macros . */
static void refreshLineWithCompletion ( struct linenoiseState * ls , linenoiseCompletions * lc , int flags ) {
/* Obtain the table of completions if the caller didn't provide one. */
linenoiseCompletions ctable = { 0 , NULL } ;
if ( lc = = NULL ) {
completionCallback ( ls - > buf , & ctable ) ;
lc = & ctable ;
}
/* Show the edited line with completion if possible, or just refresh. */
if ( ls - > completion_idx < lc - > len ) {
struct linenoiseState saved = * ls ;
ls - > len = ls - > pos = strlen ( lc - > cvec [ ls - > completion_idx ] ) ;
ls - > buf = lc - > cvec [ ls - > completion_idx ] ;
refreshLineWithFlags ( ls , flags ) ;
ls - > len = saved . len ;
ls - > pos = saved . pos ;
ls - > buf = saved . buf ;
} else {
refreshLineWithFlags ( ls , flags ) ;
}
/* Free the completions table if needed. */
if ( lc ! = & ctable ) freeCompletions ( & ctable ) ;
}
2023-11-22 20:59:51 +00:00
/* This is an helper function for linenoiseEdit*() and is called when the
2023-11-22 11:58:43 +00:00
* user types the < tab > key in order to complete the string currently in the
* input .
*
* The state of the editing is encapsulated into the pointed linenoiseState
2023-11-22 22:11:41 +00:00
* structure as described in the structure definition .
*
* If the function returns non - zero , the caller should handle the
* returned value as a byte read from the standard input , and process
* it as usually : this basically means that the function may return a byte
* read from the termianl but not processed . Otherwise , if zero is returned ,
* the input was consumed by the completeLine ( ) function to navigate the
* possible completions , and the caller should read for the next characters
* from stdin . */
2023-11-22 22:23:09 +00:00
static int completeLine ( struct linenoiseState * ls , int keypressed ) {
2023-11-22 11:58:43 +00:00
linenoiseCompletions lc = { 0 , NULL } ;
2023-11-22 22:23:09 +00:00
int nwritten ;
char c = keypressed ;
2023-11-22 11:58:43 +00:00
completionCallback ( ls - > buf , & lc ) ;
if ( lc . len = = 0 ) {
linenoiseBeep ( ) ;
2023-11-22 22:23:09 +00:00
ls - > in_completion = 0 ;
2023-11-22 11:58:43 +00:00
} else {
2023-11-22 22:23:09 +00:00
switch ( c ) {
case 9 : /* tab */
if ( ls - > in_completion = = 0 ) {
ls - > in_completion = 1 ;
ls - > completion_idx = 0 ;
} else {
ls - > completion_idx = ( ls - > completion_idx + 1 ) % ( lc . len + 1 ) ;
if ( ls - > completion_idx = = lc . len ) linenoiseBeep ( ) ;
}
c = 0 ;
break ;
case 27 : /* escape */
/* Re-show original buffer */
if ( ls - > completion_idx < lc . len ) refreshLine ( ls ) ;
ls - > in_completion = 0 ;
c = 0 ;
break ;
default :
/* Update buffer and return */
if ( ls - > completion_idx < lc . len ) {
nwritten = snprintf ( ls - > buf , ls - > buflen , " %s " ,
lc . cvec [ ls - > completion_idx ] ) ;
ls - > len = ls - > pos = nwritten ;
}
ls - > in_completion = 0 ;
break ;
}
2023-11-22 16:08:14 +00:00
2023-11-22 22:23:09 +00:00
/* Show completion or original buffer */
if ( ls - > in_completion & & ls - > completion_idx < lc . len ) {
2023-11-22 22:35:09 +00:00
refreshLineWithCompletion ( ls , & lc , REFRESH_ALL ) ;
2023-11-22 22:23:09 +00:00
} else {
refreshLine ( ls ) ;
2023-11-22 11:58:43 +00:00
}
}
freeCompletions ( & lc ) ;
return c ; /* Return last read character */
}
/* Register a callback function to be called for tab-completion. */
void linenoiseSetCompletionCallback ( linenoiseCompletionCallback * fn ) {
completionCallback = fn ;
}
/* Register a hits function to be called to show hits to the user at the
* right of the prompt . */
void linenoiseSetHintsCallback ( linenoiseHintsCallback * fn ) {
hintsCallback = fn ;
}
/* Register a function to free the hints returned by the hints callback
* registered with linenoiseSetHintsCallback ( ) . */
void linenoiseSetFreeHintsCallback ( linenoiseFreeHintsCallback * fn ) {
freeHintsCallback = fn ;
}
/* This function is used by the callback function registered by the user
* in order to add completion options given the input string when the
* user typed < tab > . See the example . c source code for a very easy to
* understand example . */
void linenoiseAddCompletion ( linenoiseCompletions * lc , const char * str ) {
size_t len = strlen ( str ) ;
char * copy , * * cvec ;
copy = malloc ( len + 1 ) ;
if ( copy = = NULL ) return ;
memcpy ( copy , str , len + 1 ) ;
cvec = realloc ( lc - > cvec , sizeof ( char * ) * ( lc - > len + 1 ) ) ;
if ( cvec = = NULL ) {
free ( copy ) ;
return ;
}
lc - > cvec = cvec ;
lc - > cvec [ lc - > len + + ] = copy ;
}
/* =========================== Line editing ================================= */
/* We define a very simple "append buffer" structure, that is an heap
* allocated string where we can append to . This is useful in order to
* write all the escape sequences in a buffer and flush them to the standard
* output in a single call , to avoid flickering effects . */
struct abuf {
char * b ;
int len ;
} ;
static void abInit ( struct abuf * ab ) {
ab - > b = NULL ;
ab - > len = 0 ;
}
static void abAppend ( struct abuf * ab , const char * s , int len ) {
char * new = realloc ( ab - > b , ab - > len + len ) ;
if ( new = = NULL ) return ;
memcpy ( new + ab - > len , s , len ) ;
ab - > b = new ;
ab - > len + = len ;
}
static void abFree ( struct abuf * ab ) {
free ( ab - > b ) ;
}
/* Helper of refreshSingleLine() and refreshMultiLine() to show hints
* to the right of the prompt . */
void refreshShowHints ( struct abuf * ab , struct linenoiseState * l , int plen ) {
char seq [ 64 ] ;
if ( hintsCallback & & plen + l - > len < l - > cols ) {
int color = - 1 , bold = 0 ;
char * hint = hintsCallback ( l - > buf , & color , & bold ) ;
if ( hint ) {
int hintlen = strlen ( hint ) ;
int hintmaxlen = l - > cols - ( plen + l - > len ) ;
if ( hintlen > hintmaxlen ) hintlen = hintmaxlen ;
if ( bold = = 1 & & color = = - 1 ) color = 37 ;
2023-11-22 17:43:11 +00:00
if ( color ! = - 1 | | bold ! = 0 )
2023-11-22 11:58:43 +00:00
snprintf ( seq , 64 , " \033 [%d;%d;49m " , bold , color ) ;
2023-11-22 17:46:15 +00:00
else
seq [ 0 ] = ' \0 ' ;
2023-11-22 17:43:11 +00:00
abAppend ( ab , seq , strlen ( seq ) ) ;
2023-11-22 11:58:43 +00:00
abAppend ( ab , hint , hintlen ) ;
if ( color ! = - 1 | | bold ! = 0 )
abAppend ( ab , " \033 [0m " , 4 ) ;
/* Call the function to free the hint returned. */
if ( freeHintsCallback ) freeHintsCallback ( hint ) ;
}
}
}
/* Single line low level line refresh.
*
* Rewrite the currently edited line accordingly to the buffer content ,
2023-11-22 21:44:22 +00:00
* cursor position , and number of columns of the terminal .
*
* Flags is REFRESH_ * macros . The function can just remove the old
* prompt , just write it , or both . */
static void refreshSingleLine ( struct linenoiseState * l , int flags ) {
2023-11-22 11:58:43 +00:00
char seq [ 64 ] ;
2023-11-22 16:08:14 +00:00
size_t plen = l - > plen ;
2023-11-22 11:58:43 +00:00
char * buf = l - > buf ;
size_t len = l - > len ;
size_t pos = l - > pos ;
struct abuf ab ;
while ( ( plen + pos ) > = l - > cols ) {
buf + + ;
len - - ;
pos - - ;
}
while ( plen + len > l - > cols ) {
len - - ;
}
abInit ( & ab ) ;
/* Cursor to left edge */
2023-11-22 21:44:22 +00:00
snprintf ( seq , sizeof ( seq ) , " \r " ) ;
2023-11-22 11:58:43 +00:00
abAppend ( & ab , seq , strlen ( seq ) ) ;
2023-11-22 21:44:22 +00:00
if ( flags & REFRESH_WRITE ) {
/* Write the prompt and the current buffer content */
abAppend ( & ab , l - > prompt , strlen ( l - > prompt ) ) ;
if ( maskmode = = 1 ) {
while ( len - - ) abAppend ( & ab , " * " , 1 ) ;
} else {
abAppend ( & ab , buf , len ) ;
}
/* Show hits if any. */
refreshShowHints ( & ab , l , plen ) ;
2023-11-22 18:04:22 +00:00
}
2023-11-22 11:58:43 +00:00
/* Erase to right */
2023-11-22 21:44:22 +00:00
snprintf ( seq , sizeof ( seq ) , " \x1b [0K " ) ;
2023-11-22 16:08:14 +00:00
abAppend ( & ab , seq , strlen ( seq ) ) ;
2023-11-22 21:44:22 +00:00
if ( flags & REFRESH_WRITE ) {
/* Move cursor to original position. */
snprintf ( seq , sizeof ( seq ) , " \r \x1b [%dC " , ( int ) ( pos + plen ) ) ;
abAppend ( & ab , seq , strlen ( seq ) ) ;
}
2023-11-22 17:43:11 +00:00
if ( fwrite ( ab . b , ab . len , 1 , stdout ) = = - 1 ) { } /* Can't recover from write error. */
2023-11-23 20:49:21 +00:00
flushWrite ( ) ;
2023-11-22 11:58:43 +00:00
abFree ( & ab ) ;
}
/* Multi line low level line refresh.
*
* Rewrite the currently edited line accordingly to the buffer content ,
2023-11-22 21:44:22 +00:00
* cursor position , and number of columns of the terminal .
*
* Flags is REFRESH_ * macros . The function can just remove the old
* prompt , just write it , or both . */
static void refreshMultiLine ( struct linenoiseState * l , int flags ) {
2023-11-22 11:58:43 +00:00
char seq [ 64 ] ;
2023-11-22 16:08:14 +00:00
int plen = l - > plen ;
2023-11-22 11:58:43 +00:00
int rows = ( plen + l - > len + l - > cols - 1 ) / l - > cols ; /* rows used by current buf. */
int rpos = ( plen + l - > oldpos + l - > cols ) / l - > cols ; /* cursor relative row. */
int rpos2 ; /* rpos after refresh. */
int col ; /* colum position, zero-based. */
2023-11-22 22:39:29 +00:00
int old_rows = l - > oldrows ;
2023-11-22 16:08:14 +00:00
int j ;
2023-11-22 11:58:43 +00:00
struct abuf ab ;
2023-11-22 22:39:29 +00:00
l - > oldrows = rows ;
2023-11-22 11:58:43 +00:00
/* First step: clear all the lines used before. To do so start by
* going to the last row . */
abInit ( & ab ) ;
2023-11-22 21:44:22 +00:00
if ( flags & REFRESH_CLEAN ) {
if ( old_rows - rpos > 0 ) {
snprintf ( seq , 64 , " \x1b [%dB " , old_rows - rpos ) ;
abAppend ( & ab , seq , strlen ( seq ) ) ;
}
2023-11-22 11:58:43 +00:00
2023-11-22 21:44:22 +00:00
/* Now for every row clear it, go up. */
for ( j = 0 ; j < old_rows - 1 ; j + + ) {
snprintf ( seq , 64 , " \r \x1b [0K \x1b [1A " ) ;
abAppend ( & ab , seq , strlen ( seq ) ) ;
}
2023-11-22 21:48:40 +00:00
}
2023-11-22 11:58:43 +00:00
2023-11-22 21:48:40 +00:00
if ( flags & REFRESH_ALL ) {
2023-11-22 21:44:22 +00:00
/* Clean the top line. */
snprintf ( seq , 64 , " \r \x1b [0K " ) ;
2023-11-22 16:08:14 +00:00
abAppend ( & ab , seq , strlen ( seq ) ) ;
}
2023-11-22 11:58:43 +00:00
2023-11-22 21:44:22 +00:00
if ( flags & REFRESH_WRITE ) {
/* Write the prompt and the current buffer content */
abAppend ( & ab , l - > prompt , strlen ( l - > prompt ) ) ;
if ( maskmode = = 1 ) {
unsigned int i ;
for ( i = 0 ; i < l - > len ; i + + ) abAppend ( & ab , " * " , 1 ) ;
} else {
abAppend ( & ab , l - > buf , l - > len ) ;
}
/* Show hits if any. */
refreshShowHints ( & ab , l , plen ) ;
/* If we are at the very end of the screen with our prompt, we need to
* emit a newline and move the prompt to the first column . */
if ( l - > pos & &
l - > pos = = l - > len & &
( l - > pos + plen ) % l - > cols = = 0 )
{
abAppend ( & ab , " \n " , 1 ) ;
snprintf ( seq , 64 , " \r " ) ;
abAppend ( & ab , seq , strlen ( seq ) ) ;
rows + + ;
2023-11-22 22:39:29 +00:00
if ( rows > ( int ) l - > oldrows ) l - > oldrows = rows ;
2023-11-22 21:44:22 +00:00
}
2023-11-22 14:50:57 +00:00
2023-11-22 21:44:22 +00:00
/* Move cursor to right position. */
rpos2 = ( plen + l - > pos + l - > cols ) / l - > cols ; /* Current cursor relative row */
/* Go up till we reach the expected positon. */
if ( rows - rpos2 > 0 ) {
snprintf ( seq , 64 , " \x1b [%dA " , rows - rpos2 ) ;
abAppend ( & ab , seq , strlen ( seq ) ) ;
}
/* Set column. */
col = ( plen + ( int ) l - > pos ) % ( int ) l - > cols ;
if ( col )
snprintf ( seq , 64 , " \r \x1b [%dC " , col ) ;
else
snprintf ( seq , 64 , " \r " ) ;
2023-11-22 11:58:43 +00:00
abAppend ( & ab , seq , strlen ( seq ) ) ;
}
l - > oldpos = l - > pos ;
2023-11-22 21:44:22 +00:00
if ( fwrite ( ab . b , ab . len , 1 , stdout ) = = - 1 ) { } /* Can't recover from write error. */
2023-11-23 20:49:21 +00:00
flushWrite ( ) ;
2023-11-22 11:58:43 +00:00
abFree ( & ab ) ;
}
/* Calls the two low level functions refreshSingleLine() or
* refreshMultiLine ( ) according to the selected mode . */
2023-11-22 22:35:09 +00:00
static void refreshLineWithFlags ( struct linenoiseState * l , int flags ) {
2023-11-22 11:58:43 +00:00
if ( mlmode )
2023-11-22 22:35:09 +00:00
refreshMultiLine ( l , flags ) ;
2023-11-22 21:44:22 +00:00
else
2023-11-22 22:35:09 +00:00
refreshSingleLine ( l , flags ) ;
}
/* Utility function to avoid specifying REFRESH_ALL all the times. */
static void refreshLine ( struct linenoiseState * l ) {
refreshLineWithFlags ( l , REFRESH_ALL ) ;
2023-11-22 21:44:22 +00:00
}
/* Hide the current line, when using the multiplexing API. */
void linenoiseHide ( struct linenoiseState * l ) {
if ( mlmode )
refreshMultiLine ( l , REFRESH_CLEAN ) ;
2023-11-22 11:58:43 +00:00
else
2023-11-22 21:44:22 +00:00
refreshSingleLine ( l , REFRESH_CLEAN ) ;
}
/* Show the current line, when using the multiplexing API. */
void linenoiseShow ( struct linenoiseState * l ) {
2023-11-22 22:35:09 +00:00
if ( l - > in_completion ) {
refreshLineWithCompletion ( l , NULL , REFRESH_WRITE ) ;
} else {
refreshLineWithFlags ( l , REFRESH_WRITE ) ;
}
2023-11-22 11:58:43 +00:00
}
/* Insert the character 'c' at cursor current position.
*
* On error writing to the terminal - 1 is returned , otherwise 0. */
int linenoiseEditInsert ( struct linenoiseState * l , char c ) {
if ( l - > len < l - > buflen ) {
if ( l - > len = = l - > pos ) {
l - > buf [ l - > pos ] = c ;
l - > pos + + ;
l - > len + + ;
l - > buf [ l - > len ] = ' \0 ' ;
if ( ( ! mlmode & & l - > plen + l - > len < l - > cols & & ! hintsCallback ) ) {
/* Avoid a full update of the line in the
* trivial case . */
2023-11-22 18:08:37 +00:00
char d = ( maskmode = = 1 ) ? ' * ' : c ;
if ( fwrite ( & d , 1 , 1 , stdout ) = = - 1 ) return - 1 ;
2023-11-23 20:49:21 +00:00
flushWrite ( ) ;
2023-11-22 11:58:43 +00:00
} else {
refreshLine ( l ) ;
}
} else {
memmove ( l - > buf + l - > pos + 1 , l - > buf + l - > pos , l - > len - l - > pos ) ;
l - > buf [ l - > pos ] = c ;
l - > len + + ;
l - > pos + + ;
l - > buf [ l - > len ] = ' \0 ' ;
refreshLine ( l ) ;
}
}
return 0 ;
}
2023-11-23 21:40:56 +00:00
int linenoiseInsertPastedChar ( struct linenoiseState * l , char c ) {
int fd = fileno ( stdout ) ;
if ( l - > len < l - > buflen & & l - > len = = l - > pos ) {
l - > buf [ l - > pos ] = c ;
l - > pos + + ;
l - > len + + ;
l - > buf [ l - > len ] = ' \0 ' ;
if ( write ( fd , & c , 1 ) = = - 1 ) {
return - 1 ;
}
flushWrite ( ) ;
}
return 0 ;
}
2023-11-22 11:58:43 +00:00
/* Move cursor on the left. */
void linenoiseEditMoveLeft ( struct linenoiseState * l ) {
if ( l - > pos > 0 ) {
l - > pos - - ;
refreshLine ( l ) ;
}
}
/* Move cursor on the right. */
void linenoiseEditMoveRight ( struct linenoiseState * l ) {
if ( l - > pos ! = l - > len ) {
l - > pos + + ;
refreshLine ( l ) ;
}
}
/* Move cursor to the start of the line. */
void linenoiseEditMoveHome ( struct linenoiseState * l ) {
if ( l - > pos ! = 0 ) {
l - > pos = 0 ;
refreshLine ( l ) ;
}
}
/* Move cursor to the end of the line. */
void linenoiseEditMoveEnd ( struct linenoiseState * l ) {
if ( l - > pos ! = l - > len ) {
l - > pos = l - > len ;
refreshLine ( l ) ;
}
}
/* Substitute the currently edited line with the next or previous history
* entry as specified by ' dir ' . */
# define LINENOISE_HISTORY_NEXT 0
# define LINENOISE_HISTORY_PREV 1
void linenoiseEditHistoryNext ( struct linenoiseState * l , int dir ) {
if ( history_len > 1 ) {
/* Update the current history entry before to
* overwrite it with the next one . */
free ( history [ history_len - 1 - l - > history_index ] ) ;
history [ history_len - 1 - l - > history_index ] = strdup ( l - > buf ) ;
/* Show the new entry */
l - > history_index + = ( dir = = LINENOISE_HISTORY_PREV ) ? 1 : - 1 ;
if ( l - > history_index < 0 ) {
l - > history_index = 0 ;
return ;
} else if ( l - > history_index > = history_len ) {
l - > history_index = history_len - 1 ;
return ;
}
strncpy ( l - > buf , history [ history_len - 1 - l - > history_index ] , l - > buflen ) ;
l - > buf [ l - > buflen - 1 ] = ' \0 ' ;
l - > len = l - > pos = strlen ( l - > buf ) ;
refreshLine ( l ) ;
}
}
/* Delete the character at the right of the cursor without altering the cursor
* position . Basically this is what happens with the " Delete " keyboard key . */
void linenoiseEditDelete ( struct linenoiseState * l ) {
if ( l - > len > 0 & & l - > pos < l - > len ) {
memmove ( l - > buf + l - > pos , l - > buf + l - > pos + 1 , l - > len - l - > pos - 1 ) ;
l - > len - - ;
l - > buf [ l - > len ] = ' \0 ' ;
refreshLine ( l ) ;
}
}
/* Backspace implementation. */
void linenoiseEditBackspace ( struct linenoiseState * l ) {
if ( l - > pos > 0 & & l - > len > 0 ) {
memmove ( l - > buf + l - > pos - 1 , l - > buf + l - > pos , l - > len - l - > pos ) ;
l - > pos - - ;
l - > len - - ;
l - > buf [ l - > len ] = ' \0 ' ;
refreshLine ( l ) ;
}
}
/* Delete the previosu word, maintaining the cursor at the start of the
* current word . */
void linenoiseEditDeletePrevWord ( struct linenoiseState * l ) {
size_t old_pos = l - > pos ;
size_t diff ;
while ( l - > pos > 0 & & l - > buf [ l - > pos - 1 ] = = ' ' )
l - > pos - - ;
while ( l - > pos > 0 & & l - > buf [ l - > pos - 1 ] ! = ' ' )
l - > pos - - ;
diff = old_pos - l - > pos ;
memmove ( l - > buf + l - > pos , l - > buf + old_pos , l - > len - old_pos + 1 ) ;
l - > len - = diff ;
refreshLine ( l ) ;
}
2023-11-23 20:49:21 +00:00
// TODO: try to make a non-blocking dumb mode
static char * linenoiseDumb ( struct linenoiseState * l ) {
/* dumb terminal, fall back to fgets */
2023-11-23 22:25:02 +00:00
// Not needed anymore, prompt is now in linenoiseEditStart
// fputs(l->prompt, stdout);
2023-11-23 20:49:21 +00:00
flushWrite ( ) ;
l - > len = 0 ; //needed?
while ( l - > len < l - > buflen ) {
int c = fgetc ( stdin ) ;
if ( c = = ' \n ' ) {
break ;
} else if ( c > = 0x1c & & c < = 0x1f ) {
continue ; /* consume arrow keys */
} else if ( c = = BACKSPACE | | c = = 0x8 ) {
if ( l - > len > 0 ) {
l - > buf [ l - > len - 1 ] = 0 ;
l - > len - - ;
}
fputs ( " \x08 " , stdout ) ; /* Windows CMD: erase symbol under cursor */
flushWrite ( ) ;
} else {
l - > buf [ l - > len ] = c ;
l - > len + + ;
}
fputc ( c , stdout ) ; /* echo */
flushWrite ( ) ;
}
fputc ( ' \n ' , stdout ) ;
flushWrite ( ) ;
2023-11-23 21:40:56 +00:00
// if (l->len == 0) return linenoiseEditMore;
2023-11-23 20:49:21 +00:00
return strdup ( l - > buf ) ;
}
2023-11-23 21:40:56 +00:00
uint32_t getMillis ( void ) {
struct timeval tv = { 0 } ;
gettimeofday ( & tv , NULL ) ;
return tv . tv_sec * 1000 + tv . tv_usec / 1000 ;
}
2023-11-22 20:59:51 +00:00
/* This function is part of the multiplexed API of Linenoise, that is used
* in order to implement the blocking variant of the API but can also be
* called by the user directly in an event driven program . It will :
2023-11-22 14:50:57 +00:00
*
2023-11-22 20:59:51 +00:00
* 1. Initialize the linenoise state passed by the user .
* 2. Put the terminal in RAW mode .
* 3. Show the prompt .
* 4. Return control to the user , that will have to call linenoiseEditFeed ( )
* each time there is some data arriving in the standard input .
2023-11-22 14:50:57 +00:00
*
2023-11-22 20:59:51 +00:00
* The user can also call linenoiseEditHide ( ) and linenoiseEditShow ( ) if it
* is required to show some input arriving asyncronously , without mixing
* it with the currently edited line .
*
* When linenoiseEditFeed ( ) returns non - NULL , the user finished with the
* line editing session ( pressed enter CTRL - D / C ) : in this case the caller
* needs to call linenoiseEditStop ( ) to put back the terminal in normal
* mode . This will not destroy the buffer , as long as the linenoiseState
* is still valid in the context of the caller .
*
* The function returns 0 on success , or - 1 if writing to standard output
* fails . If stdin_fd or stdout_fd are set to - 1 , the default is to use
* STDIN_FILENO and STDOUT_FILENO .
*/
int linenoiseEditStart ( struct linenoiseState * l , char * buf , size_t buflen , const char * prompt ) {
/* Populate the linenoise state that we pass to functions implementing
* specific editing functionalities . */
2023-11-22 22:23:09 +00:00
l - > in_completion = 0 ;
2023-11-22 20:59:51 +00:00
l - > buf = buf ;
l - > buflen = buflen ;
l - > prompt = prompt ;
l - > plen = strlen ( prompt ) ;
l - > oldpos = l - > pos = 0 ;
l - > len = 0 ;
l - > cols = getColumns ( ) ;
2023-11-22 22:39:29 +00:00
l - > oldrows = 0 ;
2023-11-22 20:59:51 +00:00
l - > history_index = 0 ;
/* Buffer starts empty. */
l - > buf [ 0 ] = ' \0 ' ;
l - > buflen - - ; /* Make sure there is always space for the nulterm */
2023-11-22 16:08:14 +00:00
2023-11-22 20:59:51 +00:00
/* The latest history entry is always our current buffer, that
* initially is just an empty string . */
2023-11-23 20:49:21 +00:00
if ( ! dumbmode ) {
linenoiseHistoryAdd ( " " ) ;
int pos1 = getCursorPosition ( ) ;
if ( fwrite ( prompt , l - > plen , 1 , stdout ) = = - 1 ) return - 1 ;
flushWrite ( ) ;
int pos2 = getCursorPosition ( ) ;
if ( pos1 > = 0 & & pos2 > = 0 ) {
l - > plen = pos2 - pos1 ;
}
} else {
if ( fwrite ( prompt , l - > plen , 1 , stdout ) = = - 1 ) return - 1 ;
flushWrite ( ) ;
2023-11-22 21:44:22 +00:00
}
2023-11-22 20:59:51 +00:00
return 0 ;
}
2023-11-22 22:07:52 +00:00
char * linenoiseEditMore = " If you see this, you are misusing the API: when linenoiseEditFeed() is called, if it returns linenoiseEditMore the user is yet editing the line. See the README file for more information. " ;
2023-11-22 20:59:51 +00:00
/* This function is part of the multiplexed API of linenoise, see the top
* comment on linenoiseEditStart ( ) for more information . Call this function
* each time there is some data to read from the standard input file
* descriptor . In the case of blocking operations , this function can just be
* called in a loop , and block .
*
2023-11-22 22:07:52 +00:00
* The function returns linenoiseEditMore to signal that line editing is still
* in progress , that is , the user didn ' t yet pressed enter / CTRL - D . Otherwise
* the function returns the pointer to the heap - allocated buffer with the
* edited line , that the user should free with linenoiseFree ( ) .
*
* On special conditions , NULL is returned and errno is populated :
2023-11-22 20:59:51 +00:00
*
* EAGAIN if the user pressed Ctrl - C
* ENOENT if the user pressed Ctrl - D
2023-11-22 22:07:52 +00:00
*
* Some other errno : I / O error .
2023-11-22 20:59:51 +00:00
*/
2023-11-22 22:07:52 +00:00
char * linenoiseEditFeed ( struct linenoiseState * l ) {
2023-11-23 20:49:21 +00:00
if ( dumbmode ) return linenoiseDumb ( l ) ;
2023-11-23 21:40:56 +00:00
uint32_t t1 = 0 ;
2023-11-22 20:59:51 +00:00
char c ;
int nread ;
char seq [ 3 ] ;
2023-11-23 21:40:56 +00:00
/*
* To determine whether the user is pasting data or typing itself , we
* need to calculate how many milliseconds elapsed between two key
* presses . Indeed , if there is less than LINENOISE_PASTE_KEY_DELAY
* ( typically 30 - 40 ms ) , then a paste is being performed , else , the
* user is typing .
* NOTE : pressing a key down without releasing it will also spend
* about 40 ms ( or even more )
*/
t1 = getMillis ( ) ;
2023-11-22 20:59:51 +00:00
nread = fread ( & c , 1 , 1 , stdin ) ;
2023-11-22 22:07:52 +00:00
if ( nread < = 0 ) return NULL ;
2023-11-23 21:40:56 +00:00
// FIXME: line printed twice after pasting something
2023-11-23 21:42:17 +00:00
if ( ( getMillis ( ) - t1 ) < LINENOISE_PASTE_KEY_DELAY & & c ! = ENTER ) {
2023-11-23 21:40:56 +00:00
/* Pasting data, insert characters without formatting.
* This can only be performed when the cursor is at the end of the
* line . */
if ( linenoiseInsertPastedChar ( l , c ) ) {
errno = EIO ;
return NULL ;
}
return linenoiseEditMore ;
}
2023-11-22 20:59:51 +00:00
/* Only autocomplete when the callback is set. It returns < 0 when
* there was an error reading from fd . Otherwise it will return the
* character that should be handled next . */
2023-11-22 22:23:09 +00:00
if ( ( l - > in_completion | | c = = 9 ) & & completionCallback ! = NULL ) {
c = completeLine ( l , c ) ;
2023-11-22 20:59:51 +00:00
/* Return on errors */
2023-11-23 22:25:02 +00:00
// TODO: how was it supposed to work? c can't be less than 0
// if (c < 0) return NULL;
2023-11-22 20:59:51 +00:00
/* Read next character when 0 */
2023-11-22 22:07:52 +00:00
if ( c = = 0 ) return linenoiseEditMore ;
2023-11-22 20:59:51 +00:00
}
2023-11-22 16:08:14 +00:00
2023-11-22 20:59:51 +00:00
switch ( c ) {
case ENTER : /* enter */
history_len - - ;
free ( history [ history_len ] ) ;
if ( mlmode ) linenoiseEditMoveEnd ( l ) ;
if ( hintsCallback ) {
/* Force a refresh without hints to leave the previous
* line as the user typed it after a newline . */
linenoiseHintsCallback * hc = hintsCallback ;
hintsCallback = NULL ;
refreshLine ( l ) ;
hintsCallback = hc ;
}
2023-11-22 22:07:52 +00:00
return strdup ( l - > buf ) ;
2023-11-22 20:59:51 +00:00
case CTRL_C : /* ctrl-c */
errno = EAGAIN ;
2023-11-22 22:07:52 +00:00
return NULL ;
2023-11-22 20:59:51 +00:00
case BACKSPACE : /* backspace */
case 8 : /* ctrl-h */
linenoiseEditBackspace ( l ) ;
break ;
case CTRL_D : /* ctrl-d, remove char at right of cursor, or if the
line is empty , act as end - of - file . */
if ( l - > len > 0 ) {
linenoiseEditDelete ( l ) ;
} else {
2023-11-22 11:58:43 +00:00
history_len - - ;
free ( history [ history_len ] ) ;
2023-11-22 20:59:51 +00:00
errno = ENOENT ;
2023-11-22 22:07:52 +00:00
return NULL ;
2023-11-22 20:59:51 +00:00
}
break ;
case CTRL_T : /* ctrl-t, swaps current character with previous. */
if ( l - > pos > 0 & & l - > pos < l - > len ) {
int aux = l - > buf [ l - > pos - 1 ] ;
l - > buf [ l - > pos - 1 ] = l - > buf [ l - > pos ] ;
l - > buf [ l - > pos ] = aux ;
if ( l - > pos ! = l - > len - 1 ) l - > pos + + ;
refreshLine ( l ) ;
}
break ;
case CTRL_B : /* ctrl-b */
linenoiseEditMoveLeft ( l ) ;
break ;
case CTRL_F : /* ctrl-f */
linenoiseEditMoveRight ( l ) ;
break ;
case CTRL_P : /* ctrl-p */
linenoiseEditHistoryNext ( l , LINENOISE_HISTORY_PREV ) ;
break ;
case CTRL_N : /* ctrl-n */
linenoiseEditHistoryNext ( l , LINENOISE_HISTORY_NEXT ) ;
break ;
case ESC : /* escape sequence */
/* Read the next two bytes representing the escape sequence.
* chars at different times . */
if ( fread ( seq , 1 , 1 , stdin ) = = - 1 ) break ;
if ( fread ( seq + 1 , 1 , 1 , stdin ) = = - 1 ) break ;
// or just if (fread(seq, 1, 2, stdin) < 2) break;
/* ESC [ sequences. */
if ( seq [ 0 ] = = ' [ ' ) {
if ( seq [ 1 ] > = ' 0 ' & & seq [ 1 ] < = ' 9 ' ) {
/* Extended escape, read additional byte. */
if ( fread ( seq + 2 , 1 , 1 , stdin ) = = - 1 ) break ;
if ( seq [ 2 ] = = ' ~ ' ) {
2023-11-22 11:58:43 +00:00
switch ( seq [ 1 ] ) {
2023-11-22 20:59:51 +00:00
case ' 3 ' : /* Delete key. */
linenoiseEditDelete ( l ) ;
2023-11-22 11:58:43 +00:00
break ;
}
}
2023-11-22 20:59:51 +00:00
} else {
2023-11-22 11:58:43 +00:00
switch ( seq [ 1 ] ) {
2023-11-22 20:59:51 +00:00
case ' A ' : /* Up */
linenoiseEditHistoryNext ( l , LINENOISE_HISTORY_PREV ) ;
break ;
case ' B ' : /* Down */
linenoiseEditHistoryNext ( l , LINENOISE_HISTORY_NEXT ) ;
break ;
case ' C ' : /* Right */
linenoiseEditMoveRight ( l ) ;
break ;
case ' D ' : /* Left */
linenoiseEditMoveLeft ( l ) ;
break ;
2023-11-22 11:58:43 +00:00
case ' H ' : /* Home */
2023-11-22 20:59:51 +00:00
linenoiseEditMoveHome ( l ) ;
2023-11-22 11:58:43 +00:00
break ;
case ' F ' : /* End*/
2023-11-22 20:59:51 +00:00
linenoiseEditMoveEnd ( l ) ;
2023-11-22 11:58:43 +00:00
break ;
}
}
}
2023-11-22 20:59:51 +00:00
/* ESC O sequences. */
else if ( seq [ 0 ] = = ' O ' ) {
switch ( seq [ 1 ] ) {
case ' H ' : /* Home */
linenoiseEditMoveHome ( l ) ;
break ;
case ' F ' : /* End*/
linenoiseEditMoveEnd ( l ) ;
break ;
}
}
break ;
default :
2023-11-22 22:07:52 +00:00
if ( linenoiseEditInsert ( l , c ) ) return NULL ;
2023-11-22 20:59:51 +00:00
break ;
case CTRL_U : /* Ctrl+u, delete the whole line. */
l - > buf [ 0 ] = ' \0 ' ;
l - > pos = l - > len = 0 ;
refreshLine ( l ) ;
break ;
case CTRL_K : /* Ctrl+k, delete from current to end of line. */
l - > buf [ l - > pos ] = ' \0 ' ;
l - > len = l - > pos ;
refreshLine ( l ) ;
break ;
case CTRL_A : /* Ctrl+a, go to the start of the line */
linenoiseEditMoveHome ( l ) ;
break ;
case CTRL_E : /* ctrl+e, go to the end of the line */
linenoiseEditMoveEnd ( l ) ;
break ;
case CTRL_L : /* ctrl+l, clear screen */
linenoiseClearScreen ( ) ;
refreshLine ( l ) ;
break ;
case CTRL_W : /* ctrl+w, delete previous word */
linenoiseEditDeletePrevWord ( l ) ;
break ;
2023-11-22 11:58:43 +00:00
}
2023-11-23 20:49:21 +00:00
flushWrite ( ) ;
2023-11-22 22:07:52 +00:00
return linenoiseEditMore ;
2023-11-22 11:58:43 +00:00
}
2023-11-22 20:59:51 +00:00
/* This is part of the multiplexed linenoise API. See linenoiseEditStart()
* for more information . This function is called when linenoiseEditFeed ( )
* returns something different than NULL . At this point the user input
* is in the buffer , and we can restore the terminal in normal mode . */
void linenoiseEditStop ( struct linenoiseState * l ) {
2023-11-23 20:49:21 +00:00
fputc ( ' \n ' , stdout ) ;
flushWrite ( ) ;
2023-11-22 20:59:51 +00:00
}
2023-11-22 14:50:57 +00:00
2023-11-22 20:59:51 +00:00
/* This just implements a blocking loop for the multiplexed API.
* In many applications that are not event - drivern , we can just call
* the blocking linenoise API , wait for the user to complete the editing
* and return the buffer . */
2023-11-22 22:07:52 +00:00
static char * linenoiseBlockingEdit ( char * buf , size_t buflen , const char * prompt )
2023-11-22 20:59:51 +00:00
{
2023-11-22 22:07:52 +00:00
struct linenoiseState l ;
2023-11-22 20:59:51 +00:00
/* Editing without a buffer is invalid. */
2023-11-22 16:08:14 +00:00
if ( buflen = = 0 ) {
errno = EINVAL ;
2023-11-22 22:07:52 +00:00
return NULL ;
2023-11-22 14:50:57 +00:00
}
2023-11-22 22:07:52 +00:00
char * res ;
2023-11-23 20:49:21 +00:00
linenoiseEditStart ( & l , buf , buflen , prompt ) ;
2023-11-22 22:07:52 +00:00
while ( ( res = linenoiseEditFeed ( & l ) ) = = linenoiseEditMore ) ;
linenoiseEditStop ( & l ) ;
return res ;
2023-11-22 16:08:14 +00:00
}
2023-11-22 14:50:57 +00:00
2023-11-23 20:49:21 +00:00
int linenoiseProbe ( ) {
/* Switch to non-blocking mode */
int stdin_fileno = fileno ( stdin ) ;
int flags = fcntl ( stdin_fileno , F_GETFL ) ;
flags | = O_NONBLOCK ;
int res = fcntl ( stdin_fileno , F_SETFL , flags ) ;
if ( res ! = 0 ) {
return - 1 ;
}
/* Device status request */
fprintf ( stdout , " \x1b [5n " ) ;
flushWrite ( ) ;
/* Try to read response */
2023-11-23 21:49:49 +00:00
int timeout_ms = 500 ;
2023-11-23 20:49:21 +00:00
const int retry_ms = 10 ;
size_t read_bytes = 0 ;
while ( timeout_ms > 0 & & read_bytes < 4 ) { // response is ESC[0n or ESC[3n
usleep ( retry_ms * 1000 ) ;
timeout_ms - = retry_ms ;
char c ;
int cb = read ( stdin_fileno , & c , 1 ) ;
if ( cb < 0 ) {
continue ;
}
2023-11-23 21:47:18 +00:00
if ( read_bytes = = 0 & & c ! = ' \x1b ' ) {
/* invalid response */
break ;
}
2023-11-23 20:49:21 +00:00
read_bytes + = cb ;
}
/* Restore old mode */
flags & = ~ O_NONBLOCK ;
res = fcntl ( stdin_fileno , F_SETFL , flags ) ;
if ( res ! = 0 ) {
return - 1 ;
}
if ( read_bytes < 4 ) {
return - 2 ;
}
return 0 ;
}
2023-11-22 16:08:14 +00:00
/* The high level function that is the main API of the linenoise library. */
char * linenoise ( const char * prompt ) {
2023-11-23 21:40:56 +00:00
char * buf = calloc ( 1 , max_cmdline_length ) ;
char * retval = linenoiseBlockingEdit ( buf , max_cmdline_length , prompt ) ;
free ( buf ) ;
2023-11-22 22:07:52 +00:00
return retval ;
2023-11-22 11:58:43 +00:00
}
/* This is just a wrapper the user may want to call in order to make sure
* the linenoise returned buffer is freed with the same allocator it was
* created with . Useful when the main program is using an alternative
* allocator . */
void linenoiseFree ( void * ptr ) {
2023-11-22 22:07:52 +00:00
if ( ptr = = linenoiseEditMore ) return ; // Protect from API misuse.
2023-11-22 11:58:43 +00:00
free ( ptr ) ;
}
/* ================================ History ================================= */
2023-11-22 17:43:11 +00:00
void linenoiseHistoryFree ( ) {
2023-11-22 11:58:43 +00:00
if ( history ) {
2023-11-22 16:08:14 +00:00
for ( int j = 0 ; j < history_len ; j + + ) {
2023-11-22 11:58:43 +00:00
free ( history [ j ] ) ;
2023-11-22 16:08:14 +00:00
}
2023-11-22 11:58:43 +00:00
free ( history ) ;
}
2023-11-22 16:08:14 +00:00
history = NULL ;
2023-11-22 11:58:43 +00:00
}
/* This is the API call to add a new entry in the linenoise history.
* It uses a fixed array of char pointers that are shifted ( memmoved )
* when the history max length is reached in order to remove the older
* entry and make room for the new one , so it is not exactly suitable for huge
* histories , but will work well for a few hundred of entries .
*
* Using a circular buffer is smarter , but a bit more complex to handle . */
int linenoiseHistoryAdd ( const char * line ) {
char * linecopy ;
if ( history_max_len = = 0 ) return 0 ;
/* Initialization on first call. */
if ( history = = NULL ) {
history = malloc ( sizeof ( char * ) * history_max_len ) ;
if ( history = = NULL ) return 0 ;
memset ( history , 0 , ( sizeof ( char * ) * history_max_len ) ) ;
}
/* Don't add duplicated lines. */
if ( history_len & & ! strcmp ( history [ history_len - 1 ] , line ) ) return 0 ;
/* Add an heap allocated copy of the line in the history.
* If we reached the max length , remove the older line . */
linecopy = strdup ( line ) ;
if ( ! linecopy ) return 0 ;
if ( history_len = = history_max_len ) {
free ( history [ 0 ] ) ;
memmove ( history , history + 1 , sizeof ( char * ) * ( history_max_len - 1 ) ) ;
history_len - - ;
}
history [ history_len ] = linecopy ;
history_len + + ;
return 1 ;
}
/* Set the maximum length for the history. This function can be called even
* if there is already some history , the function will make sure to retain
* just the latest ' len ' elements if the new history length value is smaller
* than the amount of items already inside the history . */
int linenoiseHistorySetMaxLen ( int len ) {
char * * new ;
if ( len < 1 ) return 0 ;
if ( history ) {
int tocopy = history_len ;
new = malloc ( sizeof ( char * ) * len ) ;
if ( new = = NULL ) return 0 ;
/* If we can't copy everything, free the elements we'll not use. */
if ( len < tocopy ) {
int j ;
for ( j = 0 ; j < tocopy - len ; j + + ) free ( history [ j ] ) ;
tocopy = len ;
}
memset ( new , 0 , sizeof ( char * ) * len ) ;
memcpy ( new , history + ( history_len - tocopy ) , sizeof ( char * ) * tocopy ) ;
free ( history ) ;
history = new ;
}
history_max_len = len ;
if ( history_len > history_max_len )
history_len = history_max_len ;
return 1 ;
}
/* Save the history in the specified file. On success 0 is returned
* otherwise - 1 is returned . */
int linenoiseHistorySave ( const char * filename ) {
FILE * fp ;
int j ;
fp = fopen ( filename , " w " ) ;
if ( fp = = NULL ) return - 1 ;
for ( j = 0 ; j < history_len ; j + + )
fprintf ( fp , " %s \n " , history [ j ] ) ;
fclose ( fp ) ;
return 0 ;
}
/* Load the history from the specified file. If the file does not exist
* zero is returned and no operation is performed .
*
* If the file exists and the operation succeeded 0 is returned , otherwise
* on error - 1 is returned . */
int linenoiseHistoryLoad ( const char * filename ) {
2023-11-23 21:40:56 +00:00
FILE * fp = fopen ( filename , " r " ) ;
if ( fp = = NULL ) {
return - 1 ;
}
char * buf = calloc ( 1 , max_cmdline_length ) ;
if ( buf = = NULL ) {
fclose ( fp ) ;
return - 1 ;
}
while ( fgets ( buf , max_cmdline_length , fp ) ! = NULL ) {
2023-11-22 11:58:43 +00:00
char * p ;
p = strchr ( buf , ' \r ' ) ;
if ( ! p ) p = strchr ( buf , ' \n ' ) ;
if ( p ) * p = ' \0 ' ;
linenoiseHistoryAdd ( buf ) ;
}
2023-11-23 21:40:56 +00:00
free ( buf ) ;
2023-11-22 11:58:43 +00:00
fclose ( fp ) ;
return 0 ;
}
2023-11-23 21:40:56 +00:00
/* Set line maximum length. If len parameter is smaller than
* LINENOISE_MINIMAL_MAX_LINE , - 1 is returned
* otherwise 0 is returned . */
int linenoiseSetMaxLineLen ( size_t len ) {
if ( len < LINENOISE_MINIMAL_MAX_LINE ) {
return - 1 ;
}
max_cmdline_length = len ;
return 0 ;
}