Add parser-links to the docs and add summary comment to the vte.c source file. Signed-off-by: David Herrmann <dh.herrmann@googlemail.com>
935 lines
23 KiB
C
935 lines
23 KiB
C
/*
|
|
* kmscon - VT Emulator
|
|
*
|
|
* Copyright (c) 2011 David Herrmann <dh.herrmann@googlemail.com>
|
|
* Copyright (c) 2011 University of Tuebingen
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
* a copy of this software and associated documentation files
|
|
* (the "Software"), to deal in the Software without restriction, including
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
* the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included
|
|
* in all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
|
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
|
* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
|
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
|
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*/
|
|
|
|
/*
|
|
* Virtual Terminal Emulator
|
|
* This is the VT implementation. It is written from scratch. It uses the
|
|
* console subsystem as output and is tightly bound to it. It supports
|
|
* functionality from vt100 up to vt500 series. It doesn't implement an
|
|
* explicitly selected terminal but tries to support the most important commands
|
|
* to be compatible with existing implementations. However, full vt102
|
|
* compatibility is the least that is provided.
|
|
*
|
|
* The main parser in this file controls the parser-state and dispatches the
|
|
* actions to the related handlers. The parser is based on the state-diagram
|
|
* from Paul Williams: http://vt100.net/emu/
|
|
* It is written from scratch, though.
|
|
* This parser is fully compatible up to the vt500 series. It requires UTF-8 and
|
|
* does not support any other input encoding. The G0 and G1 sets are therefore
|
|
* defined as subsets of UTF-8. You may still map G0-G3 into GL, though.
|
|
*
|
|
* However, the CSI/DCS/etc handlers are not designed after a specific VT
|
|
* series. We try to support all vt102 commands but implement several other
|
|
* often used sequences, too. Feel free to add further.
|
|
*
|
|
* See ./doc/vte.txt for more information on this VT-emulator.
|
|
*/
|
|
|
|
#include <errno.h>
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include <X11/keysym.h>
|
|
|
|
#include "console.h"
|
|
#include "input.h"
|
|
#include "log.h"
|
|
#include "unicode.h"
|
|
#include "vte.h"
|
|
|
|
/* Input parser states */
|
|
enum parser_state {
|
|
STATE_NONE, /* placeholder */
|
|
STATE_GROUND, /* initial state and ground */
|
|
STATE_ESC, /* ESC sequence was started */
|
|
STATE_ESC_INT, /* intermediate escape characters */
|
|
STATE_CSI_ENTRY, /* starting CSI sequence */
|
|
STATE_CSI_PARAM, /* CSI parameters */
|
|
STATE_CSI_INT, /* intermediate CSI characters */
|
|
STATE_CSI_IGNORE, /* CSI error; ignore this CSI sequence */
|
|
STATE_DCS_ENTRY, /* starting DCS sequence */
|
|
STATE_DCS_PARAM, /* DCS parameters */
|
|
STATE_DCS_INT, /* intermediate DCS characters */
|
|
STATE_DCS_PASS, /* DCS data passthrough */
|
|
STATE_DCS_IGNORE, /* DCS error; ignore this DCS sequence */
|
|
STATE_OSC_STRING, /* parsing OCS sequence */
|
|
STATE_ST_IGNORE, /* unimplemented seq; ignore until ST */
|
|
STATE_NUM
|
|
};
|
|
|
|
/* Input parser actions */
|
|
enum parser_action {
|
|
ACTION_NONE, /* placeholder */
|
|
ACTION_IGNORE, /* ignore the character entirely */
|
|
ACTION_PRINT, /* print the character on the console */
|
|
ACTION_EXECUTE, /* execute single control character (C0/C1) */
|
|
ACTION_CLEAR, /* clear current parameter state */
|
|
ACTION_COLLECT, /* collect intermediate character */
|
|
ACTION_PARAM, /* collect parameter character */
|
|
ACTION_ESC_DISPATCH, /* dispatch escape sequence */
|
|
ACTION_CSI_DISPATCH, /* dispatch csi sequence */
|
|
ACTION_DCS_START, /* start of DCS data */
|
|
ACTION_DCS_COLLECT, /* collect DCS data */
|
|
ACTION_DCS_END, /* end of DCS data */
|
|
ACTION_OSC_START, /* start of OSC data */
|
|
ACTION_OSC_COLLECT, /* collect OSC data */
|
|
ACTION_OSC_END, /* end of OSC data */
|
|
ACTION_NUM
|
|
};
|
|
|
|
/* max CSI arguments */
|
|
#define CSI_ARG_MAX 16
|
|
|
|
struct kmscon_vte {
|
|
unsigned long ref;
|
|
struct kmscon_symbol_table *st;
|
|
struct kmscon_console *con;
|
|
|
|
const char *kbd_sym;
|
|
struct kmscon_utf8_mach *mach;
|
|
|
|
unsigned int state;
|
|
unsigned int csi_argc;
|
|
int csi_argv[CSI_ARG_MAX];
|
|
};
|
|
|
|
int kmscon_vte_new(struct kmscon_vte **out, struct kmscon_symbol_table *st)
|
|
{
|
|
struct kmscon_vte *vte;
|
|
int ret;
|
|
|
|
if (!out)
|
|
return -EINVAL;
|
|
|
|
log_debug("vte: new vte object\n");
|
|
|
|
vte = malloc(sizeof(*vte));
|
|
if (!vte)
|
|
return -ENOMEM;
|
|
|
|
memset(vte, 0, sizeof(*vte));
|
|
vte->ref = 1;
|
|
vte->st = st;
|
|
vte->state = STATE_GROUND;
|
|
|
|
ret = kmscon_utf8_mach_new(&vte->mach);
|
|
if (ret)
|
|
goto err_free;
|
|
|
|
kmscon_symbol_table_ref(vte->st);
|
|
*out = vte;
|
|
return 0;
|
|
|
|
err_free:
|
|
free(vte);
|
|
return ret;
|
|
}
|
|
|
|
void kmscon_vte_ref(struct kmscon_vte *vte)
|
|
{
|
|
if (!vte)
|
|
return;
|
|
|
|
vte->ref++;
|
|
}
|
|
|
|
void kmscon_vte_unref(struct kmscon_vte *vte)
|
|
{
|
|
if (!vte || !vte->ref)
|
|
return;
|
|
|
|
if (--vte->ref)
|
|
return;
|
|
|
|
kmscon_console_unref(vte->con);
|
|
kmscon_utf8_mach_free(vte->mach);
|
|
kmscon_symbol_free_u8(vte->kbd_sym);
|
|
kmscon_symbol_table_unref(vte->st);
|
|
free(vte);
|
|
log_debug("vte: destroying vte object\n");
|
|
}
|
|
|
|
void kmscon_vte_bind(struct kmscon_vte *vte, struct kmscon_console *con)
|
|
{
|
|
if (!vte)
|
|
return;
|
|
|
|
kmscon_console_unref(vte->con);
|
|
vte->con = con;
|
|
kmscon_console_ref(vte->con);
|
|
}
|
|
|
|
/* execute control character (C0 or C1) */
|
|
static void do_execute(struct kmscon_vte *vte, uint32_t ctrl)
|
|
{
|
|
switch (ctrl) {
|
|
case 0x00: /* NUL */
|
|
/* Ignore on input */
|
|
break;
|
|
case 0x05: /* ENQ */
|
|
/* Transmit answerback message */
|
|
/* TODO */
|
|
break;
|
|
case 0x07: /* BEL */
|
|
/* Sound bell tone */
|
|
/* TODO: I always considered this annying, however, we
|
|
* should at least provide some way to enable it if the
|
|
* user *really* wants it.
|
|
*/
|
|
break;
|
|
case 0x08: /* BS */
|
|
/* Move cursor one position left */
|
|
/* TODO: Do we really have auto-wrap feature here? vt520
|
|
* doc says nothing about it. We should probably use
|
|
* kmscon_console_move_left() here.
|
|
*/
|
|
kmscon_console_backspace(vte->con);
|
|
break;
|
|
case 0x09: /* HT */
|
|
/* Move to next tab stop or end of line */
|
|
/* TODO */
|
|
break;
|
|
case 0x0a: /* LF */
|
|
/* Line feed or newline (CR/NL mode) */
|
|
/* TODO: implement CR/NL mode */
|
|
kmscon_console_newline(vte->con);
|
|
break;
|
|
case 0x0b: /* VT */
|
|
case 0x0c: /* FF */
|
|
/* Line feed */
|
|
/* TODO: does this depend on CR/NL, too? */
|
|
kmscon_console_move_down(vte->con, 1, true);
|
|
break;
|
|
case 0x0d: /* CR */
|
|
/* Move cursor to left margin */
|
|
kmscon_console_move_line_home(vte->con);
|
|
break;
|
|
case 0x0e: /* SO */
|
|
/* Map G1 character set into GL */
|
|
/* TODO */
|
|
break;
|
|
case 0x0f: /* SI */
|
|
/* Map G0 character set into Gl */
|
|
/* TODO */
|
|
break;
|
|
case 0x11: /* XON */
|
|
/* Resume transmission */
|
|
/* TODO */
|
|
break;
|
|
case 0x13: /* XOFF */
|
|
/* Stop transmission */
|
|
/* TODO */
|
|
break;
|
|
case 0x18: /* CAN */
|
|
/* Cancel escape sequence */
|
|
/* nothing to do here */
|
|
break;
|
|
case 0x1a: /* SUB */
|
|
/* Discard current escape sequence and show err-sym */
|
|
/* TODO: show reversed question mark */
|
|
kmscon_console_write(vte->con, '?');
|
|
break;
|
|
case 0x1b: /* ESC */
|
|
/* Invokes an escape sequence */
|
|
/* nothing to do here */
|
|
break;
|
|
case 0x84: /* IND */
|
|
/* Move down one row, perform scroll-up if needed */
|
|
/* TODO */
|
|
break;
|
|
case 0x85: /* NEL */
|
|
/* CR/NL with scroll-up if needed */
|
|
/* TODO */
|
|
break;
|
|
case 0x88: /* HTS */
|
|
/* Set tab stop at current position */
|
|
/* TODO */
|
|
break;
|
|
case 0x8d: /* RI */
|
|
/* Move up one row, perform scroll-down if needed */
|
|
/* TODO */
|
|
break;
|
|
case 0x8e: /* SS2 */
|
|
/* Temporarily map G2 into GL for next char only */
|
|
/* TODO */
|
|
break;
|
|
case 0x8f: /* SS3 */
|
|
/* Temporarily map G3 into GL for next char only */
|
|
/* TODO */
|
|
break;
|
|
case 0x9a: /* DECID */
|
|
/* Send device attributes response like ANSI DA */
|
|
/* TODO*/
|
|
break;
|
|
case 0x9c: /* ST */
|
|
/* End control string */
|
|
/* nothing to do here */
|
|
break;
|
|
default:
|
|
log_warn("vte: unhandled control char %u\n", ctrl);
|
|
}
|
|
}
|
|
|
|
static void do_clear(struct kmscon_vte *vte)
|
|
{
|
|
int i;
|
|
|
|
vte->csi_argc = 0;
|
|
for (i = 0; i < CSI_ARG_MAX; ++i)
|
|
vte->csi_argv[i] = -1;
|
|
}
|
|
|
|
static void do_collect(struct kmscon_vte *vte, uint32_t data)
|
|
{
|
|
}
|
|
|
|
static void do_param(struct kmscon_vte *vte, uint32_t data)
|
|
{
|
|
int new;
|
|
|
|
if (data == ';') {
|
|
if (vte->csi_argc < CSI_ARG_MAX)
|
|
vte->csi_argc++;
|
|
return;
|
|
}
|
|
|
|
if (vte->csi_argc >= CSI_ARG_MAX)
|
|
return;
|
|
|
|
/* avoid integer overflows; max allowed value is 16384 anyway */
|
|
if (vte->csi_argv[vte->csi_argc] > 0xffff)
|
|
return;
|
|
|
|
if (data >= '0' && data <= '9') {
|
|
new = vte->csi_argv[vte->csi_argc];
|
|
if (new <= 0)
|
|
new = data - '0';
|
|
else
|
|
new = new * 10 + data - '0';
|
|
vte->csi_argv[vte->csi_argc] = new;
|
|
}
|
|
}
|
|
|
|
static void do_esc(struct kmscon_vte *vte, uint32_t data)
|
|
{
|
|
switch (data) {
|
|
case 'D': /* IND */
|
|
/* Move down one row, perform scroll-up if needed */
|
|
/* TODO */
|
|
break;
|
|
case 'E': /* NEL */
|
|
/* CR/NL with scroll-up if needed */
|
|
/* TODO */
|
|
break;
|
|
case 'H': /* HTS */
|
|
/* Set tab stop at current position */
|
|
/* TODO */
|
|
break;
|
|
case 'M': /* RI */
|
|
/* Move up one row, perform scroll-down if needed */
|
|
/* TODO */
|
|
break;
|
|
case 'N': /* SS2 */
|
|
/* Temporarily map G2 into GL for next char only */
|
|
/* TODO */
|
|
break;
|
|
case 'O': /* SS3 */
|
|
/* Temporarily map G3 into GL for next char only */
|
|
/* TODO */
|
|
break;
|
|
case 'Z': /* DECID */
|
|
/* Send device attributes response like ANSI DA */
|
|
/* TODO*/
|
|
break;
|
|
case '\\': /* ST */
|
|
/* End control string */
|
|
/* nothing to do here */
|
|
break;
|
|
default:
|
|
log_warn("vte: unhandled escape seq %u\n", data);
|
|
}
|
|
}
|
|
|
|
static void do_csi(struct kmscon_vte *vte, uint32_t data)
|
|
{
|
|
int num;
|
|
|
|
if (vte->csi_argc < CSI_ARG_MAX)
|
|
vte->csi_argc++;
|
|
|
|
switch (data) {
|
|
case 'A':
|
|
num = vte->csi_argv[0];
|
|
if (num <= 0)
|
|
num = 1;
|
|
kmscon_console_move_up(vte->con, num, false);
|
|
break;
|
|
case 'B':
|
|
num = vte->csi_argv[0];
|
|
if (num <= 0)
|
|
num = 1;
|
|
kmscon_console_move_down(vte->con, num, false);
|
|
break;
|
|
case 'C':
|
|
num = vte->csi_argv[0];
|
|
if (num <= 0)
|
|
num = 1;
|
|
kmscon_console_move_right(vte->con, num);
|
|
break;
|
|
case 'D':
|
|
num = vte->csi_argv[0];
|
|
if (num <= 0)
|
|
num = 1;
|
|
kmscon_console_move_left(vte->con, num);
|
|
break;
|
|
case 'J':
|
|
if (vte->csi_argv[0] <= 0)
|
|
kmscon_console_erase_cursor_to_screen(vte->con);
|
|
else if (vte->csi_argv[0] == 1)
|
|
kmscon_console_erase_screen_to_cursor(vte->con);
|
|
else if (vte->csi_argv[0] == 2)
|
|
kmscon_console_erase_screen(vte->con);
|
|
break;
|
|
case 'K':
|
|
if (vte->csi_argv[0] <= 0)
|
|
kmscon_console_erase_cursor_to_end(vte->con);
|
|
else if (vte->csi_argv[0] == 1)
|
|
kmscon_console_erase_home_to_cursor(vte->con);
|
|
else if (vte->csi_argv[0] == 2)
|
|
kmscon_console_erase_current_line(vte->con);
|
|
break;
|
|
default:
|
|
log_debug("vte: unhandled CSI sequence %c\n", data);
|
|
}
|
|
}
|
|
|
|
/* perform parser action */
|
|
static void do_action(struct kmscon_vte *vte, uint32_t data, int action)
|
|
{
|
|
kmscon_symbol_t sym;
|
|
|
|
switch (action) {
|
|
case ACTION_NONE:
|
|
/* do nothing */
|
|
return;
|
|
case ACTION_IGNORE:
|
|
/* ignore character */
|
|
break;
|
|
case ACTION_PRINT:
|
|
sym = kmscon_symbol_make(data);
|
|
kmscon_console_write(vte->con, sym);
|
|
break;
|
|
case ACTION_EXECUTE:
|
|
do_execute(vte, data);
|
|
break;
|
|
case ACTION_CLEAR:
|
|
do_clear(vte);
|
|
break;
|
|
case ACTION_COLLECT:
|
|
do_collect(vte, data);
|
|
break;
|
|
case ACTION_PARAM:
|
|
do_param(vte, data);
|
|
break;
|
|
case ACTION_ESC_DISPATCH:
|
|
do_esc(vte, data);
|
|
break;
|
|
case ACTION_CSI_DISPATCH:
|
|
do_csi(vte, data);
|
|
break;
|
|
case ACTION_DCS_START:
|
|
break;
|
|
case ACTION_DCS_COLLECT:
|
|
break;
|
|
case ACTION_DCS_END:
|
|
break;
|
|
case ACTION_OSC_START:
|
|
break;
|
|
case ACTION_OSC_COLLECT:
|
|
break;
|
|
case ACTION_OSC_END:
|
|
break;
|
|
default:
|
|
log_warn("vte: invalid action %d\n", action);
|
|
}
|
|
}
|
|
|
|
/* entry actions to be performed when entering the selected state */
|
|
static int entry_action[] = {
|
|
[STATE_CSI_ENTRY] = ACTION_CLEAR,
|
|
[STATE_DCS_ENTRY] = ACTION_CLEAR,
|
|
[STATE_DCS_PASS] = ACTION_DCS_START,
|
|
[STATE_ESC] = ACTION_CLEAR,
|
|
[STATE_OSC_STRING] = ACTION_OSC_START,
|
|
[STATE_NUM] = ACTION_NONE,
|
|
};
|
|
|
|
/* exit actions to be performed when leaving the selected state */
|
|
static int exit_action[] = {
|
|
[STATE_DCS_PASS] = ACTION_DCS_END,
|
|
[STATE_OSC_STRING] = ACTION_OSC_END,
|
|
[STATE_NUM] = ACTION_NONE,
|
|
};
|
|
|
|
/* perform state transision and dispatch related actions */
|
|
static void do_trans(struct kmscon_vte *vte, uint32_t data, int state, int act)
|
|
{
|
|
if (state != STATE_NONE) {
|
|
/* A state transition occurs. Perform exit-action,
|
|
* transition-action and entry-action. Even when performing a
|
|
* transition to the same state as the current state we do this.
|
|
* Use STATE_NONE if this is not the desired behavior.
|
|
*/
|
|
do_action(vte, data, exit_action[vte->state]);
|
|
do_action(vte, data, act);
|
|
do_action(vte, data, entry_action[state]);
|
|
vte->state = state;
|
|
} else {
|
|
do_action(vte, data, act);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Escape sequence parser
|
|
* This parses the new input character \data. It performs state transition and
|
|
* calls the right callbacks for each action.
|
|
*/
|
|
static void parse_data(struct kmscon_vte *vte, uint32_t raw)
|
|
{
|
|
/* events that may occur in any state */
|
|
switch (raw) {
|
|
case 0x18:
|
|
case 0x1a:
|
|
case 0x80 ... 0x8f:
|
|
case 0x91 ... 0x97:
|
|
case 0x99:
|
|
case 0x9a:
|
|
case 0x9c:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_EXECUTE);
|
|
return;
|
|
case 0x1b:
|
|
do_trans(vte, raw, STATE_ESC, ACTION_NONE);
|
|
return;
|
|
case 0x98:
|
|
case 0x9e:
|
|
case 0x9f:
|
|
do_trans(vte, raw, STATE_ST_IGNORE, ACTION_NONE);
|
|
return;
|
|
case 0x90:
|
|
do_trans(vte, raw, STATE_DCS_ENTRY, ACTION_NONE);
|
|
return;
|
|
case 0x9d:
|
|
do_trans(vte, raw, STATE_OSC_STRING, ACTION_NONE);
|
|
return;
|
|
case 0x9b:
|
|
do_trans(vte, raw, STATE_CSI_ENTRY, ACTION_NONE);
|
|
return;
|
|
}
|
|
|
|
/* events that depend on the current state */
|
|
switch (vte->state) {
|
|
case STATE_GROUND:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
case 0x80 ... 0x8f:
|
|
case 0x91 ... 0x9a:
|
|
case 0x9c:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_EXECUTE);
|
|
return;
|
|
case 0x20 ... 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_PRINT);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_NONE, ACTION_PRINT);
|
|
return;
|
|
case STATE_ESC:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_EXECUTE);
|
|
return;
|
|
case 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x20 ... 0x2f:
|
|
do_trans(vte, raw, STATE_ESC_INT, ACTION_COLLECT);
|
|
return;
|
|
case 0x30 ... 0x4f:
|
|
case 0x51 ... 0x57:
|
|
case 0x59:
|
|
case 0x5a:
|
|
case 0x5c:
|
|
case 0x60 ... 0x7e:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_ESC_DISPATCH);
|
|
return;
|
|
case 0x5b:
|
|
do_trans(vte, raw, STATE_CSI_ENTRY, ACTION_NONE);
|
|
return;
|
|
case 0x5d:
|
|
do_trans(vte, raw, STATE_OSC_STRING, ACTION_NONE);
|
|
return;
|
|
case 0x50:
|
|
do_trans(vte, raw, STATE_DCS_ENTRY, ACTION_NONE);
|
|
return;
|
|
case 0x58:
|
|
case 0x5e:
|
|
case 0x5f:
|
|
do_trans(vte, raw, STATE_ST_IGNORE, ACTION_NONE);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_ESC_INT, ACTION_COLLECT);
|
|
return;
|
|
case STATE_ESC_INT:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_EXECUTE);
|
|
return;
|
|
case 0x20 ... 0x2f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_COLLECT);
|
|
return;
|
|
case 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x30 ... 0x7e:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_ESC_DISPATCH);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_NONE, ACTION_COLLECT);
|
|
return;
|
|
case STATE_CSI_ENTRY:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_EXECUTE);
|
|
return;
|
|
case 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x20 ... 0x2f:
|
|
do_trans(vte, raw, STATE_CSI_INT, ACTION_COLLECT);
|
|
return;
|
|
case 0x3a:
|
|
do_trans(vte, raw, STATE_CSI_IGNORE, ACTION_NONE);
|
|
return;
|
|
case 0x30 ... 0x39:
|
|
case 0x3b:
|
|
do_trans(vte, raw, STATE_CSI_PARAM, ACTION_PARAM);
|
|
return;
|
|
case 0x3c ... 0x3f:
|
|
do_trans(vte, raw, STATE_CSI_PARAM, ACTION_COLLECT);
|
|
return;
|
|
case 0x40 ... 0x7e:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_CSI_DISPATCH);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_CSI_IGNORE, ACTION_NONE);
|
|
return;
|
|
case STATE_CSI_PARAM:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_EXECUTE);
|
|
return;
|
|
case 0x30 ... 0x39:
|
|
case 0x3b:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_PARAM);
|
|
return;
|
|
case 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x3a:
|
|
case 0x3c ... 0x3f:
|
|
do_trans(vte, raw, STATE_CSI_IGNORE, ACTION_NONE);
|
|
return;
|
|
case 0x20 ... 0x2f:
|
|
do_trans(vte, raw, STATE_CSI_INT, ACTION_COLLECT);
|
|
return;
|
|
case 0x40 ... 0x7e:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_CSI_DISPATCH);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_CSI_IGNORE, ACTION_NONE);
|
|
return;
|
|
case STATE_CSI_INT:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_EXECUTE);
|
|
return;
|
|
case 0x20 ... 0x2f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_COLLECT);
|
|
return;
|
|
case 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x30 ... 0x3f:
|
|
do_trans(vte, raw, STATE_CSI_IGNORE, ACTION_NONE);
|
|
return;
|
|
case 0x40 ... 0x7e:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_CSI_DISPATCH);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_CSI_IGNORE, ACTION_NONE);
|
|
return;
|
|
case STATE_CSI_IGNORE:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_EXECUTE);
|
|
return;
|
|
case 0x20 ... 0x3f:
|
|
case 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x40 ... 0x7e:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_NONE);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case STATE_DCS_ENTRY:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
case 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x3a:
|
|
do_trans(vte, raw, STATE_DCS_IGNORE, ACTION_NONE);
|
|
return;
|
|
case 0x20 ... 0x2f:
|
|
do_trans(vte, raw, STATE_DCS_INT, ACTION_COLLECT);
|
|
return;
|
|
case 0x30 ... 0x39:
|
|
case 0x3b:
|
|
do_trans(vte, raw, STATE_DCS_PARAM, ACTION_PARAM);
|
|
return;
|
|
case 0x3c ... 0x3f:
|
|
do_trans(vte, raw, STATE_DCS_PARAM, ACTION_COLLECT);
|
|
return;
|
|
case 0x40 ... 0x7e:
|
|
do_trans(vte, raw, STATE_DCS_PASS, ACTION_NONE);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_DCS_PASS, ACTION_NONE);
|
|
return;
|
|
case STATE_DCS_PARAM:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
case 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x30 ... 0x39:
|
|
case 0x3b:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_PARAM);
|
|
return;
|
|
case 0x3a:
|
|
case 0x3c ... 0x3f:
|
|
do_trans(vte, raw, STATE_DCS_IGNORE, ACTION_NONE);
|
|
return;
|
|
case 0x20 ... 0x2f:
|
|
do_trans(vte, raw, STATE_DCS_INT, ACTION_COLLECT);
|
|
return;
|
|
case 0x40 ... 0x7e:
|
|
do_trans(vte, raw, STATE_DCS_PASS, ACTION_NONE);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_DCS_PASS, ACTION_NONE);
|
|
return;
|
|
case STATE_DCS_INT:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
case 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x20 ... 0x2f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_COLLECT);
|
|
return;
|
|
case 0x30 ... 0x3f:
|
|
do_trans(vte, raw, STATE_DCS_IGNORE, ACTION_NONE);
|
|
return;
|
|
case 0x40 ... 0x7e:
|
|
do_trans(vte, raw, STATE_DCS_PASS, ACTION_NONE);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_DCS_PASS, ACTION_NONE);
|
|
return;
|
|
case STATE_DCS_PASS:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
case 0x20 ... 0x7e:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_DCS_COLLECT);
|
|
return;
|
|
case 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x9c:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_NONE);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_NONE, ACTION_DCS_COLLECT);
|
|
return;
|
|
case STATE_DCS_IGNORE:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
case 0x20 ... 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x9c:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_NONE);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case STATE_OSC_STRING:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x20 ... 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_OSC_COLLECT);
|
|
return;
|
|
case 0x9c:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_NONE);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_NONE, ACTION_OSC_COLLECT);
|
|
return;
|
|
case STATE_ST_IGNORE:
|
|
switch (raw) {
|
|
case 0x00 ... 0x17:
|
|
case 0x19:
|
|
case 0x1c ... 0x1f:
|
|
case 0x20 ... 0x7f:
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
case 0x9c:
|
|
do_trans(vte, raw, STATE_GROUND, ACTION_NONE);
|
|
return;
|
|
}
|
|
do_trans(vte, raw, STATE_NONE, ACTION_IGNORE);
|
|
return;
|
|
}
|
|
|
|
log_warn("vte: unhandled input %u in state %d\n", raw, vte->state);
|
|
}
|
|
|
|
void kmscon_vte_input(struct kmscon_vte *vte, const char *u8, size_t len)
|
|
{
|
|
int state, i;
|
|
uint32_t ucs4;
|
|
|
|
if (!vte || !vte->con)
|
|
return;
|
|
|
|
for (i = 0; i < len; ++i) {
|
|
state = kmscon_utf8_mach_feed(vte->mach, u8[i]);
|
|
if (state == KMSCON_UTF8_ACCEPT ||
|
|
state == KMSCON_UTF8_REJECT) {
|
|
ucs4 = kmscon_utf8_mach_get(vte->mach);
|
|
parse_data(vte, ucs4);
|
|
}
|
|
}
|
|
}
|
|
|
|
int kmscon_vte_handle_keyboard(struct kmscon_vte *vte,
|
|
const struct kmscon_input_event *ev, const char **u8, size_t *len)
|
|
{
|
|
kmscon_symbol_t sym;
|
|
|
|
switch (ev->keysym) {
|
|
case XK_BackSpace:
|
|
*u8 = "\x08";
|
|
*len = 1;
|
|
return KMSCON_VTE_SEND;
|
|
case XK_Tab:
|
|
case XK_KP_Tab:
|
|
*u8 = "\x09";
|
|
*len = 1;
|
|
return KMSCON_VTE_SEND;
|
|
case XK_Linefeed:
|
|
*u8 = "\x0a";
|
|
*len = 1;
|
|
return KMSCON_VTE_SEND;
|
|
case XK_Clear:
|
|
*u8 = "\x0b";
|
|
*len = 1;
|
|
return KMSCON_VTE_SEND;
|
|
case XK_Pause:
|
|
*u8 = "\x13";
|
|
*len = 1;
|
|
return KMSCON_VTE_SEND;
|
|
case XK_Scroll_Lock:
|
|
/* TODO: do we need scroll lock impl.? */
|
|
*u8 = "\x14";
|
|
*len = 1;
|
|
return KMSCON_VTE_SEND;
|
|
case XK_Sys_Req:
|
|
*u8 = "\x15";
|
|
*len = 1;
|
|
return KMSCON_VTE_SEND;
|
|
case XK_Escape:
|
|
*u8 = "\x1b";
|
|
*len = 1;
|
|
return KMSCON_VTE_SEND;
|
|
case XK_Return:
|
|
case XK_KP_Enter:
|
|
/* TODO: im CR/LF mode send \x0d\x0a */
|
|
*u8 = "\x0d";
|
|
*len = 1;
|
|
return KMSCON_VTE_SEND;
|
|
}
|
|
|
|
if (ev->unicode != KMSCON_INPUT_INVALID) {
|
|
kmscon_symbol_free_u8(vte->kbd_sym);
|
|
sym = kmscon_symbol_make(ev->unicode);
|
|
vte->kbd_sym = kmscon_symbol_get_u8(vte->st, sym, len);
|
|
*u8 = vte->kbd_sym;
|
|
return KMSCON_VTE_SEND;
|
|
}
|
|
|
|
return KMSCON_VTE_DROP;
|
|
}
|