[MESA/OPENGL32]
[reactos.git] / reactos / dll / opengl / mesa / src / mesa / main / compiler.h
1 /*
2 * Mesa 3-D graphics library
3 * Version: 7.5
4 *
5 * Copyright (C) 1999-2008 Brian Paul All Rights Reserved.
6 * Copyright (C) 2009 VMware, Inc. All Rights Reserved.
7 *
8 * Permission is hereby granted, free of charge, to any person obtaining a
9 * copy of this software and associated documentation files (the "Software"),
10 * to deal in the Software without restriction, including without limitation
11 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
12 * and/or sell copies of the Software, and to permit persons to whom the
13 * Software is furnished to do so, subject to the following conditions:
14 *
15 * The above copyright notice and this permission notice shall be included
16 * in all copies or substantial portions of the Software.
17 *
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
21 * BRIAN PAUL BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
22 * AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
23 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
24 */
25
26
27 /**
28 * \file compiler.h
29 * Compiler-related stuff.
30 */
31
32
33 #ifndef COMPILER_H
34 #define COMPILER_H
35
36
37 #include <assert.h>
38 #include <ctype.h>
39 #if defined(__alpha__) && defined(CCPML)
40 #include <cpml.h> /* use Compaq's Fast Math Library on Alpha */
41 #else
42 #include <math.h>
43 #endif
44 #include <limits.h>
45 #include <stdlib.h>
46 #include <stdio.h>
47 #include <string.h>
48 #include <float.h>
49 #include <stdarg.h>
50
51
52 #ifdef __cplusplus
53 extern "C" {
54 #endif
55
56
57 /**
58 * Get standard integer types
59 */
60 #include <stdint.h>
61
62
63 /**
64 * Sun compilers define __i386 instead of the gcc-style __i386__
65 */
66 #ifdef __SUNPRO_C
67 # if !defined(__i386__) && defined(__i386)
68 # define __i386__
69 # elif !defined(__amd64__) && defined(__amd64)
70 # define __amd64__
71 # elif !defined(__sparc__) && defined(__sparc)
72 # define __sparc__
73 # endif
74 # if !defined(__volatile)
75 # define __volatile volatile
76 # endif
77 #endif
78
79
80 /**
81 * finite macro.
82 */
83 #if defined(_MSC_VER)
84 # define finite _finite
85 #elif defined(__WATCOMC__)
86 # define finite _finite
87 #endif
88
89
90 /**
91 * Disable assorted warnings
92 */
93 #if !defined(OPENSTEP) && (defined(__WIN32__) && !defined(__CYGWIN__)) && !defined(BUILD_FOR_SNAP)
94 # if !defined(__GNUC__) /* mingw environment */
95 # pragma warning( disable : 4068 ) /* unknown pragma */
96 # pragma warning( disable : 4710 ) /* function 'foo' not inlined */
97 # pragma warning( disable : 4711 ) /* function 'foo' selected for automatic inline expansion */
98 # pragma warning( disable : 4127 ) /* conditional expression is constant */
99 # if defined(MESA_MINWARN)
100 # pragma warning( disable : 4244 ) /* '=' : conversion from 'const double ' to 'float ', possible loss of data */
101 # pragma warning( disable : 4018 ) /* '<' : signed/unsigned mismatch */
102 # pragma warning( disable : 4305 ) /* '=' : truncation from 'const double ' to 'float ' */
103 # pragma warning( disable : 4550 ) /* 'function' undefined; assuming extern returning int */
104 # pragma warning( disable : 4761 ) /* integral size mismatch in argument; conversion supplied */
105 # endif
106 # endif
107 #endif
108 #if defined(__WATCOMC__)
109 # pragma disable_message(201) /* Disable unreachable code warnings */
110 #endif
111
112
113
114 /**
115 * Function inlining
116 */
117 #ifndef inline
118 # ifdef __cplusplus
119 /* C++ supports inline keyword */
120 # elif defined(__GNUC__)
121 # define inline __inline__
122 # elif defined(_MSC_VER)
123 # define inline __inline
124 # elif defined(__ICL)
125 # define inline __inline
126 # elif defined(__INTEL_COMPILER)
127 /* Intel compiler supports inline keyword */
128 # elif defined(__WATCOMC__) && (__WATCOMC__ >= 1100)
129 # define inline __inline
130 # elif defined(__SUNPRO_C) && defined(__C99FEATURES__)
131 /* C99 supports inline keyword */
132 # elif (__STDC_VERSION__ >= 199901L)
133 /* C99 supports inline keyword */
134 # else
135 # define inline
136 # endif
137 #endif
138 #ifndef INLINE
139 # define INLINE inline
140 #endif
141
142
143 /**
144 * PUBLIC/USED macros
145 *
146 * If we build the library with gcc's -fvisibility=hidden flag, we'll
147 * use the PUBLIC macro to mark functions that are to be exported.
148 *
149 * We also need to define a USED attribute, so the optimizer doesn't
150 * inline a static function that we later use in an alias. - ajax
151 */
152 #ifndef PUBLIC
153 # if (defined(__GNUC__) && __GNUC__ >= 4) || (defined(__SUNPRO_C) && (__SUNPRO_C >= 0x590))
154 # define PUBLIC __attribute__((visibility("default")))
155 # define USED __attribute__((used))
156 # else
157 # define PUBLIC
158 # define USED
159 # endif
160 #endif
161
162
163 /**
164 * Some compilers don't like some of Mesa's const usage. In those places use
165 * CONST instead of const. Pass -DNO_CONST to compilers where this matters.
166 */
167 #ifdef NO_CONST
168 # define CONST
169 #else
170 # define CONST const
171 #endif
172
173
174 /**
175 * __builtin_expect macros
176 */
177 #if !defined(__GNUC__)
178 # define __builtin_expect(x, y) (x)
179 #endif
180
181 #ifndef likely
182 # ifdef __GNUC__
183 # define likely(x) __builtin_expect(!!(x), 1)
184 # define unlikely(x) __builtin_expect(!!(x), 0)
185 # else
186 # define likely(x) (x)
187 # define unlikely(x) (x)
188 # endif
189 #endif
190
191 /**
192 * The __FUNCTION__ gcc variable is generally only used for debugging.
193 * If we're not using gcc, define __FUNCTION__ as a cpp symbol here.
194 * Don't define it if using a newer Windows compiler.
195 */
196 #ifndef __FUNCTION__
197 # if defined(__VMS)
198 # define __FUNCTION__ "VMS$NL:"
199 # elif !defined(__GNUC__) && !defined(__xlC__) && \
200 (!defined(_MSC_VER) || _MSC_VER < 1300)
201 # if (__STDC_VERSION__ >= 199901L) /* C99 */ || \
202 (defined(__SUNPRO_C) && defined(__C99FEATURES__))
203 # define __FUNCTION__ __func__
204 # else
205 # define __FUNCTION__ "<unknown>"
206 # endif
207 # endif
208 #endif
209 #ifndef __func__
210 # if (__STDC_VERSION__ >= 199901L) || \
211 (defined(__SUNPRO_C) && defined(__C99FEATURES__))
212 /* __func__ is part of C99 */
213 # elif defined(_MSC_VER)
214 # if _MSC_VER >= 1300
215 # define __func__ __FUNCTION__
216 # else
217 # define __func__ "<unknown>"
218 # endif
219 # endif
220 #endif
221
222
223 /**
224 * Either define MESA_BIG_ENDIAN or MESA_LITTLE_ENDIAN, and CPU_TO_LE32.
225 * Do not use these unless absolutely necessary!
226 * Try to use a runtime test instead.
227 * For now, only used by some DRI hardware drivers for color/texel packing.
228 */
229 #if defined(BYTE_ORDER) && defined(BIG_ENDIAN) && BYTE_ORDER == BIG_ENDIAN
230 #if defined(__linux__)
231 #include <byteswap.h>
232 #define CPU_TO_LE32( x ) bswap_32( x )
233 #elif defined(__APPLE__)
234 #include <CoreFoundation/CFByteOrder.h>
235 #define CPU_TO_LE32( x ) CFSwapInt32HostToLittle( x )
236 #elif (defined(_AIX) || defined(__blrts))
237 static INLINE GLuint CPU_TO_LE32(GLuint x)
238 {
239 return (((x & 0x000000ff) << 24) |
240 ((x & 0x0000ff00) << 8) |
241 ((x & 0x00ff0000) >> 8) |
242 ((x & 0xff000000) >> 24));
243 }
244 #else /*__linux__ */
245 #include <sys/endian.h>
246 #define CPU_TO_LE32( x ) bswap32( x )
247 #endif /*__linux__*/
248 #define MESA_BIG_ENDIAN 1
249 #else
250 #define CPU_TO_LE32( x ) ( x )
251 #define MESA_LITTLE_ENDIAN 1
252 #endif
253 #define LE32_TO_CPU( x ) CPU_TO_LE32( x )
254
255
256
257 #if !defined(CAPI) && defined(WIN32) && !defined(BUILD_FOR_SNAP)
258 #define CAPI _cdecl
259 #endif
260
261
262 /**
263 * Create a macro so that asm functions can be linked into compilers other
264 * than GNU C
265 */
266 #ifndef _ASMAPI
267 #if defined(WIN32) && !defined(BUILD_FOR_SNAP)/* was: !defined( __GNUC__ ) && !defined( VMS ) && !defined( __INTEL_COMPILER )*/
268 #define _ASMAPI __cdecl
269 #else
270 #define _ASMAPI
271 #endif
272 #ifdef PTR_DECL_IN_FRONT
273 #define _ASMAPIP * _ASMAPI
274 #else
275 #define _ASMAPIP _ASMAPI *
276 #endif
277 #endif
278
279 #ifdef USE_X86_ASM
280 #define _NORMAPI _ASMAPI
281 #define _NORMAPIP _ASMAPIP
282 #else
283 #define _NORMAPI
284 #define _NORMAPIP *
285 #endif
286
287
288 /* This is a macro on IRIX */
289 #ifdef _P
290 #undef _P
291 #endif
292
293
294 /* Turn off macro checking systems used by other libraries */
295 #ifdef CHECK
296 #undef CHECK
297 #endif
298
299
300 /**
301 * ASSERT macro
302 */
303 #if !defined(_WIN32_WCE)
304 #if defined(BUILD_FOR_SNAP) && defined(CHECKED)
305 # define ASSERT(X) _CHECK(X)
306 #elif defined(DEBUG)
307 # define ASSERT(X) assert(X)
308 #else
309 # define ASSERT(X)
310 #endif
311 #endif
312
313
314 /**
315 * Static (compile-time) assertion.
316 * Basically, use COND to dimension an array. If COND is false/zero the
317 * array size will be -1 and we'll get a compilation error.
318 */
319 #define STATIC_ASSERT(COND) \
320 do { \
321 typedef int static_assertion_failed[(!!(COND))*2-1]; \
322 } while (0)
323
324
325 #if (__GNUC__ >= 3)
326 #define PRINTFLIKE(f, a) __attribute__ ((format(__printf__, f, a)))
327 #else
328 #define PRINTFLIKE(f, a)
329 #endif
330
331 #ifndef NULL
332 #define NULL 0
333 #endif
334
335
336 /**
337 * LONGSTRING macro
338 * gcc -pedantic warns about long string literals, LONGSTRING silences that.
339 */
340 #if !defined(__GNUC__)
341 # define LONGSTRING
342 #else
343 # define LONGSTRING __extension__
344 #endif
345
346
347 #ifndef M_PI
348 #define M_PI (3.14159265358979323846)
349 #endif
350
351 #ifndef M_E
352 #define M_E (2.7182818284590452354)
353 #endif
354
355 #ifndef M_LOG2E
356 #define M_LOG2E (1.4426950408889634074)
357 #endif
358
359 #ifndef ONE_DIV_SQRT_LN2
360 #define ONE_DIV_SQRT_LN2 (1.201122408786449815)
361 #endif
362
363 #ifndef FLT_MAX_EXP
364 #define FLT_MAX_EXP 128
365 #endif
366
367
368 /**
369 * USE_IEEE: Determine if we're using IEEE floating point
370 */
371 #if defined(__i386__) || defined(__386__) || defined(__sparc__) || \
372 defined(__s390x__) || defined(__powerpc__) || \
373 defined(__x86_64__) || \
374 defined(ia64) || defined(__ia64__) || \
375 defined(__hppa__) || defined(hpux) || \
376 defined(__mips) || defined(_MIPS_ARCH) || \
377 defined(__arm__) || \
378 defined(__sh__) || defined(__m32r__) || \
379 (defined(__sun) && defined(_IEEE_754)) || \
380 (defined(__alpha__) && (defined(__IEEE_FLOAT) || !defined(VMS)))
381 #define USE_IEEE
382 #define IEEE_ONE 0x3f800000
383 #endif
384
385
386 /**
387 * START/END_FAST_MATH macros:
388 *
389 * START_FAST_MATH: Set x86 FPU to faster, 32-bit precision mode (and save
390 * original mode to a temporary).
391 * END_FAST_MATH: Restore x86 FPU to original mode.
392 */
393 #if defined(__GNUC__) && defined(__i386__)
394 /*
395 * Set the x86 FPU control word to guarentee only 32 bits of precision
396 * are stored in registers. Allowing the FPU to store more introduces
397 * differences between situations where numbers are pulled out of memory
398 * vs. situations where the compiler is able to optimize register usage.
399 *
400 * In the worst case, we force the compiler to use a memory access to
401 * truncate the float, by specifying the 'volatile' keyword.
402 */
403 /* Hardware default: All exceptions masked, extended double precision,
404 * round to nearest (IEEE compliant):
405 */
406 #define DEFAULT_X86_FPU 0x037f
407 /* All exceptions masked, single precision, round to nearest:
408 */
409 #define FAST_X86_FPU 0x003f
410 /* The fldcw instruction will cause any pending FP exceptions to be
411 * raised prior to entering the block, and we clear any pending
412 * exceptions before exiting the block. Hence, asm code has free
413 * reign over the FPU while in the fast math block.
414 */
415 #if defined(NO_FAST_MATH)
416 #define START_FAST_MATH(x) \
417 do { \
418 static GLuint mask = DEFAULT_X86_FPU; \
419 __asm__ ( "fnstcw %0" : "=m" (*&(x)) ); \
420 __asm__ ( "fldcw %0" : : "m" (mask) ); \
421 } while (0)
422 #else
423 #define START_FAST_MATH(x) \
424 do { \
425 static GLuint mask = FAST_X86_FPU; \
426 __asm__ ( "fnstcw %0" : "=m" (*&(x)) ); \
427 __asm__ ( "fldcw %0" : : "m" (mask) ); \
428 } while (0)
429 #endif
430 /* Restore original FPU mode, and clear any exceptions that may have
431 * occurred in the FAST_MATH block.
432 */
433 #define END_FAST_MATH(x) \
434 do { \
435 __asm__ ( "fnclex ; fldcw %0" : : "m" (*&(x)) ); \
436 } while (0)
437
438 #elif defined(__WATCOMC__) && defined(__386__)
439 #define DEFAULT_X86_FPU 0x037f /* See GCC comments above */
440 #define FAST_X86_FPU 0x003f /* See GCC comments above */
441 void _watcom_start_fast_math(unsigned short *x,unsigned short *mask);
442 #pragma aux _watcom_start_fast_math = \
443 "fnstcw word ptr [eax]" \
444 "fldcw word ptr [ecx]" \
445 parm [eax] [ecx] \
446 modify exact [];
447 void _watcom_end_fast_math(unsigned short *x);
448 #pragma aux _watcom_end_fast_math = \
449 "fnclex" \
450 "fldcw word ptr [eax]" \
451 parm [eax] \
452 modify exact [];
453 #if defined(NO_FAST_MATH)
454 #define START_FAST_MATH(x) \
455 do { \
456 static GLushort mask = DEFAULT_X86_FPU; \
457 _watcom_start_fast_math(&x,&mask); \
458 } while (0)
459 #else
460 #define START_FAST_MATH(x) \
461 do { \
462 static GLushort mask = FAST_X86_FPU; \
463 _watcom_start_fast_math(&x,&mask); \
464 } while (0)
465 #endif
466 #define END_FAST_MATH(x) _watcom_end_fast_math(&x)
467
468 #elif defined(_MSC_VER) && defined(_M_IX86)
469 #define DEFAULT_X86_FPU 0x037f /* See GCC comments above */
470 #define FAST_X86_FPU 0x003f /* See GCC comments above */
471 #if defined(NO_FAST_MATH)
472 #define START_FAST_MATH(x) do {\
473 static GLuint mask = DEFAULT_X86_FPU;\
474 __asm fnstcw word ptr [x]\
475 __asm fldcw word ptr [mask]\
476 } while(0)
477 #else
478 #define START_FAST_MATH(x) do {\
479 static GLuint mask = FAST_X86_FPU;\
480 __asm fnstcw word ptr [x]\
481 __asm fldcw word ptr [mask]\
482 } while(0)
483 #endif
484 #define END_FAST_MATH(x) do {\
485 __asm fnclex\
486 __asm fldcw word ptr [x]\
487 } while(0)
488
489 #else
490 #define START_FAST_MATH(x) x = 0
491 #define END_FAST_MATH(x) (void)(x)
492 #endif
493
494
495 #ifndef Elements
496 #define Elements(x) (sizeof(x)/sizeof(*(x)))
497 #endif
498
499
500
501 #ifdef __cplusplus
502 }
503 #endif
504
505
506 #endif /* COMPILER_H */