|  | // Copyright 2018 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_SAMPLING_HEAP_PROFILER_SAMPLING_HEAP_PROFILER_H | 
|  | #define BASE_SAMPLING_HEAP_PROFILER_SAMPLING_HEAP_PROFILER_H | 
|  |  | 
|  | #include <unordered_map> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/base_export.h" | 
|  | #include "base/macros.h" | 
|  | #include "base/synchronization/lock.h" | 
|  | #include "base/threading/thread_local.h" | 
|  |  | 
|  | namespace base { | 
|  |  | 
|  | template <typename T> | 
|  | class NoDestructor; | 
|  |  | 
|  | // The class implements sampling profiling of native memory heap. | 
|  | // It hooks on base::allocator and base::PartitionAlloc. | 
|  | // When started it selects and records allocation samples based on | 
|  | // the sampling_interval parameter. | 
|  | // The recorded samples can then be retrieved using GetSamples method. | 
|  | class BASE_EXPORT SamplingHeapProfiler { | 
|  | public: | 
|  | class BASE_EXPORT Sample { | 
|  | public: | 
|  | Sample(const Sample&); | 
|  | ~Sample(); | 
|  |  | 
|  | size_t size;   // Allocation size. | 
|  | size_t total;  // Total size attributed to the sample. | 
|  | std::vector<void*> stack; | 
|  |  | 
|  | private: | 
|  | friend class SamplingHeapProfiler; | 
|  |  | 
|  | Sample(size_t, size_t total, uint32_t ordinal); | 
|  |  | 
|  | uint32_t ordinal; | 
|  | }; | 
|  |  | 
|  | class SamplesObserver { | 
|  | public: | 
|  | virtual ~SamplesObserver() = default; | 
|  | virtual void SampleAdded(uint32_t id, size_t size, size_t total) = 0; | 
|  | virtual void SampleRemoved(uint32_t id) = 0; | 
|  | }; | 
|  |  | 
|  | // Must be called early during the process initialization. It creates and | 
|  | // reserves a TLS slot. | 
|  | static void InitTLSSlot(); | 
|  |  | 
|  | // This is an entry point for plugging in an external allocator. | 
|  | // Profiler will invoke the provided callback upon initialization. | 
|  | // The callback should install hooks onto the corresponding memory allocator | 
|  | // and make them invoke SamplingHeapProfiler::RecordAlloc and | 
|  | // SamplingHeapProfiler::RecordFree upon corresponding allocation events. | 
|  | // | 
|  | // If the method is called after profiler is initialized, the callback | 
|  | // is invoked right away. | 
|  | static void SetHooksInstallCallback(void (*hooks_install_callback)()); | 
|  |  | 
|  | void AddSamplesObserver(SamplesObserver*); | 
|  | void RemoveSamplesObserver(SamplesObserver*); | 
|  |  | 
|  | uint32_t Start(); | 
|  | void Stop(); | 
|  | void SetSamplingInterval(size_t sampling_interval); | 
|  | void SuppressRandomnessForTest(bool suppress); | 
|  |  | 
|  | std::vector<Sample> GetSamples(uint32_t profile_id); | 
|  |  | 
|  | static void RecordAlloc(void* address, size_t, uint32_t skip_frames = 0); | 
|  | static void RecordFree(void* address); | 
|  |  | 
|  | static SamplingHeapProfiler* GetInstance(); | 
|  |  | 
|  | private: | 
|  | SamplingHeapProfiler(); | 
|  | ~SamplingHeapProfiler() = delete; | 
|  |  | 
|  | static void InstallAllocatorHooksOnce(); | 
|  | static bool InstallAllocatorHooks(); | 
|  | static size_t GetNextSampleInterval(size_t base_interval); | 
|  |  | 
|  | void DoRecordAlloc(size_t total_allocated, | 
|  | size_t allocation_size, | 
|  | void* address, | 
|  | uint32_t skip_frames); | 
|  | void DoRecordFree(void* address); | 
|  | void RecordStackTrace(Sample*, uint32_t skip_frames); | 
|  | bool MayRehashOnInsert(); | 
|  |  | 
|  | base::ThreadLocalBoolean entered_; | 
|  | base::Lock mutex_; | 
|  | std::unordered_map<void*, Sample> samples_; | 
|  | std::vector<SamplesObserver*> observers_; | 
|  |  | 
|  | static SamplingHeapProfiler* instance_; | 
|  |  | 
|  | friend class base::NoDestructor<SamplingHeapProfiler>; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(SamplingHeapProfiler); | 
|  | }; | 
|  |  | 
|  | }  // namespace base | 
|  |  | 
|  | #endif  // BASE_SAMPLING_HEAP_PROFILER_SAMPLING_HEAP_PROFILER_H |