2018-06-21 10:58:58 +00:00
|
|
|
// Copyright 2015 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 <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2021-08-23 13:01:06 +00:00
|
|
|
#include "include/v8-wasm.h"
|
2019-05-17 12:13:44 +00:00
|
|
|
#include "src/api/api-inl.h"
|
2019-05-23 08:51:46 +00:00
|
|
|
#include "src/objects/objects-inl.h"
|
2018-06-21 10:58:58 +00:00
|
|
|
#include "src/snapshot/code-serializer.h"
|
2019-05-23 13:27:57 +00:00
|
|
|
#include "src/utils/version.h"
|
2018-06-21 10:58:58 +00:00
|
|
|
#include "src/wasm/module-decoder.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"
|
2020-01-20 11:22:36 +00:00
|
|
|
#include "src/wasm/wasm-serialization.h"
|
2018-06-21 10:58:58 +00:00
|
|
|
#include "test/cctest/cctest.h"
|
|
|
|
#include "test/common/wasm/flag-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"
|
|
|
|
|
2022-05-31 10:19:23 +00:00
|
|
|
namespace v8::internal::wasm {
|
2018-06-21 10:58:58 +00:00
|
|
|
|
|
|
|
// Approximate gtest TEST_F style, in case we adopt gtest.
|
|
|
|
class WasmSerializationTest {
|
|
|
|
public:
|
2022-05-31 10:19:23 +00:00
|
|
|
WasmSerializationTest() : zone_(&allocator_, ZONE_NAME) {
|
2018-06-21 10:58:58 +00:00
|
|
|
// Don't call here if we move to gtest.
|
|
|
|
SetUp();
|
|
|
|
}
|
|
|
|
|
2022-05-31 10:19:23 +00:00
|
|
|
static constexpr const char* kFunctionName = "increment";
|
|
|
|
|
2018-06-21 10:58:58 +00:00
|
|
|
static void BuildWireBytes(Zone* zone, ZoneBuffer* buffer) {
|
2020-07-09 11:51:58 +00:00
|
|
|
WasmModuleBuilder* builder = zone->New<WasmModuleBuilder>(zone);
|
2018-06-21 10:58:58 +00:00
|
|
|
TestSignatures sigs;
|
|
|
|
|
|
|
|
WasmFunctionBuilder* f = builder->AddFunction(sigs.i_i());
|
2020-12-17 16:55:33 +00:00
|
|
|
byte code[] = {WASM_LOCAL_GET(0), kExprI32Const, 1, kExprI32Add, kExprEnd};
|
2020-01-20 11:22:36 +00:00
|
|
|
f->EmitCode(code, sizeof(code));
|
2021-06-17 15:43:55 +00:00
|
|
|
builder->AddExport(base::CStrVector(kFunctionName), f);
|
2018-06-21 10:58:58 +00:00
|
|
|
|
2019-07-08 09:16:39 +00:00
|
|
|
builder->WriteTo(buffer);
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
|
2022-05-31 10:19:23 +00:00
|
|
|
void ClearSerializedData() { serialized_bytes_ = {}; }
|
2018-06-21 10:58:58 +00:00
|
|
|
|
|
|
|
void InvalidateVersion() {
|
|
|
|
uint32_t* slot = reinterpret_cast<uint32_t*>(
|
2018-12-10 12:22:30 +00:00
|
|
|
const_cast<uint8_t*>(serialized_bytes_.data()) +
|
2020-01-21 15:18:20 +00:00
|
|
|
WasmSerializer::kVersionHashOffset);
|
2018-06-21 10:58:58 +00:00
|
|
|
*slot = Version::Hash() + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
void InvalidateWireBytes() {
|
2018-12-10 12:22:30 +00:00
|
|
|
memset(const_cast<uint8_t*>(wire_bytes_.data()), 0, wire_bytes_.size() / 2);
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
|
2020-04-22 04:54:52 +00:00
|
|
|
MaybeHandle<WasmModuleObject> Deserialize(
|
2021-06-17 15:43:55 +00:00
|
|
|
base::Vector<const char> source_url = {}) {
|
2020-01-20 11:22:36 +00:00
|
|
|
return DeserializeNativeModule(CcTest::i_isolate(),
|
2021-06-17 15:43:55 +00:00
|
|
|
base::VectorOf(serialized_bytes_),
|
|
|
|
base::VectorOf(wire_bytes_), source_url);
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void DeserializeAndRun() {
|
2020-01-20 11:22:36 +00:00
|
|
|
ErrorThrower thrower(CcTest::i_isolate(), "");
|
|
|
|
Handle<WasmModuleObject> module_object;
|
|
|
|
CHECK(Deserialize().ToHandle(&module_object));
|
2018-06-21 10:58:58 +00:00
|
|
|
{
|
2020-11-20 16:57:36 +00:00
|
|
|
DisallowGarbageCollection assume_no_gc;
|
2021-06-17 15:43:55 +00:00
|
|
|
base::Vector<const byte> deserialized_module_wire_bytes =
|
2018-06-22 14:34:47 +00:00
|
|
|
module_object->native_module()->wire_bytes();
|
2018-12-10 12:22:30 +00:00
|
|
|
CHECK_EQ(deserialized_module_wire_bytes.size(), wire_bytes_.size());
|
2019-04-29 11:06:49 +00:00
|
|
|
CHECK_EQ(memcmp(deserialized_module_wire_bytes.begin(),
|
2018-12-10 12:22:30 +00:00
|
|
|
wire_bytes_.data(), wire_bytes_.size()),
|
2018-06-21 10:58:58 +00:00
|
|
|
0);
|
|
|
|
}
|
|
|
|
Handle<WasmInstanceObject> instance =
|
2021-06-18 14:29:39 +00:00
|
|
|
GetWasmEngine()
|
2020-01-20 11:22:36 +00:00
|
|
|
->SyncInstantiate(CcTest::i_isolate(), &thrower, module_object,
|
2018-06-21 10:58:58 +00:00
|
|
|
Handle<JSReceiver>::null(),
|
|
|
|
MaybeHandle<JSArrayBuffer>())
|
|
|
|
.ToHandleChecked();
|
|
|
|
Handle<Object> params[1] = {
|
2020-01-20 11:22:36 +00:00
|
|
|
Handle<Object>(Smi::FromInt(41), CcTest::i_isolate())};
|
2018-06-21 10:58:58 +00:00
|
|
|
int32_t result = testing::CallWasmFunctionForTesting(
|
2020-08-11 08:48:31 +00:00
|
|
|
CcTest::i_isolate(), instance, kFunctionName, 1, params);
|
2018-06-21 10:58:58 +00:00
|
|
|
CHECK_EQ(42, result);
|
|
|
|
}
|
|
|
|
|
2020-01-20 11:22:36 +00:00
|
|
|
void CollectGarbage() {
|
|
|
|
// Try hard to collect all garbage and will therefore also invoke all weak
|
|
|
|
// callbacks of actually unreachable persistent handles.
|
|
|
|
CcTest::i_isolate()->heap()->CollectAllAvailableGarbage(
|
|
|
|
GarbageCollectionReason::kTesting);
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Zone* zone() { return &zone_; }
|
|
|
|
|
|
|
|
void SetUp() {
|
2020-01-20 11:22:36 +00:00
|
|
|
CcTest::InitIsolateOnce();
|
2018-06-21 10:58:58 +00:00
|
|
|
ZoneBuffer buffer(&zone_);
|
|
|
|
WasmSerializationTest::BuildWireBytes(zone(), &buffer);
|
|
|
|
|
2020-01-20 11:22:36 +00:00
|
|
|
v8::Isolate::CreateParams create_params;
|
|
|
|
create_params.array_buffer_allocator =
|
|
|
|
CcTest::i_isolate()->array_buffer_allocator();
|
|
|
|
|
|
|
|
v8::Isolate* serialization_v8_isolate = v8::Isolate::New(create_params);
|
|
|
|
Isolate* serialization_isolate =
|
|
|
|
reinterpret_cast<Isolate*>(serialization_v8_isolate);
|
2018-06-21 10:58:58 +00:00
|
|
|
ErrorThrower thrower(serialization_isolate, "");
|
2020-01-23 09:47:31 +00:00
|
|
|
// Keep a weak pointer so we can check that the native module dies after
|
|
|
|
// serialization (when the isolate is disposed).
|
|
|
|
std::weak_ptr<NativeModule> weak_native_module;
|
2018-06-21 10:58:58 +00:00
|
|
|
{
|
|
|
|
HandleScope scope(serialization_isolate);
|
2020-01-20 11:22:36 +00:00
|
|
|
v8::Local<v8::Context> serialization_context =
|
|
|
|
v8::Context::New(serialization_v8_isolate);
|
|
|
|
serialization_context->Enter();
|
2018-06-21 10:58:58 +00:00
|
|
|
|
2019-11-26 16:25:14 +00:00
|
|
|
auto enabled_features = WasmFeatures::FromIsolate(serialization_isolate);
|
2018-06-21 10:58:58 +00:00
|
|
|
MaybeHandle<WasmModuleObject> maybe_module_object =
|
2021-06-18 14:29:39 +00:00
|
|
|
GetWasmEngine()->SyncCompile(
|
2018-08-08 14:54:44 +00:00
|
|
|
serialization_isolate, enabled_features, &thrower,
|
2018-06-21 10:58:58 +00:00
|
|
|
ModuleWireBytes(buffer.begin(), buffer.end()));
|
|
|
|
Handle<WasmModuleObject> module_object =
|
|
|
|
maybe_module_object.ToHandleChecked();
|
2020-01-23 09:47:31 +00:00
|
|
|
weak_native_module = module_object->shared_native_module();
|
|
|
|
// Check that the native module exists at this point.
|
|
|
|
CHECK(weak_native_module.lock());
|
2018-06-21 10:58:58 +00:00
|
|
|
|
|
|
|
v8::Local<v8::Object> v8_module_obj =
|
|
|
|
v8::Utils::ToLocal(Handle<JSObject>::cast(module_object));
|
2020-01-22 10:55:13 +00:00
|
|
|
CHECK(v8_module_obj->IsWasmModuleObject());
|
2018-06-21 10:58:58 +00:00
|
|
|
|
2018-12-10 12:22:30 +00:00
|
|
|
v8::Local<v8::WasmModuleObject> v8_module_object =
|
2018-12-05 15:43:33 +00:00
|
|
|
v8_module_obj.As<v8::WasmModuleObject>();
|
2018-12-10 12:22:30 +00:00
|
|
|
v8::CompiledWasmModule compiled_module =
|
|
|
|
v8_module_object->GetCompiledModule();
|
|
|
|
v8::MemorySpan<const uint8_t> uncompiled_bytes =
|
|
|
|
compiled_module.GetWireBytesRef();
|
|
|
|
uint8_t* bytes_copy = zone()->NewArray<uint8_t>(uncompiled_bytes.size());
|
|
|
|
memcpy(bytes_copy, uncompiled_bytes.data(), uncompiled_bytes.size());
|
|
|
|
wire_bytes_ = {bytes_copy, uncompiled_bytes.size()};
|
2022-05-31 10:19:23 +00:00
|
|
|
|
|
|
|
// Run the code until tier-up (of the single function) was observed.
|
|
|
|
Handle<WasmInstanceObject> instance =
|
|
|
|
GetWasmEngine()
|
|
|
|
->SyncInstantiate(serialization_isolate, &thrower, module_object,
|
|
|
|
{}, {})
|
|
|
|
.ToHandleChecked();
|
|
|
|
CHECK_EQ(0, data_.size);
|
|
|
|
while (data_.size == 0) {
|
|
|
|
testing::CallWasmFunctionForTesting(serialization_isolate, instance,
|
|
|
|
kFunctionName, 0, nullptr);
|
|
|
|
data_ = compiled_module.Serialize();
|
|
|
|
}
|
2020-08-11 11:19:17 +00:00
|
|
|
CHECK_LT(0, data_.size);
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
2020-01-20 11:22:36 +00:00
|
|
|
// Dispose of serialization isolate to destroy the reference to the
|
|
|
|
// NativeModule, which removes it from the module cache in the wasm engine
|
|
|
|
// and forces de-serialization in the new isolate.
|
|
|
|
serialization_v8_isolate->Dispose();
|
2018-06-21 10:58:58 +00:00
|
|
|
|
2020-01-23 09:47:31 +00:00
|
|
|
// Busy-wait for the NativeModule to really die. Background threads might
|
|
|
|
// temporarily keep it alive (happens very rarely, see
|
|
|
|
// https://crbug.com/v8/10148).
|
|
|
|
while (weak_native_module.lock()) {
|
|
|
|
}
|
|
|
|
|
2018-12-10 12:22:30 +00:00
|
|
|
serialized_bytes_ = {data_.buffer.get(), data_.size};
|
2018-06-21 10:58:58 +00:00
|
|
|
|
2020-01-20 11:22:36 +00:00
|
|
|
v8::HandleScope new_scope(CcTest::isolate());
|
2018-06-21 10:58:58 +00:00
|
|
|
v8::Local<v8::Context> deserialization_context =
|
2020-01-20 11:22:36 +00:00
|
|
|
v8::Context::New(CcTest::isolate());
|
2018-06-21 10:58:58 +00:00
|
|
|
deserialization_context->Enter();
|
|
|
|
}
|
|
|
|
|
|
|
|
v8::internal::AccountingAllocator allocator_;
|
|
|
|
Zone zone_;
|
2018-12-10 12:22:30 +00:00
|
|
|
v8::OwnedBuffer data_;
|
|
|
|
v8::MemorySpan<const uint8_t> wire_bytes_ = {nullptr, 0};
|
|
|
|
v8::MemorySpan<const uint8_t> serialized_bytes_ = {nullptr, 0};
|
2018-06-21 10:58:58 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
TEST(DeserializeValidModule) {
|
|
|
|
WasmSerializationTest test;
|
|
|
|
{
|
2020-01-20 11:22:36 +00:00
|
|
|
HandleScope scope(CcTest::i_isolate());
|
2018-06-21 10:58:58 +00:00
|
|
|
test.DeserializeAndRun();
|
|
|
|
}
|
2020-01-20 11:22:36 +00:00
|
|
|
test.CollectGarbage();
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
|
2020-04-22 04:54:52 +00:00
|
|
|
TEST(DeserializeWithSourceUrl) {
|
|
|
|
WasmSerializationTest test;
|
|
|
|
{
|
|
|
|
HandleScope scope(CcTest::i_isolate());
|
|
|
|
const std::string url = "http://example.com/example.wasm";
|
|
|
|
Handle<WasmModuleObject> module_object;
|
2021-06-17 15:43:55 +00:00
|
|
|
CHECK(test.Deserialize(base::VectorOf(url)).ToHandle(&module_object));
|
2021-03-19 11:19:31 +00:00
|
|
|
String url_str = String::cast(module_object->script().name());
|
|
|
|
CHECK_EQ(url, url_str.ToCString().get());
|
2020-04-22 04:54:52 +00:00
|
|
|
}
|
|
|
|
test.CollectGarbage();
|
|
|
|
}
|
|
|
|
|
2018-06-21 10:58:58 +00:00
|
|
|
TEST(DeserializeMismatchingVersion) {
|
|
|
|
WasmSerializationTest test;
|
|
|
|
{
|
2020-01-20 11:22:36 +00:00
|
|
|
HandleScope scope(CcTest::i_isolate());
|
2018-06-21 10:58:58 +00:00
|
|
|
test.InvalidateVersion();
|
2020-01-20 11:22:36 +00:00
|
|
|
CHECK(test.Deserialize().is_null());
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
2020-01-20 11:22:36 +00:00
|
|
|
test.CollectGarbage();
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(DeserializeNoSerializedData) {
|
|
|
|
WasmSerializationTest test;
|
|
|
|
{
|
2020-01-20 11:22:36 +00:00
|
|
|
HandleScope scope(CcTest::i_isolate());
|
2018-06-21 10:58:58 +00:00
|
|
|
test.ClearSerializedData();
|
2020-01-20 11:22:36 +00:00
|
|
|
CHECK(test.Deserialize().is_null());
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
2020-01-20 11:22:36 +00:00
|
|
|
test.CollectGarbage();
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(DeserializeWireBytesAndSerializedDataInvalid) {
|
|
|
|
WasmSerializationTest test;
|
|
|
|
{
|
2020-01-20 11:22:36 +00:00
|
|
|
HandleScope scope(CcTest::i_isolate());
|
2018-06-21 10:58:58 +00:00
|
|
|
test.InvalidateVersion();
|
|
|
|
test.InvalidateWireBytes();
|
2020-01-20 11:22:36 +00:00
|
|
|
CHECK(test.Deserialize().is_null());
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
2020-01-20 11:22:36 +00:00
|
|
|
test.CollectGarbage();
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bool False(v8::Local<v8::Context> context, v8::Local<v8::String> source) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(BlockWasmCodeGenAtDeserialization) {
|
|
|
|
WasmSerializationTest test;
|
|
|
|
{
|
2020-01-20 11:22:36 +00:00
|
|
|
HandleScope scope(CcTest::i_isolate());
|
|
|
|
CcTest::isolate()->SetAllowWasmCodeGenerationCallback(False);
|
|
|
|
CHECK(test.Deserialize().is_null());
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
2020-01-20 11:22:36 +00:00
|
|
|
test.CollectGarbage();
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
|
2019-10-09 09:47:48 +00:00
|
|
|
UNINITIALIZED_TEST(CompiledWasmModulesTransfer) {
|
2018-06-21 10:58:58 +00:00
|
|
|
v8::internal::AccountingAllocator allocator;
|
|
|
|
Zone zone(&allocator, ZONE_NAME);
|
|
|
|
|
|
|
|
ZoneBuffer buffer(&zone);
|
|
|
|
WasmSerializationTest::BuildWireBytes(&zone, &buffer);
|
|
|
|
|
2018-07-31 08:16:22 +00:00
|
|
|
v8::Isolate::CreateParams create_params;
|
|
|
|
create_params.array_buffer_allocator = CcTest::array_buffer_allocator();
|
|
|
|
v8::Isolate* from_isolate = v8::Isolate::New(create_params);
|
2019-10-09 09:47:48 +00:00
|
|
|
std::vector<v8::CompiledWasmModule> store;
|
2018-07-31 08:16:22 +00:00
|
|
|
std::shared_ptr<NativeModule> original_native_module;
|
2018-06-21 10:58:58 +00:00
|
|
|
{
|
2018-07-31 08:16:22 +00:00
|
|
|
v8::HandleScope scope(from_isolate);
|
|
|
|
LocalContext env(from_isolate);
|
|
|
|
|
|
|
|
Isolate* from_i_isolate = reinterpret_cast<Isolate*>(from_isolate);
|
|
|
|
testing::SetupIsolateForWasmModule(from_i_isolate);
|
2019-10-09 09:47:48 +00:00
|
|
|
ErrorThrower thrower(from_i_isolate, "TestCompiledWasmModulesTransfer");
|
2019-11-26 16:25:14 +00:00
|
|
|
auto enabled_features = WasmFeatures::FromIsolate(from_i_isolate);
|
2018-07-31 08:16:22 +00:00
|
|
|
MaybeHandle<WasmModuleObject> maybe_module_object =
|
2021-06-18 14:29:39 +00:00
|
|
|
GetWasmEngine()->SyncCompile(
|
2018-08-08 14:54:44 +00:00
|
|
|
from_i_isolate, enabled_features, &thrower,
|
2018-06-21 10:58:58 +00:00
|
|
|
ModuleWireBytes(buffer.begin(), buffer.end()));
|
2018-07-31 08:16:22 +00:00
|
|
|
Handle<WasmModuleObject> module_object =
|
|
|
|
maybe_module_object.ToHandleChecked();
|
2018-12-05 15:43:33 +00:00
|
|
|
v8::Local<v8::WasmModuleObject> v8_module =
|
|
|
|
v8::Local<v8::WasmModuleObject>::Cast(
|
2018-07-31 08:16:22 +00:00
|
|
|
v8::Utils::ToLocal(Handle<JSObject>::cast(module_object)));
|
2019-10-09 09:47:48 +00:00
|
|
|
store.push_back(v8_module->GetCompiledModule());
|
2018-12-11 10:37:32 +00:00
|
|
|
original_native_module = module_object->shared_native_module();
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
v8::Isolate* to_isolate = v8::Isolate::New(create_params);
|
|
|
|
{
|
2018-07-31 08:16:22 +00:00
|
|
|
v8::HandleScope scope(to_isolate);
|
|
|
|
LocalContext env(to_isolate);
|
|
|
|
|
2018-12-05 15:43:33 +00:00
|
|
|
v8::MaybeLocal<v8::WasmModuleObject> transferred_module =
|
2019-10-09 09:47:48 +00:00
|
|
|
v8::WasmModuleObject::FromCompiledModule(to_isolate, store[0]);
|
2018-07-31 08:16:22 +00:00
|
|
|
CHECK(!transferred_module.IsEmpty());
|
|
|
|
Handle<WasmModuleObject> module_object = Handle<WasmModuleObject>::cast(
|
|
|
|
v8::Utils::OpenHandle(*transferred_module.ToLocalChecked()));
|
|
|
|
std::shared_ptr<NativeModule> transferred_native_module =
|
2018-12-11 10:37:32 +00:00
|
|
|
module_object->shared_native_module();
|
2019-10-09 09:47:48 +00:00
|
|
|
CHECK_EQ(original_native_module, transferred_native_module);
|
2018-06-21 10:58:58 +00:00
|
|
|
}
|
|
|
|
to_isolate->Dispose();
|
|
|
|
}
|
2018-07-31 08:16:22 +00:00
|
|
|
original_native_module.reset();
|
|
|
|
from_isolate->Dispose();
|
|
|
|
}
|
|
|
|
|
2020-08-18 08:07:29 +00:00
|
|
|
TEST(TierDownAfterDeserialization) {
|
|
|
|
WasmSerializationTest test;
|
|
|
|
|
|
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
Handle<WasmModuleObject> module_object;
|
|
|
|
CHECK(test.Deserialize().ToHandle(&module_object));
|
|
|
|
|
|
|
|
auto* native_module = module_object->native_module();
|
|
|
|
CHECK_EQ(1, native_module->module()->functions.size());
|
|
|
|
WasmCodeRefScope code_ref_scope;
|
2022-05-31 10:19:23 +00:00
|
|
|
// The deserialized code must be TurboFan (we wait for tier-up before
|
|
|
|
// serializing).
|
2020-08-18 08:07:29 +00:00
|
|
|
auto* turbofan_code = native_module->GetCode(0);
|
2020-11-19 17:15:54 +00:00
|
|
|
CHECK_NOT_NULL(turbofan_code);
|
2020-08-18 08:07:29 +00:00
|
|
|
CHECK_EQ(ExecutionTier::kTurbofan, turbofan_code->tier());
|
|
|
|
|
2021-06-18 14:29:39 +00:00
|
|
|
GetWasmEngine()->TierDownAllModulesPerIsolate(isolate);
|
2020-08-18 08:07:29 +00:00
|
|
|
|
|
|
|
auto* liftoff_code = native_module->GetCode(0);
|
|
|
|
CHECK_EQ(ExecutionTier::kLiftoff, liftoff_code->tier());
|
|
|
|
}
|
|
|
|
|
2021-11-04 14:31:26 +00:00
|
|
|
TEST(SerializeLiftoffModuleFails) {
|
|
|
|
// Make sure that no function is tiered up to TurboFan.
|
2022-08-29 10:33:33 +00:00
|
|
|
if (!v8_flags.liftoff) return;
|
|
|
|
FlagScope<bool> no_tier_up(&v8_flags.wasm_tier_up, false);
|
2021-11-04 14:31:26 +00:00
|
|
|
v8::internal::AccountingAllocator allocator;
|
|
|
|
Zone zone(&allocator, "test_zone");
|
|
|
|
|
|
|
|
CcTest::InitIsolateOnce();
|
|
|
|
Isolate* isolate = CcTest::i_isolate();
|
|
|
|
HandleScope scope(isolate);
|
|
|
|
ZoneBuffer wire_bytes_buffer(&zone);
|
|
|
|
WasmSerializationTest::BuildWireBytes(&zone, &wire_bytes_buffer);
|
|
|
|
|
|
|
|
ErrorThrower thrower(isolate, "Test");
|
|
|
|
MaybeHandle<WasmModuleObject> maybe_module_object =
|
|
|
|
GetWasmEngine()->SyncCompile(
|
|
|
|
isolate, WasmFeatures::All(), &thrower,
|
|
|
|
ModuleWireBytes(wire_bytes_buffer.begin(), wire_bytes_buffer.end()));
|
|
|
|
Handle<WasmModuleObject> module_object =
|
|
|
|
maybe_module_object.ToHandleChecked();
|
|
|
|
|
|
|
|
NativeModule* native_module = module_object->native_module();
|
|
|
|
WasmSerializer wasm_serializer(native_module);
|
|
|
|
size_t buffer_size = wasm_serializer.GetSerializedNativeModuleSize();
|
|
|
|
std::unique_ptr<uint8_t[]> buffer(new uint8_t[buffer_size]);
|
|
|
|
// Serialization is expected to fail if there is no TurboFan function to
|
|
|
|
// serialize.
|
|
|
|
CHECK(!wasm_serializer.SerializeNativeModule({buffer.get(), buffer_size}));
|
|
|
|
}
|
|
|
|
|
2022-05-31 10:19:23 +00:00
|
|
|
} // namespace v8::internal::wasm
|