1 use crate::linker::{Definition, DefinitionType};
2 use crate::prelude::*;
3 use crate::runtime::vm::{
4     Imports, InstanceAllocationRequest, ModuleRuntimeInfo, StorePtr, VMFuncRef, VMFunctionImport,
5     VMGlobalImport, VMMemoryImport, VMOpaqueContext, VMTableImport,
6 };
7 use crate::store::{InstanceId, StoreOpaque, Stored};
8 use crate::types::matching;
9 use crate::{
10     AsContextMut, Engine, Export, Extern, Func, Global, Memory, Module, ModuleExport, SharedMemory,
11     StoreContext, StoreContextMut, Table, TypedFunc,
12 };
13 use alloc::sync::Arc;
14 use core::ptr::NonNull;
15 use wasmparser::WasmFeatures;
16 use wasmtime_environ::{
17     EntityIndex, EntityType, FuncIndex, GlobalIndex, MemoryIndex, PrimaryMap, TableIndex, TypeTrace,
18 };
19 
20 /// An instantiated WebAssembly module.
21 ///
22 /// This type represents the instantiation of a [`Module`]. Once instantiated
23 /// you can access the [`exports`](Instance::exports) which are of type
24 /// [`Extern`] and provide the ability to call functions, set globals, read
25 /// memory, etc. When interacting with any wasm code you'll want to make an
26 /// [`Instance`] to call any code or execute anything.
27 ///
28 /// Instances are owned by a [`Store`](crate::Store) which is passed in at
29 /// creation time. It's recommended to create instances with
30 /// [`Linker::instantiate`](crate::Linker::instantiate) or similar
31 /// [`Linker`](crate::Linker) methods, but a more low-level constructor is also
32 /// available as [`Instance::new`].
33 #[derive(Copy, Clone, Debug)]
34 #[repr(transparent)]
35 pub struct Instance(Stored<InstanceData>);
36 
37 pub(crate) struct InstanceData {
38     /// The id of the instance within the store, used to find the original
39     /// `InstanceHandle`.
40     id: InstanceId,
41     /// A lazily-populated list of exports of this instance. The order of
42     /// exports here matches the order of the exports in the original
43     /// module.
44     exports: Vec<Option<Extern>>,
45 }
46 
47 impl InstanceData {
48     pub fn from_id(id: InstanceId) -> InstanceData {
49         InstanceData {
50             id,
51             exports: vec![],
52         }
53     }
54 }
55 
56 impl Instance {
57     /// Creates a new [`Instance`] from the previously compiled [`Module`] and
58     /// list of `imports` specified.
59     ///
60     /// This method instantiates the `module` provided with the `imports`,
61     /// following the procedure in the [core specification][inst] to
62     /// instantiate. Instantiation can fail for a number of reasons (many
63     /// specified below), but if successful the `start` function will be
64     /// automatically run (if specified in the `module`) and then the
65     /// [`Instance`] will be returned.
66     ///
67     /// Per the WebAssembly spec, instantiation includes running the module's
68     /// start function, if it has one (not to be confused with the `_start`
69     /// function, which is not run).
70     ///
71     /// Note that this is a low-level function that just performs an
72     /// instantiation. See the [`Linker`](crate::Linker) struct for an API which
73     /// provides a convenient way to link imports and provides automatic Command
74     /// and Reactor behavior.
75     ///
76     /// ## Providing Imports
77     ///
78     /// The entries in the list of `imports` are intended to correspond 1:1
79     /// with the list of imports returned by [`Module::imports`]. Before
80     /// calling [`Instance::new`] you'll want to inspect the return value of
81     /// [`Module::imports`] and, for each import type, create an [`Extern`]
82     /// which corresponds to that type.  These [`Extern`] values are all then
83     /// collected into a list and passed to this function.
84     ///
85     /// Note that this function is intentionally relatively low level. For an
86     /// easier time passing imports by doing name-based resolution it's
87     /// recommended to instead use the [`Linker`](crate::Linker) type.
88     ///
89     /// ## Errors
90     ///
91     /// This function can fail for a number of reasons, including, but not
92     /// limited to:
93     ///
94     /// * The number of `imports` provided doesn't match the number of imports
95     ///   returned by the `module`'s [`Module::imports`] method.
96     /// * The type of any [`Extern`] doesn't match the corresponding
97     ///   [`ExternType`] entry that it maps to.
98     /// * The `start` function in the instance, if present, traps.
99     /// * Module/instance resource limits are exceeded.
100     ///
101     /// When instantiation fails it's recommended to inspect the return value to
102     /// see why it failed, or bubble it upwards. If you'd like to specifically
103     /// check for trap errors, you can use `error.downcast::<Trap>()`. For more
104     /// about error handling see the [`Trap`] documentation.
105     ///
106     /// [`Trap`]: crate::Trap
107     ///
108     /// # Panics
109     ///
110     /// This function will panic if called with a store associated with a
111     /// [`asynchronous config`](crate::Config::async_support). This function
112     /// will also panic if any [`Extern`] supplied is not owned by `store`.
113     ///
114     /// [inst]: https://webassembly.github.io/spec/core/exec/modules.html#exec-instantiation
115     /// [`ExternType`]: crate::ExternType
116     pub fn new(
117         mut store: impl AsContextMut,
118         module: &Module,
119         imports: &[Extern],
120     ) -> Result<Instance> {
121         let mut store = store.as_context_mut();
122         let imports = Instance::typecheck_externs(store.0, module, imports)?;
123         // Note that the unsafety here should be satisfied by the call to
124         // `typecheck_externs` above which satisfies the condition that all
125         // the imports are valid for this module.
126         unsafe { Instance::new_started(&mut store, module, imports.as_ref()) }
127     }
128 
129     /// Same as [`Instance::new`], except for usage in [asynchronous stores].
130     ///
131     /// For more details about this function see the documentation on
132     /// [`Instance::new`]. The only difference between these two methods is that
133     /// this one will asynchronously invoke the wasm start function in case it
134     /// calls any imported function which is an asynchronous host function (e.g.
135     /// created with [`Func::new_async`](crate::Func::new_async).
136     ///
137     /// # Panics
138     ///
139     /// This function will panic if called with a store associated with a
140     /// [`synchronous config`](crate::Config::new). This is only compatible with
141     /// stores associated with an [`asynchronous
142     /// config`](crate::Config::async_support).
143     ///
144     /// This function will also panic, like [`Instance::new`], if any [`Extern`]
145     /// specified does not belong to `store`.
146     #[cfg(feature = "async")]
147     pub async fn new_async<T>(
148         mut store: impl AsContextMut<Data = T>,
149         module: &Module,
150         imports: &[Extern],
151     ) -> Result<Instance>
152     where
153         T: Send,
154     {
155         let mut store = store.as_context_mut();
156         let imports = Instance::typecheck_externs(store.0, module, imports)?;
157         // See `new` for notes on this unsafety
158         unsafe { Instance::new_started_async(&mut store, module, imports.as_ref()).await }
159     }
160 
161     fn typecheck_externs(
162         store: &mut StoreOpaque,
163         module: &Module,
164         imports: &[Extern],
165     ) -> Result<OwnedImports> {
166         for import in imports {
167             if !import.comes_from_same_store(store) {
168                 bail!("cross-`Store` instantiation is not currently supported");
169             }
170         }
171         typecheck(module, imports, |cx, ty, item| {
172             let item = DefinitionType::from(store, item);
173             cx.definition(ty, &item)
174         })?;
175         let mut owned_imports = OwnedImports::new(module);
176         for import in imports {
177             owned_imports.push(import, store, module);
178         }
179         Ok(owned_imports)
180     }
181 
182     /// Internal function to create an instance and run the start function.
183     ///
184     /// This function's unsafety is the same as `Instance::new_raw`.
185     pub(crate) unsafe fn new_started<T>(
186         store: &mut StoreContextMut<'_, T>,
187         module: &Module,
188         imports: Imports<'_>,
189     ) -> Result<Instance> {
190         assert!(
191             !store.0.async_support(),
192             "must use async instantiation when async support is enabled",
193         );
194         Self::new_started_impl(store, module, imports)
195     }
196 
197     /// Internal function to create an instance and run the start function.
198     ///
199     /// ONLY CALL THIS IF YOU HAVE ALREADY CHECKED FOR ASYNCNESS AND HANDLED
200     /// THE FIBER NONSENSE
201     pub(crate) unsafe fn new_started_impl<T>(
202         store: &mut StoreContextMut<'_, T>,
203         module: &Module,
204         imports: Imports<'_>,
205     ) -> Result<Instance> {
206         let (instance, start) = Instance::new_raw(store.0, module, imports)?;
207         if let Some(start) = start {
208             instance.start_raw(store, start)?;
209         }
210         Ok(instance)
211     }
212 
213     /// Internal function to create an instance and run the start function.
214     ///
215     /// This function's unsafety is the same as `Instance::new_raw`.
216     #[cfg(feature = "async")]
217     async unsafe fn new_started_async<T>(
218         store: &mut StoreContextMut<'_, T>,
219         module: &Module,
220         imports: Imports<'_>,
221     ) -> Result<Instance>
222     where
223         T: Send,
224     {
225         assert!(
226             store.0.async_support(),
227             "must use sync instantiation when async support is disabled",
228         );
229 
230         store
231             .on_fiber(|store| Self::new_started_impl(store, module, imports))
232             .await?
233     }
234 
235     /// Internal function to create an instance which doesn't have its `start`
236     /// function run yet.
237     ///
238     /// This is not intended to be exposed from Wasmtime, it's intended to
239     /// refactor out common code from `new_started` and `new_started_async`.
240     ///
241     /// Note that this step needs to be run on a fiber in async mode even
242     /// though it doesn't do any blocking work because an async resource
243     /// limiter may need to yield.
244     ///
245     /// # Unsafety
246     ///
247     /// This method is unsafe because it does not type-check the `imports`
248     /// provided. The `imports` provided must be suitable for the module
249     /// provided as well.
250     unsafe fn new_raw(
251         store: &mut StoreOpaque,
252         module: &Module,
253         imports: Imports<'_>,
254     ) -> Result<(Instance, Option<FuncIndex>)> {
255         if !Engine::same(store.engine(), module.engine()) {
256             bail!("cross-`Engine` instantiation is not currently supported");
257         }
258         store.bump_resource_counts(module)?;
259 
260         // Allocate the GC heap, if necessary.
261         let _ = store.gc_store_mut()?;
262 
263         let compiled_module = module.compiled_module();
264 
265         // Register the module just before instantiation to ensure we keep the module
266         // properly referenced while in use by the store.
267         let module_id = store.modules_mut().register_module(module);
268         store.fill_func_refs();
269 
270         // The first thing we do is issue an instance allocation request
271         // to the instance allocator. This, on success, will give us an
272         // instance handle.
273         //
274         // Note that the `host_state` here is a pointer back to the
275         // `Instance` we'll be returning from this function. This is a
276         // circular reference so we can't construct it before we construct
277         // this instance, so we determine what the ID is and then assert
278         // it's the same later when we do actually insert it.
279         let instance_to_be = store.store_data().next_id::<InstanceData>();
280 
281         let mut instance_handle =
282             store
283                 .engine()
284                 .allocator()
285                 .allocate_module(InstanceAllocationRequest {
286                     runtime_info: &ModuleRuntimeInfo::Module(module.clone()),
287                     imports,
288                     host_state: Box::new(Instance(instance_to_be)),
289                     store: StorePtr::new(store.traitobj()),
290                     wmemcheck: store.engine().config().wmemcheck,
291                     pkey: store.get_pkey(),
292                 })?;
293 
294         // The instance still has lots of setup, for example
295         // data/elements/start/etc. This can all fail, but even on failure
296         // the instance may persist some state via previous successful
297         // initialization. For this reason once we have an instance handle
298         // we immediately insert it into the store to keep it alive.
299         //
300         // Note that we `clone` the instance handle just to make easier
301         // working the borrow checker here easier. Technically the `&mut
302         // instance` has somewhat of a borrow on `store` (which
303         // conflicts with the borrow on `store.engine`) but this doesn't
304         // matter in practice since initialization isn't even running any
305         // code here anyway.
306         let id = store.add_instance(instance_handle.clone(), module_id);
307 
308         // Additionally, before we start doing fallible instantiation, we
309         // do one more step which is to insert an `InstanceData`
310         // corresponding to this instance. This `InstanceData` can be used
311         // via `Caller::get_export` if our instance's state "leaks" into
312         // other instances, even if we don't return successfully from this
313         // function.
314         //
315         // We don't actually load all exports from the instance at this
316         // time, instead preferring to lazily load them as they're demanded.
317         // For module/instance exports, though, those aren't actually
318         // stored in the instance handle so we need to immediately handle
319         // those here.
320         let instance = {
321             let exports = vec![None; compiled_module.module().exports.len()];
322             let data = InstanceData { id, exports };
323             Instance::from_wasmtime(data, store)
324         };
325 
326         // double-check our guess of what the new instance's ID would be
327         // was actually correct.
328         assert_eq!(instance.0, instance_to_be);
329 
330         // Now that we've recorded all information we need to about this
331         // instance within a `Store` we can start performing fallible
332         // initialization. Note that we still defer the `start` function to
333         // later since that may need to run asynchronously.
334         //
335         // If this returns an error (or if the start function traps) then
336         // any other initialization which may have succeeded which placed
337         // items from this instance into other instances should be ok when
338         // those items are loaded and run we'll have all the metadata to
339         // look at them.
340         instance_handle.initialize(
341             compiled_module.module(),
342             store
343                 .engine()
344                 .features()
345                 .contains(WasmFeatures::BULK_MEMORY),
346         )?;
347 
348         Ok((instance, compiled_module.module().start_func))
349     }
350 
351     pub(crate) fn from_wasmtime(handle: InstanceData, store: &mut StoreOpaque) -> Instance {
352         Instance(store.store_data_mut().insert(handle))
353     }
354 
355     fn start_raw<T>(&self, store: &mut StoreContextMut<'_, T>, start: FuncIndex) -> Result<()> {
356         let id = store.0.store_data()[self.0].id;
357         // If a start function is present, invoke it. Make sure we use all the
358         // trap-handling configuration in `store` as well.
359         let instance = store.0.instance_mut(id);
360         let f = instance.get_exported_func(start);
361         let caller_vmctx = instance.vmctx();
362         unsafe {
363             super::func::invoke_wasm_and_catch_traps(store, |_default_caller| {
364                 let func = f.func_ref.as_ref().array_call;
365                 func(
366                     f.func_ref.as_ref().vmctx,
367                     VMOpaqueContext::from_vmcontext(caller_vmctx),
368                     [].as_mut_ptr(),
369                     0,
370                 )
371             })?;
372         }
373         Ok(())
374     }
375 
376     /// Get this instance's module.
377     pub fn module<'a, T: 'a>(&self, store: impl Into<StoreContext<'a, T>>) -> &'a Module {
378         self._module(store.into().0)
379     }
380 
381     fn _module<'a>(&self, store: &'a StoreOpaque) -> &'a Module {
382         let InstanceData { id, .. } = store[self.0];
383         store.module_for_instance(id).unwrap()
384     }
385 
386     /// Returns the list of exported items from this [`Instance`].
387     ///
388     /// # Panics
389     ///
390     /// Panics if `store` does not own this instance.
391     pub fn exports<'a, T: 'a>(
392         &'a self,
393         store: impl Into<StoreContextMut<'a, T>>,
394     ) -> impl ExactSizeIterator<Item = Export<'a>> + 'a {
395         self._exports(store.into().0)
396     }
397 
398     fn _exports<'a>(
399         &'a self,
400         store: &'a mut StoreOpaque,
401     ) -> impl ExactSizeIterator<Item = Export<'a>> + 'a {
402         // If this is an `Instantiated` instance then all the `exports` may not
403         // be filled in. Fill them all in now if that's the case.
404         let InstanceData { exports, id, .. } = &store[self.0];
405         if exports.iter().any(|e| e.is_none()) {
406             let module = Arc::clone(store.instance(*id).module());
407             let data = &store[self.0];
408             let id = data.id;
409 
410             for name in module.exports.keys() {
411                 let instance = store.instance(id);
412                 if let Some((export_name_index, _, &entity)) =
413                     instance.module().exports.get_full(name)
414                 {
415                     self._get_export(store, entity, export_name_index);
416                 }
417             }
418         }
419 
420         let data = &store.store_data()[self.0];
421         let module = store.instance(data.id).module();
422         module
423             .exports
424             .iter()
425             .zip(&data.exports)
426             .map(|((name, _), export)| Export::new(name, export.clone().unwrap()))
427     }
428 
429     /// Looks up an exported [`Extern`] value by name.
430     ///
431     /// This method will search the module for an export named `name` and return
432     /// the value, if found.
433     ///
434     /// Returns `None` if there was no export named `name`.
435     ///
436     /// # Panics
437     ///
438     /// Panics if `store` does not own this instance.
439     ///
440     /// # Why does `get_export` take a mutable context?
441     ///
442     /// This method requires a mutable context because an instance's exports are
443     /// lazily populated, and we cache them as they are accessed. This makes
444     /// instantiating a module faster, but also means this method requires a
445     /// mutable context.
446     pub fn get_export(&self, mut store: impl AsContextMut, name: &str) -> Option<Extern> {
447         let store = store.as_context_mut().0;
448         let data = &store[self.0];
449         let instance = store.instance(data.id);
450         let (export_name_index, _, &entity) = instance.module().exports.get_full(name)?;
451         self._get_export(store, entity, export_name_index)
452     }
453 
454     /// Looks up an exported [`Extern`] value by a [`ModuleExport`] value.
455     ///
456     /// This is similar to [`Instance::get_export`] but uses a [`ModuleExport`] value to avoid
457     /// string lookups where possible. [`ModuleExport`]s can be obtained by calling
458     /// [`Module::get_export_index`] on the [`Module`] that this instance was instantiated with.
459     ///
460     /// This method will search the module for an export with a matching entity index and return
461     /// the value, if found.
462     ///
463     /// Returns `None` if there was no export with a matching entity index.
464     /// # Panics
465     ///
466     /// Panics if `store` does not own this instance.
467     pub fn get_module_export(
468         &self,
469         mut store: impl AsContextMut,
470         export: &ModuleExport,
471     ) -> Option<Extern> {
472         let store = store.as_context_mut().0;
473 
474         // Verify the `ModuleExport` matches the module used in this instance.
475         if self._module(store).id() != export.module {
476             return None;
477         }
478 
479         self._get_export(store, export.entity, export.export_name_index)
480     }
481 
482     fn _get_export(
483         &self,
484         store: &mut StoreOpaque,
485         entity: EntityIndex,
486         export_name_index: usize,
487     ) -> Option<Extern> {
488         // Instantiated instances will lazily fill in exports, so we process
489         // all that lazy logic here.
490         let data = &store[self.0];
491 
492         if let Some(export) = &data.exports[export_name_index] {
493             return Some(export.clone());
494         }
495 
496         let instance = store.instance_mut(data.id); // Reborrow the &mut InstanceHandle
497         let item =
498             unsafe { Extern::from_wasmtime_export(instance.get_export_by_index(entity), store) };
499         let data = &mut store[self.0];
500         data.exports[export_name_index] = Some(item.clone());
501         Some(item)
502     }
503 
504     /// Looks up an exported [`Func`] value by name.
505     ///
506     /// Returns `None` if there was no export named `name`, or if there was but
507     /// it wasn't a function.
508     ///
509     /// # Panics
510     ///
511     /// Panics if `store` does not own this instance.
512     pub fn get_func(&self, store: impl AsContextMut, name: &str) -> Option<Func> {
513         self.get_export(store, name)?.into_func()
514     }
515 
516     /// Looks up an exported [`Func`] value by name and with its type.
517     ///
518     /// This function is a convenience wrapper over [`Instance::get_func`] and
519     /// [`Func::typed`]. For more information see the linked documentation.
520     ///
521     /// Returns an error if `name` isn't a function export or if the export's
522     /// type did not match `Params` or `Results`
523     ///
524     /// # Panics
525     ///
526     /// Panics if `store` does not own this instance.
527     pub fn get_typed_func<Params, Results>(
528         &self,
529         mut store: impl AsContextMut,
530         name: &str,
531     ) -> Result<TypedFunc<Params, Results>>
532     where
533         Params: crate::WasmParams,
534         Results: crate::WasmResults,
535     {
536         let f = self
537             .get_export(store.as_context_mut(), name)
538             .and_then(|f| f.into_func())
539             .ok_or_else(|| anyhow!("failed to find function export `{}`", name))?;
540         Ok(f.typed::<Params, Results>(store)
541             .with_context(|| format!("failed to convert function `{name}` to given type"))?)
542     }
543 
544     /// Looks up an exported [`Table`] value by name.
545     ///
546     /// Returns `None` if there was no export named `name`, or if there was but
547     /// it wasn't a table.
548     ///
549     /// # Panics
550     ///
551     /// Panics if `store` does not own this instance.
552     pub fn get_table(&self, store: impl AsContextMut, name: &str) -> Option<Table> {
553         self.get_export(store, name)?.into_table()
554     }
555 
556     /// Looks up an exported [`Memory`] value by name.
557     ///
558     /// Returns `None` if there was no export named `name`, or if there was but
559     /// it wasn't a memory.
560     ///
561     /// # Panics
562     ///
563     /// Panics if `store` does not own this instance.
564     pub fn get_memory(&self, store: impl AsContextMut, name: &str) -> Option<Memory> {
565         self.get_export(store, name)?.into_memory()
566     }
567 
568     /// Looks up an exported [`SharedMemory`] value by name.
569     ///
570     /// Returns `None` if there was no export named `name`, or if there was but
571     /// it wasn't a shared memory.
572     ///
573     /// # Panics
574     ///
575     /// Panics if `store` does not own this instance.
576     pub fn get_shared_memory(
577         &self,
578         mut store: impl AsContextMut,
579         name: &str,
580     ) -> Option<SharedMemory> {
581         let mut store = store.as_context_mut();
582         self.get_export(&mut store, name)?.into_shared_memory()
583     }
584 
585     /// Looks up an exported [`Global`] value by name.
586     ///
587     /// Returns `None` if there was no export named `name`, or if there was but
588     /// it wasn't a global.
589     ///
590     /// # Panics
591     ///
592     /// Panics if `store` does not own this instance.
593     pub fn get_global(&self, store: impl AsContextMut, name: &str) -> Option<Global> {
594         self.get_export(store, name)?.into_global()
595     }
596 
597     #[cfg(feature = "component-model")]
598     pub(crate) fn id(&self, store: &StoreOpaque) -> InstanceId {
599         store[self.0].id
600     }
601 
602     /// Get all globals within this instance.
603     ///
604     /// Returns both import and defined globals.
605     ///
606     /// Returns both exported and non-exported globals.
607     ///
608     /// Gives access to the full globals space.
609     pub(crate) fn all_globals<'a>(
610         &'a self,
611         store: &'a mut StoreOpaque,
612     ) -> impl ExactSizeIterator<Item = (GlobalIndex, Global)> + 'a {
613         let data = &store[self.0];
614         let instance = store.instance_mut(data.id);
615         instance
616             .all_globals()
617             .collect::<Vec<_>>()
618             .into_iter()
619             .map(|(i, g)| (i, unsafe { Global::from_wasmtime_global(g, store) }))
620     }
621 
622     /// Get all memories within this instance.
623     ///
624     /// Returns both import and defined memories.
625     ///
626     /// Returns both exported and non-exported memories.
627     ///
628     /// Gives access to the full memories space.
629     pub(crate) fn all_memories<'a>(
630         &'a self,
631         store: &'a mut StoreOpaque,
632     ) -> impl ExactSizeIterator<Item = (MemoryIndex, Memory)> + 'a {
633         let data = &store[self.0];
634         let instance = store.instance_mut(data.id);
635         instance
636             .all_memories()
637             .collect::<Vec<_>>()
638             .into_iter()
639             .map(|(i, m)| (i, unsafe { Memory::from_wasmtime_memory(m, store) }))
640     }
641 }
642 
643 pub(crate) struct OwnedImports {
644     functions: PrimaryMap<FuncIndex, VMFunctionImport>,
645     tables: PrimaryMap<TableIndex, VMTableImport>,
646     memories: PrimaryMap<MemoryIndex, VMMemoryImport>,
647     globals: PrimaryMap<GlobalIndex, VMGlobalImport>,
648 }
649 
650 impl OwnedImports {
651     fn new(module: &Module) -> OwnedImports {
652         let mut ret = OwnedImports::empty();
653         ret.reserve(module);
654         return ret;
655     }
656 
657     pub(crate) fn empty() -> OwnedImports {
658         OwnedImports {
659             functions: PrimaryMap::new(),
660             tables: PrimaryMap::new(),
661             memories: PrimaryMap::new(),
662             globals: PrimaryMap::new(),
663         }
664     }
665 
666     pub(crate) fn reserve(&mut self, module: &Module) {
667         let raw = module.compiled_module().module();
668         self.functions.reserve(raw.num_imported_funcs);
669         self.tables.reserve(raw.num_imported_tables);
670         self.memories.reserve(raw.num_imported_memories);
671         self.globals.reserve(raw.num_imported_globals);
672     }
673 
674     #[cfg(feature = "component-model")]
675     pub(crate) fn clear(&mut self) {
676         self.functions.clear();
677         self.tables.clear();
678         self.memories.clear();
679         self.globals.clear();
680     }
681 
682     fn push(&mut self, item: &Extern, store: &mut StoreOpaque, module: &Module) {
683         match item {
684             Extern::Func(i) => {
685                 self.functions.push(i.vmimport(store, module));
686             }
687             Extern::Global(i) => {
688                 self.globals.push(i.vmimport(store));
689             }
690             Extern::Table(i) => {
691                 self.tables.push(i.vmimport(store));
692             }
693             Extern::Memory(i) => {
694                 self.memories.push(i.vmimport(store));
695             }
696             Extern::SharedMemory(i) => {
697                 self.memories.push(i.vmimport(store));
698             }
699         }
700     }
701 
702     /// Note that this is unsafe as the validity of `item` is not verified and
703     /// it contains a bunch of raw pointers.
704     #[cfg(feature = "component-model")]
705     pub(crate) unsafe fn push_export(&mut self, item: &crate::runtime::vm::Export) {
706         match item {
707             crate::runtime::vm::Export::Function(f) => {
708                 let f = f.func_ref.as_ref();
709                 self.functions.push(VMFunctionImport {
710                     wasm_call: f.wasm_call.unwrap(),
711                     array_call: f.array_call,
712                     vmctx: f.vmctx,
713                 });
714             }
715             crate::runtime::vm::Export::Global(g) => {
716                 self.globals.push(VMGlobalImport { from: g.definition });
717             }
718             crate::runtime::vm::Export::Table(t) => {
719                 self.tables.push(VMTableImport {
720                     from: t.definition,
721                     vmctx: t.vmctx,
722                 });
723             }
724             crate::runtime::vm::Export::Memory(m) => {
725                 self.memories.push(VMMemoryImport {
726                     from: m.definition,
727                     vmctx: m.vmctx,
728                     index: m.index,
729                 });
730             }
731         }
732     }
733 
734     pub(crate) fn as_ref(&self) -> Imports<'_> {
735         Imports {
736             tables: self.tables.values().as_slice(),
737             globals: self.globals.values().as_slice(),
738             memories: self.memories.values().as_slice(),
739             functions: self.functions.values().as_slice(),
740         }
741     }
742 }
743 
744 /// An instance, pre-instantiation, that is ready to be instantiated.
745 ///
746 /// This structure represents an instance *just before* it was instantiated,
747 /// after all type-checking and imports have been resolved. The only thing left
748 /// to do for this instance is to actually run the process of instantiation.
749 ///
750 /// Note that an `InstancePre` may not be tied to any particular [`Store`] if
751 /// none of the imports it closed over are tied to any particular [`Store`].
752 ///
753 /// This structure is created through the [`Linker::instantiate_pre`] method,
754 /// which also has some more information and examples.
755 ///
756 /// [`Store`]: crate::Store
757 /// [`Linker::instantiate_pre`]: crate::Linker::instantiate_pre
758 pub struct InstancePre<T> {
759     module: Module,
760 
761     /// The items which this `InstancePre` use to instantiate the `module`
762     /// provided, passed to `Instance::new_started` after inserting them into a
763     /// `Store`.
764     ///
765     /// Note that this is stored as an `Arc<[T]>` to quickly move a strong
766     /// reference to everything internally into a `Store<T>` without having to
767     /// clone each individual item.
768     items: Arc<[Definition]>,
769 
770     /// A count of `Definition::HostFunc` entries in `items` above to
771     /// preallocate space in a `Store` up front for all entries to be inserted.
772     host_funcs: usize,
773 
774     /// The `VMFuncRef`s for the functions in `items` that do not
775     /// have a `wasm_call` trampoline. We pre-allocate and pre-patch these
776     /// `VMFuncRef`s so that we don't have to do it at
777     /// instantiation time.
778     ///
779     /// This is an `Arc<[T]>` for the same reason as `items`.
780     func_refs: Arc<[VMFuncRef]>,
781 
782     _marker: core::marker::PhantomData<fn() -> T>,
783 }
784 
785 /// InstancePre's clone does not require T: Clone
786 impl<T> Clone for InstancePre<T> {
787     fn clone(&self) -> Self {
788         Self {
789             module: self.module.clone(),
790             items: self.items.clone(),
791             host_funcs: self.host_funcs,
792             func_refs: self.func_refs.clone(),
793             _marker: self._marker,
794         }
795     }
796 }
797 
798 impl<T> InstancePre<T> {
799     /// Creates a new `InstancePre` which type-checks the `items` provided and
800     /// on success is ready to instantiate a new instance.
801     ///
802     /// # Unsafety
803     ///
804     /// This method is unsafe as the `T` of the `InstancePre<T>` is not
805     /// guaranteed to be the same as the `T` within the `Store`, the caller must
806     /// verify that.
807     pub(crate) unsafe fn new(module: &Module, items: Vec<Definition>) -> Result<InstancePre<T>> {
808         typecheck(module, &items, |cx, ty, item| cx.definition(ty, &item.ty()))?;
809 
810         let mut func_refs = vec![];
811         let mut host_funcs = 0;
812         for item in &items {
813             match item {
814                 Definition::Extern(_, _) => {}
815                 Definition::HostFunc(f) => {
816                     host_funcs += 1;
817                     if f.func_ref().wasm_call.is_none() {
818                         // `f` needs its `VMFuncRef::wasm_call` patched with a
819                         // Wasm-to-native trampoline.
820                         debug_assert!(matches!(f.host_ctx(), crate::HostContext::Array(_)));
821                         func_refs.push(VMFuncRef {
822                             wasm_call: module.wasm_to_array_trampoline(f.sig_index()),
823                             ..*f.func_ref()
824                         });
825                     }
826                 }
827             }
828         }
829 
830         Ok(InstancePre {
831             module: module.clone(),
832             items: items.into(),
833             host_funcs,
834             func_refs: func_refs.into(),
835             _marker: core::marker::PhantomData,
836         })
837     }
838 
839     /// Returns a reference to the module that this [`InstancePre`] will be
840     /// instantiating.
841     pub fn module(&self) -> &Module {
842         &self.module
843     }
844 
845     /// Instantiates this instance, creating a new instance within the provided
846     /// `store`.
847     ///
848     /// This function will run the actual process of instantiation to
849     /// completion. This will use all of the previously-closed-over items as
850     /// imports to instantiate the module that this was originally created with.
851     ///
852     /// For more information about instantiation see [`Instance::new`].
853     ///
854     /// # Panics
855     ///
856     /// Panics if any import closed over by this [`InstancePre`] isn't owned by
857     /// `store`, or if `store` has async support enabled. Additionally this
858     /// function will panic if the `store` provided comes from a different
859     /// [`Engine`] than the [`InstancePre`] originally came from.
860     pub fn instantiate(&self, mut store: impl AsContextMut<Data = T>) -> Result<Instance> {
861         let mut store = store.as_context_mut();
862         let imports = pre_instantiate_raw(
863             &mut store.0,
864             &self.module,
865             &self.items,
866             self.host_funcs,
867             &self.func_refs,
868         )?;
869 
870         // This unsafety should be handled by the type-checking performed by the
871         // constructor of `InstancePre` to assert that all the imports we're passing
872         // in match the module we're instantiating.
873         unsafe { Instance::new_started(&mut store, &self.module, imports.as_ref()) }
874     }
875 
876     /// Creates a new instance, running the start function asynchronously
877     /// instead of inline.
878     ///
879     /// For more information about asynchronous instantiation see the
880     /// documentation on [`Instance::new_async`].
881     ///
882     /// # Panics
883     ///
884     /// Panics if any import closed over by this [`InstancePre`] isn't owned by
885     /// `store`, or if `store` does not have async support enabled.
886     #[cfg(feature = "async")]
887     pub async fn instantiate_async(
888         &self,
889         mut store: impl AsContextMut<Data = T>,
890     ) -> Result<Instance>
891     where
892         T: Send,
893     {
894         let mut store = store.as_context_mut();
895         let imports = pre_instantiate_raw(
896             &mut store.0,
897             &self.module,
898             &self.items,
899             self.host_funcs,
900             &self.func_refs,
901         )?;
902 
903         // This unsafety should be handled by the type-checking performed by the
904         // constructor of `InstancePre` to assert that all the imports we're passing
905         // in match the module we're instantiating.
906         unsafe { Instance::new_started_async(&mut store, &self.module, imports.as_ref()).await }
907     }
908 }
909 
910 /// Helper function shared between
911 /// `InstancePre::{instantiate,instantiate_async}`
912 ///
913 /// This is an out-of-line function to avoid the generic on `InstancePre` and
914 /// get this compiled into the `wasmtime` crate to avoid having it monomorphized
915 /// elsewhere.
916 fn pre_instantiate_raw(
917     store: &mut StoreOpaque,
918     module: &Module,
919     items: &Arc<[Definition]>,
920     host_funcs: usize,
921     func_refs: &Arc<[VMFuncRef]>,
922 ) -> Result<OwnedImports> {
923     if host_funcs > 0 {
924         // Any linker-defined function of the `Definition::HostFunc` variant
925         // will insert a function into the store automatically as part of
926         // instantiation, so reserve space here to make insertion more efficient
927         // as it won't have to realloc during the instantiation.
928         store.store_data_mut().reserve_funcs(host_funcs);
929 
930         // The usage of `to_extern_store_rooted` requires that the items are
931         // rooted via another means, which happens here by cloning the list of
932         // items into the store once. This avoids cloning each individual item
933         // below.
934         store.push_rooted_funcs(items.clone());
935         store.push_instance_pre_func_refs(func_refs.clone());
936     }
937 
938     let mut func_refs = func_refs.iter().map(|f| NonNull::from(f));
939     let mut imports = OwnedImports::new(module);
940     for import in items.iter() {
941         if !import.comes_from_same_store(store) {
942             bail!("cross-`Store` instantiation is not currently supported");
943         }
944         // This unsafety should be encapsulated in the constructor of
945         // `InstancePre` where the `T` of the original item should match the
946         // `T` of the store. Additionally the rooting necessary has happened
947         // above.
948         let item = match import {
949             Definition::Extern(e, _) => e.clone(),
950             Definition::HostFunc(func) => unsafe {
951                 func.to_func_store_rooted(
952                     store,
953                     if func.func_ref().wasm_call.is_none() {
954                         Some(func_refs.next().unwrap())
955                     } else {
956                         None
957                     },
958                 )
959                 .into()
960             },
961         };
962         imports.push(&item, store, module);
963     }
964 
965     Ok(imports)
966 }
967 
968 fn typecheck<I>(
969     module: &Module,
970     import_args: &[I],
971     check: impl Fn(&matching::MatchCx<'_>, &EntityType, &I) -> Result<()>,
972 ) -> Result<()> {
973     let env_module = module.compiled_module().module();
974     let expected_len = env_module.imports().count();
975     let actual_len = import_args.len();
976     if expected_len != actual_len {
977         bail!("expected {expected_len} imports, found {actual_len}");
978     }
979     let cx = matching::MatchCx::new(module.engine());
980     for ((name, field, mut expected_ty), actual) in env_module.imports().zip(import_args) {
981         expected_ty.canonicalize_for_runtime_usage(&mut |module_index| {
982             module.signatures().shared_type(module_index).unwrap()
983         });
984 
985         check(&cx, &expected_ty, actual)
986             .with_context(|| format!("incompatible import type for `{name}::{field}`"))?;
987     }
988     Ok(())
989 }
990