Stream: git-wasmtime

Topic: wasmtime / issue #5005 Panic in `write_plt_entry_bytes:278`


view this post on Zulip Wasmtime GitHub notifications bot (Oct 04 2022 at 10:46):

mokhaled2992 edited issue #5005:

I'm getting a panic and the stacktrace points at the following line. This happens at the very early stages when I initialize my JIT and try to declare some functions. Any hints on how to solve this?

    unsafe fn write_plt_entry_bytes(plt_ptr: *mut [u8; 16], got_ptr: NonNull<AtomicPtr<u8>>) {
        ...................
        plt_val[2..6].copy_from_slice(&i32::to_ne_bytes(i32::try_from(what - at).unwrap()));
        ...................
    }

view this post on Zulip Wasmtime GitHub notifications bot (Oct 04 2022 at 10:50):

mokhaled2992 edited issue #5005:

I'm getting a panic and the stacktrace points at the following line. This happens at the very early stages when I initialize my JIT and try to declare some functions. Any hints on how to solve this?

    unsafe fn write_plt_entry_bytes(plt_ptr: *mut [u8; 16], got_ptr: NonNull<AtomicPtr<u8>>) {
        ...................
        plt_val[2..6].copy_from_slice(&i32::to_ne_bytes(i32::try_from(what - at).unwrap()));
        ...................
    }

More info:

I'm running on x86_64 and Ubuntu 20.

My JIT only declares some external rust functions and creates a wrapper function that calls those.

I ran into another memory permissions problem and was pointed to use the feature flag selinux-fix which solved that problem (see https://github.com/bytecodealliance/wasmtime/issues/4980#issue-1391090347).

view this post on Zulip Wasmtime GitHub notifications bot (Oct 04 2022 at 10:51):

mokhaled2992 edited issue #5005:

I'm getting a panic and the stacktrace points at the following line. This happens at the very early stages when I initialize my JIT and try to declare some functions. Any hints on how to solve this?

    unsafe fn write_plt_entry_bytes(plt_ptr: *mut [u8; 16], got_ptr: NonNull<AtomicPtr<u8>>) {
        ...................
        plt_val[2..6].copy_from_slice(&i32::to_ne_bytes(i32::try_from(what - at).unwrap()));
        ...................
    }

More info:

view this post on Zulip Wasmtime GitHub notifications bot (Oct 04 2022 at 10:51):

mokhaled2992 edited issue #5005:

I'm getting a panic and the stacktrace points at the following line. This happens at the very early stages when I initialize my JIT and try to declare some functions. Any hints on how to solve this?

    unsafe fn write_plt_entry_bytes(plt_ptr: *mut [u8; 16], got_ptr: NonNull<AtomicPtr<u8>>) {
        ...................
        plt_val[2..6].copy_from_slice(&i32::to_ne_bytes(i32::try_from(what - at).unwrap()));
        ...................
    }

More info:

view this post on Zulip Wasmtime GitHub notifications bot (Oct 04 2022 at 11:01):

mokhaled2992 edited issue #5005:

I'm getting a panic and the stacktrace points at the following line. This happens at the very early stages when I initialize my JIT and try to declare some functions. Any hints on how to solve this?

    unsafe fn write_plt_entry_bytes(plt_ptr: *mut [u8; 16], got_ptr: NonNull<AtomicPtr<u8>>) {
        ...................
        plt_val[2..6].copy_from_slice(&i32::to_ne_bytes(i32::try_from(what - at).unwrap()));
        ...................
    }

More info:

view this post on Zulip Wasmtime GitHub notifications bot (Oct 04 2022 at 11:06):

mokhaled2992 edited issue #5005:

I'm getting a panic and the stacktrace points at the line shown below. This happens at the very early stages when I initialize my JIT and try to declare some functions. Any hints on how to solve this?

    unsafe fn write_plt_entry_bytes(plt_ptr: *mut [u8; 16], got_ptr: NonNull<AtomicPtr<u8>>) {
        ...................
        plt_val[2..6].copy_from_slice(&i32::to_ne_bytes(i32::try_from(what - at).unwrap()));
        ...................
    }

More info:

view this post on Zulip Wasmtime GitHub notifications bot (Oct 04 2022 at 11:06):

mokhaled2992 edited issue #5005:

I'm getting a panic and the stacktrace points at the line shown below. This happens at the very early stages when I initialize my JIT and try to declare some functions. Any hints on how to solve this?

    unsafe fn write_plt_entry_bytes(plt_ptr: *mut [u8; 16], got_ptr: NonNull<AtomicPtr<u8>>) {
        ...................
        plt_val[2..6].copy_from_slice(&i32::to_ne_bytes(i32::try_from(what - at).unwrap()));
        ...................
    }

More info

view this post on Zulip Wasmtime GitHub notifications bot (Oct 04 2022 at 11:24):

bjorn3 commented on issue #5005:

When is_pic is enabled, it requires all code to be within 2GB of each other as a 32bit signed pc-relative offset is used for getting the address of called local functions. There is no code to guarantee this, thus leading to panics like this. I've known about this issue for a long time, but haven't had the time to think of a good solution.

view this post on Zulip Wasmtime GitHub notifications bot (Oct 05 2022 at 16:15):

bjorn3 commented on issue #5005:

Pretty much.

view this post on Zulip Wasmtime GitHub notifications bot (Oct 05 2022 at 16:15):

mokhaled2992 commented on issue #5005:

@bjorn3 I have been trying to wrap my head around this for a while now. So this is happening because the got and plt entries for a given function get allocated addresses too far from each other or?

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 06:46):

mokhaled2992 commented on issue #5005:

@bjorn3 I'm trying to understand why a PLT entry is always created for each function declaration in the module in pic mode. I have some questions:

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 06:47):

mokhaled2992 edited a comment on issue #5005:

@bjorn3 I'm trying to understand why a PLT entry is always created for each function declaration in the module in pic mode. I have some questions:

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 06:47):

mokhaled2992 edited a comment on issue #5005:

@bjorn3 I'm trying to understand why a PLT entry is always created for each function declaration in the module in pic mode. I have some questions:

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 06:48):

mokhaled2992 edited a comment on issue #5005:

@bjorn3 I'm trying to understand why a PLT entry is always created for each function declaration in the module in pic mode. I have some questions:

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 06:50):

mokhaled2992 edited a comment on issue #5005:

@bjorn3 I'm trying to understand why a PLT entry is always created for each function declaration in the module in pic mode. I have some questions:

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 06:51):

mokhaled2992 edited a comment on issue #5005:

@bjorn3 I'm trying to understand why a PLT entry is always created for each function declaration in the module in pic mode. I have some questions:

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 07:10):

bjorn3 commented on issue #5005:

call lowers to CallKnown. call_indirect to CallUnknown.

Are there any external or even local function calls that emit PLT relocations and go through the PLT at call sites? I saw that X86CallPLTRel4 is only generated for this special instruction ElfTlsGetAddr?

For PLT, no unless hotswapping is enabled. For GOT, yes CallKnown turns into GOT lookup followed by an indirect call. It would likely be possible to stop maintaining the PLT when hotswapping is disabled, but the GOT has the same issue.

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 07:10):

bjorn3 edited a comment on issue #5005:

call lowers to CallKnown. call_indirect to CallUnknown.

Are there any external or even local function calls that emit PLT relocations and go through the PLT at call sites? I saw that X86CallPLTRel4 is only generated for this special instruction ElfTlsGetAddr?

For PLT, no unless hotswapping is enabled. For GOT, yes CallKnown turns into GOT lookup followed by an indirect call in some cases. It would likely be possible to stop maintaining the PLT when hotswapping is disabled, but the GOT has the same issue.

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 07:58):

mokhaled2992 commented on issue #5005:

If I got it right, for the hotswapping we declare_function once and then we can (re)define that function through this sequence: define_function + finalize_definitions + get_finalized_function which compiles/loads the (new) definition in a new piece of memory and returns the address to it, i think it does not use or clean the memory of the old definition. My question is why do we need the PLT indirection if we already get the address to the newly created definition?

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 07:59):

mokhaled2992 edited a comment on issue #5005:

If I got it right, for the hotswapping we declare_function once and then we can (re)define that function through this sequence: (prepare_for_function_redefine) + define_function + finalize_definitions + get_finalized_function which compiles/loads the (new) definition in a new piece of memory and returns the address to it, i think it does not use or clean the memory of the old definition. My question is why do we need the PLT indirection if we already get the address to the newly created definition?

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 08:36):

bjorn3 commented on issue #5005:

The define_function replaces the entry in the GOT, causing all calls from the old function to be redirected to the new definition, even when done by other compiled code. If the GOT didn't exist, we would have to change all executable memory back to read+write, relocate it again and change it back. This is much slower and isn't thread safe. The PLT indirection is for cases where a direct GOT access is not possible, like for function pointers. The PLT is basically a trampoline jumping to the address in the corresponding GOT entry.

i think it does not use or clean the memory of the old definition.

Indeed. That it doesn't deallocate the old definition is something I would like to see fixed, but that will require a more complex scheme for allocating memory to put the executable code in and a way to check that the function is no longer in use by any thread.

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 09:24):

mokhaled2992 commented on issue #5005:

Oh I believe I got it now, this is done through get_address that returns the PLT entry for hotswapped mode instead of the actual function pointer. This hotswapping is probably meant for the non top level functions, so If you have X calling Y, you can get a finalized definition to X you can hotswap Y, but you can't get PLT entries to top level functions right?

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 09:40):

bjorn3 commented on issue #5005:

Oh I believe I got it now, this is done through get_address that returns the PLT entry for hotswapped mode instead of the actual function pointer

Indeed.

This hotswapping is probably meant for the non top level functions, so If you have X calling Y, you can get a finalized definition to X you can hotswap Y, but you can't get PLT entries to top level functions right?

get_finalized_function returns a PLT entry too in hotswap mode I believe.

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 09:43):

mokhaled2992 commented on issue #5005:

I think it just returns the pointer to the allocated memory where the function was loaded unless I missed some other indirection :)

https://github.com/bytecodealliance/wasmtime/blob/main/cranelift/jit/src/backend.rs#L379

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 09:46):

bjorn3 commented on issue #5005:

Indeed.

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 09:52):

mokhaled2992 commented on issue #5005:

Thanks alot @bjorn3 for all the support :heart:.

Feel free to close the issue if you have something that duplicates the original problem. I saw your suggestion in https://github.com/bytecodealliance/wasmtime/issues/4986 which can hit two birds with one stone as far as I understood.

view this post on Zulip Wasmtime GitHub notifications bot (Oct 06 2022 at 10:04):

bjorn3 commented on issue #5005:

I don't believe I had opened an issue for this already, so keeping it open is fine I think.

view this post on Zulip Wasmtime GitHub notifications bot (Jun 30 2024 at 09:50):

Dimchikkk commented on issue #5005:

Indeed. That it doesn't deallocate the old definition is something I would like to see fixed, but that will require a more complex scheme for allocating memory to put the executable code in and a way to check that the function is no longer in use by any thread.

Hi @bjorn3 , sorry for hijacking the thread but just wondering if you could share mode details on how would you start implementing swapped function deallocation... is it sort of garbage collector that tracks all definitions... if it was swapped with new definition and old definition is not used by any thread - deallocate memory?

view this post on Zulip Wasmtime GitHub notifications bot (Jun 30 2024 at 10:32):

bjorn3 commented on issue #5005:

My idea has been for the user of cranelift-jit to tell which functions are no longer necessary. Then cranelift-jit needs to store which parts of each page are still used by necessary functions and when no used function is still part of the page, the page can be deallocated. Or alternatively at the cost of memory usage each function can be stored in separate pages which would allow immediately deallocating all pages when a function is marked as unused..


Last updated: Jan 24 2025 at 00:11 UTC