xref: /aosp_15_r20/external/armnn/third-party/doctest/doctest.h (revision 89c4ff92f2867872bb9e2354d150bf0c8c502810)
1*89c4ff92SAndroid Build Coastguard Worker // ====================================================================== lgtm [cpp/missing-header-guard]
2*89c4ff92SAndroid Build Coastguard Worker // == DO NOT MODIFY THIS FILE BY HAND - IT IS AUTO GENERATED BY CMAKE! ==
3*89c4ff92SAndroid Build Coastguard Worker // ======================================================================
4*89c4ff92SAndroid Build Coastguard Worker //
5*89c4ff92SAndroid Build Coastguard Worker // doctest.h - the lightest feature-rich C++ single-header testing framework for unit tests and TDD
6*89c4ff92SAndroid Build Coastguard Worker //
7*89c4ff92SAndroid Build Coastguard Worker // Copyright (c) 2016-2021 Viktor Kirilov
8*89c4ff92SAndroid Build Coastguard Worker //
9*89c4ff92SAndroid Build Coastguard Worker // SPDX-License-Identifier: MIT
10*89c4ff92SAndroid Build Coastguard Worker //
11*89c4ff92SAndroid Build Coastguard Worker // Distributed under the MIT Software License
12*89c4ff92SAndroid Build Coastguard Worker // See accompanying file LICENSE.txt or copy at
13*89c4ff92SAndroid Build Coastguard Worker // https://opensource.org/licenses/MIT
14*89c4ff92SAndroid Build Coastguard Worker //
15*89c4ff92SAndroid Build Coastguard Worker // The documentation can be found at the library's page:
16*89c4ff92SAndroid Build Coastguard Worker // https://github.com/onqtam/doctest/blob/master/doc/markdown/readme.md
17*89c4ff92SAndroid Build Coastguard Worker //
18*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
19*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
20*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
21*89c4ff92SAndroid Build Coastguard Worker //
22*89c4ff92SAndroid Build Coastguard Worker // The library is heavily influenced by Catch - https://github.com/catchorg/Catch2
23*89c4ff92SAndroid Build Coastguard Worker // which uses the Boost Software License - Version 1.0
24*89c4ff92SAndroid Build Coastguard Worker // see here - https://github.com/catchorg/Catch2/blob/master/LICENSE.txt
25*89c4ff92SAndroid Build Coastguard Worker //
26*89c4ff92SAndroid Build Coastguard Worker // The concept of subcases (sections in Catch) and expression decomposition are from there.
27*89c4ff92SAndroid Build Coastguard Worker // Some parts of the code are taken directly:
28*89c4ff92SAndroid Build Coastguard Worker // - stringification - the detection of "ostream& operator<<(ostream&, const T&)" and StringMaker<>
29*89c4ff92SAndroid Build Coastguard Worker // - the Approx() helper class for floating point comparison
30*89c4ff92SAndroid Build Coastguard Worker // - colors in the console
31*89c4ff92SAndroid Build Coastguard Worker // - breaking into a debugger
32*89c4ff92SAndroid Build Coastguard Worker // - signal / SEH handling
33*89c4ff92SAndroid Build Coastguard Worker // - timer
34*89c4ff92SAndroid Build Coastguard Worker // - XmlWriter class - thanks to Phil Nash for allowing the direct reuse (AKA copy/paste)
35*89c4ff92SAndroid Build Coastguard Worker //
36*89c4ff92SAndroid Build Coastguard Worker // The expression decomposing templates are taken from lest - https://github.com/martinmoene/lest
37*89c4ff92SAndroid Build Coastguard Worker // which uses the Boost Software License - Version 1.0
38*89c4ff92SAndroid Build Coastguard Worker // see here - https://github.com/martinmoene/lest/blob/master/LICENSE.txt
39*89c4ff92SAndroid Build Coastguard Worker //
40*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
41*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
42*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
43*89c4ff92SAndroid Build Coastguard Worker 
44*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_LIBRARY_INCLUDED
45*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_LIBRARY_INCLUDED
46*89c4ff92SAndroid Build Coastguard Worker 
47*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
48*89c4ff92SAndroid Build Coastguard Worker // == VERSION ======================================================================================
49*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
50*89c4ff92SAndroid Build Coastguard Worker 
51*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_VERSION_MAJOR 2
52*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_VERSION_MINOR 4
53*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_VERSION_PATCH 6
54*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_VERSION_STR "2.4.6"
55*89c4ff92SAndroid Build Coastguard Worker 
56*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_VERSION                                                                            \
57*89c4ff92SAndroid Build Coastguard Worker     (DOCTEST_VERSION_MAJOR * 10000 + DOCTEST_VERSION_MINOR * 100 + DOCTEST_VERSION_PATCH)
58*89c4ff92SAndroid Build Coastguard Worker 
59*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
60*89c4ff92SAndroid Build Coastguard Worker // == COMPILER VERSION =============================================================================
61*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
62*89c4ff92SAndroid Build Coastguard Worker 
63*89c4ff92SAndroid Build Coastguard Worker // ideas for the version stuff are taken from here: https://github.com/cxxstuff/cxx_detect
64*89c4ff92SAndroid Build Coastguard Worker 
65*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_COMPILER(MAJOR, MINOR, PATCH) ((MAJOR)*10000000 + (MINOR)*100000 + (PATCH))
66*89c4ff92SAndroid Build Coastguard Worker 
67*89c4ff92SAndroid Build Coastguard Worker // GCC/Clang and GCC/MSVC are mutually exclusive, but Clang/MSVC are not because of clang-cl...
68*89c4ff92SAndroid Build Coastguard Worker #if defined(_MSC_VER) && defined(_MSC_FULL_VER)
69*89c4ff92SAndroid Build Coastguard Worker #if _MSC_VER == _MSC_FULL_VER / 10000
70*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC DOCTEST_COMPILER(_MSC_VER / 100, _MSC_VER % 100, _MSC_FULL_VER % 10000)
71*89c4ff92SAndroid Build Coastguard Worker #else // MSVC
72*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC                                                                               \
73*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_COMPILER(_MSC_VER / 100, (_MSC_FULL_VER / 100000) % 100, _MSC_FULL_VER % 100000)
74*89c4ff92SAndroid Build Coastguard Worker #endif // MSVC
75*89c4ff92SAndroid Build Coastguard Worker #endif // MSVC
76*89c4ff92SAndroid Build Coastguard Worker #if defined(__clang__) && defined(__clang_minor__)
77*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CLANG DOCTEST_COMPILER(__clang_major__, __clang_minor__, __clang_patchlevel__)
78*89c4ff92SAndroid Build Coastguard Worker #elif defined(__GNUC__) && defined(__GNUC_MINOR__) && defined(__GNUC_PATCHLEVEL__) &&              \
79*89c4ff92SAndroid Build Coastguard Worker         !defined(__INTEL_COMPILER)
80*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GCC DOCTEST_COMPILER(__GNUC__, __GNUC_MINOR__, __GNUC_PATCHLEVEL__)
81*89c4ff92SAndroid Build Coastguard Worker #endif // GCC
82*89c4ff92SAndroid Build Coastguard Worker 
83*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_MSVC
84*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC 0
85*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_MSVC
86*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CLANG
87*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CLANG 0
88*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CLANG
89*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_GCC
90*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GCC 0
91*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_GCC
92*89c4ff92SAndroid Build Coastguard Worker 
93*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
94*89c4ff92SAndroid Build Coastguard Worker // == COMPILER WARNINGS HELPERS ====================================================================
95*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
96*89c4ff92SAndroid Build Coastguard Worker 
97*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_CLANG
98*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_PRAGMA_TO_STR(x) _Pragma(#x)
99*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CLANG_SUPPRESS_WARNING_PUSH _Pragma("clang diagnostic push")
100*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CLANG_SUPPRESS_WARNING(w) DOCTEST_PRAGMA_TO_STR(clang diagnostic ignored w)
101*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CLANG_SUPPRESS_WARNING_POP _Pragma("clang diagnostic pop")
102*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH(w)                                                \
103*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_PUSH DOCTEST_CLANG_SUPPRESS_WARNING(w)
104*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CLANG
105*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CLANG_SUPPRESS_WARNING_PUSH
106*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CLANG_SUPPRESS_WARNING(w)
107*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CLANG_SUPPRESS_WARNING_POP
108*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH(w)
109*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CLANG
110*89c4ff92SAndroid Build Coastguard Worker 
111*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_GCC
112*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_PRAGMA_TO_STR(x) _Pragma(#x)
113*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GCC_SUPPRESS_WARNING_PUSH _Pragma("GCC diagnostic push")
114*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GCC_SUPPRESS_WARNING(w) DOCTEST_PRAGMA_TO_STR(GCC diagnostic ignored w)
115*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GCC_SUPPRESS_WARNING_POP _Pragma("GCC diagnostic pop")
116*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH(w)                                                  \
117*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING_PUSH DOCTEST_GCC_SUPPRESS_WARNING(w)
118*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_GCC
119*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GCC_SUPPRESS_WARNING_PUSH
120*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GCC_SUPPRESS_WARNING(w)
121*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GCC_SUPPRESS_WARNING_POP
122*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH(w)
123*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_GCC
124*89c4ff92SAndroid Build Coastguard Worker 
125*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_MSVC
126*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC_SUPPRESS_WARNING_PUSH __pragma(warning(push))
127*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC_SUPPRESS_WARNING(w) __pragma(warning(disable : w))
128*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC_SUPPRESS_WARNING_POP __pragma(warning(pop))
129*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(w)                                                 \
130*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_PUSH DOCTEST_MSVC_SUPPRESS_WARNING(w)
131*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_MSVC
132*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC_SUPPRESS_WARNING_PUSH
133*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC_SUPPRESS_WARNING(w)
134*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC_SUPPRESS_WARNING_POP
135*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(w)
136*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_MSVC
137*89c4ff92SAndroid Build Coastguard Worker 
138*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
139*89c4ff92SAndroid Build Coastguard Worker // == COMPILER WARNINGS ============================================================================
140*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
141*89c4ff92SAndroid Build Coastguard Worker 
142*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING_PUSH
143*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wunknown-pragmas")
144*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wnon-virtual-dtor")
145*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wweak-vtables")
146*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wpadded")
147*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wdeprecated")
148*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-prototypes")
149*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wunused-local-typedef")
150*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wc++98-compat")
151*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wc++98-compat-pedantic")
152*89c4ff92SAndroid Build Coastguard Worker 
153*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING_PUSH
154*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wunknown-pragmas")
155*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wpragmas")
156*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Weffc++")
157*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wstrict-overflow")
158*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wstrict-aliasing")
159*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wctor-dtor-privacy")
160*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wmissing-declarations")
161*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wnon-virtual-dtor")
162*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wunused-local-typedefs")
163*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wuseless-cast")
164*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wnoexcept")
165*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wsign-promo")
166*89c4ff92SAndroid Build Coastguard Worker 
167*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_PUSH
168*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4616) // invalid compiler warning
169*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4619) // invalid compiler warning
170*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4996) // The compiler encountered a deprecated declaration
171*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4706) // assignment within conditional expression
172*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4512) // 'class' : assignment operator could not be generated
173*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4127) // conditional expression is constant
174*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4820) // padding
175*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4625) // copy constructor was implicitly defined as deleted
176*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4626) // assignment operator was implicitly defined as deleted
177*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(5027) // move assignment operator was implicitly defined as deleted
178*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(5026) // move constructor was implicitly defined as deleted
179*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4623) // default constructor was implicitly defined as deleted
180*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4640) // construction of local static object is not thread-safe
181*89c4ff92SAndroid Build Coastguard Worker // static analysis
182*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(26439) // This kind of function may not throw. Declare it 'noexcept'
183*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(26495) // Always initialize a member variable
184*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(26451) // Arithmetic overflow ...
185*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(26444) // Avoid unnamed objects with custom construction and dtr...
186*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(26812) // Prefer 'enum class' over 'enum'
187*89c4ff92SAndroid Build Coastguard Worker 
188*89c4ff92SAndroid Build Coastguard Worker // 4548 - expression before comma has no effect; expected expression with side - effect
189*89c4ff92SAndroid Build Coastguard Worker // 4265 - class has virtual functions, but destructor is not virtual
190*89c4ff92SAndroid Build Coastguard Worker // 4986 - exception specification does not match previous declaration
191*89c4ff92SAndroid Build Coastguard Worker // 4350 - behavior change: 'member1' called instead of 'member2'
192*89c4ff92SAndroid Build Coastguard Worker // 4668 - 'x' is not defined as a preprocessor macro, replacing with '0' for '#if/#elif'
193*89c4ff92SAndroid Build Coastguard Worker // 4365 - conversion from 'int' to 'unsigned long', signed/unsigned mismatch
194*89c4ff92SAndroid Build Coastguard Worker // 4774 - format string expected in argument 'x' is not a string literal
195*89c4ff92SAndroid Build Coastguard Worker // 4820 - padding in structs
196*89c4ff92SAndroid Build Coastguard Worker 
197*89c4ff92SAndroid Build Coastguard Worker // only 4 should be disabled globally:
198*89c4ff92SAndroid Build Coastguard Worker // - 4514 # unreferenced inline function has been removed
199*89c4ff92SAndroid Build Coastguard Worker // - 4571 # SEH related
200*89c4ff92SAndroid Build Coastguard Worker // - 4710 # function not inlined
201*89c4ff92SAndroid Build Coastguard Worker // - 4711 # function 'x' selected for automatic inline expansion
202*89c4ff92SAndroid Build Coastguard Worker 
203*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MAKE_STD_HEADERS_CLEAN_FROM_WARNINGS_ON_WALL_BEGIN                                 \
204*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_PUSH                                                             \
205*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4548)                                                            \
206*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4265)                                                            \
207*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4986)                                                            \
208*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4350)                                                            \
209*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4668)                                                            \
210*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4365)                                                            \
211*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4774)                                                            \
212*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4820)                                                            \
213*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4625)                                                            \
214*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4626)                                                            \
215*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(5027)                                                            \
216*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(5026)                                                            \
217*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4623)                                                            \
218*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(5039)                                                            \
219*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(5045)                                                            \
220*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(5105)
221*89c4ff92SAndroid Build Coastguard Worker 
222*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MAKE_STD_HEADERS_CLEAN_FROM_WARNINGS_ON_WALL_END DOCTEST_MSVC_SUPPRESS_WARNING_POP
223*89c4ff92SAndroid Build Coastguard Worker 
224*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
225*89c4ff92SAndroid Build Coastguard Worker // == FEATURE DETECTION ============================================================================
226*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
227*89c4ff92SAndroid Build Coastguard Worker 
228*89c4ff92SAndroid Build Coastguard Worker // general compiler feature support table: https://en.cppreference.com/w/cpp/compiler_support
229*89c4ff92SAndroid Build Coastguard Worker // MSVC C++11 feature support table: https://msdn.microsoft.com/en-us/library/hh567368.aspx
230*89c4ff92SAndroid Build Coastguard Worker // GCC C++11 feature support table: https://gcc.gnu.org/projects/cxx-status.html
231*89c4ff92SAndroid Build Coastguard Worker // MSVC version table:
232*89c4ff92SAndroid Build Coastguard Worker // https://en.wikipedia.org/wiki/Microsoft_Visual_C%2B%2B#Internal_version_numbering
233*89c4ff92SAndroid Build Coastguard Worker // MSVC++ 14.2 (16) _MSC_VER == 1920 (Visual Studio 2019)
234*89c4ff92SAndroid Build Coastguard Worker // MSVC++ 14.1 (15) _MSC_VER == 1910 (Visual Studio 2017)
235*89c4ff92SAndroid Build Coastguard Worker // MSVC++ 14.0      _MSC_VER == 1900 (Visual Studio 2015)
236*89c4ff92SAndroid Build Coastguard Worker // MSVC++ 12.0      _MSC_VER == 1800 (Visual Studio 2013)
237*89c4ff92SAndroid Build Coastguard Worker // MSVC++ 11.0      _MSC_VER == 1700 (Visual Studio 2012)
238*89c4ff92SAndroid Build Coastguard Worker // MSVC++ 10.0      _MSC_VER == 1600 (Visual Studio 2010)
239*89c4ff92SAndroid Build Coastguard Worker // MSVC++ 9.0       _MSC_VER == 1500 (Visual Studio 2008)
240*89c4ff92SAndroid Build Coastguard Worker // MSVC++ 8.0       _MSC_VER == 1400 (Visual Studio 2005)
241*89c4ff92SAndroid Build Coastguard Worker 
242*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_MSVC && !defined(DOCTEST_CONFIG_WINDOWS_SEH)
243*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_WINDOWS_SEH
244*89c4ff92SAndroid Build Coastguard Worker #endif // MSVC
245*89c4ff92SAndroid Build Coastguard Worker #if defined(DOCTEST_CONFIG_NO_WINDOWS_SEH) && defined(DOCTEST_CONFIG_WINDOWS_SEH)
246*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CONFIG_WINDOWS_SEH
247*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_WINDOWS_SEH
248*89c4ff92SAndroid Build Coastguard Worker 
249*89c4ff92SAndroid Build Coastguard Worker #if !defined(_WIN32) && !defined(__QNX__) && !defined(DOCTEST_CONFIG_POSIX_SIGNALS) &&             \
250*89c4ff92SAndroid Build Coastguard Worker         !defined(__EMSCRIPTEN__)
251*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_POSIX_SIGNALS
252*89c4ff92SAndroid Build Coastguard Worker #endif // _WIN32
253*89c4ff92SAndroid Build Coastguard Worker #if defined(DOCTEST_CONFIG_NO_POSIX_SIGNALS) && defined(DOCTEST_CONFIG_POSIX_SIGNALS)
254*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CONFIG_POSIX_SIGNALS
255*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_POSIX_SIGNALS
256*89c4ff92SAndroid Build Coastguard Worker 
257*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_EXCEPTIONS
258*89c4ff92SAndroid Build Coastguard Worker #if !defined(__cpp_exceptions) && !defined(__EXCEPTIONS) && !defined(_CPPUNWIND)
259*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_NO_EXCEPTIONS
260*89c4ff92SAndroid Build Coastguard Worker #endif // no exceptions
261*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS
262*89c4ff92SAndroid Build Coastguard Worker 
263*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_NO_EXCEPTIONS_BUT_WITH_ALL_ASSERTS
264*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_EXCEPTIONS
265*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_NO_EXCEPTIONS
266*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS
267*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS_BUT_WITH_ALL_ASSERTS
268*89c4ff92SAndroid Build Coastguard Worker 
269*89c4ff92SAndroid Build Coastguard Worker #if defined(DOCTEST_CONFIG_NO_EXCEPTIONS) && !defined(DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS)
270*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS
271*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS && !DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS
272*89c4ff92SAndroid Build Coastguard Worker 
273*89c4ff92SAndroid Build Coastguard Worker #if defined(DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN) && !defined(DOCTEST_CONFIG_IMPLEMENT)
274*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_IMPLEMENT
275*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN
276*89c4ff92SAndroid Build Coastguard Worker 
277*89c4ff92SAndroid Build Coastguard Worker #if defined(_WIN32) || defined(__CYGWIN__)
278*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_MSVC
279*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SYMBOL_EXPORT __declspec(dllexport)
280*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SYMBOL_IMPORT __declspec(dllimport)
281*89c4ff92SAndroid Build Coastguard Worker #else // MSVC
282*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SYMBOL_EXPORT __attribute__((dllexport))
283*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SYMBOL_IMPORT __attribute__((dllimport))
284*89c4ff92SAndroid Build Coastguard Worker #endif // MSVC
285*89c4ff92SAndroid Build Coastguard Worker #else  // _WIN32
286*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SYMBOL_EXPORT __attribute__((visibility("default")))
287*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SYMBOL_IMPORT
288*89c4ff92SAndroid Build Coastguard Worker #endif // _WIN32
289*89c4ff92SAndroid Build Coastguard Worker 
290*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_IMPLEMENTATION_IN_DLL
291*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_IMPLEMENT
292*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_INTERFACE DOCTEST_SYMBOL_EXPORT
293*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_IMPLEMENT
294*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_INTERFACE DOCTEST_SYMBOL_IMPORT
295*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_IMPLEMENT
296*89c4ff92SAndroid Build Coastguard Worker #else  // DOCTEST_CONFIG_IMPLEMENTATION_IN_DLL
297*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_INTERFACE
298*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_IMPLEMENTATION_IN_DLL
299*89c4ff92SAndroid Build Coastguard Worker 
300*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_EMPTY
301*89c4ff92SAndroid Build Coastguard Worker 
302*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_MSVC
303*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_NOINLINE __declspec(noinline)
304*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_UNUSED
305*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ALIGNMENT(x)
306*89c4ff92SAndroid Build Coastguard Worker #elif DOCTEST_CLANG && DOCTEST_CLANG < DOCTEST_COMPILER(3, 5, 0)
307*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_NOINLINE
308*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_UNUSED
309*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ALIGNMENT(x)
310*89c4ff92SAndroid Build Coastguard Worker #else
311*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_NOINLINE __attribute__((noinline))
312*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_UNUSED __attribute__((unused))
313*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ALIGNMENT(x) __attribute__((aligned(x)))
314*89c4ff92SAndroid Build Coastguard Worker #endif
315*89c4ff92SAndroid Build Coastguard Worker 
316*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_NORETURN
317*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_NORETURN [[noreturn]]
318*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_NORETURN
319*89c4ff92SAndroid Build Coastguard Worker 
320*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_NOEXCEPT
321*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_NOEXCEPT noexcept
322*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_NOEXCEPT
323*89c4ff92SAndroid Build Coastguard Worker 
324*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
325*89c4ff92SAndroid Build Coastguard Worker // == FEATURE DETECTION END ========================================================================
326*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
327*89c4ff92SAndroid Build Coastguard Worker 
328*89c4ff92SAndroid Build Coastguard Worker // internal macros for string concatenation and anonymous variable name generation
329*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CAT_IMPL(s1, s2) s1##s2
330*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CAT(s1, s2) DOCTEST_CAT_IMPL(s1, s2)
331*89c4ff92SAndroid Build Coastguard Worker #ifdef __COUNTER__ // not standard and may be missing for some compilers
332*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ANONYMOUS(x) DOCTEST_CAT(x, __COUNTER__)
333*89c4ff92SAndroid Build Coastguard Worker #else // __COUNTER__
334*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ANONYMOUS(x) DOCTEST_CAT(x, __LINE__)
335*89c4ff92SAndroid Build Coastguard Worker #endif // __COUNTER__
336*89c4ff92SAndroid Build Coastguard Worker 
337*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TOSTR(x) #x
338*89c4ff92SAndroid Build Coastguard Worker 
339*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_ASSERTION_PARAMETERS_BY_VALUE
340*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REF_WRAP(x) x&
341*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_ASSERTION_PARAMETERS_BY_VALUE
342*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REF_WRAP(x) x
343*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_ASSERTION_PARAMETERS_BY_VALUE
344*89c4ff92SAndroid Build Coastguard Worker 
345*89c4ff92SAndroid Build Coastguard Worker // not using __APPLE__ because... this is how Catch does it
346*89c4ff92SAndroid Build Coastguard Worker #ifdef __MAC_OS_X_VERSION_MIN_REQUIRED
347*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_PLATFORM_MAC
348*89c4ff92SAndroid Build Coastguard Worker #elif defined(__IPHONE_OS_VERSION_MIN_REQUIRED)
349*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_PLATFORM_IPHONE
350*89c4ff92SAndroid Build Coastguard Worker #elif defined(_WIN32)
351*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_PLATFORM_WINDOWS
352*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_PLATFORM
353*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_PLATFORM_LINUX
354*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM
355*89c4ff92SAndroid Build Coastguard Worker 
356*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GLOBAL_NO_WARNINGS(var)                                                            \
357*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wglobal-constructors")                              \
358*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING("-Wunused-variable")                                            \
359*89c4ff92SAndroid Build Coastguard Worker     static const int var DOCTEST_UNUSED // NOLINT(fuchsia-statically-constructed-objects,cert-err58-cpp)
360*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GLOBAL_NO_WARNINGS_END() DOCTEST_CLANG_SUPPRESS_WARNING_POP
361*89c4ff92SAndroid Build Coastguard Worker 
362*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_BREAK_INTO_DEBUGGER
363*89c4ff92SAndroid Build Coastguard Worker // should probably take a look at https://github.com/scottt/debugbreak
364*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_LINUX
365*89c4ff92SAndroid Build Coastguard Worker #if defined(__GNUC__) && (defined(__i386) || defined(__x86_64))
366*89c4ff92SAndroid Build Coastguard Worker // Break at the location of the failing check if possible
367*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BREAK_INTO_DEBUGGER() __asm__("int $3\n" : :) // NOLINT (hicpp-no-assembler)
368*89c4ff92SAndroid Build Coastguard Worker #else
369*89c4ff92SAndroid Build Coastguard Worker #include <signal.h>
370*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BREAK_INTO_DEBUGGER() raise(SIGTRAP)
371*89c4ff92SAndroid Build Coastguard Worker #endif
372*89c4ff92SAndroid Build Coastguard Worker #elif defined(DOCTEST_PLATFORM_MAC)
373*89c4ff92SAndroid Build Coastguard Worker #if defined(__x86_64) || defined(__x86_64__) || defined(__amd64__) || defined(__i386)
374*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BREAK_INTO_DEBUGGER() __asm__("int $3\n" : :) // NOLINT (hicpp-no-assembler)
375*89c4ff92SAndroid Build Coastguard Worker #else
376*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BREAK_INTO_DEBUGGER() __asm__("brk #0"); // NOLINT (hicpp-no-assembler)
377*89c4ff92SAndroid Build Coastguard Worker #endif
378*89c4ff92SAndroid Build Coastguard Worker #elif DOCTEST_MSVC
379*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BREAK_INTO_DEBUGGER() __debugbreak()
380*89c4ff92SAndroid Build Coastguard Worker #elif defined(__MINGW32__)
381*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wredundant-decls")
382*89c4ff92SAndroid Build Coastguard Worker extern "C" __declspec(dllimport) void __stdcall DebugBreak();
383*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING_POP
384*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BREAK_INTO_DEBUGGER() ::DebugBreak()
385*89c4ff92SAndroid Build Coastguard Worker #else // linux
386*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BREAK_INTO_DEBUGGER() (static_cast<void>(0))
387*89c4ff92SAndroid Build Coastguard Worker #endif // linux
388*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_BREAK_INTO_DEBUGGER
389*89c4ff92SAndroid Build Coastguard Worker 
390*89c4ff92SAndroid Build Coastguard Worker // this is kept here for backwards compatibility since the config option was changed
391*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_USE_IOSFWD
392*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_USE_STD_HEADERS
393*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_USE_IOSFWD
394*89c4ff92SAndroid Build Coastguard Worker 
395*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_USE_STD_HEADERS
396*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
397*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
398*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
399*89c4ff92SAndroid Build Coastguard Worker #include <iosfwd>
400*89c4ff92SAndroid Build Coastguard Worker #include <cstddef>
401*89c4ff92SAndroid Build Coastguard Worker #include <ostream>
402*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_USE_STD_HEADERS
403*89c4ff92SAndroid Build Coastguard Worker 
404*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_CLANG
405*89c4ff92SAndroid Build Coastguard Worker // to detect if libc++ is being used with clang (the _LIBCPP_VERSION identifier)
406*89c4ff92SAndroid Build Coastguard Worker #include <ciso646>
407*89c4ff92SAndroid Build Coastguard Worker #endif // clang
408*89c4ff92SAndroid Build Coastguard Worker 
409*89c4ff92SAndroid Build Coastguard Worker #ifdef _LIBCPP_VERSION
410*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_STD_NAMESPACE_BEGIN _LIBCPP_BEGIN_NAMESPACE_STD
411*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_STD_NAMESPACE_END _LIBCPP_END_NAMESPACE_STD
412*89c4ff92SAndroid Build Coastguard Worker #else // _LIBCPP_VERSION
413*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_STD_NAMESPACE_BEGIN namespace std {
414*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_STD_NAMESPACE_END }
415*89c4ff92SAndroid Build Coastguard Worker #endif // _LIBCPP_VERSION
416*89c4ff92SAndroid Build Coastguard Worker 
417*89c4ff92SAndroid Build Coastguard Worker // Forward declaring 'X' in namespace std is not permitted by the C++ Standard.
418*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4643)
419*89c4ff92SAndroid Build Coastguard Worker 
420*89c4ff92SAndroid Build Coastguard Worker DOCTEST_STD_NAMESPACE_BEGIN // NOLINT (cert-dcl58-cpp)
421*89c4ff92SAndroid Build Coastguard Worker typedef decltype(nullptr) nullptr_t;
422*89c4ff92SAndroid Build Coastguard Worker template <class charT>
423*89c4ff92SAndroid Build Coastguard Worker struct char_traits;
424*89c4ff92SAndroid Build Coastguard Worker template <>
425*89c4ff92SAndroid Build Coastguard Worker struct char_traits<char>;
426*89c4ff92SAndroid Build Coastguard Worker template <class charT, class traits>
427*89c4ff92SAndroid Build Coastguard Worker class basic_ostream;
428*89c4ff92SAndroid Build Coastguard Worker typedef basic_ostream<char, char_traits<char>> ostream;
429*89c4ff92SAndroid Build Coastguard Worker template <class... Types>
430*89c4ff92SAndroid Build Coastguard Worker class tuple;
431*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_MSVC >= DOCTEST_COMPILER(19, 20, 0)
432*89c4ff92SAndroid Build Coastguard Worker // see this issue on why this is needed: https://github.com/onqtam/doctest/issues/183
433*89c4ff92SAndroid Build Coastguard Worker template <class _Ty>
434*89c4ff92SAndroid Build Coastguard Worker class allocator;
435*89c4ff92SAndroid Build Coastguard Worker template <class _Elem, class _Traits, class _Alloc>
436*89c4ff92SAndroid Build Coastguard Worker class basic_string;
437*89c4ff92SAndroid Build Coastguard Worker using string = basic_string<char, char_traits<char>, allocator<char>>;
438*89c4ff92SAndroid Build Coastguard Worker #endif // VS 2019
439*89c4ff92SAndroid Build Coastguard Worker DOCTEST_STD_NAMESPACE_END
440*89c4ff92SAndroid Build Coastguard Worker 
441*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_POP
442*89c4ff92SAndroid Build Coastguard Worker 
443*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_USE_STD_HEADERS
444*89c4ff92SAndroid Build Coastguard Worker 
445*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
446*89c4ff92SAndroid Build Coastguard Worker #include <type_traits>
447*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
448*89c4ff92SAndroid Build Coastguard Worker 
449*89c4ff92SAndroid Build Coastguard Worker namespace doctest {
450*89c4ff92SAndroid Build Coastguard Worker 
451*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE extern bool is_running_in_test;
452*89c4ff92SAndroid Build Coastguard Worker 
453*89c4ff92SAndroid Build Coastguard Worker // A 24 byte string class (can be as small as 17 for x64 and 13 for x86) that can hold strings with length
454*89c4ff92SAndroid Build Coastguard Worker // of up to 23 chars on the stack before going on the heap - the last byte of the buffer is used for:
455*89c4ff92SAndroid Build Coastguard Worker // - "is small" bit - the highest bit - if "0" then it is small - otherwise its "1" (128)
456*89c4ff92SAndroid Build Coastguard Worker // - if small - capacity left before going on the heap - using the lowest 5 bits
457*89c4ff92SAndroid Build Coastguard Worker // - if small - 2 bits are left unused - the second and third highest ones
458*89c4ff92SAndroid Build Coastguard Worker // - if small - acts as a null terminator if strlen() is 23 (24 including the null terminator)
459*89c4ff92SAndroid Build Coastguard Worker //              and the "is small" bit remains "0" ("as well as the capacity left") so its OK
460*89c4ff92SAndroid Build Coastguard Worker // Idea taken from this lecture about the string implementation of facebook/folly - fbstring
461*89c4ff92SAndroid Build Coastguard Worker // https://www.youtube.com/watch?v=kPR8h4-qZdk
462*89c4ff92SAndroid Build Coastguard Worker // TODO:
463*89c4ff92SAndroid Build Coastguard Worker // - optimizations - like not deleting memory unnecessarily in operator= and etc.
464*89c4ff92SAndroid Build Coastguard Worker // - resize/reserve/clear
465*89c4ff92SAndroid Build Coastguard Worker // - substr
466*89c4ff92SAndroid Build Coastguard Worker // - replace
467*89c4ff92SAndroid Build Coastguard Worker // - back/front
468*89c4ff92SAndroid Build Coastguard Worker // - iterator stuff
469*89c4ff92SAndroid Build Coastguard Worker // - find & friends
470*89c4ff92SAndroid Build Coastguard Worker // - push_back/pop_back
471*89c4ff92SAndroid Build Coastguard Worker // - assign/insert/erase
472*89c4ff92SAndroid Build Coastguard Worker // - relational operators as free functions - taking const char* as one of the params
473*89c4ff92SAndroid Build Coastguard Worker class DOCTEST_INTERFACE String
474*89c4ff92SAndroid Build Coastguard Worker {
475*89c4ff92SAndroid Build Coastguard Worker     static const unsigned len  = 24;      //!OCLINT avoid private static members
476*89c4ff92SAndroid Build Coastguard Worker     static const unsigned last = len - 1; //!OCLINT avoid private static members
477*89c4ff92SAndroid Build Coastguard Worker 
478*89c4ff92SAndroid Build Coastguard Worker     struct view // len should be more than sizeof(view) - because of the final byte for flags
479*89c4ff92SAndroid Build Coastguard Worker     {
480*89c4ff92SAndroid Build Coastguard Worker         char*    ptr;
481*89c4ff92SAndroid Build Coastguard Worker         unsigned size;
482*89c4ff92SAndroid Build Coastguard Worker         unsigned capacity;
483*89c4ff92SAndroid Build Coastguard Worker     };
484*89c4ff92SAndroid Build Coastguard Worker 
485*89c4ff92SAndroid Build Coastguard Worker     union
486*89c4ff92SAndroid Build Coastguard Worker     {
487*89c4ff92SAndroid Build Coastguard Worker         char buf[len];
488*89c4ff92SAndroid Build Coastguard Worker         view data;
489*89c4ff92SAndroid Build Coastguard Worker     };
490*89c4ff92SAndroid Build Coastguard Worker 
isOnStack()491*89c4ff92SAndroid Build Coastguard Worker     bool isOnStack() const { return (buf[last] & 128) == 0; }
492*89c4ff92SAndroid Build Coastguard Worker     void setOnHeap();
493*89c4ff92SAndroid Build Coastguard Worker     void setLast(unsigned in = last);
494*89c4ff92SAndroid Build Coastguard Worker 
495*89c4ff92SAndroid Build Coastguard Worker     void copy(const String& other);
496*89c4ff92SAndroid Build Coastguard Worker 
497*89c4ff92SAndroid Build Coastguard Worker public:
498*89c4ff92SAndroid Build Coastguard Worker     String();
499*89c4ff92SAndroid Build Coastguard Worker     ~String();
500*89c4ff92SAndroid Build Coastguard Worker 
501*89c4ff92SAndroid Build Coastguard Worker     // cppcheck-suppress noExplicitConstructor
502*89c4ff92SAndroid Build Coastguard Worker     String(const char* in);
503*89c4ff92SAndroid Build Coastguard Worker     String(const char* in, unsigned in_size);
504*89c4ff92SAndroid Build Coastguard Worker 
505*89c4ff92SAndroid Build Coastguard Worker     String(const String& other);
506*89c4ff92SAndroid Build Coastguard Worker     String& operator=(const String& other);
507*89c4ff92SAndroid Build Coastguard Worker 
508*89c4ff92SAndroid Build Coastguard Worker     String& operator+=(const String& other);
509*89c4ff92SAndroid Build Coastguard Worker     String  operator+(const String& other) const;
510*89c4ff92SAndroid Build Coastguard Worker 
511*89c4ff92SAndroid Build Coastguard Worker     String(String&& other);
512*89c4ff92SAndroid Build Coastguard Worker     String& operator=(String&& other);
513*89c4ff92SAndroid Build Coastguard Worker 
514*89c4ff92SAndroid Build Coastguard Worker     char  operator[](unsigned i) const;
515*89c4ff92SAndroid Build Coastguard Worker     char& operator[](unsigned i);
516*89c4ff92SAndroid Build Coastguard Worker 
517*89c4ff92SAndroid Build Coastguard Worker     // the only functions I'm willing to leave in the interface - available for inlining
c_str()518*89c4ff92SAndroid Build Coastguard Worker     const char* c_str() const { return const_cast<String*>(this)->c_str(); } // NOLINT
c_str()519*89c4ff92SAndroid Build Coastguard Worker     char*       c_str() {
520*89c4ff92SAndroid Build Coastguard Worker         if(isOnStack())
521*89c4ff92SAndroid Build Coastguard Worker             return reinterpret_cast<char*>(buf);
522*89c4ff92SAndroid Build Coastguard Worker         return data.ptr;
523*89c4ff92SAndroid Build Coastguard Worker     }
524*89c4ff92SAndroid Build Coastguard Worker 
525*89c4ff92SAndroid Build Coastguard Worker     unsigned size() const;
526*89c4ff92SAndroid Build Coastguard Worker     unsigned capacity() const;
527*89c4ff92SAndroid Build Coastguard Worker 
528*89c4ff92SAndroid Build Coastguard Worker     int compare(const char* other, bool no_case = false) const;
529*89c4ff92SAndroid Build Coastguard Worker     int compare(const String& other, bool no_case = false) const;
530*89c4ff92SAndroid Build Coastguard Worker };
531*89c4ff92SAndroid Build Coastguard Worker 
532*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE bool operator==(const String& lhs, const String& rhs);
533*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE bool operator!=(const String& lhs, const String& rhs);
534*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE bool operator<(const String& lhs, const String& rhs);
535*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE bool operator>(const String& lhs, const String& rhs);
536*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE bool operator<=(const String& lhs, const String& rhs);
537*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE bool operator>=(const String& lhs, const String& rhs);
538*89c4ff92SAndroid Build Coastguard Worker 
539*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE std::ostream& operator<<(std::ostream& s, const String& in);
540*89c4ff92SAndroid Build Coastguard Worker 
541*89c4ff92SAndroid Build Coastguard Worker namespace Color {
542*89c4ff92SAndroid Build Coastguard Worker     enum Enum
543*89c4ff92SAndroid Build Coastguard Worker     {
544*89c4ff92SAndroid Build Coastguard Worker         None = 0,
545*89c4ff92SAndroid Build Coastguard Worker         White,
546*89c4ff92SAndroid Build Coastguard Worker         Red,
547*89c4ff92SAndroid Build Coastguard Worker         Green,
548*89c4ff92SAndroid Build Coastguard Worker         Blue,
549*89c4ff92SAndroid Build Coastguard Worker         Cyan,
550*89c4ff92SAndroid Build Coastguard Worker         Yellow,
551*89c4ff92SAndroid Build Coastguard Worker         Grey,
552*89c4ff92SAndroid Build Coastguard Worker 
553*89c4ff92SAndroid Build Coastguard Worker         Bright = 0x10,
554*89c4ff92SAndroid Build Coastguard Worker 
555*89c4ff92SAndroid Build Coastguard Worker         BrightRed   = Bright | Red,
556*89c4ff92SAndroid Build Coastguard Worker         BrightGreen = Bright | Green,
557*89c4ff92SAndroid Build Coastguard Worker         LightGrey   = Bright | Grey,
558*89c4ff92SAndroid Build Coastguard Worker         BrightWhite = Bright | White
559*89c4ff92SAndroid Build Coastguard Worker     };
560*89c4ff92SAndroid Build Coastguard Worker 
561*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE std::ostream& operator<<(std::ostream& s, Color::Enum code);
562*89c4ff92SAndroid Build Coastguard Worker } // namespace Color
563*89c4ff92SAndroid Build Coastguard Worker 
564*89c4ff92SAndroid Build Coastguard Worker namespace assertType {
565*89c4ff92SAndroid Build Coastguard Worker     enum Enum
566*89c4ff92SAndroid Build Coastguard Worker     {
567*89c4ff92SAndroid Build Coastguard Worker         // macro traits
568*89c4ff92SAndroid Build Coastguard Worker 
569*89c4ff92SAndroid Build Coastguard Worker         is_warn    = 1,
570*89c4ff92SAndroid Build Coastguard Worker         is_check   = 2 * is_warn,
571*89c4ff92SAndroid Build Coastguard Worker         is_require = 2 * is_check,
572*89c4ff92SAndroid Build Coastguard Worker 
573*89c4ff92SAndroid Build Coastguard Worker         is_normal      = 2 * is_require,
574*89c4ff92SAndroid Build Coastguard Worker         is_throws      = 2 * is_normal,
575*89c4ff92SAndroid Build Coastguard Worker         is_throws_as   = 2 * is_throws,
576*89c4ff92SAndroid Build Coastguard Worker         is_throws_with = 2 * is_throws_as,
577*89c4ff92SAndroid Build Coastguard Worker         is_nothrow     = 2 * is_throws_with,
578*89c4ff92SAndroid Build Coastguard Worker 
579*89c4ff92SAndroid Build Coastguard Worker         is_false = 2 * is_nothrow,
580*89c4ff92SAndroid Build Coastguard Worker         is_unary = 2 * is_false, // not checked anywhere - used just to distinguish the types
581*89c4ff92SAndroid Build Coastguard Worker 
582*89c4ff92SAndroid Build Coastguard Worker         is_eq = 2 * is_unary,
583*89c4ff92SAndroid Build Coastguard Worker         is_ne = 2 * is_eq,
584*89c4ff92SAndroid Build Coastguard Worker 
585*89c4ff92SAndroid Build Coastguard Worker         is_lt = 2 * is_ne,
586*89c4ff92SAndroid Build Coastguard Worker         is_gt = 2 * is_lt,
587*89c4ff92SAndroid Build Coastguard Worker 
588*89c4ff92SAndroid Build Coastguard Worker         is_ge = 2 * is_gt,
589*89c4ff92SAndroid Build Coastguard Worker         is_le = 2 * is_ge,
590*89c4ff92SAndroid Build Coastguard Worker 
591*89c4ff92SAndroid Build Coastguard Worker         // macro types
592*89c4ff92SAndroid Build Coastguard Worker 
593*89c4ff92SAndroid Build Coastguard Worker         DT_WARN    = is_normal | is_warn,
594*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK   = is_normal | is_check,
595*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE = is_normal | is_require,
596*89c4ff92SAndroid Build Coastguard Worker 
597*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_FALSE    = is_normal | is_false | is_warn,
598*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_FALSE   = is_normal | is_false | is_check,
599*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_FALSE = is_normal | is_false | is_require,
600*89c4ff92SAndroid Build Coastguard Worker 
601*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_THROWS    = is_throws | is_warn,
602*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_THROWS   = is_throws | is_check,
603*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_THROWS = is_throws | is_require,
604*89c4ff92SAndroid Build Coastguard Worker 
605*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_THROWS_AS    = is_throws_as | is_warn,
606*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_THROWS_AS   = is_throws_as | is_check,
607*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_THROWS_AS = is_throws_as | is_require,
608*89c4ff92SAndroid Build Coastguard Worker 
609*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_THROWS_WITH    = is_throws_with | is_warn,
610*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_THROWS_WITH   = is_throws_with | is_check,
611*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_THROWS_WITH = is_throws_with | is_require,
612*89c4ff92SAndroid Build Coastguard Worker 
613*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_THROWS_WITH_AS    = is_throws_with | is_throws_as | is_warn,
614*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_THROWS_WITH_AS   = is_throws_with | is_throws_as | is_check,
615*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_THROWS_WITH_AS = is_throws_with | is_throws_as | is_require,
616*89c4ff92SAndroid Build Coastguard Worker 
617*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_NOTHROW    = is_nothrow | is_warn,
618*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_NOTHROW   = is_nothrow | is_check,
619*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_NOTHROW = is_nothrow | is_require,
620*89c4ff92SAndroid Build Coastguard Worker 
621*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_EQ    = is_normal | is_eq | is_warn,
622*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_EQ   = is_normal | is_eq | is_check,
623*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_EQ = is_normal | is_eq | is_require,
624*89c4ff92SAndroid Build Coastguard Worker 
625*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_NE    = is_normal | is_ne | is_warn,
626*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_NE   = is_normal | is_ne | is_check,
627*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_NE = is_normal | is_ne | is_require,
628*89c4ff92SAndroid Build Coastguard Worker 
629*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_GT    = is_normal | is_gt | is_warn,
630*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_GT   = is_normal | is_gt | is_check,
631*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_GT = is_normal | is_gt | is_require,
632*89c4ff92SAndroid Build Coastguard Worker 
633*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_LT    = is_normal | is_lt | is_warn,
634*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_LT   = is_normal | is_lt | is_check,
635*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_LT = is_normal | is_lt | is_require,
636*89c4ff92SAndroid Build Coastguard Worker 
637*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_GE    = is_normal | is_ge | is_warn,
638*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_GE   = is_normal | is_ge | is_check,
639*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_GE = is_normal | is_ge | is_require,
640*89c4ff92SAndroid Build Coastguard Worker 
641*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_LE    = is_normal | is_le | is_warn,
642*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_LE   = is_normal | is_le | is_check,
643*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_LE = is_normal | is_le | is_require,
644*89c4ff92SAndroid Build Coastguard Worker 
645*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_UNARY    = is_normal | is_unary | is_warn,
646*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_UNARY   = is_normal | is_unary | is_check,
647*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_UNARY = is_normal | is_unary | is_require,
648*89c4ff92SAndroid Build Coastguard Worker 
649*89c4ff92SAndroid Build Coastguard Worker         DT_WARN_UNARY_FALSE    = is_normal | is_false | is_unary | is_warn,
650*89c4ff92SAndroid Build Coastguard Worker         DT_CHECK_UNARY_FALSE   = is_normal | is_false | is_unary | is_check,
651*89c4ff92SAndroid Build Coastguard Worker         DT_REQUIRE_UNARY_FALSE = is_normal | is_false | is_unary | is_require,
652*89c4ff92SAndroid Build Coastguard Worker     };
653*89c4ff92SAndroid Build Coastguard Worker } // namespace assertType
654*89c4ff92SAndroid Build Coastguard Worker 
655*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE const char* assertString(assertType::Enum at);
656*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE const char* failureString(assertType::Enum at);
657*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE const char* skipPathFromFilename(const char* file);
658*89c4ff92SAndroid Build Coastguard Worker 
659*89c4ff92SAndroid Build Coastguard Worker struct DOCTEST_INTERFACE TestCaseData
660*89c4ff92SAndroid Build Coastguard Worker {
661*89c4ff92SAndroid Build Coastguard Worker     String      m_file;       // the file in which the test was registered (using String - see #350)
662*89c4ff92SAndroid Build Coastguard Worker     unsigned    m_line;       // the line where the test was registered
663*89c4ff92SAndroid Build Coastguard Worker     const char* m_name;       // name of the test case
664*89c4ff92SAndroid Build Coastguard Worker     const char* m_test_suite; // the test suite in which the test was added
665*89c4ff92SAndroid Build Coastguard Worker     const char* m_description;
666*89c4ff92SAndroid Build Coastguard Worker     bool        m_skip;
667*89c4ff92SAndroid Build Coastguard Worker     bool        m_no_breaks;
668*89c4ff92SAndroid Build Coastguard Worker     bool        m_no_output;
669*89c4ff92SAndroid Build Coastguard Worker     bool        m_may_fail;
670*89c4ff92SAndroid Build Coastguard Worker     bool        m_should_fail;
671*89c4ff92SAndroid Build Coastguard Worker     int         m_expected_failures;
672*89c4ff92SAndroid Build Coastguard Worker     double      m_timeout;
673*89c4ff92SAndroid Build Coastguard Worker };
674*89c4ff92SAndroid Build Coastguard Worker 
675*89c4ff92SAndroid Build Coastguard Worker struct DOCTEST_INTERFACE AssertData
676*89c4ff92SAndroid Build Coastguard Worker {
677*89c4ff92SAndroid Build Coastguard Worker     // common - for all asserts
678*89c4ff92SAndroid Build Coastguard Worker     const TestCaseData* m_test_case;
679*89c4ff92SAndroid Build Coastguard Worker     assertType::Enum    m_at;
680*89c4ff92SAndroid Build Coastguard Worker     const char*         m_file;
681*89c4ff92SAndroid Build Coastguard Worker     int                 m_line;
682*89c4ff92SAndroid Build Coastguard Worker     const char*         m_expr;
683*89c4ff92SAndroid Build Coastguard Worker     bool                m_failed;
684*89c4ff92SAndroid Build Coastguard Worker 
685*89c4ff92SAndroid Build Coastguard Worker     // exception-related - for all asserts
686*89c4ff92SAndroid Build Coastguard Worker     bool   m_threw;
687*89c4ff92SAndroid Build Coastguard Worker     String m_exception;
688*89c4ff92SAndroid Build Coastguard Worker 
689*89c4ff92SAndroid Build Coastguard Worker     // for normal asserts
690*89c4ff92SAndroid Build Coastguard Worker     String m_decomp;
691*89c4ff92SAndroid Build Coastguard Worker 
692*89c4ff92SAndroid Build Coastguard Worker     // for specific exception-related asserts
693*89c4ff92SAndroid Build Coastguard Worker     bool        m_threw_as;
694*89c4ff92SAndroid Build Coastguard Worker     const char* m_exception_type;
695*89c4ff92SAndroid Build Coastguard Worker     const char* m_exception_string;
696*89c4ff92SAndroid Build Coastguard Worker };
697*89c4ff92SAndroid Build Coastguard Worker 
698*89c4ff92SAndroid Build Coastguard Worker struct DOCTEST_INTERFACE MessageData
699*89c4ff92SAndroid Build Coastguard Worker {
700*89c4ff92SAndroid Build Coastguard Worker     String           m_string;
701*89c4ff92SAndroid Build Coastguard Worker     const char*      m_file;
702*89c4ff92SAndroid Build Coastguard Worker     int              m_line;
703*89c4ff92SAndroid Build Coastguard Worker     assertType::Enum m_severity;
704*89c4ff92SAndroid Build Coastguard Worker };
705*89c4ff92SAndroid Build Coastguard Worker 
706*89c4ff92SAndroid Build Coastguard Worker struct DOCTEST_INTERFACE SubcaseSignature
707*89c4ff92SAndroid Build Coastguard Worker {
708*89c4ff92SAndroid Build Coastguard Worker     String      m_name;
709*89c4ff92SAndroid Build Coastguard Worker     const char* m_file;
710*89c4ff92SAndroid Build Coastguard Worker     int         m_line;
711*89c4ff92SAndroid Build Coastguard Worker 
712*89c4ff92SAndroid Build Coastguard Worker     bool operator<(const SubcaseSignature& other) const;
713*89c4ff92SAndroid Build Coastguard Worker };
714*89c4ff92SAndroid Build Coastguard Worker 
715*89c4ff92SAndroid Build Coastguard Worker struct DOCTEST_INTERFACE IContextScope
716*89c4ff92SAndroid Build Coastguard Worker {
717*89c4ff92SAndroid Build Coastguard Worker     IContextScope();
718*89c4ff92SAndroid Build Coastguard Worker     virtual ~IContextScope();
719*89c4ff92SAndroid Build Coastguard Worker     virtual void stringify(std::ostream*) const = 0;
720*89c4ff92SAndroid Build Coastguard Worker };
721*89c4ff92SAndroid Build Coastguard Worker 
722*89c4ff92SAndroid Build Coastguard Worker namespace detail {
723*89c4ff92SAndroid Build Coastguard Worker     struct DOCTEST_INTERFACE TestCase;
724*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
725*89c4ff92SAndroid Build Coastguard Worker 
726*89c4ff92SAndroid Build Coastguard Worker struct ContextOptions //!OCLINT too many fields
727*89c4ff92SAndroid Build Coastguard Worker {
728*89c4ff92SAndroid Build Coastguard Worker     std::ostream* cout;        // stdout stream - std::cout by default
729*89c4ff92SAndroid Build Coastguard Worker     std::ostream* cerr;        // stderr stream - std::cerr by default
730*89c4ff92SAndroid Build Coastguard Worker     String        binary_name; // the test binary name
731*89c4ff92SAndroid Build Coastguard Worker 
732*89c4ff92SAndroid Build Coastguard Worker     const detail::TestCase* currentTest = nullptr;
733*89c4ff92SAndroid Build Coastguard Worker 
734*89c4ff92SAndroid Build Coastguard Worker     // == parameters from the command line
735*89c4ff92SAndroid Build Coastguard Worker     String   out;       // output filename
736*89c4ff92SAndroid Build Coastguard Worker     String   order_by;  // how tests should be ordered
737*89c4ff92SAndroid Build Coastguard Worker     unsigned rand_seed; // the seed for rand ordering
738*89c4ff92SAndroid Build Coastguard Worker 
739*89c4ff92SAndroid Build Coastguard Worker     unsigned first; // the first (matching) test to be executed
740*89c4ff92SAndroid Build Coastguard Worker     unsigned last;  // the last (matching) test to be executed
741*89c4ff92SAndroid Build Coastguard Worker 
742*89c4ff92SAndroid Build Coastguard Worker     int abort_after;           // stop tests after this many failed assertions
743*89c4ff92SAndroid Build Coastguard Worker     int subcase_filter_levels; // apply the subcase filters for the first N levels
744*89c4ff92SAndroid Build Coastguard Worker 
745*89c4ff92SAndroid Build Coastguard Worker     bool success;              // include successful assertions in output
746*89c4ff92SAndroid Build Coastguard Worker     bool case_sensitive;       // if filtering should be case sensitive
747*89c4ff92SAndroid Build Coastguard Worker     bool exit;                 // if the program should be exited after the tests are ran/whatever
748*89c4ff92SAndroid Build Coastguard Worker     bool duration;             // print the time duration of each test case
749*89c4ff92SAndroid Build Coastguard Worker     bool no_throw;             // to skip exceptions-related assertion macros
750*89c4ff92SAndroid Build Coastguard Worker     bool no_exitcode;          // if the framework should return 0 as the exitcode
751*89c4ff92SAndroid Build Coastguard Worker     bool no_run;               // to not run the tests at all (can be done with an "*" exclude)
752*89c4ff92SAndroid Build Coastguard Worker     bool no_version;           // to not print the version of the framework
753*89c4ff92SAndroid Build Coastguard Worker     bool no_colors;            // if output to the console should be colorized
754*89c4ff92SAndroid Build Coastguard Worker     bool force_colors;         // forces the use of colors even when a tty cannot be detected
755*89c4ff92SAndroid Build Coastguard Worker     bool no_breaks;            // to not break into the debugger
756*89c4ff92SAndroid Build Coastguard Worker     bool no_skip;              // don't skip test cases which are marked to be skipped
757*89c4ff92SAndroid Build Coastguard Worker     bool gnu_file_line;        // if line numbers should be surrounded with :x: and not (x):
758*89c4ff92SAndroid Build Coastguard Worker     bool no_path_in_filenames; // if the path to files should be removed from the output
759*89c4ff92SAndroid Build Coastguard Worker     bool no_line_numbers;      // if source code line numbers should be omitted from the output
760*89c4ff92SAndroid Build Coastguard Worker     bool no_debug_output;      // no output in the debug console when a debugger is attached
761*89c4ff92SAndroid Build Coastguard Worker     bool no_skipped_summary;   // don't print "skipped" in the summary !!! UNDOCUMENTED !!!
762*89c4ff92SAndroid Build Coastguard Worker     bool no_time_in_output;    // omit any time/timestamps from output !!! UNDOCUMENTED !!!
763*89c4ff92SAndroid Build Coastguard Worker 
764*89c4ff92SAndroid Build Coastguard Worker     bool help;             // to print the help
765*89c4ff92SAndroid Build Coastguard Worker     bool version;          // to print the version
766*89c4ff92SAndroid Build Coastguard Worker     bool count;            // if only the count of matching tests is to be retrieved
767*89c4ff92SAndroid Build Coastguard Worker     bool list_test_cases;  // to list all tests matching the filters
768*89c4ff92SAndroid Build Coastguard Worker     bool list_test_suites; // to list all suites matching the filters
769*89c4ff92SAndroid Build Coastguard Worker     bool list_reporters;   // lists all registered reporters
770*89c4ff92SAndroid Build Coastguard Worker };
771*89c4ff92SAndroid Build Coastguard Worker 
772*89c4ff92SAndroid Build Coastguard Worker namespace detail {
773*89c4ff92SAndroid Build Coastguard Worker     template <bool CONDITION, typename TYPE = void>
774*89c4ff92SAndroid Build Coastguard Worker     struct enable_if
775*89c4ff92SAndroid Build Coastguard Worker     {};
776*89c4ff92SAndroid Build Coastguard Worker 
777*89c4ff92SAndroid Build Coastguard Worker     template <typename TYPE>
778*89c4ff92SAndroid Build Coastguard Worker     struct enable_if<true, TYPE>
779*89c4ff92SAndroid Build Coastguard Worker     { typedef TYPE type; };
780*89c4ff92SAndroid Build Coastguard Worker 
781*89c4ff92SAndroid Build Coastguard Worker     // clang-format off
782*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct remove_reference      { typedef T type; };
783*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct remove_reference<T&>  { typedef T type; };
784*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct remove_reference<T&&> { typedef T type; };
785*89c4ff92SAndroid Build Coastguard Worker 
786*89c4ff92SAndroid Build Coastguard Worker     template<typename T, typename U = T&&> U declval(int);
787*89c4ff92SAndroid Build Coastguard Worker 
788*89c4ff92SAndroid Build Coastguard Worker     template<typename T> T declval(long);
789*89c4ff92SAndroid Build Coastguard Worker 
790*89c4ff92SAndroid Build Coastguard Worker     template<typename T> auto declval() DOCTEST_NOEXCEPT -> decltype(declval<T>(0)) ;
791*89c4ff92SAndroid Build Coastguard Worker 
792*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct is_lvalue_reference { const static bool value=false; };
793*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct is_lvalue_reference<T&> { const static bool value=true; };
794*89c4ff92SAndroid Build Coastguard Worker 
795*89c4ff92SAndroid Build Coastguard Worker     template <class T>
796*89c4ff92SAndroid Build Coastguard Worker     inline T&& forward(typename remove_reference<T>::type& t) DOCTEST_NOEXCEPT
797*89c4ff92SAndroid Build Coastguard Worker     {
798*89c4ff92SAndroid Build Coastguard Worker         return static_cast<T&&>(t);
799*89c4ff92SAndroid Build Coastguard Worker     }
800*89c4ff92SAndroid Build Coastguard Worker 
801*89c4ff92SAndroid Build Coastguard Worker     template <class T>
802*89c4ff92SAndroid Build Coastguard Worker     inline T&& forward(typename remove_reference<T>::type&& t) DOCTEST_NOEXCEPT
803*89c4ff92SAndroid Build Coastguard Worker     {
804*89c4ff92SAndroid Build Coastguard Worker         static_assert(!is_lvalue_reference<T>::value,
805*89c4ff92SAndroid Build Coastguard Worker                         "Can not forward an rvalue as an lvalue.");
806*89c4ff92SAndroid Build Coastguard Worker         return static_cast<T&&>(t);
807*89c4ff92SAndroid Build Coastguard Worker     }
808*89c4ff92SAndroid Build Coastguard Worker 
809*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct remove_const          { typedef T type; };
810*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct remove_const<const T> { typedef T type; };
811*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
812*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct is_enum : public std::is_enum<T> {};
813*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct underlying_type : public std::underlying_type<T> {};
814*89c4ff92SAndroid Build Coastguard Worker #else
815*89c4ff92SAndroid Build Coastguard Worker     // Use compiler intrinsics
816*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct is_enum { constexpr static bool value = __is_enum(T); };
817*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct underlying_type { typedef __underlying_type(T) type; };
818*89c4ff92SAndroid Build Coastguard Worker #endif
819*89c4ff92SAndroid Build Coastguard Worker     // clang-format on
820*89c4ff92SAndroid Build Coastguard Worker 
821*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
822*89c4ff92SAndroid Build Coastguard Worker     struct deferred_false
823*89c4ff92SAndroid Build Coastguard Worker     // cppcheck-suppress unusedStructMember
824*89c4ff92SAndroid Build Coastguard Worker     { static const bool value = false; };
825*89c4ff92SAndroid Build Coastguard Worker 
826*89c4ff92SAndroid Build Coastguard Worker     namespace has_insertion_operator_impl {
827*89c4ff92SAndroid Build Coastguard Worker         std::ostream &os();
828*89c4ff92SAndroid Build Coastguard Worker         template<class T>
829*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_REF_WRAP(T) val();
830*89c4ff92SAndroid Build Coastguard Worker 
831*89c4ff92SAndroid Build Coastguard Worker         template<class, class = void>
832*89c4ff92SAndroid Build Coastguard Worker         struct check {
833*89c4ff92SAndroid Build Coastguard Worker             static constexpr bool value = false;
834*89c4ff92SAndroid Build Coastguard Worker         };
835*89c4ff92SAndroid Build Coastguard Worker 
836*89c4ff92SAndroid Build Coastguard Worker         template<class T>
837*89c4ff92SAndroid Build Coastguard Worker         struct check<T, decltype(os() << val<T>(), void())> {
838*89c4ff92SAndroid Build Coastguard Worker             static constexpr bool value = true;
839*89c4ff92SAndroid Build Coastguard Worker         };
840*89c4ff92SAndroid Build Coastguard Worker     } // namespace has_insertion_operator_impl
841*89c4ff92SAndroid Build Coastguard Worker 
842*89c4ff92SAndroid Build Coastguard Worker     template<class T>
843*89c4ff92SAndroid Build Coastguard Worker     using has_insertion_operator = has_insertion_operator_impl::check<const T>;
844*89c4ff92SAndroid Build Coastguard Worker 
845*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void my_memcpy(void* dest, const void* src, unsigned num);
846*89c4ff92SAndroid Build Coastguard Worker 
847*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE std::ostream* getTlsOss(); // returns a thread-local ostringstream
848*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE String getTlsOssResult();
849*89c4ff92SAndroid Build Coastguard Worker 
850*89c4ff92SAndroid Build Coastguard Worker     template <bool C>
851*89c4ff92SAndroid Build Coastguard Worker     struct StringMakerBase
852*89c4ff92SAndroid Build Coastguard Worker     {
853*89c4ff92SAndroid Build Coastguard Worker         template <typename T>
854*89c4ff92SAndroid Build Coastguard Worker         static String convert(const DOCTEST_REF_WRAP(T)) {
855*89c4ff92SAndroid Build Coastguard Worker             return "{?}";
856*89c4ff92SAndroid Build Coastguard Worker         }
857*89c4ff92SAndroid Build Coastguard Worker     };
858*89c4ff92SAndroid Build Coastguard Worker 
859*89c4ff92SAndroid Build Coastguard Worker     template <>
860*89c4ff92SAndroid Build Coastguard Worker     struct StringMakerBase<true>
861*89c4ff92SAndroid Build Coastguard Worker     {
862*89c4ff92SAndroid Build Coastguard Worker         template <typename T>
863*89c4ff92SAndroid Build Coastguard Worker         static String convert(const DOCTEST_REF_WRAP(T) in) {
864*89c4ff92SAndroid Build Coastguard Worker             *getTlsOss() << in;
865*89c4ff92SAndroid Build Coastguard Worker             return getTlsOssResult();
866*89c4ff92SAndroid Build Coastguard Worker         }
867*89c4ff92SAndroid Build Coastguard Worker     };
868*89c4ff92SAndroid Build Coastguard Worker 
869*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE String rawMemoryToString(const void* object, unsigned size);
870*89c4ff92SAndroid Build Coastguard Worker 
871*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
872*89c4ff92SAndroid Build Coastguard Worker     String rawMemoryToString(const DOCTEST_REF_WRAP(T) object) {
873*89c4ff92SAndroid Build Coastguard Worker         return rawMemoryToString(&object, sizeof(object));
874*89c4ff92SAndroid Build Coastguard Worker     }
875*89c4ff92SAndroid Build Coastguard Worker 
876*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
877*89c4ff92SAndroid Build Coastguard Worker     const char* type_to_string() {
878*89c4ff92SAndroid Build Coastguard Worker         return "<>";
879*89c4ff92SAndroid Build Coastguard Worker     }
880*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
881*89c4ff92SAndroid Build Coastguard Worker 
882*89c4ff92SAndroid Build Coastguard Worker template <typename T>
883*89c4ff92SAndroid Build Coastguard Worker struct StringMaker : public detail::StringMakerBase<detail::has_insertion_operator<T>::value>
884*89c4ff92SAndroid Build Coastguard Worker {};
885*89c4ff92SAndroid Build Coastguard Worker 
886*89c4ff92SAndroid Build Coastguard Worker template <typename T>
887*89c4ff92SAndroid Build Coastguard Worker struct StringMaker<T*>
888*89c4ff92SAndroid Build Coastguard Worker {
889*89c4ff92SAndroid Build Coastguard Worker     template <typename U>
890*89c4ff92SAndroid Build Coastguard Worker     static String convert(U* p) {
891*89c4ff92SAndroid Build Coastguard Worker         if(p)
892*89c4ff92SAndroid Build Coastguard Worker             return detail::rawMemoryToString(p);
893*89c4ff92SAndroid Build Coastguard Worker         return "NULL";
894*89c4ff92SAndroid Build Coastguard Worker     }
895*89c4ff92SAndroid Build Coastguard Worker };
896*89c4ff92SAndroid Build Coastguard Worker 
897*89c4ff92SAndroid Build Coastguard Worker template <typename R, typename C>
898*89c4ff92SAndroid Build Coastguard Worker struct StringMaker<R C::*>
899*89c4ff92SAndroid Build Coastguard Worker {
900*89c4ff92SAndroid Build Coastguard Worker     static String convert(R C::*p) {
901*89c4ff92SAndroid Build Coastguard Worker         if(p)
902*89c4ff92SAndroid Build Coastguard Worker             return detail::rawMemoryToString(p);
903*89c4ff92SAndroid Build Coastguard Worker         return "NULL";
904*89c4ff92SAndroid Build Coastguard Worker     }
905*89c4ff92SAndroid Build Coastguard Worker };
906*89c4ff92SAndroid Build Coastguard Worker 
907*89c4ff92SAndroid Build Coastguard Worker template <typename T, typename detail::enable_if<!detail::is_enum<T>::value, bool>::type = true>
908*89c4ff92SAndroid Build Coastguard Worker String toString(const DOCTEST_REF_WRAP(T) value) {
909*89c4ff92SAndroid Build Coastguard Worker     return StringMaker<T>::convert(value);
910*89c4ff92SAndroid Build Coastguard Worker }
911*89c4ff92SAndroid Build Coastguard Worker 
912*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
913*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(char* in);
914*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(const char* in);
915*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
916*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(bool in);
917*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(float in);
918*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(double in);
919*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(double long in);
920*89c4ff92SAndroid Build Coastguard Worker 
921*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(char in);
922*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(char signed in);
923*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(char unsigned in);
924*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(int short in);
925*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(int short unsigned in);
926*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(int in);
927*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(int unsigned in);
928*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(int long in);
929*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(int long unsigned in);
930*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(int long long in);
931*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(int long long unsigned in);
932*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(std::nullptr_t in);
933*89c4ff92SAndroid Build Coastguard Worker 
934*89c4ff92SAndroid Build Coastguard Worker template <typename T, typename detail::enable_if<detail::is_enum<T>::value, bool>::type = true>
935*89c4ff92SAndroid Build Coastguard Worker String toString(const DOCTEST_REF_WRAP(T) value) {
936*89c4ff92SAndroid Build Coastguard Worker     typedef typename detail::underlying_type<T>::type UT;
937*89c4ff92SAndroid Build Coastguard Worker     return toString(static_cast<UT>(value));
938*89c4ff92SAndroid Build Coastguard Worker }
939*89c4ff92SAndroid Build Coastguard Worker 
940*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_MSVC >= DOCTEST_COMPILER(19, 20, 0)
941*89c4ff92SAndroid Build Coastguard Worker // see this issue on why this is needed: https://github.com/onqtam/doctest/issues/183
942*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(const std::string& in);
943*89c4ff92SAndroid Build Coastguard Worker #endif // VS 2019
944*89c4ff92SAndroid Build Coastguard Worker 
945*89c4ff92SAndroid Build Coastguard Worker class DOCTEST_INTERFACE Approx
946*89c4ff92SAndroid Build Coastguard Worker {
947*89c4ff92SAndroid Build Coastguard Worker public:
948*89c4ff92SAndroid Build Coastguard Worker     explicit Approx(double value);
949*89c4ff92SAndroid Build Coastguard Worker 
950*89c4ff92SAndroid Build Coastguard Worker     Approx operator()(double value) const;
951*89c4ff92SAndroid Build Coastguard Worker 
952*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
953*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
954*89c4ff92SAndroid Build Coastguard Worker     explicit Approx(const T& value,
955*89c4ff92SAndroid Build Coastguard Worker                     typename detail::enable_if<std::is_constructible<double, T>::value>::type* =
956*89c4ff92SAndroid Build Coastguard Worker                             static_cast<T*>(nullptr)) {
957*89c4ff92SAndroid Build Coastguard Worker         *this = Approx(static_cast<double>(value));
958*89c4ff92SAndroid Build Coastguard Worker     }
959*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
960*89c4ff92SAndroid Build Coastguard Worker 
961*89c4ff92SAndroid Build Coastguard Worker     Approx& epsilon(double newEpsilon);
962*89c4ff92SAndroid Build Coastguard Worker 
963*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
964*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
965*89c4ff92SAndroid Build Coastguard Worker     typename detail::enable_if<std::is_constructible<double, T>::value, Approx&>::type epsilon(
966*89c4ff92SAndroid Build Coastguard Worker             const T& newEpsilon) {
967*89c4ff92SAndroid Build Coastguard Worker         m_epsilon = static_cast<double>(newEpsilon);
968*89c4ff92SAndroid Build Coastguard Worker         return *this;
969*89c4ff92SAndroid Build Coastguard Worker     }
970*89c4ff92SAndroid Build Coastguard Worker #endif //  DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
971*89c4ff92SAndroid Build Coastguard Worker 
972*89c4ff92SAndroid Build Coastguard Worker     Approx& scale(double newScale);
973*89c4ff92SAndroid Build Coastguard Worker 
974*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
975*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
976*89c4ff92SAndroid Build Coastguard Worker     typename detail::enable_if<std::is_constructible<double, T>::value, Approx&>::type scale(
977*89c4ff92SAndroid Build Coastguard Worker             const T& newScale) {
978*89c4ff92SAndroid Build Coastguard Worker         m_scale = static_cast<double>(newScale);
979*89c4ff92SAndroid Build Coastguard Worker         return *this;
980*89c4ff92SAndroid Build Coastguard Worker     }
981*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
982*89c4ff92SAndroid Build Coastguard Worker 
983*89c4ff92SAndroid Build Coastguard Worker     // clang-format off
984*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator==(double lhs, const Approx & rhs);
985*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator==(const Approx & lhs, double rhs);
986*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator!=(double lhs, const Approx & rhs);
987*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator!=(const Approx & lhs, double rhs);
988*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator<=(double lhs, const Approx & rhs);
989*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator<=(const Approx & lhs, double rhs);
990*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator>=(double lhs, const Approx & rhs);
991*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator>=(const Approx & lhs, double rhs);
992*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator< (double lhs, const Approx & rhs);
993*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator< (const Approx & lhs, double rhs);
994*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator> (double lhs, const Approx & rhs);
995*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend bool operator> (const Approx & lhs, double rhs);
996*89c4ff92SAndroid Build Coastguard Worker 
997*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE friend String toString(const Approx& in);
998*89c4ff92SAndroid Build Coastguard Worker 
999*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
1000*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_APPROX_PREFIX \
1001*89c4ff92SAndroid Build Coastguard Worker     template <typename T> friend typename detail::enable_if<std::is_constructible<double, T>::value, bool>::type
1002*89c4ff92SAndroid Build Coastguard Worker 
1003*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator==(const T& lhs, const Approx& rhs) { return operator==(double(lhs), rhs); }
1004*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator==(const Approx& lhs, const T& rhs) { return operator==(rhs, lhs); }
1005*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator!=(const T& lhs, const Approx& rhs) { return !operator==(lhs, rhs); }
1006*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator!=(const Approx& lhs, const T& rhs) { return !operator==(rhs, lhs); }
1007*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator<=(const T& lhs, const Approx& rhs) { return double(lhs) < rhs.m_value || lhs == rhs; }
1008*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator<=(const Approx& lhs, const T& rhs) { return lhs.m_value < double(rhs) || lhs == rhs; }
1009*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator>=(const T& lhs, const Approx& rhs) { return double(lhs) > rhs.m_value || lhs == rhs; }
1010*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator>=(const Approx& lhs, const T& rhs) { return lhs.m_value > double(rhs) || lhs == rhs; }
1011*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator< (const T& lhs, const Approx& rhs) { return double(lhs) < rhs.m_value && lhs != rhs; }
1012*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator< (const Approx& lhs, const T& rhs) { return lhs.m_value < double(rhs) && lhs != rhs; }
1013*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator> (const T& lhs, const Approx& rhs) { return double(lhs) > rhs.m_value && lhs != rhs; }
1014*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_APPROX_PREFIX operator> (const Approx& lhs, const T& rhs) { return lhs.m_value > double(rhs) && lhs != rhs; }
1015*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_APPROX_PREFIX
1016*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_INCLUDE_TYPE_TRAITS
1017*89c4ff92SAndroid Build Coastguard Worker 
1018*89c4ff92SAndroid Build Coastguard Worker     // clang-format on
1019*89c4ff92SAndroid Build Coastguard Worker 
1020*89c4ff92SAndroid Build Coastguard Worker private:
1021*89c4ff92SAndroid Build Coastguard Worker     double m_epsilon;
1022*89c4ff92SAndroid Build Coastguard Worker     double m_scale;
1023*89c4ff92SAndroid Build Coastguard Worker     double m_value;
1024*89c4ff92SAndroid Build Coastguard Worker };
1025*89c4ff92SAndroid Build Coastguard Worker 
1026*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE String toString(const Approx& in);
1027*89c4ff92SAndroid Build Coastguard Worker 
1028*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE const ContextOptions* getContextOptions();
1029*89c4ff92SAndroid Build Coastguard Worker 
1030*89c4ff92SAndroid Build Coastguard Worker #if !defined(DOCTEST_CONFIG_DISABLE)
1031*89c4ff92SAndroid Build Coastguard Worker 
1032*89c4ff92SAndroid Build Coastguard Worker namespace detail {
1033*89c4ff92SAndroid Build Coastguard Worker     // clang-format off
1034*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
1035*89c4ff92SAndroid Build Coastguard Worker     template<class T>               struct decay_array       { typedef T type; };
1036*89c4ff92SAndroid Build Coastguard Worker     template<class T, unsigned N>   struct decay_array<T[N]> { typedef T* type; };
1037*89c4ff92SAndroid Build Coastguard Worker     template<class T>               struct decay_array<T[]>  { typedef T* type; };
1038*89c4ff92SAndroid Build Coastguard Worker 
1039*89c4ff92SAndroid Build Coastguard Worker     template<class T>   struct not_char_pointer              { enum { value = 1 }; };
1040*89c4ff92SAndroid Build Coastguard Worker     template<>          struct not_char_pointer<char*>       { enum { value = 0 }; };
1041*89c4ff92SAndroid Build Coastguard Worker     template<>          struct not_char_pointer<const char*> { enum { value = 0 }; };
1042*89c4ff92SAndroid Build Coastguard Worker 
1043*89c4ff92SAndroid Build Coastguard Worker     template<class T> struct can_use_op : public not_char_pointer<typename decay_array<T>::type> {};
1044*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
1045*89c4ff92SAndroid Build Coastguard Worker     // clang-format on
1046*89c4ff92SAndroid Build Coastguard Worker 
1047*89c4ff92SAndroid Build Coastguard Worker     struct DOCTEST_INTERFACE TestFailureException
1048*89c4ff92SAndroid Build Coastguard Worker     {
1049*89c4ff92SAndroid Build Coastguard Worker     };
1050*89c4ff92SAndroid Build Coastguard Worker 
1051*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE bool checkIfShouldThrow(assertType::Enum at);
1052*89c4ff92SAndroid Build Coastguard Worker 
1053*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_EXCEPTIONS
1054*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_NORETURN
1055*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS
1056*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void throwException();
1057*89c4ff92SAndroid Build Coastguard Worker 
1058*89c4ff92SAndroid Build Coastguard Worker     struct DOCTEST_INTERFACE Subcase
1059*89c4ff92SAndroid Build Coastguard Worker     {
1060*89c4ff92SAndroid Build Coastguard Worker         SubcaseSignature m_signature;
1061*89c4ff92SAndroid Build Coastguard Worker         bool             m_entered = false;
1062*89c4ff92SAndroid Build Coastguard Worker 
1063*89c4ff92SAndroid Build Coastguard Worker         Subcase(const String& name, const char* file, int line);
1064*89c4ff92SAndroid Build Coastguard Worker         ~Subcase();
1065*89c4ff92SAndroid Build Coastguard Worker 
1066*89c4ff92SAndroid Build Coastguard Worker         operator bool() const;
1067*89c4ff92SAndroid Build Coastguard Worker     };
1068*89c4ff92SAndroid Build Coastguard Worker 
1069*89c4ff92SAndroid Build Coastguard Worker     template <typename L, typename R>
1070*89c4ff92SAndroid Build Coastguard Worker     String stringifyBinaryExpr(const DOCTEST_REF_WRAP(L) lhs, const char* op,
1071*89c4ff92SAndroid Build Coastguard Worker                                const DOCTEST_REF_WRAP(R) rhs) {
1072*89c4ff92SAndroid Build Coastguard Worker         // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
1073*89c4ff92SAndroid Build Coastguard Worker         return toString(lhs) + op + toString(rhs);
1074*89c4ff92SAndroid Build Coastguard Worker     }
1075*89c4ff92SAndroid Build Coastguard Worker 
1076*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_CLANG && DOCTEST_CLANG < DOCTEST_COMPILER(3, 6, 0)
1077*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wunused-comparison")
1078*89c4ff92SAndroid Build Coastguard Worker #endif
1079*89c4ff92SAndroid Build Coastguard Worker 
1080*89c4ff92SAndroid Build Coastguard Worker // This will check if there is any way it could find a operator like member or friend and uses it.
1081*89c4ff92SAndroid Build Coastguard Worker // If not it doesn't find the operator or if the operator at global scope is defined after
1082*89c4ff92SAndroid Build Coastguard Worker // this template, the template won't be instantiated due to SFINAE. Once the template is not
1083*89c4ff92SAndroid Build Coastguard Worker // instantiated it can look for global operator using normal conversions.
1084*89c4ff92SAndroid Build Coastguard Worker #define SFINAE_OP(ret,op) decltype(doctest::detail::declval<L>() op doctest::detail::declval<R>(),static_cast<ret>(0))
1085*89c4ff92SAndroid Build Coastguard Worker 
1086*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(op, op_str, op_macro)                              \
1087*89c4ff92SAndroid Build Coastguard Worker     template <typename R>                                                                          \
1088*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_NOINLINE SFINAE_OP(Result,op) operator op(R&& rhs) {             \
1089*89c4ff92SAndroid Build Coastguard Worker 	    bool res = op_macro(doctest::detail::forward<L>(lhs), doctest::detail::forward<R>(rhs));                                                             \
1090*89c4ff92SAndroid Build Coastguard Worker         if(m_at & assertType::is_false)                                                            \
1091*89c4ff92SAndroid Build Coastguard Worker             res = !res;                                                                            \
1092*89c4ff92SAndroid Build Coastguard Worker         if(!res || doctest::getContextOptions()->success)                                          \
1093*89c4ff92SAndroid Build Coastguard Worker             return Result(res, stringifyBinaryExpr(lhs, op_str, rhs));                             \
1094*89c4ff92SAndroid Build Coastguard Worker         return Result(res);                                                                        \
1095*89c4ff92SAndroid Build Coastguard Worker     }
1096*89c4ff92SAndroid Build Coastguard Worker 
1097*89c4ff92SAndroid Build Coastguard Worker     // more checks could be added - like in Catch:
1098*89c4ff92SAndroid Build Coastguard Worker     // https://github.com/catchorg/Catch2/pull/1480/files
1099*89c4ff92SAndroid Build Coastguard Worker     // https://github.com/catchorg/Catch2/pull/1481/files
1100*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FORBIT_EXPRESSION(rt, op)                                                          \
1101*89c4ff92SAndroid Build Coastguard Worker     template <typename R>                                                                          \
1102*89c4ff92SAndroid Build Coastguard Worker     rt& operator op(const R&) {                                                                    \
1103*89c4ff92SAndroid Build Coastguard Worker         static_assert(deferred_false<R>::value,                                                    \
1104*89c4ff92SAndroid Build Coastguard Worker                       "Expression Too Complex Please Rewrite As Binary Comparison!");              \
1105*89c4ff92SAndroid Build Coastguard Worker         return *this;                                                                              \
1106*89c4ff92SAndroid Build Coastguard Worker     }
1107*89c4ff92SAndroid Build Coastguard Worker 
1108*89c4ff92SAndroid Build Coastguard Worker     struct DOCTEST_INTERFACE Result
1109*89c4ff92SAndroid Build Coastguard Worker     {
1110*89c4ff92SAndroid Build Coastguard Worker         bool   m_passed;
1111*89c4ff92SAndroid Build Coastguard Worker         String m_decomp;
1112*89c4ff92SAndroid Build Coastguard Worker 
1113*89c4ff92SAndroid Build Coastguard Worker         Result(bool passed, const String& decomposition = String());
1114*89c4ff92SAndroid Build Coastguard Worker 
1115*89c4ff92SAndroid Build Coastguard Worker         // forbidding some expressions based on this table: https://en.cppreference.com/w/cpp/language/operator_precedence
1116*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, &)
1117*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, ^)
1118*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, |)
1119*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, &&)
1120*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, ||)
1121*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, ==)
1122*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, !=)
1123*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, <)
1124*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, >)
1125*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, <=)
1126*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, >=)
1127*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, =)
1128*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, +=)
1129*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, -=)
1130*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, *=)
1131*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, /=)
1132*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, %=)
1133*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, <<=)
1134*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, >>=)
1135*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, &=)
1136*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, ^=)
1137*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Result, |=)
1138*89c4ff92SAndroid Build Coastguard Worker     };
1139*89c4ff92SAndroid Build Coastguard Worker 
1140*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_COMPARISON_WARNING_SUPPRESSION
1141*89c4ff92SAndroid Build Coastguard Worker 
1142*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_PUSH
1143*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING("-Wsign-conversion")
1144*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING("-Wsign-compare")
1145*89c4ff92SAndroid Build Coastguard Worker     //DOCTEST_CLANG_SUPPRESS_WARNING("-Wdouble-promotion")
1146*89c4ff92SAndroid Build Coastguard Worker     //DOCTEST_CLANG_SUPPRESS_WARNING("-Wconversion")
1147*89c4ff92SAndroid Build Coastguard Worker     //DOCTEST_CLANG_SUPPRESS_WARNING("-Wfloat-equal")
1148*89c4ff92SAndroid Build Coastguard Worker 
1149*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING_PUSH
1150*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING("-Wsign-conversion")
1151*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING("-Wsign-compare")
1152*89c4ff92SAndroid Build Coastguard Worker     //DOCTEST_GCC_SUPPRESS_WARNING("-Wdouble-promotion")
1153*89c4ff92SAndroid Build Coastguard Worker     //DOCTEST_GCC_SUPPRESS_WARNING("-Wconversion")
1154*89c4ff92SAndroid Build Coastguard Worker     //DOCTEST_GCC_SUPPRESS_WARNING("-Wfloat-equal")
1155*89c4ff92SAndroid Build Coastguard Worker 
1156*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_PUSH
1157*89c4ff92SAndroid Build Coastguard Worker     // https://stackoverflow.com/questions/39479163 what's the difference between 4018 and 4389
1158*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4388) // signed/unsigned mismatch
1159*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4389) // 'operator' : signed/unsigned mismatch
1160*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(4018) // 'expression' : signed/unsigned mismatch
1161*89c4ff92SAndroid Build Coastguard Worker     //DOCTEST_MSVC_SUPPRESS_WARNING(4805) // 'operation' : unsafe mix of type 'type' and type 'type' in operation
1162*89c4ff92SAndroid Build Coastguard Worker 
1163*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_COMPARISON_WARNING_SUPPRESSION
1164*89c4ff92SAndroid Build Coastguard Worker 
1165*89c4ff92SAndroid Build Coastguard Worker     // clang-format off
1166*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
1167*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_COMPARISON_RETURN_TYPE bool
1168*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
1169*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_COMPARISON_RETURN_TYPE typename enable_if<can_use_op<L>::value || can_use_op<R>::value, bool>::type
1170*89c4ff92SAndroid Build Coastguard Worker     // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
1171*89c4ff92SAndroid Build Coastguard Worker     inline bool eq(const char* lhs, const char* rhs) { return String(lhs) == String(rhs); }
1172*89c4ff92SAndroid Build Coastguard Worker     inline bool ne(const char* lhs, const char* rhs) { return String(lhs) != String(rhs); }
1173*89c4ff92SAndroid Build Coastguard Worker     inline bool lt(const char* lhs, const char* rhs) { return String(lhs) <  String(rhs); }
1174*89c4ff92SAndroid Build Coastguard Worker     inline bool gt(const char* lhs, const char* rhs) { return String(lhs) >  String(rhs); }
1175*89c4ff92SAndroid Build Coastguard Worker     inline bool le(const char* lhs, const char* rhs) { return String(lhs) <= String(rhs); }
1176*89c4ff92SAndroid Build Coastguard Worker     inline bool ge(const char* lhs, const char* rhs) { return String(lhs) >= String(rhs); }
1177*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
1178*89c4ff92SAndroid Build Coastguard Worker     // clang-format on
1179*89c4ff92SAndroid Build Coastguard Worker 
1180*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_RELATIONAL_OP(name, op)                                                            \
1181*89c4ff92SAndroid Build Coastguard Worker     template <typename L, typename R>                                                              \
1182*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_COMPARISON_RETURN_TYPE name(const DOCTEST_REF_WRAP(L) lhs,                             \
1183*89c4ff92SAndroid Build Coastguard Worker                                         const DOCTEST_REF_WRAP(R) rhs) {                           \
1184*89c4ff92SAndroid Build Coastguard Worker         return lhs op rhs;                                                                         \
1185*89c4ff92SAndroid Build Coastguard Worker     }
1186*89c4ff92SAndroid Build Coastguard Worker 
1187*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_RELATIONAL_OP(eq, ==)
1188*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_RELATIONAL_OP(ne, !=)
1189*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_RELATIONAL_OP(lt, <)
1190*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_RELATIONAL_OP(gt, >)
1191*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_RELATIONAL_OP(le, <=)
1192*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_RELATIONAL_OP(ge, >=)
1193*89c4ff92SAndroid Build Coastguard Worker 
1194*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
1195*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_EQ(l, r) l == r
1196*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_NE(l, r) l != r
1197*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_GT(l, r) l > r
1198*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_LT(l, r) l < r
1199*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_GE(l, r) l >= r
1200*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_LE(l, r) l <= r
1201*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
1202*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_EQ(l, r) eq(l, r)
1203*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_NE(l, r) ne(l, r)
1204*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_GT(l, r) gt(l, r)
1205*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_LT(l, r) lt(l, r)
1206*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_GE(l, r) ge(l, r)
1207*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CMP_LE(l, r) le(l, r)
1208*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
1209*89c4ff92SAndroid Build Coastguard Worker 
1210*89c4ff92SAndroid Build Coastguard Worker     template <typename L>
1211*89c4ff92SAndroid Build Coastguard Worker     // cppcheck-suppress copyCtorAndEqOperator
1212*89c4ff92SAndroid Build Coastguard Worker     struct Expression_lhs
1213*89c4ff92SAndroid Build Coastguard Worker     {
1214*89c4ff92SAndroid Build Coastguard Worker         L                lhs;
1215*89c4ff92SAndroid Build Coastguard Worker         assertType::Enum m_at;
1216*89c4ff92SAndroid Build Coastguard Worker 
1217*89c4ff92SAndroid Build Coastguard Worker         explicit Expression_lhs(L&& in, assertType::Enum at)
1218*89c4ff92SAndroid Build Coastguard Worker                 : lhs(doctest::detail::forward<L>(in))
1219*89c4ff92SAndroid Build Coastguard Worker                 , m_at(at) {}
1220*89c4ff92SAndroid Build Coastguard Worker 
1221*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_NOINLINE operator Result() {
1222*89c4ff92SAndroid Build Coastguard Worker // this is needed only foc MSVC 2015:
1223*89c4ff92SAndroid Build Coastguard Worker // https://ci.appveyor.com/project/onqtam/doctest/builds/38181202
1224*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4800) // 'int': forcing value to bool
1225*89c4ff92SAndroid Build Coastguard Worker             bool res = static_cast<bool>(lhs);
1226*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_POP
1227*89c4ff92SAndroid Build Coastguard Worker             if(m_at & assertType::is_false) //!OCLINT bitwise operator in conditional
1228*89c4ff92SAndroid Build Coastguard Worker                 res = !res;
1229*89c4ff92SAndroid Build Coastguard Worker 
1230*89c4ff92SAndroid Build Coastguard Worker             if(!res || getContextOptions()->success)
1231*89c4ff92SAndroid Build Coastguard Worker                 return Result(res, toString(lhs));
1232*89c4ff92SAndroid Build Coastguard Worker             return Result(res);
1233*89c4ff92SAndroid Build Coastguard Worker         }
1234*89c4ff92SAndroid Build Coastguard Worker 
1235*89c4ff92SAndroid Build Coastguard Worker 	/* This is required for user-defined conversions from Expression_lhs to L */
1236*89c4ff92SAndroid Build Coastguard Worker 	//operator L() const { return lhs; }
1237*89c4ff92SAndroid Build Coastguard Worker 	operator L() const { return lhs; }
1238*89c4ff92SAndroid Build Coastguard Worker 
1239*89c4ff92SAndroid Build Coastguard Worker         // clang-format off
1240*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(==, " == ", DOCTEST_CMP_EQ) //!OCLINT bitwise operator in conditional
1241*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(!=, " != ", DOCTEST_CMP_NE) //!OCLINT bitwise operator in conditional
1242*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(>,  " >  ", DOCTEST_CMP_GT) //!OCLINT bitwise operator in conditional
1243*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(<,  " <  ", DOCTEST_CMP_LT) //!OCLINT bitwise operator in conditional
1244*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(>=, " >= ", DOCTEST_CMP_GE) //!OCLINT bitwise operator in conditional
1245*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DO_BINARY_EXPRESSION_COMPARISON(<=, " <= ", DOCTEST_CMP_LE) //!OCLINT bitwise operator in conditional
1246*89c4ff92SAndroid Build Coastguard Worker         // clang-format on
1247*89c4ff92SAndroid Build Coastguard Worker 
1248*89c4ff92SAndroid Build Coastguard Worker         // forbidding some expressions based on this table: https://en.cppreference.com/w/cpp/language/operator_precedence
1249*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, &)
1250*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, ^)
1251*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, |)
1252*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, &&)
1253*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, ||)
1254*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, =)
1255*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, +=)
1256*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, -=)
1257*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, *=)
1258*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, /=)
1259*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, %=)
1260*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, <<=)
1261*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, >>=)
1262*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, &=)
1263*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, ^=)
1264*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, |=)
1265*89c4ff92SAndroid Build Coastguard Worker         // these 2 are unfortunate because they should be allowed - they have higher precedence over the comparisons, but the
1266*89c4ff92SAndroid Build Coastguard Worker         // ExpressionDecomposer class uses the left shift operator to capture the left operand of the binary expression...
1267*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, <<)
1268*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_FORBIT_EXPRESSION(Expression_lhs, >>)
1269*89c4ff92SAndroid Build Coastguard Worker     };
1270*89c4ff92SAndroid Build Coastguard Worker 
1271*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_COMPARISON_WARNING_SUPPRESSION
1272*89c4ff92SAndroid Build Coastguard Worker 
1273*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_POP
1274*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_POP
1275*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING_POP
1276*89c4ff92SAndroid Build Coastguard Worker 
1277*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_COMPARISON_WARNING_SUPPRESSION
1278*89c4ff92SAndroid Build Coastguard Worker 
1279*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_CLANG && DOCTEST_CLANG < DOCTEST_COMPILER(3, 6, 0)
1280*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING_POP
1281*89c4ff92SAndroid Build Coastguard Worker #endif
1282*89c4ff92SAndroid Build Coastguard Worker 
1283*89c4ff92SAndroid Build Coastguard Worker     struct DOCTEST_INTERFACE ExpressionDecomposer
1284*89c4ff92SAndroid Build Coastguard Worker     {
1285*89c4ff92SAndroid Build Coastguard Worker         assertType::Enum m_at;
1286*89c4ff92SAndroid Build Coastguard Worker 
1287*89c4ff92SAndroid Build Coastguard Worker         ExpressionDecomposer(assertType::Enum at);
1288*89c4ff92SAndroid Build Coastguard Worker 
1289*89c4ff92SAndroid Build Coastguard Worker         // The right operator for capturing expressions is "<=" instead of "<<" (based on the operator precedence table)
1290*89c4ff92SAndroid Build Coastguard Worker         // but then there will be warnings from GCC about "-Wparentheses" and since "_Pragma()" is problematic this will stay for now...
1291*89c4ff92SAndroid Build Coastguard Worker         // https://github.com/catchorg/Catch2/issues/870
1292*89c4ff92SAndroid Build Coastguard Worker         // https://github.com/catchorg/Catch2/issues/565
1293*89c4ff92SAndroid Build Coastguard Worker         template <typename L>
1294*89c4ff92SAndroid Build Coastguard Worker 	Expression_lhs<L> operator<<(L &&operand) {
1295*89c4ff92SAndroid Build Coastguard Worker             return Expression_lhs<L>(doctest::detail::forward<L>(operand), m_at);
1296*89c4ff92SAndroid Build Coastguard Worker         }
1297*89c4ff92SAndroid Build Coastguard Worker     };
1298*89c4ff92SAndroid Build Coastguard Worker 
1299*89c4ff92SAndroid Build Coastguard Worker     struct DOCTEST_INTERFACE TestSuite
1300*89c4ff92SAndroid Build Coastguard Worker     {
1301*89c4ff92SAndroid Build Coastguard Worker         const char* m_test_suite;
1302*89c4ff92SAndroid Build Coastguard Worker         const char* m_description;
1303*89c4ff92SAndroid Build Coastguard Worker         bool        m_skip;
1304*89c4ff92SAndroid Build Coastguard Worker         bool        m_no_breaks;
1305*89c4ff92SAndroid Build Coastguard Worker         bool        m_no_output;
1306*89c4ff92SAndroid Build Coastguard Worker         bool        m_may_fail;
1307*89c4ff92SAndroid Build Coastguard Worker         bool        m_should_fail;
1308*89c4ff92SAndroid Build Coastguard Worker         int         m_expected_failures;
1309*89c4ff92SAndroid Build Coastguard Worker         double      m_timeout;
1310*89c4ff92SAndroid Build Coastguard Worker 
1311*89c4ff92SAndroid Build Coastguard Worker         TestSuite& operator*(const char* in);
1312*89c4ff92SAndroid Build Coastguard Worker 
1313*89c4ff92SAndroid Build Coastguard Worker         template <typename T>
1314*89c4ff92SAndroid Build Coastguard Worker         TestSuite& operator*(const T& in) {
1315*89c4ff92SAndroid Build Coastguard Worker             in.fill(*this);
1316*89c4ff92SAndroid Build Coastguard Worker             return *this;
1317*89c4ff92SAndroid Build Coastguard Worker         }
1318*89c4ff92SAndroid Build Coastguard Worker     };
1319*89c4ff92SAndroid Build Coastguard Worker 
1320*89c4ff92SAndroid Build Coastguard Worker     typedef void (*funcType)();
1321*89c4ff92SAndroid Build Coastguard Worker 
1322*89c4ff92SAndroid Build Coastguard Worker     struct DOCTEST_INTERFACE TestCase : public TestCaseData
1323*89c4ff92SAndroid Build Coastguard Worker     {
1324*89c4ff92SAndroid Build Coastguard Worker         funcType m_test; // a function pointer to the test case
1325*89c4ff92SAndroid Build Coastguard Worker 
1326*89c4ff92SAndroid Build Coastguard Worker         const char* m_type; // for templated test cases - gets appended to the real name
1327*89c4ff92SAndroid Build Coastguard Worker         int m_template_id; // an ID used to distinguish between the different versions of a templated test case
1328*89c4ff92SAndroid Build Coastguard Worker         String m_full_name; // contains the name (only for templated test cases!) + the template type
1329*89c4ff92SAndroid Build Coastguard Worker 
1330*89c4ff92SAndroid Build Coastguard Worker         TestCase(funcType test, const char* file, unsigned line, const TestSuite& test_suite,
1331*89c4ff92SAndroid Build Coastguard Worker                  const char* type = "", int template_id = -1);
1332*89c4ff92SAndroid Build Coastguard Worker 
1333*89c4ff92SAndroid Build Coastguard Worker         TestCase(const TestCase& other);
1334*89c4ff92SAndroid Build Coastguard Worker 
1335*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(26434) // hides a non-virtual function
1336*89c4ff92SAndroid Build Coastguard Worker         TestCase& operator=(const TestCase& other);
1337*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_MSVC_SUPPRESS_WARNING_POP
1338*89c4ff92SAndroid Build Coastguard Worker 
1339*89c4ff92SAndroid Build Coastguard Worker         TestCase& operator*(const char* in);
1340*89c4ff92SAndroid Build Coastguard Worker 
1341*89c4ff92SAndroid Build Coastguard Worker         template <typename T>
1342*89c4ff92SAndroid Build Coastguard Worker         TestCase& operator*(const T& in) {
1343*89c4ff92SAndroid Build Coastguard Worker             in.fill(*this);
1344*89c4ff92SAndroid Build Coastguard Worker             return *this;
1345*89c4ff92SAndroid Build Coastguard Worker         }
1346*89c4ff92SAndroid Build Coastguard Worker 
1347*89c4ff92SAndroid Build Coastguard Worker         bool operator<(const TestCase& other) const;
1348*89c4ff92SAndroid Build Coastguard Worker     };
1349*89c4ff92SAndroid Build Coastguard Worker 
1350*89c4ff92SAndroid Build Coastguard Worker     // forward declarations of functions used by the macros
1351*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE int  regTest(const TestCase& tc);
1352*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE int  setTestSuite(const TestSuite& ts);
1353*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE bool isDebuggerActive();
1354*89c4ff92SAndroid Build Coastguard Worker 
1355*89c4ff92SAndroid Build Coastguard Worker     template<typename T>
1356*89c4ff92SAndroid Build Coastguard Worker     int instantiationHelper(const T&) { return 0; }
1357*89c4ff92SAndroid Build Coastguard Worker 
1358*89c4ff92SAndroid Build Coastguard Worker     namespace binaryAssertComparison {
1359*89c4ff92SAndroid Build Coastguard Worker         enum Enum
1360*89c4ff92SAndroid Build Coastguard Worker         {
1361*89c4ff92SAndroid Build Coastguard Worker             eq = 0,
1362*89c4ff92SAndroid Build Coastguard Worker             ne,
1363*89c4ff92SAndroid Build Coastguard Worker             gt,
1364*89c4ff92SAndroid Build Coastguard Worker             lt,
1365*89c4ff92SAndroid Build Coastguard Worker             ge,
1366*89c4ff92SAndroid Build Coastguard Worker             le
1367*89c4ff92SAndroid Build Coastguard Worker         };
1368*89c4ff92SAndroid Build Coastguard Worker     } // namespace binaryAssertComparison
1369*89c4ff92SAndroid Build Coastguard Worker 
1370*89c4ff92SAndroid Build Coastguard Worker     // clang-format off
1371*89c4ff92SAndroid Build Coastguard Worker     template <int, class L, class R> struct RelationalComparator     { bool operator()(const DOCTEST_REF_WRAP(L),     const DOCTEST_REF_WRAP(R)    ) const { return false;        } };
1372*89c4ff92SAndroid Build Coastguard Worker 
1373*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BINARY_RELATIONAL_OP(n, op) \
1374*89c4ff92SAndroid Build Coastguard Worker     template <class L, class R> struct RelationalComparator<n, L, R> { bool operator()(const DOCTEST_REF_WRAP(L) lhs, const DOCTEST_REF_WRAP(R) rhs) const { return op(lhs, rhs); } };
1375*89c4ff92SAndroid Build Coastguard Worker     // clang-format on
1376*89c4ff92SAndroid Build Coastguard Worker 
1377*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_BINARY_RELATIONAL_OP(0, doctest::detail::eq)
1378*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_BINARY_RELATIONAL_OP(1, doctest::detail::ne)
1379*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_BINARY_RELATIONAL_OP(2, doctest::detail::gt)
1380*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_BINARY_RELATIONAL_OP(3, doctest::detail::lt)
1381*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_BINARY_RELATIONAL_OP(4, doctest::detail::ge)
1382*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_BINARY_RELATIONAL_OP(5, doctest::detail::le)
1383*89c4ff92SAndroid Build Coastguard Worker 
1384*89c4ff92SAndroid Build Coastguard Worker     struct DOCTEST_INTERFACE ResultBuilder : public AssertData
1385*89c4ff92SAndroid Build Coastguard Worker     {
1386*89c4ff92SAndroid Build Coastguard Worker         ResultBuilder(assertType::Enum at, const char* file, int line, const char* expr,
1387*89c4ff92SAndroid Build Coastguard Worker                       const char* exception_type = "", const char* exception_string = "");
1388*89c4ff92SAndroid Build Coastguard Worker 
1389*89c4ff92SAndroid Build Coastguard Worker         void setResult(const Result& res);
1390*89c4ff92SAndroid Build Coastguard Worker 
1391*89c4ff92SAndroid Build Coastguard Worker         template <int comparison, typename L, typename R>
1392*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_NOINLINE void binary_assert(const DOCTEST_REF_WRAP(L) lhs,
1393*89c4ff92SAndroid Build Coastguard Worker                                             const DOCTEST_REF_WRAP(R) rhs) {
1394*89c4ff92SAndroid Build Coastguard Worker             m_failed = !RelationalComparator<comparison, L, R>()(lhs, rhs);
1395*89c4ff92SAndroid Build Coastguard Worker             if(m_failed || getContextOptions()->success)
1396*89c4ff92SAndroid Build Coastguard Worker                 m_decomp = stringifyBinaryExpr(lhs, ", ", rhs);
1397*89c4ff92SAndroid Build Coastguard Worker         }
1398*89c4ff92SAndroid Build Coastguard Worker 
1399*89c4ff92SAndroid Build Coastguard Worker         template <typename L>
1400*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_NOINLINE void unary_assert(const DOCTEST_REF_WRAP(L) val) {
1401*89c4ff92SAndroid Build Coastguard Worker             m_failed = !val;
1402*89c4ff92SAndroid Build Coastguard Worker 
1403*89c4ff92SAndroid Build Coastguard Worker             if(m_at & assertType::is_false) //!OCLINT bitwise operator in conditional
1404*89c4ff92SAndroid Build Coastguard Worker                 m_failed = !m_failed;
1405*89c4ff92SAndroid Build Coastguard Worker 
1406*89c4ff92SAndroid Build Coastguard Worker             if(m_failed || getContextOptions()->success)
1407*89c4ff92SAndroid Build Coastguard Worker                 m_decomp = toString(val);
1408*89c4ff92SAndroid Build Coastguard Worker         }
1409*89c4ff92SAndroid Build Coastguard Worker 
1410*89c4ff92SAndroid Build Coastguard Worker         void translateException();
1411*89c4ff92SAndroid Build Coastguard Worker 
1412*89c4ff92SAndroid Build Coastguard Worker         bool log();
1413*89c4ff92SAndroid Build Coastguard Worker         void react() const;
1414*89c4ff92SAndroid Build Coastguard Worker     };
1415*89c4ff92SAndroid Build Coastguard Worker 
1416*89c4ff92SAndroid Build Coastguard Worker     namespace assertAction {
1417*89c4ff92SAndroid Build Coastguard Worker         enum Enum
1418*89c4ff92SAndroid Build Coastguard Worker         {
1419*89c4ff92SAndroid Build Coastguard Worker             nothing     = 0,
1420*89c4ff92SAndroid Build Coastguard Worker             dbgbreak    = 1,
1421*89c4ff92SAndroid Build Coastguard Worker             shouldthrow = 2
1422*89c4ff92SAndroid Build Coastguard Worker         };
1423*89c4ff92SAndroid Build Coastguard Worker     } // namespace assertAction
1424*89c4ff92SAndroid Build Coastguard Worker 
1425*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void failed_out_of_a_testing_context(const AssertData& ad);
1426*89c4ff92SAndroid Build Coastguard Worker 
1427*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void decomp_assert(assertType::Enum at, const char* file, int line,
1428*89c4ff92SAndroid Build Coastguard Worker                                          const char* expr, Result result);
1429*89c4ff92SAndroid Build Coastguard Worker 
1430*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ASSERT_OUT_OF_TESTS(decomp)                                                        \
1431*89c4ff92SAndroid Build Coastguard Worker     do {                                                                                           \
1432*89c4ff92SAndroid Build Coastguard Worker         if(!is_running_in_test) {                                                                  \
1433*89c4ff92SAndroid Build Coastguard Worker             if(failed) {                                                                           \
1434*89c4ff92SAndroid Build Coastguard Worker                 ResultBuilder rb(at, file, line, expr);                                            \
1435*89c4ff92SAndroid Build Coastguard Worker                 rb.m_failed = failed;                                                              \
1436*89c4ff92SAndroid Build Coastguard Worker                 rb.m_decomp = decomp;                                                              \
1437*89c4ff92SAndroid Build Coastguard Worker                 failed_out_of_a_testing_context(rb);                                               \
1438*89c4ff92SAndroid Build Coastguard Worker                 if(isDebuggerActive() && !getContextOptions()->no_breaks)                          \
1439*89c4ff92SAndroid Build Coastguard Worker                     DOCTEST_BREAK_INTO_DEBUGGER();                                                 \
1440*89c4ff92SAndroid Build Coastguard Worker                 if(checkIfShouldThrow(at))                                                         \
1441*89c4ff92SAndroid Build Coastguard Worker                     throwException();                                                              \
1442*89c4ff92SAndroid Build Coastguard Worker             }                                                                                      \
1443*89c4ff92SAndroid Build Coastguard Worker             return;                                                                                \
1444*89c4ff92SAndroid Build Coastguard Worker         }                                                                                          \
1445*89c4ff92SAndroid Build Coastguard Worker     } while(false)
1446*89c4ff92SAndroid Build Coastguard Worker 
1447*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ASSERT_IN_TESTS(decomp)                                                            \
1448*89c4ff92SAndroid Build Coastguard Worker     ResultBuilder rb(at, file, line, expr);                                                        \
1449*89c4ff92SAndroid Build Coastguard Worker     rb.m_failed = failed;                                                                          \
1450*89c4ff92SAndroid Build Coastguard Worker     if(rb.m_failed || getContextOptions()->success)                                                \
1451*89c4ff92SAndroid Build Coastguard Worker         rb.m_decomp = decomp;                                                                      \
1452*89c4ff92SAndroid Build Coastguard Worker     if(rb.log())                                                                                   \
1453*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_BREAK_INTO_DEBUGGER();                                                             \
1454*89c4ff92SAndroid Build Coastguard Worker     if(rb.m_failed && checkIfShouldThrow(at))                                                      \
1455*89c4ff92SAndroid Build Coastguard Worker     throwException()
1456*89c4ff92SAndroid Build Coastguard Worker 
1457*89c4ff92SAndroid Build Coastguard Worker     template <int comparison, typename L, typename R>
1458*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_NOINLINE void binary_assert(assertType::Enum at, const char* file, int line,
1459*89c4ff92SAndroid Build Coastguard Worker                                         const char* expr, const DOCTEST_REF_WRAP(L) lhs,
1460*89c4ff92SAndroid Build Coastguard Worker                                         const DOCTEST_REF_WRAP(R) rhs) {
1461*89c4ff92SAndroid Build Coastguard Worker         bool failed = !RelationalComparator<comparison, L, R>()(lhs, rhs);
1462*89c4ff92SAndroid Build Coastguard Worker 
1463*89c4ff92SAndroid Build Coastguard Worker         // ###################################################################################
1464*89c4ff92SAndroid Build Coastguard Worker         // IF THE DEBUGGER BREAKS HERE - GO 1 LEVEL UP IN THE CALLSTACK FOR THE FAILING ASSERT
1465*89c4ff92SAndroid Build Coastguard Worker         // THIS IS THE EFFECT OF HAVING 'DOCTEST_CONFIG_SUPER_FAST_ASSERTS' DEFINED
1466*89c4ff92SAndroid Build Coastguard Worker         // ###################################################################################
1467*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_OUT_OF_TESTS(stringifyBinaryExpr(lhs, ", ", rhs));
1468*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_IN_TESTS(stringifyBinaryExpr(lhs, ", ", rhs));
1469*89c4ff92SAndroid Build Coastguard Worker     }
1470*89c4ff92SAndroid Build Coastguard Worker 
1471*89c4ff92SAndroid Build Coastguard Worker     template <typename L>
1472*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_NOINLINE void unary_assert(assertType::Enum at, const char* file, int line,
1473*89c4ff92SAndroid Build Coastguard Worker                                        const char* expr, const DOCTEST_REF_WRAP(L) val) {
1474*89c4ff92SAndroid Build Coastguard Worker         bool failed = !val;
1475*89c4ff92SAndroid Build Coastguard Worker 
1476*89c4ff92SAndroid Build Coastguard Worker         if(at & assertType::is_false) //!OCLINT bitwise operator in conditional
1477*89c4ff92SAndroid Build Coastguard Worker             failed = !failed;
1478*89c4ff92SAndroid Build Coastguard Worker 
1479*89c4ff92SAndroid Build Coastguard Worker         // ###################################################################################
1480*89c4ff92SAndroid Build Coastguard Worker         // IF THE DEBUGGER BREAKS HERE - GO 1 LEVEL UP IN THE CALLSTACK FOR THE FAILING ASSERT
1481*89c4ff92SAndroid Build Coastguard Worker         // THIS IS THE EFFECT OF HAVING 'DOCTEST_CONFIG_SUPER_FAST_ASSERTS' DEFINED
1482*89c4ff92SAndroid Build Coastguard Worker         // ###################################################################################
1483*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_OUT_OF_TESTS(toString(val));
1484*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_IN_TESTS(toString(val));
1485*89c4ff92SAndroid Build Coastguard Worker     }
1486*89c4ff92SAndroid Build Coastguard Worker 
1487*89c4ff92SAndroid Build Coastguard Worker     struct DOCTEST_INTERFACE IExceptionTranslator
1488*89c4ff92SAndroid Build Coastguard Worker     {
1489*89c4ff92SAndroid Build Coastguard Worker         IExceptionTranslator();
1490*89c4ff92SAndroid Build Coastguard Worker         virtual ~IExceptionTranslator();
1491*89c4ff92SAndroid Build Coastguard Worker         virtual bool translate(String&) const = 0;
1492*89c4ff92SAndroid Build Coastguard Worker     };
1493*89c4ff92SAndroid Build Coastguard Worker 
1494*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
1495*89c4ff92SAndroid Build Coastguard Worker     class ExceptionTranslator : public IExceptionTranslator //!OCLINT destructor of virtual class
1496*89c4ff92SAndroid Build Coastguard Worker     {
1497*89c4ff92SAndroid Build Coastguard Worker     public:
1498*89c4ff92SAndroid Build Coastguard Worker         explicit ExceptionTranslator(String (*translateFunction)(T))
1499*89c4ff92SAndroid Build Coastguard Worker                 : m_translateFunction(translateFunction) {}
1500*89c4ff92SAndroid Build Coastguard Worker 
1501*89c4ff92SAndroid Build Coastguard Worker         bool translate(String& res) const override {
1502*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_EXCEPTIONS
1503*89c4ff92SAndroid Build Coastguard Worker             try {
1504*89c4ff92SAndroid Build Coastguard Worker                 throw; // lgtm [cpp/rethrow-no-exception]
1505*89c4ff92SAndroid Build Coastguard Worker                 // cppcheck-suppress catchExceptionByValue
1506*89c4ff92SAndroid Build Coastguard Worker             } catch(T ex) {                    // NOLINT
1507*89c4ff92SAndroid Build Coastguard Worker                 res = m_translateFunction(ex); //!OCLINT parameter reassignment
1508*89c4ff92SAndroid Build Coastguard Worker                 return true;
1509*89c4ff92SAndroid Build Coastguard Worker             } catch(...) {}         //!OCLINT -  empty catch statement
1510*89c4ff92SAndroid Build Coastguard Worker #endif                              // DOCTEST_CONFIG_NO_EXCEPTIONS
1511*89c4ff92SAndroid Build Coastguard Worker             static_cast<void>(res); // to silence -Wunused-parameter
1512*89c4ff92SAndroid Build Coastguard Worker             return false;
1513*89c4ff92SAndroid Build Coastguard Worker         }
1514*89c4ff92SAndroid Build Coastguard Worker 
1515*89c4ff92SAndroid Build Coastguard Worker     private:
1516*89c4ff92SAndroid Build Coastguard Worker         String (*m_translateFunction)(T);
1517*89c4ff92SAndroid Build Coastguard Worker     };
1518*89c4ff92SAndroid Build Coastguard Worker 
1519*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void registerExceptionTranslatorImpl(const IExceptionTranslator* et);
1520*89c4ff92SAndroid Build Coastguard Worker 
1521*89c4ff92SAndroid Build Coastguard Worker     template <bool C>
1522*89c4ff92SAndroid Build Coastguard Worker     struct StringStreamBase
1523*89c4ff92SAndroid Build Coastguard Worker     {
1524*89c4ff92SAndroid Build Coastguard Worker         template <typename T>
1525*89c4ff92SAndroid Build Coastguard Worker         static void convert(std::ostream* s, const T& in) {
1526*89c4ff92SAndroid Build Coastguard Worker             *s << toString(in);
1527*89c4ff92SAndroid Build Coastguard Worker         }
1528*89c4ff92SAndroid Build Coastguard Worker 
1529*89c4ff92SAndroid Build Coastguard Worker         // always treat char* as a string in this context - no matter
1530*89c4ff92SAndroid Build Coastguard Worker         // if DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING is defined
1531*89c4ff92SAndroid Build Coastguard Worker         static void convert(std::ostream* s, const char* in) { *s << String(in); }
1532*89c4ff92SAndroid Build Coastguard Worker     };
1533*89c4ff92SAndroid Build Coastguard Worker 
1534*89c4ff92SAndroid Build Coastguard Worker     template <>
1535*89c4ff92SAndroid Build Coastguard Worker     struct StringStreamBase<true>
1536*89c4ff92SAndroid Build Coastguard Worker     {
1537*89c4ff92SAndroid Build Coastguard Worker         template <typename T>
1538*89c4ff92SAndroid Build Coastguard Worker         static void convert(std::ostream* s, const T& in) {
1539*89c4ff92SAndroid Build Coastguard Worker             *s << in;
1540*89c4ff92SAndroid Build Coastguard Worker         }
1541*89c4ff92SAndroid Build Coastguard Worker     };
1542*89c4ff92SAndroid Build Coastguard Worker 
1543*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
1544*89c4ff92SAndroid Build Coastguard Worker     struct StringStream : public StringStreamBase<has_insertion_operator<T>::value>
1545*89c4ff92SAndroid Build Coastguard Worker     {};
1546*89c4ff92SAndroid Build Coastguard Worker 
1547*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
1548*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, const T& value) {
1549*89c4ff92SAndroid Build Coastguard Worker         StringStream<T>::convert(s, value);
1550*89c4ff92SAndroid Build Coastguard Worker     }
1551*89c4ff92SAndroid Build Coastguard Worker 
1552*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
1553*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, char* in);
1554*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, const char* in);
1555*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
1556*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, bool in);
1557*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, float in);
1558*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, double in);
1559*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, double long in);
1560*89c4ff92SAndroid Build Coastguard Worker 
1561*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, char in);
1562*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, char signed in);
1563*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, char unsigned in);
1564*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, int short in);
1565*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, int short unsigned in);
1566*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, int in);
1567*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, int unsigned in);
1568*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, int long in);
1569*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, int long unsigned in);
1570*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, int long long in);
1571*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void toStream(std::ostream* s, int long long unsigned in);
1572*89c4ff92SAndroid Build Coastguard Worker 
1573*89c4ff92SAndroid Build Coastguard Worker     // ContextScope base class used to allow implementing methods of ContextScope
1574*89c4ff92SAndroid Build Coastguard Worker     // that don't depend on the template parameter in doctest.cpp.
1575*89c4ff92SAndroid Build Coastguard Worker     class DOCTEST_INTERFACE ContextScopeBase : public IContextScope {
1576*89c4ff92SAndroid Build Coastguard Worker     protected:
1577*89c4ff92SAndroid Build Coastguard Worker         ContextScopeBase();
1578*89c4ff92SAndroid Build Coastguard Worker 
1579*89c4ff92SAndroid Build Coastguard Worker         void destroy();
1580*89c4ff92SAndroid Build Coastguard Worker     };
1581*89c4ff92SAndroid Build Coastguard Worker 
1582*89c4ff92SAndroid Build Coastguard Worker     template <typename L> class ContextScope : public ContextScopeBase
1583*89c4ff92SAndroid Build Coastguard Worker     {
1584*89c4ff92SAndroid Build Coastguard Worker         const L lambda_;
1585*89c4ff92SAndroid Build Coastguard Worker 
1586*89c4ff92SAndroid Build Coastguard Worker     public:
1587*89c4ff92SAndroid Build Coastguard Worker         explicit ContextScope(const L &lambda) : lambda_(lambda) {}
1588*89c4ff92SAndroid Build Coastguard Worker 
1589*89c4ff92SAndroid Build Coastguard Worker         ContextScope(ContextScope &&other) : lambda_(other.lambda_) {}
1590*89c4ff92SAndroid Build Coastguard Worker 
1591*89c4ff92SAndroid Build Coastguard Worker         void stringify(std::ostream* s) const override { lambda_(s); }
1592*89c4ff92SAndroid Build Coastguard Worker 
1593*89c4ff92SAndroid Build Coastguard Worker         ~ContextScope() override { destroy(); }
1594*89c4ff92SAndroid Build Coastguard Worker     };
1595*89c4ff92SAndroid Build Coastguard Worker 
1596*89c4ff92SAndroid Build Coastguard Worker     struct DOCTEST_INTERFACE MessageBuilder : public MessageData
1597*89c4ff92SAndroid Build Coastguard Worker     {
1598*89c4ff92SAndroid Build Coastguard Worker         std::ostream* m_stream;
1599*89c4ff92SAndroid Build Coastguard Worker 
1600*89c4ff92SAndroid Build Coastguard Worker         MessageBuilder(const char* file, int line, assertType::Enum severity);
1601*89c4ff92SAndroid Build Coastguard Worker         MessageBuilder() = delete;
1602*89c4ff92SAndroid Build Coastguard Worker         ~MessageBuilder();
1603*89c4ff92SAndroid Build Coastguard Worker 
1604*89c4ff92SAndroid Build Coastguard Worker         // the preferred way of chaining parameters for stringification
1605*89c4ff92SAndroid Build Coastguard Worker         template <typename T>
1606*89c4ff92SAndroid Build Coastguard Worker         MessageBuilder& operator,(const T& in) {
1607*89c4ff92SAndroid Build Coastguard Worker             toStream(m_stream, in);
1608*89c4ff92SAndroid Build Coastguard Worker             return *this;
1609*89c4ff92SAndroid Build Coastguard Worker         }
1610*89c4ff92SAndroid Build Coastguard Worker 
1611*89c4ff92SAndroid Build Coastguard Worker         // kept here just for backwards-compatibility - the comma operator should be preferred now
1612*89c4ff92SAndroid Build Coastguard Worker         template <typename T>
1613*89c4ff92SAndroid Build Coastguard Worker         MessageBuilder& operator<<(const T& in) { return this->operator,(in); }
1614*89c4ff92SAndroid Build Coastguard Worker 
1615*89c4ff92SAndroid Build Coastguard Worker         // the `,` operator has the lowest operator precedence - if `<<` is used by the user then
1616*89c4ff92SAndroid Build Coastguard Worker         // the `,` operator will be called last which is not what we want and thus the `*` operator
1617*89c4ff92SAndroid Build Coastguard Worker         // is used first (has higher operator precedence compared to `<<`) so that we guarantee that
1618*89c4ff92SAndroid Build Coastguard Worker         // an operator of the MessageBuilder class is called first before the rest of the parameters
1619*89c4ff92SAndroid Build Coastguard Worker         template <typename T>
1620*89c4ff92SAndroid Build Coastguard Worker         MessageBuilder& operator*(const T& in) { return this->operator,(in); }
1621*89c4ff92SAndroid Build Coastguard Worker 
1622*89c4ff92SAndroid Build Coastguard Worker         bool log();
1623*89c4ff92SAndroid Build Coastguard Worker         void react();
1624*89c4ff92SAndroid Build Coastguard Worker     };
1625*89c4ff92SAndroid Build Coastguard Worker 
1626*89c4ff92SAndroid Build Coastguard Worker     template <typename L>
1627*89c4ff92SAndroid Build Coastguard Worker     ContextScope<L> MakeContextScope(const L &lambda) {
1628*89c4ff92SAndroid Build Coastguard Worker         return ContextScope<L>(lambda);
1629*89c4ff92SAndroid Build Coastguard Worker     }
1630*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
1631*89c4ff92SAndroid Build Coastguard Worker 
1632*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_DEFINE_DECORATOR(name, type, def)                                                  \
1633*89c4ff92SAndroid Build Coastguard Worker     struct name                                                                                    \
1634*89c4ff92SAndroid Build Coastguard Worker     {                                                                                              \
1635*89c4ff92SAndroid Build Coastguard Worker         type data;                                                                                 \
1636*89c4ff92SAndroid Build Coastguard Worker         name(type in = def)                                                                        \
1637*89c4ff92SAndroid Build Coastguard Worker                 : data(in) {}                                                                      \
1638*89c4ff92SAndroid Build Coastguard Worker         void fill(detail::TestCase& state) const { state.DOCTEST_CAT(m_, name) = data; }           \
1639*89c4ff92SAndroid Build Coastguard Worker         void fill(detail::TestSuite& state) const { state.DOCTEST_CAT(m_, name) = data; }          \
1640*89c4ff92SAndroid Build Coastguard Worker     }
1641*89c4ff92SAndroid Build Coastguard Worker 
1642*89c4ff92SAndroid Build Coastguard Worker DOCTEST_DEFINE_DECORATOR(test_suite, const char*, "");
1643*89c4ff92SAndroid Build Coastguard Worker DOCTEST_DEFINE_DECORATOR(description, const char*, "");
1644*89c4ff92SAndroid Build Coastguard Worker DOCTEST_DEFINE_DECORATOR(skip, bool, true);
1645*89c4ff92SAndroid Build Coastguard Worker DOCTEST_DEFINE_DECORATOR(no_breaks, bool, true);
1646*89c4ff92SAndroid Build Coastguard Worker DOCTEST_DEFINE_DECORATOR(no_output, bool, true);
1647*89c4ff92SAndroid Build Coastguard Worker DOCTEST_DEFINE_DECORATOR(timeout, double, 0);
1648*89c4ff92SAndroid Build Coastguard Worker DOCTEST_DEFINE_DECORATOR(may_fail, bool, true);
1649*89c4ff92SAndroid Build Coastguard Worker DOCTEST_DEFINE_DECORATOR(should_fail, bool, true);
1650*89c4ff92SAndroid Build Coastguard Worker DOCTEST_DEFINE_DECORATOR(expected_failures, int, 0);
1651*89c4ff92SAndroid Build Coastguard Worker 
1652*89c4ff92SAndroid Build Coastguard Worker template <typename T>
1653*89c4ff92SAndroid Build Coastguard Worker int registerExceptionTranslator(String (*translateFunction)(T)) {
1654*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wexit-time-destructors")
1655*89c4ff92SAndroid Build Coastguard Worker     static detail::ExceptionTranslator<T> exceptionTranslator(translateFunction);
1656*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_POP
1657*89c4ff92SAndroid Build Coastguard Worker     detail::registerExceptionTranslatorImpl(&exceptionTranslator);
1658*89c4ff92SAndroid Build Coastguard Worker     return 0;
1659*89c4ff92SAndroid Build Coastguard Worker }
1660*89c4ff92SAndroid Build Coastguard Worker 
1661*89c4ff92SAndroid Build Coastguard Worker } // namespace doctest
1662*89c4ff92SAndroid Build Coastguard Worker 
1663*89c4ff92SAndroid Build Coastguard Worker // in a separate namespace outside of doctest because the DOCTEST_TEST_SUITE macro
1664*89c4ff92SAndroid Build Coastguard Worker // introduces an anonymous namespace in which getCurrentTestSuite gets overridden
1665*89c4ff92SAndroid Build Coastguard Worker namespace doctest_detail_test_suite_ns {
1666*89c4ff92SAndroid Build Coastguard Worker DOCTEST_INTERFACE doctest::detail::TestSuite& getCurrentTestSuite();
1667*89c4ff92SAndroid Build Coastguard Worker } // namespace doctest_detail_test_suite_ns
1668*89c4ff92SAndroid Build Coastguard Worker 
1669*89c4ff92SAndroid Build Coastguard Worker namespace doctest {
1670*89c4ff92SAndroid Build Coastguard Worker #else  // DOCTEST_CONFIG_DISABLE
1671*89c4ff92SAndroid Build Coastguard Worker template <typename T>
1672*89c4ff92SAndroid Build Coastguard Worker int registerExceptionTranslator(String (*)(T)) {
1673*89c4ff92SAndroid Build Coastguard Worker     return 0;
1674*89c4ff92SAndroid Build Coastguard Worker }
1675*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_DISABLE
1676*89c4ff92SAndroid Build Coastguard Worker 
1677*89c4ff92SAndroid Build Coastguard Worker namespace detail {
1678*89c4ff92SAndroid Build Coastguard Worker     typedef void (*assert_handler)(const AssertData&);
1679*89c4ff92SAndroid Build Coastguard Worker     struct ContextState;
1680*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
1681*89c4ff92SAndroid Build Coastguard Worker 
1682*89c4ff92SAndroid Build Coastguard Worker class DOCTEST_INTERFACE Context
1683*89c4ff92SAndroid Build Coastguard Worker {
1684*89c4ff92SAndroid Build Coastguard Worker     detail::ContextState* p;
1685*89c4ff92SAndroid Build Coastguard Worker 
1686*89c4ff92SAndroid Build Coastguard Worker     void parseArgs(int argc, const char* const* argv, bool withDefaults = false);
1687*89c4ff92SAndroid Build Coastguard Worker 
1688*89c4ff92SAndroid Build Coastguard Worker public:
1689*89c4ff92SAndroid Build Coastguard Worker     explicit Context(int argc = 0, const char* const* argv = nullptr);
1690*89c4ff92SAndroid Build Coastguard Worker 
1691*89c4ff92SAndroid Build Coastguard Worker     ~Context();
1692*89c4ff92SAndroid Build Coastguard Worker 
1693*89c4ff92SAndroid Build Coastguard Worker     void applyCommandLine(int argc, const char* const* argv);
1694*89c4ff92SAndroid Build Coastguard Worker 
1695*89c4ff92SAndroid Build Coastguard Worker     void addFilter(const char* filter, const char* value);
1696*89c4ff92SAndroid Build Coastguard Worker     void clearFilters();
1697*89c4ff92SAndroid Build Coastguard Worker     void setOption(const char* option, int value);
1698*89c4ff92SAndroid Build Coastguard Worker     void setOption(const char* option, const char* value);
1699*89c4ff92SAndroid Build Coastguard Worker 
1700*89c4ff92SAndroid Build Coastguard Worker     bool shouldExit();
1701*89c4ff92SAndroid Build Coastguard Worker 
1702*89c4ff92SAndroid Build Coastguard Worker     void setAsDefaultForAssertsOutOfTestCases();
1703*89c4ff92SAndroid Build Coastguard Worker 
1704*89c4ff92SAndroid Build Coastguard Worker     void setAssertHandler(detail::assert_handler ah);
1705*89c4ff92SAndroid Build Coastguard Worker 
1706*89c4ff92SAndroid Build Coastguard Worker     int run();
1707*89c4ff92SAndroid Build Coastguard Worker };
1708*89c4ff92SAndroid Build Coastguard Worker 
1709*89c4ff92SAndroid Build Coastguard Worker namespace TestCaseFailureReason {
1710*89c4ff92SAndroid Build Coastguard Worker     enum Enum
1711*89c4ff92SAndroid Build Coastguard Worker     {
1712*89c4ff92SAndroid Build Coastguard Worker         None                     = 0,
1713*89c4ff92SAndroid Build Coastguard Worker         AssertFailure            = 1,   // an assertion has failed in the test case
1714*89c4ff92SAndroid Build Coastguard Worker         Exception                = 2,   // test case threw an exception
1715*89c4ff92SAndroid Build Coastguard Worker         Crash                    = 4,   // a crash...
1716*89c4ff92SAndroid Build Coastguard Worker         TooManyFailedAsserts     = 8,   // the abort-after option
1717*89c4ff92SAndroid Build Coastguard Worker         Timeout                  = 16,  // see the timeout decorator
1718*89c4ff92SAndroid Build Coastguard Worker         ShouldHaveFailedButDidnt = 32,  // see the should_fail decorator
1719*89c4ff92SAndroid Build Coastguard Worker         ShouldHaveFailedAndDid   = 64,  // see the should_fail decorator
1720*89c4ff92SAndroid Build Coastguard Worker         DidntFailExactlyNumTimes = 128, // see the expected_failures decorator
1721*89c4ff92SAndroid Build Coastguard Worker         FailedExactlyNumTimes    = 256, // see the expected_failures decorator
1722*89c4ff92SAndroid Build Coastguard Worker         CouldHaveFailedAndDid    = 512  // see the may_fail decorator
1723*89c4ff92SAndroid Build Coastguard Worker     };
1724*89c4ff92SAndroid Build Coastguard Worker } // namespace TestCaseFailureReason
1725*89c4ff92SAndroid Build Coastguard Worker 
1726*89c4ff92SAndroid Build Coastguard Worker struct DOCTEST_INTERFACE CurrentTestCaseStats
1727*89c4ff92SAndroid Build Coastguard Worker {
1728*89c4ff92SAndroid Build Coastguard Worker     int    numAssertsCurrentTest;
1729*89c4ff92SAndroid Build Coastguard Worker     int    numAssertsFailedCurrentTest;
1730*89c4ff92SAndroid Build Coastguard Worker     double seconds;
1731*89c4ff92SAndroid Build Coastguard Worker     int    failure_flags; // use TestCaseFailureReason::Enum
1732*89c4ff92SAndroid Build Coastguard Worker };
1733*89c4ff92SAndroid Build Coastguard Worker 
1734*89c4ff92SAndroid Build Coastguard Worker struct DOCTEST_INTERFACE TestCaseException
1735*89c4ff92SAndroid Build Coastguard Worker {
1736*89c4ff92SAndroid Build Coastguard Worker     String error_string;
1737*89c4ff92SAndroid Build Coastguard Worker     bool   is_crash;
1738*89c4ff92SAndroid Build Coastguard Worker };
1739*89c4ff92SAndroid Build Coastguard Worker 
1740*89c4ff92SAndroid Build Coastguard Worker struct DOCTEST_INTERFACE TestRunStats
1741*89c4ff92SAndroid Build Coastguard Worker {
1742*89c4ff92SAndroid Build Coastguard Worker     unsigned numTestCases;
1743*89c4ff92SAndroid Build Coastguard Worker     unsigned numTestCasesPassingFilters;
1744*89c4ff92SAndroid Build Coastguard Worker     unsigned numTestSuitesPassingFilters;
1745*89c4ff92SAndroid Build Coastguard Worker     unsigned numTestCasesFailed;
1746*89c4ff92SAndroid Build Coastguard Worker     int      numAsserts;
1747*89c4ff92SAndroid Build Coastguard Worker     int      numAssertsFailed;
1748*89c4ff92SAndroid Build Coastguard Worker };
1749*89c4ff92SAndroid Build Coastguard Worker 
1750*89c4ff92SAndroid Build Coastguard Worker struct QueryData
1751*89c4ff92SAndroid Build Coastguard Worker {
1752*89c4ff92SAndroid Build Coastguard Worker     const TestRunStats*  run_stats = nullptr;
1753*89c4ff92SAndroid Build Coastguard Worker     const TestCaseData** data      = nullptr;
1754*89c4ff92SAndroid Build Coastguard Worker     unsigned             num_data  = 0;
1755*89c4ff92SAndroid Build Coastguard Worker };
1756*89c4ff92SAndroid Build Coastguard Worker 
1757*89c4ff92SAndroid Build Coastguard Worker struct DOCTEST_INTERFACE IReporter
1758*89c4ff92SAndroid Build Coastguard Worker {
1759*89c4ff92SAndroid Build Coastguard Worker     // The constructor has to accept "const ContextOptions&" as a single argument
1760*89c4ff92SAndroid Build Coastguard Worker     // which has most of the options for the run + a pointer to the stdout stream
1761*89c4ff92SAndroid Build Coastguard Worker     // Reporter(const ContextOptions& in)
1762*89c4ff92SAndroid Build Coastguard Worker 
1763*89c4ff92SAndroid Build Coastguard Worker     // called when a query should be reported (listing test cases, printing the version, etc.)
1764*89c4ff92SAndroid Build Coastguard Worker     virtual void report_query(const QueryData&) = 0;
1765*89c4ff92SAndroid Build Coastguard Worker 
1766*89c4ff92SAndroid Build Coastguard Worker     // called when the whole test run starts
1767*89c4ff92SAndroid Build Coastguard Worker     virtual void test_run_start() = 0;
1768*89c4ff92SAndroid Build Coastguard Worker     // called when the whole test run ends (caching a pointer to the input doesn't make sense here)
1769*89c4ff92SAndroid Build Coastguard Worker     virtual void test_run_end(const TestRunStats&) = 0;
1770*89c4ff92SAndroid Build Coastguard Worker 
1771*89c4ff92SAndroid Build Coastguard Worker     // called when a test case is started (safe to cache a pointer to the input)
1772*89c4ff92SAndroid Build Coastguard Worker     virtual void test_case_start(const TestCaseData&) = 0;
1773*89c4ff92SAndroid Build Coastguard Worker     // called when a test case is reentered because of unfinished subcases (safe to cache a pointer to the input)
1774*89c4ff92SAndroid Build Coastguard Worker     virtual void test_case_reenter(const TestCaseData&) = 0;
1775*89c4ff92SAndroid Build Coastguard Worker     // called when a test case has ended
1776*89c4ff92SAndroid Build Coastguard Worker     virtual void test_case_end(const CurrentTestCaseStats&) = 0;
1777*89c4ff92SAndroid Build Coastguard Worker 
1778*89c4ff92SAndroid Build Coastguard Worker     // called when an exception is thrown from the test case (or it crashes)
1779*89c4ff92SAndroid Build Coastguard Worker     virtual void test_case_exception(const TestCaseException&) = 0;
1780*89c4ff92SAndroid Build Coastguard Worker 
1781*89c4ff92SAndroid Build Coastguard Worker     // called whenever a subcase is entered (don't cache pointers to the input)
1782*89c4ff92SAndroid Build Coastguard Worker     virtual void subcase_start(const SubcaseSignature&) = 0;
1783*89c4ff92SAndroid Build Coastguard Worker     // called whenever a subcase is exited (don't cache pointers to the input)
1784*89c4ff92SAndroid Build Coastguard Worker     virtual void subcase_end() = 0;
1785*89c4ff92SAndroid Build Coastguard Worker 
1786*89c4ff92SAndroid Build Coastguard Worker     // called for each assert (don't cache pointers to the input)
1787*89c4ff92SAndroid Build Coastguard Worker     virtual void log_assert(const AssertData&) = 0;
1788*89c4ff92SAndroid Build Coastguard Worker     // called for each message (don't cache pointers to the input)
1789*89c4ff92SAndroid Build Coastguard Worker     virtual void log_message(const MessageData&) = 0;
1790*89c4ff92SAndroid Build Coastguard Worker 
1791*89c4ff92SAndroid Build Coastguard Worker     // called when a test case is skipped either because it doesn't pass the filters, has a skip decorator
1792*89c4ff92SAndroid Build Coastguard Worker     // or isn't in the execution range (between first and last) (safe to cache a pointer to the input)
1793*89c4ff92SAndroid Build Coastguard Worker     virtual void test_case_skipped(const TestCaseData&) = 0;
1794*89c4ff92SAndroid Build Coastguard Worker 
1795*89c4ff92SAndroid Build Coastguard Worker     // doctest will not be managing the lifetimes of reporters given to it but this would still be nice to have
1796*89c4ff92SAndroid Build Coastguard Worker     virtual ~IReporter();
1797*89c4ff92SAndroid Build Coastguard Worker 
1798*89c4ff92SAndroid Build Coastguard Worker     // can obtain all currently active contexts and stringify them if one wishes to do so
1799*89c4ff92SAndroid Build Coastguard Worker     static int                         get_num_active_contexts();
1800*89c4ff92SAndroid Build Coastguard Worker     static const IContextScope* const* get_active_contexts();
1801*89c4ff92SAndroid Build Coastguard Worker 
1802*89c4ff92SAndroid Build Coastguard Worker     // can iterate through contexts which have been stringified automatically in their destructors when an exception has been thrown
1803*89c4ff92SAndroid Build Coastguard Worker     static int           get_num_stringified_contexts();
1804*89c4ff92SAndroid Build Coastguard Worker     static const String* get_stringified_contexts();
1805*89c4ff92SAndroid Build Coastguard Worker };
1806*89c4ff92SAndroid Build Coastguard Worker 
1807*89c4ff92SAndroid Build Coastguard Worker namespace detail {
1808*89c4ff92SAndroid Build Coastguard Worker     typedef IReporter* (*reporterCreatorFunc)(const ContextOptions&);
1809*89c4ff92SAndroid Build Coastguard Worker 
1810*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INTERFACE void registerReporterImpl(const char* name, int prio, reporterCreatorFunc c, bool isReporter);
1811*89c4ff92SAndroid Build Coastguard Worker 
1812*89c4ff92SAndroid Build Coastguard Worker     template <typename Reporter>
1813*89c4ff92SAndroid Build Coastguard Worker     IReporter* reporterCreator(const ContextOptions& o) {
1814*89c4ff92SAndroid Build Coastguard Worker         return new Reporter(o);
1815*89c4ff92SAndroid Build Coastguard Worker     }
1816*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
1817*89c4ff92SAndroid Build Coastguard Worker 
1818*89c4ff92SAndroid Build Coastguard Worker template <typename Reporter>
1819*89c4ff92SAndroid Build Coastguard Worker int registerReporter(const char* name, int priority, bool isReporter) {
1820*89c4ff92SAndroid Build Coastguard Worker     detail::registerReporterImpl(name, priority, detail::reporterCreator<Reporter>, isReporter);
1821*89c4ff92SAndroid Build Coastguard Worker     return 0;
1822*89c4ff92SAndroid Build Coastguard Worker }
1823*89c4ff92SAndroid Build Coastguard Worker } // namespace doctest
1824*89c4ff92SAndroid Build Coastguard Worker 
1825*89c4ff92SAndroid Build Coastguard Worker // if registering is not disabled
1826*89c4ff92SAndroid Build Coastguard Worker #if !defined(DOCTEST_CONFIG_DISABLE)
1827*89c4ff92SAndroid Build Coastguard Worker 
1828*89c4ff92SAndroid Build Coastguard Worker // common code in asserts - for convenience
1829*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ASSERT_LOG_AND_REACT(b)                                                            \
1830*89c4ff92SAndroid Build Coastguard Worker     if(b.log())                                                                                    \
1831*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_BREAK_INTO_DEBUGGER();                                                             \
1832*89c4ff92SAndroid Build Coastguard Worker     b.react()
1833*89c4ff92SAndroid Build Coastguard Worker 
1834*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS
1835*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WRAP_IN_TRY(x) x;
1836*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS
1837*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WRAP_IN_TRY(x)                                                                     \
1838*89c4ff92SAndroid Build Coastguard Worker     try {                                                                                          \
1839*89c4ff92SAndroid Build Coastguard Worker         x;                                                                                         \
1840*89c4ff92SAndroid Build Coastguard Worker     } catch(...) { _DOCTEST_RB.translateException(); }
1841*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_TRY_CATCH_IN_ASSERTS
1842*89c4ff92SAndroid Build Coastguard Worker 
1843*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_VOID_CAST_EXPRESSIONS
1844*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CAST_TO_VOID(...)                                                                  \
1845*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wuseless-cast")                                       \
1846*89c4ff92SAndroid Build Coastguard Worker     static_cast<void>(__VA_ARGS__);                                                                \
1847*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING_POP
1848*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_VOID_CAST_EXPRESSIONS
1849*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CAST_TO_VOID(...) __VA_ARGS__;
1850*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_VOID_CAST_EXPRESSIONS
1851*89c4ff92SAndroid Build Coastguard Worker 
1852*89c4ff92SAndroid Build Coastguard Worker // registers the test by initializing a dummy var with a function
1853*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REGISTER_FUNCTION(global_prefix, f, decorators)                                    \
1854*89c4ff92SAndroid Build Coastguard Worker     global_prefix DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_VAR_)) =              \
1855*89c4ff92SAndroid Build Coastguard Worker             doctest::detail::regTest(                                                              \
1856*89c4ff92SAndroid Build Coastguard Worker                     doctest::detail::TestCase(                                                     \
1857*89c4ff92SAndroid Build Coastguard Worker                             f, __FILE__, __LINE__,                                                 \
1858*89c4ff92SAndroid Build Coastguard Worker                             doctest_detail_test_suite_ns::getCurrentTestSuite()) *                 \
1859*89c4ff92SAndroid Build Coastguard Worker                     decorators);                                                                   \
1860*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS_END()
1861*89c4ff92SAndroid Build Coastguard Worker 
1862*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_IMPLEMENT_FIXTURE(der, base, func, decorators)                                     \
1863*89c4ff92SAndroid Build Coastguard Worker     namespace {                                                                                    \
1864*89c4ff92SAndroid Build Coastguard Worker         struct der : public base                                                                   \
1865*89c4ff92SAndroid Build Coastguard Worker         {                                                                                          \
1866*89c4ff92SAndroid Build Coastguard Worker             void f();                                                                              \
1867*89c4ff92SAndroid Build Coastguard Worker         };                                                                                         \
1868*89c4ff92SAndroid Build Coastguard Worker         static void func() {                                                                       \
1869*89c4ff92SAndroid Build Coastguard Worker             der v;                                                                                 \
1870*89c4ff92SAndroid Build Coastguard Worker             v.f();                                                                                 \
1871*89c4ff92SAndroid Build Coastguard Worker         }                                                                                          \
1872*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_REGISTER_FUNCTION(DOCTEST_EMPTY, func, decorators)                                 \
1873*89c4ff92SAndroid Build Coastguard Worker     }                                                                                              \
1874*89c4ff92SAndroid Build Coastguard Worker     inline DOCTEST_NOINLINE void der::f()
1875*89c4ff92SAndroid Build Coastguard Worker 
1876*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CREATE_AND_REGISTER_FUNCTION(f, decorators)                                        \
1877*89c4ff92SAndroid Build Coastguard Worker     static void f();                                                                               \
1878*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_REGISTER_FUNCTION(DOCTEST_EMPTY, f, decorators)                                        \
1879*89c4ff92SAndroid Build Coastguard Worker     static void f()
1880*89c4ff92SAndroid Build Coastguard Worker 
1881*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CREATE_AND_REGISTER_FUNCTION_IN_CLASS(f, proxy, decorators)                        \
1882*89c4ff92SAndroid Build Coastguard Worker     static doctest::detail::funcType proxy() { return f; }                                         \
1883*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_REGISTER_FUNCTION(inline const, proxy(), decorators)                                   \
1884*89c4ff92SAndroid Build Coastguard Worker     static void f()
1885*89c4ff92SAndroid Build Coastguard Worker 
1886*89c4ff92SAndroid Build Coastguard Worker // for registering tests
1887*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE(decorators)                                                              \
1888*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CREATE_AND_REGISTER_FUNCTION(DOCTEST_ANONYMOUS(_DOCTEST_ANON_FUNC_), decorators)
1889*89c4ff92SAndroid Build Coastguard Worker 
1890*89c4ff92SAndroid Build Coastguard Worker // for registering tests in classes - requires C++17 for inline variables!
1891*89c4ff92SAndroid Build Coastguard Worker #if __cplusplus >= 201703L || (DOCTEST_MSVC >= DOCTEST_COMPILER(19, 12, 0) && _MSVC_LANG >= 201703L)
1892*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_CLASS(decorators)                                                        \
1893*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CREATE_AND_REGISTER_FUNCTION_IN_CLASS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_FUNC_),          \
1894*89c4ff92SAndroid Build Coastguard Worker                                                   DOCTEST_ANONYMOUS(_DOCTEST_ANON_PROXY_),         \
1895*89c4ff92SAndroid Build Coastguard Worker                                                   decorators)
1896*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_TEST_CASE_CLASS
1897*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_CLASS(...)                                                               \
1898*89c4ff92SAndroid Build Coastguard Worker     TEST_CASES_CAN_BE_REGISTERED_IN_CLASSES_ONLY_IN_CPP17_MODE_OR_WITH_VS_2017_OR_NEWER
1899*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_TEST_CASE_CLASS
1900*89c4ff92SAndroid Build Coastguard Worker 
1901*89c4ff92SAndroid Build Coastguard Worker // for registering tests with a fixture
1902*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_FIXTURE(c, decorators)                                                   \
1903*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_IMPLEMENT_FIXTURE(DOCTEST_ANONYMOUS(_DOCTEST_ANON_CLASS_), c,                          \
1904*89c4ff92SAndroid Build Coastguard Worker                               DOCTEST_ANONYMOUS(_DOCTEST_ANON_FUNC_), decorators)
1905*89c4ff92SAndroid Build Coastguard Worker 
1906*89c4ff92SAndroid Build Coastguard Worker // for converting types to strings without the <typeinfo> header and demangling
1907*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TYPE_TO_STRING_IMPL(...)                                                           \
1908*89c4ff92SAndroid Build Coastguard Worker     template <>                                                                                    \
1909*89c4ff92SAndroid Build Coastguard Worker     inline const char* type_to_string<__VA_ARGS__>() {                                             \
1910*89c4ff92SAndroid Build Coastguard Worker         return "<" #__VA_ARGS__ ">";                                                               \
1911*89c4ff92SAndroid Build Coastguard Worker     }
1912*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TYPE_TO_STRING(...)                                                                \
1913*89c4ff92SAndroid Build Coastguard Worker     namespace doctest { namespace detail {                                                         \
1914*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_TYPE_TO_STRING_IMPL(__VA_ARGS__)                                               \
1915*89c4ff92SAndroid Build Coastguard Worker         }                                                                                          \
1916*89c4ff92SAndroid Build Coastguard Worker     }                                                                                              \
1917*89c4ff92SAndroid Build Coastguard Worker     typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
1918*89c4ff92SAndroid Build Coastguard Worker 
1919*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE_DEFINE_IMPL(dec, T, iter, func)                                 \
1920*89c4ff92SAndroid Build Coastguard Worker     template <typename T>                                                                          \
1921*89c4ff92SAndroid Build Coastguard Worker     static void func();                                                                            \
1922*89c4ff92SAndroid Build Coastguard Worker     namespace {                                                                                    \
1923*89c4ff92SAndroid Build Coastguard Worker         template <typename Tuple>                                                                  \
1924*89c4ff92SAndroid Build Coastguard Worker         struct iter;                                                                               \
1925*89c4ff92SAndroid Build Coastguard Worker         template <typename Type, typename... Rest>                                                 \
1926*89c4ff92SAndroid Build Coastguard Worker         struct iter<std::tuple<Type, Rest...>>                                                     \
1927*89c4ff92SAndroid Build Coastguard Worker         {                                                                                          \
1928*89c4ff92SAndroid Build Coastguard Worker             iter(const char* file, unsigned line, int index) {                                     \
1929*89c4ff92SAndroid Build Coastguard Worker                 doctest::detail::regTest(doctest::detail::TestCase(func<Type>, file, line,         \
1930*89c4ff92SAndroid Build Coastguard Worker                                             doctest_detail_test_suite_ns::getCurrentTestSuite(),   \
1931*89c4ff92SAndroid Build Coastguard Worker                                             doctest::detail::type_to_string<Type>(),               \
1932*89c4ff92SAndroid Build Coastguard Worker                                             int(line) * 1000 + index)                              \
1933*89c4ff92SAndroid Build Coastguard Worker                                          * dec);                                                   \
1934*89c4ff92SAndroid Build Coastguard Worker                 iter<std::tuple<Rest...>>(file, line, index + 1);                                  \
1935*89c4ff92SAndroid Build Coastguard Worker             }                                                                                      \
1936*89c4ff92SAndroid Build Coastguard Worker         };                                                                                         \
1937*89c4ff92SAndroid Build Coastguard Worker         template <>                                                                                \
1938*89c4ff92SAndroid Build Coastguard Worker         struct iter<std::tuple<>>                                                                  \
1939*89c4ff92SAndroid Build Coastguard Worker         {                                                                                          \
1940*89c4ff92SAndroid Build Coastguard Worker             iter(const char*, unsigned, int) {}                                                    \
1941*89c4ff92SAndroid Build Coastguard Worker         };                                                                                         \
1942*89c4ff92SAndroid Build Coastguard Worker     }                                                                                              \
1943*89c4ff92SAndroid Build Coastguard Worker     template <typename T>                                                                          \
1944*89c4ff92SAndroid Build Coastguard Worker     static void func()
1945*89c4ff92SAndroid Build Coastguard Worker 
1946*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE_DEFINE(dec, T, id)                                              \
1947*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TEST_CASE_TEMPLATE_DEFINE_IMPL(dec, T, DOCTEST_CAT(id, ITERATOR),                      \
1948*89c4ff92SAndroid Build Coastguard Worker                                            DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_))
1949*89c4ff92SAndroid Build Coastguard Worker 
1950*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE_IMPL(id, anon, ...)                                 \
1951*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_CAT(anon, DUMMY)) =                                         \
1952*89c4ff92SAndroid Build Coastguard Worker         doctest::detail::instantiationHelper(DOCTEST_CAT(id, ITERATOR)<__VA_ARGS__>(__FILE__, __LINE__, 0));\
1953*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS_END()
1954*89c4ff92SAndroid Build Coastguard Worker 
1955*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE_INVOKE(id, ...)                                                 \
1956*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE_IMPL(id, DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_), std::tuple<__VA_ARGS__>) \
1957*89c4ff92SAndroid Build Coastguard Worker     typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
1958*89c4ff92SAndroid Build Coastguard Worker 
1959*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE_APPLY(id, ...)                                                  \
1960*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE_IMPL(id, DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_), __VA_ARGS__) \
1961*89c4ff92SAndroid Build Coastguard Worker     typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
1962*89c4ff92SAndroid Build Coastguard Worker 
1963*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE_IMPL(dec, T, anon, ...)                                         \
1964*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TEST_CASE_TEMPLATE_DEFINE_IMPL(dec, T, DOCTEST_CAT(anon, ITERATOR), anon);             \
1965*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE_IMPL(anon, anon, std::tuple<__VA_ARGS__>)               \
1966*89c4ff92SAndroid Build Coastguard Worker     template <typename T>                                                                          \
1967*89c4ff92SAndroid Build Coastguard Worker     static void anon()
1968*89c4ff92SAndroid Build Coastguard Worker 
1969*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE(dec, T, ...)                                                    \
1970*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TEST_CASE_TEMPLATE_IMPL(dec, T, DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_), __VA_ARGS__)
1971*89c4ff92SAndroid Build Coastguard Worker 
1972*89c4ff92SAndroid Build Coastguard Worker // for subcases
1973*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SUBCASE(name)                                                                      \
1974*89c4ff92SAndroid Build Coastguard Worker     if(const doctest::detail::Subcase & DOCTEST_ANONYMOUS(_DOCTEST_ANON_SUBCASE_) DOCTEST_UNUSED = \
1975*89c4ff92SAndroid Build Coastguard Worker                doctest::detail::Subcase(name, __FILE__, __LINE__))
1976*89c4ff92SAndroid Build Coastguard Worker 
1977*89c4ff92SAndroid Build Coastguard Worker // for grouping tests in test suites by using code blocks
1978*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_SUITE_IMPL(decorators, ns_name)                                               \
1979*89c4ff92SAndroid Build Coastguard Worker     namespace ns_name { namespace doctest_detail_test_suite_ns {                                   \
1980*89c4ff92SAndroid Build Coastguard Worker             static DOCTEST_NOINLINE doctest::detail::TestSuite& getCurrentTestSuite() {            \
1981*89c4ff92SAndroid Build Coastguard Worker                 DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4640)                                      \
1982*89c4ff92SAndroid Build Coastguard Worker                 DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wexit-time-destructors")                \
1983*89c4ff92SAndroid Build Coastguard Worker                 DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wmissing-field-initializers")             \
1984*89c4ff92SAndroid Build Coastguard Worker                 static doctest::detail::TestSuite data{};                                          \
1985*89c4ff92SAndroid Build Coastguard Worker                 static bool                       inited = false;                                  \
1986*89c4ff92SAndroid Build Coastguard Worker                 DOCTEST_MSVC_SUPPRESS_WARNING_POP                                                  \
1987*89c4ff92SAndroid Build Coastguard Worker                 DOCTEST_CLANG_SUPPRESS_WARNING_POP                                                 \
1988*89c4ff92SAndroid Build Coastguard Worker                 DOCTEST_GCC_SUPPRESS_WARNING_POP                                                   \
1989*89c4ff92SAndroid Build Coastguard Worker                 if(!inited) {                                                                      \
1990*89c4ff92SAndroid Build Coastguard Worker                     data* decorators;                                                              \
1991*89c4ff92SAndroid Build Coastguard Worker                     inited = true;                                                                 \
1992*89c4ff92SAndroid Build Coastguard Worker                 }                                                                                  \
1993*89c4ff92SAndroid Build Coastguard Worker                 return data;                                                                       \
1994*89c4ff92SAndroid Build Coastguard Worker             }                                                                                      \
1995*89c4ff92SAndroid Build Coastguard Worker         }                                                                                          \
1996*89c4ff92SAndroid Build Coastguard Worker     }                                                                                              \
1997*89c4ff92SAndroid Build Coastguard Worker     namespace ns_name
1998*89c4ff92SAndroid Build Coastguard Worker 
1999*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_SUITE(decorators)                                                             \
2000*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TEST_SUITE_IMPL(decorators, DOCTEST_ANONYMOUS(_DOCTEST_ANON_SUITE_))
2001*89c4ff92SAndroid Build Coastguard Worker 
2002*89c4ff92SAndroid Build Coastguard Worker // for starting a testsuite block
2003*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_SUITE_BEGIN(decorators)                                                       \
2004*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_VAR_)) =                            \
2005*89c4ff92SAndroid Build Coastguard Worker             doctest::detail::setTestSuite(doctest::detail::TestSuite() * decorators);              \
2006*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS_END()                                                               \
2007*89c4ff92SAndroid Build Coastguard Worker     typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
2008*89c4ff92SAndroid Build Coastguard Worker 
2009*89c4ff92SAndroid Build Coastguard Worker // for ending a testsuite block
2010*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_SUITE_END                                                                     \
2011*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_VAR_)) =                            \
2012*89c4ff92SAndroid Build Coastguard Worker             doctest::detail::setTestSuite(doctest::detail::TestSuite() * "");                      \
2013*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS_END()                                                               \
2014*89c4ff92SAndroid Build Coastguard Worker     typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
2015*89c4ff92SAndroid Build Coastguard Worker 
2016*89c4ff92SAndroid Build Coastguard Worker // for registering exception translators
2017*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REGISTER_EXCEPTION_TRANSLATOR_IMPL(translatorName, signature)                      \
2018*89c4ff92SAndroid Build Coastguard Worker     inline doctest::String translatorName(signature);                                              \
2019*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_TRANSLATOR_)) =                     \
2020*89c4ff92SAndroid Build Coastguard Worker             doctest::registerExceptionTranslator(translatorName);                                  \
2021*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS_END()                                                               \
2022*89c4ff92SAndroid Build Coastguard Worker     doctest::String translatorName(signature)
2023*89c4ff92SAndroid Build Coastguard Worker 
2024*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REGISTER_EXCEPTION_TRANSLATOR(signature)                                           \
2025*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_REGISTER_EXCEPTION_TRANSLATOR_IMPL(DOCTEST_ANONYMOUS(_DOCTEST_ANON_TRANSLATOR_),       \
2026*89c4ff92SAndroid Build Coastguard Worker                                                signature)
2027*89c4ff92SAndroid Build Coastguard Worker 
2028*89c4ff92SAndroid Build Coastguard Worker // for registering reporters
2029*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REGISTER_REPORTER(name, priority, reporter)                                        \
2030*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_REPORTER_)) =                       \
2031*89c4ff92SAndroid Build Coastguard Worker             doctest::registerReporter<reporter>(name, priority, true);                             \
2032*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS_END() typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
2033*89c4ff92SAndroid Build Coastguard Worker 
2034*89c4ff92SAndroid Build Coastguard Worker // for registering listeners
2035*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REGISTER_LISTENER(name, priority, reporter)                                        \
2036*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS(DOCTEST_ANONYMOUS(_DOCTEST_ANON_REPORTER_)) =                       \
2037*89c4ff92SAndroid Build Coastguard Worker             doctest::registerReporter<reporter>(name, priority, false);                            \
2038*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GLOBAL_NO_WARNINGS_END() typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
2039*89c4ff92SAndroid Build Coastguard Worker 
2040*89c4ff92SAndroid Build Coastguard Worker // for logging
2041*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_INFO(...)                                                                          \
2042*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_INFO_IMPL(DOCTEST_ANONYMOUS(_DOCTEST_CAPTURE_), DOCTEST_ANONYMOUS(_DOCTEST_CAPTURE_),  \
2043*89c4ff92SAndroid Build Coastguard Worker                       __VA_ARGS__)
2044*89c4ff92SAndroid Build Coastguard Worker 
2045*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_INFO_IMPL(mb_name, s_name, ...)                                       \
2046*89c4ff92SAndroid Build Coastguard Worker     auto DOCTEST_ANONYMOUS(_DOCTEST_CAPTURE_) = doctest::detail::MakeContextScope(                 \
2047*89c4ff92SAndroid Build Coastguard Worker         [&](std::ostream* s_name) {                                                                \
2048*89c4ff92SAndroid Build Coastguard Worker         doctest::detail::MessageBuilder mb_name(__FILE__, __LINE__, doctest::assertType::is_warn); \
2049*89c4ff92SAndroid Build Coastguard Worker         mb_name.m_stream = s_name;                                                                 \
2050*89c4ff92SAndroid Build Coastguard Worker         mb_name * __VA_ARGS__;                                                                     \
2051*89c4ff92SAndroid Build Coastguard Worker     })
2052*89c4ff92SAndroid Build Coastguard Worker 
2053*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CAPTURE(x) DOCTEST_INFO(#x " := ", x)
2054*89c4ff92SAndroid Build Coastguard Worker 
2055*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ADD_AT_IMPL(type, file, line, mb, ...)                                             \
2056*89c4ff92SAndroid Build Coastguard Worker     do {                                                                                           \
2057*89c4ff92SAndroid Build Coastguard Worker         doctest::detail::MessageBuilder mb(file, line, doctest::assertType::type);                 \
2058*89c4ff92SAndroid Build Coastguard Worker         mb * __VA_ARGS__;                                                                          \
2059*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_LOG_AND_REACT(mb);                                                          \
2060*89c4ff92SAndroid Build Coastguard Worker     } while(false)
2061*89c4ff92SAndroid Build Coastguard Worker 
2062*89c4ff92SAndroid Build Coastguard Worker // clang-format off
2063*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ADD_MESSAGE_AT(file, line, ...) DOCTEST_ADD_AT_IMPL(is_warn, file, line, DOCTEST_ANONYMOUS(_DOCTEST_MESSAGE_), __VA_ARGS__)
2064*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ADD_FAIL_CHECK_AT(file, line, ...) DOCTEST_ADD_AT_IMPL(is_check, file, line, DOCTEST_ANONYMOUS(_DOCTEST_MESSAGE_), __VA_ARGS__)
2065*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ADD_FAIL_AT(file, line, ...) DOCTEST_ADD_AT_IMPL(is_require, file, line, DOCTEST_ANONYMOUS(_DOCTEST_MESSAGE_), __VA_ARGS__)
2066*89c4ff92SAndroid Build Coastguard Worker // clang-format on
2067*89c4ff92SAndroid Build Coastguard Worker 
2068*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MESSAGE(...) DOCTEST_ADD_MESSAGE_AT(__FILE__, __LINE__, __VA_ARGS__)
2069*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAIL_CHECK(...) DOCTEST_ADD_FAIL_CHECK_AT(__FILE__, __LINE__, __VA_ARGS__)
2070*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAIL(...) DOCTEST_ADD_FAIL_AT(__FILE__, __LINE__, __VA_ARGS__)
2071*89c4ff92SAndroid Build Coastguard Worker 
2072*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TO_LVALUE(...) __VA_ARGS__ // Not removed to keep backwards compatibility.
2073*89c4ff92SAndroid Build Coastguard Worker 
2074*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_SUPER_FAST_ASSERTS
2075*89c4ff92SAndroid Build Coastguard Worker 
2076*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ASSERT_IMPLEMENT_2(assert_type, ...)                                               \
2077*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Woverloaded-shift-op-parentheses")                  \
2078*89c4ff92SAndroid Build Coastguard Worker     doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__,         \
2079*89c4ff92SAndroid Build Coastguard Worker                                                __LINE__, #__VA_ARGS__);                            \
2080*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_WRAP_IN_TRY(_DOCTEST_RB.setResult(                                                     \
2081*89c4ff92SAndroid Build Coastguard Worker             doctest::detail::ExpressionDecomposer(doctest::assertType::assert_type)                \
2082*89c4ff92SAndroid Build Coastguard Worker             << __VA_ARGS__))                                                                       \
2083*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB)                                                      \
2084*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_POP
2085*89c4ff92SAndroid Build Coastguard Worker 
2086*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ASSERT_IMPLEMENT_1(assert_type, ...)                                               \
2087*89c4ff92SAndroid Build Coastguard Worker     do {                                                                                           \
2088*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_IMPLEMENT_2(assert_type, __VA_ARGS__);                                      \
2089*89c4ff92SAndroid Build Coastguard Worker     } while(false)
2090*89c4ff92SAndroid Build Coastguard Worker 
2091*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_SUPER_FAST_ASSERTS
2092*89c4ff92SAndroid Build Coastguard Worker 
2093*89c4ff92SAndroid Build Coastguard Worker // necessary for <ASSERT>_MESSAGE
2094*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ASSERT_IMPLEMENT_2 DOCTEST_ASSERT_IMPLEMENT_1
2095*89c4ff92SAndroid Build Coastguard Worker 
2096*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ASSERT_IMPLEMENT_1(assert_type, ...)                                               \
2097*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Woverloaded-shift-op-parentheses")                  \
2098*89c4ff92SAndroid Build Coastguard Worker     doctest::detail::decomp_assert(                                                                \
2099*89c4ff92SAndroid Build Coastguard Worker             doctest::assertType::assert_type, __FILE__, __LINE__, #__VA_ARGS__,                    \
2100*89c4ff92SAndroid Build Coastguard Worker             doctest::detail::ExpressionDecomposer(doctest::assertType::assert_type)                \
2101*89c4ff92SAndroid Build Coastguard Worker                     << __VA_ARGS__) DOCTEST_CLANG_SUPPRESS_WARNING_POP
2102*89c4ff92SAndroid Build Coastguard Worker 
2103*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_SUPER_FAST_ASSERTS
2104*89c4ff92SAndroid Build Coastguard Worker 
2105*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_WARN, __VA_ARGS__)
2106*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_CHECK, __VA_ARGS__)
2107*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_REQUIRE, __VA_ARGS__)
2108*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_FALSE(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_WARN_FALSE, __VA_ARGS__)
2109*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_FALSE(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_CHECK_FALSE, __VA_ARGS__)
2110*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_FALSE(...) DOCTEST_ASSERT_IMPLEMENT_1(DT_REQUIRE_FALSE, __VA_ARGS__)
2111*89c4ff92SAndroid Build Coastguard Worker 
2112*89c4ff92SAndroid Build Coastguard Worker // clang-format off
2113*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_MESSAGE(cond, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_ASSERT_IMPLEMENT_2(DT_WARN, cond); } while(false)
2114*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_MESSAGE(cond, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_ASSERT_IMPLEMENT_2(DT_CHECK, cond); } while(false)
2115*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_MESSAGE(cond, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_ASSERT_IMPLEMENT_2(DT_REQUIRE, cond); } while(false)
2116*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_FALSE_MESSAGE(cond, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_ASSERT_IMPLEMENT_2(DT_WARN_FALSE, cond); } while(false)
2117*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_FALSE_MESSAGE(cond, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_ASSERT_IMPLEMENT_2(DT_CHECK_FALSE, cond); } while(false)
2118*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_FALSE_MESSAGE(cond, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_ASSERT_IMPLEMENT_2(DT_REQUIRE_FALSE, cond); } while(false)
2119*89c4ff92SAndroid Build Coastguard Worker // clang-format on
2120*89c4ff92SAndroid Build Coastguard Worker 
2121*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ASSERT_THROWS_AS(expr, assert_type, message, ...)                                  \
2122*89c4ff92SAndroid Build Coastguard Worker     do {                                                                                           \
2123*89c4ff92SAndroid Build Coastguard Worker         if(!doctest::getContextOptions()->no_throw) {                                              \
2124*89c4ff92SAndroid Build Coastguard Worker             doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__, \
2125*89c4ff92SAndroid Build Coastguard Worker                                                        __LINE__, #expr, #__VA_ARGS__, message);    \
2126*89c4ff92SAndroid Build Coastguard Worker             try {                                                                                  \
2127*89c4ff92SAndroid Build Coastguard Worker                 DOCTEST_CAST_TO_VOID(expr)                                                         \
2128*89c4ff92SAndroid Build Coastguard Worker             } catch(const typename doctest::detail::remove_const<                                  \
2129*89c4ff92SAndroid Build Coastguard Worker                     typename doctest::detail::remove_reference<__VA_ARGS__>::type>::type&) {       \
2130*89c4ff92SAndroid Build Coastguard Worker                 _DOCTEST_RB.translateException();                                                  \
2131*89c4ff92SAndroid Build Coastguard Worker                 _DOCTEST_RB.m_threw_as = true;                                                     \
2132*89c4ff92SAndroid Build Coastguard Worker             } catch(...) { _DOCTEST_RB.translateException(); }                                     \
2133*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB);                                             \
2134*89c4ff92SAndroid Build Coastguard Worker         }                                                                                          \
2135*89c4ff92SAndroid Build Coastguard Worker     } while(false)
2136*89c4ff92SAndroid Build Coastguard Worker 
2137*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ASSERT_THROWS_WITH(expr, expr_str, assert_type, ...)                               \
2138*89c4ff92SAndroid Build Coastguard Worker     do {                                                                                           \
2139*89c4ff92SAndroid Build Coastguard Worker         if(!doctest::getContextOptions()->no_throw) {                                              \
2140*89c4ff92SAndroid Build Coastguard Worker             doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__, \
2141*89c4ff92SAndroid Build Coastguard Worker                                                        __LINE__, expr_str, "", __VA_ARGS__);       \
2142*89c4ff92SAndroid Build Coastguard Worker             try {                                                                                  \
2143*89c4ff92SAndroid Build Coastguard Worker                 DOCTEST_CAST_TO_VOID(expr)                                                         \
2144*89c4ff92SAndroid Build Coastguard Worker             } catch(...) { _DOCTEST_RB.translateException(); }                                     \
2145*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB);                                             \
2146*89c4ff92SAndroid Build Coastguard Worker         }                                                                                          \
2147*89c4ff92SAndroid Build Coastguard Worker     } while(false)
2148*89c4ff92SAndroid Build Coastguard Worker 
2149*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ASSERT_NOTHROW(assert_type, ...)                                                   \
2150*89c4ff92SAndroid Build Coastguard Worker     do {                                                                                           \
2151*89c4ff92SAndroid Build Coastguard Worker         doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__,     \
2152*89c4ff92SAndroid Build Coastguard Worker                                                    __LINE__, #__VA_ARGS__);                        \
2153*89c4ff92SAndroid Build Coastguard Worker         try {                                                                                      \
2154*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_CAST_TO_VOID(__VA_ARGS__)                                                      \
2155*89c4ff92SAndroid Build Coastguard Worker         } catch(...) { _DOCTEST_RB.translateException(); }                                         \
2156*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB);                                                 \
2157*89c4ff92SAndroid Build Coastguard Worker     } while(false)
2158*89c4ff92SAndroid Build Coastguard Worker 
2159*89c4ff92SAndroid Build Coastguard Worker // clang-format off
2160*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS(...) DOCTEST_ASSERT_THROWS_WITH((__VA_ARGS__), #__VA_ARGS__, DT_WARN_THROWS, "")
2161*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS(...) DOCTEST_ASSERT_THROWS_WITH((__VA_ARGS__), #__VA_ARGS__, DT_CHECK_THROWS, "")
2162*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS(...) DOCTEST_ASSERT_THROWS_WITH((__VA_ARGS__), #__VA_ARGS__, DT_REQUIRE_THROWS, "")
2163*89c4ff92SAndroid Build Coastguard Worker 
2164*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_AS(expr, ...) DOCTEST_ASSERT_THROWS_AS(expr, DT_WARN_THROWS_AS, "", __VA_ARGS__)
2165*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_AS(expr, ...) DOCTEST_ASSERT_THROWS_AS(expr, DT_CHECK_THROWS_AS, "", __VA_ARGS__)
2166*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_AS(expr, ...) DOCTEST_ASSERT_THROWS_AS(expr, DT_REQUIRE_THROWS_AS, "", __VA_ARGS__)
2167*89c4ff92SAndroid Build Coastguard Worker 
2168*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH(expr, ...) DOCTEST_ASSERT_THROWS_WITH(expr, #expr, DT_WARN_THROWS_WITH, __VA_ARGS__)
2169*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH(expr, ...) DOCTEST_ASSERT_THROWS_WITH(expr, #expr, DT_CHECK_THROWS_WITH, __VA_ARGS__)
2170*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH(expr, ...) DOCTEST_ASSERT_THROWS_WITH(expr, #expr, DT_REQUIRE_THROWS_WITH, __VA_ARGS__)
2171*89c4ff92SAndroid Build Coastguard Worker 
2172*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH_AS(expr, message, ...) DOCTEST_ASSERT_THROWS_AS(expr, DT_WARN_THROWS_WITH_AS, message, __VA_ARGS__)
2173*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH_AS(expr, message, ...) DOCTEST_ASSERT_THROWS_AS(expr, DT_CHECK_THROWS_WITH_AS, message, __VA_ARGS__)
2174*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH_AS(expr, message, ...) DOCTEST_ASSERT_THROWS_AS(expr, DT_REQUIRE_THROWS_WITH_AS, message, __VA_ARGS__)
2175*89c4ff92SAndroid Build Coastguard Worker 
2176*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_NOTHROW(...) DOCTEST_ASSERT_NOTHROW(DT_WARN_NOTHROW, __VA_ARGS__)
2177*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_NOTHROW(...) DOCTEST_ASSERT_NOTHROW(DT_CHECK_NOTHROW, __VA_ARGS__)
2178*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_NOTHROW(...) DOCTEST_ASSERT_NOTHROW(DT_REQUIRE_NOTHROW, __VA_ARGS__)
2179*89c4ff92SAndroid Build Coastguard Worker 
2180*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_MESSAGE(expr, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_WARN_THROWS(expr); } while(false)
2181*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_MESSAGE(expr, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_CHECK_THROWS(expr); } while(false)
2182*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_MESSAGE(expr, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_REQUIRE_THROWS(expr); } while(false)
2183*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_AS_MESSAGE(expr, ex, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_WARN_THROWS_AS(expr, ex); } while(false)
2184*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_AS_MESSAGE(expr, ex, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_CHECK_THROWS_AS(expr, ex); } while(false)
2185*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_AS_MESSAGE(expr, ex, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_REQUIRE_THROWS_AS(expr, ex); } while(false)
2186*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH_MESSAGE(expr, with, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_WARN_THROWS_WITH(expr, with); } while(false)
2187*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH_MESSAGE(expr, with, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_CHECK_THROWS_WITH(expr, with); } while(false)
2188*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH_MESSAGE(expr, with, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_REQUIRE_THROWS_WITH(expr, with); } while(false)
2189*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_WARN_THROWS_WITH_AS(expr, with, ex); } while(false)
2190*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_CHECK_THROWS_WITH_AS(expr, with, ex); } while(false)
2191*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_REQUIRE_THROWS_WITH_AS(expr, with, ex); } while(false)
2192*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_NOTHROW_MESSAGE(expr, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_WARN_NOTHROW(expr); } while(false)
2193*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_NOTHROW_MESSAGE(expr, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_CHECK_NOTHROW(expr); } while(false)
2194*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_NOTHROW_MESSAGE(expr, ...) do { DOCTEST_INFO(__VA_ARGS__); DOCTEST_REQUIRE_NOTHROW(expr); } while(false)
2195*89c4ff92SAndroid Build Coastguard Worker // clang-format on
2196*89c4ff92SAndroid Build Coastguard Worker 
2197*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_SUPER_FAST_ASSERTS
2198*89c4ff92SAndroid Build Coastguard Worker 
2199*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BINARY_ASSERT(assert_type, comp, ...)                                              \
2200*89c4ff92SAndroid Build Coastguard Worker     do {                                                                                           \
2201*89c4ff92SAndroid Build Coastguard Worker         doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__,     \
2202*89c4ff92SAndroid Build Coastguard Worker                                                    __LINE__, #__VA_ARGS__);                        \
2203*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_WRAP_IN_TRY(                                                                       \
2204*89c4ff92SAndroid Build Coastguard Worker                 _DOCTEST_RB.binary_assert<doctest::detail::binaryAssertComparison::comp>(          \
2205*89c4ff92SAndroid Build Coastguard Worker                         __VA_ARGS__))                                                              \
2206*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB);                                                 \
2207*89c4ff92SAndroid Build Coastguard Worker     } while(false)
2208*89c4ff92SAndroid Build Coastguard Worker 
2209*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_UNARY_ASSERT(assert_type, ...)                                                     \
2210*89c4ff92SAndroid Build Coastguard Worker     do {                                                                                           \
2211*89c4ff92SAndroid Build Coastguard Worker         doctest::detail::ResultBuilder _DOCTEST_RB(doctest::assertType::assert_type, __FILE__,     \
2212*89c4ff92SAndroid Build Coastguard Worker                                                    __LINE__, #__VA_ARGS__);                        \
2213*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_WRAP_IN_TRY(_DOCTEST_RB.unary_assert(__VA_ARGS__))                                 \
2214*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_LOG_AND_REACT(_DOCTEST_RB);                                                 \
2215*89c4ff92SAndroid Build Coastguard Worker     } while(false)
2216*89c4ff92SAndroid Build Coastguard Worker 
2217*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_SUPER_FAST_ASSERTS
2218*89c4ff92SAndroid Build Coastguard Worker 
2219*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BINARY_ASSERT(assert_type, comparison, ...)                                        \
2220*89c4ff92SAndroid Build Coastguard Worker     doctest::detail::binary_assert<doctest::detail::binaryAssertComparison::comparison>(           \
2221*89c4ff92SAndroid Build Coastguard Worker             doctest::assertType::assert_type, __FILE__, __LINE__, #__VA_ARGS__, __VA_ARGS__)
2222*89c4ff92SAndroid Build Coastguard Worker 
2223*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_UNARY_ASSERT(assert_type, ...)                                                     \
2224*89c4ff92SAndroid Build Coastguard Worker     doctest::detail::unary_assert(doctest::assertType::assert_type, __FILE__, __LINE__,            \
2225*89c4ff92SAndroid Build Coastguard Worker                                   #__VA_ARGS__, __VA_ARGS__)
2226*89c4ff92SAndroid Build Coastguard Worker 
2227*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_SUPER_FAST_ASSERTS
2228*89c4ff92SAndroid Build Coastguard Worker 
2229*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_EQ(...) DOCTEST_BINARY_ASSERT(DT_WARN_EQ, eq, __VA_ARGS__)
2230*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_EQ(...) DOCTEST_BINARY_ASSERT(DT_CHECK_EQ, eq, __VA_ARGS__)
2231*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_EQ(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_EQ, eq, __VA_ARGS__)
2232*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_NE(...) DOCTEST_BINARY_ASSERT(DT_WARN_NE, ne, __VA_ARGS__)
2233*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_NE(...) DOCTEST_BINARY_ASSERT(DT_CHECK_NE, ne, __VA_ARGS__)
2234*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_NE(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_NE, ne, __VA_ARGS__)
2235*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_GT(...) DOCTEST_BINARY_ASSERT(DT_WARN_GT, gt, __VA_ARGS__)
2236*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_GT(...) DOCTEST_BINARY_ASSERT(DT_CHECK_GT, gt, __VA_ARGS__)
2237*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_GT(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_GT, gt, __VA_ARGS__)
2238*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_LT(...) DOCTEST_BINARY_ASSERT(DT_WARN_LT, lt, __VA_ARGS__)
2239*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_LT(...) DOCTEST_BINARY_ASSERT(DT_CHECK_LT, lt, __VA_ARGS__)
2240*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_LT(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_LT, lt, __VA_ARGS__)
2241*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_GE(...) DOCTEST_BINARY_ASSERT(DT_WARN_GE, ge, __VA_ARGS__)
2242*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_GE(...) DOCTEST_BINARY_ASSERT(DT_CHECK_GE, ge, __VA_ARGS__)
2243*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_GE(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_GE, ge, __VA_ARGS__)
2244*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_LE(...) DOCTEST_BINARY_ASSERT(DT_WARN_LE, le, __VA_ARGS__)
2245*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_LE(...) DOCTEST_BINARY_ASSERT(DT_CHECK_LE, le, __VA_ARGS__)
2246*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_LE(...) DOCTEST_BINARY_ASSERT(DT_REQUIRE_LE, le, __VA_ARGS__)
2247*89c4ff92SAndroid Build Coastguard Worker 
2248*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_UNARY(...) DOCTEST_UNARY_ASSERT(DT_WARN_UNARY, __VA_ARGS__)
2249*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_UNARY(...) DOCTEST_UNARY_ASSERT(DT_CHECK_UNARY, __VA_ARGS__)
2250*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_UNARY(...) DOCTEST_UNARY_ASSERT(DT_REQUIRE_UNARY, __VA_ARGS__)
2251*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_UNARY_FALSE(...) DOCTEST_UNARY_ASSERT(DT_WARN_UNARY_FALSE, __VA_ARGS__)
2252*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_UNARY_FALSE(...) DOCTEST_UNARY_ASSERT(DT_CHECK_UNARY_FALSE, __VA_ARGS__)
2253*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_UNARY_FALSE(...) DOCTEST_UNARY_ASSERT(DT_REQUIRE_UNARY_FALSE, __VA_ARGS__)
2254*89c4ff92SAndroid Build Coastguard Worker 
2255*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_NO_EXCEPTIONS
2256*89c4ff92SAndroid Build Coastguard Worker 
2257*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_WARN_THROWS
2258*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CHECK_THROWS
2259*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_THROWS
2260*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_WARN_THROWS_AS
2261*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CHECK_THROWS_AS
2262*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_THROWS_AS
2263*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_WARN_THROWS_WITH
2264*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CHECK_THROWS_WITH
2265*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_THROWS_WITH
2266*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_WARN_THROWS_WITH_AS
2267*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CHECK_THROWS_WITH_AS
2268*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_THROWS_WITH_AS
2269*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_WARN_NOTHROW
2270*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CHECK_NOTHROW
2271*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_NOTHROW
2272*89c4ff92SAndroid Build Coastguard Worker 
2273*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_WARN_THROWS_MESSAGE
2274*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CHECK_THROWS_MESSAGE
2275*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_THROWS_MESSAGE
2276*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_WARN_THROWS_AS_MESSAGE
2277*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CHECK_THROWS_AS_MESSAGE
2278*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_THROWS_AS_MESSAGE
2279*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_WARN_THROWS_WITH_MESSAGE
2280*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CHECK_THROWS_WITH_MESSAGE
2281*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_THROWS_WITH_MESSAGE
2282*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_WARN_THROWS_WITH_AS_MESSAGE
2283*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CHECK_THROWS_WITH_AS_MESSAGE
2284*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_THROWS_WITH_AS_MESSAGE
2285*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_WARN_NOTHROW_MESSAGE
2286*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_CHECK_NOTHROW_MESSAGE
2287*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_NOTHROW_MESSAGE
2288*89c4ff92SAndroid Build Coastguard Worker 
2289*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_NO_EXCEPTIONS_BUT_WITH_ALL_ASSERTS
2290*89c4ff92SAndroid Build Coastguard Worker 
2291*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS(...) (static_cast<void>(0))
2292*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS(...) (static_cast<void>(0))
2293*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS(...) (static_cast<void>(0))
2294*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_AS(expr, ...) (static_cast<void>(0))
2295*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_AS(expr, ...) (static_cast<void>(0))
2296*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_AS(expr, ...) (static_cast<void>(0))
2297*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH(expr, ...) (static_cast<void>(0))
2298*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH(expr, ...) (static_cast<void>(0))
2299*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH(expr, ...) (static_cast<void>(0))
2300*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH_AS(expr, with, ...) (static_cast<void>(0))
2301*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH_AS(expr, with, ...) (static_cast<void>(0))
2302*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH_AS(expr, with, ...) (static_cast<void>(0))
2303*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_NOTHROW(...) (static_cast<void>(0))
2304*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_NOTHROW(...) (static_cast<void>(0))
2305*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_NOTHROW(...) (static_cast<void>(0))
2306*89c4ff92SAndroid Build Coastguard Worker 
2307*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_MESSAGE(expr, ...) (static_cast<void>(0))
2308*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_MESSAGE(expr, ...) (static_cast<void>(0))
2309*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_MESSAGE(expr, ...) (static_cast<void>(0))
2310*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_AS_MESSAGE(expr, ex, ...) (static_cast<void>(0))
2311*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_AS_MESSAGE(expr, ex, ...) (static_cast<void>(0))
2312*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_AS_MESSAGE(expr, ex, ...) (static_cast<void>(0))
2313*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH_MESSAGE(expr, with, ...) (static_cast<void>(0))
2314*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH_MESSAGE(expr, with, ...) (static_cast<void>(0))
2315*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH_MESSAGE(expr, with, ...) (static_cast<void>(0))
2316*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) (static_cast<void>(0))
2317*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) (static_cast<void>(0))
2318*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) (static_cast<void>(0))
2319*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_NOTHROW_MESSAGE(expr, ...) (static_cast<void>(0))
2320*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_NOTHROW_MESSAGE(expr, ...) (static_cast<void>(0))
2321*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_NOTHROW_MESSAGE(expr, ...) (static_cast<void>(0))
2322*89c4ff92SAndroid Build Coastguard Worker 
2323*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_NO_EXCEPTIONS_BUT_WITH_ALL_ASSERTS
2324*89c4ff92SAndroid Build Coastguard Worker 
2325*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE
2326*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_FALSE
2327*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_MESSAGE
2328*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_FALSE_MESSAGE
2329*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_EQ
2330*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_NE
2331*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_GT
2332*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_LT
2333*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_GE
2334*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_LE
2335*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_UNARY
2336*89c4ff92SAndroid Build Coastguard Worker #undef DOCTEST_REQUIRE_UNARY_FALSE
2337*89c4ff92SAndroid Build Coastguard Worker 
2338*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS_BUT_WITH_ALL_ASSERTS
2339*89c4ff92SAndroid Build Coastguard Worker 
2340*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS
2341*89c4ff92SAndroid Build Coastguard Worker 
2342*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
2343*89c4ff92SAndroid Build Coastguard Worker // == WHAT FOLLOWS IS VERSIONS OF THE MACROS THAT DO NOT DO ANY REGISTERING!                      ==
2344*89c4ff92SAndroid Build Coastguard Worker // == THIS CAN BE ENABLED BY DEFINING DOCTEST_CONFIG_DISABLE GLOBALLY!                            ==
2345*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
2346*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_DISABLE
2347*89c4ff92SAndroid Build Coastguard Worker 
2348*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_IMPLEMENT_FIXTURE(der, base, func, name)                                           \
2349*89c4ff92SAndroid Build Coastguard Worker     namespace {                                                                                    \
2350*89c4ff92SAndroid Build Coastguard Worker         template <typename DOCTEST_UNUSED_TEMPLATE_TYPE>                                           \
2351*89c4ff92SAndroid Build Coastguard Worker         struct der : public base                                                                   \
2352*89c4ff92SAndroid Build Coastguard Worker         { void f(); };                                                                             \
2353*89c4ff92SAndroid Build Coastguard Worker     }                                                                                              \
2354*89c4ff92SAndroid Build Coastguard Worker     template <typename DOCTEST_UNUSED_TEMPLATE_TYPE>                                               \
2355*89c4ff92SAndroid Build Coastguard Worker     inline void der<DOCTEST_UNUSED_TEMPLATE_TYPE>::f()
2356*89c4ff92SAndroid Build Coastguard Worker 
2357*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CREATE_AND_REGISTER_FUNCTION(f, name)                                              \
2358*89c4ff92SAndroid Build Coastguard Worker     template <typename DOCTEST_UNUSED_TEMPLATE_TYPE>                                               \
2359*89c4ff92SAndroid Build Coastguard Worker     static inline void f()
2360*89c4ff92SAndroid Build Coastguard Worker 
2361*89c4ff92SAndroid Build Coastguard Worker // for registering tests
2362*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE(name)                                                                    \
2363*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CREATE_AND_REGISTER_FUNCTION(DOCTEST_ANONYMOUS(_DOCTEST_ANON_FUNC_), name)
2364*89c4ff92SAndroid Build Coastguard Worker 
2365*89c4ff92SAndroid Build Coastguard Worker // for registering tests in classes
2366*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_CLASS(name)                                                              \
2367*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CREATE_AND_REGISTER_FUNCTION(DOCTEST_ANONYMOUS(_DOCTEST_ANON_FUNC_), name)
2368*89c4ff92SAndroid Build Coastguard Worker 
2369*89c4ff92SAndroid Build Coastguard Worker // for registering tests with a fixture
2370*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_FIXTURE(x, name)                                                         \
2371*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_IMPLEMENT_FIXTURE(DOCTEST_ANONYMOUS(_DOCTEST_ANON_CLASS_), x,                          \
2372*89c4ff92SAndroid Build Coastguard Worker                               DOCTEST_ANONYMOUS(_DOCTEST_ANON_FUNC_), name)
2373*89c4ff92SAndroid Build Coastguard Worker 
2374*89c4ff92SAndroid Build Coastguard Worker // for converting types to strings without the <typeinfo> header and demangling
2375*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TYPE_TO_STRING(...) typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
2376*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TYPE_TO_STRING_IMPL(...)
2377*89c4ff92SAndroid Build Coastguard Worker 
2378*89c4ff92SAndroid Build Coastguard Worker // for typed tests
2379*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE(name, type, ...)                                                \
2380*89c4ff92SAndroid Build Coastguard Worker     template <typename type>                                                                       \
2381*89c4ff92SAndroid Build Coastguard Worker     inline void DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_)()
2382*89c4ff92SAndroid Build Coastguard Worker 
2383*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE_DEFINE(name, type, id)                                          \
2384*89c4ff92SAndroid Build Coastguard Worker     template <typename type>                                                                       \
2385*89c4ff92SAndroid Build Coastguard Worker     inline void DOCTEST_ANONYMOUS(_DOCTEST_ANON_TMP_)()
2386*89c4ff92SAndroid Build Coastguard Worker 
2387*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE_INVOKE(id, ...)                                                 \
2388*89c4ff92SAndroid Build Coastguard Worker     typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
2389*89c4ff92SAndroid Build Coastguard Worker 
2390*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE_APPLY(id, ...)                                                  \
2391*89c4ff92SAndroid Build Coastguard Worker     typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
2392*89c4ff92SAndroid Build Coastguard Worker 
2393*89c4ff92SAndroid Build Coastguard Worker // for subcases
2394*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SUBCASE(name)
2395*89c4ff92SAndroid Build Coastguard Worker 
2396*89c4ff92SAndroid Build Coastguard Worker // for a testsuite block
2397*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_SUITE(name) namespace
2398*89c4ff92SAndroid Build Coastguard Worker 
2399*89c4ff92SAndroid Build Coastguard Worker // for starting a testsuite block
2400*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_SUITE_BEGIN(name) typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
2401*89c4ff92SAndroid Build Coastguard Worker 
2402*89c4ff92SAndroid Build Coastguard Worker // for ending a testsuite block
2403*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_SUITE_END typedef int DOCTEST_ANONYMOUS(_DOCTEST_ANON_FOR_SEMICOLON_)
2404*89c4ff92SAndroid Build Coastguard Worker 
2405*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REGISTER_EXCEPTION_TRANSLATOR(signature)                                           \
2406*89c4ff92SAndroid Build Coastguard Worker     template <typename DOCTEST_UNUSED_TEMPLATE_TYPE>                                               \
2407*89c4ff92SAndroid Build Coastguard Worker     static inline doctest::String DOCTEST_ANONYMOUS(_DOCTEST_ANON_TRANSLATOR_)(signature)
2408*89c4ff92SAndroid Build Coastguard Worker 
2409*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REGISTER_REPORTER(name, priority, reporter)
2410*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REGISTER_LISTENER(name, priority, reporter)
2411*89c4ff92SAndroid Build Coastguard Worker 
2412*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_INFO(...) (static_cast<void>(0))
2413*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CAPTURE(x) (static_cast<void>(0))
2414*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ADD_MESSAGE_AT(file, line, ...) (static_cast<void>(0))
2415*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ADD_FAIL_CHECK_AT(file, line, ...) (static_cast<void>(0))
2416*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ADD_FAIL_AT(file, line, ...) (static_cast<void>(0))
2417*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MESSAGE(...) (static_cast<void>(0))
2418*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAIL_CHECK(...) (static_cast<void>(0))
2419*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAIL(...) (static_cast<void>(0))
2420*89c4ff92SAndroid Build Coastguard Worker 
2421*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN(...) (static_cast<void>(0))
2422*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK(...) (static_cast<void>(0))
2423*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE(...) (static_cast<void>(0))
2424*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_FALSE(...) (static_cast<void>(0))
2425*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_FALSE(...) (static_cast<void>(0))
2426*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_FALSE(...) (static_cast<void>(0))
2427*89c4ff92SAndroid Build Coastguard Worker 
2428*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_MESSAGE(cond, ...) (static_cast<void>(0))
2429*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_MESSAGE(cond, ...) (static_cast<void>(0))
2430*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_MESSAGE(cond, ...) (static_cast<void>(0))
2431*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_FALSE_MESSAGE(cond, ...) (static_cast<void>(0))
2432*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_FALSE_MESSAGE(cond, ...) (static_cast<void>(0))
2433*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_FALSE_MESSAGE(cond, ...) (static_cast<void>(0))
2434*89c4ff92SAndroid Build Coastguard Worker 
2435*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS(...) (static_cast<void>(0))
2436*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS(...) (static_cast<void>(0))
2437*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS(...) (static_cast<void>(0))
2438*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_AS(expr, ...) (static_cast<void>(0))
2439*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_AS(expr, ...) (static_cast<void>(0))
2440*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_AS(expr, ...) (static_cast<void>(0))
2441*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH(expr, ...) (static_cast<void>(0))
2442*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH(expr, ...) (static_cast<void>(0))
2443*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH(expr, ...) (static_cast<void>(0))
2444*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH_AS(expr, with, ...) (static_cast<void>(0))
2445*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH_AS(expr, with, ...) (static_cast<void>(0))
2446*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH_AS(expr, with, ...) (static_cast<void>(0))
2447*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_NOTHROW(...) (static_cast<void>(0))
2448*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_NOTHROW(...) (static_cast<void>(0))
2449*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_NOTHROW(...) (static_cast<void>(0))
2450*89c4ff92SAndroid Build Coastguard Worker 
2451*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_MESSAGE(expr, ...) (static_cast<void>(0))
2452*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_MESSAGE(expr, ...) (static_cast<void>(0))
2453*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_MESSAGE(expr, ...) (static_cast<void>(0))
2454*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_AS_MESSAGE(expr, ex, ...) (static_cast<void>(0))
2455*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_AS_MESSAGE(expr, ex, ...) (static_cast<void>(0))
2456*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_AS_MESSAGE(expr, ex, ...) (static_cast<void>(0))
2457*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH_MESSAGE(expr, with, ...) (static_cast<void>(0))
2458*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH_MESSAGE(expr, with, ...) (static_cast<void>(0))
2459*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH_MESSAGE(expr, with, ...) (static_cast<void>(0))
2460*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) (static_cast<void>(0))
2461*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) (static_cast<void>(0))
2462*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) (static_cast<void>(0))
2463*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_NOTHROW_MESSAGE(expr, ...) (static_cast<void>(0))
2464*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_NOTHROW_MESSAGE(expr, ...) (static_cast<void>(0))
2465*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_NOTHROW_MESSAGE(expr, ...) (static_cast<void>(0))
2466*89c4ff92SAndroid Build Coastguard Worker 
2467*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_EQ(...) (static_cast<void>(0))
2468*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_EQ(...) (static_cast<void>(0))
2469*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_EQ(...) (static_cast<void>(0))
2470*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_NE(...) (static_cast<void>(0))
2471*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_NE(...) (static_cast<void>(0))
2472*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_NE(...) (static_cast<void>(0))
2473*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_GT(...) (static_cast<void>(0))
2474*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_GT(...) (static_cast<void>(0))
2475*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_GT(...) (static_cast<void>(0))
2476*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_LT(...) (static_cast<void>(0))
2477*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_LT(...) (static_cast<void>(0))
2478*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_LT(...) (static_cast<void>(0))
2479*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_GE(...) (static_cast<void>(0))
2480*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_GE(...) (static_cast<void>(0))
2481*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_GE(...) (static_cast<void>(0))
2482*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_LE(...) (static_cast<void>(0))
2483*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_LE(...) (static_cast<void>(0))
2484*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_LE(...) (static_cast<void>(0))
2485*89c4ff92SAndroid Build Coastguard Worker 
2486*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_UNARY(...) (static_cast<void>(0))
2487*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_UNARY(...) (static_cast<void>(0))
2488*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_UNARY(...) (static_cast<void>(0))
2489*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WARN_UNARY_FALSE(...) (static_cast<void>(0))
2490*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CHECK_UNARY_FALSE(...) (static_cast<void>(0))
2491*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_REQUIRE_UNARY_FALSE(...) (static_cast<void>(0))
2492*89c4ff92SAndroid Build Coastguard Worker 
2493*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_DISABLE
2494*89c4ff92SAndroid Build Coastguard Worker 
2495*89c4ff92SAndroid Build Coastguard Worker // clang-format off
2496*89c4ff92SAndroid Build Coastguard Worker // KEPT FOR BACKWARDS COMPATIBILITY - FORWARDING TO THE RIGHT MACROS
2497*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_WARN_EQ             DOCTEST_WARN_EQ
2498*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_CHECK_EQ            DOCTEST_CHECK_EQ
2499*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_REQUIRE_EQ          DOCTEST_REQUIRE_EQ
2500*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_WARN_NE             DOCTEST_WARN_NE
2501*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_CHECK_NE            DOCTEST_CHECK_NE
2502*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_REQUIRE_NE          DOCTEST_REQUIRE_NE
2503*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_WARN_GT             DOCTEST_WARN_GT
2504*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_CHECK_GT            DOCTEST_CHECK_GT
2505*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_REQUIRE_GT          DOCTEST_REQUIRE_GT
2506*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_WARN_LT             DOCTEST_WARN_LT
2507*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_CHECK_LT            DOCTEST_CHECK_LT
2508*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_REQUIRE_LT          DOCTEST_REQUIRE_LT
2509*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_WARN_GE             DOCTEST_WARN_GE
2510*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_CHECK_GE            DOCTEST_CHECK_GE
2511*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_REQUIRE_GE          DOCTEST_REQUIRE_GE
2512*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_WARN_LE             DOCTEST_WARN_LE
2513*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_CHECK_LE            DOCTEST_CHECK_LE
2514*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_REQUIRE_LE          DOCTEST_REQUIRE_LE
2515*89c4ff92SAndroid Build Coastguard Worker 
2516*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_WARN_UNARY          DOCTEST_WARN_UNARY
2517*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_CHECK_UNARY         DOCTEST_CHECK_UNARY
2518*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_REQUIRE_UNARY       DOCTEST_REQUIRE_UNARY
2519*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_WARN_UNARY_FALSE    DOCTEST_WARN_UNARY_FALSE
2520*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_CHECK_UNARY_FALSE   DOCTEST_CHECK_UNARY_FALSE
2521*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_FAST_REQUIRE_UNARY_FALSE DOCTEST_REQUIRE_UNARY_FALSE
2522*89c4ff92SAndroid Build Coastguard Worker 
2523*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE(id, ...) DOCTEST_TEST_CASE_TEMPLATE_INVOKE(id,__VA_ARGS__)
2524*89c4ff92SAndroid Build Coastguard Worker // clang-format on
2525*89c4ff92SAndroid Build Coastguard Worker 
2526*89c4ff92SAndroid Build Coastguard Worker // BDD style macros
2527*89c4ff92SAndroid Build Coastguard Worker // clang-format off
2528*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SCENARIO(name) DOCTEST_TEST_CASE("  Scenario: " name)
2529*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SCENARIO_CLASS(name) DOCTEST_TEST_CASE_CLASS("  Scenario: " name)
2530*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SCENARIO_TEMPLATE(name, T, ...)  DOCTEST_TEST_CASE_TEMPLATE("  Scenario: " name, T, __VA_ARGS__)
2531*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SCENARIO_TEMPLATE_DEFINE(name, T, id) DOCTEST_TEST_CASE_TEMPLATE_DEFINE("  Scenario: " name, T, id)
2532*89c4ff92SAndroid Build Coastguard Worker 
2533*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_GIVEN(name)     DOCTEST_SUBCASE("   Given: " name)
2534*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_WHEN(name)      DOCTEST_SUBCASE("    When: " name)
2535*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_AND_WHEN(name)  DOCTEST_SUBCASE("And when: " name)
2536*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_THEN(name)      DOCTEST_SUBCASE("    Then: " name)
2537*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_AND_THEN(name)  DOCTEST_SUBCASE("     And: " name)
2538*89c4ff92SAndroid Build Coastguard Worker // clang-format on
2539*89c4ff92SAndroid Build Coastguard Worker 
2540*89c4ff92SAndroid Build Coastguard Worker // == SHORT VERSIONS OF THE MACROS
2541*89c4ff92SAndroid Build Coastguard Worker #if !defined(DOCTEST_CONFIG_NO_SHORT_MACRO_NAMES)
2542*89c4ff92SAndroid Build Coastguard Worker 
2543*89c4ff92SAndroid Build Coastguard Worker #define TEST_CASE(name) DOCTEST_TEST_CASE(name)
2544*89c4ff92SAndroid Build Coastguard Worker #define TEST_CASE_CLASS(name) DOCTEST_TEST_CASE_CLASS(name)
2545*89c4ff92SAndroid Build Coastguard Worker #define TEST_CASE_FIXTURE(x, name) DOCTEST_TEST_CASE_FIXTURE(x, name)
2546*89c4ff92SAndroid Build Coastguard Worker #define TYPE_TO_STRING(...) DOCTEST_TYPE_TO_STRING(__VA_ARGS__)
2547*89c4ff92SAndroid Build Coastguard Worker #define TEST_CASE_TEMPLATE(name, T, ...) DOCTEST_TEST_CASE_TEMPLATE(name, T, __VA_ARGS__)
2548*89c4ff92SAndroid Build Coastguard Worker #define TEST_CASE_TEMPLATE_DEFINE(name, T, id) DOCTEST_TEST_CASE_TEMPLATE_DEFINE(name, T, id)
2549*89c4ff92SAndroid Build Coastguard Worker #define TEST_CASE_TEMPLATE_INVOKE(id, ...) DOCTEST_TEST_CASE_TEMPLATE_INVOKE(id, __VA_ARGS__)
2550*89c4ff92SAndroid Build Coastguard Worker #define TEST_CASE_TEMPLATE_APPLY(id, ...) DOCTEST_TEST_CASE_TEMPLATE_APPLY(id, __VA_ARGS__)
2551*89c4ff92SAndroid Build Coastguard Worker #define SUBCASE(name) DOCTEST_SUBCASE(name)
2552*89c4ff92SAndroid Build Coastguard Worker #define TEST_SUITE(decorators) DOCTEST_TEST_SUITE(decorators)
2553*89c4ff92SAndroid Build Coastguard Worker #define TEST_SUITE_BEGIN(name) DOCTEST_TEST_SUITE_BEGIN(name)
2554*89c4ff92SAndroid Build Coastguard Worker #define TEST_SUITE_END DOCTEST_TEST_SUITE_END
2555*89c4ff92SAndroid Build Coastguard Worker #define REGISTER_EXCEPTION_TRANSLATOR(signature) DOCTEST_REGISTER_EXCEPTION_TRANSLATOR(signature)
2556*89c4ff92SAndroid Build Coastguard Worker #define REGISTER_REPORTER(name, priority, reporter) DOCTEST_REGISTER_REPORTER(name, priority, reporter)
2557*89c4ff92SAndroid Build Coastguard Worker #define REGISTER_LISTENER(name, priority, reporter) DOCTEST_REGISTER_LISTENER(name, priority, reporter)
2558*89c4ff92SAndroid Build Coastguard Worker #define INFO(...) DOCTEST_INFO(__VA_ARGS__)
2559*89c4ff92SAndroid Build Coastguard Worker #define CAPTURE(x) DOCTEST_CAPTURE(x)
2560*89c4ff92SAndroid Build Coastguard Worker #define ADD_MESSAGE_AT(file, line, ...) DOCTEST_ADD_MESSAGE_AT(file, line, __VA_ARGS__)
2561*89c4ff92SAndroid Build Coastguard Worker #define ADD_FAIL_CHECK_AT(file, line, ...) DOCTEST_ADD_FAIL_CHECK_AT(file, line, __VA_ARGS__)
2562*89c4ff92SAndroid Build Coastguard Worker #define ADD_FAIL_AT(file, line, ...) DOCTEST_ADD_FAIL_AT(file, line, __VA_ARGS__)
2563*89c4ff92SAndroid Build Coastguard Worker #define MESSAGE(...) DOCTEST_MESSAGE(__VA_ARGS__)
2564*89c4ff92SAndroid Build Coastguard Worker #define FAIL_CHECK(...) DOCTEST_FAIL_CHECK(__VA_ARGS__)
2565*89c4ff92SAndroid Build Coastguard Worker #define FAIL(...) DOCTEST_FAIL(__VA_ARGS__)
2566*89c4ff92SAndroid Build Coastguard Worker #define TO_LVALUE(...) DOCTEST_TO_LVALUE(__VA_ARGS__)
2567*89c4ff92SAndroid Build Coastguard Worker 
2568*89c4ff92SAndroid Build Coastguard Worker #define WARN(...) DOCTEST_WARN(__VA_ARGS__)
2569*89c4ff92SAndroid Build Coastguard Worker #define WARN_FALSE(...) DOCTEST_WARN_FALSE(__VA_ARGS__)
2570*89c4ff92SAndroid Build Coastguard Worker #define WARN_THROWS(...) DOCTEST_WARN_THROWS(__VA_ARGS__)
2571*89c4ff92SAndroid Build Coastguard Worker #define WARN_THROWS_AS(expr, ...) DOCTEST_WARN_THROWS_AS(expr, __VA_ARGS__)
2572*89c4ff92SAndroid Build Coastguard Worker #define WARN_THROWS_WITH(expr, ...) DOCTEST_WARN_THROWS_WITH(expr, __VA_ARGS__)
2573*89c4ff92SAndroid Build Coastguard Worker #define WARN_THROWS_WITH_AS(expr, with, ...) DOCTEST_WARN_THROWS_WITH_AS(expr, with, __VA_ARGS__)
2574*89c4ff92SAndroid Build Coastguard Worker #define WARN_NOTHROW(...) DOCTEST_WARN_NOTHROW(__VA_ARGS__)
2575*89c4ff92SAndroid Build Coastguard Worker #define CHECK(...) DOCTEST_CHECK(__VA_ARGS__)
2576*89c4ff92SAndroid Build Coastguard Worker #define CHECK_FALSE(...) DOCTEST_CHECK_FALSE(__VA_ARGS__)
2577*89c4ff92SAndroid Build Coastguard Worker #define CHECK_THROWS(...) DOCTEST_CHECK_THROWS(__VA_ARGS__)
2578*89c4ff92SAndroid Build Coastguard Worker #define CHECK_THROWS_AS(expr, ...) DOCTEST_CHECK_THROWS_AS(expr, __VA_ARGS__)
2579*89c4ff92SAndroid Build Coastguard Worker #define CHECK_THROWS_WITH(expr, ...) DOCTEST_CHECK_THROWS_WITH(expr, __VA_ARGS__)
2580*89c4ff92SAndroid Build Coastguard Worker #define CHECK_THROWS_WITH_AS(expr, with, ...) DOCTEST_CHECK_THROWS_WITH_AS(expr, with, __VA_ARGS__)
2581*89c4ff92SAndroid Build Coastguard Worker #define CHECK_NOTHROW(...) DOCTEST_CHECK_NOTHROW(__VA_ARGS__)
2582*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE(...) DOCTEST_REQUIRE(__VA_ARGS__)
2583*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_FALSE(...) DOCTEST_REQUIRE_FALSE(__VA_ARGS__)
2584*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_THROWS(...) DOCTEST_REQUIRE_THROWS(__VA_ARGS__)
2585*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_THROWS_AS(expr, ...) DOCTEST_REQUIRE_THROWS_AS(expr, __VA_ARGS__)
2586*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_THROWS_WITH(expr, ...) DOCTEST_REQUIRE_THROWS_WITH(expr, __VA_ARGS__)
2587*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_THROWS_WITH_AS(expr, with, ...) DOCTEST_REQUIRE_THROWS_WITH_AS(expr, with, __VA_ARGS__)
2588*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_NOTHROW(...) DOCTEST_REQUIRE_NOTHROW(__VA_ARGS__)
2589*89c4ff92SAndroid Build Coastguard Worker 
2590*89c4ff92SAndroid Build Coastguard Worker #define WARN_MESSAGE(cond, ...) DOCTEST_WARN_MESSAGE(cond, __VA_ARGS__)
2591*89c4ff92SAndroid Build Coastguard Worker #define WARN_FALSE_MESSAGE(cond, ...) DOCTEST_WARN_FALSE_MESSAGE(cond, __VA_ARGS__)
2592*89c4ff92SAndroid Build Coastguard Worker #define WARN_THROWS_MESSAGE(expr, ...) DOCTEST_WARN_THROWS_MESSAGE(expr, __VA_ARGS__)
2593*89c4ff92SAndroid Build Coastguard Worker #define WARN_THROWS_AS_MESSAGE(expr, ex, ...) DOCTEST_WARN_THROWS_AS_MESSAGE(expr, ex, __VA_ARGS__)
2594*89c4ff92SAndroid Build Coastguard Worker #define WARN_THROWS_WITH_MESSAGE(expr, with, ...) DOCTEST_WARN_THROWS_WITH_MESSAGE(expr, with, __VA_ARGS__)
2595*89c4ff92SAndroid Build Coastguard Worker #define WARN_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) DOCTEST_WARN_THROWS_WITH_AS_MESSAGE(expr, with, ex, __VA_ARGS__)
2596*89c4ff92SAndroid Build Coastguard Worker #define WARN_NOTHROW_MESSAGE(expr, ...) DOCTEST_WARN_NOTHROW_MESSAGE(expr, __VA_ARGS__)
2597*89c4ff92SAndroid Build Coastguard Worker #define CHECK_MESSAGE(cond, ...) DOCTEST_CHECK_MESSAGE(cond, __VA_ARGS__)
2598*89c4ff92SAndroid Build Coastguard Worker #define CHECK_FALSE_MESSAGE(cond, ...) DOCTEST_CHECK_FALSE_MESSAGE(cond, __VA_ARGS__)
2599*89c4ff92SAndroid Build Coastguard Worker #define CHECK_THROWS_MESSAGE(expr, ...) DOCTEST_CHECK_THROWS_MESSAGE(expr, __VA_ARGS__)
2600*89c4ff92SAndroid Build Coastguard Worker #define CHECK_THROWS_AS_MESSAGE(expr, ex, ...) DOCTEST_CHECK_THROWS_AS_MESSAGE(expr, ex, __VA_ARGS__)
2601*89c4ff92SAndroid Build Coastguard Worker #define CHECK_THROWS_WITH_MESSAGE(expr, with, ...) DOCTEST_CHECK_THROWS_WITH_MESSAGE(expr, with, __VA_ARGS__)
2602*89c4ff92SAndroid Build Coastguard Worker #define CHECK_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) DOCTEST_CHECK_THROWS_WITH_AS_MESSAGE(expr, with, ex, __VA_ARGS__)
2603*89c4ff92SAndroid Build Coastguard Worker #define CHECK_NOTHROW_MESSAGE(expr, ...) DOCTEST_CHECK_NOTHROW_MESSAGE(expr, __VA_ARGS__)
2604*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_MESSAGE(cond, ...) DOCTEST_REQUIRE_MESSAGE(cond, __VA_ARGS__)
2605*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_FALSE_MESSAGE(cond, ...) DOCTEST_REQUIRE_FALSE_MESSAGE(cond, __VA_ARGS__)
2606*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_THROWS_MESSAGE(expr, ...) DOCTEST_REQUIRE_THROWS_MESSAGE(expr, __VA_ARGS__)
2607*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_THROWS_AS_MESSAGE(expr, ex, ...) DOCTEST_REQUIRE_THROWS_AS_MESSAGE(expr, ex, __VA_ARGS__)
2608*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_THROWS_WITH_MESSAGE(expr, with, ...) DOCTEST_REQUIRE_THROWS_WITH_MESSAGE(expr, with, __VA_ARGS__)
2609*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_THROWS_WITH_AS_MESSAGE(expr, with, ex, ...) DOCTEST_REQUIRE_THROWS_WITH_AS_MESSAGE(expr, with, ex, __VA_ARGS__)
2610*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_NOTHROW_MESSAGE(expr, ...) DOCTEST_REQUIRE_NOTHROW_MESSAGE(expr, __VA_ARGS__)
2611*89c4ff92SAndroid Build Coastguard Worker 
2612*89c4ff92SAndroid Build Coastguard Worker #define SCENARIO(name) DOCTEST_SCENARIO(name)
2613*89c4ff92SAndroid Build Coastguard Worker #define SCENARIO_CLASS(name) DOCTEST_SCENARIO_CLASS(name)
2614*89c4ff92SAndroid Build Coastguard Worker #define SCENARIO_TEMPLATE(name, T, ...) DOCTEST_SCENARIO_TEMPLATE(name, T, __VA_ARGS__)
2615*89c4ff92SAndroid Build Coastguard Worker #define SCENARIO_TEMPLATE_DEFINE(name, T, id) DOCTEST_SCENARIO_TEMPLATE_DEFINE(name, T, id)
2616*89c4ff92SAndroid Build Coastguard Worker #define GIVEN(name) DOCTEST_GIVEN(name)
2617*89c4ff92SAndroid Build Coastguard Worker #define WHEN(name) DOCTEST_WHEN(name)
2618*89c4ff92SAndroid Build Coastguard Worker #define AND_WHEN(name) DOCTEST_AND_WHEN(name)
2619*89c4ff92SAndroid Build Coastguard Worker #define THEN(name) DOCTEST_THEN(name)
2620*89c4ff92SAndroid Build Coastguard Worker #define AND_THEN(name) DOCTEST_AND_THEN(name)
2621*89c4ff92SAndroid Build Coastguard Worker 
2622*89c4ff92SAndroid Build Coastguard Worker #define WARN_EQ(...) DOCTEST_WARN_EQ(__VA_ARGS__)
2623*89c4ff92SAndroid Build Coastguard Worker #define CHECK_EQ(...) DOCTEST_CHECK_EQ(__VA_ARGS__)
2624*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_EQ(...) DOCTEST_REQUIRE_EQ(__VA_ARGS__)
2625*89c4ff92SAndroid Build Coastguard Worker #define WARN_NE(...) DOCTEST_WARN_NE(__VA_ARGS__)
2626*89c4ff92SAndroid Build Coastguard Worker #define CHECK_NE(...) DOCTEST_CHECK_NE(__VA_ARGS__)
2627*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_NE(...) DOCTEST_REQUIRE_NE(__VA_ARGS__)
2628*89c4ff92SAndroid Build Coastguard Worker #define WARN_GT(...) DOCTEST_WARN_GT(__VA_ARGS__)
2629*89c4ff92SAndroid Build Coastguard Worker #define CHECK_GT(...) DOCTEST_CHECK_GT(__VA_ARGS__)
2630*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_GT(...) DOCTEST_REQUIRE_GT(__VA_ARGS__)
2631*89c4ff92SAndroid Build Coastguard Worker #define WARN_LT(...) DOCTEST_WARN_LT(__VA_ARGS__)
2632*89c4ff92SAndroid Build Coastguard Worker #define CHECK_LT(...) DOCTEST_CHECK_LT(__VA_ARGS__)
2633*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_LT(...) DOCTEST_REQUIRE_LT(__VA_ARGS__)
2634*89c4ff92SAndroid Build Coastguard Worker #define WARN_GE(...) DOCTEST_WARN_GE(__VA_ARGS__)
2635*89c4ff92SAndroid Build Coastguard Worker #define CHECK_GE(...) DOCTEST_CHECK_GE(__VA_ARGS__)
2636*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_GE(...) DOCTEST_REQUIRE_GE(__VA_ARGS__)
2637*89c4ff92SAndroid Build Coastguard Worker #define WARN_LE(...) DOCTEST_WARN_LE(__VA_ARGS__)
2638*89c4ff92SAndroid Build Coastguard Worker #define CHECK_LE(...) DOCTEST_CHECK_LE(__VA_ARGS__)
2639*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_LE(...) DOCTEST_REQUIRE_LE(__VA_ARGS__)
2640*89c4ff92SAndroid Build Coastguard Worker #define WARN_UNARY(...) DOCTEST_WARN_UNARY(__VA_ARGS__)
2641*89c4ff92SAndroid Build Coastguard Worker #define CHECK_UNARY(...) DOCTEST_CHECK_UNARY(__VA_ARGS__)
2642*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_UNARY(...) DOCTEST_REQUIRE_UNARY(__VA_ARGS__)
2643*89c4ff92SAndroid Build Coastguard Worker #define WARN_UNARY_FALSE(...) DOCTEST_WARN_UNARY_FALSE(__VA_ARGS__)
2644*89c4ff92SAndroid Build Coastguard Worker #define CHECK_UNARY_FALSE(...) DOCTEST_CHECK_UNARY_FALSE(__VA_ARGS__)
2645*89c4ff92SAndroid Build Coastguard Worker #define REQUIRE_UNARY_FALSE(...) DOCTEST_REQUIRE_UNARY_FALSE(__VA_ARGS__)
2646*89c4ff92SAndroid Build Coastguard Worker 
2647*89c4ff92SAndroid Build Coastguard Worker // KEPT FOR BACKWARDS COMPATIBILITY
2648*89c4ff92SAndroid Build Coastguard Worker #define FAST_WARN_EQ(...) DOCTEST_FAST_WARN_EQ(__VA_ARGS__)
2649*89c4ff92SAndroid Build Coastguard Worker #define FAST_CHECK_EQ(...) DOCTEST_FAST_CHECK_EQ(__VA_ARGS__)
2650*89c4ff92SAndroid Build Coastguard Worker #define FAST_REQUIRE_EQ(...) DOCTEST_FAST_REQUIRE_EQ(__VA_ARGS__)
2651*89c4ff92SAndroid Build Coastguard Worker #define FAST_WARN_NE(...) DOCTEST_FAST_WARN_NE(__VA_ARGS__)
2652*89c4ff92SAndroid Build Coastguard Worker #define FAST_CHECK_NE(...) DOCTEST_FAST_CHECK_NE(__VA_ARGS__)
2653*89c4ff92SAndroid Build Coastguard Worker #define FAST_REQUIRE_NE(...) DOCTEST_FAST_REQUIRE_NE(__VA_ARGS__)
2654*89c4ff92SAndroid Build Coastguard Worker #define FAST_WARN_GT(...) DOCTEST_FAST_WARN_GT(__VA_ARGS__)
2655*89c4ff92SAndroid Build Coastguard Worker #define FAST_CHECK_GT(...) DOCTEST_FAST_CHECK_GT(__VA_ARGS__)
2656*89c4ff92SAndroid Build Coastguard Worker #define FAST_REQUIRE_GT(...) DOCTEST_FAST_REQUIRE_GT(__VA_ARGS__)
2657*89c4ff92SAndroid Build Coastguard Worker #define FAST_WARN_LT(...) DOCTEST_FAST_WARN_LT(__VA_ARGS__)
2658*89c4ff92SAndroid Build Coastguard Worker #define FAST_CHECK_LT(...) DOCTEST_FAST_CHECK_LT(__VA_ARGS__)
2659*89c4ff92SAndroid Build Coastguard Worker #define FAST_REQUIRE_LT(...) DOCTEST_FAST_REQUIRE_LT(__VA_ARGS__)
2660*89c4ff92SAndroid Build Coastguard Worker #define FAST_WARN_GE(...) DOCTEST_FAST_WARN_GE(__VA_ARGS__)
2661*89c4ff92SAndroid Build Coastguard Worker #define FAST_CHECK_GE(...) DOCTEST_FAST_CHECK_GE(__VA_ARGS__)
2662*89c4ff92SAndroid Build Coastguard Worker #define FAST_REQUIRE_GE(...) DOCTEST_FAST_REQUIRE_GE(__VA_ARGS__)
2663*89c4ff92SAndroid Build Coastguard Worker #define FAST_WARN_LE(...) DOCTEST_FAST_WARN_LE(__VA_ARGS__)
2664*89c4ff92SAndroid Build Coastguard Worker #define FAST_CHECK_LE(...) DOCTEST_FAST_CHECK_LE(__VA_ARGS__)
2665*89c4ff92SAndroid Build Coastguard Worker #define FAST_REQUIRE_LE(...) DOCTEST_FAST_REQUIRE_LE(__VA_ARGS__)
2666*89c4ff92SAndroid Build Coastguard Worker 
2667*89c4ff92SAndroid Build Coastguard Worker #define FAST_WARN_UNARY(...) DOCTEST_FAST_WARN_UNARY(__VA_ARGS__)
2668*89c4ff92SAndroid Build Coastguard Worker #define FAST_CHECK_UNARY(...) DOCTEST_FAST_CHECK_UNARY(__VA_ARGS__)
2669*89c4ff92SAndroid Build Coastguard Worker #define FAST_REQUIRE_UNARY(...) DOCTEST_FAST_REQUIRE_UNARY(__VA_ARGS__)
2670*89c4ff92SAndroid Build Coastguard Worker #define FAST_WARN_UNARY_FALSE(...) DOCTEST_FAST_WARN_UNARY_FALSE(__VA_ARGS__)
2671*89c4ff92SAndroid Build Coastguard Worker #define FAST_CHECK_UNARY_FALSE(...) DOCTEST_FAST_CHECK_UNARY_FALSE(__VA_ARGS__)
2672*89c4ff92SAndroid Build Coastguard Worker #define FAST_REQUIRE_UNARY_FALSE(...) DOCTEST_FAST_REQUIRE_UNARY_FALSE(__VA_ARGS__)
2673*89c4ff92SAndroid Build Coastguard Worker 
2674*89c4ff92SAndroid Build Coastguard Worker #define TEST_CASE_TEMPLATE_INSTANTIATE(id, ...) DOCTEST_TEST_CASE_TEMPLATE_INSTANTIATE(id, __VA_ARGS__)
2675*89c4ff92SAndroid Build Coastguard Worker 
2676*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_SHORT_MACRO_NAMES
2677*89c4ff92SAndroid Build Coastguard Worker 
2678*89c4ff92SAndroid Build Coastguard Worker #if !defined(DOCTEST_CONFIG_DISABLE)
2679*89c4ff92SAndroid Build Coastguard Worker 
2680*89c4ff92SAndroid Build Coastguard Worker // this is here to clear the 'current test suite' for the current translation unit - at the top
2681*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TEST_SUITE_END();
2682*89c4ff92SAndroid Build Coastguard Worker 
2683*89c4ff92SAndroid Build Coastguard Worker // add stringification for primitive/fundamental types
2684*89c4ff92SAndroid Build Coastguard Worker namespace doctest { namespace detail {
2685*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(bool)
2686*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(float)
2687*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(double)
2688*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(long double)
2689*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(char)
2690*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(signed char)
2691*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(unsigned char)
2692*89c4ff92SAndroid Build Coastguard Worker #if !DOCTEST_MSVC || defined(_NATIVE_WCHAR_T_DEFINED)
2693*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(wchar_t)
2694*89c4ff92SAndroid Build Coastguard Worker #endif // not MSVC or wchar_t support enabled
2695*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(short int)
2696*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(unsigned short int)
2697*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(int)
2698*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(unsigned int)
2699*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(long int)
2700*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(unsigned long int)
2701*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(long long int)
2702*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_TYPE_TO_STRING_IMPL(unsigned long long int)
2703*89c4ff92SAndroid Build Coastguard Worker }} // namespace doctest::detail
2704*89c4ff92SAndroid Build Coastguard Worker 
2705*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_DISABLE
2706*89c4ff92SAndroid Build Coastguard Worker 
2707*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING_POP
2708*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_POP
2709*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING_POP
2710*89c4ff92SAndroid Build Coastguard Worker 
2711*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_LIBRARY_INCLUDED
2712*89c4ff92SAndroid Build Coastguard Worker 
2713*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_SINGLE_HEADER
2714*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SINGLE_HEADER
2715*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_SINGLE_HEADER
2716*89c4ff92SAndroid Build Coastguard Worker 
2717*89c4ff92SAndroid Build Coastguard Worker #if defined(DOCTEST_CONFIG_IMPLEMENT) || !defined(DOCTEST_SINGLE_HEADER)
2718*89c4ff92SAndroid Build Coastguard Worker 
2719*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_SINGLE_HEADER
2720*89c4ff92SAndroid Build Coastguard Worker #include "doctest_fwd.h"
2721*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_SINGLE_HEADER
2722*89c4ff92SAndroid Build Coastguard Worker 
2723*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wunused-macros")
2724*89c4ff92SAndroid Build Coastguard Worker 
2725*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_LIBRARY_IMPLEMENTATION
2726*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_LIBRARY_IMPLEMENTATION
2727*89c4ff92SAndroid Build Coastguard Worker 
2728*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING_POP
2729*89c4ff92SAndroid Build Coastguard Worker 
2730*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING_PUSH
2731*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wunknown-pragmas")
2732*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wpadded")
2733*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wweak-vtables")
2734*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wglobal-constructors")
2735*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wexit-time-destructors")
2736*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-prototypes")
2737*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wsign-conversion")
2738*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wshorten-64-to-32")
2739*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-variable-declarations")
2740*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wswitch")
2741*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wswitch-enum")
2742*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wcovered-switch-default")
2743*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-noreturn")
2744*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wunused-local-typedef")
2745*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wdisabled-macro-expansion")
2746*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-braces")
2747*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wmissing-field-initializers")
2748*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wc++98-compat")
2749*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wc++98-compat-pedantic")
2750*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wunused-member-function")
2751*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING("-Wnonportable-system-include-path")
2752*89c4ff92SAndroid Build Coastguard Worker 
2753*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING_PUSH
2754*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wunknown-pragmas")
2755*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wpragmas")
2756*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wconversion")
2757*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Weffc++")
2758*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wsign-conversion")
2759*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wstrict-overflow")
2760*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wstrict-aliasing")
2761*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wmissing-field-initializers")
2762*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wmissing-braces")
2763*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wmissing-declarations")
2764*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wswitch")
2765*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wswitch-enum")
2766*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wswitch-default")
2767*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wunsafe-loop-optimizations")
2768*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wold-style-cast")
2769*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wunused-local-typedefs")
2770*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wuseless-cast")
2771*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wunused-function")
2772*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wmultiple-inheritance")
2773*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wnoexcept")
2774*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING("-Wsuggest-attribute")
2775*89c4ff92SAndroid Build Coastguard Worker 
2776*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_PUSH
2777*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4616) // invalid compiler warning
2778*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4619) // invalid compiler warning
2779*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4996) // The compiler encountered a deprecated declaration
2780*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4267) // 'var' : conversion from 'x' to 'y', possible loss of data
2781*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4706) // assignment within conditional expression
2782*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4512) // 'class' : assignment operator could not be generated
2783*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4127) // conditional expression is constant
2784*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4530) // C++ exception handler used, but unwind semantics not enabled
2785*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4577) // 'noexcept' used with no exception handling mode specified
2786*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4774) // format string expected in argument is not a string literal
2787*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4365) // conversion from 'int' to 'unsigned', signed/unsigned mismatch
2788*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4820) // padding in structs
2789*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4640) // construction of local static object is not thread-safe
2790*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(5039) // pointer to potentially throwing function passed to extern C
2791*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(5045) // Spectre mitigation stuff
2792*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4626) // assignment operator was implicitly defined as deleted
2793*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(5027) // move assignment operator was implicitly defined as deleted
2794*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(5026) // move constructor was implicitly defined as deleted
2795*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4625) // copy constructor was implicitly defined as deleted
2796*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(4800) // forcing value to bool 'true' or 'false' (performance warning)
2797*89c4ff92SAndroid Build Coastguard Worker // static analysis
2798*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(26439) // This kind of function may not throw. Declare it 'noexcept'
2799*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(26495) // Always initialize a member variable
2800*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(26451) // Arithmetic overflow ...
2801*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(26444) // Avoid unnamed objects with custom construction and dtor...
2802*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING(26812) // Prefer 'enum class' over 'enum'
2803*89c4ff92SAndroid Build Coastguard Worker 
2804*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MAKE_STD_HEADERS_CLEAN_FROM_WARNINGS_ON_WALL_BEGIN
2805*89c4ff92SAndroid Build Coastguard Worker 
2806*89c4ff92SAndroid Build Coastguard Worker // required includes - will go only in one translation unit!
2807*89c4ff92SAndroid Build Coastguard Worker #include <ctime>
2808*89c4ff92SAndroid Build Coastguard Worker #include <cmath>
2809*89c4ff92SAndroid Build Coastguard Worker #include <climits>
2810*89c4ff92SAndroid Build Coastguard Worker // borland (Embarcadero) compiler requires math.h and not cmath - https://github.com/onqtam/doctest/pull/37
2811*89c4ff92SAndroid Build Coastguard Worker #ifdef __BORLANDC__
2812*89c4ff92SAndroid Build Coastguard Worker #include <math.h>
2813*89c4ff92SAndroid Build Coastguard Worker #endif // __BORLANDC__
2814*89c4ff92SAndroid Build Coastguard Worker #include <new>
2815*89c4ff92SAndroid Build Coastguard Worker #include <cstdio>
2816*89c4ff92SAndroid Build Coastguard Worker #include <cstdlib>
2817*89c4ff92SAndroid Build Coastguard Worker #include <cstring>
2818*89c4ff92SAndroid Build Coastguard Worker #include <limits>
2819*89c4ff92SAndroid Build Coastguard Worker #include <utility>
2820*89c4ff92SAndroid Build Coastguard Worker #include <fstream>
2821*89c4ff92SAndroid Build Coastguard Worker #include <sstream>
2822*89c4ff92SAndroid Build Coastguard Worker #include <iostream>
2823*89c4ff92SAndroid Build Coastguard Worker #include <algorithm>
2824*89c4ff92SAndroid Build Coastguard Worker #include <iomanip>
2825*89c4ff92SAndroid Build Coastguard Worker #include <vector>
2826*89c4ff92SAndroid Build Coastguard Worker #include <atomic>
2827*89c4ff92SAndroid Build Coastguard Worker #include <mutex>
2828*89c4ff92SAndroid Build Coastguard Worker #include <set>
2829*89c4ff92SAndroid Build Coastguard Worker #include <map>
2830*89c4ff92SAndroid Build Coastguard Worker #include <exception>
2831*89c4ff92SAndroid Build Coastguard Worker #include <stdexcept>
2832*89c4ff92SAndroid Build Coastguard Worker #include <csignal>
2833*89c4ff92SAndroid Build Coastguard Worker #include <cfloat>
2834*89c4ff92SAndroid Build Coastguard Worker #include <cctype>
2835*89c4ff92SAndroid Build Coastguard Worker #include <cstdint>
2836*89c4ff92SAndroid Build Coastguard Worker 
2837*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_MAC
2838*89c4ff92SAndroid Build Coastguard Worker #include <sys/types.h>
2839*89c4ff92SAndroid Build Coastguard Worker #include <unistd.h>
2840*89c4ff92SAndroid Build Coastguard Worker #include <sys/sysctl.h>
2841*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM_MAC
2842*89c4ff92SAndroid Build Coastguard Worker 
2843*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_WINDOWS
2844*89c4ff92SAndroid Build Coastguard Worker 
2845*89c4ff92SAndroid Build Coastguard Worker // defines for a leaner windows.h
2846*89c4ff92SAndroid Build Coastguard Worker #ifndef WIN32_LEAN_AND_MEAN
2847*89c4ff92SAndroid Build Coastguard Worker #define WIN32_LEAN_AND_MEAN
2848*89c4ff92SAndroid Build Coastguard Worker #endif // WIN32_LEAN_AND_MEAN
2849*89c4ff92SAndroid Build Coastguard Worker #ifndef NOMINMAX
2850*89c4ff92SAndroid Build Coastguard Worker #define NOMINMAX
2851*89c4ff92SAndroid Build Coastguard Worker #endif // NOMINMAX
2852*89c4ff92SAndroid Build Coastguard Worker 
2853*89c4ff92SAndroid Build Coastguard Worker // not sure what AfxWin.h is for - here I do what Catch does
2854*89c4ff92SAndroid Build Coastguard Worker #ifdef __AFXDLL
2855*89c4ff92SAndroid Build Coastguard Worker #include <AfxWin.h>
2856*89c4ff92SAndroid Build Coastguard Worker #else
2857*89c4ff92SAndroid Build Coastguard Worker #include <windows.h>
2858*89c4ff92SAndroid Build Coastguard Worker #endif
2859*89c4ff92SAndroid Build Coastguard Worker #include <io.h>
2860*89c4ff92SAndroid Build Coastguard Worker 
2861*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_PLATFORM_WINDOWS
2862*89c4ff92SAndroid Build Coastguard Worker 
2863*89c4ff92SAndroid Build Coastguard Worker #include <sys/time.h>
2864*89c4ff92SAndroid Build Coastguard Worker #include <unistd.h>
2865*89c4ff92SAndroid Build Coastguard Worker 
2866*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM_WINDOWS
2867*89c4ff92SAndroid Build Coastguard Worker 
2868*89c4ff92SAndroid Build Coastguard Worker // this is a fix for https://github.com/onqtam/doctest/issues/348
2869*89c4ff92SAndroid Build Coastguard Worker // https://mail.gnome.org/archives/xml/2012-January/msg00000.html
2870*89c4ff92SAndroid Build Coastguard Worker #if !defined(HAVE_UNISTD_H) && !defined(STDOUT_FILENO)
2871*89c4ff92SAndroid Build Coastguard Worker #define STDOUT_FILENO fileno(stdout)
2872*89c4ff92SAndroid Build Coastguard Worker #endif // HAVE_UNISTD_H
2873*89c4ff92SAndroid Build Coastguard Worker 
2874*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MAKE_STD_HEADERS_CLEAN_FROM_WARNINGS_ON_WALL_END
2875*89c4ff92SAndroid Build Coastguard Worker 
2876*89c4ff92SAndroid Build Coastguard Worker // counts the number of elements in a C array
2877*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_COUNTOF(x) (sizeof(x) / sizeof(x[0]))
2878*89c4ff92SAndroid Build Coastguard Worker 
2879*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_DISABLE
2880*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BRANCH_ON_DISABLED(if_disabled, if_not_disabled) if_disabled
2881*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_DISABLE
2882*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_BRANCH_ON_DISABLED(if_disabled, if_not_disabled) if_not_disabled
2883*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_DISABLE
2884*89c4ff92SAndroid Build Coastguard Worker 
2885*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_OPTIONS_PREFIX
2886*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_OPTIONS_PREFIX "dt-"
2887*89c4ff92SAndroid Build Coastguard Worker #endif
2888*89c4ff92SAndroid Build Coastguard Worker 
2889*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_THREAD_LOCAL
2890*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_THREAD_LOCAL thread_local
2891*89c4ff92SAndroid Build Coastguard Worker #endif
2892*89c4ff92SAndroid Build Coastguard Worker 
2893*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_MULTI_LANE_ATOMICS_THREAD_LANES
2894*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MULTI_LANE_ATOMICS_THREAD_LANES 32
2895*89c4ff92SAndroid Build Coastguard Worker #endif
2896*89c4ff92SAndroid Build Coastguard Worker 
2897*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_MULTI_LANE_ATOMICS_CACHE_LINE_SIZE
2898*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_MULTI_LANE_ATOMICS_CACHE_LINE_SIZE 64
2899*89c4ff92SAndroid Build Coastguard Worker #endif
2900*89c4ff92SAndroid Build Coastguard Worker 
2901*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_NO_UNPREFIXED_OPTIONS
2902*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_OPTIONS_PREFIX_DISPLAY DOCTEST_CONFIG_OPTIONS_PREFIX
2903*89c4ff92SAndroid Build Coastguard Worker #else
2904*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_OPTIONS_PREFIX_DISPLAY ""
2905*89c4ff92SAndroid Build Coastguard Worker #endif
2906*89c4ff92SAndroid Build Coastguard Worker 
2907*89c4ff92SAndroid Build Coastguard Worker #if defined(WINAPI_FAMILY) && (WINAPI_FAMILY == WINAPI_FAMILY_APP)
2908*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_NO_MULTI_LANE_ATOMICS
2909*89c4ff92SAndroid Build Coastguard Worker #endif
2910*89c4ff92SAndroid Build Coastguard Worker 
2911*89c4ff92SAndroid Build Coastguard Worker namespace doctest {
2912*89c4ff92SAndroid Build Coastguard Worker 
2913*89c4ff92SAndroid Build Coastguard Worker bool is_running_in_test = false;
2914*89c4ff92SAndroid Build Coastguard Worker 
2915*89c4ff92SAndroid Build Coastguard Worker namespace {
2916*89c4ff92SAndroid Build Coastguard Worker     using namespace detail;
2917*89c4ff92SAndroid Build Coastguard Worker     // case insensitive strcmp
2918*89c4ff92SAndroid Build Coastguard Worker     int stricmp(const char* a, const char* b) {
2919*89c4ff92SAndroid Build Coastguard Worker         for(;; a++, b++) {
2920*89c4ff92SAndroid Build Coastguard Worker             const int d = tolower(*a) - tolower(*b);
2921*89c4ff92SAndroid Build Coastguard Worker             if(d != 0 || !*a)
2922*89c4ff92SAndroid Build Coastguard Worker                 return d;
2923*89c4ff92SAndroid Build Coastguard Worker         }
2924*89c4ff92SAndroid Build Coastguard Worker     }
2925*89c4ff92SAndroid Build Coastguard Worker 
2926*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
2927*89c4ff92SAndroid Build Coastguard Worker     String fpToString(T value, int precision) {
2928*89c4ff92SAndroid Build Coastguard Worker         std::ostringstream oss;
2929*89c4ff92SAndroid Build Coastguard Worker         oss << std::setprecision(precision) << std::fixed << value;
2930*89c4ff92SAndroid Build Coastguard Worker         std::string d = oss.str();
2931*89c4ff92SAndroid Build Coastguard Worker         size_t      i = d.find_last_not_of('0');
2932*89c4ff92SAndroid Build Coastguard Worker         if(i != std::string::npos && i != d.size() - 1) {
2933*89c4ff92SAndroid Build Coastguard Worker             if(d[i] == '.')
2934*89c4ff92SAndroid Build Coastguard Worker                 i++;
2935*89c4ff92SAndroid Build Coastguard Worker             d = d.substr(0, i + 1);
2936*89c4ff92SAndroid Build Coastguard Worker         }
2937*89c4ff92SAndroid Build Coastguard Worker         return d.c_str();
2938*89c4ff92SAndroid Build Coastguard Worker     }
2939*89c4ff92SAndroid Build Coastguard Worker 
2940*89c4ff92SAndroid Build Coastguard Worker     struct Endianness
2941*89c4ff92SAndroid Build Coastguard Worker     {
2942*89c4ff92SAndroid Build Coastguard Worker         enum Arch
2943*89c4ff92SAndroid Build Coastguard Worker         {
2944*89c4ff92SAndroid Build Coastguard Worker             Big,
2945*89c4ff92SAndroid Build Coastguard Worker             Little
2946*89c4ff92SAndroid Build Coastguard Worker         };
2947*89c4ff92SAndroid Build Coastguard Worker 
2948*89c4ff92SAndroid Build Coastguard Worker         static Arch which() {
2949*89c4ff92SAndroid Build Coastguard Worker             int x = 1;
2950*89c4ff92SAndroid Build Coastguard Worker             // casting any data pointer to char* is allowed
2951*89c4ff92SAndroid Build Coastguard Worker             auto ptr = reinterpret_cast<char*>(&x);
2952*89c4ff92SAndroid Build Coastguard Worker             if(*ptr)
2953*89c4ff92SAndroid Build Coastguard Worker                 return Little;
2954*89c4ff92SAndroid Build Coastguard Worker             return Big;
2955*89c4ff92SAndroid Build Coastguard Worker         }
2956*89c4ff92SAndroid Build Coastguard Worker     };
2957*89c4ff92SAndroid Build Coastguard Worker } // namespace
2958*89c4ff92SAndroid Build Coastguard Worker 
2959*89c4ff92SAndroid Build Coastguard Worker namespace detail {
2960*89c4ff92SAndroid Build Coastguard Worker     void my_memcpy(void* dest, const void* src, unsigned num) { memcpy(dest, src, num); }
2961*89c4ff92SAndroid Build Coastguard Worker 
2962*89c4ff92SAndroid Build Coastguard Worker     String rawMemoryToString(const void* object, unsigned size) {
2963*89c4ff92SAndroid Build Coastguard Worker         // Reverse order for little endian architectures
2964*89c4ff92SAndroid Build Coastguard Worker         int i = 0, end = static_cast<int>(size), inc = 1;
2965*89c4ff92SAndroid Build Coastguard Worker         if(Endianness::which() == Endianness::Little) {
2966*89c4ff92SAndroid Build Coastguard Worker             i   = end - 1;
2967*89c4ff92SAndroid Build Coastguard Worker             end = inc = -1;
2968*89c4ff92SAndroid Build Coastguard Worker         }
2969*89c4ff92SAndroid Build Coastguard Worker 
2970*89c4ff92SAndroid Build Coastguard Worker         unsigned const char* bytes = static_cast<unsigned const char*>(object);
2971*89c4ff92SAndroid Build Coastguard Worker         std::ostringstream   oss;
2972*89c4ff92SAndroid Build Coastguard Worker         oss << "0x" << std::setfill('0') << std::hex;
2973*89c4ff92SAndroid Build Coastguard Worker         for(; i != end; i += inc)
2974*89c4ff92SAndroid Build Coastguard Worker             oss << std::setw(2) << static_cast<unsigned>(bytes[i]);
2975*89c4ff92SAndroid Build Coastguard Worker         return oss.str().c_str();
2976*89c4ff92SAndroid Build Coastguard Worker     }
2977*89c4ff92SAndroid Build Coastguard Worker 
2978*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_THREAD_LOCAL std::ostringstream g_oss; // NOLINT(cert-err58-cpp)
2979*89c4ff92SAndroid Build Coastguard Worker 
2980*89c4ff92SAndroid Build Coastguard Worker     std::ostream* getTlsOss() {
2981*89c4ff92SAndroid Build Coastguard Worker         g_oss.clear(); // there shouldn't be anything worth clearing in the flags
2982*89c4ff92SAndroid Build Coastguard Worker         g_oss.str(""); // the slow way of resetting a string stream
2983*89c4ff92SAndroid Build Coastguard Worker         //g_oss.seekp(0); // optimal reset - as seen here: https://stackoverflow.com/a/624291/3162383
2984*89c4ff92SAndroid Build Coastguard Worker         return &g_oss;
2985*89c4ff92SAndroid Build Coastguard Worker     }
2986*89c4ff92SAndroid Build Coastguard Worker 
2987*89c4ff92SAndroid Build Coastguard Worker     String getTlsOssResult() {
2988*89c4ff92SAndroid Build Coastguard Worker         //g_oss << std::ends; // needed - as shown here: https://stackoverflow.com/a/624291/3162383
2989*89c4ff92SAndroid Build Coastguard Worker         return g_oss.str().c_str();
2990*89c4ff92SAndroid Build Coastguard Worker     }
2991*89c4ff92SAndroid Build Coastguard Worker 
2992*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_DISABLE
2993*89c4ff92SAndroid Build Coastguard Worker 
2994*89c4ff92SAndroid Build Coastguard Worker namespace timer_large_integer
2995*89c4ff92SAndroid Build Coastguard Worker {
2996*89c4ff92SAndroid Build Coastguard Worker 
2997*89c4ff92SAndroid Build Coastguard Worker #if defined(DOCTEST_PLATFORM_WINDOWS)
2998*89c4ff92SAndroid Build Coastguard Worker     typedef ULONGLONG type;
2999*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_PLATFORM_WINDOWS
3000*89c4ff92SAndroid Build Coastguard Worker     using namespace std;
3001*89c4ff92SAndroid Build Coastguard Worker     typedef uint64_t type;
3002*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM_WINDOWS
3003*89c4ff92SAndroid Build Coastguard Worker }
3004*89c4ff92SAndroid Build Coastguard Worker 
3005*89c4ff92SAndroid Build Coastguard Worker typedef timer_large_integer::type ticks_t;
3006*89c4ff92SAndroid Build Coastguard Worker 
3007*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_GETCURRENTTICKS
3008*89c4ff92SAndroid Build Coastguard Worker     ticks_t getCurrentTicks() { return DOCTEST_CONFIG_GETCURRENTTICKS(); }
3009*89c4ff92SAndroid Build Coastguard Worker #elif defined(DOCTEST_PLATFORM_WINDOWS)
3010*89c4ff92SAndroid Build Coastguard Worker     ticks_t getCurrentTicks() {
3011*89c4ff92SAndroid Build Coastguard Worker         static LARGE_INTEGER hz = {0}, hzo = {0};
3012*89c4ff92SAndroid Build Coastguard Worker         if(!hz.QuadPart) {
3013*89c4ff92SAndroid Build Coastguard Worker             QueryPerformanceFrequency(&hz);
3014*89c4ff92SAndroid Build Coastguard Worker             QueryPerformanceCounter(&hzo);
3015*89c4ff92SAndroid Build Coastguard Worker         }
3016*89c4ff92SAndroid Build Coastguard Worker         LARGE_INTEGER t;
3017*89c4ff92SAndroid Build Coastguard Worker         QueryPerformanceCounter(&t);
3018*89c4ff92SAndroid Build Coastguard Worker         return ((t.QuadPart - hzo.QuadPart) * LONGLONG(1000000)) / hz.QuadPart;
3019*89c4ff92SAndroid Build Coastguard Worker     }
3020*89c4ff92SAndroid Build Coastguard Worker #else  // DOCTEST_PLATFORM_WINDOWS
3021*89c4ff92SAndroid Build Coastguard Worker     ticks_t getCurrentTicks() {
3022*89c4ff92SAndroid Build Coastguard Worker         timeval t;
3023*89c4ff92SAndroid Build Coastguard Worker         gettimeofday(&t, nullptr);
3024*89c4ff92SAndroid Build Coastguard Worker         return static_cast<ticks_t>(t.tv_sec) * 1000000 + static_cast<ticks_t>(t.tv_usec);
3025*89c4ff92SAndroid Build Coastguard Worker     }
3026*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM_WINDOWS
3027*89c4ff92SAndroid Build Coastguard Worker 
3028*89c4ff92SAndroid Build Coastguard Worker     struct Timer
3029*89c4ff92SAndroid Build Coastguard Worker     {
3030*89c4ff92SAndroid Build Coastguard Worker         void         start() { m_ticks = getCurrentTicks(); }
3031*89c4ff92SAndroid Build Coastguard Worker         unsigned int getElapsedMicroseconds() const {
3032*89c4ff92SAndroid Build Coastguard Worker             return static_cast<unsigned int>(getCurrentTicks() - m_ticks);
3033*89c4ff92SAndroid Build Coastguard Worker         }
3034*89c4ff92SAndroid Build Coastguard Worker         //unsigned int getElapsedMilliseconds() const {
3035*89c4ff92SAndroid Build Coastguard Worker         //    return static_cast<unsigned int>(getElapsedMicroseconds() / 1000);
3036*89c4ff92SAndroid Build Coastguard Worker         //}
3037*89c4ff92SAndroid Build Coastguard Worker         double getElapsedSeconds() const { return static_cast<double>(getCurrentTicks() - m_ticks) / 1000000.0; }
3038*89c4ff92SAndroid Build Coastguard Worker 
3039*89c4ff92SAndroid Build Coastguard Worker     private:
3040*89c4ff92SAndroid Build Coastguard Worker         ticks_t m_ticks = 0;
3041*89c4ff92SAndroid Build Coastguard Worker     };
3042*89c4ff92SAndroid Build Coastguard Worker 
3043*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_NO_MULTI_LANE_ATOMICS
3044*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
3045*89c4ff92SAndroid Build Coastguard Worker     using AtomicOrMultiLaneAtomic = std::atomic<T>;
3046*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_NO_MULTI_LANE_ATOMICS
3047*89c4ff92SAndroid Build Coastguard Worker     // Provides a multilane implementation of an atomic variable that supports add, sub, load,
3048*89c4ff92SAndroid Build Coastguard Worker     // store. Instead of using a single atomic variable, this splits up into multiple ones,
3049*89c4ff92SAndroid Build Coastguard Worker     // each sitting on a separate cache line. The goal is to provide a speedup when most
3050*89c4ff92SAndroid Build Coastguard Worker     // operations are modifying. It achieves this with two properties:
3051*89c4ff92SAndroid Build Coastguard Worker     //
3052*89c4ff92SAndroid Build Coastguard Worker     // * Multiple atomics are used, so chance of congestion from the same atomic is reduced.
3053*89c4ff92SAndroid Build Coastguard Worker     // * Each atomic sits on a separate cache line, so false sharing is reduced.
3054*89c4ff92SAndroid Build Coastguard Worker     //
3055*89c4ff92SAndroid Build Coastguard Worker     // The disadvantage is that there is a small overhead due to the use of TLS, and load/store
3056*89c4ff92SAndroid Build Coastguard Worker     // is slower because all atomics have to be accessed.
3057*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
3058*89c4ff92SAndroid Build Coastguard Worker     class MultiLaneAtomic
3059*89c4ff92SAndroid Build Coastguard Worker     {
3060*89c4ff92SAndroid Build Coastguard Worker         struct CacheLineAlignedAtomic
3061*89c4ff92SAndroid Build Coastguard Worker         {
3062*89c4ff92SAndroid Build Coastguard Worker             std::atomic<T> atomic{};
3063*89c4ff92SAndroid Build Coastguard Worker             char padding[DOCTEST_MULTI_LANE_ATOMICS_CACHE_LINE_SIZE - sizeof(std::atomic<T>)];
3064*89c4ff92SAndroid Build Coastguard Worker         };
3065*89c4ff92SAndroid Build Coastguard Worker         CacheLineAlignedAtomic m_atomics[DOCTEST_MULTI_LANE_ATOMICS_THREAD_LANES];
3066*89c4ff92SAndroid Build Coastguard Worker 
3067*89c4ff92SAndroid Build Coastguard Worker         static_assert(sizeof(CacheLineAlignedAtomic) == DOCTEST_MULTI_LANE_ATOMICS_CACHE_LINE_SIZE,
3068*89c4ff92SAndroid Build Coastguard Worker                       "guarantee one atomic takes exactly one cache line");
3069*89c4ff92SAndroid Build Coastguard Worker 
3070*89c4ff92SAndroid Build Coastguard Worker     public:
3071*89c4ff92SAndroid Build Coastguard Worker         T operator++() DOCTEST_NOEXCEPT { return fetch_add(1) + 1; }
3072*89c4ff92SAndroid Build Coastguard Worker 
3073*89c4ff92SAndroid Build Coastguard Worker         T operator++(int) DOCTEST_NOEXCEPT { return fetch_add(1); }
3074*89c4ff92SAndroid Build Coastguard Worker 
3075*89c4ff92SAndroid Build Coastguard Worker         T fetch_add(T arg, std::memory_order order = std::memory_order_seq_cst) DOCTEST_NOEXCEPT {
3076*89c4ff92SAndroid Build Coastguard Worker             return myAtomic().fetch_add(arg, order);
3077*89c4ff92SAndroid Build Coastguard Worker         }
3078*89c4ff92SAndroid Build Coastguard Worker 
3079*89c4ff92SAndroid Build Coastguard Worker         T fetch_sub(T arg, std::memory_order order = std::memory_order_seq_cst) DOCTEST_NOEXCEPT {
3080*89c4ff92SAndroid Build Coastguard Worker             return myAtomic().fetch_sub(arg, order);
3081*89c4ff92SAndroid Build Coastguard Worker         }
3082*89c4ff92SAndroid Build Coastguard Worker 
3083*89c4ff92SAndroid Build Coastguard Worker         operator T() const DOCTEST_NOEXCEPT { return load(); }
3084*89c4ff92SAndroid Build Coastguard Worker 
3085*89c4ff92SAndroid Build Coastguard Worker         T load(std::memory_order order = std::memory_order_seq_cst) const DOCTEST_NOEXCEPT {
3086*89c4ff92SAndroid Build Coastguard Worker             auto result = T();
3087*89c4ff92SAndroid Build Coastguard Worker             for(auto const& c : m_atomics) {
3088*89c4ff92SAndroid Build Coastguard Worker                 result += c.atomic.load(order);
3089*89c4ff92SAndroid Build Coastguard Worker             }
3090*89c4ff92SAndroid Build Coastguard Worker             return result;
3091*89c4ff92SAndroid Build Coastguard Worker         }
3092*89c4ff92SAndroid Build Coastguard Worker 
3093*89c4ff92SAndroid Build Coastguard Worker         T operator=(T desired) DOCTEST_NOEXCEPT {
3094*89c4ff92SAndroid Build Coastguard Worker             store(desired);
3095*89c4ff92SAndroid Build Coastguard Worker             return desired;
3096*89c4ff92SAndroid Build Coastguard Worker         }
3097*89c4ff92SAndroid Build Coastguard Worker 
3098*89c4ff92SAndroid Build Coastguard Worker         void store(T desired, std::memory_order order = std::memory_order_seq_cst) DOCTEST_NOEXCEPT {
3099*89c4ff92SAndroid Build Coastguard Worker             // first value becomes desired", all others become 0.
3100*89c4ff92SAndroid Build Coastguard Worker             for(auto& c : m_atomics) {
3101*89c4ff92SAndroid Build Coastguard Worker                 c.atomic.store(desired, order);
3102*89c4ff92SAndroid Build Coastguard Worker                 desired = {};
3103*89c4ff92SAndroid Build Coastguard Worker             }
3104*89c4ff92SAndroid Build Coastguard Worker         }
3105*89c4ff92SAndroid Build Coastguard Worker 
3106*89c4ff92SAndroid Build Coastguard Worker     private:
3107*89c4ff92SAndroid Build Coastguard Worker         // Each thread has a different atomic that it operates on. If more than NumLanes threads
3108*89c4ff92SAndroid Build Coastguard Worker         // use this, some will use the same atomic. So performance will degrate a bit, but still
3109*89c4ff92SAndroid Build Coastguard Worker         // everything will work.
3110*89c4ff92SAndroid Build Coastguard Worker         //
3111*89c4ff92SAndroid Build Coastguard Worker         // The logic here is a bit tricky. The call should be as fast as possible, so that there
3112*89c4ff92SAndroid Build Coastguard Worker         // is minimal to no overhead in determining the correct atomic for the current thread.
3113*89c4ff92SAndroid Build Coastguard Worker         //
3114*89c4ff92SAndroid Build Coastguard Worker         // 1. A global static counter laneCounter counts continuously up.
3115*89c4ff92SAndroid Build Coastguard Worker         // 2. Each successive thread will use modulo operation of that counter so it gets an atomic
3116*89c4ff92SAndroid Build Coastguard Worker         //    assigned in a round-robin fashion.
3117*89c4ff92SAndroid Build Coastguard Worker         // 3. This tlsLaneIdx is stored in the thread local data, so it is directly available with
3118*89c4ff92SAndroid Build Coastguard Worker         //    little overhead.
3119*89c4ff92SAndroid Build Coastguard Worker         std::atomic<T>& myAtomic() DOCTEST_NOEXCEPT {
3120*89c4ff92SAndroid Build Coastguard Worker             static std::atomic<size_t> laneCounter;
3121*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_THREAD_LOCAL size_t tlsLaneIdx =
3122*89c4ff92SAndroid Build Coastguard Worker                     laneCounter++ % DOCTEST_MULTI_LANE_ATOMICS_THREAD_LANES;
3123*89c4ff92SAndroid Build Coastguard Worker 
3124*89c4ff92SAndroid Build Coastguard Worker             return m_atomics[tlsLaneIdx].atomic;
3125*89c4ff92SAndroid Build Coastguard Worker         }
3126*89c4ff92SAndroid Build Coastguard Worker     };
3127*89c4ff92SAndroid Build Coastguard Worker 
3128*89c4ff92SAndroid Build Coastguard Worker     template <typename T>
3129*89c4ff92SAndroid Build Coastguard Worker     using AtomicOrMultiLaneAtomic = MultiLaneAtomic<T>;
3130*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_MULTI_LANE_ATOMICS
3131*89c4ff92SAndroid Build Coastguard Worker 
3132*89c4ff92SAndroid Build Coastguard Worker     // this holds both parameters from the command line and runtime data for tests
3133*89c4ff92SAndroid Build Coastguard Worker     struct ContextState : ContextOptions, TestRunStats, CurrentTestCaseStats
3134*89c4ff92SAndroid Build Coastguard Worker     {
3135*89c4ff92SAndroid Build Coastguard Worker         AtomicOrMultiLaneAtomic<int> numAssertsCurrentTest_atomic;
3136*89c4ff92SAndroid Build Coastguard Worker         AtomicOrMultiLaneAtomic<int> numAssertsFailedCurrentTest_atomic;
3137*89c4ff92SAndroid Build Coastguard Worker 
3138*89c4ff92SAndroid Build Coastguard Worker         std::vector<std::vector<String>> filters = decltype(filters)(9); // 9 different filters
3139*89c4ff92SAndroid Build Coastguard Worker 
3140*89c4ff92SAndroid Build Coastguard Worker         std::vector<IReporter*> reporters_currently_used;
3141*89c4ff92SAndroid Build Coastguard Worker 
3142*89c4ff92SAndroid Build Coastguard Worker         assert_handler ah = nullptr;
3143*89c4ff92SAndroid Build Coastguard Worker 
3144*89c4ff92SAndroid Build Coastguard Worker         Timer timer;
3145*89c4ff92SAndroid Build Coastguard Worker 
3146*89c4ff92SAndroid Build Coastguard Worker         std::vector<String> stringifiedContexts; // logging from INFO() due to an exception
3147*89c4ff92SAndroid Build Coastguard Worker 
3148*89c4ff92SAndroid Build Coastguard Worker         // stuff for subcases
3149*89c4ff92SAndroid Build Coastguard Worker         std::vector<SubcaseSignature>     subcasesStack;
3150*89c4ff92SAndroid Build Coastguard Worker         std::set<decltype(subcasesStack)> subcasesPassed;
3151*89c4ff92SAndroid Build Coastguard Worker         int                               subcasesCurrentMaxLevel;
3152*89c4ff92SAndroid Build Coastguard Worker         bool                              should_reenter;
3153*89c4ff92SAndroid Build Coastguard Worker         std::atomic<bool>                 shouldLogCurrentException;
3154*89c4ff92SAndroid Build Coastguard Worker 
3155*89c4ff92SAndroid Build Coastguard Worker         void resetRunData() {
3156*89c4ff92SAndroid Build Coastguard Worker             numTestCases                = 0;
3157*89c4ff92SAndroid Build Coastguard Worker             numTestCasesPassingFilters  = 0;
3158*89c4ff92SAndroid Build Coastguard Worker             numTestSuitesPassingFilters = 0;
3159*89c4ff92SAndroid Build Coastguard Worker             numTestCasesFailed          = 0;
3160*89c4ff92SAndroid Build Coastguard Worker             numAsserts                  = 0;
3161*89c4ff92SAndroid Build Coastguard Worker             numAssertsFailed            = 0;
3162*89c4ff92SAndroid Build Coastguard Worker             numAssertsCurrentTest       = 0;
3163*89c4ff92SAndroid Build Coastguard Worker             numAssertsFailedCurrentTest = 0;
3164*89c4ff92SAndroid Build Coastguard Worker         }
3165*89c4ff92SAndroid Build Coastguard Worker 
3166*89c4ff92SAndroid Build Coastguard Worker         void finalizeTestCaseData() {
3167*89c4ff92SAndroid Build Coastguard Worker             seconds = timer.getElapsedSeconds();
3168*89c4ff92SAndroid Build Coastguard Worker 
3169*89c4ff92SAndroid Build Coastguard Worker             // update the non-atomic counters
3170*89c4ff92SAndroid Build Coastguard Worker             numAsserts += numAssertsCurrentTest_atomic;
3171*89c4ff92SAndroid Build Coastguard Worker             numAssertsFailed += numAssertsFailedCurrentTest_atomic;
3172*89c4ff92SAndroid Build Coastguard Worker             numAssertsCurrentTest       = numAssertsCurrentTest_atomic;
3173*89c4ff92SAndroid Build Coastguard Worker             numAssertsFailedCurrentTest = numAssertsFailedCurrentTest_atomic;
3174*89c4ff92SAndroid Build Coastguard Worker 
3175*89c4ff92SAndroid Build Coastguard Worker             if(numAssertsFailedCurrentTest)
3176*89c4ff92SAndroid Build Coastguard Worker                 failure_flags |= TestCaseFailureReason::AssertFailure;
3177*89c4ff92SAndroid Build Coastguard Worker 
3178*89c4ff92SAndroid Build Coastguard Worker             if(Approx(currentTest->m_timeout).epsilon(DBL_EPSILON) != 0 &&
3179*89c4ff92SAndroid Build Coastguard Worker                Approx(seconds).epsilon(DBL_EPSILON) > currentTest->m_timeout)
3180*89c4ff92SAndroid Build Coastguard Worker                 failure_flags |= TestCaseFailureReason::Timeout;
3181*89c4ff92SAndroid Build Coastguard Worker 
3182*89c4ff92SAndroid Build Coastguard Worker             if(currentTest->m_should_fail) {
3183*89c4ff92SAndroid Build Coastguard Worker                 if(failure_flags) {
3184*89c4ff92SAndroid Build Coastguard Worker                     failure_flags |= TestCaseFailureReason::ShouldHaveFailedAndDid;
3185*89c4ff92SAndroid Build Coastguard Worker                 } else {
3186*89c4ff92SAndroid Build Coastguard Worker                     failure_flags |= TestCaseFailureReason::ShouldHaveFailedButDidnt;
3187*89c4ff92SAndroid Build Coastguard Worker                 }
3188*89c4ff92SAndroid Build Coastguard Worker             } else if(failure_flags && currentTest->m_may_fail) {
3189*89c4ff92SAndroid Build Coastguard Worker                 failure_flags |= TestCaseFailureReason::CouldHaveFailedAndDid;
3190*89c4ff92SAndroid Build Coastguard Worker             } else if(currentTest->m_expected_failures > 0) {
3191*89c4ff92SAndroid Build Coastguard Worker                 if(numAssertsFailedCurrentTest == currentTest->m_expected_failures) {
3192*89c4ff92SAndroid Build Coastguard Worker                     failure_flags |= TestCaseFailureReason::FailedExactlyNumTimes;
3193*89c4ff92SAndroid Build Coastguard Worker                 } else {
3194*89c4ff92SAndroid Build Coastguard Worker                     failure_flags |= TestCaseFailureReason::DidntFailExactlyNumTimes;
3195*89c4ff92SAndroid Build Coastguard Worker                 }
3196*89c4ff92SAndroid Build Coastguard Worker             }
3197*89c4ff92SAndroid Build Coastguard Worker 
3198*89c4ff92SAndroid Build Coastguard Worker             bool ok_to_fail = (TestCaseFailureReason::ShouldHaveFailedAndDid & failure_flags) ||
3199*89c4ff92SAndroid Build Coastguard Worker                               (TestCaseFailureReason::CouldHaveFailedAndDid & failure_flags) ||
3200*89c4ff92SAndroid Build Coastguard Worker                               (TestCaseFailureReason::FailedExactlyNumTimes & failure_flags);
3201*89c4ff92SAndroid Build Coastguard Worker 
3202*89c4ff92SAndroid Build Coastguard Worker             // if any subcase has failed - the whole test case has failed
3203*89c4ff92SAndroid Build Coastguard Worker             if(failure_flags && !ok_to_fail)
3204*89c4ff92SAndroid Build Coastguard Worker                 numTestCasesFailed++;
3205*89c4ff92SAndroid Build Coastguard Worker         }
3206*89c4ff92SAndroid Build Coastguard Worker     };
3207*89c4ff92SAndroid Build Coastguard Worker 
3208*89c4ff92SAndroid Build Coastguard Worker     ContextState* g_cs = nullptr;
3209*89c4ff92SAndroid Build Coastguard Worker 
3210*89c4ff92SAndroid Build Coastguard Worker     // used to avoid locks for the debug output
3211*89c4ff92SAndroid Build Coastguard Worker     // TODO: figure out if this is indeed necessary/correct - seems like either there still
3212*89c4ff92SAndroid Build Coastguard Worker     // could be a race or that there wouldn't be a race even if using the context directly
3213*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_THREAD_LOCAL bool g_no_colors;
3214*89c4ff92SAndroid Build Coastguard Worker 
3215*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_DISABLE
3216*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
3217*89c4ff92SAndroid Build Coastguard Worker 
3218*89c4ff92SAndroid Build Coastguard Worker void String::setOnHeap() { *reinterpret_cast<unsigned char*>(&buf[last]) = 128; }
3219*89c4ff92SAndroid Build Coastguard Worker void String::setLast(unsigned in) { buf[last] = char(in); }
3220*89c4ff92SAndroid Build Coastguard Worker 
3221*89c4ff92SAndroid Build Coastguard Worker void String::copy(const String& other) {
3222*89c4ff92SAndroid Build Coastguard Worker     using namespace std;
3223*89c4ff92SAndroid Build Coastguard Worker     if(other.isOnStack()) {
3224*89c4ff92SAndroid Build Coastguard Worker         memcpy(buf, other.buf, len);
3225*89c4ff92SAndroid Build Coastguard Worker     } else {
3226*89c4ff92SAndroid Build Coastguard Worker         setOnHeap();
3227*89c4ff92SAndroid Build Coastguard Worker         data.size     = other.data.size;
3228*89c4ff92SAndroid Build Coastguard Worker         data.capacity = data.size + 1;
3229*89c4ff92SAndroid Build Coastguard Worker         data.ptr      = new char[data.capacity];
3230*89c4ff92SAndroid Build Coastguard Worker         memcpy(data.ptr, other.data.ptr, data.size + 1);
3231*89c4ff92SAndroid Build Coastguard Worker     }
3232*89c4ff92SAndroid Build Coastguard Worker }
3233*89c4ff92SAndroid Build Coastguard Worker 
3234*89c4ff92SAndroid Build Coastguard Worker String::String() {
3235*89c4ff92SAndroid Build Coastguard Worker     buf[0] = '\0';
3236*89c4ff92SAndroid Build Coastguard Worker     setLast();
3237*89c4ff92SAndroid Build Coastguard Worker }
3238*89c4ff92SAndroid Build Coastguard Worker 
3239*89c4ff92SAndroid Build Coastguard Worker String::~String() {
3240*89c4ff92SAndroid Build Coastguard Worker     if(!isOnStack())
3241*89c4ff92SAndroid Build Coastguard Worker         delete[] data.ptr;
3242*89c4ff92SAndroid Build Coastguard Worker     // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
3243*89c4ff92SAndroid Build Coastguard Worker }
3244*89c4ff92SAndroid Build Coastguard Worker 
3245*89c4ff92SAndroid Build Coastguard Worker String::String(const char* in)
3246*89c4ff92SAndroid Build Coastguard Worker         : String(in, strlen(in)) {}
3247*89c4ff92SAndroid Build Coastguard Worker 
3248*89c4ff92SAndroid Build Coastguard Worker String::String(const char* in, unsigned in_size) {
3249*89c4ff92SAndroid Build Coastguard Worker     using namespace std;
3250*89c4ff92SAndroid Build Coastguard Worker     if(in_size <= last) {
3251*89c4ff92SAndroid Build Coastguard Worker         memcpy(buf, in, in_size);
3252*89c4ff92SAndroid Build Coastguard Worker         buf[in_size] = '\0';
3253*89c4ff92SAndroid Build Coastguard Worker         setLast(last - in_size);
3254*89c4ff92SAndroid Build Coastguard Worker     } else {
3255*89c4ff92SAndroid Build Coastguard Worker         setOnHeap();
3256*89c4ff92SAndroid Build Coastguard Worker         data.size     = in_size;
3257*89c4ff92SAndroid Build Coastguard Worker         data.capacity = data.size + 1;
3258*89c4ff92SAndroid Build Coastguard Worker         data.ptr      = new char[data.capacity];
3259*89c4ff92SAndroid Build Coastguard Worker         memcpy(data.ptr, in, in_size);
3260*89c4ff92SAndroid Build Coastguard Worker         data.ptr[in_size] = '\0';
3261*89c4ff92SAndroid Build Coastguard Worker     }
3262*89c4ff92SAndroid Build Coastguard Worker }
3263*89c4ff92SAndroid Build Coastguard Worker 
3264*89c4ff92SAndroid Build Coastguard Worker String::String(const String& other) { copy(other); }
3265*89c4ff92SAndroid Build Coastguard Worker 
3266*89c4ff92SAndroid Build Coastguard Worker String& String::operator=(const String& other) {
3267*89c4ff92SAndroid Build Coastguard Worker     if(this != &other) {
3268*89c4ff92SAndroid Build Coastguard Worker         if(!isOnStack())
3269*89c4ff92SAndroid Build Coastguard Worker             delete[] data.ptr;
3270*89c4ff92SAndroid Build Coastguard Worker 
3271*89c4ff92SAndroid Build Coastguard Worker         copy(other);
3272*89c4ff92SAndroid Build Coastguard Worker     }
3273*89c4ff92SAndroid Build Coastguard Worker 
3274*89c4ff92SAndroid Build Coastguard Worker     return *this;
3275*89c4ff92SAndroid Build Coastguard Worker }
3276*89c4ff92SAndroid Build Coastguard Worker 
3277*89c4ff92SAndroid Build Coastguard Worker String& String::operator+=(const String& other) {
3278*89c4ff92SAndroid Build Coastguard Worker     const unsigned my_old_size = size();
3279*89c4ff92SAndroid Build Coastguard Worker     const unsigned other_size  = other.size();
3280*89c4ff92SAndroid Build Coastguard Worker     const unsigned total_size  = my_old_size + other_size;
3281*89c4ff92SAndroid Build Coastguard Worker     using namespace std;
3282*89c4ff92SAndroid Build Coastguard Worker     if(isOnStack()) {
3283*89c4ff92SAndroid Build Coastguard Worker         if(total_size < len) {
3284*89c4ff92SAndroid Build Coastguard Worker             // append to the current stack space
3285*89c4ff92SAndroid Build Coastguard Worker             memcpy(buf + my_old_size, other.c_str(), other_size + 1);
3286*89c4ff92SAndroid Build Coastguard Worker             // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
3287*89c4ff92SAndroid Build Coastguard Worker             setLast(last - total_size);
3288*89c4ff92SAndroid Build Coastguard Worker         } else {
3289*89c4ff92SAndroid Build Coastguard Worker             // alloc new chunk
3290*89c4ff92SAndroid Build Coastguard Worker             char* temp = new char[total_size + 1];
3291*89c4ff92SAndroid Build Coastguard Worker             // copy current data to new location before writing in the union
3292*89c4ff92SAndroid Build Coastguard Worker             memcpy(temp, buf, my_old_size); // skip the +1 ('\0') for speed
3293*89c4ff92SAndroid Build Coastguard Worker             // update data in union
3294*89c4ff92SAndroid Build Coastguard Worker             setOnHeap();
3295*89c4ff92SAndroid Build Coastguard Worker             data.size     = total_size;
3296*89c4ff92SAndroid Build Coastguard Worker             data.capacity = data.size + 1;
3297*89c4ff92SAndroid Build Coastguard Worker             data.ptr      = temp;
3298*89c4ff92SAndroid Build Coastguard Worker             // transfer the rest of the data
3299*89c4ff92SAndroid Build Coastguard Worker             memcpy(data.ptr + my_old_size, other.c_str(), other_size + 1);
3300*89c4ff92SAndroid Build Coastguard Worker         }
3301*89c4ff92SAndroid Build Coastguard Worker     } else {
3302*89c4ff92SAndroid Build Coastguard Worker         if(data.capacity > total_size) {
3303*89c4ff92SAndroid Build Coastguard Worker             // append to the current heap block
3304*89c4ff92SAndroid Build Coastguard Worker             data.size = total_size;
3305*89c4ff92SAndroid Build Coastguard Worker             memcpy(data.ptr + my_old_size, other.c_str(), other_size + 1);
3306*89c4ff92SAndroid Build Coastguard Worker         } else {
3307*89c4ff92SAndroid Build Coastguard Worker             // resize
3308*89c4ff92SAndroid Build Coastguard Worker             data.capacity *= 2;
3309*89c4ff92SAndroid Build Coastguard Worker             if(data.capacity <= total_size)
3310*89c4ff92SAndroid Build Coastguard Worker                 data.capacity = total_size + 1;
3311*89c4ff92SAndroid Build Coastguard Worker             // alloc new chunk
3312*89c4ff92SAndroid Build Coastguard Worker             char* temp = new char[data.capacity];
3313*89c4ff92SAndroid Build Coastguard Worker             // copy current data to new location before releasing it
3314*89c4ff92SAndroid Build Coastguard Worker             memcpy(temp, data.ptr, my_old_size); // skip the +1 ('\0') for speed
3315*89c4ff92SAndroid Build Coastguard Worker             // release old chunk
3316*89c4ff92SAndroid Build Coastguard Worker             delete[] data.ptr;
3317*89c4ff92SAndroid Build Coastguard Worker             // update the rest of the union members
3318*89c4ff92SAndroid Build Coastguard Worker             data.size = total_size;
3319*89c4ff92SAndroid Build Coastguard Worker             data.ptr  = temp;
3320*89c4ff92SAndroid Build Coastguard Worker             // transfer the rest of the data
3321*89c4ff92SAndroid Build Coastguard Worker             memcpy(data.ptr + my_old_size, other.c_str(), other_size + 1);
3322*89c4ff92SAndroid Build Coastguard Worker         }
3323*89c4ff92SAndroid Build Coastguard Worker     }
3324*89c4ff92SAndroid Build Coastguard Worker 
3325*89c4ff92SAndroid Build Coastguard Worker     return *this;
3326*89c4ff92SAndroid Build Coastguard Worker }
3327*89c4ff92SAndroid Build Coastguard Worker 
3328*89c4ff92SAndroid Build Coastguard Worker // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
3329*89c4ff92SAndroid Build Coastguard Worker String String::operator+(const String& other) const { return String(*this) += other; }
3330*89c4ff92SAndroid Build Coastguard Worker 
3331*89c4ff92SAndroid Build Coastguard Worker String::String(String&& other) {
3332*89c4ff92SAndroid Build Coastguard Worker     using namespace std;
3333*89c4ff92SAndroid Build Coastguard Worker     memcpy(buf, other.buf, len);
3334*89c4ff92SAndroid Build Coastguard Worker     other.buf[0] = '\0';
3335*89c4ff92SAndroid Build Coastguard Worker     other.setLast();
3336*89c4ff92SAndroid Build Coastguard Worker }
3337*89c4ff92SAndroid Build Coastguard Worker 
3338*89c4ff92SAndroid Build Coastguard Worker String& String::operator=(String&& other) {
3339*89c4ff92SAndroid Build Coastguard Worker     using namespace std;
3340*89c4ff92SAndroid Build Coastguard Worker     if(this != &other) {
3341*89c4ff92SAndroid Build Coastguard Worker         if(!isOnStack())
3342*89c4ff92SAndroid Build Coastguard Worker             delete[] data.ptr;
3343*89c4ff92SAndroid Build Coastguard Worker         memcpy(buf, other.buf, len);
3344*89c4ff92SAndroid Build Coastguard Worker         other.buf[0] = '\0';
3345*89c4ff92SAndroid Build Coastguard Worker         other.setLast();
3346*89c4ff92SAndroid Build Coastguard Worker     }
3347*89c4ff92SAndroid Build Coastguard Worker     return *this;
3348*89c4ff92SAndroid Build Coastguard Worker }
3349*89c4ff92SAndroid Build Coastguard Worker 
3350*89c4ff92SAndroid Build Coastguard Worker char String::operator[](unsigned i) const {
3351*89c4ff92SAndroid Build Coastguard Worker     return const_cast<String*>(this)->operator[](i); // NOLINT
3352*89c4ff92SAndroid Build Coastguard Worker }
3353*89c4ff92SAndroid Build Coastguard Worker 
3354*89c4ff92SAndroid Build Coastguard Worker char& String::operator[](unsigned i) {
3355*89c4ff92SAndroid Build Coastguard Worker     if(isOnStack())
3356*89c4ff92SAndroid Build Coastguard Worker         return reinterpret_cast<char*>(buf)[i];
3357*89c4ff92SAndroid Build Coastguard Worker     return data.ptr[i];
3358*89c4ff92SAndroid Build Coastguard Worker }
3359*89c4ff92SAndroid Build Coastguard Worker 
3360*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wmaybe-uninitialized")
3361*89c4ff92SAndroid Build Coastguard Worker unsigned String::size() const {
3362*89c4ff92SAndroid Build Coastguard Worker     if(isOnStack())
3363*89c4ff92SAndroid Build Coastguard Worker         return last - (unsigned(buf[last]) & 31); // using "last" would work only if "len" is 32
3364*89c4ff92SAndroid Build Coastguard Worker     return data.size;
3365*89c4ff92SAndroid Build Coastguard Worker }
3366*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING_POP
3367*89c4ff92SAndroid Build Coastguard Worker 
3368*89c4ff92SAndroid Build Coastguard Worker unsigned String::capacity() const {
3369*89c4ff92SAndroid Build Coastguard Worker     if(isOnStack())
3370*89c4ff92SAndroid Build Coastguard Worker         return len;
3371*89c4ff92SAndroid Build Coastguard Worker     return data.capacity;
3372*89c4ff92SAndroid Build Coastguard Worker }
3373*89c4ff92SAndroid Build Coastguard Worker 
3374*89c4ff92SAndroid Build Coastguard Worker int String::compare(const char* other, bool no_case) const {
3375*89c4ff92SAndroid Build Coastguard Worker     if(no_case)
3376*89c4ff92SAndroid Build Coastguard Worker         return doctest::stricmp(c_str(), other);
3377*89c4ff92SAndroid Build Coastguard Worker     return std::strcmp(c_str(), other);
3378*89c4ff92SAndroid Build Coastguard Worker }
3379*89c4ff92SAndroid Build Coastguard Worker 
3380*89c4ff92SAndroid Build Coastguard Worker int String::compare(const String& other, bool no_case) const {
3381*89c4ff92SAndroid Build Coastguard Worker     return compare(other.c_str(), no_case);
3382*89c4ff92SAndroid Build Coastguard Worker }
3383*89c4ff92SAndroid Build Coastguard Worker 
3384*89c4ff92SAndroid Build Coastguard Worker // clang-format off
3385*89c4ff92SAndroid Build Coastguard Worker bool operator==(const String& lhs, const String& rhs) { return lhs.compare(rhs) == 0; }
3386*89c4ff92SAndroid Build Coastguard Worker bool operator!=(const String& lhs, const String& rhs) { return lhs.compare(rhs) != 0; }
3387*89c4ff92SAndroid Build Coastguard Worker bool operator< (const String& lhs, const String& rhs) { return lhs.compare(rhs) < 0; }
3388*89c4ff92SAndroid Build Coastguard Worker bool operator> (const String& lhs, const String& rhs) { return lhs.compare(rhs) > 0; }
3389*89c4ff92SAndroid Build Coastguard Worker bool operator<=(const String& lhs, const String& rhs) { return (lhs != rhs) ? lhs.compare(rhs) < 0 : true; }
3390*89c4ff92SAndroid Build Coastguard Worker bool operator>=(const String& lhs, const String& rhs) { return (lhs != rhs) ? lhs.compare(rhs) > 0 : true; }
3391*89c4ff92SAndroid Build Coastguard Worker // clang-format on
3392*89c4ff92SAndroid Build Coastguard Worker 
3393*89c4ff92SAndroid Build Coastguard Worker std::ostream& operator<<(std::ostream& s, const String& in) { return s << in.c_str(); }
3394*89c4ff92SAndroid Build Coastguard Worker 
3395*89c4ff92SAndroid Build Coastguard Worker namespace {
3396*89c4ff92SAndroid Build Coastguard Worker     void color_to_stream(std::ostream&, Color::Enum) DOCTEST_BRANCH_ON_DISABLED({}, ;)
3397*89c4ff92SAndroid Build Coastguard Worker } // namespace
3398*89c4ff92SAndroid Build Coastguard Worker 
3399*89c4ff92SAndroid Build Coastguard Worker namespace Color {
3400*89c4ff92SAndroid Build Coastguard Worker     std::ostream& operator<<(std::ostream& s, Color::Enum code) {
3401*89c4ff92SAndroid Build Coastguard Worker         color_to_stream(s, code);
3402*89c4ff92SAndroid Build Coastguard Worker         return s;
3403*89c4ff92SAndroid Build Coastguard Worker     }
3404*89c4ff92SAndroid Build Coastguard Worker } // namespace Color
3405*89c4ff92SAndroid Build Coastguard Worker 
3406*89c4ff92SAndroid Build Coastguard Worker // clang-format off
3407*89c4ff92SAndroid Build Coastguard Worker const char* assertString(assertType::Enum at) {
3408*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4062) // enum 'x' in switch of enum 'y' is not handled
3409*89c4ff92SAndroid Build Coastguard Worker     switch(at) {  //!OCLINT missing default in switch statements
3410*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN                    : return "WARN";
3411*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK                   : return "CHECK";
3412*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE                 : return "REQUIRE";
3413*89c4ff92SAndroid Build Coastguard Worker 
3414*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_FALSE              : return "WARN_FALSE";
3415*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_FALSE             : return "CHECK_FALSE";
3416*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_FALSE           : return "REQUIRE_FALSE";
3417*89c4ff92SAndroid Build Coastguard Worker 
3418*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_THROWS             : return "WARN_THROWS";
3419*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_THROWS            : return "CHECK_THROWS";
3420*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_THROWS          : return "REQUIRE_THROWS";
3421*89c4ff92SAndroid Build Coastguard Worker 
3422*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_THROWS_AS          : return "WARN_THROWS_AS";
3423*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_THROWS_AS         : return "CHECK_THROWS_AS";
3424*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_THROWS_AS       : return "REQUIRE_THROWS_AS";
3425*89c4ff92SAndroid Build Coastguard Worker 
3426*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_THROWS_WITH        : return "WARN_THROWS_WITH";
3427*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_THROWS_WITH       : return "CHECK_THROWS_WITH";
3428*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_THROWS_WITH     : return "REQUIRE_THROWS_WITH";
3429*89c4ff92SAndroid Build Coastguard Worker 
3430*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_THROWS_WITH_AS     : return "WARN_THROWS_WITH_AS";
3431*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_THROWS_WITH_AS    : return "CHECK_THROWS_WITH_AS";
3432*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_THROWS_WITH_AS  : return "REQUIRE_THROWS_WITH_AS";
3433*89c4ff92SAndroid Build Coastguard Worker 
3434*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_NOTHROW            : return "WARN_NOTHROW";
3435*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_NOTHROW           : return "CHECK_NOTHROW";
3436*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_NOTHROW         : return "REQUIRE_NOTHROW";
3437*89c4ff92SAndroid Build Coastguard Worker 
3438*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_EQ                 : return "WARN_EQ";
3439*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_EQ                : return "CHECK_EQ";
3440*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_EQ              : return "REQUIRE_EQ";
3441*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_NE                 : return "WARN_NE";
3442*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_NE                : return "CHECK_NE";
3443*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_NE              : return "REQUIRE_NE";
3444*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_GT                 : return "WARN_GT";
3445*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_GT                : return "CHECK_GT";
3446*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_GT              : return "REQUIRE_GT";
3447*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_LT                 : return "WARN_LT";
3448*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_LT                : return "CHECK_LT";
3449*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_LT              : return "REQUIRE_LT";
3450*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_GE                 : return "WARN_GE";
3451*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_GE                : return "CHECK_GE";
3452*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_GE              : return "REQUIRE_GE";
3453*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_LE                 : return "WARN_LE";
3454*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_LE                : return "CHECK_LE";
3455*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_LE              : return "REQUIRE_LE";
3456*89c4ff92SAndroid Build Coastguard Worker 
3457*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_UNARY              : return "WARN_UNARY";
3458*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_UNARY             : return "CHECK_UNARY";
3459*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_UNARY           : return "REQUIRE_UNARY";
3460*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_WARN_UNARY_FALSE        : return "WARN_UNARY_FALSE";
3461*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_CHECK_UNARY_FALSE       : return "CHECK_UNARY_FALSE";
3462*89c4ff92SAndroid Build Coastguard Worker         case assertType::DT_REQUIRE_UNARY_FALSE     : return "REQUIRE_UNARY_FALSE";
3463*89c4ff92SAndroid Build Coastguard Worker     }
3464*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_POP
3465*89c4ff92SAndroid Build Coastguard Worker     return "";
3466*89c4ff92SAndroid Build Coastguard Worker }
3467*89c4ff92SAndroid Build Coastguard Worker // clang-format on
3468*89c4ff92SAndroid Build Coastguard Worker 
3469*89c4ff92SAndroid Build Coastguard Worker const char* failureString(assertType::Enum at) {
3470*89c4ff92SAndroid Build Coastguard Worker     if(at & assertType::is_warn) //!OCLINT bitwise operator in conditional
3471*89c4ff92SAndroid Build Coastguard Worker         return "WARNING";
3472*89c4ff92SAndroid Build Coastguard Worker     if(at & assertType::is_check) //!OCLINT bitwise operator in conditional
3473*89c4ff92SAndroid Build Coastguard Worker         return "ERROR";
3474*89c4ff92SAndroid Build Coastguard Worker     if(at & assertType::is_require) //!OCLINT bitwise operator in conditional
3475*89c4ff92SAndroid Build Coastguard Worker         return "FATAL ERROR";
3476*89c4ff92SAndroid Build Coastguard Worker     return "";
3477*89c4ff92SAndroid Build Coastguard Worker }
3478*89c4ff92SAndroid Build Coastguard Worker 
3479*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wnull-dereference")
3480*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wnull-dereference")
3481*89c4ff92SAndroid Build Coastguard Worker // depending on the current options this will remove the path of filenames
3482*89c4ff92SAndroid Build Coastguard Worker const char* skipPathFromFilename(const char* file) {
3483*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_DISABLE
3484*89c4ff92SAndroid Build Coastguard Worker     if(getContextOptions()->no_path_in_filenames) {
3485*89c4ff92SAndroid Build Coastguard Worker         auto back    = std::strrchr(file, '\\');
3486*89c4ff92SAndroid Build Coastguard Worker         auto forward = std::strrchr(file, '/');
3487*89c4ff92SAndroid Build Coastguard Worker         if(back || forward) {
3488*89c4ff92SAndroid Build Coastguard Worker             if(back > forward)
3489*89c4ff92SAndroid Build Coastguard Worker                 forward = back;
3490*89c4ff92SAndroid Build Coastguard Worker             return forward + 1;
3491*89c4ff92SAndroid Build Coastguard Worker         }
3492*89c4ff92SAndroid Build Coastguard Worker     }
3493*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_DISABLE
3494*89c4ff92SAndroid Build Coastguard Worker     return file;
3495*89c4ff92SAndroid Build Coastguard Worker }
3496*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING_POP
3497*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING_POP
3498*89c4ff92SAndroid Build Coastguard Worker 
3499*89c4ff92SAndroid Build Coastguard Worker bool SubcaseSignature::operator<(const SubcaseSignature& other) const {
3500*89c4ff92SAndroid Build Coastguard Worker     if(m_line != other.m_line)
3501*89c4ff92SAndroid Build Coastguard Worker         return m_line < other.m_line;
3502*89c4ff92SAndroid Build Coastguard Worker     if(std::strcmp(m_file, other.m_file) != 0)
3503*89c4ff92SAndroid Build Coastguard Worker         return std::strcmp(m_file, other.m_file) < 0;
3504*89c4ff92SAndroid Build Coastguard Worker     return m_name.compare(other.m_name) < 0;
3505*89c4ff92SAndroid Build Coastguard Worker }
3506*89c4ff92SAndroid Build Coastguard Worker 
3507*89c4ff92SAndroid Build Coastguard Worker IContextScope::IContextScope()  = default;
3508*89c4ff92SAndroid Build Coastguard Worker IContextScope::~IContextScope() = default;
3509*89c4ff92SAndroid Build Coastguard Worker 
3510*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
3511*89c4ff92SAndroid Build Coastguard Worker String toString(char* in) { return toString(static_cast<const char*>(in)); }
3512*89c4ff92SAndroid Build Coastguard Worker // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
3513*89c4ff92SAndroid Build Coastguard Worker String toString(const char* in) { return String("\"") + (in ? in : "{null string}") + "\""; }
3514*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
3515*89c4ff92SAndroid Build Coastguard Worker String toString(bool in) { return in ? "true" : "false"; }
3516*89c4ff92SAndroid Build Coastguard Worker String toString(float in) { return fpToString(in, 5) + "f"; }
3517*89c4ff92SAndroid Build Coastguard Worker String toString(double in) { return fpToString(in, 10); }
3518*89c4ff92SAndroid Build Coastguard Worker String toString(double long in) { return fpToString(in, 15); }
3519*89c4ff92SAndroid Build Coastguard Worker 
3520*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_TO_STRING_OVERLOAD(type, fmt)                                                      \
3521*89c4ff92SAndroid Build Coastguard Worker     String toString(type in) {                                                                     \
3522*89c4ff92SAndroid Build Coastguard Worker         char buf[64];                                                                              \
3523*89c4ff92SAndroid Build Coastguard Worker         std::sprintf(buf, fmt, in);                                                                \
3524*89c4ff92SAndroid Build Coastguard Worker         return buf;                                                                                \
3525*89c4ff92SAndroid Build Coastguard Worker     }
3526*89c4ff92SAndroid Build Coastguard Worker 
3527*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(char, "%d")
3528*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(char signed, "%d")
3529*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(char unsigned, "%u")
3530*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(int short, "%d")
3531*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(int short unsigned, "%u")
3532*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(int, "%d")
3533*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(unsigned, "%u")
3534*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(int long, "%ld")
3535*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(int long unsigned, "%lu")
3536*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(int long long, "%lld")
3537*89c4ff92SAndroid Build Coastguard Worker DOCTEST_TO_STRING_OVERLOAD(int long long unsigned, "%llu")
3538*89c4ff92SAndroid Build Coastguard Worker 
3539*89c4ff92SAndroid Build Coastguard Worker String toString(std::nullptr_t) { return "NULL"; }
3540*89c4ff92SAndroid Build Coastguard Worker 
3541*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_MSVC >= DOCTEST_COMPILER(19, 20, 0)
3542*89c4ff92SAndroid Build Coastguard Worker // see this issue on why this is needed: https://github.com/onqtam/doctest/issues/183
3543*89c4ff92SAndroid Build Coastguard Worker String toString(const std::string& in) { return in.c_str(); }
3544*89c4ff92SAndroid Build Coastguard Worker #endif // VS 2019
3545*89c4ff92SAndroid Build Coastguard Worker 
3546*89c4ff92SAndroid Build Coastguard Worker Approx::Approx(double value)
3547*89c4ff92SAndroid Build Coastguard Worker         : m_epsilon(static_cast<double>(std::numeric_limits<float>::epsilon()) * 100)
3548*89c4ff92SAndroid Build Coastguard Worker         , m_scale(1.0)
3549*89c4ff92SAndroid Build Coastguard Worker         , m_value(value) {}
3550*89c4ff92SAndroid Build Coastguard Worker 
3551*89c4ff92SAndroid Build Coastguard Worker Approx Approx::operator()(double value) const {
3552*89c4ff92SAndroid Build Coastguard Worker     Approx approx(value);
3553*89c4ff92SAndroid Build Coastguard Worker     approx.epsilon(m_epsilon);
3554*89c4ff92SAndroid Build Coastguard Worker     approx.scale(m_scale);
3555*89c4ff92SAndroid Build Coastguard Worker     return approx;
3556*89c4ff92SAndroid Build Coastguard Worker }
3557*89c4ff92SAndroid Build Coastguard Worker 
3558*89c4ff92SAndroid Build Coastguard Worker Approx& Approx::epsilon(double newEpsilon) {
3559*89c4ff92SAndroid Build Coastguard Worker     m_epsilon = newEpsilon;
3560*89c4ff92SAndroid Build Coastguard Worker     return *this;
3561*89c4ff92SAndroid Build Coastguard Worker }
3562*89c4ff92SAndroid Build Coastguard Worker Approx& Approx::scale(double newScale) {
3563*89c4ff92SAndroid Build Coastguard Worker     m_scale = newScale;
3564*89c4ff92SAndroid Build Coastguard Worker     return *this;
3565*89c4ff92SAndroid Build Coastguard Worker }
3566*89c4ff92SAndroid Build Coastguard Worker 
3567*89c4ff92SAndroid Build Coastguard Worker bool operator==(double lhs, const Approx& rhs) {
3568*89c4ff92SAndroid Build Coastguard Worker     // Thanks to Richard Harris for his help refining this formula
3569*89c4ff92SAndroid Build Coastguard Worker     return std::fabs(lhs - rhs.m_value) <
3570*89c4ff92SAndroid Build Coastguard Worker            rhs.m_epsilon * (rhs.m_scale + std::max<double>(std::fabs(lhs), std::fabs(rhs.m_value)));
3571*89c4ff92SAndroid Build Coastguard Worker }
3572*89c4ff92SAndroid Build Coastguard Worker bool operator==(const Approx& lhs, double rhs) { return operator==(rhs, lhs); }
3573*89c4ff92SAndroid Build Coastguard Worker bool operator!=(double lhs, const Approx& rhs) { return !operator==(lhs, rhs); }
3574*89c4ff92SAndroid Build Coastguard Worker bool operator!=(const Approx& lhs, double rhs) { return !operator==(rhs, lhs); }
3575*89c4ff92SAndroid Build Coastguard Worker bool operator<=(double lhs, const Approx& rhs) { return lhs < rhs.m_value || lhs == rhs; }
3576*89c4ff92SAndroid Build Coastguard Worker bool operator<=(const Approx& lhs, double rhs) { return lhs.m_value < rhs || lhs == rhs; }
3577*89c4ff92SAndroid Build Coastguard Worker bool operator>=(double lhs, const Approx& rhs) { return lhs > rhs.m_value || lhs == rhs; }
3578*89c4ff92SAndroid Build Coastguard Worker bool operator>=(const Approx& lhs, double rhs) { return lhs.m_value > rhs || lhs == rhs; }
3579*89c4ff92SAndroid Build Coastguard Worker bool operator<(double lhs, const Approx& rhs) { return lhs < rhs.m_value && lhs != rhs; }
3580*89c4ff92SAndroid Build Coastguard Worker bool operator<(const Approx& lhs, double rhs) { return lhs.m_value < rhs && lhs != rhs; }
3581*89c4ff92SAndroid Build Coastguard Worker bool operator>(double lhs, const Approx& rhs) { return lhs > rhs.m_value && lhs != rhs; }
3582*89c4ff92SAndroid Build Coastguard Worker bool operator>(const Approx& lhs, double rhs) { return lhs.m_value > rhs && lhs != rhs; }
3583*89c4ff92SAndroid Build Coastguard Worker 
3584*89c4ff92SAndroid Build Coastguard Worker String toString(const Approx& in) {
3585*89c4ff92SAndroid Build Coastguard Worker     // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
3586*89c4ff92SAndroid Build Coastguard Worker     return String("Approx( ") + doctest::toString(in.m_value) + " )";
3587*89c4ff92SAndroid Build Coastguard Worker }
3588*89c4ff92SAndroid Build Coastguard Worker const ContextOptions* getContextOptions() { return DOCTEST_BRANCH_ON_DISABLED(nullptr, g_cs); }
3589*89c4ff92SAndroid Build Coastguard Worker 
3590*89c4ff92SAndroid Build Coastguard Worker } // namespace doctest
3591*89c4ff92SAndroid Build Coastguard Worker 
3592*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_DISABLE
3593*89c4ff92SAndroid Build Coastguard Worker namespace doctest {
3594*89c4ff92SAndroid Build Coastguard Worker Context::Context(int, const char* const*) {}
3595*89c4ff92SAndroid Build Coastguard Worker Context::~Context() = default;
3596*89c4ff92SAndroid Build Coastguard Worker void Context::applyCommandLine(int, const char* const*) {}
3597*89c4ff92SAndroid Build Coastguard Worker void Context::addFilter(const char*, const char*) {}
3598*89c4ff92SAndroid Build Coastguard Worker void Context::clearFilters() {}
3599*89c4ff92SAndroid Build Coastguard Worker void Context::setOption(const char*, int) {}
3600*89c4ff92SAndroid Build Coastguard Worker void Context::setOption(const char*, const char*) {}
3601*89c4ff92SAndroid Build Coastguard Worker bool Context::shouldExit() { return false; }
3602*89c4ff92SAndroid Build Coastguard Worker void Context::setAsDefaultForAssertsOutOfTestCases() {}
3603*89c4ff92SAndroid Build Coastguard Worker void Context::setAssertHandler(detail::assert_handler) {}
3604*89c4ff92SAndroid Build Coastguard Worker int  Context::run() { return 0; }
3605*89c4ff92SAndroid Build Coastguard Worker 
3606*89c4ff92SAndroid Build Coastguard Worker IReporter::~IReporter() = default;
3607*89c4ff92SAndroid Build Coastguard Worker 
3608*89c4ff92SAndroid Build Coastguard Worker int                         IReporter::get_num_active_contexts() { return 0; }
3609*89c4ff92SAndroid Build Coastguard Worker const IContextScope* const* IReporter::get_active_contexts() { return nullptr; }
3610*89c4ff92SAndroid Build Coastguard Worker int                         IReporter::get_num_stringified_contexts() { return 0; }
3611*89c4ff92SAndroid Build Coastguard Worker const String*               IReporter::get_stringified_contexts() { return nullptr; }
3612*89c4ff92SAndroid Build Coastguard Worker 
3613*89c4ff92SAndroid Build Coastguard Worker int registerReporter(const char*, int, IReporter*) { return 0; }
3614*89c4ff92SAndroid Build Coastguard Worker 
3615*89c4ff92SAndroid Build Coastguard Worker } // namespace doctest
3616*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_DISABLE
3617*89c4ff92SAndroid Build Coastguard Worker 
3618*89c4ff92SAndroid Build Coastguard Worker #if !defined(DOCTEST_CONFIG_COLORS_NONE)
3619*89c4ff92SAndroid Build Coastguard Worker #if !defined(DOCTEST_CONFIG_COLORS_WINDOWS) && !defined(DOCTEST_CONFIG_COLORS_ANSI)
3620*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_WINDOWS
3621*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_COLORS_WINDOWS
3622*89c4ff92SAndroid Build Coastguard Worker #else // linux
3623*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_CONFIG_COLORS_ANSI
3624*89c4ff92SAndroid Build Coastguard Worker #endif // platform
3625*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_COLORS_WINDOWS && DOCTEST_CONFIG_COLORS_ANSI
3626*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_COLORS_NONE
3627*89c4ff92SAndroid Build Coastguard Worker 
3628*89c4ff92SAndroid Build Coastguard Worker namespace doctest_detail_test_suite_ns {
3629*89c4ff92SAndroid Build Coastguard Worker // holds the current test suite
3630*89c4ff92SAndroid Build Coastguard Worker doctest::detail::TestSuite& getCurrentTestSuite() {
3631*89c4ff92SAndroid Build Coastguard Worker     static doctest::detail::TestSuite data{};
3632*89c4ff92SAndroid Build Coastguard Worker     return data;
3633*89c4ff92SAndroid Build Coastguard Worker }
3634*89c4ff92SAndroid Build Coastguard Worker } // namespace doctest_detail_test_suite_ns
3635*89c4ff92SAndroid Build Coastguard Worker 
3636*89c4ff92SAndroid Build Coastguard Worker namespace doctest {
3637*89c4ff92SAndroid Build Coastguard Worker namespace {
3638*89c4ff92SAndroid Build Coastguard Worker     // the int (priority) is part of the key for automatic sorting - sadly one can register a
3639*89c4ff92SAndroid Build Coastguard Worker     // reporter with a duplicate name and a different priority but hopefully that won't happen often :|
3640*89c4ff92SAndroid Build Coastguard Worker     typedef std::map<std::pair<int, String>, reporterCreatorFunc> reporterMap;
3641*89c4ff92SAndroid Build Coastguard Worker 
3642*89c4ff92SAndroid Build Coastguard Worker     reporterMap& getReporters() {
3643*89c4ff92SAndroid Build Coastguard Worker         static reporterMap data;
3644*89c4ff92SAndroid Build Coastguard Worker         return data;
3645*89c4ff92SAndroid Build Coastguard Worker     }
3646*89c4ff92SAndroid Build Coastguard Worker     reporterMap& getListeners() {
3647*89c4ff92SAndroid Build Coastguard Worker         static reporterMap data;
3648*89c4ff92SAndroid Build Coastguard Worker         return data;
3649*89c4ff92SAndroid Build Coastguard Worker     }
3650*89c4ff92SAndroid Build Coastguard Worker } // namespace
3651*89c4ff92SAndroid Build Coastguard Worker namespace detail {
3652*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_ITERATE_THROUGH_REPORTERS(function, ...)                                           \
3653*89c4ff92SAndroid Build Coastguard Worker     for(auto& curr_rep : g_cs->reporters_currently_used)                                           \
3654*89c4ff92SAndroid Build Coastguard Worker     curr_rep->function(__VA_ARGS__)
3655*89c4ff92SAndroid Build Coastguard Worker 
3656*89c4ff92SAndroid Build Coastguard Worker     bool checkIfShouldThrow(assertType::Enum at) {
3657*89c4ff92SAndroid Build Coastguard Worker         if(at & assertType::is_require) //!OCLINT bitwise operator in conditional
3658*89c4ff92SAndroid Build Coastguard Worker             return true;
3659*89c4ff92SAndroid Build Coastguard Worker 
3660*89c4ff92SAndroid Build Coastguard Worker         if((at & assertType::is_check) //!OCLINT bitwise operator in conditional
3661*89c4ff92SAndroid Build Coastguard Worker            && getContextOptions()->abort_after > 0 &&
3662*89c4ff92SAndroid Build Coastguard Worker            (g_cs->numAssertsFailed + g_cs->numAssertsFailedCurrentTest_atomic) >=
3663*89c4ff92SAndroid Build Coastguard Worker                    getContextOptions()->abort_after)
3664*89c4ff92SAndroid Build Coastguard Worker             return true;
3665*89c4ff92SAndroid Build Coastguard Worker 
3666*89c4ff92SAndroid Build Coastguard Worker         return false;
3667*89c4ff92SAndroid Build Coastguard Worker     }
3668*89c4ff92SAndroid Build Coastguard Worker 
3669*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_EXCEPTIONS
3670*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_NORETURN void throwException() {
3671*89c4ff92SAndroid Build Coastguard Worker         g_cs->shouldLogCurrentException = false;
3672*89c4ff92SAndroid Build Coastguard Worker         throw TestFailureException();
3673*89c4ff92SAndroid Build Coastguard Worker     } // NOLINT(cert-err60-cpp)
3674*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_NO_EXCEPTIONS
3675*89c4ff92SAndroid Build Coastguard Worker     void throwException() {}
3676*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS
3677*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
3678*89c4ff92SAndroid Build Coastguard Worker 
3679*89c4ff92SAndroid Build Coastguard Worker namespace {
3680*89c4ff92SAndroid Build Coastguard Worker     using namespace detail;
3681*89c4ff92SAndroid Build Coastguard Worker     // matching of a string against a wildcard mask (case sensitivity configurable) taken from
3682*89c4ff92SAndroid Build Coastguard Worker     // https://www.codeproject.com/Articles/1088/Wildcard-string-compare-globbing
3683*89c4ff92SAndroid Build Coastguard Worker     int wildcmp(const char* str, const char* wild, bool caseSensitive) {
3684*89c4ff92SAndroid Build Coastguard Worker         const char* cp = str;
3685*89c4ff92SAndroid Build Coastguard Worker         const char* mp = wild;
3686*89c4ff92SAndroid Build Coastguard Worker 
3687*89c4ff92SAndroid Build Coastguard Worker         while((*str) && (*wild != '*')) {
3688*89c4ff92SAndroid Build Coastguard Worker             if((caseSensitive ? (*wild != *str) : (tolower(*wild) != tolower(*str))) &&
3689*89c4ff92SAndroid Build Coastguard Worker                (*wild != '?')) {
3690*89c4ff92SAndroid Build Coastguard Worker                 return 0;
3691*89c4ff92SAndroid Build Coastguard Worker             }
3692*89c4ff92SAndroid Build Coastguard Worker             wild++;
3693*89c4ff92SAndroid Build Coastguard Worker             str++;
3694*89c4ff92SAndroid Build Coastguard Worker         }
3695*89c4ff92SAndroid Build Coastguard Worker 
3696*89c4ff92SAndroid Build Coastguard Worker         while(*str) {
3697*89c4ff92SAndroid Build Coastguard Worker             if(*wild == '*') {
3698*89c4ff92SAndroid Build Coastguard Worker                 if(!*++wild) {
3699*89c4ff92SAndroid Build Coastguard Worker                     return 1;
3700*89c4ff92SAndroid Build Coastguard Worker                 }
3701*89c4ff92SAndroid Build Coastguard Worker                 mp = wild;
3702*89c4ff92SAndroid Build Coastguard Worker                 cp = str + 1;
3703*89c4ff92SAndroid Build Coastguard Worker             } else if((caseSensitive ? (*wild == *str) : (tolower(*wild) == tolower(*str))) ||
3704*89c4ff92SAndroid Build Coastguard Worker                       (*wild == '?')) {
3705*89c4ff92SAndroid Build Coastguard Worker                 wild++;
3706*89c4ff92SAndroid Build Coastguard Worker                 str++;
3707*89c4ff92SAndroid Build Coastguard Worker             } else {
3708*89c4ff92SAndroid Build Coastguard Worker                 wild = mp;   //!OCLINT parameter reassignment
3709*89c4ff92SAndroid Build Coastguard Worker                 str  = cp++; //!OCLINT parameter reassignment
3710*89c4ff92SAndroid Build Coastguard Worker             }
3711*89c4ff92SAndroid Build Coastguard Worker         }
3712*89c4ff92SAndroid Build Coastguard Worker 
3713*89c4ff92SAndroid Build Coastguard Worker         while(*wild == '*') {
3714*89c4ff92SAndroid Build Coastguard Worker             wild++;
3715*89c4ff92SAndroid Build Coastguard Worker         }
3716*89c4ff92SAndroid Build Coastguard Worker         return !*wild;
3717*89c4ff92SAndroid Build Coastguard Worker     }
3718*89c4ff92SAndroid Build Coastguard Worker 
3719*89c4ff92SAndroid Build Coastguard Worker     //// C string hash function (djb2) - taken from http://www.cse.yorku.ca/~oz/hash.html
3720*89c4ff92SAndroid Build Coastguard Worker     //unsigned hashStr(unsigned const char* str) {
3721*89c4ff92SAndroid Build Coastguard Worker     //    unsigned long hash = 5381;
3722*89c4ff92SAndroid Build Coastguard Worker     //    char          c;
3723*89c4ff92SAndroid Build Coastguard Worker     //    while((c = *str++))
3724*89c4ff92SAndroid Build Coastguard Worker     //        hash = ((hash << 5) + hash) + c; // hash * 33 + c
3725*89c4ff92SAndroid Build Coastguard Worker     //    return hash;
3726*89c4ff92SAndroid Build Coastguard Worker     //}
3727*89c4ff92SAndroid Build Coastguard Worker 
3728*89c4ff92SAndroid Build Coastguard Worker     // checks if the name matches any of the filters (and can be configured what to do when empty)
3729*89c4ff92SAndroid Build Coastguard Worker     bool matchesAny(const char* name, const std::vector<String>& filters, bool matchEmpty,
3730*89c4ff92SAndroid Build Coastguard Worker                     bool caseSensitive) {
3731*89c4ff92SAndroid Build Coastguard Worker         if(filters.empty() && matchEmpty)
3732*89c4ff92SAndroid Build Coastguard Worker             return true;
3733*89c4ff92SAndroid Build Coastguard Worker         for(auto& curr : filters)
3734*89c4ff92SAndroid Build Coastguard Worker             if(wildcmp(name, curr.c_str(), caseSensitive))
3735*89c4ff92SAndroid Build Coastguard Worker                 return true;
3736*89c4ff92SAndroid Build Coastguard Worker         return false;
3737*89c4ff92SAndroid Build Coastguard Worker     }
3738*89c4ff92SAndroid Build Coastguard Worker } // namespace
3739*89c4ff92SAndroid Build Coastguard Worker namespace detail {
3740*89c4ff92SAndroid Build Coastguard Worker 
3741*89c4ff92SAndroid Build Coastguard Worker     Subcase::Subcase(const String& name, const char* file, int line)
3742*89c4ff92SAndroid Build Coastguard Worker             : m_signature({name, file, line}) {
3743*89c4ff92SAndroid Build Coastguard Worker         auto* s = g_cs;
3744*89c4ff92SAndroid Build Coastguard Worker 
3745*89c4ff92SAndroid Build Coastguard Worker         // check subcase filters
3746*89c4ff92SAndroid Build Coastguard Worker         if(s->subcasesStack.size() < size_t(s->subcase_filter_levels)) {
3747*89c4ff92SAndroid Build Coastguard Worker             if(!matchesAny(m_signature.m_name.c_str(), s->filters[6], true, s->case_sensitive))
3748*89c4ff92SAndroid Build Coastguard Worker                 return;
3749*89c4ff92SAndroid Build Coastguard Worker             if(matchesAny(m_signature.m_name.c_str(), s->filters[7], false, s->case_sensitive))
3750*89c4ff92SAndroid Build Coastguard Worker                 return;
3751*89c4ff92SAndroid Build Coastguard Worker         }
3752*89c4ff92SAndroid Build Coastguard Worker 
3753*89c4ff92SAndroid Build Coastguard Worker         // if a Subcase on the same level has already been entered
3754*89c4ff92SAndroid Build Coastguard Worker         if(s->subcasesStack.size() < size_t(s->subcasesCurrentMaxLevel)) {
3755*89c4ff92SAndroid Build Coastguard Worker             s->should_reenter = true;
3756*89c4ff92SAndroid Build Coastguard Worker             return;
3757*89c4ff92SAndroid Build Coastguard Worker         }
3758*89c4ff92SAndroid Build Coastguard Worker 
3759*89c4ff92SAndroid Build Coastguard Worker         // push the current signature to the stack so we can check if the
3760*89c4ff92SAndroid Build Coastguard Worker         // current stack + the current new subcase have been traversed
3761*89c4ff92SAndroid Build Coastguard Worker         s->subcasesStack.push_back(m_signature);
3762*89c4ff92SAndroid Build Coastguard Worker         if(s->subcasesPassed.count(s->subcasesStack) != 0) {
3763*89c4ff92SAndroid Build Coastguard Worker             // pop - revert to previous stack since we've already passed this
3764*89c4ff92SAndroid Build Coastguard Worker             s->subcasesStack.pop_back();
3765*89c4ff92SAndroid Build Coastguard Worker             return;
3766*89c4ff92SAndroid Build Coastguard Worker         }
3767*89c4ff92SAndroid Build Coastguard Worker 
3768*89c4ff92SAndroid Build Coastguard Worker         s->subcasesCurrentMaxLevel = s->subcasesStack.size();
3769*89c4ff92SAndroid Build Coastguard Worker         m_entered = true;
3770*89c4ff92SAndroid Build Coastguard Worker 
3771*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ITERATE_THROUGH_REPORTERS(subcase_start, m_signature);
3772*89c4ff92SAndroid Build Coastguard Worker     }
3773*89c4ff92SAndroid Build Coastguard Worker 
3774*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4996) // std::uncaught_exception is deprecated in C++17
3775*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wdeprecated-declarations")
3776*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wdeprecated-declarations")
3777*89c4ff92SAndroid Build Coastguard Worker 
3778*89c4ff92SAndroid Build Coastguard Worker     Subcase::~Subcase() {
3779*89c4ff92SAndroid Build Coastguard Worker         if(m_entered) {
3780*89c4ff92SAndroid Build Coastguard Worker             // only mark the subcase stack as passed if no subcases have been skipped
3781*89c4ff92SAndroid Build Coastguard Worker             if(g_cs->should_reenter == false)
3782*89c4ff92SAndroid Build Coastguard Worker                 g_cs->subcasesPassed.insert(g_cs->subcasesStack);
3783*89c4ff92SAndroid Build Coastguard Worker             g_cs->subcasesStack.pop_back();
3784*89c4ff92SAndroid Build Coastguard Worker 
3785*89c4ff92SAndroid Build Coastguard Worker #if defined(__cpp_lib_uncaught_exceptions) && __cpp_lib_uncaught_exceptions >= 201411L && (!defined(__MAC_OS_X_VERSION_MIN_REQUIRED) || __MAC_OS_X_VERSION_MIN_REQUIRED >= 101200)
3786*89c4ff92SAndroid Build Coastguard Worker             if(std::uncaught_exceptions() > 0
3787*89c4ff92SAndroid Build Coastguard Worker #else
3788*89c4ff92SAndroid Build Coastguard Worker             if(std::uncaught_exception()
3789*89c4ff92SAndroid Build Coastguard Worker #endif
3790*89c4ff92SAndroid Build Coastguard Worker             && g_cs->shouldLogCurrentException) {
3791*89c4ff92SAndroid Build Coastguard Worker                 DOCTEST_ITERATE_THROUGH_REPORTERS(
3792*89c4ff92SAndroid Build Coastguard Worker                         test_case_exception, {"exception thrown in subcase - will translate later "
3793*89c4ff92SAndroid Build Coastguard Worker                                               "when the whole test case has been exited (cannot "
3794*89c4ff92SAndroid Build Coastguard Worker                                               "translate while there is an active exception)",
3795*89c4ff92SAndroid Build Coastguard Worker                                               false});
3796*89c4ff92SAndroid Build Coastguard Worker                 g_cs->shouldLogCurrentException = false;
3797*89c4ff92SAndroid Build Coastguard Worker             }
3798*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_ITERATE_THROUGH_REPORTERS(subcase_end, DOCTEST_EMPTY);
3799*89c4ff92SAndroid Build Coastguard Worker         }
3800*89c4ff92SAndroid Build Coastguard Worker     }
3801*89c4ff92SAndroid Build Coastguard Worker 
3802*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_POP
3803*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING_POP
3804*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_POP
3805*89c4ff92SAndroid Build Coastguard Worker 
3806*89c4ff92SAndroid Build Coastguard Worker     Subcase::operator bool() const { return m_entered; }
3807*89c4ff92SAndroid Build Coastguard Worker 
3808*89c4ff92SAndroid Build Coastguard Worker     Result::Result(bool passed, const String& decomposition)
3809*89c4ff92SAndroid Build Coastguard Worker             : m_passed(passed)
3810*89c4ff92SAndroid Build Coastguard Worker             , m_decomp(decomposition) {}
3811*89c4ff92SAndroid Build Coastguard Worker 
3812*89c4ff92SAndroid Build Coastguard Worker     ExpressionDecomposer::ExpressionDecomposer(assertType::Enum at)
3813*89c4ff92SAndroid Build Coastguard Worker             : m_at(at) {}
3814*89c4ff92SAndroid Build Coastguard Worker 
3815*89c4ff92SAndroid Build Coastguard Worker     TestSuite& TestSuite::operator*(const char* in) {
3816*89c4ff92SAndroid Build Coastguard Worker         m_test_suite = in;
3817*89c4ff92SAndroid Build Coastguard Worker         // clear state
3818*89c4ff92SAndroid Build Coastguard Worker         m_description       = nullptr;
3819*89c4ff92SAndroid Build Coastguard Worker         m_skip              = false;
3820*89c4ff92SAndroid Build Coastguard Worker         m_no_breaks         = false;
3821*89c4ff92SAndroid Build Coastguard Worker         m_no_output         = false;
3822*89c4ff92SAndroid Build Coastguard Worker         m_may_fail          = false;
3823*89c4ff92SAndroid Build Coastguard Worker         m_should_fail       = false;
3824*89c4ff92SAndroid Build Coastguard Worker         m_expected_failures = 0;
3825*89c4ff92SAndroid Build Coastguard Worker         m_timeout           = 0;
3826*89c4ff92SAndroid Build Coastguard Worker         return *this;
3827*89c4ff92SAndroid Build Coastguard Worker     }
3828*89c4ff92SAndroid Build Coastguard Worker 
3829*89c4ff92SAndroid Build Coastguard Worker     TestCase::TestCase(funcType test, const char* file, unsigned line, const TestSuite& test_suite,
3830*89c4ff92SAndroid Build Coastguard Worker                        const char* type, int template_id) {
3831*89c4ff92SAndroid Build Coastguard Worker         m_file              = file;
3832*89c4ff92SAndroid Build Coastguard Worker         m_line              = line;
3833*89c4ff92SAndroid Build Coastguard Worker         m_name              = nullptr; // will be later overridden in operator*
3834*89c4ff92SAndroid Build Coastguard Worker         m_test_suite        = test_suite.m_test_suite;
3835*89c4ff92SAndroid Build Coastguard Worker         m_description       = test_suite.m_description;
3836*89c4ff92SAndroid Build Coastguard Worker         m_skip              = test_suite.m_skip;
3837*89c4ff92SAndroid Build Coastguard Worker         m_no_breaks         = test_suite.m_no_breaks;
3838*89c4ff92SAndroid Build Coastguard Worker         m_no_output         = test_suite.m_no_output;
3839*89c4ff92SAndroid Build Coastguard Worker         m_may_fail          = test_suite.m_may_fail;
3840*89c4ff92SAndroid Build Coastguard Worker         m_should_fail       = test_suite.m_should_fail;
3841*89c4ff92SAndroid Build Coastguard Worker         m_expected_failures = test_suite.m_expected_failures;
3842*89c4ff92SAndroid Build Coastguard Worker         m_timeout           = test_suite.m_timeout;
3843*89c4ff92SAndroid Build Coastguard Worker 
3844*89c4ff92SAndroid Build Coastguard Worker         m_test        = test;
3845*89c4ff92SAndroid Build Coastguard Worker         m_type        = type;
3846*89c4ff92SAndroid Build Coastguard Worker         m_template_id = template_id;
3847*89c4ff92SAndroid Build Coastguard Worker     }
3848*89c4ff92SAndroid Build Coastguard Worker 
3849*89c4ff92SAndroid Build Coastguard Worker     TestCase::TestCase(const TestCase& other)
3850*89c4ff92SAndroid Build Coastguard Worker             : TestCaseData() {
3851*89c4ff92SAndroid Build Coastguard Worker         *this = other;
3852*89c4ff92SAndroid Build Coastguard Worker     }
3853*89c4ff92SAndroid Build Coastguard Worker 
3854*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(26434) // hides a non-virtual function
3855*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING(26437)           // Do not slice
3856*89c4ff92SAndroid Build Coastguard Worker     TestCase& TestCase::operator=(const TestCase& other) {
3857*89c4ff92SAndroid Build Coastguard Worker         static_cast<TestCaseData&>(*this) = static_cast<const TestCaseData&>(other);
3858*89c4ff92SAndroid Build Coastguard Worker 
3859*89c4ff92SAndroid Build Coastguard Worker         m_test        = other.m_test;
3860*89c4ff92SAndroid Build Coastguard Worker         m_type        = other.m_type;
3861*89c4ff92SAndroid Build Coastguard Worker         m_template_id = other.m_template_id;
3862*89c4ff92SAndroid Build Coastguard Worker         m_full_name   = other.m_full_name;
3863*89c4ff92SAndroid Build Coastguard Worker 
3864*89c4ff92SAndroid Build Coastguard Worker         if(m_template_id != -1)
3865*89c4ff92SAndroid Build Coastguard Worker             m_name = m_full_name.c_str();
3866*89c4ff92SAndroid Build Coastguard Worker         return *this;
3867*89c4ff92SAndroid Build Coastguard Worker     }
3868*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_POP
3869*89c4ff92SAndroid Build Coastguard Worker 
3870*89c4ff92SAndroid Build Coastguard Worker     TestCase& TestCase::operator*(const char* in) {
3871*89c4ff92SAndroid Build Coastguard Worker         m_name = in;
3872*89c4ff92SAndroid Build Coastguard Worker         // make a new name with an appended type for templated test case
3873*89c4ff92SAndroid Build Coastguard Worker         if(m_template_id != -1) {
3874*89c4ff92SAndroid Build Coastguard Worker             m_full_name = String(m_name) + m_type;
3875*89c4ff92SAndroid Build Coastguard Worker             // redirect the name to point to the newly constructed full name
3876*89c4ff92SAndroid Build Coastguard Worker             m_name = m_full_name.c_str();
3877*89c4ff92SAndroid Build Coastguard Worker         }
3878*89c4ff92SAndroid Build Coastguard Worker         return *this;
3879*89c4ff92SAndroid Build Coastguard Worker     }
3880*89c4ff92SAndroid Build Coastguard Worker 
3881*89c4ff92SAndroid Build Coastguard Worker     bool TestCase::operator<(const TestCase& other) const {
3882*89c4ff92SAndroid Build Coastguard Worker         // this will be used only to differentiate between test cases - not relevant for sorting
3883*89c4ff92SAndroid Build Coastguard Worker         if(m_line != other.m_line)
3884*89c4ff92SAndroid Build Coastguard Worker             return m_line < other.m_line;
3885*89c4ff92SAndroid Build Coastguard Worker         const int name_cmp = strcmp(m_name, other.m_name);
3886*89c4ff92SAndroid Build Coastguard Worker         if(name_cmp != 0)
3887*89c4ff92SAndroid Build Coastguard Worker             return name_cmp < 0;
3888*89c4ff92SAndroid Build Coastguard Worker         const int file_cmp = m_file.compare(other.m_file);
3889*89c4ff92SAndroid Build Coastguard Worker         if(file_cmp != 0)
3890*89c4ff92SAndroid Build Coastguard Worker             return file_cmp < 0;
3891*89c4ff92SAndroid Build Coastguard Worker         return m_template_id < other.m_template_id;
3892*89c4ff92SAndroid Build Coastguard Worker     }
3893*89c4ff92SAndroid Build Coastguard Worker 
3894*89c4ff92SAndroid Build Coastguard Worker     // all the registered tests
3895*89c4ff92SAndroid Build Coastguard Worker     std::set<TestCase>& getRegisteredTests() {
3896*89c4ff92SAndroid Build Coastguard Worker         static std::set<TestCase> data;
3897*89c4ff92SAndroid Build Coastguard Worker         return data;
3898*89c4ff92SAndroid Build Coastguard Worker     }
3899*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
3900*89c4ff92SAndroid Build Coastguard Worker namespace {
3901*89c4ff92SAndroid Build Coastguard Worker     using namespace detail;
3902*89c4ff92SAndroid Build Coastguard Worker     // for sorting tests by file/line
3903*89c4ff92SAndroid Build Coastguard Worker     bool fileOrderComparator(const TestCase* lhs, const TestCase* rhs) {
3904*89c4ff92SAndroid Build Coastguard Worker         // this is needed because MSVC gives different case for drive letters
3905*89c4ff92SAndroid Build Coastguard Worker         // for __FILE__ when evaluated in a header and a source file
3906*89c4ff92SAndroid Build Coastguard Worker         const int res = lhs->m_file.compare(rhs->m_file, bool(DOCTEST_MSVC));
3907*89c4ff92SAndroid Build Coastguard Worker         if(res != 0)
3908*89c4ff92SAndroid Build Coastguard Worker             return res < 0;
3909*89c4ff92SAndroid Build Coastguard Worker         if(lhs->m_line != rhs->m_line)
3910*89c4ff92SAndroid Build Coastguard Worker             return lhs->m_line < rhs->m_line;
3911*89c4ff92SAndroid Build Coastguard Worker         return lhs->m_template_id < rhs->m_template_id;
3912*89c4ff92SAndroid Build Coastguard Worker     }
3913*89c4ff92SAndroid Build Coastguard Worker 
3914*89c4ff92SAndroid Build Coastguard Worker     // for sorting tests by suite/file/line
3915*89c4ff92SAndroid Build Coastguard Worker     bool suiteOrderComparator(const TestCase* lhs, const TestCase* rhs) {
3916*89c4ff92SAndroid Build Coastguard Worker         const int res = std::strcmp(lhs->m_test_suite, rhs->m_test_suite);
3917*89c4ff92SAndroid Build Coastguard Worker         if(res != 0)
3918*89c4ff92SAndroid Build Coastguard Worker             return res < 0;
3919*89c4ff92SAndroid Build Coastguard Worker         return fileOrderComparator(lhs, rhs);
3920*89c4ff92SAndroid Build Coastguard Worker     }
3921*89c4ff92SAndroid Build Coastguard Worker 
3922*89c4ff92SAndroid Build Coastguard Worker     // for sorting tests by name/suite/file/line
3923*89c4ff92SAndroid Build Coastguard Worker     bool nameOrderComparator(const TestCase* lhs, const TestCase* rhs) {
3924*89c4ff92SAndroid Build Coastguard Worker         const int res = std::strcmp(lhs->m_name, rhs->m_name);
3925*89c4ff92SAndroid Build Coastguard Worker         if(res != 0)
3926*89c4ff92SAndroid Build Coastguard Worker             return res < 0;
3927*89c4ff92SAndroid Build Coastguard Worker         return suiteOrderComparator(lhs, rhs);
3928*89c4ff92SAndroid Build Coastguard Worker     }
3929*89c4ff92SAndroid Build Coastguard Worker 
3930*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_COLORS_WINDOWS
3931*89c4ff92SAndroid Build Coastguard Worker     HANDLE g_stdoutHandle;
3932*89c4ff92SAndroid Build Coastguard Worker     WORD   g_origFgAttrs;
3933*89c4ff92SAndroid Build Coastguard Worker     WORD   g_origBgAttrs;
3934*89c4ff92SAndroid Build Coastguard Worker     bool   g_attrsInitted = false;
3935*89c4ff92SAndroid Build Coastguard Worker 
3936*89c4ff92SAndroid Build Coastguard Worker     int colors_init() {
3937*89c4ff92SAndroid Build Coastguard Worker         if(!g_attrsInitted) {
3938*89c4ff92SAndroid Build Coastguard Worker             g_stdoutHandle = GetStdHandle(STD_OUTPUT_HANDLE);
3939*89c4ff92SAndroid Build Coastguard Worker             g_attrsInitted = true;
3940*89c4ff92SAndroid Build Coastguard Worker             CONSOLE_SCREEN_BUFFER_INFO csbiInfo;
3941*89c4ff92SAndroid Build Coastguard Worker             GetConsoleScreenBufferInfo(g_stdoutHandle, &csbiInfo);
3942*89c4ff92SAndroid Build Coastguard Worker             g_origFgAttrs = csbiInfo.wAttributes & ~(BACKGROUND_GREEN | BACKGROUND_RED |
3943*89c4ff92SAndroid Build Coastguard Worker                                                      BACKGROUND_BLUE | BACKGROUND_INTENSITY);
3944*89c4ff92SAndroid Build Coastguard Worker             g_origBgAttrs = csbiInfo.wAttributes & ~(FOREGROUND_GREEN | FOREGROUND_RED |
3945*89c4ff92SAndroid Build Coastguard Worker                                                      FOREGROUND_BLUE | FOREGROUND_INTENSITY);
3946*89c4ff92SAndroid Build Coastguard Worker         }
3947*89c4ff92SAndroid Build Coastguard Worker         return 0;
3948*89c4ff92SAndroid Build Coastguard Worker     }
3949*89c4ff92SAndroid Build Coastguard Worker 
3950*89c4ff92SAndroid Build Coastguard Worker     int dumy_init_console_colors = colors_init();
3951*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_COLORS_WINDOWS
3952*89c4ff92SAndroid Build Coastguard Worker 
3953*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wdeprecated-declarations")
3954*89c4ff92SAndroid Build Coastguard Worker     void color_to_stream(std::ostream& s, Color::Enum code) {
3955*89c4ff92SAndroid Build Coastguard Worker         static_cast<void>(s);    // for DOCTEST_CONFIG_COLORS_NONE or DOCTEST_CONFIG_COLORS_WINDOWS
3956*89c4ff92SAndroid Build Coastguard Worker         static_cast<void>(code); // for DOCTEST_CONFIG_COLORS_NONE
3957*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_COLORS_ANSI
3958*89c4ff92SAndroid Build Coastguard Worker         if(g_no_colors ||
3959*89c4ff92SAndroid Build Coastguard Worker            (isatty(STDOUT_FILENO) == false && getContextOptions()->force_colors == false))
3960*89c4ff92SAndroid Build Coastguard Worker             return;
3961*89c4ff92SAndroid Build Coastguard Worker 
3962*89c4ff92SAndroid Build Coastguard Worker         auto col = "";
3963*89c4ff92SAndroid Build Coastguard Worker         // clang-format off
3964*89c4ff92SAndroid Build Coastguard Worker             switch(code) { //!OCLINT missing break in switch statement / unnecessary default statement in covered switch statement
3965*89c4ff92SAndroid Build Coastguard Worker                 case Color::Red:         col = "[0;31m"; break;
3966*89c4ff92SAndroid Build Coastguard Worker                 case Color::Green:       col = "[0;32m"; break;
3967*89c4ff92SAndroid Build Coastguard Worker                 case Color::Blue:        col = "[0;34m"; break;
3968*89c4ff92SAndroid Build Coastguard Worker                 case Color::Cyan:        col = "[0;36m"; break;
3969*89c4ff92SAndroid Build Coastguard Worker                 case Color::Yellow:      col = "[0;33m"; break;
3970*89c4ff92SAndroid Build Coastguard Worker                 case Color::Grey:        col = "[1;30m"; break;
3971*89c4ff92SAndroid Build Coastguard Worker                 case Color::LightGrey:   col = "[0;37m"; break;
3972*89c4ff92SAndroid Build Coastguard Worker                 case Color::BrightRed:   col = "[1;31m"; break;
3973*89c4ff92SAndroid Build Coastguard Worker                 case Color::BrightGreen: col = "[1;32m"; break;
3974*89c4ff92SAndroid Build Coastguard Worker                 case Color::BrightWhite: col = "[1;37m"; break;
3975*89c4ff92SAndroid Build Coastguard Worker                 case Color::Bright: // invalid
3976*89c4ff92SAndroid Build Coastguard Worker                 case Color::None:
3977*89c4ff92SAndroid Build Coastguard Worker                 case Color::White:
3978*89c4ff92SAndroid Build Coastguard Worker                 default:                 col = "[0m";
3979*89c4ff92SAndroid Build Coastguard Worker             }
3980*89c4ff92SAndroid Build Coastguard Worker         // clang-format on
3981*89c4ff92SAndroid Build Coastguard Worker         s << "\033" << col;
3982*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_COLORS_ANSI
3983*89c4ff92SAndroid Build Coastguard Worker 
3984*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_COLORS_WINDOWS
3985*89c4ff92SAndroid Build Coastguard Worker         if(g_no_colors ||
3986*89c4ff92SAndroid Build Coastguard Worker            (isatty(fileno(stdout)) == false && getContextOptions()->force_colors == false))
3987*89c4ff92SAndroid Build Coastguard Worker             return;
3988*89c4ff92SAndroid Build Coastguard Worker 
3989*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_SET_ATTR(x) SetConsoleTextAttribute(g_stdoutHandle, x | g_origBgAttrs)
3990*89c4ff92SAndroid Build Coastguard Worker 
3991*89c4ff92SAndroid Build Coastguard Worker         // clang-format off
3992*89c4ff92SAndroid Build Coastguard Worker         switch (code) {
3993*89c4ff92SAndroid Build Coastguard Worker             case Color::White:       DOCTEST_SET_ATTR(FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_BLUE); break;
3994*89c4ff92SAndroid Build Coastguard Worker             case Color::Red:         DOCTEST_SET_ATTR(FOREGROUND_RED);                                      break;
3995*89c4ff92SAndroid Build Coastguard Worker             case Color::Green:       DOCTEST_SET_ATTR(FOREGROUND_GREEN);                                    break;
3996*89c4ff92SAndroid Build Coastguard Worker             case Color::Blue:        DOCTEST_SET_ATTR(FOREGROUND_BLUE);                                     break;
3997*89c4ff92SAndroid Build Coastguard Worker             case Color::Cyan:        DOCTEST_SET_ATTR(FOREGROUND_BLUE | FOREGROUND_GREEN);                  break;
3998*89c4ff92SAndroid Build Coastguard Worker             case Color::Yellow:      DOCTEST_SET_ATTR(FOREGROUND_RED | FOREGROUND_GREEN);                   break;
3999*89c4ff92SAndroid Build Coastguard Worker             case Color::Grey:        DOCTEST_SET_ATTR(0);                                                   break;
4000*89c4ff92SAndroid Build Coastguard Worker             case Color::LightGrey:   DOCTEST_SET_ATTR(FOREGROUND_INTENSITY);                                break;
4001*89c4ff92SAndroid Build Coastguard Worker             case Color::BrightRed:   DOCTEST_SET_ATTR(FOREGROUND_INTENSITY | FOREGROUND_RED);               break;
4002*89c4ff92SAndroid Build Coastguard Worker             case Color::BrightGreen: DOCTEST_SET_ATTR(FOREGROUND_INTENSITY | FOREGROUND_GREEN);             break;
4003*89c4ff92SAndroid Build Coastguard Worker             case Color::BrightWhite: DOCTEST_SET_ATTR(FOREGROUND_INTENSITY | FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_BLUE); break;
4004*89c4ff92SAndroid Build Coastguard Worker             case Color::None:
4005*89c4ff92SAndroid Build Coastguard Worker             case Color::Bright: // invalid
4006*89c4ff92SAndroid Build Coastguard Worker             default:                 DOCTEST_SET_ATTR(g_origFgAttrs);
4007*89c4ff92SAndroid Build Coastguard Worker         }
4008*89c4ff92SAndroid Build Coastguard Worker             // clang-format on
4009*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_COLORS_WINDOWS
4010*89c4ff92SAndroid Build Coastguard Worker     }
4011*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_POP
4012*89c4ff92SAndroid Build Coastguard Worker 
4013*89c4ff92SAndroid Build Coastguard Worker     std::vector<const IExceptionTranslator*>& getExceptionTranslators() {
4014*89c4ff92SAndroid Build Coastguard Worker         static std::vector<const IExceptionTranslator*> data;
4015*89c4ff92SAndroid Build Coastguard Worker         return data;
4016*89c4ff92SAndroid Build Coastguard Worker     }
4017*89c4ff92SAndroid Build Coastguard Worker 
4018*89c4ff92SAndroid Build Coastguard Worker     String translateActiveException() {
4019*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_EXCEPTIONS
4020*89c4ff92SAndroid Build Coastguard Worker         String res;
4021*89c4ff92SAndroid Build Coastguard Worker         auto&  translators = getExceptionTranslators();
4022*89c4ff92SAndroid Build Coastguard Worker         for(auto& curr : translators)
4023*89c4ff92SAndroid Build Coastguard Worker             if(curr->translate(res))
4024*89c4ff92SAndroid Build Coastguard Worker                 return res;
4025*89c4ff92SAndroid Build Coastguard Worker         // clang-format off
4026*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wcatch-value")
4027*89c4ff92SAndroid Build Coastguard Worker         try {
4028*89c4ff92SAndroid Build Coastguard Worker             throw;
4029*89c4ff92SAndroid Build Coastguard Worker         } catch(std::exception& ex) {
4030*89c4ff92SAndroid Build Coastguard Worker             return ex.what();
4031*89c4ff92SAndroid Build Coastguard Worker         } catch(std::string& msg) {
4032*89c4ff92SAndroid Build Coastguard Worker             return msg.c_str();
4033*89c4ff92SAndroid Build Coastguard Worker         } catch(const char* msg) {
4034*89c4ff92SAndroid Build Coastguard Worker             return msg;
4035*89c4ff92SAndroid Build Coastguard Worker         } catch(...) {
4036*89c4ff92SAndroid Build Coastguard Worker             return "unknown exception";
4037*89c4ff92SAndroid Build Coastguard Worker         }
4038*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_GCC_SUPPRESS_WARNING_POP
4039*89c4ff92SAndroid Build Coastguard Worker // clang-format on
4040*89c4ff92SAndroid Build Coastguard Worker #else  // DOCTEST_CONFIG_NO_EXCEPTIONS
4041*89c4ff92SAndroid Build Coastguard Worker         return "";
4042*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS
4043*89c4ff92SAndroid Build Coastguard Worker     }
4044*89c4ff92SAndroid Build Coastguard Worker } // namespace
4045*89c4ff92SAndroid Build Coastguard Worker 
4046*89c4ff92SAndroid Build Coastguard Worker namespace detail {
4047*89c4ff92SAndroid Build Coastguard Worker     // used by the macros for registering tests
4048*89c4ff92SAndroid Build Coastguard Worker     int regTest(const TestCase& tc) {
4049*89c4ff92SAndroid Build Coastguard Worker         getRegisteredTests().insert(tc);
4050*89c4ff92SAndroid Build Coastguard Worker         return 0;
4051*89c4ff92SAndroid Build Coastguard Worker     }
4052*89c4ff92SAndroid Build Coastguard Worker 
4053*89c4ff92SAndroid Build Coastguard Worker     // sets the current test suite
4054*89c4ff92SAndroid Build Coastguard Worker     int setTestSuite(const TestSuite& ts) {
4055*89c4ff92SAndroid Build Coastguard Worker         doctest_detail_test_suite_ns::getCurrentTestSuite() = ts;
4056*89c4ff92SAndroid Build Coastguard Worker         return 0;
4057*89c4ff92SAndroid Build Coastguard Worker     }
4058*89c4ff92SAndroid Build Coastguard Worker 
4059*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_IS_DEBUGGER_ACTIVE
4060*89c4ff92SAndroid Build Coastguard Worker     bool isDebuggerActive() { return DOCTEST_IS_DEBUGGER_ACTIVE(); }
4061*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_IS_DEBUGGER_ACTIVE
4062*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_LINUX
4063*89c4ff92SAndroid Build Coastguard Worker     class ErrnoGuard {
4064*89c4ff92SAndroid Build Coastguard Worker     public:
4065*89c4ff92SAndroid Build Coastguard Worker         ErrnoGuard() : m_oldErrno(errno) {}
4066*89c4ff92SAndroid Build Coastguard Worker         ~ErrnoGuard() { errno = m_oldErrno; }
4067*89c4ff92SAndroid Build Coastguard Worker     private:
4068*89c4ff92SAndroid Build Coastguard Worker         int m_oldErrno;
4069*89c4ff92SAndroid Build Coastguard Worker     };
4070*89c4ff92SAndroid Build Coastguard Worker     // See the comments in Catch2 for the reasoning behind this implementation:
4071*89c4ff92SAndroid Build Coastguard Worker     // https://github.com/catchorg/Catch2/blob/v2.13.1/include/internal/catch_debugger.cpp#L79-L102
4072*89c4ff92SAndroid Build Coastguard Worker     bool isDebuggerActive() {
4073*89c4ff92SAndroid Build Coastguard Worker         ErrnoGuard guard;
4074*89c4ff92SAndroid Build Coastguard Worker         std::ifstream in("/proc/self/status");
4075*89c4ff92SAndroid Build Coastguard Worker         for(std::string line; std::getline(in, line);) {
4076*89c4ff92SAndroid Build Coastguard Worker             static const int PREFIX_LEN = 11;
4077*89c4ff92SAndroid Build Coastguard Worker             if(line.compare(0, PREFIX_LEN, "TracerPid:\t") == 0) {
4078*89c4ff92SAndroid Build Coastguard Worker                 return line.length() > PREFIX_LEN && line[PREFIX_LEN] != '0';
4079*89c4ff92SAndroid Build Coastguard Worker             }
4080*89c4ff92SAndroid Build Coastguard Worker         }
4081*89c4ff92SAndroid Build Coastguard Worker         return false;
4082*89c4ff92SAndroid Build Coastguard Worker     }
4083*89c4ff92SAndroid Build Coastguard Worker #elif defined(DOCTEST_PLATFORM_MAC)
4084*89c4ff92SAndroid Build Coastguard Worker     // The following function is taken directly from the following technical note:
4085*89c4ff92SAndroid Build Coastguard Worker     // https://developer.apple.com/library/archive/qa/qa1361/_index.html
4086*89c4ff92SAndroid Build Coastguard Worker     // Returns true if the current process is being debugged (either
4087*89c4ff92SAndroid Build Coastguard Worker     // running under the debugger or has a debugger attached post facto).
4088*89c4ff92SAndroid Build Coastguard Worker     bool isDebuggerActive() {
4089*89c4ff92SAndroid Build Coastguard Worker         int        mib[4];
4090*89c4ff92SAndroid Build Coastguard Worker         kinfo_proc info;
4091*89c4ff92SAndroid Build Coastguard Worker         size_t     size;
4092*89c4ff92SAndroid Build Coastguard Worker         // Initialize the flags so that, if sysctl fails for some bizarre
4093*89c4ff92SAndroid Build Coastguard Worker         // reason, we get a predictable result.
4094*89c4ff92SAndroid Build Coastguard Worker         info.kp_proc.p_flag = 0;
4095*89c4ff92SAndroid Build Coastguard Worker         // Initialize mib, which tells sysctl the info we want, in this case
4096*89c4ff92SAndroid Build Coastguard Worker         // we're looking for information about a specific process ID.
4097*89c4ff92SAndroid Build Coastguard Worker         mib[0] = CTL_KERN;
4098*89c4ff92SAndroid Build Coastguard Worker         mib[1] = KERN_PROC;
4099*89c4ff92SAndroid Build Coastguard Worker         mib[2] = KERN_PROC_PID;
4100*89c4ff92SAndroid Build Coastguard Worker         mib[3] = getpid();
4101*89c4ff92SAndroid Build Coastguard Worker         // Call sysctl.
4102*89c4ff92SAndroid Build Coastguard Worker         size = sizeof(info);
4103*89c4ff92SAndroid Build Coastguard Worker         if(sysctl(mib, DOCTEST_COUNTOF(mib), &info, &size, 0, 0) != 0) {
4104*89c4ff92SAndroid Build Coastguard Worker             std::cerr << "\nCall to sysctl failed - unable to determine if debugger is active **\n";
4105*89c4ff92SAndroid Build Coastguard Worker             return false;
4106*89c4ff92SAndroid Build Coastguard Worker         }
4107*89c4ff92SAndroid Build Coastguard Worker         // We're being debugged if the P_TRACED flag is set.
4108*89c4ff92SAndroid Build Coastguard Worker         return ((info.kp_proc.p_flag & P_TRACED) != 0);
4109*89c4ff92SAndroid Build Coastguard Worker     }
4110*89c4ff92SAndroid Build Coastguard Worker #elif DOCTEST_MSVC || defined(__MINGW32__) || defined(__MINGW64__)
4111*89c4ff92SAndroid Build Coastguard Worker     bool isDebuggerActive() { return ::IsDebuggerPresent() != 0; }
4112*89c4ff92SAndroid Build Coastguard Worker #else
4113*89c4ff92SAndroid Build Coastguard Worker     bool isDebuggerActive() { return false; }
4114*89c4ff92SAndroid Build Coastguard Worker #endif // Platform
4115*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_IS_DEBUGGER_ACTIVE
4116*89c4ff92SAndroid Build Coastguard Worker 
4117*89c4ff92SAndroid Build Coastguard Worker     void registerExceptionTranslatorImpl(const IExceptionTranslator* et) {
4118*89c4ff92SAndroid Build Coastguard Worker         if(std::find(getExceptionTranslators().begin(), getExceptionTranslators().end(), et) ==
4119*89c4ff92SAndroid Build Coastguard Worker            getExceptionTranslators().end())
4120*89c4ff92SAndroid Build Coastguard Worker             getExceptionTranslators().push_back(et);
4121*89c4ff92SAndroid Build Coastguard Worker     }
4122*89c4ff92SAndroid Build Coastguard Worker 
4123*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
4124*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, char* in) { *s << in; }
4125*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, const char* in) { *s << in; }
4126*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_TREAT_CHAR_STAR_AS_STRING
4127*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, bool in) { *s << std::boolalpha << in << std::noboolalpha; }
4128*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, float in) { *s << in; }
4129*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, double in) { *s << in; }
4130*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, double long in) { *s << in; }
4131*89c4ff92SAndroid Build Coastguard Worker 
4132*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, char in) { *s << in; }
4133*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, char signed in) { *s << in; }
4134*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, char unsigned in) { *s << in; }
4135*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, int short in) { *s << in; }
4136*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, int short unsigned in) { *s << in; }
4137*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, int in) { *s << in; }
4138*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, int unsigned in) { *s << in; }
4139*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, int long in) { *s << in; }
4140*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, int long unsigned in) { *s << in; }
4141*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, int long long in) { *s << in; }
4142*89c4ff92SAndroid Build Coastguard Worker     void toStream(std::ostream* s, int long long unsigned in) { *s << in; }
4143*89c4ff92SAndroid Build Coastguard Worker 
4144*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_THREAD_LOCAL std::vector<IContextScope*> g_infoContexts; // for logging with INFO()
4145*89c4ff92SAndroid Build Coastguard Worker 
4146*89c4ff92SAndroid Build Coastguard Worker     ContextScopeBase::ContextScopeBase() {
4147*89c4ff92SAndroid Build Coastguard Worker         g_infoContexts.push_back(this);
4148*89c4ff92SAndroid Build Coastguard Worker     }
4149*89c4ff92SAndroid Build Coastguard Worker 
4150*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4996) // std::uncaught_exception is deprecated in C++17
4151*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING_WITH_PUSH("-Wdeprecated-declarations")
4152*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wdeprecated-declarations")
4153*89c4ff92SAndroid Build Coastguard Worker 
4154*89c4ff92SAndroid Build Coastguard Worker     // destroy cannot be inlined into the destructor because that would mean calling stringify after
4155*89c4ff92SAndroid Build Coastguard Worker     // ContextScope has been destroyed (base class destructors run after derived class destructors).
4156*89c4ff92SAndroid Build Coastguard Worker     // Instead, ContextScope calls this method directly from its destructor.
4157*89c4ff92SAndroid Build Coastguard Worker     void ContextScopeBase::destroy() {
4158*89c4ff92SAndroid Build Coastguard Worker #if defined(__cpp_lib_uncaught_exceptions) && __cpp_lib_uncaught_exceptions >= 201411L && (!defined(__MAC_OS_X_VERSION_MIN_REQUIRED) || __MAC_OS_X_VERSION_MIN_REQUIRED >= 101200)
4159*89c4ff92SAndroid Build Coastguard Worker         if(std::uncaught_exceptions() > 0) {
4160*89c4ff92SAndroid Build Coastguard Worker #else
4161*89c4ff92SAndroid Build Coastguard Worker         if(std::uncaught_exception()) {
4162*89c4ff92SAndroid Build Coastguard Worker #endif
4163*89c4ff92SAndroid Build Coastguard Worker             std::ostringstream s;
4164*89c4ff92SAndroid Build Coastguard Worker             this->stringify(&s);
4165*89c4ff92SAndroid Build Coastguard Worker             g_cs->stringifiedContexts.push_back(s.str().c_str());
4166*89c4ff92SAndroid Build Coastguard Worker         }
4167*89c4ff92SAndroid Build Coastguard Worker         g_infoContexts.pop_back();
4168*89c4ff92SAndroid Build Coastguard Worker     }
4169*89c4ff92SAndroid Build Coastguard Worker 
4170*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_CLANG_SUPPRESS_WARNING_POP
4171*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_GCC_SUPPRESS_WARNING_POP
4172*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_MSVC_SUPPRESS_WARNING_POP
4173*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
4174*89c4ff92SAndroid Build Coastguard Worker namespace {
4175*89c4ff92SAndroid Build Coastguard Worker     using namespace detail;
4176*89c4ff92SAndroid Build Coastguard Worker 
4177*89c4ff92SAndroid Build Coastguard Worker #if !defined(DOCTEST_CONFIG_POSIX_SIGNALS) && !defined(DOCTEST_CONFIG_WINDOWS_SEH)
4178*89c4ff92SAndroid Build Coastguard Worker     struct FatalConditionHandler
4179*89c4ff92SAndroid Build Coastguard Worker     {
4180*89c4ff92SAndroid Build Coastguard Worker         static void reset() {}
4181*89c4ff92SAndroid Build Coastguard Worker         static void allocateAltStackMem() {}
4182*89c4ff92SAndroid Build Coastguard Worker         static void freeAltStackMem() {}
4183*89c4ff92SAndroid Build Coastguard Worker     };
4184*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_CONFIG_POSIX_SIGNALS || DOCTEST_CONFIG_WINDOWS_SEH
4185*89c4ff92SAndroid Build Coastguard Worker 
4186*89c4ff92SAndroid Build Coastguard Worker     void reportFatal(const std::string&);
4187*89c4ff92SAndroid Build Coastguard Worker 
4188*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_WINDOWS
4189*89c4ff92SAndroid Build Coastguard Worker 
4190*89c4ff92SAndroid Build Coastguard Worker     struct SignalDefs
4191*89c4ff92SAndroid Build Coastguard Worker     {
4192*89c4ff92SAndroid Build Coastguard Worker         DWORD id;
4193*89c4ff92SAndroid Build Coastguard Worker         const char* name;
4194*89c4ff92SAndroid Build Coastguard Worker     };
4195*89c4ff92SAndroid Build Coastguard Worker     // There is no 1-1 mapping between signals and windows exceptions.
4196*89c4ff92SAndroid Build Coastguard Worker     // Windows can easily distinguish between SO and SigSegV,
4197*89c4ff92SAndroid Build Coastguard Worker     // but SigInt, SigTerm, etc are handled differently.
4198*89c4ff92SAndroid Build Coastguard Worker     SignalDefs signalDefs[] = {
4199*89c4ff92SAndroid Build Coastguard Worker             {static_cast<DWORD>(EXCEPTION_ILLEGAL_INSTRUCTION),
4200*89c4ff92SAndroid Build Coastguard Worker              "SIGILL - Illegal instruction signal"},
4201*89c4ff92SAndroid Build Coastguard Worker             {static_cast<DWORD>(EXCEPTION_STACK_OVERFLOW), "SIGSEGV - Stack overflow"},
4202*89c4ff92SAndroid Build Coastguard Worker             {static_cast<DWORD>(EXCEPTION_ACCESS_VIOLATION),
4203*89c4ff92SAndroid Build Coastguard Worker              "SIGSEGV - Segmentation violation signal"},
4204*89c4ff92SAndroid Build Coastguard Worker             {static_cast<DWORD>(EXCEPTION_INT_DIVIDE_BY_ZERO), "Divide by zero error"},
4205*89c4ff92SAndroid Build Coastguard Worker     };
4206*89c4ff92SAndroid Build Coastguard Worker 
4207*89c4ff92SAndroid Build Coastguard Worker     struct FatalConditionHandler
4208*89c4ff92SAndroid Build Coastguard Worker     {
4209*89c4ff92SAndroid Build Coastguard Worker         static LONG CALLBACK handleException(PEXCEPTION_POINTERS ExceptionInfo) {
4210*89c4ff92SAndroid Build Coastguard Worker             // Multiple threads may enter this filter/handler at once. We want the error message to be printed on the
4211*89c4ff92SAndroid Build Coastguard Worker             // console just once no matter how many threads have crashed.
4212*89c4ff92SAndroid Build Coastguard Worker             static std::mutex mutex;
4213*89c4ff92SAndroid Build Coastguard Worker             static bool execute = true;
4214*89c4ff92SAndroid Build Coastguard Worker             {
4215*89c4ff92SAndroid Build Coastguard Worker                 std::lock_guard<std::mutex> lock(mutex);
4216*89c4ff92SAndroid Build Coastguard Worker                 if(execute) {
4217*89c4ff92SAndroid Build Coastguard Worker                     bool reported = false;
4218*89c4ff92SAndroid Build Coastguard Worker                     for(size_t i = 0; i < DOCTEST_COUNTOF(signalDefs); ++i) {
4219*89c4ff92SAndroid Build Coastguard Worker                         if(ExceptionInfo->ExceptionRecord->ExceptionCode == signalDefs[i].id) {
4220*89c4ff92SAndroid Build Coastguard Worker                             reportFatal(signalDefs[i].name);
4221*89c4ff92SAndroid Build Coastguard Worker                             reported = true;
4222*89c4ff92SAndroid Build Coastguard Worker                             break;
4223*89c4ff92SAndroid Build Coastguard Worker                         }
4224*89c4ff92SAndroid Build Coastguard Worker                     }
4225*89c4ff92SAndroid Build Coastguard Worker                     if(reported == false)
4226*89c4ff92SAndroid Build Coastguard Worker                         reportFatal("Unhandled SEH exception caught");
4227*89c4ff92SAndroid Build Coastguard Worker                     if(isDebuggerActive() && !g_cs->no_breaks)
4228*89c4ff92SAndroid Build Coastguard Worker                         DOCTEST_BREAK_INTO_DEBUGGER();
4229*89c4ff92SAndroid Build Coastguard Worker                 }
4230*89c4ff92SAndroid Build Coastguard Worker                 execute = false;
4231*89c4ff92SAndroid Build Coastguard Worker             }
4232*89c4ff92SAndroid Build Coastguard Worker             std::exit(EXIT_FAILURE);
4233*89c4ff92SAndroid Build Coastguard Worker         }
4234*89c4ff92SAndroid Build Coastguard Worker 
4235*89c4ff92SAndroid Build Coastguard Worker         static void allocateAltStackMem() {}
4236*89c4ff92SAndroid Build Coastguard Worker         static void freeAltStackMem() {}
4237*89c4ff92SAndroid Build Coastguard Worker 
4238*89c4ff92SAndroid Build Coastguard Worker         FatalConditionHandler() {
4239*89c4ff92SAndroid Build Coastguard Worker             isSet = true;
4240*89c4ff92SAndroid Build Coastguard Worker             // 32k seems enough for doctest to handle stack overflow,
4241*89c4ff92SAndroid Build Coastguard Worker             // but the value was found experimentally, so there is no strong guarantee
4242*89c4ff92SAndroid Build Coastguard Worker             guaranteeSize = 32 * 1024;
4243*89c4ff92SAndroid Build Coastguard Worker             // Register an unhandled exception filter
4244*89c4ff92SAndroid Build Coastguard Worker             previousTop = SetUnhandledExceptionFilter(handleException);
4245*89c4ff92SAndroid Build Coastguard Worker             // Pass in guarantee size to be filled
4246*89c4ff92SAndroid Build Coastguard Worker             SetThreadStackGuarantee(&guaranteeSize);
4247*89c4ff92SAndroid Build Coastguard Worker 
4248*89c4ff92SAndroid Build Coastguard Worker             // On Windows uncaught exceptions from another thread, exceptions from
4249*89c4ff92SAndroid Build Coastguard Worker             // destructors, or calls to std::terminate are not a SEH exception
4250*89c4ff92SAndroid Build Coastguard Worker 
4251*89c4ff92SAndroid Build Coastguard Worker             // The terminal handler gets called when:
4252*89c4ff92SAndroid Build Coastguard Worker             // - std::terminate is called FROM THE TEST RUNNER THREAD
4253*89c4ff92SAndroid Build Coastguard Worker             // - an exception is thrown from a destructor FROM THE TEST RUNNER THREAD
4254*89c4ff92SAndroid Build Coastguard Worker             original_terminate_handler = std::get_terminate();
4255*89c4ff92SAndroid Build Coastguard Worker             std::set_terminate([]() DOCTEST_NOEXCEPT {
4256*89c4ff92SAndroid Build Coastguard Worker                 reportFatal("Terminate handler called");
4257*89c4ff92SAndroid Build Coastguard Worker                 if(isDebuggerActive() && !g_cs->no_breaks)
4258*89c4ff92SAndroid Build Coastguard Worker                     DOCTEST_BREAK_INTO_DEBUGGER();
4259*89c4ff92SAndroid Build Coastguard Worker                 std::exit(EXIT_FAILURE); // explicitly exit - otherwise the SIGABRT handler may be called as well
4260*89c4ff92SAndroid Build Coastguard Worker             });
4261*89c4ff92SAndroid Build Coastguard Worker 
4262*89c4ff92SAndroid Build Coastguard Worker             // SIGABRT is raised when:
4263*89c4ff92SAndroid Build Coastguard Worker             // - std::terminate is called FROM A DIFFERENT THREAD
4264*89c4ff92SAndroid Build Coastguard Worker             // - an exception is thrown from a destructor FROM A DIFFERENT THREAD
4265*89c4ff92SAndroid Build Coastguard Worker             // - an uncaught exception is thrown FROM A DIFFERENT THREAD
4266*89c4ff92SAndroid Build Coastguard Worker             prev_sigabrt_handler = std::signal(SIGABRT, [](int signal) DOCTEST_NOEXCEPT {
4267*89c4ff92SAndroid Build Coastguard Worker                 if(signal == SIGABRT) {
4268*89c4ff92SAndroid Build Coastguard Worker                     reportFatal("SIGABRT - Abort (abnormal termination) signal");
4269*89c4ff92SAndroid Build Coastguard Worker                     if(isDebuggerActive() && !g_cs->no_breaks)
4270*89c4ff92SAndroid Build Coastguard Worker                         DOCTEST_BREAK_INTO_DEBUGGER();
4271*89c4ff92SAndroid Build Coastguard Worker                     std::exit(EXIT_FAILURE);
4272*89c4ff92SAndroid Build Coastguard Worker                 }
4273*89c4ff92SAndroid Build Coastguard Worker             });
4274*89c4ff92SAndroid Build Coastguard Worker 
4275*89c4ff92SAndroid Build Coastguard Worker             // The following settings are taken from google test, and more
4276*89c4ff92SAndroid Build Coastguard Worker             // specifically from UnitTest::Run() inside of gtest.cc
4277*89c4ff92SAndroid Build Coastguard Worker 
4278*89c4ff92SAndroid Build Coastguard Worker             // the user does not want to see pop-up dialogs about crashes
4279*89c4ff92SAndroid Build Coastguard Worker             prev_error_mode_1 = SetErrorMode(SEM_FAILCRITICALERRORS | SEM_NOALIGNMENTFAULTEXCEPT |
4280*89c4ff92SAndroid Build Coastguard Worker                                              SEM_NOGPFAULTERRORBOX | SEM_NOOPENFILEERRORBOX);
4281*89c4ff92SAndroid Build Coastguard Worker             // This forces the abort message to go to stderr in all circumstances.
4282*89c4ff92SAndroid Build Coastguard Worker             prev_error_mode_2 = _set_error_mode(_OUT_TO_STDERR);
4283*89c4ff92SAndroid Build Coastguard Worker             // In the debug version, Visual Studio pops up a separate dialog
4284*89c4ff92SAndroid Build Coastguard Worker             // offering a choice to debug the aborted program - we want to disable that.
4285*89c4ff92SAndroid Build Coastguard Worker             prev_abort_behavior = _set_abort_behavior(0x0, _WRITE_ABORT_MSG | _CALL_REPORTFAULT);
4286*89c4ff92SAndroid Build Coastguard Worker             // In debug mode, the Windows CRT can crash with an assertion over invalid
4287*89c4ff92SAndroid Build Coastguard Worker             // input (e.g. passing an invalid file descriptor). The default handling
4288*89c4ff92SAndroid Build Coastguard Worker             // for these assertions is to pop up a dialog and wait for user input.
4289*89c4ff92SAndroid Build Coastguard Worker             // Instead ask the CRT to dump such assertions to stderr non-interactively.
4290*89c4ff92SAndroid Build Coastguard Worker             prev_report_mode = _CrtSetReportMode(_CRT_ASSERT, _CRTDBG_MODE_FILE | _CRTDBG_MODE_DEBUG);
4291*89c4ff92SAndroid Build Coastguard Worker             prev_report_file = _CrtSetReportFile(_CRT_ASSERT, _CRTDBG_FILE_STDERR);
4292*89c4ff92SAndroid Build Coastguard Worker         }
4293*89c4ff92SAndroid Build Coastguard Worker 
4294*89c4ff92SAndroid Build Coastguard Worker         static void reset() {
4295*89c4ff92SAndroid Build Coastguard Worker             if(isSet) {
4296*89c4ff92SAndroid Build Coastguard Worker                 // Unregister handler and restore the old guarantee
4297*89c4ff92SAndroid Build Coastguard Worker                 SetUnhandledExceptionFilter(previousTop);
4298*89c4ff92SAndroid Build Coastguard Worker                 SetThreadStackGuarantee(&guaranteeSize);
4299*89c4ff92SAndroid Build Coastguard Worker                 std::set_terminate(original_terminate_handler);
4300*89c4ff92SAndroid Build Coastguard Worker                 std::signal(SIGABRT, prev_sigabrt_handler);
4301*89c4ff92SAndroid Build Coastguard Worker                 SetErrorMode(prev_error_mode_1);
4302*89c4ff92SAndroid Build Coastguard Worker                 _set_error_mode(prev_error_mode_2);
4303*89c4ff92SAndroid Build Coastguard Worker                 _set_abort_behavior(prev_abort_behavior, _WRITE_ABORT_MSG | _CALL_REPORTFAULT);
4304*89c4ff92SAndroid Build Coastguard Worker                 static_cast<void>(_CrtSetReportMode(_CRT_ASSERT, prev_report_mode));
4305*89c4ff92SAndroid Build Coastguard Worker                 static_cast<void>(_CrtSetReportFile(_CRT_ASSERT, prev_report_file));
4306*89c4ff92SAndroid Build Coastguard Worker                 isSet = false;
4307*89c4ff92SAndroid Build Coastguard Worker             }
4308*89c4ff92SAndroid Build Coastguard Worker         }
4309*89c4ff92SAndroid Build Coastguard Worker 
4310*89c4ff92SAndroid Build Coastguard Worker         ~FatalConditionHandler() { reset(); }
4311*89c4ff92SAndroid Build Coastguard Worker 
4312*89c4ff92SAndroid Build Coastguard Worker     private:
4313*89c4ff92SAndroid Build Coastguard Worker         static UINT         prev_error_mode_1;
4314*89c4ff92SAndroid Build Coastguard Worker         static int          prev_error_mode_2;
4315*89c4ff92SAndroid Build Coastguard Worker         static unsigned int prev_abort_behavior;
4316*89c4ff92SAndroid Build Coastguard Worker         static int          prev_report_mode;
4317*89c4ff92SAndroid Build Coastguard Worker         static _HFILE       prev_report_file;
4318*89c4ff92SAndroid Build Coastguard Worker         static void (*prev_sigabrt_handler)(int);
4319*89c4ff92SAndroid Build Coastguard Worker         static std::terminate_handler original_terminate_handler;
4320*89c4ff92SAndroid Build Coastguard Worker         static bool isSet;
4321*89c4ff92SAndroid Build Coastguard Worker         static ULONG guaranteeSize;
4322*89c4ff92SAndroid Build Coastguard Worker         static LPTOP_LEVEL_EXCEPTION_FILTER previousTop;
4323*89c4ff92SAndroid Build Coastguard Worker     };
4324*89c4ff92SAndroid Build Coastguard Worker 
4325*89c4ff92SAndroid Build Coastguard Worker     UINT         FatalConditionHandler::prev_error_mode_1;
4326*89c4ff92SAndroid Build Coastguard Worker     int          FatalConditionHandler::prev_error_mode_2;
4327*89c4ff92SAndroid Build Coastguard Worker     unsigned int FatalConditionHandler::prev_abort_behavior;
4328*89c4ff92SAndroid Build Coastguard Worker     int          FatalConditionHandler::prev_report_mode;
4329*89c4ff92SAndroid Build Coastguard Worker     _HFILE       FatalConditionHandler::prev_report_file;
4330*89c4ff92SAndroid Build Coastguard Worker     void (*FatalConditionHandler::prev_sigabrt_handler)(int);
4331*89c4ff92SAndroid Build Coastguard Worker     std::terminate_handler FatalConditionHandler::original_terminate_handler;
4332*89c4ff92SAndroid Build Coastguard Worker     bool FatalConditionHandler::isSet = false;
4333*89c4ff92SAndroid Build Coastguard Worker     ULONG FatalConditionHandler::guaranteeSize = 0;
4334*89c4ff92SAndroid Build Coastguard Worker     LPTOP_LEVEL_EXCEPTION_FILTER FatalConditionHandler::previousTop = nullptr;
4335*89c4ff92SAndroid Build Coastguard Worker 
4336*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_PLATFORM_WINDOWS
4337*89c4ff92SAndroid Build Coastguard Worker 
4338*89c4ff92SAndroid Build Coastguard Worker     struct SignalDefs
4339*89c4ff92SAndroid Build Coastguard Worker     {
4340*89c4ff92SAndroid Build Coastguard Worker         int         id;
4341*89c4ff92SAndroid Build Coastguard Worker         const char* name;
4342*89c4ff92SAndroid Build Coastguard Worker     };
4343*89c4ff92SAndroid Build Coastguard Worker     SignalDefs signalDefs[] = {{SIGINT, "SIGINT - Terminal interrupt signal"},
4344*89c4ff92SAndroid Build Coastguard Worker                                {SIGILL, "SIGILL - Illegal instruction signal"},
4345*89c4ff92SAndroid Build Coastguard Worker                                {SIGFPE, "SIGFPE - Floating point error signal"},
4346*89c4ff92SAndroid Build Coastguard Worker                                {SIGSEGV, "SIGSEGV - Segmentation violation signal"},
4347*89c4ff92SAndroid Build Coastguard Worker                                {SIGTERM, "SIGTERM - Termination request signal"},
4348*89c4ff92SAndroid Build Coastguard Worker                                {SIGABRT, "SIGABRT - Abort (abnormal termination) signal"}};
4349*89c4ff92SAndroid Build Coastguard Worker 
4350*89c4ff92SAndroid Build Coastguard Worker     struct FatalConditionHandler
4351*89c4ff92SAndroid Build Coastguard Worker     {
4352*89c4ff92SAndroid Build Coastguard Worker         static bool             isSet;
4353*89c4ff92SAndroid Build Coastguard Worker         static struct sigaction oldSigActions[DOCTEST_COUNTOF(signalDefs)];
4354*89c4ff92SAndroid Build Coastguard Worker         static stack_t          oldSigStack;
4355*89c4ff92SAndroid Build Coastguard Worker         static size_t           altStackSize;
4356*89c4ff92SAndroid Build Coastguard Worker         static char*            altStackMem;
4357*89c4ff92SAndroid Build Coastguard Worker 
4358*89c4ff92SAndroid Build Coastguard Worker         static void handleSignal(int sig) {
4359*89c4ff92SAndroid Build Coastguard Worker             const char* name = "<unknown signal>";
4360*89c4ff92SAndroid Build Coastguard Worker             for(std::size_t i = 0; i < DOCTEST_COUNTOF(signalDefs); ++i) {
4361*89c4ff92SAndroid Build Coastguard Worker                 SignalDefs& def = signalDefs[i];
4362*89c4ff92SAndroid Build Coastguard Worker                 if(sig == def.id) {
4363*89c4ff92SAndroid Build Coastguard Worker                     name = def.name;
4364*89c4ff92SAndroid Build Coastguard Worker                     break;
4365*89c4ff92SAndroid Build Coastguard Worker                 }
4366*89c4ff92SAndroid Build Coastguard Worker             }
4367*89c4ff92SAndroid Build Coastguard Worker             reset();
4368*89c4ff92SAndroid Build Coastguard Worker             reportFatal(name);
4369*89c4ff92SAndroid Build Coastguard Worker             raise(sig);
4370*89c4ff92SAndroid Build Coastguard Worker         }
4371*89c4ff92SAndroid Build Coastguard Worker 
4372*89c4ff92SAndroid Build Coastguard Worker         static void allocateAltStackMem() {
4373*89c4ff92SAndroid Build Coastguard Worker             altStackMem = new char[altStackSize];
4374*89c4ff92SAndroid Build Coastguard Worker         }
4375*89c4ff92SAndroid Build Coastguard Worker 
4376*89c4ff92SAndroid Build Coastguard Worker         static void freeAltStackMem() {
4377*89c4ff92SAndroid Build Coastguard Worker             delete[] altStackMem;
4378*89c4ff92SAndroid Build Coastguard Worker         }
4379*89c4ff92SAndroid Build Coastguard Worker 
4380*89c4ff92SAndroid Build Coastguard Worker         FatalConditionHandler() {
4381*89c4ff92SAndroid Build Coastguard Worker             isSet = true;
4382*89c4ff92SAndroid Build Coastguard Worker             stack_t sigStack;
4383*89c4ff92SAndroid Build Coastguard Worker             sigStack.ss_sp    = altStackMem;
4384*89c4ff92SAndroid Build Coastguard Worker             sigStack.ss_size  = altStackSize;
4385*89c4ff92SAndroid Build Coastguard Worker             sigStack.ss_flags = 0;
4386*89c4ff92SAndroid Build Coastguard Worker             sigaltstack(&sigStack, &oldSigStack);
4387*89c4ff92SAndroid Build Coastguard Worker             struct sigaction sa = {};
4388*89c4ff92SAndroid Build Coastguard Worker             sa.sa_handler       = handleSignal; // NOLINT
4389*89c4ff92SAndroid Build Coastguard Worker             sa.sa_flags         = SA_ONSTACK;
4390*89c4ff92SAndroid Build Coastguard Worker             for(std::size_t i = 0; i < DOCTEST_COUNTOF(signalDefs); ++i) {
4391*89c4ff92SAndroid Build Coastguard Worker                 sigaction(signalDefs[i].id, &sa, &oldSigActions[i]);
4392*89c4ff92SAndroid Build Coastguard Worker             }
4393*89c4ff92SAndroid Build Coastguard Worker         }
4394*89c4ff92SAndroid Build Coastguard Worker 
4395*89c4ff92SAndroid Build Coastguard Worker         ~FatalConditionHandler() { reset(); }
4396*89c4ff92SAndroid Build Coastguard Worker         static void reset() {
4397*89c4ff92SAndroid Build Coastguard Worker             if(isSet) {
4398*89c4ff92SAndroid Build Coastguard Worker                 // Set signals back to previous values -- hopefully nobody overwrote them in the meantime
4399*89c4ff92SAndroid Build Coastguard Worker                 for(std::size_t i = 0; i < DOCTEST_COUNTOF(signalDefs); ++i) {
4400*89c4ff92SAndroid Build Coastguard Worker                     sigaction(signalDefs[i].id, &oldSigActions[i], nullptr);
4401*89c4ff92SAndroid Build Coastguard Worker                 }
4402*89c4ff92SAndroid Build Coastguard Worker                 // Return the old stack
4403*89c4ff92SAndroid Build Coastguard Worker                 sigaltstack(&oldSigStack, nullptr);
4404*89c4ff92SAndroid Build Coastguard Worker                 isSet = false;
4405*89c4ff92SAndroid Build Coastguard Worker             }
4406*89c4ff92SAndroid Build Coastguard Worker         }
4407*89c4ff92SAndroid Build Coastguard Worker     };
4408*89c4ff92SAndroid Build Coastguard Worker 
4409*89c4ff92SAndroid Build Coastguard Worker     bool             FatalConditionHandler::isSet = false;
4410*89c4ff92SAndroid Build Coastguard Worker     struct sigaction FatalConditionHandler::oldSigActions[DOCTEST_COUNTOF(signalDefs)] = {};
4411*89c4ff92SAndroid Build Coastguard Worker     stack_t          FatalConditionHandler::oldSigStack = {};
4412*89c4ff92SAndroid Build Coastguard Worker     size_t           FatalConditionHandler::altStackSize = 4 * SIGSTKSZ;
4413*89c4ff92SAndroid Build Coastguard Worker     char*            FatalConditionHandler::altStackMem = nullptr;
4414*89c4ff92SAndroid Build Coastguard Worker 
4415*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM_WINDOWS
4416*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_POSIX_SIGNALS || DOCTEST_CONFIG_WINDOWS_SEH
4417*89c4ff92SAndroid Build Coastguard Worker 
4418*89c4ff92SAndroid Build Coastguard Worker } // namespace
4419*89c4ff92SAndroid Build Coastguard Worker 
4420*89c4ff92SAndroid Build Coastguard Worker namespace {
4421*89c4ff92SAndroid Build Coastguard Worker     using namespace detail;
4422*89c4ff92SAndroid Build Coastguard Worker 
4423*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_WINDOWS
4424*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_OUTPUT_DEBUG_STRING(text) ::OutputDebugStringA(text)
4425*89c4ff92SAndroid Build Coastguard Worker #else
4426*89c4ff92SAndroid Build Coastguard Worker     // TODO: integration with XCode and other IDEs
4427*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_OUTPUT_DEBUG_STRING(text) // NOLINT(clang-diagnostic-unused-macros)
4428*89c4ff92SAndroid Build Coastguard Worker #endif // Platform
4429*89c4ff92SAndroid Build Coastguard Worker 
4430*89c4ff92SAndroid Build Coastguard Worker     void addAssert(assertType::Enum at) {
4431*89c4ff92SAndroid Build Coastguard Worker         if((at & assertType::is_warn) == 0) //!OCLINT bitwise operator in conditional
4432*89c4ff92SAndroid Build Coastguard Worker             g_cs->numAssertsCurrentTest_atomic++;
4433*89c4ff92SAndroid Build Coastguard Worker     }
4434*89c4ff92SAndroid Build Coastguard Worker 
4435*89c4ff92SAndroid Build Coastguard Worker     void addFailedAssert(assertType::Enum at) {
4436*89c4ff92SAndroid Build Coastguard Worker         if((at & assertType::is_warn) == 0) //!OCLINT bitwise operator in conditional
4437*89c4ff92SAndroid Build Coastguard Worker             g_cs->numAssertsFailedCurrentTest_atomic++;
4438*89c4ff92SAndroid Build Coastguard Worker     }
4439*89c4ff92SAndroid Build Coastguard Worker 
4440*89c4ff92SAndroid Build Coastguard Worker #if defined(DOCTEST_CONFIG_POSIX_SIGNALS) || defined(DOCTEST_CONFIG_WINDOWS_SEH)
4441*89c4ff92SAndroid Build Coastguard Worker     void reportFatal(const std::string& message) {
4442*89c4ff92SAndroid Build Coastguard Worker         g_cs->failure_flags |= TestCaseFailureReason::Crash;
4443*89c4ff92SAndroid Build Coastguard Worker 
4444*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_exception, {message.c_str(), true});
4445*89c4ff92SAndroid Build Coastguard Worker 
4446*89c4ff92SAndroid Build Coastguard Worker         while(g_cs->subcasesStack.size()) {
4447*89c4ff92SAndroid Build Coastguard Worker             g_cs->subcasesStack.pop_back();
4448*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_ITERATE_THROUGH_REPORTERS(subcase_end, DOCTEST_EMPTY);
4449*89c4ff92SAndroid Build Coastguard Worker         }
4450*89c4ff92SAndroid Build Coastguard Worker 
4451*89c4ff92SAndroid Build Coastguard Worker         g_cs->finalizeTestCaseData();
4452*89c4ff92SAndroid Build Coastguard Worker 
4453*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_end, *g_cs);
4454*89c4ff92SAndroid Build Coastguard Worker 
4455*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ITERATE_THROUGH_REPORTERS(test_run_end, *g_cs);
4456*89c4ff92SAndroid Build Coastguard Worker     }
4457*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_POSIX_SIGNALS || DOCTEST_CONFIG_WINDOWS_SEH
4458*89c4ff92SAndroid Build Coastguard Worker } // namespace
4459*89c4ff92SAndroid Build Coastguard Worker namespace detail {
4460*89c4ff92SAndroid Build Coastguard Worker 
4461*89c4ff92SAndroid Build Coastguard Worker     ResultBuilder::ResultBuilder(assertType::Enum at, const char* file, int line, const char* expr,
4462*89c4ff92SAndroid Build Coastguard Worker                                  const char* exception_type, const char* exception_string) {
4463*89c4ff92SAndroid Build Coastguard Worker         m_test_case        = g_cs->currentTest;
4464*89c4ff92SAndroid Build Coastguard Worker         m_at               = at;
4465*89c4ff92SAndroid Build Coastguard Worker         m_file             = file;
4466*89c4ff92SAndroid Build Coastguard Worker         m_line             = line;
4467*89c4ff92SAndroid Build Coastguard Worker         m_expr             = expr;
4468*89c4ff92SAndroid Build Coastguard Worker         m_failed           = true;
4469*89c4ff92SAndroid Build Coastguard Worker         m_threw            = false;
4470*89c4ff92SAndroid Build Coastguard Worker         m_threw_as         = false;
4471*89c4ff92SAndroid Build Coastguard Worker         m_exception_type   = exception_type;
4472*89c4ff92SAndroid Build Coastguard Worker         m_exception_string = exception_string;
4473*89c4ff92SAndroid Build Coastguard Worker #if DOCTEST_MSVC
4474*89c4ff92SAndroid Build Coastguard Worker         if(m_expr[0] == ' ') // this happens when variadic macros are disabled under MSVC
4475*89c4ff92SAndroid Build Coastguard Worker             ++m_expr;
4476*89c4ff92SAndroid Build Coastguard Worker #endif // MSVC
4477*89c4ff92SAndroid Build Coastguard Worker     }
4478*89c4ff92SAndroid Build Coastguard Worker 
4479*89c4ff92SAndroid Build Coastguard Worker     void ResultBuilder::setResult(const Result& res) {
4480*89c4ff92SAndroid Build Coastguard Worker         m_decomp = res.m_decomp;
4481*89c4ff92SAndroid Build Coastguard Worker         m_failed = !res.m_passed;
4482*89c4ff92SAndroid Build Coastguard Worker     }
4483*89c4ff92SAndroid Build Coastguard Worker 
4484*89c4ff92SAndroid Build Coastguard Worker     void ResultBuilder::translateException() {
4485*89c4ff92SAndroid Build Coastguard Worker         m_threw     = true;
4486*89c4ff92SAndroid Build Coastguard Worker         m_exception = translateActiveException();
4487*89c4ff92SAndroid Build Coastguard Worker     }
4488*89c4ff92SAndroid Build Coastguard Worker 
4489*89c4ff92SAndroid Build Coastguard Worker     bool ResultBuilder::log() {
4490*89c4ff92SAndroid Build Coastguard Worker         if(m_at & assertType::is_throws) { //!OCLINT bitwise operator in conditional
4491*89c4ff92SAndroid Build Coastguard Worker             m_failed = !m_threw;
4492*89c4ff92SAndroid Build Coastguard Worker         } else if((m_at & assertType::is_throws_as) && (m_at & assertType::is_throws_with)) { //!OCLINT
4493*89c4ff92SAndroid Build Coastguard Worker             m_failed = !m_threw_as || (m_exception != m_exception_string);
4494*89c4ff92SAndroid Build Coastguard Worker         } else if(m_at & assertType::is_throws_as) { //!OCLINT bitwise operator in conditional
4495*89c4ff92SAndroid Build Coastguard Worker             m_failed = !m_threw_as;
4496*89c4ff92SAndroid Build Coastguard Worker         } else if(m_at & assertType::is_throws_with) { //!OCLINT bitwise operator in conditional
4497*89c4ff92SAndroid Build Coastguard Worker             m_failed = m_exception != m_exception_string;
4498*89c4ff92SAndroid Build Coastguard Worker         } else if(m_at & assertType::is_nothrow) { //!OCLINT bitwise operator in conditional
4499*89c4ff92SAndroid Build Coastguard Worker             m_failed = m_threw;
4500*89c4ff92SAndroid Build Coastguard Worker         }
4501*89c4ff92SAndroid Build Coastguard Worker 
4502*89c4ff92SAndroid Build Coastguard Worker         if(m_exception.size())
4503*89c4ff92SAndroid Build Coastguard Worker             m_exception = String("\"") + m_exception + "\"";
4504*89c4ff92SAndroid Build Coastguard Worker 
4505*89c4ff92SAndroid Build Coastguard Worker         if(is_running_in_test) {
4506*89c4ff92SAndroid Build Coastguard Worker             addAssert(m_at);
4507*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_ITERATE_THROUGH_REPORTERS(log_assert, *this);
4508*89c4ff92SAndroid Build Coastguard Worker 
4509*89c4ff92SAndroid Build Coastguard Worker             if(m_failed)
4510*89c4ff92SAndroid Build Coastguard Worker                 addFailedAssert(m_at);
4511*89c4ff92SAndroid Build Coastguard Worker         } else if(m_failed) {
4512*89c4ff92SAndroid Build Coastguard Worker             failed_out_of_a_testing_context(*this);
4513*89c4ff92SAndroid Build Coastguard Worker         }
4514*89c4ff92SAndroid Build Coastguard Worker 
4515*89c4ff92SAndroid Build Coastguard Worker         return m_failed && isDebuggerActive() && !getContextOptions()->no_breaks &&
4516*89c4ff92SAndroid Build Coastguard Worker             (g_cs->currentTest == nullptr || !g_cs->currentTest->m_no_breaks); // break into debugger
4517*89c4ff92SAndroid Build Coastguard Worker     }
4518*89c4ff92SAndroid Build Coastguard Worker 
4519*89c4ff92SAndroid Build Coastguard Worker     void ResultBuilder::react() const {
4520*89c4ff92SAndroid Build Coastguard Worker         if(m_failed && checkIfShouldThrow(m_at))
4521*89c4ff92SAndroid Build Coastguard Worker             throwException();
4522*89c4ff92SAndroid Build Coastguard Worker     }
4523*89c4ff92SAndroid Build Coastguard Worker 
4524*89c4ff92SAndroid Build Coastguard Worker     void failed_out_of_a_testing_context(const AssertData& ad) {
4525*89c4ff92SAndroid Build Coastguard Worker         if(g_cs->ah)
4526*89c4ff92SAndroid Build Coastguard Worker             g_cs->ah(ad);
4527*89c4ff92SAndroid Build Coastguard Worker         else
4528*89c4ff92SAndroid Build Coastguard Worker             std::abort();
4529*89c4ff92SAndroid Build Coastguard Worker     }
4530*89c4ff92SAndroid Build Coastguard Worker 
4531*89c4ff92SAndroid Build Coastguard Worker     void decomp_assert(assertType::Enum at, const char* file, int line, const char* expr,
4532*89c4ff92SAndroid Build Coastguard Worker                        Result result) {
4533*89c4ff92SAndroid Build Coastguard Worker         bool failed = !result.m_passed;
4534*89c4ff92SAndroid Build Coastguard Worker 
4535*89c4ff92SAndroid Build Coastguard Worker         // ###################################################################################
4536*89c4ff92SAndroid Build Coastguard Worker         // IF THE DEBUGGER BREAKS HERE - GO 1 LEVEL UP IN THE CALLSTACK FOR THE FAILING ASSERT
4537*89c4ff92SAndroid Build Coastguard Worker         // THIS IS THE EFFECT OF HAVING 'DOCTEST_CONFIG_SUPER_FAST_ASSERTS' DEFINED
4538*89c4ff92SAndroid Build Coastguard Worker         // ###################################################################################
4539*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_OUT_OF_TESTS(result.m_decomp);
4540*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ASSERT_IN_TESTS(result.m_decomp);
4541*89c4ff92SAndroid Build Coastguard Worker         // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
4542*89c4ff92SAndroid Build Coastguard Worker     }
4543*89c4ff92SAndroid Build Coastguard Worker 
4544*89c4ff92SAndroid Build Coastguard Worker     MessageBuilder::MessageBuilder(const char* file, int line, assertType::Enum severity) {
4545*89c4ff92SAndroid Build Coastguard Worker         m_stream   = getTlsOss();
4546*89c4ff92SAndroid Build Coastguard Worker         m_file     = file;
4547*89c4ff92SAndroid Build Coastguard Worker         m_line     = line;
4548*89c4ff92SAndroid Build Coastguard Worker         m_severity = severity;
4549*89c4ff92SAndroid Build Coastguard Worker     }
4550*89c4ff92SAndroid Build Coastguard Worker 
4551*89c4ff92SAndroid Build Coastguard Worker     IExceptionTranslator::IExceptionTranslator()  = default;
4552*89c4ff92SAndroid Build Coastguard Worker     IExceptionTranslator::~IExceptionTranslator() = default;
4553*89c4ff92SAndroid Build Coastguard Worker 
4554*89c4ff92SAndroid Build Coastguard Worker     bool MessageBuilder::log() {
4555*89c4ff92SAndroid Build Coastguard Worker         m_string = getTlsOssResult();
4556*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ITERATE_THROUGH_REPORTERS(log_message, *this);
4557*89c4ff92SAndroid Build Coastguard Worker 
4558*89c4ff92SAndroid Build Coastguard Worker         const bool isWarn = m_severity & assertType::is_warn;
4559*89c4ff92SAndroid Build Coastguard Worker 
4560*89c4ff92SAndroid Build Coastguard Worker         // warn is just a message in this context so we don't treat it as an assert
4561*89c4ff92SAndroid Build Coastguard Worker         if(!isWarn) {
4562*89c4ff92SAndroid Build Coastguard Worker             addAssert(m_severity);
4563*89c4ff92SAndroid Build Coastguard Worker             addFailedAssert(m_severity);
4564*89c4ff92SAndroid Build Coastguard Worker         }
4565*89c4ff92SAndroid Build Coastguard Worker 
4566*89c4ff92SAndroid Build Coastguard Worker         return isDebuggerActive() && !getContextOptions()->no_breaks && !isWarn &&
4567*89c4ff92SAndroid Build Coastguard Worker             (g_cs->currentTest == nullptr || !g_cs->currentTest->m_no_breaks); // break into debugger
4568*89c4ff92SAndroid Build Coastguard Worker     }
4569*89c4ff92SAndroid Build Coastguard Worker 
4570*89c4ff92SAndroid Build Coastguard Worker     void MessageBuilder::react() {
4571*89c4ff92SAndroid Build Coastguard Worker         if(m_severity & assertType::is_require) //!OCLINT bitwise operator in conditional
4572*89c4ff92SAndroid Build Coastguard Worker             throwException();
4573*89c4ff92SAndroid Build Coastguard Worker     }
4574*89c4ff92SAndroid Build Coastguard Worker 
4575*89c4ff92SAndroid Build Coastguard Worker     MessageBuilder::~MessageBuilder() = default;
4576*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
4577*89c4ff92SAndroid Build Coastguard Worker namespace {
4578*89c4ff92SAndroid Build Coastguard Worker     using namespace detail;
4579*89c4ff92SAndroid Build Coastguard Worker 
4580*89c4ff92SAndroid Build Coastguard Worker     template <typename Ex>
4581*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_NORETURN void throw_exception(Ex const& e) {
4582*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_EXCEPTIONS
4583*89c4ff92SAndroid Build Coastguard Worker         throw e;
4584*89c4ff92SAndroid Build Coastguard Worker #else  // DOCTEST_CONFIG_NO_EXCEPTIONS
4585*89c4ff92SAndroid Build Coastguard Worker         std::cerr << "doctest will terminate because it needed to throw an exception.\n"
4586*89c4ff92SAndroid Build Coastguard Worker                   << "The message was: " << e.what() << '\n';
4587*89c4ff92SAndroid Build Coastguard Worker         std::terminate();
4588*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS
4589*89c4ff92SAndroid Build Coastguard Worker     }
4590*89c4ff92SAndroid Build Coastguard Worker 
4591*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_INTERNAL_ERROR
4592*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_INTERNAL_ERROR(msg)                                                                \
4593*89c4ff92SAndroid Build Coastguard Worker     throw_exception(std::logic_error(                                                              \
4594*89c4ff92SAndroid Build Coastguard Worker             __FILE__ ":" DOCTEST_TOSTR(__LINE__) ": Internal doctest error: " msg))
4595*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_INTERNAL_ERROR
4596*89c4ff92SAndroid Build Coastguard Worker 
4597*89c4ff92SAndroid Build Coastguard Worker     // clang-format off
4598*89c4ff92SAndroid Build Coastguard Worker 
4599*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
4600*89c4ff92SAndroid Build Coastguard Worker // The following code has been taken verbatim from Catch2/include/internal/catch_xmlwriter.h/cpp
4601*89c4ff92SAndroid Build Coastguard Worker // This is done so cherry-picking bug fixes is trivial - even the style/formatting is untouched.
4602*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
4603*89c4ff92SAndroid Build Coastguard Worker 
4604*89c4ff92SAndroid Build Coastguard Worker     class XmlEncode {
4605*89c4ff92SAndroid Build Coastguard Worker     public:
4606*89c4ff92SAndroid Build Coastguard Worker         enum ForWhat { ForTextNodes, ForAttributes };
4607*89c4ff92SAndroid Build Coastguard Worker 
4608*89c4ff92SAndroid Build Coastguard Worker         XmlEncode( std::string const& str, ForWhat forWhat = ForTextNodes );
4609*89c4ff92SAndroid Build Coastguard Worker 
4610*89c4ff92SAndroid Build Coastguard Worker         void encodeTo( std::ostream& os ) const;
4611*89c4ff92SAndroid Build Coastguard Worker 
4612*89c4ff92SAndroid Build Coastguard Worker         friend std::ostream& operator << ( std::ostream& os, XmlEncode const& xmlEncode );
4613*89c4ff92SAndroid Build Coastguard Worker 
4614*89c4ff92SAndroid Build Coastguard Worker     private:
4615*89c4ff92SAndroid Build Coastguard Worker         std::string m_str;
4616*89c4ff92SAndroid Build Coastguard Worker         ForWhat m_forWhat;
4617*89c4ff92SAndroid Build Coastguard Worker     };
4618*89c4ff92SAndroid Build Coastguard Worker 
4619*89c4ff92SAndroid Build Coastguard Worker     class XmlWriter {
4620*89c4ff92SAndroid Build Coastguard Worker     public:
4621*89c4ff92SAndroid Build Coastguard Worker 
4622*89c4ff92SAndroid Build Coastguard Worker         class ScopedElement {
4623*89c4ff92SAndroid Build Coastguard Worker         public:
4624*89c4ff92SAndroid Build Coastguard Worker             ScopedElement( XmlWriter* writer );
4625*89c4ff92SAndroid Build Coastguard Worker 
4626*89c4ff92SAndroid Build Coastguard Worker             ScopedElement( ScopedElement&& other ) DOCTEST_NOEXCEPT;
4627*89c4ff92SAndroid Build Coastguard Worker             ScopedElement& operator=( ScopedElement&& other ) DOCTEST_NOEXCEPT;
4628*89c4ff92SAndroid Build Coastguard Worker 
4629*89c4ff92SAndroid Build Coastguard Worker             ~ScopedElement();
4630*89c4ff92SAndroid Build Coastguard Worker 
4631*89c4ff92SAndroid Build Coastguard Worker             ScopedElement& writeText( std::string const& text, bool indent = true );
4632*89c4ff92SAndroid Build Coastguard Worker 
4633*89c4ff92SAndroid Build Coastguard Worker             template<typename T>
4634*89c4ff92SAndroid Build Coastguard Worker             ScopedElement& writeAttribute( std::string const& name, T const& attribute ) {
4635*89c4ff92SAndroid Build Coastguard Worker                 m_writer->writeAttribute( name, attribute );
4636*89c4ff92SAndroid Build Coastguard Worker                 return *this;
4637*89c4ff92SAndroid Build Coastguard Worker             }
4638*89c4ff92SAndroid Build Coastguard Worker 
4639*89c4ff92SAndroid Build Coastguard Worker         private:
4640*89c4ff92SAndroid Build Coastguard Worker             mutable XmlWriter* m_writer = nullptr;
4641*89c4ff92SAndroid Build Coastguard Worker         };
4642*89c4ff92SAndroid Build Coastguard Worker 
4643*89c4ff92SAndroid Build Coastguard Worker         XmlWriter( std::ostream& os = std::cout );
4644*89c4ff92SAndroid Build Coastguard Worker         ~XmlWriter();
4645*89c4ff92SAndroid Build Coastguard Worker 
4646*89c4ff92SAndroid Build Coastguard Worker         XmlWriter( XmlWriter const& ) = delete;
4647*89c4ff92SAndroid Build Coastguard Worker         XmlWriter& operator=( XmlWriter const& ) = delete;
4648*89c4ff92SAndroid Build Coastguard Worker 
4649*89c4ff92SAndroid Build Coastguard Worker         XmlWriter& startElement( std::string const& name );
4650*89c4ff92SAndroid Build Coastguard Worker 
4651*89c4ff92SAndroid Build Coastguard Worker         ScopedElement scopedElement( std::string const& name );
4652*89c4ff92SAndroid Build Coastguard Worker 
4653*89c4ff92SAndroid Build Coastguard Worker         XmlWriter& endElement();
4654*89c4ff92SAndroid Build Coastguard Worker 
4655*89c4ff92SAndroid Build Coastguard Worker         XmlWriter& writeAttribute( std::string const& name, std::string const& attribute );
4656*89c4ff92SAndroid Build Coastguard Worker 
4657*89c4ff92SAndroid Build Coastguard Worker         XmlWriter& writeAttribute( std::string const& name, const char* attribute );
4658*89c4ff92SAndroid Build Coastguard Worker 
4659*89c4ff92SAndroid Build Coastguard Worker         XmlWriter& writeAttribute( std::string const& name, bool attribute );
4660*89c4ff92SAndroid Build Coastguard Worker 
4661*89c4ff92SAndroid Build Coastguard Worker         template<typename T>
4662*89c4ff92SAndroid Build Coastguard Worker         XmlWriter& writeAttribute( std::string const& name, T const& attribute ) {
4663*89c4ff92SAndroid Build Coastguard Worker         std::stringstream rss;
4664*89c4ff92SAndroid Build Coastguard Worker             rss << attribute;
4665*89c4ff92SAndroid Build Coastguard Worker             return writeAttribute( name, rss.str() );
4666*89c4ff92SAndroid Build Coastguard Worker         }
4667*89c4ff92SAndroid Build Coastguard Worker 
4668*89c4ff92SAndroid Build Coastguard Worker         XmlWriter& writeText( std::string const& text, bool indent = true );
4669*89c4ff92SAndroid Build Coastguard Worker 
4670*89c4ff92SAndroid Build Coastguard Worker         //XmlWriter& writeComment( std::string const& text );
4671*89c4ff92SAndroid Build Coastguard Worker 
4672*89c4ff92SAndroid Build Coastguard Worker         //void writeStylesheetRef( std::string const& url );
4673*89c4ff92SAndroid Build Coastguard Worker 
4674*89c4ff92SAndroid Build Coastguard Worker         //XmlWriter& writeBlankLine();
4675*89c4ff92SAndroid Build Coastguard Worker 
4676*89c4ff92SAndroid Build Coastguard Worker         void ensureTagClosed();
4677*89c4ff92SAndroid Build Coastguard Worker 
4678*89c4ff92SAndroid Build Coastguard Worker     private:
4679*89c4ff92SAndroid Build Coastguard Worker 
4680*89c4ff92SAndroid Build Coastguard Worker         void writeDeclaration();
4681*89c4ff92SAndroid Build Coastguard Worker 
4682*89c4ff92SAndroid Build Coastguard Worker         void newlineIfNecessary();
4683*89c4ff92SAndroid Build Coastguard Worker 
4684*89c4ff92SAndroid Build Coastguard Worker         bool m_tagIsOpen = false;
4685*89c4ff92SAndroid Build Coastguard Worker         bool m_needsNewline = false;
4686*89c4ff92SAndroid Build Coastguard Worker         std::vector<std::string> m_tags;
4687*89c4ff92SAndroid Build Coastguard Worker         std::string m_indent;
4688*89c4ff92SAndroid Build Coastguard Worker         std::ostream& m_os;
4689*89c4ff92SAndroid Build Coastguard Worker     };
4690*89c4ff92SAndroid Build Coastguard Worker 
4691*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
4692*89c4ff92SAndroid Build Coastguard Worker // The following code has been taken verbatim from Catch2/include/internal/catch_xmlwriter.h/cpp
4693*89c4ff92SAndroid Build Coastguard Worker // This is done so cherry-picking bug fixes is trivial - even the style/formatting is untouched.
4694*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
4695*89c4ff92SAndroid Build Coastguard Worker 
4696*89c4ff92SAndroid Build Coastguard Worker using uchar = unsigned char;
4697*89c4ff92SAndroid Build Coastguard Worker 
4698*89c4ff92SAndroid Build Coastguard Worker namespace {
4699*89c4ff92SAndroid Build Coastguard Worker 
4700*89c4ff92SAndroid Build Coastguard Worker     size_t trailingBytes(unsigned char c) {
4701*89c4ff92SAndroid Build Coastguard Worker         if ((c & 0xE0) == 0xC0) {
4702*89c4ff92SAndroid Build Coastguard Worker             return 2;
4703*89c4ff92SAndroid Build Coastguard Worker         }
4704*89c4ff92SAndroid Build Coastguard Worker         if ((c & 0xF0) == 0xE0) {
4705*89c4ff92SAndroid Build Coastguard Worker             return 3;
4706*89c4ff92SAndroid Build Coastguard Worker         }
4707*89c4ff92SAndroid Build Coastguard Worker         if ((c & 0xF8) == 0xF0) {
4708*89c4ff92SAndroid Build Coastguard Worker             return 4;
4709*89c4ff92SAndroid Build Coastguard Worker         }
4710*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_INTERNAL_ERROR("Invalid multibyte utf-8 start byte encountered");
4711*89c4ff92SAndroid Build Coastguard Worker     }
4712*89c4ff92SAndroid Build Coastguard Worker 
4713*89c4ff92SAndroid Build Coastguard Worker     uint32_t headerValue(unsigned char c) {
4714*89c4ff92SAndroid Build Coastguard Worker         if ((c & 0xE0) == 0xC0) {
4715*89c4ff92SAndroid Build Coastguard Worker             return c & 0x1F;
4716*89c4ff92SAndroid Build Coastguard Worker         }
4717*89c4ff92SAndroid Build Coastguard Worker         if ((c & 0xF0) == 0xE0) {
4718*89c4ff92SAndroid Build Coastguard Worker             return c & 0x0F;
4719*89c4ff92SAndroid Build Coastguard Worker         }
4720*89c4ff92SAndroid Build Coastguard Worker         if ((c & 0xF8) == 0xF0) {
4721*89c4ff92SAndroid Build Coastguard Worker             return c & 0x07;
4722*89c4ff92SAndroid Build Coastguard Worker         }
4723*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_INTERNAL_ERROR("Invalid multibyte utf-8 start byte encountered");
4724*89c4ff92SAndroid Build Coastguard Worker     }
4725*89c4ff92SAndroid Build Coastguard Worker 
4726*89c4ff92SAndroid Build Coastguard Worker     void hexEscapeChar(std::ostream& os, unsigned char c) {
4727*89c4ff92SAndroid Build Coastguard Worker         std::ios_base::fmtflags f(os.flags());
4728*89c4ff92SAndroid Build Coastguard Worker         os << "\\x"
4729*89c4ff92SAndroid Build Coastguard Worker             << std::uppercase << std::hex << std::setfill('0') << std::setw(2)
4730*89c4ff92SAndroid Build Coastguard Worker             << static_cast<int>(c);
4731*89c4ff92SAndroid Build Coastguard Worker         os.flags(f);
4732*89c4ff92SAndroid Build Coastguard Worker     }
4733*89c4ff92SAndroid Build Coastguard Worker 
4734*89c4ff92SAndroid Build Coastguard Worker } // anonymous namespace
4735*89c4ff92SAndroid Build Coastguard Worker 
4736*89c4ff92SAndroid Build Coastguard Worker     XmlEncode::XmlEncode( std::string const& str, ForWhat forWhat )
4737*89c4ff92SAndroid Build Coastguard Worker     :   m_str( str ),
4738*89c4ff92SAndroid Build Coastguard Worker         m_forWhat( forWhat )
4739*89c4ff92SAndroid Build Coastguard Worker     {}
4740*89c4ff92SAndroid Build Coastguard Worker 
4741*89c4ff92SAndroid Build Coastguard Worker     void XmlEncode::encodeTo( std::ostream& os ) const {
4742*89c4ff92SAndroid Build Coastguard Worker         // Apostrophe escaping not necessary if we always use " to write attributes
4743*89c4ff92SAndroid Build Coastguard Worker         // (see: https://www.w3.org/TR/xml/#syntax)
4744*89c4ff92SAndroid Build Coastguard Worker 
4745*89c4ff92SAndroid Build Coastguard Worker         for( std::size_t idx = 0; idx < m_str.size(); ++ idx ) {
4746*89c4ff92SAndroid Build Coastguard Worker             uchar c = m_str[idx];
4747*89c4ff92SAndroid Build Coastguard Worker             switch (c) {
4748*89c4ff92SAndroid Build Coastguard Worker             case '<':   os << "&lt;"; break;
4749*89c4ff92SAndroid Build Coastguard Worker             case '&':   os << "&amp;"; break;
4750*89c4ff92SAndroid Build Coastguard Worker 
4751*89c4ff92SAndroid Build Coastguard Worker             case '>':
4752*89c4ff92SAndroid Build Coastguard Worker                 // See: https://www.w3.org/TR/xml/#syntax
4753*89c4ff92SAndroid Build Coastguard Worker                 if (idx > 2 && m_str[idx - 1] == ']' && m_str[idx - 2] == ']')
4754*89c4ff92SAndroid Build Coastguard Worker                     os << "&gt;";
4755*89c4ff92SAndroid Build Coastguard Worker                 else
4756*89c4ff92SAndroid Build Coastguard Worker                     os << c;
4757*89c4ff92SAndroid Build Coastguard Worker                 break;
4758*89c4ff92SAndroid Build Coastguard Worker 
4759*89c4ff92SAndroid Build Coastguard Worker             case '\"':
4760*89c4ff92SAndroid Build Coastguard Worker                 if (m_forWhat == ForAttributes)
4761*89c4ff92SAndroid Build Coastguard Worker                     os << "&quot;";
4762*89c4ff92SAndroid Build Coastguard Worker                 else
4763*89c4ff92SAndroid Build Coastguard Worker                     os << c;
4764*89c4ff92SAndroid Build Coastguard Worker                 break;
4765*89c4ff92SAndroid Build Coastguard Worker 
4766*89c4ff92SAndroid Build Coastguard Worker             default:
4767*89c4ff92SAndroid Build Coastguard Worker                 // Check for control characters and invalid utf-8
4768*89c4ff92SAndroid Build Coastguard Worker 
4769*89c4ff92SAndroid Build Coastguard Worker                 // Escape control characters in standard ascii
4770*89c4ff92SAndroid Build Coastguard Worker                 // see https://stackoverflow.com/questions/404107/why-are-control-characters-illegal-in-xml-1-0
4771*89c4ff92SAndroid Build Coastguard Worker                 if (c < 0x09 || (c > 0x0D && c < 0x20) || c == 0x7F) {
4772*89c4ff92SAndroid Build Coastguard Worker                     hexEscapeChar(os, c);
4773*89c4ff92SAndroid Build Coastguard Worker                     break;
4774*89c4ff92SAndroid Build Coastguard Worker                 }
4775*89c4ff92SAndroid Build Coastguard Worker 
4776*89c4ff92SAndroid Build Coastguard Worker                 // Plain ASCII: Write it to stream
4777*89c4ff92SAndroid Build Coastguard Worker                 if (c < 0x7F) {
4778*89c4ff92SAndroid Build Coastguard Worker                     os << c;
4779*89c4ff92SAndroid Build Coastguard Worker                     break;
4780*89c4ff92SAndroid Build Coastguard Worker                 }
4781*89c4ff92SAndroid Build Coastguard Worker 
4782*89c4ff92SAndroid Build Coastguard Worker                 // UTF-8 territory
4783*89c4ff92SAndroid Build Coastguard Worker                 // Check if the encoding is valid and if it is not, hex escape bytes.
4784*89c4ff92SAndroid Build Coastguard Worker                 // Important: We do not check the exact decoded values for validity, only the encoding format
4785*89c4ff92SAndroid Build Coastguard Worker                 // First check that this bytes is a valid lead byte:
4786*89c4ff92SAndroid Build Coastguard Worker                 // This means that it is not encoded as 1111 1XXX
4787*89c4ff92SAndroid Build Coastguard Worker                 // Or as 10XX XXXX
4788*89c4ff92SAndroid Build Coastguard Worker                 if (c <  0xC0 ||
4789*89c4ff92SAndroid Build Coastguard Worker                     c >= 0xF8) {
4790*89c4ff92SAndroid Build Coastguard Worker                     hexEscapeChar(os, c);
4791*89c4ff92SAndroid Build Coastguard Worker                     break;
4792*89c4ff92SAndroid Build Coastguard Worker                 }
4793*89c4ff92SAndroid Build Coastguard Worker 
4794*89c4ff92SAndroid Build Coastguard Worker                 auto encBytes = trailingBytes(c);
4795*89c4ff92SAndroid Build Coastguard Worker                 // Are there enough bytes left to avoid accessing out-of-bounds memory?
4796*89c4ff92SAndroid Build Coastguard Worker                 if (idx + encBytes - 1 >= m_str.size()) {
4797*89c4ff92SAndroid Build Coastguard Worker                     hexEscapeChar(os, c);
4798*89c4ff92SAndroid Build Coastguard Worker                     break;
4799*89c4ff92SAndroid Build Coastguard Worker                 }
4800*89c4ff92SAndroid Build Coastguard Worker                 // The header is valid, check data
4801*89c4ff92SAndroid Build Coastguard Worker                 // The next encBytes bytes must together be a valid utf-8
4802*89c4ff92SAndroid Build Coastguard Worker                 // This means: bitpattern 10XX XXXX and the extracted value is sane (ish)
4803*89c4ff92SAndroid Build Coastguard Worker                 bool valid = true;
4804*89c4ff92SAndroid Build Coastguard Worker                 uint32_t value = headerValue(c);
4805*89c4ff92SAndroid Build Coastguard Worker                 for (std::size_t n = 1; n < encBytes; ++n) {
4806*89c4ff92SAndroid Build Coastguard Worker                     uchar nc = m_str[idx + n];
4807*89c4ff92SAndroid Build Coastguard Worker                     valid &= ((nc & 0xC0) == 0x80);
4808*89c4ff92SAndroid Build Coastguard Worker                     value = (value << 6) | (nc & 0x3F);
4809*89c4ff92SAndroid Build Coastguard Worker                 }
4810*89c4ff92SAndroid Build Coastguard Worker 
4811*89c4ff92SAndroid Build Coastguard Worker                 if (
4812*89c4ff92SAndroid Build Coastguard Worker                     // Wrong bit pattern of following bytes
4813*89c4ff92SAndroid Build Coastguard Worker                     (!valid) ||
4814*89c4ff92SAndroid Build Coastguard Worker                     // Overlong encodings
4815*89c4ff92SAndroid Build Coastguard Worker                     (value < 0x80) ||
4816*89c4ff92SAndroid Build Coastguard Worker                     (                 value < 0x800   && encBytes > 2) || // removed "0x80 <= value &&" because redundant
4817*89c4ff92SAndroid Build Coastguard Worker                     (0x800 < value && value < 0x10000 && encBytes > 3) ||
4818*89c4ff92SAndroid Build Coastguard Worker                     // Encoded value out of range
4819*89c4ff92SAndroid Build Coastguard Worker                     (value >= 0x110000)
4820*89c4ff92SAndroid Build Coastguard Worker                     ) {
4821*89c4ff92SAndroid Build Coastguard Worker                     hexEscapeChar(os, c);
4822*89c4ff92SAndroid Build Coastguard Worker                     break;
4823*89c4ff92SAndroid Build Coastguard Worker                 }
4824*89c4ff92SAndroid Build Coastguard Worker 
4825*89c4ff92SAndroid Build Coastguard Worker                 // If we got here, this is in fact a valid(ish) utf-8 sequence
4826*89c4ff92SAndroid Build Coastguard Worker                 for (std::size_t n = 0; n < encBytes; ++n) {
4827*89c4ff92SAndroid Build Coastguard Worker                     os << m_str[idx + n];
4828*89c4ff92SAndroid Build Coastguard Worker                 }
4829*89c4ff92SAndroid Build Coastguard Worker                 idx += encBytes - 1;
4830*89c4ff92SAndroid Build Coastguard Worker                 break;
4831*89c4ff92SAndroid Build Coastguard Worker             }
4832*89c4ff92SAndroid Build Coastguard Worker         }
4833*89c4ff92SAndroid Build Coastguard Worker     }
4834*89c4ff92SAndroid Build Coastguard Worker 
4835*89c4ff92SAndroid Build Coastguard Worker     std::ostream& operator << ( std::ostream& os, XmlEncode const& xmlEncode ) {
4836*89c4ff92SAndroid Build Coastguard Worker         xmlEncode.encodeTo( os );
4837*89c4ff92SAndroid Build Coastguard Worker         return os;
4838*89c4ff92SAndroid Build Coastguard Worker     }
4839*89c4ff92SAndroid Build Coastguard Worker 
4840*89c4ff92SAndroid Build Coastguard Worker     XmlWriter::ScopedElement::ScopedElement( XmlWriter* writer )
4841*89c4ff92SAndroid Build Coastguard Worker     :   m_writer( writer )
4842*89c4ff92SAndroid Build Coastguard Worker     {}
4843*89c4ff92SAndroid Build Coastguard Worker 
4844*89c4ff92SAndroid Build Coastguard Worker     XmlWriter::ScopedElement::ScopedElement( ScopedElement&& other ) DOCTEST_NOEXCEPT
4845*89c4ff92SAndroid Build Coastguard Worker     :   m_writer( other.m_writer ){
4846*89c4ff92SAndroid Build Coastguard Worker         other.m_writer = nullptr;
4847*89c4ff92SAndroid Build Coastguard Worker     }
4848*89c4ff92SAndroid Build Coastguard Worker     XmlWriter::ScopedElement& XmlWriter::ScopedElement::operator=( ScopedElement&& other ) DOCTEST_NOEXCEPT {
4849*89c4ff92SAndroid Build Coastguard Worker         if ( m_writer ) {
4850*89c4ff92SAndroid Build Coastguard Worker             m_writer->endElement();
4851*89c4ff92SAndroid Build Coastguard Worker         }
4852*89c4ff92SAndroid Build Coastguard Worker         m_writer = other.m_writer;
4853*89c4ff92SAndroid Build Coastguard Worker         other.m_writer = nullptr;
4854*89c4ff92SAndroid Build Coastguard Worker         return *this;
4855*89c4ff92SAndroid Build Coastguard Worker     }
4856*89c4ff92SAndroid Build Coastguard Worker 
4857*89c4ff92SAndroid Build Coastguard Worker 
4858*89c4ff92SAndroid Build Coastguard Worker     XmlWriter::ScopedElement::~ScopedElement() {
4859*89c4ff92SAndroid Build Coastguard Worker         if( m_writer )
4860*89c4ff92SAndroid Build Coastguard Worker             m_writer->endElement();
4861*89c4ff92SAndroid Build Coastguard Worker     }
4862*89c4ff92SAndroid Build Coastguard Worker 
4863*89c4ff92SAndroid Build Coastguard Worker     XmlWriter::ScopedElement& XmlWriter::ScopedElement::writeText( std::string const& text, bool indent ) {
4864*89c4ff92SAndroid Build Coastguard Worker         m_writer->writeText( text, indent );
4865*89c4ff92SAndroid Build Coastguard Worker         return *this;
4866*89c4ff92SAndroid Build Coastguard Worker     }
4867*89c4ff92SAndroid Build Coastguard Worker 
4868*89c4ff92SAndroid Build Coastguard Worker     XmlWriter::XmlWriter( std::ostream& os ) : m_os( os )
4869*89c4ff92SAndroid Build Coastguard Worker     {
4870*89c4ff92SAndroid Build Coastguard Worker         writeDeclaration();
4871*89c4ff92SAndroid Build Coastguard Worker     }
4872*89c4ff92SAndroid Build Coastguard Worker 
4873*89c4ff92SAndroid Build Coastguard Worker     XmlWriter::~XmlWriter() {
4874*89c4ff92SAndroid Build Coastguard Worker         while( !m_tags.empty() )
4875*89c4ff92SAndroid Build Coastguard Worker             endElement();
4876*89c4ff92SAndroid Build Coastguard Worker     }
4877*89c4ff92SAndroid Build Coastguard Worker 
4878*89c4ff92SAndroid Build Coastguard Worker     XmlWriter& XmlWriter::startElement( std::string const& name ) {
4879*89c4ff92SAndroid Build Coastguard Worker         ensureTagClosed();
4880*89c4ff92SAndroid Build Coastguard Worker         newlineIfNecessary();
4881*89c4ff92SAndroid Build Coastguard Worker         m_os << m_indent << '<' << name;
4882*89c4ff92SAndroid Build Coastguard Worker         m_tags.push_back( name );
4883*89c4ff92SAndroid Build Coastguard Worker         m_indent += "  ";
4884*89c4ff92SAndroid Build Coastguard Worker         m_tagIsOpen = true;
4885*89c4ff92SAndroid Build Coastguard Worker         return *this;
4886*89c4ff92SAndroid Build Coastguard Worker     }
4887*89c4ff92SAndroid Build Coastguard Worker 
4888*89c4ff92SAndroid Build Coastguard Worker     XmlWriter::ScopedElement XmlWriter::scopedElement( std::string const& name ) {
4889*89c4ff92SAndroid Build Coastguard Worker         ScopedElement scoped( this );
4890*89c4ff92SAndroid Build Coastguard Worker         startElement( name );
4891*89c4ff92SAndroid Build Coastguard Worker         return scoped;
4892*89c4ff92SAndroid Build Coastguard Worker     }
4893*89c4ff92SAndroid Build Coastguard Worker 
4894*89c4ff92SAndroid Build Coastguard Worker     XmlWriter& XmlWriter::endElement() {
4895*89c4ff92SAndroid Build Coastguard Worker         newlineIfNecessary();
4896*89c4ff92SAndroid Build Coastguard Worker         m_indent = m_indent.substr( 0, m_indent.size()-2 );
4897*89c4ff92SAndroid Build Coastguard Worker         if( m_tagIsOpen ) {
4898*89c4ff92SAndroid Build Coastguard Worker             m_os << "/>";
4899*89c4ff92SAndroid Build Coastguard Worker             m_tagIsOpen = false;
4900*89c4ff92SAndroid Build Coastguard Worker         }
4901*89c4ff92SAndroid Build Coastguard Worker         else {
4902*89c4ff92SAndroid Build Coastguard Worker             m_os << m_indent << "</" << m_tags.back() << ">";
4903*89c4ff92SAndroid Build Coastguard Worker         }
4904*89c4ff92SAndroid Build Coastguard Worker         m_os << std::endl;
4905*89c4ff92SAndroid Build Coastguard Worker         m_tags.pop_back();
4906*89c4ff92SAndroid Build Coastguard Worker         return *this;
4907*89c4ff92SAndroid Build Coastguard Worker     }
4908*89c4ff92SAndroid Build Coastguard Worker 
4909*89c4ff92SAndroid Build Coastguard Worker     XmlWriter& XmlWriter::writeAttribute( std::string const& name, std::string const& attribute ) {
4910*89c4ff92SAndroid Build Coastguard Worker         if( !name.empty() && !attribute.empty() )
4911*89c4ff92SAndroid Build Coastguard Worker             m_os << ' ' << name << "=\"" << XmlEncode( attribute, XmlEncode::ForAttributes ) << '"';
4912*89c4ff92SAndroid Build Coastguard Worker         return *this;
4913*89c4ff92SAndroid Build Coastguard Worker     }
4914*89c4ff92SAndroid Build Coastguard Worker 
4915*89c4ff92SAndroid Build Coastguard Worker     XmlWriter& XmlWriter::writeAttribute( std::string const& name, const char* attribute ) {
4916*89c4ff92SAndroid Build Coastguard Worker         if( !name.empty() && attribute && attribute[0] != '\0' )
4917*89c4ff92SAndroid Build Coastguard Worker             m_os << ' ' << name << "=\"" << XmlEncode( attribute, XmlEncode::ForAttributes ) << '"';
4918*89c4ff92SAndroid Build Coastguard Worker         return *this;
4919*89c4ff92SAndroid Build Coastguard Worker     }
4920*89c4ff92SAndroid Build Coastguard Worker 
4921*89c4ff92SAndroid Build Coastguard Worker     XmlWriter& XmlWriter::writeAttribute( std::string const& name, bool attribute ) {
4922*89c4ff92SAndroid Build Coastguard Worker         m_os << ' ' << name << "=\"" << ( attribute ? "true" : "false" ) << '"';
4923*89c4ff92SAndroid Build Coastguard Worker         return *this;
4924*89c4ff92SAndroid Build Coastguard Worker     }
4925*89c4ff92SAndroid Build Coastguard Worker 
4926*89c4ff92SAndroid Build Coastguard Worker     XmlWriter& XmlWriter::writeText( std::string const& text, bool indent ) {
4927*89c4ff92SAndroid Build Coastguard Worker         if( !text.empty() ){
4928*89c4ff92SAndroid Build Coastguard Worker             bool tagWasOpen = m_tagIsOpen;
4929*89c4ff92SAndroid Build Coastguard Worker             ensureTagClosed();
4930*89c4ff92SAndroid Build Coastguard Worker             if( tagWasOpen && indent )
4931*89c4ff92SAndroid Build Coastguard Worker                 m_os << m_indent;
4932*89c4ff92SAndroid Build Coastguard Worker             m_os << XmlEncode( text );
4933*89c4ff92SAndroid Build Coastguard Worker             m_needsNewline = true;
4934*89c4ff92SAndroid Build Coastguard Worker         }
4935*89c4ff92SAndroid Build Coastguard Worker         return *this;
4936*89c4ff92SAndroid Build Coastguard Worker     }
4937*89c4ff92SAndroid Build Coastguard Worker 
4938*89c4ff92SAndroid Build Coastguard Worker     //XmlWriter& XmlWriter::writeComment( std::string const& text ) {
4939*89c4ff92SAndroid Build Coastguard Worker     //    ensureTagClosed();
4940*89c4ff92SAndroid Build Coastguard Worker     //    m_os << m_indent << "<!--" << text << "-->";
4941*89c4ff92SAndroid Build Coastguard Worker     //    m_needsNewline = true;
4942*89c4ff92SAndroid Build Coastguard Worker     //    return *this;
4943*89c4ff92SAndroid Build Coastguard Worker     //}
4944*89c4ff92SAndroid Build Coastguard Worker 
4945*89c4ff92SAndroid Build Coastguard Worker     //void XmlWriter::writeStylesheetRef( std::string const& url ) {
4946*89c4ff92SAndroid Build Coastguard Worker     //    m_os << "<?xml-stylesheet type=\"text/xsl\" href=\"" << url << "\"?>\n";
4947*89c4ff92SAndroid Build Coastguard Worker     //}
4948*89c4ff92SAndroid Build Coastguard Worker 
4949*89c4ff92SAndroid Build Coastguard Worker     //XmlWriter& XmlWriter::writeBlankLine() {
4950*89c4ff92SAndroid Build Coastguard Worker     //    ensureTagClosed();
4951*89c4ff92SAndroid Build Coastguard Worker     //    m_os << '\n';
4952*89c4ff92SAndroid Build Coastguard Worker     //    return *this;
4953*89c4ff92SAndroid Build Coastguard Worker     //}
4954*89c4ff92SAndroid Build Coastguard Worker 
4955*89c4ff92SAndroid Build Coastguard Worker     void XmlWriter::ensureTagClosed() {
4956*89c4ff92SAndroid Build Coastguard Worker         if( m_tagIsOpen ) {
4957*89c4ff92SAndroid Build Coastguard Worker             m_os << ">" << std::endl;
4958*89c4ff92SAndroid Build Coastguard Worker             m_tagIsOpen = false;
4959*89c4ff92SAndroid Build Coastguard Worker         }
4960*89c4ff92SAndroid Build Coastguard Worker     }
4961*89c4ff92SAndroid Build Coastguard Worker 
4962*89c4ff92SAndroid Build Coastguard Worker     void XmlWriter::writeDeclaration() {
4963*89c4ff92SAndroid Build Coastguard Worker         m_os << "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n";
4964*89c4ff92SAndroid Build Coastguard Worker     }
4965*89c4ff92SAndroid Build Coastguard Worker 
4966*89c4ff92SAndroid Build Coastguard Worker     void XmlWriter::newlineIfNecessary() {
4967*89c4ff92SAndroid Build Coastguard Worker         if( m_needsNewline ) {
4968*89c4ff92SAndroid Build Coastguard Worker             m_os << std::endl;
4969*89c4ff92SAndroid Build Coastguard Worker             m_needsNewline = false;
4970*89c4ff92SAndroid Build Coastguard Worker         }
4971*89c4ff92SAndroid Build Coastguard Worker     }
4972*89c4ff92SAndroid Build Coastguard Worker 
4973*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
4974*89c4ff92SAndroid Build Coastguard Worker // End of copy-pasted code from Catch
4975*89c4ff92SAndroid Build Coastguard Worker // =================================================================================================
4976*89c4ff92SAndroid Build Coastguard Worker 
4977*89c4ff92SAndroid Build Coastguard Worker     // clang-format on
4978*89c4ff92SAndroid Build Coastguard Worker 
4979*89c4ff92SAndroid Build Coastguard Worker     struct XmlReporter : public IReporter
4980*89c4ff92SAndroid Build Coastguard Worker     {
4981*89c4ff92SAndroid Build Coastguard Worker         XmlWriter  xml;
4982*89c4ff92SAndroid Build Coastguard Worker         std::mutex mutex;
4983*89c4ff92SAndroid Build Coastguard Worker 
4984*89c4ff92SAndroid Build Coastguard Worker         // caching pointers/references to objects of these types - safe to do
4985*89c4ff92SAndroid Build Coastguard Worker         const ContextOptions& opt;
4986*89c4ff92SAndroid Build Coastguard Worker         const TestCaseData*   tc = nullptr;
4987*89c4ff92SAndroid Build Coastguard Worker 
4988*89c4ff92SAndroid Build Coastguard Worker         XmlReporter(const ContextOptions& co)
4989*89c4ff92SAndroid Build Coastguard Worker                 : xml(*co.cout)
4990*89c4ff92SAndroid Build Coastguard Worker                 , opt(co) {}
4991*89c4ff92SAndroid Build Coastguard Worker 
4992*89c4ff92SAndroid Build Coastguard Worker         void log_contexts() {
4993*89c4ff92SAndroid Build Coastguard Worker             int num_contexts = get_num_active_contexts();
4994*89c4ff92SAndroid Build Coastguard Worker             if(num_contexts) {
4995*89c4ff92SAndroid Build Coastguard Worker                 auto              contexts = get_active_contexts();
4996*89c4ff92SAndroid Build Coastguard Worker                 std::stringstream ss;
4997*89c4ff92SAndroid Build Coastguard Worker                 for(int i = 0; i < num_contexts; ++i) {
4998*89c4ff92SAndroid Build Coastguard Worker                     contexts[i]->stringify(&ss);
4999*89c4ff92SAndroid Build Coastguard Worker                     xml.scopedElement("Info").writeText(ss.str());
5000*89c4ff92SAndroid Build Coastguard Worker                     ss.str("");
5001*89c4ff92SAndroid Build Coastguard Worker                 }
5002*89c4ff92SAndroid Build Coastguard Worker             }
5003*89c4ff92SAndroid Build Coastguard Worker         }
5004*89c4ff92SAndroid Build Coastguard Worker 
5005*89c4ff92SAndroid Build Coastguard Worker         unsigned line(unsigned l) const { return opt.no_line_numbers ? 0 : l; }
5006*89c4ff92SAndroid Build Coastguard Worker 
5007*89c4ff92SAndroid Build Coastguard Worker         void test_case_start_impl(const TestCaseData& in) {
5008*89c4ff92SAndroid Build Coastguard Worker             bool open_ts_tag = false;
5009*89c4ff92SAndroid Build Coastguard Worker             if(tc != nullptr) { // we have already opened a test suite
5010*89c4ff92SAndroid Build Coastguard Worker                 if(std::strcmp(tc->m_test_suite, in.m_test_suite) != 0) {
5011*89c4ff92SAndroid Build Coastguard Worker                     xml.endElement();
5012*89c4ff92SAndroid Build Coastguard Worker                     open_ts_tag = true;
5013*89c4ff92SAndroid Build Coastguard Worker                 }
5014*89c4ff92SAndroid Build Coastguard Worker             }
5015*89c4ff92SAndroid Build Coastguard Worker             else {
5016*89c4ff92SAndroid Build Coastguard Worker                 open_ts_tag = true; // first test case ==> first test suite
5017*89c4ff92SAndroid Build Coastguard Worker             }
5018*89c4ff92SAndroid Build Coastguard Worker 
5019*89c4ff92SAndroid Build Coastguard Worker             if(open_ts_tag) {
5020*89c4ff92SAndroid Build Coastguard Worker                 xml.startElement("TestSuite");
5021*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("name", in.m_test_suite);
5022*89c4ff92SAndroid Build Coastguard Worker             }
5023*89c4ff92SAndroid Build Coastguard Worker 
5024*89c4ff92SAndroid Build Coastguard Worker             tc = &in;
5025*89c4ff92SAndroid Build Coastguard Worker             xml.startElement("TestCase")
5026*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("name", in.m_name)
5027*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("filename", skipPathFromFilename(in.m_file.c_str()))
5028*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("line", line(in.m_line))
5029*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("description", in.m_description);
5030*89c4ff92SAndroid Build Coastguard Worker 
5031*89c4ff92SAndroid Build Coastguard Worker             if(Approx(in.m_timeout) != 0)
5032*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("timeout", in.m_timeout);
5033*89c4ff92SAndroid Build Coastguard Worker             if(in.m_may_fail)
5034*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("may_fail", true);
5035*89c4ff92SAndroid Build Coastguard Worker             if(in.m_should_fail)
5036*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("should_fail", true);
5037*89c4ff92SAndroid Build Coastguard Worker         }
5038*89c4ff92SAndroid Build Coastguard Worker 
5039*89c4ff92SAndroid Build Coastguard Worker         // =========================================================================================
5040*89c4ff92SAndroid Build Coastguard Worker         // WHAT FOLLOWS ARE OVERRIDES OF THE VIRTUAL METHODS OF THE REPORTER INTERFACE
5041*89c4ff92SAndroid Build Coastguard Worker         // =========================================================================================
5042*89c4ff92SAndroid Build Coastguard Worker 
5043*89c4ff92SAndroid Build Coastguard Worker         void report_query(const QueryData& in) override {
5044*89c4ff92SAndroid Build Coastguard Worker             test_run_start();
5045*89c4ff92SAndroid Build Coastguard Worker             if(opt.list_reporters) {
5046*89c4ff92SAndroid Build Coastguard Worker                 for(auto& curr : getListeners())
5047*89c4ff92SAndroid Build Coastguard Worker                     xml.scopedElement("Listener")
5048*89c4ff92SAndroid Build Coastguard Worker                             .writeAttribute("priority", curr.first.first)
5049*89c4ff92SAndroid Build Coastguard Worker                             .writeAttribute("name", curr.first.second);
5050*89c4ff92SAndroid Build Coastguard Worker                 for(auto& curr : getReporters())
5051*89c4ff92SAndroid Build Coastguard Worker                     xml.scopedElement("Reporter")
5052*89c4ff92SAndroid Build Coastguard Worker                             .writeAttribute("priority", curr.first.first)
5053*89c4ff92SAndroid Build Coastguard Worker                             .writeAttribute("name", curr.first.second);
5054*89c4ff92SAndroid Build Coastguard Worker             } else if(opt.count || opt.list_test_cases) {
5055*89c4ff92SAndroid Build Coastguard Worker                 for(unsigned i = 0; i < in.num_data; ++i) {
5056*89c4ff92SAndroid Build Coastguard Worker                     xml.scopedElement("TestCase").writeAttribute("name", in.data[i]->m_name)
5057*89c4ff92SAndroid Build Coastguard Worker                         .writeAttribute("testsuite", in.data[i]->m_test_suite)
5058*89c4ff92SAndroid Build Coastguard Worker                         .writeAttribute("filename", skipPathFromFilename(in.data[i]->m_file.c_str()))
5059*89c4ff92SAndroid Build Coastguard Worker                         .writeAttribute("line", line(in.data[i]->m_line));
5060*89c4ff92SAndroid Build Coastguard Worker                 }
5061*89c4ff92SAndroid Build Coastguard Worker                 xml.scopedElement("OverallResultsTestCases")
5062*89c4ff92SAndroid Build Coastguard Worker                         .writeAttribute("unskipped", in.run_stats->numTestCasesPassingFilters);
5063*89c4ff92SAndroid Build Coastguard Worker             } else if(opt.list_test_suites) {
5064*89c4ff92SAndroid Build Coastguard Worker                 for(unsigned i = 0; i < in.num_data; ++i)
5065*89c4ff92SAndroid Build Coastguard Worker                     xml.scopedElement("TestSuite").writeAttribute("name", in.data[i]->m_test_suite);
5066*89c4ff92SAndroid Build Coastguard Worker                 xml.scopedElement("OverallResultsTestCases")
5067*89c4ff92SAndroid Build Coastguard Worker                         .writeAttribute("unskipped", in.run_stats->numTestCasesPassingFilters);
5068*89c4ff92SAndroid Build Coastguard Worker                 xml.scopedElement("OverallResultsTestSuites")
5069*89c4ff92SAndroid Build Coastguard Worker                         .writeAttribute("unskipped", in.run_stats->numTestSuitesPassingFilters);
5070*89c4ff92SAndroid Build Coastguard Worker             }
5071*89c4ff92SAndroid Build Coastguard Worker             xml.endElement();
5072*89c4ff92SAndroid Build Coastguard Worker         }
5073*89c4ff92SAndroid Build Coastguard Worker 
5074*89c4ff92SAndroid Build Coastguard Worker         void test_run_start() override {
5075*89c4ff92SAndroid Build Coastguard Worker             // remove .exe extension - mainly to have the same output on UNIX and Windows
5076*89c4ff92SAndroid Build Coastguard Worker             std::string binary_name = skipPathFromFilename(opt.binary_name.c_str());
5077*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_WINDOWS
5078*89c4ff92SAndroid Build Coastguard Worker             if(binary_name.rfind(".exe") != std::string::npos)
5079*89c4ff92SAndroid Build Coastguard Worker                 binary_name = binary_name.substr(0, binary_name.length() - 4);
5080*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM_WINDOWS
5081*89c4ff92SAndroid Build Coastguard Worker 
5082*89c4ff92SAndroid Build Coastguard Worker             xml.startElement("doctest").writeAttribute("binary", binary_name);
5083*89c4ff92SAndroid Build Coastguard Worker             if(opt.no_version == false)
5084*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("version", DOCTEST_VERSION_STR);
5085*89c4ff92SAndroid Build Coastguard Worker 
5086*89c4ff92SAndroid Build Coastguard Worker             // only the consequential ones (TODO: filters)
5087*89c4ff92SAndroid Build Coastguard Worker             xml.scopedElement("Options")
5088*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("order_by", opt.order_by.c_str())
5089*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("rand_seed", opt.rand_seed)
5090*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("first", opt.first)
5091*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("last", opt.last)
5092*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("abort_after", opt.abort_after)
5093*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("subcase_filter_levels", opt.subcase_filter_levels)
5094*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("case_sensitive", opt.case_sensitive)
5095*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("no_throw", opt.no_throw)
5096*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("no_skip", opt.no_skip);
5097*89c4ff92SAndroid Build Coastguard Worker         }
5098*89c4ff92SAndroid Build Coastguard Worker 
5099*89c4ff92SAndroid Build Coastguard Worker         void test_run_end(const TestRunStats& p) override {
5100*89c4ff92SAndroid Build Coastguard Worker             if(tc) // the TestSuite tag - only if there has been at least 1 test case
5101*89c4ff92SAndroid Build Coastguard Worker                 xml.endElement();
5102*89c4ff92SAndroid Build Coastguard Worker 
5103*89c4ff92SAndroid Build Coastguard Worker             xml.scopedElement("OverallResultsAsserts")
5104*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("successes", p.numAsserts - p.numAssertsFailed)
5105*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("failures", p.numAssertsFailed);
5106*89c4ff92SAndroid Build Coastguard Worker 
5107*89c4ff92SAndroid Build Coastguard Worker             xml.startElement("OverallResultsTestCases")
5108*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("successes",
5109*89c4ff92SAndroid Build Coastguard Worker                                     p.numTestCasesPassingFilters - p.numTestCasesFailed)
5110*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("failures", p.numTestCasesFailed);
5111*89c4ff92SAndroid Build Coastguard Worker             if(opt.no_skipped_summary == false)
5112*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("skipped", p.numTestCases - p.numTestCasesPassingFilters);
5113*89c4ff92SAndroid Build Coastguard Worker             xml.endElement();
5114*89c4ff92SAndroid Build Coastguard Worker 
5115*89c4ff92SAndroid Build Coastguard Worker             xml.endElement();
5116*89c4ff92SAndroid Build Coastguard Worker         }
5117*89c4ff92SAndroid Build Coastguard Worker 
5118*89c4ff92SAndroid Build Coastguard Worker         void test_case_start(const TestCaseData& in) override {
5119*89c4ff92SAndroid Build Coastguard Worker             test_case_start_impl(in);
5120*89c4ff92SAndroid Build Coastguard Worker             xml.ensureTagClosed();
5121*89c4ff92SAndroid Build Coastguard Worker         }
5122*89c4ff92SAndroid Build Coastguard Worker 
5123*89c4ff92SAndroid Build Coastguard Worker         void test_case_reenter(const TestCaseData&) override {}
5124*89c4ff92SAndroid Build Coastguard Worker 
5125*89c4ff92SAndroid Build Coastguard Worker         void test_case_end(const CurrentTestCaseStats& st) override {
5126*89c4ff92SAndroid Build Coastguard Worker             xml.startElement("OverallResultsAsserts")
5127*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("successes",
5128*89c4ff92SAndroid Build Coastguard Worker                                     st.numAssertsCurrentTest - st.numAssertsFailedCurrentTest)
5129*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("failures", st.numAssertsFailedCurrentTest);
5130*89c4ff92SAndroid Build Coastguard Worker             if(opt.duration)
5131*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("duration", st.seconds);
5132*89c4ff92SAndroid Build Coastguard Worker             if(tc->m_expected_failures)
5133*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("expected_failures", tc->m_expected_failures);
5134*89c4ff92SAndroid Build Coastguard Worker             xml.endElement();
5135*89c4ff92SAndroid Build Coastguard Worker 
5136*89c4ff92SAndroid Build Coastguard Worker             xml.endElement();
5137*89c4ff92SAndroid Build Coastguard Worker         }
5138*89c4ff92SAndroid Build Coastguard Worker 
5139*89c4ff92SAndroid Build Coastguard Worker         void test_case_exception(const TestCaseException& e) override {
5140*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5141*89c4ff92SAndroid Build Coastguard Worker 
5142*89c4ff92SAndroid Build Coastguard Worker             xml.scopedElement("Exception")
5143*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("crash", e.is_crash)
5144*89c4ff92SAndroid Build Coastguard Worker                     .writeText(e.error_string.c_str());
5145*89c4ff92SAndroid Build Coastguard Worker         }
5146*89c4ff92SAndroid Build Coastguard Worker 
5147*89c4ff92SAndroid Build Coastguard Worker         void subcase_start(const SubcaseSignature& in) override {
5148*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5149*89c4ff92SAndroid Build Coastguard Worker 
5150*89c4ff92SAndroid Build Coastguard Worker             xml.startElement("SubCase")
5151*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("name", in.m_name)
5152*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("filename", skipPathFromFilename(in.m_file))
5153*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("line", line(in.m_line));
5154*89c4ff92SAndroid Build Coastguard Worker             xml.ensureTagClosed();
5155*89c4ff92SAndroid Build Coastguard Worker         }
5156*89c4ff92SAndroid Build Coastguard Worker 
5157*89c4ff92SAndroid Build Coastguard Worker         void subcase_end() override { xml.endElement(); }
5158*89c4ff92SAndroid Build Coastguard Worker 
5159*89c4ff92SAndroid Build Coastguard Worker         void log_assert(const AssertData& rb) override {
5160*89c4ff92SAndroid Build Coastguard Worker             if(!rb.m_failed && !opt.success)
5161*89c4ff92SAndroid Build Coastguard Worker                 return;
5162*89c4ff92SAndroid Build Coastguard Worker 
5163*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5164*89c4ff92SAndroid Build Coastguard Worker 
5165*89c4ff92SAndroid Build Coastguard Worker             xml.startElement("Expression")
5166*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("success", !rb.m_failed)
5167*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("type", assertString(rb.m_at))
5168*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("filename", skipPathFromFilename(rb.m_file))
5169*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("line", line(rb.m_line));
5170*89c4ff92SAndroid Build Coastguard Worker 
5171*89c4ff92SAndroid Build Coastguard Worker             xml.scopedElement("Original").writeText(rb.m_expr);
5172*89c4ff92SAndroid Build Coastguard Worker 
5173*89c4ff92SAndroid Build Coastguard Worker             if(rb.m_threw)
5174*89c4ff92SAndroid Build Coastguard Worker                 xml.scopedElement("Exception").writeText(rb.m_exception.c_str());
5175*89c4ff92SAndroid Build Coastguard Worker 
5176*89c4ff92SAndroid Build Coastguard Worker             if(rb.m_at & assertType::is_throws_as)
5177*89c4ff92SAndroid Build Coastguard Worker                 xml.scopedElement("ExpectedException").writeText(rb.m_exception_type);
5178*89c4ff92SAndroid Build Coastguard Worker             if(rb.m_at & assertType::is_throws_with)
5179*89c4ff92SAndroid Build Coastguard Worker                 xml.scopedElement("ExpectedExceptionString").writeText(rb.m_exception_string);
5180*89c4ff92SAndroid Build Coastguard Worker             if((rb.m_at & assertType::is_normal) && !rb.m_threw)
5181*89c4ff92SAndroid Build Coastguard Worker                 xml.scopedElement("Expanded").writeText(rb.m_decomp.c_str());
5182*89c4ff92SAndroid Build Coastguard Worker 
5183*89c4ff92SAndroid Build Coastguard Worker             log_contexts();
5184*89c4ff92SAndroid Build Coastguard Worker 
5185*89c4ff92SAndroid Build Coastguard Worker             xml.endElement();
5186*89c4ff92SAndroid Build Coastguard Worker         }
5187*89c4ff92SAndroid Build Coastguard Worker 
5188*89c4ff92SAndroid Build Coastguard Worker         void log_message(const MessageData& mb) override {
5189*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5190*89c4ff92SAndroid Build Coastguard Worker 
5191*89c4ff92SAndroid Build Coastguard Worker             xml.startElement("Message")
5192*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("type", failureString(mb.m_severity))
5193*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("filename", skipPathFromFilename(mb.m_file))
5194*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("line", line(mb.m_line));
5195*89c4ff92SAndroid Build Coastguard Worker 
5196*89c4ff92SAndroid Build Coastguard Worker             xml.scopedElement("Text").writeText(mb.m_string.c_str());
5197*89c4ff92SAndroid Build Coastguard Worker 
5198*89c4ff92SAndroid Build Coastguard Worker             log_contexts();
5199*89c4ff92SAndroid Build Coastguard Worker 
5200*89c4ff92SAndroid Build Coastguard Worker             xml.endElement();
5201*89c4ff92SAndroid Build Coastguard Worker         }
5202*89c4ff92SAndroid Build Coastguard Worker 
5203*89c4ff92SAndroid Build Coastguard Worker         void test_case_skipped(const TestCaseData& in) override {
5204*89c4ff92SAndroid Build Coastguard Worker             if(opt.no_skipped_summary == false) {
5205*89c4ff92SAndroid Build Coastguard Worker                 test_case_start_impl(in);
5206*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("skipped", "true");
5207*89c4ff92SAndroid Build Coastguard Worker                 xml.endElement();
5208*89c4ff92SAndroid Build Coastguard Worker             }
5209*89c4ff92SAndroid Build Coastguard Worker         }
5210*89c4ff92SAndroid Build Coastguard Worker     };
5211*89c4ff92SAndroid Build Coastguard Worker 
5212*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_REGISTER_REPORTER("xml", 0, XmlReporter);
5213*89c4ff92SAndroid Build Coastguard Worker 
5214*89c4ff92SAndroid Build Coastguard Worker     void fulltext_log_assert_to_stream(std::ostream& s, const AssertData& rb) {
5215*89c4ff92SAndroid Build Coastguard Worker         if((rb.m_at & (assertType::is_throws_as | assertType::is_throws_with)) ==
5216*89c4ff92SAndroid Build Coastguard Worker             0) //!OCLINT bitwise operator in conditional
5217*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << assertString(rb.m_at) << "( " << rb.m_expr << " ) "
5218*89c4ff92SAndroid Build Coastguard Worker                 << Color::None;
5219*89c4ff92SAndroid Build Coastguard Worker 
5220*89c4ff92SAndroid Build Coastguard Worker         if(rb.m_at & assertType::is_throws) { //!OCLINT bitwise operator in conditional
5221*89c4ff92SAndroid Build Coastguard Worker             s << (rb.m_threw ? "threw as expected!" : "did NOT throw at all!") << "\n";
5222*89c4ff92SAndroid Build Coastguard Worker         } else if((rb.m_at & assertType::is_throws_as) &&
5223*89c4ff92SAndroid Build Coastguard Worker                     (rb.m_at & assertType::is_throws_with)) { //!OCLINT
5224*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << assertString(rb.m_at) << "( " << rb.m_expr << ", \""
5225*89c4ff92SAndroid Build Coastguard Worker                 << rb.m_exception_string << "\", " << rb.m_exception_type << " ) " << Color::None;
5226*89c4ff92SAndroid Build Coastguard Worker             if(rb.m_threw) {
5227*89c4ff92SAndroid Build Coastguard Worker                 if(!rb.m_failed) {
5228*89c4ff92SAndroid Build Coastguard Worker                     s << "threw as expected!\n";
5229*89c4ff92SAndroid Build Coastguard Worker                 } else {
5230*89c4ff92SAndroid Build Coastguard Worker                     s << "threw a DIFFERENT exception! (contents: " << rb.m_exception << ")\n";
5231*89c4ff92SAndroid Build Coastguard Worker                 }
5232*89c4ff92SAndroid Build Coastguard Worker             } else {
5233*89c4ff92SAndroid Build Coastguard Worker                 s << "did NOT throw at all!\n";
5234*89c4ff92SAndroid Build Coastguard Worker             }
5235*89c4ff92SAndroid Build Coastguard Worker         } else if(rb.m_at &
5236*89c4ff92SAndroid Build Coastguard Worker                     assertType::is_throws_as) { //!OCLINT bitwise operator in conditional
5237*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << assertString(rb.m_at) << "( " << rb.m_expr << ", "
5238*89c4ff92SAndroid Build Coastguard Worker                 << rb.m_exception_type << " ) " << Color::None
5239*89c4ff92SAndroid Build Coastguard Worker                 << (rb.m_threw ? (rb.m_threw_as ? "threw as expected!" :
5240*89c4ff92SAndroid Build Coastguard Worker                                                 "threw a DIFFERENT exception: ") :
5241*89c4ff92SAndroid Build Coastguard Worker                                 "did NOT throw at all!")
5242*89c4ff92SAndroid Build Coastguard Worker                 << Color::Cyan << rb.m_exception << "\n";
5243*89c4ff92SAndroid Build Coastguard Worker         } else if(rb.m_at &
5244*89c4ff92SAndroid Build Coastguard Worker                     assertType::is_throws_with) { //!OCLINT bitwise operator in conditional
5245*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << assertString(rb.m_at) << "( " << rb.m_expr << ", \""
5246*89c4ff92SAndroid Build Coastguard Worker                 << rb.m_exception_string << "\" ) " << Color::None
5247*89c4ff92SAndroid Build Coastguard Worker                 << (rb.m_threw ? (!rb.m_failed ? "threw as expected!" :
5248*89c4ff92SAndroid Build Coastguard Worker                                                 "threw a DIFFERENT exception: ") :
5249*89c4ff92SAndroid Build Coastguard Worker                                 "did NOT throw at all!")
5250*89c4ff92SAndroid Build Coastguard Worker                 << Color::Cyan << rb.m_exception << "\n";
5251*89c4ff92SAndroid Build Coastguard Worker         } else if(rb.m_at & assertType::is_nothrow) { //!OCLINT bitwise operator in conditional
5252*89c4ff92SAndroid Build Coastguard Worker             s << (rb.m_threw ? "THREW exception: " : "didn't throw!") << Color::Cyan
5253*89c4ff92SAndroid Build Coastguard Worker                 << rb.m_exception << "\n";
5254*89c4ff92SAndroid Build Coastguard Worker         } else {
5255*89c4ff92SAndroid Build Coastguard Worker             s << (rb.m_threw ? "THREW exception: " :
5256*89c4ff92SAndroid Build Coastguard Worker                                 (!rb.m_failed ? "is correct!\n" : "is NOT correct!\n"));
5257*89c4ff92SAndroid Build Coastguard Worker             if(rb.m_threw)
5258*89c4ff92SAndroid Build Coastguard Worker                 s << rb.m_exception << "\n";
5259*89c4ff92SAndroid Build Coastguard Worker             else
5260*89c4ff92SAndroid Build Coastguard Worker                 s << "  values: " << assertString(rb.m_at) << "( " << rb.m_decomp << " )\n";
5261*89c4ff92SAndroid Build Coastguard Worker         }
5262*89c4ff92SAndroid Build Coastguard Worker     }
5263*89c4ff92SAndroid Build Coastguard Worker 
5264*89c4ff92SAndroid Build Coastguard Worker     // TODO:
5265*89c4ff92SAndroid Build Coastguard Worker     // - log_message()
5266*89c4ff92SAndroid Build Coastguard Worker     // - respond to queries
5267*89c4ff92SAndroid Build Coastguard Worker     // - honor remaining options
5268*89c4ff92SAndroid Build Coastguard Worker     // - more attributes in tags
5269*89c4ff92SAndroid Build Coastguard Worker     struct JUnitReporter : public IReporter
5270*89c4ff92SAndroid Build Coastguard Worker     {
5271*89c4ff92SAndroid Build Coastguard Worker         XmlWriter  xml;
5272*89c4ff92SAndroid Build Coastguard Worker         std::mutex mutex;
5273*89c4ff92SAndroid Build Coastguard Worker         Timer timer;
5274*89c4ff92SAndroid Build Coastguard Worker         std::vector<String> deepestSubcaseStackNames;
5275*89c4ff92SAndroid Build Coastguard Worker 
5276*89c4ff92SAndroid Build Coastguard Worker         struct JUnitTestCaseData
5277*89c4ff92SAndroid Build Coastguard Worker         {
5278*89c4ff92SAndroid Build Coastguard Worker             static std::string getCurrentTimestamp() {
5279*89c4ff92SAndroid Build Coastguard Worker                 // Beware, this is not reentrant because of backward compatibility issues
5280*89c4ff92SAndroid Build Coastguard Worker                 // Also, UTC only, again because of backward compatibility (%z is C++11)
5281*89c4ff92SAndroid Build Coastguard Worker                 time_t rawtime;
5282*89c4ff92SAndroid Build Coastguard Worker                 std::time(&rawtime);
5283*89c4ff92SAndroid Build Coastguard Worker                 auto const timeStampSize = sizeof("2017-01-16T17:06:45Z");
5284*89c4ff92SAndroid Build Coastguard Worker 
5285*89c4ff92SAndroid Build Coastguard Worker                 std::tm timeInfo;
5286*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_WINDOWS
5287*89c4ff92SAndroid Build Coastguard Worker                 gmtime_s(&timeInfo, &rawtime);
5288*89c4ff92SAndroid Build Coastguard Worker #else // DOCTEST_PLATFORM_WINDOWS
5289*89c4ff92SAndroid Build Coastguard Worker                 gmtime_r(&rawtime, &timeInfo);
5290*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM_WINDOWS
5291*89c4ff92SAndroid Build Coastguard Worker 
5292*89c4ff92SAndroid Build Coastguard Worker                 char timeStamp[timeStampSize];
5293*89c4ff92SAndroid Build Coastguard Worker                 const char* const fmt = "%Y-%m-%dT%H:%M:%SZ";
5294*89c4ff92SAndroid Build Coastguard Worker 
5295*89c4ff92SAndroid Build Coastguard Worker                 std::strftime(timeStamp, timeStampSize, fmt, &timeInfo);
5296*89c4ff92SAndroid Build Coastguard Worker                 return std::string(timeStamp);
5297*89c4ff92SAndroid Build Coastguard Worker             }
5298*89c4ff92SAndroid Build Coastguard Worker 
5299*89c4ff92SAndroid Build Coastguard Worker             struct JUnitTestMessage
5300*89c4ff92SAndroid Build Coastguard Worker             {
5301*89c4ff92SAndroid Build Coastguard Worker                 JUnitTestMessage(const std::string& _message, const std::string& _type, const std::string& _details)
5302*89c4ff92SAndroid Build Coastguard Worker                     : message(_message), type(_type), details(_details) {}
5303*89c4ff92SAndroid Build Coastguard Worker 
5304*89c4ff92SAndroid Build Coastguard Worker                 JUnitTestMessage(const std::string& _message, const std::string& _details)
5305*89c4ff92SAndroid Build Coastguard Worker                     : message(_message), type(), details(_details) {}
5306*89c4ff92SAndroid Build Coastguard Worker 
5307*89c4ff92SAndroid Build Coastguard Worker                 std::string message, type, details;
5308*89c4ff92SAndroid Build Coastguard Worker             };
5309*89c4ff92SAndroid Build Coastguard Worker 
5310*89c4ff92SAndroid Build Coastguard Worker             struct JUnitTestCase
5311*89c4ff92SAndroid Build Coastguard Worker             {
5312*89c4ff92SAndroid Build Coastguard Worker                 JUnitTestCase(const std::string& _classname, const std::string& _name)
5313*89c4ff92SAndroid Build Coastguard Worker                     : classname(_classname), name(_name), time(0), failures() {}
5314*89c4ff92SAndroid Build Coastguard Worker 
5315*89c4ff92SAndroid Build Coastguard Worker                 std::string classname, name;
5316*89c4ff92SAndroid Build Coastguard Worker                 double time;
5317*89c4ff92SAndroid Build Coastguard Worker                 std::vector<JUnitTestMessage> failures, errors;
5318*89c4ff92SAndroid Build Coastguard Worker             };
5319*89c4ff92SAndroid Build Coastguard Worker 
5320*89c4ff92SAndroid Build Coastguard Worker             void add(const std::string& classname, const std::string& name) {
5321*89c4ff92SAndroid Build Coastguard Worker                 testcases.emplace_back(classname, name);
5322*89c4ff92SAndroid Build Coastguard Worker             }
5323*89c4ff92SAndroid Build Coastguard Worker 
5324*89c4ff92SAndroid Build Coastguard Worker             void appendSubcaseNamesToLastTestcase(std::vector<String> nameStack) {
5325*89c4ff92SAndroid Build Coastguard Worker                 for(auto& curr: nameStack)
5326*89c4ff92SAndroid Build Coastguard Worker                     if(curr.size())
5327*89c4ff92SAndroid Build Coastguard Worker                         testcases.back().name += std::string("/") + curr.c_str();
5328*89c4ff92SAndroid Build Coastguard Worker             }
5329*89c4ff92SAndroid Build Coastguard Worker 
5330*89c4ff92SAndroid Build Coastguard Worker             void addTime(double time) {
5331*89c4ff92SAndroid Build Coastguard Worker                 if(time < 1e-4)
5332*89c4ff92SAndroid Build Coastguard Worker                     time = 0;
5333*89c4ff92SAndroid Build Coastguard Worker                 testcases.back().time = time;
5334*89c4ff92SAndroid Build Coastguard Worker                 totalSeconds += time;
5335*89c4ff92SAndroid Build Coastguard Worker             }
5336*89c4ff92SAndroid Build Coastguard Worker 
5337*89c4ff92SAndroid Build Coastguard Worker             void addFailure(const std::string& message, const std::string& type, const std::string& details) {
5338*89c4ff92SAndroid Build Coastguard Worker                 testcases.back().failures.emplace_back(message, type, details);
5339*89c4ff92SAndroid Build Coastguard Worker                 ++totalFailures;
5340*89c4ff92SAndroid Build Coastguard Worker             }
5341*89c4ff92SAndroid Build Coastguard Worker 
5342*89c4ff92SAndroid Build Coastguard Worker             void addError(const std::string& message, const std::string& details) {
5343*89c4ff92SAndroid Build Coastguard Worker                 testcases.back().errors.emplace_back(message, details);
5344*89c4ff92SAndroid Build Coastguard Worker                 ++totalErrors;
5345*89c4ff92SAndroid Build Coastguard Worker             }
5346*89c4ff92SAndroid Build Coastguard Worker 
5347*89c4ff92SAndroid Build Coastguard Worker             std::vector<JUnitTestCase> testcases;
5348*89c4ff92SAndroid Build Coastguard Worker             double totalSeconds = 0;
5349*89c4ff92SAndroid Build Coastguard Worker             int totalErrors = 0, totalFailures = 0;
5350*89c4ff92SAndroid Build Coastguard Worker         };
5351*89c4ff92SAndroid Build Coastguard Worker 
5352*89c4ff92SAndroid Build Coastguard Worker         JUnitTestCaseData testCaseData;
5353*89c4ff92SAndroid Build Coastguard Worker 
5354*89c4ff92SAndroid Build Coastguard Worker         // caching pointers/references to objects of these types - safe to do
5355*89c4ff92SAndroid Build Coastguard Worker         const ContextOptions& opt;
5356*89c4ff92SAndroid Build Coastguard Worker         const TestCaseData*   tc = nullptr;
5357*89c4ff92SAndroid Build Coastguard Worker 
5358*89c4ff92SAndroid Build Coastguard Worker         JUnitReporter(const ContextOptions& co)
5359*89c4ff92SAndroid Build Coastguard Worker                 : xml(*co.cout)
5360*89c4ff92SAndroid Build Coastguard Worker                 , opt(co) {}
5361*89c4ff92SAndroid Build Coastguard Worker 
5362*89c4ff92SAndroid Build Coastguard Worker         unsigned line(unsigned l) const { return opt.no_line_numbers ? 0 : l; }
5363*89c4ff92SAndroid Build Coastguard Worker 
5364*89c4ff92SAndroid Build Coastguard Worker         // =========================================================================================
5365*89c4ff92SAndroid Build Coastguard Worker         // WHAT FOLLOWS ARE OVERRIDES OF THE VIRTUAL METHODS OF THE REPORTER INTERFACE
5366*89c4ff92SAndroid Build Coastguard Worker         // =========================================================================================
5367*89c4ff92SAndroid Build Coastguard Worker 
5368*89c4ff92SAndroid Build Coastguard Worker         void report_query(const QueryData&) override {}
5369*89c4ff92SAndroid Build Coastguard Worker 
5370*89c4ff92SAndroid Build Coastguard Worker         void test_run_start() override {}
5371*89c4ff92SAndroid Build Coastguard Worker 
5372*89c4ff92SAndroid Build Coastguard Worker         void test_run_end(const TestRunStats& p) override {
5373*89c4ff92SAndroid Build Coastguard Worker             // remove .exe extension - mainly to have the same output on UNIX and Windows
5374*89c4ff92SAndroid Build Coastguard Worker             std::string binary_name = skipPathFromFilename(opt.binary_name.c_str());
5375*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_WINDOWS
5376*89c4ff92SAndroid Build Coastguard Worker             if(binary_name.rfind(".exe") != std::string::npos)
5377*89c4ff92SAndroid Build Coastguard Worker                 binary_name = binary_name.substr(0, binary_name.length() - 4);
5378*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM_WINDOWS
5379*89c4ff92SAndroid Build Coastguard Worker             xml.startElement("testsuites");
5380*89c4ff92SAndroid Build Coastguard Worker             xml.startElement("testsuite").writeAttribute("name", binary_name)
5381*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("errors", testCaseData.totalErrors)
5382*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("failures", testCaseData.totalFailures)
5383*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("tests", p.numAsserts);
5384*89c4ff92SAndroid Build Coastguard Worker             if(opt.no_time_in_output == false) {
5385*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("time", testCaseData.totalSeconds);
5386*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("timestamp", JUnitTestCaseData::getCurrentTimestamp());
5387*89c4ff92SAndroid Build Coastguard Worker             }
5388*89c4ff92SAndroid Build Coastguard Worker             if(opt.no_version == false)
5389*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("doctest_version", DOCTEST_VERSION_STR);
5390*89c4ff92SAndroid Build Coastguard Worker 
5391*89c4ff92SAndroid Build Coastguard Worker             for(const auto& testCase : testCaseData.testcases) {
5392*89c4ff92SAndroid Build Coastguard Worker                 xml.startElement("testcase")
5393*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("classname", testCase.classname)
5394*89c4ff92SAndroid Build Coastguard Worker                     .writeAttribute("name", testCase.name);
5395*89c4ff92SAndroid Build Coastguard Worker                 if(opt.no_time_in_output == false)
5396*89c4ff92SAndroid Build Coastguard Worker                     xml.writeAttribute("time", testCase.time);
5397*89c4ff92SAndroid Build Coastguard Worker                 // This is not ideal, but it should be enough to mimic gtest's junit output.
5398*89c4ff92SAndroid Build Coastguard Worker                 xml.writeAttribute("status", "run");
5399*89c4ff92SAndroid Build Coastguard Worker 
5400*89c4ff92SAndroid Build Coastguard Worker                 for(const auto& failure : testCase.failures) {
5401*89c4ff92SAndroid Build Coastguard Worker                     xml.scopedElement("failure")
5402*89c4ff92SAndroid Build Coastguard Worker                         .writeAttribute("message", failure.message)
5403*89c4ff92SAndroid Build Coastguard Worker                         .writeAttribute("type", failure.type)
5404*89c4ff92SAndroid Build Coastguard Worker                         .writeText(failure.details, false);
5405*89c4ff92SAndroid Build Coastguard Worker                 }
5406*89c4ff92SAndroid Build Coastguard Worker 
5407*89c4ff92SAndroid Build Coastguard Worker                 for(const auto& error : testCase.errors) {
5408*89c4ff92SAndroid Build Coastguard Worker                     xml.scopedElement("error")
5409*89c4ff92SAndroid Build Coastguard Worker                         .writeAttribute("message", error.message)
5410*89c4ff92SAndroid Build Coastguard Worker                         .writeText(error.details);
5411*89c4ff92SAndroid Build Coastguard Worker                 }
5412*89c4ff92SAndroid Build Coastguard Worker 
5413*89c4ff92SAndroid Build Coastguard Worker                 xml.endElement();
5414*89c4ff92SAndroid Build Coastguard Worker             }
5415*89c4ff92SAndroid Build Coastguard Worker             xml.endElement();
5416*89c4ff92SAndroid Build Coastguard Worker             xml.endElement();
5417*89c4ff92SAndroid Build Coastguard Worker         }
5418*89c4ff92SAndroid Build Coastguard Worker 
5419*89c4ff92SAndroid Build Coastguard Worker         void test_case_start(const TestCaseData& in) override {
5420*89c4ff92SAndroid Build Coastguard Worker             testCaseData.add(skipPathFromFilename(in.m_file.c_str()), in.m_name);
5421*89c4ff92SAndroid Build Coastguard Worker             timer.start();
5422*89c4ff92SAndroid Build Coastguard Worker         }
5423*89c4ff92SAndroid Build Coastguard Worker 
5424*89c4ff92SAndroid Build Coastguard Worker         void test_case_reenter(const TestCaseData& in) override {
5425*89c4ff92SAndroid Build Coastguard Worker             testCaseData.addTime(timer.getElapsedSeconds());
5426*89c4ff92SAndroid Build Coastguard Worker             testCaseData.appendSubcaseNamesToLastTestcase(deepestSubcaseStackNames);
5427*89c4ff92SAndroid Build Coastguard Worker             deepestSubcaseStackNames.clear();
5428*89c4ff92SAndroid Build Coastguard Worker 
5429*89c4ff92SAndroid Build Coastguard Worker             timer.start();
5430*89c4ff92SAndroid Build Coastguard Worker             testCaseData.add(skipPathFromFilename(in.m_file.c_str()), in.m_name);
5431*89c4ff92SAndroid Build Coastguard Worker         }
5432*89c4ff92SAndroid Build Coastguard Worker 
5433*89c4ff92SAndroid Build Coastguard Worker         void test_case_end(const CurrentTestCaseStats&) override {
5434*89c4ff92SAndroid Build Coastguard Worker             testCaseData.addTime(timer.getElapsedSeconds());
5435*89c4ff92SAndroid Build Coastguard Worker             testCaseData.appendSubcaseNamesToLastTestcase(deepestSubcaseStackNames);
5436*89c4ff92SAndroid Build Coastguard Worker             deepestSubcaseStackNames.clear();
5437*89c4ff92SAndroid Build Coastguard Worker         }
5438*89c4ff92SAndroid Build Coastguard Worker 
5439*89c4ff92SAndroid Build Coastguard Worker         void test_case_exception(const TestCaseException& e) override {
5440*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5441*89c4ff92SAndroid Build Coastguard Worker             testCaseData.addError("exception", e.error_string.c_str());
5442*89c4ff92SAndroid Build Coastguard Worker         }
5443*89c4ff92SAndroid Build Coastguard Worker 
5444*89c4ff92SAndroid Build Coastguard Worker         void subcase_start(const SubcaseSignature& in) override {
5445*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5446*89c4ff92SAndroid Build Coastguard Worker             deepestSubcaseStackNames.push_back(in.m_name);
5447*89c4ff92SAndroid Build Coastguard Worker         }
5448*89c4ff92SAndroid Build Coastguard Worker 
5449*89c4ff92SAndroid Build Coastguard Worker         void subcase_end() override {}
5450*89c4ff92SAndroid Build Coastguard Worker 
5451*89c4ff92SAndroid Build Coastguard Worker         void log_assert(const AssertData& rb) override {
5452*89c4ff92SAndroid Build Coastguard Worker             if(!rb.m_failed) // report only failures & ignore the `success` option
5453*89c4ff92SAndroid Build Coastguard Worker                 return;
5454*89c4ff92SAndroid Build Coastguard Worker 
5455*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5456*89c4ff92SAndroid Build Coastguard Worker 
5457*89c4ff92SAndroid Build Coastguard Worker             std::ostringstream os;
5458*89c4ff92SAndroid Build Coastguard Worker             os << skipPathFromFilename(rb.m_file) << (opt.gnu_file_line ? ":" : "(")
5459*89c4ff92SAndroid Build Coastguard Worker               << line(rb.m_line) << (opt.gnu_file_line ? ":" : "):") << std::endl;
5460*89c4ff92SAndroid Build Coastguard Worker 
5461*89c4ff92SAndroid Build Coastguard Worker             fulltext_log_assert_to_stream(os, rb);
5462*89c4ff92SAndroid Build Coastguard Worker             log_contexts(os);
5463*89c4ff92SAndroid Build Coastguard Worker             testCaseData.addFailure(rb.m_decomp.c_str(), assertString(rb.m_at), os.str());
5464*89c4ff92SAndroid Build Coastguard Worker         }
5465*89c4ff92SAndroid Build Coastguard Worker 
5466*89c4ff92SAndroid Build Coastguard Worker         void log_message(const MessageData&) override {}
5467*89c4ff92SAndroid Build Coastguard Worker 
5468*89c4ff92SAndroid Build Coastguard Worker         void test_case_skipped(const TestCaseData&) override {}
5469*89c4ff92SAndroid Build Coastguard Worker 
5470*89c4ff92SAndroid Build Coastguard Worker         void log_contexts(std::ostringstream& s) {
5471*89c4ff92SAndroid Build Coastguard Worker             int num_contexts = get_num_active_contexts();
5472*89c4ff92SAndroid Build Coastguard Worker             if(num_contexts) {
5473*89c4ff92SAndroid Build Coastguard Worker                 auto contexts = get_active_contexts();
5474*89c4ff92SAndroid Build Coastguard Worker 
5475*89c4ff92SAndroid Build Coastguard Worker                 s << "  logged: ";
5476*89c4ff92SAndroid Build Coastguard Worker                 for(int i = 0; i < num_contexts; ++i) {
5477*89c4ff92SAndroid Build Coastguard Worker                     s << (i == 0 ? "" : "          ");
5478*89c4ff92SAndroid Build Coastguard Worker                     contexts[i]->stringify(&s);
5479*89c4ff92SAndroid Build Coastguard Worker                     s << std::endl;
5480*89c4ff92SAndroid Build Coastguard Worker                 }
5481*89c4ff92SAndroid Build Coastguard Worker             }
5482*89c4ff92SAndroid Build Coastguard Worker         }
5483*89c4ff92SAndroid Build Coastguard Worker     };
5484*89c4ff92SAndroid Build Coastguard Worker 
5485*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_REGISTER_REPORTER("junit", 0, JUnitReporter);
5486*89c4ff92SAndroid Build Coastguard Worker 
5487*89c4ff92SAndroid Build Coastguard Worker     struct Whitespace
5488*89c4ff92SAndroid Build Coastguard Worker     {
5489*89c4ff92SAndroid Build Coastguard Worker         int nrSpaces;
5490*89c4ff92SAndroid Build Coastguard Worker         explicit Whitespace(int nr)
5491*89c4ff92SAndroid Build Coastguard Worker                 : nrSpaces(nr) {}
5492*89c4ff92SAndroid Build Coastguard Worker     };
5493*89c4ff92SAndroid Build Coastguard Worker 
5494*89c4ff92SAndroid Build Coastguard Worker     std::ostream& operator<<(std::ostream& out, const Whitespace& ws) {
5495*89c4ff92SAndroid Build Coastguard Worker         if(ws.nrSpaces != 0)
5496*89c4ff92SAndroid Build Coastguard Worker             out << std::setw(ws.nrSpaces) << ' ';
5497*89c4ff92SAndroid Build Coastguard Worker         return out;
5498*89c4ff92SAndroid Build Coastguard Worker     }
5499*89c4ff92SAndroid Build Coastguard Worker 
5500*89c4ff92SAndroid Build Coastguard Worker     struct ConsoleReporter : public IReporter
5501*89c4ff92SAndroid Build Coastguard Worker     {
5502*89c4ff92SAndroid Build Coastguard Worker         std::ostream&                 s;
5503*89c4ff92SAndroid Build Coastguard Worker         bool                          hasLoggedCurrentTestStart;
5504*89c4ff92SAndroid Build Coastguard Worker         std::vector<SubcaseSignature> subcasesStack;
5505*89c4ff92SAndroid Build Coastguard Worker         size_t                        currentSubcaseLevel;
5506*89c4ff92SAndroid Build Coastguard Worker         std::mutex                    mutex;
5507*89c4ff92SAndroid Build Coastguard Worker 
5508*89c4ff92SAndroid Build Coastguard Worker         // caching pointers/references to objects of these types - safe to do
5509*89c4ff92SAndroid Build Coastguard Worker         const ContextOptions& opt;
5510*89c4ff92SAndroid Build Coastguard Worker         const TestCaseData*   tc;
5511*89c4ff92SAndroid Build Coastguard Worker 
5512*89c4ff92SAndroid Build Coastguard Worker         ConsoleReporter(const ContextOptions& co)
5513*89c4ff92SAndroid Build Coastguard Worker                 : s(*co.cout)
5514*89c4ff92SAndroid Build Coastguard Worker                 , opt(co) {}
5515*89c4ff92SAndroid Build Coastguard Worker 
5516*89c4ff92SAndroid Build Coastguard Worker         ConsoleReporter(const ContextOptions& co, std::ostream& ostr)
5517*89c4ff92SAndroid Build Coastguard Worker                 : s(ostr)
5518*89c4ff92SAndroid Build Coastguard Worker                 , opt(co) {}
5519*89c4ff92SAndroid Build Coastguard Worker 
5520*89c4ff92SAndroid Build Coastguard Worker         // =========================================================================================
5521*89c4ff92SAndroid Build Coastguard Worker         // WHAT FOLLOWS ARE HELPERS USED BY THE OVERRIDES OF THE VIRTUAL METHODS OF THE INTERFACE
5522*89c4ff92SAndroid Build Coastguard Worker         // =========================================================================================
5523*89c4ff92SAndroid Build Coastguard Worker 
5524*89c4ff92SAndroid Build Coastguard Worker         void separator_to_stream() {
5525*89c4ff92SAndroid Build Coastguard Worker             s << Color::Yellow
5526*89c4ff92SAndroid Build Coastguard Worker               << "==============================================================================="
5527*89c4ff92SAndroid Build Coastguard Worker                  "\n";
5528*89c4ff92SAndroid Build Coastguard Worker         }
5529*89c4ff92SAndroid Build Coastguard Worker 
5530*89c4ff92SAndroid Build Coastguard Worker         const char* getSuccessOrFailString(bool success, assertType::Enum at,
5531*89c4ff92SAndroid Build Coastguard Worker                                            const char* success_str) {
5532*89c4ff92SAndroid Build Coastguard Worker             if(success)
5533*89c4ff92SAndroid Build Coastguard Worker                 return success_str;
5534*89c4ff92SAndroid Build Coastguard Worker             return failureString(at);
5535*89c4ff92SAndroid Build Coastguard Worker         }
5536*89c4ff92SAndroid Build Coastguard Worker 
5537*89c4ff92SAndroid Build Coastguard Worker         Color::Enum getSuccessOrFailColor(bool success, assertType::Enum at) {
5538*89c4ff92SAndroid Build Coastguard Worker             return success ? Color::BrightGreen :
5539*89c4ff92SAndroid Build Coastguard Worker                              (at & assertType::is_warn) ? Color::Yellow : Color::Red;
5540*89c4ff92SAndroid Build Coastguard Worker         }
5541*89c4ff92SAndroid Build Coastguard Worker 
5542*89c4ff92SAndroid Build Coastguard Worker         void successOrFailColoredStringToStream(bool success, assertType::Enum at,
5543*89c4ff92SAndroid Build Coastguard Worker                                                 const char* success_str = "SUCCESS") {
5544*89c4ff92SAndroid Build Coastguard Worker             s << getSuccessOrFailColor(success, at)
5545*89c4ff92SAndroid Build Coastguard Worker               << getSuccessOrFailString(success, at, success_str) << ": ";
5546*89c4ff92SAndroid Build Coastguard Worker         }
5547*89c4ff92SAndroid Build Coastguard Worker 
5548*89c4ff92SAndroid Build Coastguard Worker         void log_contexts() {
5549*89c4ff92SAndroid Build Coastguard Worker             int num_contexts = get_num_active_contexts();
5550*89c4ff92SAndroid Build Coastguard Worker             if(num_contexts) {
5551*89c4ff92SAndroid Build Coastguard Worker                 auto contexts = get_active_contexts();
5552*89c4ff92SAndroid Build Coastguard Worker 
5553*89c4ff92SAndroid Build Coastguard Worker                 s << Color::None << "  logged: ";
5554*89c4ff92SAndroid Build Coastguard Worker                 for(int i = 0; i < num_contexts; ++i) {
5555*89c4ff92SAndroid Build Coastguard Worker                     s << (i == 0 ? "" : "          ");
5556*89c4ff92SAndroid Build Coastguard Worker                     contexts[i]->stringify(&s);
5557*89c4ff92SAndroid Build Coastguard Worker                     s << "\n";
5558*89c4ff92SAndroid Build Coastguard Worker                 }
5559*89c4ff92SAndroid Build Coastguard Worker             }
5560*89c4ff92SAndroid Build Coastguard Worker 
5561*89c4ff92SAndroid Build Coastguard Worker             s << "\n";
5562*89c4ff92SAndroid Build Coastguard Worker         }
5563*89c4ff92SAndroid Build Coastguard Worker 
5564*89c4ff92SAndroid Build Coastguard Worker         // this was requested to be made virtual so users could override it
5565*89c4ff92SAndroid Build Coastguard Worker         virtual void file_line_to_stream(const char* file, int line,
5566*89c4ff92SAndroid Build Coastguard Worker                                         const char* tail = "") {
5567*89c4ff92SAndroid Build Coastguard Worker             s << Color::LightGrey << skipPathFromFilename(file) << (opt.gnu_file_line ? ":" : "(")
5568*89c4ff92SAndroid Build Coastguard Worker             << (opt.no_line_numbers ? 0 : line) // 0 or the real num depending on the option
5569*89c4ff92SAndroid Build Coastguard Worker             << (opt.gnu_file_line ? ":" : "):") << tail;
5570*89c4ff92SAndroid Build Coastguard Worker         }
5571*89c4ff92SAndroid Build Coastguard Worker 
5572*89c4ff92SAndroid Build Coastguard Worker         void logTestStart() {
5573*89c4ff92SAndroid Build Coastguard Worker             if(hasLoggedCurrentTestStart)
5574*89c4ff92SAndroid Build Coastguard Worker                 return;
5575*89c4ff92SAndroid Build Coastguard Worker 
5576*89c4ff92SAndroid Build Coastguard Worker             separator_to_stream();
5577*89c4ff92SAndroid Build Coastguard Worker             file_line_to_stream(tc->m_file.c_str(), tc->m_line, "\n");
5578*89c4ff92SAndroid Build Coastguard Worker             if(tc->m_description)
5579*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Yellow << "DESCRIPTION: " << Color::None << tc->m_description << "\n";
5580*89c4ff92SAndroid Build Coastguard Worker             if(tc->m_test_suite && tc->m_test_suite[0] != '\0')
5581*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Yellow << "TEST SUITE: " << Color::None << tc->m_test_suite << "\n";
5582*89c4ff92SAndroid Build Coastguard Worker             if(strncmp(tc->m_name, "  Scenario:", 11) != 0)
5583*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Yellow << "TEST CASE:  ";
5584*89c4ff92SAndroid Build Coastguard Worker             s << Color::None << tc->m_name << "\n";
5585*89c4ff92SAndroid Build Coastguard Worker 
5586*89c4ff92SAndroid Build Coastguard Worker             for(size_t i = 0; i < currentSubcaseLevel; ++i) {
5587*89c4ff92SAndroid Build Coastguard Worker                 if(subcasesStack[i].m_name[0] != '\0')
5588*89c4ff92SAndroid Build Coastguard Worker                     s << "  " << subcasesStack[i].m_name << "\n";
5589*89c4ff92SAndroid Build Coastguard Worker             }
5590*89c4ff92SAndroid Build Coastguard Worker 
5591*89c4ff92SAndroid Build Coastguard Worker             if(currentSubcaseLevel != subcasesStack.size()) {
5592*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Yellow << "\nDEEPEST SUBCASE STACK REACHED (DIFFERENT FROM THE CURRENT ONE):\n" << Color::None;
5593*89c4ff92SAndroid Build Coastguard Worker                 for(size_t i = 0; i < subcasesStack.size(); ++i) {
5594*89c4ff92SAndroid Build Coastguard Worker                     if(subcasesStack[i].m_name[0] != '\0')
5595*89c4ff92SAndroid Build Coastguard Worker                         s << "  " << subcasesStack[i].m_name << "\n";
5596*89c4ff92SAndroid Build Coastguard Worker                 }
5597*89c4ff92SAndroid Build Coastguard Worker             }
5598*89c4ff92SAndroid Build Coastguard Worker 
5599*89c4ff92SAndroid Build Coastguard Worker             s << "\n";
5600*89c4ff92SAndroid Build Coastguard Worker 
5601*89c4ff92SAndroid Build Coastguard Worker             hasLoggedCurrentTestStart = true;
5602*89c4ff92SAndroid Build Coastguard Worker         }
5603*89c4ff92SAndroid Build Coastguard Worker 
5604*89c4ff92SAndroid Build Coastguard Worker         void printVersion() {
5605*89c4ff92SAndroid Build Coastguard Worker             if(opt.no_version == false)
5606*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Cyan << "[doctest] " << Color::None << "doctest version is \""
5607*89c4ff92SAndroid Build Coastguard Worker                   << DOCTEST_VERSION_STR << "\"\n";
5608*89c4ff92SAndroid Build Coastguard Worker         }
5609*89c4ff92SAndroid Build Coastguard Worker 
5610*89c4ff92SAndroid Build Coastguard Worker         void printIntro() {
5611*89c4ff92SAndroid Build Coastguard Worker             printVersion();
5612*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None
5613*89c4ff92SAndroid Build Coastguard Worker               << "run with \"--" DOCTEST_OPTIONS_PREFIX_DISPLAY "help\" for options\n";
5614*89c4ff92SAndroid Build Coastguard Worker         }
5615*89c4ff92SAndroid Build Coastguard Worker 
5616*89c4ff92SAndroid Build Coastguard Worker         void printHelp() {
5617*89c4ff92SAndroid Build Coastguard Worker             int sizePrefixDisplay = static_cast<int>(strlen(DOCTEST_OPTIONS_PREFIX_DISPLAY));
5618*89c4ff92SAndroid Build Coastguard Worker             printVersion();
5619*89c4ff92SAndroid Build Coastguard Worker             // clang-format off
5620*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest]\n" << Color::None;
5621*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None;
5622*89c4ff92SAndroid Build Coastguard Worker             s << "boolean values: \"1/on/yes/true\" or \"0/off/no/false\"\n";
5623*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None;
5624*89c4ff92SAndroid Build Coastguard Worker             s << "filter  values: \"str1,str2,str3\" (comma separated strings)\n";
5625*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest]\n" << Color::None;
5626*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None;
5627*89c4ff92SAndroid Build Coastguard Worker             s << "filters use wildcards for matching strings\n";
5628*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None;
5629*89c4ff92SAndroid Build Coastguard Worker             s << "something passes a filter if any of the strings in a filter matches\n";
5630*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_UNPREFIXED_OPTIONS
5631*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest]\n" << Color::None;
5632*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None;
5633*89c4ff92SAndroid Build Coastguard Worker             s << "ALL FLAGS, OPTIONS AND FILTERS ALSO AVAILABLE WITH A \"" DOCTEST_CONFIG_OPTIONS_PREFIX "\" PREFIX!!!\n";
5634*89c4ff92SAndroid Build Coastguard Worker #endif
5635*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest]\n" << Color::None;
5636*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None;
5637*89c4ff92SAndroid Build Coastguard Worker             s << "Query flags - the program quits after them. Available:\n\n";
5638*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "?,   --" DOCTEST_OPTIONS_PREFIX_DISPLAY "help, -" DOCTEST_OPTIONS_PREFIX_DISPLAY "h                      "
5639*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*0) <<  "prints this message\n";
5640*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "v,   --" DOCTEST_OPTIONS_PREFIX_DISPLAY "version                       "
5641*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "prints the version\n";
5642*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "c,   --" DOCTEST_OPTIONS_PREFIX_DISPLAY "count                         "
5643*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "prints the number of matching tests\n";
5644*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "ltc, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "list-test-cases               "
5645*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "lists all matching tests by name\n";
5646*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "lts, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "list-test-suites              "
5647*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "lists all matching test suites\n";
5648*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "lr,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "list-reporters                "
5649*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "lists all registered reporters\n\n";
5650*89c4ff92SAndroid Build Coastguard Worker             // ================================================================================== << 79
5651*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None;
5652*89c4ff92SAndroid Build Coastguard Worker             s << "The available <int>/<string> options/filters are:\n\n";
5653*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "tc,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "test-case=<filters>           "
5654*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "filters     tests by their name\n";
5655*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "tce, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "test-case-exclude=<filters>   "
5656*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "filters OUT tests by their name\n";
5657*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "sf,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "source-file=<filters>         "
5658*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "filters     tests by their file\n";
5659*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "sfe, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "source-file-exclude=<filters> "
5660*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "filters OUT tests by their file\n";
5661*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "ts,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "test-suite=<filters>          "
5662*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "filters     tests by their test suite\n";
5663*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "tse, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "test-suite-exclude=<filters>  "
5664*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "filters OUT tests by their test suite\n";
5665*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "sc,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "subcase=<filters>             "
5666*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "filters     subcases by their name\n";
5667*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "sce, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "subcase-exclude=<filters>     "
5668*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "filters OUT subcases by their name\n";
5669*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "r,   --" DOCTEST_OPTIONS_PREFIX_DISPLAY "reporters=<filters>           "
5670*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "reporters to use (console is default)\n";
5671*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "o,   --" DOCTEST_OPTIONS_PREFIX_DISPLAY "out=<string>                  "
5672*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "output filename\n";
5673*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "ob,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "order-by=<string>             "
5674*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "how the tests should be ordered\n";
5675*89c4ff92SAndroid Build Coastguard Worker             s << Whitespace(sizePrefixDisplay*3) << "                                       <string> - [file/suite/name/rand/none]\n";
5676*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "rs,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "rand-seed=<int>               "
5677*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "seed for random ordering\n";
5678*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "f,   --" DOCTEST_OPTIONS_PREFIX_DISPLAY "first=<int>                   "
5679*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "the first test passing the filters to\n";
5680*89c4ff92SAndroid Build Coastguard Worker             s << Whitespace(sizePrefixDisplay*3) << "                                       execute - for range-based execution\n";
5681*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "l,   --" DOCTEST_OPTIONS_PREFIX_DISPLAY "last=<int>                    "
5682*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "the last test passing the filters to\n";
5683*89c4ff92SAndroid Build Coastguard Worker             s << Whitespace(sizePrefixDisplay*3) << "                                       execute - for range-based execution\n";
5684*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "aa,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "abort-after=<int>             "
5685*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "stop after <int> failed assertions\n";
5686*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "scfl,--" DOCTEST_OPTIONS_PREFIX_DISPLAY "subcase-filter-levels=<int>   "
5687*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "apply filters for the first <int> levels\n";
5688*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "\n[doctest] " << Color::None;
5689*89c4ff92SAndroid Build Coastguard Worker             s << "Bool options - can be used like flags and true is assumed. Available:\n\n";
5690*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "s,   --" DOCTEST_OPTIONS_PREFIX_DISPLAY "success=<bool>                "
5691*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "include successful assertions in output\n";
5692*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "cs,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "case-sensitive=<bool>         "
5693*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "filters being treated as case sensitive\n";
5694*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "e,   --" DOCTEST_OPTIONS_PREFIX_DISPLAY "exit=<bool>                   "
5695*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "exits after the tests finish\n";
5696*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "d,   --" DOCTEST_OPTIONS_PREFIX_DISPLAY "duration=<bool>               "
5697*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "prints the time duration of each test\n";
5698*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nt,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-throw=<bool>               "
5699*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "skips exceptions-related assert checks\n";
5700*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "ne,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-exitcode=<bool>            "
5701*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "returns (or exits) always with success\n";
5702*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nr,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-run=<bool>                 "
5703*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "skips all runtime doctest operations\n";
5704*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nv,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-version=<bool>             "
5705*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "omit the framework version in the output\n";
5706*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nc,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-colors=<bool>              "
5707*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "disables colors in output\n";
5708*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "fc,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "force-colors=<bool>           "
5709*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "use colors even when not in a tty\n";
5710*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nb,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-breaks=<bool>              "
5711*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "disables breakpoints in debuggers\n";
5712*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "ns,  --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-skip=<bool>                "
5713*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "don't skip test cases marked as skip\n";
5714*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "gfl, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "gnu-file-line=<bool>          "
5715*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << ":n: vs (n): for line numbers in output\n";
5716*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "npf, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-path-filenames=<bool>      "
5717*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "only filenames and no paths in output\n";
5718*89c4ff92SAndroid Build Coastguard Worker             s << " -" DOCTEST_OPTIONS_PREFIX_DISPLAY "nln, --" DOCTEST_OPTIONS_PREFIX_DISPLAY "no-line-numbers=<bool>        "
5719*89c4ff92SAndroid Build Coastguard Worker               << Whitespace(sizePrefixDisplay*1) << "0 instead of real line numbers in output\n";
5720*89c4ff92SAndroid Build Coastguard Worker             // ================================================================================== << 79
5721*89c4ff92SAndroid Build Coastguard Worker             // clang-format on
5722*89c4ff92SAndroid Build Coastguard Worker 
5723*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "\n[doctest] " << Color::None;
5724*89c4ff92SAndroid Build Coastguard Worker             s << "for more information visit the project documentation\n\n";
5725*89c4ff92SAndroid Build Coastguard Worker         }
5726*89c4ff92SAndroid Build Coastguard Worker 
5727*89c4ff92SAndroid Build Coastguard Worker         void printRegisteredReporters() {
5728*89c4ff92SAndroid Build Coastguard Worker             printVersion();
5729*89c4ff92SAndroid Build Coastguard Worker             auto printReporters = [this] (const reporterMap& reporters, const char* type) {
5730*89c4ff92SAndroid Build Coastguard Worker                 if(reporters.size()) {
5731*89c4ff92SAndroid Build Coastguard Worker                     s << Color::Cyan << "[doctest] " << Color::None << "listing all registered " << type << "\n";
5732*89c4ff92SAndroid Build Coastguard Worker                     for(auto& curr : reporters)
5733*89c4ff92SAndroid Build Coastguard Worker                         s << "priority: " << std::setw(5) << curr.first.first
5734*89c4ff92SAndroid Build Coastguard Worker                           << " name: " << curr.first.second << "\n";
5735*89c4ff92SAndroid Build Coastguard Worker                 }
5736*89c4ff92SAndroid Build Coastguard Worker             };
5737*89c4ff92SAndroid Build Coastguard Worker             printReporters(getListeners(), "listeners");
5738*89c4ff92SAndroid Build Coastguard Worker             printReporters(getReporters(), "reporters");
5739*89c4ff92SAndroid Build Coastguard Worker         }
5740*89c4ff92SAndroid Build Coastguard Worker 
5741*89c4ff92SAndroid Build Coastguard Worker         void list_query_results() {
5742*89c4ff92SAndroid Build Coastguard Worker             separator_to_stream();
5743*89c4ff92SAndroid Build Coastguard Worker             if(opt.count || opt.list_test_cases) {
5744*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Cyan << "[doctest] " << Color::None
5745*89c4ff92SAndroid Build Coastguard Worker                   << "unskipped test cases passing the current filters: "
5746*89c4ff92SAndroid Build Coastguard Worker                   << g_cs->numTestCasesPassingFilters << "\n";
5747*89c4ff92SAndroid Build Coastguard Worker             } else if(opt.list_test_suites) {
5748*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Cyan << "[doctest] " << Color::None
5749*89c4ff92SAndroid Build Coastguard Worker                   << "unskipped test cases passing the current filters: "
5750*89c4ff92SAndroid Build Coastguard Worker                   << g_cs->numTestCasesPassingFilters << "\n";
5751*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Cyan << "[doctest] " << Color::None
5752*89c4ff92SAndroid Build Coastguard Worker                   << "test suites with unskipped test cases passing the current filters: "
5753*89c4ff92SAndroid Build Coastguard Worker                   << g_cs->numTestSuitesPassingFilters << "\n";
5754*89c4ff92SAndroid Build Coastguard Worker             }
5755*89c4ff92SAndroid Build Coastguard Worker         }
5756*89c4ff92SAndroid Build Coastguard Worker 
5757*89c4ff92SAndroid Build Coastguard Worker         // =========================================================================================
5758*89c4ff92SAndroid Build Coastguard Worker         // WHAT FOLLOWS ARE OVERRIDES OF THE VIRTUAL METHODS OF THE REPORTER INTERFACE
5759*89c4ff92SAndroid Build Coastguard Worker         // =========================================================================================
5760*89c4ff92SAndroid Build Coastguard Worker 
5761*89c4ff92SAndroid Build Coastguard Worker         void report_query(const QueryData& in) override {
5762*89c4ff92SAndroid Build Coastguard Worker             if(opt.version) {
5763*89c4ff92SAndroid Build Coastguard Worker                 printVersion();
5764*89c4ff92SAndroid Build Coastguard Worker             } else if(opt.help) {
5765*89c4ff92SAndroid Build Coastguard Worker                 printHelp();
5766*89c4ff92SAndroid Build Coastguard Worker             } else if(opt.list_reporters) {
5767*89c4ff92SAndroid Build Coastguard Worker                 printRegisteredReporters();
5768*89c4ff92SAndroid Build Coastguard Worker             } else if(opt.count || opt.list_test_cases) {
5769*89c4ff92SAndroid Build Coastguard Worker                 if(opt.list_test_cases) {
5770*89c4ff92SAndroid Build Coastguard Worker                     s << Color::Cyan << "[doctest] " << Color::None
5771*89c4ff92SAndroid Build Coastguard Worker                       << "listing all test case names\n";
5772*89c4ff92SAndroid Build Coastguard Worker                     separator_to_stream();
5773*89c4ff92SAndroid Build Coastguard Worker                 }
5774*89c4ff92SAndroid Build Coastguard Worker 
5775*89c4ff92SAndroid Build Coastguard Worker                 for(unsigned i = 0; i < in.num_data; ++i)
5776*89c4ff92SAndroid Build Coastguard Worker                     s << Color::None << in.data[i]->m_name << "\n";
5777*89c4ff92SAndroid Build Coastguard Worker 
5778*89c4ff92SAndroid Build Coastguard Worker                 separator_to_stream();
5779*89c4ff92SAndroid Build Coastguard Worker 
5780*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Cyan << "[doctest] " << Color::None
5781*89c4ff92SAndroid Build Coastguard Worker                   << "unskipped test cases passing the current filters: "
5782*89c4ff92SAndroid Build Coastguard Worker                   << g_cs->numTestCasesPassingFilters << "\n";
5783*89c4ff92SAndroid Build Coastguard Worker 
5784*89c4ff92SAndroid Build Coastguard Worker             } else if(opt.list_test_suites) {
5785*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Cyan << "[doctest] " << Color::None << "listing all test suites\n";
5786*89c4ff92SAndroid Build Coastguard Worker                 separator_to_stream();
5787*89c4ff92SAndroid Build Coastguard Worker 
5788*89c4ff92SAndroid Build Coastguard Worker                 for(unsigned i = 0; i < in.num_data; ++i)
5789*89c4ff92SAndroid Build Coastguard Worker                     s << Color::None << in.data[i]->m_test_suite << "\n";
5790*89c4ff92SAndroid Build Coastguard Worker 
5791*89c4ff92SAndroid Build Coastguard Worker                 separator_to_stream();
5792*89c4ff92SAndroid Build Coastguard Worker 
5793*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Cyan << "[doctest] " << Color::None
5794*89c4ff92SAndroid Build Coastguard Worker                   << "unskipped test cases passing the current filters: "
5795*89c4ff92SAndroid Build Coastguard Worker                   << g_cs->numTestCasesPassingFilters << "\n";
5796*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Cyan << "[doctest] " << Color::None
5797*89c4ff92SAndroid Build Coastguard Worker                   << "test suites with unskipped test cases passing the current filters: "
5798*89c4ff92SAndroid Build Coastguard Worker                   << g_cs->numTestSuitesPassingFilters << "\n";
5799*89c4ff92SAndroid Build Coastguard Worker             }
5800*89c4ff92SAndroid Build Coastguard Worker         }
5801*89c4ff92SAndroid Build Coastguard Worker 
5802*89c4ff92SAndroid Build Coastguard Worker         void test_run_start() override { printIntro(); }
5803*89c4ff92SAndroid Build Coastguard Worker 
5804*89c4ff92SAndroid Build Coastguard Worker         void test_run_end(const TestRunStats& p) override {
5805*89c4ff92SAndroid Build Coastguard Worker             separator_to_stream();
5806*89c4ff92SAndroid Build Coastguard Worker             s << std::dec;
5807*89c4ff92SAndroid Build Coastguard Worker 
5808*89c4ff92SAndroid Build Coastguard Worker             auto totwidth = int(std::ceil(log10((std::max(p.numTestCasesPassingFilters, static_cast<unsigned>(p.numAsserts))) + 1)));
5809*89c4ff92SAndroid Build Coastguard Worker             auto passwidth = int(std::ceil(log10((std::max(p.numTestCasesPassingFilters - p.numTestCasesFailed, static_cast<unsigned>(p.numAsserts - p.numAssertsFailed))) + 1)));
5810*89c4ff92SAndroid Build Coastguard Worker             auto failwidth = int(std::ceil(log10((std::max(p.numTestCasesFailed, static_cast<unsigned>(p.numAssertsFailed))) + 1)));
5811*89c4ff92SAndroid Build Coastguard Worker             const bool anythingFailed = p.numTestCasesFailed > 0 || p.numAssertsFailed > 0;
5812*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None << "test cases: " << std::setw(totwidth)
5813*89c4ff92SAndroid Build Coastguard Worker               << p.numTestCasesPassingFilters << " | "
5814*89c4ff92SAndroid Build Coastguard Worker               << ((p.numTestCasesPassingFilters == 0 || anythingFailed) ? Color::None :
5815*89c4ff92SAndroid Build Coastguard Worker                                                                           Color::Green)
5816*89c4ff92SAndroid Build Coastguard Worker               << std::setw(passwidth) << p.numTestCasesPassingFilters - p.numTestCasesFailed << " passed"
5817*89c4ff92SAndroid Build Coastguard Worker               << Color::None << " | " << (p.numTestCasesFailed > 0 ? Color::Red : Color::None)
5818*89c4ff92SAndroid Build Coastguard Worker               << std::setw(failwidth) << p.numTestCasesFailed << " failed" << Color::None << " |";
5819*89c4ff92SAndroid Build Coastguard Worker             if(opt.no_skipped_summary == false) {
5820*89c4ff92SAndroid Build Coastguard Worker                 const int numSkipped = p.numTestCases - p.numTestCasesPassingFilters;
5821*89c4ff92SAndroid Build Coastguard Worker                 s << " " << (numSkipped == 0 ? Color::None : Color::Yellow) << numSkipped
5822*89c4ff92SAndroid Build Coastguard Worker                   << " skipped" << Color::None;
5823*89c4ff92SAndroid Build Coastguard Worker             }
5824*89c4ff92SAndroid Build Coastguard Worker             s << "\n";
5825*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None << "assertions: " << std::setw(totwidth)
5826*89c4ff92SAndroid Build Coastguard Worker               << p.numAsserts << " | "
5827*89c4ff92SAndroid Build Coastguard Worker               << ((p.numAsserts == 0 || anythingFailed) ? Color::None : Color::Green)
5828*89c4ff92SAndroid Build Coastguard Worker               << std::setw(passwidth) << (p.numAsserts - p.numAssertsFailed) << " passed" << Color::None
5829*89c4ff92SAndroid Build Coastguard Worker               << " | " << (p.numAssertsFailed > 0 ? Color::Red : Color::None) << std::setw(failwidth)
5830*89c4ff92SAndroid Build Coastguard Worker               << p.numAssertsFailed << " failed" << Color::None << " |\n";
5831*89c4ff92SAndroid Build Coastguard Worker             s << Color::Cyan << "[doctest] " << Color::None
5832*89c4ff92SAndroid Build Coastguard Worker               << "Status: " << (p.numTestCasesFailed > 0 ? Color::Red : Color::Green)
5833*89c4ff92SAndroid Build Coastguard Worker               << ((p.numTestCasesFailed > 0) ? "FAILURE!" : "SUCCESS!") << Color::None << std::endl;
5834*89c4ff92SAndroid Build Coastguard Worker         }
5835*89c4ff92SAndroid Build Coastguard Worker 
5836*89c4ff92SAndroid Build Coastguard Worker         void test_case_start(const TestCaseData& in) override {
5837*89c4ff92SAndroid Build Coastguard Worker             hasLoggedCurrentTestStart = false;
5838*89c4ff92SAndroid Build Coastguard Worker             tc                        = &in;
5839*89c4ff92SAndroid Build Coastguard Worker             subcasesStack.clear();
5840*89c4ff92SAndroid Build Coastguard Worker             currentSubcaseLevel = 0;
5841*89c4ff92SAndroid Build Coastguard Worker         }
5842*89c4ff92SAndroid Build Coastguard Worker 
5843*89c4ff92SAndroid Build Coastguard Worker         void test_case_reenter(const TestCaseData&) override {
5844*89c4ff92SAndroid Build Coastguard Worker             subcasesStack.clear();
5845*89c4ff92SAndroid Build Coastguard Worker         }
5846*89c4ff92SAndroid Build Coastguard Worker 
5847*89c4ff92SAndroid Build Coastguard Worker         void test_case_end(const CurrentTestCaseStats& st) override {
5848*89c4ff92SAndroid Build Coastguard Worker             if(tc->m_no_output)
5849*89c4ff92SAndroid Build Coastguard Worker                 return;
5850*89c4ff92SAndroid Build Coastguard Worker 
5851*89c4ff92SAndroid Build Coastguard Worker             // log the preamble of the test case only if there is something
5852*89c4ff92SAndroid Build Coastguard Worker             // else to print - something other than that an assert has failed
5853*89c4ff92SAndroid Build Coastguard Worker             if(opt.duration ||
5854*89c4ff92SAndroid Build Coastguard Worker                (st.failure_flags && st.failure_flags != TestCaseFailureReason::AssertFailure))
5855*89c4ff92SAndroid Build Coastguard Worker                 logTestStart();
5856*89c4ff92SAndroid Build Coastguard Worker 
5857*89c4ff92SAndroid Build Coastguard Worker             if(opt.duration)
5858*89c4ff92SAndroid Build Coastguard Worker                 s << Color::None << std::setprecision(6) << std::fixed << st.seconds
5859*89c4ff92SAndroid Build Coastguard Worker                   << " s: " << tc->m_name << "\n";
5860*89c4ff92SAndroid Build Coastguard Worker 
5861*89c4ff92SAndroid Build Coastguard Worker             if(st.failure_flags & TestCaseFailureReason::Timeout)
5862*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Red << "Test case exceeded time limit of " << std::setprecision(6)
5863*89c4ff92SAndroid Build Coastguard Worker                   << std::fixed << tc->m_timeout << "!\n";
5864*89c4ff92SAndroid Build Coastguard Worker 
5865*89c4ff92SAndroid Build Coastguard Worker             if(st.failure_flags & TestCaseFailureReason::ShouldHaveFailedButDidnt) {
5866*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Red << "Should have failed but didn't! Marking it as failed!\n";
5867*89c4ff92SAndroid Build Coastguard Worker             } else if(st.failure_flags & TestCaseFailureReason::ShouldHaveFailedAndDid) {
5868*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Yellow << "Failed as expected so marking it as not failed\n";
5869*89c4ff92SAndroid Build Coastguard Worker             } else if(st.failure_flags & TestCaseFailureReason::CouldHaveFailedAndDid) {
5870*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Yellow << "Allowed to fail so marking it as not failed\n";
5871*89c4ff92SAndroid Build Coastguard Worker             } else if(st.failure_flags & TestCaseFailureReason::DidntFailExactlyNumTimes) {
5872*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Red << "Didn't fail exactly " << tc->m_expected_failures
5873*89c4ff92SAndroid Build Coastguard Worker                   << " times so marking it as failed!\n";
5874*89c4ff92SAndroid Build Coastguard Worker             } else if(st.failure_flags & TestCaseFailureReason::FailedExactlyNumTimes) {
5875*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Yellow << "Failed exactly " << tc->m_expected_failures
5876*89c4ff92SAndroid Build Coastguard Worker                   << " times as expected so marking it as not failed!\n";
5877*89c4ff92SAndroid Build Coastguard Worker             }
5878*89c4ff92SAndroid Build Coastguard Worker             if(st.failure_flags & TestCaseFailureReason::TooManyFailedAsserts) {
5879*89c4ff92SAndroid Build Coastguard Worker                 s << Color::Red << "Aborting - too many failed asserts!\n";
5880*89c4ff92SAndroid Build Coastguard Worker             }
5881*89c4ff92SAndroid Build Coastguard Worker             s << Color::None; // lgtm [cpp/useless-expression]
5882*89c4ff92SAndroid Build Coastguard Worker         }
5883*89c4ff92SAndroid Build Coastguard Worker 
5884*89c4ff92SAndroid Build Coastguard Worker         void test_case_exception(const TestCaseException& e) override {
5885*89c4ff92SAndroid Build Coastguard Worker             if(tc->m_no_output)
5886*89c4ff92SAndroid Build Coastguard Worker                 return;
5887*89c4ff92SAndroid Build Coastguard Worker 
5888*89c4ff92SAndroid Build Coastguard Worker             logTestStart();
5889*89c4ff92SAndroid Build Coastguard Worker 
5890*89c4ff92SAndroid Build Coastguard Worker             file_line_to_stream(tc->m_file.c_str(), tc->m_line, " ");
5891*89c4ff92SAndroid Build Coastguard Worker             successOrFailColoredStringToStream(false, e.is_crash ? assertType::is_require :
5892*89c4ff92SAndroid Build Coastguard Worker                                                                    assertType::is_check);
5893*89c4ff92SAndroid Build Coastguard Worker             s << Color::Red << (e.is_crash ? "test case CRASHED: " : "test case THREW exception: ")
5894*89c4ff92SAndroid Build Coastguard Worker               << Color::Cyan << e.error_string << "\n";
5895*89c4ff92SAndroid Build Coastguard Worker 
5896*89c4ff92SAndroid Build Coastguard Worker             int num_stringified_contexts = get_num_stringified_contexts();
5897*89c4ff92SAndroid Build Coastguard Worker             if(num_stringified_contexts) {
5898*89c4ff92SAndroid Build Coastguard Worker                 auto stringified_contexts = get_stringified_contexts();
5899*89c4ff92SAndroid Build Coastguard Worker                 s << Color::None << "  logged: ";
5900*89c4ff92SAndroid Build Coastguard Worker                 for(int i = num_stringified_contexts; i > 0; --i) {
5901*89c4ff92SAndroid Build Coastguard Worker                     s << (i == num_stringified_contexts ? "" : "          ")
5902*89c4ff92SAndroid Build Coastguard Worker                       << stringified_contexts[i - 1] << "\n";
5903*89c4ff92SAndroid Build Coastguard Worker                 }
5904*89c4ff92SAndroid Build Coastguard Worker             }
5905*89c4ff92SAndroid Build Coastguard Worker             s << "\n" << Color::None;
5906*89c4ff92SAndroid Build Coastguard Worker         }
5907*89c4ff92SAndroid Build Coastguard Worker 
5908*89c4ff92SAndroid Build Coastguard Worker         void subcase_start(const SubcaseSignature& subc) override {
5909*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5910*89c4ff92SAndroid Build Coastguard Worker             subcasesStack.push_back(subc);
5911*89c4ff92SAndroid Build Coastguard Worker             ++currentSubcaseLevel;
5912*89c4ff92SAndroid Build Coastguard Worker             hasLoggedCurrentTestStart = false;
5913*89c4ff92SAndroid Build Coastguard Worker         }
5914*89c4ff92SAndroid Build Coastguard Worker 
5915*89c4ff92SAndroid Build Coastguard Worker         void subcase_end() override {
5916*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5917*89c4ff92SAndroid Build Coastguard Worker             --currentSubcaseLevel;
5918*89c4ff92SAndroid Build Coastguard Worker             hasLoggedCurrentTestStart = false;
5919*89c4ff92SAndroid Build Coastguard Worker         }
5920*89c4ff92SAndroid Build Coastguard Worker 
5921*89c4ff92SAndroid Build Coastguard Worker         void log_assert(const AssertData& rb) override {
5922*89c4ff92SAndroid Build Coastguard Worker             if((!rb.m_failed && !opt.success) || tc->m_no_output)
5923*89c4ff92SAndroid Build Coastguard Worker                 return;
5924*89c4ff92SAndroid Build Coastguard Worker 
5925*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5926*89c4ff92SAndroid Build Coastguard Worker 
5927*89c4ff92SAndroid Build Coastguard Worker             logTestStart();
5928*89c4ff92SAndroid Build Coastguard Worker 
5929*89c4ff92SAndroid Build Coastguard Worker             file_line_to_stream(rb.m_file, rb.m_line, " ");
5930*89c4ff92SAndroid Build Coastguard Worker             successOrFailColoredStringToStream(!rb.m_failed, rb.m_at);
5931*89c4ff92SAndroid Build Coastguard Worker 
5932*89c4ff92SAndroid Build Coastguard Worker             fulltext_log_assert_to_stream(s, rb);
5933*89c4ff92SAndroid Build Coastguard Worker 
5934*89c4ff92SAndroid Build Coastguard Worker             log_contexts();
5935*89c4ff92SAndroid Build Coastguard Worker         }
5936*89c4ff92SAndroid Build Coastguard Worker 
5937*89c4ff92SAndroid Build Coastguard Worker         void log_message(const MessageData& mb) override {
5938*89c4ff92SAndroid Build Coastguard Worker             if(tc->m_no_output)
5939*89c4ff92SAndroid Build Coastguard Worker                 return;
5940*89c4ff92SAndroid Build Coastguard Worker 
5941*89c4ff92SAndroid Build Coastguard Worker             std::lock_guard<std::mutex> lock(mutex);
5942*89c4ff92SAndroid Build Coastguard Worker 
5943*89c4ff92SAndroid Build Coastguard Worker             logTestStart();
5944*89c4ff92SAndroid Build Coastguard Worker 
5945*89c4ff92SAndroid Build Coastguard Worker             file_line_to_stream(mb.m_file, mb.m_line, " ");
5946*89c4ff92SAndroid Build Coastguard Worker             s << getSuccessOrFailColor(false, mb.m_severity)
5947*89c4ff92SAndroid Build Coastguard Worker               << getSuccessOrFailString(mb.m_severity & assertType::is_warn, mb.m_severity,
5948*89c4ff92SAndroid Build Coastguard Worker                                         "MESSAGE") << ": ";
5949*89c4ff92SAndroid Build Coastguard Worker             s << Color::None << mb.m_string << "\n";
5950*89c4ff92SAndroid Build Coastguard Worker             log_contexts();
5951*89c4ff92SAndroid Build Coastguard Worker         }
5952*89c4ff92SAndroid Build Coastguard Worker 
5953*89c4ff92SAndroid Build Coastguard Worker         void test_case_skipped(const TestCaseData&) override {}
5954*89c4ff92SAndroid Build Coastguard Worker     };
5955*89c4ff92SAndroid Build Coastguard Worker 
5956*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_REGISTER_REPORTER("console", 0, ConsoleReporter);
5957*89c4ff92SAndroid Build Coastguard Worker 
5958*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_WINDOWS
5959*89c4ff92SAndroid Build Coastguard Worker     struct DebugOutputWindowReporter : public ConsoleReporter
5960*89c4ff92SAndroid Build Coastguard Worker     {
5961*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_THREAD_LOCAL static std::ostringstream oss;
5962*89c4ff92SAndroid Build Coastguard Worker 
5963*89c4ff92SAndroid Build Coastguard Worker         DebugOutputWindowReporter(const ContextOptions& co)
5964*89c4ff92SAndroid Build Coastguard Worker                 : ConsoleReporter(co, oss) {}
5965*89c4ff92SAndroid Build Coastguard Worker 
5966*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(func, type, arg)                                    \
5967*89c4ff92SAndroid Build Coastguard Worker     void func(type arg) override {                                                                 \
5968*89c4ff92SAndroid Build Coastguard Worker         bool with_col = g_no_colors;                                                               \
5969*89c4ff92SAndroid Build Coastguard Worker         g_no_colors   = false;                                                                     \
5970*89c4ff92SAndroid Build Coastguard Worker         ConsoleReporter::func(arg);                                                                \
5971*89c4ff92SAndroid Build Coastguard Worker         if(oss.tellp() != std::streampos{}) {                                                      \
5972*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_OUTPUT_DEBUG_STRING(oss.str().c_str());                                        \
5973*89c4ff92SAndroid Build Coastguard Worker             oss.str("");                                                                           \
5974*89c4ff92SAndroid Build Coastguard Worker         }                                                                                          \
5975*89c4ff92SAndroid Build Coastguard Worker         g_no_colors = with_col;                                                                    \
5976*89c4ff92SAndroid Build Coastguard Worker     }
5977*89c4ff92SAndroid Build Coastguard Worker 
5978*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(test_run_start, DOCTEST_EMPTY, DOCTEST_EMPTY)
5979*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(test_run_end, const TestRunStats&, in)
5980*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(test_case_start, const TestCaseData&, in)
5981*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(test_case_reenter, const TestCaseData&, in)
5982*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(test_case_end, const CurrentTestCaseStats&, in)
5983*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(test_case_exception, const TestCaseException&, in)
5984*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(subcase_start, const SubcaseSignature&, in)
5985*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(subcase_end, DOCTEST_EMPTY, DOCTEST_EMPTY)
5986*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(log_assert, const AssertData&, in)
5987*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(log_message, const MessageData&, in)
5988*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_DEBUG_OUTPUT_REPORTER_OVERRIDE(test_case_skipped, const TestCaseData&, in)
5989*89c4ff92SAndroid Build Coastguard Worker     };
5990*89c4ff92SAndroid Build Coastguard Worker 
5991*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_THREAD_LOCAL std::ostringstream DebugOutputWindowReporter::oss;
5992*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM_WINDOWS
5993*89c4ff92SAndroid Build Coastguard Worker 
5994*89c4ff92SAndroid Build Coastguard Worker     // the implementation of parseOption()
5995*89c4ff92SAndroid Build Coastguard Worker     bool parseOptionImpl(int argc, const char* const* argv, const char* pattern, String* value) {
5996*89c4ff92SAndroid Build Coastguard Worker         // going from the end to the beginning and stopping on the first occurrence from the end
5997*89c4ff92SAndroid Build Coastguard Worker         for(int i = argc; i > 0; --i) {
5998*89c4ff92SAndroid Build Coastguard Worker             auto index = i - 1;
5999*89c4ff92SAndroid Build Coastguard Worker             auto temp = std::strstr(argv[index], pattern);
6000*89c4ff92SAndroid Build Coastguard Worker             if(temp && (value || strlen(temp) == strlen(pattern))) { //!OCLINT prefer early exits and continue
6001*89c4ff92SAndroid Build Coastguard Worker                 // eliminate matches in which the chars before the option are not '-'
6002*89c4ff92SAndroid Build Coastguard Worker                 bool noBadCharsFound = true;
6003*89c4ff92SAndroid Build Coastguard Worker                 auto curr            = argv[index];
6004*89c4ff92SAndroid Build Coastguard Worker                 while(curr != temp) {
6005*89c4ff92SAndroid Build Coastguard Worker                     if(*curr++ != '-') {
6006*89c4ff92SAndroid Build Coastguard Worker                         noBadCharsFound = false;
6007*89c4ff92SAndroid Build Coastguard Worker                         break;
6008*89c4ff92SAndroid Build Coastguard Worker                     }
6009*89c4ff92SAndroid Build Coastguard Worker                 }
6010*89c4ff92SAndroid Build Coastguard Worker                 if(noBadCharsFound && argv[index][0] == '-') {
6011*89c4ff92SAndroid Build Coastguard Worker                     if(value) {
6012*89c4ff92SAndroid Build Coastguard Worker                         // parsing the value of an option
6013*89c4ff92SAndroid Build Coastguard Worker                         temp += strlen(pattern);
6014*89c4ff92SAndroid Build Coastguard Worker                         const unsigned len = strlen(temp);
6015*89c4ff92SAndroid Build Coastguard Worker                         if(len) {
6016*89c4ff92SAndroid Build Coastguard Worker                             *value = temp;
6017*89c4ff92SAndroid Build Coastguard Worker                             return true;
6018*89c4ff92SAndroid Build Coastguard Worker                         }
6019*89c4ff92SAndroid Build Coastguard Worker                     } else {
6020*89c4ff92SAndroid Build Coastguard Worker                         // just a flag - no value
6021*89c4ff92SAndroid Build Coastguard Worker                         return true;
6022*89c4ff92SAndroid Build Coastguard Worker                     }
6023*89c4ff92SAndroid Build Coastguard Worker                 }
6024*89c4ff92SAndroid Build Coastguard Worker             }
6025*89c4ff92SAndroid Build Coastguard Worker         }
6026*89c4ff92SAndroid Build Coastguard Worker         return false;
6027*89c4ff92SAndroid Build Coastguard Worker     }
6028*89c4ff92SAndroid Build Coastguard Worker 
6029*89c4ff92SAndroid Build Coastguard Worker     // parses an option and returns the string after the '=' character
6030*89c4ff92SAndroid Build Coastguard Worker     bool parseOption(int argc, const char* const* argv, const char* pattern, String* value = nullptr,
6031*89c4ff92SAndroid Build Coastguard Worker                      const String& defaultVal = String()) {
6032*89c4ff92SAndroid Build Coastguard Worker         if(value)
6033*89c4ff92SAndroid Build Coastguard Worker             *value = defaultVal;
6034*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_UNPREFIXED_OPTIONS
6035*89c4ff92SAndroid Build Coastguard Worker         // offset (normally 3 for "dt-") to skip prefix
6036*89c4ff92SAndroid Build Coastguard Worker         if(parseOptionImpl(argc, argv, pattern + strlen(DOCTEST_CONFIG_OPTIONS_PREFIX), value))
6037*89c4ff92SAndroid Build Coastguard Worker             return true;
6038*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_UNPREFIXED_OPTIONS
6039*89c4ff92SAndroid Build Coastguard Worker         return parseOptionImpl(argc, argv, pattern, value);
6040*89c4ff92SAndroid Build Coastguard Worker     }
6041*89c4ff92SAndroid Build Coastguard Worker 
6042*89c4ff92SAndroid Build Coastguard Worker     // locates a flag on the command line
6043*89c4ff92SAndroid Build Coastguard Worker     bool parseFlag(int argc, const char* const* argv, const char* pattern) {
6044*89c4ff92SAndroid Build Coastguard Worker         return parseOption(argc, argv, pattern);
6045*89c4ff92SAndroid Build Coastguard Worker     }
6046*89c4ff92SAndroid Build Coastguard Worker 
6047*89c4ff92SAndroid Build Coastguard Worker     // parses a comma separated list of words after a pattern in one of the arguments in argv
6048*89c4ff92SAndroid Build Coastguard Worker     bool parseCommaSepArgs(int argc, const char* const* argv, const char* pattern,
6049*89c4ff92SAndroid Build Coastguard Worker                            std::vector<String>& res) {
6050*89c4ff92SAndroid Build Coastguard Worker         String filtersString;
6051*89c4ff92SAndroid Build Coastguard Worker         if(parseOption(argc, argv, pattern, &filtersString)) {
6052*89c4ff92SAndroid Build Coastguard Worker             // tokenize with "," as a separator
6053*89c4ff92SAndroid Build Coastguard Worker             // cppcheck-suppress strtokCalled
6054*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_CLANG_SUPPRESS_WARNING_WITH_PUSH("-Wdeprecated-declarations")
6055*89c4ff92SAndroid Build Coastguard Worker             auto pch = std::strtok(filtersString.c_str(), ","); // modifies the string
6056*89c4ff92SAndroid Build Coastguard Worker             while(pch != nullptr) {
6057*89c4ff92SAndroid Build Coastguard Worker                 if(strlen(pch))
6058*89c4ff92SAndroid Build Coastguard Worker                     res.push_back(pch);
6059*89c4ff92SAndroid Build Coastguard Worker                 // uses the strtok() internal state to go to the next token
6060*89c4ff92SAndroid Build Coastguard Worker                 // cppcheck-suppress strtokCalled
6061*89c4ff92SAndroid Build Coastguard Worker                 pch = std::strtok(nullptr, ",");
6062*89c4ff92SAndroid Build Coastguard Worker             }
6063*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_CLANG_SUPPRESS_WARNING_POP
6064*89c4ff92SAndroid Build Coastguard Worker             return true;
6065*89c4ff92SAndroid Build Coastguard Worker         }
6066*89c4ff92SAndroid Build Coastguard Worker         return false;
6067*89c4ff92SAndroid Build Coastguard Worker     }
6068*89c4ff92SAndroid Build Coastguard Worker 
6069*89c4ff92SAndroid Build Coastguard Worker     enum optionType
6070*89c4ff92SAndroid Build Coastguard Worker     {
6071*89c4ff92SAndroid Build Coastguard Worker         option_bool,
6072*89c4ff92SAndroid Build Coastguard Worker         option_int
6073*89c4ff92SAndroid Build Coastguard Worker     };
6074*89c4ff92SAndroid Build Coastguard Worker 
6075*89c4ff92SAndroid Build Coastguard Worker     // parses an int/bool option from the command line
6076*89c4ff92SAndroid Build Coastguard Worker     bool parseIntOption(int argc, const char* const* argv, const char* pattern, optionType type,
6077*89c4ff92SAndroid Build Coastguard Worker                         int& res) {
6078*89c4ff92SAndroid Build Coastguard Worker         String parsedValue;
6079*89c4ff92SAndroid Build Coastguard Worker         if(!parseOption(argc, argv, pattern, &parsedValue))
6080*89c4ff92SAndroid Build Coastguard Worker             return false;
6081*89c4ff92SAndroid Build Coastguard Worker 
6082*89c4ff92SAndroid Build Coastguard Worker         if(type == 0) {
6083*89c4ff92SAndroid Build Coastguard Worker             // boolean
6084*89c4ff92SAndroid Build Coastguard Worker             const char positive[][5] = {"1", "true", "on", "yes"};  // 5 - strlen("true") + 1
6085*89c4ff92SAndroid Build Coastguard Worker             const char negative[][6] = {"0", "false", "off", "no"}; // 6 - strlen("false") + 1
6086*89c4ff92SAndroid Build Coastguard Worker 
6087*89c4ff92SAndroid Build Coastguard Worker             // if the value matches any of the positive/negative possibilities
6088*89c4ff92SAndroid Build Coastguard Worker             for(unsigned i = 0; i < 4; i++) {
6089*89c4ff92SAndroid Build Coastguard Worker                 if(parsedValue.compare(positive[i], true) == 0) {
6090*89c4ff92SAndroid Build Coastguard Worker                     res = 1; //!OCLINT parameter reassignment
6091*89c4ff92SAndroid Build Coastguard Worker                     return true;
6092*89c4ff92SAndroid Build Coastguard Worker                 }
6093*89c4ff92SAndroid Build Coastguard Worker                 if(parsedValue.compare(negative[i], true) == 0) {
6094*89c4ff92SAndroid Build Coastguard Worker                     res = 0; //!OCLINT parameter reassignment
6095*89c4ff92SAndroid Build Coastguard Worker                     return true;
6096*89c4ff92SAndroid Build Coastguard Worker                 }
6097*89c4ff92SAndroid Build Coastguard Worker             }
6098*89c4ff92SAndroid Build Coastguard Worker         } else {
6099*89c4ff92SAndroid Build Coastguard Worker             // integer
6100*89c4ff92SAndroid Build Coastguard Worker             // TODO: change this to use std::stoi or something else! currently it uses undefined behavior - assumes '0' on failed parse...
6101*89c4ff92SAndroid Build Coastguard Worker             int theInt = std::atoi(parsedValue.c_str()); // NOLINT
6102*89c4ff92SAndroid Build Coastguard Worker             if(theInt != 0) {
6103*89c4ff92SAndroid Build Coastguard Worker                 res = theInt; //!OCLINT parameter reassignment
6104*89c4ff92SAndroid Build Coastguard Worker                 return true;
6105*89c4ff92SAndroid Build Coastguard Worker             }
6106*89c4ff92SAndroid Build Coastguard Worker         }
6107*89c4ff92SAndroid Build Coastguard Worker         return false;
6108*89c4ff92SAndroid Build Coastguard Worker     }
6109*89c4ff92SAndroid Build Coastguard Worker } // namespace
6110*89c4ff92SAndroid Build Coastguard Worker 
6111*89c4ff92SAndroid Build Coastguard Worker Context::Context(int argc, const char* const* argv)
6112*89c4ff92SAndroid Build Coastguard Worker         : p(new detail::ContextState) {
6113*89c4ff92SAndroid Build Coastguard Worker     parseArgs(argc, argv, true);
6114*89c4ff92SAndroid Build Coastguard Worker     if(argc)
6115*89c4ff92SAndroid Build Coastguard Worker         p->binary_name = argv[0];
6116*89c4ff92SAndroid Build Coastguard Worker }
6117*89c4ff92SAndroid Build Coastguard Worker 
6118*89c4ff92SAndroid Build Coastguard Worker Context::~Context() {
6119*89c4ff92SAndroid Build Coastguard Worker     if(g_cs == p)
6120*89c4ff92SAndroid Build Coastguard Worker         g_cs = nullptr;
6121*89c4ff92SAndroid Build Coastguard Worker     delete p;
6122*89c4ff92SAndroid Build Coastguard Worker }
6123*89c4ff92SAndroid Build Coastguard Worker 
6124*89c4ff92SAndroid Build Coastguard Worker void Context::applyCommandLine(int argc, const char* const* argv) {
6125*89c4ff92SAndroid Build Coastguard Worker     parseArgs(argc, argv);
6126*89c4ff92SAndroid Build Coastguard Worker     if(argc)
6127*89c4ff92SAndroid Build Coastguard Worker         p->binary_name = argv[0];
6128*89c4ff92SAndroid Build Coastguard Worker }
6129*89c4ff92SAndroid Build Coastguard Worker 
6130*89c4ff92SAndroid Build Coastguard Worker // parses args
6131*89c4ff92SAndroid Build Coastguard Worker void Context::parseArgs(int argc, const char* const* argv, bool withDefaults) {
6132*89c4ff92SAndroid Build Coastguard Worker     using namespace detail;
6133*89c4ff92SAndroid Build Coastguard Worker 
6134*89c4ff92SAndroid Build Coastguard Worker     // clang-format off
6135*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "source-file=",        p->filters[0]);
6136*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "sf=",                 p->filters[0]);
6137*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "source-file-exclude=",p->filters[1]);
6138*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "sfe=",                p->filters[1]);
6139*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "test-suite=",         p->filters[2]);
6140*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "ts=",                 p->filters[2]);
6141*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "test-suite-exclude=", p->filters[3]);
6142*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "tse=",                p->filters[3]);
6143*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "test-case=",          p->filters[4]);
6144*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "tc=",                 p->filters[4]);
6145*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "test-case-exclude=",  p->filters[5]);
6146*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "tce=",                p->filters[5]);
6147*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "subcase=",            p->filters[6]);
6148*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "sc=",                 p->filters[6]);
6149*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "subcase-exclude=",    p->filters[7]);
6150*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "sce=",                p->filters[7]);
6151*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "reporters=",          p->filters[8]);
6152*89c4ff92SAndroid Build Coastguard Worker     parseCommaSepArgs(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "r=",                  p->filters[8]);
6153*89c4ff92SAndroid Build Coastguard Worker     // clang-format on
6154*89c4ff92SAndroid Build Coastguard Worker 
6155*89c4ff92SAndroid Build Coastguard Worker     int    intRes = 0;
6156*89c4ff92SAndroid Build Coastguard Worker     String strRes;
6157*89c4ff92SAndroid Build Coastguard Worker 
6158*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_PARSE_AS_BOOL_OR_FLAG(name, sname, var, default)                                   \
6159*89c4ff92SAndroid Build Coastguard Worker     if(parseIntOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX name "=", option_bool, intRes) ||  \
6160*89c4ff92SAndroid Build Coastguard Worker        parseIntOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX sname "=", option_bool, intRes))   \
6161*89c4ff92SAndroid Build Coastguard Worker         p->var = static_cast<bool>(intRes);                                                        \
6162*89c4ff92SAndroid Build Coastguard Worker     else if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX name) ||                           \
6163*89c4ff92SAndroid Build Coastguard Worker             parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX sname))                            \
6164*89c4ff92SAndroid Build Coastguard Worker         p->var = true;                                                                             \
6165*89c4ff92SAndroid Build Coastguard Worker     else if(withDefaults)                                                                          \
6166*89c4ff92SAndroid Build Coastguard Worker     p->var = default
6167*89c4ff92SAndroid Build Coastguard Worker 
6168*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_PARSE_INT_OPTION(name, sname, var, default)                                        \
6169*89c4ff92SAndroid Build Coastguard Worker     if(parseIntOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX name "=", option_int, intRes) ||   \
6170*89c4ff92SAndroid Build Coastguard Worker        parseIntOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX sname "=", option_int, intRes))    \
6171*89c4ff92SAndroid Build Coastguard Worker         p->var = intRes;                                                                           \
6172*89c4ff92SAndroid Build Coastguard Worker     else if(withDefaults)                                                                          \
6173*89c4ff92SAndroid Build Coastguard Worker     p->var = default
6174*89c4ff92SAndroid Build Coastguard Worker 
6175*89c4ff92SAndroid Build Coastguard Worker #define DOCTEST_PARSE_STR_OPTION(name, sname, var, default)                                        \
6176*89c4ff92SAndroid Build Coastguard Worker     if(parseOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX name "=", &strRes, default) ||        \
6177*89c4ff92SAndroid Build Coastguard Worker        parseOption(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX sname "=", &strRes, default) ||       \
6178*89c4ff92SAndroid Build Coastguard Worker        withDefaults)                                                                               \
6179*89c4ff92SAndroid Build Coastguard Worker     p->var = strRes
6180*89c4ff92SAndroid Build Coastguard Worker 
6181*89c4ff92SAndroid Build Coastguard Worker     // clang-format off
6182*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_STR_OPTION("out", "o", out, "");
6183*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_STR_OPTION("order-by", "ob", order_by, "file");
6184*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_INT_OPTION("rand-seed", "rs", rand_seed, 0);
6185*89c4ff92SAndroid Build Coastguard Worker 
6186*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_INT_OPTION("first", "f", first, 0);
6187*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_INT_OPTION("last", "l", last, UINT_MAX);
6188*89c4ff92SAndroid Build Coastguard Worker 
6189*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_INT_OPTION("abort-after", "aa", abort_after, 0);
6190*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_INT_OPTION("subcase-filter-levels", "scfl", subcase_filter_levels, INT_MAX);
6191*89c4ff92SAndroid Build Coastguard Worker 
6192*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("success", "s", success, false);
6193*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("case-sensitive", "cs", case_sensitive, false);
6194*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("exit", "e", exit, false);
6195*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("duration", "d", duration, false);
6196*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-throw", "nt", no_throw, false);
6197*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-exitcode", "ne", no_exitcode, false);
6198*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-run", "nr", no_run, false);
6199*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-version", "nv", no_version, false);
6200*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-colors", "nc", no_colors, false);
6201*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("force-colors", "fc", force_colors, false);
6202*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-breaks", "nb", no_breaks, false);
6203*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-skip", "ns", no_skip, false);
6204*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("gnu-file-line", "gfl", gnu_file_line, !bool(DOCTEST_MSVC));
6205*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-path-filenames", "npf", no_path_in_filenames, false);
6206*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-line-numbers", "nln", no_line_numbers, false);
6207*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-debug-output", "ndo", no_debug_output, false);
6208*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-skipped-summary", "nss", no_skipped_summary, false);
6209*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_PARSE_AS_BOOL_OR_FLAG("no-time-in-output", "ntio", no_time_in_output, false);
6210*89c4ff92SAndroid Build Coastguard Worker     // clang-format on
6211*89c4ff92SAndroid Build Coastguard Worker 
6212*89c4ff92SAndroid Build Coastguard Worker     if(withDefaults) {
6213*89c4ff92SAndroid Build Coastguard Worker         p->help             = false;
6214*89c4ff92SAndroid Build Coastguard Worker         p->version          = false;
6215*89c4ff92SAndroid Build Coastguard Worker         p->count            = false;
6216*89c4ff92SAndroid Build Coastguard Worker         p->list_test_cases  = false;
6217*89c4ff92SAndroid Build Coastguard Worker         p->list_test_suites = false;
6218*89c4ff92SAndroid Build Coastguard Worker         p->list_reporters   = false;
6219*89c4ff92SAndroid Build Coastguard Worker     }
6220*89c4ff92SAndroid Build Coastguard Worker     if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "help") ||
6221*89c4ff92SAndroid Build Coastguard Worker        parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "h") ||
6222*89c4ff92SAndroid Build Coastguard Worker        parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "?")) {
6223*89c4ff92SAndroid Build Coastguard Worker         p->help = true;
6224*89c4ff92SAndroid Build Coastguard Worker         p->exit = true;
6225*89c4ff92SAndroid Build Coastguard Worker     }
6226*89c4ff92SAndroid Build Coastguard Worker     if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "version") ||
6227*89c4ff92SAndroid Build Coastguard Worker        parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "v")) {
6228*89c4ff92SAndroid Build Coastguard Worker         p->version = true;
6229*89c4ff92SAndroid Build Coastguard Worker         p->exit    = true;
6230*89c4ff92SAndroid Build Coastguard Worker     }
6231*89c4ff92SAndroid Build Coastguard Worker     if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "count") ||
6232*89c4ff92SAndroid Build Coastguard Worker        parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "c")) {
6233*89c4ff92SAndroid Build Coastguard Worker         p->count = true;
6234*89c4ff92SAndroid Build Coastguard Worker         p->exit  = true;
6235*89c4ff92SAndroid Build Coastguard Worker     }
6236*89c4ff92SAndroid Build Coastguard Worker     if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "list-test-cases") ||
6237*89c4ff92SAndroid Build Coastguard Worker        parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "ltc")) {
6238*89c4ff92SAndroid Build Coastguard Worker         p->list_test_cases = true;
6239*89c4ff92SAndroid Build Coastguard Worker         p->exit            = true;
6240*89c4ff92SAndroid Build Coastguard Worker     }
6241*89c4ff92SAndroid Build Coastguard Worker     if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "list-test-suites") ||
6242*89c4ff92SAndroid Build Coastguard Worker        parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "lts")) {
6243*89c4ff92SAndroid Build Coastguard Worker         p->list_test_suites = true;
6244*89c4ff92SAndroid Build Coastguard Worker         p->exit             = true;
6245*89c4ff92SAndroid Build Coastguard Worker     }
6246*89c4ff92SAndroid Build Coastguard Worker     if(parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "list-reporters") ||
6247*89c4ff92SAndroid Build Coastguard Worker        parseFlag(argc, argv, DOCTEST_CONFIG_OPTIONS_PREFIX "lr")) {
6248*89c4ff92SAndroid Build Coastguard Worker         p->list_reporters = true;
6249*89c4ff92SAndroid Build Coastguard Worker         p->exit           = true;
6250*89c4ff92SAndroid Build Coastguard Worker     }
6251*89c4ff92SAndroid Build Coastguard Worker }
6252*89c4ff92SAndroid Build Coastguard Worker 
6253*89c4ff92SAndroid Build Coastguard Worker // allows the user to add procedurally to the filters from the command line
6254*89c4ff92SAndroid Build Coastguard Worker void Context::addFilter(const char* filter, const char* value) { setOption(filter, value); }
6255*89c4ff92SAndroid Build Coastguard Worker 
6256*89c4ff92SAndroid Build Coastguard Worker // allows the user to clear all filters from the command line
6257*89c4ff92SAndroid Build Coastguard Worker void Context::clearFilters() {
6258*89c4ff92SAndroid Build Coastguard Worker     for(auto& curr : p->filters)
6259*89c4ff92SAndroid Build Coastguard Worker         curr.clear();
6260*89c4ff92SAndroid Build Coastguard Worker }
6261*89c4ff92SAndroid Build Coastguard Worker 
6262*89c4ff92SAndroid Build Coastguard Worker // allows the user to override procedurally the int/bool options from the command line
6263*89c4ff92SAndroid Build Coastguard Worker void Context::setOption(const char* option, int value) {
6264*89c4ff92SAndroid Build Coastguard Worker     setOption(option, toString(value).c_str());
6265*89c4ff92SAndroid Build Coastguard Worker     // NOLINTNEXTLINE(clang-analyzer-cplusplus.NewDeleteLeaks)
6266*89c4ff92SAndroid Build Coastguard Worker }
6267*89c4ff92SAndroid Build Coastguard Worker 
6268*89c4ff92SAndroid Build Coastguard Worker // allows the user to override procedurally the string options from the command line
6269*89c4ff92SAndroid Build Coastguard Worker void Context::setOption(const char* option, const char* value) {
6270*89c4ff92SAndroid Build Coastguard Worker     auto argv   = String("-") + option + "=" + value;
6271*89c4ff92SAndroid Build Coastguard Worker     auto lvalue = argv.c_str();
6272*89c4ff92SAndroid Build Coastguard Worker     parseArgs(1, &lvalue);
6273*89c4ff92SAndroid Build Coastguard Worker }
6274*89c4ff92SAndroid Build Coastguard Worker 
6275*89c4ff92SAndroid Build Coastguard Worker // users should query this in their main() and exit the program if true
6276*89c4ff92SAndroid Build Coastguard Worker bool Context::shouldExit() { return p->exit; }
6277*89c4ff92SAndroid Build Coastguard Worker 
6278*89c4ff92SAndroid Build Coastguard Worker void Context::setAsDefaultForAssertsOutOfTestCases() { g_cs = p; }
6279*89c4ff92SAndroid Build Coastguard Worker 
6280*89c4ff92SAndroid Build Coastguard Worker void Context::setAssertHandler(detail::assert_handler ah) { p->ah = ah; }
6281*89c4ff92SAndroid Build Coastguard Worker 
6282*89c4ff92SAndroid Build Coastguard Worker // the main function that does all the filtering and test running
6283*89c4ff92SAndroid Build Coastguard Worker int Context::run() {
6284*89c4ff92SAndroid Build Coastguard Worker     using namespace detail;
6285*89c4ff92SAndroid Build Coastguard Worker 
6286*89c4ff92SAndroid Build Coastguard Worker     // save the old context state in case such was setup - for using asserts out of a testing context
6287*89c4ff92SAndroid Build Coastguard Worker     auto old_cs = g_cs;
6288*89c4ff92SAndroid Build Coastguard Worker     // this is the current contest
6289*89c4ff92SAndroid Build Coastguard Worker     g_cs               = p;
6290*89c4ff92SAndroid Build Coastguard Worker     is_running_in_test = true;
6291*89c4ff92SAndroid Build Coastguard Worker 
6292*89c4ff92SAndroid Build Coastguard Worker     g_no_colors = p->no_colors;
6293*89c4ff92SAndroid Build Coastguard Worker     p->resetRunData();
6294*89c4ff92SAndroid Build Coastguard Worker 
6295*89c4ff92SAndroid Build Coastguard Worker     // stdout by default
6296*89c4ff92SAndroid Build Coastguard Worker     p->cout = &std::cout;
6297*89c4ff92SAndroid Build Coastguard Worker     p->cerr = &std::cerr;
6298*89c4ff92SAndroid Build Coastguard Worker 
6299*89c4ff92SAndroid Build Coastguard Worker     // or to a file if specified
6300*89c4ff92SAndroid Build Coastguard Worker     std::fstream fstr;
6301*89c4ff92SAndroid Build Coastguard Worker     if(p->out.size()) {
6302*89c4ff92SAndroid Build Coastguard Worker         fstr.open(p->out.c_str(), std::fstream::out);
6303*89c4ff92SAndroid Build Coastguard Worker         p->cout = &fstr;
6304*89c4ff92SAndroid Build Coastguard Worker     }
6305*89c4ff92SAndroid Build Coastguard Worker 
6306*89c4ff92SAndroid Build Coastguard Worker     FatalConditionHandler::allocateAltStackMem();
6307*89c4ff92SAndroid Build Coastguard Worker 
6308*89c4ff92SAndroid Build Coastguard Worker     auto cleanup_and_return = [&]() {
6309*89c4ff92SAndroid Build Coastguard Worker         FatalConditionHandler::freeAltStackMem();
6310*89c4ff92SAndroid Build Coastguard Worker 
6311*89c4ff92SAndroid Build Coastguard Worker         if(fstr.is_open())
6312*89c4ff92SAndroid Build Coastguard Worker             fstr.close();
6313*89c4ff92SAndroid Build Coastguard Worker 
6314*89c4ff92SAndroid Build Coastguard Worker         // restore context
6315*89c4ff92SAndroid Build Coastguard Worker         g_cs               = old_cs;
6316*89c4ff92SAndroid Build Coastguard Worker         is_running_in_test = false;
6317*89c4ff92SAndroid Build Coastguard Worker 
6318*89c4ff92SAndroid Build Coastguard Worker         // we have to free the reporters which were allocated when the run started
6319*89c4ff92SAndroid Build Coastguard Worker         for(auto& curr : p->reporters_currently_used)
6320*89c4ff92SAndroid Build Coastguard Worker             delete curr;
6321*89c4ff92SAndroid Build Coastguard Worker         p->reporters_currently_used.clear();
6322*89c4ff92SAndroid Build Coastguard Worker 
6323*89c4ff92SAndroid Build Coastguard Worker         if(p->numTestCasesFailed && !p->no_exitcode)
6324*89c4ff92SAndroid Build Coastguard Worker             return EXIT_FAILURE;
6325*89c4ff92SAndroid Build Coastguard Worker         return EXIT_SUCCESS;
6326*89c4ff92SAndroid Build Coastguard Worker     };
6327*89c4ff92SAndroid Build Coastguard Worker 
6328*89c4ff92SAndroid Build Coastguard Worker     // setup default reporter if none is given through the command line
6329*89c4ff92SAndroid Build Coastguard Worker     if(p->filters[8].empty())
6330*89c4ff92SAndroid Build Coastguard Worker         p->filters[8].push_back("console");
6331*89c4ff92SAndroid Build Coastguard Worker 
6332*89c4ff92SAndroid Build Coastguard Worker     // check to see if any of the registered reporters has been selected
6333*89c4ff92SAndroid Build Coastguard Worker     for(auto& curr : getReporters()) {
6334*89c4ff92SAndroid Build Coastguard Worker         if(matchesAny(curr.first.second.c_str(), p->filters[8], false, p->case_sensitive))
6335*89c4ff92SAndroid Build Coastguard Worker             p->reporters_currently_used.push_back(curr.second(*g_cs));
6336*89c4ff92SAndroid Build Coastguard Worker     }
6337*89c4ff92SAndroid Build Coastguard Worker 
6338*89c4ff92SAndroid Build Coastguard Worker     // TODO: check if there is nothing in reporters_currently_used
6339*89c4ff92SAndroid Build Coastguard Worker 
6340*89c4ff92SAndroid Build Coastguard Worker     // prepend all listeners
6341*89c4ff92SAndroid Build Coastguard Worker     for(auto& curr : getListeners())
6342*89c4ff92SAndroid Build Coastguard Worker         p->reporters_currently_used.insert(p->reporters_currently_used.begin(), curr.second(*g_cs));
6343*89c4ff92SAndroid Build Coastguard Worker 
6344*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_PLATFORM_WINDOWS
6345*89c4ff92SAndroid Build Coastguard Worker     if(isDebuggerActive() && p->no_debug_output == false)
6346*89c4ff92SAndroid Build Coastguard Worker         p->reporters_currently_used.push_back(new DebugOutputWindowReporter(*g_cs));
6347*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_PLATFORM_WINDOWS
6348*89c4ff92SAndroid Build Coastguard Worker 
6349*89c4ff92SAndroid Build Coastguard Worker     // handle version, help and no_run
6350*89c4ff92SAndroid Build Coastguard Worker     if(p->no_run || p->version || p->help || p->list_reporters) {
6351*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ITERATE_THROUGH_REPORTERS(report_query, QueryData());
6352*89c4ff92SAndroid Build Coastguard Worker 
6353*89c4ff92SAndroid Build Coastguard Worker         return cleanup_and_return();
6354*89c4ff92SAndroid Build Coastguard Worker     }
6355*89c4ff92SAndroid Build Coastguard Worker 
6356*89c4ff92SAndroid Build Coastguard Worker     std::vector<const TestCase*> testArray;
6357*89c4ff92SAndroid Build Coastguard Worker     for(auto& curr : getRegisteredTests())
6358*89c4ff92SAndroid Build Coastguard Worker         testArray.push_back(&curr);
6359*89c4ff92SAndroid Build Coastguard Worker     p->numTestCases = testArray.size();
6360*89c4ff92SAndroid Build Coastguard Worker 
6361*89c4ff92SAndroid Build Coastguard Worker     // sort the collected records
6362*89c4ff92SAndroid Build Coastguard Worker     if(!testArray.empty()) {
6363*89c4ff92SAndroid Build Coastguard Worker         if(p->order_by.compare("file", true) == 0) {
6364*89c4ff92SAndroid Build Coastguard Worker             std::sort(testArray.begin(), testArray.end(), fileOrderComparator);
6365*89c4ff92SAndroid Build Coastguard Worker         } else if(p->order_by.compare("suite", true) == 0) {
6366*89c4ff92SAndroid Build Coastguard Worker             std::sort(testArray.begin(), testArray.end(), suiteOrderComparator);
6367*89c4ff92SAndroid Build Coastguard Worker         } else if(p->order_by.compare("name", true) == 0) {
6368*89c4ff92SAndroid Build Coastguard Worker             std::sort(testArray.begin(), testArray.end(), nameOrderComparator);
6369*89c4ff92SAndroid Build Coastguard Worker         } else if(p->order_by.compare("rand", true) == 0) {
6370*89c4ff92SAndroid Build Coastguard Worker             std::srand(p->rand_seed);
6371*89c4ff92SAndroid Build Coastguard Worker 
6372*89c4ff92SAndroid Build Coastguard Worker             // random_shuffle implementation
6373*89c4ff92SAndroid Build Coastguard Worker             const auto first = &testArray[0];
6374*89c4ff92SAndroid Build Coastguard Worker             for(size_t i = testArray.size() - 1; i > 0; --i) {
6375*89c4ff92SAndroid Build Coastguard Worker                 int idxToSwap = std::rand() % (i + 1); // NOLINT
6376*89c4ff92SAndroid Build Coastguard Worker 
6377*89c4ff92SAndroid Build Coastguard Worker                 const auto temp = first[i];
6378*89c4ff92SAndroid Build Coastguard Worker 
6379*89c4ff92SAndroid Build Coastguard Worker                 first[i]         = first[idxToSwap];
6380*89c4ff92SAndroid Build Coastguard Worker                 first[idxToSwap] = temp;
6381*89c4ff92SAndroid Build Coastguard Worker             }
6382*89c4ff92SAndroid Build Coastguard Worker         } else if(p->order_by.compare("none", true) == 0) {
6383*89c4ff92SAndroid Build Coastguard Worker             // means no sorting - beneficial for death tests which call into the executable
6384*89c4ff92SAndroid Build Coastguard Worker             // with a specific test case in mind - we don't want to slow down the startup times
6385*89c4ff92SAndroid Build Coastguard Worker         }
6386*89c4ff92SAndroid Build Coastguard Worker     }
6387*89c4ff92SAndroid Build Coastguard Worker 
6388*89c4ff92SAndroid Build Coastguard Worker     std::set<String> testSuitesPassingFilt;
6389*89c4ff92SAndroid Build Coastguard Worker 
6390*89c4ff92SAndroid Build Coastguard Worker     bool                             query_mode = p->count || p->list_test_cases || p->list_test_suites;
6391*89c4ff92SAndroid Build Coastguard Worker     std::vector<const TestCaseData*> queryResults;
6392*89c4ff92SAndroid Build Coastguard Worker 
6393*89c4ff92SAndroid Build Coastguard Worker     if(!query_mode)
6394*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ITERATE_THROUGH_REPORTERS(test_run_start, DOCTEST_EMPTY);
6395*89c4ff92SAndroid Build Coastguard Worker 
6396*89c4ff92SAndroid Build Coastguard Worker     // invoke the registered functions if they match the filter criteria (or just count them)
6397*89c4ff92SAndroid Build Coastguard Worker     for(auto& curr : testArray) {
6398*89c4ff92SAndroid Build Coastguard Worker         const auto& tc = *curr;
6399*89c4ff92SAndroid Build Coastguard Worker 
6400*89c4ff92SAndroid Build Coastguard Worker         bool skip_me = false;
6401*89c4ff92SAndroid Build Coastguard Worker         if(tc.m_skip && !p->no_skip)
6402*89c4ff92SAndroid Build Coastguard Worker             skip_me = true;
6403*89c4ff92SAndroid Build Coastguard Worker 
6404*89c4ff92SAndroid Build Coastguard Worker         if(!matchesAny(tc.m_file.c_str(), p->filters[0], true, p->case_sensitive))
6405*89c4ff92SAndroid Build Coastguard Worker             skip_me = true;
6406*89c4ff92SAndroid Build Coastguard Worker         if(matchesAny(tc.m_file.c_str(), p->filters[1], false, p->case_sensitive))
6407*89c4ff92SAndroid Build Coastguard Worker             skip_me = true;
6408*89c4ff92SAndroid Build Coastguard Worker         if(!matchesAny(tc.m_test_suite, p->filters[2], true, p->case_sensitive))
6409*89c4ff92SAndroid Build Coastguard Worker             skip_me = true;
6410*89c4ff92SAndroid Build Coastguard Worker         if(matchesAny(tc.m_test_suite, p->filters[3], false, p->case_sensitive))
6411*89c4ff92SAndroid Build Coastguard Worker             skip_me = true;
6412*89c4ff92SAndroid Build Coastguard Worker         if(!matchesAny(tc.m_name, p->filters[4], true, p->case_sensitive))
6413*89c4ff92SAndroid Build Coastguard Worker             skip_me = true;
6414*89c4ff92SAndroid Build Coastguard Worker         if(matchesAny(tc.m_name, p->filters[5], false, p->case_sensitive))
6415*89c4ff92SAndroid Build Coastguard Worker             skip_me = true;
6416*89c4ff92SAndroid Build Coastguard Worker 
6417*89c4ff92SAndroid Build Coastguard Worker         if(!skip_me)
6418*89c4ff92SAndroid Build Coastguard Worker             p->numTestCasesPassingFilters++;
6419*89c4ff92SAndroid Build Coastguard Worker 
6420*89c4ff92SAndroid Build Coastguard Worker         // skip the test if it is not in the execution range
6421*89c4ff92SAndroid Build Coastguard Worker         if((p->last < p->numTestCasesPassingFilters && p->first <= p->last) ||
6422*89c4ff92SAndroid Build Coastguard Worker            (p->first > p->numTestCasesPassingFilters))
6423*89c4ff92SAndroid Build Coastguard Worker             skip_me = true;
6424*89c4ff92SAndroid Build Coastguard Worker 
6425*89c4ff92SAndroid Build Coastguard Worker         if(skip_me) {
6426*89c4ff92SAndroid Build Coastguard Worker             if(!query_mode)
6427*89c4ff92SAndroid Build Coastguard Worker                 DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_skipped, tc);
6428*89c4ff92SAndroid Build Coastguard Worker             continue;
6429*89c4ff92SAndroid Build Coastguard Worker         }
6430*89c4ff92SAndroid Build Coastguard Worker 
6431*89c4ff92SAndroid Build Coastguard Worker         // do not execute the test if we are to only count the number of filter passing tests
6432*89c4ff92SAndroid Build Coastguard Worker         if(p->count)
6433*89c4ff92SAndroid Build Coastguard Worker             continue;
6434*89c4ff92SAndroid Build Coastguard Worker 
6435*89c4ff92SAndroid Build Coastguard Worker         // print the name of the test and don't execute it
6436*89c4ff92SAndroid Build Coastguard Worker         if(p->list_test_cases) {
6437*89c4ff92SAndroid Build Coastguard Worker             queryResults.push_back(&tc);
6438*89c4ff92SAndroid Build Coastguard Worker             continue;
6439*89c4ff92SAndroid Build Coastguard Worker         }
6440*89c4ff92SAndroid Build Coastguard Worker 
6441*89c4ff92SAndroid Build Coastguard Worker         // print the name of the test suite if not done already and don't execute it
6442*89c4ff92SAndroid Build Coastguard Worker         if(p->list_test_suites) {
6443*89c4ff92SAndroid Build Coastguard Worker             if((testSuitesPassingFilt.count(tc.m_test_suite) == 0) && tc.m_test_suite[0] != '\0') {
6444*89c4ff92SAndroid Build Coastguard Worker                 queryResults.push_back(&tc);
6445*89c4ff92SAndroid Build Coastguard Worker                 testSuitesPassingFilt.insert(tc.m_test_suite);
6446*89c4ff92SAndroid Build Coastguard Worker                 p->numTestSuitesPassingFilters++;
6447*89c4ff92SAndroid Build Coastguard Worker             }
6448*89c4ff92SAndroid Build Coastguard Worker             continue;
6449*89c4ff92SAndroid Build Coastguard Worker         }
6450*89c4ff92SAndroid Build Coastguard Worker 
6451*89c4ff92SAndroid Build Coastguard Worker         // execute the test if it passes all the filtering
6452*89c4ff92SAndroid Build Coastguard Worker         {
6453*89c4ff92SAndroid Build Coastguard Worker             p->currentTest = &tc;
6454*89c4ff92SAndroid Build Coastguard Worker 
6455*89c4ff92SAndroid Build Coastguard Worker             p->failure_flags = TestCaseFailureReason::None;
6456*89c4ff92SAndroid Build Coastguard Worker             p->seconds       = 0;
6457*89c4ff92SAndroid Build Coastguard Worker 
6458*89c4ff92SAndroid Build Coastguard Worker             // reset atomic counters
6459*89c4ff92SAndroid Build Coastguard Worker             p->numAssertsFailedCurrentTest_atomic = 0;
6460*89c4ff92SAndroid Build Coastguard Worker             p->numAssertsCurrentTest_atomic       = 0;
6461*89c4ff92SAndroid Build Coastguard Worker 
6462*89c4ff92SAndroid Build Coastguard Worker             p->subcasesPassed.clear();
6463*89c4ff92SAndroid Build Coastguard Worker 
6464*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_start, tc);
6465*89c4ff92SAndroid Build Coastguard Worker 
6466*89c4ff92SAndroid Build Coastguard Worker             p->timer.start();
6467*89c4ff92SAndroid Build Coastguard Worker 
6468*89c4ff92SAndroid Build Coastguard Worker             bool run_test = true;
6469*89c4ff92SAndroid Build Coastguard Worker 
6470*89c4ff92SAndroid Build Coastguard Worker             do {
6471*89c4ff92SAndroid Build Coastguard Worker                 // reset some of the fields for subcases (except for the set of fully passed ones)
6472*89c4ff92SAndroid Build Coastguard Worker                 p->should_reenter          = false;
6473*89c4ff92SAndroid Build Coastguard Worker                 p->subcasesCurrentMaxLevel = 0;
6474*89c4ff92SAndroid Build Coastguard Worker                 p->subcasesStack.clear();
6475*89c4ff92SAndroid Build Coastguard Worker 
6476*89c4ff92SAndroid Build Coastguard Worker                 p->shouldLogCurrentException = true;
6477*89c4ff92SAndroid Build Coastguard Worker 
6478*89c4ff92SAndroid Build Coastguard Worker                 // reset stuff for logging with INFO()
6479*89c4ff92SAndroid Build Coastguard Worker                 p->stringifiedContexts.clear();
6480*89c4ff92SAndroid Build Coastguard Worker 
6481*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_EXCEPTIONS
6482*89c4ff92SAndroid Build Coastguard Worker                 try {
6483*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS
6484*89c4ff92SAndroid Build Coastguard Worker // MSVC 2015 diagnoses fatalConditionHandler as unused (because reset() is a static method)
6485*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4101) // unreferenced local variable
6486*89c4ff92SAndroid Build Coastguard Worker                     FatalConditionHandler fatalConditionHandler; // Handle signals
6487*89c4ff92SAndroid Build Coastguard Worker                     // execute the test
6488*89c4ff92SAndroid Build Coastguard Worker                     tc.m_test();
6489*89c4ff92SAndroid Build Coastguard Worker                     fatalConditionHandler.reset();
6490*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_POP
6491*89c4ff92SAndroid Build Coastguard Worker #ifndef DOCTEST_CONFIG_NO_EXCEPTIONS
6492*89c4ff92SAndroid Build Coastguard Worker                 } catch(const TestFailureException&) {
6493*89c4ff92SAndroid Build Coastguard Worker                     p->failure_flags |= TestCaseFailureReason::AssertFailure;
6494*89c4ff92SAndroid Build Coastguard Worker                 } catch(...) {
6495*89c4ff92SAndroid Build Coastguard Worker                     DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_exception,
6496*89c4ff92SAndroid Build Coastguard Worker                                                       {translateActiveException(), false});
6497*89c4ff92SAndroid Build Coastguard Worker                     p->failure_flags |= TestCaseFailureReason::Exception;
6498*89c4ff92SAndroid Build Coastguard Worker                 }
6499*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_NO_EXCEPTIONS
6500*89c4ff92SAndroid Build Coastguard Worker 
6501*89c4ff92SAndroid Build Coastguard Worker                 // exit this loop if enough assertions have failed - even if there are more subcases
6502*89c4ff92SAndroid Build Coastguard Worker                 if(p->abort_after > 0 &&
6503*89c4ff92SAndroid Build Coastguard Worker                    p->numAssertsFailed + p->numAssertsFailedCurrentTest_atomic >= p->abort_after) {
6504*89c4ff92SAndroid Build Coastguard Worker                     run_test = false;
6505*89c4ff92SAndroid Build Coastguard Worker                     p->failure_flags |= TestCaseFailureReason::TooManyFailedAsserts;
6506*89c4ff92SAndroid Build Coastguard Worker                 }
6507*89c4ff92SAndroid Build Coastguard Worker 
6508*89c4ff92SAndroid Build Coastguard Worker                 if(p->should_reenter && run_test)
6509*89c4ff92SAndroid Build Coastguard Worker                     DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_reenter, tc);
6510*89c4ff92SAndroid Build Coastguard Worker                 if(!p->should_reenter)
6511*89c4ff92SAndroid Build Coastguard Worker                     run_test = false;
6512*89c4ff92SAndroid Build Coastguard Worker             } while(run_test);
6513*89c4ff92SAndroid Build Coastguard Worker 
6514*89c4ff92SAndroid Build Coastguard Worker             p->finalizeTestCaseData();
6515*89c4ff92SAndroid Build Coastguard Worker 
6516*89c4ff92SAndroid Build Coastguard Worker             DOCTEST_ITERATE_THROUGH_REPORTERS(test_case_end, *g_cs);
6517*89c4ff92SAndroid Build Coastguard Worker 
6518*89c4ff92SAndroid Build Coastguard Worker             p->currentTest = nullptr;
6519*89c4ff92SAndroid Build Coastguard Worker 
6520*89c4ff92SAndroid Build Coastguard Worker             // stop executing tests if enough assertions have failed
6521*89c4ff92SAndroid Build Coastguard Worker             if(p->abort_after > 0 && p->numAssertsFailed >= p->abort_after)
6522*89c4ff92SAndroid Build Coastguard Worker                 break;
6523*89c4ff92SAndroid Build Coastguard Worker         }
6524*89c4ff92SAndroid Build Coastguard Worker     }
6525*89c4ff92SAndroid Build Coastguard Worker 
6526*89c4ff92SAndroid Build Coastguard Worker     if(!query_mode) {
6527*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ITERATE_THROUGH_REPORTERS(test_run_end, *g_cs);
6528*89c4ff92SAndroid Build Coastguard Worker     } else {
6529*89c4ff92SAndroid Build Coastguard Worker         QueryData qdata;
6530*89c4ff92SAndroid Build Coastguard Worker         qdata.run_stats = g_cs;
6531*89c4ff92SAndroid Build Coastguard Worker         qdata.data      = queryResults.data();
6532*89c4ff92SAndroid Build Coastguard Worker         qdata.num_data  = unsigned(queryResults.size());
6533*89c4ff92SAndroid Build Coastguard Worker         DOCTEST_ITERATE_THROUGH_REPORTERS(report_query, qdata);
6534*89c4ff92SAndroid Build Coastguard Worker     }
6535*89c4ff92SAndroid Build Coastguard Worker 
6536*89c4ff92SAndroid Build Coastguard Worker     // see these issues on the reasoning for this:
6537*89c4ff92SAndroid Build Coastguard Worker     // - https://github.com/onqtam/doctest/issues/143#issuecomment-414418903
6538*89c4ff92SAndroid Build Coastguard Worker     // - https://github.com/onqtam/doctest/issues/126
6539*89c4ff92SAndroid Build Coastguard Worker     auto DOCTEST_FIX_FOR_MACOS_LIBCPP_IOSFWD_STRING_LINK_ERRORS = []() DOCTEST_NOINLINE
6540*89c4ff92SAndroid Build Coastguard Worker         { std::cout << std::string(); };
6541*89c4ff92SAndroid Build Coastguard Worker     DOCTEST_FIX_FOR_MACOS_LIBCPP_IOSFWD_STRING_LINK_ERRORS();
6542*89c4ff92SAndroid Build Coastguard Worker 
6543*89c4ff92SAndroid Build Coastguard Worker     return cleanup_and_return();
6544*89c4ff92SAndroid Build Coastguard Worker }
6545*89c4ff92SAndroid Build Coastguard Worker 
6546*89c4ff92SAndroid Build Coastguard Worker IReporter::~IReporter() = default;
6547*89c4ff92SAndroid Build Coastguard Worker 
6548*89c4ff92SAndroid Build Coastguard Worker int IReporter::get_num_active_contexts() { return detail::g_infoContexts.size(); }
6549*89c4ff92SAndroid Build Coastguard Worker const IContextScope* const* IReporter::get_active_contexts() {
6550*89c4ff92SAndroid Build Coastguard Worker     return get_num_active_contexts() ? &detail::g_infoContexts[0] : nullptr;
6551*89c4ff92SAndroid Build Coastguard Worker }
6552*89c4ff92SAndroid Build Coastguard Worker 
6553*89c4ff92SAndroid Build Coastguard Worker int IReporter::get_num_stringified_contexts() { return detail::g_cs->stringifiedContexts.size(); }
6554*89c4ff92SAndroid Build Coastguard Worker const String* IReporter::get_stringified_contexts() {
6555*89c4ff92SAndroid Build Coastguard Worker     return get_num_stringified_contexts() ? &detail::g_cs->stringifiedContexts[0] : nullptr;
6556*89c4ff92SAndroid Build Coastguard Worker }
6557*89c4ff92SAndroid Build Coastguard Worker 
6558*89c4ff92SAndroid Build Coastguard Worker namespace detail {
6559*89c4ff92SAndroid Build Coastguard Worker     void registerReporterImpl(const char* name, int priority, reporterCreatorFunc c, bool isReporter) {
6560*89c4ff92SAndroid Build Coastguard Worker         if(isReporter)
6561*89c4ff92SAndroid Build Coastguard Worker             getReporters().insert(reporterMap::value_type(reporterMap::key_type(priority, name), c));
6562*89c4ff92SAndroid Build Coastguard Worker         else
6563*89c4ff92SAndroid Build Coastguard Worker             getListeners().insert(reporterMap::value_type(reporterMap::key_type(priority, name), c));
6564*89c4ff92SAndroid Build Coastguard Worker     }
6565*89c4ff92SAndroid Build Coastguard Worker } // namespace detail
6566*89c4ff92SAndroid Build Coastguard Worker 
6567*89c4ff92SAndroid Build Coastguard Worker } // namespace doctest
6568*89c4ff92SAndroid Build Coastguard Worker 
6569*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_DISABLE
6570*89c4ff92SAndroid Build Coastguard Worker 
6571*89c4ff92SAndroid Build Coastguard Worker #ifdef DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN
6572*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_WITH_PUSH(4007) // 'function' : must be 'attribute' - see issue #182
6573*89c4ff92SAndroid Build Coastguard Worker int main(int argc, char** argv) { return doctest::Context(argc, argv).run(); }
6574*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_POP
6575*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_IMPLEMENT_WITH_MAIN
6576*89c4ff92SAndroid Build Coastguard Worker 
6577*89c4ff92SAndroid Build Coastguard Worker DOCTEST_CLANG_SUPPRESS_WARNING_POP
6578*89c4ff92SAndroid Build Coastguard Worker DOCTEST_MSVC_SUPPRESS_WARNING_POP
6579*89c4ff92SAndroid Build Coastguard Worker DOCTEST_GCC_SUPPRESS_WARNING_POP
6580*89c4ff92SAndroid Build Coastguard Worker 
6581*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_LIBRARY_IMPLEMENTATION
6582*89c4ff92SAndroid Build Coastguard Worker #endif // DOCTEST_CONFIG_IMPLEMENT
6583