blob: 895d14d45415f455c716e2953337349c9ee20ce4 [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 "perfetto/tracing/internal/track_event_internal.h"
#include "perfetto/base/proc_utils.h"
#include "perfetto/base/thread_utils.h"
#include "perfetto/base/time.h"
#include "perfetto/tracing/core/data_source_config.h"
#include "perfetto/tracing/track_event.h"
#include "perfetto/tracing/track_event_category_registry.h"
#include "perfetto/tracing/track_event_interned_data_index.h"
#include "protos/perfetto/common/data_source_descriptor.gen.h"
#include "protos/perfetto/trace/clock_snapshot.pbzero.h"
#include "protos/perfetto/trace/interned_data/interned_data.pbzero.h"
#include "protos/perfetto/trace/trace_packet_defaults.pbzero.h"
#include "protos/perfetto/trace/track_event/debug_annotation.pbzero.h"
#include "protos/perfetto/trace/track_event/track_descriptor.pbzero.h"
namespace perfetto {
namespace internal {
BaseTrackEventInternedDataIndex::~BaseTrackEventInternedDataIndex() = default;
namespace {
std::atomic<perfetto::base::PlatformThreadId> g_main_thread;
struct InternedEventCategory
: public TrackEventInternedDataIndex<
InternedEventCategory,
perfetto::protos::pbzero::InternedData::kEventCategoriesFieldNumber,
const char*,
SmallInternedDataTraits> {
static void Add(protos::pbzero::InternedData* interned_data,
size_t iid,
const char* value) {
auto category = interned_data->add_event_categories();
category->set_iid(iid);
category->set_name(value);
}
};
struct InternedEventName
: public TrackEventInternedDataIndex<
InternedEventName,
perfetto::protos::pbzero::InternedData::kEventNamesFieldNumber,
const char*,
SmallInternedDataTraits> {
static void Add(protos::pbzero::InternedData* interned_data,
size_t iid,
const char* value) {
auto name = interned_data->add_event_names();
name->set_iid(iid);
name->set_name(value);
}
};
struct InternedDebugAnnotationName
: public TrackEventInternedDataIndex<
InternedDebugAnnotationName,
perfetto::protos::pbzero::InternedData::
kDebugAnnotationNamesFieldNumber,
const char*,
SmallInternedDataTraits> {
static void Add(protos::pbzero::InternedData* interned_data,
size_t iid,
const char* value) {
auto name = interned_data->add_debug_annotation_names();
name->set_iid(iid);
name->set_name(value);
}
};
constexpr protos::pbzero::ClockSnapshot::Clock::BuiltinClocks GetClockType() {
#if !PERFETTO_BUILDFLAG(PERFETTO_OS_MACOSX) && \
!PERFETTO_BUILDFLAG(PERFETTO_OS_WIN)
return protos::pbzero::ClockSnapshot::Clock::BOOTTIME;
#else
return protos::pbzero::ClockSnapshot::Clock::MONOTONIC;
#endif
}
} // namespace
// static
bool TrackEventInternal::Initialize(
bool (*register_data_source)(const DataSourceDescriptor&)) {
if (!g_main_thread)
g_main_thread = perfetto::base::GetThreadId();
perfetto::DataSourceDescriptor dsd;
// TODO(skyostil): Advertise the known categories.
dsd.set_name("track_event");
return register_data_source(dsd);
}
// static
void TrackEventInternal::EnableTracing(
const TrackEventCategoryRegistry& registry,
const DataSourceConfig& config,
uint32_t instance_index) {
for (size_t i = 0; i < registry.category_count(); i++) {
// TODO(skyostil): Support the full category config syntax instead of
// just strict matching.
// TODO(skyostil): Support comma-separated categories.
if (config.legacy_config().empty() ||
config.legacy_config() == registry.GetCategory(i)->name) {
registry.EnableCategoryForInstance(i, instance_index);
}
}
}
// static
void TrackEventInternal::DisableTracing(
const TrackEventCategoryRegistry& registry,
uint32_t instance_index) {
for (size_t i = 0; i < registry.category_count(); i++)
registry.DisableCategoryForInstance(i, instance_index);
}
// static
uint64_t TrackEventInternal::GetTimeNs() {
if (GetClockType() == protos::pbzero::ClockSnapshot::Clock::BOOTTIME)
return static_cast<uint64_t>(perfetto::base::GetBootTimeNs().count());
PERFETTO_DCHECK(GetClockType() ==
protos::pbzero::ClockSnapshot::Clock::MONOTONIC);
return static_cast<uint64_t>(perfetto::base::GetWallTimeNs().count());
}
// static
void TrackEventInternal::ResetIncrementalState(TraceWriterBase* trace_writer,
uint64_t timestamp) {
auto default_track = ThreadTrack::Current();
{
// Mark any incremental state before this point invalid. Also set up
// defaults so that we don't need to repeat constant data for each packet.
auto packet = NewTracePacket(
trace_writer, timestamp,
protos::pbzero::TracePacket::SEQ_INCREMENTAL_STATE_CLEARED);
auto defaults = packet->set_trace_packet_defaults();
defaults->set_timestamp_clock_id(GetClockType());
// Establish the default track for this event sequence.
auto track_defaults = defaults->set_track_event_defaults();
track_defaults->set_track_uuid(default_track.uuid);
}
// Every thread should write a descriptor for its default track, because most
// trace points won't explicitly reference it.
WriteTrackDescriptor(default_track, trace_writer);
// Additionally the main thread should dump the process descriptor.
if (perfetto::base::GetThreadId() == g_main_thread)
WriteTrackDescriptor(ProcessTrack::Current(), trace_writer);
}
// static
protozero::MessageHandle<protos::pbzero::TracePacket>
TrackEventInternal::NewTracePacket(TraceWriterBase* trace_writer,
uint64_t timestamp,
uint32_t seq_flags) {
auto packet = trace_writer->NewTracePacket();
packet->set_timestamp(timestamp);
// TODO(skyostil): Stop emitting this for every event once the trace
// processor understands trace packet defaults.
if (GetClockType() != protos::pbzero::ClockSnapshot::Clock::BOOTTIME)
packet->set_timestamp_clock_id(GetClockType());
packet->set_sequence_flags(seq_flags);
return packet;
}
// static
EventContext TrackEventInternal::WriteEvent(
TraceWriterBase* trace_writer,
TrackEventIncrementalState* incr_state,
const char* category,
const char* name,
perfetto::protos::pbzero::TrackEvent::Type type,
uint64_t timestamp) {
PERFETTO_DCHECK(category);
PERFETTO_DCHECK(g_main_thread);
if (incr_state->was_cleared) {
incr_state->was_cleared = false;
ResetIncrementalState(trace_writer, timestamp);
}
auto packet = NewTracePacket(trace_writer, timestamp);
EventContext ctx(std::move(packet), incr_state);
auto track_event = ctx.event();
if (type != protos::pbzero::TrackEvent::TYPE_UNSPECIFIED)
track_event->set_type(type);
// We assume that |category| and |name| point to strings with static lifetime.
// This means we can use their addresses as interning keys.
if (type != protos::pbzero::TrackEvent::TYPE_SLICE_END) {
// TODO(skyostil): Handle multiple categories.
size_t category_iid = InternedEventCategory::Get(&ctx, category);
track_event->add_category_iids(category_iid);
}
if (name) {
size_t name_iid = InternedEventName::Get(&ctx, name);
track_event->set_name_iid(name_iid);
}
return ctx;
}
// static
protos::pbzero::DebugAnnotation* TrackEventInternal::AddDebugAnnotation(
perfetto::EventContext* event_ctx,
const char* name) {
auto annotation = event_ctx->event()->add_debug_annotations();
annotation->set_name_iid(InternedDebugAnnotationName::Get(event_ctx, name));
return annotation;
}
} // namespace internal
} // namespace perfetto