soi-header/include/bits/soi-dbg.hpp

240 lines
6.8 KiB
C++
Raw Normal View History

2019-10-06 16:45:29 +02:00
// Copyright Johannes Kapfhammer 2019.
// Distributed under the Boost Software License, Version 1.0.
// (See accompanying file LICENSE_1_0.txt or copy at
// http://www.boost.org/LICENSE_1_0.txt)
//
// A debug macro: allows debugging with dbg(<expression>);
// initialize with dbg_init() to enable colorized output
//
#ifndef SOI_DBG
#define SOI_DBG
2019-10-06 16:45:29 +02:00
#include <cstddef>
#include <cstring>
2019-10-06 17:19:10 +02:00
#include <iostream>
2019-10-06 18:38:23 +02:00
#include <ostream>
2019-10-06 17:19:10 +02:00
#include <sstream>
2019-10-06 18:38:23 +02:00
#include <stdexcept>
2019-10-06 16:45:29 +02:00
#ifdef __GNUG__
#include <cstdlib>
#include <cxxabi.h>
2019-10-06 21:55:47 +02:00
#include <memory>
#endif
2019-10-06 16:45:29 +02:00
#if defined(__unix__) || (defined(__APPLE__) && defined(__MACH__))
#include <unistd.h>
#endif
#include "soi-pretty.hpp"
2019-10-06 18:37:37 +02:00
namespace soi {
2019-10-06 16:45:29 +02:00
namespace detail {
// ----------------------------------------------------------------------------
// pretty printing the type name
2019-10-06 16:45:29 +02:00
2019-10-06 21:55:47 +02:00
struct string_view {
char const *data;
std::size_t size;
2019-10-06 16:45:29 +02:00
};
2019-10-06 21:55:47 +02:00
template <class T> string_view get_name() {
char const *p = __PRETTY_FUNCTION__;
while (*p++ != '=')
;
for (; *p == ' '; ++p)
;
char const *p2 = p;
int count = 1;
for (;; ++p2) {
switch (*p2) {
case '[':
++count;
break;
case ']':
--count;
if (!count)
return {p, std::size_t(p2 - p)};
}
2019-10-06 21:55:47 +02:00
}
return {};
2019-10-06 16:45:29 +02:00
}
2019-10-06 21:55:47 +02:00
std::string replace_all(std::string str, const std::string &from,
const std::string &to) {
size_t start_pos = 0;
while ((start_pos = str.find(from, start_pos)) != std::string::npos) {
str.replace(start_pos, from.length(), to);
start_pos +=
to.length(); // Handles case where 'to' is a substring of 'from'
}
return str;
2019-10-06 16:45:29 +02:00
}
std::string sanitize_type(std::string s) {
auto f = [&](const char *a, const char *b) {
2019-10-06 21:55:47 +02:00
s = replace_all(std::move(s), a, b);
};
f("std::", "");
f("__debug::", "");
f("__cxx11::", "");
f("basic_string<char>", "string");
f("long int", "int");
return s;
2019-10-06 17:33:18 +02:00
}
template <class T> std::string sanitized_type_name() {
auto t = get_name<T>();
return sanitize_type(std::string(t.data, t.size));
2019-10-06 16:45:29 +02:00
}
2019-10-06 16:45:29 +02:00
// ----------------------------------------------------------------------------
// colorized output
bool tty_supports_colors() {
#if defined(__unix__) || (defined(__APPLE__) && defined(__MACH__))
return isatty(STDERR_FILENO);
#else
return true;
#endif
}
int has_environment_color_overwrite() {
2019-10-06 18:38:23 +02:00
if (const char *color_enabled = std::getenv("SOI_COLOR")) {
2019-10-06 16:45:29 +02:00
if (!std::strcmp(color_enabled, "1"))
return 1;
if (!std::strcmp(color_enabled, "0"))
return 0;
}
return -1;
}
bool are_colors_enabled() {
int c = has_environment_color_overwrite();
if (c == -1)
return tty_supports_colors();
return static_cast<bool>(c);
}
// ----------------------------------------------------------------------------
// init and static variables
static bool colors_enabled = false;
2019-10-06 18:38:23 +02:00
static const char *ANSI_DEBUG = "";
static const char *ANSI_EXPRESSION = "";
static const char *ANSI_VALUE = "";
static const char *ANSI_TYPE = "";
static const char *ANSI_MESSAGE = "";
static const char *ANSI_RESET = "";
2019-10-06 16:45:29 +02:00
void dbg_init(bool with_colors) {
if (with_colors) {
bool colors_enabled = true;
2019-10-06 17:36:04 +02:00
ANSI_DEBUG = "\x1b[37m";
2019-10-06 16:45:29 +02:00
ANSI_EXPRESSION = "\x1b[36m";
ANSI_VALUE = "\x1b[01m";
ANSI_TYPE = "\x1b[32m";
2019-10-06 17:36:04 +02:00
ANSI_MESSAGE = "\x1b[31;01m";
2019-10-06 16:45:29 +02:00
ANSI_RESET = "\x1b[0m";
} else {
bool colors_enabled = false;
ANSI_DEBUG = "";
ANSI_EXPRESSION = "";
ANSI_VALUE = "";
ANSI_TYPE = "";
ANSI_MESSAGE = "";
ANSI_RESET = "";
}
}
2019-10-06 18:38:23 +02:00
void dbg_init() { dbg_init(are_colors_enabled()); }
2019-10-06 16:45:29 +02:00
// ----------------------------------------------------------------------------
// printer
template <typename T>
T &&dbg_print(T &&value, std::string const &type, char const *file, int line,
2019-10-06 18:38:23 +02:00
char const *function_name, char const *expression) {
const T &ref = value;
std::stringstream
value_buffer; // avoid nesting of dbg macros within print functinos
2019-10-06 18:37:37 +02:00
soi::print(value_buffer, ref);
2019-10-06 16:45:29 +02:00
2019-10-06 18:38:23 +02:00
std::cerr << ANSI_DEBUG << "[" << file << ":" << line << " (" << function_name
<< ")] " << ANSI_RESET << ANSI_EXPRESSION << expression
<< ANSI_RESET << " = " << ANSI_VALUE << value_buffer.rdbuf()
<< ANSI_RESET << " (" << ANSI_TYPE << type << ANSI_RESET << ")"
2019-10-06 16:45:29 +02:00
<< '\n';
return std::forward<T>(value);
}
2019-10-06 18:38:23 +02:00
template <unsigned int N>
auto dbg_print(const char (&msg)[N], std::string const &, char const *file,
2019-10-06 18:38:23 +02:00
int line, char const *function_name, char const *expression)
-> decltype(msg) {
std::cerr << ANSI_DEBUG << "[" << file << ":" << line << " (" << function_name
<< ")] " << ANSI_RESET << ANSI_MESSAGE << msg << ANSI_RESET << '\n';
2019-10-06 16:45:29 +02:00
return msg;
}
2019-10-06 21:17:37 +02:00
void dbg_print_status(char const *file, int line, char const *function_name) {
std::cerr << ANSI_VALUE << "[" << file << ":" << line << " (" << function_name
2019-10-07 13:10:16 +02:00
<< ")]" << ANSI_RESET << '\n';
2019-10-06 21:17:37 +02:00
}
2019-10-06 18:38:23 +02:00
template <typename T> T &&identity(T &&t) { return std::forward<T>(t); }
2019-10-06 16:45:29 +02:00
} // end namespace detail
2019-10-06 18:38:23 +02:00
} // namespace soi
2019-10-06 16:45:29 +02:00
#ifdef SOI_RELEASE
2019-10-06 16:45:29 +02:00
#define dbg(...) dbg_macro::identity(__VA_ARGS__)
#else
2019-10-06 21:17:37 +02:00
#if defined(__clang__)
2019-10-06 21:55:47 +02:00
#define SOI_ARG16(_0, _1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, \
_14, _15, ...) \
_15
#define SOI_IS_NONEMPTY(...) \
SOI_ARG16(1, ##__VA_ARGS__, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 0)
2019-10-06 21:17:37 +02:00
2019-10-06 21:55:47 +02:00
#define SOI_DBG_IMPL_0() \
2019-10-06 21:17:37 +02:00
soi::detail::dbg_print_status(__FILE__, __LINE__, __func__)
2019-10-06 21:55:47 +02:00
#define SOI_DBG_IMPL_1(...) \
soi::detail::dbg_print( \
(__VA_ARGS__), \
soi::detail::sanitized_type_name<decltype(__VA_ARGS__)>(), __FILE__, \
__LINE__, __func__, #__VA_ARGS__)
2019-10-06 21:17:37 +02:00
#else
#define SOI_IS_NONEMPTY(...) __VA_OPT__(HAS)
2019-10-06 21:55:47 +02:00
#define SOI_DBG_IMPL_() \
soi::detail::dbg_print_status(__FILE__, __LINE__, __func__)
2019-10-06 21:55:47 +02:00
#define SOI_DBG_IMPL_HAS(...) \
soi::detail::dbg_print( \
(__VA_ARGS__), \
soi::detail::sanitized_type_name<decltype(__VA_ARGS__)>(), __FILE__, \
__LINE__, __func__, #__VA_ARGS__)
#endif
2019-10-06 21:17:37 +02:00
#define SOI_CAT(a, ...) PRIMITIVE_CAT(a, __VA_ARGS__)
2019-10-06 21:55:47 +02:00
#define SOI_PRIMITIVE_CAT(a, ...) a##__VA_ARGS__
2019-10-06 21:17:37 +02:00
2019-10-06 21:55:47 +02:00
#define SOI_DBG_IMPL(is_nonempty, ...) \
2019-10-06 21:17:37 +02:00
SOI_PRIMITIVE_CAT(SOI_DBG_IMPL_, is_nonempty)(__VA_ARGS__)
2019-10-06 21:55:47 +02:00
#define dbg(...) SOI_DBG_IMPL(SOI_IS_NONEMPTY(__VA_ARGS__), __VA_ARGS__)
2019-10-06 16:45:29 +02:00
#endif
2019-10-06 18:38:23 +02:00
#endif // SOI_DBG