mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
change(newlib): update newlib according to new internal structures
This commit is contained in:
parent
c94d443a1b
commit
a43c509d4b
@ -305,28 +305,10 @@ static void do_core_init(void)
|
||||
#endif
|
||||
|
||||
#if defined(CONFIG_VFS_SUPPORT_IO) && !defined(CONFIG_ESP_CONSOLE_NONE)
|
||||
const static char *default_stdio_dev = "/dev/console/";
|
||||
esp_reent_init(_GLOBAL_REENT);
|
||||
_GLOBAL_REENT->_stdin = fopen(default_stdio_dev, "r");
|
||||
_GLOBAL_REENT->_stdout = fopen(default_stdio_dev, "w");
|
||||
_GLOBAL_REENT->_stderr = fopen(default_stdio_dev, "w");
|
||||
#if ESP_ROM_NEEDS_SWSETUP_WORKAROUND
|
||||
/*
|
||||
- This workaround for printf functions using 32-bit time_t after the 64-bit time_t upgrade
|
||||
- The 32-bit time_t usage is triggered through ROM Newlib functions printf related functions calling __swsetup_r() on
|
||||
the first call to a particular file pointer (i.e., stdin, stdout, stderr)
|
||||
- Thus, we call the toolchain version of __swsetup_r() now (before any printf calls are made) to setup all of the
|
||||
file pointers. Thus, the ROM newlib code will never call the ROM version of __swsetup_r().
|
||||
- See IDFGH-7728 for more details
|
||||
*/
|
||||
extern int __swsetup_r(struct _reent *, FILE *);
|
||||
__swsetup_r(_GLOBAL_REENT, _GLOBAL_REENT->_stdout);
|
||||
__swsetup_r(_GLOBAL_REENT, _GLOBAL_REENT->_stderr);
|
||||
__swsetup_r(_GLOBAL_REENT, _GLOBAL_REENT->_stdin);
|
||||
#endif // ESP_ROM_NEEDS_SWSETUP_WORKAROUND
|
||||
#else // defined(CONFIG_VFS_SUPPORT_IO) && !defined(CONFIG_ESP_CONSOLE_NONE)
|
||||
_REENT_SMALL_CHECK_INIT(_GLOBAL_REENT);
|
||||
#endif // defined(CONFIG_VFS_SUPPORT_IO) && !defined(CONFIG_ESP_CONSOLE_NONE)
|
||||
esp_newlib_init_global_stdio(ESP_VFS_DEV_CONSOLE);
|
||||
#else
|
||||
esp_newlib_init_global_stdio(NULL);
|
||||
#endif
|
||||
|
||||
esp_err_t err __attribute__((unused));
|
||||
|
||||
|
@ -39,8 +39,6 @@
|
||||
#include "esp32p4/rom/libc_stubs.h"
|
||||
#endif
|
||||
|
||||
static struct _reent s_reent;
|
||||
|
||||
extern int _printf_float(struct _reent *rptr,
|
||||
void *pdata,
|
||||
FILE * fp,
|
||||
@ -58,6 +56,21 @@ static void raise_r_stub(struct _reent *rptr)
|
||||
_raise_r(rptr, 0);
|
||||
}
|
||||
|
||||
static void esp_cleanup_r (struct _reent *rptr)
|
||||
{
|
||||
if (_REENT_STDIN(rptr) != _REENT_STDIN(_GLOBAL_REENT)) {
|
||||
_fclose_r(rptr, _REENT_STDIN(rptr));
|
||||
}
|
||||
|
||||
if (_REENT_STDOUT(rptr) != _REENT_STDOUT(_GLOBAL_REENT)) {
|
||||
_fclose_r(rptr, _REENT_STDOUT(rptr));
|
||||
}
|
||||
|
||||
if (_REENT_STDERR(rptr) !=_REENT_STDERR(_GLOBAL_REENT)) {
|
||||
_fclose_r(rptr, _REENT_STDERR(rptr));
|
||||
}
|
||||
}
|
||||
|
||||
static struct syscall_stub_table s_stub_table = {
|
||||
.__getreent = &__getreent,
|
||||
._malloc_r = &_malloc_r,
|
||||
@ -120,14 +133,14 @@ static struct syscall_stub_table s_stub_table = {
|
||||
*/
|
||||
.__assert_func = __assert_func,
|
||||
|
||||
/* We don't expect either ROM code or IDF to ever call __sinit, so it's implemented as abort() for now.
|
||||
/* We don't expect either ROM code to ever call __sinit, so it's implemented as abort() for now.
|
||||
|
||||
esp_reent_init() does this job inside IDF.
|
||||
|
||||
Kept in the syscall table in case we find a need for it later.
|
||||
__sinit may be called in IDF side only if /dev/console used as input/output. It called only
|
||||
once for _GLOBAL_REENT. Then reuse std file pointers from _GLOBAL_REENT in another reents.
|
||||
See esp_newlib_init() and esp_reent_init() for details.
|
||||
*/
|
||||
.__sinit = (void *)abort,
|
||||
._cleanup_r = &_cleanup_r,
|
||||
._cleanup_r = &esp_cleanup_r,
|
||||
#endif
|
||||
};
|
||||
|
||||
@ -141,7 +154,17 @@ void esp_newlib_init(void)
|
||||
syscall_table_ptr = &s_stub_table;
|
||||
#endif
|
||||
|
||||
#if __NEWLIB__ > 4 || ( __NEWLIB__ == 4 && __NEWLIB_MINOR__ > 1 ) /* TODO: IDF-8134 */
|
||||
memset(&__sglue, 0, sizeof(__sglue));
|
||||
_global_impure_ptr = _GLOBAL_REENT;
|
||||
#else
|
||||
static struct _reent s_reent;
|
||||
_GLOBAL_REENT = &s_reent;
|
||||
#endif
|
||||
|
||||
/* Ensure that the initialization of sfp is prevented until esp_newlib_init_global_stdio() is explicitly invoked. */
|
||||
_GLOBAL_REENT->__cleanup = esp_cleanup_r;
|
||||
_REENT_SDIDINIT(_GLOBAL_REENT) = 1;
|
||||
|
||||
environ = malloc(sizeof(char*));
|
||||
if (environ == 0) {
|
||||
@ -154,3 +177,33 @@ void esp_newlib_init(void)
|
||||
}
|
||||
|
||||
void esp_setup_newlib_syscalls(void) __attribute__((alias("esp_newlib_init")));
|
||||
|
||||
void esp_newlib_init_global_stdio(const char *stdio_dev)
|
||||
{
|
||||
if (stdio_dev == NULL)
|
||||
{
|
||||
_GLOBAL_REENT->__cleanup = NULL;
|
||||
_REENT_SDIDINIT(_GLOBAL_REENT) = 0;
|
||||
__sinit(_GLOBAL_REENT);
|
||||
_GLOBAL_REENT->__cleanup = esp_cleanup_r;
|
||||
_REENT_SDIDINIT(_GLOBAL_REENT) = 1;
|
||||
} else {
|
||||
_REENT_STDIN(_GLOBAL_REENT) = fopen(stdio_dev, "r");
|
||||
_REENT_STDOUT(_GLOBAL_REENT) = fopen(stdio_dev, "w");
|
||||
_REENT_STDERR(_GLOBAL_REENT) = fopen(stdio_dev, "w");
|
||||
#if ESP_ROM_NEEDS_SWSETUP_WORKAROUND
|
||||
/*
|
||||
- This workaround for printf functions using 32-bit time_t after the 64-bit time_t upgrade
|
||||
- The 32-bit time_t usage is triggered through ROM Newlib functions printf related functions calling __swsetup_r() on
|
||||
the first call to a particular file pointer (i.e., stdin, stdout, stderr)
|
||||
- Thus, we call the toolchain version of __swsetup_r() now (before any printf calls are made) to setup all of the
|
||||
file pointers. Thus, the ROM newlib code will never call the ROM version of __swsetup_r().
|
||||
- See IDFGH-7728 for more details
|
||||
*/
|
||||
extern int __swsetup_r (struct _reent *, FILE *);
|
||||
__swsetup_r(_GLOBAL_REENT, _REENT_STDIN(_GLOBAL_REENT));
|
||||
__swsetup_r(_GLOBAL_REENT, _REENT_STDOUT(_GLOBAL_REENT));
|
||||
__swsetup_r(_GLOBAL_REENT, _REENT_STDERR(_GLOBAL_REENT));
|
||||
#endif /* ESP_ROM_NEEDS_SWSETUP_WORKAROUND */
|
||||
}
|
||||
}
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* SPDX-FileCopyrightText: 2015-2022 Espressif Systems (Shanghai) CO LTD
|
||||
* SPDX-FileCopyrightText: 2015-2023 Espressif Systems (Shanghai) CO LTD
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
@ -25,6 +25,17 @@ void esp_newlib_time_init(void);
|
||||
*/
|
||||
void esp_reent_init(struct _reent* r);
|
||||
|
||||
/**
|
||||
* Postponed _GLOBAL_REENT stdio FPs initialization.
|
||||
*
|
||||
* Can not be a part of esp_reent_init() because stdio device may not initialized yet.
|
||||
*
|
||||
* Called from startup code and FreeRTOS, not intended to be called from
|
||||
* application code.
|
||||
*
|
||||
*/
|
||||
void esp_newlib_init_global_stdio(const char* stdio_dev);
|
||||
|
||||
/**
|
||||
* Clean up some of lazily allocated buffers in REENT structures.
|
||||
*/
|
||||
|
@ -1,22 +1,47 @@
|
||||
/*
|
||||
* SPDX-FileCopyrightText: 2020-2022 Espressif Systems (Shanghai) CO LTD
|
||||
* SPDX-FileCopyrightText: 2020-2023 Espressif Systems (Shanghai) CO LTD
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#if __NEWLIB__ > 4 || ( __NEWLIB__ == 4 && __NEWLIB_MINOR__ > 1 ) /* TODO: IDF-8134 */
|
||||
#define _REENT_BACKWARD_BINARY_COMPAT
|
||||
#define _REENT_SDIDINIT(_ptr) ((_ptr)->_reserved_0)
|
||||
#define _REENT_SGLUE(_ptr) (__sglue)
|
||||
#else
|
||||
#define _REENT_CLEANUP(_ptr) ((_ptr)->__cleanup)
|
||||
#define _REENT_STDIN(_ptr) ((_ptr)->_stdin)
|
||||
#define _REENT_STDOUT(_ptr) ((_ptr)->_stdout)
|
||||
#define _REENT_STDERR(_ptr) ((_ptr)->_stderr)
|
||||
#define _REENT_SDIDINIT(_ptr) ((_ptr)->__sdidinit)
|
||||
#define _REENT_SGLUE(_ptr) ((_ptr)->__sglue)
|
||||
#endif
|
||||
|
||||
#include_next<sys/reent.h>
|
||||
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
#if __NEWLIB__ > 4 || ( __NEWLIB__ == 4 && __NEWLIB_MINOR__ > 1 ) /* TODO: IDF-8134 */
|
||||
|
||||
extern void __sinit (struct _reent *);
|
||||
|
||||
extern struct _glue __sglue;
|
||||
extern struct _reent * _global_impure_ptr;
|
||||
|
||||
#else /* __NEWLIB__ > 4 || ( __NEWLIB__ == 4 && __NEWLIB_MINOR__ > 1 ) */
|
||||
|
||||
/* This function is not part of the newlib API, it is defined in libc/stdio/local.h
|
||||
* There is no nice way to get __cleanup member populated while avoiding __sinit,
|
||||
* so extern declaration is used here.
|
||||
*/
|
||||
extern void _cleanup_r(struct _reent* r);
|
||||
extern void _cleanup_r(struct _reent *);
|
||||
|
||||
#endif /* __NEWLIB__ > 4 || ( __NEWLIB__ == 4 && __NEWLIB_MINOR__ > 1 ) */
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* SPDX-FileCopyrightText: 2015-2022 Espressif Systems (Shanghai) CO LTD
|
||||
* SPDX-FileCopyrightText: 2015-2023 Espressif Systems (Shanghai) CO LTD
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*/
|
||||
@ -22,14 +22,11 @@
|
||||
void IRAM_ATTR esp_reent_init(struct _reent* r)
|
||||
{
|
||||
memset(r, 0, sizeof(*r));
|
||||
r->_stdout = _GLOBAL_REENT->_stdout;
|
||||
r->_stderr = _GLOBAL_REENT->_stderr;
|
||||
r->_stdin = _GLOBAL_REENT->_stdin;
|
||||
r->__cleanup = &_cleanup_r;
|
||||
r->__sdidinit = 1;
|
||||
r->__sglue._next = NULL;
|
||||
r->__sglue._niobs = 0;
|
||||
r->__sglue._iobs = NULL;
|
||||
_REENT_STDIN(r) = _REENT_STDIN(_GLOBAL_REENT);
|
||||
_REENT_STDOUT(r) = _REENT_STDOUT(_GLOBAL_REENT);
|
||||
_REENT_STDERR(r) = _REENT_STDERR(_GLOBAL_REENT);
|
||||
_REENT_CLEANUP(r) = _REENT_CLEANUP(_GLOBAL_REENT);
|
||||
_REENT_SDIDINIT(r) = _REENT_SDIDINIT(_GLOBAL_REENT);
|
||||
}
|
||||
|
||||
/* only declared in private stdio header file, local.h */
|
||||
@ -39,26 +36,20 @@ extern void __sfp_lock_release(void);
|
||||
void esp_reent_cleanup(void)
|
||||
{
|
||||
struct _reent* r = __getreent();
|
||||
/* Clean up storage used by mprec functions */
|
||||
if (r->_mp) {
|
||||
if (_REENT_MP_FREELIST(r)) {
|
||||
for (unsigned int i = 0; i < _Kmax; ++i) {
|
||||
struct _Bigint *cur, *next;
|
||||
next = _REENT_MP_FREELIST(r)[i];
|
||||
while (next) {
|
||||
cur = next;
|
||||
next = next->_next;
|
||||
free(cur);
|
||||
}
|
||||
}
|
||||
}
|
||||
free(_REENT_MP_FREELIST(r));
|
||||
free(_REENT_MP_RESULT(r));
|
||||
}
|
||||
_reclaim_reent(r);
|
||||
|
||||
r->_emergency = NULL;
|
||||
r->_mp = NULL;
|
||||
r->_r48 = NULL;
|
||||
r->_localtime_buf = NULL;
|
||||
r->_asctime_buf = NULL;
|
||||
r->_signal_buf = NULL;
|
||||
r->_misc = NULL;
|
||||
r->_cvtbuf = NULL;
|
||||
|
||||
/* Clean up "glue" (lazily-allocated FILE objects) */
|
||||
struct _glue* prev = &_GLOBAL_REENT->__sglue;
|
||||
for (struct _glue* cur = _GLOBAL_REENT->__sglue._next; cur != NULL;) {
|
||||
struct _glue* prev = &_REENT_SGLUE(_GLOBAL_REENT);
|
||||
for (struct _glue* cur = _REENT_SGLUE(_GLOBAL_REENT)._next; cur != NULL;) {
|
||||
if (cur->_niobs == 0) {
|
||||
cur = cur->_next;
|
||||
continue;
|
||||
@ -81,14 +72,4 @@ void esp_reent_cleanup(void)
|
||||
free(cur);
|
||||
cur = next;
|
||||
}
|
||||
|
||||
/* Clean up various other buffers */
|
||||
free(r->_mp);
|
||||
r->_mp = NULL;
|
||||
free(r->_r48);
|
||||
r->_r48 = NULL;
|
||||
free(r->_localtime_buf);
|
||||
r->_localtime_buf = NULL;
|
||||
free(r->_asctime_buf);
|
||||
r->_asctime_buf = NULL;
|
||||
}
|
||||
|
@ -8,6 +8,8 @@
|
||||
|
||||
#include "esp_err.h"
|
||||
|
||||
#define ESP_VFS_DEV_CONSOLE "/dev/console"
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
@ -188,7 +188,7 @@ static const esp_vfs_t vfs = {
|
||||
|
||||
esp_err_t esp_vfs_dev_console_register(void)
|
||||
{
|
||||
return esp_vfs_register("/dev/console", &vfs, NULL);
|
||||
return esp_vfs_register(ESP_VFS_DEV_CONSOLE, &vfs, NULL);
|
||||
}
|
||||
|
||||
esp_err_t esp_vfs_console_register(void)
|
||||
|
Loading…
x
Reference in New Issue
Block a user