/* #line 1 "user/ansi_parser.rl" */ #include #include "screen.h" #include "ansi_parser.h" // Max nr of CSI parameters #define CSI_N_MAX 3 /** * \brief Handle fully received CSI ANSI sequence * \param leadchar - private range leading character, 0 if none * \param params - array of CSI_N_MAX ints holding the numeric arguments * \param keychar - the char terminating the sequence */ void ICACHE_FLASH_ATTR handle_CSI(char leadchar, int *params, char keychar) { /* Implemented codes (from Wikipedia) CSI n A CUU – Cursor Up CSI n B CUD – Cursor Down CSI n C CUF – Cursor Forward CSI n D CUB – Cursor Back CSI n E CNL – Cursor Next Line CSI n F CPL – Cursor Previous Line CSI n G CHA – Cursor Horizontal Absolute CSI n ; m H CUP – Cursor Position CSI n J ED – Erase Display CSI n K EL – Erase in Line CSI n S SU – Scroll Up CSI n T SD – Scroll Down CSI n ; m f HVP – Horizontal and Vertical Position CSI n m SGR – Select Graphic Rendition (Implemented only some) CSI 6n DSR – Device Status Report NOT IMPL CSI s SCP – Save Cursor Position CSI u RCP – Restore Cursor Position CSI ?25l DECTCEM Hides the cursor CSI ?25h DECTCEM Shows the cursor */ int n1 = params[0]; int n2 = params[1]; // int n3 = params[2]; // defaults switch (keychar) { case 'A': case 'B': case 'C': case 'D': case 'E': case 'F': case 'G': case 'S': case 'T': if (n1 == 0) n1 = 1; break; case 'H': case 'f': if (n1 == 0) n1 = 1; if (n2 == 0) n2 = 1; break; case 'J': case 'K': if (n1 > 2) n1 = 0; break; } switch (keychar) { // CUU CUD CUF CUB case 'A': screen_cursor_move(0, -n1); break; case 'B': screen_cursor_move(0, n1); break; case 'C': screen_cursor_move(n1, 0); break; case 'D': screen_cursor_move(-n1, 0); break; case 'E': // CNL screen_cursor_move(0, n1); screen_cursor_set_x(0); break; case 'F': // CPL screen_cursor_move(0, -n1); screen_cursor_set_x(0); break; // CHA case 'G': screen_cursor_set_x(n1 - 1); break; // 1-based // SU, SD case 'S': screen_scroll_up(n1); break; case 'T': screen_scroll_down(n1); break; // CUP,HVP case 'H': case 'f': screen_cursor_set(n2-1, n1-1); break; // 1-based case 'J': // ED if (n1 == 0) { screen_clear(CLEAR_TO_CURSOR); } else if (n1 == 1) { screen_clear(CLEAR_FROM_CURSOR); } else { screen_clear(CLEAR_ALL); screen_cursor_set(0, 0); } break; case 'K': // EL if (n1 == 0) { screen_clear_line(CLEAR_TO_CURSOR); } else if (n1 == 1) { screen_clear_line(CLEAR_FROM_CURSOR); } else { screen_clear_line(CLEAR_ALL); screen_cursor_set_x(0); } break; // SCP, RCP case 's': screen_cursor_save(); break; case 'u': screen_cursor_restore(); break; // DECTCEM cursor show hide case 'l': if (leadchar == '?' && n1 == 25) { screen_cursor_enable(1); } break; case 'h': if (leadchar == '?' && n1 == 25) { screen_cursor_enable(0); } break; case 'm': // SGR // iterate arguments for (int i = 0; i < CSI_N_MAX; i++) { int n = params[i]; if (i == 0 && n == 0) { // reset SGR screen_set_fg(7); screen_set_bg(0); break; // cannot combine reset with others } else if (n >= 30 && n <= 37) screen_set_fg(n-30); // ANSI normal fg else if (n >= 40 && n <= 47) screen_set_bg(n-40); // ANSI normal bg else if (n == 39) screen_set_fg(7); // default fg else if (n == 49) screen_set_bg(0); // default bg else if (n == 7) screen_inverse(1); // inverse else if (n == 27) screen_inverse(0); // positive else if (n == 1) screen_set_bright_fg(); // ANSI bold = bright fg else if (n >= 90 && n <= 97) screen_set_fg(n-90+8); // AIX bright fg else if (n >= 100 && n <= 107) screen_set_bg(n-100+8); // AIX bright bg } break; } } /** * \brief Handle a request to reset the display device */ void ICACHE_FLASH_ATTR handle_RESET_cmd(void) { screen_reset(); } /** * \brief Handle a received plain character * \param c - the character */ void ICACHE_FLASH_ATTR handle_plainchar(char c) { screen_putchar(c); } void ICACHE_FLASH_ATTR handle_OSC_FactoryReset(void) { info("OSC: Factory reset"); warn("NOT IMPLEMENTED"); // TODO } void ICACHE_FLASH_ATTR handle_OSC_SetScreenSize(int rows, int cols) { info("OSC: Set screen size to %d x %d", rows, cols); screen_resize(rows, cols); } /* Ragel constants block */ /* #line 206 "user/ansi_parser.c" */ static const char _ansi_actions[] = { 0, 1, 0, 1, 1, 1, 2, 1, 3, 1, 4, 1, 5, 1, 6, 1, 7, 1, 8, 1, 9, 1, 10 }; static const char _ansi_eof_actions[] = { 0, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 13, 0, 0, 0 }; static const int ansi_start = 1; static const int ansi_first_final = 14; static const int ansi_error = 0; static const int ansi_en_CSI_body = 3; static const int ansi_en_OSC_body = 5; static const int ansi_en_main = 1; /* #line 205 "user/ansi_parser.rl" */ /** * \brief Linear ANSI chars stream parser * * Parses a stream of bytes using a Ragel parser. The defined * grammar does not use 'unget', so the entire buffer is * always processed in a linear manner. * * \attention -> but always check the Ragel output for 'p--' * or 'p -=', that means trouble. * * \param newdata - array of new chars to process * \param len - length of the newdata buffer */ void ICACHE_FLASH_ATTR ansi_parser(const char *newdata, size_t len) { static int cs = -1; // The CSI code is built here static char csi_leading; //!< Leading char, 0 if none static int csi_ni; //!< Number of the active digit static int csi_n[CSI_N_MAX]; //!< Param digits static char csi_char; //!< CSI action char (end) if (len == 0) len = strlen(newdata); // Load new data to Ragel vars const char *p = newdata; const char *eof = NULL; const char *pe = newdata + len; // Init Ragel on the first run if (cs == -1) { /* #line 265 "user/ansi_parser.c" */ { cs = ansi_start; } /* #line 241 "user/ansi_parser.rl" */ } // The parser /* #line 275 "user/ansi_parser.c" */ { const char *_acts; unsigned int _nacts; if ( p == pe ) goto _test_eof; if ( cs == 0 ) goto _out; _resume: switch ( cs ) { case 1: if ( (*p) == 27 ) goto tr1; goto tr0; case 2: switch( (*p) ) { case 91: goto tr3; case 93: goto tr4; case 99: goto tr5; } goto tr2; case 0: goto _out; case 14: if ( (*p) == 27 ) goto tr1; goto tr0; case 3: if ( (*p) == 59 ) goto tr8; if ( (*p) < 60 ) { if ( (*p) > 47 ) { if ( 48 <= (*p) && (*p) <= 57 ) goto tr7; } else if ( (*p) >= 32 ) goto tr6; } else if ( (*p) > 64 ) { if ( (*p) > 90 ) { if ( 97 <= (*p) && (*p) <= 122 ) goto tr9; } else if ( (*p) >= 65 ) goto tr9; } else goto tr6; goto tr2; case 4: if ( (*p) == 59 ) goto tr8; if ( (*p) < 65 ) { if ( 48 <= (*p) && (*p) <= 57 ) goto tr7; } else if ( (*p) > 90 ) { if ( 97 <= (*p) && (*p) <= 122 ) goto tr9; } else goto tr9; goto tr2; case 15: goto tr2; case 5: switch( (*p) ) { case 70: goto tr10; case 87: goto tr11; } goto tr2; case 6: if ( (*p) == 82 ) goto tr12; goto tr2; case 7: switch( (*p) ) { case 7: goto tr13; case 27: goto tr14; } goto tr2; case 16: goto tr2; case 8: if ( (*p) == 92 ) goto tr13; goto tr2; case 9: if ( 48 <= (*p) && (*p) <= 57 ) goto tr15; goto tr2; case 10: if ( (*p) == 59 ) goto tr16; if ( 48 <= (*p) && (*p) <= 57 ) goto tr15; goto tr2; case 11: if ( 48 <= (*p) && (*p) <= 57 ) goto tr17; goto tr2; case 12: switch( (*p) ) { case 7: goto tr18; case 27: goto tr19; } if ( 48 <= (*p) && (*p) <= 57 ) goto tr17; goto tr2; case 13: if ( (*p) == 92 ) goto tr18; goto tr2; } tr2: cs = 0; goto f0; tr0: cs = 1; goto f1; tr1: cs = 2; goto _again; tr6: cs = 4; goto f5; tr7: cs = 4; goto f6; tr8: cs = 4; goto f7; tr10: cs = 6; goto _again; tr12: cs = 7; goto _again; tr14: cs = 8; goto _again; tr11: cs = 9; goto _again; tr15: cs = 10; goto f6; tr16: cs = 11; goto f7; tr17: cs = 12; goto f6; tr19: cs = 13; goto _again; tr3: cs = 14; goto f2; tr4: cs = 14; goto f3; tr5: cs = 14; goto f4; tr9: cs = 15; goto f8; tr13: cs = 16; goto f9; tr18: cs = 16; goto f10; f1: _acts = _ansi_actions + 1; goto execFuncs; f2: _acts = _ansi_actions + 3; goto execFuncs; f5: _acts = _ansi_actions + 5; goto execFuncs; f6: _acts = _ansi_actions + 7; goto execFuncs; f7: _acts = _ansi_actions + 9; goto execFuncs; f8: _acts = _ansi_actions + 11; goto execFuncs; f0: _acts = _ansi_actions + 13; goto execFuncs; f3: _acts = _ansi_actions + 15; goto execFuncs; f9: _acts = _ansi_actions + 17; goto execFuncs; f10: _acts = _ansi_actions + 19; goto execFuncs; f4: _acts = _ansi_actions + 21; goto execFuncs; execFuncs: _nacts = *_acts++; while ( _nacts-- > 0 ) { switch ( *_acts++ ) { case 0: /* #line 251 "user/ansi_parser.rl" */ { handle_plainchar((*p)); } break; case 1: /* #line 258 "user/ansi_parser.rl" */ { /* Reset the CSI builder */ csi_leading = csi_char = 0; csi_ni = 0; /* Zero out digits */ for(int i = 0; i < CSI_N_MAX; i++) { csi_n[i] = 0; } {cs = 3; goto _again;} } break; case 2: /* #line 271 "user/ansi_parser.rl" */ { csi_leading = (*p); } break; case 3: /* #line 275 "user/ansi_parser.rl" */ { /* x10 + digit */ if (csi_ni < CSI_N_MAX) { csi_n[csi_ni] = csi_n[csi_ni]*10 + ((*p) - '0'); } } break; case 4: /* #line 282 "user/ansi_parser.rl" */ { csi_ni++; } break; case 5: /* #line 286 "user/ansi_parser.rl" */ { csi_char = (*p); handle_CSI(csi_leading, csi_n, csi_char); {cs = 1; goto _again;} } break; case 6: /* #line 294 "user/ansi_parser.rl" */ { warn("Invalid escape sequence discarded."); {cs = 1; goto _again;} } break; case 7: /* #line 311 "user/ansi_parser.rl" */ { csi_ni = 0; // we reuse the CSI numeric buffer for(int i = 0; i < CSI_N_MAX; i++) { csi_n[i] = 0; } {cs = 5; goto _again;} } break; case 8: /* #line 322 "user/ansi_parser.rl" */ { handle_OSC_FactoryReset(); {cs = 1; goto _again;} } break; case 9: /* #line 327 "user/ansi_parser.rl" */ { handle_OSC_SetScreenSize(csi_n[0], csi_n[1]); {cs = 1; goto _again;} } break; case 10: /* #line 338 "user/ansi_parser.rl" */ { // Reset screen handle_RESET_cmd(); {cs = 1; goto _again;} } break; /* #line 517 "user/ansi_parser.c" */ } } goto _again; _again: if ( cs == 0 ) goto _out; if ( ++p != pe ) goto _resume; _test_eof: {} if ( p == eof ) { const char *__acts = _ansi_actions + _ansi_eof_actions[cs]; unsigned int __nacts = (unsigned int) *__acts++; while ( __nacts-- > 0 ) { switch ( *__acts++ ) { case 6: /* #line 294 "user/ansi_parser.rl" */ { warn("Invalid escape sequence discarded."); {cs = 1; goto _again;} } break; /* #line 541 "user/ansi_parser.c" */ } } } _out: {} } /* #line 356 "user/ansi_parser.rl" */ }