mirror of
https://github.com/DarkFlippers/unleashed-firmware
synced 2024-11-22 20:43:07 +00:00
Merge remote-tracking branch 'OFW/dev' into dev
This commit is contained in:
commit
891a6c047a
6 changed files with 89 additions and 68 deletions
51
applications/debug/unit_tests/tests/furi/furi_errno_test.c
Normal file
51
applications/debug/unit_tests/tests/furi/furi_errno_test.c
Normal file
|
@ -0,0 +1,51 @@
|
||||||
|
#include <furi.h>
|
||||||
|
#include <errno.h>
|
||||||
|
#include "../test.h" // IWYU pragma: keep
|
||||||
|
|
||||||
|
#define TAG "ErrnoTest"
|
||||||
|
#define THREAD_CNT 16
|
||||||
|
#define ITER_CNT 1000
|
||||||
|
|
||||||
|
static int32_t errno_fuzzer(void* context) {
|
||||||
|
int start_value = (int)context;
|
||||||
|
int32_t fails = 0;
|
||||||
|
|
||||||
|
for(int i = start_value; i < start_value + ITER_CNT; i++) {
|
||||||
|
errno = i;
|
||||||
|
furi_thread_yield();
|
||||||
|
if(errno != i) fails++;
|
||||||
|
}
|
||||||
|
|
||||||
|
for(int i = 0; i < ITER_CNT; i++) {
|
||||||
|
errno = 0;
|
||||||
|
furi_thread_yield();
|
||||||
|
UNUSED(strtol("123456", NULL, 10)); // -V530
|
||||||
|
furi_thread_yield();
|
||||||
|
if(errno != 0) fails++;
|
||||||
|
|
||||||
|
errno = 0;
|
||||||
|
furi_thread_yield();
|
||||||
|
UNUSED(strtol("123456123456123456123456123456123456123456123456", NULL, 10)); // -V530
|
||||||
|
furi_thread_yield();
|
||||||
|
if(errno != ERANGE) fails++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return fails;
|
||||||
|
}
|
||||||
|
|
||||||
|
void test_errno_saving(void) {
|
||||||
|
FuriThread* threads[THREAD_CNT];
|
||||||
|
|
||||||
|
for(int i = 0; i < THREAD_CNT; i++) {
|
||||||
|
int start_value = i * ITER_CNT;
|
||||||
|
threads[i] = furi_thread_alloc_ex("ErrnoFuzzer", 1024, errno_fuzzer, (void*)start_value);
|
||||||
|
furi_thread_set_priority(threads[i], FuriThreadPriorityNormal);
|
||||||
|
furi_thread_start(threads[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
for(int i = 0; i < THREAD_CNT; i++) {
|
||||||
|
furi_thread_join(threads[i]);
|
||||||
|
mu_assert_int_eq(0, furi_thread_get_return_code(threads[i]));
|
||||||
|
furi_thread_free(threads[i]);
|
||||||
|
}
|
||||||
|
}
|
|
@ -8,6 +8,7 @@ void test_furi_concurrent_access(void);
|
||||||
void test_furi_pubsub(void);
|
void test_furi_pubsub(void);
|
||||||
void test_furi_memmgr(void);
|
void test_furi_memmgr(void);
|
||||||
void test_furi_event_loop(void);
|
void test_furi_event_loop(void);
|
||||||
|
void test_errno_saving(void);
|
||||||
|
|
||||||
static int foo = 0;
|
static int foo = 0;
|
||||||
|
|
||||||
|
@ -42,6 +43,10 @@ MU_TEST(mu_test_furi_event_loop) {
|
||||||
test_furi_event_loop();
|
test_furi_event_loop();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MU_TEST(mu_test_errno_saving) {
|
||||||
|
test_errno_saving();
|
||||||
|
}
|
||||||
|
|
||||||
MU_TEST_SUITE(test_suite) {
|
MU_TEST_SUITE(test_suite) {
|
||||||
MU_SUITE_CONFIGURE(&test_setup, &test_teardown);
|
MU_SUITE_CONFIGURE(&test_setup, &test_teardown);
|
||||||
MU_RUN_TEST(test_check);
|
MU_RUN_TEST(test_check);
|
||||||
|
@ -51,6 +56,7 @@ MU_TEST_SUITE(test_suite) {
|
||||||
MU_RUN_TEST(mu_test_furi_pubsub);
|
MU_RUN_TEST(mu_test_furi_pubsub);
|
||||||
MU_RUN_TEST(mu_test_furi_memmgr);
|
MU_RUN_TEST(mu_test_furi_memmgr);
|
||||||
MU_RUN_TEST(mu_test_furi_event_loop);
|
MU_RUN_TEST(mu_test_furi_event_loop);
|
||||||
|
MU_RUN_TEST(mu_test_errno_saving);
|
||||||
}
|
}
|
||||||
|
|
||||||
int run_minunit_test_furi(void) {
|
int run_minunit_test_furi(void) {
|
||||||
|
|
|
@ -3,4 +3,3 @@ ADD_SCENE(gpio, test, Test)
|
||||||
ADD_SCENE(gpio, usb_uart, UsbUart)
|
ADD_SCENE(gpio, usb_uart, UsbUart)
|
||||||
ADD_SCENE(gpio, usb_uart_cfg, UsbUartCfg)
|
ADD_SCENE(gpio, usb_uart_cfg, UsbUartCfg)
|
||||||
ADD_SCENE(gpio, usb_uart_close_rpc, UsbUartCloseRpc)
|
ADD_SCENE(gpio, usb_uart_close_rpc, UsbUartCloseRpc)
|
||||||
ADD_SCENE(gpio, exit_confirm, ExitConfirm)
|
|
||||||
|
|
|
@ -1,43 +0,0 @@
|
||||||
#include "gpio_app_i.h"
|
|
||||||
|
|
||||||
void gpio_scene_exit_confirm_dialog_callback(DialogExResult result, void* context) {
|
|
||||||
GpioApp* app = context;
|
|
||||||
|
|
||||||
view_dispatcher_send_custom_event(app->view_dispatcher, result);
|
|
||||||
}
|
|
||||||
|
|
||||||
void gpio_scene_exit_confirm_on_enter(void* context) {
|
|
||||||
GpioApp* app = context;
|
|
||||||
DialogEx* dialog = app->dialog;
|
|
||||||
|
|
||||||
dialog_ex_set_context(dialog, app);
|
|
||||||
dialog_ex_set_left_button_text(dialog, "Exit");
|
|
||||||
dialog_ex_set_right_button_text(dialog, "Stay");
|
|
||||||
dialog_ex_set_header(dialog, "Exit USB-UART?", 22, 12, AlignLeft, AlignTop);
|
|
||||||
dialog_ex_set_result_callback(dialog, gpio_scene_exit_confirm_dialog_callback);
|
|
||||||
|
|
||||||
view_dispatcher_switch_to_view(app->view_dispatcher, GpioAppViewExitConfirm);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool gpio_scene_exit_confirm_on_event(void* context, SceneManagerEvent event) {
|
|
||||||
GpioApp* app = context;
|
|
||||||
bool consumed = false;
|
|
||||||
|
|
||||||
if(event.type == SceneManagerEventTypeCustom) {
|
|
||||||
consumed = scene_manager_previous_scene(app->scene_manager);
|
|
||||||
if(consumed && event.event == DialogExResultLeft) {
|
|
||||||
view_dispatcher_send_custom_event(app->view_dispatcher, GpioUsbUartEventStop);
|
|
||||||
}
|
|
||||||
} else if(event.type == SceneManagerEventTypeBack) {
|
|
||||||
consumed = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return consumed;
|
|
||||||
}
|
|
||||||
|
|
||||||
void gpio_scene_exit_confirm_on_exit(void* context) {
|
|
||||||
GpioApp* app = context;
|
|
||||||
|
|
||||||
// Clean view
|
|
||||||
dialog_ex_reset(app->dialog);
|
|
||||||
}
|
|
|
@ -6,7 +6,7 @@ typedef struct {
|
||||||
UsbUartState state;
|
UsbUartState state;
|
||||||
} SceneUsbUartBridge;
|
} SceneUsbUartBridge;
|
||||||
|
|
||||||
static SceneUsbUartBridge* scene_usb_uart;
|
static SceneUsbUartBridge* scene_usb_uart = NULL;
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
UsbUartSceneStateInitialize,
|
UsbUartSceneStateInitialize,
|
||||||
|
@ -19,11 +19,21 @@ void gpio_scene_usb_uart_callback(GpioCustomEvent event, void* context) {
|
||||||
view_dispatcher_send_custom_event(app->view_dispatcher, event);
|
view_dispatcher_send_custom_event(app->view_dispatcher, event);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void gpio_scene_usb_uart_dialog_callback(DialogExResult result, void* context) {
|
||||||
|
GpioApp* app = context;
|
||||||
|
if(result == DialogExResultLeft) {
|
||||||
|
usb_uart_disable(app->usb_uart_bridge);
|
||||||
|
free(scene_usb_uart);
|
||||||
|
scene_usb_uart = NULL;
|
||||||
|
scene_manager_search_and_switch_to_previous_scene(app->scene_manager, GpioSceneStart);
|
||||||
|
} else {
|
||||||
|
view_dispatcher_switch_to_view(app->view_dispatcher, GpioAppViewUsbUart);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void gpio_scene_usb_uart_on_enter(void* context) {
|
void gpio_scene_usb_uart_on_enter(void* context) {
|
||||||
GpioApp* app = context;
|
GpioApp* app = context;
|
||||||
UsbUartSceneState state =
|
if(!scene_usb_uart) {
|
||||||
scene_manager_get_scene_state(app->scene_manager, GpioAppViewUsbUart);
|
|
||||||
if(state == UsbUartSceneStateInitialize) {
|
|
||||||
scene_usb_uart = malloc(sizeof(SceneUsbUartBridge));
|
scene_usb_uart = malloc(sizeof(SceneUsbUartBridge));
|
||||||
scene_usb_uart->cfg.vcp_ch = 0;
|
scene_usb_uart->cfg.vcp_ch = 0;
|
||||||
scene_usb_uart->cfg.uart_ch = 0;
|
scene_usb_uart->cfg.uart_ch = 0;
|
||||||
|
@ -37,7 +47,6 @@ void gpio_scene_usb_uart_on_enter(void* context) {
|
||||||
usb_uart_get_state(app->usb_uart_bridge, &scene_usb_uart->state);
|
usb_uart_get_state(app->usb_uart_bridge, &scene_usb_uart->state);
|
||||||
|
|
||||||
gpio_usb_uart_set_callback(app->gpio_usb_uart, gpio_scene_usb_uart_callback, app);
|
gpio_usb_uart_set_callback(app->gpio_usb_uart, gpio_scene_usb_uart_callback, app);
|
||||||
scene_manager_set_scene_state(app->scene_manager, GpioSceneUsbUart, 0);
|
|
||||||
view_dispatcher_switch_to_view(app->view_dispatcher, GpioAppViewUsbUart);
|
view_dispatcher_switch_to_view(app->view_dispatcher, GpioAppViewUsbUart);
|
||||||
notification_message(app->notifications, &sequence_display_backlight_enforce_on);
|
notification_message(app->notifications, &sequence_display_backlight_enforce_on);
|
||||||
}
|
}
|
||||||
|
@ -45,19 +54,16 @@ void gpio_scene_usb_uart_on_enter(void* context) {
|
||||||
bool gpio_scene_usb_uart_on_event(void* context, SceneManagerEvent event) {
|
bool gpio_scene_usb_uart_on_event(void* context, SceneManagerEvent event) {
|
||||||
GpioApp* app = context;
|
GpioApp* app = context;
|
||||||
if(event.type == SceneManagerEventTypeCustom) {
|
if(event.type == SceneManagerEventTypeCustom) {
|
||||||
if(event.event == GpioUsbUartEventConfig) {
|
scene_manager_next_scene(app->scene_manager, GpioSceneUsbUartCfg);
|
||||||
scene_manager_set_scene_state(
|
|
||||||
app->scene_manager, GpioSceneUsbUart, UsbUartSceneStateKeep);
|
|
||||||
scene_manager_next_scene(app->scene_manager, GpioSceneUsbUartCfg);
|
|
||||||
} else if(event.event == GpioUsbUartEventStop) {
|
|
||||||
scene_manager_set_scene_state(
|
|
||||||
app->scene_manager, GpioSceneUsbUart, UsbUartSceneStateInitialize);
|
|
||||||
scene_manager_search_and_switch_to_previous_scene(app->scene_manager, GpioSceneStart);
|
|
||||||
}
|
|
||||||
return true;
|
return true;
|
||||||
} else if(event.type == SceneManagerEventTypeBack) {
|
} else if(event.type == SceneManagerEventTypeBack) {
|
||||||
scene_manager_set_scene_state(app->scene_manager, GpioSceneUsbUart, UsbUartSceneStateKeep);
|
DialogEx* dialog = app->dialog;
|
||||||
scene_manager_next_scene(app->scene_manager, GpioSceneExitConfirm);
|
dialog_ex_set_context(dialog, app);
|
||||||
|
dialog_ex_set_left_button_text(dialog, "Exit");
|
||||||
|
dialog_ex_set_right_button_text(dialog, "Stay");
|
||||||
|
dialog_ex_set_header(dialog, "Exit USB-UART?", 22, 12, AlignLeft, AlignTop);
|
||||||
|
dialog_ex_set_result_callback(dialog, gpio_scene_usb_uart_dialog_callback);
|
||||||
|
view_dispatcher_switch_to_view(app->view_dispatcher, GpioAppViewExitConfirm);
|
||||||
return true;
|
return true;
|
||||||
} else if(event.type == SceneManagerEventTypeTick) {
|
} else if(event.type == SceneManagerEventTypeTick) {
|
||||||
uint32_t tx_cnt_last = scene_usb_uart->state.tx_cnt;
|
uint32_t tx_cnt_last = scene_usb_uart->state.tx_cnt;
|
||||||
|
@ -75,10 +81,5 @@ bool gpio_scene_usb_uart_on_event(void* context, SceneManagerEvent event) {
|
||||||
|
|
||||||
void gpio_scene_usb_uart_on_exit(void* context) {
|
void gpio_scene_usb_uart_on_exit(void* context) {
|
||||||
GpioApp* app = context;
|
GpioApp* app = context;
|
||||||
uint32_t state = scene_manager_get_scene_state(app->scene_manager, GpioSceneUsbUart);
|
|
||||||
if(state == UsbUartSceneStateInitialize) {
|
|
||||||
usb_uart_disable(app->usb_uart_bridge);
|
|
||||||
free(scene_usb_uart);
|
|
||||||
}
|
|
||||||
notification_message(app->notifications, &sequence_display_backlight_enforce_auto);
|
notification_message(app->notifications, &sequence_display_backlight_enforce_auto);
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
#if defined(__ICCARM__) || defined(__CC_ARM) || defined(__GNUC__)
|
#if defined(__ICCARM__) || defined(__CC_ARM) || defined(__GNUC__)
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
#include <errno.h>
|
||||||
#pragma GCC diagnostic ignored "-Wredundant-decls"
|
#pragma GCC diagnostic ignored "-Wredundant-decls"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
@ -26,6 +27,7 @@
|
||||||
#define configUSE_16_BIT_TICKS 0
|
#define configUSE_16_BIT_TICKS 0
|
||||||
#define configMAX_PRIORITIES (32)
|
#define configMAX_PRIORITIES (32)
|
||||||
#define configMINIMAL_STACK_SIZE ((uint16_t)128)
|
#define configMINIMAL_STACK_SIZE ((uint16_t)128)
|
||||||
|
#define configUSE_POSIX_ERRNO 1
|
||||||
|
|
||||||
/* Heap size determined automatically by linker */
|
/* Heap size determined automatically by linker */
|
||||||
// #define configTOTAL_HEAP_SIZE ((size_t)0)
|
// #define configTOTAL_HEAP_SIZE ((size_t)0)
|
||||||
|
@ -146,9 +148,14 @@ standard names. */
|
||||||
#define configOVERRIDE_DEFAULT_TICK_CONFIGURATION \
|
#define configOVERRIDE_DEFAULT_TICK_CONFIGURATION \
|
||||||
1 /* required only for Keil but does not hurt otherwise */
|
1 /* required only for Keil but does not hurt otherwise */
|
||||||
|
|
||||||
#define traceTASK_SWITCHED_IN() \
|
#define traceTASK_SWITCHED_IN() \
|
||||||
extern void furi_hal_mpu_set_stack_protection(uint32_t* stack); \
|
extern void furi_hal_mpu_set_stack_protection(uint32_t* stack); \
|
||||||
furi_hal_mpu_set_stack_protection((uint32_t*)pxCurrentTCB->pxStack)
|
furi_hal_mpu_set_stack_protection((uint32_t*)pxCurrentTCB->pxStack); \
|
||||||
|
errno = pxCurrentTCB->iTaskErrno
|
||||||
|
// ^^^^^ acquire errno directly from TCB because FreeRTOS assigns its `FreeRTOS_errno' _after_ our hook is called
|
||||||
|
|
||||||
|
// referencing `FreeRTOS_errno' here vvvvv because FreeRTOS calls our hook _before_ copying the value into the TCB, hence a manual write to the TCB would get overwritten
|
||||||
|
#define traceTASK_SWITCHED_OUT() FreeRTOS_errno = errno
|
||||||
|
|
||||||
#define portCLEAN_UP_TCB(pxTCB) \
|
#define portCLEAN_UP_TCB(pxTCB) \
|
||||||
extern void furi_thread_cleanup_tcb_event(TaskHandle_t task); \
|
extern void furi_thread_cleanup_tcb_event(TaskHandle_t task); \
|
||||||
|
|
Loading…
Reference in a new issue