blob: c65656499022e52b71ca45de662b453804fef4d8 [file] [log] [blame]
/*
* Copyright (C) 2020 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/trace_processor/importers/proto/track_event_tracker.h"
#include "src/trace_processor/importers/common/args_tracker.h"
#include "src/trace_processor/importers/common/args_translation_table.h"
#include "src/trace_processor/importers/common/process_tracker.h"
#include "src/trace_processor/importers/common/track_tracker.h"
namespace perfetto {
namespace trace_processor {
TrackEventTracker::TrackEventTracker(TraceProcessorContext* context)
: source_key_(context->storage->InternString("source")),
source_id_key_(context->storage->InternString("source_id")),
is_root_in_scope_key_(context->storage->InternString("is_root_in_scope")),
category_key_(context->storage->InternString("category")),
has_first_packet_on_sequence_key_id_(
context->storage->InternString("has_first_packet_on_sequence")),
descriptor_source_(context->storage->InternString("descriptor")),
default_descriptor_track_name_(
context->storage->InternString("Default Track")),
context_(context) {}
void TrackEventTracker::ReserveDescriptorProcessTrack(uint64_t uuid,
StringId name,
uint32_t pid,
int64_t timestamp) {
DescriptorTrackReservation reservation;
reservation.min_timestamp = timestamp;
reservation.pid = pid;
reservation.name = name;
std::map<uint64_t, DescriptorTrackReservation>::iterator it;
bool inserted;
std::tie(it, inserted) =
reserved_descriptor_tracks_.insert(std::make_pair<>(uuid, reservation));
if (inserted)
return;
if (!it->second.IsForSameTrack(reservation)) {
// Process tracks should not be reassigned to a different pid later (neither
// should the type of the track change).
PERFETTO_DLOG("New track reservation for process track with uuid %" PRIu64
" doesn't match earlier one",
uuid);
context_->storage->IncrementStats(stats::track_event_tokenizer_errors);
return;
}
it->second.min_timestamp = std::min(it->second.min_timestamp, timestamp);
}
void TrackEventTracker::ReserveDescriptorThreadTrack(uint64_t uuid,
uint64_t parent_uuid,
StringId name,
uint32_t pid,
uint32_t tid,
int64_t timestamp,
bool use_separate_track) {
DescriptorTrackReservation reservation;
reservation.min_timestamp = timestamp;
reservation.parent_uuid = parent_uuid;
reservation.pid = pid;
reservation.tid = tid;
reservation.name = name;
reservation.use_separate_track = use_separate_track;
std::map<uint64_t, DescriptorTrackReservation>::iterator it;
bool inserted;
std::tie(it, inserted) =
reserved_descriptor_tracks_.insert(std::make_pair<>(uuid, reservation));
if (inserted)
return;
if (!it->second.IsForSameTrack(reservation)) {
// Thread tracks should not be reassigned to a different pid/tid later
// (neither should the type of the track change).
PERFETTO_DLOG("New track reservation for thread track with uuid %" PRIu64
" doesn't match earlier one",
uuid);
context_->storage->IncrementStats(stats::track_event_tokenizer_errors);
return;
}
it->second.min_timestamp = std::min(it->second.min_timestamp, timestamp);
}
void TrackEventTracker::ReserveDescriptorCounterTrack(
uint64_t uuid,
uint64_t parent_uuid,
StringId name,
StringId category,
int64_t unit_multiplier,
bool is_incremental,
uint32_t packet_sequence_id) {
DescriptorTrackReservation reservation;
reservation.parent_uuid = parent_uuid;
reservation.is_counter = true;
reservation.name = name;
reservation.category = category;
reservation.unit_multiplier = unit_multiplier;
reservation.is_incremental = is_incremental;
// Incrementally encoded counters are only valid on a single sequence.
if (is_incremental)
reservation.packet_sequence_id = packet_sequence_id;
std::map<uint64_t, DescriptorTrackReservation>::iterator it;
bool inserted;
std::tie(it, inserted) =
reserved_descriptor_tracks_.insert(std::make_pair<>(uuid, reservation));
if (inserted || it->second.IsForSameTrack(reservation))
return;
// Counter tracks should not be reassigned to a different parent track later
// (neither should the type of the track change).
PERFETTO_DLOG("New track reservation for counter track with uuid %" PRIu64
" doesn't match earlier one",
uuid);
context_->storage->IncrementStats(stats::track_event_tokenizer_errors);
}
void TrackEventTracker::ReserveDescriptorChildTrack(uint64_t uuid,
uint64_t parent_uuid,
StringId name) {
DescriptorTrackReservation reservation;
reservation.parent_uuid = parent_uuid;
reservation.name = name;
std::map<uint64_t, DescriptorTrackReservation>::iterator it;
bool inserted;
std::tie(it, inserted) =
reserved_descriptor_tracks_.insert(std::make_pair<>(uuid, reservation));
if (inserted || it->second.IsForSameTrack(reservation))
return;
// Child tracks should not be reassigned to a different parent track later
// (neither should the type of the track change).
PERFETTO_DLOG("New track reservation for child track with uuid %" PRIu64
" doesn't match earlier one",
uuid);
context_->storage->IncrementStats(stats::track_event_tokenizer_errors);
}
TrackId TrackEventTracker::InsertThreadTrack(UniqueTid utid) {
tables::ThreadTrackTable::Row row;
row.utid = utid;
auto* thread_tracks = context_->storage->mutable_thread_track_table();
return thread_tracks->Insert(row).id;
}
TrackId TrackEventTracker::InternThreadTrack(UniqueTid utid) {
auto it = thread_tracks_.find(utid);
if (it != thread_tracks_.end()) {
return it->second;
}
return thread_tracks_[utid] = InsertThreadTrack(utid);
}
std::optional<TrackId> TrackEventTracker::GetDescriptorTrack(
uint64_t uuid,
StringId event_name,
std::optional<uint32_t> packet_sequence_id) {
std::optional<TrackId> track_id =
GetDescriptorTrackImpl(uuid, packet_sequence_id);
if (!track_id || event_name.is_null())
return track_id;
// Update the name of the track if unset and the track is not the primary
// track of a process/thread or a counter track.
auto* tracks = context_->storage->mutable_track_table();
uint32_t row = *tracks->id().IndexOf(*track_id);
if (!tracks->name()[row].is_null())
return track_id;
// Check reservation for track type.
auto reservation_it = reserved_descriptor_tracks_.find(uuid);
PERFETTO_CHECK(reservation_it != reserved_descriptor_tracks_.end());
if (reservation_it->second.pid || reservation_it->second.tid ||
reservation_it->second.is_counter) {
return track_id;
}
tracks->mutable_name()->Set(row, event_name);
return track_id;
}
std::optional<TrackId> TrackEventTracker::GetDescriptorTrackImpl(
uint64_t uuid,
std::optional<uint32_t> packet_sequence_id) {
auto it = descriptor_tracks_.find(uuid);
if (it != descriptor_tracks_.end())
return it->second;
std::optional<ResolvedDescriptorTrack> resolved_track =
ResolveDescriptorTrack(uuid, nullptr);
if (!resolved_track)
return std::nullopt;
// The reservation must exist as |resolved_track| would have been std::nullopt
// otherwise.
auto reserved_it = reserved_descriptor_tracks_.find(uuid);
PERFETTO_CHECK(reserved_it != reserved_descriptor_tracks_.end());
const auto& reservation = reserved_it->second;
// We resolve parent_id here to ensure that it's going to be smaller
// than the id of the child.
std::optional<TrackId> parent_id;
if (reservation.parent_uuid != 0) {
parent_id = GetDescriptorTrackImpl(reservation.parent_uuid);
}
TrackId track_id = CreateTrackFromResolved(*resolved_track);
descriptor_tracks_[uuid] = track_id;
auto args = context_->args_tracker->AddArgsTo(track_id);
args.AddArg(source_key_, Variadic::String(descriptor_source_))
.AddArg(source_id_key_, Variadic::Integer(static_cast<int64_t>(uuid)))
.AddArg(is_root_in_scope_key_,
Variadic::Boolean(resolved_track->is_root_in_scope()));
if (!reservation.category.is_null())
args.AddArg(category_key_, Variadic::String(reservation.category));
if (packet_sequence_id &&
sequences_with_first_packet_.find(*packet_sequence_id) !=
sequences_with_first_packet_.end()) {
args.AddArg(has_first_packet_on_sequence_key_id_, Variadic::Boolean(true));
}
auto* tracks = context_->storage->mutable_track_table();
auto row_ref = *tracks->FindById(track_id);
if (parent_id) {
row_ref.set_parent_id(*parent_id);
}
if (reservation.name.is_null())
return track_id;
// Initialize the track name here, so that, if a name was given in the
// reservation, it is set immediately after resolution takes place.
row_ref.set_name(reservation.name);
return track_id;
}
TrackId TrackEventTracker::CreateTrackFromResolved(
const ResolvedDescriptorTrack& track) {
if (track.is_root_in_scope()) {
switch (track.scope()) {
case ResolvedDescriptorTrack::Scope::kThread: {
if (track.use_separate_track()) {
return InternThreadTrack(track.utid());
}
return context_->track_tracker->InternThreadTrack(track.utid());
}
case ResolvedDescriptorTrack::Scope::kProcess:
return context_->track_tracker->InternProcessTrack(track.upid());
case ResolvedDescriptorTrack::Scope::kGlobal:
// Will be handled below.
break;
}
}
switch (track.scope()) {
case ResolvedDescriptorTrack::Scope::kThread: {
if (track.is_counter()) {
tables::ThreadCounterTrackTable::Row row;
row.utid = track.utid();
auto* thread_counter_tracks =
context_->storage->mutable_thread_counter_track_table();
return thread_counter_tracks->Insert(row).id;
}
return InsertThreadTrack(track.utid());
}
case ResolvedDescriptorTrack::Scope::kProcess: {
if (track.is_counter()) {
tables::ProcessCounterTrackTable::Row row;
row.upid = track.upid();
auto* process_counter_tracks =
context_->storage->mutable_process_counter_track_table();
return process_counter_tracks->Insert(row).id;
}
tables::ProcessTrackTable::Row row;
row.upid = track.upid();
auto* process_tracks = context_->storage->mutable_process_track_table();
return process_tracks->Insert(row).id;
}
case ResolvedDescriptorTrack::Scope::kGlobal: {
if (track.is_counter())
return context_->storage->mutable_counter_track_table()->Insert({}).id;
return context_->storage->mutable_track_table()->Insert({}).id;
}
}
PERFETTO_FATAL("For GCC");
}
std::optional<TrackEventTracker::ResolvedDescriptorTrack>
TrackEventTracker::ResolveDescriptorTrack(
uint64_t uuid,
std::vector<uint64_t>* descendent_uuids) {
auto it = resolved_descriptor_tracks_.find(uuid);
if (it != resolved_descriptor_tracks_.end())
return it->second;
auto reservation_it = reserved_descriptor_tracks_.find(uuid);
if (reservation_it == reserved_descriptor_tracks_.end())
return std::nullopt;
// Resolve process and thread id for tracks produced from within a pid
// namespace.
// Get the root-level trusted_pid for the process that produces the track
// event.
auto opt_trusted_pid = context_->process_tracker->GetTrustedPid(uuid);
auto& reservation = reservation_it->second;
// Try to resolve to root-level pid and tid if the process is pid-namespaced.
if (opt_trusted_pid && reservation.tid) {
auto opt_resolved_tid = context_->process_tracker->ResolveNamespacedTid(
*opt_trusted_pid, *reservation.tid);
if (opt_resolved_tid)
reservation.tid = *opt_resolved_tid;
}
if (opt_trusted_pid && reservation.pid) {
auto opt_resolved_pid = context_->process_tracker->ResolveNamespacedTid(
*opt_trusted_pid, *reservation.pid);
if (opt_resolved_pid)
reservation.pid = *opt_resolved_pid;
}
std::optional<ResolvedDescriptorTrack> resolved_track =
ResolveDescriptorTrackImpl(uuid, reservation, descendent_uuids);
if (!resolved_track) {
return std::nullopt;
}
resolved_descriptor_tracks_[uuid] = *resolved_track;
return resolved_track;
}
std::optional<TrackEventTracker::ResolvedDescriptorTrack>
TrackEventTracker::ResolveDescriptorTrackImpl(
uint64_t uuid,
const DescriptorTrackReservation& reservation,
std::vector<uint64_t>* descendent_uuids) {
static constexpr size_t kMaxAncestors = 10;
// Try to resolve any parent tracks recursively, too.
std::optional<ResolvedDescriptorTrack> parent_resolved_track;
if (reservation.parent_uuid) {
// Input data may contain loops or extremely long ancestor track chains. To
// avoid stack overflow in these situations, we keep track of the ancestors
// seen in the recursion.
std::unique_ptr<std::vector<uint64_t>> owned_descendent_uuids;
if (!descendent_uuids) {
owned_descendent_uuids.reset(new std::vector<uint64_t>());
descendent_uuids = owned_descendent_uuids.get();
}
descendent_uuids->push_back(uuid);
if (descendent_uuids->size() > kMaxAncestors) {
PERFETTO_ELOG(
"Too many ancestors in parent_track_uuid hierarchy at track %" PRIu64
" with parent %" PRIu64,
uuid, reservation.parent_uuid);
return std::nullopt;
}
if (std::find(descendent_uuids->begin(), descendent_uuids->end(),
reservation.parent_uuid) != descendent_uuids->end()) {
PERFETTO_ELOG(
"Loop detected in parent_track_uuid hierarchy at track %" PRIu64
" with parent %" PRIu64,
uuid, reservation.parent_uuid);
return std::nullopt;
}
parent_resolved_track =
ResolveDescriptorTrack(reservation.parent_uuid, descendent_uuids);
if (!parent_resolved_track) {
PERFETTO_ELOG("Unknown parent track %" PRIu64 " for track %" PRIu64,
reservation.parent_uuid, uuid);
}
descendent_uuids->pop_back();
if (owned_descendent_uuids)
descendent_uuids = nullptr;
}
if (reservation.tid) {
UniqueTid utid = context_->process_tracker->UpdateThread(*reservation.tid,
*reservation.pid);
auto it_and_inserted =
descriptor_uuids_by_utid_.insert(std::make_pair<>(utid, uuid));
if (!it_and_inserted.second) {
// We already saw a another track with a different uuid for this thread.
// Since there should only be one descriptor track for each thread, we
// assume that its tid was reused. So, start a new thread.
uint64_t old_uuid = it_and_inserted.first->second;
PERFETTO_DCHECK(old_uuid != uuid); // Every track is only resolved once.
PERFETTO_DLOG("Detected tid reuse (pid: %" PRIu32 " tid: %" PRIu32
") from track descriptors (old uuid: %" PRIu64
" new uuid: %" PRIu64 " timestamp: %" PRId64 ")",
*reservation.pid, *reservation.tid, old_uuid, uuid,
reservation.min_timestamp);
utid = context_->process_tracker->StartNewThread(std::nullopt,
*reservation.tid);
// Associate the new thread with its process.
PERFETTO_CHECK(context_->process_tracker->UpdateThread(
*reservation.tid, *reservation.pid) == utid);
descriptor_uuids_by_utid_[utid] = uuid;
}
return ResolvedDescriptorTrack::Thread(utid, false /* is_counter */,
true /* is_root*/,
reservation.use_separate_track);
}
if (reservation.pid) {
UniquePid upid =
context_->process_tracker->GetOrCreateProcess(*reservation.pid);
auto it_and_inserted =
descriptor_uuids_by_upid_.insert(std::make_pair<>(upid, uuid));
if (!it_and_inserted.second) {
// We already saw a another track with a different uuid for this process.
// Since there should only be one descriptor track for each process, we
// assume that its pid was reused. So, start a new process.
uint64_t old_uuid = it_and_inserted.first->second;
PERFETTO_DCHECK(old_uuid != uuid); // Every track is only resolved once.
PERFETTO_DLOG("Detected pid reuse (pid: %" PRIu32
") from track descriptors (old uuid: %" PRIu64
" new uuid: %" PRIu64 " timestamp: %" PRId64 ")",
*reservation.pid, old_uuid, uuid,
reservation.min_timestamp);
upid = context_->process_tracker->StartNewProcess(
std::nullopt, std::nullopt, *reservation.pid, kNullStringId,
ThreadNamePriority::kTrackDescriptor);
descriptor_uuids_by_upid_[upid] = uuid;
}
return ResolvedDescriptorTrack::Process(upid, false /* is_counter */,
true /* is_root*/);
}
if (parent_resolved_track) {
switch (parent_resolved_track->scope()) {
case ResolvedDescriptorTrack::Scope::kThread:
// If parent is a thread track, create another thread-associated track.
return ResolvedDescriptorTrack::Thread(
parent_resolved_track->utid(), reservation.is_counter,
false /* is_root*/, parent_resolved_track->use_separate_track());
case ResolvedDescriptorTrack::Scope::kProcess:
// If parent is a process track, create another process-associated
// track.
return ResolvedDescriptorTrack::Process(parent_resolved_track->upid(),
reservation.is_counter,
false /* is_root*/);
case ResolvedDescriptorTrack::Scope::kGlobal:
break;
}
}
// Otherwise create a global track.
// The global track with no uuid is the default global track (e.g. for
// global instant events). Any other global tracks are considered children
// of the default track.
bool is_root_in_scope = !parent_resolved_track;
if (!parent_resolved_track && uuid) {
// Detect loops where the default track has a parent that itself is a
// global track (and thus should be parent of the default track).
if (descendent_uuids &&
std::find(descendent_uuids->begin(), descendent_uuids->end(),
kDefaultDescriptorTrackUuid) != descendent_uuids->end()) {
PERFETTO_ELOG(
"Loop detected in parent_track_uuid hierarchy at track %" PRIu64
" with parent %" PRIu64,
uuid, kDefaultDescriptorTrackUuid);
return std::nullopt;
}
// This track will be implicitly a child of the default global track.
is_root_in_scope = false;
}
return ResolvedDescriptorTrack::Global(reservation.is_counter,
is_root_in_scope);
}
TrackId TrackEventTracker::GetOrCreateDefaultDescriptorTrack() {
// If the default track was already reserved (e.g. because a producer emitted
// a descriptor for it) or created, resolve and return it.
std::optional<TrackId> track_id =
GetDescriptorTrack(kDefaultDescriptorTrackUuid);
if (track_id)
return *track_id;
// Otherwise reserve a new track and resolve it.
ReserveDescriptorChildTrack(kDefaultDescriptorTrackUuid, /*parent_uuid=*/0,
default_descriptor_track_name_);
return *GetDescriptorTrack(kDefaultDescriptorTrackUuid);
}
std::optional<double> TrackEventTracker::ConvertToAbsoluteCounterValue(
uint64_t counter_track_uuid,
uint32_t packet_sequence_id,
double value) {
auto reservation_it = reserved_descriptor_tracks_.find(counter_track_uuid);
if (reservation_it == reserved_descriptor_tracks_.end()) {
PERFETTO_DLOG("Unknown counter track with uuid %" PRIu64,
counter_track_uuid);
return std::nullopt;
}
DescriptorTrackReservation& reservation = reservation_it->second;
if (!reservation.is_counter) {
PERFETTO_DLOG("Track with uuid %" PRIu64 " is not a counter track",
counter_track_uuid);
return std::nullopt;
}
if (reservation.unit_multiplier > 0)
value *= static_cast<double>(reservation.unit_multiplier);
if (reservation.is_incremental) {
if (reservation.packet_sequence_id != packet_sequence_id) {
PERFETTO_DLOG(
"Incremental counter track with uuid %" PRIu64
" was updated from the wrong packet sequence (expected: %" PRIu32
" got:%" PRIu32 ")",
counter_track_uuid, reservation.packet_sequence_id,
packet_sequence_id);
return std::nullopt;
}
reservation.latest_value += value;
value = reservation.latest_value;
}
return value;
}
void TrackEventTracker::OnIncrementalStateCleared(uint32_t packet_sequence_id) {
// TODO(eseckler): Improve on the runtime complexity of this. At O(hundreds)
// of packet sequences, incremental state clearing at O(trace second), and
// total number of tracks in O(thousands), a linear scan through all tracks
// here might not be fast enough.
for (auto& entry : reserved_descriptor_tracks_) {
DescriptorTrackReservation& reservation = entry.second;
// Only consider incremental counter tracks for current sequence.
if (!reservation.is_counter || !reservation.is_incremental ||
reservation.packet_sequence_id != packet_sequence_id) {
continue;
}
// Reset their value to 0, see CounterDescriptor's |is_incremental|.
reservation.latest_value = 0;
}
}
void TrackEventTracker::OnFirstPacketOnSequence(uint32_t packet_sequence_id) {
sequences_with_first_packet_.insert(packet_sequence_id);
}
TrackEventTracker::ResolvedDescriptorTrack
TrackEventTracker::ResolvedDescriptorTrack::Process(UniquePid upid,
bool is_counter,
bool is_root) {
ResolvedDescriptorTrack track;
track.scope_ = Scope::kProcess;
track.is_counter_ = is_counter;
track.is_root_in_scope_ = is_root;
track.upid_ = upid;
return track;
}
TrackEventTracker::ResolvedDescriptorTrack
TrackEventTracker::ResolvedDescriptorTrack::Thread(UniqueTid utid,
bool is_counter,
bool is_root,
bool use_separate_track) {
ResolvedDescriptorTrack track;
track.scope_ = Scope::kThread;
track.is_counter_ = is_counter;
track.is_root_in_scope_ = is_root;
track.utid_ = utid;
track.use_separate_track_ = use_separate_track;
return track;
}
TrackEventTracker::ResolvedDescriptorTrack
TrackEventTracker::ResolvedDescriptorTrack::Global(bool is_counter,
bool is_root) {
ResolvedDescriptorTrack track;
track.scope_ = Scope::kGlobal;
track.is_counter_ = is_counter;
track.is_root_in_scope_ = is_root;
return track;
}
} // namespace trace_processor
} // namespace perfetto