2023-05-19 17:53:17 +00:00
|
|
|
/* Copyright (c) 2022 Advanced Micro Devices, Inc.
|
|
|
|
|
|
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
|
|
|
in the Software without restriction, including without limitation the rights
|
|
|
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
|
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
|
|
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
|
|
|
all copies or substantial portions of the Software.
|
|
|
|
|
|
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
|
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
|
THE SOFTWARE. */
|
|
|
|
|
|
|
|
|
|
#include <cxxabi.h>
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <sys/syscall.h>
|
|
|
|
|
#include <sys/types.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
|
|
#include <atomic>
|
|
|
|
|
#include <cassert>
|
|
|
|
|
#include <cstddef>
|
|
|
|
|
#include <cstdint>
|
|
|
|
|
#include <fstream>
|
|
|
|
|
#include <iostream>
|
2024-08-08 01:14:38 +00:00
|
|
|
#include <map>
|
2023-05-19 17:53:17 +00:00
|
|
|
#include <memory>
|
|
|
|
|
#include <optional>
|
|
|
|
|
#include <ostream>
|
|
|
|
|
#include <sstream>
|
|
|
|
|
#include <string>
|
|
|
|
|
#include <hsa/hsa.h>
|
|
|
|
|
#include <mutex>
|
|
|
|
|
#include <unordered_map>
|
|
|
|
|
|
|
|
|
|
#include "rocprofiler.h"
|
|
|
|
|
#include "rocprofiler_plugin.h"
|
|
|
|
|
#include "../utils.h"
|
|
|
|
|
|
2024-06-04 15:02:51 +05:30
|
|
|
#include "src/utils/filesystem.hpp"
|
|
|
|
|
|
|
|
|
|
namespace fs = rocprofiler::common::filesystem;
|
2023-05-19 17:53:17 +00:00
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
|
|
static std::string output_file_name;
|
|
|
|
|
class file_plugin_t {
|
|
|
|
|
private:
|
|
|
|
|
enum class output_type_t { COUNTER, TRACER, PC_SAMPLING };
|
|
|
|
|
|
|
|
|
|
class output_file_t {
|
|
|
|
|
public:
|
|
|
|
|
output_file_t(std::string name) : name_(std::move(name)) {}
|
|
|
|
|
|
|
|
|
|
std::string name() const { return name_; }
|
|
|
|
|
|
|
|
|
|
template <typename T> std::ostream& operator<<(T&& value) {
|
|
|
|
|
if (!is_open()) open();
|
|
|
|
|
return stream_ << std::forward<T>(value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::ostream& operator<<(std::ostream& (*func)(std::ostream&)) {
|
|
|
|
|
if (!is_open()) open();
|
|
|
|
|
return stream_ << func;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void open() {
|
|
|
|
|
// If the stream is already in the failed state, there's no need to try
|
|
|
|
|
// to open the file.
|
|
|
|
|
if (fail()) return;
|
|
|
|
|
|
|
|
|
|
const char* output_dir = getenv("OUTPUT_PATH");
|
|
|
|
|
output_file_name = getenv("OUT_FILE_NAME") ? std::string(getenv("OUT_FILE_NAME")) + "_" : "";
|
|
|
|
|
|
|
|
|
|
if (output_dir == nullptr && getenv("OUT_FILE_NAME") == nullptr) {
|
|
|
|
|
stream_.copyfmt(std::cout);
|
|
|
|
|
stream_.clear(std::cout.rdstate());
|
|
|
|
|
stream_.basic_ios<char>::rdbuf(std::cout.rdbuf());
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (output_dir == nullptr) output_dir = "./";
|
|
|
|
|
|
|
|
|
|
fs::path output_prefix(output_dir);
|
|
|
|
|
if (!fs::is_directory(fs::status(output_prefix))) {
|
|
|
|
|
if (!stream_.fail()) rocprofiler::warning("Cannot open output directory '%s'", output_dir);
|
|
|
|
|
stream_.setstate(std::ios_base::failbit);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
|
ss << output_file_name << GetPid() << "_" << name_;
|
|
|
|
|
stream_.open(output_prefix / ss.str());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool is_open() const { return stream_.is_open(); }
|
|
|
|
|
bool fail() const { return stream_.fail(); }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
const std::string name_;
|
|
|
|
|
std::ofstream stream_;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
output_file_t* get_output_file(output_type_t output_type, uint32_t domain = 0) {
|
|
|
|
|
switch (output_type) {
|
|
|
|
|
case output_type_t::COUNTER:
|
|
|
|
|
return &output_file_;
|
|
|
|
|
case output_type_t::TRACER:
|
|
|
|
|
switch (domain) {
|
|
|
|
|
case ACTIVITY_DOMAIN_ROCTX:
|
|
|
|
|
return &roctx_file_;
|
|
|
|
|
case ACTIVITY_DOMAIN_HSA_API:
|
|
|
|
|
return &hsa_api_file_;
|
|
|
|
|
case ACTIVITY_DOMAIN_HIP_API:
|
|
|
|
|
return &hip_api_file_;
|
|
|
|
|
case ACTIVITY_DOMAIN_HIP_OPS:
|
|
|
|
|
return &hip_activity_file_;
|
|
|
|
|
case ACTIVITY_DOMAIN_HSA_OPS:
|
|
|
|
|
return &hsa_async_copy_file_;
|
|
|
|
|
default:
|
|
|
|
|
assert(!"domain/op not supported!");
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
case output_type_t::PC_SAMPLING:
|
|
|
|
|
return &pc_sample_file_;
|
|
|
|
|
}
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public:
|
2024-08-08 01:14:38 +00:00
|
|
|
file_plugin_t() {
|
|
|
|
|
valid_ = true;
|
|
|
|
|
kernel_filters_ = GetKernelFilters();
|
|
|
|
|
}
|
2023-05-19 17:53:17 +00:00
|
|
|
|
|
|
|
|
std::mutex writing_lock;
|
|
|
|
|
|
|
|
|
|
const char* GetDomainName(rocprofiler_tracer_activity_domain_t domain) {
|
|
|
|
|
switch (domain) {
|
|
|
|
|
case ACTIVITY_DOMAIN_ROCTX:
|
|
|
|
|
return "ROCTX_DOMAIN";
|
|
|
|
|
break;
|
|
|
|
|
case ACTIVITY_DOMAIN_HIP_API:
|
|
|
|
|
return "HIP_API_DOMAIN";
|
|
|
|
|
break;
|
|
|
|
|
case ACTIVITY_DOMAIN_HIP_OPS:
|
|
|
|
|
return "HIP_OPS_DOMAIN";
|
|
|
|
|
break;
|
|
|
|
|
case ACTIVITY_DOMAIN_HSA_API:
|
|
|
|
|
return "HSA_API_DOMAIN";
|
|
|
|
|
break;
|
|
|
|
|
case ACTIVITY_DOMAIN_HSA_OPS:
|
|
|
|
|
return "HSA_OPS_DOMAIN";
|
|
|
|
|
break;
|
|
|
|
|
case ACTIVITY_DOMAIN_HSA_EVT:
|
|
|
|
|
return "HSA_EVT_DOMAIN";
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return "";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FlushTracerRecord(rocprofiler_record_tracer_t tracer_record,
|
|
|
|
|
rocprofiler_session_id_t session_id,
|
|
|
|
|
rocprofiler_buffer_id_t buffer_id = rocprofiler_buffer_id_t{0}) {
|
|
|
|
|
std::lock_guard<std::mutex> lock(writing_lock);
|
|
|
|
|
if (tracer_record.timestamps.end.value <= 0 && tracer_record.domain != ACTIVITY_DOMAIN_ROCTX)
|
|
|
|
|
return;
|
2023-05-25 23:13:51 +00:00
|
|
|
const char* operation_name_c = nullptr;
|
|
|
|
|
// ROCTX domain Operation ID doesn't have a name
|
|
|
|
|
// It depends on the user input of the roctx functions.
|
|
|
|
|
// ROCTX message is the tracer_record.name
|
|
|
|
|
if (tracer_record.domain != ACTIVITY_DOMAIN_ROCTX) {
|
|
|
|
|
CHECK_ROCPROFILER(rocprofiler_query_tracer_operation_name(
|
|
|
|
|
tracer_record.domain, tracer_record.operation_id, &operation_name_c));
|
2023-05-19 17:53:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
output_file_t* output_file = get_output_file(output_type_t::TRACER, tracer_record.domain);
|
2023-06-20 15:12:33 +00:00
|
|
|
*output_file << "Domain(" << GetDomainName(tracer_record.domain) << "), ";
|
2023-05-25 23:13:51 +00:00
|
|
|
if (operation_name_c)
|
|
|
|
|
*output_file << ((tracer_record.domain == ACTIVITY_DOMAIN_HIP_API ||
|
|
|
|
|
tracer_record.domain == ACTIVITY_DOMAIN_HSA_API)
|
|
|
|
|
? "Function("
|
|
|
|
|
: "Operation_Name(")
|
|
|
|
|
<< operation_name_c << "), ";
|
|
|
|
|
if (tracer_record.name && tracer_record.domain != ACTIVITY_DOMAIN_ROCTX)
|
|
|
|
|
*output_file << "Kernel_Name("
|
|
|
|
|
<< rocprofiler::truncate_name(rocprofiler::cxx_demangle(tracer_record.name))
|
|
|
|
|
<< "), ";
|
2023-05-19 17:53:17 +00:00
|
|
|
if (tracer_record.domain != ACTIVITY_DOMAIN_ROCTX) {
|
|
|
|
|
*output_file << "Start_Timestamp(" << tracer_record.timestamps.begin.value << "), "
|
|
|
|
|
<< "End_Timestamp(" << tracer_record.timestamps.end.value << "), "
|
|
|
|
|
<< "Correlation_ID(" << tracer_record.correlation_id.value << ")";
|
|
|
|
|
} else {
|
|
|
|
|
*output_file << "Timestamp(" << tracer_record.timestamps.begin.value << "), ";
|
|
|
|
|
}
|
2023-05-25 23:13:51 +00:00
|
|
|
if (tracer_record.domain == ACTIVITY_DOMAIN_ROCTX && tracer_record.external_id.id >= 0)
|
|
|
|
|
*output_file << "ROCTX_ID(" << tracer_record.external_id.id << "), ";
|
|
|
|
|
if (tracer_record.name && tracer_record.domain == ACTIVITY_DOMAIN_ROCTX)
|
|
|
|
|
*output_file << "ROCTX_Message(" << tracer_record.name << ")";
|
2023-05-19 17:53:17 +00:00
|
|
|
*output_file << std::endl;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FlushProfilerRecord(const rocprofiler_record_profiler_t* profiler_record,
|
|
|
|
|
rocprofiler_session_id_t session_id, rocprofiler_buffer_id_t buffer_id) {
|
2024-08-08 01:14:38 +00:00
|
|
|
auto check_filter_cache = [&](const char * k_name) {
|
|
|
|
|
auto str = std::string{k_name};
|
|
|
|
|
auto f = kernel_filter_cache_.find(k_name);
|
|
|
|
|
if (f == kernel_filter_cache_.end()) {
|
|
|
|
|
bool found_match = false;
|
|
|
|
|
for (const auto& filter : kernel_filters_) {
|
|
|
|
|
if (str.find(filter) != std::string::npos) {
|
|
|
|
|
found_match = true;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
f = kernel_filter_cache_.emplace(str, found_match).first;
|
|
|
|
|
}
|
|
|
|
|
return f->second;
|
|
|
|
|
};
|
|
|
|
|
|
2023-05-19 17:53:17 +00:00
|
|
|
std::lock_guard<std::mutex> lock(writing_lock);
|
|
|
|
|
size_t name_length = 0;
|
|
|
|
|
output_file_t* output_file{nullptr};
|
|
|
|
|
output_file = get_output_file(output_type_t::COUNTER);
|
|
|
|
|
CHECK_ROCPROFILER(rocprofiler_query_kernel_info_size(ROCPROFILER_KERNEL_NAME,
|
|
|
|
|
profiler_record->kernel_id, &name_length));
|
|
|
|
|
// Taken from rocprofiler: The size hasn't changed in recent past
|
|
|
|
|
static const uint32_t lds_block_size = 128 * 4;
|
|
|
|
|
const char* kernel_name_c = nullptr;
|
|
|
|
|
if (name_length > 1) {
|
|
|
|
|
CHECK_ROCPROFILER(rocprofiler_query_kernel_info(ROCPROFILER_KERNEL_NAME,
|
|
|
|
|
profiler_record->kernel_id, &kernel_name_c));
|
2024-08-08 01:14:38 +00:00
|
|
|
if (kernel_name_c != nullptr && !kernel_filters_.empty() && !check_filter_cache(kernel_name_c)) {
|
|
|
|
|
free(const_cast<char*>(kernel_name_c));
|
|
|
|
|
return;
|
|
|
|
|
}
|
2023-05-19 17:53:17 +00:00
|
|
|
}
|
2023-06-20 15:12:33 +00:00
|
|
|
*output_file << "Dispatch_ID(" << std::to_string(profiler_record->header.id.handle) << "), "
|
2023-05-19 17:53:17 +00:00
|
|
|
<< "GPU_ID(" << std::to_string(profiler_record->gpu_id.handle) << "), "
|
|
|
|
|
<< "Queue_ID(" << std::to_string(profiler_record->queue_id.handle) << "), "
|
|
|
|
|
<< "Process_ID(" << std::to_string(GetPid()) << "), "
|
|
|
|
|
<< "Thread_ID(" << std::to_string(profiler_record->thread_id.value) << "), "
|
|
|
|
|
<< "Grid_Size(" << std::to_string(profiler_record->kernel_properties.grid_size)
|
|
|
|
|
<< "), "
|
|
|
|
|
<< "Workgroup_Size("
|
|
|
|
|
<< std::to_string(profiler_record->kernel_properties.workgroup_size) << "), "
|
2023-08-12 04:49:49 +00:00
|
|
|
<< "LDS_Per_Workgroup("
|
2023-05-19 17:53:17 +00:00
|
|
|
<< std::to_string(
|
|
|
|
|
((profiler_record->kernel_properties.lds_size + (lds_block_size - 1)) &
|
|
|
|
|
~(lds_block_size - 1)))
|
|
|
|
|
<< "), "
|
2023-08-12 04:49:49 +00:00
|
|
|
<< "Scratch_Per_Workitem("
|
2023-05-19 17:53:17 +00:00
|
|
|
<< std::to_string(profiler_record->kernel_properties.scratch_size) << "), "
|
|
|
|
|
<< "Arch_VGPR("
|
|
|
|
|
<< std::to_string(profiler_record->kernel_properties.arch_vgpr_count) << "), "
|
2023-08-12 04:49:49 +00:00
|
|
|
<< "Accum_VGPR("
|
2023-05-19 17:53:17 +00:00
|
|
|
<< std::to_string(profiler_record->kernel_properties.accum_vgpr_count) << "), "
|
|
|
|
|
<< "SGPR(" << std::to_string(profiler_record->kernel_properties.sgpr_count)
|
|
|
|
|
<< "), "
|
|
|
|
|
<< "Wave_Size(" << std::to_string(profiler_record->kernel_properties.wave_size);
|
|
|
|
|
std::string kernel_name = "";
|
|
|
|
|
if (name_length > 1) {
|
|
|
|
|
kernel_name = rocprofiler::truncate_name(rocprofiler::cxx_demangle(kernel_name_c));
|
|
|
|
|
std::string key = "\"";
|
|
|
|
|
std::size_t found = kernel_name.rfind(key);
|
|
|
|
|
while (found != std::string::npos) {
|
|
|
|
|
kernel_name.replace(found, key.length(), "'");
|
|
|
|
|
found = kernel_name.rfind(key, found - 1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
*output_file << "), "
|
|
|
|
|
<< "Kernel_Name(\"" << kernel_name << "\"), "
|
|
|
|
|
<< "Begin_Timestamp(" << std::to_string(profiler_record->timestamps.begin.value)
|
|
|
|
|
<< "), "
|
|
|
|
|
<< "End_Timestamp(" << std::to_string(profiler_record->timestamps.end.value)
|
2023-06-20 15:12:33 +00:00
|
|
|
<< "), "
|
|
|
|
|
<< "Correlation_ID(" << std::to_string(profiler_record->correlation_id.value)
|
2023-05-19 17:53:17 +00:00
|
|
|
<< ")";
|
|
|
|
|
|
|
|
|
|
// For Counters
|
|
|
|
|
if (profiler_record->counters) {
|
|
|
|
|
*output_file << ", ";
|
|
|
|
|
for (uint64_t i = 0; i < profiler_record->counters_count.value; i++) {
|
|
|
|
|
if (profiler_record->counters[i].counter_handler.handle > 0) {
|
|
|
|
|
size_t counter_name_length = 0;
|
|
|
|
|
CHECK_ROCPROFILER(rocprofiler_query_counter_info_size(
|
|
|
|
|
session_id, ROCPROFILER_COUNTER_NAME, profiler_record->counters[i].counter_handler,
|
|
|
|
|
&counter_name_length));
|
|
|
|
|
if (counter_name_length > 1) {
|
|
|
|
|
const char* name_c = nullptr;
|
|
|
|
|
CHECK_ROCPROFILER(rocprofiler_query_counter_info(
|
|
|
|
|
session_id, ROCPROFILER_COUNTER_NAME, profiler_record->counters[i].counter_handler,
|
|
|
|
|
&name_c));
|
|
|
|
|
*output_file << name_c << "("
|
|
|
|
|
<< std::to_string(profiler_record->counters[i].value.value) << ")";
|
|
|
|
|
if (i < profiler_record->counters_count.value - 1) *output_file << ", ";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
*output_file << std::endl;
|
|
|
|
|
if (kernel_name_c) {
|
|
|
|
|
free(const_cast<char*>(kernel_name_c));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void FlushPCSamplingRecord(const rocprofiler_record_pc_sample_t* pc_sampling_record) {
|
|
|
|
|
output_file_t* output_file{nullptr};
|
|
|
|
|
output_file = get_output_file(output_type_t::PC_SAMPLING);
|
|
|
|
|
const auto& sample = pc_sampling_record->pc_sample;
|
|
|
|
|
*output_file << "dispatch[" << sample.dispatch_id.value << "], "
|
|
|
|
|
<< "timestamp(" << sample.timestamp.value << "), "
|
|
|
|
|
<< "gpu_id(" << sample.gpu_id.handle << "), "
|
|
|
|
|
<< "pc-sample(" << std::hex << std::showbase << sample.pc << "), "
|
|
|
|
|
<< "se(" << sample.se << ')' << std::endl;
|
|
|
|
|
}
|
|
|
|
|
int WriteBufferRecords(const rocprofiler_record_header_t* begin,
|
|
|
|
|
const rocprofiler_record_header_t* end,
|
|
|
|
|
rocprofiler_session_id_t session_id, rocprofiler_buffer_id_t buffer_id) {
|
|
|
|
|
while (begin < end) {
|
|
|
|
|
if (!begin) return 0;
|
|
|
|
|
switch (begin->kind) {
|
|
|
|
|
case ROCPROFILER_PROFILER_RECORD: {
|
|
|
|
|
const rocprofiler_record_profiler_t* profiler_record =
|
|
|
|
|
reinterpret_cast<const rocprofiler_record_profiler_t*>(begin);
|
|
|
|
|
FlushProfilerRecord(profiler_record, session_id, buffer_id);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case ROCPROFILER_TRACER_RECORD: {
|
|
|
|
|
rocprofiler_record_tracer_t* tracer_record = const_cast<rocprofiler_record_tracer_t*>(
|
|
|
|
|
reinterpret_cast<const rocprofiler_record_tracer_t*>(begin));
|
|
|
|
|
FlushTracerRecord(*tracer_record, session_id, buffer_id);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case ROCPROFILER_ATT_TRACER_RECORD: {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case ROCPROFILER_PC_SAMPLING_RECORD: {
|
2024-04-03 18:19:23 +05:30
|
|
|
[[deprecated("PC Sampling is deprecated")]]
|
2023-05-19 17:53:17 +00:00
|
|
|
const rocprofiler_record_pc_sample_t* pc_sampling_record =
|
|
|
|
|
reinterpret_cast<const rocprofiler_record_pc_sample_t*>(begin);
|
|
|
|
|
FlushPCSamplingRecord(pc_sampling_record);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
rocprofiler_next_record(begin, &begin, session_id, buffer_id);
|
|
|
|
|
}
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool is_valid() const { return valid_; }
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
bool valid_{false};
|
2024-08-08 01:14:38 +00:00
|
|
|
std::set<std::string> kernel_filters_;
|
|
|
|
|
std::map<std::string, bool> kernel_filter_cache_;
|
2023-05-19 17:53:17 +00:00
|
|
|
std::atomic<bool> tracer_header_written_{false};
|
|
|
|
|
std::atomic<bool> profiler_header_written_{false};
|
|
|
|
|
|
|
|
|
|
output_file_t roctx_file_{"roctx_trace.txt"}, hsa_api_file_{"hsa_api_trace.txt"},
|
|
|
|
|
hip_api_file_{"hip_api_trace.txt"}, hip_activity_file_{"hcc_ops_trace.txt"},
|
|
|
|
|
hsa_async_copy_file_{"async_copy_trace.txt"}, pc_sample_file_{"pcs_trace.txt"},
|
|
|
|
|
output_file_{"results.txt"};
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
file_plugin_t* file_plugin = nullptr;
|
|
|
|
|
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
ROCPROFILER_EXPORT int rocprofiler_plugin_initialize(uint32_t rocprofiler_major_version,
|
|
|
|
|
uint32_t rocprofiler_minor_version,
|
|
|
|
|
void* data) {
|
|
|
|
|
if (rocprofiler_major_version != ROCPROFILER_VERSION_MAJOR ||
|
|
|
|
|
rocprofiler_minor_version < ROCPROFILER_VERSION_MINOR)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
if (file_plugin != nullptr) return -1;
|
|
|
|
|
|
|
|
|
|
file_plugin = new file_plugin_t();
|
|
|
|
|
if (file_plugin->is_valid()) return 0;
|
|
|
|
|
|
|
|
|
|
// The plugin failed to initialized, destroy it and return an error.
|
|
|
|
|
delete file_plugin;
|
|
|
|
|
file_plugin = nullptr;
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ROCPROFILER_EXPORT void rocprofiler_plugin_finalize() {
|
|
|
|
|
if (!file_plugin) return;
|
|
|
|
|
delete file_plugin;
|
|
|
|
|
file_plugin = nullptr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ROCPROFILER_EXPORT int rocprofiler_plugin_write_buffer_records(
|
|
|
|
|
const rocprofiler_record_header_t* begin, const rocprofiler_record_header_t* end,
|
|
|
|
|
rocprofiler_session_id_t session_id, rocprofiler_buffer_id_t buffer_id) {
|
|
|
|
|
if (!file_plugin || !file_plugin->is_valid()) return -1;
|
|
|
|
|
return file_plugin->WriteBufferRecords(begin, end, session_id, buffer_id);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ROCPROFILER_EXPORT int rocprofiler_plugin_write_record(rocprofiler_record_tracer_t record) {
|
|
|
|
|
if (!file_plugin || !file_plugin->is_valid()) return -1;
|
|
|
|
|
if (record.header.id.handle == 0) return 0;
|
|
|
|
|
file_plugin->FlushTracerRecord(record, rocprofiler_session_id_t{0}, rocprofiler_buffer_id_t{0});
|
|
|
|
|
return 0;
|
|
|
|
|
}
|