1// Copyright (C) 2022 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3
4#ifndef QASSERT_H
5#define QASSERT_H
6
7#include <QtCore/qcompilerdetection.h>
8#include <QtCore/qtconfigmacros.h>
9#include <QtCore/qtcoreexports.h>
10#include <QtCore/qtnoop.h>
11
12#if 0
13#pragma qt_class(QtAssert)
14#pragma qt_sync_stop_processing
15#endif
16
17QT_BEGIN_NAMESPACE
18
19#if defined(__cplusplus)
20
21#if !defined(Q_CC_MSVC_ONLY)
22Q_NORETURN
23#endif
24Q_DECL_COLD_FUNCTION
25Q_CORE_EXPORT void qt_assert(const char *assertion, const char *file, int line) noexcept;
26
27#if !defined(Q_ASSERT)
28# if defined(QT_NO_DEBUG) && !defined(QT_FORCE_ASSERTS)
29# define Q_ASSERT(cond) static_cast<void>(false && (cond))
30# else
31# define Q_ASSERT(cond) ((cond) ? static_cast<void>(0) : qt_assert(#cond, __FILE__, __LINE__))
32# endif
33#endif
34
35#if !defined(Q_CC_MSVC_ONLY)
36Q_NORETURN
37#endif
38Q_DECL_COLD_FUNCTION
39Q_CORE_EXPORT
40void qt_assert_x(const char *where, const char *what, const char *file, int line) noexcept;
41
42#if !defined(Q_ASSERT_X)
43# if defined(QT_NO_DEBUG) && !defined(QT_FORCE_ASSERTS)
44# define Q_ASSERT_X(cond, where, what) static_cast<void>(false && (cond))
45# else
46# define Q_ASSERT_X(cond, where, what) ((cond) ? static_cast<void>(0) : qt_assert_x(where, what, __FILE__, __LINE__))
47# endif
48#endif
49
50Q_NORETURN Q_CORE_EXPORT void qt_check_pointer(const char *, int) noexcept;
51Q_NORETURN Q_DECL_COLD_FUNCTION
52Q_CORE_EXPORT void qBadAlloc();
53
54#ifdef QT_NO_EXCEPTIONS
55# if defined(QT_NO_DEBUG) && !defined(QT_FORCE_ASSERTS)
56# define Q_CHECK_PTR(p) qt_noop()
57# else
58# define Q_CHECK_PTR(p) do {if (!(p)) qt_check_pointer(__FILE__,__LINE__);} while (false)
59# endif
60#else
61# define Q_CHECK_PTR(p) do { if (!(p)) qBadAlloc(); } while (false)
62#endif
63
64template <typename T>
65inline T *q_check_ptr(T *p) { Q_CHECK_PTR(p); return p; }
66
67// Q_UNREACHABLE_IMPL() and Q_ASSUME_IMPL() used below are defined in qcompilerdetection.h
68#define Q_UNREACHABLE() \
69 do {\
70 Q_ASSERT_X(false, "Q_UNREACHABLE()", "Q_UNREACHABLE was reached");\
71 Q_UNREACHABLE_IMPL();\
72 } while (false)
73
74#ifndef Q_UNREACHABLE_RETURN
75# ifdef Q_COMPILER_COMPLAINS_ABOUT_RETURN_AFTER_UNREACHABLE
76# define Q_UNREACHABLE_RETURN(...) Q_UNREACHABLE()
77# else
78# define Q_UNREACHABLE_RETURN(...) do { Q_UNREACHABLE(); return __VA_ARGS__; } while (0)
79# endif
80#endif
81
82Q_DECL_DEPRECATED_X("Q_ASSUME() is deprecated because it can produce worse code than when it's absent; "
83 "use C++23 [[assume]] instead")
84inline bool qt_assume_is_deprecated(bool cond) noexcept { return cond; }
85#define Q_ASSUME(Expr) \
86 [] (bool valueOfExpression) {\
87 Q_ASSERT_X(valueOfExpression, "Q_ASSUME()", "Assumption in Q_ASSUME(\"" #Expr "\") was not correct");\
88 Q_ASSUME_IMPL(valueOfExpression);\
89 }(qt_assume_is_deprecated(Expr))
90
91// Don't use these in C++ mode, use static_assert directly.
92// These are here only to keep old code compiling.
93# define Q_STATIC_ASSERT(Condition) static_assert(bool(Condition), #Condition)
94# define Q_STATIC_ASSERT_X(Condition, Message) static_assert(bool(Condition), Message)
95
96#elif defined(Q_COMPILER_STATIC_ASSERT)
97// C11 mode - using the _S version in case <assert.h> doesn't do the right thing
98# define Q_STATIC_ASSERT(Condition) _Static_assert(!!(Condition), #Condition)
99# define Q_STATIC_ASSERT_X(Condition, Message) _Static_assert(!!(Condition), Message)
100#else
101// C89 & C99 version
102# define Q_STATIC_ASSERT_PRIVATE_JOIN(A, B) Q_STATIC_ASSERT_PRIVATE_JOIN_IMPL(A, B)
103# define Q_STATIC_ASSERT_PRIVATE_JOIN_IMPL(A, B) A ## B
104# ifdef __COUNTER__
105# define Q_STATIC_ASSERT(Condition) \
106 typedef char Q_STATIC_ASSERT_PRIVATE_JOIN(q_static_assert_result, __COUNTER__) [(Condition) ? 1 : -1];
107# else
108# define Q_STATIC_ASSERT(Condition) \
109 typedef char Q_STATIC_ASSERT_PRIVATE_JOIN(q_static_assert_result, __LINE__) [(Condition) ? 1 : -1];
110# endif /* __COUNTER__ */
111# define Q_STATIC_ASSERT_X(Condition, Message) Q_STATIC_ASSERT(Condition)
112#endif // __cplusplus
113
114QT_END_NAMESPACE
115
116#endif // QASSERT_H
117