Skip to content

Commit bd348cf

Browse files
committed
Merge remote-tracking branch 'OFW/dev' into dev
2 parents d60c034 + 8dd5e64 commit bd348cf

File tree

18 files changed

+450
-94
lines changed

18 files changed

+450
-94
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,108 @@
1+
#include <furi.h>
2+
#include <errno.h>
3+
#include <stdio.h>
4+
#include "../test.h" // IWYU pragma: keep
5+
6+
#define TAG "StdioTest"
7+
8+
#define CONTEXT_MAGIC ((void*)0xDEADBEEF)
9+
10+
// stdin
11+
12+
static char mock_in[256];
13+
static size_t mock_in_len, mock_in_pos;
14+
15+
static void set_mock_in(const char* str) {
16+
size_t len = strlen(str);
17+
strcpy(mock_in, str);
18+
mock_in_len = len;
19+
mock_in_pos = 0;
20+
}
21+
22+
static size_t mock_in_cb(char* buffer, size_t size, FuriWait wait, void* context) {
23+
UNUSED(wait);
24+
furi_check(context == CONTEXT_MAGIC);
25+
size_t remaining = mock_in_len - mock_in_pos;
26+
size = MIN(remaining, size);
27+
memcpy(buffer, mock_in + mock_in_pos, size);
28+
mock_in_pos += size;
29+
return size;
30+
}
31+
32+
void test_stdin(void) {
33+
FuriThreadStdinReadCallback in_cb = furi_thread_get_stdin_callback();
34+
furi_thread_set_stdin_callback(mock_in_cb, CONTEXT_MAGIC);
35+
char buf[256];
36+
37+
// plain in
38+
set_mock_in("Hello, World!\n");
39+
fgets(buf, sizeof(buf), stdin);
40+
mu_assert_string_eq("Hello, World!\n", buf);
41+
mu_assert_int_eq(EOF, getchar());
42+
43+
// ungetc
44+
ungetc('i', stdin);
45+
ungetc('H', stdin);
46+
fgets(buf, sizeof(buf), stdin);
47+
mu_assert_string_eq("Hi", buf);
48+
mu_assert_int_eq(EOF, getchar());
49+
50+
// ungetc + plain in
51+
set_mock_in(" World");
52+
ungetc('i', stdin);
53+
ungetc('H', stdin);
54+
fgets(buf, sizeof(buf), stdin);
55+
mu_assert_string_eq("Hi World", buf);
56+
mu_assert_int_eq(EOF, getchar());
57+
58+
// partial plain in
59+
set_mock_in("Hello, World!\n");
60+
fgets(buf, strlen("Hello") + 1, stdin);
61+
mu_assert_string_eq("Hello", buf);
62+
mu_assert_int_eq(',', getchar());
63+
fgets(buf, sizeof(buf), stdin);
64+
mu_assert_string_eq(" World!\n", buf);
65+
66+
furi_thread_set_stdin_callback(in_cb, CONTEXT_MAGIC);
67+
}
68+
69+
// stdout
70+
71+
static FuriString* mock_out;
72+
FuriThreadStdoutWriteCallback original_out_cb;
73+
74+
static void mock_out_cb(const char* data, size_t size, void* context) {
75+
furi_check(context == CONTEXT_MAGIC);
76+
// there's no furi_string_cat_strn :(
77+
for(size_t i = 0; i < size; i++) {
78+
furi_string_push_back(mock_out, data[i]);
79+
}
80+
}
81+
82+
static void assert_and_clear_mock_out(const char* expected) {
83+
// return the original stdout callback for the duration of the check
84+
// if the check fails, we don't want the error to end up in our buffer,
85+
// we want to be able to see it!
86+
furi_thread_set_stdout_callback(original_out_cb, CONTEXT_MAGIC);
87+
mu_assert_string_eq(expected, furi_string_get_cstr(mock_out));
88+
furi_thread_set_stdout_callback(mock_out_cb, CONTEXT_MAGIC);
89+
90+
furi_string_reset(mock_out);
91+
}
92+
93+
void test_stdout(void) {
94+
original_out_cb = furi_thread_get_stdout_callback();
95+
furi_thread_set_stdout_callback(mock_out_cb, CONTEXT_MAGIC);
96+
mock_out = furi_string_alloc();
97+
98+
puts("Hello, World!");
99+
assert_and_clear_mock_out("Hello, World!\n");
100+
101+
printf("He");
102+
printf("llo!");
103+
fflush(stdout);
104+
assert_and_clear_mock_out("Hello!");
105+
106+
furi_string_free(mock_out);
107+
furi_thread_set_stdout_callback(original_out_cb, CONTEXT_MAGIC);
108+
}

applications/debug/unit_tests/tests/furi/furi_test.c

+8
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,8 @@ void test_furi_memmgr(void);
1010
void test_furi_event_loop(void);
1111
void test_errno_saving(void);
1212
void test_furi_primitives(void);
13+
void test_stdin(void);
14+
void test_stdout(void);
1315

1416
static int foo = 0;
1517

@@ -52,6 +54,11 @@ MU_TEST(mu_test_furi_primitives) {
5254
test_furi_primitives();
5355
}
5456

57+
MU_TEST(mu_test_stdio) {
58+
test_stdin();
59+
test_stdout();
60+
}
61+
5562
MU_TEST_SUITE(test_suite) {
5663
MU_SUITE_CONFIGURE(&test_setup, &test_teardown);
5764
MU_RUN_TEST(test_check);
@@ -61,6 +68,7 @@ MU_TEST_SUITE(test_suite) {
6168
MU_RUN_TEST(mu_test_furi_pubsub);
6269
MU_RUN_TEST(mu_test_furi_memmgr);
6370
MU_RUN_TEST(mu_test_furi_event_loop);
71+
MU_RUN_TEST(mu_test_stdio);
6472
MU_RUN_TEST(mu_test_errno_saving);
6573
MU_RUN_TEST(mu_test_furi_primitives);
6674
}

applications/services/cli/cli.c

+5-5
Original file line numberDiff line numberDiff line change
@@ -435,9 +435,9 @@ void cli_session_open(Cli* cli, void* session) {
435435
cli->session = session;
436436
if(cli->session != NULL) {
437437
cli->session->init();
438-
furi_thread_set_stdout_callback(cli->session->tx_stdout);
438+
furi_thread_set_stdout_callback(cli->session->tx_stdout, NULL);
439439
} else {
440-
furi_thread_set_stdout_callback(NULL);
440+
furi_thread_set_stdout_callback(NULL, NULL);
441441
}
442442
furi_semaphore_release(cli->idle_sem);
443443
furi_check(furi_mutex_release(cli->mutex) == FuriStatusOk);
@@ -451,7 +451,7 @@ void cli_session_close(Cli* cli) {
451451
cli->session->deinit();
452452
}
453453
cli->session = NULL;
454-
furi_thread_set_stdout_callback(NULL);
454+
furi_thread_set_stdout_callback(NULL, NULL);
455455
furi_check(furi_mutex_release(cli->mutex) == FuriStatusOk);
456456
}
457457

@@ -465,9 +465,9 @@ int32_t cli_srv(void* p) {
465465
furi_record_create(RECORD_CLI, cli);
466466

467467
if(cli->session != NULL) {
468-
furi_thread_set_stdout_callback(cli->session->tx_stdout);
468+
furi_thread_set_stdout_callback(cli->session->tx_stdout, NULL);
469469
} else {
470-
furi_thread_set_stdout_callback(NULL);
470+
furi_thread_set_stdout_callback(NULL, NULL);
471471
}
472472

473473
if(furi_hal_rtc_get_boot_mode() == FuriHalRtcBootModeNormal) {

applications/services/cli/cli_i.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -28,8 +28,9 @@ struct CliSession {
2828
void (*init)(void);
2929
void (*deinit)(void);
3030
size_t (*rx)(uint8_t* buffer, size_t size, uint32_t timeout);
31+
size_t (*rx_stdin)(uint8_t* buffer, size_t size, uint32_t timeout, void* context);
3132
void (*tx)(const uint8_t* buffer, size_t size);
32-
void (*tx_stdout)(const char* data, size_t size);
33+
void (*tx_stdout)(const char* data, size_t size, void* context);
3334
bool (*is_connected)(void);
3435
};
3536

applications/services/cli/cli_vcp.c

+8-1
Original file line numberDiff line numberDiff line change
@@ -243,6 +243,11 @@ static size_t cli_vcp_rx(uint8_t* buffer, size_t size, uint32_t timeout) {
243243
return rx_cnt;
244244
}
245245

246+
static size_t cli_vcp_rx_stdin(uint8_t* data, size_t size, uint32_t timeout, void* context) {
247+
UNUSED(context);
248+
return cli_vcp_rx(data, size, timeout);
249+
}
250+
246251
static void cli_vcp_tx(const uint8_t* buffer, size_t size) {
247252
furi_assert(vcp);
248253
furi_assert(buffer);
@@ -268,7 +273,8 @@ static void cli_vcp_tx(const uint8_t* buffer, size_t size) {
268273
VCP_DEBUG("tx %u end", size);
269274
}
270275

271-
static void cli_vcp_tx_stdout(const char* data, size_t size) {
276+
static void cli_vcp_tx_stdout(const char* data, size_t size, void* context) {
277+
UNUSED(context);
272278
cli_vcp_tx((const uint8_t*)data, size);
273279
}
274280

@@ -311,6 +317,7 @@ CliSession cli_vcp = {
311317
cli_vcp_init,
312318
cli_vcp_deinit,
313319
cli_vcp_rx,
320+
cli_vcp_rx_stdin,
314321
cli_vcp_tx,
315322
cli_vcp_tx_stdout,
316323
cli_vcp_is_connected,

documentation/UnitTests.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ To add unit tests for your protocol, follow these steps:
4343

4444
1. Create a file named `test_<your_protocol_name>.irtest` in the [assets](https://github.com/flipperdevices/flipperzero-firmware/tree/dev/applications/debug/unit_tests/resources/unit_tests/infrared) directory.
4545
2. Fill it with the test data (more on it below).
46-
3. Add the test code to [infrared_test.c](https://github.com/flipperdevices/flipperzero-firmware/blob/dev/applications/debug/unit_tests/infrared/infrared_test.c).
46+
3. Add the test code to [infrared_test.c](https://github.com/flipperdevices/flipperzero-firmware/blob/dev/applications/debug/unit_tests/tests/infrared/infrared_test.c).
4747
4. Build and install firmware with resources, install it on your Flipper and run the tests to see if they pass.
4848

4949
##### Test data format

furi/core/memmgr.c

+3-1
Original file line numberDiff line numberDiff line change
@@ -106,5 +106,7 @@ void* aligned_malloc(size_t size, size_t alignment) {
106106
}
107107

108108
void aligned_free(void* p) {
109-
free(((void**)p)[-1]);
109+
if(p) {
110+
free(((void**)p)[-1]);
111+
}
110112
}

furi/core/string.h

+4-4
Original file line numberDiff line numberDiff line change
@@ -129,12 +129,12 @@ void furi_string_swap(FuriString* string_1, FuriString* string_2);
129129

130130
/** Move string_2 content to string_1.
131131
*
132-
* Set the string to the other one, and destroy the other one.
132+
* Copy data from one string to another and destroy the source.
133133
*
134-
* @param string_1 The FuriString instance 1
135-
* @param string_2 The FuriString instance 2
134+
* @param destination The destination FuriString
135+
* @param source The source FuriString
136136
*/
137-
void furi_string_move(FuriString* string_1, FuriString* string_2);
137+
void furi_string_move(FuriString* destination, FuriString* source);
138138

139139
/** Compute a hash for the string.
140140
*

furi/core/thread.c

+64-8
Original file line numberDiff line numberDiff line change
@@ -23,12 +23,17 @@
2323

2424
#define THREAD_MAX_STACK_SIZE (UINT16_MAX * sizeof(StackType_t))
2525

26-
typedef struct FuriThreadStdout FuriThreadStdout;
27-
28-
struct FuriThreadStdout {
26+
typedef struct {
2927
FuriThreadStdoutWriteCallback write_callback;
3028
FuriString* buffer;
31-
};
29+
void* context;
30+
} FuriThreadStdout;
31+
32+
typedef struct {
33+
FuriThreadStdinReadCallback read_callback;
34+
FuriString* unread_buffer; // <! stores data from `ungetc` and friends
35+
void* context;
36+
} FuriThreadStdin;
3237

3338
struct FuriThread {
3439
StaticTask_t container;
@@ -55,6 +60,7 @@ struct FuriThread {
5560
size_t heap_size;
5661

5762
FuriThreadStdout output;
63+
FuriThreadStdin input;
5864

5965
// Keep all non-alignable byte types in one place,
6066
// this ensures that the size of this structure is minimal
@@ -136,6 +142,7 @@ static void furi_thread_body(void* context) {
136142

137143
static void furi_thread_init_common(FuriThread* thread) {
138144
thread->output.buffer = furi_string_alloc();
145+
thread->input.unread_buffer = furi_string_alloc();
139146

140147
FuriThread* parent = NULL;
141148
if(xTaskGetSchedulerState() != taskSCHEDULER_NOT_STARTED) {
@@ -245,6 +252,7 @@ void furi_thread_free(FuriThread* thread) {
245252
}
246253

247254
furi_string_free(thread->output.buffer);
255+
furi_string_free(thread->input.unread_buffer);
248256
free(thread);
249257
}
250258

@@ -710,13 +718,22 @@ uint32_t furi_thread_get_stack_space(FuriThreadId thread_id) {
710718

711719
static size_t __furi_thread_stdout_write(FuriThread* thread, const char* data, size_t size) {
712720
if(thread->output.write_callback != NULL) {
713-
thread->output.write_callback(data, size);
721+
thread->output.write_callback(data, size, thread->output.context);
714722
} else {
715723
furi_log_tx((const uint8_t*)data, size);
716724
}
717725
return size;
718726
}
719727

728+
static size_t
729+
__furi_thread_stdin_read(FuriThread* thread, char* data, size_t size, FuriWait timeout) {
730+
if(thread->input.read_callback != NULL) {
731+
return thread->input.read_callback(data, size, timeout, thread->input.context);
732+
} else {
733+
return 0;
734+
}
735+
}
736+
720737
static int32_t __furi_thread_stdout_flush(FuriThread* thread) {
721738
FuriString* buffer = thread->output.buffer;
722739
size_t size = furi_string_size(buffer);
@@ -727,17 +744,31 @@ static int32_t __furi_thread_stdout_flush(FuriThread* thread) {
727744
return 0;
728745
}
729746

730-
void furi_thread_set_stdout_callback(FuriThreadStdoutWriteCallback callback) {
747+
FuriThreadStdoutWriteCallback furi_thread_get_stdout_callback(void) {
748+
FuriThread* thread = furi_thread_get_current();
749+
furi_check(thread);
750+
return thread->output.write_callback;
751+
}
752+
753+
FuriThreadStdinReadCallback furi_thread_get_stdin_callback(void) {
754+
FuriThread* thread = furi_thread_get_current();
755+
furi_check(thread);
756+
return thread->input.read_callback;
757+
}
758+
759+
void furi_thread_set_stdout_callback(FuriThreadStdoutWriteCallback callback, void* context) {
731760
FuriThread* thread = furi_thread_get_current();
732761
furi_check(thread);
733762
__furi_thread_stdout_flush(thread);
734763
thread->output.write_callback = callback;
764+
thread->output.context = context;
735765
}
736766

737-
FuriThreadStdoutWriteCallback furi_thread_get_stdout_callback(void) {
767+
void furi_thread_set_stdin_callback(FuriThreadStdinReadCallback callback, void* context) {
738768
FuriThread* thread = furi_thread_get_current();
739769
furi_check(thread);
740-
return thread->output.write_callback;
770+
thread->input.read_callback = callback;
771+
thread->input.context = context;
741772
}
742773

743774
size_t furi_thread_stdout_write(const char* data, size_t size) {
@@ -772,6 +803,31 @@ int32_t furi_thread_stdout_flush(void) {
772803
return __furi_thread_stdout_flush(thread);
773804
}
774805

806+
size_t furi_thread_stdin_read(char* buffer, size_t size, FuriWait timeout) {
807+
FuriThread* thread = furi_thread_get_current();
808+
furi_check(thread);
809+
810+
size_t from_buffer = MIN(furi_string_size(thread->input.unread_buffer), size);
811+
size_t from_input = size - from_buffer;
812+
size_t from_input_actual =
813+
__furi_thread_stdin_read(thread, buffer + from_buffer, from_input, timeout);
814+
memcpy(buffer, furi_string_get_cstr(thread->input.unread_buffer), from_buffer);
815+
furi_string_right(thread->input.unread_buffer, from_buffer);
816+
817+
return from_buffer + from_input_actual;
818+
}
819+
820+
void furi_thread_stdin_unread(char* buffer, size_t size) {
821+
FuriThread* thread = furi_thread_get_current();
822+
furi_check(thread);
823+
824+
FuriString* new_buf = furi_string_alloc(); // there's no furi_string_alloc_set_strn :(
825+
furi_string_set_strn(new_buf, buffer, size);
826+
furi_string_cat(new_buf, thread->input.unread_buffer);
827+
furi_string_free(thread->input.unread_buffer);
828+
thread->input.unread_buffer = new_buf;
829+
}
830+
775831
void furi_thread_suspend(FuriThreadId thread_id) {
776832
furi_check(thread_id);
777833

0 commit comments

Comments
 (0)