use crate::context::layout_vmcontext; use crate::data_segment::{declare_data_segment, emit_data_segment}; use crate::function::{declare_functions, emit_functions}; use crate::table::{declare_table, emit_table}; use anyhow::Result; use object::write::{Object, Relocation, StandardSection, Symbol, SymbolSection}; use object::{RelocationEncoding, RelocationKind, SymbolFlags, SymbolKind, SymbolScope}; use wasmtime_environ::isa::TargetFrontendConfig; use wasmtime_environ::{Compilation, DataInitializer, Module, Relocations}; fn emit_vmcontext_init( obj: &mut Object, module: &Module, target_config: &TargetFrontendConfig, ) -> Result<()> { let (data, table_relocs) = layout_vmcontext(module, target_config); let symbol_id = obj.add_symbol(Symbol { name: "_vmcontext_init".as_bytes().to_vec(), value: 0, size: 0, kind: SymbolKind::Data, scope: SymbolScope::Linkage, weak: false, section: SymbolSection::Undefined, flags: SymbolFlags::None, }); let section_id = obj.section_id(StandardSection::Data); let section_offset = obj.add_symbol_data(symbol_id, section_id, &data, 1); for reloc in table_relocs.iter() { let target_name = format!("_table_{}", reloc.index); let target_symbol = obj.symbol_id(target_name.as_bytes()).unwrap(); obj.add_relocation( section_id, Relocation { offset: section_offset + reloc.offset as u64, size: 64, // FIXME for all targets kind: RelocationKind::Absolute, encoding: RelocationEncoding::Generic, symbol: target_symbol, addend: 0, }, )?; } Ok(()) } /// Emits a module that has been emitted with the `wasmtime-environ` environment /// implementation to a native object file. pub fn emit_module( obj: &mut Object, module: &Module, compilation: &Compilation, relocations: &Relocations, data_initializers: &[DataInitializer], target_config: &TargetFrontendConfig, ) -> Result<()> { declare_functions(obj, module, relocations)?; for (i, initializer) in data_initializers.iter().enumerate() { declare_data_segment(obj, initializer, i)?; } for i in 0..module.local.table_plans.len() { declare_table(obj, i)?; } emit_functions(obj, module, compilation, relocations)?; for (i, initializer) in data_initializers.iter().enumerate() { emit_data_segment(obj, initializer, i)?; } for i in 0..module.local.table_plans.len() { emit_table(obj, i)?; } emit_vmcontext_init(obj, module, target_config)?; Ok(()) }