blob: ced2e547c03437ea8f872f4d2114a764f55c919b [file] [log] [blame]
/*
* Copyright (C) 2013 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "wtf/Partitions.h"
#include "wtf/Alias.h"
#include "wtf/MainThread.h"
#include "wtf/PartitionAllocator.h"
namespace WTF {
const char* const Partitions::kAllocatedObjectPoolName = "partition_alloc/allocated_objects";
int Partitions::s_initializationLock = 0;
bool Partitions::s_initialized = false;
PartitionAllocatorGeneric Partitions::m_fastMallocAllocator;
PartitionAllocatorGeneric Partitions::m_bufferAllocator;
SizeSpecificPartitionAllocator<3328> Partitions::m_nodeAllocator;
SizeSpecificPartitionAllocator<1024> Partitions::m_layoutAllocator;
HistogramEnumerationFunction Partitions::m_histogramEnumeration = nullptr;
void Partitions::initialize(HistogramEnumerationFunction histogramEnumeration)
{
spinLockLock(&s_initializationLock);
if (!s_initialized) {
partitionAllocGlobalInit(&Partitions::handleOutOfMemory);
m_fastMallocAllocator.init();
m_bufferAllocator.init();
m_nodeAllocator.init();
m_layoutAllocator.init();
m_histogramEnumeration = histogramEnumeration;
s_initialized = true;
}
spinLockUnlock(&s_initializationLock);
}
void Partitions::shutdown()
{
// We could ASSERT here for a memory leak within the partition, but it leads
// to very hard to diagnose ASSERTs, so it's best to leave leak checking for
// the valgrind and heapcheck bots, which run without partitions.
(void) m_layoutAllocator.shutdown();
(void) m_nodeAllocator.shutdown();
(void) m_bufferAllocator.shutdown();
(void) m_fastMallocAllocator.shutdown();
}
void Partitions::decommitFreeableMemory()
{
ASSERT(isMainThread());
partitionPurgeMemoryGeneric(bufferPartition(), PartitionPurgeDecommitEmptyPages);
partitionPurgeMemoryGeneric(fastMallocPartition(), PartitionPurgeDecommitEmptyPages);
partitionPurgeMemory(nodePartition(), PartitionPurgeDecommitEmptyPages);
partitionPurgeMemory(layoutPartition(), PartitionPurgeDecommitEmptyPages);
}
void Partitions::reportMemoryUsageHistogram()
{
static size_t supportedMaxSizeInMB = 4 * 1024;
static size_t observedMaxSizeInMB = 0;
if (!m_histogramEnumeration)
return;
// We only report the memory in the main thread.
if (!isMainThread())
return;
// +1 is for rounding up the sizeInMB.
size_t sizeInMB = Partitions::totalSizeOfCommittedPages() / 1024 / 1024 + 1;
if (sizeInMB >= supportedMaxSizeInMB)
sizeInMB = supportedMaxSizeInMB - 1;
if (sizeInMB > observedMaxSizeInMB) {
// Send a UseCounter only when we see the highest memory usage
// we've ever seen.
m_histogramEnumeration("PartitionAlloc.CommittedSize", sizeInMB, supportedMaxSizeInMB);
observedMaxSizeInMB = sizeInMB;
}
}
void Partitions::dumpMemoryStats(bool isLightDump, PartitionStatsDumper* partitionStatsDumper)
{
// Object model and rendering partitions are not thread safe and can be
// accessed only on the main thread.
ASSERT(isMainThread());
decommitFreeableMemory();
partitionDumpStatsGeneric(fastMallocPartition(), "fast_malloc", isLightDump, partitionStatsDumper);
partitionDumpStatsGeneric(bufferPartition(), "buffer", isLightDump, partitionStatsDumper);
partitionDumpStats(nodePartition(), "node", isLightDump, partitionStatsDumper);
partitionDumpStats(layoutPartition(), "layout", isLightDump, partitionStatsDumper);
}
static NEVER_INLINE void partitionsOutOfMemoryUsing2G()
{
size_t signature = 2UL * 1024 * 1024 * 1024;
alias(&signature);
IMMEDIATE_CRASH();
}
static NEVER_INLINE void partitionsOutOfMemoryUsing1G()
{
size_t signature = 1UL * 1024 * 1024 * 1024;
alias(&signature);
IMMEDIATE_CRASH();
}
static NEVER_INLINE void partitionsOutOfMemoryUsing512M()
{
size_t signature = 512 * 1024 * 1024;
alias(&signature);
IMMEDIATE_CRASH();
}
static NEVER_INLINE void partitionsOutOfMemoryUsing256M()
{
size_t signature = 256 * 1024 * 1024;
alias(&signature);
IMMEDIATE_CRASH();
}
static NEVER_INLINE void partitionsOutOfMemoryUsing128M()
{
size_t signature = 128 * 1024 * 1024;
alias(&signature);
IMMEDIATE_CRASH();
}
static NEVER_INLINE void partitionsOutOfMemoryUsing64M()
{
size_t signature = 64 * 1024 * 1024;
alias(&signature);
IMMEDIATE_CRASH();
}
static NEVER_INLINE void partitionsOutOfMemoryUsing32M()
{
size_t signature = 32 * 1024 * 1024;
alias(&signature);
IMMEDIATE_CRASH();
}
static NEVER_INLINE void partitionsOutOfMemoryUsing16M()
{
size_t signature = 16 * 1024 * 1024;
alias(&signature);
IMMEDIATE_CRASH();
}
static NEVER_INLINE void partitionsOutOfMemoryUsingLessThan16M()
{
size_t signature = 16 * 1024 * 1024 - 1;
alias(&signature);
IMMEDIATE_CRASH();
}
void Partitions::handleOutOfMemory()
{
volatile size_t totalUsage = totalSizeOfCommittedPages();
if (totalUsage >= 2UL * 1024 * 1024 * 1024)
partitionsOutOfMemoryUsing2G();
if (totalUsage >= 1UL * 1024 * 1024 * 1024)
partitionsOutOfMemoryUsing1G();
if (totalUsage >= 512 * 1024 * 1024)
partitionsOutOfMemoryUsing512M();
if (totalUsage >= 256 * 1024 * 1024)
partitionsOutOfMemoryUsing256M();
if (totalUsage >= 128 * 1024 * 1024)
partitionsOutOfMemoryUsing128M();
if (totalUsage >= 64 * 1024 * 1024)
partitionsOutOfMemoryUsing64M();
if (totalUsage >= 32 * 1024 * 1024)
partitionsOutOfMemoryUsing32M();
if (totalUsage >= 16 * 1024 * 1024)
partitionsOutOfMemoryUsing16M();
partitionsOutOfMemoryUsingLessThan16M();
}
} // namespace WTF