1#![deny(missing_docs)]
4#![warn(clippy::cast_sign_loss)]
7
8#[cfg(all(target_arch = "x86_64", target_feature = "sse"))]
10#[expect(non_camel_case_types, reason = "matching wasm conventions")]
11pub(crate) type i8x16 = core::arch::x86_64::__m128i;
12#[cfg(all(target_arch = "x86_64", target_feature = "sse"))]
13#[expect(non_camel_case_types, reason = "matching wasm conventions")]
14pub(crate) type f32x4 = core::arch::x86_64::__m128;
15#[cfg(all(target_arch = "x86_64", target_feature = "sse"))]
16#[expect(non_camel_case_types, reason = "matching wasm conventions")]
17pub(crate) type f64x2 = core::arch::x86_64::__m128d;
18
19#[cfg(not(all(target_arch = "x86_64", target_feature = "sse")))]
24#[expect(non_camel_case_types, reason = "matching wasm conventions")]
25#[derive(Copy, Clone)]
26pub(crate) struct i8x16(core::convert::Infallible);
27#[cfg(not(all(target_arch = "x86_64", target_feature = "sse")))]
28#[expect(non_camel_case_types, reason = "matching wasm conventions")]
29#[derive(Copy, Clone)]
30pub(crate) struct f32x4(core::convert::Infallible);
31#[cfg(not(all(target_arch = "x86_64", target_feature = "sse")))]
32#[expect(non_camel_case_types, reason = "matching wasm conventions")]
33#[derive(Copy, Clone)]
34pub(crate) struct f64x2(core::convert::Infallible);
35
36use crate::StoreContextMut;
37use crate::prelude::*;
38use crate::store::{StoreInner, StoreOpaque, StoreResourceLimiter};
39use crate::type_registry::RegisteredType;
40use alloc::sync::Arc;
41use core::fmt;
42use core::ops::{Deref, DerefMut};
43use core::pin::pin;
44use core::ptr::NonNull;
45use core::sync::atomic::{AtomicUsize, Ordering};
46use core::task::{Context, Poll, Waker};
47use wasmtime_environ::{DefinedMemoryIndex, HostPtr, VMOffsets, VMSharedTypeIndex};
48
49#[cfg(feature = "gc")]
50use wasmtime_environ::ModuleInternedTypeIndex;
51
52mod always_mut;
53#[cfg(feature = "component-model")]
54pub mod component;
55mod const_expr;
56mod export;
57mod gc;
58mod imports;
59mod instance;
60mod memory;
61mod mmap_vec;
62#[cfg(has_virtual_memory)]
63mod pagemap_disabled;
64mod provenance;
65mod send_sync_ptr;
66mod stack_switching;
67mod store_box;
68mod sys;
69mod table;
70#[cfg(feature = "gc")]
71mod throw;
72mod traphandlers;
73mod vmcontext;
74
75#[cfg(feature = "threads")]
76mod parking_spot;
77
78#[cfg(all(has_host_compiler_backend, feature = "debug-builtins"))]
82pub mod debug_builtins;
83pub mod libcalls;
84pub mod mpk;
85
86#[cfg(feature = "pulley")]
87pub(crate) mod interpreter;
88#[cfg(not(feature = "pulley"))]
89pub(crate) mod interpreter_disabled;
90#[cfg(not(feature = "pulley"))]
91pub(crate) use interpreter_disabled as interpreter;
92
93#[cfg(feature = "debug-builtins")]
94pub use wasmtime_jit_debug::gdb_jit_int::GdbJitImageRegistration;
95
96pub use crate::runtime::vm::always_mut::*;
97pub use crate::runtime::vm::export::*;
98pub use crate::runtime::vm::gc::*;
99pub use crate::runtime::vm::imports::Imports;
100pub use crate::runtime::vm::instance::{
101 GcHeapAllocationIndex, Instance, InstanceAllocationRequest, InstanceAllocator, InstanceHandle,
102 MemoryAllocationIndex, OnDemandInstanceAllocator, TableAllocationIndex, initialize_instance,
103};
104#[cfg(feature = "pooling-allocator")]
105pub use crate::runtime::vm::instance::{
106 InstanceLimits, PoolConcurrencyLimitError, PoolingAllocatorMetrics, PoolingInstanceAllocator,
107 PoolingInstanceAllocatorConfig,
108};
109pub use crate::runtime::vm::interpreter::*;
110pub use crate::runtime::vm::memory::{
111 Memory, MemoryBase, RuntimeLinearMemory, RuntimeMemoryCreator, SharedMemory,
112};
113pub use crate::runtime::vm::mmap_vec::MmapVec;
114pub use crate::runtime::vm::provenance::*;
115pub use crate::runtime::vm::stack_switching::*;
116pub use crate::runtime::vm::store_box::*;
117#[cfg(feature = "std")]
118pub use crate::runtime::vm::sys::mmap::open_file_for_mmap;
119#[cfg(has_host_compiler_backend)]
120pub use crate::runtime::vm::sys::unwind::UnwindRegistration;
121pub use crate::runtime::vm::table::{Table, TableElementType};
122#[cfg(feature = "gc")]
123pub use crate::runtime::vm::throw::*;
124pub use crate::runtime::vm::traphandlers::*;
125#[cfg(feature = "component-model")]
126pub use crate::runtime::vm::vmcontext::VMArrayCallFunction;
127pub use crate::runtime::vm::vmcontext::{
128 VMArrayCallHostFuncContext, VMContext, VMFuncRef, VMFunctionImport, VMGlobalDefinition,
129 VMGlobalImport, VMGlobalKind, VMMemoryDefinition, VMMemoryImport, VMOpaqueContext,
130 VMStoreContext, VMTableImport, VMTagImport, VMWasmCallFunction, ValRaw,
131};
132#[cfg(has_custom_sync)]
133pub(crate) use sys::capi;
134
135pub use send_sync_ptr::SendSyncPtr;
136pub use wasmtime_unwinder::Unwind;
137
138#[cfg(has_host_compiler_backend)]
139pub use wasmtime_unwinder::{UnwindHost, get_stack_pointer};
140
141mod module_id;
142pub use module_id::CompiledModuleId;
143
144#[cfg(has_virtual_memory)]
145mod byte_count;
146#[cfg(has_virtual_memory)]
147mod cow;
148#[cfg(not(has_virtual_memory))]
149mod cow_disabled;
150#[cfg(has_virtual_memory)]
151mod mmap;
152
153#[cfg(feature = "async")]
154mod async_yield;
155#[cfg(feature = "async")]
156pub use crate::runtime::vm::async_yield::*;
157
158#[cfg(feature = "gc-null")]
159mod send_sync_unsafe_cell;
160#[cfg(feature = "gc-null")]
161pub use send_sync_unsafe_cell::SendSyncUnsafeCell;
162
163cfg_if::cfg_if! {
164 if #[cfg(has_virtual_memory)] {
165 pub use crate::runtime::vm::byte_count::*;
166 pub use crate::runtime::vm::mmap::{Mmap, MmapOffset};
167 pub use self::cow::{MemoryImage, MemoryImageSlot, ModuleMemoryImages};
168 } else {
169 pub use self::cow_disabled::{MemoryImage, MemoryImageSlot, ModuleMemoryImages};
170 }
171}
172
173pub trait ModuleMemoryImageSource: Send + Sync + 'static {
175 fn wasm_data(&self) -> &[u8];
178
179 fn mmap(&self) -> Option<&MmapVec>;
182}
183
184pub unsafe trait VMStore: 'static {
205 fn store_opaque(&self) -> &StoreOpaque;
207
208 fn store_opaque_mut(&mut self) -> &mut StoreOpaque;
210
211 fn resource_limiter_and_store_opaque(
214 &mut self,
215 ) -> (Option<StoreResourceLimiter<'_>>, &mut StoreOpaque);
216
217 #[cfg(target_has_atomic = "64")]
221 fn new_epoch_updated_deadline(&mut self) -> Result<crate::UpdateDeadline>;
222
223 #[cfg(feature = "component-model")]
225 fn component_calls(&mut self) -> &mut component::CallContexts;
226
227 #[cfg(feature = "component-model-async")]
228 fn component_async_store(
229 &mut self,
230 ) -> &mut dyn crate::runtime::component::VMComponentAsyncStore;
231
232 #[cfg(feature = "debug")]
234 fn block_on_debug_handler(&mut self, event: crate::DebugEvent) -> anyhow::Result<()>;
235}
236
237impl Deref for dyn VMStore + '_ {
238 type Target = StoreOpaque;
239
240 fn deref(&self) -> &Self::Target {
241 self.store_opaque()
242 }
243}
244
245impl DerefMut for dyn VMStore + '_ {
246 fn deref_mut(&mut self) -> &mut Self::Target {
247 self.store_opaque_mut()
248 }
249}
250
251impl dyn VMStore + '_ {
252 pub(crate) unsafe fn unchecked_context_mut<T>(&mut self) -> StoreContextMut<'_, T> {
260 unsafe { StoreContextMut(&mut *(self as *mut dyn VMStore as *mut StoreInner<T>)) }
261 }
262}
263
264#[derive(Copy, Clone)]
279#[repr(transparent)]
280struct VMStoreRawPtr(pub NonNull<dyn VMStore>);
281
282unsafe impl Send for VMStoreRawPtr {}
285unsafe impl Sync for VMStoreRawPtr {}
286
287#[derive(Clone)]
301pub enum ModuleRuntimeInfo {
302 Module(crate::Module),
303 Bare(Box<BareModuleInfo>),
304}
305
306#[derive(Clone)]
311pub struct BareModuleInfo {
312 module: Arc<wasmtime_environ::Module>,
313 offsets: VMOffsets<HostPtr>,
314 _registered_type: Option<RegisteredType>,
315}
316
317impl ModuleRuntimeInfo {
318 pub(crate) fn bare(module: Arc<wasmtime_environ::Module>) -> Self {
319 ModuleRuntimeInfo::bare_with_registered_type(module, None)
320 }
321
322 pub(crate) fn bare_with_registered_type(
323 module: Arc<wasmtime_environ::Module>,
324 registered_type: Option<RegisteredType>,
325 ) -> Self {
326 ModuleRuntimeInfo::Bare(Box::new(BareModuleInfo {
327 offsets: VMOffsets::new(HostPtr, &module),
328 module,
329 _registered_type: registered_type,
330 }))
331 }
332
333 pub(crate) fn env_module(&self) -> &Arc<wasmtime_environ::Module> {
335 match self {
336 ModuleRuntimeInfo::Module(m) => m.env_module(),
337 ModuleRuntimeInfo::Bare(b) => &b.module,
338 }
339 }
340
341 #[cfg(feature = "gc")]
344 fn engine_type_index(&self, module_index: ModuleInternedTypeIndex) -> VMSharedTypeIndex {
345 match self {
346 ModuleRuntimeInfo::Module(m) => m
347 .engine_code()
348 .signatures()
349 .shared_type(module_index)
350 .expect("bad module-level interned type index"),
351 ModuleRuntimeInfo::Bare(_) => unreachable!(),
352 }
353 }
354
355 fn memory_image(
358 &self,
359 memory: DefinedMemoryIndex,
360 ) -> anyhow::Result<Option<&Arc<MemoryImage>>> {
361 match self {
362 ModuleRuntimeInfo::Module(m) => {
363 let images = m.memory_images()?;
364 Ok(images.and_then(|images| images.get_memory_image(memory)))
365 }
366 ModuleRuntimeInfo::Bare(_) => Ok(None),
367 }
368 }
369
370 #[cfg(feature = "pooling-allocator")]
374 fn unique_id(&self) -> Option<CompiledModuleId> {
375 match self {
376 ModuleRuntimeInfo::Module(m) => Some(m.id()),
377 ModuleRuntimeInfo::Bare(_) => None,
378 }
379 }
380
381 fn wasm_data(&self) -> &[u8] {
383 match self {
384 ModuleRuntimeInfo::Module(m) => m.engine_code().wasm_data(),
385 ModuleRuntimeInfo::Bare(_) => &[],
386 }
387 }
388
389 fn type_ids(&self) -> &[VMSharedTypeIndex] {
392 match self {
393 ModuleRuntimeInfo::Module(m) => m
394 .engine_code()
395 .signatures()
396 .as_module_map()
397 .values()
398 .as_slice(),
399 ModuleRuntimeInfo::Bare(_) => &[],
400 }
401 }
402
403 pub(crate) fn offsets(&self) -> &VMOffsets<HostPtr> {
405 match self {
406 ModuleRuntimeInfo::Module(m) => m.offsets(),
407 ModuleRuntimeInfo::Bare(b) => &b.offsets,
408 }
409 }
410}
411
412#[cfg(has_virtual_memory)]
414pub fn host_page_size() -> usize {
415 static PAGE_SIZE: AtomicUsize = AtomicUsize::new(0);
418
419 return match PAGE_SIZE.load(Ordering::Relaxed) {
420 0 => {
421 let size = sys::vm::get_page_size();
422 assert!(size != 0);
423 PAGE_SIZE.store(size, Ordering::Relaxed);
424 size
425 }
426 n => n,
427 };
428}
429
430#[derive(Copy, Clone, PartialEq, Eq, Debug)]
432pub enum WaitResult {
433 Ok = 0,
436 Mismatch = 1,
439 TimedOut = 2,
442}
443
444#[derive(Debug)]
446pub struct WasmFault {
447 pub memory_size: usize,
449 pub wasm_address: u64,
451}
452
453impl fmt::Display for WasmFault {
454 fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
455 write!(
456 f,
457 "memory fault at wasm address 0x{:x} in linear memory of size 0x{:x}",
458 self.wasm_address, self.memory_size,
459 )
460 }
461}
462
463pub fn assert_ready<F: Future>(f: F) -> F::Output {
476 one_poll(f).unwrap()
477}
478
479pub fn one_poll<F: Future>(f: F) -> Option<F::Output> {
491 let mut context = Context::from_waker(&Waker::noop());
492 match pin!(f).poll(&mut context) {
493 Poll::Ready(output) => Some(output),
494 Poll::Pending => None,
495 }
496}