1 #ifndef _DEDEFS_H
2 #define _DEDEFS_H
3 /*-------------------------------------------------------------------------
4  * drawElements Base Portability Library
5  * -------------------------------------
6  *
7  * Copyright 2014 The Android Open Source Project
8  *
9  * Licensed under the Apache License, Version 2.0 (the "License");
10  * you may not use this file except in compliance with the License.
11  * You may obtain a copy of the License at
12  *
13  *      http://www.apache.org/licenses/LICENSE-2.0
14  *
15  * Unless required by applicable law or agreed to in writing, software
16  * distributed under the License is distributed on an "AS IS" BASIS,
17  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18  * See the License for the specific language governing permissions and
19  * limitations under the License.
20  *
21  *//*!
22  * \file
23  * \brief Basic portability.
24  *//*--------------------------------------------------------------------*/
25 
26 /* Compilers. */
27 #define DE_COMPILER_VANILLA	0		/*!< Vanilla compiler. Used for disabling all platform-specific optimizations.	*/
28 #define DE_COMPILER_MSC		1		/*!< Microsoft Visual Studio.													*/
29 #define DE_COMPILER_GCC     2		/*!< Gnu C Compiler.															*/
30 #define DE_COMPILER_CLANG   3		/*!< LLVM Clang Compiler.														*/
31 
32 /* Compiler detection. */
33 #if defined(_MSC_VER)
34 #	define DE_DETAIL_DETECTED_COMPILER DE_COMPILER_MSC
35 #elif defined(__clang__)
36 #	define DE_DETAIL_DETECTED_COMPILER DE_COMPILER_CLANG
37 #elif defined(__GNUC__)
38 #   define DE_DETAIL_DETECTED_COMPILER DE_COMPILER_GCC
39 #else
40 	/* DE_DETAIL_DETECTED_COMPILER not set */
41 #endif
42 
43 /* Compiler setting. */
44 #if defined(DE_COMPILER)
45 	/* Allow definitions from outside, but fail early if it conflicts with our detection */
46 #	if defined(DE_DETAIL_DETECTED_COMPILER) && (DE_COMPILER != DE_DETAIL_DETECTED_COMPILER)
47 		/* conflict, print a nice error messages for the most common misconfigs,
48 		 * GCC and Clang, and a generic for other conflicts.
49 		 */
50 #		if (DE_DETAIL_DETECTED_COMPILER == DE_COMPILER_CLANG) && (DE_COMPILER == DE_COMPILER_GCC)
51 #			error Detected compiler is Clang, but got DE_COMPILER == DE_COMPILER_GCC
52 #		elif (DE_DETAIL_DETECTED_COMPILER == DE_COMPILER_GCC) && (DE_COMPILER == DE_COMPILER_CLANG)
53 #			error Detected compiler is GCC, but got DE_COMPILER == DE_COMPILER_CLANG
54 #		else
55 #			error Detected compiler does not match the supplied compiler.
56 #		endif
57 #	endif
58 	/* Clear autodetect vars. */
59 #	if defined(DE_DETAIL_DETECTED_COMPILER)
60 #		undef DE_DETAIL_DETECTED_COMPILER
61 #	endif
62 #else
63 	/* No definition given from outside, try to autodetect */
64 #	if defined(DE_DETAIL_DETECTED_COMPILER)
65 #		define DE_COMPILER DE_DETAIL_DETECTED_COMPILER /*!< Compiler identification (set to one of DE_COMPILER_*). */
66 #	else
67 #		error Unknown compiler.
68 #	endif
69 #endif
70 
71 /* Operating systems. */
72 #define DE_OS_VANILLA	0			/*!< Vanilla OS.								*/
73 #define DE_OS_WIN32		1			/*!< Microsoft Windows desktop					*/
74 #define DE_OS_UNIX      2			/*!< Unix (or compatible)						*/
75 #define DE_OS_WINCE		3			/*!< Windows CE, Windows Mobile or Pocket PC	*/
76 #define DE_OS_OSX		4			/*!< Mac OS X									*/
77 #define DE_OS_ANDROID	5			/*!< Android									*/
78 #define DE_OS_SYMBIAN	6			/*!< Symbian OS									*/
79 #define DE_OS_IOS		7			/*!< iOS										*/
80 
81 /* OS detection (set to one of DE_OS_*). */
82 #if defined(DE_OS)
83 	/* Allow definitions from outside. */
84 #elif defined(__ANDROID__)
85 #	define DE_OS DE_OS_ANDROID
86 #elif defined(_WIN32_WCE) || defined(UNDER_CE)
87 #	define DE_OS DE_OS_WINCE
88 #elif defined(_WIN32)
89 #	define DE_OS DE_OS_WIN32
90 #elif defined(__unix__) || defined(__linux) || defined(__linux__)
91 #   define DE_OS DE_OS_UNIX
92 #elif defined(__APPLE__)
93 #	define DE_OS DE_OS_OSX
94 #elif defined(__EPOC32__)
95 #	define DE_OS DE_OS_SYMBIAN
96 #else
97 #	error Unknown operating system.
98 #endif
99 
100 /* CPUs */
101 #define DE_CPU_VANILLA	0
102 #define DE_CPU_X86		1
103 #define DE_CPU_ARM		2
104 #define DE_CPU_X86_64	3
105 #define DE_CPU_ARM_64	4
106 #define DE_CPU_MIPS		5
107 #define DE_CPU_MIPS_64	6
108 
109 /* CPU detection. */
110 #if defined(DE_CPU)
111 	/* Allow definitions from outside. */
112 #elif defined(__aarch64__)
113 #	define DE_CPU DE_CPU_ARM_64
114 #elif defined(__arm__) || defined(__ARM__) || defined(__ARM_NEON__) || defined(ARM_BUILD)
115 #	define DE_CPU DE_CPU_ARM
116 #elif defined(_M_X64) || defined(__x86_64__) || defined(__amd64__)
117 #	define DE_CPU DE_CPU_X86_64
118 #elif defined(__i386__) || defined(_M_X86) || defined(_M_IX86) || defined(X86_BUILD)
119 #	define DE_CPU DE_CPU_X86
120 #elif defined(__mips__) && ((__mips) == 32)
121 #	define DE_CPU DE_CPU_MIPS
122 #elif defined(__mips__) && ((__mips) == 64)
123 #	define DE_CPU DE_CPU_MIPS_64
124 #else
125 #	error Unknown CPU.
126 #endif
127 
128 /* Endianness */
129 #define DE_BIG_ENDIAN		0
130 #define DE_LITTLE_ENDIAN	1
131 
132 #if defined(DE_ENDIANNESS)
133 	/* Allow definitions from outside. */
134 #elif (DE_CPU == DE_CPU_X86) || (DE_CPU == DE_CPU_X86_64)
135 	/* "detect" x86(_64) endianness */
136 #	define DE_ENDIANNESS DE_LITTLE_ENDIAN
137 #elif ((DE_CPU == DE_CPU_MIPS) || (DE_CPU == DE_CPU_MIPS_64))
138 	/* detect mips endianness using platform specific macros */
139 #	if defined(__MIPSEB__) && !defined(__MIPSEL__)
140 #		define DE_ENDIANNESS DE_BIG_ENDIAN
141 #	elif !defined(__MIPSEB__) && defined(__MIPSEL__)
142 #		define DE_ENDIANNESS DE_LITTLE_ENDIAN
143 #	else
144 #		error Invalid MIPS endianness.
145 #	endif
146 #elif defined(__BYTE_ORDER__) && defined(__ORDER_LITTLE_ENDIAN__) && (__BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__)
147 #	define DE_ENDIANNESS DE_LITTLE_ENDIAN
148 #elif defined(__BYTE_ORDER__) && defined(__ORDER_BIG_ENDIAN__) && (__BYTE_ORDER__ == __ORDER_BIG_ENDIAN__)
149 #	define DE_ENDIANNESS DE_BIG_ENDIAN
150 #else
151 #	error Unknown endianness.
152 #endif
153 
154 /* Sanity */
155 #if ((DE_CPU == DE_CPU_X86) || (DE_CPU == DE_CPU_X86_64)) && (DE_ENDIANNESS == DE_BIG_ENDIAN)
156 #	error Invalid x86(_64) endianness.
157 #endif
158 
159 /* Sized data types. */
160 typedef signed char			deInt8;
161 typedef signed short		deInt16;
162 typedef signed int			deInt32;
163 typedef unsigned char		deUint8;
164 typedef unsigned short		deUint16;
165 typedef unsigned int		deUint32;
166 
167 #if (DE_COMPILER == DE_COMPILER_MSC)
168 	typedef signed __int64		deInt64;
169 	typedef unsigned __int64	deUint64;
170 
171 #	if (DE_OS == DE_OS_WINCE)
172 #		include <basetsd.h>
173 		typedef INT_PTR			deIntptr;
174 		typedef UINT_PTR		deUintptr;
175 #	elif (DE_OS == DE_OS_WIN32)
176 #		include <crtdefs.h>
177 		typedef intptr_t		deIntptr;
178 		typedef uintptr_t		deUintptr;
179 #	else
180 #		error Define intptr types.
181 #	endif
182 
183 #elif (DE_COMPILER == DE_COMPILER_GCC) || (DE_COMPILER == DE_COMPILER_CLANG)
184 	/* \note stddef.h is needed for size_t definition. */
185 #	include <stddef.h>
186 #	include <stdint.h>
187 	typedef int64_t				deInt64;
188 	typedef uint64_t			deUint64;
189 	typedef intptr_t			deIntptr;
190 	typedef uintptr_t			deUintptr;
191 #else
192 #	error Define 64-bit and intptr types.
193 #endif
194 
195 /** Boolean type. */
196 typedef int deBool;
197 #define DE_TRUE		1		/*!< True value for deBool.		*/
198 #define DE_FALSE	0		/*!< False value for deBool.	*/
199 
200 /* Null pointer. */
201 #if defined(__cplusplus)
202 #	define DE_NULL 0
203 #else
204 #	define DE_NULL ((void*)0)		/*!< Null pointer.				*/
205 #endif
206 
207 /* Function pointer type. */
208 typedef void (*deFunctionPtr) (void);
209 
210 /* Use DE_PTR_TYPE(T) instead of T* in macros to avoid clang-tidy warning. */
211 #define DE_PTR_TYPE(T) T*  /* NOLINT(T) */
212 
213 /* Debug macro. */
214 #if defined(DE_DEBUG)
215 	/* Already defined from outside. */
216 #else
217 #	if (DE_COMPILER != DE_COMPILER_GCC)
218 #		if defined(_DEBUG)
219 #			define DE_DEBUG					/*!< Debug build enabled? Usage: #if defined(DE_DEBUG).	*/
220 #		endif
221 #	elif (DE_COMPILER == DE_COMPILER_GCC) || (DE_COMPILER == DE_COMPILER_CLANG)
222 #		if !defined(NDEBUG)
223 #			define DE_DEBUG
224 #		endif
225 #	endif
226 #endif
227 
228 /* Debug code macro. */
229 #if defined(DE_DEBUG)
230 #	define DE_DEBUG_CODE(X) X
231 #else
232 #	define DE_DEBUG_CODE(X)
233 #endif
234 
235 /* Inline. */
236 #if (DE_COMPILER == DE_COMPILER_MSC)
237 #	define DE_INLINE __forceinline
238 #elif (DE_COMPILER == DE_COMPILER_GCC) || (DE_COMPILER == DE_COMPILER_CLANG)
239 #   define DE_INLINE static __inline__
240 #else
241 #	define DE_INLINE inline			/*!< Function inline.		*/
242 #endif
243 
244 /* DE_DEV_BUILD -- only define when building on a development machine. */
245 #if !defined(DE_DEV_BUILD)
246 #	if (DE_COMPILER == DE_COMPILER_MSC)
247 #		define DE_DEV_BUILD
248 #	endif
249 #endif
250 
251 /* DE_VALGRIND_BUILD -- define this in makefile if support for Valgrind is wanted. */
252 /*#define DE_VALGRIND_BUILD*/
253 
254 /** Length of array. C++ version does compile time check that passed value is an array reference. */
255 #if defined(__cplusplus) && (DE_COMPILER == DE_COMPILER_MSC)
256 	template <typename T, size_t N> char (&deArraySizeHelper(T (&array)[N]))[N];
257 #	define DE_LENGTH_OF_ARRAY(ARRAY) ((int)(sizeof(deArraySizeHelper(ARRAY))))
258 #else
259 #	define DE_LENGTH_OF_ARRAY(ARRAY) ((int)(sizeof(ARRAY) / sizeof((ARRAY)[0])))
260 #endif
261 
262 #ifdef __cplusplus
263 extern "C" {
264 #endif
265 
266 /* Assertion macro family. */
267 void deAssertFail(const char* reason, const char* file, int line);
268 
269 /* Assertion failure callback. Requires DE_ASSERT_FAILURE_CALLBACK to be defined or otherwise has no effect. */
270 typedef void (*deAssertFailureCallbackFunc) (const char* reason, const char* file, int line);
271 void deSetAssertFailureCallback (deAssertFailureCallbackFunc callback);
272 
deGetFalse(void)273 DE_INLINE deBool deGetFalse (void) { return DE_FALSE; }
deGetTrue(void)274 DE_INLINE deBool deGetTrue (void) { return DE_TRUE; }
275 
276 /* Assertion macro. */
277 #if defined(DE_DEBUG) && !defined(DE_COVERAGE_BUILD)
278 #	define DE_ASSERT(X) do { if ((!deGetFalse() && (X)) ? DE_FALSE : DE_TRUE) deAssertFail(#X, __FILE__, __LINE__); } while(deGetFalse())
279 #else
280 #	define DE_ASSERT(X) /*@ -noeffect*/ ((void)0)	/*!< Assertion macro. */
281 #endif
282 
283 /* Verify macro. Behaves like assert in debug build, but executes statement in release build. */
284 #if defined(DE_DEBUG)
285 #	define DE_VERIFY(X) do { if ((!deGetFalse() && (X)) ? DE_FALSE : DE_TRUE) deAssertFail(#X, __FILE__, __LINE__); } while(deGetFalse())
286 #else
287 #	define DE_VERIFY(X) X
288 #endif
289 
290 /* Fatal macro. */
291 #if defined(DE_DEBUG) && !defined(DE_COVERAGE_BUILD)
292 #	define DE_FATAL(MSG) do { deAssertFail("" /* force to string literal */ MSG, __FILE__, __LINE__); } while(deGetFalse())
293 #else
294 #	define DE_FATAL(MSG) /*@ -noeffect*/ ((void)0)	/*!< Fatal macro. */
295 #endif
296 
297 /** Test assert macro for use in testers (same as DE_ASSERT, but always enabled). */
298 #define DE_TEST_ASSERT(X) do { if ((!deGetFalse() && (X)) ? DE_FALSE : DE_TRUE) deAssertFail(#X, __FILE__, __LINE__); } while(deGetFalse())
299 
300 #if (DE_COMPILER == DE_COMPILER_GCC) || (DE_COMPILER == DE_COMPILER_CLANG)
301 	/* GCC 4.8 and newer warns about unused typedefs. */
302 #	define DE_UNUSED_TYPEDEF_ATTR __attribute__((unused))
303 #else
304 #	define DE_UNUSED_TYPEDEF_ATTR
305 #endif
306 
307 /** Compile-time assertion macro. */
308 #define DE_STATIC_ASSERT(X)						typedef char DE_UNIQUE_NAME[(X) ? 1 : -1] DE_UNUSED_TYPEDEF_ATTR
309 #define DE_HEADER_STATIC_ASSERT(HEADERTOKEN, X)	typedef char DE_HEADER_UNIQUE_NAME(HEADERTOKEN)[(X) ? 1 : -1] DE_UNUSED_TYPEDEF_ATTR
310 
311 #define DE_UNIQUE_NAME						DE_MAKE_NAME(__LINE__, hoax)
312 #define DE_HEADER_UNIQUE_NAME(HEADERTOKEN)	DE_MAKE_NAME(__LINE__, HEADERTOKEN)
313 #define DE_MAKE_NAME(line, token) DE_MAKE_NAME2(line, token)
314 #define DE_MAKE_NAME2(line, token) _static_assert_##line##_##token
315 
316 /** Software breakpoint. */
317 #if (DE_CPU == DE_CPU_X86) && (DE_COMPILER == DE_COMPILER_MSC)
318 #	define DE_BREAKPOINT() do { printf("Software breakpoint encountered in %s, line %d\n", __FILE__, __LINE__); __asm { int 3 } } while (deGetFalse())
319 #elif (DE_CPU == DE_CPU_X86_64) && (DE_COMPILER == DE_COMPILER_MSC)
320 #	define DE_BREAKPOINT() do { printf("Software breakpoint encountered in %s, line %d\n", __FILE__, __LINE__); __debugbreak(); } while (deGetFalse())
321 #elif (DE_CPU == DE_CPU_ARM) && (DE_COMPILER == DE_COMPILER_GCC)
322 #	define DE_BREAKPOINT() do { printf("Software breakpoint encountered in %s, line %d\n", __FILE__, __LINE__); __asm__ __volatile__ ( "bkpt #3" ); } while (deGetFalse())
323 #elif (DE_CPU == DE_CPU_ARM) && (DE_COMPILER == DE_COMPILER_MSC)
324 #	define DE_BREAKPOINT() do { printf("Software breakpoint encountered in %s, line %d\n", __FILE__, __LINE__); DebugBreak(); } while (deGetFalse())
325 #else
326 #	define DE_BREAKPOINT() DE_FATAL("Software breakpoint encountered!")
327 #endif
328 
329 /** Swap two values. */
330 #define DE_SWAP(TYPE, A, B) do { TYPE _tmp_ = (A); (A) = (B); (B) = _tmp_; } while(deGetFalse())
331 
332 /** Offset of a struct member. */
333 #define DE_OFFSET_OF(STRUCT, MEMBER) ((deUint32)(deUintptr)(deUint8*)&(((STRUCT*)0)->MEMBER))
334 
335 /* Pointer size. */
336 #if defined(DE_PTR_SIZE)
337 	/* nada */
338 #elif defined(_M_X64) || defined(__x86_64__) || defined(__amd64__) || defined(__aarch64__) || (defined(__mips) && ((__mips) == 64)) || defined(_LP64) || defined(__LP64__)
339 #	define DE_PTR_SIZE 8
340 #else
341 #	define DE_PTR_SIZE 4	/* default to 32-bit */
342 #endif
343 
344 /** Unreferenced variable silencing. */
345 #define DE_UNREF(VAR) ((void)(VAR))
346 
347 /** DE_BEGIN_EXTERN_C and DE_END_EXTERN_C. */
348 #if defined(__cplusplus)
349 #	define DE_BEGIN_EXTERN_C extern "C" {
350 #	define DE_END_EXTERN_C }
351 #else
352 #	define DE_BEGIN_EXTERN_C
353 #	define DE_END_EXTERN_C
354 #endif
355 
356 /** DE_NULL_STATEMENT */
357 #if defined(DE_DEBUG)
358 #	define DE_NULL_STATEMENT do {} while (deGetFalse())
359 #else
360 #	define DE_NULL_STATEMENT (void)0
361 #endif
362 
363 /** GCC format string attributes */
364 #if (DE_COMPILER == DE_COMPILER_GCC) || (DE_COMPILER == DE_COMPILER_CLANG)
365 #	define DE_PRINTF_FUNC_ATTR(FORMAT_STRING, FIRST_ARG) __attribute__ ((format(printf, FORMAT_STRING, FIRST_ARG)))
366 #else
367 #	define DE_PRINTF_FUNC_ATTR(FORMAT_STRING, FIRST_ARG)
368 #endif
369 
370 /** Potentially unused func attribute to silence warnings from C templates. */
371 #if (DE_COMPILER == DE_COMPILER_GCC) || (DE_COMPILER == DE_COMPILER_CLANG)
372 #	define DE_UNUSED_FUNCTION __attribute__((unused))
373 #else
374 #	define DE_UNUSED_FUNCTION
375 #endif
376 
377 #ifdef __cplusplus
378 }
379 #endif
380 
381 #endif /* _DEDEFS_H */
382