#include #include #include #include #include #include "wasm.hh" auto get_export_global(wasm::vec& exports, size_t i) -> wasm::Global* { if (exports.size() <= i || !exports[i]->global()) { std::cout << "> Error accessing global export " << i << "!" << std::endl; exit(1); } return exports[i]->global(); } auto get_export_func(const wasm::vec& exports, size_t i) -> const wasm::Func* { if (exports.size() <= i || !exports[i]->func()) { std::cout << "> Error accessing function export " << i << "!" << std::endl; exit(1); } return exports[i]->func(); } template void check(T actual, U expected) { if (actual != expected) { std::cout << "> Error reading value, expected " << expected << ", got " << actual << std::endl; exit(1); } } auto call(const wasm::Func* func) -> wasm::Val { wasm::Val results[1]; if (func->call(nullptr, results)) { std::cout << "> Error calling function!" << std::endl; exit(1); } return results[0].copy(); } void call(const wasm::Func* func, wasm::Val&& arg) { wasm::Val args[1] = {std::move(arg)}; if (func->call(args)) { std::cout << "> Error calling function!" << std::endl; exit(1); } } void run() { // Initialize. std::cout << "Initializing..." << std::endl; auto engine = wasm::Engine::make(); auto store_ = wasm::Store::make(engine.get()); auto store = store_.get(); // Load binary. std::cout << "Loading binary..." << std::endl; std::ifstream file("global.wasm"); file.seekg(0, std::ios_base::end); auto file_size = file.tellg(); file.seekg(0); auto binary = wasm::vec::make_uninitialized(file_size); file.read(binary.get(), file_size); file.close(); if (file.fail()) { std::cout << "> Error loading module!" << std::endl; return; } // Compile. std::cout << "Compiling module..." << std::endl; auto module = wasm::Module::make(store, binary); if (!module) { std::cout << "> Error compiling module!" << std::endl; return; } // Create external globals. std::cout << "Creating globals..." << std::endl; auto const_f32_type = wasm::GlobalType::make( wasm::ValType::make(wasm::F32), wasm::CONST); auto const_i64_type = wasm::GlobalType::make( wasm::ValType::make(wasm::I64), wasm::CONST); auto var_f32_type = wasm::GlobalType::make( wasm::ValType::make(wasm::F32), wasm::VAR); auto var_i64_type = wasm::GlobalType::make( wasm::ValType::make(wasm::I64), wasm::VAR); auto const_f32_import = wasm::Global::make(store, const_f32_type.get(), wasm::Val::f32(1)); auto const_i64_import = wasm::Global::make(store, const_i64_type.get(), wasm::Val::i64(2)); auto var_f32_import = wasm::Global::make(store, var_f32_type.get(), wasm::Val::f32(3)); auto var_i64_import = wasm::Global::make(store, var_i64_type.get(), wasm::Val::i64(4)); // Instantiate. std::cout << "Instantiating module..." << std::endl; wasm::Extern* imports[] = { const_f32_import.get(), const_i64_import.get(), var_f32_import.get(), var_i64_import.get() }; auto instance = wasm::Instance::make(store, module.get(), imports); if (!instance) { std::cout << "> Error instantiating module!" << std::endl; return; } // Extract export. std::cout << "Extracting exports..." << std::endl; auto exports = instance->exports(); size_t i = 0; auto const_f32_export = get_export_global(exports, i++); auto const_i64_export = get_export_global(exports, i++); auto var_f32_export = get_export_global(exports, i++); auto var_i64_export = get_export_global(exports, i++); auto get_const_f32_import = get_export_func(exports, i++); auto get_const_i64_import = get_export_func(exports, i++); auto get_var_f32_import = get_export_func(exports, i++); auto get_var_i64_import = get_export_func(exports, i++); auto get_const_f32_export = get_export_func(exports, i++); auto get_const_i64_export = get_export_func(exports, i++); auto get_var_f32_export = get_export_func(exports, i++); auto get_var_i64_export = get_export_func(exports, i++); auto set_var_f32_import = get_export_func(exports, i++); auto set_var_i64_import = get_export_func(exports, i++); auto set_var_f32_export = get_export_func(exports, i++); auto set_var_i64_export = get_export_func(exports, i++); // Interact. std::cout << "Accessing globals..." << std::endl; // Check initial values. check(const_f32_import->get().f32(), 1); check(const_i64_import->get().i64(), 2); check(var_f32_import->get().f32(), 3); check(var_i64_import->get().i64(), 4); check(const_f32_export->get().f32(), 5); check(const_i64_export->get().i64(), 6); check(var_f32_export->get().f32(), 7); check(var_i64_export->get().i64(), 8); check(call(get_const_f32_import).f32(), 1); check(call(get_const_i64_import).i64(), 2); check(call(get_var_f32_import).f32(), 3); check(call(get_var_i64_import).i64(), 4); check(call(get_const_f32_export).f32(), 5); check(call(get_const_i64_export).i64(), 6); check(call(get_var_f32_export).f32(), 7); check(call(get_var_i64_export).i64(), 8); // Modify variables through API and check again. var_f32_import->set(wasm::Val::f32(33)); var_i64_import->set(wasm::Val::i64(34)); var_f32_export->set(wasm::Val::f32(37)); var_i64_export->set(wasm::Val::i64(38)); check(var_f32_import->get().f32(), 33); check(var_i64_import->get().i64(), 34); check(var_f32_export->get().f32(), 37); check(var_i64_export->get().i64(), 38); check(call(get_var_f32_import).f32(), 33); check(call(get_var_i64_import).i64(), 34); check(call(get_var_f32_export).f32(), 37); check(call(get_var_i64_export).i64(), 38); // Modify variables through calls and check again. call(set_var_f32_import, wasm::Val::f32(73)); call(set_var_i64_import, wasm::Val::i64(74)); call(set_var_f32_export, wasm::Val::f32(77)); call(set_var_i64_export, wasm::Val::i64(78)); check(var_f32_import->get().f32(), 73); check(var_i64_import->get().i64(), 74); check(var_f32_export->get().f32(), 77); check(var_i64_export->get().i64(), 78); check(call(get_var_f32_import).f32(), 73); check(call(get_var_i64_import).i64(), 74); check(call(get_var_f32_export).f32(), 77); check(call(get_var_i64_export).i64(), 78); // Shut down. std::cout << "Shutting down..." << std::endl; } int main(int argc, const char* argv[]) { run(); std::cout << "Done." << std::endl; return 0; }