cocos-engine-external/sources/rtm/impl/error.h

183 lines
6.1 KiB
C++

#pragma once
////////////////////////////////////////////////////////////////////////////////
// The MIT License (MIT)
//
// Copyright (c) 2017 Nicholas Frechette & Animation Compression Library contributors
// Copyright (c) 2018 Nicholas Frechette & Realtime Math contributors
//
// 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 "rtm/impl/compiler_utils.h"
RTM_IMPL_FILE_PRAGMA_PUSH
//////////////////////////////////////////////////////////////////////////
// This library uses a simple system to handle asserts. Asserts are fatal and must terminate
// otherwise the behavior is undefined if execution continues.
//
// A total of 4 behaviors are supported:
// - We can print to stderr and abort
// - We can throw and exception
// - We can call a custom function
// - Do nothing and strip the check at compile time (default behavior)
//
// Aborting:
// In order to enable the aborting behavior, simply define the macro RTM_ON_ASSERT_ABORT:
// #define RTM_ON_ASSERT_ABORT
//
// Throwing:
// In order to enable the throwing behavior, simply define the macro RTM_ON_ASSERT_THROW:
// #define RTM_ON_ASSERT_THROW
// Note that the type of the exception thrown is std::runtime_error.
//
// Custom function:
// In order to enable the custom function calling behavior, define the macro RTM_ON_ASSERT_CUSTOM
// with the name of the function to call:
// #define RTM_ON_ASSERT_CUSTOM on_custom_assert_impl
// Note that the function signature is as follow:
// void on_custom_assert_impl(const char* expression, int line, const char* file, const char* format, ...) {}
//
// You can also define your own assert implementation by defining the RTM_ASSERT macro as well:
// #define RTM_ON_ASSERT_CUSTOM
// #define RTM_ASSERT(expression, format, ...) checkf(expression, ANSI_TO_TCHAR(format), #__VA_ARGS__)
//
// No checks:
// By default if no macro mentioned above is defined, all asserts will be stripped
// at compile time.
//////////////////////////////////////////////////////////////////////////
#if defined(RTM_ON_ASSERT_ABORT)
#include <cstdio>
#include <cstdarg>
#include <cstdlib>
namespace rtm
{
namespace rtm_impl
{
RTM_DISABLE_SECURITY_COOKIE_CHECK inline void on_assert_abort(const char* expression, int line, const char* file, const char* format, ...)
{
(void)expression;
(void)line;
(void)file;
va_list args;
va_start(args, format);
std::vfprintf(stderr, format, args);
std::fprintf(stderr, "\n");
va_end(args);
std::abort();
}
}
}
#define RTM_ASSERT(expression, format, ...) do { if (!(expression)) rtm::rtm_impl::on_assert_abort(#expression, __LINE__, __FILE__, (format), ## __VA_ARGS__); } while(false)
#define RTM_HAS_ASSERT_CHECKS
#define RTM_NO_EXCEPT noexcept
#elif defined(RTM_ON_ASSERT_THROW)
#include <cstdio>
#include <cstdarg>
#include <string>
#include <stdexcept>
namespace rtm
{
class runtime_assert final : public std::runtime_error
{
public:
explicit runtime_assert(const std::string& message) : std::runtime_error(message.c_str()) {}
explicit runtime_assert(const char* message) : std::runtime_error(message) {}
};
namespace rtm_impl
{
RTM_DISABLE_SECURITY_COOKIE_CHECK inline void on_assert_throw(const char* expression, int line, const char* file, const char* format, ...)
{
(void)expression;
(void)line;
(void)file;
constexpr int buffer_size = 64 * 1024;
char buffer[buffer_size];
va_list args;
va_start(args, format);
const int count = vsnprintf(buffer, buffer_size, format, args);
va_end(args);
if (count >= 0 && count < buffer_size)
throw runtime_assert(std::string(&buffer[0], count));
else
throw runtime_assert("Failed to format assert message!\n");
}
}
}
#define RTM_ASSERT(expression, format, ...) do { if (!(expression)) rtm::rtm_impl::on_assert_throw(#expression, __LINE__, __FILE__, (format), ## __VA_ARGS__); } while(false)
#define RTM_HAS_ASSERT_CHECKS
#define RTM_NO_EXCEPT
#elif defined(RTM_ON_ASSERT_CUSTOM)
#if !defined(RTM_ASSERT)
#define RTM_ASSERT(expression, format, ...) do { if (!(expression)) RTM_ON_ASSERT_CUSTOM(#expression, __LINE__, __FILE__, (format), ## __VA_ARGS__); } while(false)
#endif
#define RTM_HAS_ASSERT_CHECKS
#define RTM_NO_EXCEPT
#else
#define RTM_ASSERT(expression, format, ...) ((void)0)
#define RTM_NO_EXCEPT noexcept
#endif
//////////////////////////////////////////////////////////////////////////
// Deprecation support
//////////////////////////////////////////////////////////////////////////
#if defined(__has_cpp_attribute) && __cplusplus >= 201402L
#if __has_cpp_attribute(deprecated)
#define RTM_DEPRECATED(msg) [[deprecated(msg)]]
#endif
#endif
#if !defined(RTM_DEPRECATED)
#if defined(RTM_COMPILER_GCC) || defined(RTM_COMPILER_CLANG)
#define RTM_DEPRECATED(msg) __attribute__((deprecated))
#elif defined(RTM_COMPILER_MSVC)
#define RTM_DEPRECATED(msg) __declspec(deprecated)
#else
#define RTM_DEPRECATED(msg)
#endif
#endif
RTM_IMPL_FILE_PRAGMA_POP