wit_bindgen_rt/
lib.rs

1#![no_std]
2
3extern crate alloc;
4
5// Re-export `bitflags` so that we can reference it from macros.
6#[cfg(feature = "bitflags")]
7#[doc(hidden)]
8pub use bitflags;
9
10/// For more information about this see `./ci/rebuild-libcabi-realloc.sh`.
11#[cfg(not(target_env = "p2"))]
12mod cabi_realloc;
13
14/// This function is called from generated bindings and will be deleted by
15/// the linker. The purpose of this function is to force a reference to the
16/// symbol `cabi_realloc` to make its way through to the final linker
17/// command line. That way `wasm-ld` will pick it up, see it needs to be
18/// exported, and then export it.
19///
20/// For more information about this see `./ci/rebuild-libcabi-realloc.sh`.
21pub fn maybe_link_cabi_realloc() {
22    #[cfg(all(target_family = "wasm", not(target_env = "p2")))]
23    {
24        extern "C" {
25            fn cabi_realloc(
26                old_ptr: *mut u8,
27                old_len: usize,
28                align: usize,
29                new_len: usize,
30            ) -> *mut u8;
31        }
32        // Force the `cabi_realloc` symbol to be referenced from here. This
33        // is done with a `#[used]` Rust `static` to ensure that this
34        // reference makes it all the way to the linker before it's
35        // considered for garbage collection. When the linker sees it it'll
36        // remove this `static` here (due to it not actually being needed)
37        // but the linker will have at that point seen the `cabi_realloc`
38        // symbol and it should get exported.
39        #[used]
40        static _NAME_DOES_NOT_MATTER: unsafe extern "C" fn(
41            *mut u8,
42            usize,
43            usize,
44            usize,
45        ) -> *mut u8 = cabi_realloc;
46    }
47}
48
49/// NB: this function is called by a generated function in the
50/// `cabi_realloc` module above. It's otherwise never explicitly called.
51///
52/// For more information about this see `./ci/rebuild-libcabi-realloc.sh`.
53#[cfg(not(target_env = "p2"))]
54pub unsafe fn cabi_realloc(
55    old_ptr: *mut u8,
56    old_len: usize,
57    align: usize,
58    new_len: usize,
59) -> *mut u8 {
60    use self::alloc::alloc::{self, Layout};
61
62    let layout;
63    let ptr = if old_len == 0 {
64        if new_len == 0 {
65            return align as *mut u8;
66        }
67        layout = Layout::from_size_align_unchecked(new_len, align);
68        alloc::alloc(layout)
69    } else {
70        debug_assert_ne!(new_len, 0, "non-zero old_len requires non-zero new_len!");
71        layout = Layout::from_size_align_unchecked(old_len, align);
72        alloc::realloc(old_ptr, layout, new_len)
73    };
74    if ptr.is_null() {
75        // Print a nice message in debug mode, but in release mode don't
76        // pull in so many dependencies related to printing so just emit an
77        // `unreachable` instruction.
78        if cfg!(debug_assertions) {
79            alloc::handle_alloc_error(layout);
80        } else {
81            #[cfg(target_arch = "wasm32")]
82            core::arch::wasm32::unreachable();
83            #[cfg(not(target_arch = "wasm32"))]
84            unreachable!();
85        }
86    }
87    return ptr;
88}
89
90/// Provide a hook for generated export functions to run static constructors at
91/// most once.
92///
93/// wit-bindgen-rust generates a call to this function at the start of all
94/// component export functions. Importantly, it is not called as part of
95/// `cabi_realloc`, which is a *core* export func, but should not execute ctors.
96#[cfg(target_arch = "wasm32")]
97pub fn run_ctors_once() {
98    static mut RUN: bool = false;
99    unsafe {
100        if !RUN {
101            // This function is synthesized by `wasm-ld` to run all static
102            // constructors. wasm-ld will either provide an implementation
103            // of this symbol, or synthesize a wrapper around each
104            // exported function to (unconditionally) run ctors. By using
105            // this function, the linked module is opting into "manually"
106            // running ctors.
107            extern "C" {
108                fn __wasm_call_ctors();
109            }
110            __wasm_call_ctors();
111            RUN = true;
112        }
113    }
114}
115
116/// Support for using the Component Model Async ABI
117#[cfg(feature = "async")]
118pub mod async_support;