blob: 792bc5e53e6df6781e992342bde2264dd55b4b3b [file] [log] [blame]
// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#ifndef BASE_TRACE_EVENT_TRACE_CATEGORY_H_
#define BASE_TRACE_EVENT_TRACE_CATEGORY_H_
#include <stdint.h>
namespace base {
namespace trace_event {
// Captures the state of an invidivual trace category. Nothing except tracing
// internals (e.g., TraceLog) is supposed to have non-const Category pointers.
struct TraceCategory {
// The TRACE_EVENT macros should only use this value as a bool.
// These enum values are effectively a public API and third_party projects
// depend on their value. Hence, never remove or recycle existing bits, unless
// you are sure that all the third-party projects that depend on this have
// been updated.
enum StateFlags : uint8_t {
ENABLED_FOR_RECORDING = 1 << 0,
// Not used anymore.
DEPRECATED_ENABLED_FOR_MONITORING = 1 << 1,
DEPRECATED_ENABLED_FOR_EVENT_CALLBACK = 1 << 2,
ENABLED_FOR_ETW_EXPORT = 1 << 3,
ENABLED_FOR_FILTERING = 1 << 4
};
static const TraceCategory* FromStatePtr(const uint8_t* state_ptr) {
static_assert(
offsetof(TraceCategory, state_) == 0,
"|state_| must be the first field of the TraceCategory class.");
return reinterpret_cast<const TraceCategory*>(state_ptr);
}
bool is_valid() const { return name_ != nullptr; }
void set_name(const char* name) { name_ = name; }
const char* name() const {
DCHECK(is_valid());
return name_;
}
// TODO(primiano): This is an intermediate solution to deal with the fact that
// today TRACE_EVENT* macros cache the state ptr. They should just cache the
// full TraceCategory ptr, which is immutable, and use these helper function
// here. This will get rid of the need of this awkward ptr getter completely.
const uint8_t* state_ptr() const {
return const_cast<const uint8_t*>(&state_);
}
uint8_t state() const {
return *const_cast<volatile const uint8_t*>(&state_);
}
bool is_enabled() const { return state() != 0; }
void set_state(uint8_t state) {
*const_cast<volatile uint8_t*>(&state_) = state;
}
void clear_state_flag(StateFlags flag) { set_state(state() & (~flag)); }
void set_state_flag(StateFlags flag) { set_state(state() | flag); }
uint32_t enabled_filters() const {
return *const_cast<volatile const uint32_t*>(&enabled_filters_);
}
bool is_filter_enabled(size_t index) const {
DCHECK(index < sizeof(enabled_filters_) * 8);
return (enabled_filters() & (1 << index)) != 0;
}
void set_enabled_filters(uint32_t enabled_filters) {
*const_cast<volatile uint32_t*>(&enabled_filters_) = enabled_filters;
}
void reset_for_testing() {
set_state(0);
set_enabled_filters(0);
}
// These fields should not be accessed directly, not even by tracing code.
// The only reason why these are not private is because it makes it impossible
// to have a global array of TraceCategory in category_registry.cc without
// creating initializers. See discussion on goo.gl/qhZN94 and
// crbug.com/{660967,660828}.
// The enabled state. TRACE_EVENT* macros will capture events if any of the
// flags here are set. Since TRACE_EVENTx macros are used in a lot of
// fast-paths, accesses to this field are non-barriered and racy by design.
// This field is mutated when starting/stopping tracing and we don't care
// about missing some events.
uint8_t state_;
// When ENABLED_FOR_FILTERING is set, this contains a bitmap to the
// corresponding filter (see event_filters.h).
uint32_t enabled_filters_;
// TraceCategory group names are long lived static strings.
const char* name_;
};
} // namespace trace_event
} // namespace base
#endif // BASE_TRACE_EVENT_TRACE_CATEGORY_H_