#ifndef BASE_TRACE_EVENT_MALLOC_DUMP_PROVIDER_H_
#define BASE_TRACE_EVENT_MALLOC_DUMP_PROVIDER_H_
#include "base/allocator/buildflags.h"
#include "base/base_export.h"
#include "base/functional/callback.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/singleton.h"
#include "base/synchronization/lock.h"
#include "base/time/time.h"
#include "base/trace_event/memory_dump_provider.h"
#include "build/build_config.h"
#include "partition_alloc/buildflags.h"
#if PA_BUILDFLAG(USE_PARTITION_ALLOC)
#include "partition_alloc/partition_stats.h"
#endif
#if BUILDFLAG(IS_LINUX) || BUILDFLAG(IS_CHROMEOS) || BUILDFLAG(IS_ANDROID) || \
BUILDFLAG(IS_WIN) || BUILDFLAG(IS_MAC)
#define MALLOC_MEMORY_TRACING_SUPPORTED
#endif
namespace base {
namespace trace_event {
class MemoryAllocatorDump;
class BASE_EXPORT MallocDumpProvider : public MemoryDumpProvider {
public:
static const char kAllocatedObjects[];
static MallocDumpProvider* GetInstance();
struct ExtremeLUDStats {
size_t size_in_bytes = 0;
size_t count = 0;
size_t cumulative_size_in_bytes = 0;
size_t cumulative_count = 0;
size_t quarantine_miss_count = 0;
size_t capacity_in_bytes = 0;
};
struct ExtremeLUDStatsSet {
ExtremeLUDStats for_small_objects{};
ExtremeLUDStats for_large_objects{};
};
#if PA_BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC)
using ExtremeLUDGetStatsCallback = RepeatingCallback<ExtremeLUDStatsSet()>;
static void SetExtremeLUDGetStatsCallback(
ExtremeLUDGetStatsCallback callback);
#endif
MallocDumpProvider(const MallocDumpProvider&) = delete;
MallocDumpProvider& operator=(const MallocDumpProvider&) = delete;
bool OnMemoryDump(const MemoryDumpArgs& args,
ProcessMemoryDump* pmd) override;
private:
struct CumulativeEludStats {
size_t quarantined_bytes = 0;
size_t quarantined_count = 0;
size_t miss_count = 0;
};
friend struct DefaultSingletonTraits<MallocDumpProvider>;
MallocDumpProvider();
~MallocDumpProvider() override;
void ReportPerMinuteStats(uint64_t syscall_count,
size_t cumulative_brp_quarantined_bytes,
size_t cumulative_brp_quarantined_count,
const ExtremeLUDStats& elud_stats_for_small_objects,
const ExtremeLUDStats& elud_stats_for_large_objects,
MemoryAllocatorDump* malloc_dump,
MemoryAllocatorDump* partition_alloc_dump,
MemoryAllocatorDump* elud_dump_for_small_objects,
MemoryAllocatorDump* elud_dump_for_large_objects);
bool emit_metrics_on_memory_dump_
GUARDED_BY(emit_metrics_on_memory_dump_lock_) = true;
base::Lock emit_metrics_on_memory_dump_lock_;
#if PA_BUILDFLAG(USE_PARTITION_ALLOC_AS_MALLOC)
static ExtremeLUDGetStatsCallback& GetExtremeLUDGetStatsCallback();
base::TimeTicks last_memory_dump_time_ = base::TimeTicks::Now();
uint64_t last_syscall_count_ = 0;
size_t last_cumulative_brp_quarantined_bytes_ = 0;
size_t last_cumulative_brp_quarantined_count_ = 0;
CumulativeEludStats last_cumulative_elud_stats_for_small_objects_{0};
CumulativeEludStats last_cumulative_elud_stats_for_large_objects_{0};
#endif
};
#if PA_BUILDFLAG(USE_PARTITION_ALLOC)
class BASE_EXPORT MemoryDumpPartitionStatsDumper final
: public partition_alloc::PartitionStatsDumper {
public:
MemoryDumpPartitionStatsDumper(const char* root_name,
ProcessMemoryDump* memory_dump,
MemoryDumpLevelOfDetail level_of_detail);
static constexpr char kPartitionsDumpName[] = "partitions";
void PartitionDumpTotals(
const char* partition_name,
const partition_alloc::PartitionMemoryStats*) override;
void PartitionsDumpBucketStats(
const char* partition_name,
const partition_alloc::PartitionBucketMemoryStats*) override;
size_t total_mmapped_bytes() const { return total_mmapped_bytes_; }
size_t total_resident_bytes() const { return total_resident_bytes_; }
size_t total_active_bytes() const { return total_active_bytes_; }
size_t total_active_count() const { return total_active_count_; }
uint64_t syscall_count() const { return syscall_count_; }
size_t cumulative_brp_quarantined_bytes() const {
return cumulative_brp_quarantined_bytes_;
}
size_t cumulative_brp_quarantined_count() const {
return cumulative_brp_quarantined_count_;
}
private:
const char* root_name_;
raw_ptr<base::trace_event::ProcessMemoryDump> memory_dump_;
uint64_t uid_ = 0;
size_t total_mmapped_bytes_ = 0;
size_t total_resident_bytes_ = 0;
size_t total_active_bytes_ = 0;
size_t total_active_count_ = 0;
uint64_t syscall_count_ = 0;
size_t cumulative_brp_quarantined_bytes_ = 0;
size_t cumulative_brp_quarantined_count_ = 0;
bool detailed_;
};
#endif
}
}
#endif