blob: d9e0a56a9af984c3f493b0586d060be01d377d59 [file] [log] [blame]
/*
* Copyright (C) 2019 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "src/profiling/memory/bookkeeping_dump.h"
namespace perfetto {
namespace profiling {
namespace {
using ::perfetto::protos::pbzero::Callstack;
using ::perfetto::protos::pbzero::ProfilePacket;
// This needs to be lower than the maximum acceptable chunk size, because this
// is checked *before* writing another submessage. We conservatively assume
// submessages can be up to 100k here for a 500k chunk size.
// DropBox has a 500k chunk limit, and each chunk needs to parse as a proto.
uint32_t kPacketSizeThreshold = 400000;
} // namespace
void DumpState::WriteMap(const Interned<Mapping> map) {
auto map_it_and_inserted = dumped_mappings_.emplace(map.id());
if (map_it_and_inserted.second) {
for (const Interned<std::string>& str : map->path_components)
WriteString(str);
WriteString(map->build_id);
if (currently_written() > kPacketSizeThreshold)
NewProfilePacket();
auto mapping = current_profile_packet_->add_mappings();
mapping->set_iid(map.id());
mapping->set_offset(map->offset);
mapping->set_start(map->start);
mapping->set_end(map->end);
mapping->set_load_bias(map->load_bias);
mapping->set_build_id(map->build_id.id());
for (const Interned<std::string>& str : map->path_components)
mapping->add_path_string_ids(str.id());
}
}
void DumpState::WriteFrame(Interned<Frame> frame) {
WriteMap(frame->mapping);
WriteString(frame->function_name);
bool inserted;
std::tie(std::ignore, inserted) = dumped_frames_.emplace(frame.id());
if (inserted) {
if (currently_written() > kPacketSizeThreshold)
NewProfilePacket();
auto frame_proto = current_profile_packet_->add_frames();
frame_proto->set_iid(frame.id());
frame_proto->set_function_name_id(frame->function_name.id());
frame_proto->set_mapping_id(frame->mapping.id());
frame_proto->set_rel_pc(frame->rel_pc);
}
}
void DumpState::WriteString(const Interned<std::string>& str) {
bool inserted;
std::tie(std::ignore, inserted) = dumped_strings_.emplace(str.id());
if (inserted) {
if (currently_written() > kPacketSizeThreshold)
NewProfilePacket();
auto interned_string = current_profile_packet_->add_strings();
interned_string->set_iid(str.id());
interned_string->set_str(reinterpret_cast<const uint8_t*>(str->c_str()),
str->size());
}
}
void DumpState::StartProcessDump(
std::function<void(protos::pbzero::ProfilePacket::ProcessHeapSamples*)>
fill_process_header) {
current_process_fill_header_ = std::move(fill_process_header);
current_process_heap_samples_ = nullptr;
current_process_idle_allocs_.clear();
}
void DumpState::WriteAllocation(
const HeapTracker::CallstackAllocations& alloc) {
callstacks_to_dump_.emplace(alloc.node);
auto* heap_samples = GetCurrentProcessHeapSamples();
ProfilePacket::HeapSample* sample = heap_samples->add_samples();
sample->set_callstack_id(alloc.node->id());
sample->set_self_allocated(alloc.allocated);
sample->set_self_freed(alloc.freed);
sample->set_alloc_count(alloc.allocation_count);
sample->set_free_count(alloc.free_count);
auto it = current_process_idle_allocs_.find(alloc.node->id());
if (it != current_process_idle_allocs_.end())
sample->set_self_idle(it->second);
}
void DumpState::DumpCallstacks(GlobalCallstackTrie* callsites) {
for (GlobalCallstackTrie::Node* node : callstacks_to_dump_) {
// There need to be two separate loops over built_callstack because
// protozero cannot interleave different messages.
auto built_callstack = callsites->BuildCallstack(node);
for (const Interned<Frame>& frame : built_callstack)
WriteFrame(frame);
Callstack* callstack = current_profile_packet_->add_callstacks();
callstack->set_iid(node->id());
for (const Interned<Frame>& frame : built_callstack)
callstack->add_frame_ids(frame.id());
}
}
void DumpState::AddIdleBytes(uintptr_t callstack_id, uint64_t bytes) {
current_process_idle_allocs_[callstack_id] += bytes;
}
void DumpState::RejectConcurrent(pid_t pid) {
ProfilePacket::ProcessHeapSamples* proto =
current_profile_packet_->add_process_dumps();
proto->set_pid(static_cast<uint64_t>(pid));
proto->set_rejected_concurrent(true);
}
ProfilePacket::ProcessHeapSamples* DumpState::GetCurrentProcessHeapSamples() {
if (currently_written() > kPacketSizeThreshold &&
current_process_heap_samples_ != nullptr) {
NewProfilePacket();
current_process_heap_samples_ = nullptr;
}
if (current_process_heap_samples_ == nullptr) {
current_process_heap_samples_ =
current_profile_packet_->add_process_dumps();
current_process_fill_header_(current_process_heap_samples_);
}
return current_process_heap_samples_;
}
} // namespace profiling
} // namespace perfetto