2017-12-15 17:59:57 +00:00
|
|
|
// 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/allocation.h"
|
|
|
|
|
|
|
|
#if V8_OS_POSIX
|
|
|
|
#include <setjmp.h>
|
|
|
|
#include <signal.h>
|
|
|
|
#include <unistd.h> // NOLINT
|
|
|
|
#endif // V8_OS_POSIX
|
|
|
|
|
|
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
// TODO(eholk): Add a windows version of permissions tests.
|
|
|
|
#if V8_OS_POSIX
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
// These tests make sure the routines to allocate memory do so with the correct
|
|
|
|
// permissions.
|
|
|
|
//
|
|
|
|
// Unfortunately, there is no API to find the protection of a memory address,
|
|
|
|
// so instead we test permissions by installing a signal handler, probing a
|
|
|
|
// memory location and recovering from the fault.
|
|
|
|
//
|
|
|
|
// We don't test the execution permission because to do so we'd have to
|
|
|
|
// dynamically generate code and test if we can execute it.
|
|
|
|
|
|
|
|
class MemoryAllocationPermissionsTest : public ::testing::Test {
|
|
|
|
static void SignalHandler(int signal, siginfo_t* info, void*) {
|
|
|
|
siglongjmp(continuation_, 1);
|
|
|
|
}
|
|
|
|
struct sigaction old_action_;
|
|
|
|
// On Mac, sometimes we get SIGBUS instead of SIGSEGV.
|
|
|
|
#if V8_OS_MACOSX
|
|
|
|
struct sigaction old_bus_action_;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
protected:
|
2018-09-14 15:34:02 +00:00
|
|
|
void SetUp() override {
|
2017-12-15 17:59:57 +00:00
|
|
|
struct sigaction action;
|
|
|
|
action.sa_sigaction = SignalHandler;
|
|
|
|
sigemptyset(&action.sa_mask);
|
|
|
|
action.sa_flags = SA_SIGINFO;
|
|
|
|
sigaction(SIGSEGV, &action, &old_action_);
|
|
|
|
#if V8_OS_MACOSX
|
|
|
|
sigaction(SIGBUS, &action, &old_bus_action_);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2018-09-14 15:34:02 +00:00
|
|
|
void TearDown() override {
|
2017-12-15 17:59:57 +00:00
|
|
|
// Be a good citizen and restore the old signal handler.
|
|
|
|
sigaction(SIGSEGV, &old_action_, nullptr);
|
|
|
|
#if V8_OS_MACOSX
|
|
|
|
sigaction(SIGBUS, &old_bus_action_, nullptr);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
static sigjmp_buf continuation_;
|
|
|
|
|
|
|
|
enum class MemoryAction { kRead, kWrite };
|
|
|
|
|
|
|
|
void ProbeMemory(volatile int* buffer, MemoryAction action,
|
|
|
|
bool should_succeed) {
|
|
|
|
const int save_sigs = 1;
|
|
|
|
if (!sigsetjmp(continuation_, save_sigs)) {
|
|
|
|
switch (action) {
|
|
|
|
case MemoryAction::kRead: {
|
|
|
|
// static_cast to remove the reference and force a memory read.
|
|
|
|
USE(static_cast<int>(*buffer));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case MemoryAction::kWrite: {
|
|
|
|
*buffer = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (should_succeed) {
|
|
|
|
SUCCEED();
|
|
|
|
} else {
|
|
|
|
FAIL();
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (should_succeed) {
|
|
|
|
FAIL();
|
|
|
|
} else {
|
|
|
|
SUCCEED();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-21 18:09:26 +00:00
|
|
|
void TestPermissions(PageAllocator::Permission permission, bool can_read,
|
2017-12-15 17:59:57 +00:00
|
|
|
bool can_write) {
|
2018-09-06 13:44:07 +00:00
|
|
|
v8::PageAllocator* page_allocator =
|
|
|
|
v8::internal::GetPlatformPageAllocator();
|
|
|
|
const size_t page_size = page_allocator->AllocatePageSize();
|
|
|
|
int* buffer = static_cast<int*>(AllocatePages(
|
|
|
|
page_allocator, nullptr, page_size, page_size, permission));
|
2017-12-15 17:59:57 +00:00
|
|
|
ProbeMemory(buffer, MemoryAction::kRead, can_read);
|
|
|
|
ProbeMemory(buffer, MemoryAction::kWrite, can_write);
|
2018-09-06 13:44:07 +00:00
|
|
|
CHECK(FreePages(page_allocator, buffer, page_size));
|
2017-12-15 17:59:57 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
sigjmp_buf MemoryAllocationPermissionsTest::continuation_;
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
TEST_F(MemoryAllocationPermissionsTest, DoTest) {
|
2017-12-21 18:09:26 +00:00
|
|
|
TestPermissions(PageAllocator::Permission::kNoAccess, false, false);
|
2018-04-16 12:06:11 +00:00
|
|
|
TestPermissions(PageAllocator::Permission::kRead, true, false);
|
2017-12-21 18:09:26 +00:00
|
|
|
TestPermissions(PageAllocator::Permission::kReadWrite, true, true);
|
2018-01-24 11:50:28 +00:00
|
|
|
TestPermissions(PageAllocator::Permission::kReadWriteExecute, true, true);
|
2018-02-16 20:54:22 +00:00
|
|
|
TestPermissions(PageAllocator::Permission::kReadExecute, true, false);
|
2017-12-15 17:59:57 +00:00
|
|
|
}
|
|
|
|
#endif // V8_OS_POSIX
|
|
|
|
|
|
|
|
// Basic tests of allocation.
|
|
|
|
|
|
|
|
class AllocationTest : public ::testing::Test {};
|
|
|
|
|
|
|
|
TEST(AllocationTest, AllocateAndFree) {
|
|
|
|
size_t page_size = v8::internal::AllocatePageSize();
|
|
|
|
CHECK_NE(0, page_size);
|
|
|
|
|
2018-09-06 13:44:07 +00:00
|
|
|
v8::PageAllocator* page_allocator = v8::internal::GetPlatformPageAllocator();
|
|
|
|
|
2017-12-15 17:59:57 +00:00
|
|
|
// A large allocation, aligned at native allocation granularity.
|
|
|
|
const size_t kAllocationSize = 1 * v8::internal::MB;
|
|
|
|
void* mem_addr = v8::internal::AllocatePages(
|
2018-09-06 13:44:07 +00:00
|
|
|
page_allocator, page_allocator->GetRandomMmapAddr(), kAllocationSize,
|
|
|
|
page_size, PageAllocator::Permission::kReadWrite);
|
2017-12-15 17:59:57 +00:00
|
|
|
CHECK_NOT_NULL(mem_addr);
|
2018-09-06 13:44:07 +00:00
|
|
|
CHECK(v8::internal::FreePages(page_allocator, mem_addr, kAllocationSize));
|
2017-12-15 17:59:57 +00:00
|
|
|
|
|
|
|
// A large allocation, aligned significantly beyond native granularity.
|
|
|
|
const size_t kBigAlignment = 64 * v8::internal::MB;
|
|
|
|
void* aligned_mem_addr = v8::internal::AllocatePages(
|
2018-09-06 13:44:07 +00:00
|
|
|
page_allocator,
|
|
|
|
AlignedAddress(page_allocator->GetRandomMmapAddr(), kBigAlignment),
|
2018-02-01 21:44:24 +00:00
|
|
|
kAllocationSize, kBigAlignment, PageAllocator::Permission::kReadWrite);
|
2017-12-15 17:59:57 +00:00
|
|
|
CHECK_NOT_NULL(aligned_mem_addr);
|
|
|
|
CHECK_EQ(aligned_mem_addr, AlignedAddress(aligned_mem_addr, kBigAlignment));
|
2018-09-06 13:44:07 +00:00
|
|
|
CHECK(v8::internal::FreePages(page_allocator, aligned_mem_addr,
|
|
|
|
kAllocationSize));
|
2017-12-15 17:59:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TEST(AllocationTest, ReserveMemory) {
|
2018-09-06 13:44:07 +00:00
|
|
|
v8::PageAllocator* page_allocator = v8::internal::GetPlatformPageAllocator();
|
2017-12-15 17:59:57 +00:00
|
|
|
size_t page_size = v8::internal::AllocatePageSize();
|
|
|
|
const size_t kAllocationSize = 1 * v8::internal::MB;
|
|
|
|
void* mem_addr = v8::internal::AllocatePages(
|
2018-09-06 13:44:07 +00:00
|
|
|
page_allocator, page_allocator->GetRandomMmapAddr(), kAllocationSize,
|
|
|
|
page_size, PageAllocator::Permission::kReadWrite);
|
2017-12-15 17:59:57 +00:00
|
|
|
CHECK_NE(0, page_size);
|
|
|
|
CHECK_NOT_NULL(mem_addr);
|
2018-09-06 13:44:07 +00:00
|
|
|
size_t commit_size = page_allocator->CommitPageSize();
|
|
|
|
CHECK(v8::internal::SetPermissions(page_allocator, mem_addr, commit_size,
|
2017-12-21 18:09:26 +00:00
|
|
|
PageAllocator::Permission::kReadWrite));
|
2017-12-15 17:59:57 +00:00
|
|
|
// Check whether we can write to memory.
|
|
|
|
int* addr = static_cast<int*>(mem_addr);
|
|
|
|
addr[v8::internal::KB - 1] = 2;
|
2018-09-06 13:44:07 +00:00
|
|
|
CHECK(v8::internal::SetPermissions(page_allocator, mem_addr, commit_size,
|
2017-12-21 18:09:26 +00:00
|
|
|
PageAllocator::Permission::kNoAccess));
|
2018-09-06 13:44:07 +00:00
|
|
|
CHECK(v8::internal::FreePages(page_allocator, mem_addr, kAllocationSize));
|
2017-12-15 17:59:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|