-
Notifications
You must be signed in to change notification settings - Fork 13
Expand file tree
/
Copy pathevents.cpp
More file actions
207 lines (182 loc) · 7.07 KB
/
events.cpp
File metadata and controls
207 lines (182 loc) · 7.07 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
#include "events.hpp"
#include <Python.h>
#include <algorithm>
#include <array>
#include <iostream>
#include <scorep/SCOREP_User_Functions.h>
#include <scorep/SCOREP_User_Variables.h>
#include <unordered_map>
namespace scorepy
{
struct region_handle
{
constexpr region_handle() = default;
~region_handle() = default;
constexpr bool operator==(const region_handle& other)
{
return this->value == other.value;
}
SCOREP_User_RegionHandle value = SCOREP_USER_INVALID_REGION;
};
constexpr region_handle uninitialised_region_handle = region_handle();
static std::unordered_map<std::uintptr_t, region_handle> regions;
static std::unordered_map<std::string, region_handle> user_regions;
static std::unordered_map<std::string, region_handle> rewind_regions;
/// Region names that are known to have no region enter event and should not report an error
/// on region exit
static const std::array<std::string, 2> EXIT_REGION_WHITELIST = {
#if PY_MAJOR_VERSION >= 3
"threading:_bootstrap_inner", "threading:_bootstrap"
#else
"threading:__bootstrap_inner", "threading:__bootstrap"
#endif
};
static region_handle create_user_region(const std::string& region_name, const CString module,
const CString file_name, const std::uint64_t line_number)
{
region_handle handle;
SCOREP_User_RegionInit(&handle.value, NULL, NULL, region_name.c_str(),
SCOREP_USER_REGION_TYPE_FUNCTION, file_name.c_str(), line_number);
// Extract main module name if module is like "mainmodule.submodule.subsubmodule"
const char* dot_pos = module.find('.');
if (dot_pos)
{
const std::string main_module(module.c_str(), dot_pos);
SCOREP_User_RegionSetGroup(handle.value, main_module.c_str());
}
else
{
SCOREP_User_RegionSetGroup(handle.value, module.c_str());
}
return handle;
}
// Used for regions, that have an identifier, aka a code object id. (instrumenter regions and
// some decorated regions)
void region_begin(const CString function_name, const CString module, const CString file_name,
const std::uint64_t line_number, const std::uintptr_t& identifier)
{
auto& region_handle = regions[identifier];
if (region_handle == uninitialised_region_handle)
{
const auto& region_name = make_region_name(module, function_name);
region_handle = create_user_region(region_name, module, file_name, line_number);
}
SCOREP_User_RegionEnter(region_handle.value);
}
// Used for regions, that only have a function name, a module, a file and a line number (user
// regions)
void region_begin(const CString function_name, const CString module, const CString file_name,
const std::uint64_t line_number)
{
const auto& region_name = make_region_name(module, function_name);
auto& region_handle = user_regions[region_name];
if (region_handle == uninitialised_region_handle)
{
region_handle = create_user_region(region_name, module, file_name, line_number);
}
SCOREP_User_RegionEnter(region_handle.value);
}
// Used for regions, that have an identifier, aka a code object id. (instrumenter regions and
// some decorated regions)
void region_end(const CString function_name, const CString module, const std::uintptr_t& identifier)
{
const auto it_region = regions.find(identifier);
if (it_region != regions.end())
{
SCOREP_User_RegionEnd(it_region->second.value);
}
else
{
auto& region_name = make_region_name(module, function_name);
region_end_error_handling(region_name);
}
}
// Used for regions, that only have a function name, a module (user regions)
void region_end(const CString function_name, const CString module)
{
auto& region_name = make_region_name(module, function_name);
const auto it_region = user_regions.find(region_name);
if (it_region != user_regions.end())
{
SCOREP_User_RegionEnd(it_region->second.value);
}
else
{
region_end_error_handling(region_name);
}
}
void region_end_error_handling(const std::string& region_name)
{
static region_handle error_region;
static SCOREP_User_ParameterHandle scorep_param = SCOREP_USER_INVALID_PARAMETER;
static bool error_printed = false;
if (std::find(EXIT_REGION_WHITELIST.begin(), EXIT_REGION_WHITELIST.end(), region_name) !=
EXIT_REGION_WHITELIST.end())
{
return;
}
if (error_region.value == SCOREP_USER_INVALID_REGION)
{
SCOREP_User_RegionInit(&error_region.value, NULL, NULL, "error_region",
SCOREP_USER_REGION_TYPE_FUNCTION, "scorep.cpp", 0);
SCOREP_User_RegionSetGroup(error_region.value, "error");
}
SCOREP_User_RegionEnter(error_region.value);
SCOREP_User_ParameterString(&scorep_param, "leave-region", region_name.c_str());
SCOREP_User_RegionEnd(error_region.value);
if (!error_printed)
{
std::cerr << "SCOREP_BINDING_PYTHON ERROR: There was a region exit without an enter!\n"
<< "SCOREP_BINDING_PYTHON ERROR: For details look for \"error_region\" in "
"the trace or profile."
<< std::endl;
error_printed = true;
}
}
void rewind_begin(std::string region_name, std::string file_name, std::uint64_t line_number)
{
auto pair = rewind_regions.emplace(make_pair(region_name, region_handle()));
bool inserted_new = pair.second;
auto& handle = pair.first->second;
if (inserted_new)
{
SCOREP_User_RegionInit(&handle.value, NULL, NULL, region_name.c_str(),
SCOREP_USER_REGION_TYPE_FUNCTION, file_name.c_str(), line_number);
}
SCOREP_User_RewindRegionEnter(handle.value);
}
void rewind_end(std::string region_name, bool value)
{
auto& handle = rewind_regions.at(region_name);
/* don't call SCOREP_ExitRewindRegion, as
* SCOREP_User_RewindRegionEnd does some additional magic
* */
SCOREP_User_RewindRegionEnd(handle.value, value);
}
void parameter_int(std::string name, int64_t value)
{
static SCOREP_User_ParameterHandle scorep_param = SCOREP_USER_INVALID_PARAMETER;
SCOREP_User_ParameterInt64(&scorep_param, name.c_str(), value);
}
void parameter_uint(std::string name, uint64_t value)
{
static SCOREP_User_ParameterHandle scorep_param = SCOREP_USER_INVALID_PARAMETER;
SCOREP_User_ParameterUint64(&scorep_param, name.c_str(), value);
}
void parameter_string(std::string name, std::string value)
{
static SCOREP_User_ParameterHandle scorep_param = SCOREP_USER_INVALID_PARAMETER;
SCOREP_User_ParameterString(&scorep_param, name.c_str(), value.c_str());
}
void oa_region_begin(std::string region_name, std::string file_name, std::uint64_t line_number)
{
auto& handle = user_regions[region_name];
SCOREP_User_OaPhaseBegin(&handle.value, NULL, NULL, region_name.c_str(),
SCOREP_USER_REGION_TYPE_FUNCTION, file_name.c_str(), line_number);
}
void oa_region_end(std::string region_name)
{
auto& handle = user_regions[region_name];
SCOREP_User_OaPhaseEnd(handle.value);
}
} // namespace scorepy