mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-11-28 18:42:26 +00:00
thread-local perf events
This commit is contained in:
parent
40b704fb61
commit
6e3bbf83e3
@ -119,6 +119,8 @@ void TasksStatsCounters::incrementProfileEvents(const ::taskstats & prev, const
|
|||||||
profile_events.increment(ProfileEvents::OSWriteBytes, safeDiff(prev.write_bytes, curr.write_bytes));
|
profile_events.increment(ProfileEvents::OSWriteBytes, safeDiff(prev.write_bytes, curr.write_bytes));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
thread_local PerfEventsCounters current_thread_counters;
|
||||||
|
|
||||||
#define SOFTWARE_EVENT(PERF_NAME, LOCAL_NAME) \
|
#define SOFTWARE_EVENT(PERF_NAME, LOCAL_NAME) \
|
||||||
PerfEventInfo \
|
PerfEventInfo \
|
||||||
{ \
|
{ \
|
||||||
@ -138,7 +140,7 @@ void TasksStatsCounters::incrementProfileEvents(const ::taskstats & prev, const
|
|||||||
}
|
}
|
||||||
|
|
||||||
// descriptions' source: http://man7.org/linux/man-pages/man2/perf_event_open.2.html
|
// descriptions' source: http://man7.org/linux/man-pages/man2/perf_event_open.2.html
|
||||||
const PerfEventInfo PerfEventsCounters::raw_events_info[] = {
|
static const PerfEventInfo raw_events_info[] = {
|
||||||
HARDWARE_EVENT(PERF_COUNT_HW_CPU_CYCLES, PerfCpuCycles),
|
HARDWARE_EVENT(PERF_COUNT_HW_CPU_CYCLES, PerfCpuCycles),
|
||||||
HARDWARE_EVENT(PERF_COUNT_HW_INSTRUCTIONS, PerfInstructions),
|
HARDWARE_EVENT(PERF_COUNT_HW_INSTRUCTIONS, PerfInstructions),
|
||||||
HARDWARE_EVENT(PERF_COUNT_HW_CACHE_REFERENCES, PerfCacheReferences),
|
HARDWARE_EVENT(PERF_COUNT_HW_CACHE_REFERENCES, PerfCacheReferences),
|
||||||
@ -159,12 +161,7 @@ const PerfEventInfo PerfEventsCounters::raw_events_info[] = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#undef HARDWARE_EVENT
|
#undef HARDWARE_EVENT
|
||||||
|
#undef SOFTWARE_EVENT
|
||||||
std::atomic<PerfEventsCounters::Id> PerfEventsCounters::latest_counters_id = 0;
|
|
||||||
|
|
||||||
thread_local PerfDescriptorsHolder PerfEventsCounters::thread_events_descriptors_holder{};
|
|
||||||
thread_local std::optional<PerfEventsCounters::Id> PerfEventsCounters::current_thread_counters_id = std::nullopt;
|
|
||||||
thread_local std::optional<PerfEventsCounters::ParsedEvents> PerfEventsCounters::last_parsed_events = std::nullopt;
|
|
||||||
|
|
||||||
Logger * PerfEventsCounters::getLogger()
|
Logger * PerfEventsCounters::getLogger()
|
||||||
{
|
{
|
||||||
@ -193,48 +190,41 @@ static int openPerfEventDisabled(Int32 perf_event_paranoid, bool has_cap_sys_adm
|
|||||||
|
|
||||||
using getLoggerFunc = Logger * ();
|
using getLoggerFunc = Logger * ();
|
||||||
|
|
||||||
static void enablePerfEvent(int event_fd, getLoggerFunc getLogger)
|
static void enablePerfEvent(int event_fd)
|
||||||
{
|
{
|
||||||
if (ioctl(event_fd, PERF_EVENT_IOC_ENABLE, 0))
|
if (ioctl(event_fd, PERF_EVENT_IOC_ENABLE, 0))
|
||||||
LOG_WARNING(getLogger(), "Can't enable perf event with file descriptor {}", event_fd);
|
LOG_WARNING(&Poco::Logger::get(__PRETTY_FUNCTION__), "Can't enable perf event with file descriptor {}", event_fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void resetPerfEvent(int event_fd, getLoggerFunc getLogger)
|
static void disablePerfEvent(int event_fd)
|
||||||
{
|
|
||||||
if (ioctl(event_fd, PERF_EVENT_IOC_RESET, 0))
|
|
||||||
LOG_WARNING(getLogger(), "Can't reset perf event with file descriptor {}",
|
|
||||||
event_fd);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void disablePerfEvent(int event_fd, getLoggerFunc getLogger)
|
|
||||||
{
|
{
|
||||||
if (ioctl(event_fd, PERF_EVENT_IOC_DISABLE, 0))
|
if (ioctl(event_fd, PERF_EVENT_IOC_DISABLE, 0))
|
||||||
LOG_WARNING(getLogger(), "Can't disable perf event with file descriptor {}" , event_fd);
|
LOG_WARNING(&Poco::Logger::get(__PRETTY_FUNCTION__), "Can't disable perf event with file descriptor {}" , event_fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void releasePerfEvent(int event_fd, getLoggerFunc getLogger)
|
static void releasePerfEvent(int event_fd)
|
||||||
{
|
{
|
||||||
if (close(event_fd))
|
if (close(event_fd))
|
||||||
{
|
{
|
||||||
LOG_WARNING(getLogger(), "Can't close perf event file descriptor {}: {} ({})", event_fd, errno, strerror(errno));
|
LOG_WARNING(&Poco::Logger::get(__PRETTY_FUNCTION__), "Can't close perf event file descriptor {}: {} ({})", event_fd, errno, strerror(errno));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool validatePerfEventDescriptor(int & fd, getLoggerFunc getLogger)
|
static bool validatePerfEventDescriptor(int & fd)
|
||||||
{
|
{
|
||||||
if (fcntl(fd, F_GETFL) != -1)
|
if (fcntl(fd, F_GETFL) != -1)
|
||||||
return true;
|
return true;
|
||||||
|
|
||||||
if (errno == EBADF)
|
if (errno == EBADF)
|
||||||
{
|
{
|
||||||
LOG_WARNING(getLogger(), "Event descriptor {} was closed from the outside; reopening", fd);
|
LOG_WARNING(&Poco::Logger::get(__PRETTY_FUNCTION__), "Event descriptor {} was closed from the outside; reopening", fd);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
LOG_WARNING(getLogger(), "Error while checking availability of event descriptor {}: {} ({})", fd, strerror(errno), errno);
|
LOG_WARNING(&Poco::Logger::get(__PRETTY_FUNCTION__), "Error while checking availability of event descriptor {}: {} ({})", fd, strerror(errno), errno);
|
||||||
|
|
||||||
disablePerfEvent(fd, getLogger);
|
disablePerfEvent(fd);
|
||||||
releasePerfEvent(fd, getLogger);
|
releasePerfEvent(fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
fd = -1;
|
fd = -1;
|
||||||
@ -264,8 +254,12 @@ bool PerfEventsCounters::processThreadLocalChanges(const std::string & needed_ev
|
|||||||
|
|
||||||
if (old_one == new_one)
|
if (old_one == new_one)
|
||||||
{
|
{
|
||||||
if (old_one && !validatePerfEventDescriptor(thread_events_descriptors_holder.descriptors[i], getLogger))
|
if (old_one
|
||||||
|
&& !validatePerfEventDescriptor(
|
||||||
|
thread_events_descriptors_holder.descriptors[i]))
|
||||||
|
{
|
||||||
events_to_open.push_back(i);
|
events_to_open.push_back(i);
|
||||||
|
}
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -279,8 +273,8 @@ bool PerfEventsCounters::processThreadLocalChanges(const std::string & needed_ev
|
|||||||
for (size_t i : events_to_release)
|
for (size_t i : events_to_release)
|
||||||
{
|
{
|
||||||
int & fd = thread_events_descriptors_holder.descriptors[i];
|
int & fd = thread_events_descriptors_holder.descriptors[i];
|
||||||
disablePerfEvent(fd, getLogger);
|
disablePerfEvent(fd);
|
||||||
releasePerfEvent(fd, getLogger);
|
releasePerfEvent(fd);
|
||||||
fd = -1;
|
fd = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -365,21 +359,20 @@ std::vector<size_t> PerfEventsCounters::eventIndicesFromString(const std::string
|
|||||||
}
|
}
|
||||||
|
|
||||||
std::vector<size_t> result;
|
std::vector<size_t> result;
|
||||||
result.reserve(PerfEventsCounters::NUMBER_OF_RAW_EVENTS);
|
result.reserve(NUMBER_OF_RAW_EVENTS);
|
||||||
if (requested_events.size() == 0
|
if (requested_events.empty()
|
||||||
|| requested_events.count("all") > 0)
|
|| requested_events.count("all") > 0)
|
||||||
{
|
{
|
||||||
for (size_t i = 0; i < PerfEventsCounters::NUMBER_OF_RAW_EVENTS; ++i)
|
for (size_t i = 0; i < NUMBER_OF_RAW_EVENTS; ++i)
|
||||||
{
|
{
|
||||||
result.push_back(i);
|
result.push_back(i);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (size_t i = 0; i < PerfEventsCounters::NUMBER_OF_RAW_EVENTS; ++i)
|
for (size_t i = 0; i < NUMBER_OF_RAW_EVENTS; ++i)
|
||||||
{
|
{
|
||||||
if (requested_events.count(
|
if (requested_events.count(raw_events_info[i].settings_name) > 0)
|
||||||
PerfEventsCounters::raw_events_info[i].settings_name) > 0)
|
|
||||||
{
|
{
|
||||||
result.push_back(i);
|
result.push_back(i);
|
||||||
}
|
}
|
||||||
@ -388,15 +381,8 @@ std::vector<size_t> PerfEventsCounters::eventIndicesFromString(const std::string
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
void PerfEventsCounters::initializeProfileEvents(PerfEventsCounters & counters, const std::string & events_list)
|
void PerfEventsCounters::initializeProfileEvents(const std::string & events_list)
|
||||||
{
|
{
|
||||||
if (current_thread_counters_id.has_value())
|
|
||||||
{
|
|
||||||
if (current_thread_counters_id != counters.id)
|
|
||||||
LOG_WARNING(getLogger(), "Only one instance of `PerfEventsCounters` can be used on the thread");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!processThreadLocalChanges(events_list))
|
if (!processThreadLocalChanges(events_list))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -405,77 +391,72 @@ void PerfEventsCounters::initializeProfileEvents(PerfEventsCounters & counters,
|
|||||||
if (fd == -1)
|
if (fd == -1)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
resetPerfEvent(fd, getLogger);
|
// We don't reset the event, because the time_running and time_enabled
|
||||||
enablePerfEvent(fd, getLogger);
|
// can't be reset anyway and we have to calculate deltas.
|
||||||
|
enablePerfEvent(fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
current_thread_counters_id = counters.id;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void PerfEventsCounters::finalizeProfileEvents(PerfEventsCounters & counters, ProfileEvents::Counters & profile_events)
|
void PerfEventsCounters::finalizeProfileEvents(ProfileEvents::Counters & profile_events)
|
||||||
{
|
{
|
||||||
if (current_thread_counters_id != counters.id)
|
// Disable all perf events.
|
||||||
return;
|
|
||||||
|
|
||||||
const auto old_values = counters.raw_event_values;
|
|
||||||
|
|
||||||
// only read counters here to have as little overhead for processing as possible
|
|
||||||
for (size_t i = 0; i < NUMBER_OF_RAW_EVENTS; ++i)
|
for (size_t i = 0; i < NUMBER_OF_RAW_EVENTS; ++i)
|
||||||
{
|
{
|
||||||
int fd = counters.thread_events_descriptors_holder.descriptors[i];
|
int fd = thread_events_descriptors_holder.descriptors[i];
|
||||||
|
if (fd == -1)
|
||||||
|
continue;
|
||||||
|
disablePerfEvent(fd);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read the counter values.
|
||||||
|
PerfEventValue current_values[NUMBER_OF_RAW_EVENTS];
|
||||||
|
for (size_t i = 0; i < NUMBER_OF_RAW_EVENTS; ++i)
|
||||||
|
{
|
||||||
|
int fd = thread_events_descriptors_holder.descriptors[i];
|
||||||
if (fd == -1)
|
if (fd == -1)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
constexpr ssize_t bytes_to_read = sizeof(counters.raw_event_values[0]);
|
constexpr ssize_t bytes_to_read = sizeof(current_values[0]);
|
||||||
const int bytes_read = read(fd, &counters.raw_event_values[i],
|
const int bytes_read = read(fd, ¤t_values[i], bytes_to_read);
|
||||||
bytes_to_read);
|
|
||||||
|
|
||||||
if (bytes_read != bytes_to_read)
|
if (bytes_read != bytes_to_read)
|
||||||
{
|
{
|
||||||
LOG_WARNING(getLogger(), "Can't read event value from file descriptor: {}", fd);
|
LOG_WARNING(getLogger(), "Can't read event value from file descriptor: {}", fd);
|
||||||
counters.raw_event_values[i] = {};
|
current_values[i] = {};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// actually process counters' values and stop measuring
|
// actually process counters' values
|
||||||
for (size_t i = 0; i < NUMBER_OF_RAW_EVENTS; ++i)
|
for (size_t i = 0; i < NUMBER_OF_RAW_EVENTS; ++i)
|
||||||
{
|
{
|
||||||
int fd = counters.thread_events_descriptors_holder.descriptors[i];
|
int fd = thread_events_descriptors_holder.descriptors[i];
|
||||||
if (fd == -1)
|
if (fd == -1)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
const PerfEventInfo & info = raw_events_info[i];
|
const PerfEventInfo & info = raw_events_info[i];
|
||||||
const PerfEventValue & old_value = old_values[i];
|
const PerfEventValue & previous_value = previous_values[i];
|
||||||
const PerfEventValue & new_value = counters.raw_event_values[i];
|
const PerfEventValue & current_value = current_values[i];
|
||||||
|
|
||||||
// Account for counter multiplexing. time_running/time_enabled are
|
// Account for counter multiplexing. time_running and time_enabled are
|
||||||
// not reset by PERF_EVENT_IOC_RESET, so we have to calculate deltas
|
// not reset by PERF_EVENT_IOC_RESET, so we don't use it and calculate
|
||||||
// from old values.
|
// deltas from old values.
|
||||||
profile_events.increment(info.profile_event,
|
const UInt64 delta = (current_value.value - previous_value.value)
|
||||||
(new_value.value - old_value.value)
|
* (current_value.time_enabled - previous_value.time_enabled)
|
||||||
* (new_value.time_running - old_value.time_running)
|
/ std::max(1.f,
|
||||||
/ std::max(1.f,
|
float(current_value.time_running - previous_value.time_running));
|
||||||
float(new_value.time_enabled - old_value.time_enabled)));
|
|
||||||
|
|
||||||
disablePerfEvent(fd, getLogger);
|
profile_events.increment(info.profile_event, delta);
|
||||||
}
|
}
|
||||||
|
|
||||||
current_thread_counters_id.reset();
|
// Store current counter values for the next profiling period.
|
||||||
|
memcpy(previous_values, current_values, sizeof(current_values));
|
||||||
}
|
}
|
||||||
|
|
||||||
void PerfEventsCounters::closeEventDescriptors()
|
void PerfEventsCounters::closeEventDescriptors()
|
||||||
{
|
{
|
||||||
if (current_thread_counters_id.has_value())
|
|
||||||
{
|
|
||||||
LOG_WARNING(getLogger(), "Tried to close event descriptors while measurements are in process; ignoring");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
thread_events_descriptors_holder.releaseResources();
|
thread_events_descriptors_holder.releaseResources();
|
||||||
}
|
}
|
||||||
|
|
||||||
PerfEventsCounters::PerfEventsCounters(): id(latest_counters_id++) {}
|
|
||||||
|
|
||||||
PerfDescriptorsHolder::PerfDescriptorsHolder()
|
PerfDescriptorsHolder::PerfDescriptorsHolder()
|
||||||
{
|
{
|
||||||
for (int & descriptor : descriptors)
|
for (int & descriptor : descriptors)
|
||||||
@ -494,16 +475,12 @@ void PerfDescriptorsHolder::releaseResources()
|
|||||||
if (descriptor == -1)
|
if (descriptor == -1)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
disablePerfEvent(descriptor, getLogger);
|
disablePerfEvent(descriptor);
|
||||||
releasePerfEvent(descriptor, getLogger);
|
releasePerfEvent(descriptor);
|
||||||
descriptor = -1;
|
descriptor = -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Logger * PerfDescriptorsHolder::getLogger()
|
|
||||||
{
|
|
||||||
return &Logger::get("PerfDescriptorsHolder");
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
@ -154,56 +154,47 @@ struct PerfEventValue
|
|||||||
UInt64 time_running = 0;
|
UInt64 time_running = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct PerfDescriptorsHolder;
|
static constexpr size_t NUMBER_OF_RAW_EVENTS = 16;
|
||||||
|
|
||||||
struct PerfEventsCounters
|
|
||||||
{
|
|
||||||
static constexpr size_t NUMBER_OF_RAW_EVENTS = 16;
|
|
||||||
static constexpr Float64 FILE_DESCRIPTORS_THRESHOLD = 0.7;
|
|
||||||
static constexpr char ALL_EVENTS_NAME[] = "all";
|
|
||||||
|
|
||||||
static const PerfEventInfo raw_events_info[NUMBER_OF_RAW_EVENTS];
|
|
||||||
|
|
||||||
static void initializeProfileEvents(PerfEventsCounters & counters, const std::string & events_list);
|
|
||||||
static void finalizeProfileEvents(PerfEventsCounters & counters, ProfileEvents::Counters & profile_events);
|
|
||||||
static void closeEventDescriptors();
|
|
||||||
|
|
||||||
PerfEventsCounters();
|
|
||||||
|
|
||||||
private:
|
|
||||||
// must be unsigned to not cause undefined behaviour on increment
|
|
||||||
using Id = UInt64;
|
|
||||||
using ParsedEvents = std::pair<std::string, std::vector<size_t>>;
|
|
||||||
|
|
||||||
static std::atomic<Id> latest_counters_id;
|
|
||||||
|
|
||||||
static thread_local PerfDescriptorsHolder thread_events_descriptors_holder;
|
|
||||||
static thread_local std::optional<Id> current_thread_counters_id;
|
|
||||||
static thread_local std::optional<ParsedEvents> last_parsed_events;
|
|
||||||
|
|
||||||
Id id;
|
|
||||||
// temp array just to not create it each time event processing finishes
|
|
||||||
PerfEventValue raw_event_values[NUMBER_OF_RAW_EVENTS]{};
|
|
||||||
|
|
||||||
static Logger * getLogger();
|
|
||||||
static bool processThreadLocalChanges(const std::string & needed_events_list);
|
|
||||||
static std::vector<size_t> eventIndicesFromString(const std::string & events_list);
|
|
||||||
};
|
|
||||||
|
|
||||||
struct PerfDescriptorsHolder : boost::noncopyable
|
struct PerfDescriptorsHolder : boost::noncopyable
|
||||||
{
|
{
|
||||||
int descriptors[PerfEventsCounters::NUMBER_OF_RAW_EVENTS]{};
|
int descriptors[NUMBER_OF_RAW_EVENTS]{};
|
||||||
|
|
||||||
PerfDescriptorsHolder();
|
PerfDescriptorsHolder();
|
||||||
|
|
||||||
~PerfDescriptorsHolder();
|
~PerfDescriptorsHolder();
|
||||||
|
|
||||||
void releaseResources();
|
void releaseResources();
|
||||||
|
|
||||||
private:
|
|
||||||
static Logger * getLogger();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct PerfEventsCounters
|
||||||
|
{
|
||||||
|
PerfDescriptorsHolder thread_events_descriptors_holder;
|
||||||
|
|
||||||
|
// time_enabled and time_running can't be reset, so we have to store the
|
||||||
|
// data from the previous profiling period and calculate deltas to them,
|
||||||
|
// to be able to properly account for counter multiplexing.
|
||||||
|
PerfEventValue previous_values[NUMBER_OF_RAW_EVENTS]{};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
static constexpr Float64 FILE_DESCRIPTORS_THRESHOLD = 0.7;
|
||||||
|
static constexpr char ALL_EVENTS_NAME[] = "all";
|
||||||
|
|
||||||
|
void initializeProfileEvents(const std::string & events_list);
|
||||||
|
void finalizeProfileEvents(ProfileEvents::Counters & profile_events);
|
||||||
|
void closeEventDescriptors();
|
||||||
|
|
||||||
|
static Logger * getLogger();
|
||||||
|
bool processThreadLocalChanges(const std::string & needed_events_list);
|
||||||
|
std::vector<size_t> eventIndicesFromString(const std::string & events_list);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Perf event creation is moderately heavy, so we create them once per thread and
|
||||||
|
// then reuse.
|
||||||
|
extern thread_local PerfEventsCounters current_thread_counters;
|
||||||
|
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
|
||||||
struct PerfEventsCounters
|
struct PerfEventsCounters
|
||||||
|
@ -26,7 +26,6 @@ ThreadStatus::ThreadStatus()
|
|||||||
: thread_id{getThreadId()}
|
: thread_id{getThreadId()}
|
||||||
{
|
{
|
||||||
last_rusage = std::make_unique<RUsageCounters>();
|
last_rusage = std::make_unique<RUsageCounters>();
|
||||||
perf_events = std::make_unique<PerfEventsCounters>();
|
|
||||||
|
|
||||||
memory_tracker.setDescription("(for thread)");
|
memory_tracker.setDescription("(for thread)");
|
||||||
log = &Poco::Logger::get("ThreadStatus");
|
log = &Poco::Logger::get("ThreadStatus");
|
||||||
|
@ -199,7 +199,6 @@ protected:
|
|||||||
/// Use ptr not to add extra dependencies in the header
|
/// Use ptr not to add extra dependencies in the header
|
||||||
std::unique_ptr<RUsageCounters> last_rusage;
|
std::unique_ptr<RUsageCounters> last_rusage;
|
||||||
std::unique_ptr<TasksStatsCounters> taskstats;
|
std::unique_ptr<TasksStatsCounters> taskstats;
|
||||||
std::unique_ptr<PerfEventsCounters> perf_events;
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void setupState(const ThreadGroupStatusPtr & thread_group_);
|
void setupState(const ThreadGroupStatusPtr & thread_group_);
|
||||||
|
@ -157,7 +157,8 @@ void ThreadStatus::initPerformanceCounters()
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
PerfEventsCounters::initializeProfileEvents(*perf_events, settings.metrics_perf_events_list);
|
current_thread_counters.initializeProfileEvents(
|
||||||
|
settings.metrics_perf_events_list);
|
||||||
}
|
}
|
||||||
catch (...)
|
catch (...)
|
||||||
{
|
{
|
||||||
@ -195,9 +196,9 @@ void ThreadStatus::finalizePerformanceCounters()
|
|||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
PerfEventsCounters::finalizeProfileEvents(*perf_events, performance_counters);
|
current_thread_counters.finalizeProfileEvents(performance_counters);
|
||||||
if (close_perf_descriptors)
|
if (close_perf_descriptors)
|
||||||
PerfEventsCounters::closeEventDescriptors();
|
current_thread_counters.closeEventDescriptors();
|
||||||
}
|
}
|
||||||
catch (...)
|
catch (...)
|
||||||
{
|
{
|
||||||
|
Loading…
Reference in New Issue
Block a user