Skip to main content

wasmtime/
config.rs

1use crate::prelude::*;
2use alloc::sync::Arc;
3use bitflags::Flags;
4use core::fmt;
5use core::num::{NonZeroU32, NonZeroUsize};
6use core::str::FromStr;
7#[cfg(any(feature = "cranelift", feature = "winch"))]
8use std::path::Path;
9pub use wasmparser::WasmFeatures;
10use wasmtime_environ::{ConfigTunables, OperatorCost, OperatorCostStrategy, TripleExt, Tunables};
11
12#[cfg(feature = "runtime")]
13use crate::memory::MemoryCreator;
14#[cfg(feature = "runtime")]
15use crate::profiling_agent::{self, ProfilingAgent};
16#[cfg(feature = "runtime")]
17use crate::runtime::vm::{
18    GcRuntime, InstanceAllocator, OnDemandInstanceAllocator, RuntimeMemoryCreator,
19};
20#[cfg(feature = "runtime")]
21use crate::trampoline::MemoryCreatorProxy;
22
23#[cfg(feature = "async")]
24use crate::stack::{StackCreator, StackCreatorProxy};
25#[cfg(feature = "async")]
26use wasmtime_fiber::RuntimeFiberStackCreator;
27
28#[cfg(feature = "runtime")]
29pub use crate::runtime::code_memory::CustomCodeMemory;
30#[cfg(feature = "cache")]
31pub use wasmtime_cache::{Cache, CacheConfig};
32#[cfg(all(feature = "incremental-cache", feature = "cranelift"))]
33pub use wasmtime_environ::CacheStore;
34pub use wasmtime_environ::Inlining;
35
36pub(crate) const DEFAULT_WASM_BACKTRACE_MAX_FRAMES: NonZeroUsize = NonZeroUsize::new(20).unwrap();
37
38/// Represents the module instance allocation strategy to use.
39#[derive(Clone)]
40#[non_exhaustive]
41pub enum InstanceAllocationStrategy {
42    /// The on-demand instance allocation strategy.
43    ///
44    /// Resources related to a module instance are allocated at instantiation time and
45    /// immediately deallocated when the `Store` referencing the instance is dropped.
46    ///
47    /// This is the default allocation strategy for Wasmtime.
48    OnDemand,
49    /// The pooling instance allocation strategy.
50    ///
51    /// A pool of resources is created in advance and module instantiation reuses resources
52    /// from the pool. Resources are returned to the pool when the `Store` referencing the instance
53    /// is dropped.
54    ///
55    /// When GC is enabled, the pooling allocator requires that the GC heap
56    /// configuration matches the linear memory configuration (i.e.,
57    /// `gc_heap_reservation` must equal `memory_reservation`, etc.). By
58    /// default, if no `gc_heap_*` tunables are explicitly configured, they
59    /// automatically inherit the `memory_*` values.
60    #[cfg(feature = "pooling-allocator")]
61    Pooling(PoolingAllocationConfig),
62}
63
64impl InstanceAllocationStrategy {
65    /// The default pooling instance allocation strategy.
66    #[cfg(feature = "pooling-allocator")]
67    pub fn pooling() -> Self {
68        Self::Pooling(Default::default())
69    }
70}
71
72impl Default for InstanceAllocationStrategy {
73    fn default() -> Self {
74        Self::OnDemand
75    }
76}
77
78#[cfg(feature = "pooling-allocator")]
79impl From<PoolingAllocationConfig> for InstanceAllocationStrategy {
80    fn from(cfg: PoolingAllocationConfig) -> InstanceAllocationStrategy {
81        InstanceAllocationStrategy::Pooling(cfg)
82    }
83}
84
85#[derive(Clone)]
86/// Configure the strategy used for versioning in serializing and deserializing [`crate::Module`].
87pub enum ModuleVersionStrategy {
88    /// Use the wasmtime crate's Cargo package version.
89    WasmtimeVersion,
90    /// Use a custom version string. Must be at most 255 bytes.
91    Custom(String),
92    /// Emit no version string in serialization, and accept all version strings in deserialization.
93    None,
94}
95
96impl Default for ModuleVersionStrategy {
97    fn default() -> Self {
98        ModuleVersionStrategy::WasmtimeVersion
99    }
100}
101
102impl core::hash::Hash for ModuleVersionStrategy {
103    fn hash<H: core::hash::Hasher>(&self, hasher: &mut H) {
104        match self {
105            Self::WasmtimeVersion => env!("CARGO_PKG_VERSION").hash(hasher),
106            Self::Custom(s) => s.hash(hasher),
107            Self::None => {}
108        };
109    }
110}
111
112impl ModuleVersionStrategy {
113    /// Get the string-encoding version of the module.
114    pub fn as_str(&self) -> &str {
115        match &self {
116            Self::WasmtimeVersion => env!("CARGO_PKG_VERSION_MAJOR"),
117            Self::Custom(c) => c,
118            Self::None => "",
119        }
120    }
121}
122
123/// Configuration for record/replay
124#[derive(Clone)]
125#[non_exhaustive]
126pub enum RRConfig {
127    #[cfg(feature = "rr")]
128    /// Recording on store is enabled
129    Recording,
130    #[cfg(feature = "rr")]
131    /// Replaying on store is enabled
132    Replaying,
133    /// No record/replay is enabled
134    None,
135}
136
137/// Global configuration options used to create an [`Engine`](crate::Engine)
138/// and customize its behavior.
139///
140/// This structure exposed a builder-like interface and is primarily consumed by
141/// [`Engine::new()`](crate::Engine::new).
142///
143/// The validation of `Config` is deferred until the engine is being built, thus
144/// a problematic config may cause `Engine::new` to fail.
145///
146/// # Defaults
147///
148/// The `Default` trait implementation and the return value from
149/// [`Config::new()`] are the same and represent the default set of
150/// configuration for an engine. The exact set of defaults will differ based on
151/// properties such as enabled Cargo features at compile time and the configured
152/// target (see [`Config::target`]). Configuration options document their
153/// default values and what the conditional value of the default is where
154/// applicable.
155#[derive(Clone)]
156pub struct Config {
157    #[cfg(any(feature = "cranelift", feature = "winch"))]
158    compiler_config: Option<CompilerConfig>,
159    target: Option<target_lexicon::Triple>,
160    #[cfg(feature = "gc")]
161    collector: Collector,
162    profiling_strategy: ProfilingStrategy,
163    tunables: ConfigTunables,
164
165    #[cfg(feature = "cache")]
166    pub(crate) cache: Option<Cache>,
167    #[cfg(feature = "runtime")]
168    pub(crate) mem_creator: Option<Arc<dyn RuntimeMemoryCreator>>,
169    #[cfg(feature = "runtime")]
170    pub(crate) custom_code_memory: Option<Arc<dyn CustomCodeMemory>>,
171    pub(crate) allocation_strategy: InstanceAllocationStrategy,
172    pub(crate) max_wasm_stack: usize,
173    /// Explicitly enabled features via `Config::wasm_*` methods. This is a
174    /// signal that the embedder specifically wants something turned on
175    /// regardless of the defaults that Wasmtime might otherwise have enabled.
176    ///
177    /// Note that this, and `disabled_features` below, start as the empty set of
178    /// features to only track explicit user requests.
179    pub(crate) enabled_features: WasmFeatures,
180    /// Same as `enabled_features`, but for those that are explicitly disabled.
181    pub(crate) disabled_features: WasmFeatures,
182    pub(crate) wasm_backtrace_details_env_used: bool,
183    pub(crate) wasm_backtrace_max_frames: Option<NonZeroUsize>,
184    pub(crate) native_unwind_info: Option<bool>,
185    #[cfg(any(feature = "async", feature = "stack-switching"))]
186    pub(crate) async_stack_size: usize,
187    #[cfg(feature = "async")]
188    pub(crate) async_stack_zeroing: bool,
189    #[cfg(feature = "async")]
190    pub(crate) stack_creator: Option<Arc<dyn RuntimeFiberStackCreator>>,
191    pub(crate) module_version: ModuleVersionStrategy,
192    pub(crate) parallel_compilation: bool,
193    pub(crate) memory_guaranteed_dense_image_size: u64,
194    pub(crate) force_memory_init_memfd: bool,
195    pub(crate) wmemcheck: bool,
196    #[cfg(feature = "coredump")]
197    pub(crate) coredump_on_trap: bool,
198    pub(crate) macos_use_mach_ports: bool,
199    pub(crate) detect_host_feature: Option<fn(&str) -> Option<bool>>,
200    pub(crate) x86_float_abi_ok: Option<bool>,
201    pub(crate) shared_memory: bool,
202    pub(crate) rr_config: RRConfig,
203}
204
205/// User-provided configuration for the compiler.
206#[cfg(any(feature = "cranelift", feature = "winch"))]
207#[derive(Debug, Clone)]
208struct CompilerConfig {
209    strategy: Option<Strategy>,
210    settings: crate::hash_map::HashMap<String, String>,
211    flags: crate::hash_set::HashSet<String>,
212    #[cfg(all(feature = "incremental-cache", feature = "cranelift"))]
213    cache_store: Option<Arc<dyn CacheStore>>,
214    clif_dir: Option<std::path::PathBuf>,
215    wmemcheck: bool,
216}
217
218#[cfg(any(feature = "cranelift", feature = "winch"))]
219impl CompilerConfig {
220    fn new() -> Self {
221        Self {
222            strategy: Strategy::Auto.not_auto(),
223            settings: Default::default(),
224            flags: Default::default(),
225            #[cfg(all(feature = "incremental-cache", feature = "cranelift"))]
226            cache_store: None,
227            clif_dir: None,
228            wmemcheck: false,
229        }
230    }
231
232    /// Ensures that the key is not set or equals to the given value.
233    /// If the key is not set, it will be set to the given value.
234    ///
235    /// # Returns
236    ///
237    /// Returns true if successfully set or already had the given setting
238    /// value, or false if the setting was explicitly set to something
239    /// else previously.
240    fn ensure_setting_unset_or_given(&mut self, k: &str, v: &str) -> bool {
241        if let Some(value) = self.settings.get(k) {
242            if value != v {
243                return false;
244            }
245        } else {
246            self.settings.insert(k.to_string(), v.to_string());
247        }
248        true
249    }
250}
251
252#[cfg(any(feature = "cranelift", feature = "winch"))]
253impl Default for CompilerConfig {
254    fn default() -> Self {
255        Self::new()
256    }
257}
258
259impl Config {
260    /// Creates a new configuration object with the default configuration
261    /// specified.
262    pub fn new() -> Self {
263        let mut ret = Self {
264            tunables: ConfigTunables::default(),
265            #[cfg(any(feature = "cranelift", feature = "winch"))]
266            compiler_config: Some(CompilerConfig::default()),
267            target: None,
268            #[cfg(feature = "gc")]
269            collector: Collector::default(),
270            #[cfg(feature = "cache")]
271            cache: None,
272            profiling_strategy: ProfilingStrategy::None,
273            #[cfg(feature = "runtime")]
274            mem_creator: None,
275            #[cfg(feature = "runtime")]
276            custom_code_memory: None,
277            allocation_strategy: InstanceAllocationStrategy::OnDemand,
278            // 512k of stack -- note that this is chosen currently to not be too
279            // big, not be too small, and be a good default for most platforms.
280            // One platform of particular note is Windows where the stack size
281            // of the main thread seems to, by default, be smaller than that of
282            // Linux and macOS. This 512k value at least lets our current test
283            // suite pass on the main thread of Windows (using `--test-threads
284            // 1` forces this), or at least it passed when this change was
285            // committed.
286            max_wasm_stack: 512 * 1024,
287            wasm_backtrace_details_env_used: false,
288            wasm_backtrace_max_frames: Some(DEFAULT_WASM_BACKTRACE_MAX_FRAMES),
289            native_unwind_info: None,
290            enabled_features: WasmFeatures::empty(),
291            disabled_features: WasmFeatures::empty(),
292            #[cfg(any(feature = "async", feature = "stack-switching"))]
293            async_stack_size: 2 << 20,
294            #[cfg(feature = "async")]
295            async_stack_zeroing: false,
296            #[cfg(feature = "async")]
297            stack_creator: None,
298            module_version: ModuleVersionStrategy::default(),
299            parallel_compilation: !cfg!(miri),
300            memory_guaranteed_dense_image_size: 16 << 20,
301            force_memory_init_memfd: false,
302            wmemcheck: false,
303            #[cfg(feature = "coredump")]
304            coredump_on_trap: false,
305            macos_use_mach_ports: !cfg!(miri),
306            #[cfg(feature = "std")]
307            detect_host_feature: Some(detect_host_feature),
308            #[cfg(not(feature = "std"))]
309            detect_host_feature: None,
310            x86_float_abi_ok: None,
311            shared_memory: false,
312            rr_config: RRConfig::None,
313        };
314        ret.wasm_backtrace_details(WasmBacktraceDetails::Environment);
315        ret
316    }
317
318    #[cfg(any(feature = "cranelift", feature = "winch"))]
319    pub(crate) fn has_compiler(&self) -> bool {
320        self.compiler_config.is_some()
321    }
322
323    #[track_caller]
324    #[cfg(any(feature = "cranelift", feature = "winch"))]
325    fn compiler_config_mut(&mut self) -> &mut CompilerConfig {
326        self.compiler_config.as_mut().expect(
327            "cannot configure compiler settings for `Config`s \
328             created by `Config::without_compiler`",
329        )
330    }
331
332    /// Configure whether Wasm compilation is enabled.
333    ///
334    /// Disabling Wasm compilation will allow you to load and run
335    /// [pre-compiled][crate::Engine::precompile_module] Wasm programs, but not
336    /// to compile and run new Wasm programs that have not already been
337    /// pre-compiled.
338    ///
339    /// Many compilation-related configuration methods will panic if compilation
340    /// has been disabled.
341    ///
342    /// Note that there are two ways to disable Wasm compilation:
343    ///
344    /// 1. Statically, by disabling the `"cranelift"` and `"winch"` cargo
345    ///    features when building Wasmtime. These builds of Wasmtime will have
346    ///    smaller code size, since they do not include any of the code to
347    ///    compile Wasm.
348    ///
349    /// 2. Dynamically, by passing `false` to this method at run-time when
350    ///    configuring Wasmtime. The Wasmtime binary will still include the code
351    ///    for compiling Wasm, it just won't be executed, so code size is larger
352    ///    than with the first approach.
353    ///
354    /// The static approach is better in most cases, however dynamically calling
355    /// `enable_compiler(false)` is useful whenever you create multiple
356    /// `Engine`s in the same process, some of which must be able to compile
357    /// Wasm and some of which should never do so. Tests are a common example of
358    /// such a situation, especially when there are multiple Rust binaries in
359    /// the same cargo workspace, and cargo's feature resolution enables the
360    /// `"cranelift"` or `"winch"` features across the whole workspace.
361    #[cfg(any(feature = "cranelift", feature = "winch"))]
362    pub fn enable_compiler(&mut self, enable: bool) -> &mut Self {
363        match (enable, &self.compiler_config) {
364            (true, Some(_)) | (false, None) => {}
365            (true, None) => {
366                self.compiler_config = Some(CompilerConfig::default());
367            }
368            (false, Some(_)) => {
369                self.compiler_config = None;
370            }
371        }
372        self
373    }
374
375    /// Configures the target platform of this [`Config`].
376    ///
377    /// This method is used to configure the output of compilation in an
378    /// [`Engine`](crate::Engine). This can be used, for example, to
379    /// cross-compile from one platform to another. By default, the host target
380    /// triple is used meaning compiled code is suitable to run on the host.
381    ///
382    /// Note that the [`Module`](crate::Module) type can only be created if the
383    /// target configured here matches the host. Otherwise if a cross-compile is
384    /// being performed where the host doesn't match the target then
385    /// [`Engine::precompile_module`](crate::Engine::precompile_module) must be
386    /// used instead.
387    ///
388    /// Target-specific flags (such as CPU features) will not be inferred by
389    /// default for the target when one is provided here. This means that this
390    /// can also be used, for example, with the host architecture to disable all
391    /// host-inferred feature flags. Configuring target-specific flags can be
392    /// done with [`Config::cranelift_flag_set`] and
393    /// [`Config::cranelift_flag_enable`].
394    ///
395    /// # Errors
396    ///
397    /// This method will error if the given target triple is not supported.
398    pub fn target(&mut self, target: &str) -> Result<&mut Self> {
399        self.target =
400            Some(target_lexicon::Triple::from_str(target).map_err(|e| crate::format_err!(e))?);
401
402        Ok(self)
403    }
404
405    /// Enables the incremental compilation cache in Cranelift, using the provided `CacheStore`
406    /// backend for storage.
407    ///
408    /// # Panics
409    ///
410    /// Panics if this configuration's compiler was [disabled][Config::enable_compiler].
411    #[cfg(all(feature = "incremental-cache", feature = "cranelift"))]
412    pub fn enable_incremental_compilation(
413        &mut self,
414        cache_store: Arc<dyn CacheStore>,
415    ) -> Result<&mut Self> {
416        self.compiler_config_mut().cache_store = Some(cache_store);
417        Ok(self)
418    }
419
420    #[doc(hidden)]
421    #[deprecated(note = "no longer has any effect")]
422    #[cfg(feature = "async")]
423    pub fn async_support(&mut self, _enable: bool) -> &mut Self {
424        self
425    }
426
427    /// Configures whether DWARF debug information will be emitted
428    /// during compilation for a native debugger on the Wasmtime
429    /// process to consume.
430    ///
431    /// Note that the `debug-builtins` compile-time Cargo feature must also be
432    /// enabled for native debuggers such as GDB or LLDB to be able to debug
433    /// guest WebAssembly programs.
434    ///
435    /// By default this option is `false`.
436    /// **Note** Enabling this option is not compatible with the Winch compiler.
437    pub fn debug_info(&mut self, enable: bool) -> &mut Self {
438        self.tunables.debug_native = Some(enable);
439        self
440    }
441
442    /// Configures whether compiled guest code will be instrumented to
443    /// provide debugging at the Wasm VM level.
444    ///
445    /// This is required in order to enable a guest-level debugging
446    /// API that can precisely examine Wasm VM state and (eventually,
447    /// once it is complete) set breakpoints and watchpoints and step
448    /// through code.
449    ///
450    /// Without this enabled, debugging can only be done via a native
451    /// debugger operating on the compiled guest code (see
452    /// [`Config::debug_info`] and is "best-effort": we may be able to
453    /// recover some Wasm locals or operand stack values, but it is
454    /// not guaranteed, even when optimizations are disabled.
455    ///
456    /// When this is enabled, additional instrumentation is inserted
457    /// that directly tracks the Wasm VM state at every step. This has
458    /// some performance impact, but allows perfect debugging
459    /// fidelity.
460    ///
461    /// Breakpoints, watchpoints, and stepping are not yet supported,
462    /// but will be added in a future version of Wasmtime.
463    ///
464    /// This enables use of the [`crate::FrameHandle`] API which is
465    /// provided by [`crate::Caller::debug_exit_frames`] or
466    /// [`crate::Store::debug_exit_frames`].
467    ///
468    /// ***Note*** Enabling this option is not compatible with the
469    /// Winch compiler.
470    #[cfg(feature = "debug")]
471    pub fn guest_debug(&mut self, enable: bool) -> &mut Self {
472        self.tunables.debug_guest = Some(enable);
473        self
474    }
475
476    /// Configures whether [`WasmBacktrace`] will be present in the context of
477    /// errors returned from Wasmtime.
478    ///
479    /// This method is deprecated in favor of
480    /// [`Config::wasm_backtrace_max_frames`]. Calling `wasm_backtrace(false)`
481    /// is equivalent to `wasm_backtrace_max_frames(None)`, and
482    /// `wasm_backtrace(true)` will leave `wasm_backtrace_max_frames` unchanged
483    /// if the value is `Some` and will otherwise restore the default `Some`
484    /// value.
485    ///
486    /// [`WasmBacktrace`]: crate::WasmBacktrace
487    #[deprecated = "use `wasm_backtrace_max_frames` instead"]
488    pub fn wasm_backtrace(&mut self, enable: bool) -> &mut Self {
489        match (enable, self.wasm_backtrace_max_frames) {
490            (false, _) => self.wasm_backtrace_max_frames = None,
491            // Wasm backtraces were disabled; enable them with the
492            // default maximum number of frames to capture.
493            (true, None) => {
494                self.wasm_backtrace_max_frames = Some(DEFAULT_WASM_BACKTRACE_MAX_FRAMES)
495            }
496            // Wasm backtraces are already enabled; keep the existing
497            // max-frames configuration.
498            (true, Some(_)) => {}
499        }
500        self
501    }
502
503    /// Configures whether backtraces in `Trap` will parse debug info in the wasm file to
504    /// have filename/line number information.
505    ///
506    /// When enabled this will causes modules to retain debugging information
507    /// found in wasm binaries. This debug information will be used when a trap
508    /// happens to symbolicate each stack frame and attempt to print a
509    /// filename/line number for each wasm frame in the stack trace.
510    ///
511    /// By default this option is `WasmBacktraceDetails::Environment`, meaning
512    /// that wasm will read `WASMTIME_BACKTRACE_DETAILS` to indicate whether
513    /// details should be parsed. Note that the `std` feature of this crate must
514    /// be active to read environment variables, otherwise this is disabled by
515    /// default.
516    pub fn wasm_backtrace_details(&mut self, enable: WasmBacktraceDetails) -> &mut Self {
517        self.wasm_backtrace_details_env_used = false;
518        self.tunables.parse_wasm_debuginfo = match enable {
519            WasmBacktraceDetails::Enable => Some(true),
520            WasmBacktraceDetails::Disable => Some(false),
521            WasmBacktraceDetails::Environment => {
522                #[cfg(feature = "std")]
523                {
524                    self.wasm_backtrace_details_env_used = true;
525                    std::env::var("WASMTIME_BACKTRACE_DETAILS")
526                        .map(|s| Some(s == "1"))
527                        .unwrap_or(Some(false))
528                }
529                #[cfg(not(feature = "std"))]
530                {
531                    Some(false)
532                }
533            }
534        };
535        self
536    }
537
538    /// Configures the maximum number of WebAssembly frames to collect in
539    /// backtraces.
540    ///
541    /// A backtrace may be collected whenever an error is returned from a host
542    /// function call through to WebAssembly or when WebAssembly itself hits a
543    /// trap condition, such as an out-of-bounds memory access. This flag
544    /// indicates, in these conditions, whether the backtrace is collected or
545    /// not and how many frames should be collected.
546    ///
547    /// Currently wasm backtraces are implemented through frame pointer walking.
548    /// This means that collecting a backtrace is expected to be a fast and
549    /// relatively cheap operation. Additionally backtrace collection is
550    /// suitable in concurrent environments since one thread capturing a
551    /// backtrace won't block other threads.
552    ///
553    /// Collected backtraces are attached via
554    /// [`Error::context`](crate::Error::context) to errors returned from host
555    /// functions. The [`WasmBacktrace`] type can be acquired via
556    /// [`Error::downcast_ref`](crate::Error::downcast_ref) to inspect the
557    /// backtrace. When this option is set to `None` then this context is never
558    /// applied to errors coming out of wasm.
559    ///
560    /// The default value is 20.
561    ///
562    /// [`WasmBacktrace`]: crate::WasmBacktrace
563    pub fn wasm_backtrace_max_frames(&mut self, limit: Option<NonZeroUsize>) -> &mut Self {
564        self.wasm_backtrace_max_frames = limit;
565        self
566    }
567
568    /// Configures whether to generate native unwind information
569    /// (e.g. `.eh_frame` on Linux).
570    ///
571    /// This configuration option only exists to help third-party stack
572    /// capturing mechanisms, such as the system's unwinder or the `backtrace`
573    /// crate, determine how to unwind through Wasm frames. It does not affect
574    /// whether Wasmtime can capture Wasm backtraces or not. The presence of
575    /// [`WasmBacktrace`] is controlled by the
576    /// [`Config::wasm_backtrace_max_frames`] option.
577    ///
578    /// Native unwind information is included:
579    /// - When targeting Windows, since the Windows ABI requires it.
580    /// - By default.
581    ///
582    /// Note that systems loading many modules may wish to disable this
583    /// configuration option instead of leaving it on-by-default. Some platforms
584    /// exhibit quadratic behavior when registering/unregistering unwinding
585    /// information which can greatly slow down the module loading/unloading
586    /// process.
587    ///
588    /// [`WasmBacktrace`]: crate::WasmBacktrace
589    pub fn native_unwind_info(&mut self, enable: bool) -> &mut Self {
590        self.native_unwind_info = Some(enable);
591        self
592    }
593
594    /// Configures whether execution of WebAssembly will "consume fuel" to
595    /// either halt or yield execution as desired.
596    ///
597    /// This can be used to deterministically prevent infinitely-executing
598    /// WebAssembly code by instrumenting generated code to consume fuel as it
599    /// executes. When fuel runs out a trap is raised, however [`Store`] can be
600    /// configured to yield execution periodically via
601    /// [`crate::Store::fuel_async_yield_interval`].
602    ///
603    /// Note that a [`Store`] starts with no fuel, so if you enable this option
604    /// you'll have to be sure to pour some fuel into [`Store`] before
605    /// executing some code.
606    ///
607    /// By default this option is `false`.
608    ///
609    /// **Note** Enabling this option is not compatible with the Winch compiler.
610    ///
611    /// [`Store`]: crate::Store
612    pub fn consume_fuel(&mut self, enable: bool) -> &mut Self {
613        self.tunables.consume_fuel = Some(enable);
614        self
615    }
616
617    /// Configures the fuel cost of each WebAssembly operator.
618    ///
619    /// This is only relevant when [`Config::consume_fuel`] is enabled.
620    pub fn operator_cost(&mut self, cost: OperatorCost) -> &mut Self {
621        self.tunables.operator_cost = Some(OperatorCostStrategy::table(cost));
622        self
623    }
624
625    /// Enables epoch-based interruption.
626    ///
627    /// When executing code in async mode, we sometimes want to
628    /// implement a form of cooperative timeslicing: long-running Wasm
629    /// guest code should periodically yield to the executor
630    /// loop. This yielding could be implemented by using "fuel" (see
631    /// [`consume_fuel`](Config::consume_fuel)). However, fuel
632    /// instrumentation is somewhat expensive: it modifies the
633    /// compiled form of the Wasm code so that it maintains a precise
634    /// instruction count, frequently checking this count against the
635    /// remaining fuel. If one does not need this precise count or
636    /// deterministic interruptions, and only needs a periodic
637    /// interrupt of some form, then It would be better to have a more
638    /// lightweight mechanism.
639    ///
640    /// Epoch-based interruption is that mechanism. There is a global
641    /// "epoch", which is a counter that divides time into arbitrary
642    /// periods (or epochs). This counter lives on the
643    /// [`Engine`](crate::Engine) and can be incremented by calling
644    /// [`Engine::increment_epoch`](crate::Engine::increment_epoch).
645    /// Epoch-based instrumentation works by setting a "deadline
646    /// epoch". The compiled code knows the deadline, and at certain
647    /// points, checks the current epoch against that deadline. It
648    /// will yield if the deadline has been reached.
649    ///
650    /// The idea is that checking an infrequently-changing counter is
651    /// cheaper than counting and frequently storing a precise metric
652    /// (instructions executed) locally. The interruptions are not
653    /// deterministic, but if the embedder increments the epoch in a
654    /// periodic way (say, every regular timer tick by a thread or
655    /// signal handler), then we can ensure that all async code will
656    /// yield to the executor within a bounded time.
657    ///
658    /// The deadline check cannot be avoided by malicious wasm code. It is safe
659    /// to use epoch deadlines to limit the execution time of untrusted
660    /// code.
661    ///
662    /// The [`Store`](crate::Store) tracks the deadline, and controls
663    /// what happens when the deadline is reached during
664    /// execution. Several behaviors are possible:
665    ///
666    /// - Trap if code is executing when the epoch deadline is
667    ///   met. See
668    ///   [`Store::epoch_deadline_trap`](crate::Store::epoch_deadline_trap).
669    ///
670    /// - Call an arbitrary function. This function may chose to trap or
671    ///   increment the epoch. See
672    ///   [`Store::epoch_deadline_callback`](crate::Store::epoch_deadline_callback).
673    ///
674    /// - Yield to the executor loop, then resume when the future is
675    ///   next polled. See
676    ///   [`Store::epoch_deadline_async_yield_and_update`](crate::Store::epoch_deadline_async_yield_and_update).
677    ///
678    /// Trapping is the default. The yielding behaviour may be used for
679    /// the timeslicing behavior described above.
680    ///
681    /// This feature is available with or without async support.
682    /// However, without async support, the timeslicing behaviour is
683    /// not available. This means epoch-based interruption can only
684    /// serve as a simple external-interruption mechanism.
685    ///
686    /// An initial deadline must be set before executing code by calling
687    /// [`Store::set_epoch_deadline`](crate::Store::set_epoch_deadline). If this
688    /// deadline is not configured then wasm will immediately trap.
689    ///
690    /// ## Interaction with blocking host calls
691    ///
692    /// Epochs (and fuel) do not assist in handling WebAssembly code blocked in
693    /// a call to the host. For example if the WebAssembly function calls
694    /// `wasi:io/poll.poll` to sleep epochs will not assist in waking this up or
695    /// timing it out. Epochs intentionally only affect running WebAssembly code
696    /// itself and it's left to the embedder to determine how best to wake up
697    /// indefinitely blocking code in the host.
698    ///
699    /// The typical solution for this, however, is to use the `async` variant of
700    /// WASI host functions. This models computation as a Rust `Future` which
701    /// means that when blocking happens the future is only suspended and
702    /// control yields back to the main event loop. This gives the embedder the
703    /// opportunity to use `tokio::time::timeout` for example on a wasm
704    /// computation and have the desired effect of cancelling a blocking
705    /// operation when a timeout expires.
706    ///
707    /// ## When to use fuel vs. epochs
708    ///
709    /// In general, epoch-based interruption results in faster
710    /// execution. This difference is sometimes significant: in some
711    /// measurements, up to 2-3x. This is because epoch-based
712    /// interruption does less work: it only watches for a global
713    /// rarely-changing counter to increment, rather than keeping a
714    /// local frequently-changing counter and comparing it to a
715    /// deadline.
716    ///
717    /// Fuel, in contrast, should be used when *deterministic*
718    /// yielding or trapping is needed. For example, if it is required
719    /// that the same function call with the same starting state will
720    /// always either complete or trap with an out-of-fuel error,
721    /// deterministically, then fuel with a fixed bound should be
722    /// used.
723    ///
724    /// **Note** Enabling this option is not compatible with the Winch compiler.
725    ///
726    /// # See Also
727    ///
728    /// - [`Engine::increment_epoch`](crate::Engine::increment_epoch)
729    /// - [`Store::set_epoch_deadline`](crate::Store::set_epoch_deadline)
730    /// - [`Store::epoch_deadline_trap`](crate::Store::epoch_deadline_trap)
731    /// - [`Store::epoch_deadline_callback`](crate::Store::epoch_deadline_callback)
732    /// - [`Store::epoch_deadline_async_yield_and_update`](crate::Store::epoch_deadline_async_yield_and_update)
733    pub fn epoch_interruption(&mut self, enable: bool) -> &mut Self {
734        self.tunables.epoch_interruption = Some(enable);
735        self
736    }
737
738    /// XXX: For internal fuzzing and debugging use only!
739    #[doc(hidden)]
740    pub fn gc_zeal_alloc_counter(&mut self, counter: Option<NonZeroU32>) -> Result<&mut Self> {
741        #[cfg(not(gc_zeal))]
742        {
743            let _ = counter;
744            bail!(
745                "cannot set `gc_zeal_alloc_counter` because Wasmtime was not built with `cfg(gc_zeal)`"
746            );
747        }
748
749        #[cfg(gc_zeal)]
750        {
751            self.tunables.gc_zeal_alloc_counter = Some(counter);
752            Ok(self)
753        }
754    }
755
756    /// Configures the maximum amount of stack space available for
757    /// executing WebAssembly code.
758    ///
759    /// WebAssembly has well-defined semantics on stack overflow. This is
760    /// intended to be a knob which can help configure how much stack space
761    /// wasm execution is allowed to consume. Note that the number here is not
762    /// super-precise, but rather wasm will take at most "pretty close to this
763    /// much" stack space.
764    ///
765    /// If a wasm call (or series of nested wasm calls) take more stack space
766    /// than the `size` specified then a stack overflow trap will be raised.
767    ///
768    /// Caveat: this knob only limits the stack space consumed by wasm code.
769    /// More importantly, it does not ensure that this much stack space is
770    /// available on the calling thread stack. Exhausting the thread stack
771    /// typically leads to an **abort** of the process.
772    ///
773    /// Here are some examples of how that could happen:
774    ///
775    /// - Let's assume this option is set to 2 MiB and then a thread that has
776    ///   a stack with 512 KiB left.
777    ///
778    ///   If wasm code consumes more than 512 KiB then the process will be aborted.
779    ///
780    /// - Assuming the same conditions, but this time wasm code does not consume
781    ///   any stack but calls into a host function. The host function consumes
782    ///   more than 512 KiB of stack space. The process will be aborted.
783    ///
784    /// There's another gotcha related to recursive calling into wasm: the stack
785    /// space consumed by a host function is counted towards this limit. The
786    /// host functions are not prevented from consuming more than this limit.
787    /// However, if the host function that used more than this limit and called
788    /// back into wasm, then the execution will trap immediately because of
789    /// stack overflow.
790    ///
791    /// When the `async` feature is enabled, this value cannot exceed the
792    /// `async_stack_size` option. Be careful not to set this value too close
793    /// to `async_stack_size` as doing so may limit how much stack space
794    /// is available for host functions.
795    ///
796    /// By default this option is 512 KiB.
797    ///
798    /// # Errors
799    ///
800    /// The `Engine::new` method will fail if the `size` specified here is
801    /// either 0 or larger than the [`Config::async_stack_size`] configuration.
802    pub fn max_wasm_stack(&mut self, size: usize) -> &mut Self {
803        self.max_wasm_stack = size;
804        self
805    }
806
807    /// Configures the size of the stacks used for asynchronous execution.
808    ///
809    /// This setting configures the size of the stacks that are allocated for
810    /// asynchronous execution. The value cannot be less than `max_wasm_stack`.
811    ///
812    /// The amount of stack space guaranteed for host functions is
813    /// `async_stack_size - max_wasm_stack`, so take care not to set these two values
814    /// close to one another; doing so may cause host functions to overflow the
815    /// stack and abort the process.
816    ///
817    /// By default this option is 2 MiB.
818    ///
819    /// # Errors
820    ///
821    /// The `Engine::new` method will fail if the value for this option is
822    /// smaller than the [`Config::max_wasm_stack`] option.
823    #[cfg(any(feature = "async", feature = "stack-switching"))]
824    pub fn async_stack_size(&mut self, size: usize) -> &mut Self {
825        self.async_stack_size = size;
826        self
827    }
828
829    /// Configures whether or not stacks used for async futures are zeroed
830    /// before (re)use.
831    ///
832    /// When the [`call_async`] variant of calling WebAssembly is used
833    /// then Wasmtime will create a separate runtime execution stack for each
834    /// future produced by [`call_async`]. By default upon allocation, depending
835    /// on the platform, these stacks might be filled with uninitialized
836    /// memory. This is safe and correct because, modulo bugs in Wasmtime,
837    /// compiled Wasm code will never read from a stack slot before it
838    /// initializes the stack slot.
839    ///
840    /// However, as a defense-in-depth mechanism, you may configure Wasmtime to
841    /// ensure that these stacks are zeroed before they are used. Notably, if
842    /// you are using the pooling allocator, stacks can be pooled and reused
843    /// across different Wasm guests; ensuring that stacks are zeroed can
844    /// prevent data leakage between Wasm guests even in the face of potential
845    /// read-of-stack-slot-before-initialization bugs in Wasmtime's compiler.
846    ///
847    /// Stack zeroing can be a costly operation in highly concurrent
848    /// environments due to modifications of the virtual address space requiring
849    /// process-wide synchronization. It can also be costly in `no-std`
850    /// environments that must manually zero memory, and cannot rely on an OS
851    /// and virtual memory to provide zeroed pages.
852    ///
853    /// This option defaults to `false`.
854    ///
855    /// [`call_async`]: crate::TypedFunc::call_async
856    #[cfg(feature = "async")]
857    pub fn async_stack_zeroing(&mut self, enable: bool) -> &mut Self {
858        self.async_stack_zeroing = enable;
859        self
860    }
861
862    /// Explicitly enables (and un-disables) a given set of [`WasmFeatures`].
863    ///
864    /// Note: this is a low-level method that does not necessarily imply that
865    /// wasmtime _supports_ a feature. It should only be used to _disable_
866    /// features that callers want to be rejected by the parser or _enable_
867    /// features callers are certain that the current configuration of wasmtime
868    /// supports.
869    ///
870    /// Feature validation is deferred until an engine is being built, thus by
871    /// enabling features here a caller may cause
872    /// [`Engine::new`](crate::Engine::new) to fail later, if the feature
873    /// configuration isn't supported.
874    pub fn wasm_features(&mut self, flag: WasmFeatures, enable: bool) -> &mut Self {
875        self.enabled_features.set(flag, enable);
876        self.disabled_features.set(flag, !enable);
877        self
878    }
879
880    /// Configures whether the WebAssembly tail calls proposal will be enabled
881    /// for compilation or not.
882    ///
883    /// The [WebAssembly tail calls proposal] introduces the `return_call` and
884    /// `return_call_indirect` instructions. These instructions allow for Wasm
885    /// programs to implement some recursive algorithms with *O(1)* stack space
886    /// usage.
887    ///
888    /// This is `true` by default except when the Winch compiler is enabled.
889    ///
890    /// [WebAssembly tail calls proposal]: https://github.com/WebAssembly/tail-call
891    pub fn wasm_tail_call(&mut self, enable: bool) -> &mut Self {
892        self.wasm_features(WasmFeatures::TAIL_CALL, enable);
893        self
894    }
895
896    /// Configures whether the WebAssembly [branch-hinting] proposal is enabled.
897    ///
898    /// When enabled, the `metadata.code.branch_hint` custom section is parsed
899    /// and used to lay out cold code paths during compilation. The hints are
900    /// advisory and never affect execution semantics.
901    ///
902    /// This is `false` by default until the proposal has been fuzzed.
903    ///
904    /// [branch-hinting]: https://github.com/WebAssembly/branch-hinting
905    pub fn wasm_branch_hinting(&mut self, enable: bool) -> &mut Self {
906        self.tunables.branch_hinting = Some(enable);
907        self
908    }
909
910    /// Configures whether the WebAssembly custom-page-sizes proposal will be
911    /// enabled for compilation or not.
912    ///
913    /// The [WebAssembly custom-page-sizes proposal] allows a memory to
914    /// customize its page sizes. By default, Wasm page sizes are 64KiB
915    /// large. This proposal allows the memory to opt into smaller page sizes
916    /// instead, allowing Wasm to run in environments with less than 64KiB RAM
917    /// available, for example.
918    ///
919    /// Note that the page size is part of the memory's type, and because
920    /// different memories may have different types, they may also have
921    /// different page sizes.
922    ///
923    /// Currently the only valid page sizes are 64KiB (the default) and 1
924    /// byte. Future extensions may relax this constraint and allow all powers
925    /// of two.
926    ///
927    /// Support for this proposal is disabled by default.
928    ///
929    /// [WebAssembly custom-page-sizes proposal]: https://github.com/WebAssembly/custom-page-sizes
930    pub fn wasm_custom_page_sizes(&mut self, enable: bool) -> &mut Self {
931        self.wasm_features(WasmFeatures::CUSTOM_PAGE_SIZES, enable);
932        self
933    }
934
935    /// Configures whether the WebAssembly [threads] proposal will be enabled
936    /// for compilation.
937    ///
938    /// This feature gates items such as shared memories and atomic
939    /// instructions. Note that the threads feature depends on the bulk memory
940    /// feature, which is enabled by default. Additionally note that while the
941    /// wasm feature is called "threads" it does not actually include the
942    /// ability to spawn threads. Spawning threads is part of the [wasi-threads]
943    /// proposal which is a separately gated feature in Wasmtime.
944    ///
945    /// Embeddings of Wasmtime are able to build their own custom threading
946    /// scheme on top of the core wasm threads proposal, however.
947    ///
948    /// The default value for this option is whether the `threads`
949    /// crate feature of Wasmtime is enabled or not. By default this crate
950    /// feature is enabled.
951    ///
952    /// [threads]: https://github.com/webassembly/threads
953    /// [wasi-threads]: https://github.com/webassembly/wasi-threads
954    #[cfg(feature = "threads")]
955    pub fn wasm_threads(&mut self, enable: bool) -> &mut Self {
956        self.wasm_features(WasmFeatures::THREADS, enable);
957        self
958    }
959
960    /// Configures whether the WebAssembly [shared-everything-threads] proposal
961    /// will be enabled for compilation.
962    ///
963    /// This feature gates extended use of the `shared` attribute on items other
964    /// than memories, extra atomic instructions, and new component model
965    /// intrinsics for spawning threads. It depends on the
966    /// [`wasm_threads`][Self::wasm_threads] being enabled.
967    ///
968    /// [shared-everything-threads]:
969    ///     https://github.com/webassembly/shared-everything-threads
970    pub fn wasm_shared_everything_threads(&mut self, enable: bool) -> &mut Self {
971        self.wasm_features(WasmFeatures::SHARED_EVERYTHING_THREADS, enable);
972        self
973    }
974
975    /// Configures whether the [WebAssembly reference types proposal][proposal]
976    /// will be enabled for compilation.
977    ///
978    /// This feature gates items such as the `externref` and `funcref` types as
979    /// well as allowing a module to define multiple tables.
980    ///
981    /// Note that the reference types proposal depends on the bulk memory proposal.
982    ///
983    /// This feature is `true` by default.
984    ///
985    /// # Errors
986    ///
987    /// The validation of this feature are deferred until the engine is being built,
988    /// and thus may cause `Engine::new` fail if the `bulk_memory` feature is disabled.
989    ///
990    /// [proposal]: https://github.com/webassembly/reference-types
991    #[cfg(feature = "gc")]
992    pub fn wasm_reference_types(&mut self, enable: bool) -> &mut Self {
993        self.wasm_features(WasmFeatures::REFERENCE_TYPES, enable);
994        self
995    }
996
997    /// Configures whether the [WebAssembly function references
998    /// proposal][proposal] will be enabled for compilation.
999    ///
1000    /// This feature gates non-nullable reference types, function reference
1001    /// types, `call_ref`, `ref.func`, and non-nullable reference related
1002    /// instructions.
1003    ///
1004    /// Note that the function references proposal depends on the reference
1005    /// types proposal.
1006    ///
1007    /// This feature is `true` by default.
1008    ///
1009    /// [proposal]: https://github.com/WebAssembly/function-references
1010    #[cfg(feature = "gc")]
1011    pub fn wasm_function_references(&mut self, enable: bool) -> &mut Self {
1012        self.wasm_features(WasmFeatures::FUNCTION_REFERENCES, enable);
1013        self
1014    }
1015
1016    /// Configures whether the [WebAssembly wide-arithmetic][proposal] will be
1017    /// enabled for compilation.
1018    ///
1019    /// This feature is `false` by default.
1020    ///
1021    /// [proposal]: https://github.com/WebAssembly/wide-arithmetic
1022    pub fn wasm_wide_arithmetic(&mut self, enable: bool) -> &mut Self {
1023        self.wasm_features(WasmFeatures::WIDE_ARITHMETIC, enable);
1024        self
1025    }
1026
1027    /// Configures whether the [WebAssembly Garbage Collection
1028    /// proposal][proposal] will be enabled for compilation.
1029    ///
1030    /// This feature gates `struct` and `array` type definitions and references,
1031    /// the `i31ref` type, and all related instructions.
1032    ///
1033    /// Note that the function references proposal depends on the typed function
1034    /// references proposal.
1035    ///
1036    /// This feature is `true` by default.
1037    ///
1038    /// [proposal]: https://github.com/WebAssembly/gc
1039    pub fn wasm_gc(&mut self, enable: bool) -> &mut Self {
1040        self.wasm_features(WasmFeatures::GC, enable);
1041        self
1042    }
1043
1044    /// Configures whether the WebAssembly SIMD proposal will be
1045    /// enabled for compilation.
1046    ///
1047    /// The [WebAssembly SIMD proposal][proposal]. This feature gates items such
1048    /// as the `v128` type and all of its operators being in a module. Note that
1049    /// this does not enable the [relaxed simd proposal].
1050    ///
1051    /// **Note**
1052    ///
1053    /// On x86_64 platforms the base CPU feature requirement for SIMD
1054    /// is SSE2 for the Cranelift compiler and AVX for the Winch compiler.
1055    ///
1056    /// This is `true` by default.
1057    ///
1058    /// [proposal]: https://github.com/webassembly/simd
1059    /// [relaxed simd proposal]: https://github.com/WebAssembly/relaxed-simd
1060    pub fn wasm_simd(&mut self, enable: bool) -> &mut Self {
1061        self.wasm_features(WasmFeatures::SIMD, enable);
1062        self
1063    }
1064
1065    /// Configures whether the WebAssembly Relaxed SIMD proposal will be
1066    /// enabled for compilation.
1067    ///
1068    /// The relaxed SIMD proposal adds new instructions to WebAssembly which,
1069    /// for some specific inputs, are allowed to produce different results on
1070    /// different hosts. More-or-less this proposal enables exposing
1071    /// platform-specific semantics of SIMD instructions in a controlled
1072    /// fashion to a WebAssembly program. From an embedder's perspective this
1073    /// means that WebAssembly programs may execute differently depending on
1074    /// whether the host is x86_64 or AArch64, for example.
1075    ///
1076    /// By default Wasmtime lowers relaxed SIMD instructions to the fastest
1077    /// lowering for the platform it's running on. This means that, by default,
1078    /// some relaxed SIMD instructions may have different results for the same
1079    /// inputs across x86_64 and AArch64. This behavior can be disabled through
1080    /// the [`Config::relaxed_simd_deterministic`] option which will force
1081    /// deterministic behavior across all platforms, as classified by the
1082    /// specification, at the cost of performance.
1083    ///
1084    /// This is `true` by default.
1085    ///
1086    /// [proposal]: https://github.com/webassembly/relaxed-simd
1087    pub fn wasm_relaxed_simd(&mut self, enable: bool) -> &mut Self {
1088        self.wasm_features(WasmFeatures::RELAXED_SIMD, enable);
1089        self
1090    }
1091
1092    /// This option can be used to control the behavior of the [relaxed SIMD
1093    /// proposal's][proposal] instructions.
1094    ///
1095    /// The relaxed SIMD proposal introduces instructions that are allowed to
1096    /// have different behavior on different architectures, primarily to afford
1097    /// an efficient implementation on all architectures. This means, however,
1098    /// that the same module may execute differently on one host than another,
1099    /// which typically is not otherwise the case. This option is provided to
1100    /// force Wasmtime to generate deterministic code for all relaxed simd
1101    /// instructions, at the cost of performance, for all architectures. When
1102    /// this option is enabled then the deterministic behavior of all
1103    /// instructions in the relaxed SIMD proposal is selected.
1104    ///
1105    /// This is `false` by default.
1106    ///
1107    /// [proposal]: https://github.com/webassembly/relaxed-simd
1108    pub fn relaxed_simd_deterministic(&mut self, enable: bool) -> &mut Self {
1109        self.tunables.relaxed_simd_deterministic = Some(enable);
1110        self
1111    }
1112
1113    /// Configures whether the [WebAssembly bulk memory operations
1114    /// proposal][proposal] will be enabled for compilation.
1115    ///
1116    /// This feature gates items such as the `memory.copy` instruction, passive
1117    /// data/table segments, etc, being in a module.
1118    ///
1119    /// This is `true` by default.
1120    ///
1121    /// Feature `reference_types`, which is also `true` by default, requires
1122    /// this feature to be enabled. Thus disabling this feature must also disable
1123    /// `reference_types` as well using [`wasm_reference_types`](crate::Config::wasm_reference_types).
1124    ///
1125    /// # Errors
1126    ///
1127    /// Disabling this feature without disabling `reference_types` will cause
1128    /// `Engine::new` to fail.
1129    ///
1130    /// [proposal]: https://github.com/webassembly/bulk-memory-operations
1131    pub fn wasm_bulk_memory(&mut self, enable: bool) -> &mut Self {
1132        self.wasm_features(WasmFeatures::BULK_MEMORY, enable);
1133        self
1134    }
1135
1136    /// Configures whether the WebAssembly multi-value [proposal] will
1137    /// be enabled for compilation.
1138    ///
1139    /// This feature gates functions and blocks returning multiple values in a
1140    /// module, for example.
1141    ///
1142    /// This is `true` by default.
1143    ///
1144    /// [proposal]: https://github.com/webassembly/multi-value
1145    pub fn wasm_multi_value(&mut self, enable: bool) -> &mut Self {
1146        self.wasm_features(WasmFeatures::MULTI_VALUE, enable);
1147        self
1148    }
1149
1150    /// Configures whether the WebAssembly multi-memory [proposal] will
1151    /// be enabled for compilation.
1152    ///
1153    /// This feature gates modules having more than one linear memory
1154    /// declaration or import.
1155    ///
1156    /// This is `true` by default.
1157    ///
1158    /// [proposal]: https://github.com/webassembly/multi-memory
1159    pub fn wasm_multi_memory(&mut self, enable: bool) -> &mut Self {
1160        self.wasm_features(WasmFeatures::MULTI_MEMORY, enable);
1161        self
1162    }
1163
1164    /// Configures whether the WebAssembly memory64 [proposal] will
1165    /// be enabled for compilation.
1166    ///
1167    /// Note that this the upstream specification is not finalized and Wasmtime
1168    /// may also have bugs for this feature since it hasn't been exercised
1169    /// much.
1170    ///
1171    /// This is `false` by default.
1172    ///
1173    /// [proposal]: https://github.com/webassembly/memory64
1174    pub fn wasm_memory64(&mut self, enable: bool) -> &mut Self {
1175        self.wasm_features(WasmFeatures::MEMORY64, enable);
1176        self
1177    }
1178
1179    /// Configures whether the WebAssembly extended-const [proposal] will
1180    /// be enabled for compilation.
1181    ///
1182    /// This is `true` by default.
1183    ///
1184    /// [proposal]: https://github.com/webassembly/extended-const
1185    pub fn wasm_extended_const(&mut self, enable: bool) -> &mut Self {
1186        self.wasm_features(WasmFeatures::EXTENDED_CONST, enable);
1187        self
1188    }
1189
1190    /// Configures whether the [WebAssembly stack switching
1191    /// proposal][proposal] will be enabled for compilation.
1192    ///
1193    /// This feature gates the use of control tags.
1194    ///
1195    /// This feature depends on the `function_reference_types` and
1196    /// `exceptions` features.
1197    ///
1198    /// This feature is `false` by default.
1199    ///
1200    /// # Errors
1201    ///
1202    /// [proposal]: https://github.com/webassembly/stack-switching
1203    pub fn wasm_stack_switching(&mut self, enable: bool) -> &mut Self {
1204        self.wasm_features(WasmFeatures::STACK_SWITCHING, enable);
1205        self
1206    }
1207
1208    /// Configures whether the WebAssembly component-model [proposal] will
1209    /// be enabled for compilation.
1210    ///
1211    /// This flag can be used to blanket disable all components within Wasmtime.
1212    /// Otherwise usage of components requires statically using
1213    /// [`Component`](crate::component::Component) instead of
1214    /// [`Module`](crate::Module) for example anyway.
1215    ///
1216    /// The default value for this option is whether the `component-model`
1217    /// crate feature of Wasmtime is enabled or not. By default this crate
1218    /// feature is enabled.
1219    ///
1220    /// [proposal]: https://github.com/webassembly/component-model
1221    #[cfg(feature = "component-model")]
1222    pub fn wasm_component_model(&mut self, enable: bool) -> &mut Self {
1223        self.wasm_features(WasmFeatures::COMPONENT_MODEL, enable);
1224        self
1225    }
1226
1227    /// Configures whether components support the async ABI [proposal] for
1228    /// lifting and lowering functions, as well as `stream`, `future`, and
1229    /// `error-context` types.
1230    ///
1231    /// Please note that Wasmtime's support for this feature is _very_
1232    /// incomplete.
1233    ///
1234    /// [proposal]:
1235    ///     https://github.com/WebAssembly/component-model/blob/main/design/mvp/Concurrency.md
1236    #[cfg(feature = "component-model-async")]
1237    pub fn wasm_component_model_async(&mut self, enable: bool) -> &mut Self {
1238        self.wasm_features(WasmFeatures::CM_ASYNC, enable);
1239        self
1240    }
1241
1242    /// This corresponds to the 🚝 emoji in the component model specification.
1243    ///
1244    /// Please note that Wasmtime's support for this feature is _very_
1245    /// incomplete.
1246    ///
1247    /// [proposal]:
1248    ///     https://github.com/WebAssembly/component-model/blob/main/design/mvp/Concurrency.md
1249    #[cfg(feature = "component-model-async")]
1250    pub fn wasm_component_model_more_async_builtins(&mut self, enable: bool) -> &mut Self {
1251        self.wasm_features(WasmFeatures::CM_MORE_ASYNC_BUILTINS, enable);
1252        self
1253    }
1254
1255    /// This corresponds to the 🚟 emoji in the component model specification.
1256    ///
1257    /// Please note that Wasmtime's support for this feature is _very_
1258    /// incomplete.
1259    ///
1260    /// [proposal]: https://github.com/WebAssembly/component-model/blob/main/design/mvp/Concurrency.md
1261    #[cfg(feature = "component-model-async")]
1262    pub fn wasm_component_model_async_stackful(&mut self, enable: bool) -> &mut Self {
1263        self.wasm_features(WasmFeatures::CM_ASYNC_STACKFUL, enable);
1264        self
1265    }
1266
1267    /// This corresponds to the 🧵 emoji in the component model specification.
1268    ///
1269    /// Please note that Wasmtime's support for this feature is _very_
1270    /// incomplete.
1271    ///
1272    /// [proposal]:
1273    ///     https://github.com/WebAssembly/component-model/pull/557
1274    #[cfg(feature = "component-model-async")]
1275    pub fn wasm_component_model_threading(&mut self, enable: bool) -> &mut Self {
1276        self.wasm_features(WasmFeatures::CM_THREADING, enable);
1277        self
1278    }
1279
1280    /// This corresponds to the 📝 emoji in the component model specification.
1281    ///
1282    /// Please note that Wasmtime's support for this feature is _very_
1283    /// incomplete.
1284    ///
1285    /// [proposal]: https://github.com/WebAssembly/component-model/blob/main/design/mvp/Concurrency.md
1286    #[cfg(feature = "component-model")]
1287    pub fn wasm_component_model_error_context(&mut self, enable: bool) -> &mut Self {
1288        self.wasm_features(WasmFeatures::CM_ERROR_CONTEXT, enable);
1289        self
1290    }
1291
1292    /// Configures whether the [GC extension to the component-model
1293    /// proposal][proposal] is enabled or not.
1294    ///
1295    /// This corresponds to the 🛸 emoji in the component model specification.
1296    ///
1297    /// Please note that Wasmtime's support for this feature is _very_
1298    /// incomplete.
1299    ///
1300    /// [proposal]: https://github.com/WebAssembly/component-model/issues/525
1301    #[cfg(feature = "component-model")]
1302    pub fn wasm_component_model_gc(&mut self, enable: bool) -> &mut Self {
1303        self.wasm_features(WasmFeatures::CM_GC, enable);
1304        self
1305    }
1306
1307    /// Configures whether the component model map type is enabled or not.
1308    ///
1309    /// This is part of the component model specification and enables the
1310    /// `map<k, v>` type in WIT and the component binary format.
1311    #[cfg(feature = "component-model")]
1312    pub fn wasm_component_model_map(&mut self, enable: bool) -> &mut Self {
1313        self.wasm_features(WasmFeatures::CM_MAP, enable);
1314        self
1315    }
1316
1317    /// This corresponds to the 🔧 emoji in the component model specification.
1318    ///
1319    /// Please note that Wasmtime's support for this feature is _very_
1320    /// incomplete.
1321    #[cfg(feature = "component-model")]
1322    pub fn wasm_component_model_fixed_length_lists(&mut self, enable: bool) -> &mut Self {
1323        self.wasm_features(WasmFeatures::CM_FIXED_LENGTH_LISTS, enable);
1324        self
1325    }
1326
1327    /// This corresponds to the 🏷️ emoji in the component model specification.
1328    ///
1329    /// Please note that Wasmtime's support for this feature is a work in
1330    /// progress.
1331    #[cfg(feature = "component-model")]
1332    pub fn wasm_component_model_implements(&mut self, enable: bool) -> &mut Self {
1333        self.wasm_features(WasmFeatures::CM_IMPLEMENTS, enable);
1334        self
1335    }
1336
1337    /// Configures whether the [Exception-handling proposal][proposal] is enabled or not.
1338    ///
1339    /// This is `true` by default.
1340    ///
1341    /// [proposal]: https://github.com/WebAssembly/exception-handling
1342    #[cfg(feature = "gc")]
1343    pub fn wasm_exceptions(&mut self, enable: bool) -> &mut Self {
1344        self.wasm_features(WasmFeatures::EXCEPTIONS, enable);
1345        self
1346    }
1347
1348    #[doc(hidden)] // FIXME(#3427) - if/when implemented then un-hide this
1349    #[deprecated = "This configuration option only exists for internal \
1350                    usage with the spec testsuite. It may be removed at \
1351                    any time and without warning. Do not rely on it!"]
1352    pub fn wasm_legacy_exceptions(&mut self, enable: bool) -> &mut Self {
1353        self.wasm_features(WasmFeatures::LEGACY_EXCEPTIONS, enable);
1354        self
1355    }
1356
1357    /// Configures which compilation strategy will be used for wasm modules.
1358    ///
1359    /// This method can be used to configure which compiler is used for wasm
1360    /// modules, and for more documentation consult the [`Strategy`] enumeration
1361    /// and its documentation.
1362    ///
1363    /// The default value for this is `Strategy::Auto`.
1364    ///
1365    /// # Panics
1366    ///
1367    /// Panics if this configuration's compiler was [disabled][Config::enable_compiler].
1368    #[cfg(any(feature = "cranelift", feature = "winch"))]
1369    pub fn strategy(&mut self, strategy: Strategy) -> &mut Self {
1370        self.compiler_config_mut().strategy = strategy.not_auto();
1371        self
1372    }
1373
1374    /// Configures which garbage collector will be used for Wasm modules.
1375    ///
1376    /// This method can be used to configure which garbage collector
1377    /// implementation is used for Wasm modules. For more documentation, consult
1378    /// the [`Collector`] enumeration and its documentation.
1379    ///
1380    /// The default value for this is `Collector::Auto`.
1381    #[cfg(feature = "gc")]
1382    pub fn collector(&mut self, collector: Collector) -> &mut Self {
1383        self.collector = collector;
1384        self
1385    }
1386
1387    /// Creates a default profiler based on the profiling strategy chosen.
1388    ///
1389    /// Profiler creation calls the type's default initializer where the purpose is
1390    /// really just to put in place the type used for profiling.
1391    ///
1392    /// Some [`ProfilingStrategy`] require specific platforms or particular feature
1393    /// to be enabled, such as `ProfilingStrategy::JitDump` requires the `jitdump`
1394    /// feature.
1395    ///
1396    /// # Errors
1397    ///
1398    /// The validation of this field is deferred until the engine is being built, and thus may
1399    /// cause `Engine::new` fail if the required feature is disabled, or the platform is not
1400    /// supported.
1401    pub fn profiler(&mut self, profile: ProfilingStrategy) -> &mut Self {
1402        self.profiling_strategy = profile;
1403        self
1404    }
1405
1406    /// Configures whether the debug verifier of Cranelift is enabled or not.
1407    ///
1408    /// When Cranelift is used as a code generation backend this will configure
1409    /// it to have the `enable_verifier` flag which will enable a number of debug
1410    /// checks inside of Cranelift. This is largely only useful for the
1411    /// developers of wasmtime itself.
1412    ///
1413    /// The default value for this is `false`
1414    ///
1415    /// # Panics
1416    ///
1417    /// Panics if this configuration's compiler was [disabled][Config::enable_compiler].
1418    #[cfg(any(feature = "cranelift", feature = "winch"))]
1419    pub fn cranelift_debug_verifier(&mut self, enable: bool) -> &mut Self {
1420        let val = if enable { "true" } else { "false" };
1421        self.compiler_config_mut()
1422            .settings
1423            .insert("enable_verifier".to_string(), val.to_string());
1424        self
1425    }
1426
1427    /// Configures whether extra debug checks are inserted into
1428    /// Wasmtime-generated code by Cranelift.
1429    ///
1430    /// The default value for this is `false`
1431    ///
1432    /// # Panics
1433    ///
1434    /// Panics if this configuration's compiler was [disabled][Config::enable_compiler].
1435    #[cfg(any(feature = "cranelift", feature = "winch"))]
1436    pub fn cranelift_wasmtime_debug_checks(&mut self, enable: bool) -> &mut Self {
1437        unsafe { self.cranelift_flag_set("wasmtime_debug_checks", &enable.to_string()) }
1438    }
1439
1440    /// Configures the Cranelift code generator optimization level.
1441    ///
1442    /// When the Cranelift code generator is used you can configure the
1443    /// optimization level used for generated code in a few various ways. For
1444    /// more information see the documentation of [`OptLevel`].
1445    ///
1446    /// The default value for this is `OptLevel::Speed`.
1447    ///
1448    /// # Panics
1449    ///
1450    /// Panics if this configuration's compiler was [disabled][Config::enable_compiler].
1451    #[cfg(any(feature = "cranelift", feature = "winch"))]
1452    pub fn cranelift_opt_level(&mut self, level: OptLevel) -> &mut Self {
1453        let val = match level {
1454            OptLevel::None => "none",
1455            OptLevel::Speed => "speed",
1456            OptLevel::SpeedAndSize => "speed_and_size",
1457        };
1458        self.compiler_config_mut()
1459            .settings
1460            .insert("opt_level".to_string(), val.to_string());
1461        self
1462    }
1463
1464    /// Configures the regalloc algorithm used by the Cranelift code generator.
1465    ///
1466    /// Cranelift can select any of several register allocator algorithms. Each
1467    /// of these algorithms generates correct code, but they represent different
1468    /// tradeoffs between compile speed (how expensive the compilation process
1469    /// is) and run-time speed (how fast the generated code runs).
1470    /// For more information see the documentation of [`RegallocAlgorithm`].
1471    ///
1472    /// The default value for this is `RegallocAlgorithm::Backtracking`.
1473    ///
1474    /// # Panics
1475    ///
1476    /// Panics if this configuration's compiler was [disabled][Config::enable_compiler].
1477    #[cfg(any(feature = "cranelift", feature = "winch"))]
1478    pub fn cranelift_regalloc_algorithm(&mut self, algo: RegallocAlgorithm) -> &mut Self {
1479        let val = match algo {
1480            RegallocAlgorithm::Backtracking => "backtracking",
1481            RegallocAlgorithm::SinglePass => "single_pass",
1482        };
1483        self.compiler_config_mut()
1484            .settings
1485            .insert("regalloc_algorithm".to_string(), val.to_string());
1486        self
1487    }
1488
1489    /// Configures whether Cranelift should perform a NaN-canonicalization pass.
1490    ///
1491    /// When Cranelift is used as a code generation backend this will configure
1492    /// it to replace NaNs with a single canonical value. This is useful for
1493    /// users requiring entirely deterministic WebAssembly computation.  This is
1494    /// not required by the WebAssembly spec, so it is not enabled by default.
1495    ///
1496    /// Note that this option affects not only WebAssembly's `f32` and `f64`
1497    /// types but additionally the `v128` type. This option will cause
1498    /// operations using any of these types to have extra checks placed after
1499    /// them to normalize NaN values as needed.
1500    ///
1501    /// The default value for this is `false`
1502    ///
1503    /// # Panics
1504    ///
1505    /// Panics if this configuration's compiler was [disabled][Config::enable_compiler].
1506    #[cfg(any(feature = "cranelift", feature = "winch"))]
1507    pub fn cranelift_nan_canonicalization(&mut self, enable: bool) -> &mut Self {
1508        let val = if enable { "true" } else { "false" };
1509        self.compiler_config_mut()
1510            .settings
1511            .insert("enable_nan_canonicalization".to_string(), val.to_string());
1512        self
1513    }
1514
1515    /// Allows setting a Cranelift boolean flag or preset. This allows
1516    /// fine-tuning of Cranelift settings.
1517    ///
1518    /// Since Cranelift flags may be unstable, this method should not be considered to be stable
1519    /// either; other `Config` functions should be preferred for stability.
1520    ///
1521    /// # Safety
1522    ///
1523    /// This is marked as unsafe, because setting the wrong flag might break invariants,
1524    /// resulting in execution hazards.
1525    ///
1526    /// # Errors
1527    ///
1528    /// The validation of the flags are deferred until the engine is being built, and thus may
1529    /// cause `Engine::new` fail if the flag's name does not exist, or the value is not appropriate
1530    /// for the flag type.
1531    ///
1532    /// # Panics
1533    ///
1534    /// Panics if this configuration's compiler was [disabled][Config::enable_compiler].
1535    #[cfg(any(feature = "cranelift", feature = "winch"))]
1536    pub unsafe fn cranelift_flag_enable(&mut self, flag: &str) -> &mut Self {
1537        self.compiler_config_mut().flags.insert(flag.to_string());
1538        self
1539    }
1540
1541    /// Allows settings another Cranelift flag defined by a flag name and value. This allows
1542    /// fine-tuning of Cranelift settings.
1543    ///
1544    /// Since Cranelift flags may be unstable, this method should not be considered to be stable
1545    /// either; other `Config` functions should be preferred for stability.
1546    ///
1547    /// # Safety
1548    ///
1549    /// This is marked as unsafe, because setting the wrong flag might break invariants,
1550    /// resulting in execution hazards.
1551    ///
1552    /// # Errors
1553    ///
1554    /// The validation of the flags are deferred until the engine is being built, and thus may
1555    /// cause `Engine::new` fail if the flag's name does not exist, or incompatible with other
1556    /// settings.
1557    ///
1558    /// For example, feature `wasm_backtrace` will set `unwind_info` to `true`, but if it's
1559    /// manually set to false then it will fail.
1560    ///
1561    /// # Panics
1562    ///
1563    /// Panics if this configuration's compiler was [disabled][Config::enable_compiler].
1564    #[cfg(any(feature = "cranelift", feature = "winch"))]
1565    pub unsafe fn cranelift_flag_set(&mut self, name: &str, value: &str) -> &mut Self {
1566        self.compiler_config_mut()
1567            .settings
1568            .insert(name.to_string(), value.to_string());
1569        self
1570    }
1571
1572    /// Set a custom [`Cache`].
1573    ///
1574    /// To load a cache configuration from a file, use [`Cache::from_file`]. Otherwise, you can
1575    /// create a new cache config using [`CacheConfig::new`] and passing that to [`Cache::new`].
1576    ///
1577    /// If you want to disable the cache, you can call this method with `None`.
1578    ///
1579    /// By default, new configs do not have caching enabled.
1580    /// Every call to [`Module::new(my_wasm)`][crate::Module::new] will recompile `my_wasm`,
1581    /// even when it is unchanged, unless an enabled `CacheConfig` is provided.
1582    ///
1583    /// This method is only available when the `cache` feature of this crate is
1584    /// enabled.
1585    ///
1586    /// [docs]: https://bytecodealliance.github.io/wasmtime/cli-cache.html
1587    #[cfg(feature = "cache")]
1588    pub fn cache(&mut self, cache: Option<Cache>) -> &mut Self {
1589        self.cache = cache;
1590        self
1591    }
1592
1593    /// Sets a custom memory creator.
1594    ///
1595    /// Custom memory creators are used when creating host `Memory` objects or when
1596    /// creating instance linear memories for the on-demand instance allocation strategy.
1597    #[cfg(feature = "runtime")]
1598    pub fn with_host_memory(&mut self, mem_creator: Arc<dyn MemoryCreator>) -> &mut Self {
1599        self.mem_creator = Some(Arc::new(MemoryCreatorProxy(mem_creator)));
1600        self
1601    }
1602
1603    /// Sets a custom stack creator.
1604    ///
1605    /// Custom memory creators are used when creating creating async instance stacks for
1606    /// the on-demand instance allocation strategy.
1607    #[cfg(feature = "async")]
1608    pub fn with_host_stack(&mut self, stack_creator: Arc<dyn StackCreator>) -> &mut Self {
1609        self.stack_creator = Some(Arc::new(StackCreatorProxy(stack_creator)));
1610        self
1611    }
1612
1613    /// Sets a custom executable-memory publisher.
1614    ///
1615    /// Custom executable-memory publishers are hooks that allow
1616    /// Wasmtime to make certain regions of memory executable when
1617    /// loading precompiled modules or compiling new modules
1618    /// in-process. In most modern operating systems, memory allocated
1619    /// for heap usage is readable and writable by default but not
1620    /// executable. To jump to machine code stored in that memory, we
1621    /// need to make it executable. For security reasons, we usually
1622    /// also make it read-only at the same time, so the executing code
1623    /// can't be modified later.
1624    ///
1625    /// By default, Wasmtime will use the appropriate system calls on
1626    /// the host platform for this work. However, it also allows
1627    /// plugging in a custom implementation via this configuration
1628    /// option. This may be useful on custom or `no_std` platforms,
1629    /// for example, especially where virtual memory is not otherwise
1630    /// used by Wasmtime (no `signals-and-traps` feature).
1631    #[cfg(feature = "runtime")]
1632    pub fn with_custom_code_memory(
1633        &mut self,
1634        custom_code_memory: Option<Arc<dyn CustomCodeMemory>>,
1635    ) -> &mut Self {
1636        self.custom_code_memory = custom_code_memory;
1637        self
1638    }
1639
1640    /// Sets the instance allocation strategy to use.
1641    ///
1642    /// This is notably used in conjunction with
1643    /// [`InstanceAllocationStrategy::Pooling`] and [`PoolingAllocationConfig`].
1644    pub fn allocation_strategy(
1645        &mut self,
1646        strategy: impl Into<InstanceAllocationStrategy>,
1647    ) -> &mut Self {
1648        self.allocation_strategy = strategy.into();
1649        self
1650    }
1651
1652    /// Specifies the capacity of linear memories, in bytes, in their initial
1653    /// allocation.
1654    ///
1655    /// > Note: this value has important performance ramifications, be sure to
1656    /// > benchmark when setting this to a non-default value and read over this
1657    /// > documentation.
1658    ///
1659    /// This function will change the size of the initial memory allocation made
1660    /// for linear memories. This setting is only applicable when the initial
1661    /// size of a linear memory is below this threshold. Linear memories are
1662    /// allocated in the virtual address space of the host process with OS APIs
1663    /// such as `mmap` and this setting affects how large the allocation will
1664    /// be.
1665    ///
1666    /// ## Background: WebAssembly Linear Memories
1667    ///
1668    /// WebAssembly linear memories always start with a minimum size and can
1669    /// possibly grow up to a maximum size. The minimum size is always specified
1670    /// in a WebAssembly module itself and the maximum size can either be
1671    /// optionally specified in the module or inherently limited by the index
1672    /// type. For example for this module:
1673    ///
1674    /// ```wasm
1675    /// (module
1676    ///     (memory $a 4)
1677    ///     (memory $b 4096 4096 (pagesize 1))
1678    ///     (memory $c i64 10)
1679    /// )
1680    /// ```
1681    ///
1682    /// * Memory `$a` initially allocates 4 WebAssembly pages (256KiB) and can
1683    ///   grow up to 4GiB, the limit of the 32-bit index space.
1684    /// * Memory `$b` initially allocates 4096 WebAssembly pages, but in this
1685    ///   case its page size is 1, so it's 4096 bytes. Memory can also grow no
1686    ///   further meaning that it will always be 4096 bytes.
1687    /// * Memory `$c` is a 64-bit linear memory which starts with 640KiB of
1688    ///   memory and can theoretically grow up to 2^64 bytes, although most
1689    ///   hosts will run out of memory long before that.
1690    ///
1691    /// All operations on linear memories done by wasm are required to be
1692    /// in-bounds. Any access beyond the end of a linear memory is considered a
1693    /// trap.
1694    ///
1695    /// ## What this setting affects: Virtual Memory
1696    ///
1697    /// This setting is used to configure the behavior of the size of the linear
1698    /// memory allocation performed for each of these memories. For example the
1699    /// initial linear memory allocation looks like this:
1700    ///
1701    /// ```text
1702    ///              memory_reservation
1703    ///                    |
1704    ///          ◄─────────┴────────────────►
1705    /// ┌───────┬─────────┬──────────────────┬───────┐
1706    /// │ guard │ initial │ ... capacity ... │ guard │
1707    /// └───────┴─────────┴──────────────────┴───────┘
1708    ///  ◄──┬──►                              ◄──┬──►
1709    ///     │                                    │
1710    ///     │                             memory_guard_size
1711    ///     │
1712    ///     │
1713    ///  memory_guard_size (if guard_before_linear_memory)
1714    /// ```
1715    ///
1716    /// Memory in the `initial` range is accessible to the instance and can be
1717    /// read/written by wasm code. Memory in the `guard` regions is never
1718    /// accessible to wasm code and memory in `capacity` is initially
1719    /// inaccessible but may become accessible through `memory.grow` instructions
1720    /// for example.
1721    ///
1722    /// This means that this setting is the size of the initial chunk of virtual
1723    /// memory that a linear memory may grow into.
1724    ///
1725    /// ## What this setting affects: Runtime Speed
1726    ///
1727    /// This is a performance-sensitive setting which is taken into account
1728    /// during the compilation process of a WebAssembly module. For example if a
1729    /// 32-bit WebAssembly linear memory has a `memory_reservation` size of 4GiB
1730    /// then bounds checks can be elided because `capacity` will be guaranteed
1731    /// to be unmapped for all addressable bytes that wasm can access (modulo a
1732    /// few details).
1733    ///
1734    /// If `memory_reservation` was something smaller like 256KiB then that
1735    /// would have a much smaller impact on virtual memory but the compile code
1736    /// would then need to have explicit bounds checks to ensure that
1737    /// loads/stores are in-bounds.
1738    ///
1739    /// The goal of this setting is to enable skipping bounds checks in most
1740    /// modules by default. Some situations which require explicit bounds checks
1741    /// though are:
1742    ///
1743    /// * When `memory_reservation` is smaller than the addressable size of the
1744    ///   linear memory. For example if 64-bit linear memories always need
1745    ///   bounds checks as they can address the entire virtual address spacce.
1746    ///   For 32-bit linear memories a `memory_reservation` minimum size of 4GiB
1747    ///   is required to elide bounds checks.
1748    ///
1749    /// * When linear memories have a page size of 1 then bounds checks are
1750    ///   required. In this situation virtual memory can't be relied upon
1751    ///   because that operates at the host page size granularity where wasm
1752    ///   requires a per-byte level granularity.
1753    ///
1754    /// * Configuration settings such as [`Config::signals_based_traps`] can be
1755    ///   used to disable the use of signal handlers and virtual memory so
1756    ///   explicit bounds checks are required.
1757    ///
1758    /// * When [`Config::memory_guard_size`] is too small a bounds check may be
1759    ///   required. For 32-bit wasm addresses are actually 33-bit effective
1760    ///   addresses because loads/stores have a 32-bit static offset to add to
1761    ///   the dynamic 32-bit address. If the static offset is larger than the
1762    ///   size of the guard region then an explicit bounds check is required.
1763    ///
1764    /// ## What this setting affects: Memory Growth Behavior
1765    ///
1766    /// In addition to affecting bounds checks emitted in compiled code this
1767    /// setting also affects how WebAssembly linear memories are grown. The
1768    /// `memory.grow` instruction can be used to make a linear memory larger and
1769    /// this is also affected by APIs such as
1770    /// [`Memory::grow`](crate::Memory::grow).
1771    ///
1772    /// In these situations when the amount being grown is small enough to fit
1773    /// within the remaining capacity then the linear memory doesn't have to be
1774    /// moved at runtime. If the capacity runs out though then a new linear
1775    /// memory allocation must be made and the contents of linear memory is
1776    /// copied over.
1777    ///
1778    /// For example here's a situation where a copy happens:
1779    ///
1780    /// * The `memory_reservation` setting is configured to 128KiB.
1781    /// * A WebAssembly linear memory starts with a single 64KiB page.
1782    /// * This memory can be grown by one page to contain the full 128KiB of
1783    ///   memory.
1784    /// * If grown by one more page, though, then a 192KiB allocation must be
1785    ///   made and the previous 128KiB of contents are copied into the new
1786    ///   allocation.
1787    ///
1788    /// This growth behavior can have a significant performance impact if lots
1789    /// of data needs to be copied on growth. Conversely if memory growth never
1790    /// needs to happen because the capacity will always be large enough then
1791    /// optimizations can be applied to cache the base pointer of linear memory.
1792    ///
1793    /// When memory is grown then the
1794    /// [`Config::memory_reservation_for_growth`] is used for the new
1795    /// memory allocation to have memory to grow into.
1796    ///
1797    /// When using the pooling allocator via [`PoolingAllocationConfig`] then
1798    /// memories are never allowed to move so requests for growth are instead
1799    /// rejected with an error.
1800    ///
1801    /// ## When this setting is not used
1802    ///
1803    /// This setting is ignored and unused when the initial size of linear
1804    /// memory is larger than this threshold. For example if this setting is set
1805    /// to 1MiB but a wasm module requires a 2MiB minimum allocation then this
1806    /// setting is ignored. In this situation the minimum size of memory will be
1807    /// allocated along with [`Config::memory_reservation_for_growth`]
1808    /// after it to grow into.
1809    ///
1810    /// That means that this value can be set to zero. That can be useful in
1811    /// benchmarking to see the overhead of bounds checks for example.
1812    /// Additionally it can be used to minimize the virtual memory allocated by
1813    /// Wasmtime.
1814    ///
1815    /// ## Default Value
1816    ///
1817    /// The default value for this property depends on the host platform. For
1818    /// 64-bit platforms there's lots of address space available, so the default
1819    /// configured here is 4GiB. When coupled with the default size of
1820    /// [`Config::memory_guard_size`] this means that 32-bit WebAssembly linear
1821    /// memories with 64KiB page sizes will skip almost all bounds checks by
1822    /// default.
1823    ///
1824    /// For 32-bit platforms this value defaults to 10MiB. This means that
1825    /// bounds checks will be required on 32-bit platforms.
1826    pub fn memory_reservation(&mut self, bytes: u64) -> &mut Self {
1827        self.tunables.memory_reservation = Some(bytes);
1828        self
1829    }
1830
1831    /// Indicates whether linear memories may relocate their base pointer at
1832    /// runtime.
1833    ///
1834    /// WebAssembly linear memories either have a maximum size that's explicitly
1835    /// listed in the type of a memory or inherently limited by the index type
1836    /// of the memory (e.g. 4GiB for 32-bit linear memories). Depending on how
1837    /// the linear memory is allocated (see [`Config::memory_reservation`]) it
1838    /// may be necessary to move the memory in the host's virtual address space
1839    /// during growth. This option controls whether this movement is allowed or
1840    /// not.
1841    ///
1842    /// An example of a linear memory needing to move is when
1843    /// [`Config::memory_reservation`] is 0 then a linear memory will be
1844    /// allocated as the minimum size of the memory plus
1845    /// [`Config::memory_reservation_for_growth`]. When memory grows beyond the
1846    /// reservation for growth then the memory needs to be relocated.
1847    ///
1848    /// When this option is set to `false` then it can have a number of impacts
1849    /// on how memories work at runtime:
1850    ///
1851    /// * Modules can be compiled with static knowledge the base pointer of
1852    ///   linear memory never changes to enable optimizations such as
1853    ///   loop invariant code motion (hoisting the base pointer out of a loop).
1854    ///
1855    /// * Memories cannot grow in excess of their original allocation. This
1856    ///   means that [`Config::memory_reservation`] and
1857    ///   [`Config::memory_reservation_for_growth`] may need tuning to ensure
1858    ///   the memory configuration works at runtime.
1859    ///
1860    /// The default value for this option is `true`.
1861    pub fn memory_may_move(&mut self, enable: bool) -> &mut Self {
1862        self.tunables.memory_may_move = Some(enable);
1863        self
1864    }
1865
1866    /// Configures the size, in bytes, of the guard region used at the end of a
1867    /// linear memory's address space reservation.
1868    ///
1869    /// > Note: this value has important performance ramifications, be sure to
1870    /// > understand what this value does before tweaking it and benchmarking.
1871    ///
1872    /// This setting controls how many bytes are guaranteed to be unmapped after
1873    /// the virtual memory allocation of a linear memory. When
1874    /// combined with sufficiently large values of
1875    /// [`Config::memory_reservation`] (e.g. 4GiB for 32-bit linear memories)
1876    /// then a guard region can be used to eliminate bounds checks in generated
1877    /// code.
1878    ///
1879    /// This setting additionally can be used to help deduplicate bounds checks
1880    /// in code that otherwise requires bounds checks. For example with a 4KiB
1881    /// guard region then a 64-bit linear memory which accesses addresses `x+8`
1882    /// and `x+16` only needs to perform a single bounds check on `x`. If that
1883    /// bounds check passes then the offset is guaranteed to either reside in
1884    /// linear memory or the guard region, resulting in deterministic behavior
1885    /// either way.
1886    ///
1887    /// ## How big should the guard be?
1888    ///
1889    /// In general, like with configuring [`Config::memory_reservation`], you
1890    /// probably don't want to change this value from the defaults. Removing
1891    /// bounds checks is dependent on a number of factors where the size of the
1892    /// guard region is only one piece of the equation. Other factors include:
1893    ///
1894    /// * [`Config::memory_reservation`]
1895    /// * The index type of the linear memory (e.g. 32-bit or 64-bit)
1896    /// * The page size of the linear memory
1897    /// * Other settings such as [`Config::signals_based_traps`]
1898    ///
1899    /// Embeddings using virtual memory almost always want at least some guard
1900    /// region, but otherwise changes from the default should be profiled
1901    /// locally to see the performance impact.
1902    ///
1903    /// ## Default
1904    ///
1905    /// The default value for this property is 32MiB on 64-bit platforms. This
1906    /// allows eliminating almost all bounds checks on loads/stores with an
1907    /// immediate offset of less than 32MiB. On 32-bit platforms this defaults
1908    /// to 64KiB.
1909    pub fn memory_guard_size(&mut self, bytes: u64) -> &mut Self {
1910        self.tunables.memory_guard_size = Some(bytes);
1911        self
1912    }
1913
1914    /// Configures the size, in bytes, of the extra virtual memory space
1915    /// reserved after a linear memory is relocated.
1916    ///
1917    /// This setting is used in conjunction with [`Config::memory_reservation`]
1918    /// to configure what happens after a linear memory is relocated in the host
1919    /// address space. If the initial size of a linear memory exceeds
1920    /// [`Config::memory_reservation`] or if it grows beyond that size
1921    /// throughout its lifetime then this setting will be used.
1922    ///
1923    /// When a linear memory is relocated it will initially look like this:
1924    ///
1925    /// ```text
1926    ///            memory.size
1927    ///                 │
1928    ///          ◄──────┴─────►
1929    /// ┌───────┬──────────────┬───────┐
1930    /// │ guard │  accessible  │ guard │
1931    /// └───────┴──────────────┴───────┘
1932    ///                         ◄──┬──►
1933    ///                            │
1934    ///                     memory_guard_size
1935    /// ```
1936    ///
1937    /// where `accessible` needs to be grown but there's no more memory to grow
1938    /// into. A new region of the virtual address space will be allocated that
1939    /// looks like this:
1940    ///
1941    /// ```text
1942    ///                           memory_reservation_for_growth
1943    ///                                       │
1944    ///            memory.size                │
1945    ///                 │                     │
1946    ///          ◄──────┴─────► ◄─────────────┴───────────►
1947    /// ┌───────┬──────────────┬───────────────────────────┬───────┐
1948    /// │ guard │  accessible  │ .. reserved for growth .. │ guard │
1949    /// └───────┴──────────────┴───────────────────────────┴───────┘
1950    ///                                                     ◄──┬──►
1951    ///                                                        │
1952    ///                                               memory_guard_size
1953    /// ```
1954    ///
1955    /// This means that up to `memory_reservation_for_growth` bytes can be
1956    /// allocated again before the entire linear memory needs to be moved again
1957    /// when another `memory_reservation_for_growth` bytes will be appended to
1958    /// the size of the allocation.
1959    ///
1960    /// Note that this is a currently simple heuristic for optimizing the growth
1961    /// of dynamic memories, primarily implemented for the memory64 proposal
1962    /// where the maximum size of memory is larger than 4GiB. This setting is
1963    /// unlikely to be a one-size-fits-all style approach and if you're an
1964    /// embedder running into issues with growth and are interested in having
1965    /// other growth strategies available here please feel free to [open an
1966    /// issue on the Wasmtime repository][issue]!
1967    ///
1968    /// [issue]: https://github.com/bytecodealliance/wasmtime/issues/new
1969    ///
1970    /// ## Default
1971    ///
1972    /// For 64-bit platforms this defaults to 2GiB, and for 32-bit platforms
1973    /// this defaults to 1MiB.
1974    pub fn memory_reservation_for_growth(&mut self, bytes: u64) -> &mut Self {
1975        self.tunables.memory_reservation_for_growth = Some(bytes);
1976        self
1977    }
1978
1979    /// Configures the initial size, in bytes, to be allocated for GC heaps.
1980    ///
1981    /// This is similar to [`Config::memory_reservation`] but applies to the GC
1982    /// heap rather than to linear memories. See that method for more details
1983    /// on what "reservation" means and the implications of this setting.
1984    ///
1985    /// ## Default
1986    ///
1987    /// If none of the `gc_heap_*` tunables are explicitly configured, they
1988    /// default to the same values as their `memory_*` counterparts. Otherwise,
1989    /// the default value for this property depends on the host platform: for
1990    /// 64-bit platforms this defaults to 4GiB, and for 32-bit platforms this
1991    /// defaults to 10MiB.
1992    pub fn gc_heap_reservation(&mut self, bytes: u64) -> &mut Self {
1993        self.tunables.gc_heap_reservation = Some(bytes);
1994        self
1995    }
1996
1997    /// Configures the size, in bytes, of the guard page region for GC heaps.
1998    ///
1999    /// This is similar to [`Config::memory_guard_size`] but applies to the GC
2000    /// heap rather than to linear memories. See that method for more details on
2001    /// what guard pages are and the implications of this setting.
2002    ///
2003    /// ## Default
2004    ///
2005    /// If none of the `gc_heap_*` tunables are explicitly configured, they
2006    /// default to the same values as their `memory_*` counterparts. Otherwise,
2007    /// the default value for this property is 32MiB on 64-bit platforms and
2008    /// 64KiB on 32-bit platforms.
2009    pub fn gc_heap_guard_size(&mut self, bytes: u64) -> &mut Self {
2010        self.tunables.gc_heap_guard_size = Some(bytes);
2011        self
2012    }
2013
2014    /// Configures the size, in bytes, of the extra virtual memory space
2015    /// reserved after a GC heap is relocated.
2016    ///
2017    /// This is similar to [`Config::memory_reservation_for_growth`] but applies
2018    /// to the GC heap rather than to linear memories. See that method for more
2019    /// details.
2020    ///
2021    /// ## Default
2022    ///
2023    /// If none of the `gc_heap_*` tunables are explicitly configured, they
2024    /// default to the same values as their `memory_*` counterparts. Otherwise,
2025    /// for 64-bit platforms this defaults to 2GiB, and for 32-bit platforms
2026    /// this defaults to 1MiB.
2027    pub fn gc_heap_reservation_for_growth(&mut self, bytes: u64) -> &mut Self {
2028        self.tunables.gc_heap_reservation_for_growth = Some(bytes);
2029        self
2030    }
2031
2032    /// Indicates whether GC heaps are allowed to be reallocated after initial
2033    /// allocation at runtime.
2034    ///
2035    /// This is similar to [`Config::memory_may_move`] but applies to the GC
2036    /// heap rather than to linear memories. See that method for more details.
2037    ///
2038    /// ## Default
2039    ///
2040    /// If none of the `gc_heap_*` tunables are explicitly configured, they
2041    /// default to the same values as their `memory_*` counterparts. Otherwise,
2042    /// the default value for this option is `true`.
2043    pub fn gc_heap_may_move(&mut self, enable: bool) -> &mut Self {
2044        self.tunables.gc_heap_may_move = Some(enable);
2045        self
2046    }
2047
2048    /// Indicates whether a guard region is present before allocations of
2049    /// linear memory.
2050    ///
2051    /// Guard regions before linear memories are never used during normal
2052    /// operation of WebAssembly modules, even if they have out-of-bounds
2053    /// loads. The only purpose for a preceding guard region in linear memory
2054    /// is extra protection against possible bugs in code generators like
2055    /// Cranelift. This setting does not affect performance in any way, but will
2056    /// result in larger virtual memory reservations for linear memories (it
2057    /// won't actually ever use more memory, just use more of the address
2058    /// space).
2059    ///
2060    /// The size of the guard region before linear memory is the same as the
2061    /// guard size that comes after linear memory, which is configured by
2062    /// [`Config::memory_guard_size`].
2063    ///
2064    /// ## Default
2065    ///
2066    /// This value defaults to `true`.
2067    pub fn guard_before_linear_memory(&mut self, enable: bool) -> &mut Self {
2068        self.tunables.guard_before_linear_memory = Some(enable);
2069        self
2070    }
2071
2072    /// Indicates whether to initialize tables lazily, so that instantiation
2073    /// is fast but indirect calls are a little slower. If false, tables
2074    /// are initialized eagerly during instantiation from any active element
2075    /// segments that apply to them.
2076    ///
2077    /// **Note** Disabling this option is not compatible with the Winch compiler.
2078    ///
2079    /// ## Default
2080    ///
2081    /// This value defaults to `true`.
2082    pub fn table_lazy_init(&mut self, table_lazy_init: bool) -> &mut Self {
2083        self.tunables.table_lazy_init = Some(table_lazy_init);
2084        self
2085    }
2086
2087    /// Configure the version information used in serialized and deserialized [`crate::Module`]s.
2088    /// This effects the behavior of [`crate::Module::serialize()`], as well as
2089    /// [`crate::Module::deserialize()`] and related functions.
2090    ///
2091    /// The default strategy is to use the wasmtime crate's Cargo package version.
2092    pub fn module_version(&mut self, strategy: ModuleVersionStrategy) -> Result<&mut Self> {
2093        match strategy {
2094            // This case requires special precondition for assertion in SerializedModule::to_bytes
2095            ModuleVersionStrategy::Custom(ref v) => {
2096                if v.as_bytes().len() > 255 {
2097                    bail!("custom module version cannot be more than 255 bytes: {v}");
2098                }
2099            }
2100            _ => {}
2101        }
2102        self.module_version = strategy;
2103        Ok(self)
2104    }
2105
2106    /// Configure whether wasmtime should compile a module using multiple
2107    /// threads.
2108    ///
2109    /// Disabling this will result in a single thread being used to compile
2110    /// the wasm bytecode.
2111    ///
2112    /// By default parallel compilation is enabled.
2113    #[cfg(feature = "parallel-compilation")]
2114    pub fn parallel_compilation(&mut self, parallel: bool) -> &mut Self {
2115        self.parallel_compilation = parallel;
2116        self
2117    }
2118
2119    /// Configures whether compiled artifacts will contain information to map
2120    /// native program addresses back to the original wasm module.
2121    ///
2122    /// This configuration option is `true` by default and, if enabled,
2123    /// generates the appropriate tables in compiled modules to map from native
2124    /// address back to wasm source addresses. This is used for displaying wasm
2125    /// program counters in backtraces as well as generating filenames/line
2126    /// numbers if so configured as well (and the original wasm module has DWARF
2127    /// debugging information present).
2128    pub fn generate_address_map(&mut self, generate: bool) -> &mut Self {
2129        self.tunables.generate_address_map = Some(generate);
2130        self
2131    }
2132
2133    /// Configures whether copy-on-write memory-mapped data is used to
2134    /// initialize a linear memory.
2135    ///
2136    /// Initializing linear memory via a copy-on-write mapping can drastically
2137    /// improve instantiation costs of a WebAssembly module because copying
2138    /// memory is deferred. Additionally if a page of memory is only ever read
2139    /// from WebAssembly and never written too then the same underlying page of
2140    /// data will be reused between all instantiations of a module meaning that
2141    /// if a module is instantiated many times this can lower the overall memory
2142    /// required needed to run that module.
2143    ///
2144    /// The main disadvantage of copy-on-write initialization, however, is that
2145    /// it may be possible for highly-parallel scenarios to be less scalable. If
2146    /// a page is read initially by a WebAssembly module then that page will be
2147    /// mapped to a read-only copy shared between all WebAssembly instances. If
2148    /// the same page is then written, however, then a private copy is created
2149    /// and swapped out from the read-only version. This also requires an [IPI],
2150    /// however, which can be a significant bottleneck in high-parallelism
2151    /// situations.
2152    ///
2153    /// This feature is only applicable when a WebAssembly module meets specific
2154    /// criteria to be initialized in this fashion, such as:
2155    ///
2156    /// * Only memories defined in the module can be initialized this way.
2157    /// * Data segments for memory must use statically known offsets.
2158    /// * Data segments for memory must all be in-bounds.
2159    ///
2160    /// Modules which do not meet these criteria will fall back to
2161    /// initialization of linear memory based on copying memory.
2162    ///
2163    /// This feature of Wasmtime is also platform-specific:
2164    ///
2165    /// * Linux - this feature is supported for all instances of [`Module`].
2166    ///   Modules backed by an existing mmap (such as those created by
2167    ///   [`Module::deserialize_file`]) will reuse that mmap to cow-initialize
2168    ///   memory. Other instance of [`Module`] may use the `memfd_create`
2169    ///   syscall to create an initialization image to `mmap`.
2170    /// * Unix (not Linux) - this feature is only supported when loading modules
2171    ///   from a precompiled file via [`Module::deserialize_file`] where there
2172    ///   is a file descriptor to use to map data into the process. Note that
2173    ///   the module must have been compiled with this setting enabled as well.
2174    /// * Windows - there is no support for this feature at this time. Memory
2175    ///   initialization will always copy bytes.
2176    ///
2177    /// By default this option is enabled.
2178    ///
2179    /// [`Module::deserialize_file`]: crate::Module::deserialize_file
2180    /// [`Module`]: crate::Module
2181    /// [IPI]: https://en.wikipedia.org/wiki/Inter-processor_interrupt
2182    pub fn memory_init_cow(&mut self, enable: bool) -> &mut Self {
2183        self.tunables.memory_init_cow = Some(enable);
2184        self
2185    }
2186
2187    /// A configuration option to force the usage of `memfd_create` on Linux to
2188    /// be used as the backing source for a module's initial memory image.
2189    ///
2190    /// When [`Config::memory_init_cow`] is enabled, which is enabled by
2191    /// default, module memory initialization images are taken from a module's
2192    /// original mmap if possible. If a precompiled module was loaded from disk
2193    /// this means that the disk's file is used as an mmap source for the
2194    /// initial linear memory contents. This option can be used to force, on
2195    /// Linux, that instead of using the original file on disk a new in-memory
2196    /// file is created with `memfd_create` to hold the contents of the initial
2197    /// image.
2198    ///
2199    /// This option can be used to avoid possibly loading the contents of memory
2200    /// from disk through a page fault. Instead with `memfd_create` the contents
2201    /// of memory are always in RAM, meaning that even page faults which
2202    /// initially populate a wasm linear memory will only work with RAM instead
2203    /// of ever hitting the disk that the original precompiled module is stored
2204    /// on.
2205    ///
2206    /// This option is disabled by default.
2207    pub fn force_memory_init_memfd(&mut self, enable: bool) -> &mut Self {
2208        self.force_memory_init_memfd = enable;
2209        self
2210    }
2211
2212    /// Configures whether or not a coredump should be generated and attached to
2213    /// the [`Error`](crate::Error) when a trap is raised.
2214    ///
2215    /// This option is disabled by default.
2216    #[cfg(feature = "coredump")]
2217    pub fn coredump_on_trap(&mut self, enable: bool) -> &mut Self {
2218        self.coredump_on_trap = enable;
2219        self
2220    }
2221
2222    /// Enables memory error checking for wasm programs.
2223    ///
2224    /// This option is disabled by default.
2225    ///
2226    /// # Panics
2227    ///
2228    /// Panics if this configuration's compiler was [disabled][Config::enable_compiler].
2229    #[cfg(any(feature = "cranelift", feature = "winch"))]
2230    pub fn wmemcheck(&mut self, enable: bool) -> &mut Self {
2231        self.wmemcheck = enable;
2232        self.compiler_config_mut().wmemcheck = enable;
2233        self
2234    }
2235
2236    /// Configures the "guaranteed dense image size" for copy-on-write
2237    /// initialized memories.
2238    ///
2239    /// When using the [`Config::memory_init_cow`] feature to initialize memory
2240    /// efficiently (which is enabled by default), compiled modules contain an
2241    /// image of the module's initial heap. If the module has a fairly sparse
2242    /// initial heap, with just a few data segments at very different offsets,
2243    /// this could result in a large region of zero bytes in the image. In
2244    /// other words, it's not very memory-efficient.
2245    ///
2246    /// We normally use a heuristic to avoid this: if less than half
2247    /// of the initialized range (first non-zero to last non-zero
2248    /// byte) of any memory in the module has pages with nonzero
2249    /// bytes, then we avoid creating a memory image for the entire module.
2250    ///
2251    /// However, if the embedder always needs the instantiation-time efficiency
2252    /// of copy-on-write initialization, and is otherwise carefully controlling
2253    /// parameters of the modules (for example, by limiting the maximum heap
2254    /// size of the modules), then it may be desirable to ensure a memory image
2255    /// is created even if this could go against the heuristic above. Thus, we
2256    /// add another condition: there is a size of initialized data region up to
2257    /// which we *always* allow a memory image. The embedder can set this to a
2258    /// known maximum heap size if they desire to always get the benefits of
2259    /// copy-on-write images.
2260    ///
2261    /// In the future we may implement a "best of both worlds"
2262    /// solution where we have a dense image up to some limit, and
2263    /// then support a sparse list of initializers beyond that; this
2264    /// would get most of the benefit of copy-on-write and pay the incremental
2265    /// cost of eager initialization only for those bits of memory
2266    /// that are out-of-bounds. However, for now, an embedder desiring
2267    /// fast instantiation should ensure that this setting is as large
2268    /// as the maximum module initial memory content size.
2269    ///
2270    /// By default this value is 16 MiB.
2271    pub fn memory_guaranteed_dense_image_size(&mut self, size_in_bytes: u64) -> &mut Self {
2272        self.memory_guaranteed_dense_image_size = size_in_bytes;
2273        self
2274    }
2275
2276    /// Whether to enable function inlining during compilation or not.
2277    ///
2278    /// This may result in faster execution at runtime, but adds additional
2279    /// compilation time. Inlining may also enlarge the size of compiled
2280    /// artifacts (for example, the size of the result of
2281    /// [`Engine::precompile_component`](crate::Engine::precompile_component)).
2282    ///
2283    /// Inlining is not supported by all of Wasmtime's compilation strategies;
2284    /// currently, it only Cranelift supports it. This setting will be ignored
2285    /// when using a compilation strategy that does not support inlining, like
2286    /// Winch.
2287    ///
2288    /// The default value for this is `Inlining::No`.
2289    pub fn compiler_inlining(&mut self, inlining: Inlining) -> &mut Self {
2290        self.tunables.inlining = Some(inlining);
2291        self
2292    }
2293
2294    /// Returns the set of features that the currently selected compiler backend
2295    /// does not support at all and may panic on.
2296    ///
2297    /// Wasmtime strives to reject unknown modules or unsupported modules with
2298    /// first-class errors instead of panics. Not all compiler backends have the
2299    /// same level of feature support on all platforms as well. This method
2300    /// returns a set of features that the currently selected compiler
2301    /// configuration is known to not support and may panic on. This acts as a
2302    /// first-level filter on incoming wasm modules/configuration to fail-fast
2303    /// instead of panicking later on.
2304    ///
2305    /// Note that if a feature is not returned here it does not mean that the
2306    /// backend fully supports the proposal. Instead that means that the backend
2307    /// doesn't ever panic on the proposal, but errors during compilation may
2308    /// still be returned. This means that features listed here are definitely
2309    /// not supported at all, but features not listed here may still be
2310    /// partially supported. For example at the time of this writing the Winch
2311    /// backend partially supports simd so it's not listed here. Winch doesn't
2312    /// fully support simd but unimplemented instructions just return errors.
2313    fn compiler_panicking_wasm_features(&self) -> WasmFeatures {
2314        // First we compute the set of features that Wasmtime itself knows;
2315        // this is a sort of "maximal set" that we invert to create a set
2316        // of features we _definitely can't support_ because wasmtime
2317        // has never heard of them.
2318        let features_known_to_wasmtime = WasmFeatures::WASM3
2319            | WasmFeatures::SHARED_EVERYTHING_THREADS
2320            | WasmFeatures::COMPONENT_MODEL
2321            | WasmFeatures::CUSTOM_PAGE_SIZES
2322            | WasmFeatures::STACK_SWITCHING
2323            | WasmFeatures::WIDE_ARITHMETIC
2324            | WasmFeatures::CM_ASYNC
2325            | WasmFeatures::CM_ASYNC_STACKFUL
2326            | WasmFeatures::CM_MORE_ASYNC_BUILTINS
2327            | WasmFeatures::CM_THREADING
2328            | WasmFeatures::CM_ERROR_CONTEXT
2329            | WasmFeatures::CM_GC
2330            | WasmFeatures::CM_MAP
2331            | WasmFeatures::CM_FIXED_LENGTH_LISTS
2332            | WasmFeatures::CM_IMPLEMENTS;
2333
2334        #[allow(unused_mut, reason = "easier to avoid #[cfg]")]
2335        let mut unsupported = !features_known_to_wasmtime;
2336
2337        #[cfg(any(feature = "cranelift", feature = "winch"))]
2338        match self.compiler_config.as_ref().and_then(|c| c.strategy) {
2339            None | Some(Strategy::Cranelift) => {
2340                // Pulley at this time fundamentally doesn't support the
2341                // `threads` proposal, notably shared memory, because Rust can't
2342                // safely implement loads/stores in the face of shared memory.
2343                // Stack switching is not implemented, either.
2344                if self.compiler_target().is_pulley() {
2345                    unsupported |= WasmFeatures::THREADS;
2346                    unsupported |= WasmFeatures::STACK_SWITCHING;
2347                }
2348
2349                use target_lexicon::*;
2350                match self.compiler_target() {
2351                    Triple {
2352                        architecture: Architecture::X86_64 | Architecture::X86_64h,
2353                        operating_system:
2354                            OperatingSystem::Linux
2355                            | OperatingSystem::MacOSX(_)
2356                            | OperatingSystem::Darwin(_),
2357                        ..
2358                    } => {
2359                        // Stack switching supported on (non-Pulley) Cranelift.
2360                    }
2361
2362                    _ => {
2363                        // On platforms other than x64 Unix-like, we don't
2364                        // support stack switching.
2365                        unsupported |= WasmFeatures::STACK_SWITCHING;
2366                    }
2367                }
2368            }
2369            Some(Strategy::Winch) => {
2370                unsupported |= WasmFeatures::GC
2371                    | WasmFeatures::FUNCTION_REFERENCES
2372                    | WasmFeatures::RELAXED_SIMD
2373                    | WasmFeatures::TAIL_CALL
2374                    | WasmFeatures::GC_TYPES
2375                    | WasmFeatures::EXCEPTIONS
2376                    | WasmFeatures::LEGACY_EXCEPTIONS
2377                    | WasmFeatures::STACK_SWITCHING
2378                    | WasmFeatures::CM_ASYNC;
2379                match self.compiler_target().architecture {
2380                    target_lexicon::Architecture::Aarch64(_) => {
2381                        unsupported |= WasmFeatures::THREADS;
2382                    }
2383
2384                    // Winch doesn't support other non-x64 architectures at this
2385                    // time either but will return an first-class error for
2386                    // them.
2387                    _ => {}
2388                }
2389            }
2390            Some(Strategy::Auto) => unreachable!(),
2391        }
2392        unsupported
2393    }
2394
2395    /// Calculates the set of features that are enabled for this `Config`.
2396    ///
2397    /// This is a bit of a subtle function which takes into account inputs such
2398    /// as the default set of features Wasmtime has enabled, the currently
2399    /// enabled compiler, the currently enabled target, compile-time crate
2400    /// features, and explicitly configured wasm proposals. This function does
2401    /// not return a fixed set of all proposals in all cases as it's a bit more
2402    /// nuanced than that.
2403    ///
2404    /// This method internally will start with an empty set of features to
2405    /// avoid being tied to wasmparser's defaults. Next Wasmtime's set of
2406    /// default features are added to this set, some of which are conditional
2407    /// depending on crate features. Finally explicitly requested features via
2408    /// `wasm_*` methods on `Config` are applied. Everything is then validated
2409    /// later in `Config::validate`.
2410    ///
2411    /// Note that the validation later on in `Config::validate` is a crucial
2412    /// step here. The returned features here might include features unsupported
2413    /// at compile time or unsupported by the selected compiler. In that case
2414    /// `Config::validate` will present a first-class error message indicating
2415    /// what's going on, and users should in theory be able to understand "ok
2416    /// yeah that's why I can't enable that feature here".
2417    fn features(&self) -> WasmFeatures {
2418        // Start with an empty set of wasm features. This notably decouples
2419        // features in Wasmtime from features in wasmparser as the two are
2420        // generally on different timelines.
2421        let mut features = WasmFeatures::empty();
2422
2423        // Next add in all on-by-default features that Wasmtime has which are
2424        // subject to the criteria at
2425        // https://docs.wasmtime.dev/contributing-implementing-wasm-proposals.html
2426        // and https://docs.wasmtime.dev/stability-wasm-proposals.html.
2427        //
2428        // Note that the first entry here, `WASM3`, is a fixed feature set that
2429        // won't change over time in wasmparser which represents the union of
2430        // all on-by-default features in Wasmtime. Also note that this is
2431        // further refined in the conditional section below based on crate
2432        // features.
2433        features |= WasmFeatures::WASM3;
2434
2435        // features |= WasmFeatures::YOUR_WASM_FEATURE;
2436        // ...
2437
2438        // NB: if you add a feature above this line please double-check
2439        // https://docs.wasmtime.dev/stability-wasm-proposals.html
2440        // to ensure all requirements are met and/or update the documentation
2441        // there too.
2442
2443        // Next configure some features further based on compile-time features
2444        // of the wasmtime crate itself. For example if "gc" is disabled then
2445        // `GC_TYPES` are disabled (a wasmparser pseudo-feature) as well as
2446        // exceptions, but reference-types is still available (e.g. new
2447        // encodings/types/etc).
2448        //
2449        // These features are all "on by default" in effect but dependent on
2450        // compile-time support being available.
2451        features.set(WasmFeatures::GC_TYPES, cfg!(feature = "gc"));
2452        features.set(WasmFeatures::EXCEPTIONS, cfg!(feature = "gc"));
2453        features.set(WasmFeatures::THREADS, cfg!(feature = "threads"));
2454        features.set(
2455            WasmFeatures::COMPONENT_MODEL,
2456            cfg!(feature = "component-model"),
2457        );
2458        features.set(
2459            WasmFeatures::CM_ASYNC,
2460            self.tunables
2461                .concurrency_support
2462                .unwrap_or(cfg!(feature = "component-model-async")),
2463        );
2464
2465        // Next disable any features which the current compiler/target do not
2466        // support. This handles cases where Winch, for example, doesn't
2467        // implement a feature yet but Cranelift does. Or maybe Cranelift only
2468        // supports one particular platform and not others. Things like that.
2469        features = features & !self.compiler_panicking_wasm_features();
2470
2471        // And, finally, process all explicitly enabled/disabled features on
2472        // behalf of the embedder's frobbing `Config::wasm_*`. These have the
2473        // highest priority since they were explicitly requested.
2474        debug_assert!((self.enabled_features & self.disabled_features).is_empty());
2475        features &= !self.disabled_features;
2476        features |= self.enabled_features;
2477
2478        features
2479    }
2480
2481    /// Returns the configured compiler target for this `Config`.
2482    pub(crate) fn compiler_target(&self) -> target_lexicon::Triple {
2483        // If a target is explicitly configured, always use that.
2484        if let Some(target) = self.target.clone() {
2485            return target;
2486        }
2487
2488        // If the `build.rs` script determined that this platform uses pulley by
2489        // default, then use Pulley.
2490        if cfg!(default_target_pulley) {
2491            return target_lexicon::Triple::pulley_host();
2492        }
2493
2494        // And at this point the target is for sure the host.
2495        target_lexicon::Triple::host()
2496    }
2497
2498    /// Returns `true` if any of the `gc_heap_*` tunables have been explicitly
2499    /// configured.
2500    fn any_gc_heap_tunables_configured(&self) -> bool {
2501        self.tunables.gc_heap_reservation.is_some()
2502            || self.tunables.gc_heap_guard_size.is_some()
2503            || self.tunables.gc_heap_reservation_for_growth.is_some()
2504            || self.tunables.gc_heap_may_move.is_some()
2505    }
2506
2507    pub(crate) fn validate(&self) -> Result<(Tunables, WasmFeatures)> {
2508        let features = self.features();
2509
2510        // First validate that the selected compiler backend and configuration
2511        // supports the set of `features` that are enabled. This will help
2512        // provide more first class errors instead of panics about unsupported
2513        // features and configurations.
2514        let unsupported = features & self.compiler_panicking_wasm_features();
2515        if !unsupported.is_empty() {
2516            for flag in WasmFeatures::FLAGS.iter() {
2517                if !unsupported.contains(*flag.value()) {
2518                    continue;
2519                }
2520                bail!(
2521                    "the wasm_{} feature is not supported on this compiler configuration",
2522                    flag.name().to_lowercase()
2523                );
2524            }
2525
2526            panic!("should have returned an error by now")
2527        }
2528
2529        #[cfg(any(feature = "async", feature = "stack-switching"))]
2530        if self.max_wasm_stack > self.async_stack_size {
2531            bail!("max_wasm_stack size cannot exceed the async_stack_size");
2532        }
2533        if self.max_wasm_stack == 0 {
2534            bail!("max_wasm_stack size cannot be zero");
2535        }
2536        if !cfg!(feature = "wmemcheck") && self.wmemcheck {
2537            bail!("wmemcheck (memory checker) was requested but is not enabled in this build");
2538        }
2539
2540        if !cfg!(feature = "gc") && features.gc_types() {
2541            bail!("support for GC was disabled at compile time")
2542        }
2543
2544        if !cfg!(feature = "gc") && features.contains(WasmFeatures::EXCEPTIONS) {
2545            bail!("exceptions support requires garbage collection (GC) to be enabled in the build");
2546        }
2547
2548        match &self.rr_config {
2549            #[cfg(feature = "rr")]
2550            RRConfig::Recording | RRConfig::Replaying => {
2551                self.validate_rr_determinism_conflicts()?;
2552            }
2553            RRConfig::None => {}
2554        };
2555
2556        let mut tunables = Tunables::default_for_target(&self.compiler_target())?;
2557
2558        // By default this is enabled with the Cargo feature, and if the feature
2559        // is missing this is disabled.
2560        tunables.concurrency_support = cfg!(feature = "component-model-async");
2561
2562        #[cfg(feature = "rr")]
2563        {
2564            tunables.recording = matches!(self.rr_config, RRConfig::Recording);
2565        }
2566
2567        // If no target is explicitly specified then further refine `tunables`
2568        // for the configuration of this host depending on what platform
2569        // features were found available at compile time. This means that anyone
2570        // cross-compiling for a customized host will need to further refine
2571        // compilation options.
2572        if self.target.is_none() {
2573            // If this platform doesn't have native signals then change some
2574            // defaults to account for that. Note that VM guards are turned off
2575            // here because that's primarily a feature of eliding
2576            // bounds-checks.
2577            if !cfg!(has_native_signals) {
2578                tunables.signals_based_traps = cfg!(has_native_signals);
2579                tunables.memory_guard_size = 0;
2580                tunables.gc_heap_guard_size = 0;
2581            }
2582
2583            // When virtual memory is not available use slightly different
2584            // defaults for tunables to be more amenable to `MallocMemory`.
2585            // Note that these can still be overridden by config options.
2586            if !cfg!(has_virtual_memory) {
2587                tunables.memory_reservation = 0;
2588                tunables.memory_reservation_for_growth = 1 << 20; // 1MB
2589                tunables.memory_init_cow = false;
2590                tunables.gc_heap_reservation = 0;
2591                tunables.gc_heap_reservation_for_growth = 1 << 20; // 1MB
2592            }
2593        }
2594
2595        // If guest-debugging is enabled, we must disable
2596        // signals-based traps. Do this before we process the user's
2597        // provided tunables settings so we can detect a conflict with
2598        // an explicit request to use signals-based traps.
2599        #[cfg(feature = "debug")]
2600        if self.tunables.debug_guest == Some(true) {
2601            tunables.signals_based_traps = false;
2602        }
2603
2604        // Inlining currently falls over with the `stack_switch` instruction.
2605        #[cfg(any(feature = "cranelift", feature = "winch"))]
2606        if features.contains(WasmFeatures::STACK_SWITCHING) {
2607            if let Some(inlining) = self.tunables.inlining
2608                && inlining != Inlining::No
2609            {
2610                bail!("cannot enable compiler inlining when stack switching is enabled");
2611            }
2612            tunables.inlining = Inlining::No;
2613        }
2614
2615        self.tunables.configure(&mut tunables);
2616
2617        // If no GC heap tunables are explicitly configured, copy the memory
2618        // tunables' configured values so that GC heaps default to the same
2619        // configuration as linear memories.
2620        if !self.any_gc_heap_tunables_configured() {
2621            tunables.gc_heap_reservation = tunables.memory_reservation;
2622            tunables.gc_heap_guard_size = tunables.memory_guard_size;
2623            tunables.gc_heap_reservation_for_growth = tunables.memory_reservation_for_growth;
2624            tunables.gc_heap_may_move = tunables.memory_may_move;
2625        }
2626
2627        // If we're going to compile with winch, we must use the winch calling convention.
2628        #[cfg(any(feature = "cranelift", feature = "winch"))]
2629        {
2630            tunables.winch_callable = self
2631                .compiler_config
2632                .as_ref()
2633                .is_some_and(|c| c.strategy == Some(Strategy::Winch));
2634        }
2635
2636        tunables.collector = if features.gc_types() {
2637            #[cfg(feature = "gc")]
2638            {
2639                use wasmtime_environ::Collector as EnvCollector;
2640                Some(match self.collector.try_not_auto()? {
2641                    Collector::DeferredReferenceCounting => EnvCollector::DeferredReferenceCounting,
2642                    Collector::Null => EnvCollector::Null,
2643                    Collector::Copying => EnvCollector::Copying,
2644                    Collector::Auto => unreachable!(),
2645                })
2646            }
2647            #[cfg(not(feature = "gc"))]
2648            bail!("cannot use GC types: the `gc` feature was disabled at compile time")
2649        } else {
2650            None
2651        };
2652
2653        if tunables.debug_guest {
2654            ensure!(
2655                cfg!(feature = "debug"),
2656                "debug instrumentation support was disabled at compile time"
2657            );
2658            ensure!(
2659                !tunables.signals_based_traps,
2660                "cannot use signals-based traps with guest debugging enabled"
2661            );
2662        }
2663
2664        // Concurrency support is required for some component model features.
2665        let requires_concurrency = WasmFeatures::CM_ASYNC
2666            | WasmFeatures::CM_MORE_ASYNC_BUILTINS
2667            | WasmFeatures::CM_ASYNC_STACKFUL
2668            | WasmFeatures::CM_THREADING
2669            | WasmFeatures::CM_ERROR_CONTEXT;
2670        if tunables.concurrency_support && !cfg!(feature = "component-model-async") {
2671            bail!(
2672                "concurrency support was requested but was not \
2673                 compiled into this build of Wasmtime"
2674            )
2675        }
2676        if !tunables.concurrency_support && features.intersects(requires_concurrency) {
2677            bail!(
2678                "concurrency support must be enabled to use the component \
2679                 model async or threading features"
2680            )
2681        }
2682
2683        // If the pooling allocator is used and GC is enabled, check that
2684        // memories and the GC heap are configured identically, since the
2685        // pooling allocator can't support differently-configured heaps.
2686        #[cfg(feature = "pooling-allocator")]
2687        if matches!(
2688            &self.allocation_strategy,
2689            InstanceAllocationStrategy::Pooling(_)
2690        ) && tunables.collector.is_some()
2691        {
2692            if tunables.memory_reservation != tunables.gc_heap_reservation {
2693                bail!(
2694                    "when using the pooling allocator with GC, `memory_reservation` ({}) \
2695                     and `gc_heap_reservation` ({}) must be the same",
2696                    tunables.memory_reservation,
2697                    tunables.gc_heap_reservation,
2698                );
2699            }
2700            if tunables.memory_guard_size != tunables.gc_heap_guard_size {
2701                bail!(
2702                    "when using the pooling allocator with GC, `memory_guard_size` ({}) \
2703                     and `gc_heap_guard_size` ({}) must be the same",
2704                    tunables.memory_guard_size,
2705                    tunables.gc_heap_guard_size,
2706                );
2707            }
2708            if tunables.memory_reservation_for_growth != tunables.gc_heap_reservation_for_growth {
2709                bail!(
2710                    "when using the pooling allocator with GC, \
2711                     `memory_reservation_for_growth` ({}) and \
2712                     `gc_heap_reservation_for_growth` ({}) must be the same",
2713                    tunables.memory_reservation_for_growth,
2714                    tunables.gc_heap_reservation_for_growth,
2715                );
2716            }
2717            if tunables.memory_may_move != tunables.gc_heap_may_move {
2718                bail!(
2719                    "when using the pooling allocator with GC, `memory_may_move` ({}) \
2720                     and `gc_heap_may_move` ({}) must be the same",
2721                    tunables.memory_may_move,
2722                    tunables.gc_heap_may_move,
2723                );
2724            }
2725        }
2726
2727        Ok((tunables, features))
2728    }
2729
2730    #[cfg(feature = "runtime")]
2731    pub(crate) fn build_allocator(
2732        &self,
2733        tunables: &Tunables,
2734    ) -> Result<Box<dyn InstanceAllocator + Send + Sync>> {
2735        #[cfg(feature = "async")]
2736        let (stack_size, stack_zeroing) = (self.async_stack_size, self.async_stack_zeroing);
2737
2738        #[cfg(not(feature = "async"))]
2739        let (stack_size, stack_zeroing) = (0, false);
2740
2741        let _ = tunables;
2742
2743        match &self.allocation_strategy {
2744            InstanceAllocationStrategy::OnDemand => {
2745                let mut _allocator = try_new::<Box<_>>(OnDemandInstanceAllocator::new(
2746                    self.mem_creator.clone(),
2747                    stack_size,
2748                    stack_zeroing,
2749                ))?;
2750                #[cfg(feature = "async")]
2751                if let Some(stack_creator) = &self.stack_creator {
2752                    _allocator.set_stack_creator(stack_creator.clone());
2753                }
2754                Ok(_allocator as _)
2755            }
2756            #[cfg(feature = "pooling-allocator")]
2757            InstanceAllocationStrategy::Pooling(config) => {
2758                let mut config = config.config;
2759                config.stack_size = stack_size;
2760                config.async_stack_zeroing = stack_zeroing;
2761                let allocator = try_new::<Box<_>>(
2762                    crate::runtime::vm::PoolingInstanceAllocator::new(&config, tunables)?,
2763                )?;
2764                Ok(allocator as _)
2765            }
2766        }
2767    }
2768
2769    #[cfg(feature = "runtime")]
2770    pub(crate) fn build_gc_runtime(&self) -> Result<Option<Arc<dyn GcRuntime>>> {
2771        if !self.features().gc_types() {
2772            return Ok(None);
2773        }
2774
2775        #[cfg(not(feature = "gc"))]
2776        bail!("cannot create a GC runtime: the `gc` feature was disabled at compile time");
2777
2778        #[cfg(feature = "gc")]
2779        #[cfg_attr(
2780            not(any(feature = "gc-null", feature = "gc-drc", feature = "gc-copying")),
2781            expect(unreachable_code, reason = "definitions known to be dummy")
2782        )]
2783        {
2784            Ok(Some(match self.collector.try_not_auto()? {
2785                #[cfg(feature = "gc-drc")]
2786                Collector::DeferredReferenceCounting => {
2787                    try_new::<Arc<_>>(crate::runtime::vm::DrcCollector::default())? as _
2788                }
2789                #[cfg(not(feature = "gc-drc"))]
2790                Collector::DeferredReferenceCounting => unreachable!(),
2791
2792                #[cfg(feature = "gc-null")]
2793                Collector::Null => {
2794                    try_new::<Arc<_>>(crate::runtime::vm::NullCollector::default())? as _
2795                }
2796                #[cfg(not(feature = "gc-null"))]
2797                Collector::Null => unreachable!(),
2798
2799                #[cfg(feature = "gc-copying")]
2800                Collector::Copying => {
2801                    try_new::<Arc<_>>(crate::runtime::vm::CopyingCollector::default())? as _
2802                }
2803                #[cfg(not(feature = "gc-copying"))]
2804                Collector::Copying => unreachable!(),
2805
2806                Collector::Auto => unreachable!(),
2807            }))
2808        }
2809    }
2810
2811    #[cfg(feature = "runtime")]
2812    pub(crate) fn build_profiler(&self) -> Result<Box<dyn ProfilingAgent>> {
2813        Ok(match self.profiling_strategy {
2814            ProfilingStrategy::PerfMap => profiling_agent::new_perfmap()?,
2815            ProfilingStrategy::JitDump => profiling_agent::new_jitdump()?,
2816            ProfilingStrategy::VTune => profiling_agent::new_vtune()?,
2817            ProfilingStrategy::None => profiling_agent::new_null(),
2818            ProfilingStrategy::Pulley => profiling_agent::new_pulley()?,
2819        })
2820    }
2821
2822    #[cfg(any(feature = "cranelift", feature = "winch"))]
2823    pub(crate) fn build_compiler(
2824        mut self,
2825        tunables: &mut Tunables,
2826        features: WasmFeatures,
2827    ) -> Result<(Self, Box<dyn wasmtime_environ::Compiler>)> {
2828        let target = self.compiler_target();
2829
2830        // The target passed to the builders below is an `Option<Triple>` where
2831        // `None` represents the current host with CPU features inferred from
2832        // the host's CPU itself. The `target` above is not an `Option`, so
2833        // switch it to `None` in the case that a target wasn't explicitly
2834        // specified (which indicates no feature inference) and the target
2835        // matches the host.
2836        let target_for_builder =
2837            if self.target.is_none() && target == target_lexicon::Triple::host() {
2838                None
2839            } else {
2840                Some(target.clone())
2841            };
2842
2843        let mut compiler = match self.compiler_config_mut().strategy {
2844            #[cfg(feature = "cranelift")]
2845            Some(Strategy::Cranelift) => wasmtime_cranelift::builder(target_for_builder)?,
2846            #[cfg(not(feature = "cranelift"))]
2847            Some(Strategy::Cranelift) => bail!("cranelift support not compiled in"),
2848            #[cfg(feature = "winch")]
2849            Some(Strategy::Winch) => wasmtime_winch::builder(target_for_builder)?,
2850            #[cfg(not(feature = "winch"))]
2851            Some(Strategy::Winch) => bail!("winch support not compiled in"),
2852
2853            None | Some(Strategy::Auto) => unreachable!(),
2854        };
2855
2856        if let Some(path) = &self.compiler_config_mut().clif_dir {
2857            compiler.clif_dir(path)?;
2858        }
2859
2860        // If probestack is enabled for a target, Wasmtime will always use the
2861        // inline strategy which doesn't require us to define a `__probestack`
2862        // function or similar.
2863        self.compiler_config_mut()
2864            .settings
2865            .insert("probestack_strategy".into(), "inline".into());
2866
2867        // We enable stack probing by default on all targets.
2868        // This is required on Windows because of the way Windows
2869        // commits its stacks, but it's also a good idea on other
2870        // platforms to ensure guard pages are hit for large frame
2871        // sizes.
2872        self.compiler_config_mut()
2873            .flags
2874            .insert("enable_probestack".into());
2875
2876        // The current wasm multivalue implementation depends on this.
2877        // FIXME(#9510) handle this in wasmtime-cranelift instead.
2878        self.compiler_config_mut()
2879            .flags
2880            .insert("enable_multi_ret_implicit_sret".into());
2881
2882        if let Some(unwind_requested) = self.native_unwind_info {
2883            if !self
2884                .compiler_config_mut()
2885                .ensure_setting_unset_or_given("unwind_info", &unwind_requested.to_string())
2886            {
2887                bail!(
2888                    "incompatible settings requested for Cranelift and Wasmtime `unwind-info` settings"
2889                );
2890            }
2891        }
2892
2893        if target.operating_system == target_lexicon::OperatingSystem::Windows {
2894            if !self
2895                .compiler_config_mut()
2896                .ensure_setting_unset_or_given("unwind_info", "true")
2897            {
2898                bail!("`native_unwind_info` cannot be disabled on Windows");
2899            }
2900        }
2901
2902        // We require frame pointers for correct stack walking, which is safety
2903        // critical in the presence of reference types, and otherwise it is just
2904        // really bad developer experience to get wrong.
2905        self.compiler_config_mut()
2906            .settings
2907            .insert("preserve_frame_pointers".into(), "true".into());
2908
2909        if !tunables.signals_based_traps {
2910            let mut ok = self
2911                .compiler_config_mut()
2912                .ensure_setting_unset_or_given("enable_table_access_spectre_mitigation", "false");
2913            ok = ok
2914                && self.compiler_config_mut().ensure_setting_unset_or_given(
2915                    "enable_heap_access_spectre_mitigation",
2916                    "false",
2917                );
2918
2919            // Right now spectre-mitigated bounds checks will load from zero so
2920            // if host-based signal handlers are disabled then that's a mismatch
2921            // and doesn't work right now. Fixing this will require more thought
2922            // of how to implement the bounds check in spectre-only mode.
2923            if !ok {
2924                bail!(
2925                    "when signals-based traps are disabled then spectre \
2926                     mitigations must also be disabled"
2927                );
2928            }
2929        }
2930
2931        if features.contains(WasmFeatures::RELAXED_SIMD) && !features.contains(WasmFeatures::SIMD) {
2932            bail!("cannot disable the simd proposal but enable the relaxed simd proposal");
2933        }
2934
2935        if features.contains(WasmFeatures::STACK_SWITCHING) {
2936            use target_lexicon::OperatingSystem;
2937            let model = match target.operating_system {
2938                OperatingSystem::Windows => "update_windows_tib",
2939                OperatingSystem::Linux
2940                | OperatingSystem::MacOSX(_)
2941                | OperatingSystem::Darwin(_) => "basic",
2942                _ => bail!("stack-switching feature not supported on this platform "),
2943            };
2944
2945            if !self
2946                .compiler_config_mut()
2947                .ensure_setting_unset_or_given("stack_switch_model", model)
2948            {
2949                bail!(
2950                    "compiler option 'stack_switch_model' must be set to '{model}' on this platform"
2951                );
2952            }
2953        }
2954
2955        // Apply compiler settings and flags
2956        compiler.set_tunables(tunables.clone())?;
2957        for (k, v) in self.compiler_config_mut().settings.iter() {
2958            compiler.set(k, v)?;
2959        }
2960        for flag in self.compiler_config_mut().flags.iter() {
2961            compiler.enable(flag)?;
2962        }
2963        *tunables = compiler.tunables().cloned().unwrap();
2964
2965        #[cfg(all(feature = "incremental-cache", feature = "cranelift"))]
2966        if let Some(cache_store) = &self.compiler_config_mut().cache_store {
2967            compiler.enable_incremental_compilation(cache_store.clone())?;
2968        }
2969
2970        compiler.wmemcheck(self.compiler_config_mut().wmemcheck);
2971
2972        Ok((self, compiler.build()?))
2973    }
2974
2975    /// Internal setting for whether adapter modules for components will have
2976    /// extra WebAssembly instructions inserted performing more debug checks
2977    /// then are necessary.
2978    #[cfg(feature = "component-model")]
2979    pub fn debug_adapter_modules(&mut self, debug: bool) -> &mut Self {
2980        self.tunables.debug_adapter_modules = Some(debug);
2981        self
2982    }
2983
2984    /// Enables clif output when compiling a WebAssembly module.
2985    #[cfg(any(feature = "cranelift", feature = "winch"))]
2986    pub fn emit_clif(&mut self, path: &Path) -> &mut Self {
2987        self.compiler_config_mut().clif_dir = Some(path.to_path_buf());
2988        self
2989    }
2990
2991    /// Configures whether, when on macOS, Mach ports are used for exception
2992    /// handling instead of traditional Unix-based signal handling.
2993    ///
2994    /// WebAssembly traps in Wasmtime are implemented with native faults, for
2995    /// example a `SIGSEGV` will occur when a WebAssembly guest accesses
2996    /// out-of-bounds memory. Handling this can be configured to either use Unix
2997    /// signals or Mach ports on macOS. By default Mach ports are used.
2998    ///
2999    /// Mach ports enable Wasmtime to work by default with foreign
3000    /// error-handling systems such as breakpad which also use Mach ports to
3001    /// handle signals. In this situation Wasmtime will continue to handle guest
3002    /// faults gracefully while any non-guest faults will get forwarded to
3003    /// process-level handlers such as breakpad. Some more background on this
3004    /// can be found in #2456.
3005    ///
3006    /// A downside of using mach ports, however, is that they don't interact
3007    /// well with `fork()`. Forking a Wasmtime process on macOS will produce a
3008    /// child process that cannot successfully run WebAssembly. In this
3009    /// situation traditional Unix signal handling should be used as that's
3010    /// inherited and works across forks.
3011    ///
3012    /// If your embedding wants to use a custom error handler which leverages
3013    /// Mach ports and you additionally wish to `fork()` the process and use
3014    /// Wasmtime in the child process that's not currently possible. Please
3015    /// reach out to us if you're in this bucket!
3016    ///
3017    /// This option defaults to `true`, using Mach ports by default.
3018    pub fn macos_use_mach_ports(&mut self, mach_ports: bool) -> &mut Self {
3019        self.macos_use_mach_ports = mach_ports;
3020        self
3021    }
3022
3023    /// Configures an embedder-provided function, `detect`, which is used to
3024    /// determine if an ISA-specific feature is available on the current host.
3025    ///
3026    /// This function is used to verify that any features enabled for a compiler
3027    /// backend, such as AVX support on x86\_64, are also available on the host.
3028    /// It is undefined behavior to execute an AVX instruction on a host that
3029    /// doesn't support AVX instructions, for example.
3030    ///
3031    /// When the `std` feature is active on this crate then this function is
3032    /// configured to a default implementation that uses the standard library's
3033    /// feature detection. When the `std` feature is disabled then there is no
3034    /// default available and this method must be called to configure a feature
3035    /// probing function.
3036    ///
3037    /// The `detect` function provided is given a string name of an ISA feature.
3038    /// The function should then return:
3039    ///
3040    /// * `Some(true)` - indicates that the feature was found on the host and it
3041    ///   is supported.
3042    /// * `Some(false)` - the feature name was recognized but it was not
3043    ///   detected on the host, for example the CPU is too old.
3044    /// * `None` - the feature name was not recognized and it's not known
3045    ///   whether it's on the host or not.
3046    ///
3047    /// Feature names passed to `detect` match the same feature name used in the
3048    /// Rust standard library. For example `"sse4.2"` is used on x86\_64.
3049    ///
3050    /// # Unsafety
3051    ///
3052    /// This function is `unsafe` because it is undefined behavior to execute
3053    /// instructions that a host does not support. This means that the result of
3054    /// `detect` must be correct for memory safe execution at runtime.
3055    pub unsafe fn detect_host_feature(&mut self, detect: fn(&str) -> Option<bool>) -> &mut Self {
3056        self.detect_host_feature = Some(detect);
3057        self
3058    }
3059
3060    /// Configures Wasmtime to not use signals-based trap handlers, for example
3061    /// disables `SIGILL` and `SIGSEGV` handler registration on Unix platforms.
3062    ///
3063    /// > **Note:** this option has important performance ramifications, be sure
3064    /// > to understand the implications. Wasm programs have been measured to
3065    /// > run up to 2x slower when signals-based traps are disabled.
3066    ///
3067    /// Wasmtime will by default leverage signals-based trap handlers (or the
3068    /// platform equivalent, for example "vectored exception handlers" on
3069    /// Windows) to make generated code more efficient. For example, when
3070    /// Wasmtime can use signals-based traps, it can elide explicit bounds
3071    /// checks for Wasm linear memory accesses, instead relying on virtual
3072    /// memory guard pages to raise a `SIGSEGV` (on Unix) for out-of-bounds
3073    /// accesses, which Wasmtime's runtime then catches and handles. Another
3074    /// example is divide-by-zero: with signals-based traps, Wasmtime can let
3075    /// the hardware raise a trap when the divisor is zero. Without
3076    /// signals-based traps, Wasmtime must explicitly emit additional
3077    /// instructions to check for zero and conditionally branch to a trapping
3078    /// code path.
3079    ///
3080    /// Some environments however may not have access to signal handlers. For
3081    /// example embedded scenarios may not support virtual memory. Other
3082    /// environments where Wasmtime is embedded within the surrounding
3083    /// environment may require that new signal handlers aren't registered due
3084    /// to the global nature of signal handlers. This option exists to disable
3085    /// the signal handler registration when required for these scenarios.
3086    ///
3087    /// When signals-based trap handlers are disabled, then Wasmtime and its
3088    /// generated code will *never* rely on segfaults or other
3089    /// signals. Generated code will be slower because bounds must be explicitly
3090    /// checked along with other conditions like division by zero.
3091    ///
3092    /// The following additional factors can also affect Wasmtime's ability to
3093    /// elide explicit bounds checks and leverage signals-based traps:
3094    ///
3095    /// * The [`Config::memory_reservation`] and [`Config::memory_guard_size`]
3096    ///   settings
3097    /// * The index type of the linear memory (e.g. 32-bit or 64-bit)
3098    /// * The page size of the linear memory
3099    ///
3100    /// When this option is disabled, the
3101    /// `enable_heap_access_spectre_mitigation` and
3102    /// `enable_table_access_spectre_mitigation` Cranelift settings must also be
3103    /// disabled. This means that generated code must have spectre mitigations
3104    /// disabled. This is because spectre mitigations rely on faults from
3105    /// loading from the null address to implement bounds checks.
3106    ///
3107    /// This option defaults to `true`: signals-based trap handlers are enabled
3108    /// by default.
3109    ///
3110    /// > **Note:** Disabling this option is not compatible with the Winch
3111    /// > compiler.
3112    pub fn signals_based_traps(&mut self, enable: bool) -> &mut Self {
3113        self.tunables.signals_based_traps = Some(enable);
3114        self
3115    }
3116
3117    /// Enable/disable GC support in Wasmtime entirely.
3118    ///
3119    /// This flag can be used to gate whether GC infrastructure is enabled or
3120    /// initialized in Wasmtime at all. Wasmtime's GC implementation is required
3121    /// for the [`Self::wasm_gc`] proposal, [`Self::wasm_function_references`],
3122    /// and [`Self::wasm_exceptions`] at this time. None of those proposal can
3123    /// be enabled without also having this option enabled.
3124    ///
3125    /// This option defaults to whether the crate `gc` feature is enabled or
3126    /// not.
3127    pub fn gc_support(&mut self, enable: bool) -> &mut Self {
3128        self.wasm_features(WasmFeatures::GC_TYPES, enable)
3129    }
3130
3131    /// Explicitly indicate or not whether the host is using a hardware float
3132    /// ABI on x86 targets.
3133    ///
3134    /// This configuration option is only applicable on the
3135    /// `x86_64-unknown-none` Rust target and has no effect on other host
3136    /// targets. The `x86_64-unknown-none` Rust target does not support hardware
3137    /// floats by default and uses a "soft float" implementation and ABI. This
3138    /// means that `f32`, for example, is passed in a general-purpose register
3139    /// between functions instead of a floating-point register. This does not
3140    /// match Cranelift's ABI for `f32` where it's passed in floating-point
3141    /// registers.  Cranelift does not have support for a "soft float"
3142    /// implementation where all floating-point operations are lowered to
3143    /// libcalls.
3144    ///
3145    /// This means that for the `x86_64-unknown-none` target the ABI between
3146    /// Wasmtime's libcalls and the host is incompatible when floats are used.
3147    /// This further means that, by default, Wasmtime is unable to load native
3148    /// code when compiled to the `x86_64-unknown-none` target. The purpose of
3149    /// this option is to explicitly allow loading code and bypass this check.
3150    ///
3151    /// Setting this configuration option to `true` indicates that either:
3152    /// (a) the Rust target is compiled with the hard-float ABI manually via
3153    /// `-Zbuild-std` and a custom target JSON configuration, or (b) sufficient
3154    /// x86 features have been enabled in the compiler such that float libcalls
3155    /// will not be used in Wasmtime. For (a) there is no way in Rust at this
3156    /// time to detect whether a hard-float or soft-float ABI is in use on
3157    /// stable Rust, so this manual opt-in is required. For (b) the only
3158    /// instance where Wasmtime passes a floating-point value in a register
3159    /// between the host and compiled wasm code is with libcalls.
3160    ///
3161    /// Float-based libcalls are only used when the compilation target for a
3162    /// wasm module has insufficient target features enabled for native
3163    /// support. For example SSE4.1 is required for the `f32.ceil` WebAssembly
3164    /// instruction to be compiled to a native instruction. If SSE4.1 is not
3165    /// enabled then `f32.ceil` is translated to a "libcall" which is
3166    /// implemented on the host. Float-based libcalls can be avoided with
3167    /// sufficient target features enabled, for example:
3168    ///
3169    /// * `self.cranelift_flag_enable("has_sse3")`
3170    /// * `self.cranelift_flag_enable("has_ssse3")`
3171    /// * `self.cranelift_flag_enable("has_sse41")`
3172    /// * `self.cranelift_flag_enable("has_sse42")`
3173    /// * `self.cranelift_flag_enable("has_fma")`
3174    ///
3175    /// Note that when these features are enabled Wasmtime will perform a
3176    /// runtime check to determine that the host actually has the feature
3177    /// present.
3178    ///
3179    /// For some more discussion see [#11506].
3180    ///
3181    /// [#11506]: https://github.com/bytecodealliance/wasmtime/issues/11506
3182    ///
3183    /// # Safety
3184    ///
3185    /// This method is not safe because it cannot be detected in Rust right now
3186    /// whether the host is compiled with a soft or hard float ABI. Additionally
3187    /// if the host is compiled with a soft float ABI disabling this check does
3188    /// not ensure that the wasm module in question has zero usage of floats
3189    /// in the boundary to the host.
3190    ///
3191    /// Safely using this method requires one of:
3192    ///
3193    /// * The host target is compiled to use hardware floats.
3194    /// * Wasm modules loaded are compiled with enough x86 Cranelift features
3195    ///   enabled to avoid float-related hostcalls.
3196    pub unsafe fn x86_float_abi_ok(&mut self, enable: bool) -> &mut Self {
3197        self.x86_float_abi_ok = Some(enable);
3198        self
3199    }
3200
3201    /// Enable or disable the ability to create a
3202    /// [`SharedMemory`](crate::SharedMemory).
3203    ///
3204    /// The WebAssembly threads proposal, configured by [`Config::wasm_threads`]
3205    /// is on-by-default but there are enough deficiencies in Wasmtime's
3206    /// implementation and API integration that creation of a shared memory is
3207    /// disabled by default. This configuration knob can be used to enable this.
3208    ///
3209    /// When enabling this method be aware that wasm threads are, at this time,
3210    /// a [tier 2
3211    /// feature](https://docs.wasmtime.dev/stability-tiers.html#tier-2) in
3212    /// Wasmtime meaning that it will not receive security updates or fixes to
3213    /// historical releases. Additionally security CVEs will not be issued for
3214    /// bugs in the implementation.
3215    ///
3216    /// This option is `false` by default.
3217    pub fn shared_memory(&mut self, enable: bool) -> &mut Self {
3218        self.shared_memory = enable;
3219        self
3220    }
3221
3222    /// Specifies whether support for concurrent execution of WebAssembly is
3223    /// supported within this store.
3224    ///
3225    /// This configuration option affects whether runtime data structures are
3226    /// initialized within a `Store` on creation to support concurrent execution
3227    /// of WebAssembly guests. This is primarily applicable to the
3228    /// [`Config::wasm_component_model_async`] configuration which is the first
3229    /// time Wasmtime has supported concurrent execution of guests. This
3230    /// configuration option, for example, enables usage of
3231    /// [`Store::run_concurrent`], [`Func::call_concurrent`], [`StreamReader`],
3232    /// etc.
3233    ///
3234    /// This configuration option can be manually disabled to avoid initializing
3235    /// data structures in the [`Store`] related to concurrent execution. When
3236    /// this option is disabled then APIs related to concurrency will all fail
3237    /// with a panic. For example [`Store::run_concurrent`] will panic, creating
3238    /// a [`StreamReader`] will panic, etc.
3239    ///
3240    /// The value of this option additionally affects whether a [`Config`] is
3241    /// valid and the default set of enabled WebAssembly features. If this
3242    /// option is disabled then component-model features related to concurrency
3243    /// will all be disabled. If this option is enabled, then the options will
3244    /// retain their normal defaults. It is not valid to create a [`Config`]
3245    /// with component-model-async explicitly enabled and this option explicitly
3246    /// disabled, however.
3247    ///
3248    /// This option defaults to `true`.
3249    ///
3250    /// [`Store`]: crate::Store
3251    /// [`Store::run_concurrent`]: crate::Store::run_concurrent
3252    /// [`Func::call_concurrent`]: crate::component::Func::call_concurrent
3253    /// [`StreamReader`]: crate::component::StreamReader
3254    pub fn concurrency_support(&mut self, enable: bool) -> &mut Self {
3255        self.tunables.concurrency_support = Some(enable);
3256        self
3257    }
3258
3259    /// Validate if the current configuration has conflicting overrides that prevent
3260    /// execution determinism. Returns an error if a conflict exists.
3261    ///
3262    /// Note: Keep this in sync with [`Config::enforce_determinism`].
3263    #[inline]
3264    #[cfg(feature = "rr")]
3265    pub(crate) fn validate_rr_determinism_conflicts(&self) -> Result<()> {
3266        if let Some(v) = self.tunables.relaxed_simd_deterministic {
3267            if v == false {
3268                bail!("Relaxed deterministic SIMD cannot be disabled when determinism is enforced");
3269            }
3270        }
3271        #[cfg(any(feature = "cranelift", feature = "winch"))]
3272        if let Some(v) = self
3273            .compiler_config
3274            .as_ref()
3275            .and_then(|c| c.settings.get("enable_nan_canonicalization"))
3276        {
3277            if v != "true" {
3278                bail!("NaN canonicalization cannot be disabled when determinism is enforced");
3279            }
3280        }
3281        Ok(())
3282    }
3283
3284    /// Enable execution trace recording or replaying to the configuration.
3285    ///
3286    /// When either recording/replaying are enabled, validation fails if settings
3287    /// that control determinism are not set appropriately. In particular, RR requires
3288    /// doing the following:
3289    /// * Enabling NaN canonicalization with [`Config::cranelift_nan_canonicalization`].
3290    /// * Enabling deterministic relaxed SIMD with [`Config::relaxed_simd_deterministic`].
3291    #[inline]
3292    pub fn rr(&mut self, cfg: RRConfig) -> &mut Self {
3293        self.rr_config = cfg;
3294        self
3295    }
3296
3297    /// Whether or not trap metadata is generated in compiled wasms for internal
3298    /// asserts in the compiled code itself.
3299    ///
3300    /// Wasmtime inserts metadata within compiled artifacts which contain a
3301    /// table of known trap codes for all instructions. If a trap via a signal
3302    /// happens, and it's not listed in these tables, then that's considered a
3303    /// fatal bug that crashes the process. This option controls whether trap
3304    /// codes are inserted into metadata for internal asserts as part of
3305    /// Wasmtime's translation process. These internal asserts should never be
3306    /// triggered, but if they are then the process dies with a signal.
3307    ///
3308    /// Inserting trap metadata into compiled artifacts can take extra space in
3309    /// the final artifact. The trap tables for the artifact will be larger as
3310    /// they contain more trap codes to contain.
3311    ///
3312    /// This is intended as a debugging option and is set to `false` by
3313    /// default.
3314    pub fn metadata_for_internal_asserts(&mut self, enable: bool) -> &mut Self {
3315        self.tunables.metadata_for_internal_asserts = Some(enable);
3316        self
3317    }
3318
3319    /// Whether or not trap metadata is generated in compiled wasms for
3320    /// detection of corruption in the GC heap.
3321    ///
3322    /// For more information about what metadata is in this scenario, see
3323    /// [`Config::metadata_for_internal_asserts`]. Note, though, that this
3324    /// option is enabled by default unlike internal asserts. This is intended
3325    /// as a defense-in-depth option for generated code in the face of GC heap
3326    /// corruption. If the GC heap is corrupted and is detected then the
3327    /// trapping instruction will be gracefully handled and delivered to the
3328    /// embedder. Otherwise if this option were set to `false` then the process
3329    /// would be aborted due to a signal.
3330    pub fn metadata_for_gc_heap_corruption(&mut self, enable: bool) -> &mut Self {
3331        self.tunables.metadata_for_gc_heap_corruption = Some(enable);
3332        self
3333    }
3334}
3335
3336impl Default for Config {
3337    fn default() -> Config {
3338        Config::new()
3339    }
3340}
3341
3342impl fmt::Debug for Config {
3343    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
3344        let mut f = f.debug_struct("Config");
3345
3346        // Not every flag in WasmFeatures can be enabled as part of creating
3347        // a Config. This impl gives a complete picture of all WasmFeatures
3348        // enabled, and doesn't require maintenance by hand (which has become out
3349        // of date in the past), at the cost of possible confusion for why
3350        // a flag in this set doesn't have a Config setter.
3351        let features = self.features();
3352        for flag in WasmFeatures::FLAGS.iter() {
3353            f.field(
3354                &format!("wasm_{}", flag.name().to_lowercase()),
3355                &features.contains(*flag.value()),
3356            );
3357        }
3358
3359        f.field("parallel_compilation", &self.parallel_compilation);
3360        #[cfg(any(feature = "cranelift", feature = "winch"))]
3361        {
3362            f.field("compiler_config", &self.compiler_config);
3363        }
3364
3365        self.tunables.format(&mut f);
3366        f.finish()
3367    }
3368}
3369
3370/// Possible Compilation strategies for a wasm module.
3371///
3372/// This is used as an argument to the [`Config::strategy`] method.
3373#[non_exhaustive]
3374#[derive(PartialEq, Eq, Clone, Debug, Copy)]
3375pub enum Strategy {
3376    /// An indicator that the compilation strategy should be automatically
3377    /// selected.
3378    ///
3379    /// This is generally what you want for most projects and indicates that the
3380    /// `wasmtime` crate itself should make the decision about what the best
3381    /// code generator for a wasm module is.
3382    ///
3383    /// Currently this always defaults to Cranelift, but the default value may
3384    /// change over time.
3385    Auto,
3386
3387    /// Currently the default backend, Cranelift aims to be a reasonably fast
3388    /// code generator which generates high quality machine code.
3389    Cranelift,
3390
3391    /// A low-latency baseline compiler for WebAssembly.
3392    /// For more details regarding ISA support and Wasm proposals support
3393    /// see <https://docs.wasmtime.dev/stability-tiers.html#current-tier-status>
3394    Winch,
3395}
3396
3397#[cfg(any(feature = "winch", feature = "cranelift"))]
3398impl Strategy {
3399    fn not_auto(&self) -> Option<Strategy> {
3400        match self {
3401            Strategy::Auto => {
3402                if cfg!(feature = "cranelift") {
3403                    Some(Strategy::Cranelift)
3404                } else if cfg!(feature = "winch") {
3405                    Some(Strategy::Winch)
3406                } else {
3407                    None
3408                }
3409            }
3410            other => Some(*other),
3411        }
3412    }
3413}
3414
3415/// Possible garbage collector implementations for Wasm.
3416///
3417/// This is used as an argument to the [`Config::collector`] method.
3418///
3419/// The properties of Wasmtime's available collectors are summarized in the
3420/// following table:
3421///
3422/// | Collector                   | Collects Garbage[^1]  | Latency[^2] | Throughput[^3] | Allocation Speed[^4] | Heap Utilization[^5] |
3423/// |-----------------------------|-----------------------|-------------|----------------|----------------------|----------------------|
3424/// | `Copying`                   | Yes, including cycles | 🙁         | 🙂             | 🙂                   | 🙁                  |
3425/// | `DeferredReferenceCounting` | Yes, but not cycles   | 🙂         | 🙁             | 😐                   | 😐                  |
3426/// | `Null`                      | No                    | 🙂         | 🙂             | 🙂                   | 🙂                  |
3427///
3428/// [^1]: Whether or not the collector is capable of collecting garbage and cyclic garbage.
3429///
3430/// [^2]: How long the Wasm program is paused during garbage
3431///       collections. Shorter is better. In general, better latency implies
3432///       worse throughput and vice versa.
3433///
3434/// [^3]: How fast the Wasm program runs when using this collector. Roughly
3435///       equivalent to the number of Wasm instructions executed per
3436///       second. Faster is better. In general, better throughput implies worse
3437///       latency and vice versa.
3438///
3439/// [^4]: How fast can individual objects be allocated?
3440///
3441/// [^5]: How many objects can the collector fit into N bytes of memory? That
3442///       is, how much space for bookkeeping and metadata does this collector
3443///       require? Less space taken up by metadata means more space for
3444///       additional objects. Reference counts are larger than mark bits and
3445///       free lists are larger than bump pointers, for example.
3446#[non_exhaustive]
3447#[derive(PartialEq, Eq, Clone, Debug, Copy)]
3448pub enum Collector {
3449    /// An indicator that the garbage collector should be automatically
3450    /// selected.
3451    ///
3452    /// This is generally what you want for most projects and indicates that the
3453    /// `wasmtime` crate itself should make the decision about what the best
3454    /// collector to use is.
3455    ///
3456    /// Currently this always defaults to the copying collector, but the default
3457    /// value may change over time.
3458    Auto,
3459
3460    /// The deferred reference-counting collector.
3461    ///
3462    /// A reference-counting collector, generally trading improved latency for
3463    /// worsened throughput. However, to avoid the largest overheads of
3464    /// reference counting, it avoids manipulating reference counts for Wasm
3465    /// objects on the stack. Instead, it will hold a reference count for an
3466    /// over-approximation of all objects that are currently on the stack, trace
3467    /// the stack during collection to find the precise set of on-stack roots,
3468    /// and decrement the reference count of any object that was in the
3469    /// over-approximation but not the precise set. This improves throughput,
3470    /// compared to "pure" reference counting, by performing many fewer
3471    /// refcount-increment and -decrement operations. The cost is the increased
3472    /// latency associated with tracing the stack.
3473    ///
3474    /// This collector cannot currently collect cycles; they will leak until the
3475    /// GC heap's store is dropped.
3476    DeferredReferenceCounting,
3477
3478    /// The null collector.
3479    ///
3480    /// This collector does not actually collect any garbage. It simply
3481    /// allocates objects until it runs out of memory, at which point further
3482    /// objects allocation attempts will trap.
3483    ///
3484    /// This collector is useful for incredibly short-running Wasm instances
3485    /// where additionally you would rather halt an over-allocating Wasm program
3486    /// than spend time collecting its garbage to allow it to keep running. It
3487    /// is also useful for measuring the overheads associated with other
3488    /// collectors, as this collector imposes as close to zero throughput and
3489    /// latency overhead as possible.
3490    Null,
3491
3492    /// The copying collector.
3493    ///
3494    /// A tracing collector that splits the GC heap in half, bump-allocates
3495    /// objects in one half until it fills up, and then does a GC and copies
3496    /// live objects into the other half, and repeats the process. It has fast
3497    /// allocation, collects cyclic garbage, and good collection throughput,
3498    /// however it suffers from poor latency due to its stop-the-world
3499    /// collections and poor heap utilization due to only using half the GC
3500    /// heap's full capacity at any given time.
3501    ///
3502    /// Note that this collector is still under construction and is not yet
3503    /// functional.
3504    Copying,
3505}
3506
3507impl Default for Collector {
3508    fn default() -> Collector {
3509        Collector::Auto
3510    }
3511}
3512
3513#[cfg(feature = "gc")]
3514impl Collector {
3515    fn not_auto(&self) -> Option<Collector> {
3516        match self {
3517            Collector::Auto => {
3518                if cfg!(feature = "gc-copying") {
3519                    Some(Collector::Copying)
3520                } else if cfg!(feature = "gc-drc") {
3521                    Some(Collector::DeferredReferenceCounting)
3522                } else if cfg!(feature = "gc-null") {
3523                    Some(Collector::Null)
3524                } else {
3525                    None
3526                }
3527            }
3528            other => Some(*other),
3529        }
3530    }
3531
3532    fn try_not_auto(&self) -> Result<Self> {
3533        match self.not_auto() {
3534            #[cfg(feature = "gc-drc")]
3535            Some(c @ Collector::DeferredReferenceCounting) => Ok(c),
3536            #[cfg(not(feature = "gc-drc"))]
3537            Some(Collector::DeferredReferenceCounting) => bail!(
3538                "cannot create an engine using the deferred reference-counting \
3539                 collector because the `gc-drc` feature was not enabled at \
3540                 compile time",
3541            ),
3542
3543            #[cfg(feature = "gc-null")]
3544            Some(c @ Collector::Null) => Ok(c),
3545            #[cfg(not(feature = "gc-null"))]
3546            Some(Collector::Null) => bail!(
3547                "cannot create an engine using the null collector because \
3548                 the `gc-null` feature was not enabled at compile time",
3549            ),
3550
3551            #[cfg(feature = "gc-copying")]
3552            Some(c @ Collector::Copying) => Ok(c),
3553            #[cfg(not(feature = "gc-copying"))]
3554            Some(Collector::Copying) => bail!(
3555                "cannot create an engine using the copying collector because \
3556                 the `gc-copying` feature was not enabled at compile time",
3557            ),
3558
3559            Some(Collector::Auto) => unreachable!(),
3560
3561            None => bail!(
3562                "cannot create an engine with GC support when none of the \
3563                 collectors are available; enable one of the following \
3564                 features: `gc-drc`, `gc-null`, `gc-copying`",
3565            ),
3566        }
3567    }
3568}
3569
3570/// Possible optimization levels for the Cranelift codegen backend.
3571#[non_exhaustive]
3572#[derive(Copy, Clone, Debug, Eq, PartialEq)]
3573pub enum OptLevel {
3574    /// No optimizations performed, minimizes compilation time by disabling most
3575    /// optimizations.
3576    None,
3577    /// Generates the fastest possible code, but may take longer.
3578    Speed,
3579    /// Similar to `speed`, but also performs transformations aimed at reducing
3580    /// code size.
3581    SpeedAndSize,
3582}
3583
3584/// Possible register allocator algorithms for the Cranelift codegen backend.
3585#[non_exhaustive]
3586#[derive(Copy, Clone, Debug, Eq, PartialEq)]
3587pub enum RegallocAlgorithm {
3588    /// Generates the fastest possible code, but may take longer.
3589    ///
3590    /// This algorithm performs "backtracking", which means that it may
3591    /// undo its earlier work and retry as it discovers conflicts. This
3592    /// results in better register utilization, producing fewer spills
3593    /// and moves, but can cause super-linear compile runtime.
3594    Backtracking,
3595    /// Generates acceptable code very quickly.
3596    ///
3597    /// This algorithm performs a single pass through the code,
3598    /// guaranteed to work in linear time.  (Note that the rest of
3599    /// Cranelift is not necessarily guaranteed to run in linear time,
3600    /// however.) It cannot undo earlier decisions, however, and it
3601    /// cannot foresee constraints or issues that may occur further
3602    /// ahead in the code, so the code may have more spills and moves as
3603    /// a result.
3604    ///
3605    /// > **Note**: This algorithm is not yet production-ready and has
3606    /// > historically had known problems. It is not recommended to enable this
3607    /// > algorithm for security-sensitive applications and the Wasmtime project
3608    /// > does not consider this configuration option for issuing security
3609    /// > advisories at this time.
3610    SinglePass,
3611}
3612
3613/// Select which profiling technique to support.
3614#[derive(Debug, Clone, Copy, PartialEq)]
3615pub enum ProfilingStrategy {
3616    /// No profiler support.
3617    None,
3618
3619    /// Collect function name information as the "perf map" file format, used with `perf` on Linux.
3620    PerfMap,
3621
3622    /// Collect profiling info for "jitdump" file format, used with `perf` on
3623    /// Linux.
3624    JitDump,
3625
3626    /// Collect profiling info using the "ittapi", used with `VTune` on Linux.
3627    VTune,
3628
3629    /// Support for profiling Pulley, Wasmtime's interpreter. Note that enabling
3630    /// this at runtime requires enabling the `profile-pulley` Cargo feature at
3631    /// compile time.
3632    Pulley,
3633}
3634
3635/// Select how wasm backtrace detailed information is handled.
3636#[derive(Debug, Clone, Copy)]
3637pub enum WasmBacktraceDetails {
3638    /// Support is unconditionally enabled and wasmtime will parse and read
3639    /// debug information.
3640    Enable,
3641
3642    /// Support is disabled, and wasmtime will not parse debug information for
3643    /// backtrace details.
3644    Disable,
3645
3646    /// Support for backtrace details is conditional on the
3647    /// `WASMTIME_BACKTRACE_DETAILS` environment variable.
3648    Environment,
3649}
3650
3651/// Describe the tri-state configuration of keys such as MPK or PAGEMAP_SCAN.
3652#[derive(Clone, Copy, Debug, Eq, PartialEq, Hash)]
3653pub enum Enabled {
3654    /// Enable this feature if it's detected on the host system, otherwise leave
3655    /// it disabled.
3656    Auto,
3657    /// Enable this feature and fail configuration if the feature is not
3658    /// detected on the host system.
3659    Yes,
3660    /// Do not enable this feature, even if the host system supports it.
3661    No,
3662}
3663
3664/// Configuration options used with [`InstanceAllocationStrategy::Pooling`] to
3665/// change the behavior of the pooling instance allocator.
3666///
3667/// This structure has a builder-style API in the same manner as [`Config`] and
3668/// is configured with [`Config::allocation_strategy`].
3669///
3670/// Note that usage of the pooling allocator does not affect compiled
3671/// WebAssembly code. Compiled `*.cwasm` files, for example, are usable both
3672/// with and without the pooling allocator.
3673///
3674/// ## Advantages of Pooled Allocation
3675///
3676/// The main benefit of the pooling allocator is to make WebAssembly
3677/// instantiation both faster and more scalable in terms of parallelism.
3678/// Allocation is faster because virtual memory is already configured and ready
3679/// to go within the pool, there's no need to [`mmap`] (for example on Unix) a
3680/// new region and configure it with guard pages. By avoiding [`mmap`] this
3681/// avoids whole-process virtual memory locks which can improve scalability and
3682/// performance through avoiding this.
3683///
3684/// Additionally with pooled allocation it's possible to create "affine slots"
3685/// to a particular WebAssembly module or component over time. For example if
3686/// the same module is multiple times over time the pooling allocator will, by
3687/// default, attempt to reuse the same slot. This mean that the slot has been
3688/// pre-configured and can retain virtual memory mappings for a copy-on-write
3689/// image, for example (see [`Config::memory_init_cow`] for more information.
3690/// This means that in a steady state instance deallocation is a single
3691/// [`madvise`] to reset linear memory to its original contents followed by a
3692/// single (optional) [`mprotect`] during the next instantiation to shrink
3693/// memory back to its original size. Compared to non-pooled allocation this
3694/// avoids the need to [`mmap`] a new region of memory, [`munmap`] it, and
3695/// [`mprotect`] regions too.
3696///
3697/// Another benefit of pooled allocation is that it's possible to configure
3698/// things such that no virtual memory management is required at all in a steady
3699/// state. For example a pooling allocator can be configured with:
3700///
3701/// * [`Config::memory_init_cow`] disabled
3702/// * [`Config::memory_guard_size`] disabled
3703/// * [`Config::memory_reservation`] shrunk to minimal size
3704/// * [`PoolingAllocationConfig::table_keep_resident`] sufficiently large
3705/// * [`PoolingAllocationConfig::linear_memory_keep_resident`] sufficiently large
3706///
3707/// With all these options in place no virtual memory tricks are used at all and
3708/// everything is manually managed by Wasmtime (for example resetting memory is
3709/// a `memset(0)`). This is not as fast in a single-threaded scenario but can
3710/// provide benefits in high-parallelism situations as no virtual memory locks
3711/// or IPIs need happen.
3712///
3713/// ## Disadvantages of Pooled Allocation
3714///
3715/// Despite the above advantages to instantiation performance the pooling
3716/// allocator is not enabled by default in Wasmtime. One reason is that the
3717/// performance advantages are not necessarily portable, for example while the
3718/// pooling allocator works on Windows it has not been tuned for performance on
3719/// Windows in the same way it has on Linux.
3720///
3721/// Additionally the main cost of the pooling allocator is that it requires a
3722/// very large reservation of virtual memory (on the order of most of the
3723/// addressable virtual address space). WebAssembly 32-bit linear memories in
3724/// Wasmtime are, by default 4G address space reservations with a small guard
3725/// region both before and after the linear memory. Memories in the pooling
3726/// allocator are contiguous which means that we only need a guard after linear
3727/// memory because the previous linear memory's slot post-guard is our own
3728/// pre-guard. This means that, by default, the pooling allocator uses roughly
3729/// 4G of virtual memory per WebAssembly linear memory slot. 4G of virtual
3730/// memory is 32 bits of a 64-bit address. Many 64-bit systems can only
3731/// actually use 48-bit addresses by default (although this can be extended on
3732/// architectures nowadays too), and of those 48 bits one of them is reserved
3733/// to indicate kernel-vs-userspace. This leaves 47-32=15 bits left,
3734/// meaning you can only have at most 32k slots of linear memories on many
3735/// systems by default. This is a relatively small number and shows how the
3736/// pooling allocator can quickly exhaust all of virtual memory.
3737///
3738/// Another disadvantage of the pooling allocator is that it may keep memory
3739/// alive when nothing is using it. A previously used slot for an instance might
3740/// have paged-in memory that will not get paged out until the
3741/// [`Engine`](crate::Engine) owning the pooling allocator is dropped. While
3742/// suitable for some applications this behavior may not be suitable for all
3743/// applications.
3744///
3745/// Finally the last disadvantage of the pooling allocator is that the
3746/// configuration values for the maximum number of instances, memories, tables,
3747/// etc, must all be fixed up-front. There's not always a clear answer as to
3748/// what these values should be so not all applications may be able to work
3749/// with this constraint.
3750///
3751/// [`madvise`]: https://man7.org/linux/man-pages/man2/madvise.2.html
3752/// [`mprotect`]: https://man7.org/linux/man-pages/man2/mprotect.2.html
3753/// [`mmap`]: https://man7.org/linux/man-pages/man2/mmap.2.html
3754/// [`munmap`]: https://man7.org/linux/man-pages/man2/munmap.2.html
3755#[cfg(feature = "pooling-allocator")]
3756#[derive(Debug, Clone, Default)]
3757pub struct PoolingAllocationConfig {
3758    config: crate::runtime::vm::PoolingInstanceAllocatorConfig,
3759}
3760
3761#[cfg(feature = "pooling-allocator")]
3762impl PoolingAllocationConfig {
3763    /// Returns a new configuration builder with all default settings
3764    /// configured.
3765    pub fn new() -> PoolingAllocationConfig {
3766        PoolingAllocationConfig::default()
3767    }
3768
3769    /// Configures the maximum number of "unused warm slots" to retain in the
3770    /// pooling allocator.
3771    ///
3772    /// The pooling allocator operates over slots to allocate from, and each
3773    /// slot is considered "cold" if it's never been used before or "warm" if
3774    /// it's been used by some module in the past. Slots in the pooling
3775    /// allocator additionally track an "affinity" flag to a particular core
3776    /// wasm module. When a module is instantiated into a slot then the slot is
3777    /// considered affine to that module, even after the instance has been
3778    /// deallocated.
3779    ///
3780    /// When a new instance is created then a slot must be chosen, and the
3781    /// current algorithm for selecting a slot is:
3782    ///
3783    /// * If there are slots that are affine to the module being instantiated,
3784    ///   then the most recently used slot is selected to be allocated from.
3785    ///   This is done to improve reuse of resources such as memory mappings and
3786    ///   additionally try to benefit from temporal locality for things like
3787    ///   caches.
3788    ///
3789    /// * Otherwise if there are more than N affine slots to other modules, then
3790    ///   one of those affine slots is chosen to be allocated. The slot chosen
3791    ///   is picked on a least-recently-used basis.
3792    ///
3793    /// * Finally, if there are less than N affine slots to other modules, then
3794    ///   the non-affine slots are allocated from.
3795    ///
3796    /// This setting, `max_unused_warm_slots`, is the value for N in the above
3797    /// algorithm. The purpose of this setting is to have a knob over the RSS
3798    /// impact of "unused slots" for a long-running wasm server.
3799    ///
3800    /// If this setting is set to 0, for example, then affine slots are
3801    /// aggressively reused on a least-recently-used basis. A "cold" slot is
3802    /// only used if there are no affine slots available to allocate from. This
3803    /// means that the set of slots used over the lifetime of a program is the
3804    /// same as the maximum concurrent number of wasm instances.
3805    ///
3806    /// If this setting is set to infinity, however, then cold slots are
3807    /// prioritized to be allocated from. This means that the set of slots used
3808    /// over the lifetime of a program will approach
3809    /// [`PoolingAllocationConfig::total_memories`], or the maximum number of
3810    /// slots in the pooling allocator.
3811    ///
3812    /// Wasmtime does not aggressively decommit all resources associated with a
3813    /// slot when the slot is not in use. For example the
3814    /// [`PoolingAllocationConfig::linear_memory_keep_resident`] option can be
3815    /// used to keep memory associated with a slot, even when it's not in use.
3816    /// This means that the total set of used slots in the pooling instance
3817    /// allocator can impact the overall RSS usage of a program.
3818    ///
3819    /// The default value for this option is `100`.
3820    pub fn max_unused_warm_slots(&mut self, max: u32) -> &mut Self {
3821        self.config.max_unused_warm_slots = max;
3822        self
3823    }
3824
3825    /// The target number of decommits to do per batch.
3826    ///
3827    /// This is not precise, as we can queue up decommits at times when we
3828    /// aren't prepared to immediately flush them, and so we may go over this
3829    /// target size occasionally.
3830    ///
3831    /// A batch size of one effectively disables batching.
3832    ///
3833    /// Defaults to `1`.
3834    pub fn decommit_batch_size(&mut self, batch_size: usize) -> &mut Self {
3835        self.config.decommit_batch_size = batch_size;
3836        self
3837    }
3838
3839    /// How much memory, in bytes, to keep resident for async stacks allocated
3840    /// with the pooling allocator.
3841    ///
3842    /// When [`Config::async_stack_zeroing`] is enabled then Wasmtime will reset
3843    /// the contents of async stacks back to zero upon deallocation. This option
3844    /// can be used to perform the zeroing operation with `memset` up to a
3845    /// certain threshold of bytes instead of using system calls to reset the
3846    /// stack to zero.
3847    ///
3848    /// Note that when using this option the memory with async stacks will
3849    /// never be decommitted.
3850    #[cfg(feature = "async")]
3851    pub fn async_stack_keep_resident(&mut self, size: usize) -> &mut Self {
3852        self.config.async_stack_keep_resident = size;
3853        self
3854    }
3855
3856    /// How much memory, in bytes, to keep resident for each linear memory
3857    /// after deallocation.
3858    ///
3859    /// This option is only applicable on Linux and has no effect on other
3860    /// platforms.
3861    ///
3862    /// By default Wasmtime will use `madvise` to reset the entire contents of
3863    /// linear memory back to zero when a linear memory is deallocated. This
3864    /// option can be used to use `memset` instead to set memory back to zero
3865    /// which can, in some configurations, reduce the number of page faults
3866    /// taken when a slot is reused.
3867    pub fn linear_memory_keep_resident(&mut self, size: usize) -> &mut Self {
3868        self.config.linear_memory_keep_resident = size;
3869        self
3870    }
3871
3872    /// How much memory, in bytes, to keep resident for each table after
3873    /// deallocation.
3874    ///
3875    /// This option is only applicable on Linux and has no effect on other
3876    /// platforms.
3877    ///
3878    /// This option is the same as
3879    /// [`PoolingAllocationConfig::linear_memory_keep_resident`] except that it
3880    /// is applicable to tables instead.
3881    pub fn table_keep_resident(&mut self, size: usize) -> &mut Self {
3882        self.config.table_keep_resident = size;
3883        self
3884    }
3885
3886    /// The maximum number of concurrent component instances supported (default
3887    /// is `1000`).
3888    ///
3889    /// This provides an upper-bound on the total size of component
3890    /// metadata-related allocations, along with
3891    /// [`PoolingAllocationConfig::max_component_instance_size`]. The upper bound is
3892    ///
3893    /// ```text
3894    /// total_component_instances * max_component_instance_size
3895    /// ```
3896    ///
3897    /// where `max_component_instance_size` is rounded up to the size and alignment
3898    /// of the internal representation of the metadata.
3899    pub fn total_component_instances(&mut self, count: u32) -> &mut Self {
3900        self.config.limits.total_component_instances = count;
3901        self
3902    }
3903
3904    /// The maximum size, in bytes, allocated for a component instance's
3905    /// `VMComponentContext` metadata as well as the aggregate size of this
3906    /// component's core instances `VMContext` metadata.
3907    ///
3908    /// The [`wasmtime::component::Instance`][crate::component::Instance] type
3909    /// has a static size but its internal `VMComponentContext` is dynamically
3910    /// sized depending on the component being instantiated. This size limit
3911    /// loosely correlates to the size of the component, taking into account
3912    /// factors such as:
3913    ///
3914    /// * number of lifted and lowered functions,
3915    /// * number of memories
3916    /// * number of inner instances
3917    /// * number of resources
3918    ///
3919    /// If the allocated size per instance is too small then instantiation of a
3920    /// module will fail at runtime with an error indicating how many bytes were
3921    /// needed.
3922    ///
3923    /// In addition to the memory in the runtime for the component itself,
3924    /// components contain one or more core module instances. Each of these
3925    /// require some memory in the runtime as described in
3926    /// [`PoolingAllocationConfig::max_core_instance_size`]. The limit here
3927    /// applies against the sum of all of these individual allocations.
3928    ///
3929    /// The default value for this is 1MiB.
3930    ///
3931    /// This provides an upper-bound on the total size of all component's
3932    /// metadata-related allocations (for both the component and its embedded
3933    /// core module instances), along with
3934    /// [`PoolingAllocationConfig::total_component_instances`]. The upper bound is
3935    ///
3936    /// ```text
3937    /// total_component_instances * max_component_instance_size
3938    /// ```
3939    ///
3940    /// where `max_component_instance_size` is rounded up to the size and alignment
3941    /// of the internal representation of the metadata.
3942    pub fn max_component_instance_size(&mut self, size: usize) -> &mut Self {
3943        self.config.limits.component_instance_size = size;
3944        self
3945    }
3946
3947    /// The maximum number of core instances a single component may contain
3948    /// (default is unlimited).
3949    ///
3950    /// This method (along with
3951    /// [`PoolingAllocationConfig::max_memories_per_component`],
3952    /// [`PoolingAllocationConfig::max_tables_per_component`], and
3953    /// [`PoolingAllocationConfig::max_component_instance_size`]) allows you to cap
3954    /// the amount of resources a single component allocation consumes.
3955    ///
3956    /// If a component will instantiate more core instances than `count`, then
3957    /// the component will fail to instantiate.
3958    pub fn max_core_instances_per_component(&mut self, count: u32) -> &mut Self {
3959        self.config.limits.max_core_instances_per_component = count;
3960        self
3961    }
3962
3963    /// The maximum number of Wasm linear memories that a single component may
3964    /// transitively contain (default is unlimited).
3965    ///
3966    /// This method (along with
3967    /// [`PoolingAllocationConfig::max_core_instances_per_component`],
3968    /// [`PoolingAllocationConfig::max_tables_per_component`], and
3969    /// [`PoolingAllocationConfig::max_component_instance_size`]) allows you to cap
3970    /// the amount of resources a single component allocation consumes.
3971    ///
3972    /// If a component transitively contains more linear memories than `count`,
3973    /// then the component will fail to instantiate.
3974    pub fn max_memories_per_component(&mut self, count: u32) -> &mut Self {
3975        self.config.limits.max_memories_per_component = count;
3976        self
3977    }
3978
3979    /// The maximum number of tables that a single component may transitively
3980    /// contain (default is unlimited).
3981    ///
3982    /// This method (along with
3983    /// [`PoolingAllocationConfig::max_core_instances_per_component`],
3984    /// [`PoolingAllocationConfig::max_memories_per_component`],
3985    /// [`PoolingAllocationConfig::max_component_instance_size`]) allows you to cap
3986    /// the amount of resources a single component allocation consumes.
3987    ///
3988    /// If a component will transitively contains more tables than `count`, then
3989    /// the component will fail to instantiate.
3990    pub fn max_tables_per_component(&mut self, count: u32) -> &mut Self {
3991        self.config.limits.max_tables_per_component = count;
3992        self
3993    }
3994
3995    /// The maximum number of concurrent Wasm linear memories supported (default
3996    /// is `1000`).
3997    ///
3998    /// This value has a direct impact on the amount of memory allocated by the pooling
3999    /// instance allocator.
4000    ///
4001    /// The pooling instance allocator allocates a memory pool, where each entry
4002    /// in the pool contains the reserved address space for each linear memory
4003    /// supported by an instance.
4004    ///
4005    /// The memory pool will reserve a large quantity of host process address
4006    /// space to elide the bounds checks required for correct WebAssembly memory
4007    /// semantics. Even with 64-bit address spaces, the address space is limited
4008    /// when dealing with a large number of linear memories.
4009    ///
4010    /// For example, on Linux x86_64, the userland address space limit is 128
4011    /// TiB. That might seem like a lot, but each linear memory will *reserve* 6
4012    /// GiB of space by default.
4013    pub fn total_memories(&mut self, count: u32) -> &mut Self {
4014        self.config.limits.total_memories = count;
4015        self
4016    }
4017
4018    /// The maximum number of concurrent tables supported (default is `1000`).
4019    ///
4020    /// This value has a direct impact on the amount of memory allocated by the
4021    /// pooling instance allocator.
4022    ///
4023    /// The pooling instance allocator allocates a table pool, where each entry
4024    /// in the pool contains the space needed for each WebAssembly table
4025    /// supported by an instance (see `table_elements` to control the size of
4026    /// each table).
4027    pub fn total_tables(&mut self, count: u32) -> &mut Self {
4028        self.config.limits.total_tables = count;
4029        self
4030    }
4031
4032    /// The maximum number of execution stacks allowed for asynchronous
4033    /// execution, when enabled (default is `1000`).
4034    ///
4035    /// This value has a direct impact on the amount of memory allocated by the
4036    /// pooling instance allocator.
4037    #[cfg(feature = "async")]
4038    pub fn total_stacks(&mut self, count: u32) -> &mut Self {
4039        self.config.limits.total_stacks = count;
4040        self
4041    }
4042
4043    /// The maximum number of concurrent core instances supported (default is
4044    /// `1000`).
4045    ///
4046    /// This provides an upper-bound on the total size of core instance
4047    /// metadata-related allocations, along with
4048    /// [`PoolingAllocationConfig::max_core_instance_size`]. The upper bound is
4049    ///
4050    /// ```text
4051    /// total_core_instances * max_core_instance_size
4052    /// ```
4053    ///
4054    /// where `max_core_instance_size` is rounded up to the size and alignment of
4055    /// the internal representation of the metadata.
4056    pub fn total_core_instances(&mut self, count: u32) -> &mut Self {
4057        self.config.limits.total_core_instances = count;
4058        self
4059    }
4060
4061    /// The maximum size, in bytes, allocated for a core instance's `VMContext`
4062    /// metadata.
4063    ///
4064    /// The [`Instance`][crate::Instance] type has a static size but its
4065    /// `VMContext` metadata is dynamically sized depending on the module being
4066    /// instantiated. This size limit loosely correlates to the size of the Wasm
4067    /// module, taking into account factors such as:
4068    ///
4069    /// * number of functions
4070    /// * number of globals
4071    /// * number of memories
4072    /// * number of tables
4073    /// * number of function types
4074    ///
4075    /// If the allocated size per instance is too small then instantiation of a
4076    /// module will fail at runtime with an error indicating how many bytes were
4077    /// needed.
4078    ///
4079    /// The default value for this is 1MiB.
4080    ///
4081    /// This provides an upper-bound on the total size of core instance
4082    /// metadata-related allocations, along with
4083    /// [`PoolingAllocationConfig::total_core_instances`]. The upper bound is
4084    ///
4085    /// ```text
4086    /// total_core_instances * max_core_instance_size
4087    /// ```
4088    ///
4089    /// where `max_core_instance_size` is rounded up to the size and alignment of
4090    /// the internal representation of the metadata.
4091    pub fn max_core_instance_size(&mut self, size: usize) -> &mut Self {
4092        self.config.limits.core_instance_size = size;
4093        self
4094    }
4095
4096    /// The maximum number of defined tables for a core module (default is `1`).
4097    ///
4098    /// This value controls the capacity of the `VMTableDefinition` table in
4099    /// each instance's `VMContext` structure.
4100    ///
4101    /// The allocated size of the table will be `tables *
4102    /// sizeof(VMTableDefinition)` for each instance regardless of how many
4103    /// tables are defined by an instance's module.
4104    pub fn max_tables_per_module(&mut self, tables: u32) -> &mut Self {
4105        self.config.limits.max_tables_per_module = tables;
4106        self
4107    }
4108
4109    /// The maximum table elements for any table defined in a module (default is
4110    /// `20000`).
4111    ///
4112    /// If a table's minimum element limit is greater than this value, the
4113    /// module will fail to instantiate.
4114    ///
4115    /// If a table's maximum element limit is unbounded or greater than this
4116    /// value, the maximum will be `table_elements` for the purpose of any
4117    /// `table.grow` instruction.
4118    ///
4119    /// This value is used to reserve the maximum space for each supported
4120    /// table; table elements are pointer-sized in the Wasmtime runtime.
4121    /// Therefore, the space reserved for each instance is `tables *
4122    /// table_elements * sizeof::<*const ()>`.
4123    pub fn table_elements(&mut self, elements: usize) -> &mut Self {
4124        self.config.limits.table_elements = elements;
4125        self
4126    }
4127
4128    /// The maximum number of defined linear memories for a module (default is
4129    /// `1`).
4130    ///
4131    /// This value controls the capacity of the `VMMemoryDefinition` table in
4132    /// each core instance's `VMContext` structure.
4133    ///
4134    /// The allocated size of the table will be `memories *
4135    /// sizeof(VMMemoryDefinition)` for each core instance regardless of how
4136    /// many memories are defined by the core instance's module.
4137    pub fn max_memories_per_module(&mut self, memories: u32) -> &mut Self {
4138        self.config.limits.max_memories_per_module = memories;
4139        self
4140    }
4141
4142    /// The maximum byte size that any WebAssembly linear memory may grow to.
4143    ///
4144    /// This option defaults to 4 GiB meaning that for 32-bit linear memories
4145    /// there is no restrictions. 64-bit linear memories will not be allowed to
4146    /// grow beyond 4 GiB by default.
4147    ///
4148    /// If a memory's minimum size is greater than this value, the module will
4149    /// fail to instantiate.
4150    ///
4151    /// If a memory's maximum size is unbounded or greater than this value, the
4152    /// maximum will be `max_memory_size` for the purpose of any `memory.grow`
4153    /// instruction.
4154    ///
4155    /// This value is used to control the maximum accessible space for each
4156    /// linear memory of a core instance. This can be thought of as a simple
4157    /// mechanism like [`Store::limiter`](crate::Store::limiter) to limit memory
4158    /// at runtime. This value can also affect striping/coloring behavior when
4159    /// used in conjunction with
4160    /// [`memory_protection_keys`](PoolingAllocationConfig::memory_protection_keys).
4161    ///
4162    /// The virtual memory reservation size of each linear memory is controlled
4163    /// by the [`Config::memory_reservation`] setting and this method's
4164    /// configuration cannot exceed [`Config::memory_reservation`].
4165    pub fn max_memory_size(&mut self, bytes: usize) -> &mut Self {
4166        self.config.limits.max_memory_size = bytes;
4167        self
4168    }
4169
4170    /// Configures whether memory protection keys (MPK) should be used for more
4171    /// efficient layout of pool-allocated memories.
4172    ///
4173    /// When using the pooling allocator (see [`Config::allocation_strategy`],
4174    /// [`InstanceAllocationStrategy::Pooling`]), memory protection keys can
4175    /// reduce the total amount of allocated virtual memory by eliminating guard
4176    /// regions between WebAssembly memories in the pool. It does so by
4177    /// "coloring" memory regions with different memory keys and setting which
4178    /// regions are accessible each time executions switches from host to guest
4179    /// (or vice versa).
4180    ///
4181    /// Leveraging MPK requires configuring a smaller-than-default
4182    /// [`max_memory_size`](PoolingAllocationConfig::max_memory_size) to enable
4183    /// this coloring/striping behavior. For example embeddings might want to
4184    /// reduce the default 4G allowance to 128M.
4185    ///
4186    /// MPK is only available on Linux (called `pku` there) and recent x86
4187    /// systems; we check for MPK support at runtime by examining the `CPUID`
4188    /// register. This configuration setting can be in three states:
4189    ///
4190    /// - `auto`: if MPK support is available the guard regions are removed; if
4191    ///   not, the guard regions remain
4192    /// - `yes`: use MPK to eliminate guard regions; fail if MPK is not
4193    ///   supported
4194    /// - `no`: never use MPK
4195    ///
4196    /// By default this value is `no`, but may become `auto` in future
4197    /// releases.
4198    ///
4199    /// __WARNING__: this configuration options is still experimental--use at
4200    /// your own risk! MPK uses kernel and CPU features to protect memory
4201    /// regions; you may observe segmentation faults if anything is
4202    /// misconfigured.
4203    #[cfg(feature = "memory-protection-keys")]
4204    pub fn memory_protection_keys(&mut self, enable: Enabled) -> &mut Self {
4205        self.config.memory_protection_keys = enable;
4206        self
4207    }
4208
4209    /// Sets an upper limit on how many memory protection keys (MPK) Wasmtime
4210    /// will use.
4211    ///
4212    /// This setting is only applicable when
4213    /// [`PoolingAllocationConfig::memory_protection_keys`] is set to `enable`
4214    /// or `auto`. Configuring this above the HW and OS limits (typically 15)
4215    /// has no effect.
4216    ///
4217    /// If multiple Wasmtime engines are used in the same process, note that all
4218    /// engines will share the same set of allocated keys; this setting will
4219    /// limit how many keys are allocated initially and thus available to all
4220    /// other engines.
4221    #[cfg(feature = "memory-protection-keys")]
4222    pub fn max_memory_protection_keys(&mut self, max: usize) -> &mut Self {
4223        self.config.max_memory_protection_keys = max;
4224        self
4225    }
4226
4227    /// Check if memory protection keys (MPK) are available on the current host.
4228    ///
4229    /// This is a convenience method for determining MPK availability using the
4230    /// same method that [`Enabled::Auto`] does. See
4231    /// [`PoolingAllocationConfig::memory_protection_keys`] for more
4232    /// information.
4233    #[cfg(feature = "memory-protection-keys")]
4234    pub fn are_memory_protection_keys_available() -> bool {
4235        crate::runtime::vm::mpk::is_supported()
4236    }
4237
4238    /// The maximum number of concurrent GC heaps supported (default is `1000`).
4239    ///
4240    /// This value has a direct impact on the amount of memory allocated by the
4241    /// pooling instance allocator.
4242    ///
4243    /// The pooling instance allocator allocates a GC heap pool, where each
4244    /// entry in the pool contains the space needed for each GC heap used by a
4245    /// store.
4246    #[cfg(feature = "gc")]
4247    pub fn total_gc_heaps(&mut self, count: u32) -> &mut Self {
4248        self.config.limits.total_gc_heaps = count;
4249        self
4250    }
4251
4252    /// Configures whether the Linux-specific [`PAGEMAP_SCAN` ioctl][ioctl] is
4253    /// used to help reset linear memory.
4254    ///
4255    /// When [`Self::linear_memory_keep_resident`] or
4256    /// [`Self::table_keep_resident`] options are configured to nonzero values
4257    /// the default behavior is to `memset` the lowest addresses of a table or
4258    /// memory back to their original contents. With the `PAGEMAP_SCAN` ioctl on
4259    /// Linux this can be done to more intelligently scan for resident pages in
4260    /// the region and only reset those pages back to their original contents
4261    /// with `memset` rather than assuming the low addresses are all resident.
4262    ///
4263    /// This ioctl has the potential to provide a number of performance benefits
4264    /// in high-reuse and high concurrency scenarios. Notably this enables
4265    /// Wasmtime to scan the entire region of WebAssembly linear memory and
4266    /// manually reset memory back to its original contents, up to
4267    /// [`Self::linear_memory_keep_resident`] bytes, possibly skipping an
4268    /// `madvise` entirely. This can be more efficient by avoiding removing
4269    /// pages from the address space entirely and additionally ensuring that
4270    /// future use of the linear memory doesn't incur page faults as the pages
4271    /// remain resident.
4272    ///
4273    /// At this time this configuration option is still being evaluated as to
4274    /// how appropriate it is for all use cases. It currently defaults to
4275    /// `no` or disabled but may change to `auto`, enable if supported, in the
4276    /// future. This option is only supported on Linux and requires a kernel
4277    /// version of 6.7 or higher.
4278    ///
4279    /// [ioctl]: https://www.man7.org/linux/man-pages/man2/PAGEMAP_SCAN.2const.html
4280    pub fn pagemap_scan(&mut self, enable: Enabled) -> &mut Self {
4281        self.config.pagemap_scan = enable;
4282        self
4283    }
4284
4285    /// Tests whether [`Self::pagemap_scan`] is available or not on the host
4286    /// system.
4287    pub fn is_pagemap_scan_available() -> bool {
4288        crate::runtime::vm::PoolingInstanceAllocatorConfig::is_pagemap_scan_available()
4289    }
4290}
4291
4292#[cfg(feature = "std")]
4293fn detect_host_feature(feature: &str) -> Option<bool> {
4294    #[cfg(target_arch = "aarch64")]
4295    {
4296        return match feature {
4297            "lse" => Some(std::arch::is_aarch64_feature_detected!("lse")),
4298            "paca" => Some(std::arch::is_aarch64_feature_detected!("paca")),
4299            "fp16" => Some(std::arch::is_aarch64_feature_detected!("fp16")),
4300
4301            _ => None,
4302        };
4303    }
4304
4305    // `is_s390x_feature_detected` is nightly only for now, so use the
4306    // STORE FACILITY LIST EXTENDED instruction as a temporary measure.
4307    #[cfg(target_arch = "s390x")]
4308    {
4309        let mut facility_list: [u64; 4] = [0; 4];
4310        unsafe {
4311            core::arch::asm!(
4312                "stfle 0({})",
4313                in(reg_addr) facility_list.as_mut_ptr() ,
4314                inout("r0") facility_list.len() as u64 - 1 => _,
4315                options(nostack)
4316            );
4317        }
4318        let get_facility_bit = |n: usize| {
4319            // NOTE: bits are numbered from the left.
4320            facility_list[n / 64] & (1 << (63 - (n % 64))) != 0
4321        };
4322
4323        return match feature {
4324            "mie3" => Some(get_facility_bit(61)),
4325            "mie4" => Some(get_facility_bit(84)),
4326            "vxrs_ext2" => Some(get_facility_bit(148)),
4327            "vxrs_ext3" => Some(get_facility_bit(198)),
4328
4329            _ => None,
4330        };
4331    }
4332
4333    #[cfg(target_arch = "riscv64")]
4334    {
4335        return match feature {
4336            // due to `is_riscv64_feature_detected` is not stable.
4337            // we cannot use it. For now lie and say all features are always
4338            // found to keep tests working.
4339            _ => Some(true),
4340        };
4341    }
4342
4343    #[cfg(target_arch = "x86_64")]
4344    {
4345        return match feature {
4346            "cmpxchg16b" => Some(std::is_x86_feature_detected!("cmpxchg16b")),
4347            "sse3" => Some(std::is_x86_feature_detected!("sse3")),
4348            "ssse3" => Some(std::is_x86_feature_detected!("ssse3")),
4349            "sse4.1" => Some(std::is_x86_feature_detected!("sse4.1")),
4350            "sse4.2" => Some(std::is_x86_feature_detected!("sse4.2")),
4351            "popcnt" => Some(std::is_x86_feature_detected!("popcnt")),
4352            "avx" => Some(std::is_x86_feature_detected!("avx")),
4353            "avx2" => Some(std::is_x86_feature_detected!("avx2")),
4354            "fma" => Some(std::is_x86_feature_detected!("fma")),
4355            "bmi1" => Some(std::is_x86_feature_detected!("bmi1")),
4356            "bmi2" => Some(std::is_x86_feature_detected!("bmi2")),
4357            "avx512bitalg" => Some(std::is_x86_feature_detected!("avx512bitalg")),
4358            "avx512dq" => Some(std::is_x86_feature_detected!("avx512dq")),
4359            "avx512f" => Some(std::is_x86_feature_detected!("avx512f")),
4360            "avx512vl" => Some(std::is_x86_feature_detected!("avx512vl")),
4361            "avx512vbmi" => Some(std::is_x86_feature_detected!("avx512vbmi")),
4362            "lzcnt" => Some(std::is_x86_feature_detected!("lzcnt")),
4363
4364            _ => None,
4365        };
4366    }
4367
4368    #[allow(
4369        unreachable_code,
4370        reason = "reachable or not depending on if a target above matches"
4371    )]
4372    {
4373        let _ = feature;
4374        return None;
4375    }
4376}