// 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/debug.hpp" #include "library/binary/address_range.hpp" #include "library/state.hpp" #include #include #include #include #include namespace omnitrace { namespace debug { namespace { struct source_location_history { std::array data = {}; size_t size = 0; }; auto& get_source_location_history() { static thread_local auto _v = source_location_history{}; return _v; } auto _protect_lock = std::atomic{ false }; auto _protect_unlock = std::atomic{ false }; } // namespace void set_source_location(source_location&& _v) { auto& _hist = get_source_location_history(); auto _idx = _hist.size++; _hist.data.at(_idx % _hist.data.size()) = _v; } lock::lock() : m_lk{ tim::type_mutex(), std::defer_lock } { if(!m_lk.owns_lock() && !_protect_lock) { _protect_lock.store(true); push_thread_state(ThreadState::Internal); m_lk.lock(); _protect_lock.store(false); } } lock::~lock() { if(m_lk.owns_lock() && !_protect_unlock) { _protect_unlock.store(true); m_lk.unlock(); pop_thread_state(); _protect_unlock.store(false); } } FILE* get_file() { static FILE* _v = []() { auto&& _fname = tim::get_env("OMNITRACE_LOG_FILE", ""); if(!_fname.empty()) tim::log::monochrome() = true; return (_fname.empty()) ? stderr : tim::filepath::fopen(_fname, "w"); }(); return _v; } } // namespace debug template std::string as_hex(Tp _v, size_t _width) { std::stringstream _ss; _ss.fill('0'); _ss << "0x" << std::hex << std::setw(_width) << _v; return _ss.str(); } template <> std::string as_hex(address_range_t _v, size_t _width) { return (_v.is_range()) ? JOIN('-', as_hex(_v.low, _width), as_hex(_v.high, _width)) : as_hex(_v.low, _width); } template std::string as_hex(int32_t, size_t); template std::string as_hex(uint32_t, size_t); template std::string as_hex(int64_t, size_t); template std::string as_hex(uint64_t, size_t); template std::string as_hex(void*, size_t); } // namespace omnitrace