blob: c4bd3c713c4c55413cf5225cf2b9f1030f7a8fc3 [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.
*/
#ifndef SRC_TRACE_PROCESSOR_IMPORTERS_COMMON_TRACK_TRACKER_H_
#define SRC_TRACE_PROCESSOR_IMPORTERS_COMMON_TRACK_TRACKER_H_
#include "src/trace_processor/storage/trace_storage.h"
#include "src/trace_processor/types/trace_processor_context.h"
namespace perfetto {
namespace trace_processor {
// Tracks and stores tracks based on track types, ids and scopes.
class TrackTracker {
public:
explicit TrackTracker(TraceProcessorContext*);
// Interns a thread track into the storage.
TrackId InternThreadTrack(UniqueTid utid);
// Interns a process track into the storage.
TrackId InternProcessTrack(UniquePid upid);
// Interns a Fuchsia async track into the storage.
TrackId InternFuchsiaAsyncTrack(StringId name, int64_t correlation_id);
// Interns a given GPU track into the storage.
TrackId InternGpuTrack(const tables::GpuTrackTable::Row& row);
// Interns a legacy Chrome async event track into the storage.
TrackId InternLegacyChromeAsyncTrack(StringId name,
uint32_t upid,
int64_t source_id,
bool source_id_is_process_scoped,
StringId source_scope);
// Interns a Android async track into the storage.
TrackId InternAndroidAsyncTrack(StringId name,
UniquePid upid,
int64_t cookie);
// Interns a track for perf event stack samples, with process-wide grouping.
TrackId InternPerfStackTrack(UniquePid upid);
// Interns a track for legacy Chrome process-scoped instant events into the
// storage.
TrackId InternLegacyChromeProcessInstantTrack(UniquePid upid);
// Lazily creates the track for legacy Chrome global instant events.
TrackId GetOrCreateLegacyChromeGlobalInstantTrack();
// Associate a TrackDescriptor track identified by the given |uuid| with a
// process's |pid|. This is called during tokenization. If a reservation for
// the same |uuid| already exists, verifies that the present reservation
// matches the new one.
//
// The track will be resolved to the process track (see InternProcessTrack())
// upon the first call to GetDescriptorTrack() with the same |uuid|. At this
// time, |pid| will also be resolved to a |upid|.
void ReserveDescriptorProcessTrack(uint64_t uuid,
StringId name,
uint32_t pid,
int64_t timestamp);
// Associate a TrackDescriptor track identified by the given |uuid| with a
// thread's |pid| and |tid|. This is called during tokenization. If a
// reservation for the same |uuid| already exists, verifies that the present
// reservation matches the new one.
//
// The track will be resolved to the thread track (see InternThreadTrack())
// upon the first call to GetDescriptorTrack() with the same |uuid|. At this
// time, |pid| will also be resolved to a |upid|.
void ReserveDescriptorThreadTrack(uint64_t uuid,
uint64_t parent_uuid,
StringId name,
uint32_t pid,
uint32_t tid,
int64_t timestamp);
// Associate a TrackDescriptor track identified by the given |uuid| with a
// parent track (usually a process- or thread-associated track). This is
// called during tokenization. If a reservation for the same |uuid| already
// exists, will attempt to update it.
//
// The track will be created upon the first call to GetDescriptorTrack() with
// the same |uuid|. If |parent_uuid| is 0, the track will become a global
// track. Otherwise, it will become a new track of the same type as its parent
// track.
void ReserveDescriptorChildTrack(uint64_t uuid,
uint64_t parent_uuid,
StringId name);
// Associate a counter-type TrackDescriptor track identified by the given
// |uuid| with a parent track (usually a process or thread track). This is
// called during tokenization. If a reservation for the same |uuid| already
// exists, will attempt to update it. The provided |category| will be stored
// into the track's args.
//
// If |is_incremental| is true, the counter will only be valid on the packet
// sequence identified by |packet_sequence_id|. |unit_multiplier| is an
// optional multiplication factor applied to counter values. Values for the
// counter will be translated during tokenization via
// ConvertToAbsoluteCounterValue().
//
// The track will be created upon the first call to GetDescriptorTrack() with
// the same |uuid|. If |parent_uuid| is 0, the track will become a global
// track. Otherwise, it will become a new counter track for the same
// process/thread as its parent track.
void ReserveDescriptorCounterTrack(uint64_t uuid,
uint64_t parent_uuid,
StringId name,
StringId category,
int64_t unit_multiplier,
bool is_incremental,
uint32_t packet_sequence_id);
// Returns the ID of the track for the TrackDescriptor with the given |uuid|.
// This is called during parsing. The first call to GetDescriptorTrack() for
// each |uuid| resolves and inserts the track (and its parent tracks,
// following the parent_uuid chain recursively) based on reservations made for
// the |uuid|. Returns nullopt if no track for a descriptor with this |uuid|
// has been reserved.
base::Optional<TrackId> GetDescriptorTrack(uint64_t uuid);
// Returns the ID of the implicit trace-global default TrackDescriptor track.
TrackId GetOrCreateDefaultDescriptorTrack();
// Returns the ID of the implicit trace-global default track for triggers
// received by the service.
TrackId GetOrCreateTriggerTrack();
// Interns a global counter track into the storage.
TrackId InternGlobalCounterTrack(StringId name);
// Interns a counter track associated with a cpu into the storage.
TrackId InternCpuCounterTrack(StringId name, uint32_t cpu);
// Interns a counter track associated with a thread into the storage.
TrackId InternThreadCounterTrack(StringId name, UniqueTid utid);
// Interns a counter track associated with a process into the storage.
TrackId InternProcessCounterTrack(StringId name, UniquePid upid);
// Interns a counter track associated with an irq into the storage.
TrackId InternIrqCounterTrack(StringId name, int32_t irq);
// Interns a counter track associated with an softirq into the storage.
TrackId InternSoftirqCounterTrack(StringId name, int32_t softirq);
// Interns a counter track associated with a GPU into the storage.
TrackId InternGpuCounterTrack(StringId name, uint32_t gpu_id);
// Creates a counter track associated with a GPU into the storage.
TrackId CreateGpuCounterTrack(StringId name,
uint32_t gpu_id,
StringId description = StringId::Null(),
StringId unit = StringId::Null());
// Converts the given counter value to an absolute value in the unit of the
// counter, applying incremental delta encoding or unit multipliers as
// necessary. If the counter uses incremental encoding, |packet_sequence_id|
// must match the one in its track reservation. Returns base::nullopt if the
// counter track is unknown or an invalid |packet_sequence_id| was passed.
base::Optional<int64_t> ConvertToAbsoluteCounterValue(
uint64_t counter_track_uuid,
uint32_t packet_sequence_id,
int64_t value);
// Called by ProtoTraceTokenizer whenever incremental state is cleared on a
// packet sequence. Resets counter values for any incremental counters of
// the sequence identified by |packet_sequence_id|.
void OnIncrementalStateCleared(uint32_t packet_sequence_id);
private:
struct GpuTrackTuple {
StringId track_name;
StringId scope;
int64_t context_id;
friend bool operator<(const GpuTrackTuple& l, const GpuTrackTuple& r) {
return std::tie(l.track_name, l.scope, l.context_id) <
std::tie(r.track_name, r.scope, r.context_id);
}
};
struct ChromeTrackTuple {
base::Optional<int64_t> upid;
int64_t source_id = 0;
StringId source_scope = StringId::Null();
friend bool operator<(const ChromeTrackTuple& l,
const ChromeTrackTuple& r) {
return std::tie(l.source_id, l.upid, l.source_scope) <
std::tie(r.source_id, r.upid, r.source_scope);
}
};
struct AndroidAsyncTrackTuple {
UniquePid upid;
int64_t cookie;
StringId name;
friend bool operator<(const AndroidAsyncTrackTuple& l,
const AndroidAsyncTrackTuple& r) {
return std::tie(l.upid, l.cookie, l.name) <
std::tie(r.upid, r.cookie, r.name);
}
};
struct DescriptorTrackReservation {
uint64_t parent_uuid = 0;
base::Optional<uint32_t> pid;
base::Optional<uint32_t> tid;
int64_t min_timestamp = 0; // only set if |pid| and/or |tid| is set.
StringId name = kNullStringId;
// For counter tracks.
bool is_counter = false;
StringId category = kNullStringId;
int64_t unit_multiplier = 1;
bool is_incremental = false;
uint32_t packet_sequence_id = 0;
int64_t latest_value = 0;
// Whether |other| is a valid descriptor for this track reservation. A track
// should always remain nested underneath its original parent.
bool IsForSameTrack(const DescriptorTrackReservation& other) {
// Note that |min_timestamp|, |latest_value|, and |name| are ignored for
// this comparison.
return std::tie(parent_uuid, pid, tid, is_counter, category,
unit_multiplier, is_incremental, packet_sequence_id) ==
std::tie(other.parent_uuid, pid, tid, is_counter, category,
unit_multiplier, is_incremental, packet_sequence_id);
}
};
base::Optional<TrackId> GetDescriptorTrackImpl(
uint64_t uuid,
std::vector<uint64_t>* descendent_uuids = nullptr);
TrackId ResolveDescriptorTrack(uint64_t uuid,
const DescriptorTrackReservation&,
std::vector<uint64_t>* descendent_uuids);
static constexpr uint64_t kDefaultDescriptorTrackUuid = 0u;
std::map<UniqueTid, TrackId> thread_tracks_;
std::map<UniquePid, TrackId> process_tracks_;
std::map<int64_t /* correlation_id */, TrackId> fuchsia_async_tracks_;
std::map<GpuTrackTuple, TrackId> gpu_tracks_;
std::map<ChromeTrackTuple, TrackId> chrome_tracks_;
std::map<AndroidAsyncTrackTuple, TrackId> android_async_tracks_;
std::map<UniquePid, TrackId> chrome_process_instant_tracks_;
base::Optional<TrackId> chrome_global_instant_track_id_;
std::map<uint64_t /* uuid */, DescriptorTrackReservation>
reserved_descriptor_tracks_;
std::map<uint64_t /* uuid */, TrackId> resolved_descriptor_tracks_;
std::map<UniquePid, TrackId> perf_stack_tracks_;
std::map<StringId, TrackId> global_counter_tracks_by_name_;
std::map<std::pair<StringId, uint32_t>, TrackId> cpu_counter_tracks_;
std::map<std::pair<StringId, UniqueTid>, TrackId> utid_counter_tracks_;
std::map<std::pair<StringId, UniquePid>, TrackId> upid_counter_tracks_;
std::map<std::pair<StringId, int32_t>, TrackId> irq_counter_tracks_;
std::map<std::pair<StringId, int32_t>, TrackId> softirq_counter_tracks_;
std::map<std::pair<StringId, uint32_t>, TrackId> gpu_counter_tracks_;
// Stores the descriptor uuid used for the primary process/thread track
// for the given upid / utid. Used for pid/tid reuse detection.
std::map<UniquePid, uint64_t /*uuid*/> descriptor_uuids_by_upid_;
std::map<UniqueTid, uint64_t /*uuid*/> descriptor_uuids_by_utid_;
base::Optional<TrackId> trigger_track_id_;
const StringId source_key_ = kNullStringId;
const StringId source_id_key_ = kNullStringId;
const StringId source_id_is_process_scoped_key_ = kNullStringId;
const StringId source_scope_key_ = kNullStringId;
const StringId parent_track_id_key_ = kNullStringId;
const StringId category_key_ = kNullStringId;
const StringId fuchsia_source_ = kNullStringId;
const StringId chrome_source_ = kNullStringId;
const StringId android_source_ = kNullStringId;
const StringId descriptor_source_ = kNullStringId;
const StringId default_descriptor_track_name_ = kNullStringId;
TraceProcessorContext* const context_;
};
} // namespace trace_processor
} // namespace perfetto
#endif // SRC_TRACE_PROCESSOR_IMPORTERS_COMMON_TRACK_TRACKER_H_