// Copyright 2020 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. #include #include "src/base/vector.h" #include "src/utils/utils.h" #include "src/wasm/module-decoder.h" #include "src/wasm/struct-types.h" #include "src/wasm/wasm-arguments.h" #include "src/wasm/wasm-engine.h" #include "src/wasm/wasm-module-builder.h" #include "src/wasm/wasm-module.h" #include "src/wasm/wasm-objects-inl.h" #include "src/wasm/wasm-opcodes.h" #include "test/cctest/cctest.h" #include "test/cctest/compiler/value-helper.h" #include "test/cctest/wasm/wasm-run-utils.h" #include "test/common/wasm/test-signatures.h" #include "test/common/wasm/wasm-macro-gen.h" #include "test/common/wasm/wasm-module-runner.h" namespace v8 { namespace internal { namespace wasm { namespace test_gc { using F = std::pair; class WasmGCTester { public: explicit WasmGCTester( TestExecutionTier execution_tier = TestExecutionTier::kTurbofan) : flag_gc(&v8::internal::FLAG_experimental_wasm_gc, true), flag_reftypes(&v8::internal::FLAG_experimental_wasm_reftypes, true), flag_typedfuns(&v8::internal::FLAG_experimental_wasm_typed_funcref, true), flag_liftoff( &v8::internal::FLAG_liftoff, execution_tier == TestExecutionTier::kTurbofan ? false : true), flag_liftoff_only( &v8::internal::FLAG_liftoff_only, execution_tier == TestExecutionTier::kLiftoff ? true : false), flag_tierup(&v8::internal::FLAG_wasm_tier_up, false), zone_(&allocator, ZONE_NAME), builder_(&zone_), isolate_(CcTest::InitIsolateOnce()), scope(isolate_), thrower(isolate_, "Test wasm GC") { testing::SetupIsolateForWasmModule(isolate_); } byte AddGlobal(ValueType type, bool mutability, WasmInitExpr init) { return builder_.AddGlobal(type, mutability, std::move(init)); } byte DefineFunction(FunctionSig* sig, std::initializer_list locals, std::initializer_list code) { WasmFunctionBuilder* fun = builder_.AddFunction(sig); for (ValueType local : locals) { fun->AddLocal(local); } fun->EmitCode(code.begin(), static_cast(code.size())); return fun->func_index(); } void DefineExportedFunction(const char* name, FunctionSig* sig, std::initializer_list code) { WasmFunctionBuilder* fun = builder_.AddFunction(sig); fun->EmitCode(code.begin(), static_cast(code.size())); builder_.AddExport(base::CStrVector(name), fun); } MaybeHandle CallExportedFunction(const char* name, int argc, Handle args[]) { Handle func = testing::GetExportedFunction(isolate_, instance_, name) .ToHandleChecked(); return Execution::Call(isolate_, func, isolate_->factory()->undefined_value(), argc, args); } byte DefineStruct(std::initializer_list fields) { StructType::Builder type_builder(&zone_, static_cast(fields.size())); for (F field : fields) { type_builder.AddField(field.first, field.second); } return builder_.AddStructType(type_builder.Build()); } byte DefineArray(ValueType element_type, bool mutability) { return builder_.AddArrayType( zone_.New(element_type, mutability)); } byte DefineSignature(FunctionSig* sig) { return builder_.AddSignature(sig); } byte DefineTable(ValueType type, uint32_t min_size, uint32_t max_size) { return builder_.AddTable(type, min_size, max_size); } void CompileModule() { ZoneBuffer buffer(&zone_); builder_.WriteTo(&buffer); MaybeHandle maybe_instance = testing::CompileAndInstantiateForTesting( isolate_, &thrower, ModuleWireBytes(buffer.begin(), buffer.end())); if (thrower.error()) FATAL("%s", thrower.error_msg()); instance_ = maybe_instance.ToHandleChecked(); } void CheckResult(uint32_t function_index, int32_t expected) { const FunctionSig* sig = sigs.i_v(); DCHECK(*sig == *instance_->module()->functions[function_index].sig); CWasmArgumentsPacker packer(CWasmArgumentsPacker::TotalSize(sig)); CheckResultImpl(function_index, sig, &packer, expected); } void CheckResult(uint32_t function_index, int32_t expected, int32_t arg) { const FunctionSig* sig = sigs.i_i(); DCHECK(*sig == *instance_->module()->functions[function_index].sig); CWasmArgumentsPacker packer(CWasmArgumentsPacker::TotalSize(sig)); packer.Push(arg); CheckResultImpl(function_index, sig, &packer, expected); } MaybeHandle GetResultObject(uint32_t function_index) { const FunctionSig* sig = instance_->module()->functions[function_index].sig; DCHECK_EQ(sig->parameter_count(), 0); DCHECK_EQ(sig->return_count(), 1); CWasmArgumentsPacker packer(CWasmArgumentsPacker::TotalSize(sig)); CallFunctionImpl(function_index, sig, &packer); CHECK(!isolate_->has_pending_exception()); packer.Reset(); return Handle(Object(packer.Pop
()), isolate_); } MaybeHandle GetResultObject(uint32_t function_index, int32_t arg) { const FunctionSig* sig = instance_->module()->functions[function_index].sig; DCHECK_EQ(sig->parameter_count(), 1); DCHECK_EQ(sig->return_count(), 1); DCHECK(sig->parameters()[0] == kWasmI32); CWasmArgumentsPacker packer(CWasmArgumentsPacker::TotalSize(sig)); packer.Push(arg); CallFunctionImpl(function_index, sig, &packer); CHECK(!isolate_->has_pending_exception()); packer.Reset(); return Handle(Object(packer.Pop
()), isolate_); } void CheckHasThrown(uint32_t function_index, const char* expected = "") { const FunctionSig* sig = instance_->module()->functions[function_index].sig; DCHECK_EQ(sig->parameter_count(), 0); CWasmArgumentsPacker packer(CWasmArgumentsPacker::TotalSize(sig)); CheckHasThrownImpl(function_index, sig, &packer, expected); } void CheckHasThrown(uint32_t function_index, int32_t arg, const char* expected = "") { const FunctionSig* sig = instance_->module()->functions[function_index].sig; DCHECK_EQ(sig->parameter_count(), 1); DCHECK(sig->parameters()[0] == kWasmI32); CWasmArgumentsPacker packer(CWasmArgumentsPacker::TotalSize(sig)); packer.Push(arg); CheckHasThrownImpl(function_index, sig, &packer, expected); } Handle instance() { return instance_; } Isolate* isolate() { return isolate_; } WasmModuleBuilder* builder() { return &builder_; } Zone* zone() { return &zone_; } TestSignatures sigs; private: const FlagScope flag_gc; const FlagScope flag_reftypes; const FlagScope flag_typedfuns; const FlagScope flag_liftoff; const FlagScope flag_liftoff_only; const FlagScope flag_tierup; void CheckResultImpl(uint32_t function_index, const FunctionSig* sig, CWasmArgumentsPacker* packer, int32_t expected) { CallFunctionImpl(function_index, sig, packer); if (isolate_->has_pending_exception()) { Handle message = ErrorUtils::ToString(isolate_, handle(isolate_->pending_exception(), isolate_)) .ToHandleChecked(); FATAL("%s", message->ToCString().get()); } packer->Reset(); CHECK_EQ(expected, packer->Pop()); } void CheckHasThrownImpl(uint32_t function_index, const FunctionSig* sig, CWasmArgumentsPacker* packer, const char* expected) { CallFunctionImpl(function_index, sig, packer); CHECK(isolate_->has_pending_exception()); Handle message = ErrorUtils::ToString(isolate_, handle(isolate_->pending_exception(), isolate_)) .ToHandleChecked(); std::string message_str(message->ToCString().get()); CHECK_NE(message_str.find(expected), std::string::npos); isolate_->clear_pending_exception(); } void CallFunctionImpl(uint32_t function_index, const FunctionSig* sig, CWasmArgumentsPacker* packer) { WasmCodeRefScope scope; NativeModule* native_module = instance_->module_object().native_module(); WasmCode* code = native_module->GetCode(function_index); Address wasm_call_target = code->instruction_start(); Handle object_ref = instance_; Handle c_wasm_entry = compiler::CompileCWasmEntry(isolate_, sig, native_module->module()); Execution::CallWasm(isolate_, c_wasm_entry, wasm_call_target, object_ref, packer->argv()); } v8::internal::AccountingAllocator allocator; Zone zone_; WasmModuleBuilder builder_; Isolate* const isolate_; const HandleScope scope; Handle instance_; ErrorThrower thrower; }; ValueType ref(uint32_t type_index) { return ValueType::Ref(type_index, kNonNullable); } ValueType optref(uint32_t type_index) { return ValueType::Ref(type_index, kNullable); } WASM_COMPILED_EXEC_TEST(WasmBasicStruct) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineStruct({F(kWasmI32, true), F(kWasmI32, true)}); const byte empty_struct_index = tester.DefineStruct({}); ValueType kRefType = ref(type_index); ValueType kEmptyStructType = ref(empty_struct_index); ValueType kOptRefType = optref(type_index); FunctionSig sig_q_v(1, 0, &kRefType); FunctionSig sig_qe_v(1, 0, &kEmptyStructType); // Test struct.new and struct.get. const byte kGet1 = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_STRUCT_GET( type_index, 0, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_I32V(64), WASM_RTT_CANON(type_index))), kExprEnd}); // Test struct.new and struct.get. const byte kGet2 = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_STRUCT_GET( type_index, 1, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_I32V(64), WASM_RTT_CANON(type_index))), kExprEnd}); // Test struct.new, returning struct reference. const byte kGetStruct = tester.DefineFunction( &sig_q_v, {}, {WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_I32V(64), WASM_RTT_CANON(type_index)), kExprEnd}); // Test struct.new, returning reference to an empty struct. const byte kGetEmptyStruct = tester.DefineFunction( &sig_qe_v, {}, {WASM_STRUCT_NEW_WITH_RTT(empty_struct_index, WASM_RTT_CANON(empty_struct_index)), kExprEnd}); // Test struct.set, struct refs types in locals. const byte j_local_index = 0; const byte j_field_index = 0; const byte kSet = tester.DefineFunction( tester.sigs.i_v(), {kOptRefType}, {WASM_LOCAL_SET( j_local_index, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_I32V(64), WASM_RTT_CANON(type_index))), WASM_STRUCT_SET(type_index, j_field_index, WASM_LOCAL_GET(j_local_index), WASM_I32V(-99)), WASM_STRUCT_GET(type_index, j_field_index, WASM_LOCAL_GET(j_local_index)), kExprEnd}); tester.CompileModule(); tester.CheckResult(kGet1, 42); tester.CheckResult(kGet2, 64); CHECK(tester.GetResultObject(kGetStruct).ToHandleChecked()->IsWasmStruct()); CHECK(tester.GetResultObject(kGetEmptyStruct) .ToHandleChecked() ->IsWasmStruct()); tester.CheckResult(kSet, -99); } // Test struct.set, ref.as_non_null, // struct refs types in globals and if-results. WASM_COMPILED_EXEC_TEST(WasmRefAsNonNull) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineStruct({F(kWasmI32, true), F(kWasmI32, true)}); ValueType kRefTypes[] = {ref(type_index)}; ValueType kOptRefType = optref(type_index); FunctionSig sig_q_v(1, 0, kRefTypes); const byte global_index = tester.AddGlobal(kOptRefType, true, WasmInitExpr::RefNullConst( static_cast(type_index))); const byte field_index = 0; const byte kFunc = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_GLOBAL_SET( global_index, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(55), WASM_I32V(66), WASM_RTT_CANON(type_index))), WASM_STRUCT_GET( type_index, field_index, WASM_REF_AS_NON_NULL(WASM_IF_ELSE_R(kOptRefType, WASM_I32V(1), WASM_GLOBAL_GET(global_index), WASM_REF_NULL(type_index)))), kExprEnd}); tester.CompileModule(); tester.CheckResult(kFunc, 55); } WASM_COMPILED_EXEC_TEST(WasmBrOnNull) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineStruct({F(kWasmI32, true), F(kWasmI32, true)}); ValueType kRefTypes[] = {ref(type_index)}; ValueType kOptRefType = optref(type_index); FunctionSig sig_q_v(1, 0, kRefTypes); const byte local_index = 0; const byte kTaken = tester.DefineFunction( tester.sigs.i_v(), {kOptRefType}, {WASM_BLOCK_I(WASM_I32V(42), // Branch will be taken. // 42 left on stack outside the block (not 52). WASM_BR_ON_NULL(0, WASM_LOCAL_GET(local_index)), WASM_I32V(52), WASM_BR(0)), kExprEnd}); const byte field_index = 0; const byte kNotTaken = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_BLOCK_I( WASM_I32V(42), WASM_STRUCT_GET( type_index, field_index, // Branch will not be taken. // 52 left on stack outside the block (not 42). WASM_BR_ON_NULL(0, WASM_STRUCT_NEW_WITH_RTT( type_index, WASM_I32V(52), WASM_I32V(62), WASM_RTT_CANON(type_index)))), WASM_BR(0)), kExprEnd}); tester.CompileModule(); tester.CheckResult(kTaken, 42); tester.CheckResult(kNotTaken, 52); } WASM_COMPILED_EXEC_TEST(WasmBrOnNonNull) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineStruct({F(kWasmI32, true), F(kWasmI32, true)}); ValueType kRefType = ref(type_index); ValueType kOptRefType = optref(type_index); FunctionSig sig_q_v(1, 0, &kRefType); const byte field_index = 0; const byte kTaken = tester.DefineFunction( tester.sigs.i_v(), {kOptRefType, kOptRefType}, {WASM_LOCAL_SET( 0, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(52), WASM_I32V(62), WASM_RTT_CANON(type_index))), WASM_LOCAL_SET( 1, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(11), WASM_I32V(22), WASM_RTT_CANON(type_index))), WASM_STRUCT_GET(type_index, field_index, WASM_BLOCK_R(ref(type_index), // Branch will be taken, and the block will // return struct(52, 62). WASM_BR_ON_NON_NULL(0, WASM_LOCAL_GET(0)), WASM_REF_AS_NON_NULL(WASM_LOCAL_GET(1)))), kExprEnd}); const byte kNotTaken = tester.DefineFunction( tester.sigs.i_v(), {kOptRefType, kOptRefType}, {WASM_LOCAL_SET(0, WASM_REF_NULL(type_index)), WASM_LOCAL_SET( 1, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(11), WASM_I32V(22), WASM_RTT_CANON(type_index))), WASM_STRUCT_GET(type_index, field_index, WASM_BLOCK_R(ref(type_index), // Branch will not be taken, and the block // will return struct(11, 22). WASM_BR_ON_NON_NULL(0, WASM_LOCAL_GET(0)), WASM_REF_AS_NON_NULL(WASM_LOCAL_GET(1)))), kExprEnd}); tester.CompileModule(); tester.CheckResult(kTaken, 52); tester.CheckResult(kNotTaken, 11); } WASM_COMPILED_EXEC_TEST(BrOnCast) { WasmGCTester tester(execution_tier); ValueType kDataRefNull = ValueType::Ref(HeapType::kData, kNullable); const byte type_index = tester.DefineStruct({F(kWasmI32, true)}); const byte other_type_index = tester.DefineStruct({F(kWasmF32, true)}); const byte rtt_index = tester.AddGlobal(ValueType::Rtt(type_index, 0), false, WasmInitExpr::RttCanon( static_cast(type_index))); const byte kTestStruct = tester.DefineFunction( tester.sigs.i_v(), {kWasmI32, kDataRefNull}, {WASM_BLOCK_R(ValueType::Ref(type_index, kNullable), WASM_LOCAL_SET(0, WASM_I32V(111)), // Pipe a struct through a local so it's statically typed // as dataref. WASM_LOCAL_SET(1, WASM_STRUCT_NEW_WITH_RTT( other_type_index, WASM_F32(1.0), WASM_RTT_CANON(other_type_index))), WASM_LOCAL_GET(1), // The type check fails, so this branch isn't taken. WASM_BR_ON_CAST(0, WASM_GLOBAL_GET(rtt_index)), WASM_DROP, WASM_LOCAL_SET(0, WASM_I32V(221)), // (Final result) - 1 WASM_LOCAL_SET(1, WASM_STRUCT_NEW_WITH_RTT( type_index, WASM_I32V(1), WASM_GLOBAL_GET(rtt_index))), WASM_LOCAL_GET(1), // This branch is taken. WASM_BR_ON_CAST(0, WASM_GLOBAL_GET(rtt_index)), WASM_GLOBAL_GET(rtt_index), WASM_GC_OP(kExprRefCast), // Not executed due to the branch. WASM_LOCAL_SET(0, WASM_I32V(333))), WASM_GC_OP(kExprStructGet), type_index, 0, WASM_LOCAL_GET(0), kExprI32Add, kExprEnd}); const byte kTestNull = tester.DefineFunction( tester.sigs.i_v(), {kWasmI32, kDataRefNull}, {WASM_BLOCK_R(ValueType::Ref(type_index, kNullable), WASM_LOCAL_SET(0, WASM_I32V(111)), WASM_LOCAL_GET(1), // Put a nullref onto the value stack. // Not taken for nullref. WASM_BR_ON_CAST(0, WASM_GLOBAL_GET(rtt_index)), WASM_RTT_CANON(type_index), WASM_GC_OP(kExprRefCast), WASM_LOCAL_SET(0, WASM_I32V(222))), // Final result. WASM_DROP, WASM_LOCAL_GET(0), kExprEnd}); const byte kTypedAfterBranch = tester.DefineFunction( tester.sigs.i_v(), {kWasmI32, kDataRefNull}, {WASM_LOCAL_SET(1, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_GLOBAL_GET(rtt_index))), WASM_BLOCK_I( // The inner block should take the early branch with a struct // on the stack. WASM_BLOCK_R(ValueType::Ref(type_index, kNonNullable), WASM_LOCAL_GET(1), WASM_BR_ON_CAST(0, WASM_GLOBAL_GET(rtt_index)), // Returning 123 is the unreachable failure case. WASM_I32V(123), WASM_BR(1)), // The outer block catches the struct left behind by the inner block // and reads its field. WASM_GC_OP(kExprStructGet), type_index, 0), kExprEnd}); tester.CompileModule(); tester.CheckResult(kTestStruct, 222); tester.CheckResult(kTestNull, 222); tester.CheckResult(kTypedAfterBranch, 42); } WASM_COMPILED_EXEC_TEST(BrOnCastFail) { WasmGCTester tester(execution_tier); ValueType kDataRefNull = ValueType::Ref(HeapType::kData, kNullable); const byte type0 = tester.DefineStruct({F(kWasmI32, true)}); const byte type1 = tester.DefineStruct({F(kWasmI64, true), F(kWasmI32, true)}); const int field0_value = 5; const int field1_value = 25; const int null_value = 45; // local_0 = value; // if (!(local_0 instanceof type0)) goto block1; // return static_cast(local_0).field_0; // block1: // if (local_0 == nullptr) goto block2; // return static_cast(local_0).field_1; // block2: // return null_value; #define FUNCTION_BODY(value) \ WASM_LOCAL_SET(0, WASM_SEQ(value)), \ WASM_BLOCK( \ WASM_BLOCK_R(kDataRefNull, WASM_LOCAL_GET(0), \ WASM_BR_ON_CAST_FAIL(0, WASM_RTT_CANON(type0)), \ WASM_GC_OP(kExprStructGet), type0, 0, kExprReturn), \ kExprBrOnNull, 0, WASM_RTT_CANON(type1), WASM_GC_OP(kExprRefCast), \ WASM_GC_OP(kExprStructGet), type1, 1, kExprReturn), \ WASM_I32V(null_value), kExprEnd const byte kBranchTaken = tester.DefineFunction(tester.sigs.i_v(), {kDataRefNull}, {FUNCTION_BODY(WASM_STRUCT_NEW_WITH_RTT( type1, WASM_I64V(10), WASM_I32V(field1_value), WASM_RTT_CANON(type1)))}); const byte kBranchNotTaken = tester.DefineFunction( tester.sigs.i_v(), {kDataRefNull}, {FUNCTION_BODY(WASM_STRUCT_NEW_WITH_RTT(type0, WASM_I32V(field0_value), WASM_RTT_CANON(type0)))}); const byte kNull = tester.DefineFunction( tester.sigs.i_v(), {kDataRefNull}, {FUNCTION_BODY(WASM_REF_NULL(type0))}); const byte kUnrelatedTypes = tester.DefineFunction( tester.sigs.i_v(), {ValueType::Ref(type1, kNullable)}, {FUNCTION_BODY(WASM_STRUCT_NEW_WITH_RTT(type1, WASM_I64V(10), WASM_I32V(field1_value), WASM_RTT_CANON(type1)))}); #undef FUNCTION_BODY tester.CompileModule(); tester.CheckResult(kBranchTaken, field1_value); tester.CheckResult(kBranchNotTaken, field0_value); tester.CheckResult(kNull, null_value); tester.CheckResult(kUnrelatedTypes, field1_value); } WASM_COMPILED_EXEC_TEST(WasmRefEq) { WasmGCTester tester(execution_tier); byte type_index = tester.DefineStruct({F(kWasmI32, true), F(kWasmI32, true)}); ValueType kRefTypes[] = {ref(type_index)}; ValueType kOptRefType = optref(type_index); FunctionSig sig_q_v(1, 0, kRefTypes); byte local_index = 0; const byte kFunc = tester.DefineFunction( tester.sigs.i_v(), {kOptRefType}, {WASM_LOCAL_SET(local_index, WASM_STRUCT_NEW_WITH_RTT( type_index, WASM_I32V(55), WASM_I32V(66), WASM_RTT_CANON(type_index))), WASM_I32_ADD( WASM_I32_SHL( WASM_REF_EQ( // true WASM_LOCAL_GET(local_index), WASM_LOCAL_GET(local_index)), WASM_I32V(0)), WASM_I32_ADD( WASM_I32_SHL(WASM_REF_EQ( // false WASM_LOCAL_GET(local_index), WASM_STRUCT_NEW_WITH_RTT( type_index, WASM_I32V(55), WASM_I32V(66), WASM_RTT_CANON(type_index))), WASM_I32V(1)), WASM_I32_ADD(WASM_I32_SHL( // false WASM_REF_EQ(WASM_LOCAL_GET(local_index), WASM_REF_NULL(type_index)), WASM_I32V(2)), WASM_I32_SHL(WASM_REF_EQ( // true WASM_REF_NULL(type_index), WASM_REF_NULL(type_index)), WASM_I32V(3))))), kExprEnd}); tester.CompileModule(); tester.CheckResult(kFunc, 0b1001); } WASM_COMPILED_EXEC_TEST(WasmPackedStructU) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineStruct( {F(kWasmI8, true), F(kWasmI16, true), F(kWasmI32, true)}); ValueType struct_type = optref(type_index); const byte local_index = 0; int32_t expected_output_0 = 0x1234; int32_t expected_output_1 = -1; const byte kF0 = tester.DefineFunction( tester.sigs.i_v(), {struct_type}, {WASM_LOCAL_SET(local_index, WASM_STRUCT_NEW_WITH_RTT( type_index, WASM_I32V(expected_output_0), WASM_I32V(expected_output_1), WASM_I32V(0x12345678), WASM_RTT_CANON(type_index))), WASM_STRUCT_GET_U(type_index, 0, WASM_LOCAL_GET(local_index)), kExprEnd}); const byte kF1 = tester.DefineFunction( tester.sigs.i_v(), {struct_type}, {WASM_LOCAL_SET(local_index, WASM_STRUCT_NEW_WITH_RTT( type_index, WASM_I32V(expected_output_0), WASM_I32V(expected_output_1), WASM_I32V(0x12345678), WASM_RTT_CANON(type_index))), WASM_STRUCT_GET_U(type_index, 1, WASM_LOCAL_GET(local_index)), kExprEnd}); tester.CompileModule(); tester.CheckResult(kF0, static_cast(expected_output_0)); tester.CheckResult(kF1, static_cast(expected_output_1)); } WASM_COMPILED_EXEC_TEST(WasmPackedStructS) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineStruct( {F(kWasmI8, true), F(kWasmI16, true), F(kWasmI32, true)}); ValueType struct_type = optref(type_index); const byte local_index = 0; int32_t expected_output_0 = 0x80; int32_t expected_output_1 = 42; const byte kF0 = tester.DefineFunction( tester.sigs.i_v(), {struct_type}, {WASM_LOCAL_SET( local_index, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(expected_output_0), WASM_I32V(expected_output_1), WASM_I32V(0), WASM_RTT_CANON(type_index))), WASM_STRUCT_GET_S(type_index, 0, WASM_LOCAL_GET(local_index)), kExprEnd}); const byte kF1 = tester.DefineFunction( tester.sigs.i_v(), {struct_type}, {WASM_LOCAL_SET( local_index, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(0x80), WASM_I32V(expected_output_1), WASM_I32V(0), WASM_RTT_CANON(type_index))), WASM_STRUCT_GET_S(type_index, 1, WASM_LOCAL_GET(local_index)), kExprEnd}); tester.CompileModule(); tester.CheckResult(kF0, static_cast(expected_output_0)); tester.CheckResult(kF1, static_cast(expected_output_1)); } TEST(WasmLetInstruction) { WasmGCTester tester; const byte type_index = tester.DefineStruct({F(kWasmI32, true), F(kWasmI32, true)}); const byte let_local_index = 0; const byte let_field_index = 0; const byte kLetTest1 = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_LET_1_I( WASM_SEQ(kRefCode, type_index), WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_I32V(52), WASM_RTT_CANON(type_index)), WASM_STRUCT_GET(type_index, let_field_index, WASM_LOCAL_GET(let_local_index))), kExprEnd}); const byte let_2_field_index = 0; const byte kLetTest2 = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_LET_2_I( kI32Code, WASM_I32_ADD(WASM_I32V(42), WASM_I32V(-32)), WASM_SEQ(kRefCode, type_index), WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_I32V(52), WASM_RTT_CANON(type_index)), WASM_I32_MUL(WASM_STRUCT_GET(type_index, let_2_field_index, WASM_LOCAL_GET(1)), WASM_LOCAL_GET(0))), kExprEnd}); const byte kLetTestLocals = tester.DefineFunction( tester.sigs.i_i(), {kWasmI32}, {WASM_LOCAL_SET(1, WASM_I32V(100)), WASM_LET_2_I( kI32Code, WASM_I32V(1), kI32Code, WASM_I32V(10), WASM_I32_SUB(WASM_I32_ADD(WASM_LOCAL_GET(0), // 1st let-local WASM_LOCAL_GET(2)), // Parameter WASM_I32_ADD(WASM_LOCAL_GET(1), // 2nd let-local WASM_LOCAL_GET(3)))), // Function local kExprEnd}); // Result: (1 + 1000) - (10 + 100) = 891 const byte let_erase_local_index = 0; const byte kLetTestErase = tester.DefineFunction( tester.sigs.i_v(), {kWasmI32}, {WASM_LOCAL_SET(let_erase_local_index, WASM_I32V(0)), WASM_LET_1_V(kI32Code, WASM_I32V(1), WASM_NOP), WASM_LOCAL_GET(let_erase_local_index), kExprEnd}); // The result should be 0 and not 1, as local_get(0) refers to the original // local. const byte kLetInLoop = tester.DefineFunction( tester.sigs.i_i(), {}, {WASM_LOOP(WASM_LET_1_V( kI32Code, WASM_I32V(10), // -- WASM_LOCAL_SET(1, WASM_I32_SUB(WASM_LOCAL_GET(1), WASM_I32V(10))), WASM_BR_IF(1, WASM_I32_GES(WASM_LOCAL_GET(1), WASM_LOCAL_GET(0))))), WASM_LOCAL_GET(0), WASM_END}); const byte kLetInBlock = tester.DefineFunction( tester.sigs.i_i(), {}, {WASM_BLOCK(WASM_LET_1_V( kI32Code, WASM_I32V(10), // -- WASM_BR_IF(1, WASM_I32_GES(WASM_LOCAL_GET(1), WASM_LOCAL_GET(0))), WASM_LOCAL_SET(1, WASM_I32V(30)))), WASM_LOCAL_GET(0), WASM_END}); tester.CompileModule(); tester.CheckResult(kLetTest1, 42); tester.CheckResult(kLetTest2, 420); tester.CheckResult(kLetTestLocals, 891, 1000); tester.CheckResult(kLetTestErase, 0); tester.CheckResult(kLetInLoop, 2, 52); tester.CheckResult(kLetInLoop, -11, -1); tester.CheckResult(kLetInBlock, 15, 15); tester.CheckResult(kLetInBlock, 30, 5); } WASM_COMPILED_EXEC_TEST(WasmBasicArray) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineArray(wasm::kWasmI32, true); const byte fp_type_index = tester.DefineArray(wasm::kWasmF64, true); ValueType kRefTypes[] = {ref(type_index)}; FunctionSig sig_q_v(1, 0, kRefTypes); ValueType kOptRefType = optref(type_index); // f: a = [12, 12, 12]; a[1] = 42; return a[arg0] const byte local_index = 1; const byte kGetElem = tester.DefineFunction( tester.sigs.i_i(), {kOptRefType}, {WASM_LOCAL_SET(local_index, WASM_ARRAY_NEW_WITH_RTT( type_index, WASM_I32V(12), WASM_I32V(3), WASM_RTT_CANON(type_index))), WASM_ARRAY_SET(type_index, WASM_LOCAL_GET(local_index), WASM_I32V(1), WASM_I32V(42)), WASM_ARRAY_GET(type_index, WASM_LOCAL_GET(local_index), WASM_LOCAL_GET(0)), kExprEnd}); // Reads and returns an array's length. const byte kGetLength = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_ARRAY_LEN(type_index, WASM_ARRAY_NEW_WITH_RTT( type_index, WASM_I32V(0), WASM_I32V(42), WASM_RTT_CANON(type_index))), kExprEnd}); // Create an array of length 2, initialized to [42, 42]. const byte kAllocate = tester.DefineFunction( &sig_q_v, {}, {WASM_ARRAY_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_I32V(2), WASM_RTT_CANON(type_index)), kExprEnd}); const uint32_t kLongLength = 1u << 16; const byte kAllocateLarge = tester.DefineFunction( &sig_q_v, {}, {WASM_ARRAY_NEW_DEFAULT(type_index, WASM_I32V(kLongLength), WASM_RTT_CANON(type_index)), kExprEnd}); const uint32_t kTooLong = kV8MaxWasmArrayLength + 1; const byte kAllocateTooLarge = tester.DefineFunction( &sig_q_v, {}, {WASM_ARRAY_NEW_DEFAULT(type_index, WASM_I32V(kTooLong), WASM_RTT_CANON(type_index)), kExprEnd}); // Tests that fp arrays work properly. // f: a = [10.0, 10.0, 10.0]; a[1] = 42.42; return static_cast(a[1]); double result_value = 42.42; const byte kTestFpArray = tester.DefineFunction( tester.sigs.i_v(), {optref(fp_type_index)}, {WASM_LOCAL_SET(0, WASM_ARRAY_NEW_WITH_RTT( fp_type_index, WASM_F64(10.0), WASM_I32V(3), WASM_RTT_CANON(fp_type_index))), WASM_ARRAY_SET(fp_type_index, WASM_LOCAL_GET(0), WASM_I32V(1), WASM_F64(result_value)), WASM_I32_SCONVERT_F64( WASM_ARRAY_GET(fp_type_index, WASM_LOCAL_GET(0), WASM_I32V(1))), kExprEnd}); tester.CompileModule(); tester.CheckResult(kGetElem, 12, 0); tester.CheckResult(kGetElem, 42, 1); tester.CheckResult(kGetElem, 12, 2); tester.CheckHasThrown(kGetElem, 3); tester.CheckHasThrown(kGetElem, -1); tester.CheckResult(kGetLength, 42); tester.CheckResult(kTestFpArray, static_cast(result_value)); MaybeHandle h_result = tester.GetResultObject(kAllocate); CHECK(h_result.ToHandleChecked()->IsWasmArray()); #if OBJECT_PRINT h_result.ToHandleChecked()->Print(); #endif MaybeHandle maybe_large_result = tester.GetResultObject(kAllocateLarge); Handle large_result = maybe_large_result.ToHandleChecked(); CHECK(large_result->IsWasmArray()); CHECK(Handle::cast(large_result)->Size() > kMaxRegularHeapObjectSize); tester.CheckHasThrown(kAllocateTooLarge, "requested new array is too large"); } WASM_COMPILED_EXEC_TEST(WasmPackedArrayU) { WasmGCTester tester(execution_tier); const byte array_index = tester.DefineArray(kWasmI8, true); ValueType array_type = optref(array_index); const byte param_index = 0; const byte local_index = 1; int32_t expected_output_3 = 258; const byte kF = tester.DefineFunction( tester.sigs.i_i(), {array_type}, {WASM_LOCAL_SET(local_index, WASM_ARRAY_NEW_WITH_RTT( array_index, WASM_I32V(0), WASM_I32V(4), WASM_RTT_CANON(array_index))), WASM_ARRAY_SET(array_index, WASM_LOCAL_GET(local_index), WASM_I32V(0), WASM_I32V(1)), WASM_ARRAY_SET(array_index, WASM_LOCAL_GET(local_index), WASM_I32V(1), WASM_I32V(10)), WASM_ARRAY_SET(array_index, WASM_LOCAL_GET(local_index), WASM_I32V(2), WASM_I32V(200)), WASM_ARRAY_SET(array_index, WASM_LOCAL_GET(local_index), WASM_I32V(3), WASM_I32V(expected_output_3)), WASM_ARRAY_GET_U(array_index, WASM_LOCAL_GET(local_index), WASM_LOCAL_GET(param_index)), kExprEnd}); tester.CompileModule(); tester.CheckResult(kF, 1, 0); tester.CheckResult(kF, 10, 1); tester.CheckResult(kF, 200, 2); // Only the 2 lsb's of 258 should be stored in the array. tester.CheckResult(kF, static_cast(expected_output_3), 3); } WASM_COMPILED_EXEC_TEST(WasmPackedArrayS) { WasmGCTester tester(execution_tier); const byte array_index = tester.DefineArray(kWasmI16, true); ValueType array_type = optref(array_index); int32_t array_elements[] = {0x12345678, 10, 0xFEDC, 0xFF1234}; const byte param_index = 0; const byte local_index = 1; const byte kF = tester.DefineFunction( tester.sigs.i_i(), {array_type}, {WASM_LOCAL_SET( local_index, WASM_ARRAY_NEW_WITH_RTT(array_index, WASM_I32V(array_elements[0]), WASM_I32V(4), WASM_RTT_CANON(array_index))), WASM_ARRAY_SET(array_index, WASM_LOCAL_GET(local_index), WASM_I32V(1), WASM_I32V(array_elements[1])), WASM_ARRAY_SET(array_index, WASM_LOCAL_GET(local_index), WASM_I32V(2), WASM_I32V(array_elements[2])), WASM_ARRAY_SET(array_index, WASM_LOCAL_GET(local_index), WASM_I32V(3), WASM_I32V(array_elements[3])), WASM_ARRAY_GET_S(array_index, WASM_LOCAL_GET(local_index), WASM_LOCAL_GET(param_index)), kExprEnd}); tester.CompileModule(); // Exactly the 2 lsb's should be stored by array.new. tester.CheckResult(kF, static_cast(array_elements[0]), 0); tester.CheckResult(kF, static_cast(array_elements[1]), 1); // Sign should be extended. tester.CheckResult(kF, static_cast(array_elements[2]), 2); // Exactly the 2 lsb's should be stored by array.set. tester.CheckResult(kF, static_cast(array_elements[3]), 3); } WASM_COMPILED_EXEC_TEST(WasmArrayCopy) { FLAG_SCOPE(experimental_wasm_gc_experiments); WasmGCTester tester(execution_tier); const byte array32_index = tester.DefineArray(kWasmI32, true); const byte array16_index = tester.DefineArray(kWasmI16, true); const byte arrayref_index = tester.DefineArray(optref(array32_index), true); // Copies i32 ranges: local1[0..3] to local2[6..9]. const byte kCopyI32 = tester.DefineFunction( tester.sigs.i_i(), {optref(array32_index), optref(array32_index)}, {WASM_LOCAL_SET(1, WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(10), WASM_RTT_CANON(array32_index))), WASM_ARRAY_SET(array32_index, WASM_LOCAL_GET(1), WASM_I32V(0), WASM_I32V(0)), WASM_ARRAY_SET(array32_index, WASM_LOCAL_GET(1), WASM_I32V(1), WASM_I32V(1)), WASM_ARRAY_SET(array32_index, WASM_LOCAL_GET(1), WASM_I32V(2), WASM_I32V(2)), WASM_ARRAY_SET(array32_index, WASM_LOCAL_GET(1), WASM_I32V(3), WASM_I32V(3)), WASM_LOCAL_SET(2, WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(10), WASM_RTT_CANON(array32_index))), WASM_ARRAY_COPY(array32_index, array32_index, WASM_LOCAL_GET(2), WASM_I32V(6), WASM_LOCAL_GET(1), WASM_I32V(0), WASM_I32V(4)), WASM_ARRAY_GET(array32_index, WASM_LOCAL_GET(2), WASM_LOCAL_GET(0)), kExprEnd}); // Copies i16 ranges: local1[0..3] to local2[6..9]. const byte kCopyI16 = tester.DefineFunction( tester.sigs.i_i(), {optref(array16_index), optref(array16_index)}, {WASM_LOCAL_SET(1, WASM_ARRAY_NEW_DEFAULT(array16_index, WASM_I32V(10), WASM_RTT_CANON(array16_index))), WASM_ARRAY_SET(array16_index, WASM_LOCAL_GET(1), WASM_I32V(0), WASM_I32V(0)), WASM_ARRAY_SET(array16_index, WASM_LOCAL_GET(1), WASM_I32V(1), WASM_I32V(1)), WASM_ARRAY_SET(array16_index, WASM_LOCAL_GET(1), WASM_I32V(2), WASM_I32V(2)), WASM_ARRAY_SET(array16_index, WASM_LOCAL_GET(1), WASM_I32V(3), WASM_I32V(3)), WASM_LOCAL_SET(2, WASM_ARRAY_NEW_DEFAULT(array16_index, WASM_I32V(10), WASM_RTT_CANON(array16_index))), WASM_ARRAY_COPY(array16_index, array16_index, WASM_LOCAL_GET(2), WASM_I32V(6), WASM_LOCAL_GET(1), WASM_I32V(0), WASM_I32V(4)), WASM_ARRAY_GET_S(array16_index, WASM_LOCAL_GET(2), WASM_LOCAL_GET(0)), kExprEnd}); // Copies reference ranges: local1[0..3] to local2[6..9]. const byte kCopyRef = tester.DefineFunction( FunctionSig::Build(tester.zone(), {optref(array32_index)}, {kWasmI32}), {optref(arrayref_index), optref(arrayref_index)}, {WASM_LOCAL_SET(1, WASM_ARRAY_NEW_DEFAULT(arrayref_index, WASM_I32V(10), WASM_RTT_CANON(arrayref_index))), WASM_ARRAY_SET(arrayref_index, WASM_LOCAL_GET(1), WASM_I32V(0), WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(6), WASM_RTT_CANON(array32_index))), WASM_ARRAY_SET(arrayref_index, WASM_LOCAL_GET(1), WASM_I32V(1), WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(7), WASM_RTT_CANON(array32_index))), WASM_ARRAY_SET(arrayref_index, WASM_LOCAL_GET(1), WASM_I32V(2), WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(8), WASM_RTT_CANON(array32_index))), WASM_ARRAY_SET(arrayref_index, WASM_LOCAL_GET(1), WASM_I32V(3), WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(9), WASM_RTT_CANON(array32_index))), WASM_LOCAL_SET(2, WASM_ARRAY_NEW_DEFAULT(arrayref_index, WASM_I32V(10), WASM_RTT_CANON(arrayref_index))), WASM_ARRAY_COPY(arrayref_index, arrayref_index, WASM_LOCAL_GET(2), WASM_I32V(6), WASM_LOCAL_GET(1), WASM_I32V(0), WASM_I32V(4)), WASM_ARRAY_GET(arrayref_index, WASM_LOCAL_GET(2), WASM_LOCAL_GET(0)), kExprEnd}); // Copies overlapping reference ranges: local1[0..3] to local1[2..5]. const byte kCopyRefOverlapping = tester.DefineFunction( FunctionSig::Build(tester.zone(), {optref(array32_index)}, {kWasmI32}), {optref(arrayref_index)}, {WASM_LOCAL_SET(1, WASM_ARRAY_NEW_DEFAULT(arrayref_index, WASM_I32V(10), WASM_RTT_CANON(arrayref_index))), WASM_ARRAY_SET(arrayref_index, WASM_LOCAL_GET(1), WASM_I32V(0), WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(2), WASM_RTT_CANON(array32_index))), WASM_ARRAY_SET(arrayref_index, WASM_LOCAL_GET(1), WASM_I32V(1), WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(3), WASM_RTT_CANON(array32_index))), WASM_ARRAY_SET(arrayref_index, WASM_LOCAL_GET(1), WASM_I32V(2), WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(4), WASM_RTT_CANON(array32_index))), WASM_ARRAY_SET(arrayref_index, WASM_LOCAL_GET(1), WASM_I32V(3), WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(5), WASM_RTT_CANON(array32_index))), WASM_ARRAY_COPY(arrayref_index, arrayref_index, WASM_LOCAL_GET(1), WASM_I32V(2), WASM_LOCAL_GET(1), WASM_I32V(0), WASM_I32V(4)), WASM_ARRAY_GET(arrayref_index, WASM_LOCAL_GET(1), WASM_LOCAL_GET(0)), kExprEnd}); const byte kOobSource = tester.DefineFunction( tester.sigs.v_v(), {optref(array32_index), optref(array32_index)}, {WASM_LOCAL_SET(0, WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(10), WASM_RTT_CANON(array32_index))), WASM_LOCAL_SET(1, WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(10), WASM_RTT_CANON(array32_index))), WASM_ARRAY_COPY(array32_index, array32_index, WASM_LOCAL_GET(1), WASM_I32V(6), WASM_LOCAL_GET(0), WASM_I32V(8), WASM_I32V(4)), kExprEnd}); const byte kOobDestination = tester.DefineFunction( tester.sigs.v_v(), {optref(array32_index), optref(array32_index)}, {WASM_LOCAL_SET(0, WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(10), WASM_RTT_CANON(array32_index))), WASM_LOCAL_SET(1, WASM_ARRAY_NEW_DEFAULT(array32_index, WASM_I32V(10), WASM_RTT_CANON(array32_index))), WASM_ARRAY_COPY(array32_index, array32_index, WASM_LOCAL_GET(1), WASM_I32V(6), WASM_LOCAL_GET(0), WASM_I32V(3), WASM_I32V(5)), kExprEnd}); tester.CompileModule(); tester.CheckResult(kCopyI32, 0, 5); tester.CheckResult(kCopyI32, 0, 6); tester.CheckResult(kCopyI32, 1, 7); tester.CheckResult(kCopyI32, 2, 8); tester.CheckResult(kCopyI32, 3, 9); tester.CheckResult(kCopyI16, 0, 5); tester.CheckResult(kCopyI16, 0, 6); tester.CheckResult(kCopyI16, 1, 7); tester.CheckResult(kCopyI16, 2, 8); tester.CheckResult(kCopyI16, 3, 9); { Handle result5 = tester.GetResultObject(kCopyRef, 5).ToHandleChecked(); CHECK(result5->IsNull()); for (int i = 6; i <= 9; i++) { Handle res = tester.GetResultObject(kCopyRef, i).ToHandleChecked(); CHECK(res->IsWasmArray()); CHECK_EQ(Handle::cast(res)->length(), static_cast(i)); } } CHECK(tester.GetResultObject(kCopyRefOverlapping, 6) .ToHandleChecked() ->IsNull()); Handle res0 = tester.GetResultObject(kCopyRefOverlapping, 0).ToHandleChecked(); CHECK(res0->IsWasmArray()); CHECK_EQ(Handle::cast(res0)->length(), static_cast(2)); for (int i = 2; i <= 5; i++) { Handle res = tester.GetResultObject(kCopyRefOverlapping, i).ToHandleChecked(); CHECK(res->IsWasmArray()); CHECK_EQ(Handle::cast(res)->length(), static_cast(i)); } tester.CheckHasThrown(kOobSource); tester.CheckHasThrown(kOobDestination); } WASM_COMPILED_EXEC_TEST(NewDefault) { WasmGCTester tester(execution_tier); const byte struct_type = tester.DefineStruct( {F(wasm::kWasmI32, true), F(wasm::kWasmF64, true), F(optref(0), true)}); const byte array_type = tester.DefineArray(wasm::kWasmI32, true); // Returns: struct[0] + f64_to_i32(struct[1]) + (struct[2].is_null ^ 1) == 0. const byte allocate_struct = tester.DefineFunction( tester.sigs.i_v(), {optref(struct_type)}, {WASM_LOCAL_SET(0, WASM_STRUCT_NEW_DEFAULT(struct_type, WASM_RTT_CANON(struct_type))), WASM_I32_ADD( WASM_I32_ADD(WASM_STRUCT_GET(struct_type, 0, WASM_LOCAL_GET(0)), WASM_I32_SCONVERT_F64(WASM_STRUCT_GET( struct_type, 1, WASM_LOCAL_GET(0)))), WASM_I32_XOR(WASM_REF_IS_NULL( WASM_STRUCT_GET(struct_type, 2, WASM_LOCAL_GET(0))), WASM_I32V(1))), kExprEnd}); const byte allocate_array = tester.DefineFunction( tester.sigs.i_v(), {optref(array_type)}, {WASM_LOCAL_SET(0, WASM_ARRAY_NEW_DEFAULT(array_type, WASM_I32V(2), WASM_RTT_CANON(array_type))), WASM_I32_ADD( WASM_ARRAY_GET(array_type, WASM_LOCAL_GET(0), WASM_I32V(0)), WASM_ARRAY_GET(array_type, WASM_LOCAL_GET(0), WASM_I32V(1))), kExprEnd}); tester.CompileModule(); tester.CheckResult(allocate_struct, 0); tester.CheckResult(allocate_array, 0); } WASM_COMPILED_EXEC_TEST(BasicRtt) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineStruct({F(wasm::kWasmI32, true)}); const byte subtype_index = tester.DefineStruct({F(wasm::kWasmI32, true), F(wasm::kWasmI32, true)}); ValueType kRttTypes[] = {ValueType::Rtt(type_index, 0)}; FunctionSig sig_t_v(1, 0, kRttTypes); ValueType kRttSubtypes[] = {ValueType::Rtt(subtype_index, 1)}; FunctionSig sig_t2_v(1, 0, kRttSubtypes); ValueType kRttTypesDeeper[] = {ValueType::Rtt(type_index, 1)}; FunctionSig sig_t3_v(1, 0, kRttTypesDeeper); ValueType kRefTypes[] = {ref(type_index)}; FunctionSig sig_q_v(1, 0, kRefTypes); const byte kRttCanon = tester.DefineFunction( &sig_t_v, {}, {WASM_RTT_CANON(type_index), kExprEnd}); const byte kRttSub = tester.DefineFunction( &sig_t2_v, {}, {WASM_RTT_SUB(subtype_index, WASM_RTT_CANON(type_index)), kExprEnd}); const byte kStructWithRtt = tester.DefineFunction( &sig_q_v, {}, {WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_RTT_CANON(type_index)), kExprEnd}); const int kFieldIndex = 1; const int kStructIndexCode = 0; // This implements the following function: // var local_struct: type0; // local_struct = new type1 with rtt 'kRttSub()'; // return (ref.test local_struct kRttSub()) + // ((ref.cast local_struct kRttSub())[field0]); // } // The expected return value is 1+42 = 43. const byte kRefCast = tester.DefineFunction( tester.sigs.i_v(), {optref(type_index)}, {WASM_LOCAL_SET( kStructIndexCode, WASM_STRUCT_NEW_WITH_RTT(subtype_index, WASM_I32V(11), WASM_I32V(42), WASM_CALL_FUNCTION0(kRttSub))), WASM_I32_ADD( WASM_REF_TEST(WASM_LOCAL_GET(kStructIndexCode), WASM_CALL_FUNCTION0(kRttSub)), WASM_STRUCT_GET(subtype_index, kFieldIndex, WASM_REF_CAST(WASM_LOCAL_GET(kStructIndexCode), WASM_CALL_FUNCTION0(kRttSub)))), kExprEnd}); tester.CompileModule(); Handle ref_result = tester.GetResultObject(kRttCanon).ToHandleChecked(); CHECK(ref_result->IsMap()); Handle map = Handle::cast(ref_result); CHECK(map->IsWasmStructMap()); CHECK_EQ(reinterpret_cast
( tester.instance()->module()->struct_type(type_index)), map->wasm_type_info().foreign_address()); Handle subref_result = tester.GetResultObject(kRttSub).ToHandleChecked(); CHECK(subref_result->IsMap()); Handle submap = Handle::cast(subref_result); CHECK_EQ(reinterpret_cast
( tester.instance()->module()->struct_type(subtype_index)), submap->wasm_type_info().foreign_address()); Handle subref_result_canonicalized = tester.GetResultObject(kRttSub).ToHandleChecked(); CHECK(subref_result.is_identical_to(subref_result_canonicalized)); Handle s = tester.GetResultObject(kStructWithRtt).ToHandleChecked(); CHECK(s->IsWasmStruct()); CHECK_EQ(Handle::cast(s)->map(), *map); tester.CheckResult(kRefCast, 43); } WASM_COMPILED_EXEC_TEST(RttFreshSub) { WasmGCTester tester(execution_tier); FlagScope flag_gc_experiments(&FLAG_experimental_wasm_gc_experiments, true); const byte kType = tester.DefineStruct({F(wasm::kWasmI32, true)}); HeapType::Representation type_repr = static_cast(kType); const byte kRtt = tester.AddGlobal( ValueType::Rtt(kType, 1), false, WasmInitExpr::RttFreshSub(type_repr, WasmInitExpr::RttCanon(type_repr))); // A struct allocated with a fresh RTT does not match other fresh RTTs // created for the same type. const byte kRefTest = tester.DefineFunction( tester.sigs.i_v(), {optref(kType)}, {WASM_LOCAL_SET(0, WASM_STRUCT_NEW_WITH_RTT( kType, WASM_I32V(11), WASM_RTT_FRESH_SUB(kType, WASM_RTT_CANON(kType)))), WASM_I32_ADD( WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_RTT_FRESH_SUB(kType, WASM_RTT_CANON(kType))), WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_GLOBAL_GET(kRtt))), kExprEnd}); tester.CompileModule(); tester.CheckResult(kRefTest, 0); } WASM_COMPILED_EXEC_TEST(RefTrivialCasts) { WasmGCTester tester(execution_tier); byte type_index = tester.DefineStruct({F(wasm::kWasmI32, true)}); byte subtype_index = tester.DefineStruct({F(wasm::kWasmI32, true), F(wasm::kWasmS128, false)}); ValueType sig_types[] = {kWasmS128, kWasmI32, kWasmF64}; FunctionSig sig(1, 2, sig_types); byte sig_index = tester.DefineSignature(&sig); const byte kRefTestNull = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_TEST(WASM_REF_NULL(type_index), WASM_RTT_CANON(subtype_index)), kExprEnd}); const byte kRefTestUpcast = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_TEST( WASM_STRUCT_NEW_DEFAULT( subtype_index, WASM_RTT_SUB(subtype_index, WASM_RTT_CANON(type_index))), WASM_RTT_CANON(type_index)), kExprEnd}); const byte kRefTestUpcastNull = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_TEST(WASM_REF_NULL(subtype_index), WASM_RTT_CANON(type_index)), kExprEnd}); const byte kRefTestUnrelated = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_TEST( WASM_STRUCT_NEW_DEFAULT( subtype_index, WASM_RTT_SUB(subtype_index, WASM_RTT_CANON(type_index))), WASM_RTT_CANON(sig_index)), kExprEnd}); const byte kRefTestUnrelatedNull = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_TEST(WASM_REF_NULL(subtype_index), WASM_RTT_CANON(sig_index)), kExprEnd}); const byte kRefTestUnrelatedNonNullable = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_TEST( WASM_STRUCT_NEW_DEFAULT(type_index, WASM_RTT_CANON(type_index)), WASM_RTT_CANON(sig_index)), kExprEnd}); const byte kRefCastNull = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_IS_NULL(WASM_REF_CAST(WASM_REF_NULL(type_index), WASM_RTT_CANON(subtype_index))), kExprEnd}); const byte kRefCastUpcast = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_IS_NULL(WASM_REF_CAST( WASM_STRUCT_NEW_DEFAULT( subtype_index, WASM_RTT_SUB(subtype_index, WASM_RTT_CANON(type_index))), WASM_RTT_CANON(type_index))), kExprEnd}); const byte kRefCastUpcastNull = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_IS_NULL(WASM_REF_CAST(WASM_REF_NULL(subtype_index), WASM_RTT_CANON(type_index))), kExprEnd}); const byte kRefCastUnrelated = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_IS_NULL(WASM_REF_CAST( WASM_STRUCT_NEW_DEFAULT( subtype_index, WASM_RTT_SUB(subtype_index, WASM_RTT_CANON(type_index))), WASM_RTT_CANON(sig_index))), kExprEnd}); const byte kRefCastUnrelatedNull = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_IS_NULL(WASM_REF_CAST(WASM_REF_NULL(subtype_index), WASM_RTT_CANON(sig_index))), kExprEnd}); const byte kRefCastUnrelatedNonNullable = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_IS_NULL(WASM_REF_CAST( WASM_STRUCT_NEW_DEFAULT(type_index, WASM_RTT_CANON(type_index)), WASM_RTT_CANON(sig_index))), kExprEnd}); tester.CompileModule(); tester.CheckResult(kRefTestNull, 0); tester.CheckResult(kRefTestUpcast, 1); tester.CheckResult(kRefTestUpcastNull, 0); tester.CheckResult(kRefTestUnrelated, 0); tester.CheckResult(kRefTestUnrelatedNull, 0); tester.CheckResult(kRefTestUnrelatedNonNullable, 0); tester.CheckResult(kRefCastNull, 1); tester.CheckResult(kRefCastUpcast, 0); tester.CheckResult(kRefCastUpcastNull, 1); tester.CheckHasThrown(kRefCastUnrelated); tester.CheckResult(kRefCastUnrelatedNull, 1); tester.CheckHasThrown(kRefCastUnrelatedNonNullable); } WASM_EXEC_TEST(NoDepthRtt) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineStruct({F(wasm::kWasmI32, true)}); const byte subtype_index = tester.DefineStruct({F(wasm::kWasmI32, true), F(wasm::kWasmI32, true)}); const byte empty_struct_index = tester.DefineStruct({}); ValueType kRttTypeNoDepth = ValueType::Rtt(type_index); FunctionSig sig_t1_v_nd(1, 0, &kRttTypeNoDepth); ValueType kRttSubtypeNoDepth = ValueType::Rtt(subtype_index); FunctionSig sig_t2_v_nd(1, 0, &kRttSubtypeNoDepth); const byte kRttTypeCanon = tester.DefineFunction( &sig_t1_v_nd, {}, {WASM_RTT_CANON(type_index), kExprEnd}); const byte kRttSubtypeCanon = tester.DefineFunction( &sig_t2_v_nd, {}, {WASM_RTT_CANON(subtype_index), kExprEnd}); const byte kRttSubtypeSub = tester.DefineFunction( &sig_t2_v_nd, {}, {WASM_RTT_SUB(subtype_index, WASM_CALL_FUNCTION0(kRttTypeCanon)), kExprEnd}); const byte kTestCanon = tester.DefineFunction( tester.sigs.i_v(), {optref(type_index)}, {WASM_LOCAL_SET(0, WASM_STRUCT_NEW_WITH_RTT( subtype_index, WASM_I32V(11), WASM_I32V(42), WASM_RTT_CANON(subtype_index))), WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_CALL_FUNCTION0(kRttSubtypeCanon)), kExprEnd}); const byte kTestSub = tester.DefineFunction( tester.sigs.i_v(), {optref(type_index)}, {WASM_LOCAL_SET( 0, WASM_STRUCT_NEW_WITH_RTT( subtype_index, WASM_I32V(11), WASM_I32V(42), WASM_RTT_SUB(subtype_index, WASM_RTT_CANON(type_index)))), WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_CALL_FUNCTION0(kRttSubtypeSub)), kExprEnd}); const byte kTestSubVsEmpty = tester.DefineFunction( tester.sigs.i_v(), {optref(type_index)}, {WASM_LOCAL_SET(0, WASM_STRUCT_NEW_WITH_RTT( subtype_index, WASM_I32V(11), WASM_I32V(42), WASM_RTT_SUB(subtype_index, WASM_RTT_CANON(empty_struct_index)))), WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_CALL_FUNCTION0(kRttSubtypeSub)), kExprEnd}); const byte kTestSubVsCanon = tester.DefineFunction( tester.sigs.i_v(), {optref(type_index)}, {WASM_LOCAL_SET(0, WASM_STRUCT_NEW_WITH_RTT( subtype_index, WASM_I32V(11), WASM_I32V(42), WASM_RTT_CANON(subtype_index))), WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_CALL_FUNCTION0(kRttSubtypeSub)), kExprEnd}); const byte kTestCanonVsSub = tester.DefineFunction( tester.sigs.i_v(), {optref(type_index)}, {WASM_LOCAL_SET( 0, WASM_STRUCT_NEW_WITH_RTT( subtype_index, WASM_I32V(11), WASM_I32V(42), WASM_RTT_SUB(subtype_index, WASM_RTT_CANON(type_index)))), WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_CALL_FUNCTION0(kRttSubtypeCanon)), kExprEnd}); const byte kTestSuperVsSub = tester.DefineFunction( tester.sigs.i_v(), {optref(type_index)}, {WASM_LOCAL_SET(0, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_RTT_CANON(type_index))), WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_CALL_FUNCTION0(kRttSubtypeCanon)), kExprEnd}); tester.CompileModule(); tester.CheckResult(kTestCanon, 1); tester.CheckResult(kTestSub, 1); tester.CheckResult(kTestSubVsEmpty, 0); tester.CheckResult(kTestSubVsCanon, 0); tester.CheckResult(kTestCanonVsSub, 0); tester.CheckResult(kTestSuperVsSub, 0); } WASM_COMPILED_EXEC_TEST(ArrayNewMap) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineArray(kWasmI32, true); ValueType array_type = ValueType::Ref(type_index, kNonNullable); FunctionSig sig(1, 0, &array_type); const byte array_new_with_rtt = tester.DefineFunction( &sig, {}, {WASM_ARRAY_NEW_WITH_RTT(type_index, WASM_I32V(10), WASM_I32V(42), WASM_RTT_CANON(type_index)), kExprEnd}); ValueType rtt_type = ValueType::Rtt(type_index, 0); FunctionSig rtt_canon_sig(1, 0, &rtt_type); const byte kRttCanon = tester.DefineFunction( &rtt_canon_sig, {}, {WASM_RTT_CANON(type_index), kExprEnd}); tester.CompileModule(); Handle map = tester.GetResultObject(kRttCanon).ToHandleChecked(); Handle result = tester.GetResultObject(array_new_with_rtt).ToHandleChecked(); CHECK(result->IsWasmArray()); CHECK_EQ(Handle::cast(result)->map(), *map); } WASM_COMPILED_EXEC_TEST(FunctionRefs) { WasmGCTester tester(execution_tier); const byte func_index = tester.DefineFunction(tester.sigs.i_v(), {}, {WASM_I32V(42), kExprEnd}); const byte sig_index = 0; const byte other_sig_index = tester.DefineSignature(tester.sigs.d_d()); // This is just so func_index counts as "declared". tester.AddGlobal(ValueType::Ref(sig_index, kNullable), false, WasmInitExpr::RefFuncConst(func_index)); ValueType func_type = ValueType::Ref(sig_index, kNullable); FunctionSig sig_func(1, 0, &func_type); ValueType rtt0 = ValueType::Rtt(sig_index, 0); FunctionSig sig_rtt0(1, 0, &rtt0); const byte rtt_canon = tester.DefineFunction( &sig_rtt0, {}, {WASM_RTT_CANON(sig_index), kExprEnd}); const byte cast = tester.DefineFunction( &sig_func, {kWasmFuncRef}, {WASM_LOCAL_SET(0, WASM_REF_FUNC(func_index)), WASM_REF_CAST(WASM_LOCAL_GET(0), WASM_RTT_CANON(sig_index)), kExprEnd}); const byte cast_reference = tester.DefineFunction( &sig_func, {}, {WASM_REF_FUNC(sig_index), kExprEnd}); const byte test = tester.DefineFunction( tester.sigs.i_v(), {kWasmFuncRef}, {WASM_LOCAL_SET(0, WASM_REF_FUNC(func_index)), WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_RTT_CANON(sig_index)), kExprEnd}); const byte test_fail_1 = tester.DefineFunction( tester.sigs.i_v(), {kWasmFuncRef}, {WASM_LOCAL_SET(0, WASM_REF_FUNC(func_index)), WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_RTT_CANON(other_sig_index)), kExprEnd}); const byte test_fail_2 = tester.DefineFunction( tester.sigs.i_v(), {kWasmFuncRef}, {WASM_LOCAL_SET(0, WASM_REF_FUNC(func_index)), WASM_REF_TEST(WASM_LOCAL_GET(0), WASM_RTT_SUB(sig_index, WASM_RTT_CANON(sig_index))), kExprEnd}); tester.CompileModule(); Handle result_canon = tester.GetResultObject(rtt_canon).ToHandleChecked(); CHECK(result_canon->IsMap()); Handle map_canon = Handle::cast(result_canon); CHECK(map_canon->IsJSFunctionMap()); Handle result_cast = tester.GetResultObject(cast).ToHandleChecked(); CHECK(result_cast->IsJSFunction()); Handle cast_function = Handle::cast(result_cast); Handle result_cast_reference = tester.GetResultObject(cast_reference).ToHandleChecked(); CHECK(result_cast_reference->IsJSFunction()); Handle cast_function_reference = Handle::cast(result_cast_reference); CHECK_EQ(cast_function->code().raw_instruction_start(), cast_function_reference->code().raw_instruction_start()); tester.CheckResult(test, 1); tester.CheckResult(test_fail_1, 0); tester.CheckResult(test_fail_2, 0); } WASM_COMPILED_EXEC_TEST(CallRef) { WasmGCTester tester(execution_tier); byte callee = tester.DefineFunction( tester.sigs.i_ii(), {}, {WASM_I32_ADD(WASM_LOCAL_GET(0), WASM_LOCAL_GET(1)), kExprEnd}); byte caller = tester.DefineFunction( tester.sigs.i_i(), {}, {WASM_CALL_REF(WASM_REF_FUNC(callee), WASM_I32V(42), WASM_LOCAL_GET(0)), kExprEnd}); // This is just so func_index counts as "declared". tester.AddGlobal(ValueType::Ref(0, kNullable), false, WasmInitExpr::RefFuncConst(callee)); tester.CompileModule(); tester.CheckResult(caller, 47, 5); } WASM_COMPILED_EXEC_TEST(CallReftypeParameters) { WasmGCTester tester(execution_tier); byte type_index = tester.DefineStruct({F(wasm::kWasmI32, true)}); ValueType kRefType{optref(type_index)}; ValueType sig_types[] = {kWasmI32, kRefType, kRefType, kRefType, kRefType, kWasmI32, kWasmI32, kWasmI32, kWasmI32}; FunctionSig sig(1, 8, sig_types); byte adder = tester.DefineFunction( &sig, {}, {WASM_I32_ADD( WASM_STRUCT_GET(type_index, 0, WASM_LOCAL_GET(0)), WASM_I32_ADD( WASM_STRUCT_GET(type_index, 0, WASM_LOCAL_GET(1)), WASM_I32_ADD( WASM_STRUCT_GET(type_index, 0, WASM_LOCAL_GET(2)), WASM_I32_ADD( WASM_STRUCT_GET(type_index, 0, WASM_LOCAL_GET(3)), WASM_I32_ADD( WASM_LOCAL_GET(4), WASM_I32_ADD(WASM_LOCAL_GET(5), WASM_I32_ADD(WASM_LOCAL_GET(6), WASM_LOCAL_GET(7)))))))), kExprEnd}); byte caller = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_CALL_FUNCTION(adder, WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(2), WASM_RTT_CANON(type_index)), WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(4), WASM_RTT_CANON(type_index)), WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(8), WASM_RTT_CANON(type_index)), WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(16), WASM_RTT_CANON(type_index)), WASM_I32V(32), WASM_I32V(64), WASM_I32V(128), WASM_I32V(256)), kExprEnd}); tester.CompileModule(); tester.CheckResult(caller, 510); } WASM_COMPILED_EXEC_TEST(AbstractTypeChecks) { WasmGCTester tester(execution_tier); byte array_index = tester.DefineArray(kWasmI32, true); byte function_index = tester.DefineFunction(tester.sigs.v_v(), {}, {kExprEnd}); byte sig_index = 1; // This is just so func_index counts as "declared". tester.AddGlobal(ValueType::Ref(sig_index, kNullable), false, WasmInitExpr::RefFuncConst(function_index)); byte kDataCheckNull = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_IS_DATA(WASM_REF_NULL(kAnyRefCode)), kExprEnd}); byte kFuncCheckNull = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_IS_FUNC(WASM_REF_NULL(kAnyRefCode)), kExprEnd}); byte kI31CheckNull = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_REF_IS_I31(WASM_REF_NULL(kAnyRefCode)), kExprEnd}); byte kDataCastNull = tester.DefineFunction(tester.sigs.i_v(), {}, {WASM_REF_AS_DATA(WASM_REF_NULL(kAnyRefCode)), WASM_DROP, WASM_I32V(1), kExprEnd}); byte kFuncCastNull = tester.DefineFunction(tester.sigs.i_v(), {}, {WASM_REF_AS_FUNC(WASM_REF_NULL(kAnyRefCode)), WASM_DROP, WASM_I32V(1), kExprEnd}); byte kI31CastNull = tester.DefineFunction(tester.sigs.i_v(), {}, {WASM_REF_AS_I31(WASM_REF_NULL(kAnyRefCode)), WASM_DROP, WASM_I32V(1), kExprEnd}); #define TYPE_CHECK(type, value) \ tester.DefineFunction(tester.sigs.i_v(), {kWasmAnyRef}, \ {WASM_LOCAL_SET(0, WASM_SEQ(value)), \ WASM_REF_IS_##type(WASM_LOCAL_GET(0)), kExprEnd}) byte kDataCheckSuccess = TYPE_CHECK(DATA, WASM_ARRAY_NEW_DEFAULT(array_index, WASM_I32V(10), WASM_RTT_CANON(array_index))); byte kDataCheckFailure = TYPE_CHECK(DATA, WASM_I31_NEW(WASM_I32V(42))); byte kFuncCheckSuccess = TYPE_CHECK(FUNC, WASM_REF_FUNC(function_index)); byte kFuncCheckFailure = TYPE_CHECK(FUNC, WASM_ARRAY_NEW_DEFAULT(array_index, WASM_I32V(10), WASM_RTT_CANON(array_index))); byte kI31CheckSuccess = TYPE_CHECK(I31, WASM_I31_NEW(WASM_I32V(42))); byte kI31CheckFailure = TYPE_CHECK(I31, WASM_ARRAY_NEW_DEFAULT(array_index, WASM_I32V(10), WASM_RTT_CANON(array_index))); #undef TYPE_CHECK #define TYPE_CAST(type, value) \ tester.DefineFunction(tester.sigs.i_v(), {kWasmAnyRef}, \ {WASM_LOCAL_SET(0, WASM_SEQ(value)), \ WASM_REF_AS_##type(WASM_LOCAL_GET(0)), WASM_DROP, \ WASM_I32V(1), kExprEnd}) byte kDataCastSuccess = TYPE_CAST(DATA, WASM_ARRAY_NEW_DEFAULT(array_index, WASM_I32V(10), WASM_RTT_CANON(array_index))); byte kDataCastFailure = TYPE_CAST(DATA, WASM_I31_NEW(WASM_I32V(42))); byte kFuncCastSuccess = TYPE_CAST(FUNC, WASM_REF_FUNC(function_index)); byte kFuncCastFailure = TYPE_CAST(FUNC, WASM_ARRAY_NEW_DEFAULT(array_index, WASM_I32V(10), WASM_RTT_CANON(array_index))); byte kI31CastSuccess = TYPE_CAST(I31, WASM_I31_NEW(WASM_I32V(42))); byte kI31CastFailure = TYPE_CAST(I31, WASM_ARRAY_NEW_DEFAULT(array_index, WASM_I32V(10), WASM_RTT_CANON(array_index))); #undef TYPE_CAST // If the branch is not taken, we return 0. If it is taken, then the respective // type check should succeed, and we return 1. #define BR_ON(TYPE, type, value) \ tester.DefineFunction( \ tester.sigs.i_v(), {kWasmAnyRef}, \ {WASM_LOCAL_SET(0, WASM_SEQ(value)), \ WASM_REF_IS_##TYPE(WASM_BLOCK_R(kWasm##type##Ref, WASM_LOCAL_GET(0), \ WASM_BR_ON_##TYPE(0), \ WASM_RETURN(WASM_I32V(0)))), \ kExprEnd}) byte kBrOnDataTaken = BR_ON(DATA, Data, WASM_ARRAY_NEW_DEFAULT(array_index, WASM_I32V(10), WASM_RTT_CANON(array_index))); byte kBrOnDataNotTaken = BR_ON(DATA, Data, WASM_REF_FUNC(function_index)); byte kBrOnFuncTaken = BR_ON(FUNC, Func, WASM_REF_FUNC(function_index)); byte kBrOnFuncNotTaken = BR_ON(FUNC, Func, WASM_I31_NEW(WASM_I32V(42))); byte kBrOnI31Taken = BR_ON(I31, I31, WASM_I31_NEW(WASM_I32V(42))); byte kBrOnI31NotTaken = BR_ON(I31, I31, WASM_ARRAY_NEW_DEFAULT(array_index, WASM_I32V(10), WASM_RTT_CANON(array_index))); #undef BR_ON // If the branch is not taken, we return 1. If it is taken, then the respective // type check should fail, and we return 0. #define BR_ON_NON(TYPE, type, value) \ tester.DefineFunction( \ tester.sigs.i_v(), {kWasmAnyRef}, \ {WASM_LOCAL_SET(0, WASM_SEQ(value)), \ WASM_REF_IS_##TYPE(WASM_BLOCK_R(kWasmAnyRef, WASM_LOCAL_GET(0), \ WASM_BR_ON_NON_##TYPE(0), \ WASM_RETURN(WASM_I32V(1)))), \ kExprEnd}) byte kBrOnNonDataNotTaken = BR_ON_NON(DATA, Data, WASM_ARRAY_NEW_DEFAULT(array_index, WASM_I32V(10), WASM_RTT_CANON(array_index))); byte kBrOnNonDataTaken = BR_ON_NON(DATA, Data, WASM_REF_FUNC(function_index)); byte kBrOnNonFuncNotTaken = BR_ON_NON(FUNC, Func, WASM_REF_FUNC(function_index)); byte kBrOnNonFuncTaken = BR_ON_NON(FUNC, Func, WASM_I31_NEW(WASM_I32V(42))); byte kBrOnNonI31NotTaken = BR_ON_NON(I31, I31, WASM_I31_NEW(WASM_I32V(42))); byte kBrOnNonI31Taken = BR_ON_NON(I31, I31, WASM_ARRAY_NEW_DEFAULT(array_index, WASM_I32V(10), WASM_RTT_CANON(array_index))); #undef BR_ON_NON tester.CompileModule(); tester.CheckResult(kDataCheckNull, 0); tester.CheckResult(kFuncCheckNull, 0); tester.CheckResult(kI31CheckNull, 0); tester.CheckHasThrown(kDataCastNull); tester.CheckHasThrown(kFuncCastNull); tester.CheckHasThrown(kI31CastNull); tester.CheckResult(kDataCheckSuccess, 1); tester.CheckResult(kFuncCheckSuccess, 1); tester.CheckResult(kI31CheckSuccess, 1); tester.CheckResult(kDataCheckFailure, 0); tester.CheckResult(kFuncCheckFailure, 0); tester.CheckResult(kI31CheckFailure, 0); tester.CheckResult(kDataCastSuccess, 1); tester.CheckResult(kFuncCastSuccess, 1); tester.CheckResult(kI31CastSuccess, 1); tester.CheckHasThrown(kDataCastFailure); tester.CheckHasThrown(kFuncCastFailure); tester.CheckHasThrown(kI31CastFailure); tester.CheckResult(kBrOnDataTaken, 1); tester.CheckResult(kBrOnDataNotTaken, 0); tester.CheckResult(kBrOnFuncTaken, 1); tester.CheckResult(kBrOnFuncNotTaken, 0); tester.CheckResult(kBrOnI31Taken, 1); tester.CheckResult(kBrOnI31NotTaken, 0); tester.CheckResult(kBrOnNonDataTaken, 0); tester.CheckResult(kBrOnNonDataNotTaken, 1); tester.CheckResult(kBrOnNonFuncTaken, 0); tester.CheckResult(kBrOnNonFuncNotTaken, 1); tester.CheckResult(kBrOnNonI31Taken, 0); tester.CheckResult(kBrOnNonI31NotTaken, 1); } WASM_COMPILED_EXEC_TEST(BasicI31) { WasmGCTester tester(execution_tier); const byte kSigned = tester.DefineFunction( tester.sigs.i_i(), {}, {WASM_I31_GET_S(WASM_I31_NEW(WASM_LOCAL_GET(0))), kExprEnd}); const byte kUnsigned = tester.DefineFunction( tester.sigs.i_i(), {}, {WASM_I31_GET_U(WASM_I31_NEW(WASM_LOCAL_GET(0))), kExprEnd}); tester.CompileModule(); tester.CheckResult(kSigned, 123, 123); tester.CheckResult(kUnsigned, 123, 123); // Truncation: tester.CheckResult(kSigned, 0x1234, static_cast(0x80001234)); tester.CheckResult(kUnsigned, 0x1234, static_cast(0x80001234)); // Sign/zero extension: tester.CheckResult(kSigned, -1, 0x7FFFFFFF); tester.CheckResult(kUnsigned, 0x7FFFFFFF, 0x7FFFFFFF); } // This flushed out a few bugs, so it serves as a regression test. It can also // be modified (made to run longer) to measure performance of casts. WASM_COMPILED_EXEC_TEST(CastsBenchmark) { WasmGCTester tester(execution_tier); const byte SuperType = tester.DefineStruct({F(wasm::kWasmI32, true)}); const byte SubType = tester.DefineStruct({F(wasm::kWasmI32, true), F(wasm::kWasmI32, true)}); ValueType kDataRefNull = ValueType::Ref(HeapType::kData, kNullable); const byte ListType = tester.DefineArray(kDataRefNull, true); const byte List = tester.AddGlobal(ValueType::Ref(ListType, kNullable), true, WasmInitExpr::RefNullConst( static_cast(ListType))); const byte RttSuper = tester.AddGlobal( ValueType::Rtt(SuperType, 0), false, WasmInitExpr::RttCanon(static_cast(SuperType))); const byte RttSub = tester.AddGlobal( ValueType::Rtt(SubType, 1), false, WasmInitExpr::RttSub(static_cast(SubType), WasmInitExpr::GlobalGet(RttSuper))); const byte RttList = tester.AddGlobal( ValueType::Rtt(ListType, 0), false, WasmInitExpr::RttCanon(static_cast(ListType))); const uint32_t kListLength = 1024; const uint32_t i = 0; const byte Prepare = tester.DefineFunction( tester.sigs.i_v(), {wasm::kWasmI32}, {// List = new eqref[kListLength]; WASM_GLOBAL_SET(List, WASM_ARRAY_NEW_DEFAULT(ListType, WASM_I32V(kListLength), WASM_GLOBAL_GET(RttList))), // for (int i = 0; i < kListLength; ) { // List[i] = new Super(i); // i++; // List[i] = new Sub(i, 0); // i++; // } WASM_LOCAL_SET(i, WASM_I32V_1(0)), WASM_LOOP( WASM_ARRAY_SET(ListType, WASM_GLOBAL_GET(List), WASM_LOCAL_GET(i), WASM_STRUCT_NEW_WITH_RTT(SuperType, WASM_LOCAL_GET(i), WASM_GLOBAL_GET(RttSuper))), WASM_LOCAL_SET(i, WASM_I32_ADD(WASM_LOCAL_GET(i), WASM_I32V_1(1))), WASM_ARRAY_SET(ListType, WASM_GLOBAL_GET(List), WASM_LOCAL_GET(i), WASM_STRUCT_NEW_WITH_RTT(SubType, WASM_LOCAL_GET(i), WASM_I32V_1(0), WASM_GLOBAL_GET(RttSub))), WASM_LOCAL_SET(i, WASM_I32_ADD(WASM_LOCAL_GET(i), WASM_I32V_1(1))), WASM_BR_IF(0, WASM_I32_NE(WASM_LOCAL_GET(i), WASM_I32V(kListLength)))), // return 42; // Dummy value, due to test framework. WASM_I32V_1(42), kExprEnd}); const uint32_t sum = 1; // Index of the local. const uint32_t list = 2; const uint32_t kLoops = 2; const uint32_t kIterations = kLoops * kListLength; const byte Main = tester.DefineFunction( tester.sigs.i_v(), { wasm::kWasmI32, wasm::kWasmI32, ValueType::Ref(ListType, kNullable), }, {WASM_LOCAL_SET(list, WASM_GLOBAL_GET(List)), // sum = 0; WASM_LOCAL_SET(sum, WASM_I32V_1(0)), // for (int i = 0; i < kIterations; i++) { // sum += ref.cast(List[i & kListLength]).x // } WASM_LOCAL_SET(i, WASM_I32V_1(0)), WASM_LOOP( WASM_LOCAL_SET( sum, WASM_I32_ADD( WASM_LOCAL_GET(sum), WASM_STRUCT_GET( SuperType, 0, WASM_REF_CAST( WASM_ARRAY_GET( ListType, WASM_LOCAL_GET(list), WASM_I32_AND(WASM_LOCAL_GET(i), WASM_I32V(kListLength - 1))), WASM_GLOBAL_GET(RttSuper))))), WASM_LOCAL_SET(i, WASM_I32_ADD(WASM_LOCAL_GET(i), WASM_I32V_1(1))), WASM_BR_IF(0, WASM_I32_LTS(WASM_LOCAL_GET(i), WASM_I32V(kIterations)))), // return sum; WASM_LOCAL_GET(sum), kExprEnd}); tester.CompileModule(); tester.CheckResult(Prepare, 42); // Time this section to get a benchmark for subtyping checks. // Note: if you bump kIterations or kListLength, you may have to take i32 // overflow into account. tester.CheckResult(Main, (kListLength * (kListLength - 1) / 2) * kLoops); } WASM_COMPILED_EXEC_TEST(GlobalInitReferencingGlobal) { WasmGCTester tester(execution_tier); const byte from = tester.AddGlobal(kWasmI32, false, WasmInitExpr(42)); const byte to = tester.AddGlobal(kWasmI32, false, WasmInitExpr::GlobalGet(from)); const byte func = tester.DefineFunction(tester.sigs.i_v(), {}, {WASM_GLOBAL_GET(to), kExprEnd}); tester.CompileModule(); tester.CheckResult(func, 42); } WASM_COMPILED_EXEC_TEST(GCTables) { WasmGCTester tester(execution_tier); byte super_struct = tester.DefineStruct({F(kWasmI32, false)}); byte sub_struct = tester.DefineStruct({F(kWasmI32, false), F(kWasmI32, true)}); FunctionSig* super_sig = FunctionSig::Build(tester.zone(), {kWasmI32}, {optref(sub_struct)}); byte super_sig_index = tester.DefineSignature(super_sig); FunctionSig* sub_sig = FunctionSig::Build(tester.zone(), {kWasmI32}, {optref(super_struct)}); byte sub_sig_index = tester.DefineSignature(sub_sig); tester.DefineTable(optref(super_sig_index), 10, 10); byte super_func = tester.DefineFunction( super_sig, {}, {WASM_I32_ADD(WASM_STRUCT_GET(sub_struct, 0, WASM_LOCAL_GET(0)), WASM_STRUCT_GET(sub_struct, 1, WASM_LOCAL_GET(0))), WASM_END}); byte sub_func = tester.DefineFunction( sub_sig, {}, {WASM_STRUCT_GET(super_struct, 0, WASM_LOCAL_GET(0)), WASM_END}); byte setup_func = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_TABLE_SET(0, WASM_I32V(0), WASM_REF_NULL(super_sig_index)), WASM_TABLE_SET(0, WASM_I32V(1), WASM_REF_FUNC(super_func)), WASM_TABLE_SET(0, WASM_I32V(2), WASM_REF_FUNC(sub_func)), WASM_I32V(0), WASM_END}); byte super_struct_producer = tester.DefineFunction( FunctionSig::Build(tester.zone(), {ref(super_struct)}, {}), {}, {WASM_STRUCT_NEW_WITH_RTT(super_struct, WASM_I32V(-5), WASM_RTT_CANON(super_struct)), WASM_END}); byte sub_struct_producer = tester.DefineFunction( FunctionSig::Build(tester.zone(), {ref(sub_struct)}, {}), {}, {WASM_STRUCT_NEW_WITH_RTT(sub_struct, WASM_I32V(7), WASM_I32V(11), WASM_RTT_CANON(sub_struct)), WASM_END}); // Calling a null entry should trap. byte call_null = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_CALL_INDIRECT(super_sig_index, WASM_CALL_FUNCTION0(sub_struct_producer), WASM_I32V(0)), WASM_END}); // Calling with a signature identical to the type of the table should work, // provided the entry has the same signature. byte call_same_type = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_CALL_INDIRECT(super_sig_index, WASM_CALL_FUNCTION0(sub_struct_producer), WASM_I32V(1)), WASM_END}); // Calling with a signature that is a subtype of the type of the table should // work, provided the entry has the same signature. byte call_subtype = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_CALL_INDIRECT(sub_sig_index, WASM_CALL_FUNCTION0(super_struct_producer), WASM_I32V(2)), WASM_END}); // Calling with a signature that is mismatched to that of the entry should // trap. byte call_type_mismatch = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_CALL_INDIRECT(super_sig_index, WASM_CALL_FUNCTION0(sub_struct_producer), WASM_I32V(2)), WASM_END}); // Getting a table element and then calling it with call_ref should work. byte table_get_and_call_ref = tester.DefineFunction( tester.sigs.i_v(), {}, {WASM_CALL_REF(WASM_TABLE_GET(0, WASM_I32V(2)), WASM_CALL_FUNCTION0(sub_struct_producer)), WASM_END}); // Only here so these functions count as "declared". tester.AddGlobal(optref(super_sig_index), false, WasmInitExpr::RefFuncConst(super_func)); tester.AddGlobal(optref(sub_sig_index), false, WasmInitExpr::RefFuncConst(sub_func)); tester.CompileModule(); tester.CheckResult(setup_func, 0); tester.CheckHasThrown(call_null); tester.CheckResult(call_same_type, 18); tester.CheckResult(call_subtype, -5); tester.CheckHasThrown(call_type_mismatch); tester.CheckResult(table_get_and_call_ref, 7); } WASM_COMPILED_EXEC_TEST(JsAccess) { WasmGCTester tester(execution_tier); const byte type_index = tester.DefineStruct({F(wasm::kWasmI32, true)}); ValueType kRefType = ref(type_index); ValueType kSupertypeToI[] = {kWasmI32, kWasmDataRef}; FunctionSig sig_t_v(1, 0, &kRefType); FunctionSig sig_super_v(1, 0, &kWasmDataRef); FunctionSig sig_i_super(1, 1, kSupertypeToI); tester.DefineExportedFunction( "disallowed", &sig_t_v, {WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_RTT_CANON(type_index)), kExprEnd}); // Same code, different signature. tester.DefineExportedFunction( "producer", &sig_super_v, {WASM_STRUCT_NEW_WITH_RTT(type_index, WASM_I32V(42), WASM_RTT_CANON(type_index)), kExprEnd}); tester.DefineExportedFunction( "consumer", &sig_i_super, {WASM_STRUCT_GET( type_index, 0, WASM_REF_CAST(WASM_LOCAL_GET(0), WASM_RTT_CANON(type_index))), kExprEnd}); tester.CompileModule(); Isolate* isolate = tester.isolate(); TryCatch try_catch(reinterpret_cast(isolate)); MaybeHandle maybe_result = tester.CallExportedFunction("disallowed", 0, nullptr); CHECK(maybe_result.is_null()); CHECK(try_catch.HasCaught()); try_catch.Reset(); isolate->clear_pending_exception(); maybe_result = tester.CallExportedFunction("producer", 0, nullptr); if (maybe_result.is_null()) { FATAL("Calling 'producer' failed: %s", *v8::String::Utf8Value(reinterpret_cast(isolate), try_catch.Message()->Get())); } { Handle args[] = {maybe_result.ToHandleChecked()}; maybe_result = tester.CallExportedFunction("consumer", 1, args); } if (maybe_result.is_null()) { FATAL("Calling 'consumer' failed: %s", *v8::String::Utf8Value(reinterpret_cast(isolate), try_catch.Message()->Get())); } Handle result = maybe_result.ToHandleChecked(); CHECK(result->IsSmi()); CHECK_EQ(42, Smi::cast(*result).value()); // Calling {consumer} with any other object (e.g. the Smi we just got as // {result}) should trap. { Handle args[] = {result}; maybe_result = tester.CallExportedFunction("consumer", 1, args); } CHECK(maybe_result.is_null()); CHECK(try_catch.HasCaught()); try_catch.Reset(); isolate->clear_pending_exception(); } } // namespace test_gc } // namespace wasm } // namespace internal } // namespace v8