1 /**
2 * Copyright 2020-2021 Huawei Technologies Co., Ltd
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17 #include "profiler/device/gpu/gpu_profiling.h"
18
19 #include <cxxabi.h>
20 #include <chrono>
21 #include <cmath>
22 #include <ctime>
23 #include "profiler/device/gpu/cupti_interface.h"
24 #include "profiler/device/gpu/gpu_data_saver.h"
25 #include "pybind_api/api_register.h"
26 #include "utils/log_adapter.h"
27 #include "utils/utils.h"
28 #include "utils/profile.h"
29 #include "utils/ms_context.h"
30
31 namespace mindspore {
32 namespace profiler {
33 namespace gpu {
34 const size_t BUF_SIZE = 32 * 1024;
35 const size_t ALIGN_SIZE = 8;
36 #define CHECK_CUPTI_RET_WITH_ERROR(expression, message) \
37 if ((expression) != CUPTI_SUCCESS) { \
38 const char *errstr; \
39 CuptiGetResultString(expression, &errstr); \
40 MS_LOG(ERROR) << "CUPTI Error:" << errstr << " function:" << (message) \
41 << ". You may not have access to the NVIDIA GPU performance counters on " \
42 << "the target device. Please use the root account to run profiling or " \
43 << "configure permissions. If there is still the problem, please refer to the" \
44 << " GPU performance tuning document on the official website of mindinsight."; \
45 }
46
47 #define CHECK_CUPTI_RET_WITH_EXCEPT(expression, message) \
48 if ((expression) != CUPTI_SUCCESS) { \
49 const char *errstr; \
50 CuptiGetResultString(expression, &errstr); \
51 MS_LOG(EXCEPTION) << "CUPTI Error:" << errstr << " function:" << (message); \
52 }
53 #define CHECK_CUDA_RET_WITH_ERROR(expression, message) \
54 do { \
55 cudaError_t status = (expression); \
56 if (status != cudaSuccess) { \
57 MS_LOG(ERROR) << "CUDA Error: " << (message) << " | Error Number: " << status << " " \
58 << cudaGetErrorString(status); \
59 } \
60 } while (0)
61 #define PROFILER_ERROR_IF_NULLPTR(ptr) \
62 do { \
63 if ((ptr) == nullptr) { \
64 MS_LOG(ERROR) << ": The pointer[" << #ptr << "] is null."; \
65 return; \
66 } \
67 } while (0)
68
69 std::shared_ptr<GPUProfiler> GPUProfiler::profiler_inst_ = std::make_shared<GPUProfiler>();
70
GetThreadID()71 int32_t GetThreadID() {
72 uint32_t thread_id = static_cast<uint32_t>(pthread_self());
73 return thread_id;
74 }
75
GetStreamID(const CUcontext context,const void * stream)76 uint32_t GetStreamID(const CUcontext context, const void *stream) {
77 uint32_t stream_id = 0;
78 if (stream != nullptr) {
79 CHECK_CUPTI_RET_WITH_ERROR(CuptiGetStreamId(context, (CUstream)stream, &stream_id), "CuptiGetStreamId");
80 if (CuptiGetStreamId(context, (CUstream)stream, &stream_id) != CUPTI_SUCCESS) {
81 MS_LOG(ERROR) << "Training process unexpectedly stopped, profiling data cannot be write to file"
82 << "To obtain the profiling data, do not interrupt the training process.";
83 }
84 }
85 return stream_id;
86 }
87
GetCUPTITimeStamp()88 uint64_t GetCUPTITimeStamp() {
89 uint64_t time_stamp = 0l;
90 CHECK_CUPTI_RET_WITH_ERROR(CuptiGetTimestamp(&time_stamp), "CuptiGetTimestamp");
91 return time_stamp;
92 }
93
GetHostTimeStamp()94 uint64_t GetHostTimeStamp() {
95 auto cur_sys_clock = std::chrono::system_clock::now();
96 uint64_t cur_time_stamp =
97 std::chrono::duration_cast<std::chrono::nanoseconds>(cur_sys_clock.time_since_epoch()).count();
98 return cur_time_stamp;
99 }
100
GetKernelFunc(const char * name)101 std::string GetKernelFunc(const char *name) {
102 char *demangledName = abi::__cxa_demangle(name, nullptr, nullptr, nullptr);
103 if (demangledName != nullptr) {
104 return demangledName;
105 } else {
106 return name;
107 }
108 }
109
IsMemcpyAsyncEvent(CUpti_CallbackId cb_id)110 bool IsMemcpyAsyncEvent(CUpti_CallbackId cb_id) {
111 switch (cb_id) {
112 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyAsync:
113 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyHtoDAsync_v2:
114 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyDtoHAsync_v2:
115 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyDtoDAsync_v2:
116 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyAtoHAsync_v2:
117 case CUPTI_DRIVER_TRACE_CBID_cuMemcpy2DAsync_v2:
118 case CUPTI_DRIVER_TRACE_CBID_cuMemcpy3DAsync_v2:
119 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyHtoAAsync_v2:
120 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyPeerAsync:
121 return true;
122 default:
123 return false;
124 }
125 return false;
126 }
127
IsMemcpySyncEvent(CUpti_CallbackId cb_id)128 bool IsMemcpySyncEvent(CUpti_CallbackId cb_id) {
129 switch (cb_id) {
130 case CUPTI_DRIVER_TRACE_CBID_cuMemcpy:
131 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyHtoD_v2:
132 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyDtoH_v2:
133 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyDtoD_v2:
134 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyAtoH_v2:
135 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyAtoD_v2:
136 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyDtoA_v2:
137 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyAtoA_v2:
138 case CUPTI_DRIVER_TRACE_CBID_cuMemcpy2D_v2:
139 case CUPTI_DRIVER_TRACE_CBID_cuMemcpy2DUnaligned_v2:
140 case CUPTI_DRIVER_TRACE_CBID_cuMemcpy3D_v2:
141 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyHtoA_v2:
142 case CUPTI_DRIVER_TRACE_CBID_cuMemcpyPeer:
143 return true;
144 default:
145 return false;
146 }
147 return false;
148 }
149
CUPTIApiExit(const std::shared_ptr<GPUProfiler> & gpu_profiler_inst,CUpti_CallbackId cb_id,const CUpti_CallbackData * cb_data)150 void CUPTIApiExit(const std::shared_ptr<GPUProfiler> &gpu_profiler_inst, CUpti_CallbackId cb_id,
151 const CUpti_CallbackData *cb_data) {
152 uint64_t start_timestamp = *cb_data->correlationData;
153 uint64_t end_timestamp = GetCUPTITimeStamp();
154 switch (cb_id) {
155 case CUPTI_DRIVER_TRACE_CBID_cuLaunchKernel:
156 case CUPTI_DRIVER_TRACE_CBID_cuLaunchCooperativeKernel:
157 case CUPTI_DRIVER_TRACE_CBID_cuLaunchCooperativeKernelMultiDevice:
158 gpu_profiler_inst->EventHandleProcess(cb_id, cb_data, "cuLaunchKernel", start_timestamp, end_timestamp);
159 break;
160 case CUPTI_DRIVER_TRACE_CBID_cuMemAlloc:
161 case CUPTI_DRIVER_TRACE_CBID_cuMemAlloc_v2:
162 gpu_profiler_inst->EventHandleProcess(cb_id, cb_data, "cuMemAlloc", start_timestamp, end_timestamp);
163 break;
164 case CUPTI_DRIVER_TRACE_CBID_cuEventCreate:
165 case CUPTI_DRIVER_TRACE_CBID_cuEventDestroy_v2:
166 case CUPTI_DRIVER_TRACE_CBID_cuEventRecord:
167 case CUPTI_DRIVER_TRACE_CBID_cuEventSynchronize:
168 case CUPTI_DRIVER_TRACE_CBID_cuEventElapsedTime:
169 // In some cases, the callback of cuctxsetcurrent is only exist
170 // without entry, so this callback is ignored
171 case CUPTI_DRIVER_TRACE_CBID_cuCtxSetCurrent:
172 break;
173 default:
174 gpu_profiler_inst->EventHandleProcess(cb_id, cb_data, "others_api", start_timestamp, end_timestamp);
175 break;
176 }
177 if (IsMemcpyAsyncEvent(cb_id) || IsMemcpySyncEvent(cb_id)) {
178 gpu_profiler_inst->EventHandleProcess(cb_id, cb_data, "cuMemcpy", start_timestamp, end_timestamp);
179 }
180 }
181
CUPTICallBackFunc(void * user_data,CUpti_CallbackDomain domain,CUpti_CallbackId cb_id,const CUpti_CallbackData * cb_data)182 void CUPTICallBackFunc(void *user_data, CUpti_CallbackDomain domain, CUpti_CallbackId cb_id,
183 const CUpti_CallbackData *cb_data) {
184 if (domain != CUPTI_CB_DOMAIN_DRIVER_API) {
185 return;
186 }
187 auto gpu_profiler_inst = GPUProfiler::GetInstance();
188 PROFILER_ERROR_IF_NULLPTR(gpu_profiler_inst);
189 if (!gpu_profiler_inst->GetEnableFlag()) {
190 return;
191 }
192
193 PROFILER_ERROR_IF_NULLPTR(cb_data);
194 if (cb_data->context == nullptr) {
195 MS_LOG(DEBUG) << "Callback data context is null , correlation Id:" << cb_data->correlationId
196 << " callback id:" << cb_id;
197 return;
198 }
199
200 if (cb_data->callbackSite == CUPTI_API_ENTER) {
201 *cb_data->correlationData = GetCUPTITimeStamp();
202 } else if (cb_data->callbackSite == CUPTI_API_EXIT) {
203 CUPTIApiExit(gpu_profiler_inst, cb_id, cb_data);
204 }
205 }
206
GetKernelFuncName(std::string kernel_name)207 std::string GetKernelFuncName(std::string kernel_name) {
208 // remove the return type name (void) in kernel_name.
209 std::string search_pattern("void ");
210 auto func_name_begin_iter = kernel_name.find(search_pattern);
211 if (func_name_begin_iter == kernel_name.npos) {
212 func_name_begin_iter = 0;
213 } else {
214 func_name_begin_iter += search_pattern.length();
215 }
216 return kernel_name.substr(func_name_begin_iter);
217 }
218
GetInstance()219 std::shared_ptr<GPUProfiler> &GPUProfiler::GetInstance() {
220 MS_EXCEPTION_IF_NULL(profiler_inst_);
221 return profiler_inst_;
222 }
223
SyncEnable(const bool enable_flag)224 void GPUProfiler::SyncEnable(const bool enable_flag) {
225 MS_LOG(INFO) << "GPU Profiler synchronous enable flag:" << enable_flag;
226 sync_enable_flag_ = enable_flag;
227 }
228
StepProfilingEnable(const bool enable_flag)229 void GPUProfiler::StepProfilingEnable(const bool enable_flag) {
230 MS_LOG(INFO) << "GPU Profiler enable flag:" << enable_flag;
231 CHECK_CUPTI_RET_WITH_ERROR(CuptiActivityFlushAll(0), "CuptiActivityFlushAll");
232 enable_flag_ = enable_flag;
233 }
234
FixOpNameByCorrelationId(Event * event)235 void GPUProfiler::FixOpNameByCorrelationId(Event *event) {
236 PROFILER_ERROR_IF_NULLPTR(event);
237 if (event->api_type != CUPTIApiType::kActivity) {
238 return;
239 }
240 auto iter = op_name_map_.find(event->correlation_id);
241 if (iter != op_name_map_.end()) {
242 event->op_name = std::move(iter->second);
243 }
244 }
245
AddEvent(Event && event)246 void GPUProfiler::AddEvent(Event &&event) {
247 // protect callback concurrency for driver api and activity
248 std::unique_lock<std::mutex> lock(event_mutex_);
249 switch (event.api_type) {
250 case CUPTIApiType::kCallback: {
251 if (cupti_callback_events_count_ < max_cupti_callback_events_) {
252 events_.emplace_back(std::move(event));
253 cupti_callback_events_count_++;
254 } else {
255 cupti_callback_events_drop_count_++;
256 }
257 break;
258 }
259 case CUPTIApiType::kActivity: {
260 if (cupti_activity_events_count_ < max_cupti_activity_events_) {
261 events_.emplace_back(std::move(event));
262 cupti_activity_events_count_++;
263 } else {
264 cupti_activity_events_drop_count_++;
265 }
266 break;
267 }
268 default:
269 break;
270 }
271 }
272
EventLog(const Event & event)273 void GPUProfiler::EventLog(const Event &event) {
274 MS_LOG(DEBUG) << "GPUProfiler"
275 << ",\"kernel_name:" << event.kernel_name << "\",kernel_type:" << event.kernel_type
276 << ",api_type:" << static_cast<int>(event.api_type) << ",start_time_stamp:" << event.start_time_stamp
277 << ",end_time_stamp:" << event.end_time_stamp << ",cost:,"
278 << (event.end_time_stamp - event.start_time_stamp) / kTimeUnit << ",op_name:" << event.op_name
279 << ",device_id:" << event.device_id << ",correlation_id:" << event.correlation_id
280 << ",thread_id:" << event.thread_id << ",context_id:" << event.context_id
281 << ",stream_id:" << event.stream_id << ",cb_id:" << event.cb_id;
282 }
283
ProcessEvents()284 void GPUProfiler::ProcessEvents() {
285 for (Event &event : events_) {
286 if (event.op_name.empty()) {
287 FixOpNameByCorrelationId(&event);
288 }
289
290 EventLog(event);
291
292 if (event.op_name.empty() || event.cb_id == CUPTI_DRIVER_TRACE_CBID_cuStreamSynchronize) {
293 continue;
294 }
295
296 auto iter = op_info_map_.find(event.op_name);
297 if (iter != op_info_map_.end()) {
298 switch (event.api_type) {
299 case CUPTIApiType::kCallback: {
300 iter->second.op_kernel_api_count += 1;
301 // The time unit from ns to us
302 iter->second.cupti_api_call_time += (event.end_time_stamp - event.start_time_stamp) / kTimeUnit;
303 break;
304 }
305 case CUPTIApiType::kActivity: {
306 iter->second.op_kernel_count += 1;
307 // The time unit from ns to us
308 iter->second.cupti_activity_time += (event.end_time_stamp - event.start_time_stamp) / kTimeUnit;
309 break;
310 }
311 default:
312 break;
313 }
314 }
315 }
316 }
317
OpsParser()318 void GPUProfiler::OpsParser() {
319 MS_LOG(INFO) << "Count the number of events size:" << events_.size()
320 << " callback api:" << cupti_callback_events_count_ << " activity:" << cupti_activity_events_count_;
321
322 if (cupti_activity_events_drop_count_ > 0 || cupti_callback_events_drop_count_ > 0) {
323 MS_LOG(WARNING)
324 << "The total number of events exceeded the profiler's processing capacity, some events were discarded."
325 << " activity api events:" << cupti_activity_events_drop_count_
326 << " callback api events:" << cupti_callback_events_drop_count_;
327 }
328
329 if (events_.size() == 0) {
330 return;
331 }
332
333 ProcessEvents();
334 MS_LOG(DEBUG) << "GPU_profiler, op_name, op_count , kernel_count, kernel_api_count,|"
335 ",cupti_activity_total_time, cupti_api_call_total_time, op_host_cost_total_time,|"
336 ",cupti_activity_average_time,cupti_api_call_average_time, op_host_cost_average_time"
337 << std::endl;
338
339 std::vector<std::pair<std::string, OpInfo>> order_vec(op_info_map_.begin(), op_info_map_.end());
340
341 auto cmp_func = [](const std::pair<std::string, OpInfo> &a, const std::pair<std::string, OpInfo> &b) {
342 return a.second.cupti_activity_time > b.second.cupti_activity_time;
343 };
344 std::sort(order_vec.begin(), order_vec.end(), cmp_func);
345
346 for (auto iter = order_vec.begin(); iter != order_vec.end(); iter++) {
347 if (iter->second.op_count == 0) {
348 MS_LOG(ERROR) << "The num of operations can not be 0.";
349 return;
350 }
351 MS_LOG(DEBUG) << "GPU_profiler"
352 << "," << iter->first << "," << iter->second.op_count << "," << iter->second.op_kernel_count << ","
353 << iter->second.op_kernel_api_count << ","
354 << "|," << iter->second.cupti_activity_time << "," << iter->second.cupti_api_call_time << ","
355 << iter->second.op_host_cost_time << ","
356 << "|," << round(iter->second.cupti_activity_time / iter->second.op_count) << ","
357 << round(iter->second.cupti_api_call_time / iter->second.op_count) << ","
358 << round(iter->second.op_host_cost_time / iter->second.op_count) << std::endl;
359 }
360 }
361
EventHandleProcess(CUpti_CallbackId cbid,const CUpti_CallbackData * cbdata,const std::string & typestring,uint64_t startTimestamp,uint64_t endTimestamp)362 void GPUProfiler::EventHandleProcess(CUpti_CallbackId cbid, const CUpti_CallbackData *cbdata,
363 const std::string &typestring, uint64_t startTimestamp, uint64_t endTimestamp) {
364 Event event;
365 uint32_t device_id = -1;
366 CuptiGetDeviceId(cbdata->context, &device_id);
367 event.kernel_name = cbdata->symbolName ? GetKernelFunc(cbdata->symbolName) : cbdata->functionName;
368 event.kernel_name = GetKernelFuncName(event.kernel_name);
369 event.kernel_type = typestring;
370 event.api_type = CUPTIApiType::kCallback;
371 event.start_time_stamp = startTimestamp;
372 event.end_time_stamp = endTimestamp;
373 event.op_name = op_name_;
374 event.device_id = device_id;
375 event.correlation_id = cbdata->correlationId;
376 event.thread_id = GetThreadID();
377 event.context_id = cbdata->contextUid;
378 event.stream_id = GetStreamID(cbdata->context, stream_);
379 event.cb_id = cbid;
380 op_name_map_[event.correlation_id] = event.op_name;
381 AddEvent(std::move(event));
382 }
383
384 void CUPTIAPI ActivityAllocBuffer(uint8_t **buffer, size_t *size, size_t *maxNumRecords);
385
386 void CUPTIAPI ActivityProcessBuffer(CUcontext ctx, uint32_t streamId, uint8_t *buffer, size_t size, size_t validSize);
387
Init(const std::string & profileDataPath="")388 void GPUProfiler::Init(const std::string &profileDataPath = "") {
389 MS_LOG(INFO) << "Initialize GPU Profiling";
390 if (subscriber_ != nullptr) {
391 StopCUPTI();
392 MS_LOG(EXCEPTION)
393 << "Repeated initialization, Please check whether you have created the Profiler object multiple times";
394 }
395 CHECK_CUPTI_RET_WITH_EXCEPT(CuptiSubscribe(&subscriber_, (CUpti_CallbackFunc)CUPTICallBackFunc, this),
396 "CuptiSubscribe");
397 CHECK_CUPTI_RET_WITH_EXCEPT(CuptiEnableDomain(1, subscriber_, CUPTI_CB_DOMAIN_DRIVER_API), "CuptiEnableDomain");
398
399 activities_enable_.emplace_back(CUPTI_ACTIVITY_KIND_MEMCPY);
400 activities_enable_.emplace_back(CUPTI_ACTIVITY_KIND_MEMCPY2);
401 activities_enable_.emplace_back(CUPTI_ACTIVITY_KIND_KERNEL);
402
403 for (std::vector<CUpti_ActivityKind>::iterator it = activities_enable_.begin(); it != activities_enable_.end();
404 ++it) {
405 CHECK_CUPTI_RET_WITH_EXCEPT(CuptiActivityEnable(*it), "CuptiActivityEnable");
406 }
407
408 CHECK_CUPTI_RET_WITH_EXCEPT(CuptiActivityRegisterCallbacks(ActivityAllocBuffer, ActivityProcessBuffer),
409 "CuptiActivityRegisterCallbacks");
410
411 base_time_.gpu_start_time = GetCUPTITimeStamp();
412 base_time_.host_start_time = GetHostTimeStamp();
413 base_time_.host_start_monotonic_raw_time = GetHostMonoTimeStamp();
414
415 profile_data_path_ = profileDataPath;
416 MS_LOG(INFO) << "GPU start time(ns):" << base_time_.gpu_start_time
417 << " Host start time(ns):" << base_time_.host_start_time << " profile data path: " << profile_data_path_;
418 }
419
SetRunTimeData(const std::string & op_name,void * stream)420 void GPUProfiler::SetRunTimeData(const std::string &op_name, void *stream) {
421 auto iter = op_info_map_.find(op_name);
422 if (iter != op_info_map_.end()) {
423 iter->second.op_count += 1;
424 } else {
425 OpInfo op_info;
426 op_info.op_name = op_name;
427 op_info.stream = stream;
428 op_info.op_count = 1;
429 op_info_map_[op_name] = op_info;
430 }
431 op_name_ = op_name;
432 stream_ = stream;
433 }
434
OpDataProducerBegin(const std::string op_name,void * stream)435 void GPUProfiler::OpDataProducerBegin(const std::string op_name, void *stream) {
436 if (sync_enable_flag_) {
437 CHECK_CUDA_RET_WITH_ERROR(cudaEventCreate(&op_event_start_), "cudaEventCreate op event start failed");
438 CHECK_CUDA_RET_WITH_ERROR(cudaEventCreate(&op_event_stop_), "cudaEventCreate op event stop failed");
439 CHECK_CUDA_RET_WITH_ERROR(cudaEventRecord(op_event_start_, (CUstream)stream_),
440 "cudaEventRecord op event start failed");
441 op_host_time_start_ = GetHostTimeStamp();
442 op_cupti_time_start_ = GetCUPTITimeStamp();
443 } else {
444 op_host_time_start_ = GetHostTimeStamp();
445 op_cupti_time_start_ = GetCUPTITimeStamp();
446 }
447 SetRunTimeData(op_name, stream);
448
449 if (MsContext::GetInstance()->get_param<bool>(MS_CTX_ENABLE_MINDRT)) {
450 RecordOneStepStartEndInfo(op_name);
451 }
452 }
453
SingleOpLaunchTimeProcess(float op_time_elapsed)454 void GPUProfiler::SingleOpLaunchTimeProcess(float op_time_elapsed) {
455 auto launch_end_time = GetTime();
456 double launch_start_time = launch_end_time - op_time_elapsed / kTimeUnit / kTimeUnit;
457 SetSingleOpLaunchTime(std::make_pair(launch_start_time, launch_end_time));
458 }
459
OpDataProducerEnd()460 void GPUProfiler::OpDataProducerEnd() {
461 float op_time_elapsed = 0;
462 if (sync_enable_flag_) {
463 CHECK_CUDA_RET_WITH_ERROR(cudaEventRecord(op_event_stop_, (CUstream)stream_),
464 "cudaEventRecord op event stop failed");
465 CHECK_CUDA_RET_WITH_ERROR(cudaEventSynchronize(op_event_start_), "cudaEventSynchronize op event start failed");
466 CHECK_CUDA_RET_WITH_ERROR(cudaEventSynchronize(op_event_stop_), "cudaEventSynchronize op event stop failed");
467 CHECK_CUDA_RET_WITH_ERROR(cudaEventElapsedTime(&op_time_elapsed, op_event_start_, op_event_stop_),
468 "cudaEventElapsedTime failed");
469 CHECK_CUDA_RET_WITH_ERROR(cudaEventDestroy(op_event_start_), "cudaEventDestroy op event start failed");
470 CHECK_CUDA_RET_WITH_ERROR(cudaEventDestroy(op_event_stop_), "cudaEventDestroy op event stop failed");
471 op_time_elapsed = op_time_elapsed * kTimeUnit;
472 op_host_time_stop_ = GetHostTimeStamp();
473 SingleOpLaunchTimeProcess(op_time_elapsed);
474 } else {
475 op_host_time_stop_ = GetHostTimeStamp();
476 op_time_elapsed = (op_host_time_stop_ - op_host_time_start_) / kTimeUnit;
477 SingleOpLaunchTimeProcess(op_time_elapsed);
478 }
479 MS_LOG(DEBUG) << "Host Time Elapsed(us)," << op_name_ << "," << op_time_elapsed;
480 Profiler::SetRunTimeData(op_name_, op_time_elapsed);
481 Profiler::SetRunTimeData(op_name_, op_cupti_time_start_, op_time_elapsed);
482 }
483
StopCUPTI()484 void GPUProfiler::StopCUPTI() {
485 if (subscriber_ != nullptr) {
486 CHECK_CUPTI_RET_WITH_ERROR(CuptiUnsubscribe(subscriber_), "CuptiUnsubscribe");
487 CHECK_CUPTI_RET_WITH_ERROR(CuptiActivityFlushAll(0), "CuptiActivityFlushAll");
488 for (std::vector<CUpti_ActivityKind>::iterator it = activities_enable_.begin(); it != activities_enable_.end();
489 ++it) {
490 CHECK_CUPTI_RET_WITH_ERROR(CuptiActivityDisable(*it), "CuptiActivityDisable");
491 }
492 subscriber_ = nullptr;
493 }
494 }
495
Stop()496 void GPUProfiler::Stop() {
497 MS_LOG(INFO) << "Stop GPU Profiling";
498 StopCUPTI();
499 OpsParser();
500 SaveProfileData();
501 ClearInst();
502 }
503
SaveExtraProfileData()504 void GPUProfiler::SaveExtraProfileData() {
505 for (auto op : profiling_op_) {
506 op.second->SaveProfilingData();
507 }
508 MS_LOG(INFO) << "Save extra profiling data end.";
509 }
510
SaveProfileData()511 void GPUProfiler::SaveProfileData() {
512 if (profile_data_path_.empty()) {
513 MS_LOG(WARNING) << "Profile data path is empty, skip save profile data.";
514 } else {
515 GpuDataSaver dataSaver(step_trace_op_name_, all_step_start_end_info_);
516 dataSaver.ParseOpInfo(op_info_map_);
517 dataSaver.ParseEvent(events_);
518 dataSaver.WriteFile(profile_data_path_, base_time_);
519 SaveExtraProfileData();
520 }
521 }
522
ClearInst()523 void GPUProfiler::ClearInst() {
524 op_info_map_.clear();
525 op_name_map_.clear();
526 events_.clear();
527 activities_enable_.clear();
528 enable_flag_ = false;
529 sync_enable_flag_ = true;
530 cupti_callback_events_count_ = 0l;
531 cupti_callback_events_drop_count_ = 0l;
532 cupti_activity_events_count_ = 0l;
533 cupti_activity_events_drop_count_ = 0l;
534 }
535
ActivityAllocBuffer(uint8_t ** buffer,size_t * size,size_t * maxNumRecords)536 void CUPTIAPI ActivityAllocBuffer(uint8_t **buffer, size_t *size, size_t *maxNumRecords) {
537 auto gpu_profiler_inst = GPUProfiler::GetInstance();
538 if (gpu_profiler_inst == nullptr) {
539 MS_LOG(ERROR) << "GPU profiler instance is nullptr";
540 return;
541 }
542 gpu_profiler_inst->AllocBuffer(buffer, size, maxNumRecords);
543 }
544
ActivityProcessBuffer(CUcontext ctx,uint32_t streamId,uint8_t * buffer,size_t size,size_t validSize)545 void CUPTIAPI ActivityProcessBuffer(CUcontext ctx, uint32_t streamId, uint8_t *buffer, size_t size, size_t validSize) {
546 PROFILER_ERROR_IF_NULLPTR(buffer);
547 auto gpu_profiler_inst = GPUProfiler::GetInstance();
548 if (gpu_profiler_inst == nullptr) {
549 MS_LOG(ERROR) << "GPU profiler instance is nullptr";
550 return;
551 }
552 gpu_profiler_inst->ProcessBuffer(ctx, streamId, buffer, size, validSize);
553 }
554
ProcessActivityMemcpyRecord(Event * profilingData,CUpti_Activity * record,CUpti_ActivityMemcpy * memcpy)555 void ProcessActivityMemcpyRecord(Event *profilingData, CUpti_Activity *record, CUpti_ActivityMemcpy *memcpy) {
556 switch (memcpy->copyKind) {
557 case CUPTI_ACTIVITY_MEMCPY_KIND_HTOD:
558 profilingData->activity_type = ActivityType::kMemcpyH2D;
559 profilingData->kernel_name = "MemcpyH2D";
560 break;
561 case CUPTI_ACTIVITY_MEMCPY_KIND_DTOH:
562 profilingData->activity_type = ActivityType::kMemcpyD2H;
563 profilingData->kernel_name = "MemcpyD2H";
564 break;
565 case CUPTI_ACTIVITY_MEMCPY_KIND_HTOA:
566 profilingData->activity_type = ActivityType::kMemcpyH2A;
567 profilingData->kernel_name = "MemcpyH2A";
568 break;
569 case CUPTI_ACTIVITY_MEMCPY_KIND_ATOH:
570 profilingData->activity_type = ActivityType::kMemcpyA2H;
571 profilingData->kernel_name = "MemcpyA2H";
572 break;
573 case CUPTI_ACTIVITY_MEMCPY_KIND_ATOD:
574 profilingData->activity_type = ActivityType::kMemcpyA2D;
575 profilingData->kernel_name = "MemcpyA2D";
576 break;
577 case CUPTI_ACTIVITY_MEMCPY_KIND_DTOA:
578 profilingData->activity_type = ActivityType::kMemcpyD2A;
579 profilingData->kernel_name = "MemcpyD2A";
580 break;
581 case CUPTI_ACTIVITY_MEMCPY_KIND_DTOD:
582 profilingData->activity_type = ActivityType::kMemcpyD2D;
583 profilingData->kernel_name = "MemcpyD2D";
584 break;
585 case CUPTI_ACTIVITY_MEMCPY_KIND_HTOH:
586 profilingData->activity_type = ActivityType::kMemcpyH2H;
587 profilingData->kernel_name = "MemcpyH2H";
588 break;
589 case CUPTI_ACTIVITY_MEMCPY_KIND_PTOP:
590 profilingData->activity_type = ActivityType::kMemcpyP2P;
591 profilingData->kernel_name = "MemcpyP2P";
592 break;
593 default:
594 profilingData->activity_type = ActivityType::kMemcpyUnknown;
595 profilingData->kernel_name = "MemcpyUnknown";
596 break;
597 }
598 }
599
HandleActivityMemcpyRecord(Event * profilingData,CUpti_Activity * record)600 void HandleActivityMemcpyRecord(Event *profilingData, CUpti_Activity *record) {
601 CUpti_ActivityMemcpy *memcpy = reinterpret_cast<CUpti_ActivityMemcpy *>(record);
602 ProcessActivityMemcpyRecord(profilingData, record, memcpy);
603
604 profilingData->kernel_type = "cuMemcpy";
605 profilingData->api_type = CUPTIApiType::kActivity;
606 profilingData->start_time_stamp = memcpy->start;
607 profilingData->end_time_stamp = memcpy->end;
608 profilingData->device_id = memcpy->deviceId;
609 profilingData->context_id = memcpy->contextId;
610 profilingData->stream_id = memcpy->streamId;
611 profilingData->correlation_id = memcpy->correlationId;
612 profilingData->memcpy_info.bytes = memcpy->bytes;
613 profilingData->memcpy_info.src_kind = memcpy->srcKind;
614 profilingData->memcpy_info.dst_kind = memcpy->dstKind;
615 }
616
HandleActivityMemcpy2Record(Event * profilingData,CUpti_Activity * record)617 void HandleActivityMemcpy2Record(Event *profilingData, CUpti_Activity *record) {
618 CUpti_ActivityMemcpy2 *memcpyP2P = reinterpret_cast<CUpti_ActivityMemcpy2 *>(record);
619 profilingData->activity_type = ActivityType::kMemcpyP2P;
620 profilingData->kernel_name = "MemcpyP2P";
621 profilingData->kernel_type = "cuMemcpy";
622 profilingData->api_type = CUPTIApiType::kActivity;
623 profilingData->start_time_stamp = memcpyP2P->start;
624 profilingData->end_time_stamp = memcpyP2P->end;
625 profilingData->device_id = memcpyP2P->deviceId;
626 profilingData->context_id = memcpyP2P->contextId;
627 profilingData->stream_id = memcpyP2P->streamId;
628 profilingData->correlation_id = memcpyP2P->correlationId;
629 profilingData->memcpy_info.bytes = memcpyP2P->bytes;
630 profilingData->memcpy_info.src_kind = memcpyP2P->srcKind;
631 profilingData->memcpy_info.dst_kind = memcpyP2P->dstKind;
632 }
633
HandleActivityMemsetRecord(Event * profilingData,CUpti_Activity * record)634 void HandleActivityMemsetRecord(Event *profilingData, CUpti_Activity *record) {
635 CUpti_ActivityMemset *memset = reinterpret_cast<CUpti_ActivityMemset *>(record);
636 profilingData->activity_type = ActivityType::kMemset;
637 profilingData->kernel_name = "MemorySet";
638 profilingData->api_type = CUPTIApiType::kActivity;
639 profilingData->start_time_stamp = memset->start;
640 profilingData->end_time_stamp = memset->end;
641 profilingData->device_id = memset->deviceId;
642 profilingData->context_id = memset->contextId;
643 profilingData->stream_id = memset->streamId;
644 profilingData->correlation_id = memset->correlationId;
645 profilingData->memcpy_info.bytes = memset->bytes;
646 }
647
HandleActivityKernelRecord(Event * profilingData,CUpti_Activity * record)648 void HandleActivityKernelRecord(Event *profilingData, CUpti_Activity *record) {
649 CUpti_ActivityKernel4 *kernel = reinterpret_cast<CUpti_ActivityKernel4 *>(record);
650 profilingData->activity_type = ActivityType::kKernel;
651 profilingData->api_type = CUPTIApiType::kActivity;
652 profilingData->kernel_name = GetKernelFunc(kernel->name);
653 profilingData->kernel_name = GetKernelFuncName(profilingData->kernel_name);
654 profilingData->kernel_type = "cuLaunchKernel";
655 profilingData->start_time_stamp = kernel->start;
656 profilingData->end_time_stamp = kernel->end;
657 profilingData->device_id = kernel->deviceId;
658 profilingData->context_id = kernel->contextId;
659 profilingData->stream_id = kernel->streamId;
660 profilingData->correlation_id = kernel->correlationId;
661 profilingData->kernel_info.registers_per_thread = kernel->registersPerThread;
662 profilingData->kernel_info.static_shared_memory = kernel->staticSharedMemory;
663 profilingData->kernel_info.dynamic_shared_memory = kernel->dynamicSharedMemory;
664 profilingData->kernel_info.block_x = kernel->blockX;
665 profilingData->kernel_info.block_y = kernel->blockY;
666 profilingData->kernel_info.block_z = kernel->blockZ;
667 profilingData->kernel_info.grid_x = kernel->gridX;
668 profilingData->kernel_info.grid_y = kernel->gridY;
669 profilingData->kernel_info.grid_z = kernel->gridZ;
670 }
671
HandleActivityRecord(CUpti_Activity * record)672 void GPUProfiler::HandleActivityRecord(CUpti_Activity *record) {
673 PROFILER_ERROR_IF_NULLPTR(record);
674 Event profilingData;
675 profilingData.cb_id = 0;
676 switch (record->kind) {
677 case CUPTI_ACTIVITY_KIND_MEMCPY: {
678 HandleActivityMemcpyRecord(&profilingData, record);
679 break;
680 }
681 case CUPTI_ACTIVITY_KIND_MEMCPY2: {
682 HandleActivityMemcpy2Record(&profilingData, record);
683 break;
684 }
685 case CUPTI_ACTIVITY_KIND_MEMSET: {
686 HandleActivityMemsetRecord(&profilingData, record);
687 break;
688 }
689 case CUPTI_ACTIVITY_KIND_KERNEL:
690 case CUPTI_ACTIVITY_KIND_CONCURRENT_KERNEL: {
691 HandleActivityKernelRecord(&profilingData, record);
692 break;
693 }
694 default:
695 MS_LOG(WARNING) << "Unknown activity type!";
696 return;
697 }
698
699 AddEvent(std::move(profilingData));
700 }
701
SetStepTraceOpName(ProfilingTraceInfo trace_op_name)702 void GPUProfiler::SetStepTraceOpName(ProfilingTraceInfo trace_op_name) { step_trace_op_name_ = trace_op_name; }
703
RegisterProfilingOp(std::shared_ptr<ProfilingOp> node)704 void GPUProfiler::RegisterProfilingOp(std::shared_ptr<ProfilingOp> node) {
705 PROFILER_ERROR_IF_NULLPTR(node);
706 if (profiling_op_.find(node->Name()) != profiling_op_.end()) {
707 return;
708 }
709 node->Init();
710 profiling_op_[node->Name()] = node;
711 }
712
AllocBuffer(uint8_t ** buffer,size_t * size,size_t * maxNumRecords)713 void CUPTIAPI GPUProfiler::AllocBuffer(uint8_t **buffer, size_t *size, size_t *maxNumRecords) {
714 PROFILER_ERROR_IF_NULLPTR(size);
715 PROFILER_ERROR_IF_NULLPTR(maxNumRecords);
716 int stat = posix_memalign(reinterpret_cast<void **>(buffer), ALIGN_SIZE, BUF_SIZE);
717 if (stat) {
718 MS_LOG(ERROR) << "Out of memory, activity buffer alloc failed.";
719 return;
720 }
721 MS_LOG(DEBUG) << "Alloc activity buffer, buffer size: " << BUF_SIZE;
722 *size = BUF_SIZE;
723 *maxNumRecords = 0;
724 }
725
ProcessBuffer(CUcontext ctx,uint32_t streamId,uint8_t * buffer,size_t size,size_t validSize)726 void CUPTIAPI GPUProfiler::ProcessBuffer(CUcontext ctx, uint32_t streamId, uint8_t *buffer, size_t size,
727 size_t validSize) {
728 if (!enable_flag_) {
729 MS_LOG(DEBUG) << "Profiler is not enable, skip to process activity record.";
730 free(buffer);
731 return;
732 }
733 CUptiResult status;
734 CUpti_Activity *record = NULL;
735
736 MS_LOG(DEBUG) << "Process activity buffer, valid size:" << validSize << ",Stream ID:" << streamId;
737 if (validSize > 0) {
738 do {
739 status = CuptiActivityGetNextRecord(buffer, validSize, &record);
740 if (status == CUPTI_SUCCESS) {
741 HandleActivityRecord(record);
742 } else if (status == CUPTI_ERROR_MAX_LIMIT_REACHED) {
743 break;
744 } else {
745 CHECK_CUPTI_RET_WITH_ERROR(status, "CuptiActivityGetNextRecord");
746 }
747 } while (1);
748
749 // report any records dropped from the queue
750 size_t dropped;
751 CHECK_CUPTI_RET_WITH_ERROR(CuptiActivityGetNumDroppedRecords(ctx, streamId, &dropped),
752 "CuptiActivityGetNumDroppedRecords");
753 if (dropped != 0) {
754 MS_LOG(INFO) << "Dropped " << (unsigned int)dropped << " activity records\n";
755 }
756 }
757
758 free(buffer);
759 }
760
__anon16dda4f60202(const py::module *m) 761 REGISTER_PYBIND_DEFINE(GPUProfiler_, ([](const py::module *m) {
762 (void)py::class_<GPUProfiler, std::shared_ptr<GPUProfiler>>(*m, "GPUProfiler")
763 .def_static("get_instance", &GPUProfiler::GetInstance, "GPUProfiler get_instance.")
764 .def("init", &GPUProfiler::Init, py::arg("profile_data_path"), "init")
765 .def("stop", &GPUProfiler::Stop, "stop")
766 .def("step_profiling_enable", &GPUProfiler::StepProfilingEnable, py::arg("enable_flag"),
767 "enable or disable step profiling")
768 .def("sync_enable", &GPUProfiler::SyncEnable, py::arg("enable_flag"),
769 "enable or disable synchronization profiling");
770 }));
771 } // namespace gpu
772 } // namespace profiler
773 } // namespace mindspore
774