aboutsummaryrefslogtreecommitdiffstats
path: root/vt100.l
diff options
context:
space:
mode:
Diffstat (limited to 'vt100.l')
-rw-r--r--vt100.l487
1 files changed, 487 insertions, 0 deletions
diff --git a/vt100.l b/vt100.l
new file mode 100644
index 0000000..8e6ed5e
--- /dev/null
+++ b/vt100.l
@@ -0,0 +1,487 @@
+%{
+#include <string.h>
+
+#include "runes.h"
+
+#define RUNES_VT100_CSI_MAX_PARAMS 256
+%}
+
+%option reentrant noyywrap nodefault batch
+%option prefix="runes_vt100_yy"
+%option extra-type="RunesTerm *"
+
+CTRL [\000-\037\177]
+TEXT [^\000-\037\177]
+
+ST \007
+BEL \007
+BS \010
+TAB \011
+LF \012
+VT \013
+FF \014
+CR \015
+ESC \033
+
+CSI {ESC}\[
+OSC {ESC}\]
+RI {ESC}M
+
+DECCSI {CSI}\?
+CSIPARAM1 ([0-9]+)?
+CSIPARAM2 ([0-9]+(;[0-9]+)?)?
+CSIPARAMS ([0-9]+(;[0-9]+)*)?
+SMPARAMS ([<=?]?[0-9]+(;[<=?]?[0-9]+)*)?
+
+CUU {CSI}{CSIPARAM1}A
+CUD {CSI}{CSIPARAM1}B
+CUF {CSI}{CSIPARAM1}C
+CUB {CSI}{CSIPARAM1}D
+CUP {CSI}{CSIPARAM2}H
+ED {CSI}{CSIPARAM1}J
+EL {CSI}{CSIPARAM1}K
+SM {CSI}{SMPARAMS}h
+RM {CSI}{SMPARAMS}l
+SGR {CSI}{CSIPARAMS}m
+
+DECSED {DECCSI}{CSIPARAM1}J
+DECSEL {DECCSI}{CSIPARAM1}K
+
+OSC0 {OSC}0;{TEXT}*{ST}
+OSC1 {OSC}1;{TEXT}*{ST}
+OSC2 {OSC}2;{TEXT}*{ST}
+
+%{
+static void runes_vt100_handle_text(RunesTerm *t, char *text, size_t len);
+static void runes_vt100_handle_bel(RunesTerm *t);
+static void runes_vt100_handle_bs(RunesTerm *t);
+static void runes_vt100_handle_tab(RunesTerm *t);
+static void runes_vt100_handle_lf(RunesTerm *t);
+static void runes_vt100_handle_cr(RunesTerm *t);
+static void runes_vt100_handle_ri(RunesTerm *t);
+static void runes_vt100_extract_csi_params(
+ char *buf, size_t len, int *params, int *nparams);
+static void runes_vt100_extract_sm_params(
+ char *buf, size_t len, char *modes, int *params, int *nparams);
+static void runes_vt100_handle_cuu(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_cud(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_cuf(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_cub(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_cup(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_ed(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_el(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_sm(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_rm(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_sgr(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_decsed(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_decsel(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_osc0(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_osc1(RunesTerm *t, char *buf, size_t len);
+static void runes_vt100_handle_osc2(RunesTerm *t, char *buf, size_t len);
+%}
+
+%%
+
+{TEXT}+ runes_vt100_handle_text(yyextra, yytext, yyleng);
+
+{BEL} runes_vt100_handle_bel(yyextra);
+{BS} runes_vt100_handle_bs(yyextra);
+{TAB} runes_vt100_handle_tab(yyextra);
+{LF} |
+{VT} |
+{FF} runes_vt100_handle_lf(yyextra);
+{CR} runes_vt100_handle_cr(yyextra);
+
+{RI} runes_vt100_handle_ri(yyextra);
+
+{CUU} runes_vt100_handle_cuu(yyextra, yytext, yyleng);
+{CUD} runes_vt100_handle_cud(yyextra, yytext, yyleng);
+{CUF} runes_vt100_handle_cuf(yyextra, yytext, yyleng);
+{CUB} runes_vt100_handle_cub(yyextra, yytext, yyleng);
+{CUP} runes_vt100_handle_cup(yyextra, yytext, yyleng);
+{ED} runes_vt100_handle_ed(yyextra, yytext, yyleng);
+{EL} runes_vt100_handle_el(yyextra, yytext, yyleng);
+{SM} runes_vt100_handle_sm(yyextra, yytext, yyleng);
+{RM} runes_vt100_handle_rm(yyextra, yytext, yyleng);
+{SGR} runes_vt100_handle_sgr(yyextra, yytext, yyleng);
+
+{DECSED} runes_vt100_handle_decsed(yyextra, yytext, yyleng);
+{DECSEL} runes_vt100_handle_decsel(yyextra, yytext, yyleng);
+
+{OSC0} runes_vt100_handle_osc0(yyextra, yytext, yyleng);
+{OSC1} runes_vt100_handle_osc1(yyextra, yytext, yyleng);
+{OSC2} runes_vt100_handle_osc2(yyextra, yytext, yyleng);
+
+{CSI}\??{CSIPARAMS}(?s:.) {
+ fprintf(
+ stderr, "unhandled CSI sequence: \\033%*s\n",
+ (int)yyleng - 1, yytext + 1);
+}
+
+{OSC}(?s:.) {
+ fprintf(
+ stderr, "unhandled OSC sequence: \\033%*s\n",
+ (int)yyleng - 1, yytext + 1);
+}
+
+{ESC} {
+ fprintf(stderr, "unhandled escape sequence\n");
+}
+
+{CTRL} {
+ fprintf(stderr, "unhandled control character: \\%hho\n", yytext[0]);
+}
+
+%%
+
+void runes_vt100_process_string(RunesTerm *t, char *buf, size_t len)
+{
+ YY_BUFFER_STATE state;
+ yyscan_t scanner;
+
+ yylex_init_extra(t, &scanner);
+ state = runes_vt100_yy_scan_bytes(buf, len, scanner);
+ runes_vt100_yylex(scanner);
+ runes_vt100_yy_delete_buffer(state, scanner);
+ yylex_destroy(scanner);
+}
+
+static void runes_vt100_handle_text(RunesTerm *t, char *text, size_t len)
+{
+ runes_display_show_string(t, text, len);
+}
+
+static void runes_vt100_handle_bel(RunesTerm *t)
+{
+ /* XXX */
+}
+
+static void runes_vt100_handle_bs(RunesTerm *t)
+{
+ runes_display_backspace(t);
+}
+
+static void runes_vt100_handle_tab(RunesTerm *t)
+{
+ runes_display_move_to(t, t->row, t->col - (t->col % 8) + 8);
+}
+
+static void runes_vt100_handle_lf(RunesTerm *t)
+{
+ runes_display_move_to(t, t->row + 1, t->col);
+}
+
+static void runes_vt100_handle_cr(RunesTerm *t)
+{
+ runes_display_move_to(t, t->row, 0);
+}
+
+static void runes_vt100_handle_ri(RunesTerm *t)
+{
+ runes_display_move_to(t, t->row - 1, t->col);
+}
+
+static void runes_vt100_extract_csi_params(
+ char *buf, size_t len, int *params, int *nparams)
+{
+ runes_vt100_extract_sm_params(buf, len, NULL, params, nparams);
+}
+
+static void runes_vt100_extract_sm_params(
+ char *buf, size_t len, char *modes, int *params, int *nparams)
+{
+ char tmp = buf[len];
+ char *pos = buf;
+
+ /* this assumes that it will only ever be called on a fully matched CSI
+ * sequence: accessing one character beyond the end is safe because CSI
+ * sequences always have one character after the parameters (to determine
+ * the type of sequence), and the parameters can only ever be digits,
+ * separated by semicolons. */
+ buf[len] = '\0';
+ *nparams = 0;
+ for (;;) {
+ if (*nparams >= RUNES_VT100_CSI_MAX_PARAMS) {
+ fprintf(stderr, "max CSI parameter length exceeded\n");
+ break;
+ }
+
+ if (modes && (size_t)(pos - buf) < len) {
+ if (strstr(pos, "0123456789")) {
+ modes[*nparams] = '\0';
+ }
+ else {
+ modes[*nparams] = *pos++;
+ }
+ }
+
+ params[(*nparams)++] = atoi(pos);
+
+ pos = strchr(pos, ';');
+ if (pos) {
+ pos++;
+ }
+ else {
+ break;
+ }
+ }
+}
+
+static void runes_vt100_handle_cuu(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS] = { 1 }, nparams;
+
+ runes_vt100_extract_csi_params(buf + 2, len - 3, params, &nparams);
+ runes_display_move_to(t, t->row - params[0], t->col);
+}
+
+static void runes_vt100_handle_cud(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS] = { 1 }, nparams;
+
+ runes_vt100_extract_csi_params(buf + 2, len - 3, params, &nparams);
+ runes_display_move_to(t, t->row + params[0], t->col);
+}
+
+static void runes_vt100_handle_cuf(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS] = { 1 }, nparams;
+
+ runes_vt100_extract_csi_params(buf + 2, len - 3, params, &nparams);
+ runes_display_move_to(t, t->row, t->col + params[0]);
+}
+
+static void runes_vt100_handle_cub(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS] = { 1 }, nparams;
+
+ runes_vt100_extract_csi_params(buf + 2, len - 3, params, &nparams);
+ runes_display_move_to(t, t->row, t->col - params[0]);
+}
+
+static void runes_vt100_handle_cup(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS] = { 1, 1 }, nparams;
+
+ runes_vt100_extract_csi_params(buf + 2, len - 3, params, &nparams);
+ runes_display_move_to(t, params[0] - 1, params[1] - 1);
+}
+
+static void runes_vt100_handle_ed(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS] = { 0 }, nparams;
+
+ runes_vt100_extract_csi_params(buf + 2, len - 3, params, &nparams);
+ switch (params[0]) {
+ case 0:
+ runes_display_clear_screen_forward(t);
+ break;
+ case 1:
+ /* XXX */
+ fprintf(stderr, "unhandled ED parameter 1\n");
+ break;
+ case 2:
+ runes_display_clear_screen(t);
+ break;
+ default:
+ fprintf(stderr, "unknown ED parameter %d\n", params[0]);
+ break;
+ }
+}
+
+static void runes_vt100_handle_el(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS] = { 0 }, nparams;
+
+ runes_vt100_extract_csi_params(buf + 2, len - 3, params, &nparams);
+ switch (params[0]) {
+ case 0:
+ runes_display_kill_line_forward(t);
+ break;
+ case 1:
+ /* XXX */
+ fprintf(stderr, "unhandled EL parameter 1\n");
+ break;
+ case 2:
+ /* XXX */
+ fprintf(stderr, "unhandled EL parameter 2\n");
+ break;
+ default:
+ fprintf(stderr, "unknown EL parameter %d\n", params[0]);
+ break;
+ }
+}
+
+static void runes_vt100_handle_sm(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS], nparams, i;
+ char modes[RUNES_VT100_CSI_MAX_PARAMS] = { 0 };
+
+ runes_vt100_extract_sm_params(buf + 2, len - 3, modes, params, &nparams);
+ for (i = 0; i < nparams; ++i) {
+ switch (modes[i]) {
+ case '?':
+ switch (params[i]) {
+ case 25:
+ runes_display_show_cursor(t);
+ break;
+ default:
+ fprintf(
+ stderr, "unknown SM parameter: %c%d\n",
+ modes[i], params[i]);
+ break;
+ }
+ break;
+ default:
+ fprintf(
+ stderr, "unknown SM parameter: %c%d\n",
+ modes[i], params[i]);
+ break;
+ }
+ }
+}
+
+static void runes_vt100_handle_rm(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS], nparams, i;
+ char modes[RUNES_VT100_CSI_MAX_PARAMS] = { 0 };
+
+ runes_vt100_extract_sm_params(buf + 2, len - 3, modes, params, &nparams);
+ for (i = 0; i < nparams; ++i) {
+ switch (modes[i]) {
+ case '?':
+ switch (params[i]) {
+ case 25:
+ runes_display_hide_cursor(t);
+ break;
+ default:
+ fprintf(
+ stderr, "unknown RM parameter: %c%d\n",
+ modes[i], params[i]);
+ break;
+ }
+ break;
+ default:
+ fprintf(
+ stderr, "unknown RM parameter: %c%d\n",
+ modes[i], params[i]);
+ break;
+ }
+ }
+}
+
+static void runes_vt100_handle_sgr(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS] = { 0 }, nparams, i;
+
+ runes_vt100_extract_csi_params(buf + 2, len - 3, params, &nparams);
+ if (nparams < 1) {
+ nparams = 1;
+ }
+ for (i = 0; i < nparams; ++i) {
+ switch (params[i]) {
+ case 0:
+ runes_display_reset_text_attributes(t);
+ break;
+ case 1:
+ runes_display_set_bold(t);
+ break;
+ case 3:
+ runes_display_set_italic(t);
+ break;
+ case 4:
+ runes_display_set_underline(t);
+ break;
+ case 22:
+ runes_display_reset_bold(t);
+ break;
+ case 23:
+ runes_display_reset_italic(t);
+ break;
+ case 24:
+ runes_display_reset_underline(t);
+ break;
+ case 30: case 31: case 32: case 33:
+ case 34: case 35: case 36: case 37:
+ runes_display_set_fg_color(t, t->colors[params[i] - 30]);
+ break;
+ case 39:
+ runes_display_reset_fg_color(t);
+ break;
+ case 40: case 41: case 42: case 43:
+ case 44: case 45: case 46: case 47:
+ runes_display_set_bg_color(t, t->colors[params[i] - 40]);
+ break;
+ case 49:
+ runes_display_reset_bg_color(t);
+ break;
+ default:
+ fprintf(stderr, "unknown SGR parameter: %d\n", params[i]);
+ break;
+ }
+ }
+}
+
+static void runes_vt100_handle_decsed(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS] = { 0 }, nparams;
+
+ runes_vt100_extract_csi_params(buf + 2, len - 3, params, &nparams);
+ switch (params[0]) {
+ case 0:
+ /* XXX not quite correct */
+ runes_display_clear_screen_forward(t);
+ break;
+ case 1:
+ /* XXX */
+ fprintf(stderr, "unhandled DECSED parameter 1\n");
+ break;
+ case 2:
+ /* XXX not quite correct */
+ runes_display_clear_screen(t);
+ break;
+ default:
+ fprintf(stderr, "unknown DECSED parameter %d\n", params[0]);
+ break;
+ }
+}
+
+static void runes_vt100_handle_decsel(RunesTerm *t, char *buf, size_t len)
+{
+ int params[RUNES_VT100_CSI_MAX_PARAMS] = { 0 }, nparams;
+
+ runes_vt100_extract_csi_params(buf + 2, len - 3, params, &nparams);
+ switch (params[0]) {
+ case 0:
+ /* XXX not quite correct */
+ runes_display_kill_line_forward(t);
+ break;
+ case 1:
+ /* XXX */
+ fprintf(stderr, "unhandled DECSEL parameter 1\n");
+ break;
+ case 2:
+ /* XXX */
+ fprintf(stderr, "unhandled DECSEL parameter 2\n");
+ break;
+ default:
+ fprintf(stderr, "unknown DECSEL parameter %d\n", params[0]);
+ break;
+ }
+}
+
+static void runes_vt100_handle_osc0(RunesTerm *t, char *buf, size_t len)
+{
+ runes_window_backend_set_icon_name(t, buf + 4, len - 5);
+ runes_window_backend_set_window_title(t, buf + 4, len - 5);
+}
+
+static void runes_vt100_handle_osc1(RunesTerm *t, char *buf, size_t len)
+{
+ runes_window_backend_set_icon_name(t, buf + 4, len - 5);
+}
+
+static void runes_vt100_handle_osc2(RunesTerm *t, char *buf, size_t len)
+{
+ runes_window_backend_set_window_title(t, buf + 4, len - 5);
+}