mirror of
https://git.tartarus.org/simon/putty.git
synced 2025-01-25 09:12:24 +00:00
4da67d8fa6
In the changes around commit 420fe75552
, I made the terminal
suspend output processing while it waited for a term_size() callback
in response to a resize request. Because on X11 there are unusual
circumstances in which you never receive that callback, I also added a
last-ditch 5-second timeout, so that eventually we'll resume terminal
output processing regardless.
But the timeout lives in terminal.c, in the cross-platform code. This
is pointless on Windows (where resize processing is synchronous, so we
always finish it before the timer code next gets called anyway), but I
decided it was easier to keep the whole mechanism in terminal.c in the
absence of a good reason not to.
Now I've found that reason. We _also_ generate window resizes locally
to the GTK front end, in response to the key combinations that change
the font size, and _those_ still have an asynchrony problem.
So, to begin with, I'm refactoring the request_resize system so that
now there's an explicit callback from the frontend to the terminal to
say 'Your resize request has now been processed, whether or not you've
received a term_size() call'. On Windows, this simplifies matters
greatly because we always know exactly when to call that, and don't
have to keep a 'have we called term_size() already?' flag. On GTK, the
timing complexity previously in terminal.c has moved into window.c.
No functional change (I hope). The payoff will be in the next commit.
564 lines
18 KiB
C
564 lines
18 KiB
C
/*
|
|
* Internals of the Terminal structure, for those other modules
|
|
* which need to look inside it. It would be nice if this could be
|
|
* folded back into terminal.c in future, with an abstraction layer
|
|
* to handle everything that other modules need to know about it;
|
|
* but for the moment, this will do.
|
|
*/
|
|
|
|
#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 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 */
|
|
|
|
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;
|
|
char *answerback;
|
|
int answerbacklen;
|
|
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;
|
|
};
|
|
|
|
static inline bool in_utf(Terminal *term)
|
|
{
|
|
return term->utf || term->ucsdata->line_codepage == CP_UTF8;
|
|
}
|
|
|
|
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)
|
|
|
|
#endif
|