a76e7b4b38
This API allows allocating shared memory mappings inside a virtual address space from a platform-specific handle to a shared memory object. This will make it possible to allocate shared memory inside the sandbox, for example as backing memory for ArrayBuffers. Bug: chromium:1218005 Change-Id: I4f1f50baec50734e846496cff78046e4fffe75c5 Cq-Include-Trybots: luci.v8.try:v8_linux64_heap_sandbox_dbg_ng,v8_linux_arm64_sim_heap_sandbox_dbg_ng Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3383777 Reviewed-by: Victor Gomes <victorgomes@chromium.org> Reviewed-by: Michael Lippautz <mlippautz@chromium.org> Commit-Queue: Samuel Groß <saelo@chromium.org> Cr-Commit-Position: refs/heads/main@{#79123}
266 lines
10 KiB
C++
266 lines
10 KiB
C++
// Copyright 2022 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/virtual-address-space.h"
|
|
|
|
#include "src/base/emulated-virtual-address-subspace.h"
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
|
|
|
namespace v8 {
|
|
namespace base {
|
|
|
|
constexpr size_t KB = 1024;
|
|
constexpr size_t MB = KB * 1024;
|
|
|
|
void TestRandomPageAddressGeneration(v8::VirtualAddressSpace* space) {
|
|
space->SetRandomSeed(::testing::FLAGS_gtest_random_seed);
|
|
for (int i = 0; i < 10; i++) {
|
|
Address addr = space->RandomPageAddress();
|
|
EXPECT_GE(addr, space->base());
|
|
EXPECT_LT(addr, space->base() + space->size());
|
|
}
|
|
}
|
|
|
|
void TestBasicPageAllocation(v8::VirtualAddressSpace* space) {
|
|
// Allocation sizes in KB.
|
|
const size_t allocation_sizes[] = {4, 8, 12, 16, 32, 64, 128,
|
|
256, 512, 768, 1024, 768, 512, 256,
|
|
128, 64, 32, 16, 12, 8, 4};
|
|
|
|
std::vector<Address> allocations;
|
|
size_t alignment = space->allocation_granularity();
|
|
for (size_t i = 0; i < arraysize(allocation_sizes); i++) {
|
|
size_t size = allocation_sizes[i] * KB;
|
|
if (!IsAligned(size, space->allocation_granularity())) continue;
|
|
Address allocation =
|
|
space->AllocatePages(VirtualAddressSpace::kNoHint, size, alignment,
|
|
PagePermissions::kReadWrite);
|
|
|
|
ASSERT_NE(kNullAddress, allocation);
|
|
EXPECT_GE(allocation, space->base());
|
|
EXPECT_LT(allocation, space->base() + space->size());
|
|
|
|
allocations.push_back(allocation);
|
|
|
|
// Memory must be writable
|
|
*reinterpret_cast<size_t*>(allocation) = size;
|
|
}
|
|
|
|
// Windows has an allocation granularity of 64KB and macOS could have 16KB, so
|
|
// we won't necessarily have managed to obtain all allocations, but we
|
|
// should've gotten all that are >= 64KB.
|
|
EXPECT_GE(allocations.size(), 11UL);
|
|
|
|
for (Address allocation : allocations) {
|
|
//... and readable
|
|
size_t size = *reinterpret_cast<size_t*>(allocation);
|
|
EXPECT_TRUE(space->FreePages(allocation, size));
|
|
}
|
|
}
|
|
|
|
void TestPageAllocationAlignment(v8::VirtualAddressSpace* space) {
|
|
// In multiples of the allocation_granularity.
|
|
const size_t alignments[] = {1, 2, 4, 8, 16, 32, 64};
|
|
const size_t size = space->allocation_granularity();
|
|
|
|
for (size_t i = 0; i < arraysize(alignments); i++) {
|
|
size_t alignment = alignments[i] * space->allocation_granularity();
|
|
Address allocation =
|
|
space->AllocatePages(VirtualAddressSpace::kNoHint, size, alignment,
|
|
PagePermissions::kReadWrite);
|
|
|
|
ASSERT_NE(kNullAddress, allocation);
|
|
EXPECT_TRUE(IsAligned(allocation, alignment));
|
|
EXPECT_GE(allocation, space->base());
|
|
EXPECT_LT(allocation, space->base() + space->size());
|
|
|
|
EXPECT_TRUE(space->FreePages(allocation, size));
|
|
}
|
|
}
|
|
|
|
void TestParentSpaceCannotAllocateInChildSpace(v8::VirtualAddressSpace* parent,
|
|
v8::VirtualAddressSpace* child) {
|
|
child->SetRandomSeed(::testing::FLAGS_gtest_random_seed);
|
|
|
|
size_t chunksize = parent->allocation_granularity();
|
|
size_t alignment = chunksize;
|
|
Address start = child->base();
|
|
Address end = start + child->size();
|
|
|
|
for (int i = 0; i < 10; i++) {
|
|
Address hint = child->RandomPageAddress();
|
|
Address allocation = parent->AllocatePages(hint, chunksize, alignment,
|
|
PagePermissions::kNoAccess);
|
|
ASSERT_NE(kNullAddress, allocation);
|
|
EXPECT_TRUE(allocation < start || allocation >= end);
|
|
EXPECT_TRUE(parent->FreePages(allocation, chunksize));
|
|
}
|
|
}
|
|
|
|
void TestSharedPageAllocation(v8::VirtualAddressSpace* space) {
|
|
const size_t size = 2 * space->allocation_granularity();
|
|
|
|
PlatformSharedMemoryHandle handle =
|
|
OS::CreateSharedMemoryHandleForTesting(size);
|
|
if (handle == kInvalidSharedMemoryHandle) return;
|
|
|
|
Address mapping1 =
|
|
space->AllocateSharedPages(VirtualAddressSpace::kNoHint, size,
|
|
PagePermissions::kReadWrite, handle, 0);
|
|
ASSERT_NE(kNullAddress, mapping1);
|
|
Address mapping2 =
|
|
space->AllocateSharedPages(VirtualAddressSpace::kNoHint, size,
|
|
PagePermissions::kReadWrite, handle, 0);
|
|
ASSERT_NE(kNullAddress, mapping2);
|
|
ASSERT_NE(mapping1, mapping2);
|
|
|
|
int value = 0x42;
|
|
EXPECT_EQ(0, *reinterpret_cast<int*>(mapping2));
|
|
*reinterpret_cast<int*>(mapping1) = value;
|
|
EXPECT_EQ(value, *reinterpret_cast<int*>(mapping2));
|
|
|
|
EXPECT_TRUE(space->FreeSharedPages(mapping1, size));
|
|
EXPECT_TRUE(space->FreeSharedPages(mapping2, size));
|
|
|
|
OS::DestroySharedMemoryHandle(handle);
|
|
}
|
|
|
|
TEST(VirtualAddressSpaceTest, TestPagePermissionSubsets) {
|
|
const PagePermissions kNoAccess = PagePermissions::kNoAccess;
|
|
const PagePermissions kRead = PagePermissions::kRead;
|
|
const PagePermissions kReadWrite = PagePermissions::kReadWrite;
|
|
const PagePermissions kReadWriteExecute = PagePermissions::kReadWriteExecute;
|
|
const PagePermissions kReadExecute = PagePermissions::kReadExecute;
|
|
|
|
EXPECT_TRUE(IsSubset(kNoAccess, kNoAccess));
|
|
EXPECT_FALSE(IsSubset(kRead, kNoAccess));
|
|
EXPECT_FALSE(IsSubset(kReadWrite, kNoAccess));
|
|
EXPECT_FALSE(IsSubset(kReadWriteExecute, kNoAccess));
|
|
EXPECT_FALSE(IsSubset(kReadExecute, kNoAccess));
|
|
|
|
EXPECT_TRUE(IsSubset(kNoAccess, kRead));
|
|
EXPECT_TRUE(IsSubset(kRead, kRead));
|
|
EXPECT_FALSE(IsSubset(kReadWrite, kRead));
|
|
EXPECT_FALSE(IsSubset(kReadWriteExecute, kRead));
|
|
EXPECT_FALSE(IsSubset(kReadExecute, kRead));
|
|
|
|
EXPECT_TRUE(IsSubset(kNoAccess, kReadWrite));
|
|
EXPECT_TRUE(IsSubset(kRead, kReadWrite));
|
|
EXPECT_TRUE(IsSubset(kReadWrite, kReadWrite));
|
|
EXPECT_FALSE(IsSubset(kReadWriteExecute, kReadWrite));
|
|
EXPECT_FALSE(IsSubset(kReadExecute, kReadWrite));
|
|
|
|
EXPECT_TRUE(IsSubset(kNoAccess, kReadWriteExecute));
|
|
EXPECT_TRUE(IsSubset(kRead, kReadWriteExecute));
|
|
EXPECT_TRUE(IsSubset(kReadWrite, kReadWriteExecute));
|
|
EXPECT_TRUE(IsSubset(kReadWriteExecute, kReadWriteExecute));
|
|
EXPECT_TRUE(IsSubset(kReadExecute, kReadWriteExecute));
|
|
|
|
EXPECT_TRUE(IsSubset(kNoAccess, kReadExecute));
|
|
EXPECT_TRUE(IsSubset(kRead, kReadExecute));
|
|
EXPECT_FALSE(IsSubset(kReadWrite, kReadExecute));
|
|
EXPECT_FALSE(IsSubset(kReadWriteExecute, kReadExecute));
|
|
EXPECT_TRUE(IsSubset(kReadExecute, kReadExecute));
|
|
}
|
|
|
|
TEST(VirtualAddressSpaceTest, TestRootSpace) {
|
|
VirtualAddressSpace rootspace;
|
|
|
|
TestRandomPageAddressGeneration(&rootspace);
|
|
TestBasicPageAllocation(&rootspace);
|
|
TestPageAllocationAlignment(&rootspace);
|
|
TestSharedPageAllocation(&rootspace);
|
|
}
|
|
|
|
TEST(VirtualAddressSpaceTest, TestSubspace) {
|
|
constexpr size_t kSubspaceSize = 32 * MB;
|
|
constexpr size_t kSubSubspaceSize = 16 * MB;
|
|
|
|
VirtualAddressSpace rootspace;
|
|
|
|
if (!rootspace.CanAllocateSubspaces()) return;
|
|
size_t subspace_alignment = rootspace.allocation_granularity();
|
|
auto subspace = rootspace.AllocateSubspace(VirtualAddressSpace::kNoHint,
|
|
kSubspaceSize, subspace_alignment,
|
|
PagePermissions::kReadWrite);
|
|
ASSERT_TRUE(subspace);
|
|
EXPECT_NE(kNullAddress, subspace->base());
|
|
EXPECT_EQ(kSubspaceSize, subspace->size());
|
|
EXPECT_EQ(PagePermissions::kReadWrite, subspace->max_page_permissions());
|
|
|
|
TestRandomPageAddressGeneration(subspace.get());
|
|
TestBasicPageAllocation(subspace.get());
|
|
TestPageAllocationAlignment(subspace.get());
|
|
TestParentSpaceCannotAllocateInChildSpace(&rootspace, subspace.get());
|
|
TestSharedPageAllocation(subspace.get());
|
|
|
|
// Test sub-subspaces
|
|
if (!subspace->CanAllocateSubspaces()) return;
|
|
size_t subsubspace_alignment = subspace->allocation_granularity();
|
|
auto subsubspace = subspace->AllocateSubspace(
|
|
VirtualAddressSpace::kNoHint, kSubSubspaceSize, subsubspace_alignment,
|
|
PagePermissions::kReadWrite);
|
|
ASSERT_TRUE(subsubspace);
|
|
EXPECT_NE(kNullAddress, subsubspace->base());
|
|
EXPECT_EQ(kSubSubspaceSize, subsubspace->size());
|
|
EXPECT_EQ(PagePermissions::kReadWrite, subsubspace->max_page_permissions());
|
|
|
|
TestRandomPageAddressGeneration(subsubspace.get());
|
|
TestBasicPageAllocation(subsubspace.get());
|
|
TestPageAllocationAlignment(subsubspace.get());
|
|
TestParentSpaceCannotAllocateInChildSpace(subspace.get(), subsubspace.get());
|
|
TestSharedPageAllocation(subsubspace.get());
|
|
}
|
|
|
|
TEST(VirtualAddressSpaceTest, TestEmulatedSubspace) {
|
|
constexpr size_t kSubspaceSize = 32 * MB;
|
|
// Size chosen so page allocation tests will obtain pages in both the mapped
|
|
// and the unmapped region.
|
|
constexpr size_t kSubspaceMappedSize = 1 * MB;
|
|
|
|
VirtualAddressSpace rootspace;
|
|
|
|
size_t subspace_alignment = rootspace.allocation_granularity();
|
|
ASSERT_TRUE(
|
|
IsAligned(kSubspaceMappedSize, rootspace.allocation_granularity()));
|
|
Address reservation = kNullAddress;
|
|
for (int i = 0; i < 10; i++) {
|
|
// Reserve the full size first at a random address, then free it again to
|
|
// ensure that there's enough free space behind the final reservation.
|
|
Address hint = rootspace.RandomPageAddress();
|
|
reservation = rootspace.AllocatePages(hint, kSubspaceSize,
|
|
rootspace.allocation_granularity(),
|
|
PagePermissions::kNoAccess);
|
|
ASSERT_NE(kNullAddress, reservation);
|
|
hint = reservation;
|
|
EXPECT_TRUE(rootspace.FreePages(reservation, kSubspaceSize));
|
|
reservation =
|
|
rootspace.AllocatePages(hint, kSubspaceMappedSize, subspace_alignment,
|
|
PagePermissions::kNoAccess);
|
|
if (reservation == hint) {
|
|
break;
|
|
} else {
|
|
EXPECT_TRUE(rootspace.FreePages(reservation, kSubspaceMappedSize));
|
|
reservation = kNullAddress;
|
|
}
|
|
}
|
|
ASSERT_NE(kNullAddress, reservation);
|
|
|
|
EmulatedVirtualAddressSubspace subspace(&rootspace, reservation,
|
|
kSubspaceMappedSize, kSubspaceSize);
|
|
EXPECT_EQ(reservation, subspace.base());
|
|
EXPECT_EQ(kSubspaceSize, subspace.size());
|
|
EXPECT_EQ(rootspace.max_page_permissions(), subspace.max_page_permissions());
|
|
|
|
TestRandomPageAddressGeneration(&subspace);
|
|
TestBasicPageAllocation(&subspace);
|
|
TestPageAllocationAlignment(&subspace);
|
|
// An emulated subspace does *not* guarantee that the parent space cannot
|
|
// allocate pages in it, so no TestParentSpaceCannotAllocateInChildSpace.
|
|
TestSharedPageAllocation(&subspace);
|
|
}
|
|
|
|
} // namespace base
|
|
} // namespace v8
|