2015-07-22 13:38:36 +02:00
|
|
|
#pragma once
|
|
|
|
#ifndef MESSMER_CPP_UTILS_ASSERT_ASSERT_H
|
|
|
|
#define MESSMER_CPP_UTILS_ASSERT_ASSERT_H
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This implements an ASSERT(expr, msg) macro.
|
|
|
|
* In a debug build, it will crash and halt the program on an assert failure.
|
|
|
|
* In a release build, it will throw an AssertFailed exception instead, which can then be caught.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "AssertFailed.h"
|
|
|
|
#include <iostream>
|
|
|
|
#include <string>
|
2015-09-29 22:43:55 +02:00
|
|
|
#include <execinfo.h>
|
2015-07-22 13:38:36 +02:00
|
|
|
|
|
|
|
namespace cpputils {
|
|
|
|
namespace _assert {
|
|
|
|
inline std::string format(const char *expr, const char *message, const char *file, int line) {
|
2015-09-29 22:43:55 +02:00
|
|
|
// get void*'s for all entries on the stack
|
|
|
|
void *array[10];
|
|
|
|
size_t size = backtrace(array, sizeof(array));
|
|
|
|
char **backtrace_str = backtrace_symbols(array, sizeof(backtrace_str));
|
|
|
|
std::string result = std::string()+"Assertion ["+expr+"] failed in "+file+":"+std::to_string(line)+": "+message+"\n\n";
|
|
|
|
for (unsigned int i = 0; i < size; ++i) {
|
|
|
|
result += std::string(backtrace_str[i]) + "\n";
|
|
|
|
}
|
|
|
|
free(backtrace_str);
|
|
|
|
return result;
|
2015-07-22 13:38:36 +02:00
|
|
|
}
|
|
|
|
|
2015-07-24 20:27:46 +02:00
|
|
|
inline void assert_fail_release [[noreturn]] (const char *expr, const char *message, const char *file, int line) {
|
2015-07-22 13:38:36 +02:00
|
|
|
throw AssertFailed(format(expr, message, file, line));
|
|
|
|
}
|
|
|
|
|
2015-07-24 20:27:46 +02:00
|
|
|
inline void assert_fail_debug [[noreturn]] (const char *expr, const char *message, const char *file, int line) {
|
2015-07-22 13:38:36 +02:00
|
|
|
std::cerr << format(expr, message, file, line) << std::endl;
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef NDEBUG
|
|
|
|
//TODO Check whether disabling assertions in prod affects speed.
|
|
|
|
# define ASSERT(expr, msg) (void)((expr) || (cpputils::_assert::assert_fail_release(#expr, msg, __FILE__, __LINE__),0))
|
|
|
|
#else
|
|
|
|
# define ASSERT(expr, msg) (void)((expr) || (cpputils::_assert::assert_fail_debug(#expr, msg, __FILE__, __LINE__),0))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#endif
|