// MIT License // // Copyright (c) 2022 Advanced Micro Devices, Inc. All Rights Reserved. // // 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 "library/sampling.hpp" #include "library/config.hpp" #include "library/debug.hpp" #include "library/ptl.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace omnitrace { namespace sampling { using bundle_t = tim::lightweight_tuple; using sampler_t = tim::sampling::sampler; } // namespace sampling } // namespace omnitrace TIMEMORY_DEFINE_CONCRETE_TRAIT(check_signals, omnitrace::sampling::sampler_t, std::false_type) TIMEMORY_DEFINE_CONCRETE_TRAIT(buffer_size, omnitrace::sampling::sampler_t, TIMEMORY_ESC(std::integral_constant)) namespace omnitrace { namespace sampling { using signal_type_instances = omnitrace_thread_data, api::sampling>; using backtrace_init_instances = omnitrace_thread_data; using sampler_running_instances = omnitrace_thread_data; using papi_vector_instances = omnitrace_thread_data; namespace { std::unique_ptr& get_papi_vector(int64_t _tid) { static auto& _v = papi_vector_instances::instances(); if(_tid == threading::get_id()) papi_vector_instances::construct(); return _v.at(_tid); } std::unique_ptr& get_backtrace_init(int64_t _tid) { static auto& _v = backtrace_init_instances::instances(); return _v.at(_tid); } std::unique_ptr& get_sampler_running(int64_t _tid) { static auto& _v = sampler_running_instances::instances(); return _v.at(_tid); } std::unique_ptr>& get_signal_types(int64_t _tid) { static auto& _v = signal_type_instances::instances(); // on the main thread, use both SIGALRM and SIGPROF. // on secondary threads, only use SIGPROF. signal_type_instances::construct((_tid == 0) ? std::set{ SIGALRM, SIGPROF } : std::set{ SIGPROF }); return _v.at(_tid); } template void thread_sigmask(Args... _args) { auto _err = pthread_sigmask(_args...); if(_err != 0) { errno = _err; perror("pthread_sigmask"); exit(EXIT_FAILURE); } } template sigset_t get_signal_set(Tp&& _v) { sigset_t _sigset; sigemptyset(&_sigset); for(auto itr : _v) sigaddset(&_sigset, itr); return _sigset; } template std::string get_signal_names(Tp&& _v) { std::string _sig_names{}; for(auto&& itr : _v) _sig_names += std::get<0>(tim::signal_settings::get_info( static_cast(itr))) + " "; return _sig_names.substr(0, _sig_names.length() - 1); } } // namespace std::set setup() { return backtrace::configure(true); } std::set shutdown() { return backtrace::configure(false); } void block_signals(std::set _signals) { if(_signals.empty()) _signals = *get_signal_types(threading::get_id()); if(_signals.empty()) { OMNITRACE_PRINT("No signals to block...\n"); return; } OMNITRACE_DEBUG("Blocking signals [%s] on thread #%lu...\n", get_signal_names(_signals).c_str(), threading::get_id()); sigset_t _v = get_signal_set(_signals); thread_sigmask(SIG_BLOCK, &_v, nullptr); } void unblock_signals(std::set _signals) { if(_signals.empty()) _signals = *get_signal_types(threading::get_id()); if(_signals.empty()) { OMNITRACE_PRINT("No signals to unblock...\n"); return; } OMNITRACE_DEBUG("Unblocking signals [%s] on thread #%lu...\n", get_signal_names(_signals).c_str(), threading::get_id()); sigset_t _v = get_signal_set(_signals); thread_sigmask(SIG_UNBLOCK, &_v, nullptr); } std::unique_ptr& get_sampler(int64_t _tid) { static auto& _v = sampler_instances::instances(); return _v.at(_tid); } } // namespace sampling } // namespace omnitrace