1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-25 09:12:24 +00:00

Lots of things, notably:

* Tidied state machine, with better handling of unknown escape and control
   sequences.
 * Support for automatic newline mode, with lfhascr now being a session
   variable.
 * #include <string.h> for mem*() and str*().

[originally from svn r74]
This commit is contained in:
Ben Harris 1999-03-07 23:19:02 +00:00
parent a403857032
commit 4336d540db

View File

@ -2,6 +2,7 @@
#include <windows.h> #include <windows.h>
#endif /* not macintosh */ #endif /* not macintosh */
#include <string.h>
#include <stdio.h> #include <stdio.h>
#include <stdlib.h> #include <stdlib.h>
@ -27,6 +28,7 @@ static int curs_x, curs_y; /* cursor */
static int save_x, save_y; /* saved cursor position */ static int save_x, save_y; /* saved cursor position */
static int marg_t, marg_b; /* scroll margins */ static int marg_t, marg_b; /* scroll margins */
static int dec_om; /* DEC origin mode flag */ static int dec_om; /* DEC origin mode flag */
static int lfhascr; /* Auto-cr mode flag */
static int wrap, wrapnext; /* wrap flags */ static int wrap, wrapnext; /* wrap flags */
static int insert; /* insert-mode flag */ static int insert; /* insert-mode flag */
static int cset; /* 0 or 1: which char set */ static int cset; /* 0 or 1: which char set */
@ -61,9 +63,10 @@ static int nl_count;
static enum { static enum {
TOPLEVEL, IGNORE_NEXT, TOPLEVEL, IGNORE_NEXT,
SEEN_ESC, SEEN_CSI, SET_GL, SET_GR, SEEN_ESC, SEEN_CSI, SEEN_GZD4, SEEN_G1D4,
SEEN_OSC, SEEN_OSC_P, SEEN_OSC_W, OSC_STRING, OSC_MAYBE_ST, SEEN_OSC, SEEN_OSC_P, SEEN_OSC_W, OSC_STRING, OSC_MAYBE_ST,
SEEN_ESCHASH SEEN_ESCHASH,
SEEN_ESC_CONFUSED, SEEN_CSI_CONFUSED,
} termstate; } termstate;
static enum { static enum {
@ -118,6 +121,7 @@ static void power_on(void) {
alt_cset = cset = 0; alt_cset = cset = 0;
cset_attr[0] = cset_attr[1] = ATTR_ASCII; cset_attr[0] = cset_attr[1] = ATTR_ASCII;
rvideo = 0; rvideo = 0;
lfhascr = cfg.lfhascr;
save_attr = curr_attr = ATTR_DEFAULT; save_attr = curr_attr = ATTR_DEFAULT;
app_cursor_keys = cfg.app_cursor; app_cursor_keys = cfg.app_cursor;
app_keypad_keys = cfg.app_keypad; app_keypad_keys = cfg.app_keypad;
@ -552,6 +556,9 @@ static void toggle_mode (int mode, int query, int state) {
case 4: /* set insert mode */ case 4: /* set insert mode */
insert = state; insert = state;
break; break;
case 20: /* line feed/new line mode */
lfhascr = state;
break;
} }
} }
@ -579,6 +586,20 @@ static void do_osc(void) {
} }
} }
enum c0 {
NUL, SOH, STX, ETX, EOT, ENQ, ACK, BEL,
BS, HT, LF, VT, FF, CR, SO, SI,
DLE, DC1, DC2, DC3, DC4, NAK, SYN, ETB,
CAN, EM, SUB, ESC, IS1, IS2, IS3, IS4
};
enum c1 {
BPH = 0x82, NBH, IND, NEL, SSA, ESA,
HTS, HTJ, VTS, PLD, PLU, RI, SS1, SS2,
DCS, PU1, PU2, STS, CCH, MW, SPA, EPA,
SOS, SCI = 0x9a, CSI, ST, OSC, PM, APC
};
/* /*
* Remove everything currently in `inbuf' and stick it up on the * Remove everything currently in `inbuf' and stick it up on the
* in-memory display. There's a big state machine in here to * in-memory display. There's a big state machine in here to
@ -587,15 +608,17 @@ static void do_osc(void) {
void term_out(void) { void term_out(void) {
int c; int c;
int must_update = FALSE; int must_update = FALSE;
int reprocess = FALSE;
while ( (c = inbuf_getc()) != -1) { while (reprocess || (c = inbuf_getc()) != -1) {
#ifdef LOG #ifdef LOG
{ if (!reprocess) {
static FILE *fp = NULL; static FILE *fp = NULL;
if (!fp) fp = fopen("putty.log", "wb"); if (!fp) fp = fopen("putty.log", "wb");
if (fp) fputc (c, fp); if (fp) fputc (c, fp);
} }
#endif #endif
reprocess = FALSE;
switch (termstate) { switch (termstate) {
case TOPLEVEL: case TOPLEVEL:
do_toplevel: do_toplevel:
@ -619,25 +642,6 @@ void term_out(void) {
disptop = scrtop; disptop = scrtop;
must_update = TRUE; must_update = TRUE;
break; break;
case '\016':
cset = 1;
break;
case '\017':
cset = 0;
break;
case '\033':
termstate = SEEN_ESC;
break;
case 0233:
termstate = SEEN_CSI;
esc_nargs = 1;
esc_args[0] = ARG_DEFAULT;
esc_query = FALSE;
break;
case 0235:
termstate = SEEN_OSC;
esc_args[0] = 0;
break;
case '\015': case '\015':
curs_x = 0; curs_x = 0;
wrapnext = FALSE; wrapnext = FALSE;
@ -648,11 +652,12 @@ void term_out(void) {
case '\013': case '\013':
case '\014': case '\014':
case '\012': case '\012':
case 'IND':
if (curs_y == marg_b) if (curs_y == marg_b)
scroll (marg_t, marg_b, 1, TRUE); scroll (marg_t, marg_b, 1, TRUE);
else if (curs_y < rows-1) else if (curs_y < rows-1)
curs_y++; curs_y++;
if (cfg.lfhascr) if (lfhascr)
curs_x = 0; curs_x = 0;
fix_cpos; fix_cpos;
wrapnext = FALSE; wrapnext = FALSE;
@ -673,8 +678,56 @@ void term_out(void) {
disptop = scrtop; disptop = scrtop;
must_update = TRUE; must_update = TRUE;
break; break;
case '\016':
cset = 1;
break;
case '\017':
cset = 0;
break;
case '\033':
termstate = SEEN_ESC;
break;
case NEL: /* exactly equivalent to CR-LF */
curs_x = 0;
wrapnext = FALSE;
if (curs_y == marg_b)
scroll (marg_t, marg_b, 1, TRUE);
else if (curs_y < rows-1)
curs_y++;
fix_cpos;
wrapnext = FALSE;
nl_count++;
disptop = scrtop;
break;
case HTS: /* set a tab */
tabs[curs_x] = TRUE;
break;
case RI: /* reverse index - backwards LF */
if (curs_y == marg_t)
scroll (marg_t, marg_b, -1, TRUE);
else if (curs_y > 0)
curs_y--;
fix_cpos;
wrapnext = FALSE;
disptop = scrtop;
must_update = TRUE;
break;
case SCI: /* terminal type query */
/* This sequence is standardised as something else entirely. */
back->send ("\033[?6c", 5);
break;
case 0233:
termstate = SEEN_CSI;
esc_nargs = 1;
esc_args[0] = ARG_DEFAULT;
esc_query = FALSE;
break;
case 0235:
termstate = SEEN_OSC;
esc_args[0] = 0;
break;
default: default:
if (c >= ' ' && c != 0234) { if (c >= ' ' && c < 0x7f || c >= 0xa0 ) {
if (wrapnext) { if (wrapnext) {
cpos[1] = ATTR_WRAPPED; cpos[1] = ATTR_WRAPPED;
if (curs_y == marg_b) if (curs_y == marg_b)
@ -699,6 +752,7 @@ void term_out(void) {
} }
disptop = scrtop; disptop = scrtop;
} }
break;
} }
break; break;
case IGNORE_NEXT: case IGNORE_NEXT:
@ -717,32 +771,29 @@ void term_out(void) {
} }
/* else fall through */ /* else fall through */
case SEEN_ESC: case SEEN_ESC:
/*
* According to ECMA-35, an escape sequence consists of
* ESC, a sequence (possibly empty) of intermediate bytes
* from column 02 (SPACE--/), and a final byte from
* columns 03-07 (0--~).
*/
termstate = TOPLEVEL; termstate = TOPLEVEL;
switch (c) { if (c >= 0x40 && c < 0x60) {
case '\005': case '\007': case '\b': case '\016': case '\017': /* Fe sequences -- C1 control as an escape sequence */
case '\033': case 0233: case 0234: case 0235: case '\015': c += 0x40;
case '\013': case '\014': case '\012': case '\t': reprocess = TRUE;
termstate = TOPLEVEL; } else switch (c) {
goto do_toplevel; /* hack... */ /* nF sequences -- with intermediate bytes */
case ' ': /* some weird sequence? */ case '#': /* Single control functions */
termstate = IGNORE_NEXT; termstate = SEEN_ESCHASH;
break; break;
case '[': /* enter CSI mode */ case '(': /* GZD4: should set G0 */
termstate = SEEN_CSI; termstate = SEEN_GZD4;
esc_nargs = 1;
esc_args[0] = ARG_DEFAULT;
esc_query = FALSE;
break; break;
case ']': /* xterm escape sequences */ case ')': /* G1D4: should set G1 */
termstate = SEEN_OSC; termstate = SEEN_G1D4;
esc_args[0] = 0;
break;
case '(': /* should set GL */
termstate = SET_GL;
break;
case ')': /* should set GR */
termstate = SET_GR;
break; break;
/* Fp sequences -- private control functions */
case '7': /* save cursor */ case '7': /* save cursor */
save_cursor (TRUE); save_cursor (TRUE);
break; break;
@ -757,63 +808,47 @@ void term_out(void) {
case '>': case '>':
app_keypad_keys = FALSE; app_keypad_keys = FALSE;
break; break;
case 'D': /* exactly equivalent to LF */ /* Fs sequences -- standardised control functions */
if (curs_y == marg_b) case 'c': /* RIS: restore power-on settings */
scroll (marg_t, marg_b, 1, TRUE);
else if (curs_y < rows-1)
curs_y++;
fix_cpos;
wrapnext = FALSE;
disptop = scrtop;
nl_count++;
break;
case 'E': /* exactly equivalent to CR-LF */
curs_x = 0;
wrapnext = FALSE;
if (curs_y == marg_b)
scroll (marg_t, marg_b, 1, TRUE);
else if (curs_y < rows-1)
curs_y++;
fix_cpos;
wrapnext = FALSE;
nl_count++;
disptop = scrtop;
break;
case 'M': /* reverse index - backwards LF */
if (curs_y == marg_t)
scroll (marg_t, marg_b, -1, TRUE);
else if (curs_y > 0)
curs_y--;
fix_cpos;
wrapnext = FALSE;
disptop = scrtop;
must_update = TRUE;
break;
case 'Z': /* terminal type query */
back->send ("\033[?6c", 5);
break;
case 'c': /* restore power-on settings */
power_on(); power_on();
fix_cpos; fix_cpos;
disptop = scrtop; disptop = scrtop;
must_update = TRUE; must_update = TRUE;
break; break;
case '#': /* ESC # 8 fills screen with Es :-) */ default:
termstate = SEEN_ESCHASH; termstate = SEEN_ESC_CONFUSED;
break; reprocess = TRUE;
case 'H': /* set a tab */
tabs[curs_x] = TRUE;
break; break;
} }
break; break;
case SEEN_ESC_CONFUSED:
/*
* We're in an escape sequence, but we no longer know what
* it means and we just want it to go away
*/
termstate = TOPLEVEL;
if (c < 0x20 || c >= 0x7f)
/*
* ECMA-35 says this isn't allowed, so we can do what
* we like.
*/
reprocess = TRUE;
else if (c <= 0x30)
/* Intermediate byte -- more to come */
termstate = SEEN_ESC_CONFUSED;
/* Otherwise, that was a final byte and we're free! */
break;
case SEEN_CSI: case SEEN_CSI:
/*
* In theory, a control sequence consists of CSI, then a
* sequence (possibly empty) of parameter bytes (0--?)
* then a sequence (possibly empty) of intermediate bytes
* (SPACE--/), then a final byte (@--~). We're rather
* more relaxed, and don't differentiate between parameter
* and intermediate bytes.
*/
termstate = TOPLEVEL; /* default */ termstate = TOPLEVEL; /* default */
switch (c) { switch (c) {
case '\005': case '\007': case '\b': case '\016': case '\017':
case '\033': case 0233: case 0234: case 0235: case '\015':
case '\013': case '\014': case '\012': case '\t':
termstate = TOPLEVEL;
goto do_toplevel; /* hack... */
case '0': case '1': case '2': case '3': case '4': case '0': case '1': case '2': case '3': case '4':
case '5': case '6': case '7': case '8': case '9': case '5': case '6': case '7': case '8': case '9':
if (esc_nargs <= ARGS_MAX) { if (esc_nargs <= ARGS_MAX) {
@ -1059,19 +1094,29 @@ void term_out(void) {
} }
} }
break; break;
default:
termstate = SEEN_CSI_CONFUSED;
reprocess = TRUE;
break;
} }
break; break;
case SET_GL: case SEEN_CSI_CONFUSED:
case SET_GR: termstate = TOPLEVEL;
if (c < 0x20 || c >= 0x7f)
reprocess = TRUE;
else if (c < 0x40)
termstate = SEEN_CSI_CONFUSED;
case SEEN_GZD4:
case SEEN_G1D4:
switch (c) { switch (c) {
case 'A': case 'A':
cset_attr[termstate == SET_GL ? 0 : 1] = ATTR_GBCHR; cset_attr[termstate == SEEN_GZD4 ? 0 : 1] = ATTR_GBCHR;
break; break;
case '0': case '0':
cset_attr[termstate == SET_GL ? 0 : 1] = ATTR_LINEDRW; cset_attr[termstate == SEEN_GZD4 ? 0 : 1] = ATTR_LINEDRW;
break; break;
default: /* specifically, 'B' */ default: /* specifically, 'B' */
cset_attr[termstate == SET_GL ? 0 : 1] = ATTR_ASCII; cset_attr[termstate == SEEN_GZD4 ? 0 : 1] = ATTR_ASCII;
break; break;
} }
termstate = TOPLEVEL; termstate = TOPLEVEL;