/* Copyright (c) 2018-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 "inc/roctx.h" #include "inc/roctracer_roctx.h" #include #include #include #include #include "inc/ext/prof_protocol.h" #include "util/exception.h" #include "util/logger.h" #define PUBLIC_API __attribute__((visibility("default"))) #define CONSTRUCTOR_API __attribute__((constructor)) #define DESTRUCTOR_API __attribute__((destructor)) #define API_METHOD_PREFIX \ roctx_status_t err = ROCTX_STATUS_SUCCESS; \ try { #define API_METHOD_SUFFIX \ } \ catch (std::exception & e) { \ ERR_LOGGING(__FUNCTION__ << "(), " << e.what()); \ err = roctx::GetExcStatus(e); \ } \ return (err == ROCTX_STATUS_SUCCESS) ? 0 : -1; #define API_METHOD_SUFFIX_NRET \ } \ catch (std::exception & e) { \ ERR_LOGGING(__FUNCTION__ << "(), " << e.what()); \ err = roctx::GetExcStatus(e); \ } \ (void)err; #define API_METHOD_CATCH(X) \ } \ catch (std::exception & e) { \ ERR_LOGGING(__FUNCTION__ << "(), " << e.what()); \ } \ (void)err; \ return X; inline uint32_t GetPid() { return syscall(__NR_getpid); } inline uint32_t GetTid() { return syscall(__NR_gettid); } //////////////////////////////////////////////////////////////////////////////// // Library errors enumaration typedef enum { ROCTX_STATUS_SUCCESS = 0, ROCTX_STATUS_ERROR = 1, } roctx_status_t; /////////////////////////////////////////////////////////////////////////////////////////////////// // Library implementation // namespace roctx { typedef std::stack message_stack_t; typedef std::map thread_map_t; typedef std::mutex map_mutex_t; map_mutex_t map_mutex; thread_map_t thread_map; static thread_local message_stack_t* message_stack = NULL; roctx_status_t GetExcStatus(const std::exception& e) { const roctracer::util::exception* roctx_exc_ptr = dynamic_cast(&e); return (roctx_exc_ptr) ? static_cast(roctx_exc_ptr->status()) : ROCTX_STATUS_ERROR; } void thread_data_init() { message_stack = new message_stack_t; const auto tid = GetTid(); std::lock_guard lck(map_mutex); thread_map[tid] = message_stack; } // callbacks table extern cb_table_t cb_table; } // namespace roctx // Logger instantiation roctracer::util::Logger::mutex_t roctracer::util::Logger::mutex_; std::atomic roctracer::util::Logger::instance_{}; std::atomic roctx_range_counter(0); /////////////////////////////////////////////////////////////////////////////////////////////////// // Public library methods // extern "C" { PUBLIC_API uint32_t roctx_version_major() { return ROCTX_VERSION_MAJOR; } PUBLIC_API uint32_t roctx_version_minor() { return ROCTX_VERSION_MINOR; } PUBLIC_API const char* roctracer_error_string() { return strdup(roctracer::util::Logger::LastMessage().c_str()); } PUBLIC_API void roctxMarkA(const char* message) { API_METHOD_PREFIX roctx_api_data_t api_data{}; api_data.args.roctxMarkA.message = strdup(message); activity_rtapi_callback_t api_callback_fun = NULL; void* api_callback_arg = NULL; roctx::cb_table.get(ROCTX_API_ID_roctxMarkA, &api_callback_fun, &api_callback_arg); if (api_callback_fun) api_callback_fun(ACTIVITY_DOMAIN_ROCTX, ROCTX_API_ID_roctxMarkA, &api_data, api_callback_arg); API_METHOD_SUFFIX_NRET } PUBLIC_API int roctxRangePushA(const char* message) { API_METHOD_PREFIX if (roctx::message_stack == NULL) roctx::thread_data_init(); roctx_api_data_t api_data{}; api_data.args.roctxRangePushA.message = strdup(message); activity_rtapi_callback_t api_callback_fun = NULL; void* api_callback_arg = NULL; roctx::cb_table.get(ROCTX_API_ID_roctxRangePushA, &api_callback_fun, &api_callback_arg); if (api_callback_fun) api_callback_fun(ACTIVITY_DOMAIN_ROCTX, ROCTX_API_ID_roctxRangePushA, &api_data, api_callback_arg); roctx::message_stack->push(strdup(message)); return roctx::message_stack->size() - 1; API_METHOD_CATCH(-1); } PUBLIC_API int roctxRangePop() { API_METHOD_PREFIX if (roctx::message_stack == NULL) roctx::thread_data_init(); roctx_api_data_t api_data{}; activity_rtapi_callback_t api_callback_fun = NULL; void* api_callback_arg = NULL; roctx::cb_table.get(ROCTX_API_ID_roctxRangePop, &api_callback_fun, &api_callback_arg); if (api_callback_fun) api_callback_fun(ACTIVITY_DOMAIN_ROCTX, ROCTX_API_ID_roctxRangePop, &api_data, api_callback_arg); if (roctx::message_stack->empty()) { EXC_ABORT(ROCTX_STATUS_ERROR, "Pop from empty stack!"); } else { roctx::message_stack->pop(); } return roctx::message_stack->size(); API_METHOD_CATCH(-1) } PUBLIC_API roctx_range_id_t roctxRangeStartA(const char* message) { API_METHOD_PREFIX roctx_range_counter++; roctx_api_data_t api_data{}; api_data.args.roctxRangeStartA.message = strdup(message); api_data.args.roctxRangeStartA.id = roctx_range_counter; activity_rtapi_callback_t api_callback_fun = NULL; void* api_callback_arg = NULL; roctx::cb_table.get(ROCTX_API_ID_roctxRangeStartA, &api_callback_fun, &api_callback_arg); if (api_callback_fun) api_callback_fun(ACTIVITY_DOMAIN_ROCTX, ROCTX_API_ID_roctxRangeStartA, &api_data, api_callback_arg); return roctx_range_counter; API_METHOD_CATCH(-1); } PUBLIC_API void roctxRangeStop(roctx_range_id_t rangeId) { API_METHOD_PREFIX roctx_api_data_t api_data{}; api_data.args.roctxRangeStop.id = rangeId; activity_rtapi_callback_t api_callback_fun = NULL; void* api_callback_arg = NULL; roctx::cb_table.get(ROCTX_API_ID_roctxRangeStop, &api_callback_fun, &api_callback_arg); if (api_callback_fun) api_callback_fun(ACTIVITY_DOMAIN_ROCTX, ROCTX_API_ID_roctxRangeStop, &api_data, api_callback_arg); API_METHOD_SUFFIX_NRET } PUBLIC_API void RangeStackIterate(roctx_range_iterate_cb_t callback, void* arg) { for (const auto& entry : roctx::thread_map) { const auto tid = entry.first; for (roctx::message_stack_t stack = *(entry.second); !stack.empty(); stack.pop()) { std::string message = stack.top(); roctx_range_data_t data{}; data.message = message.c_str(); data.tid = tid; callback(&data, arg); } } } } // extern "C"