1 | // Licensed to the .NET Foundation under one or more agreements. |
2 | // The .NET Foundation licenses this file to you under the MIT license. |
3 | // See the LICENSE file in the project root for more information. |
4 | |
5 | // |
6 | |
7 | // |
8 | // common.h - precompiled headers include for the COM+ Execution Engine |
9 | // |
10 | |
11 | // |
12 | // Make sure _ASSERTE is defined before including this header file |
13 | // Other than that, please keep this header self-contained so that it can be included in |
14 | // all dlls |
15 | // |
16 | |
17 | |
18 | #ifndef _stdmacros_h_ |
19 | #define _stdmacros_h_ |
20 | |
21 | #include "specstrings.h" |
22 | #include "contract.h" |
23 | |
24 | #ifndef _ASSERTE |
25 | #error Please define _ASSERTE before including StdMacros.h |
26 | #endif |
27 | |
28 | #ifdef _DEBUG |
29 | #define DEBUG_ARG(x) , x |
30 | #define DEBUG_ARG1(x) x |
31 | #else |
32 | #define DEBUG_ARG(x) |
33 | #define DEBUG_ARG1(x) |
34 | #endif |
35 | |
36 | #ifdef DACCESS_COMPILE |
37 | #define DAC_ARG(x) , x |
38 | #else |
39 | #define DAC_ARG(x) |
40 | #endif |
41 | |
42 | |
43 | /********************************************/ |
44 | /* Portability macros */ |
45 | /********************************************/ |
46 | |
47 | #ifdef _TARGET_AMD64_ |
48 | #define AMD64_FIRST_ARG(x) x , |
49 | #define AMD64_ARG(x) , x |
50 | #define AMD64_ONLY(x) x |
51 | #define NOT_AMD64(x) |
52 | #define NOT_AMD64_ARG(x) |
53 | #else |
54 | #define AMD64_FIRST_ARG(x) |
55 | #define AMD64_ARG(x) |
56 | #define AMD64_ONLY(x) |
57 | #define NOT_AMD64(x) x |
58 | #define NOT_AMD64_ARG(x) , x |
59 | #endif |
60 | |
61 | #ifdef _TARGET_X86_ |
62 | #define X86_FIRST_ARG(x) x , |
63 | #define X86_ARG(x) , x |
64 | #define X86_ONLY(x) x |
65 | #define NOT_X86(x) |
66 | #define NOT_X86_ARG(x) |
67 | #else |
68 | #define X86_FIRST_ARG(x) |
69 | #define X86_ARG(x) |
70 | #define X86_ONLY(x) |
71 | #define NOT_X86(x) x |
72 | #define NOT_X86_ARG(x) , x |
73 | #endif |
74 | |
75 | #ifdef _WIN64 |
76 | #define WIN64_ARG(x) , x |
77 | #define WIN64_ONLY(x) x |
78 | #define NOT_WIN64(x) |
79 | #define NOT_WIN64_ARG(x) |
80 | #else |
81 | #define WIN64_ARG(x) |
82 | #define WIN64_ONLY(x) |
83 | #define NOT_WIN64(x) x |
84 | #define NOT_WIN64_ARG(x) , x |
85 | #endif // _WIN64 |
86 | |
87 | #ifdef _TARGET_ARM_ |
88 | #define ARM_FIRST_ARG(x) x , |
89 | #define ARM_ARG(x) , x |
90 | #define ARM_ONLY(x) x |
91 | #define NOT_ARM(x) |
92 | #define NOT_ARM_ARG(x) |
93 | #else |
94 | #define ARM_FIRST_ARG(x) |
95 | #define ARM_ARG(x) |
96 | #define ARM_ONLY(x) |
97 | #define NOT_ARM(x) x |
98 | #define NOT_ARM_ARG(x) , x |
99 | #endif |
100 | |
101 | #ifdef _TARGET_ARM64_ |
102 | #define ARM64_FIRST_ARG(x) x , |
103 | #define ARM64_ARG(x) , x |
104 | #define ARM64_ONLY(x) x |
105 | #define NOT_ARM64(x) |
106 | #define NOT_ARM64_ARG(x) |
107 | #else |
108 | #define ARM64_FIRST_ARG(x) |
109 | #define ARM64_ARG(x) |
110 | #define ARM64_ONLY(x) |
111 | #define NOT_ARM64(x) x |
112 | #define NOT_ARM64_ARG(x) , x |
113 | #endif |
114 | |
115 | #ifdef _TARGET_64BIT_ |
116 | #define LOG2_PTRSIZE 3 |
117 | #else |
118 | #define LOG2_PTRSIZE 2 |
119 | #endif |
120 | |
121 | #ifdef _WIN64 |
122 | #define INVALID_POINTER_CC 0xcccccccccccccccc |
123 | #define INVALID_POINTER_CD 0xcdcdcdcdcdcdcdcd |
124 | #define FMT_ADDR " %08x`%08x " |
125 | #define LFMT_ADDR W(" %08x`%08x ") |
126 | #define DBG_ADDR(ptr) (((UINT_PTR) (ptr)) >> 32), (((UINT_PTR) (ptr)) & 0xffffffff) |
127 | #else // _WIN64 |
128 | #define INVALID_POINTER_CC 0xcccccccc |
129 | #define INVALID_POINTER_CD 0xcdcdcdcd |
130 | #define FMT_ADDR " %08x " |
131 | #define LFMT_ADDR W(" %08x ") |
132 | #define DBG_ADDR(ptr) ((UINT_PTR)(ptr)) |
133 | #endif // _WIN64 |
134 | |
135 | #ifdef _TARGET_ARM_ |
136 | #define ALIGN_ACCESS ((1<<LOG2_PTRSIZE)-1) |
137 | #endif |
138 | |
139 | |
140 | #ifndef ALLOC_ALIGN_CONSTANT |
141 | #define ALLOC_ALIGN_CONSTANT (sizeof(void*)-1) |
142 | #endif |
143 | |
144 | |
145 | inline void *GetTopMemoryAddress(void) |
146 | { |
147 | WRAPPER_NO_CONTRACT; |
148 | |
149 | static void *result; // = NULL; |
150 | if( NULL == result ) |
151 | { |
152 | SYSTEM_INFO sysInfo; |
153 | GetSystemInfo( &sysInfo ); |
154 | result = sysInfo.lpMaximumApplicationAddress; |
155 | } |
156 | return result; |
157 | } |
158 | inline void *GetBotMemoryAddress(void) |
159 | { |
160 | WRAPPER_NO_CONTRACT; |
161 | |
162 | static void *result; // = NULL; |
163 | if( NULL == result ) |
164 | { |
165 | SYSTEM_INFO sysInfo; |
166 | GetSystemInfo( &sysInfo ); |
167 | result = sysInfo.lpMinimumApplicationAddress; |
168 | } |
169 | return result; |
170 | } |
171 | |
172 | #define TOP_MEMORY (GetTopMemoryAddress()) |
173 | #define BOT_MEMORY (GetBotMemoryAddress()) |
174 | |
175 | |
176 | // |
177 | // This macro returns val rounded up as necessary to be a multiple of alignment; alignment must be a power of 2 |
178 | // |
179 | inline size_t ALIGN_UP( size_t val, size_t alignment ) |
180 | { |
181 | LIMITED_METHOD_DAC_CONTRACT; |
182 | |
183 | // alignment must be a power of 2 for this implementation to work (need modulo otherwise) |
184 | _ASSERTE( 0 == (alignment & (alignment - 1)) ); |
185 | size_t result = (val + (alignment - 1)) & ~(alignment - 1); |
186 | _ASSERTE( result >= val ); // check for overflow |
187 | return result; |
188 | } |
189 | inline void* ALIGN_UP( void* val, size_t alignment ) |
190 | { |
191 | WRAPPER_NO_CONTRACT; |
192 | |
193 | return (void*) ALIGN_UP( (size_t)val, alignment ); |
194 | } |
195 | inline uint8_t* ALIGN_UP( uint8_t* val, size_t alignment ) |
196 | { |
197 | WRAPPER_NO_CONTRACT; |
198 | |
199 | return (uint8_t*) ALIGN_UP( (size_t)val, alignment ); |
200 | } |
201 | |
202 | inline size_t ALIGN_DOWN( size_t val, size_t alignment ) |
203 | { |
204 | LIMITED_METHOD_CONTRACT; |
205 | |
206 | // alignment must be a power of 2 for this implementation to work (need modulo otherwise) |
207 | _ASSERTE( 0 == (alignment & (alignment - 1)) ); |
208 | size_t result = val & ~(alignment - 1); |
209 | return result; |
210 | } |
211 | inline void* ALIGN_DOWN( void* val, size_t alignment ) |
212 | { |
213 | WRAPPER_NO_CONTRACT; |
214 | return (void*) ALIGN_DOWN( (size_t)val, alignment ); |
215 | } |
216 | inline uint8_t* ALIGN_DOWN( uint8_t* val, size_t alignment ) |
217 | { |
218 | WRAPPER_NO_CONTRACT; |
219 | return (uint8_t*) ALIGN_DOWN( (size_t)val, alignment ); |
220 | } |
221 | |
222 | inline BOOL IS_ALIGNED( size_t val, size_t alignment ) |
223 | { |
224 | LIMITED_METHOD_CONTRACT; |
225 | SUPPORTS_DAC; |
226 | |
227 | // alignment must be a power of 2 for this implementation to work (need modulo otherwise) |
228 | _ASSERTE( 0 == (alignment & (alignment - 1)) ); |
229 | return 0 == (val & (alignment - 1)); |
230 | } |
231 | inline BOOL IS_ALIGNED( const void* val, size_t alignment ) |
232 | { |
233 | WRAPPER_NO_CONTRACT; |
234 | return IS_ALIGNED( (size_t) val, alignment ); |
235 | } |
236 | |
237 | // Rounds a ULONG up to the nearest power of two number. |
238 | inline ULONG RoundUpToPower2(ULONG x) |
239 | { |
240 | if (x == 0) return 1; |
241 | |
242 | x = x - 1; |
243 | x = x | (x >> 1); |
244 | x = x | (x >> 2); |
245 | x = x | (x >> 4); |
246 | x = x | (x >> 8); |
247 | x = x | (x >> 16); |
248 | return x + 1; |
249 | } |
250 | |
251 | #ifdef ALIGN_ACCESS |
252 | |
253 | // NOTE: pSrc is evaluated three times!!! |
254 | #define MAYBE_UNALIGNED_READ(pSrc, bits) (IS_ALIGNED((size_t)(pSrc), sizeof(UINT##bits)) ? \ |
255 | (*(UINT##bits*) (pSrc)) : \ |
256 | (GET_UNALIGNED_##bits(pSrc)) ) |
257 | |
258 | #define MAYBE_UNALIGNED_WRITE(pDst, bits, expr) do { if (IS_ALIGNED((size_t)(pDst), sizeof(UINT##bits))) \ |
259 | *(UINT##bits*)(pDst) = (UINT##bits)(expr); else \ |
260 | SET_UNALIGNED_##bits(pDst, (UINT##bits)(expr)); } while (0) |
261 | |
262 | // these are necessary for MAYBE_UNALIGNED_XXX to work with UINT_PTR |
263 | #define GET_UNALIGNED__PTR(x) GET_UNALIGNED_PTR(x) |
264 | #define SET_UNALIGNED__PTR(p,x) SET_UNALIGNED_PTR(p,x) |
265 | |
266 | #else // ALIGN_ACCESS |
267 | #define MAYBE_UNALIGNED_READ(pSrc, bits) (*(UINT##bits*)(pSrc)) |
268 | #define MAYBE_UNALIGNED_WRITE(pDst, bits, expr) do { *(UINT##bits*)(pDst) = (UINT##bits)(expr); } while(0) |
269 | #endif // ALIGN_ACCESS |
270 | |
271 | // |
272 | // define some useful macros for logging object |
273 | // |
274 | |
275 | #define FMT_OBJECT "object" FMT_ADDR |
276 | #define FMT_HANDLE "handle" FMT_ADDR |
277 | #define FMT_CLASS "%s" |
278 | #define FMT_REG "r%d " |
279 | #define FMT_STK "sp%s0x%02x " |
280 | #define FMT_PIPTR "%s%s pointer " |
281 | |
282 | |
283 | #define DBG_GET_CLASS_NAME(pMT) \ |
284 | (((pMT) == NULL) ? NULL : (pMT)->GetClass()->GetDebugClassName()) |
285 | |
286 | #define DBG_CLASS_NAME_MT(pMT) \ |
287 | (DBG_GET_CLASS_NAME(pMT) == NULL) ? "<null-class>" : DBG_GET_CLASS_NAME(pMT) |
288 | |
289 | #define DBG_GET_MT_FROM_OBJ(obj) \ |
290 | (MethodTable*)((size_t)((Object*) (obj))->GetGCSafeMethodTable()) |
291 | |
292 | #define DBG_CLASS_NAME_OBJ(obj) \ |
293 | ((obj) == NULL) ? "null" : DBG_CLASS_NAME_MT(DBG_GET_MT_FROM_OBJ(obj)) |
294 | |
295 | #define DBG_CLASS_NAME_IPTR2(obj,iptr) \ |
296 | ((iptr) != 0) ? "" : DBG_CLASS_NAME_MT(DBG_GET_MT_FROM_OBJ(obj)) |
297 | |
298 | #define DBG_CLASS_NAME_IPTR(obj,iptr) \ |
299 | ((obj) == NULL) ? "null" : DBG_CLASS_NAME_IPTR2(obj,iptr) |
300 | |
301 | #define DBG_STK(off) \ |
302 | (off >= 0) ? "+" : "-", \ |
303 | (off >= 0) ? off : -off |
304 | |
305 | #define DBG_PIN_NAME(pin) \ |
306 | (pin) ? "pinned " : "" |
307 | |
308 | #define DBG_IPTR_NAME(iptr) \ |
309 | (iptr) ? "interior" : "base" |
310 | |
311 | #define LOG_HANDLE_OBJECT_CLASS(str1, hnd, str2, obj) \ |
312 | str1 FMT_HANDLE str2 FMT_OBJECT FMT_CLASS "\n", \ |
313 | DBG_ADDR(hnd), DBG_ADDR(obj), DBG_CLASS_NAME_OBJ(obj) |
314 | |
315 | #define LOG_OBJECT_CLASS(obj) \ |
316 | FMT_OBJECT FMT_CLASS "\n", \ |
317 | DBG_ADDR(obj), DBG_CLASS_NAME_OBJ(obj) |
318 | |
319 | #define LOG_PIPTR_OBJECT_CLASS(obj, pin, iptr) \ |
320 | FMT_PIPTR FMT_ADDR FMT_CLASS "\n", \ |
321 | DBG_PIN_NAME(pin), DBG_IPTR_NAME(iptr), \ |
322 | DBG_ADDR(obj), DBG_CLASS_NAME_IPTR(obj,iptr) |
323 | |
324 | #define LOG_HANDLE_OBJECT(str1, hnd, str2, obj) \ |
325 | str1 FMT_HANDLE str2 FMT_OBJECT "\n", \ |
326 | DBG_ADDR(hnd), DBG_ADDR(obj) |
327 | |
328 | #define LOG_PIPTR_OBJECT(obj, pin, iptr) \ |
329 | FMT_PIPTR FMT_ADDR "\n", \ |
330 | DBG_PIN_NAME(pin), DBG_IPTR_NAME(iptr), \ |
331 | DBG_ADDR(obj) |
332 | |
333 | #define UNIQUE_LABEL_DEF(a,x) a##x |
334 | #define UNIQUE_LABEL_DEF_X(a,x) UNIQUE_LABEL_DEF(a,x) |
335 | #ifdef _MSC_VER |
336 | #define UNIQUE_LABEL(a) UNIQUE_LABEL_DEF_X(_unique_label_##a##_, __COUNTER__) |
337 | #else |
338 | #define UNIQUE_LABEL(a) UNIQUE_LABEL_DEF_X(_unique_label_##a##_, __LINE__) |
339 | #endif |
340 | |
341 | |
342 | #ifndef _countof |
343 | #define _countof(_array) (sizeof(_array)/sizeof(_array[0])) |
344 | #endif |
345 | |
346 | |
347 | // This is temporary. LKG should provide these macros and we should then |
348 | // remove STRUNCATE and _TRUNCATE from here. |
349 | |
350 | /* error codes */ |
351 | #if !defined(STRUNCATE) |
352 | #define STRUNCATE 80 |
353 | #endif |
354 | |
355 | /* _TRUNCATE */ |
356 | #if !defined(_TRUNCATE) |
357 | #define _TRUNCATE ((size_t)-1) |
358 | #endif |
359 | |
360 | #endif //_stdmacros_h_ |
361 | |