23
23
24
24
#define THREAD_MAX_STACK_SIZE (UINT16_MAX * sizeof(StackType_t))
25
25
26
- typedef struct FuriThreadStdout FuriThreadStdout ;
27
-
28
- struct FuriThreadStdout {
26
+ typedef struct {
29
27
FuriThreadStdoutWriteCallback write_callback ;
30
28
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 ;
32
37
33
38
struct FuriThread {
34
39
StaticTask_t container ;
@@ -55,6 +60,7 @@ struct FuriThread {
55
60
size_t heap_size ;
56
61
57
62
FuriThreadStdout output ;
63
+ FuriThreadStdin input ;
58
64
59
65
// Keep all non-alignable byte types in one place,
60
66
// this ensures that the size of this structure is minimal
@@ -136,6 +142,7 @@ static void furi_thread_body(void* context) {
136
142
137
143
static void furi_thread_init_common (FuriThread * thread ) {
138
144
thread -> output .buffer = furi_string_alloc ();
145
+ thread -> input .unread_buffer = furi_string_alloc ();
139
146
140
147
FuriThread * parent = NULL ;
141
148
if (xTaskGetSchedulerState () != taskSCHEDULER_NOT_STARTED ) {
@@ -245,6 +252,7 @@ void furi_thread_free(FuriThread* thread) {
245
252
}
246
253
247
254
furi_string_free (thread -> output .buffer );
255
+ furi_string_free (thread -> input .unread_buffer );
248
256
free (thread );
249
257
}
250
258
@@ -710,13 +718,22 @@ uint32_t furi_thread_get_stack_space(FuriThreadId thread_id) {
710
718
711
719
static size_t __furi_thread_stdout_write (FuriThread * thread , const char * data , size_t size ) {
712
720
if (thread -> output .write_callback != NULL ) {
713
- thread -> output .write_callback (data , size );
721
+ thread -> output .write_callback (data , size , thread -> output . context );
714
722
} else {
715
723
furi_log_tx ((const uint8_t * )data , size );
716
724
}
717
725
return size ;
718
726
}
719
727
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
+
720
737
static int32_t __furi_thread_stdout_flush (FuriThread * thread ) {
721
738
FuriString * buffer = thread -> output .buffer ;
722
739
size_t size = furi_string_size (buffer );
@@ -727,17 +744,31 @@ static int32_t __furi_thread_stdout_flush(FuriThread* thread) {
727
744
return 0 ;
728
745
}
729
746
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 ) {
731
760
FuriThread * thread = furi_thread_get_current ();
732
761
furi_check (thread );
733
762
__furi_thread_stdout_flush (thread );
734
763
thread -> output .write_callback = callback ;
764
+ thread -> output .context = context ;
735
765
}
736
766
737
- FuriThreadStdoutWriteCallback furi_thread_get_stdout_callback ( void ) {
767
+ void furi_thread_set_stdin_callback ( FuriThreadStdinReadCallback callback , void * context ) {
738
768
FuriThread * thread = furi_thread_get_current ();
739
769
furi_check (thread );
740
- return thread -> output .write_callback ;
770
+ thread -> input .read_callback = callback ;
771
+ thread -> input .context = context ;
741
772
}
742
773
743
774
size_t furi_thread_stdout_write (const char * data , size_t size ) {
@@ -772,6 +803,31 @@ int32_t furi_thread_stdout_flush(void) {
772
803
return __furi_thread_stdout_flush (thread );
773
804
}
774
805
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
+
775
831
void furi_thread_suspend (FuriThreadId thread_id ) {
776
832
furi_check (thread_id );
777
833
0 commit comments