| OLD | NEW |
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | 1 // Copyright 2015 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "platform/heap/BlinkGCMemoryDumpProvider.h" | 5 #include "platform/heap/BlinkGCMemoryDumpProvider.h" |
| 6 | 6 |
| 7 #include "base/trace_event/heap_profiler_allocation_context_tracker.h" |
| 8 #include "base/trace_event/heap_profiler_allocation_register.h" |
| 9 #include "base/trace_event/trace_event_memory_overhead.h" |
| 7 #include "platform/heap/Handle.h" | 10 #include "platform/heap/Handle.h" |
| 8 #include "public/platform/Platform.h" | 11 #include "public/platform/Platform.h" |
| 9 #include "public/platform/WebMemoryAllocatorDump.h" | 12 #include "public/platform/WebMemoryAllocatorDump.h" |
| 10 #include "public/platform/WebProcessMemoryDump.h" | 13 #include "public/platform/WebProcessMemoryDump.h" |
| 11 #include "wtf/StdLibExtras.h" | 14 #include "wtf/StdLibExtras.h" |
| 12 #include "wtf/Threading.h" | 15 #include "wtf/Threading.h" |
| 13 | 16 |
| 14 namespace blink { | 17 namespace blink { |
| 15 namespace { | 18 namespace { |
| 16 | 19 |
| 17 void dumpMemoryTotals(blink::WebProcessMemoryDump* memoryDump) | 20 void dumpMemoryTotals(blink::WebProcessMemoryDump* memoryDump) |
| 18 { | 21 { |
| 19 String dumpName = String::format("blink_gc"); | 22 String dumpName = String::format("blink_gc"); |
| 20 WebMemoryAllocatorDump* allocatorDump = memoryDump->createMemoryAllocatorDum
p(dumpName); | 23 WebMemoryAllocatorDump* allocatorDump = memoryDump->createMemoryAllocatorDum
p(dumpName); |
| 21 allocatorDump->addScalar("size", "bytes", Heap::allocatedSpace()); | 24 allocatorDump->addScalar("size", "bytes", Heap::allocatedSpace()); |
| 22 | 25 |
| 23 dumpName.append("/allocated_objects"); | 26 dumpName.append("/allocated_objects"); |
| 24 WebMemoryAllocatorDump* objectsDump = memoryDump->createMemoryAllocatorDump(
dumpName); | 27 WebMemoryAllocatorDump* objectsDump = memoryDump->createMemoryAllocatorDump(
dumpName); |
| 25 | 28 |
| 26 // Heap::markedObjectSize() can be underestimated if we're still in the | 29 // Heap::markedObjectSize() can be underestimated if we're still in the |
| 27 // process of lazy sweeping. | 30 // process of lazy sweeping. |
| 28 objectsDump->addScalar("size", "bytes", Heap::allocatedObjectSize() + Heap::
markedObjectSize()); | 31 objectsDump->addScalar("size", "bytes", Heap::allocatedObjectSize() + Heap::
markedObjectSize()); |
| 29 } | 32 } |
| 30 | 33 |
| 34 void reportAllocation(Address address, size_t size) |
| 35 { |
| 36 BlinkGCMemoryDumpProvider::instance()->insert(address, size); |
| 37 } |
| 38 |
| 39 void reportFree(Address address) |
| 40 { |
| 41 BlinkGCMemoryDumpProvider::instance()->remove(address); |
| 42 } |
| 43 |
| 31 } // namespace | 44 } // namespace |
| 32 | 45 |
| 33 BlinkGCMemoryDumpProvider* BlinkGCMemoryDumpProvider::instance() | 46 BlinkGCMemoryDumpProvider* BlinkGCMemoryDumpProvider::instance() |
| 34 { | 47 { |
| 35 DEFINE_STATIC_LOCAL(BlinkGCMemoryDumpProvider, instance, ()); | 48 DEFINE_STATIC_LOCAL(BlinkGCMemoryDumpProvider, instance, ()); |
| 36 return &instance; | 49 return &instance; |
| 37 } | 50 } |
| 38 | 51 |
| 39 BlinkGCMemoryDumpProvider::~BlinkGCMemoryDumpProvider() | 52 BlinkGCMemoryDumpProvider::~BlinkGCMemoryDumpProvider() |
| 40 { | 53 { |
| 41 } | 54 } |
| 42 | 55 |
| 43 bool BlinkGCMemoryDumpProvider::onMemoryDump(WebMemoryDumpLevelOfDetail levelOfD
etail, blink::WebProcessMemoryDump* memoryDump) | 56 bool BlinkGCMemoryDumpProvider::onMemoryDump(WebMemoryDumpLevelOfDetail levelOfD
etail, blink::WebProcessMemoryDump* memoryDump) |
| 44 { | 57 { |
| 45 if (levelOfDetail == WebMemoryDumpLevelOfDetail::Light) { | 58 if (levelOfDetail == WebMemoryDumpLevelOfDetail::Light) { |
| 46 dumpMemoryTotals(memoryDump); | 59 dumpMemoryTotals(memoryDump); |
| 47 return true; | 60 return true; |
| 48 } | 61 } |
| 49 | 62 |
| 50 Heap::collectGarbage(BlinkGC::NoHeapPointersOnStack, BlinkGC::TakeSnapshot,
BlinkGC::ForcedGC); | 63 Heap::collectGarbage(BlinkGC::NoHeapPointersOnStack, BlinkGC::TakeSnapshot,
BlinkGC::ForcedGC); |
| 51 dumpMemoryTotals(memoryDump); | 64 dumpMemoryTotals(memoryDump); |
| 52 | 65 |
| 66 if (m_isHeapProfilingEnabled) { |
| 67 base::trace_event::TraceEventMemoryOverhead overhead; |
| 68 base::hash_map<base::trace_event::AllocationContext, size_t> bytesByCont
ext; |
| 69 { |
| 70 MutexLocker locker(m_allocationRegisterMutex); |
| 71 for (const auto& allocSize : *m_allocationRegister) |
| 72 bytesByContext[allocSize.context] += allocSize.size; |
| 73 |
| 74 m_allocationRegister->EstimateTraceMemoryOverhead(&overhead); |
| 75 } |
| 76 memoryDump->dumpHeapUsage(bytesByContext, overhead, "blink_gc"); |
| 77 } |
| 78 |
| 53 // Merge all dumps collected by Heap::collectGarbage. | 79 // Merge all dumps collected by Heap::collectGarbage. |
| 54 memoryDump->takeAllDumpsFrom(m_currentProcessMemoryDump.get()); | 80 memoryDump->takeAllDumpsFrom(m_currentProcessMemoryDump.get()); |
| 55 return true; | 81 return true; |
| 56 } | 82 } |
| 57 | 83 |
| 84 void BlinkGCMemoryDumpProvider::onHeapProfilingEnabled(bool enabled) |
| 85 { |
| 86 if (enabled) { |
| 87 { |
| 88 MutexLocker locker(m_allocationRegisterMutex); |
| 89 if (!m_allocationRegister) |
| 90 m_allocationRegister = adoptPtr(new base::trace_event::Allocatio
nRegister()); |
| 91 } |
| 92 HeapAllocHooks::setAllocationHook(reportAllocation); |
| 93 HeapAllocHooks::setFreeHook(reportFree); |
| 94 } else { |
| 95 HeapAllocHooks::setAllocationHook(nullptr); |
| 96 HeapAllocHooks::setFreeHook(nullptr); |
| 97 } |
| 98 m_isHeapProfilingEnabled = enabled; |
| 99 } |
| 100 |
| 58 WebMemoryAllocatorDump* BlinkGCMemoryDumpProvider::createMemoryAllocatorDumpForC
urrentGC(const String& absoluteName) | 101 WebMemoryAllocatorDump* BlinkGCMemoryDumpProvider::createMemoryAllocatorDumpForC
urrentGC(const String& absoluteName) |
| 59 { | 102 { |
| 60 return m_currentProcessMemoryDump->createMemoryAllocatorDump(absoluteName); | 103 return m_currentProcessMemoryDump->createMemoryAllocatorDump(absoluteName); |
| 61 } | 104 } |
| 62 | 105 |
| 63 void BlinkGCMemoryDumpProvider::clearProcessDumpForCurrentGC() | 106 void BlinkGCMemoryDumpProvider::clearProcessDumpForCurrentGC() |
| 64 { | 107 { |
| 65 m_currentProcessMemoryDump->clear(); | 108 m_currentProcessMemoryDump->clear(); |
| 66 } | 109 } |
| 67 | 110 |
| 68 BlinkGCMemoryDumpProvider::BlinkGCMemoryDumpProvider() | 111 BlinkGCMemoryDumpProvider::BlinkGCMemoryDumpProvider() |
| 69 : m_currentProcessMemoryDump(adoptPtr(Platform::current()->createProcessMemo
ryDump())) | 112 : m_currentProcessMemoryDump(adoptPtr(Platform::current()->createProcessMemo
ryDump())) |
| 113 , m_isHeapProfilingEnabled(false) |
| 70 { | 114 { |
| 71 } | 115 } |
| 72 | 116 |
| 117 void BlinkGCMemoryDumpProvider::insert(Address address, size_t size) |
| 118 { |
| 119 base::trace_event::AllocationContext context = base::trace_event::Allocation
ContextTracker::GetContextSnapshot(); |
| 120 // TODO(hajimehoshi): Implement to use a correct type name. |
| 121 context.type_name = ""; |
| 122 MutexLocker locker(m_allocationRegisterMutex); |
| 123 if (m_allocationRegister) |
| 124 m_allocationRegister->Insert(address, size, context); |
| 125 } |
| 126 |
| 127 void BlinkGCMemoryDumpProvider::remove(Address address) |
| 128 { |
| 129 MutexLocker locker(m_allocationRegisterMutex); |
| 130 if (m_allocationRegister) |
| 131 m_allocationRegister->Remove(address); |
| 132 } |
| 133 |
| 73 } // namespace blink | 134 } // namespace blink |
| OLD | NEW |