IAssert.h#
Fully qualified name: carb/assert/IAssert.h
File members: carb/assert/IAssert.h
// SPDX-FileCopyrightText: Copyright (c) 2019-2025 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
// SPDX-License-Identifier: LicenseRef-NvidiaProprietary
//
// NVIDIA CORPORATION, its affiliates and licensors retain all intellectual
// property and proprietary rights in and to this material, related
// documentation and any modifications thereto. Any use, reproduction,
// disclosure or distribution of this material and related documentation
// without an express license agreement from NVIDIA CORPORATION or
// its affiliates is strictly prohibited.
#pragma once
#include "../Interface.h"
#include <cstdarg>
#define carb_assert_IAssert_latest CARB_HEXVERSION(1, 1)
#ifndef carb_assert_IAssert
# define carb_assert_IAssert CARB_HEXVERSION(1, 0)
#endif
// NOTE: this interface is included and used in a very low level header. No more headers than are
// absolutely necessary should be included from here.
#include "../Interface.h"
#include <stdarg.h>
namespace carb::assert
{
struct IAssert;
}
CARB_WEAKLINK CARB_HIDDEN carb::assert::IAssert* g_carbAssert;
namespace carb
{
namespace assert
{
enum class Type : uint16_t
{
eAssert,
eCheck,
eReleaseAssert,
eFatal
};
enum class Result
{
eCallNext,
eContinueExecution,
eHalt,
};
struct AssertEntry
{
uint16_t sizeOf{ uint16_t(sizeof(AssertEntry)) };
Type type;
int32_t line;
const char* condition;
const char* file;
const char* func;
constexpr AssertEntry(Type type, const char* condition, const char* func, const char* file, int32_t line) noexcept
: type(type), line(line), condition(condition), file(file), func(func)
{
}
};
using VectoredAssertionHandler = Result (*)(const AssertEntry& entry, const char* extraInfo, void* userData);
enum class [[nodiscard]] Handle : uintptr_t{
kInvalid = 0
};
using AssertFlags = uint32_t;
constexpr AssertFlags fAssertSkipDialog = 0x00000001;
constexpr AssertFlags fAssertSkipBreakpoint = 0x00000002;
constexpr AssertFlags fAssertNoConsole = 0x00000004;
constexpr AssertFlags fAssertNoMetadata = 0x00000008;
struct IAssert
{
CARB_PLUGIN_INTERFACE_EX("carb::assert::IAssert", carb_assert_IAssert_latest, carb_assert_IAssert)
AssertFlags(CARB_ABI* setAssertionFlags)(AssertFlags set, AssertFlags clear);
uint64_t(CARB_ABI* getAssertionFailureCount)();
bool(CARB_ABI* reportFailedAssertionV)(
const char* condition, const char* file, const char* func, int32_t line, const char* fmt, ...);
#if CARB_VERSION_ATLEAST(carb_assert_IAssert, 1, 1)
bool(CARB_ABI* reportFailedAssertion2)(const AssertEntry& entry, const char* fmt, std::va_list* ap);
Handle(CARB_ABI* registerAssertionHandler)(bool first, VectoredAssertionHandler handler, void* userData);
bool(CARB_ABI* unregisterAssertionHandler)(Handle h);
#endif
// ^^^ API functions
// vvv Inline helper functions
bool reportFailedAssertion(
const char* condition, const char* file, const char* func, int32_t line, const char* fmt = nullptr, ...)
{
std::va_list args;
bool result;
va_start(args, fmt);
result = reportFailedAssertionV(condition, file, func, line, fmt, &args);
va_end(args);
return result;
}
};
namespace detail
{
inline CARB_NOINLINE bool dispatchAssert(const ::carb::assert::AssertEntry& entry, const char* fmt = nullptr, ...) noexcept
{
std::va_list ap;
va_start(ap, fmt);
bool breakpoint = true;
if (g_carbAssert)
{
#if CARB_VERSION_ATLEAST(carb_assert_IAssert, 1, 1)
breakpoint = g_carbAssert->reportFailedAssertion2(entry, fmt, &ap);
#else
breakpoint = g_carbAssert->reportFailedAssertionV(entry.condition, entry.file, entry.func, entry.line, fmt, &ap);
#endif
}
else
{
char* buf = nullptr;
if (fmt)
{
// measure string
std::va_list ap2;
va_copy(ap2, ap);
auto icount = ptrdiff_t(std::vsnprintf(nullptr, 0, fmt, ap2));
va_end(ap2);
if (icount >= 0)
{
size_t count = size_t(icount);
buf = CARB_STACK_ALLOC(char, count + 1);
if (std::vsnprintf(buf, count + 1, fmt, ap) < 0)
buf = nullptr; // an error occurred, so ignore the buffer
}
}
fprintf(stderr, "%s:%" PRId32 " (%s):: Assertion (%s) failed%c %s\n", entry.file, entry.line, entry.func,
entry.condition, buf ? ':' : '.', buf ? buf : "");
}
va_end(ap);
return breakpoint;
}
} // namespace detail
} // namespace assert
} // namespace carb