// Copyright 2017 The Chromium 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 "base/memory/protected_memory.h" #include #include #include #if defined(OS_LINUX) #include #endif // defined(OS_LINUX) #if defined(OS_MACOSX) && !defined(OS_IOS) #include #include #endif // defined(OS_MACOSX) && !defined(OS_IOS) #include "base/posix/eintr_wrapper.h" #include "base/synchronization/lock.h" #include "build/build_config.h" namespace base { #if !defined(COMPONENT_BUILD) PROTECTED_MEMORY_SECTION int AutoWritableMemory::writers = 0; #endif // !defined(COMPONENT_BUILD) base::LazyInstance::Leaky AutoWritableMemory::writers_lock = LAZY_INSTANCE_INITIALIZER; static uintptr_t page_mask() { return ~(static_cast(getpagesize()) - 1); } static bool SetMemory(void* start, void* end, int prot) { const uintptr_t page_start = reinterpret_cast(start) & page_mask(); return mprotect(reinterpret_cast(page_start), reinterpret_cast(end) - page_start, prot) == 0; } bool AutoWritableMemory::SetMemoryReadWrite(void* start, void* end) { return SetMemory(start, end, PROT_READ | PROT_WRITE); } bool AutoWritableMemory::SetMemoryReadOnly(void* start, void* end) { return SetMemory(start, end, PROT_READ); } #if defined(OS_LINUX) void AssertMemoryIsReadOnly(const void* ptr) { #if DCHECK_IS_ON() const uintptr_t page_start = reinterpret_cast(ptr) & page_mask(); // Note: We've casted away const here, which should not be meaningful since // if the memory is written to we will abort immediately. int result = getrlimit(RLIMIT_NPROC, reinterpret_cast(page_start)); DCHECK_EQ(result, -1); DCHECK_EQ(errno, EFAULT); #endif // DCHECK_IS_ON() } #elif defined(OS_MACOSX) && !defined(OS_IOS) void AssertMemoryIsReadOnly(const void* ptr) { #if DCHECK_IS_ON() mach_port_t object_name; vm_region_basic_info_64 region_info; mach_vm_size_t size = 1; mach_msg_type_number_t count = VM_REGION_BASIC_INFO_COUNT_64; kern_return_t kr = mach_vm_region( mach_task_self(), reinterpret_cast(&ptr), &size, VM_REGION_BASIC_INFO_64, reinterpret_cast(®ion_info), &count, &object_name); DCHECK_EQ(kr, KERN_SUCCESS); DCHECK_EQ(region_info.protection, VM_PROT_READ); #endif // DCHECK_IS_ON() } #endif // defined(OS_LINUX) || (defined(OS_MACOSX) && !defined(OS_IOS)) } // namespace base