1
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
53
59
62
63
64
65
66
67
68
69
70
71
81
84
87
88
89
90
91
92
93
94
99
100
101
110
113
114
115
116
117
118
119
120
129
132
133
134
135
136
137
138
143
144
150
151
167
168
173
174
175
176
177
178
179
180
190
191
198
199
200
201
202
203
204
205
206
208
209
210
211
213
214
215
216
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
237
238
239
240
242
243
244
245
247
248
249
250
251
252
253
254
255
262
263
266
267
281
282
287
288
289
290
291
292
293
294
301
302
307
308
314
315
316
317
318
319
324
325
326
331
332
337
338
339
340
343
344
345
346
347
354
355
356
362
363
364
365
366
367
368
369
370
373
376
379
382
389
390
391
/* ... */
#include <string.h>
#include <stdio.h>
#include <stdarg.h>
#include "pico.h"
#if LIB_PICO_PRINTF_PICO
#include "pico/printf.h"
#endif
#include "pico/stdio.h"
#include "pico/stdio/driver.h"
#include "pico/time.h"
#if PICO_STDOUT_MUTEX
#include "pico/mutex.h"
#endif
#if LIB_PICO_STDIO_UART
#include "pico/stdio_uart.h"
#endif
#if LIB_PICO_STDIO_USB
#include "pico/stdio_usb.h"
#endif
#if LIB_PICO_STDIO_SEMIHOSTING
#include "pico/stdio_semihosting.h"
#endif
#if LIB_PICO_STDIO_RTT
#include "pico/stdio_rtt.h"
#endif
static stdio_driver_t *drivers;
static stdio_driver_t *filter;
#if PICO_STDOUT_MUTEX
auto_init_mutex(print_mutex);
bool stdout_serialize_begin(void) {
return mutex_try_enter_block_until(&print_mutex, make_timeout_time_ms(PICO_STDIO_DEADLOCK_TIMEOUT_MS));
}{ ... }
void stdout_serialize_end(void) {
mutex_exit(&print_mutex);
}{ ... }
/* ... */#else
static bool stdout_serialize_begin(void) {
return true;
}stdout_serialize_begin (void) { ... }
static void stdout_serialize_end(void) {
}stdout_serialize_end (void) { ... }
/* ... */#endif
static void stdio_out_chars_no_crlf(stdio_driver_t *driver, const char *s, int len) {
driver->out_chars(s, len);
}{ ... }
static void stdio_out_chars_crlf(stdio_driver_t *driver, const char *s, int len) {
#if PICO_STDIO_ENABLE_CRLF_SUPPORT
if (!driver->crlf_enabled) {
driver->out_chars(s, len);
return;
}if (!driver->crlf_enabled) { ... }
int first_of_chunk = 0;
static const char crlf_str[] = {'\r', '\n'};
for (int i = 0; i < len; i++) {
bool prev_char_was_cr = i > 0 ? s[i - 1] == '\r' : driver->last_ended_with_cr;
if (s[i] == '\n' && !prev_char_was_cr) {
if (i > first_of_chunk) {
driver->out_chars(&s[first_of_chunk], i - first_of_chunk);
}if (i > first_of_chunk) { ... }
driver->out_chars(crlf_str, 2);
first_of_chunk = i + 1;
}if (s[i] == '\n' && !prev_char_was_cr) { ... }
}for (int i = 0; i < len; i++) { ... }
if (first_of_chunk < len) {
driver->out_chars(&s[first_of_chunk], len - first_of_chunk);
}if (first_of_chunk < len) { ... }
if (len > 0) {
driver->last_ended_with_cr = s[len - 1] == '\r';
}if (len > 0) { ... }
/* ... */#else
driver->out_chars(s, len);
#endif
}{ ... }
int stdio_put_string(const char *s, int len, bool newline, bool cr_translation) {
bool serialized = stdout_serialize_begin();
if (!serialized) {
#if PICO_STDIO_IGNORE_NESTED_STDOUT
return 0;
#endif
}if (!serialized) { ... }
if (len == -1) len = (int)strlen(s);
void (*out_func)(stdio_driver_t *, const char *, int) = cr_translation ? stdio_out_chars_crlf : stdio_out_chars_no_crlf;
for (stdio_driver_t *driver = drivers; driver; driver = driver->next) {
if (!driver->out_chars) continue;
if (filter && filter != driver) continue;
out_func(driver, s, len);
if (newline) {
const char c = '\n';
out_func(driver, &c, 1);
}if (newline) { ... }
}for (stdio_driver_t *driver = drivers; driver; driver = driver->next) { ... }
if (serialized) {
stdout_serialize_end();
}if (serialized) { ... }
return len;
}{ ... }
int stdio_get_until(char *buf, int len, absolute_time_t until) {
do {
for (stdio_driver_t *driver = drivers; driver; driver = driver->next) {
if (filter && filter != driver) continue;
if (driver->in_chars) {
int read = driver->in_chars(buf, len);
if (read > 0) {
return read;
}if (read > 0) { ... }
}if (driver->in_chars) { ... }
}for (stdio_driver_t *driver = drivers; driver; driver = driver->next) { ... }
if (time_reached(until)) {
return PICO_ERROR_TIMEOUT;
}if (time_reached(until)) { ... }
busy_wait_us(1);
...} while (true);
}{ ... }
int stdio_putchar_raw(int c) {
char cc = (char)c;
stdio_put_string(&cc, 1, false, false);
return c;
}{ ... }
int stdio_puts_raw(const char *s) {
int len = (int)strlen(s);
stdio_put_string(s, len, true, false);
stdio_flush();
return len;
}{ ... }
void stdio_set_driver_enabled(stdio_driver_t *driver, bool enable) {
stdio_driver_t **prev = &drivers;
while (*prev) {
if (*prev == driver) {
if (!enable) {
*prev = driver->next;
driver->next = NULL;
}if (!enable) { ... }
return;
}if (*prev == driver) { ... }
prev = &(*prev)->next;
}while (*prev) { ... }
if (enable) {
*prev = driver;
}if (enable) { ... }
}{ ... }
void stdio_flush(void) {
for (stdio_driver_t *d = drivers; d; d = d->next) {
if (d->out_flush) d->out_flush();
}for (stdio_driver_t *d = drivers; d; d = d->next) { ... }
}{ ... }
#if LIB_PICO_PRINTF_PICO
typedef struct stdio_stack_buffer {
int used;
char buf[PICO_STDIO_STACK_BUFFER_SIZE];
...} stdio_stack_buffer_t;
static void stdio_stack_buffer_flush(stdio_stack_buffer_t *buffer) {
if (buffer->used) {
for (stdio_driver_t *d = drivers; d; d = d->next) {
if (!d->out_chars) continue;
if (filter && filter != d) continue;
stdio_out_chars_crlf(d, buffer->buf, buffer->used);
}for (stdio_driver_t *d = drivers; d; d = d->next) { ... }
buffer->used = 0;
}if (buffer->used) { ... }
}{ ... }
static void stdio_buffered_printer(char c, void *arg) {
stdio_stack_buffer_t *buffer = (stdio_stack_buffer_t *)arg;
if (buffer->used == PICO_STDIO_STACK_BUFFER_SIZE) {
stdio_stack_buffer_flush(buffer);
}if (buffer->used == PICO_STDIO_STACK_BUFFER_SIZE) { ... }
buffer->buf[buffer->used++] = c;
}{ ... }
#endif/* ... */
bool stdio_init_all(void) {
bool rc = false;
#if LIB_PICO_STDIO_UART
stdio_uart_init();
rc = true;/* ... */
#endif
#if LIB_PICO_STDIO_SEMIHOSTING
stdio_semihosting_init();
rc = true;/* ... */
#endif
#if LIB_PICO_STDIO_RTT
stdio_rtt_init();
rc = true;/* ... */
#endif
#if LIB_PICO_STDIO_USB
rc |= stdio_usb_init();
#endif
return rc;
}{ ... }
bool stdio_deinit_all(void) {
stdio_flush();
bool rc = false;
#if LIB_PICO_STDIO_UART
stdio_uart_deinit();
rc = true;/* ... */
#endif
#if LIB_PICO_STDIO_SEMIHOSTING
stdio_semihosting_deinit();
rc = true;/* ... */
#endif
#if LIB_PICO_STDIO_RTT
stdio_rtt_deinit();
rc = true;/* ... */
#endif
#if LIB_PICO_STDIO_USB
rc = stdio_usb_deinit();
#endif
return rc;
}{ ... }
int stdio_getchar_timeout_us(uint32_t timeout_us) {
char buf[1];
int rc = stdio_get_until(buf, sizeof(buf), make_timeout_time_us(timeout_us));
if (rc < 0) return rc;
assert(rc);
return (uint8_t)buf[0];
}{ ... }
void stdio_filter_driver(stdio_driver_t *driver) {
filter = driver;
}{ ... }
void stdio_set_translate_crlf(stdio_driver_t *driver, bool enabled) {
#if PICO_STDIO_ENABLE_CRLF_SUPPORT
if (enabled && !driver->crlf_enabled) {
driver->last_ended_with_cr = false;
}if (enabled && !driver->crlf_enabled) { ... }
driver->crlf_enabled = enabled;/* ... */
#else
(void)driver;
(void)enabled;
panic_unsupported();/* ... */
#endif
}{ ... }
void stdio_set_chars_available_callback(void (*fn)(void*), void *param) {
for (stdio_driver_t *s = drivers; s; s = s->next) {
if (s->set_chars_available_callback) s->set_chars_available_callback(fn, param);
}for (stdio_driver_t *s = drivers; s; s = s->next) { ... }
}{ ... }
#if PICO_STDIO_SHORT_CIRCUIT_CLIB_FUNCS
#define PRIMARY_STDIO_FUNC(x) WRAPPER_FUNC(x)
#else
#define PRIMARY_STDIO_FUNC(x) stdio_ ## x
#endif
int PRIMARY_STDIO_FUNC(getchar)(void) {
char buf[1];
int len = stdio_get_until(buf, 1, at_the_end_of_time);
if (len < 0) return len;
assert(len == 1);
return (uint8_t)buf[0];
...}
int PRIMARY_STDIO_FUNC(putchar)(int c) {
char cc = (char)c;
stdio_put_string(&cc, 1, false, true);
return c;
...}
int PRIMARY_STDIO_FUNC(puts)(const char *s) {
int len = (int)strlen(s);
stdio_put_string(s, len, true, true);
stdio_flush();
return len;
...}
int REAL_FUNC(vprintf)(const char *format, va_list va);
int PRIMARY_STDIO_FUNC(vprintf)(const char *format, va_list va) {
bool serialzed = stdout_serialize_begin();
if (!serialzed) {
#if PICO_STDIO_IGNORE_NESTED_STDOUT
return 0;
#endif
}if (!serialzed) { ... }
int ret;
#if LIB_PICO_PRINTF_PICO
struct stdio_stack_buffer buffer;
buffer.used = 0;
ret = vfctprintf(stdio_buffered_printer, &buffer, format, va);
stdio_stack_buffer_flush(&buffer);
stdio_flush();/* ... */
#elif LIB_PICO_PRINTF_NONE
((void)format);
((void)va);
extern void printf_none_assert(void);
printf_none_assert();
ret = 0;/* ... */
#else
ret = REAL_FUNC(vprintf)(format, va);
#endif
if (serialzed) {
stdout_serialize_end();
}if (serialzed) { ... }
return ret;
...}
int __printflike(1, 0) PRIMARY_STDIO_FUNC(printf)(const char* format, ...)
{
va_list va;
va_start(va, format);
int ret = vprintf(format, va);
va_end(va);
return ret;
...}
#if PICO_STDIO_SHORT_CIRCUIT_CLIB_FUNCS
int stdio_getchar(void) __attribute__((alias(__XSTRING(WRAPPER_FUNC(getchar)))));
int stdio_putchar(int) __attribute__((alias(__XSTRING(WRAPPER_FUNC(putchar)))));
int stdio_puts(const char *s) __attribute__((alias(__XSTRING(WRAPPER_FUNC(puts)))));
int stdio_vprintf(const char *format, va_list va) __attribute__((alias(__XSTRING(WRAPPER_FUNC(vprintf)))));
int __printflike(1, 0) stdio_printf(const char* format, ...) __attribute__((alias(__XSTRING(WRAPPER_FUNC(printf)))));/* ... */
#else
int REAL_FUNC(getchar)(void);
int REAL_FUNC(putchar)(int);
int REAL_FUNC(puts)(const char *s);
int __printflike(1, 0) REAL_FUNC(printf)(const char* format, ...);
int WRAPPER_FUNC(getchar)(void) {
return REAL_FUNC(getchar)();
...}
int WRAPPER_FUNC(putchar)(int c) {
return REAL_FUNC(putchar)(c);
...}
int WRAPPER_FUNC(puts)(const char *s) {
return REAL_FUNC(puts)(s);
...}
int WRAPPER_FUNC(vprintf)(const char *format, va_list va) {
return REAL_FUNC(vprintf)(format, va);
...}
int __printflike(1, 0) WRAPPER_FUNC(printf)(const char* format, ...) {
va_list va;
va_start(va, format);
int ret = REAL_FUNC(vprintf)(format, va);
va_end(va);
return ret;
...}/* ... */
#endif