123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618 |
- /*
- * 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 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;
- bool no_bracketed_paste; /* disabled in configuration */
- 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
- bool osc_is_apc;
- 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;
- size_t 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
|