/* * \brief Escape-sequence decoder * \author Norman Feske * \date 2011-06-06 */ /* * Copyright (C) 2011-2017 Genode Labs GmbH * * This file is part of the Genode OS framework, which is distributed * under the terms of the GNU Affero General Public License version 3. */ #ifndef _TERMINAL__DECODER_H_ #define _TERMINAL__DECODER_H_ #include <terminal/character_screen.h> #include <terminal/print.h> namespace Terminal { class Decoder; } class Terminal::Decoder { private: /** * Return digit value of character */ static inline int digit(char c) { if (c >= '0' && c <= '9') return c - '0'; return -1; } /** * Return true if character is a digit */ static inline bool is_digit(char c) { return (digit(c) >= 0); } /** * Return true if number starts with the specified digit * * \param digit digit 0..9 to test for */ static inline bool starts_with_digit(int digit, int number) { for (; number > 9; number /= 10); return (number == digit); } /** * Return number with the first digit removed */ static inline int remove_first_digit(int number) { int factor = 1; for (; number/factor > 9; factor *= 10); return number % factor; } enum State { STATE_IDLE, STATE_ESC_CSI, /* read CONTROL SEQUENCE INTRODUCER */ STATE_ESC_ECMA, /* read an ECMA-48 escape sequence */ STATE_ESC_SCS, /* read an Select Character Set sequence */ STATE_ESC_VT100, /* read a VT100 escape sequence */ STATE_ESC_OSC /* skip an Operating System Command */ }; /** * Buffer used for collecting escape sequences */ class Escape_stack { private: Log_buffer _dump_log { }; public: struct Entry { enum { INVALID, NUMBER, CODE }; int type = INVALID; int value = 0; void print(Genode::Output &out, State state) const { if (type == NUMBER) { Genode::print(out, value); } else if (state == STATE_ESC_ECMA) { Ecma(value).print(out); } else { Ascii(value).print(out); } } void print(Genode::Output &out) const { print(out, STATE_ESC_VT100); } }; struct Number_entry : Entry { Number_entry(int number) { type = NUMBER, value = number; } }; struct Code_entry : Entry { Code_entry(int code) { type = CODE, value = code; } }; struct Invalid_entry : Entry { }; private: enum { MAX_ENTRIES = 32 }; Entry _entries[MAX_ENTRIES]; int _index; void _dump(State state) { _dump_log.print("ESC"); for (int i = 0; i < _index; i++) { _dump_log.out_char(' '); _entries[i].print(_dump_log, state); } } public: Escape_stack() : _index(0) { } void reset() { _index = 0; } void discard(State state = STATE_ESC_VT100) { _dump_log.print("unhandled sequence "); _dump(state); _dump_log.flush_warning(); _index = 0; } void push(Entry const &entry) { if (_index == MAX_ENTRIES - 1) { Genode::error("escape stack overflow"); _dump(STATE_ESC_VT100); _dump_log.flush_error(); reset(); return; } _entries[_index++] = entry; } /** * Return number of stack elements */ unsigned num_elem() const { return _index; } /** * Return Nth stack entry * * 'index' is relative to the bottom of the stack. */ Entry operator [] (int index) const { return (index <= _index) ? _entries[index] : Invalid_entry(); } } _escape_stack { }; Character_screen &_screen; State _state = STATE_IDLE; int _number = -1; /* current number argument supplied in escape sequence */ void _append_to_number(char c) { _number = (_number < 0 ? 0 : _number)*10 + digit(c); } void _enter_state_idle() { _state = STATE_IDLE; _escape_stack.reset(); _number = -1; } void _enter_state_esc_csi() { _state = STATE_ESC_CSI; _escape_stack.reset(); } void _enter_state_esc_ecma() { _state = STATE_ESC_ECMA; } void _enter_state_esc_vt100() { _state = STATE_ESC_VT100; } void _enter_state_esc_osc() { _state = STATE_ESC_OSC; } bool _sgr(int const p) { if (p < 30) return (_screen.sgr(p), true); /* p starting with digit '3' -> set foreground color */ if (starts_with_digit(3, p)) return (_screen.setaf(remove_first_digit(p)), true); /* p starting with digit '4' -> set background color */ if (starts_with_digit(4, p)) return (_screen.setab(remove_first_digit(p)), true); return false; } /** * Try to handle single-element escape sequence * * \return true if escape sequence was handled */ bool _handle_esc_seq_1() { switch (_escape_stack[0].value) { case 'H': return (_screen.hts(), true); case 'c': return true; /* prefixes 'rs2' */ case 'E': return (_screen.nel(), true); case '>': return true; /* follows 'rmkx' */ case '=': return true; /* follows 'smkx' */ default: return false; } } bool _handle_esc_seq_2() { switch (_escape_stack[0].value) { case '[': switch (_escape_stack[1].value) { case 'A': return (_screen.cuu(), true); case 'B': return (_screen.cud(), true); case 'C': return (_screen.cuf(), true); case 'D': return (_screen.cub(), true); case 'G': return (_screen.cha(), true); case 'H': return (_screen.cup(1,1), true); case 'J': return (_screen.ed(), true); case 'K': return (_screen.el(), true); case 'L': return (_screen.il(), true); case 'M': return (_screen.dl(), true); case 'P': return (_screen.dch(), true); case 'm': return _sgr(0); case 'S': return (_screen.su(), true); case 'T': return (_screen.sd(), true); case 'c': return (_screen.da(), true); case 'd': return (_screen.vpa(), true); case 'n': return (_screen.vpb(), true); case '@': return (_screen.ich(), true); default: return false; } break; default: return false; } } bool _handle_esc_seq_3() { /* * All three-element sequences have the form \E[<NUMBER><COMMAND> */ if ((_escape_stack[0].value != '[') || (_escape_stack[1].type != Escape_stack::Entry::NUMBER)) return false; int const p1 = _escape_stack[1].value; char const command = _escape_stack[2].value; switch (command) { case 'A': return (_screen.cuu(p1), true); case 'B': return (_screen.cud(p1), true); case 'C': return (_screen.cuf(p1), true); case 'D': return (_screen.cub(p1), true); case 'd': return (_screen.vpa(p1), true); case 'g': return (p1 == 3) && (_screen.tbc(), true); case 'G': return (_screen.cha(p1), true); case 'h': return (_screen.decsm(p1), true); case 'l': return (_screen.decrm(p1), true); case 'J': return (_screen.ed(p1), true); case 'K': return (_screen.el(p1), true); case 'L': return (_screen.il(p1), true); case 'M': return (_screen.dl(p1), true); case 'm': return _sgr(p1); case 'n': return (_screen.vpb(p1), true); case 'P': return (_screen.dch(p1), true); case '@': return (_screen.ich(p1), true); case 'S': return (_screen.su(p1), true); case 'T': return (_screen.sd(p1), true); case 'X': return (_screen.ech(p1), true); default: break; } return false; } bool _handle_esc_seq_4() { /* * All four-element escape sequences have the form * \E[?<NUMBER><COMMAND> */ if ((_escape_stack[0].value != '[') || (_escape_stack[1].value != '?') || (_escape_stack[2].type != Escape_stack::Entry::NUMBER)) return false; int const p1 = _escape_stack[2].value; char const command = _escape_stack[3].value; switch (command) { case 'h': return (_screen.decsm(p1), true); case 'l': return (_screen.decrm(p1), true); default: break; } return false; } bool _handle_esc_seq_5() { /* * All five-element escape sequences have the form * \E[<NUMBER1>;<NUMBER2><COMMAND> */ if ((_escape_stack[0].value != '[') || (_escape_stack[1].type != Escape_stack::Entry::NUMBER) || (_escape_stack[2].value != ';') || (_escape_stack[3].type != Escape_stack::Entry::NUMBER)) return false; int const p[2] = { _escape_stack[1].value, _escape_stack[3].value }; int const command = _escape_stack[4].value; switch (command) { case 'r': return (_screen.csr(p[0], p[1]), true); case 'H': return (_screen.cup(p[0], p[1]), true); case 'm': if (p[0] == 39 && p[1] == 49) return (_screen.op(), true); for (int i = 0; i < 2; i++) if (!_sgr(p[i])) Genode::warning("Number ", p[i], " in sequence '[", p[0], ";", p[1], "m' is not implemented"); return true; default: return false; } } bool _handle_esc_seq_6() { /* * All five-element escape sequences have the form * \E[?<NUMBER1>;<NUMBER2><COMMAND> */ if ((_escape_stack[0].value != '[') || (_escape_stack[1].value != '?') || (_escape_stack[2].type != Escape_stack::Entry::NUMBER) || (_escape_stack[3].value != ';') || (_escape_stack[4].type != Escape_stack::Entry::NUMBER)) return false; int const p[2] = { _escape_stack[2].value, _escape_stack[4].value }; switch (_escape_stack[5].value) { case 'h': return (_screen.decsm(p[0], p[1]), true); case 'l': return (_screen.decrm(p[0], p[1]), true); default: return false; } } bool _handle_esc_seq_7() { /* * All six-element escape sequences have the form * \E[<NUMBER1>;<NUMBER2>;<NUMBER3><COMMAND> */ if ((_escape_stack[0].value != '[') || (_escape_stack[1].type != Escape_stack::Entry::NUMBER) || (_escape_stack[2].value != ';') || (_escape_stack[3].type != Escape_stack::Entry::NUMBER) || (_escape_stack[4].value != ';') || (_escape_stack[5].type != Escape_stack::Entry::NUMBER)) return false; int const p[3] = { _escape_stack[1].value, _escape_stack[3].value, _escape_stack[5].value }; int const command = _escape_stack[6].value; switch (command) { case 'm': for (int i = 0; i < 3; i++) if (!_sgr(p[i])) Genode::warning("Number ", p[i], " in sequence '[", p[0], ";", p[1], ";", p[2], "m' is not implemented"); return true; default: return false; } return true; } bool _complete() { return (((_escape_stack.num_elem() == 1) && _handle_esc_seq_1()) || ((_escape_stack.num_elem() == 2) && _handle_esc_seq_2()) || ((_escape_stack.num_elem() == 3) && _handle_esc_seq_3()) || ((_escape_stack.num_elem() == 4) && _handle_esc_seq_4()) || ((_escape_stack.num_elem() == 5) && _handle_esc_seq_5()) || ((_escape_stack.num_elem() == 6) && _handle_esc_seq_6()) || ((_escape_stack.num_elem() == 7) && _handle_esc_seq_7())); } public: Decoder(Character_screen &screen) : _screen(screen) { } void insert(unsigned char c) { switch (_state) { case STATE_IDLE: enum { ESC_PREFIX = 0x1b }; if (c == ESC_PREFIX) { _enter_state_esc_csi(); break; } /* handle special characters */ /* handle normal characters */ _screen.output(c); break; case STATE_ESC_CSI: /* check that the second byte is in set C1 - ECMA-48 5.3 */ switch (c) { case '7': _screen.decsc(); _enter_state_idle(); break; case '8': _screen.decrc(); _enter_state_idle(); break; case '(': case ')': _escape_stack.push(Escape_stack::Code_entry(c)); _state = STATE_ESC_SCS; break; case ']': _enter_state_esc_osc(); break; case 'M': _screen.reverse_index(); _enter_state_idle(); break; case '=': case '>': /* keypad mode, not useful enough to handle */ _enter_state_idle(); break; default: if (0x40 <= c && c <= 0x5f) { _escape_stack.push(Escape_stack::Code_entry(c)); _enter_state_esc_ecma(); break; } Genode::error("unknown CSI ESC", Ascii(c)); _enter_state_idle(); } break; case STATE_ESC_ECMA: case STATE_ESC_VT100: /* * We received the prefix character of an escape sequence, * collect the escape-sequence elements until we detect the * completion of the sequence. */ /* check for start of a number argument */ if (is_digit(c)) { _append_to_number(c); } else /* non-number character of escape sequence */ { if (-1 < _number) { _escape_stack.push(Escape_stack::Number_entry(_number)); _number = -1; } _escape_stack.push(Escape_stack::Code_entry(c)); /* check for Final Byte - ECMA-48 5.4 */ if (_state == STATE_ESC_ECMA && c > 0x3f && c < 0x7f) { if (!_complete()) { _escape_stack.discard(_state); } _enter_state_idle(); } else { if (_complete()) _enter_state_idle(); } } break; case STATE_ESC_SCS: switch (_escape_stack[0].value) { case '(': _screen.scs_g0(c); break; case ')': _screen.scs_g1(c); break; } _enter_state_idle(); break; case STATE_ESC_OSC: enum { BELL = 07 }; _escape_stack.push(Escape_stack::Code_entry(c)); if (c == BELL) { _escape_stack.discard(_state); _enter_state_idle(); } break; } }; }; #endif /* _TERMINAL__DECODER_H_ */