// Copyright 2014 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. #ifndef BASE_MAC_SCOPED_MACH_VM_H_ #define BASE_MAC_SCOPED_MACH_VM_H_ #include <mach/mach.h> #include <stddef.h> #include <algorithm> #include "base/base_export.h" #include "base/logging.h" #include "base/macros.h" // Use ScopedMachVM to supervise ownership of pages in the current process // through the Mach VM subsystem. Pages allocated with vm_allocate can be // released when exiting a scope with ScopedMachVM. // // The Mach VM subsystem operates on a page-by-page basis, and a single VM // allocation managed by a ScopedMachVM object may span multiple pages. As far // as Mach is concerned, allocated pages may be deallocated individually. This // is in contrast to higher-level allocators such as malloc, where the base // address of an allocation implies the size of an allocated block. // Consequently, it is not sufficient to just pass the base address of an // allocation to ScopedMachVM, it also needs to know the size of the // allocation. To avoid any confusion, both the base address and size must // be page-aligned. // // When dealing with Mach VM, base addresses will naturally be page-aligned, // but user-specified sizes may not be. If there's a concern that a size is // not page-aligned, use the mach_vm_round_page macro to correct it. // // Example: // // vm_address_t address = 0; // vm_size_t size = 12345; // This requested size is not page-aligned. // kern_return_t kr = // vm_allocate(mach_task_self(), &address, size, VM_FLAGS_ANYWHERE); // if (kr != KERN_SUCCESS) { // return false; // } // ScopedMachVM vm_owner(address, mach_vm_round_page(size)); namespace base { namespace mac { class BASE_EXPORT ScopedMachVM { public: explicit ScopedMachVM(vm_address_t address = 0, vm_size_t size = 0) : address_(address), size_(size) { DCHECK_EQ(address % PAGE_SIZE, 0u); DCHECK_EQ(size % PAGE_SIZE, 0u); } ~ScopedMachVM() { if (size_) { vm_deallocate(mach_task_self(), address_, size_); } } void reset(vm_address_t address = 0, vm_size_t size = 0); vm_address_t address() const { return address_; } vm_size_t size() const { return size_; } void swap(ScopedMachVM& that) { std::swap(address_, that.address_); std::swap(size_, that.size_); } void release() { address_ = 0; size_ = 0; } private: vm_address_t address_; vm_size_t size_; DISALLOW_COPY_AND_ASSIGN(ScopedMachVM); }; } // namespace mac } // namespace base #endif // BASE_MAC_SCOPED_MACH_VM_H_