2000-03-19 05:28:55 +00:00
|
|
|
/* vi: set sw=4 ts=4: */
|
2000-03-16 08:09:57 +00:00
|
|
|
/*
|
2010-09-03 12:51:36 +02:00
|
|
|
* Command line editing.
|
2000-03-16 08:09:57 +00:00
|
|
|
*
|
2003-07-29 06:38:40 +00:00
|
|
|
* Copyright (c) 1986-2003 may safely be consumed by a BSD or GPL license.
|
2001-07-12 20:26:32 +00:00
|
|
|
* Written by: Vladimir Oleynik <dzo@simtreas.ru>
|
2001-02-16 18:36:04 +00:00
|
|
|
*
|
|
|
|
* Used ideas:
|
|
|
|
* Adam Rogoyski <rogoyski@cs.utexas.edu>
|
|
|
|
* Dave Cinege <dcinege@psychosis.com>
|
|
|
|
* Jakub Jelinek (c) 1995
|
2003-07-29 06:38:40 +00:00
|
|
|
* Erik Andersen <andersen@codepoet.org> (Majorly adjusted for busybox)
|
2000-03-16 08:09:57 +00:00
|
|
|
*
|
2001-02-16 18:36:04 +00:00
|
|
|
* This code is 'as is' with no warranty.
|
2000-03-16 08:09:57 +00:00
|
|
|
*/
|
|
|
|
/*
|
2008-06-28 21:03:43 +00:00
|
|
|
* Usage and known bugs:
|
2009-10-25 23:50:56 +01:00
|
|
|
* Terminal key codes are not extensive, more needs to be added.
|
|
|
|
* This version was created on Debian GNU/Linux 2.x.
|
2008-06-28 21:03:43 +00:00
|
|
|
* Delete, Backspace, Home, End, and the arrow keys were tested
|
|
|
|
* to work in an Xterm and console. Ctrl-A also works as Home.
|
|
|
|
* Ctrl-E also works as End.
|
|
|
|
*
|
2009-10-25 23:50:56 +01:00
|
|
|
* The following readline-like commands are not implemented:
|
|
|
|
* CTL-t -- Transpose two characters
|
|
|
|
*
|
2008-06-28 21:03:43 +00:00
|
|
|
* lineedit does not know that the terminal escape sequences do not
|
|
|
|
* take up space on the screen. The redisplay code assumes, unless
|
|
|
|
* told otherwise, that each character in the prompt is a printable
|
|
|
|
* character that takes up one character position on the screen.
|
|
|
|
* You need to tell lineedit that some sequences of characters
|
|
|
|
* in the prompt take up no screen space. Compatibly with readline,
|
|
|
|
* use the \[ escape to begin a sequence of non-printing characters,
|
|
|
|
* and the \] escape to signal the end of such a sequence. Example:
|
|
|
|
*
|
|
|
|
* PS1='\[\033[01;32m\]\u@\h\[\033[01;34m\] \w \$\[\033[00m\] '
|
2013-08-19 16:45:04 +02:00
|
|
|
*
|
|
|
|
* Unicode in PS1 is not fully supported: prompt length calulation is wrong,
|
|
|
|
* resulting in line wrap problems with long (multi-line) input.
|
2001-02-16 18:36:04 +00:00
|
|
|
*/
|
2015-05-29 11:31:40 +01:00
|
|
|
#include "busybox.h"
|
|
|
|
#include "NUM_APPLETS.h"
|
2009-07-11 21:36:13 +02:00
|
|
|
#include "unicode.h"
|
2011-05-20 04:26:29 +02:00
|
|
|
#ifndef _POSIX_VDISABLE
|
|
|
|
# define _POSIX_VDISABLE '\0'
|
|
|
|
#endif
|
|
|
|
|
2004-01-22 12:42:23 +00:00
|
|
|
|
2006-12-19 01:10:25 +00:00
|
|
|
#ifdef TEST
|
2009-07-10 18:40:49 +02:00
|
|
|
# define ENABLE_FEATURE_EDITING 0
|
|
|
|
# define ENABLE_FEATURE_TAB_COMPLETION 0
|
|
|
|
# define ENABLE_FEATURE_USERNAME_COMPLETION 0
|
|
|
|
#endif
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2001-02-20 06:42:29 +00:00
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
/* Entire file (except TESTing part) sits inside this #if */
|
2007-01-22 09:03:07 +00:00
|
|
|
#if ENABLE_FEATURE_EDITING
|
2000-03-16 08:09:57 +00:00
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
|
2020-12-16 11:14:08 +01:00
|
|
|
#if !ENABLE_SHELL_ASH && !ENABLE_SHELL_HUSH
|
|
|
|
/* so far only shells use these features */
|
|
|
|
# undef ENABLE_FEATURE_EDITING_FANCY_PROMPT
|
|
|
|
# undef ENABLE_FEATURE_TAB_COMPLETION
|
|
|
|
# undef ENABLE_FEATURE_USERNAME_COMPLETION
|
|
|
|
# define ENABLE_FEATURE_EDITING_FANCY_PROMPT 0
|
|
|
|
# define ENABLE_FEATURE_TAB_COMPLETION 0
|
|
|
|
# define ENABLE_FEATURE_USERNAME_COMPLETION 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2010-07-18 22:21:24 +02:00
|
|
|
#define ENABLE_USERNAME_OR_HOMEDIR \
|
2007-11-10 01:35:47 +00:00
|
|
|
(ENABLE_FEATURE_USERNAME_COMPLETION || ENABLE_FEATURE_EDITING_FANCY_PROMPT)
|
2010-07-18 22:21:24 +02:00
|
|
|
#if ENABLE_USERNAME_OR_HOMEDIR
|
|
|
|
# define IF_USERNAME_OR_HOMEDIR(...) __VA_ARGS__
|
2020-12-16 11:14:08 +01:00
|
|
|
#else
|
|
|
|
# define IF_USERNAME_OR_HOMEDIR(...) /*nothing*/
|
2007-11-10 01:35:47 +00:00
|
|
|
#endif
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2009-07-10 18:40:49 +02:00
|
|
|
|
|
|
|
#undef CHAR_T
|
2010-03-26 14:06:56 +01:00
|
|
|
#if ENABLE_UNICODE_SUPPORT
|
2010-04-29 13:43:39 +02:00
|
|
|
# define BB_NUL ((wchar_t)0)
|
2009-07-10 18:40:49 +02:00
|
|
|
# define CHAR_T wchar_t
|
2019-05-14 17:26:47 +02:00
|
|
|
static bool BB_isspace(CHAR_T c)
|
|
|
|
{
|
|
|
|
return ((unsigned)c < 256 && isspace(c));
|
|
|
|
}
|
2009-12-12 02:42:35 +01:00
|
|
|
# if ENABLE_FEATURE_EDITING_VI
|
2019-05-14 17:26:47 +02:00
|
|
|
static bool BB_isalnum_or_underscore(CHAR_T c)
|
|
|
|
{
|
2016-08-14 23:30:29 +02:00
|
|
|
return ((unsigned)c < 256 && isalnum(c)) || c == '_';
|
|
|
|
}
|
2009-12-12 02:42:35 +01:00
|
|
|
# endif
|
2019-05-14 17:26:47 +02:00
|
|
|
static bool BB_ispunct(CHAR_T c)
|
|
|
|
{
|
|
|
|
return ((unsigned)c < 256 && ispunct(c));
|
|
|
|
}
|
2009-07-10 18:40:49 +02:00
|
|
|
# undef isspace
|
|
|
|
# undef isalnum
|
|
|
|
# undef ispunct
|
|
|
|
# undef isprint
|
|
|
|
# define isspace isspace_must_not_be_used
|
|
|
|
# define isalnum isalnum_must_not_be_used
|
|
|
|
# define ispunct ispunct_must_not_be_used
|
|
|
|
# define isprint isprint_must_not_be_used
|
|
|
|
#else
|
|
|
|
# define BB_NUL '\0'
|
|
|
|
# define CHAR_T char
|
|
|
|
# define BB_isspace(c) isspace(c)
|
2016-08-14 23:30:29 +02:00
|
|
|
# if ENABLE_FEATURE_EDITING_VI
|
2017-12-31 17:30:02 +01:00
|
|
|
static bool BB_isalnum_or_underscore(CHAR_T c)
|
|
|
|
{
|
2020-06-24 09:31:30 +02:00
|
|
|
return isalnum(c) || c == '_';
|
2016-08-14 23:30:29 +02:00
|
|
|
}
|
|
|
|
# endif
|
2009-07-10 18:40:49 +02:00
|
|
|
# define BB_ispunct(c) ispunct(c)
|
|
|
|
#endif
|
2010-07-18 22:21:24 +02:00
|
|
|
#if ENABLE_UNICODE_PRESERVE_BROKEN
|
|
|
|
# define unicode_mark_raw_byte(wc) ((wc) | 0x20000000)
|
|
|
|
# define unicode_is_raw_byte(wc) ((wc) & 0x20000000)
|
|
|
|
#else
|
|
|
|
# define unicode_is_raw_byte(wc) 0
|
|
|
|
#endif
|
2009-07-10 18:40:49 +02:00
|
|
|
|
|
|
|
|
2010-10-28 21:34:56 +02:00
|
|
|
#define ESC "\033"
|
|
|
|
|
|
|
|
#define SEQ_CLEAR_TILL_END_OF_SCREEN ESC"[J"
|
|
|
|
//#define SEQ_CLEAR_TILL_END_OF_LINE ESC"[K"
|
2010-04-29 13:43:39 +02:00
|
|
|
|
|
|
|
|
2007-11-10 01:35:47 +00:00
|
|
|
enum {
|
|
|
|
MAX_LINELEN = CONFIG_FEATURE_EDITING_MAX_LEN < 0x7ff0
|
2008-01-07 16:13:14 +00:00
|
|
|
? CONFIG_FEATURE_EDITING_MAX_LEN
|
2007-11-10 01:35:47 +00:00
|
|
|
: 0x7ff0
|
|
|
|
};
|
|
|
|
|
2010-07-18 22:21:24 +02:00
|
|
|
#if ENABLE_USERNAME_OR_HOMEDIR
|
2007-11-10 01:35:47 +00:00
|
|
|
static const char null_str[] ALIGN1 = "";
|
|
|
|
#endif
|
2002-12-03 22:45:46 +00:00
|
|
|
|
2007-11-10 01:35:47 +00:00
|
|
|
/* We try to minimize both static and stack usage. */
|
2008-04-22 00:08:27 +00:00
|
|
|
struct lineedit_statics {
|
2007-11-10 01:35:47 +00:00
|
|
|
line_input_t *state;
|
2000-04-21 01:26:49 +00:00
|
|
|
|
2016-11-27 22:25:07 +01:00
|
|
|
unsigned cmdedit_termw; /* = 80; */ /* actual terminal width */
|
2000-05-20 00:40:08 +00:00
|
|
|
|
2009-05-17 16:44:54 +02:00
|
|
|
unsigned cmdedit_x; /* real x (col) terminal position */
|
|
|
|
unsigned cmdedit_y; /* pseudoreal y (row) terminal position */
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
unsigned cmdedit_prmt_len; /* on-screen length of last/sole prompt line */
|
2000-12-18 20:25:50 +00:00
|
|
|
|
2007-11-10 01:35:47 +00:00
|
|
|
unsigned cursor;
|
2009-09-29 00:00:12 +02:00
|
|
|
int command_len; /* must be signed */
|
|
|
|
/* signed maxsize: we want x in "if (x > S.maxsize)"
|
2009-07-12 02:50:35 +02:00
|
|
|
* to _not_ be promoted to unsigned */
|
|
|
|
int maxsize;
|
2009-07-10 18:40:49 +02:00
|
|
|
CHAR_T *command_ps;
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2007-11-10 01:35:47 +00:00
|
|
|
const char *cmdedit_prompt;
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
const char *prompt_last_line; /* last/sole prompt line */
|
2007-11-10 01:35:47 +00:00
|
|
|
|
2010-07-18 22:21:24 +02:00
|
|
|
#if ENABLE_USERNAME_OR_HOMEDIR
|
2007-11-10 01:35:47 +00:00
|
|
|
char *user_buf;
|
|
|
|
char *home_pwd_buf; /* = (char*)null_str; */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if ENABLE_FEATURE_TAB_COMPLETION
|
|
|
|
char **matches;
|
|
|
|
unsigned num_matches;
|
2001-02-16 18:36:04 +00:00
|
|
|
#endif
|
|
|
|
|
2018-02-25 20:09:54 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_WINCH
|
2016-11-27 22:25:07 +01:00
|
|
|
unsigned SIGWINCH_saved;
|
|
|
|
volatile unsigned SIGWINCH_count;
|
|
|
|
volatile smallint ok_to_redraw;
|
2018-02-25 20:09:54 +01:00
|
|
|
#endif
|
2016-11-27 22:25:07 +01:00
|
|
|
|
2007-11-10 01:35:47 +00:00
|
|
|
#if ENABLE_FEATURE_EDITING_VI
|
2010-07-18 22:21:24 +02:00
|
|
|
# define DELBUFSIZ 128
|
2007-11-10 01:35:47 +00:00
|
|
|
smallint newdelflag; /* whether delbuf should be reused yet */
|
2016-11-27 22:25:07 +01:00
|
|
|
CHAR_T *delptr;
|
2009-07-10 18:40:49 +02:00
|
|
|
CHAR_T delbuf[DELBUFSIZ]; /* a place to store deleted characters */
|
2007-11-10 01:35:47 +00:00
|
|
|
#endif
|
2009-10-27 10:34:06 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_ASK_TERMINAL
|
2009-10-27 10:47:49 +01:00
|
|
|
smallint sent_ESC_br6n;
|
2009-10-27 10:34:06 +01:00
|
|
|
#endif
|
2016-11-27 22:25:07 +01:00
|
|
|
|
2018-02-25 20:09:54 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_WINCH
|
2016-11-27 22:25:07 +01:00
|
|
|
/* Largish struct, keeping it last results in smaller code */
|
|
|
|
struct sigaction SIGWINCH_handler;
|
2018-02-25 20:09:54 +01:00
|
|
|
#endif
|
2007-11-10 01:35:47 +00:00
|
|
|
};
|
|
|
|
|
2008-04-22 00:08:27 +00:00
|
|
|
/* See lineedit_ptr_hack.c */
|
|
|
|
extern struct lineedit_statics *const lineedit_ptr_to_statics;
|
2007-11-10 01:35:47 +00:00
|
|
|
|
2008-04-22 00:08:27 +00:00
|
|
|
#define S (*lineedit_ptr_to_statics)
|
2007-11-10 01:35:47 +00:00
|
|
|
#define state (S.state )
|
|
|
|
#define cmdedit_termw (S.cmdedit_termw )
|
|
|
|
#define cmdedit_x (S.cmdedit_x )
|
|
|
|
#define cmdedit_y (S.cmdedit_y )
|
|
|
|
#define cmdedit_prmt_len (S.cmdedit_prmt_len)
|
|
|
|
#define cursor (S.cursor )
|
|
|
|
#define command_len (S.command_len )
|
|
|
|
#define command_ps (S.command_ps )
|
|
|
|
#define cmdedit_prompt (S.cmdedit_prompt )
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
#define prompt_last_line (S.prompt_last_line)
|
2007-12-24 14:09:19 +00:00
|
|
|
#define user_buf (S.user_buf )
|
2007-11-10 01:35:47 +00:00
|
|
|
#define home_pwd_buf (S.home_pwd_buf )
|
|
|
|
#define matches (S.matches )
|
|
|
|
#define num_matches (S.num_matches )
|
|
|
|
#define delptr (S.delptr )
|
|
|
|
#define newdelflag (S.newdelflag )
|
|
|
|
#define delbuf (S.delbuf )
|
|
|
|
|
|
|
|
#define INIT_S() do { \
|
2019-10-25 12:12:22 +02:00
|
|
|
(*(struct lineedit_statics**)not_const_pp(&lineedit_ptr_to_statics)) = xzalloc(sizeof(S)); \
|
2008-02-27 18:41:59 +00:00
|
|
|
barrier(); \
|
2007-11-10 01:35:47 +00:00
|
|
|
} while (0)
|
2011-05-20 04:26:29 +02:00
|
|
|
|
2007-11-10 01:35:47 +00:00
|
|
|
static void deinit_S(void)
|
|
|
|
{
|
|
|
|
#if ENABLE_FEATURE_EDITING_FANCY_PROMPT
|
|
|
|
/* This one is allocated only if FANCY_PROMPT is on
|
2010-09-03 12:51:36 +02:00
|
|
|
* (otherwise it points to verbatim prompt (NOT malloced)) */
|
2007-11-10 01:35:47 +00:00
|
|
|
free((char*)cmdedit_prompt);
|
|
|
|
#endif
|
2010-07-18 22:21:24 +02:00
|
|
|
#if ENABLE_USERNAME_OR_HOMEDIR
|
2007-11-10 01:35:47 +00:00
|
|
|
free(user_buf);
|
|
|
|
if (home_pwd_buf != null_str)
|
|
|
|
free(home_pwd_buf);
|
2007-01-21 19:18:19 +00:00
|
|
|
#endif
|
2008-04-22 00:08:27 +00:00
|
|
|
free(lineedit_ptr_to_statics);
|
2007-11-10 01:35:47 +00:00
|
|
|
}
|
|
|
|
#define DEINIT_S() deinit_S()
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2008-04-25 18:44:35 +00:00
|
|
|
|
2010-03-26 14:06:56 +01:00
|
|
|
#if ENABLE_UNICODE_SUPPORT
|
2011-07-11 07:36:59 +02:00
|
|
|
static size_t load_string(const char *src)
|
2009-07-10 18:40:49 +02:00
|
|
|
{
|
2011-03-27 01:18:07 +01:00
|
|
|
if (unicode_status == UNICODE_ON) {
|
2011-07-11 07:36:59 +02:00
|
|
|
ssize_t len = mbstowcs(command_ps, src, S.maxsize - 1);
|
2011-03-27 01:18:07 +01:00
|
|
|
if (len < 0)
|
|
|
|
len = 0;
|
|
|
|
command_ps[len] = BB_NUL;
|
|
|
|
return len;
|
|
|
|
} else {
|
|
|
|
unsigned i = 0;
|
2011-07-11 07:36:59 +02:00
|
|
|
while (src[i] && i < S.maxsize - 1) {
|
|
|
|
command_ps[i] = src[i];
|
2011-03-27 01:18:07 +01:00
|
|
|
i++;
|
2011-07-11 07:36:59 +02:00
|
|
|
}
|
|
|
|
command_ps[i] = BB_NUL;
|
2011-03-27 01:18:07 +01:00
|
|
|
return i;
|
|
|
|
}
|
2009-07-10 18:40:49 +02:00
|
|
|
}
|
2010-04-29 13:43:39 +02:00
|
|
|
static unsigned save_string(char *dst, unsigned maxsize)
|
2009-07-10 18:40:49 +02:00
|
|
|
{
|
2011-03-27 01:18:07 +01:00
|
|
|
if (unicode_status == UNICODE_ON) {
|
2010-05-11 14:49:13 +02:00
|
|
|
# if !ENABLE_UNICODE_PRESERVE_BROKEN
|
2011-03-27 01:18:07 +01:00
|
|
|
ssize_t len = wcstombs(dst, command_ps, maxsize - 1);
|
|
|
|
if (len < 0)
|
|
|
|
len = 0;
|
|
|
|
dst[len] = '\0';
|
|
|
|
return len;
|
2010-05-11 14:49:13 +02:00
|
|
|
# else
|
2011-03-27 01:18:07 +01:00
|
|
|
unsigned dstpos = 0;
|
|
|
|
unsigned srcpos = 0;
|
|
|
|
|
|
|
|
maxsize--;
|
|
|
|
while (dstpos < maxsize) {
|
|
|
|
wchar_t wc;
|
|
|
|
int n = srcpos;
|
2010-04-29 13:43:39 +02:00
|
|
|
|
2011-03-27 01:18:07 +01:00
|
|
|
/* Convert up to 1st invalid byte (or up to end) */
|
|
|
|
while ((wc = command_ps[srcpos]) != BB_NUL
|
|
|
|
&& !unicode_is_raw_byte(wc)
|
|
|
|
) {
|
|
|
|
srcpos++;
|
|
|
|
}
|
|
|
|
command_ps[srcpos] = BB_NUL;
|
|
|
|
n = wcstombs(dst + dstpos, command_ps + n, maxsize - dstpos);
|
|
|
|
if (n < 0) /* should not happen */
|
|
|
|
break;
|
|
|
|
dstpos += n;
|
|
|
|
if (wc == BB_NUL) /* usually is */
|
|
|
|
break;
|
2010-09-02 12:01:11 +02:00
|
|
|
|
2011-03-27 01:18:07 +01:00
|
|
|
/* We do have invalid byte here! */
|
|
|
|
command_ps[srcpos] = wc; /* restore it */
|
2010-04-29 13:43:39 +02:00
|
|
|
srcpos++;
|
2011-03-27 01:18:07 +01:00
|
|
|
if (dstpos == maxsize)
|
|
|
|
break;
|
|
|
|
dst[dstpos++] = (char) wc;
|
2010-04-29 13:43:39 +02:00
|
|
|
}
|
2011-03-27 01:18:07 +01:00
|
|
|
dst[dstpos] = '\0';
|
|
|
|
return dstpos;
|
2010-05-11 14:49:13 +02:00
|
|
|
# endif
|
2011-03-27 01:18:07 +01:00
|
|
|
} else {
|
|
|
|
unsigned i = 0;
|
|
|
|
while ((dst[i] = command_ps[i]) != 0)
|
|
|
|
i++;
|
|
|
|
return i;
|
|
|
|
}
|
2009-07-10 18:40:49 +02:00
|
|
|
}
|
|
|
|
/* I thought just fputwc(c, stdout) would work. But no... */
|
|
|
|
static void BB_PUTCHAR(wchar_t c)
|
|
|
|
{
|
2011-03-27 01:18:07 +01:00
|
|
|
if (unicode_status == UNICODE_ON) {
|
|
|
|
char buf[MB_CUR_MAX + 1];
|
|
|
|
mbstate_t mbst = { 0 };
|
|
|
|
ssize_t len = wcrtomb(buf, c, &mbst);
|
|
|
|
if (len > 0) {
|
|
|
|
buf[len] = '\0';
|
2021-02-03 20:47:14 +01:00
|
|
|
fputs_stdout(buf);
|
2011-03-27 01:18:07 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* In this case, c is always one byte */
|
|
|
|
putchar(c);
|
2009-07-10 18:40:49 +02:00
|
|
|
}
|
|
|
|
}
|
2010-05-16 20:46:53 +02:00
|
|
|
# if ENABLE_UNICODE_COMBINING_WCHARS || ENABLE_UNICODE_WIDE_WCHARS
|
|
|
|
static wchar_t adjust_width_and_validate_wc(unsigned *width_adj, wchar_t wc)
|
|
|
|
# else
|
|
|
|
static wchar_t adjust_width_and_validate_wc(wchar_t wc)
|
|
|
|
# define adjust_width_and_validate_wc(width_adj, wc) \
|
|
|
|
((*(width_adj))++, adjust_width_and_validate_wc(wc))
|
|
|
|
# endif
|
|
|
|
{
|
|
|
|
int w = 1;
|
|
|
|
|
|
|
|
if (unicode_status == UNICODE_ON) {
|
2010-05-16 21:15:03 +02:00
|
|
|
if (wc > CONFIG_LAST_SUPPORTED_WCHAR) {
|
|
|
|
/* note: also true for unicode_is_raw_byte(wc) */
|
2010-05-16 20:46:53 +02:00
|
|
|
goto subst;
|
|
|
|
}
|
|
|
|
w = wcwidth(wc);
|
|
|
|
if ((ENABLE_UNICODE_COMBINING_WCHARS && w < 0)
|
|
|
|
|| (!ENABLE_UNICODE_COMBINING_WCHARS && w <= 0)
|
|
|
|
|| (!ENABLE_UNICODE_WIDE_WCHARS && w > 1)
|
|
|
|
) {
|
|
|
|
subst:
|
|
|
|
w = 1;
|
|
|
|
wc = CONFIG_SUBST_WCHAR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
# if ENABLE_UNICODE_COMBINING_WCHARS || ENABLE_UNICODE_WIDE_WCHARS
|
|
|
|
*width_adj += w;
|
|
|
|
#endif
|
|
|
|
return wc;
|
|
|
|
}
|
|
|
|
#else /* !UNICODE */
|
2011-07-11 07:36:59 +02:00
|
|
|
static size_t load_string(const char *src)
|
2009-07-10 18:40:49 +02:00
|
|
|
{
|
2011-07-11 07:36:59 +02:00
|
|
|
safe_strncpy(command_ps, src, S.maxsize);
|
2009-07-10 18:40:49 +02:00
|
|
|
return strlen(command_ps);
|
|
|
|
}
|
2009-07-15 20:02:19 +02:00
|
|
|
# if ENABLE_FEATURE_TAB_COMPLETION
|
2010-04-29 13:43:39 +02:00
|
|
|
static void save_string(char *dst, unsigned maxsize)
|
2009-07-10 18:40:49 +02:00
|
|
|
{
|
|
|
|
safe_strncpy(dst, command_ps, maxsize);
|
|
|
|
}
|
2009-07-15 18:27:47 +02:00
|
|
|
# endif
|
|
|
|
# define BB_PUTCHAR(c) bb_putchar(c)
|
2010-05-16 20:46:53 +02:00
|
|
|
/* Should never be called: */
|
|
|
|
int adjust_width_and_validate_wc(unsigned *width_adj, int wc);
|
2009-07-10 18:40:49 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
/* Put 'command_ps[cursor]', cursor++.
|
|
|
|
* Advance cursor on screen. If we reached right margin, scroll text up
|
|
|
|
* and remove terminal margin effect by printing 'next_char' */
|
2008-04-25 18:44:35 +00:00
|
|
|
#define HACK_FOR_WRONG_WIDTH 1
|
2010-05-11 14:49:13 +02:00
|
|
|
static void put_cur_glyph_and_inc_cursor(void)
|
2001-02-16 18:36:04 +00:00
|
|
|
{
|
2009-07-10 18:40:49 +02:00
|
|
|
CHAR_T c = command_ps[cursor];
|
2010-05-16 20:46:53 +02:00
|
|
|
unsigned width = 0;
|
|
|
|
int ofs_to_right;
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2009-07-10 18:40:49 +02:00
|
|
|
if (c == BB_NUL) {
|
2007-01-21 19:18:19 +00:00
|
|
|
/* erase character after end of input string */
|
|
|
|
c = ' ';
|
2010-05-11 14:49:13 +02:00
|
|
|
} else {
|
|
|
|
/* advance cursor only if we aren't at the end yet */
|
|
|
|
cursor++;
|
2010-05-16 20:46:53 +02:00
|
|
|
if (unicode_status == UNICODE_ON) {
|
|
|
|
IF_UNICODE_WIDE_WCHARS(width = cmdedit_x;)
|
|
|
|
c = adjust_width_and_validate_wc(&cmdedit_x, c);
|
|
|
|
IF_UNICODE_WIDE_WCHARS(width = cmdedit_x - width;)
|
|
|
|
} else {
|
|
|
|
cmdedit_x++;
|
|
|
|
}
|
2007-01-21 19:18:19 +00:00
|
|
|
}
|
2010-05-11 14:49:13 +02:00
|
|
|
|
2010-05-16 20:46:53 +02:00
|
|
|
ofs_to_right = cmdedit_x - cmdedit_termw;
|
|
|
|
if (!ENABLE_UNICODE_WIDE_WCHARS || ofs_to_right <= 0) {
|
|
|
|
/* c fits on this line */
|
2009-07-10 18:40:49 +02:00
|
|
|
BB_PUTCHAR(c);
|
2006-12-21 22:27:10 +00:00
|
|
|
}
|
2010-05-16 20:46:53 +02:00
|
|
|
|
|
|
|
if (ofs_to_right >= 0) {
|
|
|
|
/* we go to the next line */
|
2008-04-25 18:44:35 +00:00
|
|
|
#if HACK_FOR_WRONG_WIDTH
|
|
|
|
/* This works better if our idea of term width is wrong
|
|
|
|
* and it is actually wider (often happens on serial lines).
|
|
|
|
* Printing CR,LF *forces* cursor to next line.
|
|
|
|
* OTOH if terminal width is correct AND terminal does NOT
|
|
|
|
* have automargin (IOW: it is moving cursor to next line
|
|
|
|
* by itself (which is wrong for VT-10x terminals)),
|
|
|
|
* this will break things: there will be one extra empty line */
|
|
|
|
puts("\r"); /* + implicit '\n' */
|
|
|
|
#else
|
2010-05-11 14:49:13 +02:00
|
|
|
/* VT-10x terminals don't wrap cursor to next line when last char
|
|
|
|
* on the line is printed - cursor stays "over" this char.
|
|
|
|
* Need to print _next_ char too (first one to appear on next line)
|
|
|
|
* to make cursor move down to next line.
|
|
|
|
*/
|
|
|
|
/* Works ok only if cmdedit_termw is correct. */
|
|
|
|
c = command_ps[cursor];
|
|
|
|
if (c == BB_NUL)
|
|
|
|
c = ' ';
|
|
|
|
BB_PUTCHAR(c);
|
2007-09-27 10:20:47 +00:00
|
|
|
bb_putchar('\b');
|
2008-04-25 18:44:35 +00:00
|
|
|
#endif
|
2010-05-16 20:46:53 +02:00
|
|
|
cmdedit_y++;
|
|
|
|
if (!ENABLE_UNICODE_WIDE_WCHARS || ofs_to_right == 0) {
|
|
|
|
width = 0;
|
|
|
|
} else { /* ofs_to_right > 0 */
|
|
|
|
/* wide char c didn't fit on prev line */
|
|
|
|
BB_PUTCHAR(c);
|
|
|
|
}
|
|
|
|
cmdedit_x = width;
|
2000-03-19 05:28:55 +00:00
|
|
|
}
|
2000-03-16 08:09:57 +00:00
|
|
|
}
|
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
/* Move to end of line (by printing all chars till the end) */
|
2010-05-11 14:49:13 +02:00
|
|
|
static void put_till_end_and_adv_cursor(void)
|
2001-02-16 18:36:04 +00:00
|
|
|
{
|
2007-01-22 07:21:38 +00:00
|
|
|
while (cursor < command_len)
|
2010-05-11 14:49:13 +02:00
|
|
|
put_cur_glyph_and_inc_cursor();
|
2001-01-26 20:42:23 +00:00
|
|
|
}
|
2000-07-04 06:22:18 +00:00
|
|
|
|
2001-01-26 20:42:23 +00:00
|
|
|
/* Go to the next line */
|
2001-02-16 18:36:04 +00:00
|
|
|
static void goto_new_line(void)
|
|
|
|
{
|
2010-05-11 14:49:13 +02:00
|
|
|
put_till_end_and_adv_cursor();
|
2018-04-06 17:58:21 +02:00
|
|
|
/* "cursor == 0" is only if prompt is "" and user input is empty */
|
|
|
|
if (cursor == 0 || cmdedit_x != 0)
|
2007-09-27 10:20:47 +00:00
|
|
|
bb_putchar('\n');
|
2001-01-26 20:42:23 +00:00
|
|
|
}
|
2000-07-04 06:22:18 +00:00
|
|
|
|
2006-08-29 19:41:06 +00:00
|
|
|
static void beep(void)
|
2001-02-16 18:36:04 +00:00
|
|
|
{
|
2007-09-27 10:20:47 +00:00
|
|
|
bb_putchar('\007');
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
2000-03-16 08:09:57 +00:00
|
|
|
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
/* Full or last/sole prompt line, reset edit cursor, calculate terminal cursor.
|
|
|
|
* cmdedit_y is always calculated for the last/sole prompt line.
|
|
|
|
*/
|
|
|
|
static void put_prompt_custom(bool is_full)
|
2010-05-17 00:45:44 +02:00
|
|
|
{
|
2021-02-03 20:47:14 +01:00
|
|
|
fputs_stdout((is_full ? cmdedit_prompt : prompt_last_line));
|
2010-05-17 00:45:44 +02:00
|
|
|
cursor = 0;
|
2016-11-27 22:25:07 +01:00
|
|
|
cmdedit_y = cmdedit_prmt_len / cmdedit_termw; /* new quasireal y */
|
|
|
|
cmdedit_x = cmdedit_prmt_len % cmdedit_termw;
|
2010-05-17 00:45:44 +02:00
|
|
|
}
|
|
|
|
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
#define put_prompt_last_line() put_prompt_custom(0)
|
|
|
|
#define put_prompt() put_prompt_custom(1)
|
|
|
|
|
2004-04-14 17:51:38 +00:00
|
|
|
/* Move back one character */
|
2007-01-21 19:18:59 +00:00
|
|
|
/* (optimized for slow terminals) */
|
|
|
|
static void input_backward(unsigned num)
|
2001-01-26 20:42:23 +00:00
|
|
|
{
|
2001-02-16 18:36:04 +00:00
|
|
|
if (num > cursor)
|
|
|
|
num = cursor;
|
2010-05-16 20:46:53 +02:00
|
|
|
if (num == 0)
|
2007-01-21 19:18:59 +00:00
|
|
|
return;
|
|
|
|
cursor -= num;
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2010-05-16 20:46:53 +02:00
|
|
|
if ((ENABLE_UNICODE_COMBINING_WCHARS || ENABLE_UNICODE_WIDE_WCHARS)
|
|
|
|
&& unicode_status == UNICODE_ON
|
|
|
|
) {
|
|
|
|
/* correct NUM to be equal to _screen_ width */
|
|
|
|
int n = num;
|
|
|
|
num = 0;
|
|
|
|
while (--n >= 0)
|
|
|
|
adjust_width_and_validate_wc(&num, command_ps[cursor + n]);
|
|
|
|
if (num == 0)
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2008-05-25 21:52:03 +00:00
|
|
|
if (cmdedit_x >= num) {
|
2001-02-16 18:36:04 +00:00
|
|
|
cmdedit_x -= num;
|
2007-01-21 19:18:59 +00:00
|
|
|
if (num <= 4) {
|
2008-02-18 22:28:03 +00:00
|
|
|
/* This is longer by 5 bytes on x86.
|
2008-05-25 21:52:03 +00:00
|
|
|
* Also gets miscompiled for ARM users
|
|
|
|
* (busybox.net/bugs/view.php?id=2274).
|
2008-02-18 22:28:03 +00:00
|
|
|
* printf(("\b\b\b\b" + 4) - num);
|
|
|
|
* return;
|
|
|
|
*/
|
|
|
|
do {
|
|
|
|
bb_putchar('\b');
|
|
|
|
} while (--num);
|
2007-01-21 19:18:59 +00:00
|
|
|
return;
|
|
|
|
}
|
2010-10-28 21:34:56 +02:00
|
|
|
printf(ESC"[%uD", num);
|
2007-01-21 19:18:59 +00:00
|
|
|
return;
|
2001-01-26 20:42:23 +00:00
|
|
|
}
|
2007-01-21 19:18:59 +00:00
|
|
|
|
|
|
|
/* Need to go one or more lines up */
|
2010-05-17 00:45:44 +02:00
|
|
|
if (ENABLE_UNICODE_WIDE_WCHARS) {
|
|
|
|
/* With wide chars, it is hard to "backtrack"
|
|
|
|
* and reliably figure out where to put cursor.
|
|
|
|
* Example (<> is a wide char; # is an ordinary char, _ cursor):
|
|
|
|
* |prompt: <><> |
|
|
|
|
* |<><><><><><> |
|
|
|
|
* |_ |
|
|
|
|
* and user presses left arrow. num = 1, cmdedit_x = 0,
|
|
|
|
* We need to go up one line, and then - how do we know that
|
|
|
|
* we need to go *10* positions to the right? Because
|
|
|
|
* |prompt: <>#<>|
|
|
|
|
* |<><><>#<><><>|
|
|
|
|
* |_ |
|
|
|
|
* in this situation we need to go *11* positions to the right.
|
|
|
|
*
|
|
|
|
* A simpler thing to do is to redraw everything from the start
|
|
|
|
* up to new cursor position (which is already known):
|
|
|
|
*/
|
|
|
|
unsigned sv_cursor;
|
2010-05-17 04:05:53 +02:00
|
|
|
/* go to 1st column; go up to first line */
|
2010-10-28 21:34:56 +02:00
|
|
|
printf("\r" ESC"[%uA", cmdedit_y);
|
2010-05-17 00:45:44 +02:00
|
|
|
cmdedit_y = 0;
|
|
|
|
sv_cursor = cursor;
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
put_prompt_last_line(); /* sets cursor to 0 */
|
2010-05-17 00:45:44 +02:00
|
|
|
while (cursor < sv_cursor)
|
|
|
|
put_cur_glyph_and_inc_cursor();
|
|
|
|
} else {
|
2010-05-17 12:30:44 +02:00
|
|
|
int lines_up;
|
|
|
|
/* num = chars to go back from the beginning of current line: */
|
2010-05-17 00:45:44 +02:00
|
|
|
num -= cmdedit_x;
|
2010-05-17 12:30:44 +02:00
|
|
|
/* num=1...w: one line up, w+1...2w: two, etc: */
|
2016-11-27 22:25:07 +01:00
|
|
|
lines_up = 1 + (num - 1) / cmdedit_termw;
|
|
|
|
cmdedit_x = (cmdedit_termw * cmdedit_y - num) % cmdedit_termw;
|
2010-05-17 12:30:44 +02:00
|
|
|
cmdedit_y -= lines_up;
|
|
|
|
/* go to 1st column; go up */
|
2010-10-28 21:34:56 +02:00
|
|
|
printf("\r" ESC"[%uA", lines_up);
|
2010-05-17 12:30:44 +02:00
|
|
|
/* go to correct column.
|
2010-05-17 12:33:13 +02:00
|
|
|
* xterm, konsole, Linux VT interpret 0 as 1 below! wow.
|
|
|
|
* need to *make sure* we skip it if cmdedit_x == 0 */
|
2010-05-17 12:30:44 +02:00
|
|
|
if (cmdedit_x)
|
2010-10-28 21:34:56 +02:00
|
|
|
printf(ESC"[%uC", cmdedit_x);
|
2008-05-25 21:52:03 +00:00
|
|
|
}
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
2000-03-16 08:09:57 +00:00
|
|
|
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
/* See redraw and draw_full below */
|
|
|
|
static void draw_custom(int y, int back_cursor, bool is_full)
|
2001-02-16 18:36:04 +00:00
|
|
|
{
|
2010-05-17 04:05:53 +02:00
|
|
|
if (y > 0) /* up y lines */
|
2010-10-28 21:34:56 +02:00
|
|
|
printf(ESC"[%uA", y);
|
2007-09-27 10:20:47 +00:00
|
|
|
bb_putchar('\r');
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
put_prompt_custom(is_full);
|
2010-05-11 14:49:13 +02:00
|
|
|
put_till_end_and_adv_cursor();
|
|
|
|
printf(SEQ_CLEAR_TILL_END_OF_SCREEN);
|
2001-02-16 18:36:04 +00:00
|
|
|
input_backward(back_cursor);
|
2000-03-16 08:09:57 +00:00
|
|
|
}
|
|
|
|
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
/* Move y lines up, draw last/sole prompt line, editor line[s], and clear tail.
|
|
|
|
* goal: redraw the prompt+input+cursor in-place, overwriting the previous */
|
|
|
|
#define redraw(y, back_cursor) draw_custom((y), (back_cursor), 0)
|
|
|
|
|
|
|
|
/* Like above, but without moving up, and while using all the prompt lines.
|
|
|
|
* goal: draw a full prompt+input+cursor unrelated to a previous position.
|
|
|
|
* note: cmdedit_y always ends up relating to the last/sole prompt line */
|
|
|
|
#define draw_full(back_cursor) draw_custom(0, (back_cursor), 1)
|
|
|
|
|
2005-08-04 19:04:46 +00:00
|
|
|
/* Delete the char in front of the cursor, optionally saving it
|
|
|
|
* for later putback */
|
2008-03-17 09:04:04 +00:00
|
|
|
#if !ENABLE_FEATURE_EDITING_VI
|
|
|
|
static void input_delete(void)
|
|
|
|
#define input_delete(save) input_delete()
|
|
|
|
#else
|
2005-08-04 19:04:46 +00:00
|
|
|
static void input_delete(int save)
|
2008-03-17 09:04:04 +00:00
|
|
|
#endif
|
2000-04-12 17:49:52 +00:00
|
|
|
{
|
2001-01-26 20:42:23 +00:00
|
|
|
int j = cursor;
|
2000-04-09 18:27:46 +00:00
|
|
|
|
2008-05-13 02:27:31 +00:00
|
|
|
if (j == (int)command_len)
|
2000-04-12 17:49:52 +00:00
|
|
|
return;
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2007-01-22 09:03:07 +00:00
|
|
|
#if ENABLE_FEATURE_EDITING_VI
|
2005-08-04 19:04:46 +00:00
|
|
|
if (save) {
|
|
|
|
if (newdelflag) {
|
2007-11-10 01:35:47 +00:00
|
|
|
delptr = delbuf;
|
2005-08-04 19:04:46 +00:00
|
|
|
newdelflag = 0;
|
|
|
|
}
|
2007-11-10 01:35:47 +00:00
|
|
|
if ((delptr - delbuf) < DELBUFSIZ)
|
|
|
|
*delptr++ = command_ps[j];
|
2005-08-04 19:04:46 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2009-07-10 18:40:49 +02:00
|
|
|
memmove(command_ps + j, command_ps + j + 1,
|
2009-07-16 14:33:16 +02:00
|
|
|
/* (command_len + 1 [because of NUL]) - (j + 1)
|
|
|
|
* simplified into (command_len - j) */
|
|
|
|
(command_len - j) * sizeof(command_ps[0]));
|
2007-01-22 07:21:38 +00:00
|
|
|
command_len--;
|
2010-05-11 14:49:13 +02:00
|
|
|
put_till_end_and_adv_cursor();
|
|
|
|
/* Last char is still visible, erase it (and more) */
|
|
|
|
printf(SEQ_CLEAR_TILL_END_OF_SCREEN);
|
2003-07-28 09:56:35 +00:00
|
|
|
input_backward(cursor - j); /* back to old pos cursor */
|
2001-01-26 20:42:23 +00:00
|
|
|
}
|
|
|
|
|
2007-01-22 09:03:07 +00:00
|
|
|
#if ENABLE_FEATURE_EDITING_VI
|
2005-08-04 19:04:46 +00:00
|
|
|
static void put(void)
|
|
|
|
{
|
2007-01-21 19:18:19 +00:00
|
|
|
int ocursor;
|
2007-11-10 01:35:47 +00:00
|
|
|
int j = delptr - delbuf;
|
2007-01-21 19:18:19 +00:00
|
|
|
|
2005-08-04 19:04:46 +00:00
|
|
|
if (j == 0)
|
|
|
|
return;
|
|
|
|
ocursor = cursor;
|
|
|
|
/* open hole and then fill it */
|
2009-07-10 18:40:49 +02:00
|
|
|
memmove(command_ps + cursor + j, command_ps + cursor,
|
|
|
|
(command_len - cursor + 1) * sizeof(command_ps[0]));
|
|
|
|
memcpy(command_ps + cursor, delbuf, j * sizeof(command_ps[0]));
|
2007-01-22 08:34:44 +00:00
|
|
|
command_len += j;
|
2010-05-11 14:49:13 +02:00
|
|
|
put_till_end_and_adv_cursor();
|
2006-12-21 22:27:10 +00:00
|
|
|
input_backward(cursor - ocursor - j + 1); /* at end of new text */
|
2005-08-04 19:04:46 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2001-01-26 20:42:23 +00:00
|
|
|
/* Delete the char in back of the cursor */
|
|
|
|
static void input_backspace(void)
|
|
|
|
{
|
|
|
|
if (cursor > 0) {
|
2001-02-16 18:36:04 +00:00
|
|
|
input_backward(1);
|
2005-08-04 19:04:46 +00:00
|
|
|
input_delete(0);
|
2000-04-12 17:49:52 +00:00
|
|
|
}
|
2001-01-26 20:42:23 +00:00
|
|
|
}
|
2000-04-12 17:49:52 +00:00
|
|
|
|
2004-04-14 17:51:38 +00:00
|
|
|
/* Move forward one character */
|
2001-01-26 20:42:23 +00:00
|
|
|
static void input_forward(void)
|
|
|
|
{
|
2007-01-22 07:21:38 +00:00
|
|
|
if (cursor < command_len)
|
2010-05-11 14:49:13 +02:00
|
|
|
put_cur_glyph_and_inc_cursor();
|
2000-04-12 17:49:52 +00:00
|
|
|
}
|
|
|
|
|
2007-01-22 09:03:07 +00:00
|
|
|
#if ENABLE_FEATURE_TAB_COMPLETION
|
2001-01-26 20:42:23 +00:00
|
|
|
|
2010-11-22 03:49:18 +01:00
|
|
|
//FIXME:
|
|
|
|
//needs to be more clever: currently it thinks that "foo\ b<TAB>
|
|
|
|
//matches the file named "foo bar", which is untrue.
|
|
|
|
//Also, perhaps "foo b<TAB> needs to complete to "foo bar" <cursor>,
|
|
|
|
//not "foo bar <cursor>...
|
|
|
|
|
2007-01-21 19:19:46 +00:00
|
|
|
static void free_tab_completion_data(void)
|
|
|
|
{
|
|
|
|
if (matches) {
|
|
|
|
while (num_matches)
|
|
|
|
free(matches[--num_matches]);
|
|
|
|
free(matches);
|
|
|
|
matches = NULL;
|
|
|
|
}
|
|
|
|
}
|
2006-01-25 11:53:47 +00:00
|
|
|
|
2006-12-21 22:24:46 +00:00
|
|
|
static void add_match(char *matched)
|
2006-01-25 11:53:47 +00:00
|
|
|
{
|
2017-11-07 18:09:29 +01:00
|
|
|
unsigned char *p = (unsigned char*)matched;
|
|
|
|
while (*p) {
|
|
|
|
/* ESC attack fix: drop any string with control chars */
|
|
|
|
if (*p < ' '
|
|
|
|
|| (!ENABLE_UNICODE_SUPPORT && *p >= 0x7f)
|
|
|
|
|| (ENABLE_UNICODE_SUPPORT && *p == 0x7f)
|
|
|
|
) {
|
|
|
|
free(matched);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
p++;
|
|
|
|
}
|
2008-07-08 05:14:36 +00:00
|
|
|
matches = xrealloc_vector(matches, 4, num_matches);
|
|
|
|
matches[num_matches] = matched;
|
2006-01-25 11:53:47 +00:00
|
|
|
num_matches++;
|
|
|
|
}
|
|
|
|
|
2010-11-22 03:49:18 +01:00
|
|
|
# if ENABLE_FEATURE_USERNAME_COMPLETION
|
2010-09-02 12:01:11 +02:00
|
|
|
/* Replace "~user/..." with "/homedir/...".
|
|
|
|
* The parameter is malloced, free it or return it
|
|
|
|
* unchanged if no user is matched.
|
|
|
|
*/
|
|
|
|
static char *username_path_completion(char *ud)
|
2000-03-17 01:12:41 +00:00
|
|
|
{
|
2001-02-16 18:36:04 +00:00
|
|
|
struct passwd *entry;
|
2010-09-02 12:01:11 +02:00
|
|
|
char *tilde_name = ud;
|
|
|
|
char *home = NULL;
|
|
|
|
|
|
|
|
ud++; /* skip ~ */
|
|
|
|
if (*ud == '/') { /* "~/..." */
|
|
|
|
home = home_pwd_buf;
|
|
|
|
} else {
|
|
|
|
/* "~user/..." */
|
|
|
|
ud = strchr(ud, '/');
|
|
|
|
*ud = '\0'; /* "~user" */
|
|
|
|
entry = getpwnam(tilde_name + 1);
|
|
|
|
*ud = '/'; /* restore "~user/..." */
|
|
|
|
if (entry)
|
|
|
|
home = entry->pw_dir;
|
|
|
|
}
|
|
|
|
if (home) {
|
|
|
|
ud = concat_path_file(home, ud);
|
|
|
|
free(tilde_name);
|
|
|
|
tilde_name = ud;
|
|
|
|
}
|
|
|
|
return tilde_name;
|
|
|
|
}
|
|
|
|
|
2010-09-03 12:56:36 +02:00
|
|
|
/* ~use<tab> - find all users with this prefix.
|
|
|
|
* Return the length of the prefix used for matching.
|
|
|
|
*/
|
|
|
|
static NOINLINE unsigned complete_username(const char *ud)
|
2010-09-02 12:01:11 +02:00
|
|
|
{
|
2015-02-07 21:21:02 +01:00
|
|
|
struct passwd *pw;
|
2010-09-03 12:56:36 +02:00
|
|
|
unsigned userlen;
|
2001-01-26 20:42:23 +00:00
|
|
|
|
2010-09-02 12:01:11 +02:00
|
|
|
ud++; /* skip ~ */
|
2001-02-16 18:36:04 +00:00
|
|
|
userlen = strlen(ud);
|
2001-01-26 20:42:23 +00:00
|
|
|
|
2010-09-02 12:01:11 +02:00
|
|
|
setpwent();
|
2015-02-07 21:21:02 +01:00
|
|
|
while ((pw = getpwent()) != NULL) {
|
2010-09-02 12:01:11 +02:00
|
|
|
/* Null usernames should result in all users as possible completions. */
|
2015-03-12 17:48:34 +01:00
|
|
|
if (/* !ud[0] || */ is_prefixed_with(pw->pw_name, ud)) {
|
2015-02-07 21:21:02 +01:00
|
|
|
add_match(xasprintf("~%s/", pw->pw_name));
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
|
|
|
}
|
2015-02-07 21:21:02 +01:00
|
|
|
endpwent(); /* don't keep password file open */
|
2010-09-03 12:56:36 +02:00
|
|
|
|
|
|
|
return 1 + userlen;
|
2000-03-17 01:12:41 +00:00
|
|
|
}
|
2010-11-22 03:49:18 +01:00
|
|
|
# endif /* FEATURE_USERNAME_COMPLETION */
|
2001-01-26 20:42:23 +00:00
|
|
|
|
|
|
|
enum {
|
2001-02-16 18:36:04 +00:00
|
|
|
FIND_EXE_ONLY = 0,
|
|
|
|
FIND_DIR_ONLY = 1,
|
2001-01-26 20:42:23 +00:00
|
|
|
FIND_FILE_ONLY = 2,
|
|
|
|
};
|
2000-03-19 10:46:06 +00:00
|
|
|
|
2020-12-16 10:59:20 +01:00
|
|
|
static unsigned path_parse(char ***p)
|
2001-01-26 20:42:23 +00:00
|
|
|
{
|
2020-12-16 10:59:20 +01:00
|
|
|
unsigned npth;
|
2007-01-22 07:21:38 +00:00
|
|
|
const char *pth;
|
2007-01-22 08:34:44 +00:00
|
|
|
char *tmp;
|
2007-01-22 07:21:38 +00:00
|
|
|
char **res;
|
2001-01-26 20:42:23 +00:00
|
|
|
|
2020-12-16 11:14:08 +01:00
|
|
|
# if EDITING_HAS_path_lookup
|
2007-01-22 07:21:38 +00:00
|
|
|
if (state->flags & WITH_PATH_LOOKUP)
|
|
|
|
pth = state->path_lookup;
|
|
|
|
else
|
2020-12-16 11:14:08 +01:00
|
|
|
# endif
|
2007-01-22 07:21:38 +00:00
|
|
|
pth = getenv("PATH");
|
2010-09-02 12:01:11 +02:00
|
|
|
|
|
|
|
/* PATH="" or PATH=":"? */
|
2006-12-21 22:27:10 +00:00
|
|
|
if (!pth || !pth[0] || LONE_CHAR(pth, ':'))
|
2001-01-26 20:42:23 +00:00
|
|
|
return 1;
|
|
|
|
|
2007-01-22 08:34:44 +00:00
|
|
|
tmp = (char*)pth;
|
2020-12-16 10:59:20 +01:00
|
|
|
npth = 1; /* path component count */
|
2006-12-21 22:27:10 +00:00
|
|
|
while (1) {
|
2001-01-26 20:42:23 +00:00
|
|
|
tmp = strchr(tmp, ':');
|
2006-12-21 22:27:10 +00:00
|
|
|
if (!tmp)
|
2001-01-26 20:42:23 +00:00
|
|
|
break;
|
2010-09-02 12:01:11 +02:00
|
|
|
tmp++;
|
2007-01-22 07:21:38 +00:00
|
|
|
npth++;
|
2001-01-26 20:42:23 +00:00
|
|
|
}
|
|
|
|
|
2020-12-16 10:59:20 +01:00
|
|
|
*p = res = xzalloc((npth + 1) * sizeof(res[0]));
|
2007-01-22 08:34:44 +00:00
|
|
|
res[0] = tmp = xstrdup(pth);
|
2007-01-22 07:21:38 +00:00
|
|
|
npth = 1;
|
2006-12-21 22:27:10 +00:00
|
|
|
while (1) {
|
2001-01-26 20:42:23 +00:00
|
|
|
tmp = strchr(tmp, ':');
|
2006-12-21 22:27:10 +00:00
|
|
|
if (!tmp)
|
2001-01-26 20:42:23 +00:00
|
|
|
break;
|
2007-01-22 07:21:38 +00:00
|
|
|
*tmp++ = '\0'; /* ':' -> '\0' */
|
|
|
|
res[npth++] = tmp;
|
2001-01-26 20:42:23 +00:00
|
|
|
}
|
2020-12-16 10:59:20 +01:00
|
|
|
/* special case: "match subdirectories of the current directory" */
|
|
|
|
/*res[npth++] = NULL; - filled by xzalloc() */
|
2001-01-26 20:42:23 +00:00
|
|
|
return npth;
|
|
|
|
}
|
|
|
|
|
2010-09-03 12:56:36 +02:00
|
|
|
/* Complete command, directory or file name.
|
|
|
|
* Return the length of the prefix used for matching.
|
|
|
|
*/
|
|
|
|
static NOINLINE unsigned complete_cmd_dir_file(const char *command, int type)
|
2001-02-16 18:36:04 +00:00
|
|
|
{
|
|
|
|
char *path1[1];
|
|
|
|
char **paths = path1;
|
2020-12-16 10:59:20 +01:00
|
|
|
unsigned npaths;
|
|
|
|
unsigned i;
|
|
|
|
unsigned baselen;
|
|
|
|
const char *basecmd;
|
2010-09-02 12:01:11 +02:00
|
|
|
char *dirbuf = NULL;
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
npaths = 1;
|
2007-01-29 22:51:58 +00:00
|
|
|
path1[0] = (char*)".";
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2020-12-16 10:59:20 +01:00
|
|
|
basecmd = strrchr(command, '/');
|
|
|
|
if (!basecmd) {
|
2010-09-02 12:01:11 +02:00
|
|
|
if (type == FIND_EXE_ONLY)
|
|
|
|
npaths = path_parse(&paths);
|
2020-12-16 10:59:20 +01:00
|
|
|
basecmd = command;
|
2001-01-26 20:42:23 +00:00
|
|
|
} else {
|
2010-09-02 12:01:11 +02:00
|
|
|
/* point to 'l' in "..../last_component" */
|
2020-12-16 10:59:20 +01:00
|
|
|
basecmd++;
|
2007-01-21 19:18:19 +00:00
|
|
|
/* dirbuf = ".../.../.../" */
|
2020-12-16 10:59:20 +01:00
|
|
|
dirbuf = xstrndup(command, basecmd - command);
|
2010-11-22 03:49:18 +01:00
|
|
|
# if ENABLE_FEATURE_USERNAME_COMPLETION
|
2003-07-28 09:56:35 +00:00
|
|
|
if (dirbuf[0] == '~') /* ~/... or ~user/... */
|
2010-09-02 12:01:11 +02:00
|
|
|
dirbuf = username_path_completion(dirbuf);
|
2010-11-22 03:49:18 +01:00
|
|
|
# endif
|
2010-09-02 12:44:39 +02:00
|
|
|
path1[0] = dirbuf;
|
2001-01-26 20:42:23 +00:00
|
|
|
}
|
2020-12-16 10:59:20 +01:00
|
|
|
baselen = strlen(basecmd);
|
2000-03-19 10:46:06 +00:00
|
|
|
|
2016-10-24 01:25:05 +02:00
|
|
|
if (type == FIND_EXE_ONLY && !dirbuf) {
|
2020-01-21 16:01:58 +00:00
|
|
|
# if ENABLE_FEATURE_SH_STANDALONE && NUM_APPLETS != 1
|
2018-11-17 17:48:14 +00:00
|
|
|
const char *p = applet_names;
|
2016-04-08 11:57:20 +01:00
|
|
|
while (*p) {
|
2020-12-16 10:59:20 +01:00
|
|
|
if (strncmp(basecmd, p, baselen) == 0)
|
2015-05-29 11:31:40 +01:00
|
|
|
add_match(xstrdup(p));
|
2018-11-17 17:48:14 +00:00
|
|
|
while (*p++ != '\0')
|
2016-04-08 11:57:20 +01:00
|
|
|
continue;
|
2015-05-29 11:31:40 +01:00
|
|
|
}
|
2017-07-29 00:59:24 +02:00
|
|
|
# endif
|
2020-01-21 16:01:58 +00:00
|
|
|
# if EDITING_HAS_get_exe_name
|
|
|
|
if (state->get_exe_name) {
|
|
|
|
i = 0;
|
|
|
|
for (;;) {
|
|
|
|
const char *b = state->get_exe_name(i++);
|
|
|
|
if (!b)
|
|
|
|
break;
|
2020-12-16 10:59:20 +01:00
|
|
|
if (strncmp(basecmd, b, baselen) == 0)
|
2020-01-21 16:01:58 +00:00
|
|
|
add_match(xstrdup(b));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
# endif
|
|
|
|
}
|
2015-05-29 11:31:40 +01:00
|
|
|
|
2001-02-16 18:36:04 +00:00
|
|
|
for (i = 0; i < npaths; i++) {
|
2010-09-02 12:01:11 +02:00
|
|
|
DIR *dir;
|
|
|
|
struct dirent *next;
|
|
|
|
struct stat st;
|
|
|
|
char *found;
|
2021-02-18 09:50:29 +00:00
|
|
|
const char *lpath;
|
2010-09-02 12:01:11 +02:00
|
|
|
|
2020-12-16 10:59:20 +01:00
|
|
|
if (paths[i] == NULL) { /* path_parse()'s last component? */
|
|
|
|
/* in PATH completion, current dir's subdir names
|
|
|
|
* can be completions (but only subdirs, not files).
|
|
|
|
*/
|
2020-12-11 12:34:21 +00:00
|
|
|
type = FIND_DIR_ONLY;
|
|
|
|
paths[i] = (char *)".";
|
|
|
|
}
|
|
|
|
|
2021-02-18 09:50:29 +00:00
|
|
|
lpath = *paths[i] ? paths[i] : ".";
|
|
|
|
dir = opendir(lpath);
|
2008-04-24 04:42:52 +00:00
|
|
|
if (!dir)
|
|
|
|
continue; /* don't print an error */
|
2001-02-16 18:36:04 +00:00
|
|
|
|
|
|
|
while ((next = readdir(dir)) != NULL) {
|
2010-09-03 14:11:08 +02:00
|
|
|
unsigned len;
|
2010-09-02 12:01:11 +02:00
|
|
|
const char *name_found = next->d_name;
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2010-09-02 12:01:11 +02:00
|
|
|
/* .../<tab>: bash 3.2.0 shows dotfiles, but not . and .. */
|
2020-12-16 10:59:20 +01:00
|
|
|
if (!basecmd[0] && DOT_OR_DOTDOT(name_found))
|
2001-01-26 20:42:23 +00:00
|
|
|
continue;
|
2010-09-02 12:01:11 +02:00
|
|
|
/* match? */
|
2020-12-16 10:59:20 +01:00
|
|
|
if (strncmp(basecmd, name_found, baselen) != 0)
|
2010-09-02 12:01:11 +02:00
|
|
|
continue; /* no */
|
|
|
|
|
2021-02-18 09:50:29 +00:00
|
|
|
found = concat_path_file(lpath, name_found);
|
2008-04-24 04:42:52 +00:00
|
|
|
/* NB: stat() first so that we see is it a directory;
|
|
|
|
* but if that fails, use lstat() so that
|
|
|
|
* we still match dangling links */
|
|
|
|
if (stat(found, &st) && lstat(found, &st))
|
2010-09-02 12:01:11 +02:00
|
|
|
goto cont; /* hmm, remove in progress? */
|
2006-12-21 22:24:46 +00:00
|
|
|
|
2010-09-03 14:11:08 +02:00
|
|
|
/* Save only name */
|
|
|
|
len = strlen(name_found);
|
|
|
|
found = xrealloc(found, len + 2); /* +2: for slash and NUL */
|
|
|
|
strcpy(found, name_found);
|
2006-12-21 22:24:46 +00:00
|
|
|
|
2001-01-26 20:42:23 +00:00
|
|
|
if (S_ISDIR(st.st_mode)) {
|
2020-12-10 14:44:57 +00:00
|
|
|
/* skip directories if searching PATH */
|
|
|
|
if (type == FIND_EXE_ONLY && !dirbuf)
|
|
|
|
goto cont;
|
2010-09-03 14:11:08 +02:00
|
|
|
/* name is a directory, add slash */
|
|
|
|
found[len] = '/';
|
|
|
|
found[len + 1] = '\0';
|
2001-01-26 20:42:23 +00:00
|
|
|
} else {
|
2010-09-02 12:01:11 +02:00
|
|
|
/* skip files if looking for dirs only (example: cd) */
|
2003-07-28 09:56:35 +00:00
|
|
|
if (type == FIND_DIR_ONLY)
|
2001-04-09 22:48:12 +00:00
|
|
|
goto cont;
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
2010-09-02 12:01:11 +02:00
|
|
|
/* add it to the list */
|
2006-12-21 22:24:46 +00:00
|
|
|
add_match(found);
|
2006-01-25 11:53:47 +00:00
|
|
|
continue;
|
2006-12-21 22:27:10 +00:00
|
|
|
cont:
|
2001-04-09 22:48:12 +00:00
|
|
|
free(found);
|
2000-03-19 10:46:06 +00:00
|
|
|
}
|
2001-02-16 18:36:04 +00:00
|
|
|
closedir(dir);
|
2010-09-02 12:01:11 +02:00
|
|
|
} /* for every path */
|
|
|
|
|
2001-02-16 18:36:04 +00:00
|
|
|
if (paths != path1) {
|
2008-04-24 04:42:52 +00:00
|
|
|
free(paths[0]); /* allocated memory is only in first member */
|
2001-02-16 18:36:04 +00:00
|
|
|
free(paths);
|
2000-03-19 10:46:06 +00:00
|
|
|
}
|
2010-09-03 14:11:08 +02:00
|
|
|
free(dirbuf);
|
2010-09-03 12:56:36 +02:00
|
|
|
|
2020-12-16 10:59:20 +01:00
|
|
|
return baselen;
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
2001-01-26 20:42:23 +00:00
|
|
|
|
2010-09-03 12:51:36 +02:00
|
|
|
/* build_match_prefix:
|
2010-09-03 14:09:24 +02:00
|
|
|
* On entry, match_buf contains everything up to cursor at the moment <tab>
|
2010-09-03 12:51:36 +02:00
|
|
|
* was pressed. This function looks at it, figures out what part of it
|
|
|
|
* constitutes the command/file/directory prefix to use for completion,
|
2010-09-03 14:09:24 +02:00
|
|
|
* and rewrites match_buf to contain only that part.
|
2010-09-03 12:51:36 +02:00
|
|
|
*/
|
2010-09-03 14:09:24 +02:00
|
|
|
#define dbg_bmp 0
|
2010-09-03 12:51:36 +02:00
|
|
|
/* Helpers: */
|
2009-07-16 02:19:39 +02:00
|
|
|
/* QUOT is used on elements of int_buf[], which are bytes,
|
|
|
|
* not Unicode chars. Therefore it works correctly even in Unicode mode.
|
|
|
|
*/
|
2006-12-21 22:27:10 +00:00
|
|
|
#define QUOT (UCHAR_MAX+1)
|
2010-09-03 13:02:47 +02:00
|
|
|
static void remove_chunk(int16_t *int_buf, int beg, int end)
|
2009-07-16 02:19:39 +02:00
|
|
|
{
|
2010-09-03 12:51:36 +02:00
|
|
|
/* beg must be <= end */
|
2010-09-03 12:53:15 +02:00
|
|
|
if (beg == end)
|
|
|
|
return;
|
2010-09-03 12:59:15 +02:00
|
|
|
|
|
|
|
while ((int_buf[beg] = int_buf[end]) != 0)
|
|
|
|
beg++, end++;
|
|
|
|
|
2010-09-03 12:53:15 +02:00
|
|
|
if (dbg_bmp) {
|
|
|
|
int i;
|
|
|
|
for (i = 0; int_buf[i]; i++)
|
|
|
|
bb_putchar((unsigned char)int_buf[i]);
|
|
|
|
bb_putchar('\n');
|
|
|
|
}
|
2009-07-16 02:19:39 +02:00
|
|
|
}
|
2010-09-03 14:09:24 +02:00
|
|
|
/* Caller ensures that match_buf points to a malloced buffer
|
|
|
|
* big enough to hold strlen(match_buf)*2 + 2
|
|
|
|
*/
|
|
|
|
static NOINLINE int build_match_prefix(char *match_buf)
|
2001-02-16 18:36:04 +00:00
|
|
|
{
|
|
|
|
int i, j;
|
|
|
|
int command_mode;
|
2010-09-03 14:09:24 +02:00
|
|
|
int16_t *int_buf = (int16_t*)match_buf;
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2010-09-03 14:09:24 +02:00
|
|
|
if (dbg_bmp) printf("\n%s\n", match_buf);
|
2010-09-03 12:53:15 +02:00
|
|
|
|
2010-09-03 14:09:24 +02:00
|
|
|
/* Copy in reverse order, since they overlap */
|
|
|
|
i = strlen(match_buf);
|
|
|
|
do {
|
|
|
|
int_buf[i] = (unsigned char)match_buf[i];
|
|
|
|
i--;
|
|
|
|
} while (i >= 0);
|
2000-04-12 17:49:52 +00:00
|
|
|
|
2010-09-03 12:51:36 +02:00
|
|
|
/* Mark every \c as "quoted c" */
|
2010-09-03 14:09:24 +02:00
|
|
|
for (i = 0; int_buf[i]; i++) {
|
|
|
|
if (int_buf[i] == '\\') {
|
|
|
|
remove_chunk(int_buf, i, i + 1);
|
|
|
|
int_buf[i] |= QUOT;
|
2000-04-12 17:49:52 +00:00
|
|
|
}
|
2010-05-16 20:46:53 +02:00
|
|
|
}
|
2010-09-03 12:59:15 +02:00
|
|
|
/* Quote-mark "chars" and 'chars', drop delimiters */
|
2010-09-03 12:53:15 +02:00
|
|
|
{
|
|
|
|
int in_quote = 0;
|
2010-09-03 12:59:15 +02:00
|
|
|
i = 0;
|
|
|
|
while (int_buf[i]) {
|
2010-09-03 12:53:15 +02:00
|
|
|
int cur = int_buf[i];
|
2010-09-03 12:59:15 +02:00
|
|
|
if (!cur)
|
|
|
|
break;
|
2010-09-03 12:53:15 +02:00
|
|
|
if (cur == '\'' || cur == '"') {
|
2010-09-03 12:59:15 +02:00
|
|
|
if (!in_quote || (cur == in_quote)) {
|
|
|
|
in_quote ^= cur;
|
2010-09-03 13:02:47 +02:00
|
|
|
remove_chunk(int_buf, i, i + 1);
|
2010-09-03 12:59:15 +02:00
|
|
|
continue;
|
|
|
|
}
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
2010-09-03 12:59:15 +02:00
|
|
|
if (in_quote)
|
|
|
|
int_buf[i] = cur | QUOT;
|
|
|
|
i++;
|
2010-09-03 12:53:15 +02:00
|
|
|
}
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
2001-01-26 20:42:23 +00:00
|
|
|
|
2010-09-03 12:51:36 +02:00
|
|
|
/* Remove everything up to command delimiters:
|
|
|
|
* ';' ';;' '&' '|' '&&' '||',
|
|
|
|
* but careful with '>&' '<&' '>|'
|
|
|
|
*/
|
2001-02-16 18:36:04 +00:00
|
|
|
for (i = 0; int_buf[i]; i++) {
|
2010-09-03 12:53:15 +02:00
|
|
|
int cur = int_buf[i];
|
|
|
|
if (cur == ';' || cur == '&' || cur == '|') {
|
|
|
|
int prev = i ? int_buf[i - 1] : 0;
|
|
|
|
if (cur == '&' && (prev == '>' || prev == '<')) {
|
|
|
|
continue;
|
|
|
|
} else if (cur == '|' && prev == '>') {
|
|
|
|
continue;
|
|
|
|
}
|
2010-09-03 13:02:47 +02:00
|
|
|
remove_chunk(int_buf, 0, i + 1 + (cur == int_buf[i + 1]));
|
2010-09-03 12:53:15 +02:00
|
|
|
i = -1; /* back to square 1 */
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
|
|
|
}
|
2010-09-03 12:51:36 +02:00
|
|
|
/* Remove all `cmd` */
|
2009-07-16 02:19:39 +02:00
|
|
|
for (i = 0; int_buf[i]; i++) {
|
2001-02-16 18:36:04 +00:00
|
|
|
if (int_buf[i] == '`') {
|
2010-09-03 12:51:36 +02:00
|
|
|
for (j = i + 1; int_buf[j]; j++) {
|
2001-02-16 18:36:04 +00:00
|
|
|
if (int_buf[j] == '`') {
|
2010-09-03 12:59:15 +02:00
|
|
|
/* `cmd` should count as a word:
|
|
|
|
* `cmd` c<tab> should search for files c*,
|
|
|
|
* not commands c*. Therefore we don't drop
|
|
|
|
* `cmd` entirely, we replace it with single `.
|
|
|
|
*/
|
2010-09-03 13:02:47 +02:00
|
|
|
remove_chunk(int_buf, i, j);
|
2010-09-03 12:53:15 +02:00
|
|
|
goto next;
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
2010-09-03 12:51:36 +02:00
|
|
|
}
|
2010-09-03 12:53:15 +02:00
|
|
|
/* No closing ` - command mode, remove all up to ` */
|
2010-09-03 13:02:47 +02:00
|
|
|
remove_chunk(int_buf, 0, i + 1);
|
2010-09-03 12:53:15 +02:00
|
|
|
break;
|
2010-09-03 12:59:15 +02:00
|
|
|
next: ;
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
2009-07-16 02:19:39 +02:00
|
|
|
}
|
2001-01-26 20:42:23 +00:00
|
|
|
|
2010-09-03 12:59:15 +02:00
|
|
|
/* Remove "cmd (" and "cmd {"
|
|
|
|
* Example: "if { c<tab>"
|
|
|
|
* In this example, c should be matched as command pfx.
|
|
|
|
*/
|
|
|
|
for (i = 0; int_buf[i]; i++) {
|
|
|
|
if (int_buf[i] == '(' || int_buf[i] == '{') {
|
2010-09-03 13:02:47 +02:00
|
|
|
remove_chunk(int_buf, 0, i + 1);
|
2010-09-03 14:08:24 +02:00
|
|
|
i = -1; /* back to square 1 */
|
2001-01-26 20:42:23 +00:00
|
|
|
}
|
2009-07-16 02:19:39 +02:00
|
|
|
}
|
2000-04-12 17:49:52 +00:00
|
|
|
|
2010-09-03 12:51:36 +02:00
|
|
|
/* Remove leading unquoted spaces */
|
2001-02-16 18:36:04 +00:00
|
|
|
for (i = 0; int_buf[i]; i++)
|
|
|
|
if (int_buf[i] != ' ')
|
|
|
|
break;
|
2010-09-03 13:02:47 +02:00
|
|
|
remove_chunk(int_buf, 0, i);
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2010-09-03 12:51:36 +02:00
|
|
|
/* Determine completion mode */
|
2001-02-16 18:36:04 +00:00
|
|
|
command_mode = FIND_EXE_ONLY;
|
2009-07-16 02:19:39 +02:00
|
|
|
for (i = 0; int_buf[i]; i++) {
|
2001-02-16 18:36:04 +00:00
|
|
|
if (int_buf[i] == ' ' || int_buf[i] == '<' || int_buf[i] == '>') {
|
2010-09-03 12:51:36 +02:00
|
|
|
if (int_buf[i] == ' '
|
|
|
|
&& command_mode == FIND_EXE_ONLY
|
2010-09-03 12:59:15 +02:00
|
|
|
&& (char)int_buf[0] == 'c'
|
|
|
|
&& (char)int_buf[1] == 'd'
|
|
|
|
&& i == 2 /* -> int_buf[2] == ' ' */
|
2006-12-21 22:27:10 +00:00
|
|
|
) {
|
2001-02-16 18:36:04 +00:00
|
|
|
command_mode = FIND_DIR_ONLY;
|
2006-12-21 22:27:10 +00:00
|
|
|
} else {
|
2001-02-16 18:36:04 +00:00
|
|
|
command_mode = FIND_FILE_ONLY;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-07-16 02:19:39 +02:00
|
|
|
}
|
2010-09-03 12:53:15 +02:00
|
|
|
if (dbg_bmp) printf("command_mode(0:exe/1:dir/2:file):%d\n", command_mode);
|
2010-09-03 12:51:36 +02:00
|
|
|
|
|
|
|
/* Remove everything except last word */
|
2010-09-02 12:01:11 +02:00
|
|
|
for (i = 0; int_buf[i]; i++) /* quasi-strlen(int_buf) */
|
|
|
|
continue;
|
2001-02-16 18:36:04 +00:00
|
|
|
for (--i; i >= 0; i--) {
|
2010-09-03 12:53:15 +02:00
|
|
|
int cur = int_buf[i];
|
2021-04-09 17:02:00 +02:00
|
|
|
if (cur == ' ' || cur == '<' || cur == '>' || cur == '|' || cur == '&' || cur == '=') {
|
2010-09-03 13:02:47 +02:00
|
|
|
remove_chunk(int_buf, 0, i + 1);
|
2001-02-16 18:36:04 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-09-03 14:09:24 +02:00
|
|
|
/* Convert back to string of _chars_ */
|
2010-09-03 12:59:15 +02:00
|
|
|
i = 0;
|
2010-09-03 14:09:24 +02:00
|
|
|
while ((match_buf[i] = int_buf[i]) != '\0')
|
2010-09-03 12:59:15 +02:00
|
|
|
i++;
|
2010-09-03 13:02:47 +02:00
|
|
|
|
2010-09-03 14:09:24 +02:00
|
|
|
if (dbg_bmp) printf("final match_buf:'%s'\n", match_buf);
|
2001-02-16 18:36:04 +00:00
|
|
|
|
|
|
|
return command_mode;
|
2009-07-16 14:14:34 +02:00
|
|
|
}
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2003-01-06 01:11:50 +00:00
|
|
|
/*
|
2010-09-03 12:51:36 +02:00
|
|
|
* Display by column (original idea from ls applet,
|
|
|
|
* very optimized by me [Vladimir] :)
|
2007-01-21 19:19:46 +00:00
|
|
|
*/
|
2006-01-25 11:53:47 +00:00
|
|
|
static void showfiles(void)
|
2003-01-06 01:11:50 +00:00
|
|
|
{
|
|
|
|
int ncols, row;
|
|
|
|
int column_width = 0;
|
2006-01-25 11:53:47 +00:00
|
|
|
int nfiles = num_matches;
|
2003-01-06 01:11:50 +00:00
|
|
|
int nrows = nfiles;
|
2006-01-25 11:53:47 +00:00
|
|
|
int l;
|
2003-01-06 01:11:50 +00:00
|
|
|
|
2009-07-16 02:19:39 +02:00
|
|
|
/* find the longest file name - use that as the column width */
|
2003-01-06 01:11:50 +00:00
|
|
|
for (row = 0; row < nrows; row++) {
|
2010-06-01 08:33:18 +02:00
|
|
|
l = unicode_strwidth(matches[row]);
|
2003-01-06 01:11:50 +00:00
|
|
|
if (column_width < l)
|
|
|
|
column_width = l;
|
|
|
|
}
|
|
|
|
column_width += 2; /* min space for columns */
|
|
|
|
ncols = cmdedit_termw / column_width;
|
|
|
|
|
|
|
|
if (ncols > 1) {
|
|
|
|
nrows /= ncols;
|
2006-12-19 01:10:25 +00:00
|
|
|
if (nfiles % ncols)
|
2003-01-06 01:11:50 +00:00
|
|
|
nrows++; /* round up fractionals */
|
|
|
|
} else {
|
|
|
|
ncols = 1;
|
|
|
|
}
|
|
|
|
for (row = 0; row < nrows; row++) {
|
|
|
|
int n = row;
|
|
|
|
int nc;
|
2006-01-25 11:53:47 +00:00
|
|
|
|
2006-12-21 22:27:10 +00:00
|
|
|
for (nc = 1; nc < ncols && n+nrows < nfiles; n += nrows, nc++) {
|
2006-12-21 22:24:46 +00:00
|
|
|
printf("%s%-*s", matches[n],
|
2010-06-01 08:33:18 +02:00
|
|
|
(int)(column_width - unicode_strwidth(matches[n])), ""
|
2009-07-16 02:19:39 +02:00
|
|
|
);
|
2006-01-25 11:53:47 +00:00
|
|
|
}
|
2010-06-01 08:33:18 +02:00
|
|
|
if (ENABLE_UNICODE_SUPPORT)
|
2018-09-30 16:56:56 +02:00
|
|
|
puts(printable_string(matches[n]));
|
2010-06-01 08:33:18 +02:00
|
|
|
else
|
|
|
|
puts(matches[n]);
|
2003-01-06 01:11:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-22 03:49:18 +01:00
|
|
|
static const char *is_special_char(char c)
|
|
|
|
{
|
|
|
|
return strchr(" `\"#$%^&*()=+{}[]:;'|\\<>", c);
|
|
|
|
}
|
|
|
|
|
|
|
|
static char *quote_special_chars(char *found)
|
2007-01-21 19:18:19 +00:00
|
|
|
{
|
|
|
|
int l = 0;
|
2009-07-16 02:19:39 +02:00
|
|
|
char *s = xzalloc((strlen(found) + 1) * 2);
|
2007-01-21 19:18:19 +00:00
|
|
|
|
|
|
|
while (*found) {
|
2010-11-22 03:49:18 +01:00
|
|
|
if (is_special_char(*found))
|
2007-01-21 19:18:19 +00:00
|
|
|
s[l++] = '\\';
|
|
|
|
s[l++] = *found++;
|
|
|
|
}
|
2009-07-16 02:19:39 +02:00
|
|
|
/* s[l] = '\0'; - already is */
|
2007-01-21 19:18:19 +00:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2007-01-21 19:19:46 +00:00
|
|
|
/* Do TAB completion */
|
2010-09-03 12:51:36 +02:00
|
|
|
static NOINLINE void input_tab(smallint *lastWasTab)
|
2001-02-16 18:36:04 +00:00
|
|
|
{
|
2010-09-03 14:08:24 +02:00
|
|
|
char *chosen_match;
|
2010-09-03 14:09:24 +02:00
|
|
|
char *match_buf;
|
2010-09-03 14:08:24 +02:00
|
|
|
size_t len_found;
|
|
|
|
/* Length of string used for matching */
|
|
|
|
unsigned match_pfx_len = match_pfx_len;
|
|
|
|
int find_type;
|
2010-11-22 03:49:18 +01:00
|
|
|
# if ENABLE_UNICODE_SUPPORT
|
2010-09-03 14:08:24 +02:00
|
|
|
/* cursor pos in command converted to multibyte form */
|
|
|
|
int cursor_mb;
|
2010-11-22 03:49:18 +01:00
|
|
|
# endif
|
2010-09-03 14:08:24 +02:00
|
|
|
if (!(state->flags & TAB_COMPLETION))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (*lastWasTab) {
|
|
|
|
/* The last char was a TAB too.
|
|
|
|
* Print a list of all the available choices.
|
|
|
|
*/
|
|
|
|
if (num_matches > 0) {
|
|
|
|
/* cursor will be changed by goto_new_line() */
|
|
|
|
int sav_cursor = cursor;
|
|
|
|
goto_new_line();
|
|
|
|
showfiles();
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
draw_full(command_len - sav_cursor);
|
2010-09-03 14:08:24 +02:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2010-09-03 14:08:24 +02:00
|
|
|
*lastWasTab = 1;
|
2010-09-03 14:09:24 +02:00
|
|
|
chosen_match = NULL;
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2010-09-03 14:09:24 +02:00
|
|
|
/* Make a local copy of the string up to the position of the cursor.
|
|
|
|
* build_match_prefix will expand it into int16_t's, need to allocate
|
|
|
|
* twice as much as the string_len+1.
|
|
|
|
* (we then also (ab)use this extra space later - see (**))
|
|
|
|
*/
|
|
|
|
match_buf = xmalloc(MAX_LINELEN * sizeof(int16_t));
|
2010-11-22 03:49:18 +01:00
|
|
|
# if !ENABLE_UNICODE_SUPPORT
|
2010-09-03 14:09:24 +02:00
|
|
|
save_string(match_buf, cursor + 1); /* +1 for NUL */
|
2010-11-22 03:49:18 +01:00
|
|
|
# else
|
2010-09-03 14:08:24 +02:00
|
|
|
{
|
|
|
|
CHAR_T wc = command_ps[cursor];
|
|
|
|
command_ps[cursor] = BB_NUL;
|
2010-09-03 14:09:24 +02:00
|
|
|
save_string(match_buf, MAX_LINELEN);
|
2010-09-03 14:08:24 +02:00
|
|
|
command_ps[cursor] = wc;
|
2010-09-03 14:09:24 +02:00
|
|
|
cursor_mb = strlen(match_buf);
|
2010-09-03 14:08:24 +02:00
|
|
|
}
|
2010-11-22 03:49:18 +01:00
|
|
|
# endif
|
2010-09-03 14:09:24 +02:00
|
|
|
find_type = build_match_prefix(match_buf);
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2010-09-03 14:08:24 +02:00
|
|
|
/* Free up any memory already allocated */
|
|
|
|
free_tab_completion_data();
|
2000-04-12 17:49:52 +00:00
|
|
|
|
2010-11-22 03:49:18 +01:00
|
|
|
# if ENABLE_FEATURE_USERNAME_COMPLETION
|
|
|
|
/* If the word starts with ~ and there is no slash in the word,
|
2010-09-03 14:08:24 +02:00
|
|
|
* then try completing this word as a username. */
|
|
|
|
if (state->flags & USERNAME_COMPLETION)
|
2010-09-03 14:09:24 +02:00
|
|
|
if (match_buf[0] == '~' && strchr(match_buf, '/') == NULL)
|
|
|
|
match_pfx_len = complete_username(match_buf);
|
2010-11-22 03:49:18 +01:00
|
|
|
# endif
|
|
|
|
/* If complete_username() did not match,
|
|
|
|
* try to match a command in $PATH, or a directory, or a file */
|
2010-09-03 14:08:24 +02:00
|
|
|
if (!matches)
|
2010-09-03 14:09:24 +02:00
|
|
|
match_pfx_len = complete_cmd_dir_file(match_buf, find_type);
|
2010-11-22 03:49:18 +01:00
|
|
|
|
|
|
|
/* Account for backslashes which will be inserted
|
|
|
|
* by quote_special_chars() later */
|
|
|
|
{
|
|
|
|
const char *e = match_buf + strlen(match_buf);
|
|
|
|
const char *s = e - match_pfx_len;
|
|
|
|
while (s < e)
|
|
|
|
if (is_special_char(*s++))
|
|
|
|
match_pfx_len++;
|
|
|
|
}
|
|
|
|
|
2010-09-03 14:08:24 +02:00
|
|
|
/* Remove duplicates */
|
|
|
|
if (matches) {
|
2010-11-22 03:49:18 +01:00
|
|
|
unsigned i, n = 0;
|
2010-09-03 14:08:24 +02:00
|
|
|
qsort_string_vector(matches, num_matches);
|
|
|
|
for (i = 0; i < num_matches - 1; ++i) {
|
|
|
|
//if (matches[i] && matches[i+1]) { /* paranoia */
|
|
|
|
if (strcmp(matches[i], matches[i+1]) == 0) {
|
|
|
|
free(matches[i]);
|
|
|
|
//matches[i] = NULL; /* paranoia */
|
|
|
|
} else {
|
|
|
|
matches[n++] = matches[i];
|
|
|
|
}
|
|
|
|
//}
|
2001-06-26 02:06:08 +00:00
|
|
|
}
|
2010-09-03 14:08:24 +02:00
|
|
|
matches[n++] = matches[i];
|
|
|
|
num_matches = n;
|
|
|
|
}
|
2010-11-22 03:49:18 +01:00
|
|
|
|
2010-09-03 14:08:24 +02:00
|
|
|
/* Did we find exactly one match? */
|
|
|
|
if (num_matches != 1) { /* no */
|
|
|
|
char *cp;
|
|
|
|
beep();
|
|
|
|
if (!matches)
|
2010-09-03 14:09:24 +02:00
|
|
|
goto ret; /* no matches at all */
|
2010-09-03 14:08:24 +02:00
|
|
|
/* Find common prefix */
|
|
|
|
chosen_match = xstrdup(matches[0]);
|
|
|
|
for (cp = chosen_match; *cp; cp++) {
|
|
|
|
unsigned n;
|
|
|
|
for (n = 1; n < num_matches; n++) {
|
|
|
|
if (matches[n][cp - chosen_match] != *cp) {
|
|
|
|
goto stop;
|
2009-07-12 02:50:35 +02:00
|
|
|
}
|
|
|
|
}
|
2010-09-03 14:08:24 +02:00
|
|
|
}
|
2010-09-03 12:56:36 +02:00
|
|
|
stop:
|
2010-09-03 14:08:24 +02:00
|
|
|
if (cp == chosen_match) { /* have unique prefix? */
|
2010-09-03 14:09:24 +02:00
|
|
|
goto ret; /* no */
|
2010-09-03 14:08:24 +02:00
|
|
|
}
|
|
|
|
*cp = '\0';
|
2010-11-22 03:49:18 +01:00
|
|
|
cp = quote_special_chars(chosen_match);
|
2010-09-03 14:08:24 +02:00
|
|
|
free(chosen_match);
|
|
|
|
chosen_match = cp;
|
|
|
|
len_found = strlen(chosen_match);
|
|
|
|
} else { /* exactly one match */
|
|
|
|
/* Next <tab> is not a double-tab */
|
|
|
|
*lastWasTab = 0;
|
|
|
|
|
2010-11-22 03:49:18 +01:00
|
|
|
chosen_match = quote_special_chars(matches[0]);
|
2010-09-03 14:08:24 +02:00
|
|
|
len_found = strlen(chosen_match);
|
|
|
|
if (chosen_match[len_found-1] != '/') {
|
|
|
|
chosen_match[len_found] = ' ';
|
|
|
|
chosen_match[++len_found] = '\0';
|
2001-01-26 20:42:23 +00:00
|
|
|
}
|
2010-09-03 14:08:24 +02:00
|
|
|
}
|
2009-07-12 02:50:35 +02:00
|
|
|
|
2010-11-22 03:49:18 +01:00
|
|
|
# if !ENABLE_UNICODE_SUPPORT
|
2010-09-03 14:08:24 +02:00
|
|
|
/* Have space to place the match? */
|
|
|
|
/* The result consists of three parts with these lengths: */
|
|
|
|
/* cursor + (len_found - match_pfx_len) + (command_len - cursor) */
|
|
|
|
/* it simplifies into: */
|
|
|
|
if ((int)(len_found - match_pfx_len + command_len) < S.maxsize) {
|
|
|
|
int pos;
|
|
|
|
/* save tail */
|
2010-09-03 14:09:24 +02:00
|
|
|
strcpy(match_buf, &command_ps[cursor]);
|
2010-09-03 14:08:24 +02:00
|
|
|
/* add match and tail */
|
2010-09-03 14:09:24 +02:00
|
|
|
sprintf(&command_ps[cursor], "%s%s", chosen_match + match_pfx_len, match_buf);
|
2010-09-03 14:08:24 +02:00
|
|
|
command_len = strlen(command_ps);
|
|
|
|
/* new pos */
|
|
|
|
pos = cursor + len_found - match_pfx_len;
|
|
|
|
/* write out the matched command */
|
|
|
|
redraw(cmdedit_y, command_len - pos);
|
|
|
|
}
|
2010-11-22 03:49:18 +01:00
|
|
|
# else
|
2010-09-03 14:08:24 +02:00
|
|
|
{
|
2010-09-03 14:09:24 +02:00
|
|
|
/* Use 2nd half of match_buf as scratch space - see (**) */
|
|
|
|
char *command = match_buf + MAX_LINELEN;
|
|
|
|
int len = save_string(command, MAX_LINELEN);
|
2010-09-03 12:56:36 +02:00
|
|
|
/* Have space to place the match? */
|
2010-09-03 14:08:24 +02:00
|
|
|
/* cursor_mb + (len_found - match_pfx_len) + (len - cursor_mb) */
|
|
|
|
if ((int)(len_found - match_pfx_len + len) < MAX_LINELEN) {
|
2010-09-03 12:56:36 +02:00
|
|
|
int pos;
|
2009-07-12 02:50:35 +02:00
|
|
|
/* save tail */
|
2010-09-03 14:09:24 +02:00
|
|
|
strcpy(match_buf, &command[cursor_mb]);
|
2010-09-03 14:08:24 +02:00
|
|
|
/* where do we want to have cursor after all? */
|
|
|
|
strcpy(&command[cursor_mb], chosen_match + match_pfx_len);
|
2011-07-11 07:36:59 +02:00
|
|
|
len = load_string(command);
|
2009-07-12 02:50:35 +02:00
|
|
|
/* add match and tail */
|
2021-04-14 17:52:18 +02:00
|
|
|
stpcpy(stpcpy(&command[cursor_mb], chosen_match + match_pfx_len), match_buf);
|
2011-07-11 07:36:59 +02:00
|
|
|
command_len = load_string(command);
|
2009-07-12 02:50:35 +02:00
|
|
|
/* write out the matched command */
|
2010-09-03 14:08:24 +02:00
|
|
|
/* paranoia: load_string can return 0 on conv error,
|
|
|
|
* prevent passing pos = (0 - 12) to redraw */
|
|
|
|
pos = command_len - len;
|
|
|
|
redraw(cmdedit_y, pos >= 0 ? pos : 0);
|
2009-07-12 02:50:35 +02:00
|
|
|
}
|
2010-09-03 14:08:24 +02:00
|
|
|
}
|
2010-11-22 03:49:18 +01:00
|
|
|
# endif
|
2010-09-03 14:09:24 +02:00
|
|
|
ret:
|
2010-09-03 14:08:24 +02:00
|
|
|
free(chosen_match);
|
2010-09-03 14:09:24 +02:00
|
|
|
free(match_buf);
|
2000-04-12 17:49:52 +00:00
|
|
|
}
|
2007-01-21 19:19:46 +00:00
|
|
|
|
2010-09-03 12:51:36 +02:00
|
|
|
#endif /* FEATURE_TAB_COMPLETION */
|
2000-04-12 17:49:52 +00:00
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
|
2009-03-23 06:33:37 +00:00
|
|
|
line_input_t* FAST_FUNC new_line_input_t(int flags)
|
|
|
|
{
|
|
|
|
line_input_t *n = xzalloc(sizeof(*n));
|
|
|
|
n->flags = flags;
|
2017-08-02 17:27:28 +02:00
|
|
|
n->timeout = -1;
|
2014-01-10 14:38:26 +01:00
|
|
|
#if MAX_HISTORY > 0
|
2011-03-31 13:16:52 +02:00
|
|
|
n->max_history = MAX_HISTORY;
|
2014-01-10 14:38:26 +01:00
|
|
|
#endif
|
2009-03-23 06:33:37 +00:00
|
|
|
return n;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-11-02 22:09:37 +00:00
|
|
|
#if MAX_HISTORY > 0
|
2007-01-21 19:18:19 +00:00
|
|
|
|
2013-04-07 18:47:24 +02:00
|
|
|
unsigned FAST_FUNC size_from_HISTFILESIZE(const char *hp)
|
2011-03-31 13:16:52 +02:00
|
|
|
{
|
|
|
|
int size = MAX_HISTORY;
|
|
|
|
if (hp) {
|
|
|
|
size = atoi(hp);
|
|
|
|
if (size <= 0)
|
|
|
|
return 1;
|
|
|
|
if (size > MAX_HISTORY)
|
|
|
|
return MAX_HISTORY;
|
|
|
|
}
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
2008-09-27 01:28:56 +00:00
|
|
|
static void save_command_ps_at_cur_history(void)
|
|
|
|
{
|
2009-07-10 18:40:49 +02:00
|
|
|
if (command_ps[0] != BB_NUL) {
|
2008-09-27 01:28:56 +00:00
|
|
|
int cur = state->cur_history;
|
|
|
|
free(state->history[cur]);
|
2009-07-10 18:40:49 +02:00
|
|
|
|
2010-03-26 14:06:56 +01:00
|
|
|
# if ENABLE_UNICODE_SUPPORT
|
2009-07-10 18:40:49 +02:00
|
|
|
{
|
|
|
|
char tbuf[MAX_LINELEN];
|
|
|
|
save_string(tbuf, sizeof(tbuf));
|
|
|
|
state->history[cur] = xstrdup(tbuf);
|
|
|
|
}
|
2009-09-27 02:48:53 +02:00
|
|
|
# else
|
2008-09-27 01:28:56 +00:00
|
|
|
state->history[cur] = xstrdup(command_ps);
|
2009-09-27 02:48:53 +02:00
|
|
|
# endif
|
2008-09-27 01:28:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-01-22 07:21:38 +00:00
|
|
|
/* state->flags is already checked to be nonzero */
|
2008-09-27 01:28:56 +00:00
|
|
|
static int get_previous_history(void)
|
2000-04-12 17:49:52 +00:00
|
|
|
{
|
2008-09-27 01:28:56 +00:00
|
|
|
if ((state->flags & DO_HISTORY) && state->cur_history) {
|
|
|
|
save_command_ps_at_cur_history();
|
|
|
|
state->cur_history--;
|
|
|
|
return 1;
|
2002-11-27 09:29:49 +00:00
|
|
|
}
|
2008-09-27 01:28:56 +00:00
|
|
|
beep();
|
|
|
|
return 0;
|
2000-04-12 17:49:52 +00:00
|
|
|
}
|
|
|
|
|
2002-11-27 09:29:49 +00:00
|
|
|
static int get_next_history(void)
|
2000-04-12 17:49:52 +00:00
|
|
|
{
|
2007-01-22 07:21:38 +00:00
|
|
|
if (state->flags & DO_HISTORY) {
|
2008-09-27 01:28:56 +00:00
|
|
|
if (state->cur_history < state->cnt_history) {
|
|
|
|
save_command_ps_at_cur_history(); /* save the current history line */
|
|
|
|
return ++state->cur_history;
|
2007-01-22 07:21:38 +00:00
|
|
|
}
|
2002-11-27 09:29:49 +00:00
|
|
|
}
|
2007-01-22 07:21:38 +00:00
|
|
|
beep();
|
|
|
|
return 0;
|
2000-04-12 17:49:52 +00:00
|
|
|
}
|
2002-12-03 22:45:46 +00:00
|
|
|
|
2013-04-07 18:47:24 +02:00
|
|
|
/* Lists command history. Used by shell 'history' builtins */
|
|
|
|
void FAST_FUNC show_history(const line_input_t *st)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!st)
|
|
|
|
return;
|
|
|
|
for (i = 0; i < st->cnt_history; i++)
|
|
|
|
printf("%4d %s\n", i, st->history[i]);
|
|
|
|
}
|
|
|
|
|
2020-12-21 21:36:58 +01:00
|
|
|
# if ENABLE_FEATURE_EDITING_SAVEHISTORY
|
2018-12-27 18:03:20 +01:00
|
|
|
void FAST_FUNC free_line_input_t(line_input_t *n)
|
|
|
|
{
|
2020-12-21 21:36:58 +01:00
|
|
|
if (n) {
|
|
|
|
int i = n->cnt_history;
|
|
|
|
while (i > 0)
|
|
|
|
free(n->history[--i]);
|
|
|
|
free(n);
|
|
|
|
}
|
2018-12-27 18:03:20 +01:00
|
|
|
}
|
2020-12-21 21:36:58 +01:00
|
|
|
# else
|
|
|
|
/* #defined to free() in libbb.h */
|
|
|
|
# endif
|
2018-12-27 18:03:20 +01:00
|
|
|
|
2009-09-27 02:48:53 +02:00
|
|
|
# if ENABLE_FEATURE_EDITING_SAVEHISTORY
|
2009-03-22 19:00:05 +00:00
|
|
|
/* We try to ensure that concurrent additions to the history
|
2009-03-23 06:33:37 +00:00
|
|
|
* do not overwrite each other.
|
2009-03-22 19:00:05 +00:00
|
|
|
* Otherwise shell users get unhappy.
|
|
|
|
*
|
|
|
|
* History file is trimmed lazily, when it grows several times longer
|
|
|
|
* than configured MAX_HISTORY lines.
|
|
|
|
*/
|
|
|
|
|
2007-01-22 07:21:38 +00:00
|
|
|
/* state->flags is already checked to be nonzero */
|
2009-03-23 06:33:37 +00:00
|
|
|
static void load_history(line_input_t *st_parm)
|
2002-12-09 11:10:40 +00:00
|
|
|
{
|
2009-03-22 19:00:05 +00:00
|
|
|
char *temp_h[MAX_HISTORY];
|
|
|
|
char *line;
|
2002-12-03 22:45:46 +00:00
|
|
|
FILE *fp;
|
2009-03-22 19:00:05 +00:00
|
|
|
unsigned idx, i, line_len;
|
2002-12-03 22:45:46 +00:00
|
|
|
|
2008-03-26 13:32:30 +00:00
|
|
|
/* NB: do not trash old history if file can't be opened */
|
2002-12-03 22:45:46 +00:00
|
|
|
|
2009-03-23 06:33:37 +00:00
|
|
|
fp = fopen_for_read(st_parm->hist_file);
|
2006-12-21 22:27:10 +00:00
|
|
|
if (fp) {
|
2008-03-26 13:32:30 +00:00
|
|
|
/* clean up old history */
|
2009-03-23 06:33:37 +00:00
|
|
|
for (idx = st_parm->cnt_history; idx > 0;) {
|
2009-03-22 19:00:05 +00:00
|
|
|
idx--;
|
2009-03-23 06:33:37 +00:00
|
|
|
free(st_parm->history[idx]);
|
|
|
|
st_parm->history[idx] = NULL;
|
2008-03-26 13:32:30 +00:00
|
|
|
}
|
|
|
|
|
2009-03-22 19:00:05 +00:00
|
|
|
/* fill temp_h[], retaining only last MAX_HISTORY lines */
|
|
|
|
memset(temp_h, 0, sizeof(temp_h));
|
2011-09-04 16:12:33 +02:00
|
|
|
idx = 0;
|
2012-04-24 22:40:58 +02:00
|
|
|
st_parm->cnt_history_in_file = 0;
|
2009-03-22 19:00:05 +00:00
|
|
|
while ((line = xmalloc_fgetline(fp)) != NULL) {
|
|
|
|
if (line[0] == '\0') {
|
|
|
|
free(line);
|
2002-12-09 11:10:40 +00:00
|
|
|
continue;
|
|
|
|
}
|
2009-03-22 19:00:05 +00:00
|
|
|
free(temp_h[idx]);
|
|
|
|
temp_h[idx] = line;
|
2012-04-24 22:40:58 +02:00
|
|
|
st_parm->cnt_history_in_file++;
|
2009-03-22 19:00:05 +00:00
|
|
|
idx++;
|
2011-03-31 13:16:52 +02:00
|
|
|
if (idx == st_parm->max_history)
|
2009-03-22 19:00:05 +00:00
|
|
|
idx = 0;
|
2002-12-03 22:45:46 +00:00
|
|
|
}
|
2006-12-21 22:27:10 +00:00
|
|
|
fclose(fp);
|
2009-03-22 19:00:05 +00:00
|
|
|
|
|
|
|
/* find first non-NULL temp_h[], if any */
|
2009-03-23 06:33:37 +00:00
|
|
|
if (st_parm->cnt_history_in_file) {
|
2009-03-22 19:00:05 +00:00
|
|
|
while (temp_h[idx] == NULL) {
|
|
|
|
idx++;
|
2011-03-31 13:16:52 +02:00
|
|
|
if (idx == st_parm->max_history)
|
2009-03-22 19:00:05 +00:00
|
|
|
idx = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-23 06:33:37 +00:00
|
|
|
/* copy temp_h[] to st_parm->history[] */
|
2011-03-31 13:16:52 +02:00
|
|
|
for (i = 0; i < st_parm->max_history;) {
|
2009-03-22 19:00:05 +00:00
|
|
|
line = temp_h[idx];
|
|
|
|
if (!line)
|
|
|
|
break;
|
|
|
|
idx++;
|
2011-03-31 13:16:52 +02:00
|
|
|
if (idx == st_parm->max_history)
|
2009-03-22 19:00:05 +00:00
|
|
|
idx = 0;
|
|
|
|
line_len = strlen(line);
|
|
|
|
if (line_len >= MAX_LINELEN)
|
|
|
|
line[MAX_LINELEN-1] = '\0';
|
2009-03-23 06:33:37 +00:00
|
|
|
st_parm->history[i++] = line;
|
2009-03-22 19:00:05 +00:00
|
|
|
}
|
2009-03-23 06:33:37 +00:00
|
|
|
st_parm->cnt_history = i;
|
2011-09-04 16:12:33 +02:00
|
|
|
if (ENABLE_FEATURE_EDITING_SAVE_ON_EXIT)
|
|
|
|
st_parm->cnt_history_in_file = i;
|
2002-12-03 22:45:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-09-04 16:12:33 +02:00
|
|
|
# if ENABLE_FEATURE_EDITING_SAVE_ON_EXIT
|
|
|
|
void save_history(line_input_t *st)
|
|
|
|
{
|
|
|
|
FILE *fp;
|
|
|
|
|
2020-12-21 21:36:58 +01:00
|
|
|
if (!st || !st->hist_file)
|
2011-09-04 16:12:33 +02:00
|
|
|
return;
|
|
|
|
if (st->cnt_history <= st->cnt_history_in_file)
|
|
|
|
return;
|
|
|
|
|
|
|
|
fp = fopen(st->hist_file, "a");
|
|
|
|
if (fp) {
|
|
|
|
int i, fd;
|
|
|
|
char *new_name;
|
|
|
|
line_input_t *st_temp;
|
|
|
|
|
|
|
|
for (i = st->cnt_history_in_file; i < st->cnt_history; i++)
|
|
|
|
fprintf(fp, "%s\n", st->history[i]);
|
|
|
|
fclose(fp);
|
|
|
|
|
|
|
|
/* we may have concurrently written entries from others.
|
|
|
|
* load them */
|
|
|
|
st_temp = new_line_input_t(st->flags);
|
|
|
|
st_temp->hist_file = st->hist_file;
|
|
|
|
st_temp->max_history = st->max_history;
|
|
|
|
load_history(st_temp);
|
|
|
|
|
|
|
|
/* write out temp file and replace hist_file atomically */
|
|
|
|
new_name = xasprintf("%s.%u.new", st->hist_file, (int) getpid());
|
|
|
|
fd = open(new_name, O_WRONLY | O_CREAT | O_TRUNC, 0600);
|
|
|
|
if (fd >= 0) {
|
|
|
|
fp = xfdopen_for_write(fd);
|
|
|
|
for (i = 0; i < st_temp->cnt_history; i++)
|
|
|
|
fprintf(fp, "%s\n", st_temp->history[i]);
|
|
|
|
fclose(fp);
|
|
|
|
if (rename(new_name, st->hist_file) == 0)
|
|
|
|
st->cnt_history_in_file = st_temp->cnt_history;
|
|
|
|
}
|
|
|
|
free(new_name);
|
|
|
|
free_line_input_t(st_temp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
# else
|
2009-03-22 19:00:05 +00:00
|
|
|
static void save_history(char *str)
|
2002-12-03 22:45:46 +00:00
|
|
|
{
|
2009-03-22 19:00:05 +00:00
|
|
|
int fd;
|
2009-03-23 06:33:37 +00:00
|
|
|
int len, len2;
|
2003-07-28 09:56:35 +00:00
|
|
|
|
2011-09-04 16:12:33 +02:00
|
|
|
if (!state->hist_file)
|
|
|
|
return;
|
|
|
|
|
2010-11-15 02:58:28 +01:00
|
|
|
fd = open(state->hist_file, O_WRONLY | O_CREAT | O_APPEND, 0600);
|
2009-03-22 19:00:05 +00:00
|
|
|
if (fd < 0)
|
|
|
|
return;
|
2009-03-23 06:33:37 +00:00
|
|
|
xlseek(fd, 0, SEEK_END); /* paranoia */
|
|
|
|
len = strlen(str);
|
|
|
|
str[len] = '\n'; /* we (try to) do atomic write */
|
|
|
|
len2 = full_write(fd, str, len + 1);
|
|
|
|
str[len] = '\0';
|
2009-03-22 19:00:05 +00:00
|
|
|
close(fd);
|
2009-03-23 06:33:37 +00:00
|
|
|
if (len2 != len + 1)
|
|
|
|
return; /* "wtf?" */
|
2009-03-22 19:00:05 +00:00
|
|
|
|
|
|
|
/* did we write so much that history file needs trimming? */
|
2009-03-23 06:33:37 +00:00
|
|
|
state->cnt_history_in_file++;
|
2011-03-31 13:16:52 +02:00
|
|
|
if (state->cnt_history_in_file > state->max_history * 4) {
|
2009-03-22 19:00:05 +00:00
|
|
|
char *new_name;
|
2009-03-23 06:33:37 +00:00
|
|
|
line_input_t *st_temp;
|
|
|
|
|
|
|
|
/* we may have concurrently written entries from others.
|
|
|
|
* load them */
|
|
|
|
st_temp = new_line_input_t(state->flags);
|
|
|
|
st_temp->hist_file = state->hist_file;
|
2011-03-31 14:39:38 +02:00
|
|
|
st_temp->max_history = state->max_history;
|
2009-03-23 06:33:37 +00:00
|
|
|
load_history(st_temp);
|
2009-03-22 19:00:05 +00:00
|
|
|
|
2009-03-23 06:33:37 +00:00
|
|
|
/* write out temp file and replace hist_file atomically */
|
|
|
|
new_name = xasprintf("%s.%u.new", state->hist_file, (int) getpid());
|
2011-09-04 15:28:03 +02:00
|
|
|
fd = open(new_name, O_WRONLY | O_CREAT | O_TRUNC, 0600);
|
2010-11-15 02:58:28 +01:00
|
|
|
if (fd >= 0) {
|
|
|
|
FILE *fp;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
fp = xfdopen_for_write(fd);
|
2009-03-23 06:33:37 +00:00
|
|
|
for (i = 0; i < st_temp->cnt_history; i++)
|
|
|
|
fprintf(fp, "%s\n", st_temp->history[i]);
|
2009-03-22 19:00:05 +00:00
|
|
|
fclose(fp);
|
2009-03-23 06:33:37 +00:00
|
|
|
if (rename(new_name, state->hist_file) == 0)
|
|
|
|
state->cnt_history_in_file = st_temp->cnt_history;
|
2009-03-22 19:00:05 +00:00
|
|
|
}
|
|
|
|
free(new_name);
|
2009-03-23 06:33:37 +00:00
|
|
|
free_line_input_t(st_temp);
|
2002-12-03 22:45:46 +00:00
|
|
|
}
|
|
|
|
}
|
2011-09-04 16:12:33 +02:00
|
|
|
# endif
|
2009-09-27 02:48:53 +02:00
|
|
|
# else
|
|
|
|
# define load_history(a) ((void)0)
|
|
|
|
# define save_history(a) ((void)0)
|
|
|
|
# endif /* FEATURE_COMMAND_SAVEHISTORY */
|
2002-12-03 22:45:46 +00:00
|
|
|
|
2009-03-22 19:00:05 +00:00
|
|
|
static void remember_in_history(char *str)
|
2007-01-22 07:21:38 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!(state->flags & DO_HISTORY))
|
|
|
|
return;
|
2008-09-27 01:28:56 +00:00
|
|
|
if (str[0] == '\0')
|
|
|
|
return;
|
2007-01-22 07:21:38 +00:00
|
|
|
i = state->cnt_history;
|
2008-09-27 01:28:56 +00:00
|
|
|
/* Don't save dupes */
|
|
|
|
if (i && strcmp(state->history[i-1], str) == 0)
|
|
|
|
return;
|
|
|
|
|
2011-03-31 13:16:52 +02:00
|
|
|
free(state->history[state->max_history]); /* redundant, paranoia */
|
|
|
|
state->history[state->max_history] = NULL; /* redundant, paranoia */
|
2008-09-27 01:28:56 +00:00
|
|
|
|
|
|
|
/* If history[] is full, remove the oldest command */
|
2011-03-31 13:16:52 +02:00
|
|
|
/* we need to keep history[state->max_history] empty, hence >=, not > */
|
|
|
|
if (i >= state->max_history) {
|
2007-01-22 07:21:38 +00:00
|
|
|
free(state->history[0]);
|
2011-03-31 13:16:52 +02:00
|
|
|
for (i = 0; i < state->max_history-1; i++)
|
2007-01-22 07:21:38 +00:00
|
|
|
state->history[i] = state->history[i+1];
|
2011-03-31 13:16:52 +02:00
|
|
|
/* i == state->max_history-1 */
|
2011-11-04 01:09:09 +01:00
|
|
|
# if ENABLE_FEATURE_EDITING_SAVE_ON_EXIT
|
|
|
|
if (state->cnt_history_in_file)
|
2011-09-04 16:12:33 +02:00
|
|
|
state->cnt_history_in_file--;
|
2011-11-04 01:09:09 +01:00
|
|
|
# endif
|
2007-01-22 07:21:38 +00:00
|
|
|
}
|
2011-03-31 13:16:52 +02:00
|
|
|
/* i <= state->max_history-1 */
|
2007-01-22 07:21:38 +00:00
|
|
|
state->history[i++] = xstrdup(str);
|
2011-03-31 13:16:52 +02:00
|
|
|
/* i <= state->max_history */
|
2007-01-22 07:21:38 +00:00
|
|
|
state->cur_history = i;
|
|
|
|
state->cnt_history = i;
|
2011-09-04 16:12:33 +02:00
|
|
|
# if ENABLE_FEATURE_EDITING_SAVEHISTORY && !ENABLE_FEATURE_EDITING_SAVE_ON_EXIT
|
|
|
|
save_history(str);
|
2009-09-27 02:48:53 +02:00
|
|
|
# endif
|
2007-01-22 07:21:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#else /* MAX_HISTORY == 0 */
|
2009-09-27 02:48:53 +02:00
|
|
|
# define remember_in_history(a) ((void)0)
|
2007-01-22 07:21:38 +00:00
|
|
|
#endif /* MAX_HISTORY */
|
2001-02-16 18:36:04 +00:00
|
|
|
|
|
|
|
|
2009-10-25 23:50:56 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_VI
|
2000-03-17 01:12:41 +00:00
|
|
|
/*
|
2007-01-21 19:18:19 +00:00
|
|
|
* vi mode implemented 2005 by Paul Fox <pgf@foxharp.boston.ma.us>
|
2000-03-17 01:12:41 +00:00
|
|
|
*/
|
2005-09-22 12:59:26 +00:00
|
|
|
static void
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_Word_motion(int eat)
|
2005-08-04 19:04:46 +00:00
|
|
|
{
|
2009-07-10 18:40:49 +02:00
|
|
|
CHAR_T *command = command_ps;
|
|
|
|
|
|
|
|
while (cursor < command_len && !BB_isspace(command[cursor]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
2009-07-10 18:40:49 +02:00
|
|
|
if (eat) while (cursor < command_len && BB_isspace(command[cursor]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
|
|
|
}
|
|
|
|
|
2005-09-22 12:59:26 +00:00
|
|
|
static void
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_word_motion(int eat)
|
2005-08-04 19:04:46 +00:00
|
|
|
{
|
2009-07-10 18:40:49 +02:00
|
|
|
CHAR_T *command = command_ps;
|
|
|
|
|
2016-08-14 23:30:29 +02:00
|
|
|
if (BB_isalnum_or_underscore(command[cursor])) {
|
2007-01-22 08:34:44 +00:00
|
|
|
while (cursor < command_len
|
2016-08-14 23:30:29 +02:00
|
|
|
&& (BB_isalnum_or_underscore(command[cursor+1]))
|
2009-07-12 00:51:15 +02:00
|
|
|
) {
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
2009-07-12 00:51:15 +02:00
|
|
|
}
|
2009-07-10 18:40:49 +02:00
|
|
|
} else if (BB_ispunct(command[cursor])) {
|
|
|
|
while (cursor < command_len && BB_ispunct(command[cursor+1]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
|
|
|
}
|
|
|
|
|
2007-01-22 08:34:44 +00:00
|
|
|
if (cursor < command_len)
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
|
|
|
|
2009-07-12 00:51:15 +02:00
|
|
|
if (eat) {
|
2009-07-10 18:40:49 +02:00
|
|
|
while (cursor < command_len && BB_isspace(command[cursor]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
2009-07-12 00:51:15 +02:00
|
|
|
}
|
2005-08-04 19:04:46 +00:00
|
|
|
}
|
|
|
|
|
2005-09-22 12:59:26 +00:00
|
|
|
static void
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_End_motion(void)
|
2005-08-04 19:04:46 +00:00
|
|
|
{
|
2009-07-10 18:40:49 +02:00
|
|
|
CHAR_T *command = command_ps;
|
|
|
|
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
2009-07-10 18:40:49 +02:00
|
|
|
while (cursor < command_len && BB_isspace(command[cursor]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
2009-07-10 18:40:49 +02:00
|
|
|
while (cursor < command_len-1 && !BB_isspace(command[cursor+1]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
|
|
|
}
|
|
|
|
|
2005-09-22 12:59:26 +00:00
|
|
|
static void
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_end_motion(void)
|
2005-08-04 19:04:46 +00:00
|
|
|
{
|
2009-07-10 18:40:49 +02:00
|
|
|
CHAR_T *command = command_ps;
|
|
|
|
|
2007-01-22 08:34:44 +00:00
|
|
|
if (cursor >= command_len-1)
|
2005-08-04 19:04:46 +00:00
|
|
|
return;
|
|
|
|
input_forward();
|
2009-07-10 18:40:49 +02:00
|
|
|
while (cursor < command_len-1 && BB_isspace(command[cursor]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
2007-01-22 08:34:44 +00:00
|
|
|
if (cursor >= command_len-1)
|
2005-08-04 19:04:46 +00:00
|
|
|
return;
|
2016-08-14 23:30:29 +02:00
|
|
|
if (BB_isalnum_or_underscore(command[cursor])) {
|
2007-01-22 08:34:44 +00:00
|
|
|
while (cursor < command_len-1
|
2016-08-14 23:30:29 +02:00
|
|
|
&& (BB_isalnum_or_underscore(command[cursor+1]))
|
2006-12-21 22:27:10 +00:00
|
|
|
) {
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
2006-12-21 22:27:10 +00:00
|
|
|
}
|
2009-07-10 18:40:49 +02:00
|
|
|
} else if (BB_ispunct(command[cursor])) {
|
|
|
|
while (cursor < command_len-1 && BB_ispunct(command[cursor+1]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-09-22 12:59:26 +00:00
|
|
|
static void
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_Back_motion(void)
|
2005-08-04 19:04:46 +00:00
|
|
|
{
|
2009-07-10 18:40:49 +02:00
|
|
|
CHAR_T *command = command_ps;
|
|
|
|
|
|
|
|
while (cursor > 0 && BB_isspace(command[cursor-1]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_backward(1);
|
2009-07-10 18:40:49 +02:00
|
|
|
while (cursor > 0 && !BB_isspace(command[cursor-1]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_backward(1);
|
|
|
|
}
|
|
|
|
|
2005-09-22 12:59:26 +00:00
|
|
|
static void
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_back_motion(void)
|
2005-08-04 19:04:46 +00:00
|
|
|
{
|
2009-07-10 18:40:49 +02:00
|
|
|
CHAR_T *command = command_ps;
|
|
|
|
|
2005-08-04 19:04:46 +00:00
|
|
|
if (cursor <= 0)
|
|
|
|
return;
|
|
|
|
input_backward(1);
|
2009-07-10 18:40:49 +02:00
|
|
|
while (cursor > 0 && BB_isspace(command[cursor]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_backward(1);
|
|
|
|
if (cursor <= 0)
|
|
|
|
return;
|
2016-08-14 23:30:29 +02:00
|
|
|
if (BB_isalnum_or_underscore(command[cursor])) {
|
2006-12-21 22:27:10 +00:00
|
|
|
while (cursor > 0
|
2016-08-14 23:30:29 +02:00
|
|
|
&& (BB_isalnum_or_underscore(command[cursor-1]))
|
2006-12-21 22:27:10 +00:00
|
|
|
) {
|
2005-08-04 19:04:46 +00:00
|
|
|
input_backward(1);
|
2006-12-21 22:27:10 +00:00
|
|
|
}
|
2009-07-10 18:40:49 +02:00
|
|
|
} else if (BB_ispunct(command[cursor])) {
|
|
|
|
while (cursor > 0 && BB_ispunct(command[cursor-1]))
|
2005-08-04 19:04:46 +00:00
|
|
|
input_backward(1);
|
|
|
|
}
|
|
|
|
}
|
2007-01-21 19:18:19 +00:00
|
|
|
#endif
|
|
|
|
|
2009-10-25 23:50:56 +01:00
|
|
|
/* Modelled after bash 4.0 behavior of Ctrl-<arrow> */
|
|
|
|
static void ctrl_left(void)
|
|
|
|
{
|
|
|
|
CHAR_T *command = command_ps;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
CHAR_T c;
|
|
|
|
|
|
|
|
input_backward(1);
|
|
|
|
if (cursor == 0)
|
|
|
|
break;
|
|
|
|
c = command[cursor];
|
|
|
|
if (c != ' ' && !BB_ispunct(c)) {
|
|
|
|
/* we reached a "word" delimited by spaces/punct.
|
|
|
|
* go to its beginning */
|
|
|
|
while (1) {
|
|
|
|
c = command[cursor - 1];
|
|
|
|
if (c == ' ' || BB_ispunct(c))
|
|
|
|
break;
|
|
|
|
input_backward(1);
|
|
|
|
if (cursor == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
static void ctrl_right(void)
|
|
|
|
{
|
|
|
|
CHAR_T *command = command_ps;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
CHAR_T c;
|
|
|
|
|
|
|
|
c = command[cursor];
|
|
|
|
if (c == BB_NUL)
|
|
|
|
break;
|
|
|
|
if (c != ' ' && !BB_ispunct(c)) {
|
|
|
|
/* we reached a "word" delimited by spaces/punct.
|
|
|
|
* go to its end + 1 */
|
|
|
|
while (1) {
|
|
|
|
input_forward();
|
|
|
|
c = command[cursor];
|
|
|
|
if (c == BB_NUL || c == ' ' || BB_ispunct(c))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
input_forward();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
|
|
|
|
/*
|
2007-01-22 07:21:38 +00:00
|
|
|
* read_line_input and its helpers
|
2007-01-21 19:18:19 +00:00
|
|
|
*/
|
|
|
|
|
2009-11-11 03:19:30 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_ASK_TERMINAL
|
|
|
|
static void ask_terminal(void)
|
|
|
|
{
|
|
|
|
/* Ask terminal where is the cursor now.
|
|
|
|
* lineedit_read_key handles response and corrects
|
|
|
|
* our idea of current cursor position.
|
|
|
|
* Testcase: run "echo -n long_line_long_line_long_line",
|
|
|
|
* then type in a long, wrapping command and try to
|
|
|
|
* delete it using backspace key.
|
|
|
|
* Note: we print it _after_ prompt, because
|
|
|
|
* prompt may contain CR. Example: PS1='\[\r\n\]\w '
|
|
|
|
*/
|
|
|
|
/* Problem: if there is buffered input on stdin,
|
|
|
|
* the response will be delivered later,
|
|
|
|
* possibly to an unsuspecting application.
|
|
|
|
* Testcase: "sleep 1; busybox ash" + press and hold [Enter].
|
|
|
|
* Result:
|
|
|
|
* ~/srcdevel/bbox/fix/busybox.t4 #
|
|
|
|
* ~/srcdevel/bbox/fix/busybox.t4 #
|
|
|
|
* ^[[59;34~/srcdevel/bbox/fix/busybox.t4 # <-- garbage
|
|
|
|
* ~/srcdevel/bbox/fix/busybox.t4 #
|
|
|
|
*
|
|
|
|
* Checking for input with poll only makes the race narrower,
|
|
|
|
* I still can trigger it. Strace:
|
|
|
|
*
|
|
|
|
* write(1, "~/srcdevel/bbox/fix/busybox.t4 # ", 33) = 33
|
|
|
|
* poll([{fd=0, events=POLLIN}], 1, 0) = 0 (Timeout) <-- no input exists
|
|
|
|
* write(1, "\33[6n", 4) = 4 <-- send the ESC sequence, quick!
|
2011-05-20 04:26:29 +02:00
|
|
|
* poll([{fd=0, events=POLLIN}], 1, -1) = 1 ([{fd=0, revents=POLLIN}])
|
2009-11-11 03:19:30 +01:00
|
|
|
* read(0, "\n", 1) = 1 <-- oh crap, user's input got in first
|
|
|
|
*/
|
2010-07-25 00:06:41 +02:00
|
|
|
struct pollfd pfd;
|
|
|
|
|
|
|
|
pfd.fd = STDIN_FILENO;
|
|
|
|
pfd.events = POLLIN;
|
|
|
|
if (safe_poll(&pfd, 1, 0) == 0) {
|
|
|
|
S.sent_ESC_br6n = 1;
|
2021-02-03 20:47:14 +01:00
|
|
|
fputs_stdout(ESC"[6n");
|
2010-07-25 00:06:41 +02:00
|
|
|
fflush_all(); /* make terminal see it ASAP! */
|
2009-11-11 03:19:30 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
#define ask_terminal() ((void)0)
|
|
|
|
#endif
|
|
|
|
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
/* Note about multi-line PS1 (e.g. "\n\w \u@\h\n> ") and prompt redrawing:
|
|
|
|
*
|
|
|
|
* If the prompt has any newlines, after we print it once we use only its last
|
|
|
|
* line to redraw in-place, which makes it simpler to calculate how many lines
|
|
|
|
* we should move the cursor up to align the redraw (cmdedit_y). The earlier
|
|
|
|
* prompt lines just stay on screen and we redraw below them.
|
|
|
|
*
|
|
|
|
* Use cases for all prompt lines beyond the initial draw:
|
|
|
|
* - After clear-screen (^L) or after displaying tab-completion choices, we
|
|
|
|
* print the full prompt, as it isn't redrawn in-place.
|
|
|
|
* - During terminal resize we could try to redraw all lines, but we don't,
|
|
|
|
* because it requires delicate alignment, it's good enough with only the
|
|
|
|
* last line, and doing it wrong is arguably worse than not doing it at all.
|
|
|
|
*
|
|
|
|
* Terminology wise, if it doesn't mention "full", then it means the last/sole
|
|
|
|
* prompt line. We use the prompt (last/sole line) while redrawing in-place,
|
|
|
|
* and the full where we need a fresh one unrelated to an earlier position.
|
|
|
|
*
|
|
|
|
* If PS1 is not multiline, the last/sole line and the full are the same string.
|
|
|
|
*/
|
|
|
|
|
2013-08-19 16:45:04 +02:00
|
|
|
/* Called just once at read_line_input() init time */
|
2007-01-22 09:03:07 +00:00
|
|
|
#if !ENABLE_FEATURE_EDITING_FANCY_PROMPT
|
2007-11-10 01:35:47 +00:00
|
|
|
static void parse_and_put_prompt(const char *prmt_ptr)
|
2007-01-21 19:18:19 +00:00
|
|
|
{
|
2013-08-19 16:45:04 +02:00
|
|
|
const char *p;
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
cmdedit_prompt = prompt_last_line = prmt_ptr;
|
2013-08-19 16:45:04 +02:00
|
|
|
p = strrchr(prmt_ptr, '\n');
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
if (p)
|
|
|
|
prompt_last_line = p + 1;
|
|
|
|
cmdedit_prmt_len = unicode_strwidth(prompt_last_line);
|
2007-01-21 19:18:19 +00:00
|
|
|
put_prompt();
|
|
|
|
}
|
|
|
|
#else
|
2007-11-10 01:35:47 +00:00
|
|
|
static void parse_and_put_prompt(const char *prmt_ptr)
|
2007-01-21 19:18:19 +00:00
|
|
|
{
|
2013-08-19 16:45:04 +02:00
|
|
|
int prmt_size = 0;
|
2007-01-21 19:18:19 +00:00
|
|
|
char *prmt_mem_ptr = xzalloc(1);
|
2013-03-29 13:09:05 +01:00
|
|
|
char *cwd_buf = NULL;
|
2013-08-19 16:45:04 +02:00
|
|
|
char flg_not_length = '[';
|
2007-12-03 10:45:14 +00:00
|
|
|
char cbuf[2];
|
2007-01-21 19:18:19 +00:00
|
|
|
|
2013-08-19 16:44:05 +02:00
|
|
|
/*cmdedit_prmt_len = 0; - already is */
|
2007-01-22 07:30:26 +00:00
|
|
|
|
2007-12-03 10:45:14 +00:00
|
|
|
cbuf[1] = '\0'; /* never changes */
|
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
while (*prmt_ptr) {
|
2013-08-19 16:45:04 +02:00
|
|
|
char timebuf[sizeof("HH:MM:SS")];
|
2007-12-03 10:45:14 +00:00
|
|
|
char *free_me = NULL;
|
2013-08-19 16:45:04 +02:00
|
|
|
char *pbuf;
|
|
|
|
char c;
|
2007-12-03 10:45:14 +00:00
|
|
|
|
|
|
|
pbuf = cbuf;
|
2007-01-21 19:18:19 +00:00
|
|
|
c = *prmt_ptr++;
|
|
|
|
if (c == '\\') {
|
2013-03-29 13:09:05 +01:00
|
|
|
const char *cp;
|
2007-01-21 19:18:19 +00:00
|
|
|
int l;
|
2013-03-28 13:20:12 +01:00
|
|
|
/*
|
|
|
|
* Supported via bb_process_escape_sequence:
|
|
|
|
* \a ASCII bell character (07)
|
|
|
|
* \e ASCII escape character (033)
|
|
|
|
* \n newline
|
|
|
|
* \r carriage return
|
|
|
|
* \\ backslash
|
|
|
|
* \nnn char with octal code nnn
|
|
|
|
* Supported:
|
|
|
|
* \$ if the effective UID is 0, a #, otherwise a $
|
|
|
|
* \w current working directory, with $HOME abbreviated with a tilde
|
|
|
|
* Note: we do not support $PROMPT_DIRTRIM=n feature
|
2013-03-29 13:09:05 +01:00
|
|
|
* \W basename of the current working directory, with $HOME abbreviated with a tilde
|
2013-03-28 13:20:12 +01:00
|
|
|
* \h hostname up to the first '.'
|
|
|
|
* \H hostname
|
|
|
|
* \u username
|
|
|
|
* \[ begin a sequence of non-printing characters
|
|
|
|
* \] end a sequence of non-printing characters
|
2013-03-29 13:09:05 +01:00
|
|
|
* \T current time in 12-hour HH:MM:SS format
|
|
|
|
* \@ current time in 12-hour am/pm format
|
|
|
|
* \A current time in 24-hour HH:MM format
|
|
|
|
* \t current time in 24-hour HH:MM:SS format
|
|
|
|
* (all of the above work as \A)
|
2013-03-28 13:20:12 +01:00
|
|
|
* Not supported:
|
2013-03-29 13:09:05 +01:00
|
|
|
* \! history number of this command
|
2013-03-28 13:20:12 +01:00
|
|
|
* \# command number of this command
|
|
|
|
* \j number of jobs currently managed by the shell
|
|
|
|
* \l basename of the shell's terminal device name
|
|
|
|
* \s name of the shell, the basename of $0 (the portion following the final slash)
|
|
|
|
* \V release of bash, version + patch level (e.g., 2.00.0)
|
|
|
|
* \d date in "Weekday Month Date" format (e.g., "Tue May 26")
|
|
|
|
* \D{format}
|
|
|
|
* format is passed to strftime(3).
|
|
|
|
* An empty format results in a locale-specific time representation.
|
|
|
|
* The braces are required.
|
|
|
|
* Mishandled by bb_process_escape_sequence:
|
|
|
|
* \v version of bash (e.g., 2.00)
|
|
|
|
*/
|
2013-03-29 13:09:05 +01:00
|
|
|
cp = prmt_ptr;
|
|
|
|
c = *cp;
|
|
|
|
if (c != 't') /* don't treat \t as tab */
|
|
|
|
c = bb_process_escape_sequence(&prmt_ptr);
|
2007-01-21 19:18:19 +00:00
|
|
|
if (prmt_ptr == cp) {
|
2007-11-10 01:35:47 +00:00
|
|
|
if (*cp == '\0')
|
2007-01-21 19:18:19 +00:00
|
|
|
break;
|
|
|
|
c = *prmt_ptr++;
|
2007-12-03 10:45:14 +00:00
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
switch (c) {
|
|
|
|
case 'u':
|
2007-09-27 10:17:16 +00:00
|
|
|
pbuf = user_buf ? user_buf : (char*)"";
|
2007-01-21 19:18:19 +00:00
|
|
|
break;
|
2013-03-28 13:20:12 +01:00
|
|
|
case 'H':
|
2007-01-21 19:18:19 +00:00
|
|
|
case 'h':
|
2008-02-25 23:23:58 +00:00
|
|
|
pbuf = free_me = safe_gethostname();
|
2013-03-28 13:20:12 +01:00
|
|
|
if (c == 'h')
|
|
|
|
strchrnul(pbuf, '.')[0] = '\0';
|
2007-01-21 19:18:19 +00:00
|
|
|
break;
|
|
|
|
case '$':
|
2007-01-21 19:19:46 +00:00
|
|
|
c = (geteuid() == 0 ? '#' : '$');
|
2007-01-21 19:18:19 +00:00
|
|
|
break;
|
2013-03-29 13:09:05 +01:00
|
|
|
case 'T': /* 12-hour HH:MM:SS format */
|
|
|
|
case '@': /* 12-hour am/pm format */
|
|
|
|
case 'A': /* 24-hour HH:MM format */
|
|
|
|
case 't': /* 24-hour HH:MM:SS format */
|
|
|
|
/* We show all of them as 24-hour HH:MM */
|
|
|
|
strftime_HHMMSS(timebuf, sizeof(timebuf), NULL)[-3] = '\0';
|
|
|
|
pbuf = timebuf;
|
|
|
|
break;
|
|
|
|
case 'w': /* current dir */
|
|
|
|
case 'W': /* basename of cur dir */
|
|
|
|
if (!cwd_buf) {
|
|
|
|
cwd_buf = xrealloc_getcwd_or_warn(NULL);
|
|
|
|
if (!cwd_buf)
|
|
|
|
cwd_buf = (char *)bb_msg_unknown;
|
2015-03-12 17:48:34 +01:00
|
|
|
else if (home_pwd_buf[0]) {
|
|
|
|
char *after_home_user;
|
|
|
|
|
2013-03-29 13:09:05 +01:00
|
|
|
/* /home/user[/something] -> ~[/something] */
|
2015-03-12 17:48:34 +01:00
|
|
|
after_home_user = is_prefixed_with(cwd_buf, home_pwd_buf);
|
|
|
|
if (after_home_user
|
|
|
|
&& (*after_home_user == '/' || *after_home_user == '\0')
|
2013-03-29 13:09:05 +01:00
|
|
|
) {
|
|
|
|
cwd_buf[0] = '~';
|
2015-03-12 17:48:34 +01:00
|
|
|
overlapping_strcpy(cwd_buf + 1, after_home_user);
|
2013-03-29 13:09:05 +01:00
|
|
|
}
|
|
|
|
}
|
2007-01-21 19:18:19 +00:00
|
|
|
}
|
2007-12-03 10:45:14 +00:00
|
|
|
pbuf = cwd_buf;
|
2013-03-29 13:09:05 +01:00
|
|
|
if (c == 'w')
|
|
|
|
break;
|
2007-06-30 08:04:05 +00:00
|
|
|
cp = strrchr(pbuf, '/');
|
2013-03-29 13:21:53 +01:00
|
|
|
if (cp)
|
2013-03-29 13:09:05 +01:00
|
|
|
pbuf = (char*)cp + 1;
|
2007-01-21 19:18:19 +00:00
|
|
|
break;
|
2013-03-28 13:20:12 +01:00
|
|
|
// bb_process_escape_sequence does this now:
|
|
|
|
// case 'e': case 'E': /* \e \E = \033 */
|
|
|
|
// c = '\033';
|
|
|
|
// break;
|
2007-12-03 10:45:14 +00:00
|
|
|
case 'x': case 'X': {
|
|
|
|
char buf2[4];
|
2007-01-21 19:18:19 +00:00
|
|
|
for (l = 0; l < 3;) {
|
2007-12-03 10:45:14 +00:00
|
|
|
unsigned h;
|
2007-01-21 19:18:19 +00:00
|
|
|
buf2[l++] = *prmt_ptr;
|
2007-12-03 10:45:14 +00:00
|
|
|
buf2[l] = '\0';
|
|
|
|
h = strtoul(buf2, &pbuf, 16);
|
2007-01-21 19:18:19 +00:00
|
|
|
if (h > UCHAR_MAX || (pbuf - buf2) < l) {
|
2007-12-03 10:45:14 +00:00
|
|
|
buf2[--l] = '\0';
|
2007-01-21 19:18:19 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
prmt_ptr++;
|
|
|
|
}
|
2007-12-03 10:45:14 +00:00
|
|
|
c = (char)strtoul(buf2, NULL, 16);
|
2007-01-21 19:18:19 +00:00
|
|
|
if (c == 0)
|
|
|
|
c = '?';
|
2007-12-03 10:45:14 +00:00
|
|
|
pbuf = cbuf;
|
2007-01-21 19:18:19 +00:00
|
|
|
break;
|
2007-12-03 10:45:14 +00:00
|
|
|
}
|
2007-01-21 19:18:19 +00:00
|
|
|
case '[': case ']':
|
|
|
|
if (c == flg_not_length) {
|
2013-08-19 16:44:05 +02:00
|
|
|
/* Toggle '['/']' hex 5b/5d */
|
|
|
|
flg_not_length ^= 6;
|
2007-01-21 19:18:19 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
break;
|
2007-12-03 10:45:14 +00:00
|
|
|
} /* switch */
|
|
|
|
} /* if */
|
|
|
|
} /* if */
|
|
|
|
cbuf[0] = c;
|
2013-08-19 16:45:04 +02:00
|
|
|
{
|
|
|
|
int n = strlen(pbuf);
|
|
|
|
prmt_size += n;
|
|
|
|
if (c == '\n')
|
|
|
|
cmdedit_prmt_len = 0;
|
|
|
|
else if (flg_not_length != ']') {
|
2020-11-21 12:26:39 +01:00
|
|
|
#if ENABLE_UNICODE_SUPPORT
|
|
|
|
if (n == 1) {
|
|
|
|
/* Only count single-byte characters and the first of multi-byte characters */
|
|
|
|
if ((unsigned char)*pbuf < 0x80 /* single byte character */
|
|
|
|
|| (unsigned char)*pbuf >= 0xc0 /* first of multi-byte characters */
|
|
|
|
) {
|
|
|
|
cmdedit_prmt_len += n;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
cmdedit_prmt_len += unicode_strwidth(pbuf);
|
|
|
|
}
|
2013-08-19 16:44:05 +02:00
|
|
|
#else
|
2013-08-19 16:45:04 +02:00
|
|
|
cmdedit_prmt_len += n;
|
2013-08-19 16:44:05 +02:00
|
|
|
#endif
|
2013-08-19 16:45:04 +02:00
|
|
|
}
|
2013-08-19 16:44:05 +02:00
|
|
|
}
|
2013-08-19 16:45:04 +02:00
|
|
|
prmt_mem_ptr = strcat(xrealloc(prmt_mem_ptr, prmt_size+1), pbuf);
|
2007-12-03 10:45:14 +00:00
|
|
|
free(free_me);
|
|
|
|
} /* while */
|
|
|
|
|
|
|
|
if (cwd_buf != (char *)bb_msg_unknown)
|
|
|
|
free(cwd_buf);
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
/* see comment (above this function) about multiline prompt redrawing */
|
|
|
|
cmdedit_prompt = prompt_last_line = prmt_mem_ptr;
|
|
|
|
prmt_ptr = strrchr(cmdedit_prompt, '\n');
|
|
|
|
if (prmt_ptr)
|
|
|
|
prompt_last_line = prmt_ptr + 1;
|
2007-01-21 19:18:19 +00:00
|
|
|
put_prompt();
|
|
|
|
}
|
2020-12-16 11:14:08 +01:00
|
|
|
#endif /* FEATURE_EDITING_FANCY_PROMPT */
|
2005-08-04 19:04:46 +00:00
|
|
|
|
2018-02-25 20:09:54 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_WINCH
|
2016-11-28 01:10:16 +01:00
|
|
|
static void cmdedit_setwidth(void)
|
2007-01-21 19:19:46 +00:00
|
|
|
{
|
2016-11-28 01:10:16 +01:00
|
|
|
int new_y;
|
|
|
|
|
|
|
|
cmdedit_termw = get_terminal_width(STDIN_FILENO);
|
|
|
|
/* new y for current cursor */
|
|
|
|
new_y = (cursor + cmdedit_prmt_len) / cmdedit_termw;
|
|
|
|
/* redraw */
|
|
|
|
redraw((new_y >= cmdedit_y ? new_y : cmdedit_y), command_len - cursor);
|
2007-01-21 19:19:46 +00:00
|
|
|
}
|
|
|
|
|
2016-11-27 22:25:07 +01:00
|
|
|
static void win_changed(int nsig UNUSED_PARAM)
|
2007-01-21 19:19:46 +00:00
|
|
|
{
|
2016-11-27 22:25:07 +01:00
|
|
|
if (S.ok_to_redraw) {
|
|
|
|
/* We are in read_key(), safe to redraw immediately */
|
|
|
|
int sv_errno = errno;
|
2016-11-28 01:10:16 +01:00
|
|
|
cmdedit_setwidth();
|
|
|
|
fflush_all();
|
2016-11-27 22:25:07 +01:00
|
|
|
errno = sv_errno;
|
|
|
|
} else {
|
|
|
|
/* Signal main loop that redraw is necessary */
|
|
|
|
S.SIGWINCH_count++;
|
|
|
|
}
|
2007-01-21 19:19:46 +00:00
|
|
|
}
|
2018-02-25 20:09:54 +01:00
|
|
|
#endif
|
2007-01-21 19:19:46 +00:00
|
|
|
|
2011-02-08 05:07:02 +01:00
|
|
|
static int lineedit_read_key(char *read_key_buffer, int timeout)
|
2009-05-15 03:27:53 +02:00
|
|
|
{
|
2009-05-17 16:44:54 +02:00
|
|
|
int64_t ic;
|
2010-03-26 14:06:56 +01:00
|
|
|
#if ENABLE_UNICODE_SUPPORT
|
2009-07-10 18:40:49 +02:00
|
|
|
char unicode_buf[MB_CUR_MAX + 1];
|
|
|
|
int unicode_idx = 0;
|
|
|
|
#endif
|
2009-05-17 16:44:54 +02:00
|
|
|
|
2016-11-28 01:10:16 +01:00
|
|
|
fflush_all();
|
2010-03-11 21:17:55 +01:00
|
|
|
while (1) {
|
|
|
|
/* Wait for input. TIMEOUT = -1 makes read_key wait even
|
|
|
|
* on nonblocking stdin, TIMEOUT = 50 makes sure we won't
|
|
|
|
* insist on full MB_CUR_MAX buffer to declare input like
|
|
|
|
* "\xff\n",pause,"ls\n" invalid and thus won't lose "ls".
|
|
|
|
*
|
|
|
|
* Note: read_key sets errno to 0 on success.
|
|
|
|
*/
|
2018-02-25 20:09:54 +01:00
|
|
|
IF_FEATURE_EDITING_WINCH(S.ok_to_redraw = 1;)
|
2010-03-11 21:17:55 +01:00
|
|
|
ic = read_key(STDIN_FILENO, read_key_buffer, timeout);
|
2018-02-25 20:09:54 +01:00
|
|
|
IF_FEATURE_EDITING_WINCH(S.ok_to_redraw = 0;)
|
2010-03-11 21:17:55 +01:00
|
|
|
if (errno) {
|
2010-03-26 14:06:56 +01:00
|
|
|
#if ENABLE_UNICODE_SUPPORT
|
2010-03-11 21:17:55 +01:00
|
|
|
if (errno == EAGAIN && unicode_idx != 0)
|
|
|
|
goto pushback;
|
2009-10-29 03:45:26 +01:00
|
|
|
#endif
|
2010-03-11 21:17:55 +01:00
|
|
|
break;
|
2009-05-29 10:39:06 +02:00
|
|
|
}
|
2009-10-14 12:53:04 +02:00
|
|
|
|
2009-10-27 10:34:06 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_ASK_TERMINAL
|
|
|
|
if ((int32_t)ic == KEYCODE_CURSOR_POS
|
2009-10-27 10:47:49 +01:00
|
|
|
&& S.sent_ESC_br6n
|
2009-05-17 16:44:54 +02:00
|
|
|
) {
|
2009-10-27 10:47:49 +01:00
|
|
|
S.sent_ESC_br6n = 0;
|
2009-10-27 10:34:06 +01:00
|
|
|
if (cursor == 0) { /* otherwise it may be bogus */
|
|
|
|
int col = ((ic >> 32) & 0x7fff) - 1;
|
2013-08-19 16:44:05 +02:00
|
|
|
/*
|
|
|
|
* Is col > cmdedit_prmt_len?
|
|
|
|
* If yes (terminal says cursor is farther to the right
|
|
|
|
* of where we think it should be),
|
|
|
|
* the prompt wasn't printed starting at col 1,
|
|
|
|
* there was additional text before it.
|
|
|
|
*/
|
|
|
|
if ((int)(col - cmdedit_prmt_len) > 0) {
|
|
|
|
/* Fix our understanding of current x position */
|
2009-10-27 10:34:06 +01:00
|
|
|
cmdedit_x += (col - cmdedit_prmt_len);
|
|
|
|
while (cmdedit_x >= cmdedit_termw) {
|
|
|
|
cmdedit_x -= cmdedit_termw;
|
|
|
|
cmdedit_y++;
|
|
|
|
}
|
2009-05-17 16:44:54 +02:00
|
|
|
}
|
|
|
|
}
|
2010-03-11 21:17:55 +01:00
|
|
|
continue;
|
2009-05-17 16:44:54 +02:00
|
|
|
}
|
2009-10-27 10:34:06 +01:00
|
|
|
#endif
|
2009-07-10 18:40:49 +02:00
|
|
|
|
2010-03-26 14:06:56 +01:00
|
|
|
#if ENABLE_UNICODE_SUPPORT
|
2010-03-09 14:09:24 +01:00
|
|
|
if (unicode_status == UNICODE_ON) {
|
2009-07-10 18:40:49 +02:00
|
|
|
wchar_t wc;
|
|
|
|
|
|
|
|
if ((int32_t)ic < 0) /* KEYCODE_xxx */
|
2010-03-11 21:17:55 +01:00
|
|
|
break;
|
|
|
|
// TODO: imagine sequence like: 0xff,<left-arrow>: we are currently losing 0xff...
|
2010-03-09 14:09:24 +01:00
|
|
|
|
2009-07-10 18:40:49 +02:00
|
|
|
unicode_buf[unicode_idx++] = ic;
|
|
|
|
unicode_buf[unicode_idx] = '\0';
|
2010-03-09 14:09:24 +01:00
|
|
|
if (mbstowcs(&wc, unicode_buf, 1) != 1) {
|
|
|
|
/* Not (yet?) a valid unicode char */
|
|
|
|
if (unicode_idx < MB_CUR_MAX) {
|
2010-03-11 21:17:55 +01:00
|
|
|
timeout = 50;
|
|
|
|
continue;
|
2010-03-09 14:09:24 +01:00
|
|
|
}
|
2010-03-11 21:17:55 +01:00
|
|
|
pushback:
|
2010-03-09 14:09:24 +01:00
|
|
|
/* Invalid sequence. Save all "bad bytes" except first */
|
2010-03-11 21:17:55 +01:00
|
|
|
read_key_ungets(read_key_buffer, unicode_buf + 1, unicode_idx - 1);
|
2010-04-29 13:43:39 +02:00
|
|
|
# if !ENABLE_UNICODE_PRESERVE_BROKEN
|
2010-03-09 14:09:24 +01:00
|
|
|
ic = CONFIG_SUBST_WCHAR;
|
2010-04-29 13:43:39 +02:00
|
|
|
# else
|
2010-05-16 20:46:53 +02:00
|
|
|
ic = unicode_mark_raw_byte(unicode_buf[0]);
|
2010-04-29 13:43:39 +02:00
|
|
|
# endif
|
2010-03-09 14:09:24 +01:00
|
|
|
} else {
|
|
|
|
/* Valid unicode char, return its code */
|
|
|
|
ic = wc;
|
2009-07-10 18:40:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2010-03-11 21:17:55 +01:00
|
|
|
break;
|
|
|
|
}
|
2009-07-10 18:40:49 +02:00
|
|
|
|
2009-05-15 03:27:53 +02:00
|
|
|
return ic;
|
|
|
|
}
|
|
|
|
|
2010-03-18 18:35:37 +01:00
|
|
|
#if ENABLE_UNICODE_BIDI_SUPPORT
|
|
|
|
static int isrtl_str(void)
|
|
|
|
{
|
|
|
|
int idx = cursor;
|
2010-03-26 13:13:24 +01:00
|
|
|
|
|
|
|
while (idx < command_len && unicode_bidi_is_neutral_wchar(command_ps[idx]))
|
2010-03-18 18:35:37 +01:00
|
|
|
idx++;
|
2010-03-26 13:13:24 +01:00
|
|
|
return unicode_bidi_isrtl(command_ps[idx]);
|
2010-03-18 18:35:37 +01:00
|
|
|
}
|
|
|
|
#else
|
|
|
|
# define isrtl_str() 0
|
|
|
|
#endif
|
|
|
|
|
2006-03-07 20:26:11 +00:00
|
|
|
/* leave out the "vi-mode"-only case labels if vi editing isn't
|
|
|
|
* configured. */
|
2009-07-12 00:51:15 +02:00
|
|
|
#define vi_case(caselabel) IF_FEATURE_EDITING_VI(case caselabel)
|
2005-08-04 19:04:46 +00:00
|
|
|
|
|
|
|
/* convert uppercase ascii to equivalent control char, for readability */
|
2007-01-21 19:18:19 +00:00
|
|
|
#undef CTRL
|
|
|
|
#define CTRL(a) ((a) & ~0x40)
|
2005-08-04 19:04:46 +00:00
|
|
|
|
2011-07-11 07:36:59 +02:00
|
|
|
enum {
|
|
|
|
VI_CMDMODE_BIT = 0x40000000,
|
|
|
|
/* 0x80000000 bit flags KEYCODE_xxx */
|
|
|
|
};
|
|
|
|
|
|
|
|
#if ENABLE_FEATURE_REVERSE_SEARCH
|
|
|
|
/* Mimic readline Ctrl-R reverse history search.
|
|
|
|
* When invoked, it shows the following prompt:
|
|
|
|
* (reverse-i-search)'': user_input [cursor pos unchanged by Ctrl-R]
|
|
|
|
* and typing results in search being performed:
|
|
|
|
* (reverse-i-search)'tmp': cd /tmp [cursor under t in /tmp]
|
|
|
|
* Search is performed by looking at progressively older lines in history.
|
|
|
|
* Ctrl-R again searches for the next match in history.
|
|
|
|
* Backspace deletes last matched char.
|
|
|
|
* Control keys exit search and return to normal editing (at current history line).
|
|
|
|
*/
|
2017-08-02 17:27:28 +02:00
|
|
|
static int32_t reverse_i_search(int timeout)
|
2011-07-11 07:36:59 +02:00
|
|
|
{
|
|
|
|
char match_buf[128]; /* for user input */
|
|
|
|
char read_key_buffer[KEYCODE_BUFFER_SIZE];
|
|
|
|
const char *matched_history_line;
|
|
|
|
const char *saved_prompt;
|
2013-08-19 16:44:05 +02:00
|
|
|
unsigned saved_prmt_len;
|
2011-07-11 07:36:59 +02:00
|
|
|
int32_t ic;
|
|
|
|
|
|
|
|
matched_history_line = NULL;
|
|
|
|
read_key_buffer[0] = 0;
|
|
|
|
match_buf[0] = '\0';
|
|
|
|
|
|
|
|
/* Save and replace the prompt */
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
saved_prompt = prompt_last_line;
|
2013-08-19 16:44:05 +02:00
|
|
|
saved_prmt_len = cmdedit_prmt_len;
|
2011-07-11 07:36:59 +02:00
|
|
|
goto set_prompt;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
int h;
|
|
|
|
unsigned match_buf_len = strlen(match_buf);
|
|
|
|
|
2017-08-02 17:27:28 +02:00
|
|
|
//FIXME: correct timeout? (i.e. count it down?)
|
|
|
|
ic = lineedit_read_key(read_key_buffer, timeout);
|
2011-07-11 07:36:59 +02:00
|
|
|
|
|
|
|
switch (ic) {
|
|
|
|
case CTRL('R'): /* searching for the next match */
|
|
|
|
break;
|
|
|
|
|
|
|
|
case '\b':
|
|
|
|
case '\x7f':
|
|
|
|
/* Backspace */
|
|
|
|
if (unicode_status == UNICODE_ON) {
|
|
|
|
while (match_buf_len != 0) {
|
|
|
|
uint8_t c = match_buf[--match_buf_len];
|
|
|
|
if ((c & 0xc0) != 0x80) /* start of UTF-8 char? */
|
|
|
|
break; /* yes */
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (match_buf_len != 0)
|
|
|
|
match_buf_len--;
|
|
|
|
}
|
|
|
|
match_buf[match_buf_len] = '\0';
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (ic < ' '
|
|
|
|
|| (!ENABLE_UNICODE_SUPPORT && ic >= 256)
|
|
|
|
|| (ENABLE_UNICODE_SUPPORT && ic >= VI_CMDMODE_BIT)
|
|
|
|
) {
|
|
|
|
goto ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Append this char */
|
|
|
|
#if ENABLE_UNICODE_SUPPORT
|
|
|
|
if (unicode_status == UNICODE_ON) {
|
|
|
|
mbstate_t mbstate = { 0 };
|
|
|
|
char buf[MB_CUR_MAX + 1];
|
|
|
|
int len = wcrtomb(buf, ic, &mbstate);
|
|
|
|
if (len > 0) {
|
|
|
|
buf[len] = '\0';
|
|
|
|
if (match_buf_len + len < sizeof(match_buf))
|
|
|
|
strcpy(match_buf + match_buf_len, buf);
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
#endif
|
|
|
|
if (match_buf_len < sizeof(match_buf) - 1) {
|
|
|
|
match_buf[match_buf_len] = ic;
|
|
|
|
match_buf[match_buf_len + 1] = '\0';
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
} /* switch (ic) */
|
|
|
|
|
|
|
|
/* Search in history for match_buf */
|
|
|
|
h = state->cur_history;
|
|
|
|
if (ic == CTRL('R'))
|
|
|
|
h--;
|
|
|
|
while (h >= 0) {
|
|
|
|
if (state->history[h]) {
|
|
|
|
char *match = strstr(state->history[h], match_buf);
|
|
|
|
if (match) {
|
|
|
|
state->cur_history = h;
|
|
|
|
matched_history_line = state->history[h];
|
|
|
|
command_len = load_string(matched_history_line);
|
|
|
|
cursor = match - matched_history_line;
|
|
|
|
//FIXME: cursor position for Unicode case
|
|
|
|
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
free((char*)prompt_last_line);
|
2011-07-11 07:36:59 +02:00
|
|
|
set_prompt:
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
prompt_last_line = xasprintf("(reverse-i-search)'%s': ", match_buf);
|
|
|
|
cmdedit_prmt_len = unicode_strwidth(prompt_last_line);
|
2011-07-11 07:36:59 +02:00
|
|
|
goto do_redraw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
h--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Not found */
|
|
|
|
match_buf[match_buf_len] = '\0';
|
|
|
|
beep();
|
|
|
|
continue;
|
|
|
|
|
|
|
|
do_redraw:
|
|
|
|
redraw(cmdedit_y, command_len - cursor);
|
|
|
|
} /* while (1) */
|
|
|
|
|
|
|
|
ret:
|
|
|
|
if (matched_history_line)
|
|
|
|
command_len = load_string(matched_history_line);
|
|
|
|
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
free((char*)prompt_last_line);
|
|
|
|
prompt_last_line = saved_prompt;
|
2013-08-19 16:44:05 +02:00
|
|
|
cmdedit_prmt_len = saved_prmt_len;
|
2011-07-11 07:36:59 +02:00
|
|
|
redraw(cmdedit_y, command_len - cursor);
|
|
|
|
|
|
|
|
return ic;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2018-12-08 15:45:46 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_WINCH
|
2018-12-08 13:49:15 +01:00
|
|
|
static void sigaction2(int sig, struct sigaction *act)
|
|
|
|
{
|
|
|
|
// Grr... gcc 8.1.1:
|
|
|
|
// "passing argument 3 to restrict-qualified parameter aliases with argument 2"
|
|
|
|
// dance around that...
|
|
|
|
struct sigaction *oact FIX_ALIASING;
|
|
|
|
oact = act;
|
|
|
|
sigaction(sig, act, oact);
|
|
|
|
}
|
2018-12-08 15:45:46 +01:00
|
|
|
#endif
|
2018-12-08 13:49:15 +01:00
|
|
|
|
2009-07-16 14:14:34 +02:00
|
|
|
/* maxsize must be >= 2.
|
|
|
|
* Returns:
|
2008-02-02 18:35:55 +00:00
|
|
|
* -1 on read errors or EOF, or on bare Ctrl-D,
|
|
|
|
* 0 on ctrl-C (the line entered is still returned in 'command'),
|
2016-11-25 03:41:03 +01:00
|
|
|
* (in both cases the cursor remains on the input line, '\n' is not printed)
|
2007-09-25 18:35:28 +00:00
|
|
|
* >0 length of input string, including terminating '\n'
|
|
|
|
*/
|
2017-08-02 17:27:28 +02:00
|
|
|
int FAST_FUNC read_line_input(line_input_t *st, const char *prompt, char *command, int maxsize)
|
2000-03-16 08:09:57 +00:00
|
|
|
{
|
2017-09-15 17:14:01 +02:00
|
|
|
int len, n;
|
2017-08-02 17:27:28 +02:00
|
|
|
int timeout;
|
2007-11-10 01:35:47 +00:00
|
|
|
#if ENABLE_FEATURE_TAB_COMPLETION
|
2010-09-03 12:51:36 +02:00
|
|
|
smallint lastWasTab = 0;
|
2007-11-10 01:35:47 +00:00
|
|
|
#endif
|
2007-01-21 19:18:19 +00:00
|
|
|
smallint break_out = 0;
|
2007-01-22 09:03:07 +00:00
|
|
|
#if ENABLE_FEATURE_EDITING_VI
|
2007-01-21 19:18:19 +00:00
|
|
|
smallint vi_cmdmode = 0;
|
2005-08-04 19:04:46 +00:00
|
|
|
#endif
|
2007-11-10 01:35:47 +00:00
|
|
|
struct termios initial_settings;
|
|
|
|
struct termios new_settings;
|
2009-05-15 03:27:53 +02:00
|
|
|
char read_key_buffer[KEYCODE_BUFFER_SIZE];
|
2007-01-22 07:21:38 +00:00
|
|
|
|
2007-11-10 01:35:47 +00:00
|
|
|
INIT_S();
|
2020-12-21 22:50:23 +01:00
|
|
|
//command_len = 0; - done by INIT_S()
|
|
|
|
//cmdedit_y = 0; /* quasireal y, not true if line > xt*yt */
|
|
|
|
cmdedit_termw = 80;
|
|
|
|
IF_USERNAME_OR_HOMEDIR(home_pwd_buf = (char*)null_str;)
|
|
|
|
IF_FEATURE_EDITING_VI(delptr = delbuf;)
|
2007-11-10 01:35:47 +00:00
|
|
|
|
2017-09-15 17:14:01 +02:00
|
|
|
n = get_termios_and_make_raw(STDIN_FILENO, &new_settings, &initial_settings, 0
|
|
|
|
| TERMIOS_CLEAR_ISIG /* turn off INTR (ctrl-C), QUIT, SUSP */
|
|
|
|
);
|
|
|
|
if (n != 0 || (initial_settings.c_lflag & (ECHO|ICANON)) == ICANON) {
|
2014-12-10 17:22:13 +01:00
|
|
|
/* Happens when e.g. stty -echo was run before.
|
|
|
|
* But if ICANON is not set, we don't come here.
|
|
|
|
* (example: interactive python ^Z-backgrounded,
|
|
|
|
* tty is still in "raw mode").
|
|
|
|
*/
|
2007-11-10 01:35:47 +00:00
|
|
|
parse_and_put_prompt(prompt);
|
2016-11-28 01:10:16 +01:00
|
|
|
fflush_all();
|
2007-12-09 10:03:28 +00:00
|
|
|
if (fgets(command, maxsize, stdin) == NULL)
|
|
|
|
len = -1; /* EOF or error */
|
|
|
|
else
|
|
|
|
len = strlen(command);
|
2007-11-10 01:35:47 +00:00
|
|
|
DEINIT_S();
|
|
|
|
return len;
|
2007-10-20 18:30:38 +00:00
|
|
|
}
|
|
|
|
|
2010-01-04 20:49:58 +01:00
|
|
|
init_unicode();
|
2009-07-11 21:36:13 +02:00
|
|
|
|
2007-01-22 07:21:38 +00:00
|
|
|
// FIXME: audit & improve this
|
2007-06-10 15:08:44 +00:00
|
|
|
if (maxsize > MAX_LINELEN)
|
|
|
|
maxsize = MAX_LINELEN;
|
2009-07-12 02:50:35 +02:00
|
|
|
S.maxsize = maxsize;
|
2007-01-22 07:21:38 +00:00
|
|
|
|
2017-08-02 17:27:28 +02:00
|
|
|
timeout = -1;
|
|
|
|
/* Make state->flags == 0 if st is NULL.
|
|
|
|
* With zeroed flags, no other fields are ever referenced.
|
|
|
|
*/
|
|
|
|
state = (line_input_t*) &const_int_0;
|
|
|
|
if (st) {
|
|
|
|
state = st;
|
|
|
|
timeout = st->timeout;
|
|
|
|
}
|
2009-09-27 02:48:53 +02:00
|
|
|
#if MAX_HISTORY > 0
|
2019-02-04 16:16:30 +01:00
|
|
|
if (state->flags & DO_HISTORY) {
|
2009-09-27 02:48:53 +02:00
|
|
|
# if ENABLE_FEATURE_EDITING_SAVEHISTORY
|
2019-02-04 16:16:30 +01:00
|
|
|
if (state->hist_file)
|
|
|
|
if (state->cnt_history == 0)
|
|
|
|
load_history(state);
|
2009-09-27 02:48:53 +02:00
|
|
|
# endif
|
2008-11-02 00:41:05 +00:00
|
|
|
state->cur_history = state->cnt_history;
|
2019-02-04 16:16:30 +01:00
|
|
|
}
|
2009-09-27 02:48:53 +02:00
|
|
|
#endif
|
2007-01-22 07:21:38 +00:00
|
|
|
|
2001-02-16 18:36:04 +00:00
|
|
|
/* prepare before init handlers */
|
2010-03-26 14:06:56 +01:00
|
|
|
#if ENABLE_UNICODE_SUPPORT
|
2009-07-10 18:40:49 +02:00
|
|
|
command_ps = xzalloc(maxsize * sizeof(command_ps[0]));
|
|
|
|
#else
|
2001-01-26 20:42:23 +00:00
|
|
|
command_ps = command;
|
2007-01-21 19:18:59 +00:00
|
|
|
command[0] = '\0';
|
2009-07-10 18:40:49 +02:00
|
|
|
#endif
|
|
|
|
#define command command_must_not_be_used
|
2001-01-26 20:42:23 +00:00
|
|
|
|
2008-11-05 13:20:58 +00:00
|
|
|
tcsetattr_stdin_TCSANOW(&new_settings);
|
2001-01-26 20:42:23 +00:00
|
|
|
|
2010-07-18 22:21:24 +02:00
|
|
|
#if ENABLE_USERNAME_OR_HOMEDIR
|
2007-01-22 07:30:26 +00:00
|
|
|
{
|
|
|
|
struct passwd *entry;
|
|
|
|
|
|
|
|
entry = getpwuid(geteuid());
|
|
|
|
if (entry) {
|
|
|
|
user_buf = xstrdup(entry->pw_name);
|
|
|
|
home_pwd_buf = xstrdup(entry->pw_dir);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2008-09-27 01:28:56 +00:00
|
|
|
|
|
|
|
#if 0
|
2011-03-31 13:16:52 +02:00
|
|
|
for (i = 0; i <= state->max_history; i++)
|
2009-07-10 18:40:49 +02:00
|
|
|
bb_error_msg("history[%d]:'%s'", i, state->history[i]);
|
2008-09-27 01:28:56 +00:00
|
|
|
bb_error_msg("cur_history:%d cnt_history:%d", state->cur_history, state->cnt_history);
|
|
|
|
#endif
|
|
|
|
|
2017-11-08 13:38:12 +01:00
|
|
|
/* Get width (before printing prompt) */
|
|
|
|
cmdedit_termw = get_terminal_width(STDIN_FILENO);
|
2009-11-11 03:19:30 +01:00
|
|
|
/* Print out the command prompt, optionally ask where cursor is */
|
2007-11-10 01:35:47 +00:00
|
|
|
parse_and_put_prompt(prompt);
|
2009-11-11 03:19:30 +01:00
|
|
|
ask_terminal();
|
2001-01-04 11:08:45 +00:00
|
|
|
|
2018-02-25 20:09:54 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_WINCH
|
2011-05-20 04:26:29 +02:00
|
|
|
/* Install window resize handler (NB: after *all* init is complete) */
|
2016-11-27 22:25:07 +01:00
|
|
|
S.SIGWINCH_handler.sa_handler = win_changed;
|
|
|
|
S.SIGWINCH_handler.sa_flags = SA_RESTART;
|
2018-12-08 13:49:15 +01:00
|
|
|
sigaction2(SIGWINCH, &S.SIGWINCH_handler);
|
2018-02-25 20:09:54 +01:00
|
|
|
#endif
|
2009-05-17 19:28:14 +02:00
|
|
|
read_key_buffer[0] = 0;
|
2000-03-19 05:28:55 +00:00
|
|
|
while (1) {
|
2009-07-10 18:40:49 +02:00
|
|
|
/*
|
|
|
|
* The emacs and vi modes share much of the code in the big
|
|
|
|
* command loop. Commands entered when in vi's command mode
|
|
|
|
* (aka "escape mode") get an extra bit added to distinguish
|
|
|
|
* them - this keeps them from being self-inserted. This
|
|
|
|
* clutters the big switch a bit, but keeps all the code
|
|
|
|
* in one place.
|
|
|
|
*/
|
2009-10-14 12:53:04 +02:00
|
|
|
int32_t ic, ic_raw;
|
2018-02-25 20:09:54 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_WINCH
|
2016-11-27 22:25:07 +01:00
|
|
|
unsigned count;
|
|
|
|
|
|
|
|
count = S.SIGWINCH_count;
|
|
|
|
if (S.SIGWINCH_saved != count) {
|
|
|
|
S.SIGWINCH_saved = count;
|
2016-11-28 01:10:16 +01:00
|
|
|
cmdedit_setwidth();
|
2016-11-27 22:25:07 +01:00
|
|
|
}
|
2018-02-25 20:09:54 +01:00
|
|
|
#endif
|
2011-02-08 05:07:02 +01:00
|
|
|
ic = ic_raw = lineedit_read_key(read_key_buffer, timeout);
|
2006-03-07 20:26:11 +00:00
|
|
|
|
2011-07-11 07:36:59 +02:00
|
|
|
#if ENABLE_FEATURE_REVERSE_SEARCH
|
|
|
|
again:
|
|
|
|
#endif
|
2007-01-22 09:03:07 +00:00
|
|
|
#if ENABLE_FEATURE_EDITING_VI
|
2005-08-04 19:04:46 +00:00
|
|
|
newdelflag = 1;
|
2009-05-15 03:27:53 +02:00
|
|
|
if (vi_cmdmode) {
|
|
|
|
/* btw, since KEYCODE_xxx are all < 0, this doesn't
|
|
|
|
* change ic if it contains one of them: */
|
|
|
|
ic |= VI_CMDMODE_BIT;
|
|
|
|
}
|
2005-08-04 19:04:46 +00:00
|
|
|
#endif
|
2009-05-15 03:27:53 +02:00
|
|
|
|
2006-12-21 22:27:10 +00:00
|
|
|
switch (ic) {
|
2000-04-12 17:49:52 +00:00
|
|
|
case '\n':
|
|
|
|
case '\r':
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case('\n'|VI_CMDMODE_BIT:)
|
|
|
|
vi_case('\r'|VI_CMDMODE_BIT:)
|
2000-04-12 17:49:52 +00:00
|
|
|
/* Enter */
|
2001-02-16 18:36:04 +00:00
|
|
|
goto_new_line();
|
2000-04-12 17:49:52 +00:00
|
|
|
break_out = 1;
|
|
|
|
break;
|
2007-01-21 19:18:19 +00:00
|
|
|
case CTRL('A'):
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case('0'|VI_CMDMODE_BIT:)
|
2000-03-19 05:28:55 +00:00
|
|
|
/* Control-a -- Beginning of line */
|
2001-02-16 18:36:04 +00:00
|
|
|
input_backward(cursor);
|
2001-01-26 20:42:23 +00:00
|
|
|
break;
|
2007-01-21 19:18:19 +00:00
|
|
|
case CTRL('B'):
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case('h'|VI_CMDMODE_BIT:)
|
2010-03-18 18:35:37 +01:00
|
|
|
vi_case('\b'|VI_CMDMODE_BIT:) /* ^H */
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case('\x7f'|VI_CMDMODE_BIT:) /* DEL */
|
2010-03-18 18:35:37 +01:00
|
|
|
input_backward(1); /* Move back one character */
|
2000-04-21 01:26:49 +00:00
|
|
|
break;
|
2007-01-21 19:18:19 +00:00
|
|
|
case CTRL('E'):
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case('$'|VI_CMDMODE_BIT:)
|
2000-04-12 17:49:52 +00:00
|
|
|
/* Control-e -- End of line */
|
2010-05-11 14:49:13 +02:00
|
|
|
put_till_end_and_adv_cursor();
|
2000-04-12 17:49:52 +00:00
|
|
|
break;
|
2007-01-21 19:18:19 +00:00
|
|
|
case CTRL('F'):
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case('l'|VI_CMDMODE_BIT:)
|
|
|
|
vi_case(' '|VI_CMDMODE_BIT:)
|
2010-03-18 18:35:37 +01:00
|
|
|
input_forward(); /* Move forward one character */
|
2000-03-19 05:28:55 +00:00
|
|
|
break;
|
2010-03-18 18:35:37 +01:00
|
|
|
case '\b': /* ^H */
|
2007-01-21 19:18:19 +00:00
|
|
|
case '\x7f': /* DEL */
|
2010-03-18 18:35:37 +01:00
|
|
|
if (!isrtl_str())
|
|
|
|
input_backspace();
|
|
|
|
else
|
|
|
|
input_delete(0);
|
|
|
|
break;
|
|
|
|
case KEYCODE_DELETE:
|
|
|
|
if (!isrtl_str())
|
|
|
|
input_delete(0);
|
|
|
|
else
|
|
|
|
input_backspace();
|
2000-04-12 17:49:52 +00:00
|
|
|
break;
|
2007-11-10 01:35:47 +00:00
|
|
|
#if ENABLE_FEATURE_TAB_COMPLETION
|
2000-04-12 17:49:52 +00:00
|
|
|
case '\t':
|
2001-02-16 18:36:04 +00:00
|
|
|
input_tab(&lastWasTab);
|
2000-03-19 05:28:55 +00:00
|
|
|
break;
|
2007-11-10 01:35:47 +00:00
|
|
|
#endif
|
2007-01-21 19:18:19 +00:00
|
|
|
case CTRL('K'):
|
2003-07-28 09:56:35 +00:00
|
|
|
/* Control-k -- clear to end of line */
|
2009-07-10 18:40:49 +02:00
|
|
|
command_ps[cursor] = BB_NUL;
|
2007-01-22 07:21:38 +00:00
|
|
|
command_len = cursor;
|
2010-05-11 14:49:13 +02:00
|
|
|
printf(SEQ_CLEAR_TILL_END_OF_SCREEN);
|
2002-04-13 13:26:49 +00:00
|
|
|
break;
|
2007-01-21 19:18:19 +00:00
|
|
|
case CTRL('L'):
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case(CTRL('L')|VI_CMDMODE_BIT:)
|
2003-12-23 20:24:51 +00:00
|
|
|
/* Control-l -- clear screen */
|
lineedit: improve multiline PS1 - redraw using last PS1 line. Closes 10381
This patch only affects prompts with newlines.
We redraw the prompt [+ input] occasionally, e.g. during tab completion,
history browsing or search, etc, and we expect it to align with prior
redraws, such that the visible effect is that only the input changes.
With multi-line PS1, redraw always printed the prompt some lines below
the old one, which resulted in terminal scroll during every redraw.
Now we only redraw the last PS1 line, so vertical alignment is easier to
manage (we already calculated it using only the last line, but re-drew
all lines - that was the culprit), which fixes those extra scrolls.
Notes:
- We now use the full prompt for the initial draw, after clear-screen (^L),
and after tab-completion choices are displayed. Everything else now
redraws using the last/sole prompt line.
- During terminal resize we now only redraw the last[/sole] prompt line,
which is arguably better because it's hard to do right (and we never did).
- Good side effect for reverse-i-search: its prompt now replaces only the
last line of the original prompt - like other shells do.
function old new delta
put_prompt_custom - 66 +66
draw_custom - 66 +66
parse_and_put_prompt 766 806 +40
read_line_input 3867 3884 +17
input_tab 1069 1076 +7
cmdedit_setwidth 61 63 +2
redraw 59 47 -12
put_prompt 46 - -46
------------------------------------------------------------------------------
(add/remove: 2/1 grow/shrink: 4/1 up/down: 198/-58) Total: 140 bytes
Signed-off-by: Avi Halachmi <avihpit@yahoo.com>
Signed-off-by: Denys Vlasenko <vda.linux@googlemail.com>
2017-10-12 16:38:35 +02:00
|
|
|
/* cursor to top,left; clear to the end of screen */
|
|
|
|
printf(ESC"[H" ESC"[J");
|
|
|
|
draw_full(command_len - cursor);
|
2001-12-21 11:20:15 +00:00
|
|
|
break;
|
2006-11-02 22:09:37 +00:00
|
|
|
#if MAX_HISTORY > 0
|
2007-01-21 19:18:19 +00:00
|
|
|
case CTRL('N'):
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case(CTRL('N')|VI_CMDMODE_BIT:)
|
|
|
|
vi_case('j'|VI_CMDMODE_BIT:)
|
2000-04-12 17:49:52 +00:00
|
|
|
/* Control-n -- Get next command in history */
|
2002-11-27 09:29:49 +00:00
|
|
|
if (get_next_history())
|
2000-04-12 17:49:52 +00:00
|
|
|
goto rewrite_line;
|
2000-03-19 05:28:55 +00:00
|
|
|
break;
|
2007-01-21 19:18:19 +00:00
|
|
|
case CTRL('P'):
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case(CTRL('P')|VI_CMDMODE_BIT:)
|
|
|
|
vi_case('k'|VI_CMDMODE_BIT:)
|
2000-04-12 17:49:52 +00:00
|
|
|
/* Control-p -- Get previous command from history */
|
2008-09-27 01:28:56 +00:00
|
|
|
if (get_previous_history())
|
2000-04-12 17:49:52 +00:00
|
|
|
goto rewrite_line;
|
2000-03-19 05:28:55 +00:00
|
|
|
break;
|
2002-11-27 09:29:49 +00:00
|
|
|
#endif
|
2007-01-21 19:18:19 +00:00
|
|
|
case CTRL('U'):
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case(CTRL('U')|VI_CMDMODE_BIT:)
|
2001-02-16 18:36:04 +00:00
|
|
|
/* Control-U -- Clear line before cursor */
|
|
|
|
if (cursor) {
|
2007-01-22 07:21:38 +00:00
|
|
|
command_len -= cursor;
|
2009-07-10 18:40:49 +02:00
|
|
|
memmove(command_ps, command_ps + cursor,
|
|
|
|
(command_len + 1) * sizeof(command_ps[0]));
|
2007-01-22 07:21:38 +00:00
|
|
|
redraw(cmdedit_y, command_len);
|
2001-02-16 18:36:04 +00:00
|
|
|
}
|
|
|
|
break;
|
2007-01-21 19:18:19 +00:00
|
|
|
case CTRL('W'):
|
2009-05-15 03:27:53 +02:00
|
|
|
vi_case(CTRL('W')|VI_CMDMODE_BIT:)
|
2003-12-23 20:24:51 +00:00
|
|
|
/* Control-W -- Remove the last word */
|
2009-07-10 18:40:49 +02:00
|
|
|
while (cursor > 0 && BB_isspace(command_ps[cursor-1]))
|
2003-12-23 20:24:51 +00:00
|
|
|
input_backspace();
|
2009-07-10 18:40:49 +02:00
|
|
|
while (cursor > 0 && !BB_isspace(command_ps[cursor-1]))
|
2003-12-23 20:24:51 +00:00
|
|
|
input_backspace();
|
|
|
|
break;
|
2016-11-24 15:04:00 +01:00
|
|
|
case KEYCODE_ALT_D: {
|
|
|
|
/* Delete word forward */
|
|
|
|
int nc, sc = cursor;
|
|
|
|
ctrl_right();
|
|
|
|
nc = cursor - sc;
|
|
|
|
input_backward(nc);
|
|
|
|
while (--nc >= 0)
|
|
|
|
input_delete(1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case KEYCODE_ALT_BACKSPACE: {
|
|
|
|
/* Delete word backward */
|
|
|
|
int sc = cursor;
|
|
|
|
ctrl_left();
|
|
|
|
while (sc-- > cursor)
|
|
|
|
input_delete(1);
|
|
|
|
break;
|
|
|
|
}
|
2011-07-11 07:36:59 +02:00
|
|
|
#if ENABLE_FEATURE_REVERSE_SEARCH
|
|
|
|
case CTRL('R'):
|
2017-08-02 17:27:28 +02:00
|
|
|
ic = ic_raw = reverse_i_search(timeout);
|
2011-07-11 07:36:59 +02:00
|
|
|
goto again;
|
|
|
|
#endif
|
2007-01-21 19:21:21 +00:00
|
|
|
|
2007-01-22 09:03:07 +00:00
|
|
|
#if ENABLE_FEATURE_EDITING_VI
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'i'|VI_CMDMODE_BIT:
|
2005-08-04 19:04:46 +00:00
|
|
|
vi_cmdmode = 0;
|
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'I'|VI_CMDMODE_BIT:
|
2005-08-04 19:04:46 +00:00
|
|
|
input_backward(cursor);
|
|
|
|
vi_cmdmode = 0;
|
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'a'|VI_CMDMODE_BIT:
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
|
|
|
vi_cmdmode = 0;
|
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'A'|VI_CMDMODE_BIT:
|
2010-05-11 14:49:13 +02:00
|
|
|
put_till_end_and_adv_cursor();
|
2005-08-04 19:04:46 +00:00
|
|
|
vi_cmdmode = 0;
|
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'x'|VI_CMDMODE_BIT:
|
2005-08-04 19:04:46 +00:00
|
|
|
input_delete(1);
|
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'X'|VI_CMDMODE_BIT:
|
2005-08-04 19:04:46 +00:00
|
|
|
if (cursor > 0) {
|
|
|
|
input_backward(1);
|
|
|
|
input_delete(1);
|
|
|
|
}
|
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'W'|VI_CMDMODE_BIT:
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_Word_motion(1);
|
2005-08-04 19:04:46 +00:00
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'w'|VI_CMDMODE_BIT:
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_word_motion(1);
|
2005-08-04 19:04:46 +00:00
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'E'|VI_CMDMODE_BIT:
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_End_motion();
|
2005-08-04 19:04:46 +00:00
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'e'|VI_CMDMODE_BIT:
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_end_motion();
|
2005-08-04 19:04:46 +00:00
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'B'|VI_CMDMODE_BIT:
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_Back_motion();
|
2005-08-04 19:04:46 +00:00
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'b'|VI_CMDMODE_BIT:
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_back_motion();
|
2005-08-04 19:04:46 +00:00
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'C'|VI_CMDMODE_BIT:
|
2005-08-04 19:04:46 +00:00
|
|
|
vi_cmdmode = 0;
|
|
|
|
/* fall through */
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'D'|VI_CMDMODE_BIT:
|
2005-08-04 19:04:46 +00:00
|
|
|
goto clear_to_eol;
|
|
|
|
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'c'|VI_CMDMODE_BIT:
|
2005-08-04 19:04:46 +00:00
|
|
|
vi_cmdmode = 0;
|
|
|
|
/* fall through */
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'd'|VI_CMDMODE_BIT: {
|
2006-12-21 22:27:10 +00:00
|
|
|
int nc, sc;
|
2009-05-15 03:27:53 +02:00
|
|
|
|
2011-02-08 05:07:02 +01:00
|
|
|
ic = lineedit_read_key(read_key_buffer, timeout);
|
2009-05-15 03:27:53 +02:00
|
|
|
if (errno) /* error */
|
2010-09-07 18:40:53 +02:00
|
|
|
goto return_error_indicator;
|
2009-10-14 12:53:04 +02:00
|
|
|
if (ic == ic_raw) { /* "cc", "dd" */
|
2006-12-21 22:27:10 +00:00
|
|
|
input_backward(cursor);
|
|
|
|
goto clear_to_eol;
|
|
|
|
break;
|
|
|
|
}
|
2009-10-14 12:53:04 +02:00
|
|
|
|
|
|
|
sc = cursor;
|
2009-05-15 03:27:53 +02:00
|
|
|
switch (ic) {
|
2006-12-21 22:27:10 +00:00
|
|
|
case 'w':
|
|
|
|
case 'W':
|
|
|
|
case 'e':
|
|
|
|
case 'E':
|
2009-05-15 03:27:53 +02:00
|
|
|
switch (ic) {
|
2006-12-21 22:27:10 +00:00
|
|
|
case 'w': /* "dw", "cw" */
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_word_motion(vi_cmdmode);
|
2006-10-03 19:56:34 +00:00
|
|
|
break;
|
2006-12-21 22:27:10 +00:00
|
|
|
case 'W': /* 'dW', 'cW' */
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_Word_motion(vi_cmdmode);
|
2006-10-03 19:56:34 +00:00
|
|
|
break;
|
2006-12-21 22:27:10 +00:00
|
|
|
case 'e': /* 'de', 'ce' */
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_end_motion();
|
2006-12-21 22:27:10 +00:00
|
|
|
input_forward();
|
2006-10-03 19:56:34 +00:00
|
|
|
break;
|
2006-12-21 22:27:10 +00:00
|
|
|
case 'E': /* 'dE', 'cE' */
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_End_motion();
|
2006-12-21 22:27:10 +00:00
|
|
|
input_forward();
|
2006-10-03 19:56:34 +00:00
|
|
|
break;
|
|
|
|
}
|
2006-12-21 22:27:10 +00:00
|
|
|
nc = cursor;
|
|
|
|
input_backward(cursor - sc);
|
|
|
|
while (nc-- > cursor)
|
|
|
|
input_delete(1);
|
|
|
|
break;
|
|
|
|
case 'b': /* "db", "cb" */
|
|
|
|
case 'B': /* implemented as B */
|
2009-05-15 03:27:53 +02:00
|
|
|
if (ic == 'b')
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_back_motion();
|
2006-12-21 22:27:10 +00:00
|
|
|
else
|
2009-07-10 18:40:49 +02:00
|
|
|
vi_Back_motion();
|
2006-12-21 22:27:10 +00:00
|
|
|
while (sc-- > cursor)
|
|
|
|
input_delete(1);
|
|
|
|
break;
|
|
|
|
case ' ': /* "d ", "c " */
|
|
|
|
input_delete(1);
|
|
|
|
break;
|
|
|
|
case '$': /* "d$", "c$" */
|
2009-05-15 03:27:53 +02:00
|
|
|
clear_to_eol:
|
2007-01-22 07:21:38 +00:00
|
|
|
while (cursor < command_len)
|
2006-12-21 22:27:10 +00:00
|
|
|
input_delete(1);
|
|
|
|
break;
|
2005-08-04 19:04:46 +00:00
|
|
|
}
|
|
|
|
break;
|
2006-12-21 22:27:10 +00:00
|
|
|
}
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'p'|VI_CMDMODE_BIT:
|
2005-08-04 19:04:46 +00:00
|
|
|
input_forward();
|
|
|
|
/* fallthrough */
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'P'|VI_CMDMODE_BIT:
|
2005-08-04 19:04:46 +00:00
|
|
|
put();
|
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case 'r'|VI_CMDMODE_BIT:
|
2009-10-14 12:53:04 +02:00
|
|
|
//FIXME: unicode case?
|
2011-02-08 05:07:02 +01:00
|
|
|
ic = lineedit_read_key(read_key_buffer, timeout);
|
2009-05-15 03:27:53 +02:00
|
|
|
if (errno) /* error */
|
2010-09-07 18:40:53 +02:00
|
|
|
goto return_error_indicator;
|
2009-05-15 03:27:53 +02:00
|
|
|
if (ic < ' ' || ic > 255) {
|
2005-08-04 19:04:46 +00:00
|
|
|
beep();
|
2009-05-15 03:27:53 +02:00
|
|
|
} else {
|
2009-07-10 18:40:49 +02:00
|
|
|
command_ps[cursor] = ic;
|
2009-05-15 03:27:53 +02:00
|
|
|
bb_putchar(ic);
|
2007-09-27 10:20:47 +00:00
|
|
|
bb_putchar('\b');
|
2005-08-04 19:04:46 +00:00
|
|
|
}
|
|
|
|
break;
|
2007-01-21 19:18:19 +00:00
|
|
|
case '\x1b': /* ESC */
|
2007-01-22 07:21:38 +00:00
|
|
|
if (state->flags & VI_MODE) {
|
2009-05-15 03:27:53 +02:00
|
|
|
/* insert mode --> command mode */
|
2005-08-04 19:04:46 +00:00
|
|
|
vi_cmdmode = 1;
|
|
|
|
input_backward(1);
|
2004-01-22 12:42:23 +00:00
|
|
|
}
|
2009-05-15 03:27:53 +02:00
|
|
|
break;
|
|
|
|
#endif /* FEATURE_COMMAND_EDITING_VI */
|
2007-01-21 19:21:21 +00:00
|
|
|
|
2006-11-02 22:09:37 +00:00
|
|
|
#if MAX_HISTORY > 0
|
2009-05-15 03:27:53 +02:00
|
|
|
case KEYCODE_UP:
|
|
|
|
if (get_previous_history())
|
|
|
|
goto rewrite_line;
|
|
|
|
beep();
|
|
|
|
break;
|
|
|
|
case KEYCODE_DOWN:
|
|
|
|
if (!get_next_history())
|
2001-02-16 18:36:04 +00:00
|
|
|
break;
|
2007-01-21 19:18:19 +00:00
|
|
|
rewrite_line:
|
2009-05-15 03:27:53 +02:00
|
|
|
/* Rewrite the line with the selected history item */
|
|
|
|
/* change command */
|
2009-09-06 02:36:23 +02:00
|
|
|
command_len = load_string(state->history[state->cur_history] ?
|
2011-07-11 07:36:59 +02:00
|
|
|
state->history[state->cur_history] : "");
|
2009-05-15 03:27:53 +02:00
|
|
|
/* redraw and go to eol (bol, in vi) */
|
|
|
|
redraw(cmdedit_y, (state->flags & VI_MODE) ? 9999 : 0);
|
|
|
|
break;
|
2002-11-27 09:29:49 +00:00
|
|
|
#endif
|
2009-05-15 03:27:53 +02:00
|
|
|
case KEYCODE_RIGHT:
|
|
|
|
input_forward();
|
|
|
|
break;
|
|
|
|
case KEYCODE_LEFT:
|
|
|
|
input_backward(1);
|
|
|
|
break;
|
2009-10-25 23:50:56 +01:00
|
|
|
case KEYCODE_CTRL_LEFT:
|
2011-11-03 13:28:22 +01:00
|
|
|
case KEYCODE_ALT_LEFT: /* bash doesn't do it */
|
2009-10-25 23:50:56 +01:00
|
|
|
ctrl_left();
|
|
|
|
break;
|
|
|
|
case KEYCODE_CTRL_RIGHT:
|
2011-11-03 13:28:22 +01:00
|
|
|
case KEYCODE_ALT_RIGHT: /* bash doesn't do it */
|
2009-10-25 23:50:56 +01:00
|
|
|
ctrl_right();
|
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
case KEYCODE_HOME:
|
|
|
|
input_backward(cursor);
|
|
|
|
break;
|
|
|
|
case KEYCODE_END:
|
2010-05-11 14:49:13 +02:00
|
|
|
put_till_end_and_adv_cursor();
|
2001-02-16 18:36:04 +00:00
|
|
|
break;
|
2000-03-19 05:28:55 +00:00
|
|
|
|
2009-05-15 03:27:53 +02:00
|
|
|
default:
|
2009-10-14 12:53:04 +02:00
|
|
|
if (initial_settings.c_cc[VINTR] != 0
|
|
|
|
&& ic_raw == initial_settings.c_cc[VINTR]
|
|
|
|
) {
|
|
|
|
/* Ctrl-C (usually) - stop gathering input */
|
|
|
|
command_len = 0;
|
|
|
|
break_out = -1; /* "do not append '\n'" */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (initial_settings.c_cc[VEOF] != 0
|
|
|
|
&& ic_raw == initial_settings.c_cc[VEOF]
|
|
|
|
) {
|
|
|
|
/* Ctrl-D (usually) - delete one character,
|
|
|
|
* or exit if len=0 and no chars to delete */
|
|
|
|
if (command_len == 0) {
|
|
|
|
errno = 0;
|
2010-09-07 18:40:53 +02:00
|
|
|
|
|
|
|
case -1: /* error (e.g. EIO when tty is destroyed) */
|
|
|
|
IF_FEATURE_EDITING_VI(return_error_indicator:)
|
2009-10-14 12:53:04 +02:00
|
|
|
break_out = command_len = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
input_delete(0);
|
|
|
|
break;
|
|
|
|
}
|
2009-05-15 03:27:53 +02:00
|
|
|
// /* Control-V -- force insert of next char */
|
|
|
|
// if (c == CTRL('V')) {
|
|
|
|
// if (safe_read(STDIN_FILENO, &c, 1) < 1)
|
2010-09-07 18:40:53 +02:00
|
|
|
// goto return_error_indicator;
|
2009-05-15 03:27:53 +02:00
|
|
|
// if (c == 0) {
|
|
|
|
// beep();
|
|
|
|
// break;
|
|
|
|
// }
|
|
|
|
// }
|
2009-07-10 18:40:49 +02:00
|
|
|
if (ic < ' '
|
2010-03-26 14:06:56 +01:00
|
|
|
|| (!ENABLE_UNICODE_SUPPORT && ic >= 256)
|
|
|
|
|| (ENABLE_UNICODE_SUPPORT && ic >= VI_CMDMODE_BIT)
|
2009-07-10 18:40:49 +02:00
|
|
|
) {
|
2009-05-15 03:27:53 +02:00
|
|
|
/* If VI_CMDMODE_BIT is set, ic is >= 256
|
2009-10-14 12:53:04 +02:00
|
|
|
* and vi mode ignores unexpected chars.
|
2009-05-15 03:27:53 +02:00
|
|
|
* Otherwise, we are here if ic is a
|
|
|
|
* control char or an unhandled ESC sequence,
|
|
|
|
* which is also ignored.
|
|
|
|
*/
|
2007-01-21 19:21:21 +00:00
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
}
|
|
|
|
if ((int)command_len >= (maxsize - 2)) {
|
|
|
|
/* Not enough space for the char and EOL */
|
2000-03-19 05:28:55 +00:00
|
|
|
break;
|
2009-05-15 03:27:53 +02:00
|
|
|
}
|
2000-03-19 05:28:55 +00:00
|
|
|
|
2007-01-22 07:21:38 +00:00
|
|
|
command_len++;
|
2009-05-15 03:27:53 +02:00
|
|
|
if (cursor == (command_len - 1)) {
|
|
|
|
/* We are at the end, append */
|
2009-07-10 18:40:49 +02:00
|
|
|
command_ps[cursor] = ic;
|
|
|
|
command_ps[cursor + 1] = BB_NUL;
|
2010-05-11 14:49:13 +02:00
|
|
|
put_cur_glyph_and_inc_cursor();
|
2010-03-26 13:13:24 +01:00
|
|
|
if (unicode_bidi_isrtl(ic))
|
2010-03-18 18:35:37 +01:00
|
|
|
input_backward(1);
|
2009-05-15 03:27:53 +02:00
|
|
|
} else {
|
|
|
|
/* In the middle, insert */
|
2001-02-16 18:36:04 +00:00
|
|
|
int sc = cursor;
|
2000-03-19 05:28:55 +00:00
|
|
|
|
2009-07-10 18:40:49 +02:00
|
|
|
memmove(command_ps + sc + 1, command_ps + sc,
|
|
|
|
(command_len - sc) * sizeof(command_ps[0]));
|
|
|
|
command_ps[sc] = ic;
|
2010-03-26 13:13:24 +01:00
|
|
|
/* is right-to-left char, or neutral one (e.g. comma) was just added to rtl text? */
|
|
|
|
if (!isrtl_str())
|
|
|
|
sc++; /* no */
|
2010-05-11 14:49:13 +02:00
|
|
|
put_till_end_and_adv_cursor();
|
2001-01-26 20:42:23 +00:00
|
|
|
/* to prev x pos + 1 */
|
2001-02-16 18:36:04 +00:00
|
|
|
input_backward(cursor - sc);
|
2000-03-19 05:28:55 +00:00
|
|
|
}
|
|
|
|
break;
|
2009-10-14 12:53:04 +02:00
|
|
|
} /* switch (ic) */
|
2009-05-15 03:27:53 +02:00
|
|
|
|
|
|
|
if (break_out)
|
2000-03-19 05:28:55 +00:00
|
|
|
break;
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2007-11-10 01:35:47 +00:00
|
|
|
#if ENABLE_FEATURE_TAB_COMPLETION
|
2009-10-14 12:53:04 +02:00
|
|
|
if (ic_raw != '\t')
|
2010-09-03 12:51:36 +02:00
|
|
|
lastWasTab = 0;
|
2007-11-10 01:35:47 +00:00
|
|
|
#endif
|
2009-05-15 03:27:53 +02:00
|
|
|
} /* while (1) */
|
2000-03-16 08:09:57 +00:00
|
|
|
|
2009-10-27 10:34:06 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_ASK_TERMINAL
|
2009-10-27 10:47:49 +01:00
|
|
|
if (S.sent_ESC_br6n) {
|
2009-10-27 10:34:06 +01:00
|
|
|
/* "sleep 1; busybox ash" + hold [Enter] to trigger.
|
|
|
|
* We sent "ESC [ 6 n", but got '\n' first, and
|
|
|
|
* KEYCODE_CURSOR_POS response is now buffered from terminal.
|
|
|
|
* It's bad already and not much can be done with it
|
|
|
|
* (it _will_ be visible for the next process to read stdin),
|
|
|
|
* but without this delay it even shows up on the screen
|
|
|
|
* as garbage because we restore echo settings with tcsetattr
|
|
|
|
* before it comes in. UGLY!
|
|
|
|
*/
|
|
|
|
usleep(20*1000);
|
2020-12-21 21:55:03 +01:00
|
|
|
// MAYBE? tcflush(STDIN_FILENO, TCIFLUSH); /* flushes data received but not read */
|
2009-10-27 10:34:06 +01:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-07-18 22:21:24 +02:00
|
|
|
/* End of bug-catching "command_must_not_be_used" trick */
|
2009-07-10 18:40:49 +02:00
|
|
|
#undef command
|
|
|
|
|
2010-03-26 14:06:56 +01:00
|
|
|
#if ENABLE_UNICODE_SUPPORT
|
2009-09-29 00:00:12 +02:00
|
|
|
command[0] = '\0';
|
|
|
|
if (command_len > 0)
|
|
|
|
command_len = save_string(command, maxsize - 1);
|
2009-07-10 18:40:49 +02:00
|
|
|
free(command_ps);
|
|
|
|
#endif
|
|
|
|
|
2013-04-07 18:47:24 +02:00
|
|
|
if (command_len > 0) {
|
2007-01-22 07:21:38 +00:00
|
|
|
remember_in_history(command);
|
2013-04-07 18:47:24 +02:00
|
|
|
}
|
2007-01-21 19:18:19 +00:00
|
|
|
|
2003-12-23 20:24:51 +00:00
|
|
|
if (break_out > 0) {
|
2007-01-22 07:21:38 +00:00
|
|
|
command[command_len++] = '\n';
|
|
|
|
command[command_len] = '\0';
|
2001-07-17 01:12:36 +00:00
|
|
|
}
|
2007-01-21 19:18:19 +00:00
|
|
|
|
2007-11-10 01:35:47 +00:00
|
|
|
#if ENABLE_FEATURE_TAB_COMPLETION
|
2007-01-21 19:19:46 +00:00
|
|
|
free_tab_completion_data();
|
2001-02-16 18:36:04 +00:00
|
|
|
#endif
|
2007-01-21 19:18:19 +00:00
|
|
|
|
2007-01-22 07:30:26 +00:00
|
|
|
/* restore initial_settings */
|
2008-11-05 13:20:58 +00:00
|
|
|
tcsetattr_stdin_TCSANOW(&initial_settings);
|
2018-02-25 20:09:54 +01:00
|
|
|
#if ENABLE_FEATURE_EDITING_WINCH
|
2007-01-22 07:30:26 +00:00
|
|
|
/* restore SIGWINCH handler */
|
2016-11-27 22:25:07 +01:00
|
|
|
sigaction_set(SIGWINCH, &S.SIGWINCH_handler);
|
2018-02-25 20:09:54 +01:00
|
|
|
#endif
|
2009-11-02 14:19:51 +01:00
|
|
|
fflush_all();
|
2007-11-10 01:35:47 +00:00
|
|
|
|
2008-08-20 00:46:32 +00:00
|
|
|
len = command_len;
|
2007-11-10 01:35:47 +00:00
|
|
|
DEINIT_S();
|
|
|
|
|
2008-08-20 00:46:32 +00:00
|
|
|
return len; /* can't return command_len, DEINIT_S() destroys it */
|
2007-01-22 07:21:38 +00:00
|
|
|
}
|
|
|
|
|
2010-07-18 22:21:24 +02:00
|
|
|
#else /* !FEATURE_EDITING */
|
2007-01-22 07:21:38 +00:00
|
|
|
|
|
|
|
#undef read_line_input
|
2008-06-27 02:52:20 +00:00
|
|
|
int FAST_FUNC read_line_input(const char* prompt, char* command, int maxsize)
|
2007-01-22 07:21:38 +00:00
|
|
|
{
|
2021-02-03 20:47:14 +01:00
|
|
|
fputs_stdout(prompt);
|
2009-11-02 14:19:51 +01:00
|
|
|
fflush_all();
|
2012-09-27 16:03:49 +02:00
|
|
|
if (!fgets(command, maxsize, stdin))
|
|
|
|
return -1;
|
2007-01-22 07:21:38 +00:00
|
|
|
return strlen(command);
|
2000-03-16 08:09:57 +00:00
|
|
|
}
|
|
|
|
|
2010-07-18 22:21:24 +02:00
|
|
|
#endif /* !FEATURE_EDITING */
|
2001-02-16 18:36:04 +00:00
|
|
|
|
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
/*
|
|
|
|
* Testing
|
|
|
|
*/
|
2001-02-16 18:36:04 +00:00
|
|
|
|
2007-01-21 19:18:19 +00:00
|
|
|
#ifdef TEST
|
2001-04-09 22:48:12 +00:00
|
|
|
|
2001-03-15 20:51:09 +00:00
|
|
|
#include <locale.h>
|
2007-01-21 19:18:19 +00:00
|
|
|
|
|
|
|
const char *applet_name = "debug stuff usage";
|
2001-03-15 20:51:09 +00:00
|
|
|
|
2001-02-16 18:36:04 +00:00
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
2007-06-10 15:08:44 +00:00
|
|
|
char buff[MAX_LINELEN];
|
2001-02-16 18:36:04 +00:00
|
|
|
char *prompt =
|
2007-01-22 09:03:07 +00:00
|
|
|
#if ENABLE_FEATURE_EDITING_FANCY_PROMPT
|
2006-12-21 22:27:10 +00:00
|
|
|
"\\[\\033[32;1m\\]\\u@\\[\\x1b[33;1m\\]\\h:"
|
|
|
|
"\\[\\033[34;1m\\]\\w\\[\\033[35;1m\\] "
|
2017-09-13 22:48:30 +02:00
|
|
|
"\\!\\[\\e[36;1m\\]\\$ \\[\\E[m\\]";
|
2001-02-16 18:36:04 +00:00
|
|
|
#else
|
|
|
|
"% ";
|
|
|
|
#endif
|
|
|
|
|
2006-12-19 01:10:25 +00:00
|
|
|
while (1) {
|
2001-03-13 22:57:56 +00:00
|
|
|
int l;
|
2007-01-22 07:21:38 +00:00
|
|
|
l = read_line_input(prompt, buff);
|
2006-12-21 22:27:10 +00:00
|
|
|
if (l <= 0 || buff[l-1] != '\n')
|
2003-12-23 20:24:51 +00:00
|
|
|
break;
|
2010-09-03 12:51:36 +02:00
|
|
|
buff[l-1] = '\0';
|
2007-01-22 07:21:38 +00:00
|
|
|
printf("*** read_line_input() returned line =%s=\n", buff);
|
2001-07-12 20:26:32 +00:00
|
|
|
}
|
2007-01-22 07:21:38 +00:00
|
|
|
printf("*** read_line_input() detect ^D\n");
|
2001-02-16 18:36:04 +00:00
|
|
|
return 0;
|
|
|
|
}
|
2000-07-28 15:14:45 +00:00
|
|
|
|
2003-07-28 09:56:35 +00:00
|
|
|
#endif /* TEST */
|