Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[libc-0.2] posix_spawn_file_actions_t cannot be used on Linux after #3602 #3608

Closed
japaric opened this issue Mar 4, 2024 · 10 comments · Fixed by #3809
Closed

[libc-0.2] posix_spawn_file_actions_t cannot be used on Linux after #3602 #3608

japaric opened this issue Mar 4, 2024 · 10 comments · Fixed by #3809
Labels
C-bug Category: bug

Comments

@japaric
Copy link
Member

japaric commented Mar 4, 2024

We found an issue when updating ferrocene's libc submodule (ferrocene/ferrocene#356) to revision 947a185 . the only change included in our libc update was PR #3602 .

when building stage 2 of libstd to x86_64-linux the build failed to execute the build script of the quote crate with the following error:

thread 'main' panicked at library/std/src/sys_common/process.rs:155:17:
called Result::unwrap() on an Err value: Os { code: 9, kind: Uncategorized, message: "Bad file descriptor" } 

where the mentioned line corresponds to this revision.

Tracking down the error led us to this usage of the posix_spawn API:

let mut file_actions = MaybeUninit::uninit();
cvt_nz(libc::posix_spawn_file_actions_init(file_actions.as_mut_ptr()))?;

This code allocates a posix_spawn_file_actions_init type on the stack. before #3602, the file_actions stack variable had the correct size and alignment but after #3602, it now has a size and alignment of *mut c_void (8 bytes on x86_64) and that results in UB when posix_spawn_file_actions_init is called.

GLIBC implements posix_spawn_file_actions_init as a memset operation that zeroes the struct. In Rust syntax, that would be more or less this code:

#[repr(C)]
struct posix_spawn_file_actions_t {
    // fields and padding here
}

unsafe fn posix_spawn_file_actions_init(actions: *mut posix_spawn_file_actions_t) -> c_int {
    ptr::write_bytes(actions, 0, 1);
    0
}

Bionic implements posix_spawn_file_actions_init differently. It uses a heap allocation as indirection. The Rust version of the bionic implementation looks roughly like this:

#[repr(C)]
struct __actual_posix_spawn_file_actions_t {
    // fields and padding here
}

type posix_spawn_file_actions_t = *mut c_void;

unsafe fn posix_spawn_file_actions_init(actions: *mut posix_spawn_file_actions_t) -> c_int {
    let mut alloc = Box::new(MaybeUninit::<__actual_posix_spawn_file_actions_t>::uninit());
    ptr::write_bytes(alloc.as_mut_ptr(), 0, 1);
    *actions = Box::into_raw(alloc).cast();
    0
}

This usage of posix_spawn_file_actions_t in libstd:

let mut file_actions = MaybeUninit::uninit();
cvt_nz(libc::posix_spawn_file_actions_init(file_actions.as_mut_ptr()))?;

is compatible with both GLIBC and Bionic but the libc crate needs to provide the correct size and alignment on Linux.

So, the conditional code should produce this on Linux

pub struct posix_spawn_file_actions_t {
    __allocated: ::c_int,
    __used: ::c_int,
    __actions: *mut ::c_int,
    __pad: [::c_int; 16],
}

but this on Android

pub type posix_spawn_file_actions_t = *mut ::c_void;

All this probably also applies to the posix_spawnattr_t type but I have yet to look into the details of the GLIBC code around that type.

I'll send up a fix PR after I have done some more testing.

@madsmtm
Copy link
Contributor

madsmtm commented Apr 30, 2024

@JohnTitor: This is blocking updating libc in the standard library, specifically because Cargo build scripts seem to break.

@madsmtm
Copy link
Contributor

madsmtm commented May 5, 2024

The problematic PR has been reverted in #3678.

@JohnTitor
Copy link
Member

JohnTitor commented May 5, 2024

Note that #3608 exists since 0.2.153, #3678 just fixes #3677. We still have to merge #3609.

@silence-coding
Copy link

silence-coding commented May 6, 2024

After wasting a day, I also found out that posix_spawnattr_init is fine in 0.2.153, but 0.2.154 causes Segmentation fault......

@madsmtm
Copy link
Contributor

madsmtm commented May 7, 2024

Note that #3608 exists since 0.2.153, #3678 just fixes #3677. We still have to merge #3609.

Hmm, I'm confused? The problem described in this issue is not present in v0.2.153, and the issue is resolved by #3678?

And #3677 is just a duplicate of this?

Or do you mean that we need to merge #3683? (EDIT: #3690)

@coolbluewater
Copy link

@madsmtm it looks like #3683 supersedes #3609. Additionally all checks have passed for #3683, so that needs to be merged asap.

@ngg
Copy link

ngg commented May 13, 2024

#3690 supersedes #3683 now. What's the blocker here?

@cuviper
Copy link
Member

cuviper commented May 16, 2024

I have yanked 0.2.154 with my T-libs crate access, but I don't think I have any permission in the repo to help get a new version out.

@newpavlov
Copy link
Contributor

It would be nice to get v0.2.155 published as soon as possible. Yanking of v0.2.154 breaks the latest version of getrandom, see rust-random/getrandom#423.

@madsmtm
Copy link
Contributor

madsmtm commented May 17, 2024

#3690 has been merged, and 0.2.155 has been released, so I believe this can be closed.

bors added a commit to rust-lang-ci/rust that referenced this issue May 20, 2024
Update libc to 0.2.155

Motivation: To fix `-Zbuild-std` / Xargo for visionOS targets.

EDIT: Blocked on ~rust-lang/libc#3608 / rust-lang/libc#3609 ~rust-lang/libc#3682 and rust-lang/libc#3690 No longer blocked.
RalfJung pushed a commit to RalfJung/miri that referenced this issue May 22, 2024
Update libc to 0.2.155

Motivation: To fix `-Zbuild-std` / Xargo for visionOS targets.

EDIT: Blocked on ~rust-lang/libc#3608 / rust-lang/libc#3609 ~rust-lang/libc#3682 and rust-lang/libc#3690 No longer blocked.
flip1995 pushed a commit to flip1995/rust-clippy that referenced this issue May 24, 2024
Update libc to 0.2.155

Motivation: To fix `-Zbuild-std` / Xargo for visionOS targets.

EDIT: Blocked on ~rust-lang/libc#3608 / rust-lang/libc#3609 ~rust-lang/libc#3682 and rust-lang/libc#3690 No longer blocked.
bors added a commit to rust-lang/rust-analyzer that referenced this issue Jun 20, 2024
Update libc to 0.2.155

Motivation: To fix `-Zbuild-std` / Xargo for visionOS targets.

EDIT: Blocked on ~rust-lang/libc#3608 / rust-lang/libc#3609 ~rust-lang/libc#3682 and rust-lang/libc#3690 No longer blocked.
tgross35 pushed a commit to tgross35/rust-libc that referenced this issue Aug 12, 2024
Fixes: rust-lang#3709
Fixes: rust-lang#3608
Fixes: rust-lang#3677

Co-authored-by: Jorge Aparicio <jorge.aparicio@ferrous-systems.com>

[resolved conflict, update commit message - Trevor]
(apply <rust-lang#3690> to `main`)
(cherry picked from commit b2b2fd7)

Signed-off-by: Trevor Gross <tmgross@umich.edu>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
C-bug Category: bug
Projects
None yet
8 participants