Skip to content

refactor: fix definition of time_t and off_t in Newlib#5132

Open
dybucc wants to merge 2 commits into
rust-lang:mainfrom
dybucc:newlib-fix-time_t
Open

refactor: fix definition of time_t and off_t in Newlib#5132
dybucc wants to merge 2 commits into
rust-lang:mainfrom
dybucc:newlib-fix-time_t

Conversation

@dybucc

@dybucc dybucc commented Jun 1, 2026

Copy link
Copy Markdown
Contributor

Description

This PR adresses bit-width representation issues under the newlib target environment concerned with both the time_t and off_t types.

In newlib, time_t is conditionally defined as being a c_longlong under the horizon and espidf target operating systems. Otherwise, it always defaults to using a 32-bit signed integer. off_t is always defined as a 64-bit signed integer except in espidf and in vita, where they are defined as c_long and c_int, respectively.

This is possibly incorrect. The newlib upstream repo maintained by Cygwin defines time_t conditionally as being either one of a c_long or a signed integer type with at least 64 bits precision. The definition will always default to the latter unless one issues --enable-newlib-long-time_t to the configure build script, or if the host system is known to have c_longs larger than 32 bits. In much the same vein, the espressif upstream repo for their fork of newlib uses the exact same definition, and so does the devkitARM shipped for development on horizon. Support for espidf with 32-bit time_t is likely provided for compatibility purposes following the analogous compatibility shims espressif has set in place in their upstream repos [1] [2].

Upstream also defines off_t in an ever so slightly different way to the declaration in the libc crate. The definition in the "official" repo maintained by the Cygwin developers will always provide a 64-bit value. The forks for vita, espressif's esp-32, devkitA64 (and by extension horizon in the Nintendo Switch but not in the Nintendo 3DS,) and RTEMS, don't define this type in the machine/_types.h header file. This header file is the one where the first definition for the type appears, and is the one that takes priority if it exists. A macro checks that in sys/_types.h, and will otherwise define the type in terms of a c_long. This only happens in the afore mentioned platforms, where the definition of off_t is removed from machine/_types.h. Currently, the exposed type is not completely wrong, but is quite defintely wrong in vita.

This patch changes the time_t definition to fit a 64-bit signed integer unless compiling for the afore mentioned environment (the one for time_t) with the existing cfg(espidf_time32) enabled. Note this option is "documented" as being meant for end users of the libc crate to enable, but no information on it is provided anywhere in all of the libc crate. It seems like the only users of that cfg are the projects in the esp-rs org. Some of the other targets using newlib have been verified to also follow this definition for time_t (sources below.) Among those checked are the ones for which we have custom modules at the end of the newlib/mod.rs module. Notably, no architecture-specific definitions were found for aarch64.

A simple test has been ran in all platforms except vita and aarch64 (which here means the Nintendo Switch with devkitPro's devkitA64) for both time_t and off_t. The former used a two-element array of time_t values and ensured that a write through an FFI routine (like time) only overwrote one of the values and not both. The latter required a bit more intricate testing, as no C standard library routine takes a mutable pointer to an off_t. All of the affected platforms required emulation and to some extent, supported embedding a virtual file system to the compiled binary. This was used to embed files with different sizes and use stream seeking operations with values for off_t both larger and smaller than 32 bits.

Testing was not possible in vita because the emulator did not support the type of homebrew package that the Rust toolchain for vita builds. In horizon for the Nintendo Switch under aarch64 with devkitA64, testing was neither possible for the same reason as in vita.

It would also be great to know whether the libc crate considers horizon to be the OS for both the Nintendo Switch and the Nintendo 3DS targets, or only for the Nintendo 3DS target. It just so happens that the newlib module has submodules for all three of them with a note that mentions testing only with devkitARM. Using cfg's for different architecures may not be the right choice, as there could be other systems whose ISA matched aarch64 with changes to their newlib fork that differ from those in devkitA64 (or any other devkitPro SDK, for that matter.)

Sources

Checklist

  • Relevant tests in libc-test/semver have been updated
  • No placeholder or unstable values like *LAST or *MAX are included (see #3131)
  • Tested locally (cd libc-test && cargo test --target mytarget); especially relevant for platforms that may not be checked in CI

@dybucc

dybucc commented Jun 2, 2026

Copy link
Copy Markdown
Contributor Author

CI seems to be failing for reasons unrelated to the changes introduced in the patch. A rerun should
do it.

@dybucc dybucc changed the title refactor: fix definition of time_t in Newlib refactor: fix definition of time_t and off_t in Newlib Jun 3, 2026
@dybucc dybucc force-pushed the newlib-fix-time_t branch 4 times, most recently from 7152f77 to e821035 Compare June 7, 2026 16:34
@dybucc dybucc marked this pull request as ready for review June 7, 2026 16:35
@dybucc dybucc force-pushed the newlib-fix-time_t branch from e821035 to de697e2 Compare June 9, 2026 07:11
@rustbot

This comment has been minimized.

@dybucc

dybucc commented Jun 9, 2026

Copy link
Copy Markdown
Contributor Author

CI actually passes. There seems to be an issue with a glob import that is not used, but this has not
been changed in the patch (it's not even part of it, for that matter.) For some reason, rebasing
onto main with dependabot updates has ended up with a warning across all of my open PRs due to
that one (now apparently unused) import.

dybucc added 2 commits June 15, 2026 17:10
At present, the newlib module uses a faulty definition of the `time_t`
type that falls back to a 32-bti signed integer if the target does not
match either one of the `horizon` operating system or the espressif
ESP-IDF with the 32-bit compatibility shim for `time_t` values.

In all current upstream repos forking newlib, the definition follows a
default where `time_t` is defined as a 64-bit type. It only falls back
to a 32-bit type if a non-default build-time option is issues to the
`configure` script, or otherwise if the host machine is detected to
support `c_long`s larger than 32 bits.
The newlib definition is either one of a signed 64 bit integer or a C
`long`. The former is only present when `_off_t` (note the starting
underscore) is defined under `machine/_types.h`. `off_t` is defined as a
`c_long` in all other cases. There's some special casing for Cygwin, but
that is unimportant in the `libc` crate because the Cygwin module is not
a child module of the newlib module.

Except in `vita`, espressif's platforms, devkitA64 for `aarch64`, and
RTEMS, this type is always defined under `machine/_types.h`. Otherwise,
it's defined as a `c_long` under `sys/_types.h`.

More details can be found in the accompanying PR.
@dybucc dybucc force-pushed the newlib-fix-time_t branch from de697e2 to 0e027c7 Compare June 15, 2026 15:11
@rustbot

rustbot commented Jun 15, 2026

Copy link
Copy Markdown
Collaborator

This PR was rebased onto a different main commit. Here's a range-diff highlighting what actually changed.

Rebasing is a normal part of keeping PRs up to date, so no action is needed—this note is just to help reviewers.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Projects

None yet

Development

Successfully merging this pull request may close these issues.

2 participants