// Copyright 2013 The Chromium Authors // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "base/process/memory.h" #include #include "base/allocator/buildflags.h" #include "base/allocator/partition_allocator/shim/allocator_interception_mac.h" #include "base/allocator/partition_allocator/shim/allocator_shim.h" #include "build/build_config.h" namespace base { namespace { void oom_killer_new() { TerminateBecauseOutOfMemory(0); } } // namespace void EnableTerminationOnHeapCorruption() { #if !ARCH_CPU_64_BITS DLOG(WARNING) << "EnableTerminationOnHeapCorruption only works on 64-bit"; #endif } bool UncheckedMalloc(size_t size, void** result) { #if BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC) // Unlike use_allocator="none", the default malloc zone is replaced with // PartitionAlloc, so the allocator shim functions work best. *result = allocator_shim::UncheckedAlloc(size); return *result != nullptr; #else // BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC) return allocator_shim::UncheckedMallocMac(size, result); #endif // BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC) } // The standard version is defined in memory.cc in case of // USE_PARTITION_ALLOC_AS_MALLOC. #if !BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC) bool UncheckedCalloc(size_t num_items, size_t size, void** result) { return allocator_shim::UncheckedCallocMac(num_items, size, result); } #endif // !BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC) void EnableTerminationOnOutOfMemory() { // Step 1: Enable OOM killer on C++ failures. std::set_new_handler(oom_killer_new); // Step 2: Enable OOM killer on C-malloc failures for the default zone (if we // have a shim). #if BUILDFLAG(USE_ALLOCATOR_SHIM) allocator_shim::SetCallNewHandlerOnMallocFailure(true); #endif // Step 3: Enable OOM killer on all other malloc zones (or just "all" without // "other" if shim is disabled). allocator_shim::InterceptAllocationsMac(); } void UncheckedFree(void* ptr) { #if BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC) // Important: might be different from free(), because in some cases, free() // does not necessarily know about allocator_shim::* functions. allocator_shim::UncheckedFree(ptr); #else // BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC) free(ptr); #endif // BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC) } } // namespace base