|
|
@@ -5,17 +5,13 @@
|
|
|
#include <lib/toolbox/args.h>
|
|
|
#include <furi_hal_usb_hid.h>
|
|
|
#include <storage/storage.h>
|
|
|
-#include "bad_usb_script.h"
|
|
|
-#include "mnemonic.h"
|
|
|
+#include "ducky_script.h"
|
|
|
+#include "ducky_script_i.h"
|
|
|
#include <dolphin/dolphin.h>
|
|
|
|
|
|
#define TAG "BadUSB"
|
|
|
#define WORKER_TAG TAG "Worker"
|
|
|
|
|
|
-#define SCRIPT_STATE_ERROR (-1)
|
|
|
-#define SCRIPT_STATE_END (-2)
|
|
|
-#define SCRIPT_STATE_NEXT_LINE (-3)
|
|
|
-
|
|
|
#define BADUSB_ASCII_TO_KEY(script, x) \
|
|
|
(((uint8_t)x < 128) ? (script->layout[(uint8_t)x]) : HID_KEYBOARD_NONE)
|
|
|
|
|
|
@@ -26,87 +22,20 @@ typedef enum {
|
|
|
WorkerEvtDisconnect = (1 << 3),
|
|
|
} WorkerEvtFlags;
|
|
|
|
|
|
-typedef struct {
|
|
|
- char* name;
|
|
|
- uint16_t keycode;
|
|
|
-} DuckyKey;
|
|
|
-
|
|
|
-static const DuckyKey ducky_keys[] = {
|
|
|
- {"CTRL-ALT", KEY_MOD_LEFT_CTRL | KEY_MOD_LEFT_ALT},
|
|
|
- {"CTRL-SHIFT", KEY_MOD_LEFT_CTRL | KEY_MOD_LEFT_SHIFT},
|
|
|
- {"ALT-SHIFT", KEY_MOD_LEFT_ALT | KEY_MOD_LEFT_SHIFT},
|
|
|
- {"ALT-GUI", KEY_MOD_LEFT_ALT | KEY_MOD_LEFT_GUI},
|
|
|
- {"GUI-SHIFT", KEY_MOD_LEFT_GUI | KEY_MOD_LEFT_SHIFT},
|
|
|
- {"GUI-CTRL", KEY_MOD_LEFT_GUI | KEY_MOD_LEFT_CTRL},
|
|
|
-
|
|
|
- {"CTRL", KEY_MOD_LEFT_CTRL},
|
|
|
- {"CONTROL", KEY_MOD_LEFT_CTRL},
|
|
|
- {"SHIFT", KEY_MOD_LEFT_SHIFT},
|
|
|
- {"ALT", KEY_MOD_LEFT_ALT},
|
|
|
- {"GUI", KEY_MOD_LEFT_GUI},
|
|
|
- {"WINDOWS", KEY_MOD_LEFT_GUI},
|
|
|
-
|
|
|
- {"DOWNARROW", HID_KEYBOARD_DOWN_ARROW},
|
|
|
- {"DOWN", HID_KEYBOARD_DOWN_ARROW},
|
|
|
- {"LEFTARROW", HID_KEYBOARD_LEFT_ARROW},
|
|
|
- {"LEFT", HID_KEYBOARD_LEFT_ARROW},
|
|
|
- {"RIGHTARROW", HID_KEYBOARD_RIGHT_ARROW},
|
|
|
- {"RIGHT", HID_KEYBOARD_RIGHT_ARROW},
|
|
|
- {"UPARROW", HID_KEYBOARD_UP_ARROW},
|
|
|
- {"UP", HID_KEYBOARD_UP_ARROW},
|
|
|
-
|
|
|
- {"ENTER", HID_KEYBOARD_RETURN},
|
|
|
- {"BREAK", HID_KEYBOARD_PAUSE},
|
|
|
- {"PAUSE", HID_KEYBOARD_PAUSE},
|
|
|
- {"CAPSLOCK", HID_KEYBOARD_CAPS_LOCK},
|
|
|
- {"DELETE", HID_KEYBOARD_DELETE_FORWARD},
|
|
|
- {"BACKSPACE", HID_KEYBOARD_DELETE},
|
|
|
- {"END", HID_KEYBOARD_END},
|
|
|
- {"ESC", HID_KEYBOARD_ESCAPE},
|
|
|
- {"ESCAPE", HID_KEYBOARD_ESCAPE},
|
|
|
- {"HOME", HID_KEYBOARD_HOME},
|
|
|
- {"INSERT", HID_KEYBOARD_INSERT},
|
|
|
- {"NUMLOCK", HID_KEYPAD_NUMLOCK},
|
|
|
- {"PAGEUP", HID_KEYBOARD_PAGE_UP},
|
|
|
- {"PAGEDOWN", HID_KEYBOARD_PAGE_DOWN},
|
|
|
- {"PRINTSCREEN", HID_KEYBOARD_PRINT_SCREEN},
|
|
|
- {"SCROLLLOCK", HID_KEYBOARD_SCROLL_LOCK},
|
|
|
- {"SPACE", HID_KEYBOARD_SPACEBAR},
|
|
|
- {"TAB", HID_KEYBOARD_TAB},
|
|
|
- {"MENU", HID_KEYBOARD_APPLICATION},
|
|
|
- {"APP", HID_KEYBOARD_APPLICATION},
|
|
|
-
|
|
|
- {"F1", HID_KEYBOARD_F1},
|
|
|
- {"F2", HID_KEYBOARD_F2},
|
|
|
- {"F3", HID_KEYBOARD_F3},
|
|
|
- {"F4", HID_KEYBOARD_F4},
|
|
|
- {"F5", HID_KEYBOARD_F5},
|
|
|
- {"F6", HID_KEYBOARD_F6},
|
|
|
- {"F7", HID_KEYBOARD_F7},
|
|
|
- {"F8", HID_KEYBOARD_F8},
|
|
|
- {"F9", HID_KEYBOARD_F9},
|
|
|
- {"F10", HID_KEYBOARD_F10},
|
|
|
- {"F11", HID_KEYBOARD_F11},
|
|
|
- {"F12", HID_KEYBOARD_F12},
|
|
|
-};
|
|
|
-
|
|
|
-static const char ducky_cmd_comment[] = {"REM"};
|
|
|
static const char ducky_cmd_id[] = {"ID"};
|
|
|
-static const char ducky_cmd_delay[] = {"DELAY "};
|
|
|
-static const char ducky_cmd_string[] = {"STRING "};
|
|
|
-static const char ducky_cmd_stringln[] = {"STRINGLN "};
|
|
|
-static const char ducky_cmd_defdelay_1[] = {"DEFAULT_DELAY "};
|
|
|
-static const char ducky_cmd_defdelay_2[] = {"DEFAULTDELAY "};
|
|
|
-static const char ducky_cmd_stringdelay_1[] = {"STRINGDELAY "};
|
|
|
-static const char ducky_cmd_stringdelay_2[] = {"STRING_DELAY "};
|
|
|
-static const char ducky_cmd_repeat[] = {"REPEAT "};
|
|
|
-static const char ducky_cmd_sysrq[] = {"SYSRQ "};
|
|
|
-static const char ducky_cmd_hold[] = {"HOLD "};
|
|
|
-static const char ducky_cmd_release[] = {"RELEASE "};
|
|
|
-
|
|
|
-static const char ducky_cmd_altchar[] = {"ALTCHAR "};
|
|
|
-static const char ducky_cmd_altstr_1[] = {"ALTSTRING "};
|
|
|
-static const char ducky_cmd_altstr_2[] = {"ALTCODE "};
|
|
|
+
|
|
|
+static const uint8_t numpad_keys[10] = {
|
|
|
+ HID_KEYPAD_0,
|
|
|
+ HID_KEYPAD_1,
|
|
|
+ HID_KEYPAD_2,
|
|
|
+ HID_KEYPAD_3,
|
|
|
+ HID_KEYPAD_4,
|
|
|
+ HID_KEYPAD_5,
|
|
|
+ HID_KEYPAD_6,
|
|
|
+ HID_KEYPAD_7,
|
|
|
+ HID_KEYPAD_8,
|
|
|
+ HID_KEYPAD_9,
|
|
|
+};
|
|
|
|
|
|
uint32_t ducky_get_command_len(const char* line) {
|
|
|
uint32_t len = strlen(line);
|
|
|
@@ -121,76 +50,150 @@ bool ducky_is_line_end(const char chr) {
|
|
|
}
|
|
|
|
|
|
uint16_t ducky_get_keycode(BadUsbScript* bad_usb, const char* param, bool accept_chars) {
|
|
|
- for(size_t i = 0; i < (sizeof(ducky_keys) / sizeof(ducky_keys[0])); i++) {
|
|
|
- size_t key_cmd_len = strlen(ducky_keys[i].name);
|
|
|
- if((strncmp(param, ducky_keys[i].name, key_cmd_len) == 0) &&
|
|
|
- (ducky_is_line_end(param[key_cmd_len]))) {
|
|
|
- return ducky_keys[i].keycode;
|
|
|
- }
|
|
|
+ uint16_t keycode = ducky_get_keycode_by_name(param);
|
|
|
+ if(keycode != HID_KEYBOARD_NONE) {
|
|
|
+ return keycode;
|
|
|
}
|
|
|
+
|
|
|
if((accept_chars) && (strlen(param) > 0)) {
|
|
|
return (BADUSB_ASCII_TO_KEY(bad_usb, param[0]) & 0xFF);
|
|
|
}
|
|
|
return 0;
|
|
|
}
|
|
|
|
|
|
-static int32_t
|
|
|
- ducky_parse_line(BadUsbScript* bad_usb, FuriString* line, char* error, size_t error_len) {
|
|
|
+bool ducky_get_number(const char* param, uint32_t* val) {
|
|
|
+ uint32_t value = 0;
|
|
|
+ if(sscanf(param, "%lu", &value) == 1) {
|
|
|
+ *val = value;
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+ return false;
|
|
|
+}
|
|
|
+
|
|
|
+void ducky_numlock_on() {
|
|
|
+ if((furi_hal_hid_get_led_state() & HID_KB_LED_NUM) == 0) {
|
|
|
+ furi_hal_hid_kb_press(HID_KEYBOARD_LOCK_NUM_LOCK);
|
|
|
+ furi_hal_hid_kb_release(HID_KEYBOARD_LOCK_NUM_LOCK);
|
|
|
+ }
|
|
|
+}
|
|
|
+bool ducky_numpad_press(const char num) {
|
|
|
+ if((num < '0') || (num > '9')) return false;
|
|
|
+
|
|
|
+ uint16_t key = numpad_keys[num - '0'];
|
|
|
+ furi_hal_hid_kb_press(key);
|
|
|
+ furi_hal_hid_kb_release(key);
|
|
|
+
|
|
|
+ return true;
|
|
|
+}
|
|
|
+
|
|
|
+bool ducky_altchar(const char* charcode) {
|
|
|
+ uint8_t i = 0;
|
|
|
+ bool state = false;
|
|
|
+
|
|
|
+ furi_hal_hid_kb_press(KEY_MOD_LEFT_ALT);
|
|
|
+
|
|
|
+ while(!ducky_is_line_end(charcode[i])) {
|
|
|
+ state = ducky_numpad_press(charcode[i]);
|
|
|
+ if(state == false) break;
|
|
|
+ i++;
|
|
|
+ }
|
|
|
+
|
|
|
+ furi_hal_hid_kb_release(KEY_MOD_LEFT_ALT);
|
|
|
+ return state;
|
|
|
+}
|
|
|
+
|
|
|
+bool ducky_altstring(const char* param) {
|
|
|
+ uint32_t i = 0;
|
|
|
+ bool state = false;
|
|
|
+
|
|
|
+ while(param[i] != '\0') {
|
|
|
+ if((param[i] < ' ') || (param[i] > '~')) {
|
|
|
+ i++;
|
|
|
+ continue; // Skip non-printable chars
|
|
|
+ }
|
|
|
+
|
|
|
+ char temp_str[4];
|
|
|
+ snprintf(temp_str, 4, "%u", param[i]);
|
|
|
+
|
|
|
+ state = ducky_altchar(temp_str);
|
|
|
+ if(state == false) break;
|
|
|
+ i++;
|
|
|
+ }
|
|
|
+ return state;
|
|
|
+}
|
|
|
+
|
|
|
+int32_t ducky_error(BadUsbScript* bad_usb, const char* text, ...) {
|
|
|
+ va_list args;
|
|
|
+ va_start(args, text);
|
|
|
+
|
|
|
+ vsnprintf(bad_usb->st.error, sizeof(bad_usb->st.error), text, args);
|
|
|
+
|
|
|
+ va_end(args);
|
|
|
+ return SCRIPT_STATE_ERROR;
|
|
|
+}
|
|
|
+
|
|
|
+bool ducky_string(BadUsbScript* bad_usb, const char* param) {
|
|
|
+ uint32_t i = 0;
|
|
|
+
|
|
|
+ while(param[i] != '\0') {
|
|
|
+ if(param[i] != '\n') {
|
|
|
+ uint16_t keycode = BADUSB_ASCII_TO_KEY(bad_usb, param[i]);
|
|
|
+ if(keycode != HID_KEYBOARD_NONE) {
|
|
|
+ furi_hal_hid_kb_press(keycode);
|
|
|
+ furi_hal_hid_kb_release(keycode);
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ furi_hal_hid_kb_press(HID_KEYBOARD_RETURN);
|
|
|
+ furi_hal_hid_kb_release(HID_KEYBOARD_RETURN);
|
|
|
+ }
|
|
|
+ i++;
|
|
|
+ }
|
|
|
+ bad_usb->stringdelay = 0;
|
|
|
+ return true;
|
|
|
+}
|
|
|
+
|
|
|
+static bool ducky_string_next(BadUsbScript* bad_usb) {
|
|
|
+ if(bad_usb->string_print_pos >= furi_string_size(bad_usb->string_print)) {
|
|
|
+ return true;
|
|
|
+ }
|
|
|
+
|
|
|
+ char print_char = furi_string_get_char(bad_usb->string_print, bad_usb->string_print_pos);
|
|
|
+
|
|
|
+ if(print_char != '\n') {
|
|
|
+ uint16_t keycode = BADUSB_ASCII_TO_KEY(bad_usb, print_char);
|
|
|
+ if(keycode != HID_KEYBOARD_NONE) {
|
|
|
+ furi_hal_hid_kb_press(keycode);
|
|
|
+ furi_hal_hid_kb_release(keycode);
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ furi_hal_hid_kb_press(HID_KEYBOARD_RETURN);
|
|
|
+ furi_hal_hid_kb_release(HID_KEYBOARD_RETURN);
|
|
|
+ }
|
|
|
+
|
|
|
+ bad_usb->string_print_pos++;
|
|
|
+
|
|
|
+ return false;
|
|
|
+}
|
|
|
+
|
|
|
+static int32_t ducky_parse_line(BadUsbScript* bad_usb, FuriString* line) {
|
|
|
uint32_t line_len = furi_string_size(line);
|
|
|
const char* line_tmp = furi_string_get_cstr(line);
|
|
|
- const char* ducky_cmd_table[] = {
|
|
|
- ducky_cmd_comment,
|
|
|
- ducky_cmd_id,
|
|
|
- ducky_cmd_delay,
|
|
|
- ducky_cmd_string,
|
|
|
- ducky_cmd_defdelay_1,
|
|
|
- ducky_cmd_defdelay_2,
|
|
|
- ducky_cmd_stringdelay_1,
|
|
|
- ducky_cmd_stringdelay_2,
|
|
|
- ducky_cmd_repeat,
|
|
|
- ducky_cmd_sysrq,
|
|
|
- ducky_cmd_altchar,
|
|
|
- ducky_cmd_altstr_1,
|
|
|
- ducky_cmd_altstr_2,
|
|
|
- ducky_cmd_stringln,
|
|
|
- ducky_cmd_hold,
|
|
|
- ducky_cmd_release,
|
|
|
- NULL};
|
|
|
- int32_t (*fnc_ptr[])(BadUsbScript*, FuriString*, const char*, char*, size_t) = {
|
|
|
- &ducky_fnc_noop,
|
|
|
- &ducky_fnc_noop,
|
|
|
- &ducky_fnc_delay,
|
|
|
- &ducky_fnc_string,
|
|
|
- &ducky_fnc_defdelay,
|
|
|
- &ducky_fnc_defdelay,
|
|
|
- &ducky_fnc_strdelay,
|
|
|
- &ducky_fnc_strdelay,
|
|
|
- &ducky_fnc_repeat,
|
|
|
- &ducky_fnc_sysrq,
|
|
|
- &ducky_fnc_altchar,
|
|
|
- &ducky_fnc_altstring,
|
|
|
- &ducky_fnc_altstring,
|
|
|
- &ducky_fnc_stringln,
|
|
|
- &ducky_fnc_hold,
|
|
|
- &ducky_fnc_release,
|
|
|
- NULL};
|
|
|
|
|
|
if(line_len == 0) {
|
|
|
return SCRIPT_STATE_NEXT_LINE; // Skip empty lines
|
|
|
}
|
|
|
FURI_LOG_D(WORKER_TAG, "line:%s", line_tmp);
|
|
|
+
|
|
|
// Ducky Lang Functions
|
|
|
- for(size_t i = 0; ducky_cmd_table[i]; i++) {
|
|
|
- if(strncmp(line_tmp, ducky_cmd_table[i], strlen(ducky_cmd_table[i])) == 0)
|
|
|
- return ((fnc_ptr[i])(bad_usb, line, line_tmp, error, error_len));
|
|
|
+ int32_t cmd_result = ducky_execute_cmd(bad_usb, line_tmp);
|
|
|
+ if(cmd_result != SCRIPT_STATE_CMD_UNKNOWN) {
|
|
|
+ return cmd_result;
|
|
|
}
|
|
|
+
|
|
|
// Special keys + modifiers
|
|
|
uint16_t key = ducky_get_keycode(bad_usb, line_tmp, false);
|
|
|
if(key == HID_KEYBOARD_NONE) {
|
|
|
- if(error != NULL) {
|
|
|
- snprintf(error, error_len, "No keycode defined for %s", line_tmp);
|
|
|
- }
|
|
|
- return SCRIPT_STATE_ERROR;
|
|
|
+ return ducky_error(bad_usb, "No keycode defined for %s", line_tmp);
|
|
|
}
|
|
|
if((key & 0xFF00) != 0) {
|
|
|
// It's a modifier key
|
|
|
@@ -199,7 +202,7 @@ static int32_t
|
|
|
}
|
|
|
furi_hal_hid_kb_press(key);
|
|
|
furi_hal_hid_kb_release(key);
|
|
|
- return (0);
|
|
|
+ return 0;
|
|
|
}
|
|
|
|
|
|
static bool ducky_set_usb_id(BadUsbScript* bad_usb, const char* line) {
|
|
|
@@ -277,8 +280,7 @@ static int32_t ducky_script_execute_next(BadUsbScript* bad_usb, File* script_fil
|
|
|
|
|
|
if(bad_usb->repeat_cnt > 0) {
|
|
|
bad_usb->repeat_cnt--;
|
|
|
- delay_val = ducky_parse_line(
|
|
|
- bad_usb, bad_usb->line_prev, bad_usb->st.error, sizeof(bad_usb->st.error));
|
|
|
+ delay_val = ducky_parse_line(bad_usb, bad_usb->line_prev);
|
|
|
if(delay_val == SCRIPT_STATE_NEXT_LINE) { // Empty line
|
|
|
return 0;
|
|
|
} else if(delay_val < 0) { // Script error
|
|
|
@@ -313,10 +315,11 @@ static int32_t ducky_script_execute_next(BadUsbScript* bad_usb, File* script_fil
|
|
|
bad_usb->buf_len = bad_usb->buf_len + bad_usb->buf_start - (i + 1);
|
|
|
bad_usb->buf_start = i + 1;
|
|
|
furi_string_trim(bad_usb->line);
|
|
|
- delay_val = ducky_parse_line(
|
|
|
- bad_usb, bad_usb->line, bad_usb->st.error, sizeof(bad_usb->st.error));
|
|
|
+ delay_val = ducky_parse_line(bad_usb, bad_usb->line);
|
|
|
if(delay_val == SCRIPT_STATE_NEXT_LINE) { // Empty line
|
|
|
return 0;
|
|
|
+ } else if(delay_val == SCRIPT_STATE_STRING_START) { // Print string with delays
|
|
|
+ return delay_val;
|
|
|
} else if(delay_val < 0) {
|
|
|
bad_usb->st.error_line = bad_usb->st.line_cur;
|
|
|
FURI_LOG_E(WORKER_TAG, "Unknown command at line %u", bad_usb->st.line_cur);
|
|
|
@@ -339,10 +342,11 @@ static void bad_usb_hid_state_callback(bool state, void* context) {
|
|
|
furi_assert(context);
|
|
|
BadUsbScript* bad_usb = context;
|
|
|
|
|
|
- if(state == true)
|
|
|
+ if(state == true) {
|
|
|
furi_thread_flags_set(furi_thread_get_id(bad_usb->thread), WorkerEvtConnect);
|
|
|
- else
|
|
|
+ } else {
|
|
|
furi_thread_flags_set(furi_thread_get_id(bad_usb->thread), WorkerEvtDisconnect);
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
static uint32_t bad_usb_flags_get(uint32_t flags_mask, uint32_t timeout) {
|
|
|
@@ -368,6 +372,7 @@ static int32_t bad_usb_worker(void* context) {
|
|
|
File* script_file = storage_file_alloc(furi_record_open(RECORD_STORAGE));
|
|
|
bad_usb->line = furi_string_alloc();
|
|
|
bad_usb->line_prev = furi_string_alloc();
|
|
|
+ bad_usb->string_print = furi_string_alloc();
|
|
|
|
|
|
furi_hal_hid_set_state_callback(bad_usb_hid_state_callback, bad_usb);
|
|
|
|
|
|
@@ -420,6 +425,7 @@ static int32_t bad_usb_worker(void* context) {
|
|
|
bad_usb->defdelay = 0;
|
|
|
bad_usb->stringdelay = 0;
|
|
|
bad_usb->repeat_cnt = 0;
|
|
|
+ bad_usb->key_hold_nb = 0;
|
|
|
bad_usb->file_end = false;
|
|
|
storage_file_seek(script_file, 0, true);
|
|
|
worker_state = BadUsbStateRunning;
|
|
|
@@ -492,12 +498,17 @@ static int32_t bad_usb_worker(void* context) {
|
|
|
delay_val = 0;
|
|
|
worker_state = BadUsbStateScriptError;
|
|
|
bad_usb->st.state = worker_state;
|
|
|
+ furi_hal_hid_kb_release_all();
|
|
|
} else if(delay_val == SCRIPT_STATE_END) { // End of script
|
|
|
delay_val = 0;
|
|
|
worker_state = BadUsbStateIdle;
|
|
|
bad_usb->st.state = BadUsbStateDone;
|
|
|
furi_hal_hid_kb_release_all();
|
|
|
continue;
|
|
|
+ } else if(delay_val == SCRIPT_STATE_STRING_START) { // Start printing string with delays
|
|
|
+ delay_val = bad_usb->defdelay;
|
|
|
+ bad_usb->string_print_pos = 0;
|
|
|
+ worker_state = BadUsbStateStringDelay;
|
|
|
} else if(delay_val > 1000) {
|
|
|
bad_usb->st.state = BadUsbStateDelay; // Show long delays
|
|
|
bad_usb->st.delay_remain = delay_val / 1000;
|
|
|
@@ -505,7 +516,35 @@ static int32_t bad_usb_worker(void* context) {
|
|
|
} else {
|
|
|
furi_check((flags & FuriFlagError) == 0);
|
|
|
}
|
|
|
+ } else if(worker_state == BadUsbStateStringDelay) { // State: print string with delays
|
|
|
+ uint32_t flags = furi_thread_flags_wait(
|
|
|
+ WorkerEvtEnd | WorkerEvtToggle | WorkerEvtDisconnect,
|
|
|
+ FuriFlagWaitAny,
|
|
|
+ bad_usb->stringdelay);
|
|
|
|
|
|
+ if(!(flags & FuriFlagError)) {
|
|
|
+ if(flags & WorkerEvtEnd) {
|
|
|
+ break;
|
|
|
+ } else if(flags & WorkerEvtToggle) {
|
|
|
+ worker_state = BadUsbStateIdle; // Stop executing script
|
|
|
+ furi_hal_hid_kb_release_all();
|
|
|
+ } else if(flags & WorkerEvtDisconnect) {
|
|
|
+ worker_state = BadUsbStateNotConnected; // USB disconnected
|
|
|
+ furi_hal_hid_kb_release_all();
|
|
|
+ }
|
|
|
+ bad_usb->st.state = worker_state;
|
|
|
+ continue;
|
|
|
+ } else if(
|
|
|
+ (flags == (unsigned)FuriFlagErrorTimeout) ||
|
|
|
+ (flags == (unsigned)FuriFlagErrorResource)) {
|
|
|
+ bool string_end = ducky_string_next(bad_usb);
|
|
|
+ if(string_end) {
|
|
|
+ bad_usb->stringdelay = 0;
|
|
|
+ worker_state = BadUsbStateRunning;
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ furi_check((flags & FuriFlagError) == 0);
|
|
|
+ }
|
|
|
} else if(
|
|
|
(worker_state == BadUsbStateFileError) ||
|
|
|
(worker_state == BadUsbStateScriptError)) { // State: error
|
|
|
@@ -524,6 +563,7 @@ static int32_t bad_usb_worker(void* context) {
|
|
|
storage_file_free(script_file);
|
|
|
furi_string_free(bad_usb->line);
|
|
|
furi_string_free(bad_usb->line_prev);
|
|
|
+ furi_string_free(bad_usb->string_print);
|
|
|
|
|
|
FURI_LOG_I(WORKER_TAG, "End");
|
|
|
|