VirtualBox

source: vbox/trunk/include/iprt/assertcompile.h@ 68862

Last change on this file since 68862 was 68686, checked in by vboxsync, 7 years ago

Split out the compile time assertions from iprt/assert.h and put them in iprt/assertcompile.h.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 8.1 KB
Line 
1/** @file
2 * IPRT - Compile Time Assertions.
3 */
4
5/*
6 * Copyright (C) 2006-2017 Oracle Corporation
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * The contents of this file may alternatively be used under the terms
17 * of the Common Development and Distribution License Version 1.0
18 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
19 * VirtualBox OSE distribution, in which case the provisions of the
20 * CDDL are applicable instead of those of the GPL.
21 *
22 * You may elect to license modified versions of this file under the
23 * terms and conditions of either the GPL or the CDDL or both.
24 */
25
26#ifndef ___iprt_assertcompile_h
27#define ___iprt_assertcompile_h
28
29#include <iprt/cdefs.h>
30
31/** @defgroup grp_rt_assert_compile Compile time assertions
32 * @ingroup grp_rt
33 *
34 * These assertions are used to check structure sizes, member/size alignments
35 * and similar compile time expressions.
36 *
37 * @remarks As you might have noticed, the AssertCompile macros don't follow the
38 * coding guidelines wrt to macros supposedly being all uppercase and
39 * underscored. For various reasons they don't, and nobody has
40 * complained yet.
41 *
42 * @{
43 */
44
45/**
46 * RTASSERTTYPE is the type the AssertCompile() macro redefines.
47 * It has no other function and shouldn't be used.
48 * Visual C++ uses this.
49 */
50typedef int RTASSERTTYPE[1];
51
52/**
53 * RTASSERTVAR is the type the AssertCompile() macro redefines.
54 * It has no other function and shouldn't be used.
55 * GCC uses this.
56 */
57#ifdef __GNUC__
58RT_C_DECLS_BEGIN
59#endif
60extern int RTASSERTVAR[1];
61#ifdef __GNUC__
62RT_C_DECLS_END
63#endif
64
65/** @def RTASSERT_HAVE_STATIC_ASSERT
66 * Indicates that the compiler implements static_assert(expr, msg).
67 */
68#ifdef _MSC_VER
69# if _MSC_VER >= 1600 && defined(__cplusplus)
70# define RTASSERT_HAVE_STATIC_ASSERT
71# endif
72#endif
73#if defined(__GNUC__) && defined(__GXX_EXPERIMENTAL_CXX0X__)
74# define RTASSERT_HAVE_STATIC_ASSERT
75#endif
76#if RT_CLANG_PREREQ(6, 0)
77# if __has_feature(cxx_static_assert) || __has_feature(c_static_assert)
78# define RTASSERT_HAVE_STATIC_ASSERT
79# endif
80#endif
81#ifdef DOXYGEN_RUNNING
82# define RTASSERT_HAVE_STATIC_ASSERT
83#endif
84
85/** @def AssertCompileNS
86 * Asserts that a compile-time expression is true. If it's not break the build.
87 *
88 * This differs from AssertCompile in that it accepts some more expressions
89 * than what C++0x allows - NS = Non-standard.
90 *
91 * @param expr Expression which should be true.
92 */
93#ifdef __GNUC__
94# define AssertCompileNS(expr) extern int RTASSERTVAR[1] __attribute__((__unused__)), RTASSERTVAR[(expr) ? 1 : 0] __attribute__((__unused__))
95#elif defined(__IBMC__) || defined(__IBMCPP__)
96# define AssertCompileNS(expr) extern int RTASSERTVAR[(expr) ? 1 : 0]
97#else
98# define AssertCompileNS(expr) typedef int RTASSERTTYPE[(expr) ? 1 : 0]
99#endif
100
101/** @def AssertCompile
102 * Asserts that a C++0x compile-time expression is true. If it's not break the
103 * build.
104 * @param expr Expression which should be true.
105 */
106#ifdef RTASSERT_HAVE_STATIC_ASSERT
107# define AssertCompile(expr) static_assert(!!(expr), #expr)
108#else
109# define AssertCompile(expr) AssertCompileNS(expr)
110#endif
111
112/** @def RTASSERT_OFFSET_OF()
113 * A offsetof() macro suitable for compile time assertions.
114 * Both GCC v4 and VisualAge for C++ v3.08 has trouble using RT_OFFSETOF.
115 */
116#if defined(__GNUC__)
117# if __GNUC__ >= 4
118# define RTASSERT_OFFSET_OF(a_Type, a_Member) __builtin_offsetof(a_Type, a_Member)
119# else
120# define RTASSERT_OFFSET_OF(a_Type, a_Member) RT_OFFSETOF(a_Type, a_Member)
121# endif
122#elif (defined(__IBMC__) || defined(__IBMCPP__)) && defined(RT_OS_OS2)
123# define RTASSERT_OFFSET_OF(a_Type, a_Member) __offsetof(a_Type, a_Member)
124#elif (defined(__WATCOMC__) && defined(__cplusplus))
125# define RTASSERT_OFFSET_OF(a_Type, a_Member) __offsetof(a_Type, a_Member)
126#else
127# define RTASSERT_OFFSET_OF(a_Type, a_Member) RT_OFFSETOF(a_Type, a_Member)
128#endif
129
130
131/** @def AssertCompileSize
132 * Asserts a size at compile.
133 * @param type The type.
134 * @param size The expected type size.
135 */
136#define AssertCompileSize(type, size) \
137 AssertCompile(sizeof(type) == (size))
138
139/** @def AssertCompileSizeAlignment
140 * Asserts a size alignment at compile.
141 * @param type The type.
142 * @param align The size alignment to assert.
143 */
144#define AssertCompileSizeAlignment(type, align) \
145 AssertCompile(!(sizeof(type) & ((align) - 1)))
146
147/** @def AssertCompileMemberSize
148 * Asserts a member offset alignment at compile.
149 * @param type The type.
150 * @param member The member.
151 * @param size The member size to assert.
152 */
153#define AssertCompileMemberSize(type, member, size) \
154 AssertCompile(RT_SIZEOFMEMB(type, member) == (size))
155
156/** @def AssertCompileMemberSizeAlignment
157 * Asserts a member size alignment at compile.
158 * @param type The type.
159 * @param member The member.
160 * @param align The member size alignment to assert.
161 */
162#define AssertCompileMemberSizeAlignment(type, member, align) \
163 AssertCompile(!(RT_SIZEOFMEMB(type, member) & ((align) - 1)))
164
165/** @def AssertCompileMemberAlignment
166 * Asserts a member offset alignment at compile.
167 * @param type The type.
168 * @param member The member.
169 * @param align The member offset alignment to assert.
170 */
171#define AssertCompileMemberAlignment(type, member, align) \
172 AssertCompile(!(RTASSERT_OFFSET_OF(type, member) & ((align) - 1)))
173
174/** @def AssertCompileMemberOffset
175 * Asserts an offset of a structure member at compile.
176 * @param type The type.
177 * @param member The member.
178 * @param off The expected offset.
179 */
180#define AssertCompileMemberOffset(type, member, off) \
181 AssertCompile(RTASSERT_OFFSET_OF(type, member) == (off))
182
183/** @def AssertCompile2MemberOffsets
184 * Asserts that two (sub-structure) members in union have the same offset.
185 * @param type The type.
186 * @param member1 The first member.
187 * @param member2 The second member.
188 */
189#define AssertCompile2MemberOffsets(type, member1, member2) \
190 AssertCompile(RTASSERT_OFFSET_OF(type, member1) == RTASSERT_OFFSET_OF(type, member2))
191
192/** @def AssertCompileAdjacentMembers
193 * Asserts that two structure members are adjacent.
194 * @param type The type.
195 * @param member1 The first member.
196 * @param member2 The second member.
197 */
198#define AssertCompileAdjacentMembers(type, member1, member2) \
199 AssertCompile(RTASSERT_OFFSET_OF(type, member1) + RT_SIZEOFMEMB(type, member1) == RTASSERT_OFFSET_OF(type, member2))
200
201/** @def AssertCompileMembersAtSameOffset
202 * Asserts that members of two different structures are at the same offset.
203 * @param type1 The first type.
204 * @param member1 The first member.
205 * @param type2 The second type.
206 * @param member2 The second member.
207 */
208#define AssertCompileMembersAtSameOffset(type1, member1, type2, member2) \
209 AssertCompile(RTASSERT_OFFSET_OF(type1, member1) == RTASSERT_OFFSET_OF(type2, member2))
210
211/** @def AssertCompileMembersSameSize
212 * Asserts that members of two different structures have the same size.
213 * @param type1 The first type.
214 * @param member1 The first member.
215 * @param type2 The second type.
216 * @param member2 The second member.
217 */
218#define AssertCompileMembersSameSize(type1, member1, type2, member2) \
219 AssertCompile(RT_SIZEOFMEMB(type1, member1) == RT_SIZEOFMEMB(type2, member2))
220
221/** @def AssertCompileMembersSameSizeAndOffset
222 * Asserts that members of two different structures have the same size and are
223 * at the same offset.
224 * @param type1 The first type.
225 * @param member1 The first member.
226 * @param type2 The second type.
227 * @param member2 The second member.
228 */
229#define AssertCompileMembersSameSizeAndOffset(type1, member1, type2, member2) \
230 AssertCompile( RTASSERT_OFFSET_OF(type1, member1) == RTASSERT_OFFSET_OF(type2, member2) \
231 && RT_SIZEOFMEMB(type1, member1) == RT_SIZEOFMEMB(type2, member2))
232
233/** @} */
234
235#endif
236
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette