2023-06-20 14:33:09 +10:00
|
|
|
#ifndef STAR_EXCEPTION_HPP
|
|
|
|
#define STAR_EXCEPTION_HPP
|
|
|
|
|
2023-06-28 00:20:22 +10:00
|
|
|
#include "StarMemory.hpp"
|
|
|
|
#include "StarOutputProxy.hpp"
|
|
|
|
|
|
|
|
#include <string>
|
2023-06-29 02:31:12 +10:00
|
|
|
#include <sstream>
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
namespace Star {
|
|
|
|
|
2023-06-28 00:20:22 +10:00
|
|
|
template <typename... T>
|
|
|
|
std::string strf(fmt::format_string<T...> fmt, T&&... args);
|
|
|
|
|
2023-06-20 14:33:09 +10:00
|
|
|
class StarException : public std::exception {
|
|
|
|
public:
|
|
|
|
template <typename... Args>
|
2023-06-27 19:24:35 +10:00
|
|
|
static StarException format(fmt::format_string<Args...> fmt, Args const&... args);
|
2023-06-20 14:33:09 +10:00
|
|
|
|
|
|
|
StarException() noexcept;
|
|
|
|
virtual ~StarException() noexcept;
|
|
|
|
|
2023-06-25 18:12:54 +10:00
|
|
|
explicit StarException(std::string message, bool genStackTrace = true) noexcept;
|
2023-06-20 14:33:09 +10:00
|
|
|
explicit StarException(std::exception const& cause) noexcept;
|
|
|
|
StarException(std::string message, std::exception const& cause) noexcept;
|
|
|
|
|
|
|
|
virtual char const* what() const noexcept override;
|
|
|
|
|
|
|
|
// If the given exception is really StarException, then this will call
|
|
|
|
// StarException::printException, otherwise just prints std::exception::what.
|
|
|
|
friend void printException(std::ostream& os, std::exception const& e, bool fullStacktrace);
|
|
|
|
friend std::string printException(std::exception const& e, bool fullStacktrace);
|
|
|
|
friend OutputProxy outputException(std::exception const& e, bool fullStacktrace);
|
|
|
|
|
|
|
|
protected:
|
2023-06-25 18:12:54 +10:00
|
|
|
StarException(char const* type, std::string message, bool genStackTrace = true) noexcept;
|
2023-06-20 14:33:09 +10:00
|
|
|
StarException(char const* type, std::string message, std::exception const& cause) noexcept;
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Takes the ostream to print to, whether to print the full stacktrace. Must
|
|
|
|
// not bind 'this', may outlive the exception in the case of chained
|
|
|
|
// exception causes.
|
|
|
|
function<void(std::ostream&, bool)> m_printException;
|
|
|
|
|
|
|
|
// m_printException will be called without the stack-trace to print
|
|
|
|
// m_whatBuffer, if the what() method is invoked.
|
|
|
|
mutable std::string m_whatBuffer;
|
|
|
|
};
|
|
|
|
|
|
|
|
void printException(std::ostream& os, std::exception const& e, bool fullStacktrace);
|
|
|
|
std::string printException(std::exception const& e, bool fullStacktrace);
|
|
|
|
OutputProxy outputException(std::exception const& e, bool fullStacktrace);
|
|
|
|
|
|
|
|
void printStack(char const* message);
|
|
|
|
|
|
|
|
// Log error and stack-trace and possibly show a dialog box if available, then
|
|
|
|
// abort.
|
|
|
|
void fatalError(char const* message, bool showStackTrace);
|
|
|
|
void fatalException(std::exception const& e, bool showStackTrace);
|
|
|
|
|
|
|
|
#ifdef STAR_DEBUG
|
|
|
|
#define debugPrintStack() \
|
|
|
|
{ Star::printStack("Debug: file " STAR_STR(__FILE__) " line " STAR_STR(__LINE__)); }
|
|
|
|
#define starAssert(COND) \
|
|
|
|
{ \
|
|
|
|
if (COND) \
|
|
|
|
; \
|
|
|
|
else \
|
|
|
|
Star::fatalError("assert failure in file " STAR_STR(__FILE__) " line " STAR_STR(__LINE__), true); \
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
#define debugPrintStack() \
|
|
|
|
{}
|
|
|
|
#define starAssert(COND) \
|
|
|
|
{}
|
|
|
|
#endif
|
|
|
|
|
2023-06-28 00:20:22 +10:00
|
|
|
#define STAR_EXCEPTION(ClassName, BaseName) \
|
|
|
|
class ClassName : public BaseName { \
|
|
|
|
public: \
|
|
|
|
template <typename... Args> \
|
|
|
|
static ClassName format(fmt::format_string<Args...> fmt, Args const&... args) { \
|
|
|
|
return ClassName(strf(fmt, args...)); \
|
|
|
|
} \
|
|
|
|
ClassName() : BaseName(#ClassName, std::string()) {} \
|
2023-06-25 18:12:54 +10:00
|
|
|
explicit ClassName(std::string message, bool genStackTrace = true) : BaseName(#ClassName, move(message), genStackTrace) {} \
|
2023-06-28 00:20:22 +10:00
|
|
|
explicit ClassName(std::exception const& cause) : BaseName(#ClassName, std::string(), cause) {} \
|
|
|
|
ClassName(std::string message, std::exception const& cause) : BaseName(#ClassName, move(message), cause) {} \
|
|
|
|
\
|
|
|
|
protected: \
|
2023-06-25 18:12:54 +10:00
|
|
|
ClassName(char const* type, std::string message, bool genStackTrace = true) : BaseName(type, move(message), genStackTrace) {} \
|
2023-06-28 00:20:22 +10:00
|
|
|
ClassName(char const* type, std::string message, std::exception const& cause) \
|
|
|
|
: BaseName(type, move(message), cause) {} \
|
2023-06-20 14:33:09 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
STAR_EXCEPTION(OutOfRangeException, StarException);
|
|
|
|
STAR_EXCEPTION(IOException, StarException);
|
|
|
|
STAR_EXCEPTION(MemoryException, StarException);
|
|
|
|
|
|
|
|
template <typename... Args>
|
2023-06-27 19:24:35 +10:00
|
|
|
StarException StarException::format(fmt::format_string<Args...> fmt, Args const&... args) {
|
2023-06-20 14:33:09 +10:00
|
|
|
return StarException(strf(fmt, args...));
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|