Commit c5a16a39 authored by Clemens Hammacher's avatar Clemens Hammacher Committed by Commit Bot

[wasm] Factor out {WasmCodeAllocator}

This factors out a {WasmCodeAllocator} which manages all code
reservations and allocations for a {NativeModule}. This will allow for
better testing of this component (which will be added in a separate CL).

R=titzer@chromium.org

Change-Id: I09727c30afc533e95569276147792d0e641b0507
Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/1605738
Commit-Queue: Clemens Hammacher <clemensh@chromium.org>
Reviewed-by: 's avatarBen Titzer <titzer@chromium.org>
Cr-Commit-Position: refs/heads/master@{#61445}
parent 5343d789
...@@ -400,18 +400,198 @@ void WasmCode::DecrementRefCount(Vector<WasmCode* const> code_vec) { ...@@ -400,18 +400,198 @@ void WasmCode::DecrementRefCount(Vector<WasmCode* const> code_vec) {
if (engine) engine->FreeDeadCode(dead_code); if (engine) engine->FreeDeadCode(dead_code);
} }
WasmCodeAllocator::WasmCodeAllocator(WasmCodeManager* code_manager,
VirtualMemory code_space,
bool can_request_more)
: code_manager_(code_manager),
free_code_space_(code_space.region()),
can_request_more_memory_(can_request_more) {
owned_code_space_.reserve(can_request_more ? 4 : 1);
owned_code_space_.emplace_back(std::move(code_space));
}
WasmCodeAllocator::~WasmCodeAllocator() {
code_manager_->FreeNativeModule(VectorOf(owned_code_space_),
committed_code_space());
}
Vector<byte> WasmCodeAllocator::AllocateForCode(NativeModule* native_module,
size_t size) {
base::MutexGuard lock(&mutex_);
DCHECK_EQ(code_manager_, native_module->engine()->code_manager());
DCHECK_LT(0, size);
v8::PageAllocator* page_allocator = GetPlatformPageAllocator();
// This happens under a lock assumed by the caller.
size = RoundUp<kCodeAlignment>(size);
base::AddressRegion code_space = free_code_space_.Allocate(size);
if (code_space.is_empty()) {
if (!can_request_more_memory_) {
V8::FatalProcessOutOfMemory(nullptr, "wasm code reservation");
UNREACHABLE();
}
Address hint = owned_code_space_.empty() ? kNullAddress
: owned_code_space_.back().end();
VirtualMemory new_mem =
code_manager_->TryAllocate(size, reinterpret_cast<void*>(hint));
if (!new_mem.IsReserved()) {
V8::FatalProcessOutOfMemory(nullptr, "wasm code reservation");
UNREACHABLE();
}
code_manager_->AssignRanges(new_mem.address(), new_mem.end(),
native_module);
free_code_space_.Merge(new_mem.region());
owned_code_space_.emplace_back(std::move(new_mem));
code_space = free_code_space_.Allocate(size);
DCHECK(!code_space.is_empty());
}
const Address page_size = page_allocator->AllocatePageSize();
Address commit_start = RoundUp(code_space.begin(), page_size);
Address commit_end = RoundUp(code_space.end(), page_size);
// {commit_start} will be either code_space.start or the start of the next
// page. {commit_end} will be the start of the page after the one in which
// the allocation ends.
// We start from an aligned start, and we know we allocated vmem in
// page multiples.
// We just need to commit what's not committed. The page in which we
// start is already committed (or we start at the beginning of a page).
// The end needs to be committed all through the end of the page.
if (commit_start < commit_end) {
committed_code_space_.fetch_add(commit_end - commit_start);
// Committed code cannot grow bigger than maximum code space size.
DCHECK_LE(committed_code_space_.load(), kMaxWasmCodeMemory);
#if V8_OS_WIN
// On Windows, we cannot commit a region that straddles different
// reservations of virtual memory. Because we bump-allocate, and because, if
// we need more memory, we append that memory at the end of the
// owned_code_space_ list, we traverse that list in reverse order to find
// the reservation(s) that guide how to chunk the region to commit.
for (auto& vmem : base::Reversed(owned_code_space_)) {
if (commit_end <= vmem.address() || vmem.end() <= commit_start) continue;
Address start = std::max(commit_start, vmem.address());
Address end = std::min(commit_end, vmem.end());
size_t commit_size = static_cast<size_t>(end - start);
if (!code_manager_->Commit(start, commit_size)) {
V8::FatalProcessOutOfMemory(nullptr, "wasm code commit");
UNREACHABLE();
}
// Opportunistically reduce the commit range. This might terminate the
// loop early.
if (commit_start == start) commit_start = end;
if (commit_end == end) commit_end = start;
if (commit_start >= commit_end) break;
}
#else
if (!code_manager_->Commit(commit_start, commit_end - commit_start)) {
V8::FatalProcessOutOfMemory(nullptr, "wasm code commit");
UNREACHABLE();
}
#endif
}
DCHECK(IsAligned(code_space.begin(), kCodeAlignment));
allocated_code_space_.Merge(code_space);
generated_code_size_.fetch_add(code_space.size(), std::memory_order_relaxed);
TRACE_HEAP("Code alloc for %p: %" PRIxPTR ",+%zu\n", this, code_space.begin(),
size);
return {reinterpret_cast<byte*>(code_space.begin()), code_space.size()};
}
bool WasmCodeAllocator::SetExecutable(bool executable) {
base::MutexGuard lock(&mutex_);
if (is_executable_ == executable) return true;
TRACE_HEAP("Setting module %p as executable: %d.\n", this, executable);
v8::PageAllocator* page_allocator = GetPlatformPageAllocator();
if (FLAG_wasm_write_protect_code_memory) {
PageAllocator::Permission permission =
executable ? PageAllocator::kReadExecute : PageAllocator::kReadWrite;
#if V8_OS_WIN
// On windows, we need to switch permissions per separate virtual memory
// reservation. This is really just a problem when the NativeModule is
// growable (meaning can_request_more_memory_). That's 32-bit in production,
// or unittests.
// For now, in that case, we commit at reserved memory granularity.
// Technically, that may be a waste, because we may reserve more than we
// use. On 32-bit though, the scarce resource is the address space -
// committed or not.
if (can_request_more_memory_) {
for (auto& vmem : owned_code_space_) {
if (!SetPermissions(page_allocator, vmem.address(), vmem.size(),
permission)) {
return false;
}
TRACE_HEAP("Set %p:%p to executable:%d\n", vmem.address(), vmem.end(),
executable);
}
is_executable_ = executable;
return true;
}
#endif
for (auto& region : allocated_code_space_.regions()) {
// allocated_code_space_ is fine-grained, so we need to
// page-align it.
size_t region_size =
RoundUp(region.size(), page_allocator->AllocatePageSize());
if (!SetPermissions(page_allocator, region.begin(), region_size,
permission)) {
return false;
}
TRACE_HEAP("Set %p:%p to executable:%d\n",
reinterpret_cast<void*>(region.begin()),
reinterpret_cast<void*>(region.end()), executable);
}
}
is_executable_ = executable;
return true;
}
void WasmCodeAllocator::FreeCode(Vector<WasmCode* const> codes) {
// Zap code area and collect freed code regions.
DisjointAllocationPool freed_regions;
size_t code_size = 0;
for (WasmCode* code : codes) {
ZapCode(code->instruction_start(), code->instructions().size());
FlushInstructionCache(code->instruction_start(),
code->instructions().size());
code_size += code->instructions().size();
freed_regions.Merge(base::AddressRegion{code->instruction_start(),
code->instructions().size()});
}
freed_code_size_.fetch_add(code_size);
// Merge {freed_regions} into {freed_code_space_} and discard full pages.
base::MutexGuard guard(&mutex_);
PageAllocator* allocator = GetPlatformPageAllocator();
size_t page_size = allocator->AllocatePageSize();
for (auto region : freed_regions.regions()) {
auto merged_region = freed_code_space_.Merge(region);
Address discard_start = std::max(RoundUp(merged_region.begin(), page_size),
RoundDown(region.begin(), page_size));
Address discard_end = std::min(RoundDown(merged_region.end(), page_size),
RoundUp(region.end(), page_size));
if (discard_start >= discard_end) continue;
// TODO(clemensh): Reenable after fixing https://crbug.com/960707.
// allocator->DiscardSystemPages(reinterpret_cast<void*>(discard_start),
// discard_end - discard_start);
}
}
NativeModule::NativeModule(WasmEngine* engine, const WasmFeatures& enabled, NativeModule::NativeModule(WasmEngine* engine, const WasmFeatures& enabled,
bool can_request_more, VirtualMemory code_space, bool can_request_more, VirtualMemory code_space,
std::shared_ptr<const WasmModule> module, std::shared_ptr<const WasmModule> module,
std::shared_ptr<Counters> async_counters, std::shared_ptr<Counters> async_counters,
std::shared_ptr<NativeModule>* shared_this) std::shared_ptr<NativeModule>* shared_this)
: enabled_features_(enabled), : code_allocator_(engine->code_manager(), std::move(code_space),
can_request_more),
enabled_features_(enabled),
module_(std::move(module)), module_(std::move(module)),
import_wrapper_cache_(std::unique_ptr<WasmImportWrapperCache>( import_wrapper_cache_(std::unique_ptr<WasmImportWrapperCache>(
new WasmImportWrapperCache(this))), new WasmImportWrapperCache(this))),
free_code_space_(code_space.region()),
engine_(engine), engine_(engine),
can_request_more_memory_(can_request_more),
use_trap_handler_(trap_handler::IsTrapHandlerEnabled() ? kUseTrapHandler use_trap_handler_(trap_handler::IsTrapHandlerEnabled() ? kUseTrapHandler
: kNoTrapHandler) { : kNoTrapHandler) {
// We receive a pointer to an empty {std::shared_ptr}, and install ourselve // We receive a pointer to an empty {std::shared_ptr}, and install ourselve
...@@ -422,8 +602,6 @@ NativeModule::NativeModule(WasmEngine* engine, const WasmFeatures& enabled, ...@@ -422,8 +602,6 @@ NativeModule::NativeModule(WasmEngine* engine, const WasmFeatures& enabled,
compilation_state_ = compilation_state_ =
CompilationState::New(*shared_this, std::move(async_counters)); CompilationState::New(*shared_this, std::move(async_counters));
DCHECK_NOT_NULL(module_); DCHECK_NOT_NULL(module_);
owned_code_space_.reserve(can_request_more ? 4 : 1);
owned_code_space_.emplace_back(std::move(code_space));
#if defined(V8_OS_WIN_X64) #if defined(V8_OS_WIN_X64)
// On some platforms, specifically Win64, we need to reserve some pages at // On some platforms, specifically Win64, we need to reserve some pages at
...@@ -433,7 +611,7 @@ NativeModule::NativeModule(WasmEngine* engine, const WasmFeatures& enabled, ...@@ -433,7 +611,7 @@ NativeModule::NativeModule(WasmEngine* engine, const WasmFeatures& enabled,
// for details. // for details.
if (engine_->code_manager() if (engine_->code_manager()
->CanRegisterUnwindInfoForNonABICompliantCodeRange()) { ->CanRegisterUnwindInfoForNonABICompliantCodeRange()) {
AllocateForCode(Heap::GetCodeRangeReservedAreaSize()); code_allocator_.AllocateForCode(this, Heap::GetCodeRangeReservedAreaSize());
} }
#endif #endif
...@@ -584,7 +762,8 @@ WasmCode* NativeModule::AddAndPublishAnonymousCode(Handle<Code> code, ...@@ -584,7 +762,8 @@ WasmCode* NativeModule::AddAndPublishAnonymousCode(Handle<Code> code,
const size_t code_comments_offset = const size_t code_comments_offset =
static_cast<size_t>(code->code_comments_offset()); static_cast<size_t>(code->code_comments_offset());
Vector<uint8_t> dst_code_bytes = AllocateForCode(instructions.size()); Vector<uint8_t> dst_code_bytes =
code_allocator_.AllocateForCode(this, instructions.size());
memcpy(dst_code_bytes.begin(), instructions.begin(), instructions.size()); memcpy(dst_code_bytes.begin(), instructions.begin(), instructions.size());
// Apply the relocation delta by iterating over the RelocInfo. // Apply the relocation delta by iterating over the RelocInfo.
...@@ -642,10 +821,10 @@ std::unique_ptr<WasmCode> NativeModule::AddCode( ...@@ -642,10 +821,10 @@ std::unique_ptr<WasmCode> NativeModule::AddCode(
OwnedVector<trap_handler::ProtectedInstructionData> protected_instructions, OwnedVector<trap_handler::ProtectedInstructionData> protected_instructions,
OwnedVector<const byte> source_position_table, WasmCode::Kind kind, OwnedVector<const byte> source_position_table, WasmCode::Kind kind,
ExecutionTier tier) { ExecutionTier tier) {
return AddCodeWithCodeSpace(index, desc, stack_slots, tagged_parameter_slots, return AddCodeWithCodeSpace(
std::move(protected_instructions), index, desc, stack_slots, tagged_parameter_slots,
std::move(source_position_table), kind, tier, std::move(protected_instructions), std::move(source_position_table), kind,
AllocateForCode(desc.instr_size)); tier, code_allocator_.AllocateForCode(this, desc.instr_size));
} }
std::unique_ptr<WasmCode> NativeModule::AddCodeWithCodeSpace( std::unique_ptr<WasmCode> NativeModule::AddCodeWithCodeSpace(
...@@ -802,7 +981,8 @@ WasmCode* NativeModule::AddDeserializedCode( ...@@ -802,7 +981,8 @@ WasmCode* NativeModule::AddDeserializedCode(
OwnedVector<const byte> reloc_info, OwnedVector<const byte> reloc_info,
OwnedVector<const byte> source_position_table, WasmCode::Kind kind, OwnedVector<const byte> source_position_table, WasmCode::Kind kind,
ExecutionTier tier) { ExecutionTier tier) {
Vector<uint8_t> dst_code_bytes = AllocateForCode(instructions.size()); Vector<uint8_t> dst_code_bytes =
code_allocator_.AllocateForCode(this, instructions.size());
memcpy(dst_code_bytes.begin(), instructions.begin(), instructions.size()); memcpy(dst_code_bytes.begin(), instructions.begin(), instructions.size());
std::unique_ptr<WasmCode> code{new WasmCode{ std::unique_ptr<WasmCode> code{new WasmCode{
...@@ -846,7 +1026,8 @@ bool NativeModule::HasCode(uint32_t index) const { ...@@ -846,7 +1026,8 @@ bool NativeModule::HasCode(uint32_t index) const {
WasmCode* NativeModule::CreateEmptyJumpTable(uint32_t jump_table_size) { WasmCode* NativeModule::CreateEmptyJumpTable(uint32_t jump_table_size) {
// Only call this if we really need a jump table. // Only call this if we really need a jump table.
DCHECK_LT(0, jump_table_size); DCHECK_LT(0, jump_table_size);
Vector<uint8_t> code_space = AllocateForCode(jump_table_size); Vector<uint8_t> code_space =
code_allocator_.AllocateForCode(this, jump_table_size);
ZapCode(reinterpret_cast<Address>(code_space.begin()), code_space.size()); ZapCode(reinterpret_cast<Address>(code_space.begin()), code_space.size());
std::unique_ptr<WasmCode> code{new WasmCode{ std::unique_ptr<WasmCode> code{new WasmCode{
this, // native_module this, // native_module
...@@ -867,93 +1048,6 @@ WasmCode* NativeModule::CreateEmptyJumpTable(uint32_t jump_table_size) { ...@@ -867,93 +1048,6 @@ WasmCode* NativeModule::CreateEmptyJumpTable(uint32_t jump_table_size) {
return PublishCode(std::move(code)); return PublishCode(std::move(code));
} }
Vector<byte> NativeModule::AllocateForCode(size_t size) {
base::MutexGuard lock(&allocation_mutex_);
DCHECK_LT(0, size);
v8::PageAllocator* page_allocator = GetPlatformPageAllocator();
// This happens under a lock assumed by the caller.
size = RoundUp<kCodeAlignment>(size);
base::AddressRegion code_space = free_code_space_.Allocate(size);
if (code_space.is_empty()) {
if (!can_request_more_memory_) {
V8::FatalProcessOutOfMemory(nullptr,
"NativeModule::AllocateForCode reservation");
UNREACHABLE();
}
Address hint = owned_code_space_.empty() ? kNullAddress
: owned_code_space_.back().end();
VirtualMemory new_mem = engine_->code_manager()->TryAllocate(
size, reinterpret_cast<void*>(hint));
if (!new_mem.IsReserved()) {
V8::FatalProcessOutOfMemory(nullptr,
"NativeModule::AllocateForCode reservation");
UNREACHABLE();
}
engine_->code_manager()->AssignRanges(new_mem.address(), new_mem.end(),
this);
free_code_space_.Merge(new_mem.region());
owned_code_space_.emplace_back(std::move(new_mem));
code_space = free_code_space_.Allocate(size);
DCHECK(!code_space.is_empty());
}
const Address page_size = page_allocator->AllocatePageSize();
Address commit_start = RoundUp(code_space.begin(), page_size);
Address commit_end = RoundUp(code_space.end(), page_size);
// {commit_start} will be either code_space.start or the start of the next
// page. {commit_end} will be the start of the page after the one in which
// the allocation ends.
// We start from an aligned start, and we know we allocated vmem in
// page multiples.
// We just need to commit what's not committed. The page in which we
// start is already committed (or we start at the beginning of a page).
// The end needs to be committed all through the end of the page.
if (commit_start < commit_end) {
committed_code_space_.fetch_add(commit_end - commit_start);
// Committed code cannot grow bigger than maximum code space size.
DCHECK_LE(committed_code_space_.load(), kMaxWasmCodeMemory);
#if V8_OS_WIN
// On Windows, we cannot commit a region that straddles different
// reservations of virtual memory. Because we bump-allocate, and because, if
// we need more memory, we append that memory at the end of the
// owned_code_space_ list, we traverse that list in reverse order to find
// the reservation(s) that guide how to chunk the region to commit.
for (auto& vmem : base::Reversed(owned_code_space_)) {
if (commit_end <= vmem.address() || vmem.end() <= commit_start) continue;
Address start = std::max(commit_start, vmem.address());
Address end = std::min(commit_end, vmem.end());
size_t commit_size = static_cast<size_t>(end - start);
if (!engine_->code_manager()->Commit(start, commit_size)) {
V8::FatalProcessOutOfMemory(nullptr,
"NativeModule::AllocateForCode commit");
UNREACHABLE();
}
// Opportunistically reduce the commit range. This might terminate the
// loop early.
if (commit_start == start) commit_start = end;
if (commit_end == end) commit_end = start;
if (commit_start >= commit_end) break;
}
#else
if (!engine_->code_manager()->Commit(commit_start,
commit_end - commit_start)) {
V8::FatalProcessOutOfMemory(nullptr,
"NativeModule::AllocateForCode commit");
UNREACHABLE();
}
#endif
}
DCHECK(IsAligned(code_space.begin(), kCodeAlignment));
allocated_code_space_.Merge(code_space);
generated_code_size_.fetch_add(code_space.size(), std::memory_order_relaxed);
TRACE_HEAP("Code alloc for %p: %" PRIxPTR ",+%zu\n", this, code_space.begin(),
size);
return {reinterpret_cast<byte*>(code_space.begin()), code_space.size()};
}
namespace { namespace {
class NativeModuleWireBytesStorage final : public WireBytesStorage { class NativeModuleWireBytesStorage final : public WireBytesStorage {
public: public:
...@@ -1025,7 +1119,7 @@ const char* NativeModule::GetRuntimeStubName(Address runtime_stub_entry) const { ...@@ -1025,7 +1119,7 @@ const char* NativeModule::GetRuntimeStubName(Address runtime_stub_entry) const {
} }
NativeModule::~NativeModule() { NativeModule::~NativeModule() {
TRACE_HEAP("Deleting native module: %p\n", reinterpret_cast<void*>(this)); TRACE_HEAP("Deleting native module: %p\n", this);
// Cancel all background compilation before resetting any field of the // Cancel all background compilation before resetting any field of the
// NativeModule or freeing anything. // NativeModule or freeing anything.
compilation_state_->AbortCompilation(); compilation_state_->AbortCompilation();
...@@ -1189,7 +1283,7 @@ std::shared_ptr<NativeModule> WasmCodeManager::NewNativeModule( ...@@ -1189,7 +1283,7 @@ std::shared_ptr<NativeModule> WasmCodeManager::NewNativeModule(
code_space = TryAllocate(code_vmem_size); code_space = TryAllocate(code_vmem_size);
if (code_space.IsReserved()) break; if (code_space.IsReserved()) break;
if (retries == kAllocationRetries) { if (retries == kAllocationRetries) {
V8::FatalProcessOutOfMemory(isolate, "WasmCodeManager::NewNativeModule"); V8::FatalProcessOutOfMemory(isolate, "NewNativeModule");
UNREACHABLE(); UNREACHABLE();
} }
// Run one GC, then try the allocation again. // Run one GC, then try the allocation again.
...@@ -1220,60 +1314,11 @@ std::shared_ptr<NativeModule> WasmCodeManager::NewNativeModule( ...@@ -1220,60 +1314,11 @@ std::shared_ptr<NativeModule> WasmCodeManager::NewNativeModule(
return ret; return ret;
} }
bool NativeModule::SetExecutable(bool executable) {
if (is_executable_ == executable) return true;
TRACE_HEAP("Setting module %p as executable: %d.\n", this, executable);
v8::PageAllocator* page_allocator = GetPlatformPageAllocator();
if (FLAG_wasm_write_protect_code_memory) {
PageAllocator::Permission permission =
executable ? PageAllocator::kReadExecute : PageAllocator::kReadWrite;
#if V8_OS_WIN
// On windows, we need to switch permissions per separate virtual memory
// reservation. This is really just a problem when the NativeModule is
// growable (meaning can_request_more_memory_). That's 32-bit in production,
// or unittests.
// For now, in that case, we commit at reserved memory granularity.
// Technically, that may be a waste, because we may reserve more than we
// use. On 32-bit though, the scarce resource is the address space -
// committed or not.
if (can_request_more_memory_) {
for (auto& vmem : owned_code_space_) {
if (!SetPermissions(page_allocator, vmem.address(), vmem.size(),
permission)) {
return false;
}
TRACE_HEAP("Set %p:%p to executable:%d\n", vmem.address(), vmem.end(),
executable);
}
is_executable_ = executable;
return true;
}
#endif
for (auto& region : allocated_code_space_.regions()) {
// allocated_code_space_ is fine-grained, so we need to
// page-align it.
size_t region_size =
RoundUp(region.size(), page_allocator->AllocatePageSize());
if (!SetPermissions(page_allocator, region.begin(), region_size,
permission)) {
return false;
}
TRACE_HEAP("Set %p:%p to executable:%d\n",
reinterpret_cast<void*>(region.begin()),
reinterpret_cast<void*>(region.end()), executable);
}
}
is_executable_ = executable;
return true;
}
void NativeModule::SampleCodeSize( void NativeModule::SampleCodeSize(
Counters* counters, NativeModule::CodeSamplingTime sampling_time) const { Counters* counters, NativeModule::CodeSamplingTime sampling_time) const {
size_t code_size = sampling_time == kSampling size_t code_size = sampling_time == kSampling
? committed_code_space() ? code_allocator_.committed_code_space()
: generated_code_size_.load(std::memory_order_relaxed); : code_allocator_.generated_code_size();
int code_size_mb = static_cast<int>(code_size / MB); int code_size_mb = static_cast<int>(code_size / MB);
Histogram* histogram = nullptr; Histogram* histogram = nullptr;
switch (sampling_time) { switch (sampling_time) {
...@@ -1286,12 +1331,12 @@ void NativeModule::SampleCodeSize( ...@@ -1286,12 +1331,12 @@ void NativeModule::SampleCodeSize(
case kSampling: { case kSampling: {
histogram = counters->wasm_module_code_size_mb(); histogram = counters->wasm_module_code_size_mb();
// Also, add a sample of freed code size, absolute and relative. // Also, add a sample of freed code size, absolute and relative.
size_t freed_size = freed_code_size_.load(std::memory_order_relaxed); size_t freed_size = code_allocator_.freed_code_size();
DCHECK_LE(freed_size, generated_code_size_); size_t generated_size = code_allocator_.generated_code_size();
DCHECK_LE(freed_size, generated_size);
int total_freed_mb = static_cast<int>(freed_size / MB); int total_freed_mb = static_cast<int>(freed_size / MB);
counters->wasm_module_freed_code_size_mb()->AddSample(total_freed_mb); counters->wasm_module_freed_code_size_mb()->AddSample(total_freed_mb);
int freed_percent = int freed_percent = static_cast<int>(100 * freed_size / generated_size);
static_cast<int>(100 * freed_code_size_ / generated_code_size_);
counters->wasm_module_freed_code_size_percent()->AddSample(freed_percent); counters->wasm_module_freed_code_size_percent()->AddSample(freed_percent);
break; break;
} }
...@@ -1312,7 +1357,8 @@ std::vector<WasmCode*> NativeModule::AddCompiledCode( ...@@ -1312,7 +1357,8 @@ std::vector<WasmCode*> NativeModule::AddCompiledCode(
DCHECK(result.succeeded()); DCHECK(result.succeeded());
total_code_space += RoundUp<kCodeAlignment>(result.code_desc.instr_size); total_code_space += RoundUp<kCodeAlignment>(result.code_desc.instr_size);
} }
Vector<byte> code_space = AllocateForCode(total_code_space); Vector<byte> code_space =
code_allocator_.AllocateForCode(this, total_code_space);
std::vector<std::unique_ptr<WasmCode>> generated_code; std::vector<std::unique_ptr<WasmCode>> generated_code;
generated_code.reserve(results.size()); generated_code.reserve(results.size());
...@@ -1351,18 +1397,8 @@ bool NativeModule::IsRedirectedToInterpreter(uint32_t func_index) { ...@@ -1351,18 +1397,8 @@ bool NativeModule::IsRedirectedToInterpreter(uint32_t func_index) {
} }
void NativeModule::FreeCode(Vector<WasmCode* const> codes) { void NativeModule::FreeCode(Vector<WasmCode* const> codes) {
// Zap code area and collect freed code regions. // Free the code space.
DisjointAllocationPool freed_regions; code_allocator_.FreeCode(codes);
size_t code_size = 0;
for (WasmCode* code : codes) {
ZapCode(code->instruction_start(), code->instructions().size());
FlushInstructionCache(code->instruction_start(),
code->instructions().size());
code_size += code->instructions().size();
freed_regions.Merge(base::AddressRegion{code->instruction_start(),
code->instructions().size()});
}
freed_code_size_.fetch_add(code_size);
// Free the {WasmCode} objects. This will also unregister trap handler data. // Free the {WasmCode} objects. This will also unregister trap handler data.
base::MutexGuard guard(&allocation_mutex_); base::MutexGuard guard(&allocation_mutex_);
...@@ -1370,27 +1406,12 @@ void NativeModule::FreeCode(Vector<WasmCode* const> codes) { ...@@ -1370,27 +1406,12 @@ void NativeModule::FreeCode(Vector<WasmCode* const> codes) {
DCHECK_EQ(1, owned_code_.count(code->instruction_start())); DCHECK_EQ(1, owned_code_.count(code->instruction_start()));
owned_code_.erase(code->instruction_start()); owned_code_.erase(code->instruction_start());
} }
// Merge {freed_regions} into {freed_code_space_} and discard full pages.
PageAllocator* allocator = GetPlatformPageAllocator();
size_t page_size = allocator->AllocatePageSize();
for (auto region : freed_regions.regions()) {
auto merged_region = freed_code_space_.Merge(region);
Address discard_start = std::max(RoundUp(merged_region.begin(), page_size),
RoundDown(region.begin(), page_size));
Address discard_end = std::min(RoundDown(merged_region.end(), page_size),
RoundUp(region.end(), page_size));
if (discard_start >= discard_end) continue;
// TODO(clemensh): Reenable after fixing https://crbug.com/960707.
// allocator->DiscardSystemPages(reinterpret_cast<void*>(discard_start),
// discard_end - discard_start);
}
} }
void WasmCodeManager::FreeNativeModule(NativeModule* native_module) { void WasmCodeManager::FreeNativeModule(Vector<VirtualMemory> owned_code_space,
size_t committed_size) {
base::MutexGuard lock(&native_modules_mutex_); base::MutexGuard lock(&native_modules_mutex_);
TRACE_HEAP("Freeing NativeModule %p\n", native_module); for (auto& code_space : owned_code_space) {
for (auto& code_space : native_module->owned_code_space_) {
DCHECK(code_space.IsReserved()); DCHECK(code_space.IsReserved());
TRACE_HEAP("VMem Release: %" PRIxPTR ":%" PRIxPTR " (%zu)\n", TRACE_HEAP("VMem Release: %" PRIxPTR ":%" PRIxPTR " (%zu)\n",
code_space.address(), code_space.end(), code_space.size()); code_space.address(), code_space.end(), code_space.size());
...@@ -1407,12 +1428,10 @@ void WasmCodeManager::FreeNativeModule(NativeModule* native_module) { ...@@ -1407,12 +1428,10 @@ void WasmCodeManager::FreeNativeModule(NativeModule* native_module) {
code_space.Free(); code_space.Free();
DCHECK(!code_space.IsReserved()); DCHECK(!code_space.IsReserved());
} }
native_module->owned_code_space_.clear();
size_t code_size = native_module->committed_code_space_.load(); DCHECK(IsAligned(committed_size, AllocatePageSize()));
DCHECK(IsAligned(code_size, AllocatePageSize())); size_t old_committed = total_committed_code_space_.fetch_sub(committed_size);
size_t old_committed = total_committed_code_space_.fetch_sub(code_size); DCHECK_LE(committed_size, old_committed);
DCHECK_LE(code_size, old_committed);
USE(old_committed); USE(old_committed);
} }
......
...@@ -272,6 +272,64 @@ class V8_EXPORT_PRIVATE WasmCode final { ...@@ -272,6 +272,64 @@ class V8_EXPORT_PRIVATE WasmCode final {
// Return a textual description of the kind. // Return a textual description of the kind.
const char* GetWasmCodeKindAsString(WasmCode::Kind); const char* GetWasmCodeKindAsString(WasmCode::Kind);
// Manages the code reservations and allocations of a single {NativeModule}.
class WasmCodeAllocator {
public:
WasmCodeAllocator(WasmCodeManager*, VirtualMemory code_space,
bool can_request_more);
~WasmCodeAllocator();
size_t committed_code_space() const {
return committed_code_space_.load(std::memory_order_acquire);
}
size_t generated_code_size() const {
return generated_code_size_.load(std::memory_order_acquire);
}
size_t freed_code_size() const {
return freed_code_size_.load(std::memory_order_acquire);
}
// Allocate code space. Returns a valid buffer or fails with OOM (crash).
Vector<byte> AllocateForCode(NativeModule*, size_t size);
// Sets permissions of all owned code space to executable, or read-write (if
// {executable} is false). Returns true on success.
V8_EXPORT_PRIVATE bool SetExecutable(bool executable);
// Free memory pages of all given code objects. Used for wasm code GC.
void FreeCode(Vector<WasmCode* const>);
private:
// The engine-wide wasm code manager.
WasmCodeManager* const code_manager_;
mutable base::Mutex mutex_;
//////////////////////////////////////////////////////////////////////////////
// Protected by {mutex_}:
// Code space that was reserved and is available for allocations (subset of
// {owned_code_space_}).
DisjointAllocationPool free_code_space_;
// Code space that was allocated for code (subset of {owned_code_space_}).
DisjointAllocationPool allocated_code_space_;
// Code space that was allocated before but is dead now. Full pages within
// this region are discarded. It's still a subset of {owned_code_space_}).
DisjointAllocationPool freed_code_space_;
std::vector<VirtualMemory> owned_code_space_;
// End of fields protected by {mutex_}.
//////////////////////////////////////////////////////////////////////////////
std::atomic<size_t> committed_code_space_{0};
std::atomic<size_t> generated_code_size_{0};
std::atomic<size_t> freed_code_size_{0};
bool is_executable_ = false;
const bool can_request_more_memory_;
};
class V8_EXPORT_PRIVATE NativeModule final { class V8_EXPORT_PRIVATE NativeModule final {
public: public:
#if V8_TARGET_ARCH_X64 || V8_TARGET_ARCH_S390X || V8_TARGET_ARCH_ARM64 #if V8_TARGET_ARCH_X64 || V8_TARGET_ARCH_S390X || V8_TARGET_ARCH_ARM64
...@@ -357,7 +415,9 @@ class V8_EXPORT_PRIVATE NativeModule final { ...@@ -357,7 +415,9 @@ class V8_EXPORT_PRIVATE NativeModule final {
// above {GetCallTargetForFunction} returns) to a function index. // above {GetCallTargetForFunction} returns) to a function index.
uint32_t GetFunctionIndexFromJumpTableSlot(Address slot_address) const; uint32_t GetFunctionIndexFromJumpTableSlot(Address slot_address) const;
bool SetExecutable(bool executable); bool SetExecutable(bool executable) {
return code_allocator_.SetExecutable(executable);
}
// For cctests, where we build both WasmModule and the runtime objects // For cctests, where we build both WasmModule and the runtime objects
// on the fly, and bypass the instance builder pipeline. // on the fly, and bypass the instance builder pipeline.
...@@ -384,7 +444,9 @@ class V8_EXPORT_PRIVATE NativeModule final { ...@@ -384,7 +444,9 @@ class V8_EXPORT_PRIVATE NativeModule final {
Vector<const uint8_t> wire_bytes() const { return wire_bytes_->as_vector(); } Vector<const uint8_t> wire_bytes() const { return wire_bytes_->as_vector(); }
const WasmModule* module() const { return module_.get(); } const WasmModule* module() const { return module_.get(); }
std::shared_ptr<const WasmModule> shared_module() const { return module_; } std::shared_ptr<const WasmModule> shared_module() const { return module_; }
size_t committed_code_space() const { return committed_code_space_.load(); } size_t committed_code_space() const {
return code_allocator_.committed_code_space();
}
WasmEngine* engine() const { return engine_; } WasmEngine* engine() const { return engine_; }
void SetWireBytes(OwnedVector<const uint8_t> wire_bytes); void SetWireBytes(OwnedVector<const uint8_t> wire_bytes);
...@@ -442,8 +504,6 @@ class V8_EXPORT_PRIVATE NativeModule final { ...@@ -442,8 +504,6 @@ class V8_EXPORT_PRIVATE NativeModule final {
// Add and publish anonymous code. // Add and publish anonymous code.
WasmCode* AddAndPublishAnonymousCode(Handle<Code>, WasmCode::Kind kind, WasmCode* AddAndPublishAnonymousCode(Handle<Code>, WasmCode::Kind kind,
const char* name = nullptr); const char* name = nullptr);
// Allocate code space. Returns a valid buffer or fails with OOM (crash).
Vector<byte> AllocateForCode(size_t size);
WasmCode* CreateEmptyJumpTable(uint32_t jump_table_size); WasmCode* CreateEmptyJumpTable(uint32_t jump_table_size);
...@@ -471,6 +531,10 @@ class V8_EXPORT_PRIVATE NativeModule final { ...@@ -471,6 +531,10 @@ class V8_EXPORT_PRIVATE NativeModule final {
byte |= 1 << (bitset_idx % kBitsPerByte); byte |= 1 << (bitset_idx % kBitsPerByte);
} }
// {WasmCodeAllocator} manages all code reservations and allocations for this
// {NativeModule}.
WasmCodeAllocator code_allocator_;
// Features enabled for this module. We keep a copy of the features that // Features enabled for this module. We keep a copy of the features that
// were enabled at the time of the creation of this native module, // were enabled at the time of the creation of this native module,
// to be consistent across asynchronous compilations later. // to be consistent across asynchronous compilations later.
...@@ -517,27 +581,12 @@ class V8_EXPORT_PRIVATE NativeModule final { ...@@ -517,27 +581,12 @@ class V8_EXPORT_PRIVATE NativeModule final {
// this module marking those functions that have been redirected. // this module marking those functions that have been redirected.
std::unique_ptr<uint8_t[]> interpreter_redirections_; std::unique_ptr<uint8_t[]> interpreter_redirections_;
// Code space that was reserved and is available for allocations (subset of
// {owned_code_space_}).
DisjointAllocationPool free_code_space_;
// Code space that was allocated for code (subset of {owned_code_space_}).
DisjointAllocationPool allocated_code_space_;
// Code space that was allocated before but is dead now. Full pages within
// this region are discarded. It's still a subset of {owned_code_space_}).
DisjointAllocationPool freed_code_space_;
std::vector<VirtualMemory> owned_code_space_;
// End of fields protected by {allocation_mutex_}. // End of fields protected by {allocation_mutex_}.
////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////
WasmEngine* const engine_; WasmEngine* const engine_;
std::atomic<size_t> committed_code_space_{0};
std::atomic<size_t> generated_code_size_{0};
std::atomic<size_t> freed_code_size_{0};
int modification_scope_depth_ = 0; int modification_scope_depth_ = 0;
bool can_request_more_memory_;
UseTrapHandler use_trap_handler_ = kNoTrapHandler; UseTrapHandler use_trap_handler_ = kNoTrapHandler;
bool is_executable_ = false;
bool lazy_compile_frozen_ = false; bool lazy_compile_frozen_ = false;
DISALLOW_COPY_AND_ASSIGN(NativeModule); DISALLOW_COPY_AND_ASSIGN(NativeModule);
...@@ -577,7 +626,7 @@ class V8_EXPORT_PRIVATE WasmCodeManager final { ...@@ -577,7 +626,7 @@ class V8_EXPORT_PRIVATE WasmCodeManager final {
static size_t EstimateNativeModuleNonCodeSize(const WasmModule* module); static size_t EstimateNativeModuleNonCodeSize(const WasmModule* module);
private: private:
friend class NativeModule; friend class WasmCodeAllocator;
friend class WasmEngine; friend class WasmEngine;
std::shared_ptr<NativeModule> NewNativeModule( std::shared_ptr<NativeModule> NewNativeModule(
...@@ -592,7 +641,8 @@ class V8_EXPORT_PRIVATE WasmCodeManager final { ...@@ -592,7 +641,8 @@ class V8_EXPORT_PRIVATE WasmCodeManager final {
// for the freed memory size. We do that in FreeNativeModule. // for the freed memory size. We do that in FreeNativeModule.
// There's no separate Uncommit. // There's no separate Uncommit.
void FreeNativeModule(NativeModule*); void FreeNativeModule(Vector<VirtualMemory> owned_code,
size_t committed_size);
void AssignRanges(Address start, Address end, NativeModule*); void AssignRanges(Address start, Address end, NativeModule*);
......
...@@ -614,50 +614,46 @@ std::shared_ptr<NativeModule> WasmEngine::NewNativeModule( ...@@ -614,50 +614,46 @@ std::shared_ptr<NativeModule> WasmEngine::NewNativeModule(
} }
void WasmEngine::FreeNativeModule(NativeModule* native_module) { void WasmEngine::FreeNativeModule(NativeModule* native_module) {
{ base::MutexGuard guard(&mutex_);
base::MutexGuard guard(&mutex_); auto it = native_modules_.find(native_module);
auto it = native_modules_.find(native_module); DCHECK_NE(native_modules_.end(), it);
DCHECK_NE(native_modules_.end(), it); for (Isolate* isolate : it->second->isolates) {
for (Isolate* isolate : it->second->isolates) { DCHECK_EQ(1, isolates_.count(isolate));
DCHECK_EQ(1, isolates_.count(isolate)); IsolateInfo* info = isolates_[isolate].get();
IsolateInfo* info = isolates_[isolate].get(); DCHECK_EQ(1, info->native_modules.count(native_module));
DCHECK_EQ(1, info->native_modules.count(native_module)); info->native_modules.erase(native_module);
info->native_modules.erase(native_module); // If there are {WasmCode} objects of the deleted {NativeModule}
// If there are {WasmCode} objects of the deleted {NativeModule} // outstanding to be logged in this isolate, remove them. Decrementing the
// outstanding to be logged in this isolate, remove them. Decrementing the // ref count is not needed, since the {NativeModule} dies anyway.
// ref count is not needed, since the {NativeModule} dies anyway. size_t remaining = info->code_to_log.size();
size_t remaining = info->code_to_log.size(); if (remaining > 0) {
if (remaining > 0) { for (size_t i = 0; i < remaining; ++i) {
for (size_t i = 0; i < remaining; ++i) { while (i < remaining &&
while (i < remaining && info->code_to_log[i]->native_module() == native_module) {
info->code_to_log[i]->native_module() == native_module) { // Move the last remaining item to this slot (this can be the same
// Move the last remaining item to this slot (this can be the same // as {i}, which is OK).
// as {i}, which is OK). info->code_to_log[i] = info->code_to_log[--remaining];
info->code_to_log[i] = info->code_to_log[--remaining];
}
} }
info->code_to_log.resize(remaining);
} }
info->code_to_log.resize(remaining);
} }
// If there is a GC running which has references to code contained in the }
// deleted {NativeModule}, remove those references. // If there is a GC running which has references to code contained in the
if (current_gc_info_) { // deleted {NativeModule}, remove those references.
for (auto it = current_gc_info_->dead_code.begin(), if (current_gc_info_) {
end = current_gc_info_->dead_code.end(); for (auto it = current_gc_info_->dead_code.begin(),
it != end;) { end = current_gc_info_->dead_code.end();
if ((*it)->native_module() == native_module) { it != end;) {
it = current_gc_info_->dead_code.erase(it); if ((*it)->native_module() == native_module) {
} else { it = current_gc_info_->dead_code.erase(it);
++it; } else {
} ++it;
} }
TRACE_CODE_GC(
"Native module %p died, reducing dead code objects to %zu.\n",
native_module, current_gc_info_->dead_code.size());
} }
native_modules_.erase(it); TRACE_CODE_GC("Native module %p died, reducing dead code objects to %zu.\n",
native_module, current_gc_info_->dead_code.size());
} }
code_manager_.FreeNativeModule(native_module); native_modules_.erase(it);
} }
namespace { namespace {
......
...@@ -213,7 +213,7 @@ TEST_P(WasmCodeManagerTest, EmptyCase) { ...@@ -213,7 +213,7 @@ TEST_P(WasmCodeManagerTest, EmptyCase) {
CHECK_EQ(0, manager()->committed_code_space()); CHECK_EQ(0, manager()->committed_code_space());
ASSERT_DEATH_IF_SUPPORTED(AllocModule(page_size, GetParam()), ASSERT_DEATH_IF_SUPPORTED(AllocModule(page_size, GetParam()),
"OOM in NativeModule::AllocateForCode commit"); "OOM in wasm code commit");
} }
TEST_P(WasmCodeManagerTest, AllocateAndGoOverLimit) { TEST_P(WasmCodeManagerTest, AllocateAndGoOverLimit) {
...@@ -244,7 +244,7 @@ TEST_P(WasmCodeManagerTest, AllocateAndGoOverLimit) { ...@@ -244,7 +244,7 @@ TEST_P(WasmCodeManagerTest, AllocateAndGoOverLimit) {
// WasmCodeManager). Hence don't check for that part of the OOM message. // WasmCodeManager). Hence don't check for that part of the OOM message.
ASSERT_DEATH_IF_SUPPORTED( ASSERT_DEATH_IF_SUPPORTED(
AddCode(native_module.get(), index++, 1 * kCodeAlignment), AddCode(native_module.get(), index++, 1 * kCodeAlignment),
"OOM in NativeModule::AllocateForCode"); "OOM in wasm code");
} }
TEST_P(WasmCodeManagerTest, TotalLimitIrrespectiveOfModuleCount) { TEST_P(WasmCodeManagerTest, TotalLimitIrrespectiveOfModuleCount) {
...@@ -260,7 +260,7 @@ TEST_P(WasmCodeManagerTest, TotalLimitIrrespectiveOfModuleCount) { ...@@ -260,7 +260,7 @@ TEST_P(WasmCodeManagerTest, TotalLimitIrrespectiveOfModuleCount) {
CHECK_NOT_NULL(code); CHECK_NOT_NULL(code);
ASSERT_DEATH_IF_SUPPORTED( ASSERT_DEATH_IF_SUPPORTED(
AddCode(nm2.get(), 0, 2 * page_size - kJumpTableSize), AddCode(nm2.get(), 0, 2 * page_size - kJumpTableSize),
"OOM in NativeModule::AllocateForCode commit"); "OOM in wasm code commit");
} }
TEST_P(WasmCodeManagerTest, GrowingVsFixedModule) { TEST_P(WasmCodeManagerTest, GrowingVsFixedModule) {
...@@ -275,7 +275,7 @@ TEST_P(WasmCodeManagerTest, GrowingVsFixedModule) { ...@@ -275,7 +275,7 @@ TEST_P(WasmCodeManagerTest, GrowingVsFixedModule) {
// grow. // grow.
ASSERT_DEATH_IF_SUPPORTED( ASSERT_DEATH_IF_SUPPORTED(
AddCode(nm.get(), 0, remaining_space_in_module + kCodeAlignment), AddCode(nm.get(), 0, remaining_space_in_module + kCodeAlignment),
"OOM in NativeModule::AllocateForCode"); "OOM in wasm code reservation");
} else { } else {
// The module grows by one page. One page remains uncommitted. // The module grows by one page. One page remains uncommitted.
WasmCodeRefScope code_ref_scope; WasmCodeRefScope code_ref_scope;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment