mirror of
https://github.com/fish-shell/fish-shell
synced 2025-01-14 22:14:53 +00:00
29c488c6dc
darcs-hash:20060123204014-ac50b-b9d2a10c7f00945b5d9d244cb3b8c1489fe280de.gz
3223 lines
62 KiB
C
3223 lines
62 KiB
C
/** \file reader.c
|
|
|
|
Functions for reading data from stdin and passing to the
|
|
parser. If stdin is a keyboard, it supplies a killring, history,
|
|
syntax highlighting, tab-completion and various other interactive features.
|
|
|
|
Internally the interactive mode functions rely in the functions of the
|
|
input library to read individual characters of input.
|
|
|
|
Token search is handled incrementally. Actual searches are only done
|
|
on when searching backwards, since the previous results are saved. The
|
|
last search position is remembered and a new search continues from the
|
|
last search position. All search results are saved in the list
|
|
'search_prev'. When the user searches forward, i.e. presses Alt-down,
|
|
the list is consulted for previous search result, and subsequent
|
|
backwards searches are also handled by consultiung the list up until
|
|
the end of the list is reached, at which point regular searching will
|
|
commence.
|
|
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include <stdlib.h>
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <errno.h>
|
|
#include <termios.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/time.h>
|
|
#include <sys/wait.h>
|
|
#include <sys/poll.h>
|
|
#include <unistd.h>
|
|
#include <wctype.h>
|
|
|
|
#if HAVE_NCURSES_H
|
|
#include <ncurses.h>
|
|
#else
|
|
#include <curses.h>
|
|
#endif
|
|
|
|
#if HAVE_TERMIO_H
|
|
#include <termio.h>
|
|
#endif
|
|
|
|
#if HAVE_TERM_H
|
|
#include <term.h>
|
|
#elif HAVE_NCURSES_TERM_H
|
|
#include <ncurses/term.h>
|
|
#endif
|
|
|
|
#include <signal.h>
|
|
#include <fcntl.h>
|
|
#include <dirent.h>
|
|
#include <time.h>
|
|
#include <wchar.h>
|
|
|
|
#include <assert.h>
|
|
|
|
#include "util.h"
|
|
#include "wutil.h"
|
|
#include "highlight.h"
|
|
#include "reader.h"
|
|
#include "proc.h"
|
|
#include "parser.h"
|
|
#include "complete.h"
|
|
#include "history.h"
|
|
#include "common.h"
|
|
#include "sanity.h"
|
|
#include "env.h"
|
|
#include "exec.h"
|
|
#include "expand.h"
|
|
#include "tokenizer.h"
|
|
#include "kill.h"
|
|
#include "input_common.h"
|
|
#include "input.h"
|
|
#include "function.h"
|
|
#include "output.h"
|
|
#include "signal.h"
|
|
#include "translate.h"
|
|
|
|
/**
|
|
Maximum length of prefix string when printing completion
|
|
list. Longer prefixes will be ellipsized.
|
|
*/
|
|
#define PREFIX_MAX_LEN 8
|
|
|
|
/**
|
|
A simple prompt for reading shell commands that does not rely on
|
|
fish specific commands, meaning it will work even if fish is not
|
|
installed. This is used by read_i.
|
|
*/
|
|
#define DEFAULT_PROMPT L"whoami; echo @; hostname|cut -d . -f 1; echo \" \"; pwd; printf '> ';"
|
|
|
|
/**
|
|
The default title for the reader. This is used by reader_readline.
|
|
*/
|
|
#define DEFAULT_TITLE L"echo $_ \" \"; pwd"
|
|
|
|
/**
|
|
The maximum number of characters to read from the keyboard without
|
|
repainting. Note that this readahead will only occur if new
|
|
characters are avaialble for reading, fish will never block for
|
|
more input without repainting.
|
|
*/
|
|
#define READAHEAD_MAX 256
|
|
|
|
/**
|
|
A struct describing the state of the interactive reader. These
|
|
states can be stacked, in case reader_readline is called from
|
|
input_read().
|
|
*/
|
|
typedef struct reader_data
|
|
{
|
|
/**
|
|
Buffer containing the current commandline
|
|
*/
|
|
wchar_t *buff;
|
|
|
|
/**
|
|
The output string, may be different than buff if buff can't fit on one line.
|
|
*/
|
|
wchar_t *output;
|
|
|
|
/**
|
|
The number of characters used by the prompt
|
|
*/
|
|
int prompt_width;
|
|
|
|
/**
|
|
Buffer containing the current search item
|
|
*/
|
|
wchar_t *search_buff;
|
|
/**
|
|
Saved position used by token history search
|
|
*/
|
|
int token_history_pos;
|
|
|
|
/**
|
|
Saved search string for token history search. Not handled by check_size.
|
|
*/
|
|
const wchar_t *token_history_buff;
|
|
|
|
/**
|
|
List for storing previous search results. Used to avoid duplicates.
|
|
*/
|
|
array_list_t search_prev;
|
|
|
|
/**
|
|
The current position in search_prev
|
|
*/
|
|
|
|
int search_pos;
|
|
|
|
/**
|
|
Current size of the buffers
|
|
*/
|
|
size_t buff_sz;
|
|
|
|
/**
|
|
Length of the command in buff. (Not the length of buff itself)
|
|
*/
|
|
|
|
size_t buff_len;
|
|
|
|
/**
|
|
The current position of the cursor in buff.
|
|
*/
|
|
size_t buff_pos;
|
|
|
|
/**
|
|
The current position of the cursor in output buffer.
|
|
*/
|
|
size_t output_pos;
|
|
|
|
/**
|
|
Name of the current application
|
|
*/
|
|
wchar_t *name;
|
|
|
|
/** The prompt text */
|
|
wchar_t *prompt;
|
|
|
|
/**
|
|
Color is the syntax highlighting for buff. The format is that
|
|
color[i] is the classification (according to the enum in
|
|
highlight.h) of buff[i].
|
|
*/
|
|
int *color;
|
|
|
|
/**
|
|
New color buffer, used for syntax highlighting.
|
|
*/
|
|
int *new_color;
|
|
|
|
/**
|
|
Color for the actual output string.
|
|
*/
|
|
int *output_color;
|
|
|
|
/**
|
|
Should the prompt command be reexecuted on the next repaint
|
|
*/
|
|
int exec_prompt;
|
|
|
|
/**
|
|
Function for tab completion
|
|
*/
|
|
void (*complete_func)( const wchar_t *,
|
|
array_list_t * );
|
|
|
|
/**
|
|
Function for syntax highlighting
|
|
*/
|
|
void (*highlight_func)( wchar_t *,
|
|
int *,
|
|
int,
|
|
array_list_t * );
|
|
|
|
/**
|
|
Function for testing if the string can be returned
|
|
*/
|
|
int (*test_func)( wchar_t * );
|
|
|
|
/**
|
|
When this is true, the reader will exit
|
|
*/
|
|
int end_loop;
|
|
|
|
/**
|
|
Pointer to previous reader_data
|
|
*/
|
|
struct reader_data *next;
|
|
}
|
|
reader_data_t;
|
|
|
|
/**
|
|
The current interactive reading context
|
|
*/
|
|
static reader_data_t *data=0;
|
|
|
|
|
|
/**
|
|
Flag for ending non-interactive shell
|
|
*/
|
|
static int end_loop = 0;
|
|
|
|
/**
|
|
The list containing names of files that are being parsed
|
|
*/
|
|
static array_list_t current_filename;
|
|
|
|
/**
|
|
These status buffers are used to check if any output has occurred
|
|
other than from fish's main loop, in which case we need to redraw.
|
|
*/
|
|
static struct stat prev_buff_1, prev_buff_2, post_buff_1, post_buff_2;
|
|
|
|
|
|
|
|
/**
|
|
List containing strings which make up the prompt
|
|
*/
|
|
static array_list_t prompt_list;
|
|
|
|
/**
|
|
Stores the previous termios mode so we can reset the modes when
|
|
we execute programs and when the shell exits.
|
|
*/
|
|
static struct termios saved_modes;
|
|
|
|
|
|
/**
|
|
Store the pid of the parent process, so the exit function knows whether it should reset the terminal or not.
|
|
*/
|
|
static pid_t original_pid;
|
|
|
|
/**
|
|
This variable is set to true by the signal handler when ^C is pressed
|
|
*/
|
|
static int interupted=0;
|
|
|
|
/**
|
|
Original terminal mode when fish was started
|
|
*/
|
|
static struct termios old_modes;
|
|
|
|
/*
|
|
Prototypes for a bunch of functions defined later on.
|
|
*/
|
|
|
|
static void reader_save_status();
|
|
static void reader_check_status();
|
|
static void reader_super_highlight_me_plenty( wchar_t * buff, int *color, int pos, array_list_t *error );
|
|
|
|
|
|
/**
|
|
Give up control of terminal
|
|
*/
|
|
static void term_donate()
|
|
{
|
|
tcgetattr(0,&old_modes); /* get the current terminal modes */
|
|
|
|
set_color(FISH_COLOR_NORMAL, FISH_COLOR_NORMAL);
|
|
|
|
while( 1 )
|
|
{
|
|
if( tcsetattr(0,TCSANOW,&saved_modes) )
|
|
{
|
|
if( errno != EINTR )
|
|
{
|
|
debug( 1, _( L"Could not set terminal mode for new job" ) );
|
|
wperror( L"tcsetattr" );
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
|
|
|
|
}
|
|
|
|
/**
|
|
Grab control of terminal
|
|
*/
|
|
static void term_steal()
|
|
{
|
|
|
|
while( 1 )
|
|
{
|
|
if( tcsetattr(0,TCSANOW,&shell_modes) )
|
|
{
|
|
if( errno != EINTR )
|
|
{
|
|
debug( 1, _( L"Could not set terminal mode for shell" ) );
|
|
wperror( L"tcsetattr" );
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
|
|
common_handle_winch(0 );
|
|
|
|
if( tcsetattr(0,TCSANOW,&old_modes))/* return to previous mode */
|
|
{
|
|
wperror(L"tcsetattr");
|
|
exit(1);
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
Test if there is space between the time fields of struct stat to
|
|
use for sub second information. If so, we assume this space
|
|
contains the desired information.
|
|
*/
|
|
static int room_for_usec(struct stat *st)
|
|
{
|
|
int res = ((&(st->st_atime) + 2) == &(st->st_mtime) &&
|
|
(&(st->st_atime) + 4) == &(st->st_ctime));
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
string_buffer used as temporary storage for the reader_readline function
|
|
*/
|
|
static string_buffer_t *readline_buffer=0;
|
|
|
|
void reader_handle_int( int sig )
|
|
{
|
|
block_t *c = current_block;
|
|
while( c )
|
|
{
|
|
c->skip=1;
|
|
c=c->outer;
|
|
}
|
|
interupted = 1;
|
|
|
|
}
|
|
|
|
wchar_t *reader_current_filename()
|
|
{
|
|
return (wchar_t *)al_peek( ¤t_filename );
|
|
}
|
|
|
|
|
|
void reader_push_current_filename( wchar_t *fn )
|
|
{
|
|
al_push( ¤t_filename, fn );
|
|
}
|
|
|
|
|
|
wchar_t *reader_pop_current_filename()
|
|
{
|
|
return (wchar_t *)al_pop( ¤t_filename );
|
|
}
|
|
|
|
|
|
/**
|
|
Make sure buffers are large enough to hold current data plus one extra character.
|
|
*/
|
|
static int check_size()
|
|
{
|
|
if( data->buff_sz < data->buff_len + 2 )
|
|
{
|
|
data->buff_sz = maxi( 128, data->buff_len*2 );
|
|
|
|
data->buff = realloc( data->buff,
|
|
sizeof(wchar_t)*data->buff_sz);
|
|
data->search_buff = realloc( data->search_buff,
|
|
sizeof(wchar_t)*data->buff_sz);
|
|
data->output = realloc( data->output,
|
|
sizeof(wchar_t)*data->buff_sz);
|
|
|
|
data->color = realloc( data->color,
|
|
sizeof(int)*data->buff_sz);
|
|
data->new_color = realloc( data->new_color,
|
|
sizeof(int)*data->buff_sz);
|
|
data->output_color = realloc( data->output_color,
|
|
sizeof(int)*data->buff_sz);
|
|
|
|
if( data->buff==0 ||
|
|
data->search_buff==0 ||
|
|
data->color==0 ||
|
|
data->new_color == 0 )
|
|
{
|
|
die_mem();
|
|
|
|
}
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
Check if the screen is not wide enough for the buffer, which means
|
|
the buffer must be scrolled on input and cursor movement.
|
|
*/
|
|
static int force_repaint()
|
|
{
|
|
int max_width = common_get_width() - data->prompt_width;
|
|
int pref_width = my_wcswidth( data->buff ) + (data->buff_pos==data->buff_len);
|
|
return pref_width >= max_width;
|
|
}
|
|
|
|
|
|
/**
|
|
Calculate what part of the buffer should be visible
|
|
|
|
\return returns 1 screen needs repainting, 0 otherwise
|
|
*/
|
|
static int calc_output()
|
|
{
|
|
int max_width = common_get_width() - data->prompt_width;
|
|
int pref_width = my_wcswidth( data->buff ) + (data->buff_pos==data->buff_len);
|
|
if( pref_width <= max_width )
|
|
{
|
|
wcscpy( data->output, data->buff );
|
|
memcpy( data->output_color, data->color, sizeof(int) * data->buff_len );
|
|
data->output_pos=data->buff_pos;
|
|
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
int offset = data->buff_pos;
|
|
int offset_end = data->buff_pos;
|
|
int w = 0;
|
|
wchar_t *pos=data->output;
|
|
*pos=0;
|
|
|
|
|
|
w = (data->buff_pos==data->buff_len)?1:wcwidth( data->buff[offset] );
|
|
while( 1 )
|
|
{
|
|
int inc=0;
|
|
int ellipsis_width;
|
|
|
|
ellipsis_width = wcwidth(ellipsis_char)*((offset?1:0)+(offset_end<data->buff_len?1:0));
|
|
|
|
if( offset > 0 && (ellipsis_width + w + wcwidth( data->buff[offset-1] ) <= max_width ) )
|
|
{
|
|
inc=1;
|
|
offset--;
|
|
w+= wcwidth( data->buff[offset]);
|
|
}
|
|
|
|
ellipsis_width = wcwidth(ellipsis_char)*((offset?1:0)+(offset_end<data->buff_len?1:0));
|
|
|
|
if( offset_end < data->buff_len && (ellipsis_width + w + wcwidth( data->buff[offset_end+1] ) <= max_width ) )
|
|
{
|
|
inc = 1;
|
|
offset_end++;
|
|
w+= wcwidth( data->buff[offset_end]);
|
|
}
|
|
|
|
if( !inc )
|
|
break;
|
|
|
|
}
|
|
|
|
data->output_pos = data->buff_pos - offset + (offset?1:0);
|
|
|
|
if( offset )
|
|
{
|
|
data->output[0]=ellipsis_char;
|
|
data->output[1]=0;
|
|
|
|
}
|
|
|
|
wcsncat( data->output,
|
|
data->buff+offset,
|
|
offset_end-offset );
|
|
|
|
if( offset_end<data->buff_len )
|
|
{
|
|
int l = wcslen(data->output);
|
|
|
|
data->output[l]=ellipsis_char;
|
|
data->output[l+1]=0;
|
|
|
|
}
|
|
|
|
*data->output_color=HIGHLIGHT_NORMAL;
|
|
|
|
memcpy( data->output_color+(offset?1:0),
|
|
data->color+offset,
|
|
sizeof(int) * (data->buff_len-offset) );
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
Compare two completions, ignoring their description.
|
|
*/
|
|
static int fldcmp( wchar_t *a, wchar_t *b )
|
|
{
|
|
while( 1 )
|
|
{
|
|
if( *a != *b )
|
|
return *a-*b;
|
|
if( ( (*a == COMPLETE_SEP) || (*a == L'\0') ) &&
|
|
( (*b == COMPLETE_SEP) || (*b == L'\0') ) )
|
|
return 0;
|
|
a++;
|
|
b++;
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
Remove any duplicate completions in the list. This relies on the
|
|
list first beeing sorted.
|
|
*/
|
|
static void remove_duplicates( array_list_t *l )
|
|
{
|
|
int in, out;
|
|
wchar_t *prev;
|
|
if( al_get_count( l ) == 0 )
|
|
return;
|
|
|
|
prev = (wchar_t *)al_get( l, 0 );
|
|
for( in=1, out=1; in < al_get_count( l ); in++ )
|
|
{
|
|
wchar_t *curr = (wchar_t *)al_get( l, in );
|
|
if( fldcmp( prev, curr )==0 )
|
|
{
|
|
free( curr );
|
|
}
|
|
else
|
|
{
|
|
al_set( l, out++, curr );
|
|
prev = curr;
|
|
}
|
|
}
|
|
al_truncate( l, out );
|
|
}
|
|
|
|
|
|
/**
|
|
Translate a highlighting code ()Such as as returned by the highlight function
|
|
into a color code which is then passed on to set_color.
|
|
*/
|
|
static void set_color_translated( int c )
|
|
{
|
|
set_color( highlight_get_color( c & 0xff ),
|
|
highlight_get_color( (c>>8)&0xff ) );
|
|
}
|
|
|
|
int reader_interupted()
|
|
{
|
|
int res=interupted;
|
|
if( res )
|
|
interupted=0;
|
|
return res;
|
|
}
|
|
|
|
void reader_write_title()
|
|
{
|
|
wchar_t *title;
|
|
array_list_t l;
|
|
wchar_t *term = env_get( L"TERM" );
|
|
|
|
/*
|
|
This is a pretty lame heuristic for detecting terminals that do
|
|
not support setting the title. If we recognise the terminal name
|
|
as that of a virtual terminal, we assume it supports setting the
|
|
title. Otherwise we check the ttyname.
|
|
*/
|
|
if( !term || !contains_str( term, L"xterm", L"screen", L"nxterm", L"rxvt", 0 ) )
|
|
{
|
|
char *n = ttyname( STDIN_FILENO );
|
|
if( strstr( n, "tty" ) || strstr( n, "/vc/") )
|
|
return;
|
|
}
|
|
|
|
title = function_exists( L"fish_title" )?L"fish_title":DEFAULT_TITLE;
|
|
|
|
if( wcslen( title ) ==0 )
|
|
return;
|
|
|
|
al_init( &l );
|
|
|
|
if( exec_subshell( title, &l ) != -1 )
|
|
{
|
|
int i;
|
|
writestr( L"\e]2;" );
|
|
for( i=0; i<al_get_count( &l ); i++ )
|
|
{
|
|
writestr( (wchar_t *)al_get( &l, i ) );
|
|
}
|
|
writestr( L"\7" );
|
|
}
|
|
al_foreach( &l, (void (*)(const void *))&free );
|
|
al_destroy( &l );
|
|
set_color( FISH_COLOR_RESET, FISH_COLOR_RESET );
|
|
}
|
|
|
|
/**
|
|
Tests if the specified narrow character sequence is present at the
|
|
specified position of the specified wide character string. All of
|
|
\c seq must match, but str may be longer than seq.
|
|
*/
|
|
static int try_sequence( char *seq, wchar_t *str )
|
|
{
|
|
int i;
|
|
|
|
for( i=0;; i++ )
|
|
{
|
|
if( !seq[i] )
|
|
return i;
|
|
|
|
if( seq[i] != str[i] )
|
|
return 0;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
Calculate the width of the specified prompt. Does some clever magic
|
|
to detect common escape sequences that may be embeded in a prompt,
|
|
such as color codes.
|
|
*/
|
|
static int calc_prompt_width( array_list_t *arr )
|
|
{
|
|
int res = 0;
|
|
int i, j, k;
|
|
|
|
for( i=0; i<al_get_count( arr ); i++ )
|
|
{
|
|
wchar_t *next = (wchar_t *)al_get( arr, i );
|
|
|
|
for( j=0; next[j]; j++ )
|
|
{
|
|
if( next[j] == L'\e' )
|
|
{
|
|
/*
|
|
This is the start of an escape code. Try to guess it's width.
|
|
*/
|
|
int l;
|
|
int len=0;
|
|
int found = 0;
|
|
|
|
/*
|
|
Test these color escapes with parameter value 0..7
|
|
*/
|
|
char * esc[] =
|
|
{
|
|
set_a_foreground,
|
|
set_a_background,
|
|
set_foreground,
|
|
set_background,
|
|
}
|
|
;
|
|
|
|
/*
|
|
Test these regular escapes without any parameter values
|
|
*/
|
|
char *esc2[] =
|
|
{
|
|
enter_bold_mode,
|
|
exit_attribute_mode,
|
|
enter_underline_mode,
|
|
exit_underline_mode,
|
|
enter_standout_mode,
|
|
exit_standout_mode,
|
|
flash_screen
|
|
}
|
|
;
|
|
|
|
for( l=0; l < (sizeof(esc)/sizeof(char *)) && !found; l++ )
|
|
{
|
|
if( !esc[l] )
|
|
continue;
|
|
|
|
for( k=0; k<8; k++ )
|
|
{
|
|
len = try_sequence( tparm(esc[l],k), &next[j] );
|
|
if( len )
|
|
{
|
|
j += (len-1);
|
|
found = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
for( l=0; l < (sizeof(esc2)/sizeof(char *)) && !found; l++ )
|
|
{
|
|
if( !esc2[l] )
|
|
continue;
|
|
/*
|
|
Test both padded and unpadded version, just to
|
|
be safe. Most versions of tparm don't actually
|
|
seem to do anything these days.
|
|
*/
|
|
len = maxi( try_sequence( tparm(esc2[l]), &next[j] ),
|
|
try_sequence( esc2[l], &next[j] ));
|
|
|
|
if( len )
|
|
{
|
|
j += (len-1);
|
|
found = 1;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
Ordinary decent character. Just add width.
|
|
*/
|
|
res += wcwidth( next[j] );
|
|
}
|
|
|
|
}
|
|
}
|
|
return res;
|
|
}
|
|
|
|
|
|
/**
|
|
Write the prompt to screen. If data->exec_prompt is set, the prompt
|
|
command is first evaluated, and the title will be reexecuted as
|
|
well.
|
|
*/
|
|
static void write_prompt()
|
|
{
|
|
int i;
|
|
set_color( FISH_COLOR_NORMAL, FISH_COLOR_NORMAL );
|
|
|
|
/*
|
|
Check if we need to reexecute the prompt command
|
|
*/
|
|
if( data->exec_prompt )
|
|
{
|
|
|
|
al_foreach( &prompt_list, (void (*)(const void *))&free );
|
|
al_truncate( &prompt_list, 0 );
|
|
|
|
if( data->prompt )
|
|
{
|
|
if( exec_subshell( data->prompt, &prompt_list ) == -1 )
|
|
{
|
|
/* If executing the prompt fails, make sure we at least don't print any junk */
|
|
al_foreach( &prompt_list, (void (*)(const void *))&free );
|
|
al_destroy( &prompt_list );
|
|
al_init( &prompt_list );
|
|
}
|
|
}
|
|
|
|
data->prompt_width=calc_prompt_width( &prompt_list );
|
|
|
|
data->exec_prompt = 0;
|
|
reader_write_title();
|
|
}
|
|
|
|
/*
|
|
Write out the prompt strings
|
|
*/
|
|
|
|
for( i=0; i<al_get_count( &prompt_list); i++ )
|
|
{
|
|
writestr( (wchar_t *)al_get( &prompt_list, i ) );
|
|
}
|
|
set_color( FISH_COLOR_RESET, FISH_COLOR_RESET );
|
|
|
|
}
|
|
|
|
/**
|
|
Write the whole command line (but not the prompt) to the screen. Do
|
|
not set the cursor correctly afterwards.
|
|
*/
|
|
static void write_cmdline()
|
|
{
|
|
int i;
|
|
|
|
for( i=0; data->output[i]; i++ )
|
|
{
|
|
set_color_translated( data->output_color[i] );
|
|
writech( data->output[i] );
|
|
}
|
|
}
|
|
|
|
/**
|
|
perm_left_cursor and parm_right_cursor don't seem to be defined as
|
|
often as cursor_left and cursor_right, so we use this workalike.
|
|
*/
|
|
static void move_cursor( int steps )
|
|
{
|
|
int i;
|
|
|
|
if( steps < 0 ){
|
|
for( i=0; i>steps; i--)
|
|
{
|
|
writembs(cursor_left);
|
|
}
|
|
}
|
|
else
|
|
for( i=0; i<steps; i++)
|
|
writembs(cursor_right);
|
|
}
|
|
|
|
|
|
void reader_init()
|
|
{
|
|
al_init( ¤t_filename);
|
|
}
|
|
|
|
|
|
void reader_destroy()
|
|
{
|
|
al_destroy( ¤t_filename);
|
|
if( readline_buffer )
|
|
{
|
|
sb_destroy( readline_buffer );
|
|
free( readline_buffer );
|
|
readline_buffer=0;
|
|
}
|
|
}
|
|
|
|
|
|
void reader_exit( int do_exit )
|
|
{
|
|
if( is_interactive )
|
|
data->end_loop=do_exit;
|
|
else
|
|
end_loop=do_exit;
|
|
}
|
|
|
|
void repaint()
|
|
{
|
|
int steps;
|
|
|
|
calc_output();
|
|
set_color( FISH_COLOR_RESET, FISH_COLOR_RESET );
|
|
writech('\r');
|
|
writembs(clr_eol);
|
|
write_prompt();
|
|
write_cmdline();
|
|
|
|
/*
|
|
fwprintf( stderr, L"Width of \'%ls\' (length is %d): ",
|
|
&data->buff[data->buff_pos],
|
|
wcslen(&data->buff[data->buff_pos]));
|
|
fwprintf( stderr, L"%d\n", my_wcswidth(&data->buff[data->buff_pos]));
|
|
*/
|
|
|
|
steps = my_wcswidth( &data->output[data->output_pos]);
|
|
if( steps )
|
|
move_cursor( -steps );
|
|
|
|
set_color( FISH_COLOR_NORMAL, -1 );
|
|
reader_save_status();
|
|
}
|
|
|
|
/**
|
|
Make sure color values are correct, and repaint if they are not.
|
|
*/
|
|
static void check_colors()
|
|
{
|
|
reader_super_highlight_me_plenty( data->buff, data->new_color, data->buff_pos, 0 );
|
|
if( memcmp( data->new_color, data->color, sizeof(int)*data->buff_len )!=0 )
|
|
{
|
|
memcpy( data->color, data->new_color, sizeof(int)*data->buff_len );
|
|
|
|
repaint();
|
|
}
|
|
}
|
|
|
|
/**
|
|
Stat stdout and stderr and save result.
|
|
|
|
This should be done before calling a function that may cause output.
|
|
*/
|
|
|
|
static void reader_save_status()
|
|
{
|
|
|
|
#ifdef HAVE_FUTIMES
|
|
/*
|
|
This futimes call tries to trick the system into using st_mtime
|
|
as a tampering flag. This of course only works on systems where
|
|
futimes is defined, but it should make the status saving stuff
|
|
failsafe.
|
|
*/
|
|
struct timeval t[]=
|
|
{
|
|
{
|
|
time(0)-1,
|
|
0
|
|
}
|
|
,
|
|
{
|
|
time(0)-1,
|
|
0
|
|
}
|
|
}
|
|
;
|
|
|
|
futimes( 1, t );
|
|
futimes( 2, t );
|
|
#endif
|
|
|
|
fstat( 1, &prev_buff_1 );
|
|
fstat( 2, &prev_buff_2 );
|
|
}
|
|
|
|
/**
|
|
Stat stdout and stderr and compare result to previous result in
|
|
reader_save_status. Repaint if modification time has changed.
|
|
|
|
Unfortunately, for some reason this call seems to give a lot of
|
|
false positives, at least under Linux.
|
|
*/
|
|
|
|
static void reader_check_status()
|
|
{
|
|
fflush( stdout );
|
|
fflush( stderr );
|
|
|
|
fstat( 1, &post_buff_1 );
|
|
fstat( 2, &post_buff_2 );
|
|
|
|
int changed = ( prev_buff_1.st_mtime != post_buff_1.st_mtime ) ||
|
|
( prev_buff_2.st_mtime != post_buff_2.st_mtime );
|
|
|
|
if (room_for_usec( &post_buff_1))
|
|
{
|
|
changed = changed || ( (&prev_buff_1.st_mtime)[1] != (&post_buff_1.st_mtime)[1] ) ||
|
|
( (&prev_buff_2.st_mtime)[1] != (&post_buff_2.st_mtime)[1] );
|
|
}
|
|
|
|
if( changed )
|
|
{
|
|
repaint();
|
|
set_color( FISH_COLOR_RESET, FISH_COLOR_RESET );
|
|
}
|
|
}
|
|
|
|
/**
|
|
Remove the previous character in the character buffer and on the
|
|
screen using syntax highlighting, etc.
|
|
*/
|
|
static void remove_backward()
|
|
{
|
|
int wdt;
|
|
|
|
if( data->buff_pos <= 0 )
|
|
return;
|
|
|
|
if( data->buff_pos < data->buff_len )
|
|
{
|
|
memmove( &data->buff[data->buff_pos-1],
|
|
&data->buff[data->buff_pos],
|
|
sizeof(wchar_t)*(data->buff_len-data->buff_pos+1) );
|
|
|
|
memmove( &data->color[data->buff_pos-1],
|
|
&data->color[data->buff_pos],
|
|
sizeof(wchar_t)*(data->buff_len-data->buff_pos+1) );
|
|
}
|
|
data->buff_pos--;
|
|
data->buff_len--;
|
|
|
|
wdt=wcwidth(data->buff[data->buff_pos]);
|
|
move_cursor(-wdt);
|
|
data->buff[data->buff_len]='\0';
|
|
// wcscpy(data->search_buff,data->buff);
|
|
|
|
reader_super_highlight_me_plenty( data->buff,
|
|
data->new_color,
|
|
data->buff_pos,
|
|
0 );
|
|
if( (!force_repaint()) && ( memcmp( data->new_color,
|
|
data->color,
|
|
sizeof(int)*data->buff_len )==0 ) &&
|
|
( delete_character != 0) && (wdt==1) )
|
|
{
|
|
/*
|
|
Only do this if delete mode functions, and only for a column
|
|
wide characters, since terminfo seems to break for other
|
|
characters. This last check should be removed when terminfo
|
|
is fixed.
|
|
*/
|
|
if( enter_delete_mode != 0 )
|
|
writembs(enter_delete_mode);
|
|
writembs(delete_character);
|
|
if( exit_delete_mode != 0 )
|
|
writembs(exit_delete_mode);
|
|
}
|
|
else
|
|
{
|
|
memcpy( data->color,
|
|
data->new_color,
|
|
sizeof(int) * data->buff_len );
|
|
|
|
repaint();
|
|
}
|
|
}
|
|
|
|
/**
|
|
Remove the current character in the character buffer and on the
|
|
screen using syntax highlighting, etc.
|
|
*/
|
|
static void remove_forward()
|
|
{
|
|
if( data->buff_pos >= data->buff_len )
|
|
return;
|
|
|
|
move_cursor(wcwidth(data->buff[data->buff_pos]));
|
|
data->buff_pos++;
|
|
|
|
remove_backward();
|
|
}
|
|
|
|
/**
|
|
Insert the character into the command line buffer and print it to
|
|
the screen using syntax highlighting, etc.
|
|
*/
|
|
static int insert_char( int c )
|
|
{
|
|
|
|
if( !check_size() )
|
|
return 0;
|
|
|
|
/* Insert space for extra character at the right position */
|
|
if( data->buff_pos < data->buff_len )
|
|
{
|
|
memmove( &data->buff[data->buff_pos+1],
|
|
&data->buff[data->buff_pos],
|
|
sizeof(wchar_t)*(data->buff_len-data->buff_pos) );
|
|
|
|
memmove( &data->color[data->buff_pos+1],
|
|
&data->color[data->buff_pos],
|
|
sizeof(int)*(data->buff_len-data->buff_pos) );
|
|
}
|
|
/* Set character */
|
|
data->buff[data->buff_pos]=c;
|
|
|
|
/* Update lengths, etc */
|
|
data->buff_pos++;
|
|
data->buff_len++;
|
|
data->buff[data->buff_len]='\0';
|
|
|
|
/* Syntax highlight */
|
|
|
|
reader_super_highlight_me_plenty( data->buff,
|
|
data->new_color,
|
|
data->buff_pos-1,
|
|
0 );
|
|
data->color[data->buff_pos-1] = data->new_color[data->buff_pos-1];
|
|
|
|
/* Check if the coloring has changed */
|
|
if( (!force_repaint()) && ( memcmp( data->new_color,
|
|
data->color,
|
|
sizeof(int)*data->buff_len )==0 ) &&
|
|
( insert_character ||
|
|
( data->buff_pos == data->buff_len ) ||
|
|
enter_insert_mode) )
|
|
{
|
|
/*
|
|
Colors look ok, so we set the right color and insert a
|
|
character
|
|
*/
|
|
set_color_translated( data->color[data->buff_pos-1] );
|
|
if( data->buff_pos < data->buff_len )
|
|
{
|
|
if( enter_insert_mode != 0 )
|
|
writembs(enter_insert_mode);
|
|
else
|
|
writembs(insert_character);
|
|
writech(c);
|
|
if( insert_padding != 0 )
|
|
writembs(insert_padding);
|
|
if( exit_insert_mode != 0 )
|
|
writembs(exit_insert_mode);
|
|
}
|
|
else
|
|
writech(c);
|
|
set_color( FISH_COLOR_NORMAL, -1 );
|
|
}
|
|
else
|
|
{
|
|
/* Nope, colors are off, so we repaint the entire command line */
|
|
memcpy( data->color, data->new_color, sizeof(int) * data->buff_len );
|
|
|
|
repaint();
|
|
}
|
|
// wcscpy(data->search_buff,data->buff);
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
Insert the characters of the string into the command line buffer
|
|
and print them to the screen using syntax highlighting, etc.
|
|
*/
|
|
static int insert_str(wchar_t *str)
|
|
{
|
|
int len = wcslen( str );
|
|
if( len < 4 )
|
|
{
|
|
while( (*str)!=0 )
|
|
if(!insert_char( *str++ ))
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
int old_len = data->buff_len;
|
|
|
|
data->buff_len += len;
|
|
check_size();
|
|
|
|
/* Insert space for extra characters at the right position */
|
|
if( data->buff_pos < old_len )
|
|
{
|
|
memmove( &data->buff[data->buff_pos+len],
|
|
&data->buff[data->buff_pos],
|
|
sizeof(wchar_t)*(data->buff_len-data->buff_pos) );
|
|
}
|
|
memmove( &data->buff[data->buff_pos], str, sizeof(wchar_t)*len );
|
|
data->buff_pos += len;
|
|
data->buff[data->buff_len]='\0';
|
|
|
|
/* Syntax highlight */
|
|
|
|
reader_super_highlight_me_plenty( data->buff,
|
|
data->new_color,
|
|
data->buff_pos-1,
|
|
0 );
|
|
memcpy( data->color, data->new_color, sizeof(int) * data->buff_len );
|
|
|
|
/* repaint */
|
|
|
|
repaint();
|
|
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
/**
|
|
Calculate the length of the common prefix substring of two strings.
|
|
*/
|
|
static int comp_len( wchar_t *a, wchar_t *b )
|
|
{
|
|
int i;
|
|
for( i=0;
|
|
a[i] != '\0' && b[i] != '\0' && a[i]==b[i];
|
|
i++ )
|
|
;
|
|
return i;
|
|
}
|
|
|
|
/**
|
|
Find the outermost quoting style of current token. Returns 0 if token is not quoted.
|
|
|
|
*/
|
|
static wchar_t get_quote( wchar_t *cmd, int l )
|
|
{
|
|
int i=0;
|
|
wchar_t res=0;
|
|
|
|
// fwprintf( stderr, L"Woot %ls\n", cmd );
|
|
|
|
while( 1 )
|
|
{
|
|
if( !cmd[i] )
|
|
break;
|
|
|
|
if( cmd[i] == L'\'' || cmd[i] == L'\"' )
|
|
{
|
|
wchar_t *end = quote_end( &cmd[i] );
|
|
//fwprintf( stderr, L"Jump %d\n", end-cmd );
|
|
if(( end == 0 ) || (!*end) || (end-cmd > l))
|
|
{
|
|
res = cmd[i];
|
|
break;
|
|
}
|
|
i = end-cmd+1;
|
|
}
|
|
else
|
|
i++;
|
|
|
|
}
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
Calculates information on the parameter at the specified index.
|
|
|
|
\param cmd The command to be analyzed
|
|
\param pos An index in the string which is inside the parameter
|
|
\param quote If not 0, store the type of quote this parameter has, can be either ', " or \\0, meaning the string is not quoted.
|
|
\param offset If not 0, get_param will store a pointer to the beginning of the parameter.
|
|
\param string If not 0, get_parm will store a copy of the parameter string as returned by the tokenizer.
|
|
\param type If not 0, get_param will store the token type as returned by tok_last.
|
|
*/
|
|
static void get_param( wchar_t *cmd,
|
|
int pos,
|
|
wchar_t *quote,
|
|
wchar_t **offset,
|
|
wchar_t **string,
|
|
int *type )
|
|
{
|
|
int prev_pos=0;
|
|
wchar_t last_quote = '\0';
|
|
int unfinished;
|
|
|
|
tokenizer tok;
|
|
tok_init( &tok, cmd, TOK_ACCEPT_UNFINISHED );
|
|
|
|
for( ; tok_has_next( &tok ); tok_next( &tok ) )
|
|
{
|
|
if( tok_get_pos( &tok ) > pos )
|
|
break;
|
|
|
|
if( tok_last_type( &tok ) == TOK_STRING )
|
|
last_quote = get_quote( tok_last( &tok ),
|
|
pos - tok_get_pos( &tok ) );
|
|
|
|
if( type != 0 )
|
|
*type = tok_last_type( &tok );
|
|
if( string != 0 )
|
|
wcscpy( *string, tok_last( &tok ) );
|
|
|
|
prev_pos = tok_get_pos( &tok );
|
|
}
|
|
|
|
tok_destroy( &tok );
|
|
|
|
wchar_t c = cmd[pos];
|
|
cmd[pos]=0;
|
|
int cmdlen = wcslen( cmd );
|
|
unfinished = (cmdlen==0);
|
|
if( !unfinished )
|
|
{
|
|
unfinished = (quote != 0);
|
|
|
|
if( !unfinished )
|
|
{
|
|
if( wcschr( L" \t\n\r", cmd[cmdlen-1] ) != 0 )
|
|
{
|
|
if( ( cmdlen == 1) || (cmd[cmdlen-2] != L'\\') )
|
|
{
|
|
unfinished=1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if( quote )
|
|
*quote = last_quote;
|
|
|
|
if( offset != 0 )
|
|
{
|
|
if( !unfinished )
|
|
{
|
|
while( (cmd[prev_pos] != 0) && (wcschr( L";|",cmd[prev_pos])!= 0) )
|
|
prev_pos++;
|
|
|
|
*offset = cmd + prev_pos;
|
|
}
|
|
else
|
|
{
|
|
*offset = cmd + pos;
|
|
}
|
|
}
|
|
cmd[pos]=c;
|
|
}
|
|
|
|
/**
|
|
Insert the string at the current cursor position. The function
|
|
checks if the string is quoted or not and correctly escapes the
|
|
string.
|
|
|
|
\param val the string to insert
|
|
\param is_complete Whether this completion is the whole string or
|
|
just the common prefix of several completions. If the former, end by
|
|
printing a space (and an end quote if the parameter is quoted).
|
|
*/
|
|
static void completion_insert( wchar_t *val, int is_complete )
|
|
{
|
|
wchar_t *replaced;
|
|
|
|
wchar_t quote;
|
|
|
|
get_param( data->buff,
|
|
data->buff_pos,
|
|
"e,
|
|
0, 0, 0 );
|
|
|
|
if( quote == L'\0' )
|
|
{
|
|
replaced = expand_escape( val, 1 );
|
|
}
|
|
else
|
|
{
|
|
int unescapable=0;
|
|
|
|
wchar_t *pin, *pout;
|
|
|
|
replaced = pout =
|
|
malloc( sizeof(wchar_t)*(wcslen(val) + 1) );
|
|
|
|
for( pin=val; *pin; pin++ )
|
|
{
|
|
switch( *pin )
|
|
{
|
|
case L'\n':
|
|
case L'\t':
|
|
case L'\b':
|
|
case L'\r':
|
|
unescapable=1;
|
|
break;
|
|
default:
|
|
*pout++ = *pin;
|
|
break;
|
|
}
|
|
}
|
|
if( unescapable )
|
|
{
|
|
free( replaced );
|
|
wchar_t *tmp = expand_escape( val, 1 );
|
|
replaced = wcsdupcat( L" ", tmp );
|
|
free( tmp);
|
|
replaced[0]=quote;
|
|
}
|
|
else
|
|
*pout = 0;
|
|
}
|
|
|
|
if( insert_str( replaced ) )
|
|
{
|
|
|
|
if( is_complete ) /* Print trailing space since this is the only completion */
|
|
{
|
|
|
|
if( (quote) &&
|
|
(data->buff[data->buff_pos] != quote ) ) /* This is a quoted parameter, first print a quote */
|
|
{
|
|
insert_char( quote );
|
|
}
|
|
insert_char( L' ' );
|
|
}
|
|
}
|
|
|
|
free(replaced);
|
|
}
|
|
|
|
/**
|
|
Run the fish_pager command to display the completion list, and
|
|
insert the result into the backbuffer.
|
|
*/
|
|
|
|
static void run_pager( wchar_t *prefix, int is_quoted, array_list_t *comp )
|
|
{
|
|
int i;
|
|
string_buffer_t cmd;
|
|
wchar_t * prefix_esc;
|
|
|
|
if( !prefix || (wcslen(prefix)==0))
|
|
prefix_esc = wcsdup(L"\"\"");
|
|
else
|
|
prefix_esc = escape( prefix,1);
|
|
|
|
sb_init( &cmd );
|
|
sb_printf( &cmd,
|
|
L"fish_pager %d %ls",
|
|
is_quoted,
|
|
prefix_esc );
|
|
|
|
free( prefix_esc );
|
|
|
|
for( i=0; i<al_get_count( comp); i++ )
|
|
{
|
|
wchar_t *el = escape( (wchar_t*)al_get( comp, i ),1);
|
|
sb_printf( &cmd, L" %ls", el );
|
|
free(el);
|
|
}
|
|
|
|
term_donate();
|
|
|
|
io_data_t *out = io_buffer_create();
|
|
|
|
eval( (wchar_t *)cmd.buff, out, TOP);
|
|
term_steal();
|
|
|
|
io_buffer_read( out );
|
|
|
|
sb_destroy( &cmd );
|
|
|
|
|
|
int nil=0;
|
|
b_append( out->param2.out_buffer, &nil, 1 );
|
|
|
|
wchar_t *tmp;
|
|
wchar_t *str = str2wcs((char *)out->param2.out_buffer->buff);
|
|
|
|
if( str )
|
|
{
|
|
for( tmp = str + wcslen(str)-1; tmp >= str; tmp-- )
|
|
{
|
|
input_unreadch( *tmp );
|
|
}
|
|
free( str );
|
|
}
|
|
|
|
io_buffer_destroy( out);
|
|
|
|
}
|
|
|
|
/**
|
|
Handle the list of completions. This means the following:
|
|
|
|
- If the list is empty, flash the terminal.
|
|
- If the list contains one element, write the whole element, and if
|
|
the element does not end on a '/', '@', ':', or a '=', also write a trailing
|
|
space.
|
|
- If the list contains multiple elements with a common prefix, write
|
|
the prefix.
|
|
- If the list contains multiple elements without
|
|
a common prefix, call run_pager to display a list of completions
|
|
|
|
\param comp the list of completion strings
|
|
*/
|
|
|
|
|
|
static int handle_completions( array_list_t *comp )
|
|
{
|
|
int i;
|
|
|
|
if( al_get_count( comp ) == 0 )
|
|
{
|
|
if( flash_screen != 0 )
|
|
writembs( flash_screen );
|
|
return 0;
|
|
}
|
|
else if( al_get_count( comp ) == 1 )
|
|
{
|
|
wchar_t *comp_str = wcsdup((wchar_t *)al_get( comp, 0 ));
|
|
wchar_t *woot = wcschr( comp_str, COMPLETE_SEP );
|
|
if( woot != 0 )
|
|
*woot = L'\0';
|
|
completion_insert( comp_str,
|
|
( wcslen(comp_str) == 0 ) ||
|
|
( wcschr( L"/=@:",
|
|
comp_str[wcslen(comp_str)-1] ) == 0 ) );
|
|
free( comp_str );
|
|
return 1;
|
|
}
|
|
else
|
|
{
|
|
wchar_t *base = wcsdup( (wchar_t *)al_get( comp, 0 ) );
|
|
int len = wcslen( base );
|
|
for( i=1; i<al_get_count( comp ); i++ )
|
|
{
|
|
int new_len = comp_len( base, (wchar_t *)al_get( comp, i ) );
|
|
len = new_len < len ? new_len: len;
|
|
}
|
|
if( len > 0 )
|
|
{
|
|
base[len]=L'\0';
|
|
wchar_t *woot = wcschr( base, COMPLETE_SEP );
|
|
if( woot != 0 )
|
|
*woot = L'\0';
|
|
completion_insert(base, 0);
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
There is no common prefix in the completions, and show_list
|
|
is true, so we print the list
|
|
*/
|
|
int len;
|
|
wchar_t * prefix;
|
|
wchar_t * prefix_start;
|
|
get_param( data->buff,
|
|
data->buff_pos,
|
|
0,
|
|
&prefix_start,
|
|
0,
|
|
0 );
|
|
|
|
len = &data->buff[data->buff_pos]-prefix_start+1;
|
|
|
|
if( len <= PREFIX_MAX_LEN )
|
|
{
|
|
prefix = malloc( sizeof(wchar_t)*(len+1) );
|
|
wcslcpy( prefix, prefix_start, len );
|
|
prefix[len]=L'\0';
|
|
}
|
|
else
|
|
{
|
|
wchar_t tmp[2]=
|
|
{
|
|
ellipsis_char,
|
|
0
|
|
}
|
|
;
|
|
|
|
prefix = wcsdupcat( tmp,
|
|
prefix_start + (len - PREFIX_MAX_LEN) );
|
|
prefix[PREFIX_MAX_LEN] = 0;
|
|
|
|
}
|
|
|
|
{
|
|
int is_quoted;
|
|
|
|
wchar_t quote;
|
|
get_param( data->buff, data->buff_pos, "e, 0, 0, 0 );
|
|
is_quoted = (quote != L'\0');
|
|
|
|
writech(L'\n');
|
|
|
|
run_pager( prefix, is_quoted, comp );
|
|
|
|
|
|
/*
|
|
Try to print a list of completions. First try with five
|
|
columns, then four, etc. completion_try_print always
|
|
succeeds with one column.
|
|
*/
|
|
/*
|
|
*/
|
|
}
|
|
|
|
free( prefix );
|
|
repaint();
|
|
|
|
}
|
|
|
|
free( base );
|
|
return len;
|
|
}
|
|
}
|
|
|
|
/**
|
|
Reset the terminal. This function is placed in the list of
|
|
functions to call when exiting by using the atexit function. It
|
|
checks whether it is the original parent process that is exiting
|
|
and not a subshell, and if it is the parent, it restores the
|
|
terminal.
|
|
*/
|
|
static void exit_func()
|
|
{
|
|
if( getpid() == original_pid )
|
|
tcsetattr(0, TCSANOW, &saved_modes);
|
|
}
|
|
|
|
/**
|
|
Initialize data for interactive use
|
|
*/
|
|
static void reader_interactive_init()
|
|
{
|
|
/* See if we are running interactively. */
|
|
pid_t shell_pgid;
|
|
|
|
input_init();
|
|
kill_init();
|
|
shell_pgid = getpgrp ();
|
|
|
|
/* Loop until we are in the foreground. */
|
|
while (tcgetpgrp( 0 ) != shell_pgid)
|
|
{
|
|
kill (- shell_pgid, SIGTTIN);
|
|
}
|
|
|
|
/* Put ourselves in our own process group. */
|
|
shell_pgid = getpid ();
|
|
if( getpgrp() != shell_pgid )
|
|
{
|
|
if (setpgid (shell_pgid, shell_pgid) < 0)
|
|
{
|
|
debug( 1,
|
|
_( L"Couldn't put the shell in its own process group" ));
|
|
wperror( L"setpgid" );
|
|
exit (1);
|
|
}
|
|
}
|
|
|
|
/* Grab control of the terminal. */
|
|
if( tcsetpgrp (STDIN_FILENO, shell_pgid) )
|
|
{
|
|
debug( 1,
|
|
_( L"Couldn't grab control of terminal" ) );
|
|
wperror( L"tcsetpgrp" );
|
|
exit(1);
|
|
}
|
|
|
|
|
|
al_init( &prompt_list );
|
|
history_init();
|
|
|
|
|
|
common_handle_winch(0);
|
|
|
|
tcgetattr(0,&shell_modes); /* get the current terminal modes */
|
|
memcpy( &saved_modes,
|
|
&shell_modes,
|
|
sizeof(saved_modes)); /* save a copy so we can reset the terminal later */
|
|
|
|
shell_modes.c_lflag &= ~ICANON; /* turn off canonical mode */
|
|
shell_modes.c_lflag &= ~ECHO; /* turn off echo mode */
|
|
shell_modes.c_cc[VMIN]=1;
|
|
shell_modes.c_cc[VTIME]=0;
|
|
|
|
if( tcsetattr(0,TCSANOW,&shell_modes)) /* set the new modes */
|
|
{
|
|
wperror(L"tcsetattr");
|
|
exit(1);
|
|
}
|
|
|
|
/* We need to know the parents pid so we'll know if we are a subshell */
|
|
original_pid = getpid();
|
|
|
|
if( atexit( &exit_func ) )
|
|
debug( 1, _( L"Could not set exit function" ) );
|
|
|
|
env_set( L"_", L"fish", ENV_GLOBAL );
|
|
}
|
|
|
|
/**
|
|
Destroy data for interactive use
|
|
*/
|
|
static void reader_interactive_destroy()
|
|
{
|
|
kill_destroy();
|
|
al_foreach( &prompt_list, (void (*)(const void *))&free );
|
|
al_destroy( &prompt_list );
|
|
history_destroy();
|
|
|
|
writestr( L"\n" );
|
|
set_color( FISH_COLOR_RESET, FISH_COLOR_RESET );
|
|
input_destroy();
|
|
}
|
|
|
|
|
|
void reader_sanity_check()
|
|
{
|
|
if( is_interactive)
|
|
{
|
|
if(!( data->buff_pos <= data->buff_len ))
|
|
sanity_lose();
|
|
if(!( data->buff_len == wcslen( data->buff ) ))
|
|
sanity_lose();
|
|
}
|
|
}
|
|
|
|
void reader_current_subshell_extent( wchar_t **a, wchar_t **b )
|
|
{
|
|
wchar_t *begin, *end;
|
|
wchar_t *pos;
|
|
|
|
if( a )
|
|
*a=0;
|
|
if( b )
|
|
*b = 0;
|
|
|
|
if( !data )
|
|
return;
|
|
|
|
pos = data->buff;
|
|
|
|
while( 1 )
|
|
{
|
|
int bc, ec;
|
|
|
|
if( expand_locate_subshell( pos,
|
|
&begin,
|
|
&end,
|
|
1 ) <= 0)
|
|
{
|
|
begin=data->buff;
|
|
end = data->buff + wcslen(data->buff);
|
|
break;
|
|
}
|
|
|
|
if( !end )
|
|
{
|
|
end = data->buff + wcslen(data->buff);
|
|
}
|
|
|
|
bc = begin-data->buff;
|
|
ec = end-data->buff;
|
|
|
|
if(( bc < data->buff_pos ) && (ec >= data->buff_pos) )
|
|
{
|
|
begin++;
|
|
break;
|
|
}
|
|
pos = end+1;
|
|
}
|
|
if( a )
|
|
*a = begin;
|
|
if( b )
|
|
*b = end;
|
|
}
|
|
|
|
/**
|
|
Get the beginning and dend of the job or process definition under the cursor
|
|
*/
|
|
static void reader_current_job_or_process_extent( wchar_t **a,
|
|
wchar_t **b,
|
|
int process )
|
|
{
|
|
wchar_t *begin, *end;
|
|
int pos;
|
|
wchar_t *buffcpy;
|
|
int finished=0;
|
|
|
|
tokenizer tok;
|
|
|
|
if( a )
|
|
*a=0;
|
|
if( b )
|
|
*b = 0;
|
|
|
|
reader_current_subshell_extent( &begin, &end );
|
|
if( !end || !begin )
|
|
return;
|
|
|
|
pos = data->buff_pos - (begin - data->buff);
|
|
// fwprintf( stderr, L"Subshell extent: %d %d %d\n", begin-data->buff, end-data->buff, pos );
|
|
|
|
if( a )
|
|
{
|
|
*a = begin;
|
|
}
|
|
|
|
if( b )
|
|
{
|
|
*b = end;
|
|
}
|
|
|
|
buffcpy = wcsndup( begin, end-begin );
|
|
|
|
if( !buffcpy )
|
|
{
|
|
die_mem();
|
|
}
|
|
// fwprintf( stderr, L"Strlen: %d\n", wcslen(buffcpy ) );
|
|
|
|
for( tok_init( &tok, buffcpy, TOK_ACCEPT_UNFINISHED );
|
|
tok_has_next( &tok ) && !finished;
|
|
tok_next( &tok ) )
|
|
{
|
|
int tok_begin = tok_get_pos( &tok );
|
|
// fwprintf( stderr, L".");
|
|
|
|
switch( tok_last_type( &tok ) )
|
|
{
|
|
case TOK_PIPE:
|
|
if( !process )
|
|
break;
|
|
|
|
case TOK_END:
|
|
case TOK_BACKGROUND:
|
|
{
|
|
|
|
// fwprintf( stderr, L"New cmd at %d\n", tok_begin );
|
|
|
|
if( tok_begin >= pos )
|
|
{
|
|
finished=1;
|
|
if( b )
|
|
*b = data->buff + tok_begin;
|
|
}
|
|
else
|
|
{
|
|
if( a )
|
|
*a = data->buff + tok_begin+1;
|
|
}
|
|
break;
|
|
|
|
}
|
|
}
|
|
}
|
|
|
|
// fwprintf( stderr, L"Res: %d %d\n", *a-data->buff, *b-data->buff );
|
|
free( buffcpy);
|
|
|
|
tok_destroy( &tok );
|
|
|
|
}
|
|
|
|
void reader_current_process_extent( wchar_t **a, wchar_t **b )
|
|
{
|
|
reader_current_job_or_process_extent( a, b, 1 );
|
|
}
|
|
|
|
void reader_current_job_extent( wchar_t **a, wchar_t **b )
|
|
{
|
|
reader_current_job_or_process_extent( a, b, 0 );
|
|
}
|
|
|
|
|
|
void reader_current_token_extent( wchar_t **tok_begin,
|
|
wchar_t **tok_end,
|
|
wchar_t **prev_begin,
|
|
wchar_t **prev_end )
|
|
{
|
|
wchar_t *begin, *end;
|
|
int pos;
|
|
wchar_t *buffcpy;
|
|
|
|
tokenizer tok;
|
|
|
|
wchar_t *a, *b, *pa, *pb;
|
|
|
|
|
|
a = b = pa = pb = 0;
|
|
|
|
reader_current_subshell_extent( &begin, &end );
|
|
|
|
if( !end || !begin )
|
|
return;
|
|
|
|
pos = data->buff_pos - (begin - data->buff);
|
|
|
|
a = data->buff + pos;
|
|
b = a;
|
|
pa = data->buff + pos;
|
|
pb = pa;
|
|
|
|
assert( begin >= data->buff );
|
|
assert( begin <= (data->buff+wcslen(data->buff) ) );
|
|
assert( end >= begin );
|
|
assert( end <= (data->buff+wcslen(data->buff) ) );
|
|
|
|
buffcpy = wcsndup( begin, end-begin );
|
|
|
|
if( !buffcpy )
|
|
{
|
|
die_mem();
|
|
}
|
|
|
|
for( tok_init( &tok, buffcpy, TOK_ACCEPT_UNFINISHED );
|
|
tok_has_next( &tok );
|
|
tok_next( &tok ) )
|
|
{
|
|
int tok_begin = tok_get_pos( &tok );
|
|
int tok_end=tok_begin;
|
|
|
|
/*
|
|
Calculate end of token
|
|
*/
|
|
if( tok_last_type( &tok ) == TOK_STRING )
|
|
tok_end +=wcslen(tok_last(&tok));
|
|
|
|
/*
|
|
Cursor was before beginning of this token, means that the
|
|
cursor is between two tokens, so we set it to a zero element
|
|
string and break
|
|
*/
|
|
if( tok_begin > pos )
|
|
{
|
|
a = b = data->buff + pos;
|
|
break;
|
|
}
|
|
|
|
/*
|
|
If cursor is inside the token, this is the token we are
|
|
looking for. If so, set a and b and break
|
|
*/
|
|
if( tok_end >= pos )
|
|
{
|
|
a = begin + tok_get_pos( &tok );
|
|
b = a + wcslen(tok_last(&tok));
|
|
|
|
// fwprintf( stderr, L"Whee %ls\n", *a );
|
|
|
|
break;
|
|
}
|
|
|
|
/*
|
|
Remember previous string token
|
|
*/
|
|
if( tok_last_type( &tok ) == TOK_STRING )
|
|
{
|
|
pa = begin + tok_get_pos( &tok );
|
|
pb = pa + wcslen(tok_last(&tok));
|
|
}
|
|
}
|
|
|
|
free( buffcpy);
|
|
|
|
tok_destroy( &tok );
|
|
|
|
if( tok_begin )
|
|
*tok_begin = a;
|
|
if( tok_end )
|
|
*tok_end = b;
|
|
if( prev_begin )
|
|
*prev_begin = pa;
|
|
if( prev_end )
|
|
*prev_end = pb;
|
|
|
|
assert( pa >= data->buff );
|
|
assert( pa <= (data->buff+wcslen(data->buff) ) );
|
|
assert( pb >= pa );
|
|
assert( pb <= (data->buff+wcslen(data->buff) ) );
|
|
|
|
}
|
|
|
|
|
|
void reader_replace_current_token( wchar_t *new_token )
|
|
{
|
|
|
|
wchar_t *begin, *end;
|
|
string_buffer_t sb;
|
|
int new_pos;
|
|
|
|
/*
|
|
Find current token
|
|
*/
|
|
reader_current_token_extent( &begin, &end, 0, 0 );
|
|
|
|
if( !begin || !end )
|
|
return;
|
|
|
|
// fwprintf( stderr, L"%d %d, %d\n", begin-data->buff, end-data->buff, data->buff_len );
|
|
|
|
/*
|
|
Make new string
|
|
*/
|
|
sb_init( &sb );
|
|
sb_append_substring( &sb, data->buff, begin-data->buff );
|
|
sb_append( &sb, new_token );
|
|
sb_append( &sb, end );
|
|
|
|
new_pos = (begin-data->buff) + wcslen(new_token);
|
|
|
|
reader_set_buffer( (wchar_t *)sb.buff, new_pos );
|
|
sb_destroy( &sb );
|
|
|
|
}
|
|
|
|
|
|
/**
|
|
Set the specified string from the history as the current buffer. Do
|
|
not modify prefix_width.
|
|
*/
|
|
static void handle_history( const wchar_t *new_str )
|
|
{
|
|
data->buff_len = wcslen( new_str );
|
|
check_size();
|
|
wcscpy( data->buff, new_str );
|
|
data->buff_pos=wcslen(data->buff);
|
|
reader_super_highlight_me_plenty( data->buff, data->color, data->buff_pos, 0 );
|
|
|
|
repaint();
|
|
}
|
|
|
|
/**
|
|
Check if the specified string is contained in the list, using
|
|
wcscmp as a comparison function
|
|
*/
|
|
static int contains( const wchar_t *needle,
|
|
array_list_t *haystack )
|
|
{
|
|
int i;
|
|
for( i=0; i<al_get_count( haystack ); i++ )
|
|
{
|
|
if( !wcscmp( needle, al_get( haystack, i ) ) )
|
|
return 1;
|
|
}
|
|
return 0;
|
|
|
|
}
|
|
|
|
/**
|
|
Reset the data structures associated with the token search
|
|
*/
|
|
static void reset_token_history()
|
|
{
|
|
wchar_t *begin, *end;
|
|
|
|
reader_current_token_extent( &begin, &end, 0, 0 );
|
|
if( begin )
|
|
{
|
|
wcslcpy(data->search_buff, begin, end-begin+1);
|
|
}
|
|
else
|
|
data->search_buff[0]=0;
|
|
|
|
data->token_history_pos = -1;
|
|
data->search_pos=0;
|
|
al_foreach( &data->search_prev, (void (*)(const void *))&free );
|
|
al_truncate( &data->search_prev, 0 );
|
|
al_push( &data->search_prev, wcsdup( data->search_buff ) );
|
|
}
|
|
|
|
|
|
/**
|
|
Handles a token search command.
|
|
|
|
\param forward if the search should be forward or reverse
|
|
\param reset whether the current token should be made the new search token
|
|
*/
|
|
static void handle_token_history( int forward, int reset )
|
|
{
|
|
wchar_t *str=0;
|
|
int current_pos;
|
|
tokenizer tok;
|
|
|
|
if( reset )
|
|
{
|
|
/*
|
|
Start a new token search using the current token
|
|
*/
|
|
reset_token_history();
|
|
|
|
}
|
|
|
|
|
|
current_pos = data->token_history_pos;
|
|
|
|
if( forward || data->search_pos < (al_get_count( &data->search_prev )-1) )
|
|
{
|
|
if( forward )
|
|
{
|
|
if( data->search_pos > 0 )
|
|
{
|
|
data->search_pos--;
|
|
}
|
|
str = (wchar_t *)al_get( &data->search_prev, data->search_pos );
|
|
}
|
|
else
|
|
{
|
|
data->search_pos++;
|
|
str = (wchar_t *)al_get( &data->search_prev, data->search_pos );
|
|
}
|
|
|
|
reader_replace_current_token( str );
|
|
reader_super_highlight_me_plenty( data->buff, data->color, data->buff_pos, 0 );
|
|
repaint();
|
|
}
|
|
else
|
|
{
|
|
if( current_pos == -1 )
|
|
{
|
|
/*
|
|
Move to previous line
|
|
*/
|
|
free( (void *)data->token_history_buff );
|
|
data->token_history_buff = wcsdup( history_prev_match(L"") );
|
|
current_pos = wcslen(data->token_history_buff);
|
|
}
|
|
|
|
if( ! wcslen( data->token_history_buff ) )
|
|
{
|
|
/*
|
|
We have reached the end of the history - check if the
|
|
history already contains the search string itself, if so
|
|
return, otherwise add it.
|
|
*/
|
|
|
|
const wchar_t *last = al_get( &data->search_prev, al_get_count( &data->search_prev ) -1 );
|
|
if( wcscmp( last, data->search_buff ) )
|
|
{
|
|
str = wcsdup(data->search_buff);
|
|
}
|
|
else
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
|
|
debug( 3, L"new '%ls'", data->token_history_buff );
|
|
|
|
for( tok_init( &tok, data->token_history_buff, TOK_ACCEPT_UNFINISHED );
|
|
tok_has_next( &tok);
|
|
tok_next( &tok ))
|
|
{
|
|
switch( tok_last_type( &tok ) )
|
|
{
|
|
case TOK_STRING:
|
|
{
|
|
if( wcsstr( tok_last( &tok ), data->search_buff ) )
|
|
{
|
|
debug( 3, L"Found token at pos %d\n", tok_get_pos( &tok ) );
|
|
if( tok_get_pos( &tok ) >= current_pos )
|
|
{
|
|
break;
|
|
}
|
|
debug( 3, L"ok pos" );
|
|
|
|
if( !contains( tok_last( &tok ), &data->search_prev ) )
|
|
{
|
|
free(str);
|
|
data->token_history_pos = tok_get_pos( &tok );
|
|
str = wcsdup(tok_last( &tok ));
|
|
}
|
|
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
tok_destroy( &tok );
|
|
}
|
|
|
|
if( str )
|
|
{
|
|
reader_replace_current_token( str );
|
|
reader_super_highlight_me_plenty( data->buff, data->color, data->buff_pos, 0 );
|
|
repaint();
|
|
al_push( &data->search_prev, str );
|
|
data->search_pos = al_get_count( &data->search_prev )-1;
|
|
}
|
|
else
|
|
{
|
|
data->token_history_pos=-1;
|
|
handle_token_history( 0, 0 );
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/**
|
|
Move buffer position one word or erase one word. This function
|
|
updates both the internal buffer and the screen. It is used by
|
|
M-left, M-right and ^W to do block movement or block erase.
|
|
|
|
\param dir Direction to move/erase. 0 means move left, 1 means move right.
|
|
\param erase Whether to erase the characters along the way or only move past them.
|
|
|
|
*/
|
|
static void move_word( int dir, int erase )
|
|
{
|
|
int end_buff_pos=data->buff_pos;
|
|
int mode=0;
|
|
int step = dir?1:-1;
|
|
|
|
while( mode < 2 )
|
|
{
|
|
if( !dir )
|
|
{
|
|
if( end_buff_pos == 0 )
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
if( end_buff_pos == data->buff_len )
|
|
break;
|
|
}
|
|
end_buff_pos+=step;
|
|
|
|
if( end_buff_pos < data->buff_len )
|
|
{
|
|
switch( mode )
|
|
{
|
|
case 0:
|
|
if( iswalnum(data->buff[end_buff_pos] ) )
|
|
mode++;
|
|
break;
|
|
|
|
case 1:
|
|
if( !iswalnum(data->buff[end_buff_pos] ) )
|
|
{
|
|
if( !dir )
|
|
end_buff_pos -= step;
|
|
mode++;
|
|
}
|
|
break;
|
|
/*
|
|
case 2:
|
|
if( !iswspace(data->buff[end_buff_pos] ) )
|
|
{
|
|
mode++;
|
|
if( !dir )
|
|
end_buff_pos-=step;
|
|
}
|
|
break;
|
|
*/
|
|
}
|
|
}
|
|
|
|
if( mode==2)
|
|
break;
|
|
|
|
}
|
|
|
|
if( erase )
|
|
{
|
|
int remove_count = abs(data->buff_pos - end_buff_pos);
|
|
int first_char = mini( data->buff_pos, end_buff_pos );
|
|
wchar_t *woot = wcsndup( data->buff + first_char, remove_count);
|
|
// fwprintf( stderr, L"Remove from %d to %d\n", first_char, first_char+remove_count );
|
|
|
|
kill_add( woot );
|
|
free( woot );
|
|
memmove( data->buff + first_char, data->buff + first_char+remove_count, sizeof(wchar_t)*(data->buff_len-first_char-remove_count) );
|
|
data->buff_len -= remove_count;
|
|
data->buff_pos = first_char;
|
|
data->buff[data->buff_len]=0;
|
|
|
|
reader_super_highlight_me_plenty( data->buff, data->color, data->buff_pos, 0 );
|
|
|
|
repaint();
|
|
}
|
|
else
|
|
{
|
|
/* move_cursor(end_buff_pos-data->buff_pos);
|
|
data->buff_pos = end_buff_pos;
|
|
*/
|
|
if( end_buff_pos < data->buff_pos )
|
|
{
|
|
while( data->buff_pos != end_buff_pos )
|
|
{
|
|
data->buff_pos--;
|
|
move_cursor( -wcwidth(data->buff[data->buff_pos]));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
while( data->buff_pos != end_buff_pos )
|
|
{
|
|
move_cursor( wcwidth(data->buff[data->buff_pos]));
|
|
data->buff_pos++;
|
|
check_colors();
|
|
}
|
|
}
|
|
|
|
repaint();
|
|
// check_colors();
|
|
}
|
|
}
|
|
|
|
|
|
wchar_t *reader_get_buffer()
|
|
{
|
|
return data?data->buff:0;
|
|
}
|
|
|
|
void reader_set_buffer( wchar_t *b, int p )
|
|
{
|
|
int l = wcslen( b );
|
|
|
|
if( !data )
|
|
return;
|
|
|
|
data->buff_len = l;
|
|
check_size();
|
|
wcscpy( data->buff, b );
|
|
|
|
if( p>=0 )
|
|
{
|
|
data->buff_pos=p;
|
|
}
|
|
else
|
|
{
|
|
data->buff_pos=l;
|
|
// fwprintf( stderr, L"Pos %d\n", l );
|
|
}
|
|
|
|
reader_super_highlight_me_plenty( data->buff,
|
|
data->color,
|
|
data->buff_pos,
|
|
0 );
|
|
}
|
|
|
|
|
|
int reader_get_cursor_pos()
|
|
{
|
|
if( !data )
|
|
return -1;
|
|
|
|
return data->buff_pos;
|
|
}
|
|
|
|
|
|
void reader_run_command( wchar_t *cmd )
|
|
{
|
|
|
|
wchar_t *ft;
|
|
|
|
ft= tok_first( cmd );
|
|
|
|
if( ft != 0 )
|
|
env_set( L"_", ft, ENV_GLOBAL );
|
|
free(ft);
|
|
|
|
reader_write_title();
|
|
|
|
term_donate();
|
|
|
|
eval( cmd, 0, TOP );
|
|
job_reap( 1 );
|
|
|
|
term_steal();
|
|
|
|
env_set( L"_", L"fish", ENV_GLOBAL );
|
|
|
|
#ifdef HAVE__PROC_SELF_STAT
|
|
proc_update_jiffies();
|
|
#endif
|
|
|
|
|
|
}
|
|
|
|
|
|
/**
|
|
Test if the given shell command contains errors. Uses parser_test
|
|
for testing.
|
|
*/
|
|
|
|
static int shell_test( wchar_t *b )
|
|
{
|
|
return !wcslen(b);
|
|
}
|
|
|
|
/**
|
|
Test if the given string contains error. Since this is the error
|
|
detection for general purpose, there are no invalid strings, so
|
|
this function always returns false.
|
|
*/
|
|
static int default_test( wchar_t *b )
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void reader_push( wchar_t *name )
|
|
{
|
|
reader_data_t *n = calloc( 1, sizeof( reader_data_t ) );
|
|
n->name = wcsdup( name );
|
|
n->next = data;
|
|
data=n;
|
|
check_size();
|
|
data->buff[0]=data->search_buff[0]=0;
|
|
data->exec_prompt=1;
|
|
|
|
if( data->next == 0 )
|
|
{
|
|
reader_interactive_init();
|
|
}
|
|
reader_set_highlight_function( &highlight_universal );
|
|
reader_set_test_function( &default_test );
|
|
reader_set_prompt( L"" );
|
|
history_set_mode( name );
|
|
|
|
al_init( &data->search_prev );
|
|
data->token_history_buff=0;
|
|
|
|
}
|
|
|
|
void reader_pop()
|
|
{
|
|
reader_data_t *n = data;
|
|
|
|
if( data == 0 )
|
|
{
|
|
debug( 0, _( L"Pop null reader block" ) );
|
|
sanity_lose();
|
|
return;
|
|
}
|
|
|
|
data=data->next;
|
|
|
|
free(n->name );
|
|
free( n->prompt );
|
|
free( n->buff );
|
|
free( n->color );
|
|
free( n->new_color );
|
|
free( n->search_buff );
|
|
free( n->output );
|
|
free( n->output_color );
|
|
|
|
/*
|
|
Clean up after history search
|
|
*/
|
|
al_foreach( &n->search_prev, (void (*)(const void *))&free );
|
|
al_destroy( &n->search_prev );
|
|
free( (void *)n->token_history_buff);
|
|
|
|
free(n);
|
|
|
|
if( data == 0 )
|
|
{
|
|
reader_interactive_destroy();
|
|
}
|
|
else
|
|
{
|
|
history_set_mode( data->name );
|
|
data->exec_prompt=1;
|
|
}
|
|
}
|
|
|
|
void reader_set_prompt( wchar_t *new_prompt )
|
|
{
|
|
free( data->prompt );
|
|
data->prompt=wcsdup(new_prompt);
|
|
}
|
|
|
|
void reader_set_complete_function( void (*f)( const wchar_t *,
|
|
array_list_t * ) )
|
|
{
|
|
data->complete_func = f;
|
|
}
|
|
|
|
void reader_set_highlight_function( void (*f)( wchar_t *,
|
|
int *,
|
|
int,
|
|
array_list_t * ) )
|
|
{
|
|
data->highlight_func = f;
|
|
}
|
|
|
|
void reader_set_test_function( int (*f)( wchar_t * ) )
|
|
{
|
|
data->test_func = f;
|
|
}
|
|
|
|
/**
|
|
Call specified external highlighting function and then do search
|
|
highlighting.
|
|
*/
|
|
static void reader_super_highlight_me_plenty( wchar_t * buff, int *color, int pos, array_list_t *error )
|
|
{
|
|
data->highlight_func( buff, color, pos, error );
|
|
if( wcslen(data->search_buff) )
|
|
{
|
|
wchar_t * match = wcsstr( buff, data->search_buff );
|
|
if( match )
|
|
{
|
|
int start = match-buff;
|
|
int count = wcslen(data->search_buff );
|
|
int i;
|
|
// fwprintf( stderr, L"WEE color from %d to %d\n", start, start+count );
|
|
|
|
for( i=0; i<count; i++ )
|
|
{
|
|
/*
|
|
Do not overwrite previous highlighting color
|
|
*/
|
|
if( color[start+i]>>8 == 0 )
|
|
{
|
|
color[start+i] |= HIGHLIGHT_SEARCH_MATCH<<8;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
int exit_status()
|
|
{
|
|
if( is_interactive )
|
|
return first_job == 0 && data->end_loop;
|
|
else
|
|
return end_loop;
|
|
}
|
|
|
|
/**
|
|
Read interactively. Read input from stdin while providing editing
|
|
facilities.
|
|
*/
|
|
static int read_i()
|
|
{
|
|
int prev_end_loop=0;
|
|
|
|
reader_push(L"fish");
|
|
reader_set_complete_function( &complete );
|
|
reader_set_highlight_function( &highlight_shell );
|
|
reader_set_test_function( &shell_test );
|
|
|
|
data->prompt_width=60;
|
|
|
|
while( (!data->end_loop) && (!sanity_check()) )
|
|
{
|
|
wchar_t *tmp;
|
|
|
|
if( function_exists( L"fish_prompt" ) )
|
|
reader_set_prompt( L"fish_prompt" );
|
|
else
|
|
reader_set_prompt( DEFAULT_PROMPT );
|
|
|
|
/*
|
|
Put buff in temporary string and clear buff, so
|
|
that we can handle a call to reader_set_buffer
|
|
during evaluation.
|
|
*/
|
|
|
|
tmp = wcsdup( reader_readline() );
|
|
|
|
data->buff_pos=data->buff_len=0;
|
|
data->buff[data->buff_len]=L'\0';
|
|
reader_run_command( tmp );
|
|
free( tmp );
|
|
|
|
if( data->end_loop)
|
|
{
|
|
if( !prev_end_loop && first_job != 0 )
|
|
{
|
|
writestr(_( L"There are stopped jobs\n" ));
|
|
write_prompt();
|
|
data->end_loop = 0;
|
|
prev_end_loop=1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
prev_end_loop=0;
|
|
}
|
|
error_reset();
|
|
|
|
}
|
|
reader_pop();
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
Test if there are bytes available for reading on the specified file
|
|
descriptor
|
|
*/
|
|
static int can_read( int fd )
|
|
{
|
|
struct timeval can_read_timeout = { 0, 0 };
|
|
fd_set fds;
|
|
|
|
FD_ZERO(&fds);
|
|
FD_SET(fd, &fds);
|
|
return select(fd + 1, &fds, 0, 0, &can_read_timeout) == 1;
|
|
}
|
|
|
|
/**
|
|
Test if the specified character is in the private use area that
|
|
fish uses to store internal characters
|
|
*/
|
|
static int wchar_private( wchar_t c )
|
|
{
|
|
return ( (c >= 0xe000) && (c <= 0xf8ff ) );
|
|
}
|
|
|
|
wchar_t *reader_readline()
|
|
{
|
|
|
|
wint_t c;
|
|
int i;
|
|
int last_char=0, yank=0;
|
|
wchar_t *yank_str;
|
|
array_list_t comp;
|
|
int comp_empty=1;
|
|
int finished=0;
|
|
struct termios old_modes;
|
|
|
|
check_size();
|
|
data->search_buff[0]=data->buff[data->buff_len]='\0';
|
|
|
|
|
|
al_init( &comp );
|
|
|
|
data->exec_prompt=1;
|
|
|
|
reader_super_highlight_me_plenty( data->buff, data->color, data->buff_pos, 0 );
|
|
repaint();
|
|
|
|
tcgetattr(0,&old_modes); /* get the current terminal modes */
|
|
if( tcsetattr(0,TCSANOW,&shell_modes)) /* set the new modes */
|
|
{
|
|
wperror(L"tcsetattr");
|
|
exit(1);
|
|
}
|
|
|
|
while( !finished && !data->end_loop)
|
|
{
|
|
|
|
/*
|
|
Save the terminal status so we know if we have to redraw
|
|
*/
|
|
|
|
reader_save_status();
|
|
|
|
/*
|
|
Sometimes strange input sequences seem to generate a zero
|
|
byte. I believe these simply mean a character was pressed
|
|
but it should be ignored. (Example: Trying to add a tilde
|
|
(~) to digit)
|
|
*/
|
|
while( 1 )
|
|
{
|
|
c=input_readch();
|
|
if( ( (!wchar_private(c))) && (c>31) && (c != 127) )
|
|
{
|
|
if( can_read(0) )
|
|
{
|
|
|
|
wchar_t arr[READAHEAD_MAX+1];
|
|
int i;
|
|
|
|
memset( arr, 0, sizeof( arr ) );
|
|
arr[0] = c;
|
|
|
|
for( i=1; i<READAHEAD_MAX; i++ )
|
|
{
|
|
|
|
if( !can_read( 0 ) )
|
|
{
|
|
c = 0;
|
|
break;
|
|
}
|
|
c = input_readch();
|
|
if( (!wchar_private(c)) && (c>31) && (c != 127) )
|
|
{
|
|
arr[i]=c;
|
|
c=0;
|
|
}
|
|
else
|
|
break;
|
|
}
|
|
|
|
insert_str( arr );
|
|
|
|
}
|
|
}
|
|
|
|
if( c != 0 )
|
|
break;
|
|
}
|
|
|
|
reader_check_status();
|
|
|
|
if( (last_char == R_COMPLETE) && (c != R_COMPLETE) && (!comp_empty) )
|
|
{
|
|
al_foreach( &comp, (void (*)(const void *))&free );
|
|
al_truncate( &comp, 0 );
|
|
comp_empty = 1;
|
|
}
|
|
|
|
if( last_char != R_YANK && last_char != R_YANK_POP )
|
|
yank=0;
|
|
|
|
switch( c )
|
|
{
|
|
|
|
/* go to beginning of line*/
|
|
case R_BEGINNING_OF_LINE:
|
|
{
|
|
data->buff_pos = 0;
|
|
|
|
repaint();
|
|
break;
|
|
}
|
|
|
|
/* go to EOL*/
|
|
case R_END_OF_LINE:
|
|
{
|
|
data->buff_pos = data->buff_len;
|
|
|
|
repaint();
|
|
break;
|
|
}
|
|
|
|
case R_NULL:
|
|
{
|
|
data->exec_prompt=1;
|
|
repaint();
|
|
break;
|
|
}
|
|
|
|
/* complete */
|
|
case R_COMPLETE:
|
|
{
|
|
|
|
// fwprintf( stderr, L"aaa\n" );
|
|
if( !data->complete_func )
|
|
break;
|
|
|
|
if( !comp_empty && last_char == R_COMPLETE )
|
|
break;
|
|
|
|
if( comp_empty )
|
|
{
|
|
wchar_t *begin, *end;
|
|
wchar_t *buffcpy;
|
|
|
|
reader_current_subshell_extent( &begin, &end );
|
|
|
|
int len = data->buff_pos - (data->buff - begin);
|
|
buffcpy = wcsndup( begin, len );
|
|
|
|
//fwprintf( stderr, L"String is %ls\n", buffcpy );
|
|
|
|
reader_save_status();
|
|
data->complete_func( buffcpy, &comp );
|
|
reader_check_status();
|
|
|
|
sort_list( &comp );
|
|
remove_duplicates( &comp );
|
|
|
|
free( buffcpy );
|
|
}
|
|
if( (comp_empty =
|
|
handle_completions( &comp ) ) )
|
|
{
|
|
al_foreach( &comp, (void (*)(const void *))&free );
|
|
al_truncate( &comp, 0 );
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
/* kill*/
|
|
case R_KILL_LINE:
|
|
{
|
|
kill_add( &data->buff[data->buff_pos] );
|
|
data->buff_len = data->buff_pos;
|
|
data->buff[data->buff_len]=L'\0';
|
|
|
|
|
|
repaint();
|
|
// wcscpy(data->search_buff,data->buff);
|
|
break;
|
|
}
|
|
|
|
case R_BACKWARD_KILL_LINE:
|
|
{
|
|
wchar_t *str = wcsndup( data->buff, data->buff_pos );
|
|
if( !str )
|
|
die_mem();
|
|
|
|
kill_add( str );
|
|
free( str );
|
|
|
|
data->buff_len = wcslen(data->buff +data->buff_pos);
|
|
memmove( data->buff, data->buff +data->buff_pos, sizeof(wchar_t)*data->buff_len );
|
|
data->buff[data->buff_len]=L'\0';
|
|
data->buff_pos=0;
|
|
reader_super_highlight_me_plenty( data->buff, data->color, data->buff_pos, 0 );
|
|
|
|
repaint();
|
|
// wcscpy(data->search_buff,data->buff);
|
|
break;
|
|
}
|
|
|
|
case R_KILL_WHOLE_LINE:
|
|
{
|
|
kill_add( data->buff );
|
|
data->buff_len = data->buff_pos = 0;
|
|
data->buff[data->buff_len]=L'\0';
|
|
reader_super_highlight_me_plenty( data->buff, data->color, data->buff_pos, 0 );
|
|
|
|
repaint();
|
|
// wcscpy(data->search_buff,data->buff);
|
|
break;
|
|
}
|
|
|
|
/* yank*/
|
|
case R_YANK:
|
|
{ yank_str = kill_yank();
|
|
insert_str( yank_str );
|
|
yank = wcslen( yank_str );
|
|
// wcscpy(data->search_buff,data->buff);
|
|
break;
|
|
}
|
|
|
|
/* rotate killring*/
|
|
case R_YANK_POP:
|
|
{
|
|
if( yank )
|
|
{
|
|
for( i=0; i<yank; i++ )
|
|
remove_backward();
|
|
|
|
yank_str = kill_yank_rotate();
|
|
insert_str(yank_str);
|
|
yank = wcslen(yank_str);
|
|
}
|
|
break;
|
|
}
|
|
|
|
/* Escape was pressed */
|
|
case L'\e':
|
|
{
|
|
if( *data->search_buff )
|
|
{
|
|
if( data->token_history_pos==-1 )
|
|
{
|
|
history_reset();
|
|
reader_set_buffer( data->search_buff,
|
|
wcslen(data->search_buff ) );
|
|
}
|
|
else
|
|
{
|
|
reader_replace_current_token( data->search_buff );
|
|
}
|
|
*data->search_buff=0;
|
|
check_colors();
|
|
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
/* delete backward*/
|
|
case R_BACKWARD_DELETE_CHAR:
|
|
{
|
|
remove_backward();
|
|
break;
|
|
}
|
|
|
|
/* delete forward*/
|
|
case R_DELETE_CHAR:
|
|
{
|
|
remove_forward();
|
|
break;
|
|
}
|
|
|
|
/* exit, but only if line is empty */
|
|
case R_EXIT:
|
|
{
|
|
if( data->buff_len == 0 )
|
|
{
|
|
writestr( L"\n" );
|
|
data->end_loop=1;
|
|
}
|
|
break;
|
|
}
|
|
|
|
/* Newline, evaluate*/
|
|
case L'\n':
|
|
{
|
|
data->buff[data->buff_len]=L'\0';
|
|
|
|
if( !data->test_func( data->buff ) )
|
|
{
|
|
|
|
if( wcslen( data->buff ) )
|
|
{
|
|
// wcscpy(data->search_buff,L"");
|
|
history_add( data->buff );
|
|
}
|
|
finished=1;
|
|
data->buff_pos=data->buff_len;
|
|
check_colors();
|
|
writestr( L"\n" );
|
|
}
|
|
else
|
|
{
|
|
writech('\r');
|
|
writembs(clr_eol);
|
|
writech('\n');
|
|
repaint();
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
/* History up */
|
|
case R_HISTORY_SEARCH_BACKWARD:
|
|
{
|
|
if( (last_char != R_HISTORY_SEARCH_BACKWARD) &&
|
|
(last_char != R_HISTORY_SEARCH_FORWARD) )
|
|
{
|
|
wcscpy(data->search_buff, data->buff );
|
|
data->search_buff[data->buff_pos]=0;
|
|
}
|
|
|
|
handle_history(history_prev_match(data->search_buff));
|
|
break;
|
|
}
|
|
|
|
/* History down */
|
|
case R_HISTORY_SEARCH_FORWARD:
|
|
{
|
|
if( (last_char != R_HISTORY_SEARCH_BACKWARD) &&
|
|
(last_char != R_HISTORY_SEARCH_FORWARD) )
|
|
{
|
|
wcscpy(data->search_buff, data->buff );
|
|
data->search_buff[data->buff_pos]=0;
|
|
}
|
|
|
|
handle_history(history_next_match(data->search_buff));
|
|
break;
|
|
}
|
|
|
|
/* Token search for a earlier match */
|
|
case R_HISTORY_TOKEN_SEARCH_BACKWARD:
|
|
{
|
|
int reset=0;
|
|
if( (last_char != R_HISTORY_TOKEN_SEARCH_BACKWARD) &&
|
|
(last_char != R_HISTORY_TOKEN_SEARCH_FORWARD) )
|
|
{
|
|
reset=1;
|
|
|
|
}
|
|
|
|
handle_token_history( 0, reset );
|
|
|
|
break;
|
|
}
|
|
|
|
/* Token search for a later match */
|
|
case R_HISTORY_TOKEN_SEARCH_FORWARD:
|
|
{
|
|
int reset=0;
|
|
|
|
if( (last_char != R_HISTORY_TOKEN_SEARCH_BACKWARD) &&
|
|
(last_char != R_HISTORY_TOKEN_SEARCH_FORWARD) )
|
|
{
|
|
reset=1;
|
|
}
|
|
|
|
handle_token_history( 1, reset );
|
|
|
|
break;
|
|
}
|
|
|
|
|
|
/* Move left*/
|
|
case R_BACKWARD_CHAR:
|
|
if( data->buff_pos > 0 )
|
|
{
|
|
data->buff_pos--;
|
|
if( !force_repaint() )
|
|
{
|
|
move_cursor( -wcwidth(data->buff[data->buff_pos]));
|
|
check_colors();
|
|
}
|
|
else
|
|
{
|
|
repaint();
|
|
}
|
|
}
|
|
break;
|
|
|
|
/* Move right*/
|
|
case R_FORWARD_CHAR:
|
|
{
|
|
if( data->buff_pos < data->buff_len )
|
|
{
|
|
if( !force_repaint() )
|
|
{
|
|
move_cursor( wcwidth(data->buff[data->buff_pos]));
|
|
data->buff_pos++;
|
|
check_colors();
|
|
}
|
|
else
|
|
{
|
|
data->buff_pos++;
|
|
|
|
repaint();
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
case R_DELETE_LINE:
|
|
{
|
|
data->buff[0]=0;
|
|
data->buff_len=0;
|
|
data->buff_pos=0;
|
|
repaint();
|
|
break;
|
|
}
|
|
|
|
/* kill one word left */
|
|
case R_BACKWARD_KILL_WORD:
|
|
{
|
|
move_word(0,1);
|
|
break;
|
|
}
|
|
|
|
/* kill one word right */
|
|
case R_KILL_WORD:
|
|
{
|
|
move_word(1,1);
|
|
break;
|
|
}
|
|
|
|
/* move one word left*/
|
|
case R_BACKWARD_WORD:
|
|
{
|
|
move_word(0,0);
|
|
break;
|
|
}
|
|
|
|
/* move one word right*/
|
|
case R_FORWARD_WORD:
|
|
{
|
|
move_word( 1,0);
|
|
break;
|
|
}
|
|
|
|
case R_CLEAR_SCREEN:
|
|
{
|
|
writembs( clear_screen );
|
|
|
|
repaint();
|
|
break;
|
|
}
|
|
|
|
case R_BEGINNING_OF_HISTORY:
|
|
{
|
|
history_first();
|
|
break;
|
|
}
|
|
|
|
case R_END_OF_HISTORY:
|
|
{
|
|
history_reset();
|
|
break;
|
|
}
|
|
|
|
/* Other, if a normal character, we add it to the command */
|
|
default:
|
|
{
|
|
if( (!wchar_private(c)) && (c>31) && (c != 127) )
|
|
insert_char( c );
|
|
else
|
|
debug( 0, _( L"Unknown keybinding %d" ), c );
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
if( (c != R_HISTORY_SEARCH_BACKWARD) &&
|
|
(c != R_HISTORY_SEARCH_FORWARD) &&
|
|
(c != R_HISTORY_TOKEN_SEARCH_BACKWARD) &&
|
|
(c != R_HISTORY_TOKEN_SEARCH_FORWARD) )
|
|
{
|
|
data->search_buff[0]=0;
|
|
history_reset();
|
|
data->token_history_pos=-1;
|
|
|
|
}
|
|
|
|
|
|
last_char = c;
|
|
}
|
|
|
|
al_destroy( &comp );
|
|
if( tcsetattr(0,TCSANOW,&old_modes)) /* return to previous mode */
|
|
{
|
|
wperror(L"tcsetattr");
|
|
exit(1);
|
|
}
|
|
|
|
set_color( FISH_COLOR_RESET, FISH_COLOR_RESET );
|
|
|
|
return data->buff;
|
|
}
|
|
|
|
/**
|
|
Read non-interactively. Read input from stdin without displaying
|
|
the prompt, using syntax highlighting. This is used for reading
|
|
scripts and init files.
|
|
*/
|
|
static int read_ni( int fd )
|
|
{
|
|
FILE *in_stream;
|
|
wchar_t *buff=0;
|
|
buffer_t acc;
|
|
|
|
int des = fd == 0 ? dup(0) : fd;
|
|
int res=0;
|
|
|
|
if (des == -1)
|
|
{
|
|
wperror( L"dup" );
|
|
return 1;
|
|
}
|
|
|
|
b_init( &acc );
|
|
|
|
in_stream = fdopen( des, "r" );
|
|
if( in_stream != 0 )
|
|
{
|
|
wchar_t *str;
|
|
int acc_used;
|
|
|
|
while(!feof( in_stream ))
|
|
{
|
|
char buff[4096];
|
|
int c;
|
|
|
|
c = fread(buff, 1, 4096, in_stream);
|
|
if( ferror( in_stream ) )
|
|
{
|
|
debug( 1,
|
|
_( L"Error while reading commands" ) );
|
|
|
|
/*
|
|
Reset buffer. We won't evaluate incomplete files.
|
|
*/
|
|
acc.used=0;
|
|
break;
|
|
}
|
|
|
|
b_append( &acc, buff, c );
|
|
}
|
|
b_append( &acc, "\0", 1 );
|
|
acc_used = acc.used;
|
|
str = str2wcs( acc.buff );
|
|
b_destroy( &acc );
|
|
|
|
if( fclose( in_stream ))
|
|
{
|
|
debug( 1,
|
|
_( L"Error while closing input stream" ) );
|
|
wperror( L"fclose" );
|
|
res = 1;
|
|
}
|
|
|
|
// fwprintf( stderr, L"Woot is %d chars\n", wcslen( acc.buff ) );
|
|
|
|
if( str )
|
|
{
|
|
if( !parser_test( str, 1 ) )
|
|
{
|
|
//fwprintf( stderr, L"We parse it\n" );
|
|
eval( str, 0, TOP );
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
No error reporting - parser_test did that for us
|
|
*/
|
|
res = 1;
|
|
}
|
|
free( str );
|
|
}
|
|
else
|
|
{
|
|
if( acc_used > 1 )
|
|
{
|
|
debug( 1,
|
|
_( L"Could not convert input. Read %d bytes." ),
|
|
acc_used-1 );
|
|
}
|
|
else
|
|
{
|
|
debug( 1,
|
|
_( L"Could not read input stream" ) );
|
|
}
|
|
res=1;
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
debug( 1,
|
|
_( L"Error while opening input stream" ) );
|
|
wperror( L"fdopen" );
|
|
free( buff );
|
|
res=1;
|
|
}
|
|
error_reset();
|
|
return res;
|
|
}
|
|
|
|
int reader_read( int fd )
|
|
{
|
|
int res;
|
|
/*
|
|
If reader_read is called recursively through the '.' builtin,
|
|
we need to preserve is_interactive, so we save the
|
|
original state. We also update the signal handlers.
|
|
*/
|
|
int shell_was_interactive = is_interactive;
|
|
|
|
is_interactive = ((fd == 0) && isatty(STDIN_FILENO));
|
|
signal_set_handlers();
|
|
|
|
res= is_interactive?read_i():read_ni( fd );
|
|
|
|
/*
|
|
If the exit command was called in a script, only exit the
|
|
script, not the program
|
|
*/
|
|
end_loop = 0;
|
|
|
|
is_interactive = shell_was_interactive;
|
|
signal_set_handlers();
|
|
return res;
|
|
}
|