blob: 1dac872449b9950ca2066e171d40a2a8a33b2b3b [file] [log] [blame]
// Copyright 2017 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/base/page-allocator.h"
#include "src/base/platform/platform.h"
#include "src/base/platform/wrappers.h"
#if V8_OS_MACOSX
#include <sys/mman.h> // For MAP_JIT.
#endif
namespace v8 {
namespace base {
#define STATIC_ASSERT_ENUM(a, b) \
static_assert(static_cast<int>(a) == static_cast<int>(b), \
"mismatching enum: " #a)
STATIC_ASSERT_ENUM(PageAllocator::kNoAccess,
base::OS::MemoryPermission::kNoAccess);
STATIC_ASSERT_ENUM(PageAllocator::kReadWrite,
base::OS::MemoryPermission::kReadWrite);
STATIC_ASSERT_ENUM(PageAllocator::kReadWriteExecute,
base::OS::MemoryPermission::kReadWriteExecute);
STATIC_ASSERT_ENUM(PageAllocator::kReadExecute,
base::OS::MemoryPermission::kReadExecute);
STATIC_ASSERT_ENUM(PageAllocator::kNoAccessWillJitLater,
base::OS::MemoryPermission::kNoAccessWillJitLater);
#undef STATIC_ASSERT_ENUM
PageAllocator::PageAllocator()
: allocate_page_size_(base::OS::AllocatePageSize()),
commit_page_size_(base::OS::CommitPageSize()) {}
void PageAllocator::SetRandomMmapSeed(int64_t seed) {
base::OS::SetRandomMmapSeed(seed);
}
void* PageAllocator::GetRandomMmapAddr() {
return base::OS::GetRandomMmapAddr();
}
void* PageAllocator::AllocatePages(void* hint, size_t size, size_t alignment,
PageAllocator::Permission access) {
#if !(V8_OS_MACOSX && V8_HOST_ARCH_ARM64 && defined(MAP_JIT))
// kNoAccessWillJitLater is only used on Apple Silicon. Map it to regular
// kNoAccess on other platforms, so code doesn't have to handle both enum
// values.
if (access == PageAllocator::kNoAccessWillJitLater) {
access = PageAllocator::kNoAccess;
}
#endif
return base::OS::Allocate(hint, size, alignment,
static_cast<base::OS::MemoryPermission>(access));
}
class SharedMemoryMapping : public ::v8::PageAllocator::SharedMemoryMapping {
public:
explicit SharedMemoryMapping(PageAllocator* page_allocator, void* ptr,
size_t size)
: page_allocator_(page_allocator), ptr_(ptr), size_(size) {}
~SharedMemoryMapping() override { page_allocator_->FreePages(ptr_, size_); }
void* GetMemory() const override { return ptr_; }
private:
PageAllocator* page_allocator_;
void* ptr_;
size_t size_;
};
class SharedMemory : public ::v8::PageAllocator::SharedMemory {
public:
SharedMemory(PageAllocator* allocator, void* memory, size_t size)
: allocator_(allocator), ptr_(memory), size_(size) {}
void* GetMemory() const override { return ptr_; }
size_t GetSize() const override { return size_; }
std::unique_ptr<::v8::PageAllocator::SharedMemoryMapping> RemapTo(
void* new_address) const override {
if (allocator_->RemapShared(ptr_, new_address, size_)) {
return std::make_unique<SharedMemoryMapping>(allocator_, new_address,
size_);
} else {
return {};
}
}
~SharedMemory() override { allocator_->FreePages(ptr_, size_); }
private:
PageAllocator* allocator_;
void* ptr_;
size_t size_;
};
bool PageAllocator::CanAllocateSharedPages() {
#ifdef V8_OS_LINUX
return true;
#else
return false;
#endif
}
std::unique_ptr<v8::PageAllocator::SharedMemory>
PageAllocator::AllocateSharedPages(size_t size, const void* original_address) {
#ifdef V8_OS_LINUX
void* ptr =
base::OS::AllocateShared(size, base::OS::MemoryPermission::kReadWrite);
CHECK_NOT_NULL(ptr);
base::Memcpy(ptr, original_address, size);
bool success = base::OS::SetPermissions(
ptr, size, base::OS::MemoryPermission::kReadWrite);
CHECK(success);
auto shared_memory =
std::make_unique<v8::base::SharedMemory>(this, ptr, size);
return shared_memory;
#else
return {};
#endif
}
void* PageAllocator::RemapShared(void* old_address, void* new_address,
size_t size) {
#ifdef V8_OS_LINUX
return base::OS::RemapShared(old_address, new_address, size);
#else
return nullptr;
#endif
}
bool PageAllocator::FreePages(void* address, size_t size) {
return base::OS::Free(address, size);
}
bool PageAllocator::ReleasePages(void* address, size_t size, size_t new_size) {
DCHECK_LT(new_size, size);
return base::OS::Release(reinterpret_cast<uint8_t*>(address) + new_size,
size - new_size);
}
bool PageAllocator::SetPermissions(void* address, size_t size,
PageAllocator::Permission access) {
return base::OS::SetPermissions(
address, size, static_cast<base::OS::MemoryPermission>(access));
}
bool PageAllocator::DiscardSystemPages(void* address, size_t size) {
return base::OS::DiscardSystemPages(address, size);
}
} // namespace base
} // namespace v8