2019-03-15 13:01:26 -07:00
|
|
|
// Matrix Construct
|
|
|
|
//
|
|
|
|
// Copyright (C) Matrix Construct Developers, Authors & Contributors
|
|
|
|
// Copyright (C) 2016-2019 Jason Volk <jason@zemos.net>
|
|
|
|
//
|
|
|
|
// Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
// purpose with or without fee is hereby granted, provided that the above
|
|
|
|
// copyright notice and this permission notice is present in all copies. The
|
|
|
|
// full license for this software is available in the LICENSE file.
|
|
|
|
|
|
|
|
// This header allows for the use of soft-assertions. It is included before
|
|
|
|
// the standard <assert.h> and these declarations will take precedence.
|
|
|
|
//
|
|
|
|
// These declarations exist because the system declarations may have
|
|
|
|
// __attribute__((noreturn)) and our only modification here is to remove
|
|
|
|
// that. Alternative definitions to the standard library also exist in
|
|
|
|
// ircd/assert.cc
|
|
|
|
|
2019-05-28 15:53:25 -07:00
|
|
|
#pragma once
|
|
|
|
#define HAVE_IRCD_ASSERT_H
|
|
|
|
|
2022-07-16 13:09:14 -07:00
|
|
|
// This file has to be included prior to the standard library assert headers
|
2019-07-08 16:49:43 -07:00
|
|
|
#if defined(RB_ASSERT) && defined(_ASSERT_H_DECLS) && defined(RB_DEBUG)
|
2019-05-28 15:53:25 -07:00
|
|
|
#error "Do not include <assert.h> or <cassert> first."
|
|
|
|
#endif
|
|
|
|
|
2022-07-16 13:09:14 -07:00
|
|
|
// Define an indicator for whether we override standard assert() behavior in
|
|
|
|
// this build if the conditions are appropriate.
|
2019-03-15 13:01:26 -07:00
|
|
|
#if defined(RB_ASSERT) && !defined(NDEBUG)
|
2019-08-11 19:02:50 -07:00
|
|
|
#define _ASSERT_H_DECLS
|
|
|
|
#endif
|
2019-03-15 13:01:26 -07:00
|
|
|
|
2019-08-11 19:02:50 -07:00
|
|
|
// Our utils
|
|
|
|
namespace ircd
|
|
|
|
{
|
|
|
|
void debugtrap() noexcept;
|
2022-07-16 13:09:14 -07:00
|
|
|
|
|
|
|
template<class expr>
|
|
|
|
void always_assert(expr&&) noexcept;
|
|
|
|
|
|
|
|
[[gnu::cold]]
|
|
|
|
void print_assertion(const char *, const char *, const unsigned, const char *) noexcept;
|
|
|
|
|
|
|
|
#if defined(RB_ASSERT_OPTIMISTIC)
|
|
|
|
struct assertion extern assertion;
|
|
|
|
#endif
|
2019-08-11 19:02:50 -07:00
|
|
|
}
|
|
|
|
|
2022-07-16 13:09:14 -07:00
|
|
|
// Alternative declaration of __assert_fail() which may return; our definitions
|
|
|
|
// take on different behaviors depending on the value of RB_ASSERT.
|
|
|
|
#if defined(RB_ASSERT)
|
2019-03-15 13:01:26 -07:00
|
|
|
extern "C" void
|
2019-05-14 15:11:29 -07:00
|
|
|
__attribute__((visibility("default")))
|
2019-08-11 19:02:50 -07:00
|
|
|
__assert_fail(const char *__assertion,
|
|
|
|
const char *__file,
|
|
|
|
unsigned int __line,
|
2022-07-16 13:09:14 -07:00
|
|
|
const char *__function) noexcept;
|
2019-03-15 13:01:26 -07:00
|
|
|
#endif
|
2019-08-08 16:12:52 -07:00
|
|
|
|
2020-12-09 21:37:39 -08:00
|
|
|
// Custom assert
|
2022-07-16 13:09:14 -07:00
|
|
|
#if defined(RB_ASSERT_INTRINSIC) && !defined(NDEBUG)
|
2020-12-09 21:37:39 -08:00
|
|
|
#undef assert
|
2022-07-16 13:09:14 -07:00
|
|
|
#define assert(expr) \
|
|
|
|
({ \
|
|
|
|
if(__builtin_expect(!static_cast<bool>(expr), 0)) \
|
|
|
|
__assert_fail(#expr, __FILE__, __LINE__, __FUNCTION__); \
|
|
|
|
})
|
|
|
|
#elif defined(RB_ASSERT_OPTIMISTIC) && !defined(NDEBUG)
|
|
|
|
#undef assert
|
|
|
|
#define assert(expr) \
|
|
|
|
({ \
|
|
|
|
ircd::assertion.ok &= static_cast<bool>(expr); \
|
2020-12-09 21:37:39 -08:00
|
|
|
})
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// Custom addl cases to avoid any trouble w/ clang
|
2022-07-16 13:09:14 -07:00
|
|
|
#if defined(RB_ASSERT) && defined(__clang__) && !defined(assert) && defined(NDEBUG)
|
|
|
|
#define assert(expr) (static_cast<void>(0))
|
2020-12-09 21:37:39 -08:00
|
|
|
#endif
|
|
|
|
|
2022-07-16 13:09:14 -07:00
|
|
|
#if defined(RB_ASSERT_OPTIMISTIC)
|
|
|
|
/// State structure for optimistic assertion mode.
|
|
|
|
struct ircd::assertion
|
|
|
|
{
|
|
|
|
bool ok alignas(8) {true};
|
|
|
|
const char *__assertion;
|
|
|
|
const char *__file;
|
|
|
|
unsigned int __line;
|
|
|
|
const char *__function;
|
|
|
|
|
|
|
|
bool point();
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(RB_ASSERT_OPTIMISTIC)
|
|
|
|
/// Test-and-reset for pending assertion.
|
|
|
|
[[gnu::hot]]
|
|
|
|
inline bool
|
|
|
|
ircd::assertion::point()
|
|
|
|
{
|
|
|
|
if(__builtin_expect(!ok, 0))
|
|
|
|
{
|
|
|
|
#if defined(__SSE2__) && defined(__clang__)
|
|
|
|
asm volatile ("lfence");
|
|
|
|
#endif
|
|
|
|
__assert_fail(__assertion, __file, __line, __function);
|
|
|
|
ok = true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else return false;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#if defined(RB_ASSERT_INTRINSIC)
|
2020-02-01 17:20:50 -08:00
|
|
|
extern "C" // for clang
|
|
|
|
{
|
2022-07-16 13:09:14 -07:00
|
|
|
/// Override the standard assert behavior, if enabled, to trap into the
|
|
|
|
/// debugger as close as possible to the offending site.
|
2020-02-01 17:20:50 -08:00
|
|
|
extern inline void
|
2023-02-08 18:16:34 -08:00
|
|
|
__attribute__((cold, flatten, always_inline, gnu_inline, artificial, nodebug))
|
2021-09-03 23:32:00 -07:00
|
|
|
__assert_fail(const char *const __assertion,
|
|
|
|
const char *const __file,
|
2020-02-01 17:20:50 -08:00
|
|
|
unsigned int __line,
|
2021-09-03 23:32:00 -07:00
|
|
|
const char *const __function)
|
2022-07-16 13:09:14 -07:00
|
|
|
noexcept
|
2020-02-01 17:20:50 -08:00
|
|
|
{
|
2020-12-05 00:35:50 -08:00
|
|
|
#if defined(__SSE2__) && defined(__clang__)
|
2022-07-16 13:09:14 -07:00
|
|
|
asm volatile ("lfence");
|
2020-12-05 00:35:50 -08:00
|
|
|
#endif
|
2020-02-01 17:20:50 -08:00
|
|
|
ircd::print_assertion(__assertion, __file, __line, __function);
|
|
|
|
ircd::debugtrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-08-11 19:02:50 -07:00
|
|
|
/// Intrinsic to halt execution for examination by a tracing debugger without
|
|
|
|
/// aborting the program.
|
|
|
|
///
|
2019-08-08 16:12:52 -07:00
|
|
|
extern inline void
|
2023-02-08 18:16:34 -08:00
|
|
|
__attribute__((always_inline, gnu_inline, artificial, nodebug))
|
2019-08-08 16:12:52 -07:00
|
|
|
ircd::debugtrap()
|
2019-08-11 19:02:50 -07:00
|
|
|
noexcept
|
2019-08-08 16:12:52 -07:00
|
|
|
{
|
|
|
|
#if defined(__clang__)
|
|
|
|
static_assert(__has_builtin(__builtin_debugtrap));
|
|
|
|
__builtin_debugtrap();
|
|
|
|
#elif defined(__x86_64__)
|
2019-09-02 19:58:04 -07:00
|
|
|
__asm__ volatile ("int $3 # IRCd ASSERTION DEBUG TRAP !!! ");
|
2019-08-08 16:12:52 -07:00
|
|
|
#else
|
|
|
|
__builtin_trap();
|
|
|
|
#endif
|
|
|
|
}
|
2019-12-26 14:38:51 -08:00
|
|
|
|
|
|
|
/// Trap on false expression whether or not NDEBUG.
|
2020-02-23 20:53:27 -08:00
|
|
|
template<class expr>
|
2019-12-26 14:38:51 -08:00
|
|
|
extern inline void
|
2023-02-08 18:16:34 -08:00
|
|
|
__attribute__((always_inline, gnu_inline, artificial, nodebug))
|
2020-02-23 20:53:27 -08:00
|
|
|
ircd::always_assert(expr&& x)
|
2019-12-26 14:38:51 -08:00
|
|
|
noexcept
|
|
|
|
{
|
2020-02-23 20:53:27 -08:00
|
|
|
if(__builtin_expect(!bool(x), 0))
|
2019-12-26 14:38:51 -08:00
|
|
|
debugtrap();
|
|
|
|
}
|