mirror of
https://github.com/DarkFlippers/unleashed-firmware.git
synced 2025-12-12 12:42:30 +04:00
* feat: FuriThread stdin * ci: fix f18 * feat: stdio callback context * feat: FuriPipe * POTENTIALLY EXPLOSIVE pipe welding * fix: non-explosive welding * Revert welding * docs: furi_pipe * feat: pipe event loop integration * update f18 sdk * f18 * docs: make doxygen happy * fix: event loop not triggering when pipe attached to stdio * fix: partial stdout in pipe * allow simultaneous in and out subscription in event loop * feat: vcp i/o * feat: cli ansi stuffs and history * feat: more line editing * working but slow cli rewrite * restore previous speed after 4 days of debugging 🥲 * fix: cli_app_should_stop * fix: cli and event_loop memory leaks * style: remove commented out code * ci: fix pvs warnings * fix: unit tests, event_loop crash * ci: fix build * ci: silence pvs warning * feat: cli gpio * ci: fix formatting * Fix memory leak during event loop unsubscription * Event better memory leak fix * feat: cli completions * Merge remote-tracking branch 'origin/dev' into portasynthinca3/3928-cli-threads * merge fixups * temporarily exclude speaker_debug app * pvs and unit tests fixups * feat: commands in fals * move commands out of flash, code cleanup * ci: fix errors * fix: run commands in buffer when stopping session * speedup cli file transfer * fix f18 * separate cli_shell into modules * fix pvs warning * fix qflipper refusing to connect * remove temp debug logs * remove erroneous conclusion * Fix memory leak during event loop unsubscription * Event better memory leak fix * unit test for the fix * improve thread stdio callback signatures * pipe stdout timeout * update api symbols * fix f18, formatting * fix pvs warnings * increase stack size, hope to fix unit tests * cli: revert flag changes * cli: fix formatting * cli, fbt: loopback perf benchmark * thread, event_loop: subscribing to thread flags * cli: signal internal events using thread flags, improve performance * fix f18, formatting * event_loop: fix crash * storage_cli: increase write_chunk buffer size again * cli: explanation for order=0 * thread, event_loop: thread flags callback refactor * cli: increase stack size * cli: rename cli_app_should_stop -> cli_is_pipe_broken_or_is_etx_next_char * cli: use plain array instead of mlib for history * cli: prepend file name to static fns * cli: fix formatting * cli_shell: increase stack size * cli: fix rpc lockup * cli: better lockup fix * cli: fix f18 * fix merge --------- Co-authored-by: Georgii Surkov <georgii.surkov@outlook.com> Co-authored-by: あく <alleteam@gmail.com>
124 lines
3.7 KiB
C
124 lines
3.7 KiB
C
#include "cli_ansi.h"
|
|
|
|
typedef enum {
|
|
CliAnsiParserStateInitial,
|
|
CliAnsiParserStateEscape,
|
|
CliAnsiParserStateEscapeBrace,
|
|
CliAnsiParserStateEscapeBraceOne,
|
|
CliAnsiParserStateEscapeBraceOneSemicolon,
|
|
CliAnsiParserStateEscapeBraceOneSemicolonModifiers,
|
|
} CliAnsiParserState;
|
|
|
|
struct CliAnsiParser {
|
|
CliAnsiParserState state;
|
|
CliModKey modifiers;
|
|
};
|
|
|
|
CliAnsiParser* cli_ansi_parser_alloc(void) {
|
|
CliAnsiParser* parser = malloc(sizeof(CliAnsiParser));
|
|
return parser;
|
|
}
|
|
|
|
void cli_ansi_parser_free(CliAnsiParser* parser) {
|
|
free(parser);
|
|
}
|
|
|
|
/**
|
|
* @brief Converts a single character representing a special key into the enum
|
|
* representation
|
|
*/
|
|
static CliKey cli_ansi_key_from_mnemonic(char c) {
|
|
switch(c) {
|
|
case 'A':
|
|
return CliKeyUp;
|
|
case 'B':
|
|
return CliKeyDown;
|
|
case 'C':
|
|
return CliKeyRight;
|
|
case 'D':
|
|
return CliKeyLeft;
|
|
case 'F':
|
|
return CliKeyEnd;
|
|
case 'H':
|
|
return CliKeyHome;
|
|
default:
|
|
return CliKeyUnrecognized;
|
|
}
|
|
}
|
|
|
|
#define PARSER_RESET_AND_RETURN(parser, modifiers_val, key_val) \
|
|
do { \
|
|
parser->state = CliAnsiParserStateInitial; \
|
|
return (CliAnsiParserResult){ \
|
|
.is_done = true, \
|
|
.result = (CliKeyCombo){ \
|
|
.modifiers = modifiers_val, \
|
|
.key = key_val, \
|
|
}}; \
|
|
} while(0);
|
|
|
|
CliAnsiParserResult cli_ansi_parser_feed(CliAnsiParser* parser, char c) {
|
|
switch(parser->state) {
|
|
case CliAnsiParserStateInitial:
|
|
// <key> -> <key>
|
|
if(c != CliKeyEsc) PARSER_RESET_AND_RETURN(parser, CliModKeyNo, c); // -V1048
|
|
|
|
// <ESC> ...
|
|
parser->state = CliAnsiParserStateEscape;
|
|
break;
|
|
|
|
case CliAnsiParserStateEscape:
|
|
// <ESC> <ESC> -> <ESC>
|
|
if(c == CliKeyEsc) PARSER_RESET_AND_RETURN(parser, CliModKeyNo, c);
|
|
|
|
// <ESC> <key> -> Alt + <key>
|
|
if(c != '[') PARSER_RESET_AND_RETURN(parser, CliModKeyAlt, c);
|
|
|
|
// <ESC> [ ...
|
|
parser->state = CliAnsiParserStateEscapeBrace;
|
|
break;
|
|
|
|
case CliAnsiParserStateEscapeBrace:
|
|
// <ESC> [ <key mnemonic> -> <key>
|
|
if(c != '1') PARSER_RESET_AND_RETURN(parser, CliModKeyNo, cli_ansi_key_from_mnemonic(c));
|
|
|
|
// <ESC> [ 1 ...
|
|
parser->state = CliAnsiParserStateEscapeBraceOne;
|
|
break;
|
|
|
|
case CliAnsiParserStateEscapeBraceOne:
|
|
// <ESC> [ 1 <non-;> -> error
|
|
if(c != ';') PARSER_RESET_AND_RETURN(parser, CliModKeyNo, CliKeyUnrecognized);
|
|
|
|
// <ESC> [ 1 ; ...
|
|
parser->state = CliAnsiParserStateEscapeBraceOneSemicolon;
|
|
break;
|
|
|
|
case CliAnsiParserStateEscapeBraceOneSemicolon:
|
|
// <ESC> [ 1 ; <modifiers> ...
|
|
parser->modifiers = (c - '0');
|
|
parser->modifiers &= ~1;
|
|
parser->state = CliAnsiParserStateEscapeBraceOneSemicolonModifiers;
|
|
break;
|
|
|
|
case CliAnsiParserStateEscapeBraceOneSemicolonModifiers:
|
|
// <ESC> [ 1 ; <modifiers> <key mnemonic> -> <modifiers> + <key>
|
|
PARSER_RESET_AND_RETURN(parser, parser->modifiers, cli_ansi_key_from_mnemonic(c));
|
|
}
|
|
|
|
return (CliAnsiParserResult){.is_done = false};
|
|
}
|
|
|
|
CliAnsiParserResult cli_ansi_parser_feed_timeout(CliAnsiParser* parser) {
|
|
CliAnsiParserResult result = {.is_done = false};
|
|
|
|
if(parser->state == CliAnsiParserStateEscape) {
|
|
result.is_done = true;
|
|
result.result.key = CliKeyEsc;
|
|
result.result.modifiers = CliModKeyNo;
|
|
}
|
|
|
|
parser->state = CliAnsiParserStateInitial;
|
|
return result;
|
|
}
|