1
0
mirror of https://github.com/rkd77/elinks.git synced 2025-01-03 14:57:44 -05:00

Bug 724: terminal: Correct parsing of ECMA-48 control sequences.

The previous version assumed the first non-digit after the CSI was the
Final Byte, for example the first semicolon in the "\E[?1;2c" report.
It then treated all subsequent bytes as typed characters.
According to Standard ECMA-48 (Fifth Edition - June 1991), there may
be any number of Parameter Bytes in the range 0x30 to 0x3F, and then
any number of Intermediate Bytes in the range 0x20 to 0x2F, between
the CSI and the Final Byte.

This version still does not support control sequences longer than
ITRM_IN_QUEUE_SIZE bytes.
This commit is contained in:
Kalle Olavi Niemitalo 2006-10-29 18:41:16 +02:00 committed by Kalle Olavi Niemitalo
parent 23ca645054
commit b7319a75db

View File

@ -629,25 +629,69 @@ free_and_return:
} }
/* Returns the length of the escape sequence */ /* Parse an ECMA-48 control sequence that was received from a
* terminal. Extract the Final Byte (if there are no Intermediate
* Bytes) and the value of the first parameter (if it is an integer).
*
* This function assumes the control sequence begins with a CSI -
* CONTROL SEQUENCE INTRODUCER encoded as ESC [. (ECMA-48 also allows
* 0x9B as a single-byte CSI, but we don't support that here.)
*
* Return one of:
* -1 if the control sequence is not yet complete; the caller sets a timer.
* 0 if the control sequence does not comply with ECMA-48.
* The length of the control sequence otherwise. */
static inline int static inline int
get_esc_code(unsigned char *str, int len, unsigned char *code, int *num) get_esc_code(unsigned char *str, int len, unsigned char *final_byte,
int *first_param_value)
{ {
const int parameter_pos = 2;
int intermediate_pos;
int final_pos;
int pos; int pos;
*num = 0; *final_byte = '\0';
*code = '\0'; *first_param_value = 0;
for (pos = 2; pos < len; pos++) { /* Parameter Bytes */
if (!isdigit(str[pos]) || pos > 7) { pos = parameter_pos;
*code = str[pos]; while (pos < len && str[pos] >= 0x30 && str[pos] <= 0x3F)
++pos;
return pos + 1; /* Intermediate Bytes */
} intermediate_pos = pos;
*num = *num * 10 + str[pos] - '0'; while (pos < len && str[pos] >= 0x20 && str[pos] <= 0x2F)
} ++pos;
return 0; /* Final Byte */
final_pos = pos;
if (pos >= len)
return -1;
if (!(str[pos] >= 0x40 && str[pos] <= 0x7E))
return 0;
/* The control sequence seems OK. If the first Parameter
* Byte indicates that the parameter string is formatted
* as specified in clause 5.4.2 of ECMA-48, and the first
* parameter is an integer, then compute its value.
* (We need not check @len here because the loop cannot get
* past the Final Byte.) */
for (pos = parameter_pos; str[pos] >= 0x30 && str[pos] <= 0x39; ++pos)
*first_param_value = *first_param_value * 10 + str[pos] - 0x30;
/* If the first parameter contains an embedded separator, then
* the value is not an integer, so discard what we computed. */
if (str[pos] == 0x3A)
*first_param_value = 0;
/* The meaning of the Final Byte depends on the Intermediate
* Bytes. Because we don't currently need to recognize any
* control sequences that use Intermediate Bytes, we just
* discard the Final Byte if there are any Intermediate
* Bytes. */
if (intermediate_pos == final_pos)
*final_byte = str[final_pos];
return final_pos + 1;
} }
/* Define it to dump queue content in a readable form, /* Define it to dump queue content in a readable form,
@ -659,8 +703,16 @@ get_esc_code(unsigned char *str, int len, unsigned char *code, int *num)
#include <ctype.h> /* isprint() isspace() */ #include <ctype.h> /* isprint() isspace() */
#endif #endif
/* Returns length of the escape sequence or -1 if the caller needs to set up /* Decode a control sequence that begins with CSI (CONTROL SEQUENCE
* the ESC delay timer. */ * INTRODUCER) encoded as ESC [, and set *@ev accordingly.
* (ECMA-48 also allows 0x9B as a single-byte CSI, but we don't
* support that here.)
*
* Return one of:
* -1 if the control sequence is not yet complete; the caller sets a timer.
* 0 if the control sequence should be parsed by some other function.
* The length of the control sequence otherwise.
* Returning >0 does not imply this function has altered *ev. */
static int static int
decode_terminal_escape_sequence(struct itrm *itrm, struct interlink_event *ev) decode_terminal_escape_sequence(struct itrm *itrm, struct interlink_event *ev)
{ {
@ -674,7 +726,8 @@ decode_terminal_escape_sequence(struct itrm *itrm, struct interlink_event *ev)
if (itrm->in.queue.data[2] == '[') { if (itrm->in.queue.data[2] == '[') {
/* The terminfo entry for linux has "kf1=\E[[A", etc. /* The terminfo entry for linux has "kf1=\E[[A", etc.
* These are not control sequences compliant with * These are not control sequences compliant with
* clause 5.4 of ECMA-48. */ * clause 5.4 of ECMA-48. (According to ECMA-48,
* "\E[[" is SRS - START REVERSED STRING.) */
if (itrm->in.queue.len >= 4 if (itrm->in.queue.len >= 4
&& itrm->in.queue.data[3] >= 'A' && itrm->in.queue.data[3] >= 'A'
&& itrm->in.queue.data[3] <= 'L') { && itrm->in.queue.data[3] <= 'L') {
@ -687,6 +740,17 @@ decode_terminal_escape_sequence(struct itrm *itrm, struct interlink_event *ev)
} }
el = get_esc_code(itrm->in.queue.data, itrm->in.queue.len, &c, &v); el = get_esc_code(itrm->in.queue.data, itrm->in.queue.len, &c, &v);
if (el == -1) {
/* If the control sequence is incomplete but itrm->in.queue
* is already full, then we must not wait for more input:
* kbd_timeout might call in_kbd and thus process_input
* and come right back here. Better just reject the whole
* thing and let the initial CSI be handled as Alt-[. */
if (itrm->in.queue.len == ITRM_IN_QUEUE_SIZE)
return 0;
else
return -1;
}
#ifdef DEBUG_ITRM_QUEUE #ifdef DEBUG_ITRM_QUEUE
fprintf(stderr, "esc code: %c v=%d c=%c el=%d\n", itrm->in.queue.data[1], v, c, el); fprintf(stderr, "esc code: %c v=%d c=%c el=%d\n", itrm->in.queue.data[1], v, c, el);
fflush(stderr); fflush(stderr);
@ -711,9 +775,8 @@ decode_terminal_escape_sequence(struct itrm *itrm, struct interlink_event *ev)
* this escape sequence with the meaning expected by * this escape sequence with the meaning expected by
* ELinks. Escape sequences with no known terminal may end * ELinks. Escape sequences with no known terminal may end
* up being removed from ELinks when bug 96 is fixed. * up being removed from ELinks when bug 96 is fixed.
*/ */ /* ECMA-48 Terminfo $TERM */
switch (c) { /* ECMA-48 Terminfo $TERM */ switch (c) { /* ------- -------- ----- */
case 0: return -1; /* ------- -------- ----- */
case 'A': kbd.key = KBD_UP; break; /* CUU kcuu1 vt200 */ case 'A': kbd.key = KBD_UP; break; /* CUU kcuu1 vt200 */
case 'B': kbd.key = KBD_DOWN; break; /* CUD kcud1 vt200 */ case 'B': kbd.key = KBD_DOWN; break; /* CUD kcud1 vt200 */
case 'C': kbd.key = KBD_RIGHT; break; /* CUF kcuf1 vt200 */ case 'C': kbd.key = KBD_RIGHT; break; /* CUF kcuf1 vt200 */