Skip to content

GlobalDlmalloc reentrancy on panics #43

@hanna-kruppe

Description

@hanna-kruppe

While looking through the global allocator implementations, I've noticed that the code doesn't explicitly guard against reentrancy when acquiring the lock. If the allocator code panics for any reasons (most likely debug assertions) then the #[panic_handler] or panic hook will run and potentially allocate recursively. This would be problematic because:

  1. It creates a second &mut Dlmalloc to the global DLMALLOC instance while another mutable reference to it exists in another stack frame. Although you could argue that it's fine because execution will be aborted so the first one is no longer live.
  2. More importantly, the allocator logic most likely isn't written to support reentrancy, and even if it was, the panic happened because some invariant was broken so further allocations can't be serviced reliably.

Looking through the existing platform-specific implementations of the global lock:

  • On Unix, a normal ("fast") pthread mutex is used. According to the man page this will deadlock reliably if the same thread tries to acquire the mutex again, which is not ideal but safe.
  • On windows, an SRWLOCK is used in exclusive mode, which doesn't support recursive acquisition but the documentation implies that it will reliably deadlock in exclusive mode (involving shared mode is a crapshoot). Again, not great but safe.
  • Xous doesn't implement this lock because it doesn't have global allocator support.
  • On wasm, there's no lock at all.

The following program, compiled to wasm32-unknown-unknown (Rust 1.81, debug mode, crate-type cdylib), demonstrates that reentrancy really happens. The bogus dealloc call causes an assertion failure and then the use of format! in the panic hook allocates a new string.

use std::alloc::{self, Layout};

extern "C" {
    fn exfiltrate_str(data: *const u8, len: usize);
}

#[no_mangle]
pub extern fn entry_point() {
    std::panic::set_hook(Box::new(|info| {
        let msg = format!("panic at {:?}", info.location());
        unsafe { exfiltrate_str(msg.as_ptr(), msg.len()); }
    }));
    let layout = Layout::from_size_align(32, 8).unwrap();
    unsafe {
        alloc::dealloc(64 as *mut u8, layout);
    }
}

I cobbled together a wasmtime embedder that provides a definition for exfiltrate_str and the output is: panic at Some(Location { file: "/rust/deps/dlmalloc-0.2.6/src/dlmalloc.rs", line: 1192, col: 9 }).

How important is this? Probably not very much, at least as long as it's constrained to wasm. Triggering an assertion in the allocator almost certainly means you already have UB somewhere. If the reentrancy causes a crash during execution of the panic hook/handler instead of right afterwards, at worst it makes debugging a bit harder. But since I already dug into it, I figured I might as well report it.


Finally, after writing this up, I realized that you can cause similar issues on non-wasm targets by compiling with panic=unwind and catching the panic. While this could cause more damage, properly supporting other targets explicitly isn't a goal of the crate so it's less interesting.

Metadata

Metadata

Assignees

No one assigned

    Labels

    No labels
    No labels

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions