Source code of Windows XP (NT5)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

278 lines
9.3 KiB

  1. /***
  2. *stdarg.h - defines ANSI-style macros for variable argument functions
  3. *
  4. * Copyright (c) 1985-2001, Microsoft Corporation. All rights reserved.
  5. *
  6. *Purpose:
  7. * This file defines ANSI-style macros for accessing arguments
  8. * of functions which take a variable number of arguments.
  9. * [ANSI]
  10. *
  11. * [Public]
  12. *
  13. *Revision History:
  14. * 05-03-89 JCR Added _INTERNAL_IFSTRIP for relinc usage
  15. * 08-15-89 GJF Cleanup, now specific to OS/2 2.0 (i.e., 386 flat model)
  16. * 10-30-89 GJF Fixed copyright
  17. * 11-02-89 JCR Changed "DLL" to "_DLL"
  18. * 01-05-90 JCR Added NULL definition
  19. * 03-02-90 GJF Added #ifndef _INC_STDARG stuff. Also, removed some
  20. * (now) useless preprocessor directives.
  21. * 05-29-90 GJF Replaced sizeof() with _INTSIZEOF() and revised the
  22. * va_arg() macro (fixes PTM 60)
  23. * 05-31-90 GJF Revised va_end() macro (propagated 5-25-90 change to
  24. * crt7 version by WAJ)
  25. * 10-30-90 GJF Moved the real definitions into cruntime.h (for NT
  26. * folks) and relinc.sed (to release ANSI compatible
  27. * version). Ugly compromise.
  28. * 08-20-91 JCR C++ and ANSI naming
  29. * 11-01-91 GDP MIPS Compiler support. Moved real definitions back here
  30. * 10-16-92 SKS Replaced "#ifdef i386" with "#ifdef _M_IX86".
  31. * 11-03-92 GJF Fixed several conditionals, dropped _DOSX32_ support.
  32. * 01-03-93 SRW Fold in ALPHA changes
  33. * 01-09-93 SRW Remove usage of MIPS and ALPHA to conform to ANSI
  34. * Use _MIPS_ and _ALPHA_ instead.
  35. * 01-25-93 GJF Fix va_list definition.
  36. * 10-04-93 SRW Fix ifdefs for MIPS and ALPHA to only check for _M_?????? defines
  37. * 10-12-93 GJF Merged NT version into Cuda version. Also, replaced
  38. * _ALPHA_ with _M_ALPHA.
  39. * 11-11-93 GJF Minor cosmetic changes.
  40. * 04-05-94 SKS Add prototype of __builtin_va_start for ALPHA
  41. * 10-02-94 BWT Add PPC support.
  42. * 11-03-94 GJF Ensure 8 byte alignment.
  43. * 12-28-94 JCF Merged with mac header.
  44. * 02-11-95 CFW Add _CRTBLD to avoid users getting wrong headers.
  45. * 02-14-95 CFW Clean up Mac merge.
  46. * 12-14-95 JWM Add "#pragma once".
  47. * 02-24-97 GJF Detab-ed.
  48. * 10-07-97 RDL Added IA64.
  49. * 11-07-97 RDL Soft23 definitions.
  50. * 02-06-98 GJF Changes for Win64: fixed _APALIGN() macro (fix from
  51. * Intel)
  52. * 05-17-99 PML Remove all Macintosh support.
  53. * 10-06-99 PML Add _W64 modifier to types which are 32 bits in Win32,
  54. * 64 bits in Win64.
  55. * 10-25-99 PML Add support for _M_CEE (VS7#54572).
  56. * 01-20-00 PML Remove __epcg__.
  57. * 05-17-00 PML Use __alignof in _APALIGN macro for IA64.
  58. * 09-07-00 PML Remove va_list definition for _M_CEE (vs7#159777)
  59. * 09-21-00 PML Change IA64 definition of va_start for C++ (vs7#103357)
  60. * 02-05-01 PML Fix va_start for classes with operator& (vs7#201535)
  61. * 03-19-01 BWT Add AMD64 changes
  62. * 03-26-01 GB Added va_args for AMD64
  63. *
  64. ****/
  65. #if _MSC_VER > 1000 /*IFSTRIP=IGN*/
  66. #pragma once
  67. #endif
  68. #ifndef _INC_STDARG
  69. #define _INC_STDARG
  70. #if !defined(_WIN32)
  71. #error ERROR: Only Win32 target supported!
  72. #endif
  73. #ifndef _CRTBLD
  74. /* This version of the header files is NOT for user programs.
  75. * It is intended for use when building the C runtimes ONLY.
  76. * The version intended for public use will not have this message.
  77. */
  78. #error ERROR: Use of C runtime library internal header file.
  79. #endif /* _CRTBLD */
  80. #ifdef _MSC_VER
  81. /*
  82. * Currently, all MS C compilers for Win32 platforms default to 8 byte
  83. * alignment.
  84. */
  85. #pragma pack(push,8)
  86. #endif /* _MSC_VER */
  87. #ifdef __cplusplus
  88. extern "C" {
  89. #endif
  90. #ifndef _INTERNAL_IFSTRIP_
  91. #include <cruntime.h>
  92. #endif /* _INTERNAL_IFSTRIP_ */
  93. #if !defined(_W64)
  94. #if !defined(__midl) && (defined(_X86_) || defined(_M_IX86)) && _MSC_VER >= 1300 /*IFSTRIP=IGN*/
  95. #define _W64 __w64
  96. #else
  97. #define _W64
  98. #endif
  99. #endif
  100. #ifndef _UINTPTR_T_DEFINED
  101. #ifdef _WIN64
  102. typedef unsigned __int64 uintptr_t;
  103. #else
  104. typedef _W64 unsigned int uintptr_t;
  105. #endif
  106. #define _UINTPTR_T_DEFINED
  107. #endif
  108. #ifndef _VA_LIST_DEFINED
  109. #ifdef _M_ALPHA
  110. typedef struct {
  111. char *a0; /* pointer to first homed integer argument */
  112. int offset; /* byte offset of next parameter */
  113. } va_list;
  114. #else
  115. typedef char * va_list;
  116. #endif
  117. #define _VA_LIST_DEFINED
  118. #endif
  119. #ifdef __cplusplus
  120. #define _ADDRESSOF(v) ( &reinterpret_cast<const char &>(v) )
  121. #else
  122. #define _ADDRESSOF(v) ( &(v) )
  123. #endif
  124. #if defined(_M_CEE)
  125. extern void __cdecl __va_start(va_list*, ...);
  126. extern void * __cdecl __va_arg(va_list*, ...);
  127. extern void __cdecl __va_end(va_list*);
  128. #define va_start(ap,v) ( __va_start(&ap, _ADDRESSOF(v), sizeof(v), \
  129. __builtin_alignof(v), _ADDRESSOF(v)) )
  130. #define va_arg(ap,t) ( *(t *)__va_arg(&ap, sizeof(t), \
  131. __builtin_alignof(t), (t *)0) )
  132. #define va_end(ap) ( __va_end(&ap) )
  133. #elif defined(_M_IX86)
  134. #define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
  135. #define va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )
  136. #define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
  137. #define va_end(ap) ( ap = (va_list)0 )
  138. #elif defined(_M_MRX000)
  139. /* Use these types and definitions if generating code for MIPS */
  140. #define va_start(ap,v) ap = (va_list)_ADDRESSOF(v) + sizeof(v)
  141. #define va_end(list)
  142. #define va_arg(list, mode) ((mode *)(list =\
  143. (char *) ((((int)list + (__builtin_alignof(mode)<=4?3:7)) &\
  144. (__builtin_alignof(mode)<=4?-4:-8))+sizeof(mode))))[-1]
  145. /* +++++++++++++++++++++++++++++++++++++++++++
  146. Because of parameter passing conventions in C:
  147. use mode=int for char, and short types
  148. use mode=double for float types
  149. use a pointer for array types
  150. +++++++++++++++++++++++++++++++++++++++++++ */
  151. #elif defined(_M_ALPHA)
  152. /* Use these types and definitions if generating code for ALPHA */
  153. /*
  154. * The Alpha compiler supports two builtin functions that are used to
  155. * implement stdarg/varargs. The __builtin_va_start function is used
  156. * by va_start to initialize the data structure that locates the next
  157. * argument. The __builtin_isfloat function is used by va_arg to pick
  158. * which part of the home area a given register argument is stored in.
  159. * The home area is where up to six integer and/or six floating point
  160. * register arguments are stored down (so they can also be referenced
  161. * by a pointer like any arguments passed on the stack).
  162. */
  163. extern void * __builtin_va_start(va_list, ...);
  164. #ifdef _CFRONT
  165. #define __builtin_isfloat(a) __builtin_alignof(a)
  166. #endif
  167. #define va_start(list, v) __builtin_va_start(list, v, 1)
  168. #define va_end(list)
  169. #define va_arg(list, mode) \
  170. ( *( ((list).offset += ((int)sizeof(mode) + 7) & -8) , \
  171. (mode *)((list).a0 + (list).offset - \
  172. ((__builtin_isfloat(mode) && (list).offset <= (6 * 8)) ? \
  173. (6 * 8) + 8 : ((int)sizeof(mode) + 7) & -8) \
  174. ) \
  175. ) \
  176. )
  177. #elif defined(_M_PPC)
  178. /* Microsoft C8 front end (used in Motorola Merged compiler) */
  179. /* bytes that a type occupies in the argument list */
  180. #define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
  181. /* return 'ap' adjusted for type 't' in arglist */
  182. #define _ALIGNIT(ap,t) \
  183. ((((int)(ap))+(sizeof(t)<8?3:7)) & (sizeof(t)<8?~3:~7))
  184. #define va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )
  185. #define va_arg(ap,t) ( *(t *)((ap = (char *) (_ALIGNIT(ap, t) + _INTSIZEOF(t))) - _INTSIZEOF(t)) )
  186. #define va_end(ap) ( ap = (va_list)0 )
  187. #elif defined(_M_IA64)
  188. #define _VA_STRUCT_ALIGN 16
  189. #define _VA_ALIGN 8
  190. #define _ALIGNOF(ap) ((((ap)+_VA_STRUCT_ALIGN - 1) & ~(_VA_STRUCT_ALIGN -1)) \
  191. - (ap))
  192. #define _APALIGN(t,ap) (__alignof(t) > 8 ? _ALIGNOF((uintptr_t) ap) : 0)
  193. #define _SLOTSIZEOF(t) ( (sizeof(t) + _VA_ALIGN - 1) & ~(_VA_ALIGN - 1) )
  194. #ifdef __cplusplus
  195. extern void __cdecl __va_start(va_list*, ...);
  196. #define va_start(ap,v) ( __va_start(&ap, _ADDRESSOF(v), _SLOTSIZEOF(v), \
  197. _ADDRESSOF(v)) )
  198. #else
  199. #define va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _SLOTSIZEOF(v) )
  200. #endif
  201. #define va_arg(ap,t) (*(t *)((ap += _SLOTSIZEOF(t)+ _APALIGN(t,ap)) \
  202. -_SLOTSIZEOF(t)))
  203. #define va_end(ap) ( ap = (va_list)0 )
  204. #elif defined(_M_AMD64)
  205. extern void __cdecl __va_start(va_list *, ...);
  206. #define va_start(ap, x) ( __va_start(&ap, x) )
  207. #define va_arg(ap, t) \
  208. ( ( sizeof(t) > sizeof(__int64) || ( sizeof(t) & (sizeof(t) - 1) ) != 0 ) \
  209. ? **(t **)( ( ap += sizeof(__int64) ) - sizeof(__int64) ) \
  210. : *(t *)( ( ap += sizeof(__int64) ) - sizeof(__int64) ) )
  211. #define va_end(ap) ( ap = (va_list)0 )
  212. #else
  213. /* A guess at the proper definitions for other platforms */
  214. #define _INTSIZEOF(n) ( (sizeof(n) + sizeof(int) - 1) & ~(sizeof(int) - 1) )
  215. #define va_start(ap,v) ( ap = (va_list)_ADDRESSOF(v) + _INTSIZEOF(v) )
  216. #define va_arg(ap,t) ( *(t *)((ap += _INTSIZEOF(t)) - _INTSIZEOF(t)) )
  217. #define va_end(ap) ( ap = (va_list)0 )
  218. #endif
  219. #ifdef __cplusplus
  220. }
  221. #endif
  222. #ifdef _MSC_VER
  223. #pragma pack(pop)
  224. #endif /* _MSC_VER */
  225. #endif /* _INC_STDARG */