1 | /** Static assertion macros |
---|
2 | * Borrowed from boost and Loki. |
---|
3 | */ |
---|
4 | #ifndef GMTL_STATIC_ASSERT_HPP |
---|
5 | #define GMTL_STATIC_ASSERT_HPP |
---|
6 | |
---|
7 | namespace gmtl |
---|
8 | { |
---|
9 | /** |
---|
10 | * Define a structure that will contain our static assert; the name |
---|
11 | * CompileTimeError will hopefully be displayed and catch the eye of the |
---|
12 | * programmer. |
---|
13 | */ |
---|
14 | template <int> struct CompileTimeError; |
---|
15 | template <> struct CompileTimeError<true> {}; |
---|
16 | } |
---|
17 | /** |
---|
18 | * GMTL_STATIC_ASSERT macro. |
---|
19 | * This macro will evaluate a compile time integral or pointer expression; |
---|
20 | * if the expression is zero, the macro will generate a message in the form |
---|
21 | * of an undefined identifier. |
---|
22 | * |
---|
23 | * @param expr the expression to evaluate. |
---|
24 | * @param msg the message to display if expr is zero; msg cannot |
---|
25 | * contain spaces! |
---|
26 | */ |
---|
27 | #define GMTL_STATIC_ASSERT(expr, msg) \ |
---|
28 | { gmtl::CompileTimeError<((expr) != 0)> ERROR_##msg; (void)ERROR_##msg; } |
---|
29 | |
---|
30 | // -- OLD Static assert --- // |
---|
31 | // -- To be used if the new one causes problems -- // |
---|
32 | /* |
---|
33 | #include <gmtl/Defines.h> |
---|
34 | #include <gmtl/Util/Meta.h> |
---|
35 | |
---|
36 | #ifdef __BORLANDC__ |
---|
37 | // |
---|
38 | // workaround for buggy integral-constant expression support: |
---|
39 | #define GMTL_BUGGY_INTEGRAL_CONSTANT_EXPRESSIONS |
---|
40 | #endif |
---|
41 | |
---|
42 | namespace gmtl |
---|
43 | { |
---|
44 | |
---|
45 | // HP aCC cannot deal with missing names for template value parameters |
---|
46 | template <bool x> struct STATIC_ASSERTION_FAILURE; |
---|
47 | |
---|
48 | template <> struct STATIC_ASSERTION_FAILURE<true>{}; |
---|
49 | |
---|
50 | // HP aCC cannot deal with missing names for template value parameters |
---|
51 | template<int x> struct static_assert_test{}; |
---|
52 | |
---|
53 | } |
---|
54 | |
---|
55 | // |
---|
56 | // Implicit instantiation requires that all member declarations be |
---|
57 | // instantiated, but that the definitions are *not* instantiated. |
---|
58 | // |
---|
59 | // It's not particularly clear how this applies to enum's or typedefs; |
---|
60 | // both are described as declarations [7.1.3] and [7.2] in the standard, |
---|
61 | // however some compilers use "delayed evaluation" of one or more of |
---|
62 | // these when implicitly instantiating templates. We use typedef declarations |
---|
63 | // by default, but try defining BOOST_USE_ENUM_STATIC_ASSERT if the enum |
---|
64 | // version gets better results from your compiler... |
---|
65 | // |
---|
66 | // Implementation: |
---|
67 | // Both of these versions rely on sizeof(incomplete_type) generating an error |
---|
68 | // message containing the name of the incomplete type. We use |
---|
69 | // "STATIC_ASSERTION_FAILURE" as the type name here to generate |
---|
70 | // an eye catching error message. The result of the sizeof expression is either |
---|
71 | // used as an enum initialiser, or as a template argument depending which version |
---|
72 | // is in use... |
---|
73 | // Note that the argument to the assert is explicitly cast to bool using old- |
---|
74 | // style casts: too many compilers currently have problems with static_cast |
---|
75 | // when used inside integral constant expressions. |
---|
76 | // |
---|
77 | #if !defined(GMTL_BUGGY_INTEGRAL_CONSTANT_EXPRESSIONS) && !defined(__MWERKS__) |
---|
78 | #ifndef GMTL_MSVC |
---|
79 | #define GMTL_STATIC_ASSERT( B ) \ |
---|
80 | typedef ::gmtl::static_assert_test<\ |
---|
81 | sizeof(::gmtl::STATIC_ASSERTION_FAILURE< (bool)( B ) >)>\ |
---|
82 | GMTL_JOIN(gmtl_static_assert_typedef_, __LINE__) |
---|
83 | #else |
---|
84 | // __LINE__ macro broken when -ZI is used see Q199057 |
---|
85 | // fortunately MSVC ignores duplicate typedef's. |
---|
86 | #define GMTL_STATIC_ASSERT( B ) \ |
---|
87 | typedef ::gmtl::static_assert_test<\ |
---|
88 | sizeof(::gmtl::STATIC_ASSERTION_FAILURE< (bool)( B ) >)\ |
---|
89 | > gmtl_static_assert_typedef_ |
---|
90 | #endif |
---|
91 | #else |
---|
92 | // alternative enum based implementation: |
---|
93 | #define GMTL_STATIC_ASSERT( B ) \ |
---|
94 | enum { GMTL_JOIN(gmtl_static_assert_enum_, __LINE__) \ |
---|
95 | = sizeof(::gmtl::STATIC_ASSERTION_FAILURE< (bool)( B ) >) } |
---|
96 | #endif |
---|
97 | |
---|
98 | */ |
---|
99 | |
---|
100 | #endif // GMTL_STATIC_ASSERT_HPP |
---|