mirror of
https://github.com/DarkFlippers/unleashed-firmware.git
synced 2025-12-12 20:49:49 +04:00
* furi: update heap4 to latest * debug: heap under/overflow testing app * fix formatting * silence pvs warnings * Linker: symbols without type * Infrared: fix crash in universal remotes on long back press * Infrared: properly fix incorrect input handling behavior and crash in universal remote. Fix same issue in hid_app. * FreeRTOSConfig: explicit cast to uint in configTOTAL_HEAP_SIZE * Format sources * C and C++ compatible version of stm32wb55_linker.h Co-authored-by: あく <alleteam@gmail.com>
187 lines
5.5 KiB
C
187 lines
5.5 KiB
C
#include <furi_hal.h>
|
|
#include <furi.h>
|
|
|
|
#include <gui/gui.h>
|
|
#include <gui/view_dispatcher.h>
|
|
#include <gui/modules/submenu.h>
|
|
|
|
#define TAG "CrashTest"
|
|
|
|
typedef struct {
|
|
Gui* gui;
|
|
ViewDispatcher* view_dispatcher;
|
|
Submenu* submenu;
|
|
} CrashTest;
|
|
|
|
typedef enum {
|
|
CrashTestViewSubmenu,
|
|
} CrashTestView;
|
|
|
|
typedef enum {
|
|
CrashTestSubmenuCheck,
|
|
CrashTestSubmenuCheckMessage,
|
|
CrashTestSubmenuAssert,
|
|
CrashTestSubmenuAssertMessage,
|
|
CrashTestSubmenuCrash,
|
|
CrashTestSubmenuHalt,
|
|
CrashTestSubmenuHeapUnderflow,
|
|
CrashTestSubmenuHeapOverflow,
|
|
} CrashTestSubmenu;
|
|
|
|
static void crash_test_corrupt_heap_underflow(void) {
|
|
const size_t block_size = 1000;
|
|
const size_t underflow_size = 123;
|
|
uint8_t* block = malloc(block_size);
|
|
|
|
#pragma GCC diagnostic push
|
|
#pragma GCC diagnostic ignored "-Wstringop-overflow" // that's what we want!
|
|
memset(block - underflow_size, 0xDD, underflow_size); // -V769
|
|
#pragma GCC diagnostic pop
|
|
|
|
free(block); // should crash here (if compiled with DEBUG=1)
|
|
|
|
// If we got here, the heap wasn't able to detect our corruption and crash
|
|
furi_crash("Test failed, should've crashed with \"FreeRTOS Assert\" error");
|
|
}
|
|
|
|
static void crash_test_corrupt_heap_overflow(void) {
|
|
const size_t block_size = 1000;
|
|
const size_t overflow_size = 123;
|
|
uint8_t* block1 = malloc(block_size);
|
|
uint8_t* block2 = malloc(block_size);
|
|
memset(block2, 12, 34); // simulate use to avoid optimization // -V597 // -V1086
|
|
|
|
#pragma GCC diagnostic push
|
|
#pragma GCC diagnostic ignored "-Wstringop-overflow" // that's what we want!
|
|
memset(block1 + block_size, 0xDD, overflow_size); // -V769 // -V512
|
|
#pragma GCC diagnostic pop
|
|
|
|
uint8_t* block3 = malloc(block_size);
|
|
memset(block3, 12, 34); // simulate use to avoid optimization // -V597 // -V1086
|
|
|
|
free(block3); // should crash here (if compiled with DEBUG=1)
|
|
free(block2);
|
|
free(block1);
|
|
|
|
// If we got here, the heap wasn't able to detect our corruption and crash
|
|
furi_crash("Test failed, should've crashed with \"FreeRTOS Assert\" error");
|
|
}
|
|
|
|
static void crash_test_submenu_callback(void* context, uint32_t index) {
|
|
CrashTest* instance = (CrashTest*)context;
|
|
UNUSED(instance);
|
|
|
|
switch(index) {
|
|
case CrashTestSubmenuCheck:
|
|
furi_check(false);
|
|
break;
|
|
case CrashTestSubmenuCheckMessage:
|
|
furi_check(false, "Crash test: furi_check with message");
|
|
break;
|
|
case CrashTestSubmenuAssert:
|
|
furi_assert(false);
|
|
break;
|
|
case CrashTestSubmenuAssertMessage:
|
|
furi_assert(false, "Crash test: furi_assert with message");
|
|
break;
|
|
case CrashTestSubmenuCrash:
|
|
furi_crash("Crash test: furi_crash");
|
|
break;
|
|
case CrashTestSubmenuHalt:
|
|
furi_halt("Crash test: furi_halt");
|
|
break;
|
|
case CrashTestSubmenuHeapUnderflow:
|
|
crash_test_corrupt_heap_underflow();
|
|
break;
|
|
case CrashTestSubmenuHeapOverflow:
|
|
crash_test_corrupt_heap_overflow();
|
|
break;
|
|
default:
|
|
furi_crash();
|
|
}
|
|
}
|
|
|
|
static uint32_t crash_test_exit_callback(void* context) {
|
|
UNUSED(context);
|
|
return VIEW_NONE;
|
|
}
|
|
|
|
CrashTest* crash_test_alloc(void) {
|
|
CrashTest* instance = malloc(sizeof(CrashTest));
|
|
|
|
View* view = NULL;
|
|
|
|
instance->gui = furi_record_open(RECORD_GUI);
|
|
instance->view_dispatcher = view_dispatcher_alloc();
|
|
view_dispatcher_attach_to_gui(
|
|
instance->view_dispatcher, instance->gui, ViewDispatcherTypeFullscreen);
|
|
|
|
// Menu
|
|
instance->submenu = submenu_alloc();
|
|
view = submenu_get_view(instance->submenu);
|
|
view_set_previous_callback(view, crash_test_exit_callback);
|
|
view_dispatcher_add_view(instance->view_dispatcher, CrashTestViewSubmenu, view);
|
|
submenu_add_item(
|
|
instance->submenu, "Check", CrashTestSubmenuCheck, crash_test_submenu_callback, instance);
|
|
submenu_add_item(
|
|
instance->submenu,
|
|
"Check with message",
|
|
CrashTestSubmenuCheckMessage,
|
|
crash_test_submenu_callback,
|
|
instance);
|
|
submenu_add_item(
|
|
instance->submenu, "Assert", CrashTestSubmenuAssert, crash_test_submenu_callback, instance);
|
|
submenu_add_item(
|
|
instance->submenu,
|
|
"Assert with message",
|
|
CrashTestSubmenuAssertMessage,
|
|
crash_test_submenu_callback,
|
|
instance);
|
|
submenu_add_item(
|
|
instance->submenu, "Crash", CrashTestSubmenuCrash, crash_test_submenu_callback, instance);
|
|
submenu_add_item(
|
|
instance->submenu, "Halt", CrashTestSubmenuHalt, crash_test_submenu_callback, instance);
|
|
submenu_add_item(
|
|
instance->submenu,
|
|
"Heap underflow",
|
|
CrashTestSubmenuHeapUnderflow,
|
|
crash_test_submenu_callback,
|
|
instance);
|
|
submenu_add_item(
|
|
instance->submenu,
|
|
"Heap overflow",
|
|
CrashTestSubmenuHeapOverflow,
|
|
crash_test_submenu_callback,
|
|
instance);
|
|
|
|
return instance;
|
|
}
|
|
|
|
void crash_test_free(CrashTest* instance) {
|
|
view_dispatcher_remove_view(instance->view_dispatcher, CrashTestViewSubmenu);
|
|
submenu_free(instance->submenu);
|
|
|
|
view_dispatcher_free(instance->view_dispatcher);
|
|
furi_record_close(RECORD_GUI);
|
|
|
|
free(instance);
|
|
}
|
|
|
|
int32_t crash_test_run(CrashTest* instance) {
|
|
view_dispatcher_switch_to_view(instance->view_dispatcher, CrashTestViewSubmenu);
|
|
view_dispatcher_run(instance->view_dispatcher);
|
|
return 0;
|
|
}
|
|
|
|
int32_t crash_test_app(void* p) {
|
|
UNUSED(p);
|
|
|
|
CrashTest* instance = crash_test_alloc();
|
|
|
|
int32_t ret = crash_test_run(instance);
|
|
|
|
crash_test_free(instance);
|
|
|
|
return ret;
|
|
}
|