// Copyright 2018 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/snapshot/embedded/embedded-data.h" #include "src/codegen/assembler-inl.h" #include "src/codegen/callable.h" #include "src/objects/objects-inl.h" #include "src/snapshot/snapshot-utils.h" #include "src/snapshot/snapshot.h" namespace v8 { namespace internal { // static bool InstructionStream::PcIsOffHeap(Isolate* isolate, Address pc) { const Address start = reinterpret_cast
(isolate->embedded_blob_code()); return start <= pc && pc < start + isolate->embedded_blob_code_size(); } // static Code InstructionStream::TryLookupCode(Isolate* isolate, Address address) { if (!PcIsOffHeap(isolate, address)) return Code(); EmbeddedData d = EmbeddedData::FromBlob(); if (address < d.InstructionStartOfBuiltin(0)) return Code(); // Note: Addresses within the padding section between builtins (i.e. within // start + size <= address < start + padded_size) are interpreted as belonging // to the preceding builtin. int l = 0, r = Builtins::builtin_count; while (l < r) { const int mid = (l + r) / 2; Address start = d.InstructionStartOfBuiltin(mid); Address end = start + d.PaddedInstructionSizeOfBuiltin(mid); if (address < start) { r = mid; } else if (address >= end) { l = mid + 1; } else { return isolate->builtins()->builtin(mid); } } UNREACHABLE(); } // static void InstructionStream::CreateOffHeapInstructionStream(Isolate* isolate, uint8_t** code, uint32_t* code_size, uint8_t** data, uint32_t* data_size) { // Create the embedded blob from scratch using the current Isolate's heap. EmbeddedData d = EmbeddedData::FromIsolate(isolate); // Allocate the backing store that will contain the embedded blob in this // Isolate. The backing store is on the native heap, *not* on V8's garbage- // collected heap. v8::PageAllocator* page_allocator = v8::internal::GetPlatformPageAllocator(); const uint32_t alignment = static_cast(page_allocator->AllocatePageSize()); void* const requested_allocation_code_address = AlignedAddress(isolate->heap()->GetRandomMmapAddr(), alignment); const uint32_t allocation_code_size = RoundUp(d.code_size(), alignment); uint8_t* allocated_code_bytes = static_cast(AllocatePages( page_allocator, requested_allocation_code_address, allocation_code_size, alignment, PageAllocator::kReadWrite)); CHECK_NOT_NULL(allocated_code_bytes); void* const requested_allocation_data_address = AlignedAddress(isolate->heap()->GetRandomMmapAddr(), alignment); const uint32_t allocation_data_size = RoundUp(d.data_size(), alignment); uint8_t* allocated_data_bytes = static_cast(AllocatePages( page_allocator, requested_allocation_data_address, allocation_data_size, alignment, PageAllocator::kReadWrite)); CHECK_NOT_NULL(allocated_data_bytes); // Copy the embedded blob into the newly allocated backing store. Switch // permissions to read-execute since builtin code is immutable from now on // and must be executable in case any JS execution is triggered. // // Once this backing store is set as the current_embedded_blob, V8 cannot tell // the difference between a 'real' embedded build (where the blob is embedded // in the binary) and what we are currently setting up here (where the blob is // on the native heap). std::memcpy(allocated_code_bytes, d.code(), d.code_size()); CHECK(SetPermissions(page_allocator, allocated_code_bytes, allocation_code_size, PageAllocator::kReadExecute)); std::memcpy(allocated_data_bytes, d.data(), d.data_size()); CHECK(SetPermissions(page_allocator, allocated_data_bytes, allocation_data_size, PageAllocator::kRead)); *code = allocated_code_bytes; *code_size = d.code_size(); *data = allocated_data_bytes; *data_size = d.data_size(); d.Dispose(); } // static void InstructionStream::FreeOffHeapInstructionStream(uint8_t* code, uint32_t code_size, uint8_t* data, uint32_t data_size) { v8::PageAllocator* page_allocator = v8::internal::GetPlatformPageAllocator(); const uint32_t page_size = static_cast(page_allocator->AllocatePageSize()); CHECK(FreePages(page_allocator, code, RoundUp(code_size, page_size))); CHECK(FreePages(page_allocator, data, RoundUp(data_size, page_size))); } namespace { bool BuiltinAliasesOffHeapTrampolineRegister(Isolate* isolate, Code code) { DCHECK(Builtins::IsIsolateIndependent(code.builtin_index())); switch (Builtins::KindOf(code.builtin_index())) { case Builtins::CPP: case Builtins::TFC: case Builtins::TFH: case Builtins::TFJ: case Builtins::TFS: break; // Bytecode handlers will only ever be used by the interpreter and so there // will never be a need to use trampolines with them. case Builtins::BCH: case Builtins::ASM: // TODO(jgruber): Extend checks to remaining kinds. return false; } Callable callable = Builtins::CallableFor( isolate, static_cast(code.builtin_index())); CallInterfaceDescriptor descriptor = callable.descriptor(); if (descriptor.ContextRegister() == kOffHeapTrampolineRegister) { return true; } for (int i = 0; i < descriptor.GetRegisterParameterCount(); i++) { Register reg = descriptor.GetRegisterParameter(i); if (reg == kOffHeapTrampolineRegister) return true; } return false; } void FinalizeEmbeddedCodeTargets(Isolate* isolate, EmbeddedData* blob) { static const int kRelocMask = RelocInfo::ModeMask(RelocInfo::CODE_TARGET) | RelocInfo::ModeMask(RelocInfo::RELATIVE_CODE_TARGET); STATIC_ASSERT(Builtins::kAllBuiltinsAreIsolateIndependent); for (int i = 0; i < Builtins::builtin_count; i++) { Code code = isolate->builtins()->builtin(i); RelocIterator on_heap_it(code, kRelocMask); RelocIterator off_heap_it(blob, code, kRelocMask); #if defined(V8_TARGET_ARCH_X64) || defined(V8_TARGET_ARCH_ARM64) || \ defined(V8_TARGET_ARCH_ARM) || defined(V8_TARGET_ARCH_MIPS) || \ defined(V8_TARGET_ARCH_IA32) || defined(V8_TARGET_ARCH_S390) // On these platforms we emit relative builtin-to-builtin // jumps for isolate independent builtins in the snapshot. This fixes up the // relative jumps to the right offsets in the snapshot. // See also: Code::IsIsolateIndependent. while (!on_heap_it.done()) { DCHECK(!off_heap_it.done()); RelocInfo* rinfo = on_heap_it.rinfo(); DCHECK_EQ(rinfo->rmode(), off_heap_it.rinfo()->rmode()); Code target = Code::GetCodeFromTargetAddress(rinfo->target_address()); CHECK(Builtins::IsIsolateIndependentBuiltin(target)); // Do not emit write-barrier for off-heap writes. off_heap_it.rinfo()->set_target_address( blob->InstructionStartOfBuiltin(target.builtin_index()), SKIP_WRITE_BARRIER); on_heap_it.next(); off_heap_it.next(); } DCHECK(off_heap_it.done()); #else // Architectures other than x64 and arm/arm64 do not use pc-relative calls // and thus must not contain embedded code targets. Instead, we use an // indirection through the root register. CHECK(on_heap_it.done()); CHECK(off_heap_it.done()); #endif // defined(V8_TARGET_ARCH_X64) || defined(V8_TARGET_ARCH_ARM64) } } } // namespace // static EmbeddedData EmbeddedData::FromIsolate(Isolate* isolate) { Builtins* builtins = isolate->builtins(); // Store instruction stream lengths and offsets. std::vector layout_descriptions(kTableSize); bool saw_unsafe_builtin = false; uint32_t raw_code_size = 0; uint32_t raw_data_size = 0; STATIC_ASSERT(Builtins::kAllBuiltinsAreIsolateIndependent); for (int i = 0; i < Builtins::builtin_count; i++) { Code code = builtins->builtin(i); // Sanity-check that the given builtin is isolate-independent and does not // use the trampoline register in its calling convention. if (!code.IsIsolateIndependent(isolate)) { saw_unsafe_builtin = true; fprintf(stderr, "%s is not isolate-independent.\n", Builtins::name(i)); } if (BuiltinAliasesOffHeapTrampolineRegister(isolate, code)) { saw_unsafe_builtin = true; fprintf(stderr, "%s aliases the off-heap trampoline register.\n", Builtins::name(i)); } uint32_t instruction_size = static_cast(code.raw_instruction_size()); uint32_t metadata_size = static_cast(code.raw_metadata_size()); DCHECK_EQ(0, raw_code_size % kCodeAlignment); layout_descriptions[i].instruction_offset = raw_code_size; layout_descriptions[i].instruction_length = instruction_size; layout_descriptions[i].metadata_offset = raw_data_size; layout_descriptions[i].metadata_length = metadata_size; // Align the start of each section. raw_code_size += PadAndAlignCode(instruction_size); raw_data_size += PadAndAlignData(metadata_size); } CHECK_WITH_MSG( !saw_unsafe_builtin, "One or more builtins marked as isolate-independent either contains " "isolate-dependent code or aliases the off-heap trampoline register. " "If in doubt, ask jgruber@"); // Allocate space for the code section, value-initialized to 0. STATIC_ASSERT(RawCodeOffset() == 0); const uint32_t blob_code_size = RawCodeOffset() + raw_code_size; uint8_t* const blob_code = new uint8_t[blob_code_size](); // Allocate space for the data section, value-initialized to 0. STATIC_ASSERT(IsAligned(FixedDataSize(), Code::kMetadataAlignment)); const uint32_t blob_data_size = FixedDataSize() + raw_data_size; uint8_t* const blob_data = new uint8_t[blob_data_size](); // Initially zap the entire blob, effectively padding the alignment area // between two builtins with int3's (on x64/ia32). ZapCode(reinterpret_cast
(blob_code), blob_code_size); // Hash relevant parts of the Isolate's heap and store the result. { STATIC_ASSERT(IsolateHashSize() == kSizetSize); const size_t hash = isolate->HashIsolateForEmbeddedBlob(); std::memcpy(blob_data + IsolateHashOffset(), &hash, IsolateHashSize()); } // Write the layout_descriptions tables. DCHECK_EQ(LayoutDescriptionTableSize(), sizeof(layout_descriptions[0]) * layout_descriptions.size()); std::memcpy(blob_data + LayoutDescriptionTableOffset(), layout_descriptions.data(), LayoutDescriptionTableSize()); // .. and the variable-size data section. uint8_t* const raw_metadata_start = blob_data + RawMetadataOffset(); STATIC_ASSERT(Builtins::kAllBuiltinsAreIsolateIndependent); for (int i = 0; i < Builtins::builtin_count; i++) { Code code = builtins->builtin(i); uint32_t offset = layout_descriptions[i].metadata_offset; uint8_t* dst = raw_metadata_start + offset; DCHECK_LE(RawMetadataOffset() + offset + code.raw_metadata_size(), blob_data_size); std::memcpy(dst, reinterpret_cast(code.raw_metadata_start()), code.raw_metadata_size()); } // .. and the variable-size code section. uint8_t* const raw_code_start = blob_code + RawCodeOffset(); STATIC_ASSERT(Builtins::kAllBuiltinsAreIsolateIndependent); for (int i = 0; i < Builtins::builtin_count; i++) { Code code = builtins->builtin(i); uint32_t offset = layout_descriptions[i].instruction_offset; uint8_t* dst = raw_code_start + offset; DCHECK_LE(RawCodeOffset() + offset + code.raw_instruction_size(), blob_code_size); std::memcpy(dst, reinterpret_cast(code.raw_instruction_start()), code.raw_instruction_size()); } EmbeddedData d(blob_code, blob_code_size, blob_data, blob_data_size); // Fix up call targets that point to other embedded builtins. FinalizeEmbeddedCodeTargets(isolate, &d); // Hash the blob and store the result. { STATIC_ASSERT(EmbeddedBlobDataHashSize() == kSizetSize); const size_t data_hash = d.CreateEmbeddedBlobDataHash(); std::memcpy(blob_data + EmbeddedBlobDataHashOffset(), &data_hash, EmbeddedBlobDataHashSize()); STATIC_ASSERT(EmbeddedBlobCodeHashSize() == kSizetSize); const size_t code_hash = d.CreateEmbeddedBlobCodeHash(); std::memcpy(blob_data + EmbeddedBlobCodeHashOffset(), &code_hash, EmbeddedBlobCodeHashSize()); DCHECK_EQ(data_hash, d.CreateEmbeddedBlobDataHash()); DCHECK_EQ(data_hash, d.EmbeddedBlobDataHash()); DCHECK_EQ(code_hash, d.CreateEmbeddedBlobCodeHash()); DCHECK_EQ(code_hash, d.EmbeddedBlobCodeHash()); } if (FLAG_serialization_statistics) d.PrintStatistics(); return d; } Address EmbeddedData::InstructionStartOfBuiltin(int i) const { DCHECK(Builtins::IsBuiltinId(i)); const struct LayoutDescription* descs = LayoutDescription(); const uint8_t* result = RawCode() + descs[i].instruction_offset; DCHECK_LT(result, code_ + code_size_); return reinterpret_cast
(result); } uint32_t EmbeddedData::InstructionSizeOfBuiltin(int i) const { DCHECK(Builtins::IsBuiltinId(i)); const struct LayoutDescription* descs = LayoutDescription(); return descs[i].instruction_length; } Address EmbeddedData::MetadataStartOfBuiltin(int i) const { DCHECK(Builtins::IsBuiltinId(i)); const struct LayoutDescription* descs = LayoutDescription(); const uint8_t* result = RawMetadata() + descs[i].metadata_offset; DCHECK_LE(descs[i].metadata_offset, data_size_); return reinterpret_cast
(result); } uint32_t EmbeddedData::MetadataSizeOfBuiltin(int i) const { DCHECK(Builtins::IsBuiltinId(i)); const struct LayoutDescription* descs = LayoutDescription(); return descs[i].metadata_length; } Address EmbeddedData::InstructionStartOfBytecodeHandlers() const { return InstructionStartOfBuiltin(Builtins::kFirstBytecodeHandler); } Address EmbeddedData::InstructionEndOfBytecodeHandlers() const { STATIC_ASSERT(Builtins::kFirstBytecodeHandler + kNumberOfBytecodeHandlers + 2 * kNumberOfWideBytecodeHandlers == Builtins::builtin_count); int lastBytecodeHandler = Builtins::builtin_count - 1; return InstructionStartOfBuiltin(lastBytecodeHandler) + InstructionSizeOfBuiltin(lastBytecodeHandler); } size_t EmbeddedData::CreateEmbeddedBlobDataHash() const { STATIC_ASSERT(EmbeddedBlobDataHashOffset() == 0); STATIC_ASSERT(EmbeddedBlobCodeHashOffset() == EmbeddedBlobDataHashSize()); STATIC_ASSERT(IsolateHashOffset() == EmbeddedBlobCodeHashOffset() + EmbeddedBlobCodeHashSize()); static constexpr uint32_t kFirstHashedDataOffset = IsolateHashOffset(); // Hash the entire data section except the embedded blob hash fields // themselves. Vector payload(data_ + kFirstHashedDataOffset, data_size_ - kFirstHashedDataOffset); return Checksum(payload); } size_t EmbeddedData::CreateEmbeddedBlobCodeHash() const { CHECK(FLAG_text_is_readable); Vector payload(code_, code_size_); return Checksum(payload); } void EmbeddedData::PrintStatistics() const { DCHECK(FLAG_serialization_statistics); constexpr int kCount = Builtins::builtin_count; int sizes[kCount]; STATIC_ASSERT(Builtins::kAllBuiltinsAreIsolateIndependent); for (int i = 0; i < kCount; i++) { sizes[i] = InstructionSizeOfBuiltin(i); } // Sort for percentiles. std::sort(&sizes[0], &sizes[kCount]); const int k50th = kCount * 0.5; const int k75th = kCount * 0.75; const int k90th = kCount * 0.90; const int k99th = kCount * 0.99; PrintF("EmbeddedData:\n"); PrintF(" Total size: %d\n", static_cast(code_size() + data_size())); PrintF(" Data size: %d\n", static_cast(data_size())); PrintF(" Code size: %d\n", static_cast(code_size())); PrintF(" Instruction size (50th percentile): %d\n", sizes[k50th]); PrintF(" Instruction size (75th percentile): %d\n", sizes[k75th]); PrintF(" Instruction size (90th percentile): %d\n", sizes[k90th]); PrintF(" Instruction size (99th percentile): %d\n", sizes[k99th]); PrintF("\n"); } } // namespace internal } // namespace v8