14#include "arcane/utils/IPerformanceCounterService.h"
15#include "arcane/utils/TraceAccessor.h"
16#include "arcane/utils/FatalErrorException.h"
18#include "arcane/FactoryService.h"
20#include <linux/perf_event.h>
21#include <linux/hw_breakpoint.h>
36class LinuxPerfPerformanceCounterService
46 ~LinuxPerfPerformanceCounterService()
61 bool _addEvent(
int event_type,
int event_config,
bool is_optional =
false)
63 struct perf_event_attr attr;
64 memset(&attr, 0,
sizeof(attr));
66 attr.type = event_type,
67 attr.config = event_config;
68 attr.size =
sizeof(
struct perf_event_attr);
69 attr.exclude_kernel = 1;
76 unsigned long flags = 0;
77 long long_fd = syscall(__NR_perf_event_open, &attr, m_process_id, cpu, group_fd, flags);
78 info(4) <<
"AddEvent type=" << attr.type <<
" id=" << attr.config <<
" fd=" << long_fd;
82 ARCANE_FATAL(
"ERROR for event type={0} id={1} error={2}", attr.type, attr.config, strerror(errno));
84 int fd =
static_cast<int>(long_fd);
85 m_events_file_descriptor.add(fd);
94 info(4) <<
"LinuxPerf: Start";
95 for (
int fd : m_events_file_descriptor) {
96 int r = ::ioctl(fd, PERF_EVENT_IOC_ENABLE);
98 ARCANE_FATAL(
"Error starting event r={0} error={1}", r, strerror(errno));
106 info(4) <<
"LinuxPerf: Stop";
107 for (
int fd : m_events_file_descriptor) {
108 int r = ::ioctl(fd, PERF_EVENT_IOC_DISABLE);
110 ARCANE_FATAL(
"Error stopping event r={0} error={1}", r, strerror(errno));
112 m_is_started =
false;
122 for (
Int32 index = 0, n = m_events_file_descriptor.size(); index < n; ++index) {
123 Int64 value = _getOneCounter(index);
124 Int64 current_value = counters[index];
125 counters[index] = (do_substract) ? value - current_value : value;
132 return _getOneCounter(0);
138 bool m_is_started =
false;
139 bool m_is_init =
false;
140 pid_t m_process_id = -1;
146 for (
int fd : m_events_file_descriptor) {
150 m_events_file_descriptor.
clear();
155 int fd = m_events_file_descriptor[index];
156 uint64_t value[1] = { 0 };
157 const size_t s =
sizeof(uint64_t);
158 size_t nb_read = ::read(fd, value, s);
164 void _checkInitialize()
169 info() <<
"Initialize LinuxPerfPerformanceCounterService";
170 m_process_id = ::getpid();
172 _addEvent(PERF_TYPE_HARDWARE, PERF_COUNT_HW_CPU_CYCLES);
174 _addEvent(PERF_TYPE_HARDWARE, PERF_COUNT_HW_INSTRUCTIONS);
182 const bool is_optional =
true;
185 int cache_access = (PERF_COUNT_HW_CACHE_OP_READ << 8) | (PERF_COUNT_HW_CACHE_RESULT_MISS << 16);
186 is_bad =
_addEvent(PERF_TYPE_HW_CACHE, PERF_COUNT_HW_CACHE_LL | cache_access, is_optional);
189 is_bad =
_addEvent(PERF_TYPE_HARDWARE, PERF_COUNT_HW_STALLED_CYCLES_FRONTEND, is_optional);
191 _addEvent(PERF_TYPE_HARDWARE, PERF_COUNT_HW_CACHE_MISSES, is_optional);
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
#define ARCANE_SERVICE_INTERFACE(ainterface)
Macro pour déclarer une interface lors de l'enregistrement d'un service.
void clear()
Supprime les éléments du tableau.
virtual ITraceMng * traceMng() const =0
Gestionnaire de traces.
IApplication * application() const
Accès à l'application IApplication associé.
Structure contenant les informations pour créer un service.
Propriétés de création d'un service.
TraceAccessor(ITraceMng *m)
Construit un accesseur via le gestionnaire de trace m.
TraceMessage info() const
Flot pour un message d'information.
Vecteur 1D de données avec sémantique par valeur (style STL).
#define ARCANE_REGISTER_SERVICE(aclass, a_service_property,...)
Macro pour enregistrer un service.
-*- tab-width: 2; indent-tabs-mode: nil; coding: utf-8-with-signature -*-
ArrayView< Int64 > Int64ArrayView
Equivalent C d'un tableau à une dimension d'entiers 64 bits.
std::int64_t Int64
Type entier signé sur 64 bits.
Int32 Integer
Type représentant un entier.
@ ST_Application
Le service s'utilise au niveau de l'application.
std::int32_t Int32
Type entier signé sur 32 bits.