5a7dec797e
Change-Id: Ib720a81105af13898ff745ce0cbc2a48c1f4a980
[ROCm/rocprofiler commit: 08fc21ac31]
182 righe
5.7 KiB
C++
182 righe
5.7 KiB
C++
/* 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 "helper.h"
|
|
|
|
#include <cstdio>
|
|
#include <cstdarg>
|
|
#include <iomanip>
|
|
#include <iostream>
|
|
#include <sstream>
|
|
#include <string>
|
|
#include <amd_comgr/amd_comgr.h>
|
|
|
|
#define amd_comgr_(call) \
|
|
do { \
|
|
if (amd_comgr_status_t status = amd_comgr_##call; status != AMD_COMGR_STATUS_SUCCESS) { \
|
|
const char* reason = ""; \
|
|
amd_comgr_status_string(status, &reason); \
|
|
fatal(#call " failed: %s", reason); \
|
|
} \
|
|
} while (false)
|
|
|
|
namespace rocmtools {
|
|
|
|
std::string string_vprintf(const char* format, va_list va) {
|
|
va_list copy;
|
|
|
|
va_copy(copy, va);
|
|
size_t size = vsnprintf(NULL, 0, format, copy);
|
|
va_end(copy);
|
|
|
|
std::string str(size, '\0');
|
|
vsprintf(&str[0], format, va);
|
|
|
|
return str;
|
|
}
|
|
|
|
std::string string_printf(const char* format, ...) {
|
|
va_list va;
|
|
va_start(va, format);
|
|
std::string str(string_vprintf(format, va));
|
|
va_end(va);
|
|
|
|
return str;
|
|
}
|
|
|
|
[[maybe_unused]] void warning(const char* format, ...) {
|
|
va_list va;
|
|
va_start(va, format);
|
|
vfprintf(stderr, format, va);
|
|
va_end(va);
|
|
}
|
|
|
|
[[maybe_unused]] void fatal [[noreturn]] (const char* format, ...) {
|
|
va_list va;
|
|
va_start(va, format);
|
|
std::string message = string_vprintf(format, va);
|
|
va_end(va);
|
|
|
|
#if defined(ENABLE_BACKTRACE)
|
|
BackTraceInfo info;
|
|
|
|
info.sstream << std::endl << "Backtrace:";
|
|
info.state = ::backtrace_create_state("/proc/self/exe", 0, errorCallback, &info);
|
|
::backtrace_full(info.state, 1, fullCallback, errorCallback, &info);
|
|
|
|
message += info.sstream.str();
|
|
#endif /* defined (ENABLE_BACKTRACE) */
|
|
|
|
std::string errmsg("ROCProfiler: fatal error: " + message);
|
|
fputs(errmsg.c_str(), stderr);
|
|
|
|
// throw(errmsg);
|
|
abort();
|
|
}
|
|
|
|
/* The function extracts the kernel name from
|
|
input string. By using the iterators it finds the
|
|
window in the string which contains only the kernel name.
|
|
For example 'Foo<int, float>::foo(a[], int (int))' -> 'foo'*/
|
|
std::string truncate_name(const std::string& name) {
|
|
auto rit = name.rbegin();
|
|
auto rend = name.rend();
|
|
uint32_t counter = 0;
|
|
char open_token = 0;
|
|
char close_token = 0;
|
|
while (rit != rend) {
|
|
if (counter == 0) {
|
|
switch (*rit) {
|
|
case ')':
|
|
counter = 1;
|
|
open_token = ')';
|
|
close_token = '(';
|
|
break;
|
|
case '>':
|
|
counter = 1;
|
|
open_token = '>';
|
|
close_token = '<';
|
|
break;
|
|
case ']':
|
|
counter = 1;
|
|
open_token = ']';
|
|
close_token = '[';
|
|
break;
|
|
case ' ':
|
|
++rit;
|
|
continue;
|
|
}
|
|
if (counter == 0) break;
|
|
} else {
|
|
if (*rit == open_token) counter++;
|
|
if (*rit == close_token) counter--;
|
|
}
|
|
++rit;
|
|
}
|
|
auto rbeg = rit;
|
|
while ((rit != rend) && (*rit != ' ') && (*rit != ':')) rit++;
|
|
return name.substr(rend - rit, rit - rbeg);
|
|
}
|
|
|
|
// C++ symbol demangle
|
|
std::string cxx_demangle(const std::string& symbol) {
|
|
amd_comgr_data_t mangled_data;
|
|
amd_comgr_(create_data(AMD_COMGR_DATA_KIND_BYTES, &mangled_data));
|
|
amd_comgr_(set_data(mangled_data, symbol.size(), symbol.data()));
|
|
|
|
amd_comgr_data_t demangled_data;
|
|
amd_comgr_(demangle_symbol_name(mangled_data, &demangled_data));
|
|
|
|
size_t demangled_size = 0;
|
|
amd_comgr_(get_data(demangled_data, &demangled_size, nullptr));
|
|
|
|
std::string demangled_str;
|
|
demangled_str.resize(demangled_size);
|
|
// amd_comgr_(get_data(demangled_data, &demangled_size, demangled_str.data())); // TODO: uncomment
|
|
|
|
amd_comgr_(release_data(mangled_data));
|
|
amd_comgr_(release_data(demangled_data));
|
|
return demangled_str;
|
|
}
|
|
|
|
// check if string has special char
|
|
bool has_special_char(std::string const& str) {
|
|
return std::find_if(str.begin(), str.end(), [](unsigned char ch) {
|
|
return !(isalnum(ch) || ch == '_' || ch == ':' || ch == ' ');
|
|
}) != str.end();
|
|
}
|
|
|
|
// check if string has correct counter format
|
|
bool has_counter_format(std::string const& str) {
|
|
return std::find_if(str.begin(), str.end(), [](unsigned char ch) {
|
|
return (isalnum(ch) || ch == '_' || ch != ':');
|
|
}) != str.end();
|
|
}
|
|
|
|
// trims the begining of the line for spaces
|
|
std::string left_trim(const std::string& s) {
|
|
const std::string WHITESPACE = " \n\r\t\f\v";
|
|
size_t start = s.find_first_not_of(WHITESPACE);
|
|
return (start == std::string::npos) ? "" : s.substr(start);
|
|
}
|
|
|
|
|
|
} // namespace rocmtools
|