// Copyright 2016 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #ifndef WASM_RUN_UTILS_H #define WASM_RUN_UTILS_H #include #include #include #include #include #include #include "src/base/utils/random-number-generator.h" #include "src/compiler/compiler-source-position-table.h" #include "src/compiler/int64-lowering.h" #include "src/compiler/js-graph.h" #include "src/compiler/node.h" #include "src/compiler/wasm-compiler.h" #include "src/trap-handler/trap-handler.h" #include "src/wasm/canonical-types.h" #include "src/wasm/function-body-decoder.h" #include "src/wasm/local-decl-encoder.h" #include "src/wasm/wasm-code-manager.h" #include "src/wasm/wasm-external-refs.h" #include "src/wasm/wasm-js.h" #include "src/wasm/wasm-module.h" #include "src/wasm/wasm-objects-inl.h" #include "src/wasm/wasm-objects.h" #include "src/wasm/wasm-opcodes.h" #include "src/wasm/wasm-tier.h" #include "src/zone/accounting-allocator.h" #include "src/zone/zone.h" #include "test/cctest/cctest.h" #include "test/cctest/compiler/graph-and-builders.h" #include "test/common/call-tester.h" #include "test/common/value-helper.h" #include "test/common/wasm/flag-utils.h" #include "test/common/wasm/wasm-interpreter.h" namespace v8 { namespace internal { namespace wasm { enum class TestExecutionTier : int8_t { kLiftoff = static_cast(ExecutionTier::kLiftoff), kTurbofan = static_cast(ExecutionTier::kTurbofan), kInterpreter, kLiftoffForFuzzing }; static_assert( std::is_same::type, std::underlying_type::type>::value, "enum types match"); enum TestingModuleMemoryType { kMemory32, kMemory64 }; using base::ReadLittleEndianValue; using base::WriteLittleEndianValue; constexpr uint32_t kMaxFunctions = 10; constexpr uint32_t kMaxGlobalsSize = 128; // Don't execute more than 16k steps. constexpr int kMaxNumSteps = 16 * 1024; using compiler::CallDescriptor; using compiler::MachineTypeForC; using compiler::Node; // TODO(titzer): check traps more robustly in tests. // Currently, in tests, we just return 0xDEADBEEF from the function in which // the trap occurs if the runtime context is not available to throw a JavaScript // exception. #define CHECK_TRAP32(x) \ CHECK_EQ(0xDEADBEEF, (base::bit_cast(x)) & 0xFFFFFFFF) #define CHECK_TRAP64(x) \ CHECK_EQ(0xDEADBEEFDEADBEEF, \ (base::bit_cast(x)) & 0xFFFFFFFFFFFFFFFF) #define CHECK_TRAP(x) CHECK_TRAP32(x) #define WASM_WRAPPER_RETURN_VALUE 8754 #define BUILD(r, ...) \ do { \ byte __code[] = {__VA_ARGS__}; \ r.Build(__code, __code + arraysize(__code)); \ } while (false) #define ADD_CODE(vec, ...) \ do { \ byte __buf[] = {__VA_ARGS__}; \ for (size_t __i = 0; __i < sizeof(__buf); __i++) \ vec.push_back(__buf[__i]); \ } while (false) // For tests that must manually import a JSFunction with source code. struct ManuallyImportedJSFunction { const FunctionSig* sig; Handle js_function; }; // Helper Functions. bool IsSameNan(float expected, float actual); bool IsSameNan(double expected, double actual); // A Wasm module builder. Globals are pre-set, however, memory and code may be // progressively added by a test. In turn, we piecemeal update the runtime // objects, i.e. {WasmInstanceObject}, {WasmModuleObject} and, if necessary, // the interpreter. class TestingModuleBuilder { public: TestingModuleBuilder(Zone*, ManuallyImportedJSFunction*, TestExecutionTier, RuntimeExceptionSupport, TestingModuleMemoryType, Isolate* isolate); ~TestingModuleBuilder(); void ChangeOriginToAsmjs() { test_module_->origin = kAsmJsSloppyOrigin; } byte* AddMemory(uint32_t size, SharedFlag shared = SharedFlag::kNotShared); size_t CodeTableLength() const { return native_module_->num_functions(); } template T* AddMemoryElems(uint32_t count) { AddMemory(count * sizeof(T)); return raw_mem_start(); } template T* AddGlobal(ValueType type = ValueType::For(MachineTypeForC())) { const WasmGlobal* global = AddGlobal(type); return reinterpret_cast(globals_data_ + global->offset); } byte AddSignature(const FunctionSig* sig) { test_module_->add_signature(sig, kNoSuperType); GetTypeCanonicalizer()->AddRecursiveGroup(test_module_.get(), 1); instance_object_->set_isorecursive_canonical_types( test_module_->isorecursive_canonical_type_ids.data()); size_t size = test_module_->types.size(); CHECK_GT(127, size); return static_cast(size - 1); } uint32_t mem_size() { return mem_size_; } template T* raw_mem_start() { DCHECK(mem_start_); return reinterpret_cast(mem_start_); } template T* raw_mem_end() { DCHECK(mem_start_); return reinterpret_cast(mem_start_ + mem_size_); } template T raw_mem_at(int i) { DCHECK(mem_start_); return ReadMemory(&(reinterpret_cast(mem_start_)[i])); } template T raw_val_at(int i) { return ReadMemory(reinterpret_cast(mem_start_ + i)); } template void WriteMemory(T* p, T val) { WriteLittleEndianValue(reinterpret_cast
(p), val); } template T ReadMemory(T* p) { return ReadLittleEndianValue(reinterpret_cast
(p)); } // Zero-initialize the memory. void BlankMemory() { byte* raw = raw_mem_start(); memset(raw, 0, mem_size_); } // Pseudo-randomly initialize the memory. void RandomizeMemory(unsigned int seed = 88) { byte* raw = raw_mem_start(); byte* end = raw_mem_end(); v8::base::RandomNumberGenerator rng; rng.SetSeed(seed); rng.NextBytes(raw, end - raw); } void SetMaxMemPages(uint32_t maximum_pages) { test_module_->maximum_pages = maximum_pages; if (instance_object()->has_memory_object()) { instance_object()->memory_object().set_maximum_pages(maximum_pages); } } void SetHasSharedMemory() { test_module_->has_shared_memory = true; } enum FunctionType { kImport, kWasm }; uint32_t AddFunction(const FunctionSig* sig, const char* name, FunctionType type); // Freezes the signature map of the module and allocates the storage for // export wrappers. void InitializeWrapperCache(); // Wrap the code so it can be called as a JS function. Handle WrapCode(uint32_t index); // If function_indexes is {nullptr}, the contents of the table will be // initialized with null functions. void AddIndirectFunctionTable(const uint16_t* function_indexes, uint32_t table_size, ValueType table_type = kWasmFuncRef); uint32_t AddBytes(base::Vector bytes); uint32_t AddException(const FunctionSig* sig); uint32_t AddPassiveDataSegment(base::Vector bytes); uint32_t AddPassiveElementSegment(const std::vector& entries); WasmFunction* GetFunctionAt(int index) { return &test_module_->functions[index]; } WasmInterpreter* interpreter() const { return interpreter_.get(); } bool interpret() const { return interpreter_ != nullptr; } Isolate* isolate() const { return isolate_; } Handle instance_object() const { return instance_object_; } WasmCode* GetFunctionCode(uint32_t index) const { return native_module_->GetCode(index); } Address globals_start() const { return reinterpret_cast
(globals_data_); } void SetTieredDown() { native_module_->SetTieringState(kTieredDown); execution_tier_ = TestExecutionTier::kLiftoff; } void TierDown() { SetTieredDown(); native_module_->RecompileForTiering(); } CompilationEnv CreateCompilationEnv(); TestExecutionTier test_execution_tier() const { return execution_tier_; } ExecutionTier execution_tier() const { switch (execution_tier_) { case TestExecutionTier::kTurbofan: return ExecutionTier::kTurbofan; case TestExecutionTier::kLiftoff: return ExecutionTier::kLiftoff; default: UNREACHABLE(); } } RuntimeExceptionSupport runtime_exception_support() const { return runtime_exception_support_; } void set_max_steps(int n) { max_steps_ = n; } int* max_steps_ptr() { return &max_steps_; } int32_t nondeterminism() { return nondeterminism_; } int32_t* non_determinism_ptr() { return &nondeterminism_; } void EnableFeature(WasmFeature feature) { enabled_features_.Add(feature); } private: std::shared_ptr test_module_; Isolate* isolate_; WasmFeatures enabled_features_; uint32_t global_offset = 0; byte* mem_start_ = nullptr; uint32_t mem_size_ = 0; byte* globals_data_ = nullptr; std::unique_ptr interpreter_; TestExecutionTier execution_tier_; Handle instance_object_; NativeModule* native_module_ = nullptr; RuntimeExceptionSupport runtime_exception_support_; int32_t max_steps_ = kMaxNumSteps; int32_t nondeterminism_ = 0; // Data segment arrays that are normally allocated on the instance. std::vector data_segment_data_; std::vector
data_segment_starts_; std::vector data_segment_sizes_; std::vector dropped_elem_segments_; const WasmGlobal* AddGlobal(ValueType type); Handle InitInstanceObject(); }; void TestBuildingGraph(Zone* zone, compiler::JSGraph* jsgraph, CompilationEnv* env, const FunctionSig* sig, compiler::SourcePositionTable* source_position_table, const byte* start, const byte* end); class WasmFunctionWrapper : private compiler::GraphAndBuilders { public: WasmFunctionWrapper(Zone* zone, int num_params); void Init(CallDescriptor* call_descriptor, MachineType return_type, base::Vector param_types); template void Init(CallDescriptor* call_descriptor) { std::array param_machine_types{ {MachineTypeForC()...}}; base::Vector param_vec(param_machine_types.data(), param_machine_types.size()); Init(call_descriptor, MachineTypeForC(), param_vec); } void SetInnerCode(WasmCode* code) { intptr_t address = static_cast(code->instruction_start()); compiler::NodeProperties::ChangeOp( inner_code_node_, common()->ExternalConstant(ExternalReference::FromRawAddress(address))); } const compiler::Operator* IntPtrConstant(intptr_t value) { return machine()->Is32() ? common()->Int32Constant(static_cast(value)) : common()->Int64Constant(static_cast(value)); } void SetInstance(Handle instance) { compiler::NodeProperties::ChangeOp(context_address_, common()->HeapConstant(instance)); } Handle GetWrapperCode(Isolate* isolate = nullptr); Signature* signature() const { return signature_; } private: Node* inner_code_node_; Node* context_address_; MaybeHandle code_; Signature* signature_; }; // A helper for compiling wasm functions for testing. // It contains the internal state for compilation (i.e. TurboFan graph) and // interpretation (by adding to the interpreter manually). class WasmFunctionCompiler : public compiler::GraphAndBuilders { public: ~WasmFunctionCompiler(); Isolate* isolate() { return builder_->isolate(); } CallDescriptor* descriptor() { if (descriptor_ == nullptr) { descriptor_ = compiler::GetWasmCallDescriptor(zone(), sig); } return descriptor_; } uint32_t function_index() { return function_->func_index; } uint32_t sig_index() { return function_->sig_index; } void Build(const byte* start, const byte* end); byte AllocateLocal(ValueType type) { uint32_t index = local_decls.AddLocals(1, type); byte result = static_cast(index); DCHECK_EQ(index, result); return result; } void SetSigIndex(int sig_index) { function_->sig_index = sig_index; } private: friend class WasmRunnerBase; WasmFunctionCompiler(Zone* zone, const FunctionSig* sig, TestingModuleBuilder* builder, const char* name); compiler::JSGraph jsgraph; const FunctionSig* sig; // The call descriptor is initialized when the function is compiled. CallDescriptor* descriptor_; TestingModuleBuilder* builder_; WasmFunction* function_; LocalDeclEncoder local_decls; compiler::SourcePositionTable source_position_table_; WasmInterpreter* interpreter_; }; // A helper class to build a module around Wasm bytecode, generate machine // code, and run that code. class WasmRunnerBase : public InitializedHandleScope { public: WasmRunnerBase(ManuallyImportedJSFunction* maybe_import, TestExecutionTier execution_tier, int num_params, RuntimeExceptionSupport runtime_exception_support = kNoRuntimeExceptionSupport, TestingModuleMemoryType mem_type = kMemory32, Isolate* isolate = nullptr) : InitializedHandleScope(isolate), zone_(&allocator_, ZONE_NAME, kCompressGraphZone), builder_(&zone_, maybe_import, execution_tier, runtime_exception_support, mem_type, isolate), wrapper_(&zone_, num_params) {} static void SetUpTrapCallback() { WasmRunnerBase::trap_happened = false; auto trap_callback = []() -> void { WasmRunnerBase::trap_happened = true; set_trap_callback_for_testing(nullptr); }; set_trap_callback_for_testing(trap_callback); } // Builds a graph from the given Wasm code and generates the machine // code and call wrapper for that graph. This method must not be called // more than once. void Build(const byte* start, const byte* end) { CHECK(!compiled_); compiled_ = true; functions_[0]->Build(start, end); } // Resets the state for building the next function. // The main function called will always be the first function. template WasmFunctionCompiler& NewFunction(const char* name = nullptr) { return NewFunction(CreateSig(), name); } // Resets the state for building the next function. // The main function called will be the last generated function. // Returns the index of the previously built function. WasmFunctionCompiler& NewFunction(const FunctionSig* sig, const char* name = nullptr) { functions_.emplace_back( new WasmFunctionCompiler(&zone_, sig, &builder_, name)); byte sig_index = builder().AddSignature(sig); functions_.back()->SetSigIndex(sig_index); return *functions_.back(); } byte AllocateLocal(ValueType type) { return functions_[0]->AllocateLocal(type); } uint32_t function_index() { return functions_[0]->function_index(); } WasmFunction* function() { return functions_[0]->function_; } WasmInterpreter* interpreter() { DCHECK(interpret()); return functions_[0]->interpreter_; } bool possible_nondeterminism() { return possible_nondeterminism_; } TestingModuleBuilder& builder() { return builder_; } Zone* zone() { return &zone_; } bool interpret() { return builder_.interpret(); } void TierDown() { builder_.TierDown(); } template FunctionSig* CreateSig() { return WasmRunnerBase::CreateSig(&zone_); } template static FunctionSig* CreateSig(Zone* zone) { std::array param_machine_types{ {MachineTypeForC()...}}; base::Vector param_vec(param_machine_types.data(), param_machine_types.size()); return CreateSig(zone, MachineTypeForC(), param_vec); } void CheckCallApplyViaJS(double expected, uint32_t function_index, Handle* buffer, int count) { Isolate* isolate = builder_.isolate(); SetUpTrapCallback(); if (jsfuncs_.size() <= function_index) { jsfuncs_.resize(function_index + 1); } if (jsfuncs_[function_index].is_null()) { jsfuncs_[function_index] = builder_.WrapCode(function_index); } Handle jsfunc = jsfuncs_[function_index]; Handle global(isolate->context().global_object(), isolate); MaybeHandle retval = Execution::TryCall(isolate, jsfunc, global, count, buffer, Execution::MessageHandling::kReport, nullptr); if (retval.is_null() || WasmRunnerBase::trap_happened) { CHECK_EQ(expected, static_cast(0xDEADBEEF)); } else { Handle result = retval.ToHandleChecked(); if (result->IsSmi()) { CHECK_EQ(expected, Smi::ToInt(*result)); } else { CHECK(result->IsHeapNumber()); CHECK_DOUBLE_EQ(expected, HeapNumber::cast(*result).value()); } } if (builder_.interpret()) { CHECK_GT(builder_.interpreter()->NumInterpretedCalls(), 0); } } Handle GetWrapperCode() { return wrapper_.GetWrapperCode(main_isolate()); } private: static FunctionSig* CreateSig(Zone* zone, MachineType return_type, base::Vector param_types); protected: wasm::WasmCodeRefScope code_ref_scope_; std::vector> jsfuncs_; v8::internal::AccountingAllocator allocator_; Zone zone_; TestingModuleBuilder builder_; std::vector> functions_; WasmFunctionWrapper wrapper_; bool compiled_ = false; bool possible_nondeterminism_ = false; int32_t main_fn_index_ = 0; static void SetThreadInWasmFlag() { *reinterpret_cast(trap_handler::GetThreadInWasmThreadLocalAddress()) = true; } static void ClearThreadInWasmFlag() { *reinterpret_cast(trap_handler::GetThreadInWasmThreadLocalAddress()) = false; } public: // This field has to be static. Otherwise, gcc complains about the use in // the lambda context below. static bool trap_happened; }; template inline WasmValue WasmValueInitializer(T value) { return WasmValue(value); } template <> inline WasmValue WasmValueInitializer(int8_t value) { return WasmValue(static_cast(value)); } template <> inline WasmValue WasmValueInitializer(int16_t value) { return WasmValue(static_cast(value)); } template class WasmRunner : public WasmRunnerBase { public: explicit WasmRunner(TestExecutionTier execution_tier, ManuallyImportedJSFunction* maybe_import = nullptr, const char* main_fn_name = "main", RuntimeExceptionSupport runtime_exception_support = kNoRuntimeExceptionSupport, TestingModuleMemoryType mem_type = kMemory32, Isolate* isolate = nullptr) : WasmRunnerBase(maybe_import, execution_tier, sizeof...(ParamTypes), runtime_exception_support, mem_type, isolate) { WasmFunctionCompiler& main_fn = NewFunction(main_fn_name); // Non-zero if there is an import. main_fn_index_ = main_fn.function_index(); if (!interpret()) { wrapper_.Init(main_fn.descriptor()); } } ReturnType Call(ParamTypes... p) { // Save the original context, because CEntry (for runtime calls) will // reset / invalidate it when returning. SaveContext save_context(main_isolate()); DCHECK(compiled_); if (interpret()) return CallInterpreter(p...); ReturnType return_value = static_cast(0xDEADBEEFDEADBEEF); SetUpTrapCallback(); wrapper_.SetInnerCode(builder_.GetFunctionCode(main_fn_index_)); wrapper_.SetInstance(builder_.instance_object()); Handle wrapper_code = GetWrapperCode(); compiler::CodeRunner runner(main_isolate(), wrapper_code, wrapper_.signature()); int32_t result; { SetThreadInWasmFlag(); result = runner.Call(static_cast(&p)..., static_cast(&return_value)); ClearThreadInWasmFlag(); } CHECK_EQ(WASM_WRAPPER_RETURN_VALUE, result); return WasmRunnerBase::trap_happened ? static_cast(0xDEADBEEFDEADBEEF) : return_value; } ReturnType CallInterpreter(ParamTypes... p) { interpreter()->Reset(); std::array args{{WasmValueInitializer(p)...}}; interpreter()->InitFrame(function(), args.data()); interpreter()->Run(); CHECK_GT(interpreter()->NumInterpretedCalls(), 0); if (interpreter()->state() == WasmInterpreter::FINISHED) { WasmValue val = interpreter()->GetReturnValue(); possible_nondeterminism_ |= interpreter()->PossibleNondeterminism(); return val.to(); } else if (interpreter()->state() == WasmInterpreter::TRAPPED) { // TODO(titzer): return the correct trap code int64_t result = 0xDEADBEEFDEADBEEF; return static_cast(result); } else { // TODO(titzer): falling off end return ReturnType{0}; } } void CheckCallViaJS(double expected, ParamTypes... p) { Isolate* isolate = builder_.isolate(); // MSVC doesn't allow empty arrays, so include a dummy at the end. Handle buffer[] = {isolate->factory()->NewNumber(p)..., Handle()}; CheckCallApplyViaJS(expected, function()->func_index, buffer, sizeof...(p)); } void CheckCallViaJSTraps(ParamTypes... p) { CheckCallViaJS(static_cast(0xDEADBEEF), p...); } void SetMaxSteps(int n) { builder_.set_max_steps(n); } bool HasNondeterminism() { return builder_.nondeterminism(); } }; // A macro to define tests that run in different engine configurations. #define WASM_EXEC_TEST(name) \ void RunWasm_##name(TestExecutionTier execution_tier); \ TEST(RunWasmTurbofan_##name) { \ RunWasm_##name(TestExecutionTier::kTurbofan); \ } \ TEST(RunWasmLiftoff_##name) { RunWasm_##name(TestExecutionTier::kLiftoff); } \ TEST(RunWasmInterpreter_##name) { \ RunWasm_##name(TestExecutionTier::kInterpreter); \ } \ void RunWasm_##name(TestExecutionTier execution_tier) #define UNINITIALIZED_WASM_EXEC_TEST(name) \ void RunWasm_##name(TestExecutionTier execution_tier); \ UNINITIALIZED_TEST(RunWasmTurbofan_##name) { \ RunWasm_##name(TestExecutionTier::kTurbofan); \ } \ UNINITIALIZED_TEST(RunWasmLiftoff_##name) { \ RunWasm_##name(TestExecutionTier::kLiftoff); \ } \ UNINITIALIZED_TEST(RunWasmInterpreter_##name) { \ RunWasm_##name(TestExecutionTier::kInterpreter); \ } \ void RunWasm_##name(TestExecutionTier execution_tier) #define WASM_COMPILED_EXEC_TEST(name) \ void RunWasm_##name(TestExecutionTier execution_tier); \ TEST(RunWasmTurbofan_##name) { \ RunWasm_##name(TestExecutionTier::kTurbofan); \ } \ TEST(RunWasmLiftoff_##name) { RunWasm_##name(TestExecutionTier::kLiftoff); } \ void RunWasm_##name(TestExecutionTier execution_tier) } // namespace wasm } // namespace internal } // namespace v8 #endif