mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-12-13 09:52:38 +00:00
139 lines
3.8 KiB
C++
139 lines
3.8 KiB
C++
#pragma once
|
|
|
|
#include <Common/VariableContext.h>
|
|
#include "base/types.h"
|
|
#include <atomic>
|
|
#include <memory>
|
|
#include <stddef.h>
|
|
|
|
/** Implements global counters for various events happening in the application
|
|
* - for high level profiling.
|
|
* See .cpp for list of events.
|
|
*/
|
|
|
|
namespace ProfileEvents
|
|
{
|
|
/// Event identifier (index in array).
|
|
using Event = size_t;
|
|
using Count = size_t;
|
|
using Increment = Int64;
|
|
using Counter = std::atomic<Count>;
|
|
class Counters;
|
|
|
|
/// Counters - how many times each event happened
|
|
extern Counters global_counters;
|
|
|
|
class Counters
|
|
{
|
|
Counter * counters = nullptr;
|
|
std::unique_ptr<Counter[]> counters_holder;
|
|
/// Used to propagate increments
|
|
Counters * parent = nullptr;
|
|
|
|
public:
|
|
|
|
VariableContext level = VariableContext::Thread;
|
|
|
|
/// By default, any instance have to increment global counters
|
|
Counters(VariableContext level_ = VariableContext::Thread, Counters * parent_ = &global_counters);
|
|
|
|
/// Global level static initializer
|
|
Counters(Counter * allocated_counters) noexcept
|
|
: counters(allocated_counters), parent(nullptr), level(VariableContext::Global) {}
|
|
|
|
Counter & operator[] (Event event)
|
|
{
|
|
return counters[event];
|
|
}
|
|
|
|
const Counter & operator[] (Event event) const
|
|
{
|
|
return counters[event];
|
|
}
|
|
|
|
inline void increment(Event event, Count amount = 1)
|
|
{
|
|
Counters * current = this;
|
|
do
|
|
{
|
|
current->counters[event].fetch_add(amount, std::memory_order_relaxed);
|
|
current = current->parent;
|
|
} while (current != nullptr);
|
|
}
|
|
|
|
struct Snapshot
|
|
{
|
|
Snapshot();
|
|
Snapshot(Snapshot &&) = default;
|
|
|
|
Count operator[] (Event event) const noexcept
|
|
{
|
|
return counters_holder[event];
|
|
}
|
|
|
|
Snapshot & operator=(Snapshot &&) = default;
|
|
private:
|
|
std::unique_ptr<Count[]> counters_holder;
|
|
|
|
friend class Counters;
|
|
friend struct CountersIncrement;
|
|
};
|
|
|
|
/// Every single value is fetched atomically, but not all values as a whole.
|
|
Snapshot getPartiallyAtomicSnapshot() const;
|
|
|
|
/// Reset all counters to zero and reset parent.
|
|
void reset();
|
|
|
|
/// Get parent (thread unsafe)
|
|
Counters * getParent()
|
|
{
|
|
return parent;
|
|
}
|
|
|
|
/// Set parent (thread unsafe)
|
|
void setParent(Counters * parent_)
|
|
{
|
|
parent = parent_;
|
|
}
|
|
|
|
/// Set all counters to zero
|
|
void resetCounters();
|
|
|
|
static const Event num_counters;
|
|
};
|
|
|
|
/// Increment a counter for event. Thread-safe.
|
|
void increment(Event event, Count amount = 1);
|
|
|
|
/// Get name of event by identifier. Returns statically allocated string.
|
|
const char * getName(Event event);
|
|
|
|
/// Get description of event by identifier. Returns statically allocated string.
|
|
const char * getDocumentation(Event event);
|
|
|
|
/// Get index just after last event identifier.
|
|
Event end();
|
|
|
|
struct CountersIncrement
|
|
{
|
|
CountersIncrement() noexcept = default;
|
|
explicit CountersIncrement(Counters::Snapshot const & snapshot);
|
|
CountersIncrement(Counters::Snapshot const & after, Counters::Snapshot const & before);
|
|
|
|
CountersIncrement(CountersIncrement &&) = default;
|
|
CountersIncrement & operator=(CountersIncrement &&) = default;
|
|
|
|
Increment operator[](Event event) const noexcept
|
|
{
|
|
return increment_holder[event];
|
|
}
|
|
private:
|
|
void init();
|
|
|
|
static_assert(sizeof(Count) == sizeof(Increment), "Sizes of counter and increment differ");
|
|
|
|
std::unique_ptr<Increment[]> increment_holder;
|
|
};
|
|
}
|