| /* |
| * 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/trace_processor/track_tracker.h" |
| |
| #include "src/trace_processor/args_tracker.h" |
| #include "src/trace_processor/process_tracker.h" |
| |
| namespace perfetto { |
| namespace trace_processor { |
| |
| // static |
| constexpr uint64_t TrackTracker::kDefaultDescriptorTrackUuid; |
| |
| TrackTracker::TrackTracker(TraceProcessorContext* context) |
| : source_key_(context->storage->InternString("source")), |
| source_id_key_(context->storage->InternString("source_id")), |
| source_id_is_process_scoped_key_( |
| context->storage->InternString("source_id_is_process_scoped")), |
| source_scope_key_(context->storage->InternString("source_scope")), |
| fuchsia_source_(context->storage->InternString("fuchsia")), |
| chrome_source_(context->storage->InternString("chrome")), |
| android_source_(context->storage->InternString("android")), |
| descriptor_source_(context->storage->InternString("descriptor")), |
| default_descriptor_track_name_( |
| context->storage->InternString("Default Track")), |
| context_(context) {} |
| |
| TrackId TrackTracker::InternThreadTrack(UniqueTid utid) { |
| auto it = thread_tracks_.find(utid); |
| if (it != thread_tracks_.end()) |
| return it->second; |
| |
| tables::ThreadTrackTable::Row row; |
| row.utid = utid; |
| auto id = context_->storage->mutable_thread_track_table()->Insert(row); |
| thread_tracks_[utid] = id; |
| return id; |
| } |
| |
| TrackId TrackTracker::InternProcessTrack(UniquePid upid) { |
| auto it = process_tracks_.find(upid); |
| if (it != process_tracks_.end()) |
| return it->second; |
| |
| tables::ProcessTrackTable::Row row; |
| row.upid = upid; |
| auto id = context_->storage->mutable_process_track_table()->Insert(row); |
| process_tracks_[upid] = id; |
| return id; |
| } |
| |
| TrackId TrackTracker::InternFuchsiaAsyncTrack(StringId name, |
| int64_t correlation_id) { |
| auto it = fuchsia_async_tracks_.find(correlation_id); |
| if (it != fuchsia_async_tracks_.end()) |
| return it->second; |
| |
| tables::TrackTable::Row row(name); |
| auto id = context_->storage->mutable_track_table()->Insert(row); |
| fuchsia_async_tracks_[correlation_id] = id; |
| |
| context_->args_tracker->AddArgsTo(id) |
| .AddArg(source_key_, Variadic::String(fuchsia_source_)) |
| .AddArg(source_id_key_, Variadic::Integer(correlation_id)); |
| |
| return id; |
| } |
| |
| TrackId TrackTracker::InternGpuTrack(const tables::GpuTrackTable::Row& row) { |
| GpuTrackTuple tuple{row.name.id, row.scope, row.context_id.value_or(0)}; |
| |
| auto it = gpu_tracks_.find(tuple); |
| if (it != gpu_tracks_.end()) |
| return it->second; |
| |
| auto id = context_->storage->mutable_gpu_track_table()->Insert(row); |
| gpu_tracks_[tuple] = id; |
| return id; |
| } |
| |
| TrackId TrackTracker::InternLegacyChromeAsyncTrack( |
| StringId name, |
| uint32_t upid, |
| int64_t source_id, |
| bool source_id_is_process_scoped, |
| StringId source_scope) { |
| ChromeTrackTuple tuple; |
| if (source_id_is_process_scoped) |
| tuple.upid = upid; |
| tuple.source_id = source_id; |
| tuple.source_scope = source_scope; |
| |
| auto it = chrome_tracks_.find(tuple); |
| if (it != chrome_tracks_.end()) |
| return it->second; |
| |
| // Legacy async tracks are always drawn in the context of a process, even if |
| // the ID's scope is global. |
| tables::ProcessTrackTable::Row track(name); |
| track.upid = upid; |
| TrackId id = context_->storage->mutable_process_track_table()->Insert(track); |
| chrome_tracks_[tuple] = id; |
| |
| context_->args_tracker->AddArgsTo(id) |
| .AddArg(source_key_, Variadic::String(chrome_source_)) |
| .AddArg(source_id_key_, Variadic::Integer(source_id)) |
| .AddArg(source_id_is_process_scoped_key_, |
| Variadic::Boolean(source_id_is_process_scoped)) |
| .AddArg(source_scope_key_, Variadic::String(source_scope)); |
| |
| return id; |
| } |
| |
| TrackId TrackTracker::InternAndroidAsyncTrack(StringId name, |
| UniquePid upid, |
| int64_t cookie) { |
| AndroidAsyncTrackTuple tuple{upid, cookie, name}; |
| |
| auto it = android_async_tracks_.find(tuple); |
| if (it != android_async_tracks_.end()) |
| return it->second; |
| |
| tables::ProcessTrackTable::Row row(name); |
| row.upid = upid; |
| auto id = context_->storage->mutable_process_track_table()->Insert(row); |
| android_async_tracks_[tuple] = id; |
| |
| context_->args_tracker->AddArgsTo(id) |
| .AddArg(source_key_, Variadic::String(android_source_)) |
| .AddArg(source_id_key_, Variadic::Integer(cookie)); |
| |
| return id; |
| } |
| |
| TrackId TrackTracker::InternLegacyChromeProcessInstantTrack(UniquePid upid) { |
| auto it = chrome_process_instant_tracks_.find(upid); |
| if (it != chrome_process_instant_tracks_.end()) |
| return it->second; |
| |
| tables::ProcessTrackTable::Row row; |
| row.upid = upid; |
| auto id = context_->storage->mutable_process_track_table()->Insert(row); |
| chrome_process_instant_tracks_[upid] = id; |
| |
| context_->args_tracker->AddArgsTo(id).AddArg( |
| source_key_, Variadic::String(chrome_source_)); |
| |
| return id; |
| } |
| |
| TrackId TrackTracker::GetOrCreateLegacyChromeGlobalInstantTrack() { |
| if (!chrome_global_instant_track_id_) { |
| chrome_global_instant_track_id_ = |
| context_->storage->mutable_track_table()->Insert({}); |
| |
| context_->args_tracker->AddArgsTo(*chrome_global_instant_track_id_) |
| .AddArg(source_key_, Variadic::String(chrome_source_)); |
| } |
| return *chrome_global_instant_track_id_; |
| } |
| |
| void TrackTracker::ReserveDescriptorProcessTrack(uint64_t uuid, |
| uint32_t pid, |
| int64_t timestamp) { |
| DescriptorTrackReservation reservation; |
| reservation.min_timestamp = timestamp; |
| reservation.pid = pid; |
| |
| 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 TrackTracker::ReserveDescriptorThreadTrack(uint64_t uuid, |
| uint64_t parent_uuid, |
| uint32_t pid, |
| uint32_t tid, |
| int64_t timestamp) { |
| DescriptorTrackReservation reservation; |
| reservation.min_timestamp = timestamp; |
| reservation.parent_uuid = parent_uuid; |
| reservation.pid = pid; |
| reservation.tid = tid; |
| |
| 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 TrackTracker::ReserveDescriptorChildTrack(uint64_t uuid, |
| uint64_t parent_uuid) { |
| DescriptorTrackReservation reservation; |
| reservation.parent_uuid = parent_uuid; |
| |
| 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); |
| } |
| |
| base::Optional<TrackId> TrackTracker::GetDescriptorTrack(uint64_t uuid) { |
| auto it = resolved_descriptor_tracks_.find(uuid); |
| if (it == resolved_descriptor_tracks_.end()) { |
| auto reservation_it = reserved_descriptor_tracks_.find(uuid); |
| if (reservation_it == reserved_descriptor_tracks_.end()) |
| return base::nullopt; |
| TrackId track_id = ResolveDescriptorTrack(uuid, reservation_it->second); |
| resolved_descriptor_tracks_[uuid] = track_id; |
| return track_id; |
| } |
| return it->second; |
| } |
| |
| TrackId TrackTracker::ResolveDescriptorTrack( |
| uint64_t uuid, |
| const DescriptorTrackReservation& reservation) { |
| base::Optional<TrackId> parent_track_id; |
| if (reservation.parent_uuid) { |
| // Ensure that parent track is resolved. |
| parent_track_id = GetDescriptorTrack(reservation.parent_uuid); |
| if (!parent_track_id) { |
| PERFETTO_ELOG("Unknown parent track %" PRIu64 " for track %" PRIu64, |
| reservation.parent_uuid, uuid); |
| } |
| } |
| |
| 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( |
| base::nullopt, *reservation.tid, kNullStringId); |
| |
| // 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 InternThreadTrack(utid); |
| } |
| |
| 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( |
| base::nullopt, base::nullopt, *reservation.pid, kNullStringId); |
| |
| descriptor_uuids_by_upid_[upid] = uuid; |
| } |
| return InternProcessTrack(upid); |
| } |
| |
| base::Optional<TrackId> track_id; |
| if (parent_track_id) { |
| // If parent is a thread track, create another thread-associated track. |
| base::Optional<uint32_t> thread_track_index = |
| context_->storage->thread_track_table().id().IndexOf(*parent_track_id); |
| if (thread_track_index) { |
| auto* thread_tracks = context_->storage->mutable_thread_track_table(); |
| tables::ThreadTrackTable::Row row; |
| row.utid = thread_tracks->utid()[*thread_track_index]; |
| track_id = thread_tracks->Insert(row); |
| } else { |
| // If parent is a process track, create another process-associated track. |
| base::Optional<uint32_t> process_track_index = |
| context_->storage->process_track_table().id().IndexOf( |
| *parent_track_id); |
| if (process_track_index) { |
| auto* process_tracks = context_->storage->mutable_process_track_table(); |
| tables::ProcessTrackTable::Row track; |
| track.upid = process_tracks->upid()[*process_track_index]; |
| track_id = process_tracks->Insert(track); |
| } |
| } |
| } |
| |
| // Otherwise create a global track. |
| if (!track_id) { |
| tables::TrackTable::Row track; |
| track_id = context_->storage->mutable_track_table()->Insert(track); |
| } |
| |
| context_->args_tracker->AddArgsTo(*track_id) |
| .AddArg(source_key_, Variadic::String(descriptor_source_)) |
| .AddArg(source_id_key_, Variadic::Integer(static_cast<int64_t>(uuid))); |
| return *track_id; |
| } |
| |
| TrackId TrackTracker::GetOrCreateDefaultDescriptorTrack() { |
| // If the default track was already reserved (e.g. because a producer emitted |
| // a descriptor for it) or created, resolve and return it. |
| base::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); |
| track_id = GetDescriptorTrack(kDefaultDescriptorTrackUuid); |
| |
| auto* tracks = context_->storage->mutable_track_table(); |
| tracks->mutable_name()->Set(*tracks->id().IndexOf(*track_id), |
| default_descriptor_track_name_); |
| return *track_id; |
| } |
| |
| TrackId TrackTracker::InternGlobalCounterTrack(StringId name) { |
| auto it = global_counter_tracks_by_name_.find(name); |
| if (it != global_counter_tracks_by_name_.end()) { |
| return it->second; |
| } |
| |
| tables::CounterTrackTable::Row row(name); |
| TrackId track = context_->storage->mutable_counter_track_table()->Insert(row); |
| global_counter_tracks_by_name_[name] = track; |
| return track; |
| } |
| |
| TrackId TrackTracker::InternCpuCounterTrack(StringId name, uint32_t cpu) { |
| auto it = cpu_counter_tracks_.find(std::make_pair(name, cpu)); |
| if (it != cpu_counter_tracks_.end()) { |
| return it->second; |
| } |
| |
| tables::CpuCounterTrackTable::Row row(name); |
| row.cpu = cpu; |
| |
| TrackId track = |
| context_->storage->mutable_cpu_counter_track_table()->Insert(row); |
| cpu_counter_tracks_[std::make_pair(name, cpu)] = track; |
| return track; |
| } |
| |
| TrackId TrackTracker::InternThreadCounterTrack(StringId name, UniqueTid utid) { |
| auto it = utid_counter_tracks_.find(std::make_pair(name, utid)); |
| if (it != utid_counter_tracks_.end()) { |
| return it->second; |
| } |
| |
| tables::ThreadCounterTrackTable::Row row(name); |
| row.utid = utid; |
| |
| TrackId track = |
| context_->storage->mutable_thread_counter_track_table()->Insert(row); |
| utid_counter_tracks_[std::make_pair(name, utid)] = track; |
| return track; |
| } |
| |
| TrackId TrackTracker::InternProcessCounterTrack(StringId name, UniquePid upid) { |
| auto it = upid_counter_tracks_.find(std::make_pair(name, upid)); |
| if (it != upid_counter_tracks_.end()) { |
| return it->second; |
| } |
| |
| tables::ProcessCounterTrackTable::Row row(name); |
| row.upid = upid; |
| |
| TrackId track = |
| context_->storage->mutable_process_counter_track_table()->Insert(row); |
| upid_counter_tracks_[std::make_pair(name, upid)] = track; |
| return track; |
| } |
| |
| TrackId TrackTracker::InternIrqCounterTrack(StringId name, int32_t irq) { |
| auto it = irq_counter_tracks_.find(std::make_pair(name, irq)); |
| if (it != irq_counter_tracks_.end()) { |
| return it->second; |
| } |
| |
| tables::IrqCounterTrackTable::Row row(name); |
| row.irq = irq; |
| |
| TrackId track = |
| context_->storage->mutable_irq_counter_track_table()->Insert(row); |
| irq_counter_tracks_[std::make_pair(name, irq)] = track; |
| return track; |
| } |
| |
| TrackId TrackTracker::InternSoftirqCounterTrack(StringId name, |
| int32_t softirq) { |
| auto it = softirq_counter_tracks_.find(std::make_pair(name, softirq)); |
| if (it != softirq_counter_tracks_.end()) { |
| return it->second; |
| } |
| |
| tables::SoftirqCounterTrackTable::Row row(name); |
| row.softirq = softirq; |
| |
| TrackId track = |
| context_->storage->mutable_softirq_counter_track_table()->Insert(row); |
| softirq_counter_tracks_[std::make_pair(name, softirq)] = track; |
| return track; |
| } |
| |
| TrackId TrackTracker::InternGpuCounterTrack(StringId name, uint32_t gpu_id) { |
| auto it = gpu_counter_tracks_.find(std::make_pair(name, gpu_id)); |
| if (it != gpu_counter_tracks_.end()) { |
| return it->second; |
| } |
| TrackId track = CreateGpuCounterTrack(name, gpu_id); |
| gpu_counter_tracks_[std::make_pair(name, gpu_id)] = track; |
| return track; |
| } |
| |
| TrackId TrackTracker::CreateGpuCounterTrack(StringId name, |
| uint32_t gpu_id, |
| StringId description, |
| StringId unit) { |
| tables::GpuCounterTrackTable::Row row(name); |
| row.gpu_id = gpu_id; |
| row.description = description; |
| row.unit = unit; |
| |
| return context_->storage->mutable_gpu_counter_track_table()->Insert(row); |
| } |
| |
| } // namespace trace_processor |
| } // namespace perfetto |