// 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 "src/wasm/wasm-opcodes-inl.h" #include "test/cctest/cctest.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 { template class Memory64Runner : public WasmRunner { public: explicit Memory64Runner(TestExecutionTier execution_tier) : WasmRunner(execution_tier) { this->builder().EnableFeature(kFeature_memory64); this->builder().SetMemory64(); } }; WASM_EXEC_TEST(Load) { // TODO(clemensb): Implement memory64 in the interpreter. if (execution_tier == TestExecutionTier::kInterpreter) return; Memory64Runner r(execution_tier); uint32_t* memory = r.builder().AddMemoryElems(kWasmPageSize / sizeof(int32_t)); BUILD(r, WASM_LOAD_MEM(MachineType::Int32(), WASM_LOCAL_GET(0))); CHECK_EQ(0, r.Call(0)); #if V8_TARGET_BIG_ENDIAN memory[0] = 0x78563412; #else memory[0] = 0x12345678; #endif CHECK_EQ(0x12345678, r.Call(0)); CHECK_EQ(0x123456, r.Call(1)); CHECK_EQ(0x1234, r.Call(2)); CHECK_EQ(0x12, r.Call(3)); CHECK_EQ(0x0, r.Call(4)); CHECK_TRAP(r.Call(-1)); CHECK_TRAP(r.Call(kWasmPageSize)); CHECK_TRAP(r.Call(kWasmPageSize - 3)); CHECK_EQ(0x0, r.Call(kWasmPageSize - 4)); CHECK_TRAP(r.Call(uint64_t{1} << 32)); } // TODO(clemensb): Test atomic instructions. WASM_EXEC_TEST(InitExpression) { EXPERIMENTAL_FLAG_SCOPE(memory64); Isolate* isolate = CcTest::InitIsolateOnce(); HandleScope scope(isolate); ErrorThrower thrower(isolate, "TestMemory64InitExpression"); const byte data[] = { WASM_MODULE_HEADER, // SECTION(Memory, // ENTRY_COUNT(1), // kMemory64WithMaximum, // type 1, // initial size 2), // maximum size SECTION(Data, // ENTRY_COUNT(1), // 0, // linear memory index WASM_I64V_3(0xFFFF), kExprEnd, // destination offset U32V_1(1), // source size 'c') // data bytes }; testing::CompileAndInstantiateForTesting( isolate, &thrower, ModuleWireBytes(data, data + arraysize(data))); if (thrower.error()) { thrower.Reify()->Print(); FATAL("compile or instantiate error"); } } WASM_EXEC_TEST(MemorySize) { // TODO(clemensb): Implement memory64 in the interpreter. if (execution_tier == TestExecutionTier::kInterpreter) return; Memory64Runner r(execution_tier); constexpr int kNumPages = 13; r.builder().AddMemoryElems(kNumPages * kWasmPageSize); BUILD(r, WASM_MEMORY_SIZE); CHECK_EQ(kNumPages, r.Call()); } WASM_EXEC_TEST(MemoryGrow) { // TODO(clemensb): Implement memory64 in the interpreter. if (execution_tier == TestExecutionTier::kInterpreter) return; Memory64Runner r(execution_tier); r.builder().SetMaxMemPages(13); r.builder().AddMemory(kWasmPageSize); BUILD(r, WASM_MEMORY_GROW(WASM_LOCAL_GET(0))); CHECK_EQ(1, r.Call(6)); CHECK_EQ(7, r.Call(1)); CHECK_EQ(-1, r.Call(-1)); CHECK_EQ(-1, r.Call(int64_t{1} << 31)); CHECK_EQ(-1, r.Call(int64_t{1} << 32)); CHECK_EQ(-1, r.Call(int64_t{1} << 33)); CHECK_EQ(-1, r.Call(int64_t{1} << 63)); CHECK_EQ(-1, r.Call(6)); // Above the maximum of 13. CHECK_EQ(8, r.Call(5)); // Just at the maximum of 13. } } // namespace wasm } // namespace internal } // namespace v8