143 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			143 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| 
 | |
| #pragma once
 | |
| 
 | |
| #if !defined(MIJIN_DEBUG_STACKTRACE_HPP_INCLUDED)
 | |
| #define MIJIN_DEBUG_STACKTRACE_HPP_INCLUDED 1
 | |
| 
 | |
| #include <cmath>
 | |
| #include <iomanip>
 | |
| #include <vector>
 | |
| #if __has_include(<fmt/format.h>)
 | |
| #   include <fmt/format.h>
 | |
| #endif
 | |
| #include "./symbol_info.hpp"
 | |
| #include "../internal/common.hpp"
 | |
| #include "../types/result.hpp"
 | |
| #include "../util/iterators.hpp"
 | |
| 
 | |
| namespace mijin
 | |
| {
 | |
| 
 | |
| //
 | |
| // public defines
 | |
| //
 | |
| 
 | |
| //
 | |
| // public constants
 | |
| //
 | |
| 
 | |
| //
 | |
| // public types
 | |
| //
 | |
| 
 | |
| struct Stackframe
 | |
| {
 | |
|     void* address;
 | |
|     std::string filename;
 | |
|     std::string function;
 | |
|     int lineNumber;
 | |
| };
 | |
| 
 | |
| class Stacktrace
 | |
| {
 | |
| private:
 | |
|     std::vector<Stackframe> frames_;
 | |
| public:
 | |
|     Stacktrace() = default;
 | |
|     Stacktrace(const Stacktrace&) = default;
 | |
|     Stacktrace(Stacktrace&&) = default;
 | |
|     explicit Stacktrace(std::vector<Stackframe> frames) MIJIN_NOEXCEPT : frames_(std::move(frames)) {}
 | |
| 
 | |
|     Stacktrace& operator=(const Stacktrace&) = default;
 | |
|     Stacktrace& operator=(Stacktrace&&) = default;
 | |
| 
 | |
|     [[nodiscard]] const std::vector<Stackframe>& getFrames() const MIJIN_NOEXCEPT { return frames_; }
 | |
| };
 | |
| 
 | |
| //
 | |
| // public functions
 | |
| //
 | |
| 
 | |
| [[nodiscard]] Result<Stacktrace> captureStacktrace(unsigned skipFrames = 0) MIJIN_NOEXCEPT;
 | |
| [[nodiscard]] const Optional<Stacktrace>& getExceptionStacktrace() MIJIN_NOEXCEPT;
 | |
| 
 | |
| template<typename TStream>
 | |
| TStream& operator<<(TStream& stream, const Stackframe& stackframe)
 | |
| {
 | |
|     stream << "[" << stackframe.address << "] " << stackframe.filename << ":" << stackframe.lineNumber << " in " << demangleCPPIdentifier(stackframe.function.c_str());
 | |
| 
 | |
|     return stream;
 | |
| }
 | |
| 
 | |
| template<typename TStream>
 | |
| TStream& operator<<(TStream& stream, const Stacktrace& stacktrace)
 | |
| {
 | |
|     const std::streamsize oldWidth = stream.width();
 | |
|     const std::ios::fmtflags oldFlags = stream.flags();
 | |
|     const int numDigits = static_cast<int>(std::ceil(std::log10(stacktrace.getFrames().size())));
 | |
|     stream << std::left;
 | |
|     for (const auto& [idx, frame] : mijin::enumerate(stacktrace.getFrames()))
 | |
|     {
 | |
|         stream << "  #" << std::setw(numDigits) << idx << std::setw(0) << " at " << frame << "\n";
 | |
|     }
 | |
|     stream << std::setw(oldWidth);
 | |
|     stream.flags(oldFlags);
 | |
|     return stream;
 | |
| }
 | |
| 
 | |
| } // namespace mijin
 | |
| 
 | |
| #if __has_include(<fmt/format.h>)
 | |
| template<>
 | |
| struct fmt::formatter<mijin::Stackframe>
 | |
| {
 | |
|     constexpr auto parse(format_parse_context& ctx) -> decltype(ctx.begin())
 | |
|     {
 | |
|         auto it = ctx.begin();
 | |
|         auto end = ctx.end();
 | |
| 
 | |
|         if (it != end && *it != '}') throw format_error("invalid format");
 | |
| 
 | |
|         return it;
 | |
|     }
 | |
| 
 | |
|     template<typename TContext>
 | |
|     auto format(const mijin::Stackframe& stackframe, TContext& ctx) const -> decltype(ctx.out())
 | |
|     {
 | |
|         auto it = ctx.out();
 | |
|         it = fmt::format_to(it, "[{}] {}:{} in {}", stackframe.address, stackframe.filename,
 | |
|                             stackframe.lineNumber, mijin::demangleCPPIdentifier(stackframe.function.c_str()));
 | |
|         return it;
 | |
|     }
 | |
| };
 | |
| 
 | |
| template<>
 | |
| struct fmt::formatter<mijin::Stacktrace>
 | |
| {
 | |
|     constexpr auto parse(format_parse_context& ctx) -> decltype(ctx.begin())
 | |
|     {
 | |
|         auto it = ctx.begin();
 | |
|         auto end = ctx.end();
 | |
| 
 | |
|         if (it != end && *it != '}') throw format_error("invalid format");
 | |
| 
 | |
|         return it;
 | |
|     }
 | |
| 
 | |
|     template<typename TContext>
 | |
|     auto format(const mijin::Stacktrace& stacktrace, TContext& ctx) const -> decltype(ctx.out())
 | |
|     {
 | |
|         const int numDigits = static_cast<int>(std::ceil(std::log10(stacktrace.getFrames().size())));
 | |
|         auto it = ctx.out();
 | |
|         it = fmt::format_to(it, "[{} frames]", stacktrace.getFrames().size());
 | |
|         for (const auto& [idx, frame] : mijin::enumerate(stacktrace.getFrames()))
 | |
|         {
 | |
|             it = fmt::format_to(it, "\n  #{:<{}} at {}", idx, numDigits, frame);
 | |
|         }
 | |
|         return it;
 | |
|     }
 | |
| };
 | |
| #endif // __has_include(<fmt/format.h>)
 | |
| 
 | |
| #endif // !defined(MIJIN_DEBUG_STACKTRACE_HPP_INCLUDED)
 |