mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-25 01:02:24 +00:00
57536cb7a3
This has all the basic necessities to become a test of the terminal's behaviour, in terms of how its data structures evolve as output is sent to it, and perhaps also (by filling in the stub TermWin more usefully) testing what it draws during updates and what it sends in response to query sequences. For the moment, all I've done is to set up the framework, and add one demo test of printing some ordinary text and observing that it appears in the data structures and the cursor has moved. I expect that writing a full test of terminal.c will be a very big job. But perhaps I or someone else will find time to prod it gradually in the background of other work. In particular, when I'm _modifying_ any part of the terminal code, it would be good to add some tests for the part I'm changing, before making the change, and check they still work afterwards.
617 lines
20 KiB
C
617 lines
20 KiB
C
/*
|
|
* Internals of the Terminal structure, used by other modules in the
|
|
* terminal subdirectory and by test suites.
|
|
*/
|
|
|
|
#ifndef PUTTY_TERMINAL_H
|
|
#define PUTTY_TERMINAL_H
|
|
|
|
#include "tree234.h"
|
|
|
|
struct beeptime {
|
|
struct beeptime *next;
|
|
unsigned long ticks;
|
|
};
|
|
|
|
#define TRUST_SIGIL_WIDTH 3
|
|
#define TRUST_SIGIL_CHAR 0xDFFE
|
|
|
|
typedef struct {
|
|
int y, x;
|
|
} pos;
|
|
|
|
typedef struct termchar termchar;
|
|
typedef struct termline termline;
|
|
|
|
struct termchar {
|
|
/*
|
|
* Any code in terminal.c which definitely needs to be changed
|
|
* when extra fields are added here is labelled with a comment
|
|
* saying FULL-TERMCHAR.
|
|
*/
|
|
unsigned long chr;
|
|
unsigned long attr;
|
|
truecolour truecolour;
|
|
|
|
/*
|
|
* The cc_next field is used to link multiple termchars
|
|
* together into a list, so as to fit more than one character
|
|
* into a character cell (Unicode combining characters).
|
|
*
|
|
* cc_next is a relative offset into the current array of
|
|
* termchars. I.e. to advance to the next character in a list,
|
|
* one does `tc += tc->next'.
|
|
*
|
|
* Zero means end of list.
|
|
*/
|
|
int cc_next;
|
|
};
|
|
|
|
struct termline {
|
|
unsigned short lattr;
|
|
int cols; /* number of real columns on the line */
|
|
int size; /* number of allocated termchars
|
|
* (cc-lists may make this > cols) */
|
|
bool temporary; /* true if decompressed from scrollback */
|
|
int cc_free; /* offset to first cc in free list */
|
|
struct termchar *chars;
|
|
bool trusted;
|
|
};
|
|
|
|
struct bidi_cache_entry {
|
|
int width;
|
|
bool trusted;
|
|
struct termchar *chars;
|
|
int *forward, *backward; /* the permutations of line positions */
|
|
};
|
|
|
|
struct term_utf8_decode {
|
|
int state; /* Is there a pending UTF-8 character */
|
|
int chr; /* and what is it so far? */
|
|
int size; /* The size of the UTF character. */
|
|
};
|
|
|
|
struct term_userpass_state;
|
|
|
|
struct terminal_tag {
|
|
|
|
int compatibility_level;
|
|
|
|
tree234 *scrollback; /* lines scrolled off top of screen */
|
|
tree234 *screen; /* lines on primary screen */
|
|
tree234 *alt_screen; /* lines on alternate screen */
|
|
int disptop; /* distance scrolled back (0 or -ve) */
|
|
int tempsblines; /* number of lines of .scrollback that
|
|
can be retrieved onto the terminal
|
|
("temporary scrollback") */
|
|
|
|
termline **disptext; /* buffer of text on real screen */
|
|
int dispcursx, dispcursy; /* location of cursor on real screen */
|
|
int curstype; /* type of cursor on real screen */
|
|
|
|
#define VBELL_TIMEOUT (TICKSPERSEC/10) /* visual bell lasts 1/10 sec */
|
|
|
|
struct beeptime *beephead, *beeptail;
|
|
int nbeeps;
|
|
bool beep_overloaded;
|
|
long lastbeep;
|
|
|
|
#define TTYPE termchar
|
|
#define TSIZE (sizeof(TTYPE))
|
|
|
|
int default_attr, curr_attr, save_attr;
|
|
truecolour curr_truecolour, save_truecolour;
|
|
termchar basic_erase_char, erase_char;
|
|
|
|
bufchain inbuf; /* terminal input buffer */
|
|
|
|
pos curs; /* cursor */
|
|
pos savecurs; /* saved cursor position */
|
|
int marg_t, marg_b; /* scroll margins */
|
|
bool dec_om; /* DEC origin mode flag */
|
|
bool wrap, wrapnext; /* wrap flags */
|
|
bool insert; /* insert-mode flag */
|
|
int cset; /* 0 or 1: which char set */
|
|
int save_cset, save_csattr; /* saved with cursor position */
|
|
bool save_utf, save_wnext; /* saved with cursor position */
|
|
bool rvideo; /* global reverse video flag */
|
|
unsigned long rvbell_startpoint; /* for ESC[?5hESC[?5l vbell */
|
|
bool cursor_on; /* cursor enabled flag */
|
|
bool reset_132; /* Flag ESC c resets to 80 cols */
|
|
bool use_bce; /* Use Background coloured erase */
|
|
bool cblinker; /* When blinking is the cursor on ? */
|
|
bool tblinker; /* When the blinking text is on */
|
|
bool blink_is_real; /* Actually blink blinking text */
|
|
int sco_acs, save_sco_acs; /* CSI 10,11,12m -> OEM charset */
|
|
bool vt52_bold; /* Force bold on non-bold colours */
|
|
bool utf; /* Are we in toggleable UTF-8 mode? */
|
|
term_utf8_decode utf8; /* If so, here's our decoding state */
|
|
bool printing, only_printing; /* Are we doing ANSI printing? */
|
|
int print_state; /* state of print-end-sequence scan */
|
|
bufchain printer_buf; /* buffered data for printer */
|
|
printer_job *print_job;
|
|
|
|
/* ESC 7 saved state for the alternate screen */
|
|
pos alt_savecurs;
|
|
int alt_save_attr;
|
|
truecolour alt_save_truecolour;
|
|
int alt_save_cset, alt_save_csattr;
|
|
bool alt_save_utf;
|
|
bool alt_save_wnext;
|
|
int alt_save_sco_acs;
|
|
|
|
int rows, cols, savelines;
|
|
bool has_focus;
|
|
bool in_vbell;
|
|
long vbell_end;
|
|
bool app_cursor_keys, app_keypad_keys, vt52_mode;
|
|
bool repeat_off, srm_echo, cr_lf_return;
|
|
bool seen_disp_event;
|
|
bool big_cursor;
|
|
|
|
bool xterm_mouse_forbidden;
|
|
int xterm_mouse; /* send mouse messages to host */
|
|
bool xterm_extended_mouse;
|
|
bool urxvt_extended_mouse;
|
|
int mouse_is_down; /* used while tracking mouse buttons */
|
|
int raw_mouse_reported_x;
|
|
int raw_mouse_reported_y;
|
|
|
|
bool bracketed_paste, bracketed_paste_active;
|
|
|
|
int cset_attr[2];
|
|
|
|
/*
|
|
* Saved settings on the alternate screen.
|
|
*/
|
|
int alt_x, alt_y;
|
|
bool alt_wnext, alt_ins;
|
|
bool alt_om, alt_wrap;
|
|
int alt_cset, alt_sco_acs;
|
|
bool alt_utf;
|
|
int alt_t, alt_b;
|
|
int alt_which;
|
|
int alt_sblines; /* # of lines on alternate screen that should be used for scrollback. */
|
|
|
|
#define ARGS_MAX 32 /* max # of esc sequence arguments */
|
|
#define ARG_DEFAULT 0 /* if an arg isn't specified */
|
|
#define def(a,d) ( (a) == ARG_DEFAULT ? (d) : (a) )
|
|
unsigned esc_args[ARGS_MAX];
|
|
int esc_nargs;
|
|
int esc_query;
|
|
#define ANSI(x,y) ((x)+((y)*256))
|
|
#define ANSI_QUE(x) ANSI(x,1)
|
|
|
|
#define OSC_STR_MAX 2048
|
|
int osc_strlen;
|
|
char osc_string[OSC_STR_MAX + 1];
|
|
bool osc_w;
|
|
|
|
char id_string[1024];
|
|
|
|
unsigned char *tabs;
|
|
|
|
enum {
|
|
TOPLEVEL,
|
|
SEEN_ESC,
|
|
SEEN_CSI,
|
|
SEEN_OSC,
|
|
SEEN_OSC_W,
|
|
|
|
DO_CTRLS,
|
|
|
|
SEEN_OSC_P,
|
|
OSC_STRING, OSC_MAYBE_ST, OSC_MAYBE_ST_UTF8,
|
|
VT52_ESC,
|
|
VT52_Y1,
|
|
VT52_Y2,
|
|
VT52_FG,
|
|
VT52_BG
|
|
} termstate;
|
|
|
|
enum {
|
|
NO_SELECTION, ABOUT_TO, DRAGGING, SELECTED
|
|
} selstate;
|
|
enum {
|
|
LEXICOGRAPHIC, RECTANGULAR
|
|
} seltype;
|
|
enum {
|
|
SM_CHAR, SM_WORD, SM_LINE
|
|
} selmode;
|
|
pos selstart, selend, selanchor;
|
|
|
|
short wordness[256];
|
|
|
|
/* Mask of attributes to pay attention to when painting. */
|
|
int attr_mask;
|
|
|
|
wchar_t *paste_buffer;
|
|
int paste_len, paste_pos;
|
|
|
|
Backend *backend;
|
|
|
|
Ldisc *ldisc;
|
|
|
|
TermWin *win;
|
|
|
|
LogContext *logctx;
|
|
|
|
struct unicode_data *ucsdata;
|
|
|
|
unsigned long last_graphic_char;
|
|
|
|
/*
|
|
* We maintain a full copy of a Conf here, not merely a pointer
|
|
* to it. That way, when we're passed a new one for
|
|
* reconfiguration, we can check the differences and adjust the
|
|
* _current_ setting of (e.g.) auto wrap mode rather than only
|
|
* the default.
|
|
*/
|
|
Conf *conf;
|
|
|
|
/*
|
|
* GUI implementations of seat_output call term_out, but it can
|
|
* also be called from the ldisc if the ldisc is called _within_
|
|
* term_out. So we have to guard against re-entrancy - if
|
|
* seat_output is called recursively like this, it will simply add
|
|
* data to the end of the buffer term_out is in the process of
|
|
* working through.
|
|
*/
|
|
bool in_term_out;
|
|
|
|
/*
|
|
* We don't permit window updates too close together, to avoid CPU
|
|
* churn pointlessly redrawing the window faster than the user can
|
|
* read. So after an update, we set window_update_cooldown = true
|
|
* and schedule a timer to reset it to false. In between those
|
|
* times, window updates are not performed, and instead we set
|
|
* window_update_pending = true, which will remind us to perform
|
|
* the deferred redraw when the cooldown period ends and
|
|
* window_update_cooldown is reset to false.
|
|
*/
|
|
bool window_update_pending, window_update_cooldown;
|
|
long window_update_cooldown_end;
|
|
|
|
/*
|
|
* Track pending blinks and tblinks.
|
|
*/
|
|
bool tblink_pending, cblink_pending;
|
|
long next_tblink, next_cblink;
|
|
|
|
/*
|
|
* These are buffers used by the bidi and Arabic shaping code.
|
|
*/
|
|
termchar *ltemp;
|
|
int ltemp_size;
|
|
bidi_char *wcFrom, *wcTo;
|
|
int wcFromTo_size;
|
|
struct bidi_cache_entry *pre_bidi_cache, *post_bidi_cache;
|
|
size_t bidi_cache_size;
|
|
|
|
/*
|
|
* Current trust state, used to annotate every line of the
|
|
* terminal that a graphic character is output to.
|
|
*/
|
|
bool trusted;
|
|
|
|
/*
|
|
* We copy a bunch of stuff out of the Conf structure into local
|
|
* fields in the Terminal structure, to avoid the repeated
|
|
* tree234 lookups which would be involved in fetching them from
|
|
* the former every time.
|
|
*/
|
|
bool ansi_colour;
|
|
strbuf *answerback;
|
|
bool no_arabicshaping;
|
|
int beep;
|
|
bool bellovl;
|
|
int bellovl_n;
|
|
int bellovl_s;
|
|
int bellovl_t;
|
|
bool no_bidi;
|
|
bool bksp_is_delete;
|
|
bool blink_cur;
|
|
bool blinktext;
|
|
bool cjk_ambig_wide;
|
|
int conf_height;
|
|
int conf_width;
|
|
bool crhaslf;
|
|
bool erase_to_scrollback;
|
|
int funky_type, sharrow_type;
|
|
bool lfhascr;
|
|
bool logflush;
|
|
int logtype;
|
|
bool mouse_override;
|
|
bool nethack_keypad;
|
|
bool no_alt_screen;
|
|
bool no_applic_c;
|
|
bool no_applic_k;
|
|
bool no_dbackspace;
|
|
bool no_mouse_rep;
|
|
bool no_remote_charset;
|
|
bool no_remote_resize;
|
|
bool no_remote_wintitle;
|
|
bool no_remote_clearscroll;
|
|
bool rawcnp;
|
|
bool utf8linedraw;
|
|
bool rect_select;
|
|
int remote_qtitle_action;
|
|
bool rxvt_homeend;
|
|
bool scroll_on_disp;
|
|
bool scroll_on_key;
|
|
bool xterm_256_colour;
|
|
bool true_colour;
|
|
|
|
wchar_t *last_selected_text;
|
|
int *last_selected_attr;
|
|
truecolour *last_selected_tc;
|
|
size_t last_selected_len;
|
|
int mouse_select_clipboards[N_CLIPBOARDS];
|
|
int n_mouse_select_clipboards;
|
|
int mouse_paste_clipboard;
|
|
|
|
char *window_title, *icon_title;
|
|
int wintitle_codepage, icontitle_codepage;
|
|
bool minimised;
|
|
|
|
BidiContext *bidi_ctx;
|
|
|
|
/* Multi-layered colour palette. The colours from Conf (plus the
|
|
* default xterm-256 ones that don't have Conf ids at all) have
|
|
* lowest priority, followed by platform overrides if any,
|
|
* followed by escape-sequence overrides during the session. */
|
|
struct term_subpalette {
|
|
rgb values[OSC4_NCOLOURS];
|
|
bool present[OSC4_NCOLOURS];
|
|
} subpalettes[3];
|
|
#define SUBPAL_CONF 0
|
|
#define SUBPAL_PLATFORM 1
|
|
#define SUBPAL_SESSION 2
|
|
|
|
/* The composite palette that we make out of the above */
|
|
rgb palette[OSC4_NCOLOURS];
|
|
|
|
unsigned winpos_x, winpos_y, winpixsize_x, winpixsize_y;
|
|
|
|
/*
|
|
* Assorted 'pending' flags for ancillary window changes performed
|
|
* in term_update. Generally, to trigger one of these operations,
|
|
* you set the pending flag and/or the parameters here, then call
|
|
* term_schedule_update.
|
|
*/
|
|
bool win_move_pending;
|
|
int win_move_pending_x, win_move_pending_y;
|
|
bool win_zorder_pending;
|
|
bool win_zorder_top;
|
|
bool win_minimise_pending;
|
|
bool win_minimise_enable;
|
|
bool win_maximise_pending;
|
|
bool win_maximise_enable;
|
|
bool win_title_pending, win_icon_title_pending;
|
|
bool win_pointer_shape_pending;
|
|
bool win_pointer_shape_raw;
|
|
bool win_refresh_pending;
|
|
bool win_scrollbar_update_pending;
|
|
bool win_palette_pending;
|
|
unsigned win_palette_pending_min, win_palette_pending_limit;
|
|
|
|
/*
|
|
* Unlike the rest of the above 'pending' flags, the one for
|
|
* window resizing has to be more complicated, because it's very
|
|
* likely that a server sending a window-resize escape sequence is
|
|
* going to follow it up immediately with further terminal output
|
|
* that draws a full-screen application expecting the terminal to
|
|
* be the new size.
|
|
*
|
|
* So, once we've requested a window resize from the TermWin, we
|
|
* have to stop processing terminal data until we get back the
|
|
* notification that our window really has changed size (or until
|
|
* we find out that it's not going to).
|
|
*
|
|
* Hence, window resizes go through a small state machine with two
|
|
* different kinds of 'pending'. NEED_SEND is the state where
|
|
* we've received an escape sequence asking for a new size but not
|
|
* yet sent it to the TermWin via win_request_resize; AWAIT_REPLY
|
|
* is the state where we've sent it to the TermWin and are
|
|
* expecting a call back to term_size().
|
|
*
|
|
* So _both_ of those 'pending' states inhibit terminal output
|
|
* processing.
|
|
*
|
|
* (Hence, once we're in either state, we should never handle
|
|
* another resize sequence, so the only possible path through this
|
|
* state machine is to get all the way back to the ground state
|
|
* before doing anything else interesting.)
|
|
*/
|
|
enum {
|
|
WIN_RESIZE_NO, WIN_RESIZE_NEED_SEND, WIN_RESIZE_AWAIT_REPLY
|
|
} win_resize_pending;
|
|
int win_resize_pending_w, win_resize_pending_h;
|
|
|
|
/*
|
|
* Indicates whether term_get_userpass_input is currently using
|
|
* the terminal to present a password prompt or similar, and if
|
|
* so, whether it's overridden the terminal into UTF-8 mode.
|
|
*/
|
|
struct term_userpass_state *userpass_state;
|
|
bool userpass_utf8_override;
|
|
};
|
|
|
|
static inline bool in_utf(Terminal *term)
|
|
{
|
|
return (term->utf ||
|
|
term->ucsdata->line_codepage == CP_UTF8 ||
|
|
(term->userpass_state && term->userpass_utf8_override));
|
|
}
|
|
|
|
unsigned long term_translate(
|
|
Terminal *term, term_utf8_decode *utf8, unsigned char c);
|
|
static inline int term_char_width(Terminal *term, unsigned int c)
|
|
{
|
|
return term->cjk_ambig_wide ? mk_wcwidth_cjk(c) : mk_wcwidth(c);
|
|
}
|
|
|
|
/*
|
|
* UCSINCOMPLETE is returned from term_translate if it's successfully
|
|
* absorbed a byte but not emitted a complete character yet.
|
|
* UCSTRUNCATED indicates a truncated multibyte sequence (so the
|
|
* caller emits an error character and then calls term_translate again
|
|
* with the same input byte). UCSINVALID indicates some other invalid
|
|
* multibyte sequence, such as an overlong synonym, or a standalone
|
|
* continuation byte, or a completely illegal thing like 0xFE. These
|
|
* values are not stored in the terminal data structures at all.
|
|
*/
|
|
#define UCSINCOMPLETE 0x8000003FU /* '?' */
|
|
#define UCSTRUNCATED 0x80000021U /* '!' */
|
|
#define UCSINVALID 0x8000002AU /* '*' */
|
|
|
|
/*
|
|
* Maximum number of combining characters we're willing to store in a
|
|
* character cell. Our linked-list data representation permits an
|
|
* unlimited number of these in principle, but if we allowed that in
|
|
* practice then it would be an easy DoS to just squirt a squillion
|
|
* identical combining characters to someone's terminal and cause
|
|
* their PuTTY or pterm to consume lots of memory and CPU pointlessly.
|
|
*
|
|
* The precise figure of 32 is more or less arbitrary, but one point
|
|
* supporting it is UAX #15's comment that 30 combining characters is
|
|
* "significantly beyond what is required for any linguistic or
|
|
* technical usage".
|
|
*/
|
|
#define CC_LIMIT 32
|
|
|
|
/* ----------------------------------------------------------------------
|
|
* Helper functions for dealing with the small 'pos' structure.
|
|
*/
|
|
|
|
static inline bool poslt(pos p1, pos p2)
|
|
{
|
|
if (p1.y != p2.y)
|
|
return p1.y < p2.y;
|
|
return p1.x < p2.x;
|
|
}
|
|
|
|
static inline bool posle(pos p1, pos p2)
|
|
{
|
|
if (p1.y != p2.y)
|
|
return p1.y < p2.y;
|
|
return p1.x <= p2.x;
|
|
}
|
|
|
|
static inline bool poseq(pos p1, pos p2)
|
|
{
|
|
return p1.y == p2.y && p1.x == p2.x;
|
|
}
|
|
|
|
static inline int posdiff_fn(pos p1, pos p2, int cols)
|
|
{
|
|
return (p1.y - p2.y) * (cols+1) + (p1.x - p2.x);
|
|
}
|
|
|
|
/* Convenience wrapper on posdiff_fn which uses the 'Terminal *term'
|
|
* that more or less every function in terminal.c will have in scope.
|
|
* For safety's sake I include a TYPECHECK that ensures it really is a
|
|
* structure pointer of the right type. */
|
|
#define GET_TERM_COLS TYPECHECK(term == (Terminal *)0, term->cols)
|
|
#define posdiff(p1,p2) posdiff_fn(p1, p2, GET_TERM_COLS)
|
|
|
|
/* Product-order comparisons for rectangular block selection. */
|
|
|
|
static inline bool posPle(pos p1, pos p2)
|
|
{
|
|
return p1.y <= p2.y && p1.x <= p2.x;
|
|
}
|
|
|
|
static inline bool posPle_left(pos p1, pos p2)
|
|
{
|
|
/*
|
|
* This function is used for checking whether a given character
|
|
* cell of the terminal ought to be highlighted as part of the
|
|
* selection, by comparing with term->selend. term->selend stores
|
|
* the location one space to the right of the last highlighted
|
|
* character. So we want to highlight the characters that are
|
|
* less-or-equal (in the product order) to the character just left
|
|
* of p2.
|
|
*
|
|
* (Setting up term->selend that way was the easiest way to get
|
|
* rectangular selection working at all, in a code base that had
|
|
* done lexicographic selection the way I happened to have done
|
|
* it.)
|
|
*/
|
|
return p1.y <= p2.y && p1.x < p2.x;
|
|
}
|
|
|
|
static inline bool incpos_fn(pos *p, int cols)
|
|
{
|
|
if (p->x == cols) {
|
|
p->x = 0;
|
|
p->y++;
|
|
return true;
|
|
}
|
|
p->x++;
|
|
return false;
|
|
}
|
|
|
|
static inline bool decpos_fn(pos *p, int cols)
|
|
{
|
|
if (p->x == 0) {
|
|
p->x = cols;
|
|
p->y--;
|
|
return true;
|
|
}
|
|
p->x--;
|
|
return false;
|
|
}
|
|
|
|
/* Convenience wrappers on incpos and decpos which use term->cols
|
|
* (similarly to posdiff above), and also (for mild convenience and
|
|
* mostly historical inertia) let you leave off the & at every call
|
|
* site. */
|
|
#define incpos(p) incpos_fn(&(p), GET_TERM_COLS)
|
|
#define decpos(p) decpos_fn(&(p), GET_TERM_COLS)
|
|
|
|
struct TermLineEditorCallbackReceiverVtable {
|
|
void (*to_terminal)(TermLineEditorCallbackReceiver *rcv, ptrlen data);
|
|
void (*to_backend)(TermLineEditorCallbackReceiver *rcv, ptrlen data);
|
|
void (*special)(TermLineEditorCallbackReceiver *rcv,
|
|
SessionSpecialCode code, int arg);
|
|
void (*newline)(TermLineEditorCallbackReceiver *rcv);
|
|
};
|
|
struct TermLineEditorCallbackReceiver {
|
|
const TermLineEditorCallbackReceiverVtable *vt;
|
|
};
|
|
TermLineEditor *lineedit_new(Terminal *term, unsigned flags,
|
|
TermLineEditorCallbackReceiver *receiver);
|
|
void lineedit_free(TermLineEditor *le);
|
|
void lineedit_input(TermLineEditor *le, char ch, bool dedicated);
|
|
void lineedit_modify_flags(TermLineEditor *le, unsigned clr, unsigned flip);
|
|
void lineedit_send_line(TermLineEditor *le);
|
|
|
|
/*
|
|
* Flags controlling the behaviour of TermLineEditor.
|
|
*/
|
|
#define LINEEDIT_FLAGS(X) \
|
|
X(LE_INTERRUPT) /* pass SS_IP back to client on ^C */ \
|
|
X(LE_SUSPEND) /* pass SS_SUSP back to client on ^Z */ \
|
|
X(LE_ABORT) /* pass SS_ABORT back to client on ^\ */ \
|
|
X(LE_EOF_ALWAYS) /* pass SS_EOF to client on *any* ^Z
|
|
* (X(not)just if the line buffer is empty) */ \
|
|
X(LE_ESC_ERASES) /* make ESC erase the line, as well as ^U */ \
|
|
X(LE_CRLF_NEWLINE) /* interpret manual ^M^J the same as Return */ \
|
|
/* end of list */
|
|
enum {
|
|
#define ALLOCATE_BIT_POSITION(flag) flag ## _bitpos,
|
|
LINEEDIT_FLAGS(ALLOCATE_BIT_POSITION)
|
|
#undef ALLOCATE_BIT_POSITION
|
|
};
|
|
enum {
|
|
#define DEFINE_FLAG_BIT(flag) flag = 1 << flag ## _bitpos,
|
|
LINEEDIT_FLAGS(DEFINE_FLAG_BIT)
|
|
#undef DEFINE_FLAG_BIT
|
|
};
|
|
|
|
termline *term_get_line(Terminal *term, int y);
|
|
void term_release_line(termline *line);
|
|
|
|
#endif
|