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 | // method.hpp |
6 | // |
7 | |
8 | // |
9 | // See the book of the runtime entry for overall design: |
10 | // file:../../doc/BookOfTheRuntime/ClassLoader/MethodDescDesign.doc |
11 | // |
12 | |
13 | #ifndef _METHOD_H |
14 | #define _METHOD_H |
15 | |
16 | #include "cor.h" |
17 | #include "util.hpp" |
18 | #include "clsload.hpp" |
19 | #include "codeman.h" |
20 | #include "class.h" |
21 | #include "siginfo.hpp" |
22 | #include "declsec.h" |
23 | #include "methodimpl.h" |
24 | #include "typedesc.h" |
25 | #include <stddef.h> |
26 | #include "eeconfig.h" |
27 | #include "precode.h" |
28 | #include "codeversion.h" |
29 | |
30 | #ifndef FEATURE_PREJIT |
31 | #include "fixuppointer.h" |
32 | #endif |
33 | |
34 | class Stub; |
35 | class FCallMethodDesc; |
36 | class FieldDesc; |
37 | class NDirect; |
38 | class MethodDescChunk; |
39 | struct LayoutRawFieldInfo; |
40 | class InstantiatedMethodDesc; |
41 | class DictionaryLayout; |
42 | class Dictionary; |
43 | class GCCoverageInfo; |
44 | class DynamicMethodDesc; |
45 | class ReJitManager; |
46 | class CodeVersionManager; |
47 | class PrepareCodeConfig; |
48 | class CallCounter; |
49 | |
50 | typedef DPTR(FCallMethodDesc) PTR_FCallMethodDesc; |
51 | typedef DPTR(ArrayMethodDesc) PTR_ArrayMethodDesc; |
52 | typedef DPTR(DynamicMethodDesc) PTR_DynamicMethodDesc; |
53 | typedef DPTR(InstantiatedMethodDesc) PTR_InstantiatedMethodDesc; |
54 | typedef DPTR(GCCoverageInfo) PTR_GCCoverageInfo; // see code:GCCoverageInfo::savedCode |
55 | |
56 | #ifdef FEATURE_MINIMETADATA_IN_TRIAGEDUMPS |
57 | GVAL_DECL(DWORD, g_MiniMetaDataBuffMaxSize); |
58 | GVAL_DECL(TADDR, g_MiniMetaDataBuffAddress); |
59 | #endif // FEATURE_MINIMETADATA_IN_TRIAGEDUMPS |
60 | |
61 | EXTERN_C VOID STDCALL NDirectImportThunk(); |
62 | |
63 | #define METHOD_TOKEN_REMAINDER_BIT_COUNT 14 |
64 | #define METHOD_TOKEN_REMAINDER_MASK ((1 << METHOD_TOKEN_REMAINDER_BIT_COUNT) - 1) |
65 | #define METHOD_TOKEN_RANGE_BIT_COUNT (24 - METHOD_TOKEN_REMAINDER_BIT_COUNT) |
66 | #define METHOD_TOKEN_RANGE_MASK ((1 << METHOD_TOKEN_RANGE_BIT_COUNT) - 1) |
67 | |
68 | //============================================================= |
69 | // Splits methoddef token into two pieces for |
70 | // storage inside a methoddesc. |
71 | //============================================================= |
72 | FORCEINLINE UINT16 GetTokenRange(mdToken tok) |
73 | { |
74 | LIMITED_METHOD_CONTRACT; |
75 | return (UINT16)((tok>>METHOD_TOKEN_REMAINDER_BIT_COUNT) & METHOD_TOKEN_RANGE_MASK); |
76 | } |
77 | |
78 | FORCEINLINE VOID SplitToken(mdToken tok, UINT16 *ptokrange, UINT16 *ptokremainder) |
79 | { |
80 | LIMITED_METHOD_CONTRACT; |
81 | *ptokrange = (UINT16)((tok>>METHOD_TOKEN_REMAINDER_BIT_COUNT) & METHOD_TOKEN_RANGE_MASK); |
82 | *ptokremainder = (UINT16)(tok & METHOD_TOKEN_REMAINDER_MASK); |
83 | } |
84 | |
85 | FORCEINLINE mdToken MergeToken(UINT16 tokrange, UINT16 tokremainder) |
86 | { |
87 | LIMITED_METHOD_DAC_CONTRACT; |
88 | return (tokrange << METHOD_TOKEN_REMAINDER_BIT_COUNT) | tokremainder | mdtMethodDef; |
89 | } |
90 | |
91 | // The MethodDesc is a union of several types. The following |
92 | // 3-bit field determines which type it is. Note that JIT'ed/non-JIT'ed |
93 | // is not represented here because this isn't known until the |
94 | // method is executed for the first time. Because any thread could |
95 | // change this bit, it has to be done in a place where access is |
96 | // synchronized. |
97 | |
98 | // **** NOTE: if you add any new flags, make sure you add them to ClearFlagsOnUpdate |
99 | // so that when a method is replaced its relevant flags are updated |
100 | |
101 | // Used in MethodDesc |
102 | enum MethodClassification |
103 | { |
104 | mcIL = 0, // IL |
105 | mcFCall = 1, // FCall (also includes tlbimped ctor, Delegate ctor) |
106 | mcNDirect = 2, // N/Direct |
107 | mcEEImpl = 3, // special method; implementation provided by EE (like Delegate Invoke) |
108 | mcArray = 4, // Array ECall |
109 | mcInstantiated = 5, // Instantiated generic methods, including descriptors |
110 | // for both shared and unshared code (see InstantiatedMethodDesc) |
111 | |
112 | #ifdef FEATURE_COMINTEROP |
113 | // This needs a little explanation. There are MethodDescs on MethodTables |
114 | // which are Interfaces. These have the mdcInterface bit set. Then there |
115 | // are MethodDescs on MethodTables that are Classes, where the method is |
116 | // exposed through an interface. These do not have the mdcInterface bit set. |
117 | // |
118 | // So, today, a dispatch through an 'mdcInterface' MethodDesc is either an |
119 | // error (someone forgot to look up the method in a class' VTable) or it is |
120 | // a case of COM Interop. |
121 | |
122 | mcComInterop = 6, |
123 | #endif // FEATURE_COMINTEROP |
124 | mcDynamic = 7, // for method desc with no metadata behind |
125 | mcCount, |
126 | }; |
127 | |
128 | |
129 | // All flags in the MethodDesc now reside in a single 16-bit field. |
130 | |
131 | enum MethodDescClassification |
132 | { |
133 | // Method is IL, FCall etc., see MethodClassification above. |
134 | mdcClassification = 0x0007, |
135 | mdcClassificationCount = mdcClassification+1, |
136 | |
137 | // Note that layout of code:MethodDesc::s_ClassificationSizeTable depends on the exact values |
138 | // of mdcHasNonVtableSlot and mdcMethodImpl |
139 | |
140 | // Has local slot (vs. has real slot in MethodTable) |
141 | mdcHasNonVtableSlot = 0x0008, |
142 | |
143 | // Method is a body for a method impl (MI_MethodDesc, MI_NDirectMethodDesc, etc) |
144 | // where the function explicitly implements IInterface.foo() instead of foo(). |
145 | mdcMethodImpl = 0x0010, |
146 | |
147 | // Method is static |
148 | mdcStatic = 0x0020, |
149 | |
150 | // unused = 0x0040, |
151 | // unused = 0x0080, |
152 | // unused = 0x0100, |
153 | // unused = 0x0200, |
154 | |
155 | // Duplicate method. When a method needs to be placed in multiple slots in the |
156 | // method table, because it could not be packed into one slot. For eg, a method |
157 | // providing implementation for two interfaces, MethodImpl, etc |
158 | mdcDuplicate = 0x0400, |
159 | |
160 | // Has this method been verified? |
161 | mdcVerifiedState = 0x0800, |
162 | |
163 | // Is the method verifiable? It needs to be verified first to determine this |
164 | mdcVerifiable = 0x1000, |
165 | |
166 | // Is this method ineligible for inlining? |
167 | mdcNotInline = 0x2000, |
168 | |
169 | // Is the method synchronized |
170 | mdcSynchronized = 0x4000, |
171 | |
172 | // Does the method's slot number require all 16 bits |
173 | mdcRequiresFullSlotNumber = 0x8000 |
174 | }; |
175 | |
176 | #define METHOD_MAX_RVA 0x7FFFFFFF |
177 | |
178 | |
179 | // The size of this structure needs to be a multiple of MethodDesc::ALIGNMENT |
180 | // |
181 | // @GENERICS: |
182 | // Method descriptors for methods belonging to instantiated types may be shared between compatible instantiations |
183 | // Hence for reflection and elsewhere where exact types are important it's necessary to pair a method desc |
184 | // with the exact owning type handle. |
185 | // |
186 | // See genmeth.cpp for details of instantiated generic method descriptors. |
187 | // |
188 | // A MethodDesc is the representation of a method of a type. These live in code:MethodDescChunk which in |
189 | // turn lives in code:EEClass. They are conceptually cold (we do not expect to access them in normal |
190 | // program exectution, but we often fall short of that goal. |
191 | // |
192 | // A Method desc knows how to get at its metadata token code:GetMemberDef, its chunk |
193 | // code:MethodDescChunk, which in turns knows how to get at its type code:MethodTable. |
194 | // It also knows how to get at its IL code (code:IMAGE_COR_ILMETHOD) |
195 | class MethodDesc |
196 | { |
197 | friend class EEClass; |
198 | friend class MethodTableBuilder; |
199 | friend class ArrayClass; |
200 | friend class NDirect; |
201 | friend class MethodDescChunk; |
202 | friend class InstantiatedMethodDesc; |
203 | friend class MethodImpl; |
204 | friend class CheckAsmOffsets; |
205 | friend class ClrDataAccess; |
206 | |
207 | friend class MethodDescCallSite; |
208 | #ifdef DACCESS_COMPILE |
209 | friend class NativeImageDumper; |
210 | #endif |
211 | |
212 | public: |
213 | |
214 | enum |
215 | { |
216 | #ifdef _WIN64 |
217 | ALIGNMENT_SHIFT = 3, |
218 | #else |
219 | ALIGNMENT_SHIFT = 2, |
220 | #endif |
221 | ALIGNMENT = (1<<ALIGNMENT_SHIFT), |
222 | ALIGNMENT_MASK = (ALIGNMENT-1) |
223 | }; |
224 | |
225 | #ifdef _DEBUG |
226 | |
227 | // These are set only for MethodDescs but every time we want to use the debugger |
228 | // to examine these fields, the code has the thing stored in a MethodDesc*. |
229 | // So... |
230 | LPCUTF8 m_pszDebugMethodName; |
231 | LPCUTF8 m_pszDebugClassName; |
232 | LPCUTF8 m_pszDebugMethodSignature; |
233 | FixupPointer<PTR_MethodTable> m_pDebugMethodTable; |
234 | |
235 | PTR_GCCoverageInfo m_GcCover; |
236 | |
237 | #endif // _DEBUG |
238 | |
239 | inline BOOL HasStableEntryPoint() |
240 | { |
241 | LIMITED_METHOD_DAC_CONTRACT; |
242 | |
243 | return (m_bFlags2 & enum_flag2_HasStableEntryPoint) != 0; |
244 | } |
245 | |
246 | inline PCODE GetStableEntryPoint() |
247 | { |
248 | LIMITED_METHOD_DAC_CONTRACT; |
249 | |
250 | _ASSERTE(HasStableEntryPoint()); |
251 | return GetMethodEntryPoint(); |
252 | } |
253 | |
254 | BOOL SetStableEntryPointInterlocked(PCODE addr); |
255 | |
256 | BOOL HasTemporaryEntryPoint(); |
257 | PCODE GetTemporaryEntryPoint(); |
258 | |
259 | void SetTemporaryEntryPoint(LoaderAllocator *pLoaderAllocator, AllocMemTracker *pamTracker); |
260 | |
261 | inline BOOL HasPrecode() |
262 | { |
263 | LIMITED_METHOD_DAC_CONTRACT; |
264 | |
265 | return (m_bFlags2 & enum_flag2_HasPrecode) != 0; |
266 | } |
267 | |
268 | inline Precode* GetPrecode() |
269 | { |
270 | LIMITED_METHOD_DAC_CONTRACT; |
271 | |
272 | PRECONDITION(HasPrecode()); |
273 | Precode* pPrecode = Precode::GetPrecodeFromEntryPoint(GetStableEntryPoint()); |
274 | PREFIX_ASSUME(pPrecode != NULL); |
275 | return pPrecode; |
276 | } |
277 | |
278 | inline BOOL MayHavePrecode() |
279 | { |
280 | CONTRACTL |
281 | { |
282 | THROWS; |
283 | GC_TRIGGERS; |
284 | MODE_ANY; |
285 | } |
286 | CONTRACTL_END |
287 | |
288 | return !MayHaveNativeCode() || IsVersionableWithPrecode(); |
289 | } |
290 | |
291 | void InterlockedUpdateFlags2(BYTE bMask, BOOL fSet); |
292 | |
293 | Precode* GetOrCreatePrecode(); |
294 | |
295 | #ifdef FEATURE_PREJIT |
296 | Precode * GetSavedPrecode(DataImage *image); |
297 | Precode * GetSavedPrecodeOrNull(DataImage *image); |
298 | #endif // FEATURE_PREJIT |
299 | |
300 | // Given a code address return back the MethodDesc whenever possible |
301 | // |
302 | static MethodDesc * GetMethodDescFromStubAddr(PCODE addr, BOOL fSpeculative = FALSE); |
303 | |
304 | |
305 | DWORD GetAttrs() const; |
306 | |
307 | DWORD GetImplAttrs(); |
308 | |
309 | // This function can lie if a method impl was used to implement |
310 | // more than one method on this class. Use GetName(int) to indicate |
311 | // which slot you are interested in. |
312 | // See the TypeString class for better control over name formatting. |
313 | LPCUTF8 GetName(); |
314 | |
315 | LPCUTF8 GetName(USHORT slot); |
316 | |
317 | void PrecomputeNameHash(); |
318 | BOOL MightHaveName(ULONG nameHashValue); |
319 | |
320 | FORCEINLINE LPCUTF8 GetNameOnNonArrayClass() |
321 | { |
322 | WRAPPER_NO_CONTRACT; |
323 | LPCSTR szName; |
324 | if (FAILED(GetMDImport()->GetNameOfMethodDef(GetMemberDef(), &szName))) |
325 | { |
326 | szName = NULL; |
327 | } |
328 | return szName; |
329 | } |
330 | |
331 | COUNT_T GetStableHash(); |
332 | |
333 | // Non-zero for InstantiatedMethodDescs |
334 | DWORD GetNumGenericMethodArgs(); |
335 | |
336 | // Return the number of class type parameters that are in scope for this method |
337 | DWORD GetNumGenericClassArgs() |
338 | { |
339 | WRAPPER_NO_CONTRACT; |
340 | SUPPORTS_DAC; |
341 | return GetMethodTable()->GetNumGenericArgs(); |
342 | } |
343 | |
344 | // True if this is a method descriptor for an instantiated generic method |
345 | // whose method type arguments are the formal type parameters of the generic method |
346 | // NOTE: the declaring class may not be the generic type definition e.g. consider C<int>.m<T> |
347 | BOOL IsGenericMethodDefinition() const; |
348 | |
349 | // True if the declaring type or instantiation of method (if any) contains formal generic type parameters |
350 | BOOL ContainsGenericVariables(); |
351 | |
352 | Module* GetDefiningModuleForOpenMethod(); |
353 | |
354 | // True if this is a class and method instantiation that on <__Canon,...,__Canon> |
355 | BOOL IsTypicalSharedInstantiation(); |
356 | |
357 | |
358 | // True if and only if this is a method desriptor for : |
359 | // 1. a non-generic method or a generic method at its typical method instantiation |
360 | // 2. in a non-generic class or a typical instantiation of a generic class |
361 | // This method can be called on a non-restored method desc |
362 | BOOL IsTypicalMethodDefinition() const; |
363 | |
364 | // Force a load of the (typical) constraints on the type parameters of a typical method definition, |
365 | // detecting cyclic bounds on class and method type parameters. |
366 | void LoadConstraintsForTypicalMethodDefinition(BOOL *pfHasCircularClassConstraints, |
367 | BOOL *pfHasCircularMethodConstraints, |
368 | ClassLoadLevel level = CLASS_LOADED); |
369 | |
370 | DWORD IsClassConstructor() |
371 | { |
372 | WRAPPER_NO_CONTRACT; |
373 | return IsMdClassConstructor(GetAttrs(), GetName()); |
374 | } |
375 | |
376 | DWORD IsClassConstructorOrCtor() |
377 | { |
378 | WRAPPER_NO_CONTRACT; |
379 | DWORD dwAttrs = GetAttrs(); |
380 | if (IsMdRTSpecialName(dwAttrs)) |
381 | { |
382 | LPCUTF8 name = GetName(); |
383 | return IsMdInstanceInitializer(dwAttrs, name) || IsMdClassConstructor(dwAttrs, name); |
384 | } |
385 | return FALSE; |
386 | } |
387 | |
388 | inline void SetHasMethodImplSlot() |
389 | { |
390 | m_wFlags |= mdcMethodImpl; |
391 | } |
392 | |
393 | inline BOOL HasMethodImplSlot() |
394 | { |
395 | LIMITED_METHOD_DAC_CONTRACT; |
396 | return (mdcMethodImpl & m_wFlags); |
397 | } |
398 | |
399 | FORCEINLINE BOOL IsMethodImpl() |
400 | { |
401 | LIMITED_METHOD_DAC_CONTRACT; |
402 | // Once we stop allocating dummy MethodImplSlot in MethodTableBuilder::WriteMethodImplData, |
403 | // the check for NULL will become unnecessary. |
404 | return HasMethodImplSlot() && (GetMethodImpl()->GetSlots() != NULL); |
405 | } |
406 | |
407 | inline DWORD IsStatic() |
408 | { |
409 | LIMITED_METHOD_DAC_CONTRACT; |
410 | |
411 | // This bit caches the IsMdStatic(GetAttrs()) check. We used to assert it here, but not doing it anymore. GetAttrs() |
412 | // accesses metadata that is not compatible with contracts of this method. The metadata access can fail, the metadata |
413 | // are not available during shutdown, the metadata access can take locks. It is not worth it to code around all these |
414 | // just for the assert. |
415 | // _ASSERTE((((m_wFlags & mdcStatic) != 0) == (IsMdStatic(flags) != 0))); |
416 | |
417 | return (m_wFlags & mdcStatic) != 0; |
418 | } |
419 | |
420 | inline void SetStatic() |
421 | { |
422 | LIMITED_METHOD_CONTRACT; |
423 | m_wFlags |= mdcStatic; |
424 | } |
425 | |
426 | inline void ClearStatic() |
427 | { |
428 | LIMITED_METHOD_CONTRACT; |
429 | m_wFlags &= ~mdcStatic; |
430 | } |
431 | |
432 | inline BOOL IsIL() |
433 | { |
434 | LIMITED_METHOD_DAC_CONTRACT; |
435 | return mcIL == GetClassification() || mcInstantiated == GetClassification(); |
436 | } |
437 | |
438 | //================================================================ |
439 | // Generics-related predicates etc. |
440 | |
441 | // True if the method descriptor is an instantiation of a generic method. |
442 | inline BOOL HasMethodInstantiation() const; |
443 | |
444 | // True if the method descriptor is either an instantiation of |
445 | // a generic method or is an instance method in an instantiated class (or both). |
446 | BOOL HasClassOrMethodInstantiation() |
447 | { |
448 | LIMITED_METHOD_DAC_CONTRACT; |
449 | return (HasClassInstantiation() || HasMethodInstantiation()); |
450 | } |
451 | |
452 | BOOL HasClassOrMethodInstantiation_NoLogging() const |
453 | { |
454 | LIMITED_METHOD_DAC_CONTRACT; |
455 | return (HasClassInstantiation_NoLogging() || HasMethodInstantiation()); |
456 | } |
457 | |
458 | inline BOOL HasClassInstantiation() const |
459 | { |
460 | LIMITED_METHOD_DAC_CONTRACT; |
461 | return GetMethodTable()->HasInstantiation(); |
462 | } |
463 | |
464 | inline BOOL HasClassInstantiation_NoLogging() const |
465 | { |
466 | LIMITED_METHOD_DAC_CONTRACT; |
467 | return GetMethodTable_NoLogging()->HasInstantiation(); |
468 | } |
469 | |
470 | // Return the instantiation for an instantiated generic method |
471 | // Return NULL if not an instantiated method |
472 | // To get the (representative) instantiation of the declaring class use GetMethodTable()->GetInstantiation() |
473 | // NOTE: This will assert if you try to get the instantiation of a generic method def in a non-typical class |
474 | // e.g. C<int>.m<U> will fail but C<T>.m<U> will succeed |
475 | Instantiation GetMethodInstantiation() const; |
476 | |
477 | // As above, but will succeed on C<int>.m<U> |
478 | // To do this it might force a load of the typical parent |
479 | Instantiation LoadMethodInstantiation(); |
480 | |
481 | // Return a pointer to the method dictionary for an instantiated generic method |
482 | // The initial slots in a method dictionary are the type arguments themselves |
483 | // Return NULL if not an instantiated method |
484 | Dictionary* GetMethodDictionary(); |
485 | DictionaryLayout* GetDictionaryLayout(); |
486 | |
487 | InstantiatedMethodDesc* AsInstantiatedMethodDesc() const; |
488 | |
489 | BaseDomain *GetDomain(); |
490 | |
491 | #ifdef FEATURE_CODE_VERSIONING |
492 | CodeVersionManager* GetCodeVersionManager(); |
493 | #endif |
494 | #ifdef FEATURE_TIERED_COMPILATION |
495 | CallCounter* GetCallCounter(); |
496 | #endif |
497 | |
498 | PTR_LoaderAllocator GetLoaderAllocator(); |
499 | |
500 | // GetLoaderAllocatorForCode returns the allocator with the responsibility for allocation. |
501 | // This is called from GetMulticallableAddrOfCode when allocating a small trampoline stub for the method. |
502 | // Normally a method in a shared domain will allocate memory for stubs in the shared domain. |
503 | // That has to be different for DynamicMethod as they need to be allocated always in the AppDomain |
504 | // that created the method. |
505 | LoaderAllocator * GetLoaderAllocatorForCode(); |
506 | |
507 | // GetDomainSpecificLoaderAllocator returns the collectable loader allocator for collectable types |
508 | // and the loader allocator in the current domain for non-collectable types |
509 | LoaderAllocator * GetDomainSpecificLoaderAllocator(); |
510 | |
511 | Module* GetLoaderModule(); |
512 | |
513 | Module* GetZapModule(); |
514 | |
515 | // Does this immediate item live in an NGEN module? |
516 | BOOL IsZapped(); |
517 | |
518 | // Strip off method and class instantiation if present and replace by the typical instantiation |
519 | // e.g. C<int>.m<string> -> C<T>.m<U>. Does not modify the MethodDesc, but returns |
520 | // the appropriate stripped MethodDesc. |
521 | // This is the identity function on non-instantiated method descs in non-instantiated classes |
522 | MethodDesc* LoadTypicalMethodDefinition(); |
523 | |
524 | // Strip off the method instantiation (if present) and replace by the typical instantiation |
525 | // e.g. // C<int>.m<string> -> C<int>.m<U>. Does not modify the MethodDesc, but returns |
526 | // the appropriate stripped MethodDesc. |
527 | // This is the identity function on non-instantiated method descs |
528 | MethodDesc* StripMethodInstantiation(); |
529 | |
530 | // Return the instantiation of a method's enclosing class |
531 | // Return NULL if the enclosing class is not instantiated |
532 | // If the method code is shared then this might be a *representative* instantiation |
533 | // |
534 | // See GetExactClassInstantiation if you need to get the exact |
535 | // instantiation of a shared method desc. |
536 | Instantiation GetClassInstantiation() const; |
537 | |
538 | // Is the code shared between multiple instantiations of class or method? |
539 | // If so, then when compiling the code we might need to look up tokens |
540 | // in the class or method dictionary. Also, when debugging the exact generic arguments |
541 | // need to be ripped off the stack, either from the this pointer or from one of the |
542 | // extra args below. |
543 | BOOL IsSharedByGenericInstantiations(); // shared code of any kind |
544 | |
545 | BOOL IsSharedByGenericMethodInstantiations(); // shared due to method instantiation |
546 | |
547 | // How does a method shared between generic instantiations get at |
548 | // the extra instantiation information at runtime? Only one of the following three |
549 | // will ever hold: |
550 | // |
551 | // AcquiresInstMethodTableFromThis() |
552 | // The method is in a generic class but is not itself a |
553 | // generic method (the normal case). Furthermore a "this" pointer |
554 | // is available and we can get the exact instantiation from it. |
555 | // |
556 | // RequiresInstMethodTableArg() |
557 | // The method is shared between generic classes but is not |
558 | // itself generic. Furthermore no "this" pointer is given |
559 | // (e.g. a value type method), so we pass in the exact-instantiation |
560 | // method table as an extra argument. |
561 | // i.e. per-inst static methods in shared-code instantiated generic |
562 | // classes (e.g. static void MyClass<string>::m()) |
563 | // i.e. shared-code instance methods in instantiated generic |
564 | // structs (e.g. void MyValueType<string>::m()) |
565 | // |
566 | // RequiresInstMethodDescArg() |
567 | // The method is itself generic and is shared between generic |
568 | // instantiations but is not itself generic. Furthermore |
569 | // no "this" pointer is given (e.g. a value type method), so we pass in the |
570 | // exact-instantiation method table as an extra argument. |
571 | // i.e. shared-code instantiated generic methods |
572 | // |
573 | // These are used for direct calls to instantiated generic methods |
574 | // e.g. call void C::m<string>() implemented by calculating dict(m<string>) at compile-time and passing it as an extra parameter |
575 | // call void C::m<!0>() implemented by calculating dict(m<!0>) at run-time (if the caller lives in shared-class code) |
576 | |
577 | BOOL AcquiresInstMethodTableFromThis(); |
578 | BOOL RequiresInstMethodTableArg(); |
579 | BOOL RequiresInstMethodDescArg(); |
580 | BOOL RequiresInstArg(); |
581 | |
582 | // Can this method handle be given out to reflection for use in a MethodInfo |
583 | // object? |
584 | BOOL IsRuntimeMethodHandle(); |
585 | |
586 | // Given a method table of an object and a method that comes from some |
587 | // superclass of the class of that object, find that superclass. |
588 | MethodTable * GetExactDeclaringType(MethodTable * ownerOrSubType); |
589 | |
590 | // Given a type handle of an object and a method that comes from some |
591 | // superclass of the class of that object, find the instantiation of |
592 | // that superclass, i.e. the class instantiation which will be relevant |
593 | // to interpreting the signature of the method. The type handle of |
594 | // the object does not need to be given in all circumstances, in |
595 | // particular it is only needed for MethodDescs pMD that |
596 | // return true for pMD->RequiresInstMethodTableArg() or |
597 | // pMD->RequiresInstMethodDescArg(). In other cases it is |
598 | // allowed to be null. |
599 | // |
600 | // Will return NULL if the method is not in a generic class. |
601 | Instantiation GetExactClassInstantiation(TypeHandle possibleObjType); |
602 | |
603 | |
604 | BOOL SatisfiesMethodConstraints(TypeHandle thParent, BOOL fThrowIfNotSatisfied = FALSE); |
605 | |
606 | |
607 | BOOL HasSameMethodDefAs(MethodDesc * pMD); |
608 | |
609 | //================================================================ |
610 | // Classifications of kinds of MethodDescs. |
611 | |
612 | inline BOOL IsRuntimeSupplied() |
613 | { |
614 | LIMITED_METHOD_DAC_CONTRACT; |
615 | return mcFCall == GetClassification() |
616 | || mcArray == GetClassification(); |
617 | } |
618 | |
619 | |
620 | inline DWORD IsArray() const |
621 | { |
622 | LIMITED_METHOD_DAC_CONTRACT; |
623 | return mcArray == GetClassification(); |
624 | } |
625 | |
626 | inline DWORD IsEEImpl() const |
627 | { |
628 | LIMITED_METHOD_DAC_CONTRACT; |
629 | return mcEEImpl == GetClassification(); |
630 | } |
631 | |
632 | inline DWORD IsNoMetadata() const |
633 | { |
634 | LIMITED_METHOD_DAC_CONTRACT; |
635 | return (mcDynamic == GetClassification()); |
636 | } |
637 | |
638 | inline PTR_DynamicMethodDesc AsDynamicMethodDesc(); |
639 | inline bool IsDynamicMethod(); |
640 | inline bool IsILStub(); |
641 | inline bool IsLCGMethod(); |
642 | |
643 | inline DWORD IsNDirect() |
644 | { |
645 | LIMITED_METHOD_DAC_CONTRACT; |
646 | return mcNDirect == GetClassification(); |
647 | } |
648 | |
649 | inline DWORD IsInterface() |
650 | { |
651 | WRAPPER_NO_CONTRACT; |
652 | return GetMethodTable()->IsInterface(); |
653 | } |
654 | |
655 | void ComputeSuppressUnmanagedCodeAccessAttr(IMDInternalImport *pImport); |
656 | BOOL HasNativeCallableAttribute(); |
657 | |
658 | #ifdef FEATURE_COMINTEROP |
659 | inline DWORD IsComPlusCall() |
660 | { |
661 | WRAPPER_NO_CONTRACT; |
662 | return mcComInterop == GetClassification(); |
663 | } |
664 | inline DWORD IsGenericComPlusCall(); |
665 | inline void SetupGenericComPlusCall(); |
666 | #else // !FEATURE_COMINTEROP |
667 | // hardcoded to return FALSE to improve code readibility |
668 | inline DWORD IsComPlusCall() |
669 | { |
670 | LIMITED_METHOD_CONTRACT; |
671 | return FALSE; |
672 | } |
673 | inline DWORD IsGenericComPlusCall() |
674 | { |
675 | LIMITED_METHOD_CONTRACT; |
676 | return FALSE; |
677 | } |
678 | #endif // !FEATURE_COMINTEROP |
679 | |
680 | // Update flags in a thread safe manner. |
681 | WORD InterlockedUpdateFlags(WORD wMask, BOOL fSet); |
682 | |
683 | // If the method is in an Edit and Contine (EnC) module, then |
684 | // we DON'T want to backpatch this, ever. We MUST always call |
685 | // through the precode so that we can update the method. |
686 | inline DWORD IsEnCMethod() |
687 | { |
688 | WRAPPER_NO_CONTRACT; |
689 | Module *pModule = GetModule(); |
690 | PREFIX_ASSUME(pModule != NULL); |
691 | return pModule->IsEditAndContinueEnabled(); |
692 | } |
693 | |
694 | inline BOOL IsNotInline() |
695 | { |
696 | LIMITED_METHOD_CONTRACT; |
697 | return (m_wFlags & mdcNotInline); |
698 | } |
699 | |
700 | inline void SetNotInline(BOOL set) |
701 | { |
702 | WRAPPER_NO_CONTRACT; |
703 | InterlockedUpdateFlags(mdcNotInline, set); |
704 | } |
705 | |
706 | #ifndef DACCESS_COMPILE |
707 | VOID EnsureActive(); |
708 | #endif |
709 | CHECK CheckActivated(); |
710 | |
711 | //================================================================ |
712 | // FCalls. |
713 | BOOL IsFCall() |
714 | { |
715 | WRAPPER_NO_CONTRACT; |
716 | return mcFCall == GetClassification(); |
717 | } |
718 | |
719 | BOOL IsFCallOrIntrinsic(); |
720 | |
721 | BOOL IsQCall(); |
722 | |
723 | //================================================================ |
724 | // |
725 | |
726 | inline void ClearFlagsOnUpdate() |
727 | { |
728 | WRAPPER_NO_CONTRACT; |
729 | SetNotInline(FALSE); |
730 | } |
731 | |
732 | // Restore the MethodDesc to it's initial, pristine state, so that |
733 | // it can be reused for new code (eg. for EnC, method rental, etc.) |
734 | // |
735 | // Things to think about before calling this: |
736 | // |
737 | // Does the caller need to free up the jitted code for the old IL |
738 | // (including any other IJitManager datastructures) ? |
739 | // Does the caller guarantee thread-safety ? |
740 | // |
741 | void Reset(); |
742 | |
743 | //================================================================ |
744 | // About the signature. |
745 | |
746 | BOOL IsVarArg(); |
747 | BOOL IsVoid(); |
748 | BOOL HasRetBuffArg(); |
749 | |
750 | // Returns the # of bytes of stack used by arguments. Does not include |
751 | // arguments passed in registers. |
752 | UINT SizeOfArgStack(); |
753 | |
754 | // Returns the # of bytes of stack used by arguments in a call from native to this function. |
755 | // Does not include arguments passed in registers. |
756 | UINT SizeOfNativeArgStack(); |
757 | |
758 | // Returns the # of bytes to pop after a call. Not necessary the |
759 | // same as SizeOfArgStack()! |
760 | UINT CbStackPop(); |
761 | |
762 | //================================================================ |
763 | // Unboxing stubs. |
764 | // |
765 | // Return TRUE if this is this a special stub used to implement delegates to an |
766 | // instance method in a value class and/or virtual methods on a value class. |
767 | // |
768 | // For every BoxedEntryPointStub there is associated unboxed-this-MethodDesc |
769 | // which accepts an unboxed "this" pointer. |
770 | // |
771 | // The action of a typical BoxedEntryPointStub is to |
772 | // bump up the this pointer by one word so that it points to the interior of the object |
773 | // and then call the underlying unboxed-this-MethodDesc. |
774 | // |
775 | // Additionally, if the non-BoxedEntryPointStub is RequiresInstMethodTableArg() |
776 | // then pass on the MethodTable as an extra argument to the |
777 | // underlying unboxed-this-MethodDesc. |
778 | BOOL IsUnboxingStub() |
779 | { |
780 | LIMITED_METHOD_DAC_CONTRACT; |
781 | |
782 | return (m_bFlags2 & enum_flag2_IsUnboxingStub) != 0; |
783 | } |
784 | |
785 | void SetIsUnboxingStub() |
786 | { |
787 | LIMITED_METHOD_CONTRACT; |
788 | m_bFlags2 |= enum_flag2_IsUnboxingStub; |
789 | } |
790 | |
791 | |
792 | //================================================================ |
793 | // Instantiating Stubs |
794 | // |
795 | // Return TRUE if this is this a special stub used to implement an |
796 | // instantiated generic method or per-instantiation static method. |
797 | // The action of an instantiating stub is |
798 | // * pass on a MethodTable or InstantiatedMethodDesc extra argument to shared code |
799 | BOOL IsInstantiatingStub(); |
800 | |
801 | |
802 | // A wrapper stub is either an unboxing stub or an instantiating stub |
803 | BOOL IsWrapperStub(); |
804 | MethodDesc *GetWrappedMethodDesc(); |
805 | MethodDesc *GetExistingWrappedMethodDesc(); |
806 | |
807 | //================================================================== |
808 | // Access the underlying metadata |
809 | |
810 | BOOL () |
811 | { |
812 | CONTRACTL |
813 | { |
814 | NOTHROW; |
815 | GC_NOTRIGGER; |
816 | SO_TOLERANT; |
817 | MODE_ANY; |
818 | } |
819 | CONTRACTL_END; |
820 | return IsIL() && !IsUnboxingStub() && GetRVA(); |
821 | } |
822 | |
823 | COR_ILMETHOD* (BOOL fAllowOverrides = FALSE); |
824 | |
825 | BOOL HasStoredSig() |
826 | { |
827 | LIMITED_METHOD_DAC_CONTRACT; |
828 | return IsEEImpl() || IsArray() || IsNoMetadata(); |
829 | } |
830 | |
831 | PCCOR_SIGNATURE GetSig(); |
832 | |
833 | void GetSig(PCCOR_SIGNATURE *ppSig, DWORD *pcSig); |
834 | SigParser GetSigParser(); |
835 | |
836 | // Convenience methods for common signature wrapper types. |
837 | SigPointer GetSigPointer(); |
838 | Signature GetSignature(); |
839 | |
840 | |
841 | void GetSigFromMetadata(IMDInternalImport * importer, |
842 | PCCOR_SIGNATURE * ppSig, |
843 | DWORD * pcSig); |
844 | |
845 | |
846 | IMDInternalImport* GetMDImport() const |
847 | { |
848 | WRAPPER_NO_CONTRACT; |
849 | Module *pModule = GetModule(); |
850 | PREFIX_ASSUME(pModule != NULL); |
851 | return pModule->GetMDImport(); |
852 | } |
853 | |
854 | #ifndef DACCESS_COMPILE |
855 | IMetaDataEmit* GetEmitter() |
856 | { |
857 | WRAPPER_NO_CONTRACT; |
858 | Module *pModule = GetModule(); |
859 | PREFIX_ASSUME(pModule != NULL); |
860 | return pModule->GetEmitter(); |
861 | } |
862 | |
863 | IMetaDataImport* GetRWImporter() |
864 | { |
865 | WRAPPER_NO_CONTRACT; |
866 | Module *pModule = GetModule(); |
867 | PREFIX_ASSUME(pModule != NULL); |
868 | return pModule->GetRWImporter(); |
869 | } |
870 | #endif // !DACCESS_COMPILE |
871 | |
872 | #ifdef FEATURE_COMINTEROP |
873 | WORD GetComSlot(); |
874 | LONG GetComDispid(); |
875 | #endif // FEATURE_COMINTEROP |
876 | |
877 | inline DWORD IsCtor() |
878 | { |
879 | WRAPPER_NO_CONTRACT; |
880 | return IsMdInstanceInitializer(GetAttrs(), GetName()); |
881 | } |
882 | |
883 | inline DWORD IsFinal() |
884 | { |
885 | WRAPPER_NO_CONTRACT; |
886 | return IsMdFinal(GetAttrs()); |
887 | } |
888 | |
889 | inline DWORD IsPrivate() |
890 | { |
891 | WRAPPER_NO_CONTRACT; |
892 | return IsMdPrivate(GetAttrs()); |
893 | } |
894 | |
895 | inline DWORD IsPublic() const |
896 | { |
897 | WRAPPER_NO_CONTRACT; |
898 | return IsMdPublic(GetAttrs()); |
899 | } |
900 | |
901 | inline DWORD IsProtected() const |
902 | { |
903 | WRAPPER_NO_CONTRACT; |
904 | return IsMdFamily(GetAttrs()); |
905 | } |
906 | |
907 | inline DWORD IsVirtual() |
908 | { |
909 | WRAPPER_NO_CONTRACT; |
910 | return IsMdVirtual(GetAttrs()); |
911 | } |
912 | |
913 | inline DWORD IsAbstract() |
914 | { |
915 | WRAPPER_NO_CONTRACT; |
916 | return IsMdAbstract(GetAttrs()); |
917 | } |
918 | |
919 | //================================================================== |
920 | // Flags.. |
921 | |
922 | inline void SetSynchronized() |
923 | { |
924 | LIMITED_METHOD_CONTRACT; |
925 | m_wFlags |= mdcSynchronized; |
926 | } |
927 | |
928 | inline DWORD IsSynchronized() |
929 | { |
930 | LIMITED_METHOD_DAC_CONTRACT; |
931 | return (m_wFlags & mdcSynchronized) != 0; |
932 | } |
933 | |
934 | // Be careful about races with profiler when using this method. The profiler can |
935 | // replace preimplemented code of the method with jitted code. |
936 | // Avoid code patterns like if(IsPreImplemented()) { PCODE pCode = GetPreImplementedCode(); ... }. |
937 | // Use PCODE pCode = GetPreImplementedCode(); if (pCode != NULL) { ... } instead. |
938 | BOOL IsPreImplemented() |
939 | { |
940 | LIMITED_METHOD_DAC_CONTRACT; |
941 | |
942 | return GetPreImplementedCode() != NULL; |
943 | } |
944 | |
945 | //================================================================== |
946 | // The MethodDesc in relation to the VTable it is associated with. |
947 | // WARNING: Not all MethodDescs have slots, nor do they all have |
948 | // entries in MethodTables. Beware. |
949 | |
950 | // Does the method has virtual slot? Note that methods implementing interfaces |
951 | // on value types do not have virtual slots, but they are marked as virtual in metadata. |
952 | inline BOOL IsVtableMethod() |
953 | { |
954 | LIMITED_METHOD_CONTRACT; |
955 | MethodTable *pMT = GetMethodTable(); |
956 | g_IBCLogger.LogMethodTableAccess(pMT); |
957 | return |
958 | !IsEnCAddedMethod() |
959 | // The slot numbers are currently meaningless for |
960 | // some unboxed-this-generic-method-instantiations |
961 | && !(pMT->IsValueType() && !IsStatic() && !IsUnboxingStub()) |
962 | && GetSlot() < pMT->GetNumVirtuals(); |
963 | } |
964 | |
965 | // Is this a default interface method (virtual non-abstract instance method) |
966 | inline BOOL IsDefaultInterfaceMethod() |
967 | { |
968 | LIMITED_METHOD_CONTRACT; |
969 | |
970 | #ifdef FEATURE_DEFAULT_INTERFACES |
971 | return (GetMethodTable()->IsInterface() && !IsStatic() && IsVirtual() && !IsAbstract()); |
972 | #else |
973 | return false; |
974 | #endif // FEATURE_DEFAULT_INTERFACES |
975 | } |
976 | |
977 | inline BOOL HasNonVtableSlot(); |
978 | |
979 | void SetHasNonVtableSlot() |
980 | { |
981 | LIMITED_METHOD_CONTRACT; |
982 | m_wFlags |= mdcHasNonVtableSlot; |
983 | } |
984 | |
985 | // duplicate methods |
986 | inline BOOL IsDuplicate() |
987 | { |
988 | LIMITED_METHOD_CONTRACT; |
989 | return (m_wFlags & mdcDuplicate) == mdcDuplicate; |
990 | } |
991 | |
992 | void SetDuplicate() |
993 | { |
994 | LIMITED_METHOD_CONTRACT; |
995 | // method table is not setup yet |
996 | //_ASSERTE(!GetClass()->IsInterface()); |
997 | m_wFlags |= mdcDuplicate; |
998 | } |
999 | |
1000 | //================================================================== |
1001 | // EnC |
1002 | |
1003 | inline BOOL IsEnCAddedMethod(); |
1004 | |
1005 | //================================================================== |
1006 | // |
1007 | |
1008 | inline EEClass* GetClass() |
1009 | { |
1010 | WRAPPER_NO_CONTRACT; |
1011 | MethodTable *pMT = GetMethodTable_NoLogging(); |
1012 | g_IBCLogger.LogEEClassAndMethodTableAccess(pMT); |
1013 | EEClass *pClass = pMT->GetClass_NoLogging(); |
1014 | PREFIX_ASSUME(pClass != NULL); |
1015 | return pClass; |
1016 | } |
1017 | |
1018 | inline PTR_MethodTable GetMethodTable() const; |
1019 | inline PTR_MethodTable GetMethodTable_NoLogging() const; |
1020 | |
1021 | inline DPTR(RelativeFixupPointer<PTR_MethodTable>) GetMethodTablePtr() const; |
1022 | |
1023 | public: |
1024 | inline MethodDescChunk* GetMethodDescChunk() const; |
1025 | inline int GetMethodDescIndex() const; |
1026 | // If this is an method desc. (whether non-generic shared-instantiated or exact-instantiated) |
1027 | // inside a shared class then get the method table for the representative |
1028 | // class. |
1029 | inline MethodTable* GetCanonicalMethodTable(); |
1030 | |
1031 | Module *GetModule() const; |
1032 | Module *GetModule_NoLogging() const; |
1033 | |
1034 | Assembly *GetAssembly() const |
1035 | { |
1036 | WRAPPER_NO_CONTRACT; |
1037 | Module *pModule = GetModule(); |
1038 | PREFIX_ASSUME(pModule != NULL); |
1039 | return pModule->GetAssembly(); |
1040 | } |
1041 | |
1042 | //================================================================== |
1043 | // The slot number of this method in the corresponding method table. |
1044 | // |
1045 | // Use with extreme caution. The slot number will not be |
1046 | // valid for EnC code or for MethodDescs representing instantiation |
1047 | // of generic methods. It may also not mean what you think it will mean |
1048 | // for strange method descs such as BoxedEntryPointStubs. |
1049 | // |
1050 | // In any case we should be moving to use slot numbers a lot less |
1051 | // since they make the EE code inflexible. |
1052 | |
1053 | inline WORD GetSlot() |
1054 | { |
1055 | LIMITED_METHOD_DAC_CONTRACT; |
1056 | #ifndef DACCESS_COMPILE |
1057 | // The DAC build uses this method to test for "sanity" of a MethodDesc, and |
1058 | // doesn't need the assert. |
1059 | _ASSERTE(! IsEnCAddedMethod() || !"Cannot get slot for method added via EnC" ); |
1060 | #endif // !DACCESS_COMPILE |
1061 | |
1062 | // Check if this MD is using the packed slot layout |
1063 | if (!RequiresFullSlotNumber()) |
1064 | { |
1065 | return (m_wSlotNumber & enum_packedSlotLayout_SlotMask); |
1066 | } |
1067 | |
1068 | return m_wSlotNumber; |
1069 | } |
1070 | |
1071 | inline VOID SetSlot(WORD wSlotNum) |
1072 | { |
1073 | LIMITED_METHOD_CONTRACT; |
1074 | |
1075 | // Check if we have to avoid using the packed slot layout |
1076 | if (wSlotNum > enum_packedSlotLayout_SlotMask) |
1077 | { |
1078 | SetRequiresFullSlotNumber(); |
1079 | } |
1080 | |
1081 | // Set only the portion of m_wSlotNumber we are using |
1082 | if (!RequiresFullSlotNumber()) |
1083 | { |
1084 | m_wSlotNumber &= ~enum_packedSlotLayout_SlotMask; |
1085 | m_wSlotNumber |= wSlotNum; |
1086 | } |
1087 | else |
1088 | { |
1089 | m_wSlotNumber = wSlotNum; |
1090 | } |
1091 | } |
1092 | |
1093 | inline BOOL IsVirtualSlot() |
1094 | { |
1095 | return GetSlot() < GetMethodTable()->GetNumVirtuals(); |
1096 | } |
1097 | inline BOOL IsVtableSlot() |
1098 | { |
1099 | return IsVirtualSlot() && !HasNonVtableSlot(); |
1100 | } |
1101 | |
1102 | TADDR GetAddrOfSlot(); |
1103 | |
1104 | PTR_MethodDesc GetDeclMethodDesc(UINT32 slotNumber); |
1105 | |
1106 | protected: |
1107 | inline void SetRequiresFullSlotNumber() |
1108 | { |
1109 | LIMITED_METHOD_CONTRACT; |
1110 | m_wFlags |= mdcRequiresFullSlotNumber; |
1111 | } |
1112 | |
1113 | inline DWORD RequiresFullSlotNumber() |
1114 | { |
1115 | LIMITED_METHOD_DAC_CONTRACT; |
1116 | return (m_wFlags & mdcRequiresFullSlotNumber) != 0; |
1117 | } |
1118 | |
1119 | public: |
1120 | mdMethodDef GetMemberDef() const; |
1121 | mdMethodDef GetMemberDef_NoLogging() const; |
1122 | |
1123 | #ifdef _DEBUG |
1124 | BOOL SanityCheck(); |
1125 | #endif // _DEBUG |
1126 | |
1127 | public: |
1128 | |
1129 | void SetMemberDef(mdMethodDef mb); |
1130 | |
1131 | //================================================================ |
1132 | // Set the offset of this method desc in a chunk table (which allows us |
1133 | // to work back to the method table/module pointer stored at the head of |
1134 | // the table. |
1135 | void SetChunkIndex(MethodDescChunk *pChunk); |
1136 | |
1137 | BOOL IsPointingToPrestub(); |
1138 | |
1139 | public: |
1140 | |
1141 | // TRUE iff it is possible to change the code this method will run using |
1142 | // the CodeVersionManager. |
1143 | // Note: EnC currently returns FALSE here because it uses its own seperate |
1144 | // scheme to manage versionability. We will likely want to converge them |
1145 | // at some point. |
1146 | BOOL IsVersionable() |
1147 | { |
1148 | #ifndef FEATURE_CODE_VERSIONING |
1149 | return FALSE; |
1150 | #else |
1151 | return IsVersionableWithPrecode() || IsVersionableWithJumpStamp(); |
1152 | #endif |
1153 | } |
1154 | |
1155 | // If true, these methods version using the CodeVersionManager and |
1156 | // switch between different code versions by updating the target of the precode. |
1157 | // Note: EnC returns FALSE - even though it uses precode updates it does not |
1158 | // use the CodeVersionManager right now |
1159 | BOOL IsVersionableWithPrecode() |
1160 | { |
1161 | #ifdef FEATURE_CODE_VERSIONING |
1162 | return |
1163 | // policy: which things do we want to version with a precode if possible |
1164 | IsEligibleForTieredCompilation() && |
1165 | |
1166 | // functional requirements: |
1167 | !IsZapped() && // NGEN directly invokes the pre-generated native code. |
1168 | // without necessarily going through the prestub or |
1169 | // precode |
1170 | HasNativeCodeSlot(); // the stable entry point will need to point at our |
1171 | // precode and not directly contain the native code. |
1172 | #else |
1173 | return FALSE; |
1174 | #endif |
1175 | } |
1176 | |
1177 | // If true, these methods version using the CodeVersionManager and switch between |
1178 | // different code versions by overwriting the first bytes of the method's initial |
1179 | // native code with a jmp instruction. |
1180 | BOOL IsVersionableWithJumpStamp() |
1181 | { |
1182 | #if defined(FEATURE_CODE_VERSIONING) && defined(FEATURE_JUMPSTAMP) |
1183 | return |
1184 | // for native image code this is policy, but for jitted code it is a functional requirement |
1185 | // to ensure the prolog is sufficiently large |
1186 | ReJitManager::IsReJITEnabled() && |
1187 | |
1188 | // functional requirement - the runtime doesn't expect both options to be possible |
1189 | !IsVersionableWithPrecode() && |
1190 | |
1191 | // functional requirement - we must be able to evacuate the prolog and the prolog must be big |
1192 | // enough, both of which are only designed to work on jitted code |
1193 | (IsIL() || IsNoMetadata()) && |
1194 | !IsUnboxingStub() && |
1195 | !IsInstantiatingStub() && |
1196 | |
1197 | // functional requirement - code version manager can't handle what would happen if the code |
1198 | // was collected |
1199 | !GetLoaderAllocator()->IsCollectible(); |
1200 | #else |
1201 | return FALSE; |
1202 | #endif |
1203 | } |
1204 | |
1205 | #ifdef FEATURE_TIERED_COMPILATION |
1206 | // Is this method allowed to be recompiled and the entrypoint redirected so that we |
1207 | // can optimize its performance? Eligibility is invariant for the lifetime of a method. |
1208 | BOOL IsEligibleForTieredCompilation() |
1209 | { |
1210 | LIMITED_METHOD_DAC_CONTRACT; |
1211 | |
1212 | // Keep in-sync with MethodTableBuilder::NeedsNativeCodeSlot(bmtMDMethod * pMDMethod) |
1213 | // to ensure native slots are available where needed. |
1214 | return g_pConfig->TieredCompilation() && |
1215 | !IsZapped() && |
1216 | !IsEnCMethod() && |
1217 | HasNativeCodeSlot() && |
1218 | !IsUnboxingStub() && |
1219 | !IsInstantiatingStub() && |
1220 | !IsDynamicMethod() && |
1221 | !GetLoaderAllocator()->IsCollectible() && |
1222 | !CORDisableJITOptimizations(GetModule()->GetDebuggerInfoBits()) && |
1223 | !CORProfilerDisableTieredCompilation(); |
1224 | } |
1225 | #endif |
1226 | |
1227 | bool RequestedAggressiveOptimization() |
1228 | { |
1229 | WRAPPER_NO_CONTRACT; |
1230 | |
1231 | return |
1232 | IsIL() && // only makes sense for IL methods, and this implies !IsNoMetadata() |
1233 | IsMiAggressiveOptimization(GetImplAttrs()); |
1234 | } |
1235 | |
1236 | // Does this method force the NativeCodeSlot to stay fixed after it |
1237 | // is first initialized to native code? Consumers of the native code |
1238 | // pointer need to be very careful about if and when they cache it |
1239 | // if it is not stable. |
1240 | // |
1241 | // The stability of the native code pointer is separate from the |
1242 | // stability of the entrypoint. A stable entrypoint can be a precode |
1243 | // which dispatches to an unstable native code pointer. |
1244 | BOOL IsNativeCodeStableAfterInit() |
1245 | { |
1246 | LIMITED_METHOD_DAC_CONTRACT; |
1247 | |
1248 | #if defined(FEATURE_JIT_PITCHING) |
1249 | if (IsPitchable()) |
1250 | return false; |
1251 | #endif |
1252 | |
1253 | return |
1254 | #ifdef FEATURE_TIERED_COMPILATION |
1255 | !IsEligibleForTieredCompilation() && |
1256 | #endif |
1257 | !IsEnCMethod(); |
1258 | } |
1259 | |
1260 | //Is this method currently pointing to native code that will never change? |
1261 | BOOL IsPointingToStableNativeCode() |
1262 | { |
1263 | LIMITED_METHOD_DAC_CONTRACT; |
1264 | |
1265 | if (!IsNativeCodeStableAfterInit()) |
1266 | return FALSE; |
1267 | |
1268 | return IsPointingToNativeCode(); |
1269 | } |
1270 | |
1271 | // Note: We are skipping the prestub based on addition information from the JIT. |
1272 | // (e.g. that the call is on same this ptr or that the this ptr is not null). |
1273 | // Thus we can end up with a running NGENed method for which IsPointingToNativeCode is false! |
1274 | BOOL IsPointingToNativeCode() |
1275 | { |
1276 | LIMITED_METHOD_DAC_CONTRACT; |
1277 | |
1278 | if (!HasStableEntryPoint()) |
1279 | return FALSE; |
1280 | |
1281 | if (!HasPrecode()) |
1282 | return TRUE; |
1283 | |
1284 | return GetPrecode()->IsPointingToNativeCode(GetNativeCode()); |
1285 | } |
1286 | |
1287 | // Be careful about races with profiler when using this method. The profiler can |
1288 | // replace preimplemented code of the method with jitted code. |
1289 | // Avoid code patterns like if(HasNativeCode()) { PCODE pCode = GetNativeCode(); ... }. |
1290 | // Use PCODE pCode = GetNativeCode(); if (pCode != NULL) { ... } instead. |
1291 | BOOL HasNativeCode() |
1292 | { |
1293 | LIMITED_METHOD_DAC_CONTRACT; |
1294 | |
1295 | return GetNativeCode() != NULL; |
1296 | } |
1297 | |
1298 | BOOL SetNativeCodeInterlocked(PCODE addr, PCODE pExpected = NULL); |
1299 | |
1300 | TADDR GetAddrOfNativeCodeSlot(); |
1301 | |
1302 | BOOL MayHaveNativeCode(); |
1303 | |
1304 | ULONG GetRVA(); |
1305 | |
1306 | public: |
1307 | |
1308 | // Returns preimplemented code of the method if method has one. |
1309 | // Returns NULL if method has no preimplemented code. |
1310 | // Be careful about races with profiler when using this method. The profiler can |
1311 | // replace preimplemented code of the method with jitted code. |
1312 | PCODE GetPreImplementedCode(); |
1313 | |
1314 | // Returns address of code to call. The address is good for one immediate invocation only. |
1315 | // Use GetMultiCallableAddrOfCode() to get address that can be invoked multiple times. |
1316 | // |
1317 | // Only call GetSingleCallableAddrOfCode() if you can guarantee that no virtualization is |
1318 | // necessary, or if you can guarantee that it has already happened. For instance, the frame of a |
1319 | // stackwalk has obviously been virtualized as much as it will be. |
1320 | // |
1321 | PCODE GetSingleCallableAddrOfCode() |
1322 | { |
1323 | WRAPPER_NO_CONTRACT; |
1324 | _ASSERTE(!IsGenericMethodDefinition()); |
1325 | return GetMethodEntryPoint(); |
1326 | } |
1327 | |
1328 | // This one is used to implement "ldftn". |
1329 | PCODE GetMultiCallableAddrOfCode(CORINFO_ACCESS_FLAGS accessFlags = CORINFO_ACCESS_LDFTN); |
1330 | |
1331 | // Internal version of GetMultiCallableAddrOfCode. Returns NULL if attempt to acquire directly |
1332 | // callable entrypoint would result into unnecesary allocation of indirection stub. Caller should use |
1333 | // indirect call via slot in this case. |
1334 | PCODE TryGetMultiCallableAddrOfCode(CORINFO_ACCESS_FLAGS accessFlags); |
1335 | |
1336 | // These return an address after resolving "virtual methods" correctly, including any |
1337 | // handling of context proxies, other thunking layers and also including |
1338 | // instantiation of generic virtual methods if required. |
1339 | // The first one returns an address which cannot be invoked |
1340 | // multiple times. Use GetMultiCallableAddrOfVirtualizedCode() for that. |
1341 | // |
1342 | // The code that implements these was taken verbatim from elsewhere in the |
1343 | // codebase, and there may be subtle differences between the two, e.g. with |
1344 | // regard to thunking. |
1345 | PCODE GetSingleCallableAddrOfVirtualizedCode(OBJECTREF *orThis, TypeHandle staticTH); |
1346 | PCODE GetMultiCallableAddrOfVirtualizedCode(OBJECTREF *orThis, TypeHandle staticTH); |
1347 | |
1348 | // The current method entrypoint. It is simply the value of the current method slot. |
1349 | // GetMethodEntryPoint() should be used to get an opaque method entrypoint, for instance |
1350 | // when copying or searching vtables. It should not be used to get address to call. |
1351 | // |
1352 | // GetSingleCallableAddrOfCode() and GetStableEntryPoint() are aliases with stricter preconditions. |
1353 | // Use of these aliases is as appropriate. |
1354 | // |
1355 | PCODE GetMethodEntryPoint(); |
1356 | |
1357 | //******************************************************************************* |
1358 | // Returns the address of the native code. The native code can be one of: |
1359 | // - jitted code if !IsPreImplemented() |
1360 | // - ngened code if IsPreImplemented() |
1361 | PCODE GetNativeCode(); |
1362 | |
1363 | #if defined(FEATURE_JIT_PITCHING) |
1364 | bool IsPitchable(); |
1365 | void PitchNativeCode(); |
1366 | #endif |
1367 | |
1368 | //================================================================ |
1369 | // FindOrCreateAssociatedMethodDesc |
1370 | // |
1371 | // You might think that every MethodDef in the metadata had |
1372 | // one and only one MethodDesc in the source... Well, how wrong |
1373 | // you are :-) |
1374 | // |
1375 | // Some MethodDefs can be associated with more than one MethodDesc. |
1376 | // This can happen because: |
1377 | // (1) The method is an instance method in a struct, which |
1378 | // can be called with either an unboxed "this" pointer or |
1379 | // a "boxed" this pointer.. There is a different MethodDesc for |
1380 | // these two cases. |
1381 | // (2) The method is a generic method. There is one primary |
1382 | // MethodDesc for each generic method, called the GenericMethodDefinition. |
1383 | // This is the one stored in the vtable. New MethodDescs will |
1384 | // be created for instantiations according to the scheme described |
1385 | // elsewhere in this file. |
1386 | // There are also various other stubs associated with MethodDesc, but these stubs |
1387 | // do not result in new MethodDescs. |
1388 | // |
1389 | // All of the above MethodDescs are called "associates" of the primary MethodDesc. |
1390 | // Note that the primary MethodDesc for an instance method on a struct is |
1391 | // the one that accepts an unboxed "this" pointer. |
1392 | // |
1393 | // FindOrCreateAssociatedMethodDesc is the _primary_ routine |
1394 | // in the codebase for getting an associated MethodDesc from a primary MethodDesc. |
1395 | // You should treat this routine as a black box, i.e. just specify the right |
1396 | // parameters and it will do all the hard work of finding the right |
1397 | // MethodDesc for you. |
1398 | // |
1399 | // This routine can be used for "normal" MethodDescs that have nothing |
1400 | // to do with generics. For example, if you need an BoxedEntryPointStub then |
1401 | // you may call this routine to get it. It may also return |
1402 | // the Primary MethodDesc itself if that MethodDesc is suitable given the |
1403 | // parameters. |
1404 | // |
1405 | // NOTE: The behaviour of this method is not thoroughly defined |
1406 | // if pPrimaryMD is not really a "primary" MD. Primary MDs are: |
1407 | // 1. Primary MDs are:never a generic method instantiation, |
1408 | // but are instead the "uninstantiated" generic MD. |
1409 | // 2. Primary MDs are never instantiating stubs. |
1410 | // 3. Primary MDs are never BoxedEntryPointStubs. |
1411 | // |
1412 | // We assert if cases (1) or (2) occur. However, some places in the |
1413 | // code pass in an BoxedEntryPointStub when pPrimaryMD is a virtual/interface method on |
1414 | // a struct. These cases are confusing and should be rooted |
1415 | // out: it is probably preferable in terms |
1416 | // of correctness to pass in the the corresponding non-unboxing MD. |
1417 | // |
1418 | // allowCreate may be set to FALSE to enforce that the method searched |
1419 | // should already be in existence - thus preventing creation and GCs during |
1420 | // inappropriate times. |
1421 | // |
1422 | static MethodDesc* FindOrCreateAssociatedMethodDesc(MethodDesc* pPrimaryMD, |
1423 | MethodTable *pExactMT, |
1424 | BOOL forceBoxedEntryPoint, |
1425 | Instantiation methodInst, |
1426 | BOOL allowInstParam, |
1427 | BOOL forceRemotableMethod = FALSE, |
1428 | BOOL allowCreate = TRUE, |
1429 | ClassLoadLevel level = CLASS_LOADED); |
1430 | |
1431 | // Normalize methoddesc for reflection |
1432 | static MethodDesc* FindOrCreateAssociatedMethodDescForReflection(MethodDesc *pMethod, |
1433 | TypeHandle instType, |
1434 | Instantiation methodInst); |
1435 | |
1436 | // True if a MD is an funny BoxedEntryPointStub (not from the method table) or |
1437 | // an MD for a generic instantiation...In other words the MethodDescs and the |
1438 | // MethodTable are guaranteed to be "tightly-knit", i.e. if one is present in |
1439 | // an NGEN image then then other will be, and if one is "used" at runtime then |
1440 | // the other will be too. |
1441 | BOOL IsTightlyBoundToMethodTable(); |
1442 | |
1443 | // For method descriptors which are non-generic this is the identity function |
1444 | // (except it returns the primary descriptor, not an BoxedEntryPointStub). |
1445 | // |
1446 | // For a generic method definition C<T>.m<U> this will return |
1447 | // C<__Canon>.m<__Canon> |
1448 | // |
1449 | // allowCreate may be set to FALSE to enforce that the method searched |
1450 | // should already be in existence - thus preventing creation and GCs during |
1451 | // inappropriate times. |
1452 | // |
1453 | MethodDesc * FindOrCreateTypicalSharedInstantiation(BOOL allowCreate = TRUE); |
1454 | |
1455 | // Given an object and an method descriptor for an instantiation of |
1456 | // a virtualized generic method, get the |
1457 | // corresponding instantiation of the target of a call. |
1458 | MethodDesc *ResolveGenericVirtualMethod(OBJECTREF *orThis); |
1459 | |
1460 | |
1461 | public: |
1462 | |
1463 | // does this function return an object reference? |
1464 | MetaSig::RETURNTYPE ReturnsObject( |
1465 | #ifdef _DEBUG |
1466 | bool supportStringConstructors = false, |
1467 | #endif |
1468 | MethodTable** pMT = NULL |
1469 | ); |
1470 | |
1471 | |
1472 | void Destruct(); |
1473 | |
1474 | public: |
1475 | // In general you don't want to call GetCallTarget - you want to |
1476 | // use either "call" directly or call MethodDesc::GetSingleCallableAddrOfVirtualizedCode and |
1477 | // then "CallTarget". Note that GetCallTarget is approximately GetSingleCallableAddrOfCode |
1478 | // but the additional wierdness that class-based-virtual calls (but not interface calls nor calls |
1479 | // on proxies) are resolved to their target. Because of this, many clients of "Call" (see above) |
1480 | // end up doing some resolution for interface calls and/or proxies themselves. |
1481 | PCODE GetCallTarget(OBJECTREF* pThisObj, TypeHandle ownerType = TypeHandle()); |
1482 | |
1483 | MethodImpl *GetMethodImpl(); |
1484 | |
1485 | |
1486 | #if defined(FEATURE_PREJIT ) && !defined(DACCESS_COMPILE) |
1487 | //================================================================ |
1488 | // Precompilation (NGEN) |
1489 | |
1490 | void Save(DataImage *image); |
1491 | void Fixup(DataImage *image); |
1492 | void FixupSlot(DataImage *image, PVOID p, SSIZE_T offset, ZapRelocationType type = IMAGE_REL_BASED_PTR); |
1493 | |
1494 | // |
1495 | // Helper class used to regroup MethodDesc chunks before saving them into NGen image. |
1496 | // The regrouping takes into account IBC data and optional NGen-specific MethodDesc members. |
1497 | // |
1498 | class SaveChunk |
1499 | { |
1500 | DataImage * m_pImage; |
1501 | |
1502 | ZapStoredStructure * m_pFirstNode; |
1503 | MethodDescChunk * m_pLastChunk; |
1504 | |
1505 | typedef enum _MethodPriorityEnum |
1506 | { |
1507 | NoFlags = -1, |
1508 | HotMethodDesc = 0x0, |
1509 | WriteableMethodDesc = 0x1, |
1510 | ColdMethodDesc = 0x2, |
1511 | ColdWriteableMethodDesc= ColdMethodDesc | WriteableMethodDesc |
1512 | |
1513 | } MethodPriorityEnum; |
1514 | |
1515 | struct MethodInfo |
1516 | { |
1517 | MethodDesc * m_pMD; |
1518 | //MethodPriorityEnum |
1519 | BYTE m_priority; |
1520 | |
1521 | BOOL m_fHasPrecode:1; |
1522 | BOOL m_fHasNativeCodeSlot:1; |
1523 | BOOL m_fHasFixupList:1; |
1524 | }; |
1525 | |
1526 | InlineSArray<MethodInfo, 20> m_methodInfos; |
1527 | |
1528 | static int __cdecl MethodInfoCmp(const void* a_, const void* b_); |
1529 | |
1530 | SIZE_T GetSavedMethodDescSize(MethodInfo * pMethodInfo); |
1531 | |
1532 | void SaveOneChunk(COUNT_T start, COUNT_T count, ULONG size, DWORD priority); |
1533 | |
1534 | public: |
1535 | SaveChunk(DataImage * image) |
1536 | : m_pImage(image), m_pFirstNode(NULL), m_pLastChunk(NULL) |
1537 | { |
1538 | LIMITED_METHOD_CONTRACT; |
1539 | } |
1540 | |
1541 | void Append(MethodDesc * pMD); |
1542 | |
1543 | ZapStoredStructure * Save(); |
1544 | }; |
1545 | |
1546 | bool CanSkipDoPrestub(MethodDesc * callerMD, |
1547 | CorInfoIndirectCallReason *pReason, |
1548 | CORINFO_ACCESS_FLAGS accessFlags = CORINFO_ACCESS_ANY); |
1549 | |
1550 | // This is different from !IsRestored() in that it checks if restoring |
1551 | // will ever be needed for this ngened data-structure. |
1552 | // This is to be used at ngen time of a dependent module to determine |
1553 | // if it can be accessed directly, or if the restoring mechanism needs |
1554 | // to be hooked in. |
1555 | BOOL NeedsRestore(DataImage *image, BOOL fAssumeMethodTableRestored = FALSE) |
1556 | { |
1557 | WRAPPER_NO_CONTRACT; |
1558 | return ComputeNeedsRestore(image, NULL, fAssumeMethodTableRestored); |
1559 | } |
1560 | |
1561 | BOOL ComputeNeedsRestore(DataImage *image, TypeHandleList *pVisited, BOOL fAssumeMethodTableRestored = FALSE); |
1562 | |
1563 | // |
1564 | // After the zapper compiles all code in a module it may attempt |
1565 | // to populate entries in all dictionaries |
1566 | // associated with instantiations of generic methods. This is an optional step - nothing will |
1567 | // go wrong at runtime except we may get more one-off calls to JIT_GenericHandle. |
1568 | // Although these are one-off we prefer to avoid them since they touch metadata |
1569 | // pages. |
1570 | // |
1571 | // Fully populating a dictionary may in theory load more types, methods etc. However |
1572 | // for the moment only those entries that refer to things that |
1573 | // are already loaded will be filled in. |
1574 | void PrepopulateDictionary(DataImage * image, BOOL nonExpansive); |
1575 | |
1576 | #endif // FEATURE_PREJIT && !DACCESS_COMPILE |
1577 | |
1578 | TADDR GetFixupList(); |
1579 | |
1580 | BOOL IsRestored_NoLogging(); |
1581 | BOOL IsRestored(); |
1582 | void CheckRestore(ClassLoadLevel level = CLASS_LOADED); |
1583 | |
1584 | //================================================================ |
1585 | // Running the Prestub preparation step. |
1586 | |
1587 | // The stub produced by prestub requires method desc to be passed |
1588 | // in dedicated register. Used to implement stubs shared between |
1589 | // MethodDescs (e.g. PInvoke stubs) |
1590 | BOOL RequiresMethodDescCallingConvention(BOOL fEstimateForChunk = FALSE); |
1591 | |
1592 | // Returns true if the method has to have stable entrypoint always. |
1593 | BOOL RequiresStableEntryPoint(BOOL fEstimateForChunk = FALSE); |
1594 | |
1595 | // |
1596 | // Backpatch method slots |
1597 | // |
1598 | // Arguments: |
1599 | // pMT - cached value of code:MethodDesc::GetMethodTable() |
1600 | // pDispatchingMT - method table of the object that the method is being dispatched on, can be NULL. |
1601 | // fFullBackPatch - indicates whether to patch all possible slots, including the ones |
1602 | // expensive to patch |
1603 | // |
1604 | // Return value: |
1605 | // stable entry point (code:MethodDesc::GetStableEntryPoint()) |
1606 | // |
1607 | PCODE DoBackpatch(MethodTable * pMT, MethodTable * pDispatchingMT, BOOL fFullBackPatch); |
1608 | |
1609 | PCODE DoPrestub(MethodTable *pDispatchingMT); |
1610 | |
1611 | VOID GetMethodInfo(SString &namespaceOrClassName, SString &methodName, SString &methodSignature); |
1612 | VOID GetMethodInfoWithNewSig(SString &namespaceOrClassName, SString &methodName, SString &methodSignature); |
1613 | VOID GetMethodInfoNoSig(SString &namespaceOrClassName, SString &methodName); |
1614 | VOID GetFullMethodInfo(SString& fullMethodSigName); |
1615 | |
1616 | BOOL HasTypeEquivalentStructParameters(); |
1617 | |
1618 | typedef void (*WalkValueTypeParameterFnPtr)(Module *pModule, mdToken token, Module *pDefModule, mdToken tkDefToken, const SigParser *ptr, SigTypeContext *pTypeContext, void *pData); |
1619 | |
1620 | void WalkValueTypeParameters(MethodTable *pMT, WalkValueTypeParameterFnPtr function, void *pData); |
1621 | |
1622 | void PrepareForUseAsADependencyOfANativeImage() |
1623 | { |
1624 | WRAPPER_NO_CONTRACT; |
1625 | if (!IsZapped() && !HaveValueTypeParametersBeenWalked()) |
1626 | PrepareForUseAsADependencyOfANativeImageWorker(); |
1627 | } |
1628 | |
1629 | void PrepareForUseAsADependencyOfANativeImageWorker(); |
1630 | |
1631 | //================================================================ |
1632 | // The actual data stored in a MethodDesc follows. |
1633 | |
1634 | protected: |
1635 | enum { |
1636 | // There are flags available for use here (currently 5 flags bits are available); however, new bits are hard to come by, so any new flags bits should |
1637 | // have a fairly strong justification for existence. |
1638 | enum_flag3_TokenRemainderMask = 0x3FFF, // This must equal METHOD_TOKEN_REMAINDER_MASK calculated higher in this file |
1639 | // These are seperate to allow the flags space available and used to be obvious here |
1640 | // and for the logic that splits the token to be algorithmically generated based on the |
1641 | // #define |
1642 | enum_flag3_HasForwardedValuetypeParameter = 0x4000, // Indicates that a type-forwarded type is used as a valuetype parameter (this flag is only valid for ngenned items) |
1643 | enum_flag3_ValueTypeParametersWalked = 0x4000, // Indicates that all typeref's in the signature of the method have been resolved to typedefs (or that process failed) (this flag is only valid for non-ngenned methods) |
1644 | enum_flag3_DoesNotHaveEquivalentValuetypeParameters = 0x8000, // Indicates that we have verified that there are no equivalent valuetype parameters for this method |
1645 | }; |
1646 | UINT16 m_wFlags3AndTokenRemainder; |
1647 | |
1648 | BYTE m_chunkIndex; |
1649 | |
1650 | enum { |
1651 | // enum_flag2_HasPrecode implies that enum_flag2_HasStableEntryPoint is set. |
1652 | enum_flag2_HasStableEntryPoint = 0x01, // The method entrypoint is stable (either precode or actual code) |
1653 | enum_flag2_HasPrecode = 0x02, // Precode has been allocated for this method |
1654 | |
1655 | enum_flag2_IsUnboxingStub = 0x04, |
1656 | enum_flag2_HasNativeCodeSlot = 0x08, // Has slot for native code |
1657 | |
1658 | enum_flag2_IsJitIntrinsic = 0x10, // Jit may expand method as an intrinsic |
1659 | |
1660 | // unused = 0x20, |
1661 | // unused = 0x40, |
1662 | // unused = 0x80, |
1663 | }; |
1664 | BYTE m_bFlags2; |
1665 | |
1666 | // The slot number of this MethodDesc in the vtable array. |
1667 | // Note that we may store other information in the high bits if available -- |
1668 | // see enum_packedSlotLayout and mdcRequiresFullSlotNumber for details. |
1669 | WORD m_wSlotNumber; |
1670 | |
1671 | enum { |
1672 | enum_packedSlotLayout_SlotMask = 0x03FF, |
1673 | enum_packedSlotLayout_NameHashMask = 0xFC00 |
1674 | }; |
1675 | |
1676 | WORD m_wFlags; |
1677 | |
1678 | |
1679 | |
1680 | public: |
1681 | #ifdef DACCESS_COMPILE |
1682 | void EnumMemoryRegions(CLRDataEnumMemoryFlags flags); |
1683 | #endif |
1684 | |
1685 | public: |
1686 | inline DWORD GetClassification() const |
1687 | { |
1688 | LIMITED_METHOD_DAC_CONTRACT; |
1689 | |
1690 | return (m_wFlags & mdcClassification); |
1691 | } |
1692 | |
1693 | inline void SetClassification(DWORD classification) |
1694 | { |
1695 | LIMITED_METHOD_CONTRACT; |
1696 | _ASSERTE((m_wFlags & mdcClassification) == 0); |
1697 | m_wFlags |= classification; |
1698 | } |
1699 | |
1700 | inline BOOL HasNativeCodeSlot() |
1701 | { |
1702 | LIMITED_METHOD_DAC_CONTRACT; |
1703 | return (m_bFlags2 & enum_flag2_HasNativeCodeSlot) != 0; |
1704 | } |
1705 | |
1706 | inline void SetHasNativeCodeSlot() |
1707 | { |
1708 | LIMITED_METHOD_CONTRACT; |
1709 | m_bFlags2 |= enum_flag2_HasNativeCodeSlot; |
1710 | } |
1711 | |
1712 | inline BOOL IsJitIntrinsic() |
1713 | { |
1714 | LIMITED_METHOD_DAC_CONTRACT; |
1715 | return (m_bFlags2 & enum_flag2_IsJitIntrinsic) != 0; |
1716 | } |
1717 | |
1718 | inline void SetIsJitIntrinsic() |
1719 | { |
1720 | LIMITED_METHOD_CONTRACT; |
1721 | m_bFlags2 |= enum_flag2_IsJitIntrinsic; |
1722 | } |
1723 | |
1724 | static const SIZE_T s_ClassificationSizeTable[]; |
1725 | |
1726 | static SIZE_T GetBaseSize(DWORD classification) |
1727 | { |
1728 | LIMITED_METHOD_DAC_CONTRACT; |
1729 | _ASSERTE(classification < mdcClassificationCount); |
1730 | return s_ClassificationSizeTable[classification]; |
1731 | } |
1732 | |
1733 | SIZE_T GetBaseSize() |
1734 | { |
1735 | LIMITED_METHOD_DAC_CONTRACT; |
1736 | return GetBaseSize(GetClassification()); |
1737 | } |
1738 | |
1739 | SIZE_T SizeOf(); |
1740 | |
1741 | WORD InterlockedUpdateFlags3(WORD wMask, BOOL fSet); |
1742 | |
1743 | #ifdef FEATURE_TYPEEQUIVALENCE |
1744 | inline BOOL DoesNotHaveEquivalentValuetypeParameters() |
1745 | { |
1746 | LIMITED_METHOD_DAC_CONTRACT; |
1747 | return (m_wFlags3AndTokenRemainder & enum_flag3_DoesNotHaveEquivalentValuetypeParameters) != 0; |
1748 | } |
1749 | |
1750 | inline void SetDoesNotHaveEquivalentValuetypeParameters() |
1751 | { |
1752 | LIMITED_METHOD_CONTRACT; |
1753 | InterlockedUpdateFlags3(enum_flag3_DoesNotHaveEquivalentValuetypeParameters, TRUE); |
1754 | } |
1755 | #endif // FEATURE_TYPEEQUIVALENCE |
1756 | |
1757 | inline BOOL HasForwardedValuetypeParameter() |
1758 | { |
1759 | LIMITED_METHOD_DAC_CONTRACT; |
1760 | // This should only be asked of Zapped MethodDescs |
1761 | _ASSERTE(IsZapped()); |
1762 | return (m_wFlags3AndTokenRemainder & enum_flag3_HasForwardedValuetypeParameter) != 0; |
1763 | } |
1764 | |
1765 | inline void SetHasForwardedValuetypeParameter() |
1766 | { |
1767 | LIMITED_METHOD_CONTRACT; |
1768 | InterlockedUpdateFlags3(enum_flag3_HasForwardedValuetypeParameter, TRUE); |
1769 | } |
1770 | |
1771 | inline BOOL HaveValueTypeParametersBeenWalked() |
1772 | { |
1773 | LIMITED_METHOD_DAC_CONTRACT; |
1774 | |
1775 | // This should only be asked of non-Zapped MethodDescs, and only during execution (not compilation) |
1776 | _ASSERTE(!IsZapped() && !IsCompilationProcess()); |
1777 | |
1778 | return (m_wFlags3AndTokenRemainder & enum_flag3_ValueTypeParametersWalked) != 0; |
1779 | } |
1780 | |
1781 | inline void SetValueTypeParametersWalked() |
1782 | { |
1783 | LIMITED_METHOD_CONTRACT; |
1784 | |
1785 | _ASSERTE(!IsZapped() && !IsCompilationProcess()); |
1786 | |
1787 | InterlockedUpdateFlags3(enum_flag3_ValueTypeParametersWalked, TRUE); |
1788 | } |
1789 | |
1790 | // |
1791 | // Optional MethodDesc slots appear after the end of base MethodDesc in this order: |
1792 | // |
1793 | |
1794 | // class MethodImpl; // Present if HasMethodImplSlot() is true |
1795 | |
1796 | typedef RelativePointer<PCODE> NonVtableSlot; // Present if HasNonVtableSlot() is true |
1797 | // RelativePointer for NGen, PCODE for JIT |
1798 | |
1799 | #define FIXUP_LIST_MASK 1 |
1800 | typedef RelativePointer<TADDR> NativeCodeSlot; // Present if HasNativeCodeSlot() is true |
1801 | // lower order bit (FIXUP_LIST_MASK) used to determine if FixupListSlot is present |
1802 | typedef RelativePointer<TADDR> FixupListSlot; |
1803 | |
1804 | // Stub Dispatch code |
1805 | public: |
1806 | MethodDesc *GetInterfaceMD(); |
1807 | |
1808 | // StubMethodInfo for use in creating RuntimeMethodHandles |
1809 | REFLECTMETHODREF GetStubMethodInfo(); |
1810 | |
1811 | PrecodeType GetPrecodeType(); |
1812 | |
1813 | |
1814 | // --------------------------------------------------------------------------------- |
1815 | // IL based Code generation pipeline |
1816 | // --------------------------------------------------------------------------------- |
1817 | |
1818 | #ifndef DACCESS_COMPILE |
1819 | public: |
1820 | PCODE PrepareInitialCode(); |
1821 | PCODE PrepareCode(NativeCodeVersion codeVersion); |
1822 | PCODE PrepareCode(PrepareCodeConfig* pConfig); |
1823 | |
1824 | private: |
1825 | PCODE PrepareILBasedCode(PrepareCodeConfig* pConfig); |
1826 | PCODE GetPrecompiledCode(PrepareCodeConfig* pConfig); |
1827 | PCODE GetPrecompiledNgenCode(PrepareCodeConfig* pConfig); |
1828 | PCODE GetPrecompiledR2RCode(PrepareCodeConfig* pConfig); |
1829 | PCODE GetMulticoreJitCode(); |
1830 | COR_ILMETHOD_DECODER* GetAndVerifyILHeader(PrepareCodeConfig* pConfig, COR_ILMETHOD_DECODER* pIlDecoderMemory); |
1831 | COR_ILMETHOD_DECODER* GetAndVerifyMetadataILHeader(PrepareCodeConfig* pConfig, COR_ILMETHOD_DECODER* pIlDecoderMemory); |
1832 | COR_ILMETHOD_DECODER* GetAndVerifyNoMetadataILHeader(); |
1833 | PCODE JitCompileCode(PrepareCodeConfig* pConfig); |
1834 | PCODE JitCompileCodeLockedEventWrapper(PrepareCodeConfig* pConfig, JitListLockEntry* pEntry); |
1835 | PCODE JitCompileCodeLocked(PrepareCodeConfig* pConfig, JitListLockEntry* pLockEntry, ULONG* pSizeOfCode, CORJIT_FLAGS* pFlags); |
1836 | #endif // DACCESS_COMPILE |
1837 | |
1838 | #ifdef HAVE_GCCOVER |
1839 | private: |
1840 | static CrstStatic m_GCCoverCrst; |
1841 | |
1842 | public: |
1843 | static void Init(); |
1844 | #endif |
1845 | }; |
1846 | |
1847 | #ifndef DACCESS_COMPILE |
1848 | class PrepareCodeConfig |
1849 | { |
1850 | public: |
1851 | PrepareCodeConfig(); |
1852 | PrepareCodeConfig(NativeCodeVersion nativeCodeVersion, BOOL needsMulticoreJitNotification, BOOL mayUsePrecompiledCode); |
1853 | MethodDesc* GetMethodDesc(); |
1854 | NativeCodeVersion GetCodeVersion(); |
1855 | BOOL NeedsMulticoreJitNotification(); |
1856 | BOOL MayUsePrecompiledCode(); |
1857 | virtual PCODE IsJitCancellationRequested(); |
1858 | virtual BOOL SetNativeCode(PCODE pCode, PCODE * ppAlternateCodeToUse); |
1859 | virtual COR_ILMETHOD* (); |
1860 | virtual CORJIT_FLAGS GetJitCompilationFlags(); |
1861 | BOOL ProfilerRejectedPrecompiledCode(); |
1862 | BOOL ReadyToRunRejectedPrecompiledCode(); |
1863 | void SetProfilerRejectedPrecompiledCode(); |
1864 | void SetReadyToRunRejectedPrecompiledCode(); |
1865 | |
1866 | protected: |
1867 | MethodDesc* m_pMethodDesc; |
1868 | NativeCodeVersion m_nativeCodeVersion; |
1869 | BOOL m_needsMulticoreJitNotification; |
1870 | BOOL m_mayUsePrecompiledCode; |
1871 | BOOL m_ProfilerRejectedPrecompiledCode; |
1872 | BOOL m_ReadyToRunRejectedPrecompiledCode; |
1873 | }; |
1874 | |
1875 | #ifdef FEATURE_CODE_VERSIONING |
1876 | class VersionedPrepareCodeConfig : public PrepareCodeConfig |
1877 | { |
1878 | public: |
1879 | VersionedPrepareCodeConfig(); |
1880 | VersionedPrepareCodeConfig(NativeCodeVersion codeVersion); |
1881 | HRESULT FinishConfiguration(); |
1882 | virtual PCODE IsJitCancellationRequested(); |
1883 | virtual BOOL SetNativeCode(PCODE pCode, PCODE * ppAlternateCodeToUse); |
1884 | virtual COR_ILMETHOD* (); |
1885 | virtual CORJIT_FLAGS GetJitCompilationFlags(); |
1886 | private: |
1887 | ILCodeVersion m_ilCodeVersion; |
1888 | }; |
1889 | #endif // FEATURE_CODE_VERSIONING |
1890 | #endif // DACCESS_COMPILE |
1891 | |
1892 | /******************************************************************/ |
1893 | |
1894 | // A code:MethodDescChunk is a container that holds one or more code:MethodDesc. Logically it is just |
1895 | // compression. Basically fields that are common among methods descs in the chunk are stored in the chunk |
1896 | // and the MethodDescs themselves just store and index that allows them to find their Chunk. Semantically |
1897 | // a code:MethodDescChunk is just a set of code:MethodDesc. |
1898 | class MethodDescChunk |
1899 | { |
1900 | friend class MethodDesc; |
1901 | friend class CheckAsmOffsets; |
1902 | #if defined(FEATURE_PREJIT) && !defined(DACCESS_COMPILE) |
1903 | friend class MethodDesc::SaveChunk; |
1904 | #endif |
1905 | #ifdef DACCESS_COMPILE |
1906 | friend class NativeImageDumper; |
1907 | #endif // DACCESS_COMPILE |
1908 | |
1909 | enum { |
1910 | enum_flag_TokenRangeMask = 0x03FF, // This must equal METHOD_TOKEN_RANGE_MASK calculated higher in this file |
1911 | // These are seperate to allow the flags space available and used to be obvious here |
1912 | // and for the logic that splits the token to be algorithmically generated based on the |
1913 | // #define |
1914 | enum_flag_HasCompactEntrypoints = 0x4000, // Compact temporary entry points |
1915 | enum_flag_IsZapped = 0x8000, // This chunk lives in NGen module |
1916 | }; |
1917 | |
1918 | public: |
1919 | // |
1920 | // Allocates methodDescCount identical MethodDescs in smallest possible number of chunks. |
1921 | // If methodDescCount is zero, one chunk with maximum number of MethodDescs is allocated. |
1922 | // |
1923 | static MethodDescChunk *CreateChunk(LoaderHeap *pHeap, DWORD methodDescCount, |
1924 | DWORD classification, |
1925 | BOOL fNonVtableSlot, |
1926 | BOOL fNativeCodeSlot, |
1927 | BOOL fComPlusCallInfo, |
1928 | MethodTable *initialMT, |
1929 | class AllocMemTracker *pamTracker); |
1930 | |
1931 | BOOL HasTemporaryEntryPoints() |
1932 | { |
1933 | LIMITED_METHOD_CONTRACT; |
1934 | return !IsZapped(); |
1935 | } |
1936 | |
1937 | TADDR GetTemporaryEntryPoints() |
1938 | { |
1939 | LIMITED_METHOD_CONTRACT; |
1940 | _ASSERTE(HasTemporaryEntryPoints()); |
1941 | return *(dac_cast<DPTR(TADDR)>(this) - 1); |
1942 | } |
1943 | |
1944 | PCODE GetTemporaryEntryPoint(int index); |
1945 | |
1946 | void EnsureTemporaryEntryPointsCreated(LoaderAllocator *pLoaderAllocator, AllocMemTracker *pamTracker) |
1947 | { |
1948 | CONTRACTL |
1949 | { |
1950 | THROWS; |
1951 | GC_NOTRIGGER; |
1952 | MODE_ANY; |
1953 | } |
1954 | CONTRACTL_END; |
1955 | |
1956 | if (GetTemporaryEntryPoints() == NULL) |
1957 | CreateTemporaryEntryPoints(pLoaderAllocator, pamTracker); |
1958 | } |
1959 | |
1960 | void CreateTemporaryEntryPoints(LoaderAllocator *pLoaderAllocator, AllocMemTracker *pamTracker); |
1961 | |
1962 | #ifdef HAS_COMPACT_ENTRYPOINTS |
1963 | // |
1964 | // There two implementation options for temporary entrypoints: |
1965 | // |
1966 | // (1) Compact entrypoints. They provide as dense entrypoints as possible, but can't be patched |
1967 | // to point to the final code. The call to unjitted method is indirect call via slot. |
1968 | // |
1969 | // (2) Precodes. The precode will be patched to point to the final code eventually, thus |
1970 | // the temporary entrypoint can be embedded in the code. The call to unjitted method is |
1971 | // direct call to direct jump. |
1972 | // |
1973 | // We use (1) for x86 and (2) for 64-bit to get the best performance on each platform. |
1974 | // For ARM (1) is used. |
1975 | |
1976 | TADDR AllocateCompactEntryPoints(LoaderAllocator *pLoaderAllocator, AllocMemTracker *pamTracker); |
1977 | |
1978 | static MethodDesc* GetMethodDescFromCompactEntryPoint(PCODE addr, BOOL fSpeculative = FALSE); |
1979 | static SIZE_T SizeOfCompactEntryPoints(int count); |
1980 | |
1981 | static BOOL IsCompactEntryPointAtAddress(PCODE addr); |
1982 | |
1983 | #ifdef _TARGET_ARM_ |
1984 | static int GetCompactEntryPointMaxCount (); |
1985 | #endif // _TARGET_ARM_ |
1986 | #endif // HAS_COMPACT_ENTRYPOINTS |
1987 | |
1988 | FORCEINLINE PTR_MethodTable GetMethodTable() |
1989 | { |
1990 | LIMITED_METHOD_DAC_CONTRACT; |
1991 | return m_methodTable.GetValue(PTR_HOST_MEMBER_TADDR(MethodDescChunk, this, m_methodTable)); |
1992 | } |
1993 | |
1994 | inline DPTR(RelativeFixupPointer<PTR_MethodTable>) GetMethodTablePtr() const |
1995 | { |
1996 | LIMITED_METHOD_DAC_CONTRACT; |
1997 | return dac_cast<DPTR(RelativeFixupPointer<PTR_MethodTable>)>(PTR_HOST_MEMBER_TADDR(MethodDescChunk, this, m_methodTable)); |
1998 | } |
1999 | |
2000 | #ifndef DACCESS_COMPILE |
2001 | inline void SetMethodTable(MethodTable * pMT) |
2002 | { |
2003 | LIMITED_METHOD_CONTRACT; |
2004 | _ASSERTE(m_methodTable.IsNull()); |
2005 | _ASSERTE(pMT != NULL); |
2006 | m_methodTable.SetValue(pMT); |
2007 | } |
2008 | |
2009 | inline void SetSizeAndCount(ULONG sizeOfMethodDescs, COUNT_T methodDescCount) |
2010 | { |
2011 | LIMITED_METHOD_CONTRACT; |
2012 | |
2013 | _ASSERTE(FitsIn<BYTE>((sizeOfMethodDescs / MethodDesc::ALIGNMENT) - 1)); |
2014 | m_size = static_cast<BYTE>((sizeOfMethodDescs / MethodDesc::ALIGNMENT) - 1); |
2015 | _ASSERTE(SizeOf() == sizeof(MethodDescChunk) + sizeOfMethodDescs); |
2016 | |
2017 | _ASSERTE(FitsIn<BYTE>(methodDescCount - 1)); |
2018 | m_count = static_cast<BYTE>(methodDescCount - 1); |
2019 | _ASSERTE(GetCount() == methodDescCount); |
2020 | } |
2021 | #endif // !DACCESS_COMPILE |
2022 | |
2023 | #ifdef FEATURE_PREJIT |
2024 | #ifndef DACCESS_COMPILE |
2025 | inline void RestoreMTPointer(ClassLoadLevel level = CLASS_LOADED) |
2026 | { |
2027 | LIMITED_METHOD_CONTRACT; |
2028 | Module::RestoreMethodTablePointer(&m_methodTable, NULL, level); |
2029 | } |
2030 | #endif // !DACCESS_COMPILE |
2031 | #endif // FEATURE_PREJIT |
2032 | |
2033 | #ifndef DACCESS_COMPILE |
2034 | void SetNextChunk(MethodDescChunk *chunk) |
2035 | { |
2036 | LIMITED_METHOD_CONTRACT; |
2037 | m_next.SetValueMaybeNull(chunk); |
2038 | } |
2039 | #endif // !DACCESS_COMPILE |
2040 | |
2041 | PTR_MethodDescChunk GetNextChunk() |
2042 | { |
2043 | LIMITED_METHOD_CONTRACT; |
2044 | return m_next.GetValueMaybeNull(PTR_HOST_MEMBER_TADDR(MethodDescChunk, this, m_next)); |
2045 | } |
2046 | |
2047 | UINT32 GetCount() |
2048 | { |
2049 | LIMITED_METHOD_DAC_CONTRACT; |
2050 | return m_count + 1; |
2051 | } |
2052 | |
2053 | BOOL IsZapped() |
2054 | { |
2055 | LIMITED_METHOD_DAC_CONTRACT; |
2056 | #ifdef FEATURE_PREJIT |
2057 | return (m_flagsAndTokenRange & enum_flag_IsZapped) != 0; |
2058 | #else |
2059 | return FALSE; |
2060 | #endif |
2061 | } |
2062 | |
2063 | inline BOOL HasCompactEntryPoints() |
2064 | { |
2065 | LIMITED_METHOD_DAC_CONTRACT; |
2066 | |
2067 | #ifdef HAS_COMPACT_ENTRYPOINTS |
2068 | return (m_flagsAndTokenRange & enum_flag_HasCompactEntrypoints) != 0; |
2069 | #else |
2070 | return FALSE; |
2071 | #endif |
2072 | } |
2073 | |
2074 | inline UINT16 GetTokRange() |
2075 | { |
2076 | LIMITED_METHOD_DAC_CONTRACT; |
2077 | return m_flagsAndTokenRange & enum_flag_TokenRangeMask; |
2078 | } |
2079 | |
2080 | inline SIZE_T SizeOf() |
2081 | { |
2082 | LIMITED_METHOD_DAC_CONTRACT; |
2083 | return sizeof(MethodDescChunk) + (m_size + 1) * MethodDesc::ALIGNMENT; |
2084 | } |
2085 | |
2086 | inline MethodDesc *GetFirstMethodDesc() |
2087 | { |
2088 | LIMITED_METHOD_DAC_CONTRACT; |
2089 | return PTR_MethodDesc(dac_cast<TADDR>(this) + sizeof(MethodDescChunk)); |
2090 | } |
2091 | |
2092 | // Maximum size of one chunk (corresponts to the maximum of m_size = 0xFF) |
2093 | static const SIZE_T MaxSizeOfMethodDescs = 0x100 * MethodDesc::ALIGNMENT; |
2094 | |
2095 | #ifdef DACCESS_COMPILE |
2096 | void EnumMemoryRegions(CLRDataEnumMemoryFlags flags); |
2097 | #endif |
2098 | |
2099 | private: |
2100 | void SetIsZapped() |
2101 | { |
2102 | LIMITED_METHOD_CONTRACT; |
2103 | m_flagsAndTokenRange |= enum_flag_IsZapped; |
2104 | } |
2105 | |
2106 | void SetHasCompactEntryPoints() |
2107 | { |
2108 | LIMITED_METHOD_CONTRACT; |
2109 | m_flagsAndTokenRange |= enum_flag_HasCompactEntrypoints; |
2110 | } |
2111 | |
2112 | void SetTokenRange(UINT16 tokenRange) |
2113 | { |
2114 | LIMITED_METHOD_CONTRACT; |
2115 | _ASSERTE((tokenRange & ~enum_flag_TokenRangeMask) == 0); |
2116 | static_assert_no_msg(enum_flag_TokenRangeMask == METHOD_TOKEN_RANGE_MASK); |
2117 | m_flagsAndTokenRange = (m_flagsAndTokenRange & ~enum_flag_TokenRangeMask) | tokenRange; |
2118 | } |
2119 | |
2120 | RelativeFixupPointer<PTR_MethodTable> m_methodTable; |
2121 | |
2122 | RelativePointer<PTR_MethodDescChunk> m_next; |
2123 | |
2124 | BYTE m_size; // The size of this chunk minus 1 (in multiples of MethodDesc::ALIGNMENT) |
2125 | BYTE m_count; // The number of MethodDescs in this chunk minus 1 |
2126 | UINT16 m_flagsAndTokenRange; |
2127 | |
2128 | // Followed by array of method descs... |
2129 | }; |
2130 | |
2131 | inline int MethodDesc::GetMethodDescIndex() const |
2132 | { |
2133 | LIMITED_METHOD_DAC_CONTRACT; |
2134 | |
2135 | return m_chunkIndex; |
2136 | } |
2137 | |
2138 | inline MethodDescChunk *MethodDesc::GetMethodDescChunk() const |
2139 | { |
2140 | LIMITED_METHOD_DAC_CONTRACT; |
2141 | |
2142 | return |
2143 | PTR_MethodDescChunk(dac_cast<TADDR>(this) - |
2144 | (sizeof(MethodDescChunk) + (GetMethodDescIndex() * MethodDesc::ALIGNMENT))); |
2145 | } |
2146 | |
2147 | // convert an entry point into a MethodDesc |
2148 | MethodDesc* Entry2MethodDesc(PCODE entryPoint, MethodTable *pMT); |
2149 | |
2150 | |
2151 | typedef DPTR(class StoredSigMethodDesc) PTR_StoredSigMethodDesc; |
2152 | class StoredSigMethodDesc : public MethodDesc |
2153 | { |
2154 | public: |
2155 | // Put the sig RVA in here - this allows us to avoid |
2156 | // touching the method desc table when mscorlib is prejitted. |
2157 | |
2158 | RelativePointer<TADDR> m_pSig; |
2159 | DWORD m_cSig; |
2160 | #ifdef _WIN64 |
2161 | // m_dwExtendedFlags is not used by StoredSigMethodDesc itself. |
2162 | // It is used by child classes. We allocate the space here to get |
2163 | // optimal layout. |
2164 | DWORD m_dwExtendedFlags; |
2165 | #endif |
2166 | |
2167 | TADDR GetSigRVA() |
2168 | { |
2169 | LIMITED_METHOD_DAC_CONTRACT; |
2170 | return RelativePointer<TADDR>::GetValueMaybeNullAtPtr(PTR_HOST_MEMBER_TADDR(StoredSigMethodDesc, this, m_pSig)); |
2171 | } |
2172 | |
2173 | bool HasStoredMethodSig(void) |
2174 | { |
2175 | LIMITED_METHOD_DAC_CONTRACT; |
2176 | return !m_pSig.IsNull(); |
2177 | } |
2178 | PCCOR_SIGNATURE GetStoredMethodSig(DWORD* sigLen = NULL) |
2179 | { |
2180 | LIMITED_METHOD_DAC_CONTRACT; |
2181 | if (sigLen) |
2182 | { |
2183 | *sigLen = m_cSig; |
2184 | } |
2185 | #ifdef DACCESS_COMPILE |
2186 | return (PCCOR_SIGNATURE) |
2187 | DacInstantiateTypeByAddress(GetSigRVA(), m_cSig, true); |
2188 | #else // !DACCESS_COMPILE |
2189 | g_IBCLogger.LogNDirectCodeAccess(this); |
2190 | return (PCCOR_SIGNATURE) m_pSig.GetValueMaybeNull(); |
2191 | #endif // !DACCESS_COMPILE |
2192 | } |
2193 | void SetStoredMethodSig(PCCOR_SIGNATURE sig, DWORD sigBytes) |
2194 | { |
2195 | #ifndef DACCESS_COMPILE |
2196 | m_pSig.SetValueMaybeNull((TADDR)sig); |
2197 | m_cSig = sigBytes; |
2198 | #endif // !DACCESS_COMPILE |
2199 | } |
2200 | |
2201 | #ifdef DACCESS_COMPILE |
2202 | void EnumMemoryRegions(CLRDataEnumMemoryFlags flags); |
2203 | #endif |
2204 | }; |
2205 | |
2206 | //----------------------------------------------------------------------- |
2207 | // Operations specific to FCall methods. We use a derived class to get |
2208 | // the compiler involved in enforcing proper method type usage. |
2209 | // DO NOT ADD FIELDS TO THIS CLASS. |
2210 | //----------------------------------------------------------------------- |
2211 | |
2212 | class FCallMethodDesc : public MethodDesc |
2213 | { |
2214 | #ifdef DACCESS_COMPILE |
2215 | friend class NativeImageDumper; |
2216 | #endif |
2217 | |
2218 | DWORD m_dwECallID; |
2219 | #ifdef _WIN64 |
2220 | DWORD m_padding; |
2221 | #endif |
2222 | |
2223 | public: |
2224 | void SetECallID(DWORD dwID) |
2225 | { |
2226 | LIMITED_METHOD_CONTRACT; |
2227 | m_dwECallID = dwID; |
2228 | } |
2229 | |
2230 | DWORD GetECallID() |
2231 | { |
2232 | LIMITED_METHOD_CONTRACT; |
2233 | return m_dwECallID; |
2234 | } |
2235 | }; |
2236 | |
2237 | class HostCodeHeap; |
2238 | class LCGMethodResolver; |
2239 | typedef DPTR(LCGMethodResolver) PTR_LCGMethodResolver; |
2240 | class ILStubResolver; |
2241 | typedef DPTR(ILStubResolver) PTR_ILStubResolver; |
2242 | class DynamicResolver; |
2243 | typedef DPTR(DynamicResolver) PTR_DynamicResolver; |
2244 | |
2245 | class DynamicMethodDesc : public StoredSigMethodDesc |
2246 | { |
2247 | friend class ILStubCache; |
2248 | friend class ILStubState; |
2249 | friend class DynamicMethodTable; |
2250 | friend class MethodDesc; |
2251 | #ifdef DACCESS_COMPILE |
2252 | friend class NativeImageDumper; |
2253 | #endif |
2254 | |
2255 | protected: |
2256 | RelativePointer<PTR_CUTF8> m_pszMethodName; |
2257 | PTR_DynamicResolver m_pResolver; |
2258 | |
2259 | #ifndef _WIN64 |
2260 | // We use m_dwExtendedFlags from StoredSigMethodDesc on WIN64 |
2261 | DWORD m_dwExtendedFlags; // see DynamicMethodDesc::ExtendedFlags enum |
2262 | #endif |
2263 | |
2264 | typedef enum ExtendedFlags |
2265 | { |
2266 | nomdAttrs = 0x0000FFFF, // method attributes (LCG) |
2267 | nomdILStubAttrs = mdMemberAccessMask | mdStatic, // method attributes (IL stubs) |
2268 | |
2269 | // attributes (except mdStatic and mdMemberAccessMask) have different meaning for IL stubs |
2270 | // mdMemberAccessMask = 0x0007, |
2271 | nomdReverseStub = 0x0008, |
2272 | // mdStatic = 0x0010, |
2273 | nomdCALLIStub = 0x0020, |
2274 | nomdDelegateStub = 0x0040, |
2275 | nomdCopyCtorArgs = 0x0080, |
2276 | nomdUnbreakable = 0x0100, |
2277 | nomdDelegateCOMStub = 0x0200, // CLR->COM or COM->CLR call via a delegate (WinRT specific) |
2278 | nomdSignatureNeedsRestore = 0x0400, |
2279 | nomdStubNeedsCOMStarted = 0x0800, // EnsureComStarted must be called before executing the method |
2280 | nomdMulticastStub = 0x1000, |
2281 | nomdUnboxingILStub = 0x2000, |
2282 | nomdSecureDelegateStub = 0x4000, |
2283 | |
2284 | nomdILStub = 0x00010000, |
2285 | nomdLCGMethod = 0x00020000, |
2286 | nomdStackArgSize = 0xFFFC0000, // native stack arg size for IL stubs |
2287 | } ExtendedFlags; |
2288 | |
2289 | public: |
2290 | bool IsILStub() { LIMITED_METHOD_DAC_CONTRACT; return !!(m_dwExtendedFlags & nomdILStub); } |
2291 | bool IsLCGMethod() { LIMITED_METHOD_DAC_CONTRACT; return !!(m_dwExtendedFlags & nomdLCGMethod); } |
2292 | |
2293 | inline PTR_DynamicResolver GetResolver(); |
2294 | inline PTR_LCGMethodResolver GetLCGMethodResolver(); |
2295 | inline PTR_ILStubResolver GetILStubResolver(); |
2296 | |
2297 | PTR_CUTF8 GetMethodName() |
2298 | { |
2299 | LIMITED_METHOD_DAC_CONTRACT; |
2300 | return RelativePointer<PTR_CUTF8>::GetValueMaybeNullAtPtr(PTR_HOST_MEMBER_TADDR(DynamicMethodDesc, this, m_pszMethodName)); |
2301 | } |
2302 | |
2303 | WORD GetAttrs() |
2304 | { |
2305 | LIMITED_METHOD_CONTRACT; |
2306 | return (IsILStub() ? (m_dwExtendedFlags & nomdILStubAttrs) : (m_dwExtendedFlags & nomdAttrs)); |
2307 | } |
2308 | |
2309 | DWORD GetExtendedFlags() |
2310 | { |
2311 | LIMITED_METHOD_CONTRACT; |
2312 | return m_dwExtendedFlags; |
2313 | } |
2314 | |
2315 | WORD GetNativeStackArgSize() |
2316 | { |
2317 | LIMITED_METHOD_DAC_CONTRACT; |
2318 | _ASSERTE(IsILStub()); |
2319 | return (WORD)((m_dwExtendedFlags & nomdStackArgSize) >> 16); |
2320 | } |
2321 | |
2322 | void SetNativeStackArgSize(WORD cbArgSize) |
2323 | { |
2324 | LIMITED_METHOD_CONTRACT; |
2325 | _ASSERTE(IsILStub() && (cbArgSize % sizeof(SLOT)) == 0); |
2326 | m_dwExtendedFlags = (m_dwExtendedFlags & ~nomdStackArgSize) | ((DWORD)cbArgSize << 16); |
2327 | } |
2328 | |
2329 | void SetHasCopyCtorArgs(bool value) |
2330 | { |
2331 | LIMITED_METHOD_CONTRACT; |
2332 | if (value) |
2333 | { |
2334 | m_dwExtendedFlags |= nomdCopyCtorArgs; |
2335 | } |
2336 | } |
2337 | |
2338 | void SetUnbreakable(bool value) |
2339 | { |
2340 | LIMITED_METHOD_CONTRACT; |
2341 | if (value) |
2342 | { |
2343 | m_dwExtendedFlags |= nomdUnbreakable; |
2344 | } |
2345 | } |
2346 | |
2347 | void SetSignatureNeedsRestore(bool value) |
2348 | { |
2349 | LIMITED_METHOD_CONTRACT; |
2350 | if (value) |
2351 | { |
2352 | m_dwExtendedFlags |= nomdSignatureNeedsRestore; |
2353 | } |
2354 | } |
2355 | |
2356 | void SetStubNeedsCOMStarted(bool value) |
2357 | { |
2358 | LIMITED_METHOD_CONTRACT; |
2359 | if (value) |
2360 | { |
2361 | m_dwExtendedFlags |= nomdStubNeedsCOMStarted; |
2362 | } |
2363 | } |
2364 | |
2365 | bool IsRestored() |
2366 | { |
2367 | LIMITED_METHOD_CONTRACT; |
2368 | |
2369 | if (IsSignatureNeedsRestore()) |
2370 | { |
2371 | // Since we don't update the signatreNeedsRestore bit when we actually |
2372 | // restore the signature, the bit will have a stall value. The signature |
2373 | // bit in the metadata will always contain the correct, up-to-date |
2374 | // information. |
2375 | Volatile<BYTE> *pVolatileSig = (Volatile<BYTE> *)GetStoredMethodSig(); |
2376 | if ((*pVolatileSig & IMAGE_CEE_CS_CALLCONV_NEEDSRESTORE) != 0) |
2377 | return false; |
2378 | } |
2379 | |
2380 | return true; |
2381 | } |
2382 | |
2383 | bool IsReverseStub() { LIMITED_METHOD_DAC_CONTRACT; _ASSERTE(IsILStub()); return (0 != (m_dwExtendedFlags & nomdReverseStub)); } |
2384 | bool IsCALLIStub() { LIMITED_METHOD_DAC_CONTRACT; _ASSERTE(IsILStub()); return (0 != (m_dwExtendedFlags & nomdCALLIStub)); } |
2385 | bool IsDelegateStub() { LIMITED_METHOD_DAC_CONTRACT; _ASSERTE(IsILStub()); return (0 != (m_dwExtendedFlags & nomdDelegateStub)); } |
2386 | bool IsCLRToCOMStub() { LIMITED_METHOD_CONTRACT; _ASSERTE(IsILStub()); return ((0 == (m_dwExtendedFlags & mdStatic)) && !IsReverseStub() && !IsDelegateStub()); } |
2387 | bool IsCOMToCLRStub() { LIMITED_METHOD_CONTRACT; _ASSERTE(IsILStub()); return ((0 == (m_dwExtendedFlags & mdStatic)) && IsReverseStub()); } |
2388 | bool IsPInvokeStub() { LIMITED_METHOD_CONTRACT; _ASSERTE(IsILStub()); return ((0 != (m_dwExtendedFlags & mdStatic)) && !IsReverseStub() && !IsCALLIStub()); } |
2389 | bool HasCopyCtorArgs() { LIMITED_METHOD_CONTRACT; _ASSERTE(IsILStub()); return (0 != (m_dwExtendedFlags & nomdCopyCtorArgs)); } |
2390 | bool IsUnbreakable() { LIMITED_METHOD_CONTRACT; _ASSERTE(IsILStub()); return (0 != (m_dwExtendedFlags & nomdUnbreakable)); } |
2391 | bool IsDelegateCOMStub() { LIMITED_METHOD_CONTRACT; _ASSERTE(IsILStub()); return (0 != (m_dwExtendedFlags & nomdDelegateCOMStub)); } |
2392 | bool IsSignatureNeedsRestore() { LIMITED_METHOD_CONTRACT; _ASSERTE(IsILStub()); return (0 != (m_dwExtendedFlags & nomdSignatureNeedsRestore)); } |
2393 | bool IsStubNeedsCOMStarted() { LIMITED_METHOD_CONTRACT; _ASSERTE(IsILStub()); return (0 != (m_dwExtendedFlags & nomdStubNeedsCOMStarted)); } |
2394 | #ifdef FEATURE_MULTICASTSTUB_AS_IL |
2395 | bool IsMulticastStub() { |
2396 | LIMITED_METHOD_DAC_CONTRACT; |
2397 | _ASSERTE(IsILStub()); |
2398 | return !!(m_dwExtendedFlags & nomdMulticastStub); |
2399 | } |
2400 | #endif |
2401 | #ifdef FEATURE_STUBS_AS_IL |
2402 | bool IsSecureDelegateStub() { |
2403 | LIMITED_METHOD_DAC_CONTRACT; |
2404 | _ASSERTE(IsILStub()); |
2405 | return !!(m_dwExtendedFlags & nomdSecureDelegateStub); |
2406 | } |
2407 | bool IsUnboxingILStub() { |
2408 | LIMITED_METHOD_DAC_CONTRACT; |
2409 | _ASSERTE(IsILStub()); |
2410 | return !!(m_dwExtendedFlags & nomdUnboxingILStub); |
2411 | } |
2412 | #endif |
2413 | |
2414 | // Whether the stub takes a context argument that is an interop MethodDesc. |
2415 | bool HasMDContextArg() |
2416 | { |
2417 | LIMITED_METHOD_CONTRACT; |
2418 | return ((IsCLRToCOMStub() && !IsDelegateCOMStub()) || IsPInvokeStub()); |
2419 | } |
2420 | |
2421 | void Restore(); |
2422 | void Fixup(DataImage* image); |
2423 | // |
2424 | // following implementations defined in DynamicMethod.cpp |
2425 | // |
2426 | void Destroy(BOOL fDomainUnload = FALSE); |
2427 | }; |
2428 | |
2429 | |
2430 | class ArrayMethodDesc : public StoredSigMethodDesc |
2431 | { |
2432 | public: |
2433 | // The VTABLE for an array look like |
2434 | |
2435 | // System.Object Vtable |
2436 | // System.Array Vtable |
2437 | // type[] Vtable |
2438 | // Get(<rank specific) |
2439 | // Set(<rank specific) |
2440 | // Address(<rank specific) |
2441 | // .ctor(int) // Possibly more |
2442 | |
2443 | enum { |
2444 | ARRAY_FUNC_GET = 0, |
2445 | ARRAY_FUNC_SET = 1, |
2446 | ARRAY_FUNC_ADDRESS = 2, |
2447 | ARRAY_FUNC_CTOR = 3, // Anything >= ARRAY_FUNC_CTOR is .ctor |
2448 | }; |
2449 | |
2450 | // Get the index of runtime provided array method |
2451 | DWORD GetArrayFuncIndex() |
2452 | { |
2453 | LIMITED_METHOD_DAC_CONTRACT; |
2454 | |
2455 | // The ru |
2456 | DWORD dwSlot = GetSlot(); |
2457 | DWORD dwVirtuals = GetMethodTable()->GetNumVirtuals(); |
2458 | _ASSERTE(dwSlot >= dwVirtuals); |
2459 | return dwSlot - dwVirtuals; |
2460 | } |
2461 | |
2462 | LPCUTF8 GetMethodName(); |
2463 | DWORD GetAttrs(); |
2464 | CorInfoIntrinsics GetIntrinsicID(); |
2465 | }; |
2466 | |
2467 | #ifdef HAS_NDIRECT_IMPORT_PRECODE |
2468 | typedef NDirectImportPrecode NDirectImportThunkGlue; |
2469 | #else // HAS_NDIRECT_IMPORT_PRECODE |
2470 | |
2471 | class NDirectImportThunkGlue |
2472 | { |
2473 | PVOID m_dummy; // Dummy field to make the alignment right |
2474 | |
2475 | public: |
2476 | LPVOID GetEntrypoint() |
2477 | { |
2478 | LIMITED_METHOD_CONTRACT; |
2479 | return NULL; |
2480 | } |
2481 | void Init(MethodDesc *pMethod) |
2482 | { |
2483 | LIMITED_METHOD_CONTRACT; |
2484 | } |
2485 | }; |
2486 | #ifdef FEATURE_PREJIT |
2487 | PORTABILITY_WARNING("NDirectImportThunkGlue" ); |
2488 | #endif // FEATURE_PREJIT |
2489 | |
2490 | #endif // HAS_NDIRECT_IMPORT_PRECODE |
2491 | |
2492 | typedef DPTR(NDirectImportThunkGlue) PTR_NDirectImportThunkGlue; |
2493 | |
2494 | |
2495 | // |
2496 | // This struct consolidates the writeable parts of the NDirectMethodDesc |
2497 | // so that we can eventually layout a read-only NDirectMethodDesc with a pointer |
2498 | // to the writeable parts in an ngen image |
2499 | // |
2500 | class NDirectWriteableData |
2501 | { |
2502 | public: |
2503 | // The JIT generates an indirect call through this location in some cases. |
2504 | // Initialized to NDirectImportThunkGlue. Patched to the true target or |
2505 | // host interceptor stub or alignment thunk after linking. |
2506 | LPVOID m_pNDirectTarget; |
2507 | }; |
2508 | |
2509 | typedef DPTR(NDirectWriteableData) PTR_NDirectWriteableData; |
2510 | |
2511 | //----------------------------------------------------------------------- |
2512 | // Operations specific to NDirect methods. We use a derived class to get |
2513 | // the compiler involved in enforcing proper method type usage. |
2514 | // DO NOT ADD FIELDS TO THIS CLASS. |
2515 | //----------------------------------------------------------------------- |
2516 | class NDirectMethodDesc : public MethodDesc |
2517 | { |
2518 | public: |
2519 | struct temp1 |
2520 | { |
2521 | // If we are hosted, stack imbalance MDA is active, or alignment thunks are needed, |
2522 | // we will intercept m_pNDirectTarget. The true target is saved here. |
2523 | LPVOID m_pNativeNDirectTarget; |
2524 | |
2525 | // Information about the entrypoint |
2526 | RelativePointer<PTR_CUTF8> m_pszEntrypointName; |
2527 | |
2528 | union |
2529 | { |
2530 | RelativePointer<PTR_CUTF8> m_pszLibName; |
2531 | DWORD m_dwECallID; // ECallID for QCalls |
2532 | }; |
2533 | |
2534 | // The writeable part of the methoddesc. |
2535 | #if defined(FEATURE_NGEN_RELOCS_OPTIMIZATIONS) |
2536 | RelativePointer<PTR_NDirectWriteableData> m_pWriteableData; |
2537 | #else |
2538 | PlainPointer<PTR_NDirectWriteableData> m_pWriteableData; |
2539 | #endif |
2540 | |
2541 | #ifdef HAS_NDIRECT_IMPORT_PRECODE |
2542 | RelativePointer<PTR_NDirectImportThunkGlue> m_pImportThunkGlue; |
2543 | #else // HAS_NDIRECT_IMPORT_PRECODE |
2544 | NDirectImportThunkGlue m_ImportThunkGlue; |
2545 | #endif // HAS_NDIRECT_IMPORT_PRECODE |
2546 | |
2547 | ULONG m_DefaultDllImportSearchPathsAttributeValue; // DefaultDllImportSearchPathsAttribute is saved. |
2548 | |
2549 | // Various attributes needed at runtime. |
2550 | WORD m_wFlags; |
2551 | |
2552 | #if defined(_TARGET_X86_) |
2553 | // Size of outgoing arguments (on stack). Note that in order to get the @n stdcall name decoration, |
2554 | // it may be necessary to subtract 4 as the hidden large structure pointer parameter does not count. |
2555 | // See code:kStdCallWithRetBuf |
2556 | WORD m_cbStackArgumentSize; |
2557 | #endif // defined(_TARGET_X86_) |
2558 | |
2559 | // This field gets set only when this MethodDesc is marked as PreImplemented |
2560 | RelativePointer<PTR_MethodDesc> m_pStubMD; |
2561 | |
2562 | } ndirect; |
2563 | |
2564 | enum Flags |
2565 | { |
2566 | // There are two groups of flag bits here each which gets initialized |
2567 | // at different times. |
2568 | |
2569 | // |
2570 | // Group 1: The init group. |
2571 | // |
2572 | // This group is set during MethodDesc construction. No race issues |
2573 | // here since they are initialized before the MD is ever published |
2574 | // and never change after that. |
2575 | |
2576 | kEarlyBound = 0x0001, // IJW managed->unmanaged thunk. Standard [sysimport] stuff otherwise. |
2577 | |
2578 | kHasSuppressUnmanagedCodeAccess = 0x0002, |
2579 | |
2580 | kDefaultDllImportSearchPathsIsCached = 0x0004, // set if we cache attribute value. |
2581 | |
2582 | // kUnusedMask = 0x0008 |
2583 | |
2584 | // |
2585 | // Group 2: The runtime group. |
2586 | // |
2587 | // This group is set during runtime potentially by multiple threads |
2588 | // at the same time. All flags in this category has to be set via interlocked operation. |
2589 | // |
2590 | kIsMarshalingRequiredCached = 0x0010, // Set if we have cached the results of marshaling required computation |
2591 | kCachedMarshalingRequired = 0x0020, // The result of the marshaling required computation |
2592 | |
2593 | kNativeAnsi = 0x0040, |
2594 | |
2595 | kLastError = 0x0080, // setLastError keyword specified |
2596 | kNativeNoMangle = 0x0100, // nomangle keyword specified |
2597 | |
2598 | kVarArgs = 0x0200, |
2599 | kStdCall = 0x0400, |
2600 | kThisCall = 0x0800, |
2601 | |
2602 | kIsQCall = 0x1000, |
2603 | |
2604 | kDefaultDllImportSearchPathsStatus = 0x2000, // either method has custom attribute or not. |
2605 | |
2606 | kHasCopyCtorArgs = 0x4000, |
2607 | |
2608 | kStdCallWithRetBuf = 0x8000, // Call returns large structure, only valid if kStdCall is also set |
2609 | |
2610 | }; |
2611 | |
2612 | // Retrieves the cached result of marshaling required computation, or performs the computation |
2613 | // if the result is not cached yet. |
2614 | BOOL MarshalingRequired() |
2615 | { |
2616 | STANDARD_VM_CONTRACT; |
2617 | |
2618 | if ((ndirect.m_wFlags & kIsMarshalingRequiredCached) == 0) |
2619 | { |
2620 | // Compute the flag and cache the result |
2621 | InterlockedSetNDirectFlags(kIsMarshalingRequiredCached | |
2622 | (ComputeMarshalingRequired() ? kCachedMarshalingRequired : 0)); |
2623 | } |
2624 | _ASSERTE((ndirect.m_wFlags & kIsMarshalingRequiredCached) != 0); |
2625 | return (ndirect.m_wFlags & kCachedMarshalingRequired) != 0; |
2626 | } |
2627 | |
2628 | BOOL ComputeMarshalingRequired(); |
2629 | |
2630 | // Atomically set specified flags. Only setting of the bits is supported. |
2631 | void InterlockedSetNDirectFlags(WORD wFlags); |
2632 | |
2633 | void SetIsEarlyBound() |
2634 | { |
2635 | LIMITED_METHOD_CONTRACT; |
2636 | ndirect.m_wFlags |= kEarlyBound; |
2637 | } |
2638 | |
2639 | BOOL IsEarlyBound() |
2640 | { |
2641 | LIMITED_METHOD_CONTRACT; |
2642 | return (ndirect.m_wFlags & kEarlyBound) != 0; |
2643 | } |
2644 | |
2645 | BOOL IsNativeAnsi() const |
2646 | { |
2647 | LIMITED_METHOD_CONTRACT; |
2648 | |
2649 | return (ndirect.m_wFlags & kNativeAnsi) != 0; |
2650 | } |
2651 | |
2652 | BOOL IsNativeNoMangled() const |
2653 | { |
2654 | LIMITED_METHOD_CONTRACT; |
2655 | |
2656 | return (ndirect.m_wFlags & kNativeNoMangle) != 0; |
2657 | } |
2658 | |
2659 | |
2660 | DWORD GetECallID() const |
2661 | { |
2662 | LIMITED_METHOD_CONTRACT; |
2663 | |
2664 | _ASSERTE(IsQCall()); |
2665 | return ndirect.m_dwECallID; |
2666 | } |
2667 | |
2668 | void SetECallID(DWORD dwID) |
2669 | { |
2670 | LIMITED_METHOD_CONTRACT; |
2671 | |
2672 | _ASSERTE(IsQCall()); |
2673 | ndirect.m_dwECallID = dwID; |
2674 | } |
2675 | |
2676 | PTR_CUTF8 GetLibNameRaw() |
2677 | { |
2678 | LIMITED_METHOD_DAC_CONTRACT; |
2679 | |
2680 | return RelativePointer<PTR_CUTF8>::GetValueMaybeNullAtPtr(PTR_HOST_MEMBER_TADDR(NDirectMethodDesc, this, ndirect.m_pszLibName)); |
2681 | } |
2682 | |
2683 | #ifndef DACCESS_COMPILE |
2684 | LPCUTF8 GetLibName() const |
2685 | { |
2686 | LIMITED_METHOD_CONTRACT; |
2687 | |
2688 | return IsQCall() ? "QCall" : ndirect.m_pszLibName.GetValueMaybeNull(); |
2689 | } |
2690 | #endif // !DACCESS_COMPILE |
2691 | |
2692 | PTR_CUTF8 GetEntrypointName() const |
2693 | { |
2694 | LIMITED_METHOD_DAC_CONTRACT; |
2695 | |
2696 | return RelativePointer<PTR_CUTF8>::GetValueMaybeNullAtPtr(PTR_HOST_MEMBER_TADDR(NDirectMethodDesc, this, ndirect.m_pszEntrypointName)); |
2697 | } |
2698 | |
2699 | BOOL IsVarArgs() const |
2700 | { |
2701 | LIMITED_METHOD_DAC_CONTRACT; |
2702 | |
2703 | return (ndirect.m_wFlags & kVarArgs) != 0; |
2704 | } |
2705 | |
2706 | BOOL IsStdCall() const |
2707 | { |
2708 | LIMITED_METHOD_DAC_CONTRACT; |
2709 | |
2710 | return (ndirect.m_wFlags & kStdCall) != 0; |
2711 | } |
2712 | |
2713 | BOOL IsThisCall() const |
2714 | { |
2715 | LIMITED_METHOD_DAC_CONTRACT; |
2716 | |
2717 | return (ndirect.m_wFlags & kThisCall) != 0; |
2718 | } |
2719 | |
2720 | // Returns TRUE if this MethodDesc is internal call from mscorlib to mscorwks |
2721 | BOOL IsQCall() const |
2722 | { |
2723 | LIMITED_METHOD_DAC_CONTRACT; |
2724 | |
2725 | return (ndirect.m_wFlags & kIsQCall) != 0; |
2726 | } |
2727 | |
2728 | BOOL HasDefaultDllImportSearchPathsAttribute(); |
2729 | |
2730 | BOOL IsDefaultDllImportSearchPathsAttributeCached() |
2731 | { |
2732 | LIMITED_METHOD_CONTRACT; |
2733 | return (ndirect.m_wFlags & kDefaultDllImportSearchPathsIsCached) != 0; |
2734 | } |
2735 | |
2736 | ULONG DefaultDllImportSearchPathsAttributeCachedValue() |
2737 | { |
2738 | LIMITED_METHOD_CONTRACT; |
2739 | return ndirect.m_DefaultDllImportSearchPathsAttributeValue & 0xFFFFFFFD; |
2740 | } |
2741 | |
2742 | BOOL DllImportSearchAssemblyDirectory() |
2743 | { |
2744 | LIMITED_METHOD_CONTRACT; |
2745 | return (ndirect.m_DefaultDllImportSearchPathsAttributeValue & 0x2) != 0; |
2746 | } |
2747 | |
2748 | BOOL HasCopyCtorArgs() const |
2749 | { |
2750 | LIMITED_METHOD_DAC_CONTRACT; |
2751 | |
2752 | return (ndirect.m_wFlags & kHasCopyCtorArgs) != 0; |
2753 | } |
2754 | |
2755 | void SetHasCopyCtorArgs(BOOL value) |
2756 | { |
2757 | WRAPPER_NO_CONTRACT; |
2758 | |
2759 | if (value) |
2760 | { |
2761 | InterlockedSetNDirectFlags(kHasCopyCtorArgs); |
2762 | } |
2763 | } |
2764 | |
2765 | BOOL IsStdCallWithRetBuf() const |
2766 | { |
2767 | LIMITED_METHOD_DAC_CONTRACT; |
2768 | |
2769 | return (ndirect.m_wFlags & kStdCallWithRetBuf) != 0; |
2770 | } |
2771 | |
2772 | PTR_NDirectWriteableData GetWriteableData() const |
2773 | { |
2774 | LIMITED_METHOD_DAC_CONTRACT; |
2775 | |
2776 | return ReadPointer(this, &NDirectMethodDesc::ndirect, &decltype(NDirectMethodDesc::ndirect)::m_pWriteableData); |
2777 | } |
2778 | |
2779 | PTR_NDirectImportThunkGlue GetNDirectImportThunkGlue() |
2780 | { |
2781 | LIMITED_METHOD_DAC_CONTRACT; |
2782 | |
2783 | TADDR base = PTR_HOST_MEMBER_TADDR(NDirectMethodDesc, this, ndirect.m_pImportThunkGlue); |
2784 | |
2785 | #ifdef HAS_NDIRECT_IMPORT_PRECODE |
2786 | return RelativePointer<PTR_NDirectImportThunkGlue>::GetValueAtPtr(base); |
2787 | #else |
2788 | return dac_cast<PTR_NDirectImportThunkGlue>(base); |
2789 | #endif |
2790 | } |
2791 | |
2792 | LPVOID GetNDirectTarget() |
2793 | { |
2794 | LIMITED_METHOD_CONTRACT; |
2795 | |
2796 | _ASSERTE(IsNDirect()); |
2797 | return GetWriteableData()->m_pNDirectTarget; |
2798 | } |
2799 | |
2800 | LPVOID GetNativeNDirectTarget() |
2801 | { |
2802 | LIMITED_METHOD_CONTRACT; |
2803 | |
2804 | _ASSERTE(IsNDirect()); |
2805 | _ASSERTE_IMPL(!NDirectTargetIsImportThunk()); |
2806 | |
2807 | LPVOID pNativeNDirectTarget = ndirect.m_pNativeNDirectTarget; |
2808 | if (pNativeNDirectTarget != NULL) |
2809 | return pNativeNDirectTarget; |
2810 | |
2811 | return GetNDirectTarget(); |
2812 | } |
2813 | |
2814 | VOID SetNDirectTarget(LPVOID pTarget); |
2815 | |
2816 | #ifndef DACCESS_COMPILE |
2817 | BOOL NDirectTargetIsImportThunk() |
2818 | { |
2819 | WRAPPER_NO_CONTRACT; |
2820 | |
2821 | _ASSERTE(IsNDirect()); |
2822 | |
2823 | return (GetNDirectTarget() == GetNDirectImportThunkGlue()->GetEntrypoint()); |
2824 | } |
2825 | #endif // !DACCESS_COMPILE |
2826 | |
2827 | // Find the entry point name and function address |
2828 | // based on the module and data from NDirectMethodDesc |
2829 | // |
2830 | LPVOID FindEntryPoint(HINSTANCE hMod) const; |
2831 | |
2832 | private: |
2833 | FARPROC FindEntryPointWithMangling(HINSTANCE mod, PTR_CUTF8 entryPointName) const; |
2834 | |
2835 | #ifdef MDA_SUPPORTED |
2836 | Stub* GenerateStubForMDA(LPVOID pNativeTarget, Stub *pInnerStub); |
2837 | #endif // MDA_SUPPORTED |
2838 | |
2839 | public: |
2840 | |
2841 | void SetStackArgumentSize(WORD cbDstBuffer, CorPinvokeMap unmgdCallConv) |
2842 | { |
2843 | LIMITED_METHOD_CONTRACT; |
2844 | |
2845 | #if defined(_TARGET_X86_) |
2846 | // thiscall passes the this pointer in ECX |
2847 | if (unmgdCallConv == pmCallConvThiscall) |
2848 | { |
2849 | _ASSERTE(cbDstBuffer >= sizeof(SLOT)); |
2850 | cbDstBuffer -= sizeof(SLOT); |
2851 | } |
2852 | |
2853 | // Don't write to the field if it's already initialized to avoid creating private pages (NGEN) |
2854 | if (ndirect.m_cbStackArgumentSize == 0xFFFF) |
2855 | { |
2856 | ndirect.m_cbStackArgumentSize = cbDstBuffer; |
2857 | } |
2858 | else |
2859 | { |
2860 | _ASSERTE(ndirect.m_cbStackArgumentSize == cbDstBuffer); |
2861 | } |
2862 | #endif // defined(_TARGET_X86_) |
2863 | } |
2864 | |
2865 | #if defined(_TARGET_X86_) |
2866 | WORD GetStackArgumentSize() const |
2867 | { |
2868 | LIMITED_METHOD_DAC_CONTRACT; |
2869 | |
2870 | _ASSERTE(ndirect.m_cbStackArgumentSize != 0xFFFF); |
2871 | |
2872 | // If we have a methoddesc, stackArgSize is the number of bytes of |
2873 | // the outgoing marshalling buffer. |
2874 | return ndirect.m_cbStackArgumentSize; |
2875 | } |
2876 | #endif // defined(_TARGET_X86_) |
2877 | |
2878 | VOID InitEarlyBoundNDirectTarget(); |
2879 | |
2880 | // In AppDomains, we can trigger declarer's cctor when we link the P/Invoke, |
2881 | // which takes care of inlined calls as well. See code:NDirect.NDirectLink. |
2882 | // Although the cctor is guaranteed to run in the shared domain before the |
2883 | // target is invoked (code:IsClassConstructorTriggeredByILStub), we will |
2884 | // trigger at it link time as well because linking may depend on it - the |
2885 | // cctor may change the target DLL, change DLL search path etc. |
2886 | BOOL IsClassConstructorTriggeredAtLinkTime() |
2887 | { |
2888 | LIMITED_METHOD_CONTRACT; |
2889 | MethodTable * pMT = GetMethodTable(); |
2890 | // Try to avoid touching the EEClass if possible |
2891 | if (pMT->IsClassPreInited()) |
2892 | return FALSE; |
2893 | return !pMT->GetClass()->IsBeforeFieldInit(); |
2894 | } |
2895 | |
2896 | #ifndef DACCESS_COMPILE |
2897 | // In the shared domain and in NGENed code, we will trigger declarer's cctor |
2898 | // in the marshaling stub by calling code:StubHelpers.InitDeclaringType. If |
2899 | // this returns TRUE, the call must not be inlined. |
2900 | BOOL IsClassConstructorTriggeredByILStub() |
2901 | { |
2902 | WRAPPER_NO_CONTRACT; |
2903 | |
2904 | return (IsClassConstructorTriggeredAtLinkTime() && |
2905 | (IsZapped() || GetDomain()->IsSharedDomain() || SystemDomain::GetCurrentDomain()->IsCompilationDomain())); |
2906 | } |
2907 | #endif //!DACCESS_COMPILE |
2908 | }; //class NDirectMethodDesc |
2909 | |
2910 | |
2911 | //----------------------------------------------------------------------- |
2912 | // Operations specific to EEImplCall methods. We use a derived class to get |
2913 | // the compiler involved in enforcing proper method type usage. |
2914 | // |
2915 | // For now, the only EE impl is the delegate Invoke method. If we |
2916 | // add other EE impl types in the future, may need a discriminator |
2917 | // field here. |
2918 | //----------------------------------------------------------------------- |
2919 | class EEImplMethodDesc : public StoredSigMethodDesc |
2920 | { }; |
2921 | |
2922 | #ifdef FEATURE_COMINTEROP |
2923 | |
2924 | // This is the extra information needed to be associated with a method in order to use it for |
2925 | // CLR->COM calls. It is currently used by code:ComPlusCallMethodDesc (ordinary CLR->COM calls), |
2926 | // code:InstantiatedMethodDesc (optional field, CLR->COM calls on shared generic interfaces), |
2927 | // and code:DelegateEEClass (delegate->COM calls for WinRT). |
2928 | typedef DPTR(struct ComPlusCallInfo) PTR_ComPlusCallInfo; |
2929 | struct ComPlusCallInfo |
2930 | { |
2931 | // Returns ComPlusCallInfo associated with a method. pMD must be a ComPlusCallMethodDesc or |
2932 | // EEImplMethodDesc that has already been initialized for COM interop. |
2933 | inline static ComPlusCallInfo *FromMethodDesc(MethodDesc *pMD); |
2934 | |
2935 | enum Flags |
2936 | { |
2937 | kHasSuppressUnmanagedCodeAccess = 0x1, |
2938 | kRequiresArgumentWrapping = 0x2, |
2939 | kHasCopyCtorArgs = 0x4, |
2940 | }; |
2941 | |
2942 | union |
2943 | { |
2944 | // IL stub for CLR to COM call |
2945 | PCODE m_pILStub; |
2946 | |
2947 | // MethodDesc of the COM event provider to forward the call to (COM event interfaces) |
2948 | MethodDesc *m_pEventProviderMD; |
2949 | }; |
2950 | |
2951 | // method table of the interface which this represents |
2952 | PTR_MethodTable m_pInterfaceMT; |
2953 | |
2954 | // We need only 3 bits here, see enum Flags below. |
2955 | BYTE m_flags; |
2956 | |
2957 | // ComSlot() (is cached when we first invoke the method and generate |
2958 | // the stubs for it. There's probably a better place to do this |
2959 | // caching but I'm not sure I know all the places these things are |
2960 | // created.) |
2961 | WORD m_cachedComSlot; |
2962 | |
2963 | PCODE * GetAddrOfILStubField() |
2964 | { |
2965 | LIMITED_METHOD_CONTRACT; |
2966 | return &m_pILStub; |
2967 | } |
2968 | |
2969 | #ifdef _TARGET_X86_ |
2970 | // Size of outgoing arguments (on stack). This is currently used only |
2971 | // on x86 when we have an InlinedCallFrame representing a CLR->COM call. |
2972 | WORD m_cbStackArgumentSize; |
2973 | |
2974 | void SetHasCopyCtorArgs(BOOL value) |
2975 | { |
2976 | LIMITED_METHOD_CONTRACT; |
2977 | if (value) |
2978 | FastInterlockOr(reinterpret_cast<DWORD *>(&m_flags), kHasCopyCtorArgs); |
2979 | } |
2980 | |
2981 | BOOL HasCopyCtorArgs() |
2982 | { |
2983 | LIMITED_METHOD_CONTRACT; |
2984 | return ((m_flags & kHasCopyCtorArgs) != 0); |
2985 | } |
2986 | |
2987 | void InitStackArgumentSize() |
2988 | { |
2989 | LIMITED_METHOD_CONTRACT; |
2990 | |
2991 | m_cbStackArgumentSize = 0xFFFF; |
2992 | } |
2993 | |
2994 | void SetStackArgumentSize(WORD cbDstBuffer) |
2995 | { |
2996 | LIMITED_METHOD_CONTRACT; |
2997 | |
2998 | // Don't write to the field if it's already initialized to avoid creating private pages (NGEN) |
2999 | if (m_cbStackArgumentSize == 0xFFFF) |
3000 | { |
3001 | m_cbStackArgumentSize = cbDstBuffer; |
3002 | } |
3003 | _ASSERTE(m_cbStackArgumentSize == cbDstBuffer); |
3004 | } |
3005 | |
3006 | WORD GetStackArgumentSize() |
3007 | { |
3008 | LIMITED_METHOD_DAC_CONTRACT; |
3009 | |
3010 | _ASSERTE(m_cbStackArgumentSize != 0xFFFF); |
3011 | return m_cbStackArgumentSize; |
3012 | } |
3013 | |
3014 | union |
3015 | { |
3016 | LPVOID m_pRetThunk; // used for late-bound calls |
3017 | LPVOID m_pInterceptStub; // used for early-bound IL stub calls |
3018 | }; |
3019 | |
3020 | #else // _TARGET_X86_ |
3021 | void InitStackArgumentSize() |
3022 | { |
3023 | LIMITED_METHOD_CONTRACT; |
3024 | } |
3025 | |
3026 | void SetStackArgumentSize(WORD cbDstBuffer) |
3027 | { |
3028 | LIMITED_METHOD_CONTRACT; |
3029 | } |
3030 | #endif // _TARGET_X86_ |
3031 | |
3032 | // This field gets set only when this MethodDesc is marked as PreImplemented |
3033 | RelativePointer<PTR_MethodDesc> m_pStubMD; |
3034 | |
3035 | #ifdef FEATURE_PREJIT |
3036 | BOOL ShouldSave(DataImage *image); |
3037 | void Fixup(DataImage *image); |
3038 | #endif |
3039 | }; |
3040 | |
3041 | |
3042 | //----------------------------------------------------------------------- |
3043 | // Operations specific to ComPlusCall methods. We use a derived class to get |
3044 | // the compiler involved in enforcing proper method type usage. |
3045 | // DO NOT ADD FIELDS TO THIS CLASS. |
3046 | //----------------------------------------------------------------------- |
3047 | class ComPlusCallMethodDesc : public MethodDesc |
3048 | { |
3049 | public: |
3050 | ComPlusCallInfo *m_pComPlusCallInfo; // initialized in code:ComPlusCall.PopulateComPlusCallMethodDesc |
3051 | |
3052 | void InitRetThunk(); |
3053 | void InitComEventCallInfo(); |
3054 | |
3055 | PCODE * GetAddrOfILStubField() |
3056 | { |
3057 | LIMITED_METHOD_CONTRACT; |
3058 | return m_pComPlusCallInfo->GetAddrOfILStubField(); |
3059 | } |
3060 | |
3061 | MethodTable* GetInterfaceMethodTable() |
3062 | { |
3063 | LIMITED_METHOD_CONTRACT; |
3064 | _ASSERTE(m_pComPlusCallInfo->m_pInterfaceMT != NULL); |
3065 | return m_pComPlusCallInfo->m_pInterfaceMT; |
3066 | } |
3067 | |
3068 | MethodDesc* GetEventProviderMD() |
3069 | { |
3070 | LIMITED_METHOD_CONTRACT; |
3071 | |
3072 | return m_pComPlusCallInfo->m_pEventProviderMD; |
3073 | } |
3074 | |
3075 | |
3076 | BOOL RequiresArgumentWrapping() |
3077 | { |
3078 | LIMITED_METHOD_CONTRACT; |
3079 | |
3080 | return (m_pComPlusCallInfo->m_flags & ComPlusCallInfo::kRequiresArgumentWrapping) != 0; |
3081 | } |
3082 | |
3083 | void SetLateBoundFlags(BYTE newFlags) |
3084 | { |
3085 | LIMITED_METHOD_CONTRACT; |
3086 | |
3087 | FastInterlockOr(reinterpret_cast<DWORD *>(&m_pComPlusCallInfo->m_flags), newFlags); |
3088 | } |
3089 | |
3090 | #ifdef _TARGET_X86_ |
3091 | BOOL HasCopyCtorArgs() |
3092 | { |
3093 | LIMITED_METHOD_CONTRACT; |
3094 | return m_pComPlusCallInfo->HasCopyCtorArgs(); |
3095 | } |
3096 | |
3097 | void SetHasCopyCtorArgs(BOOL value) |
3098 | { |
3099 | LIMITED_METHOD_CONTRACT; |
3100 | m_pComPlusCallInfo->SetHasCopyCtorArgs(value); |
3101 | } |
3102 | |
3103 | WORD GetStackArgumentSize() |
3104 | { |
3105 | LIMITED_METHOD_DAC_CONTRACT; |
3106 | return m_pComPlusCallInfo->GetStackArgumentSize(); |
3107 | } |
3108 | |
3109 | void SetStackArgumentSize(WORD cbDstBuffer) |
3110 | { |
3111 | LIMITED_METHOD_CONTRACT; |
3112 | m_pComPlusCallInfo->SetStackArgumentSize(cbDstBuffer); |
3113 | } |
3114 | #else // _TARGET_X86_ |
3115 | void SetStackArgumentSize(WORD cbDstBuffer) |
3116 | { |
3117 | LIMITED_METHOD_CONTRACT; |
3118 | } |
3119 | #endif // _TARGET_X86_ |
3120 | }; |
3121 | #endif // FEATURE_COMINTEROP |
3122 | |
3123 | //----------------------------------------------------------------------- |
3124 | // InstantiatedMethodDesc's are used for generics and |
3125 | // come in four flavours, discriminated by the |
3126 | // low order bits of the first field: |
3127 | // |
3128 | // 00 --> GenericMethodDefinition |
3129 | // 01 --> UnsharedMethodInstantiation |
3130 | // 10 --> SharedMethodInstantiation |
3131 | // 11 --> WrapperStubWithInstantiations - and unboxing or instantiating stub |
3132 | // |
3133 | // A SharedMethodInstantiation descriptor extends MethodDesc |
3134 | // with a pointer to dictionary layout and a representative instantiation. |
3135 | // |
3136 | // A GenericMethodDefinition is the instantiation of a |
3137 | // generic method at its formals, used for verifying the method and |
3138 | // also for reflection. |
3139 | // |
3140 | // A WrapperStubWithInstantiations extends MethodDesc with: |
3141 | // (1) a method instantiation |
3142 | // (2) an "underlying" method descriptor. |
3143 | // A WrapperStubWithInstantiations may be placed in a MethodChunk for |
3144 | // a method table which specifies an exact instantiation for the class/struct. |
3145 | // A WrapperStubWithInstantiations may be either |
3146 | // an BoxedEntryPointStub or an exact-instantiation stub. |
3147 | // |
3148 | // Exact-instantiation stubs are used as extra type-context parameters. When |
3149 | // used as an entry, instantiating stubs pass an instantiation |
3150 | // dictionary on to the underlying method. These entries are required to |
3151 | // implement ldftn instructions on instantiations of shared generic |
3152 | // methods, as the InstantiatingStub's pointer does not expect a |
3153 | // dictionary argument; instead, it passes itself on to the shared |
3154 | // code as the dictionary. |
3155 | // |
3156 | // An UnsharedMethodInstantiation contains just an instantiation. |
3157 | // These are fully-specialized wrt method and class type parameters. |
3158 | // These satisfy (!IMD_IsGenericMethodDefinition() && |
3159 | // !IMD_IsSharedByGenericMethodInstantiations() && |
3160 | // !IMD_IsWrapperStubWithInstantiations()) |
3161 | // |
3162 | // Note that plain MethodDescs may represent shared code w.r.t. class type |
3163 | // parameters (see MethodDesc::IsSharedByGenericInstantiations()). |
3164 | //----------------------------------------------------------------------- |
3165 | |
3166 | class InstantiatedMethodDesc : public MethodDesc |
3167 | { |
3168 | #ifdef DACCESS_COMPILE |
3169 | friend class NativeImageDumper; |
3170 | #endif |
3171 | |
3172 | public: |
3173 | |
3174 | // All varities of InstantiatedMethodDesc's support this method. |
3175 | BOOL IMD_HasMethodInstantiation() |
3176 | { |
3177 | LIMITED_METHOD_DAC_CONTRACT; |
3178 | |
3179 | if (IMD_IsGenericMethodDefinition()) |
3180 | return TRUE; |
3181 | else |
3182 | return !m_pPerInstInfo.IsNull(); |
3183 | } |
3184 | |
3185 | // All varieties of InstantiatedMethodDesc's support this method. |
3186 | Instantiation IMD_GetMethodInstantiation() |
3187 | { |
3188 | LIMITED_METHOD_DAC_CONTRACT; |
3189 | |
3190 | return Instantiation(IMD_GetMethodDictionary()->GetInstantiation(), m_wNumGenericArgs); |
3191 | } |
3192 | |
3193 | PTR_Dictionary IMD_GetMethodDictionary() |
3194 | { |
3195 | LIMITED_METHOD_DAC_CONTRACT; |
3196 | |
3197 | return ReadPointerMaybeNull(this, &InstantiatedMethodDesc::m_pPerInstInfo); |
3198 | } |
3199 | |
3200 | PTR_Dictionary IMD_GetMethodDictionaryNonNull() |
3201 | { |
3202 | LIMITED_METHOD_DAC_CONTRACT; |
3203 | |
3204 | return ReadPointer(this, &InstantiatedMethodDesc::m_pPerInstInfo); |
3205 | } |
3206 | |
3207 | BOOL IMD_IsGenericMethodDefinition() |
3208 | { |
3209 | LIMITED_METHOD_DAC_CONTRACT; |
3210 | |
3211 | return((m_wFlags2 & KindMask) == GenericMethodDefinition); |
3212 | } |
3213 | |
3214 | BOOL IMD_IsSharedByGenericMethodInstantiations() |
3215 | { |
3216 | LIMITED_METHOD_DAC_CONTRACT; |
3217 | |
3218 | return((m_wFlags2 & KindMask) == SharedMethodInstantiation); |
3219 | } |
3220 | BOOL IMD_IsWrapperStubWithInstantiations() |
3221 | { |
3222 | LIMITED_METHOD_DAC_CONTRACT; |
3223 | |
3224 | return((m_wFlags2 & KindMask) == WrapperStubWithInstantiations); |
3225 | } |
3226 | |
3227 | BOOL IMD_IsEnCAddedMethod() |
3228 | { |
3229 | LIMITED_METHOD_CONTRACT; |
3230 | |
3231 | #ifdef EnC_SUPPORTED |
3232 | return((m_wFlags2 & KindMask) == EnCAddedMethod); |
3233 | #else |
3234 | return FALSE; |
3235 | #endif |
3236 | } |
3237 | |
3238 | #ifdef FEATURE_COMINTEROP |
3239 | BOOL IMD_HasComPlusCallInfo() |
3240 | { |
3241 | LIMITED_METHOD_CONTRACT; |
3242 | return ((m_wFlags2 & HasComPlusCallInfo) != 0); |
3243 | } |
3244 | |
3245 | void IMD_SetupGenericComPlusCall() |
3246 | { |
3247 | LIMITED_METHOD_CONTRACT; |
3248 | |
3249 | m_wFlags2 |= InstantiatedMethodDesc::HasComPlusCallInfo; |
3250 | |
3251 | IMD_GetComPlusCallInfo()->InitStackArgumentSize(); |
3252 | } |
3253 | |
3254 | PTR_ComPlusCallInfo IMD_GetComPlusCallInfo() |
3255 | { |
3256 | LIMITED_METHOD_CONTRACT; |
3257 | |
3258 | _ASSERTE(IMD_HasComPlusCallInfo()); |
3259 | SIZE_T size = s_ClassificationSizeTable[m_wFlags & (mdcClassification | mdcHasNonVtableSlot | mdcMethodImpl)]; |
3260 | |
3261 | if (HasNativeCodeSlot()) |
3262 | { |
3263 | size += (*dac_cast<PTR_TADDR>(dac_cast<TADDR>(this) + size) & FIXUP_LIST_MASK) ? |
3264 | (sizeof(NativeCodeSlot) + sizeof(FixupListSlot)) : sizeof(NativeCodeSlot); |
3265 | } |
3266 | |
3267 | return dac_cast<PTR_ComPlusCallInfo>(dac_cast<TADDR>(this) + size); |
3268 | } |
3269 | #endif // FEATURE_COMINTEROP |
3270 | |
3271 | PTR_DictionaryLayout GetDictLayoutRaw() |
3272 | { |
3273 | LIMITED_METHOD_DAC_CONTRACT; |
3274 | return RelativePointer<PTR_DictionaryLayout>::GetValueMaybeNullAtPtr(PTR_HOST_MEMBER_TADDR(InstantiatedMethodDesc, this, m_pDictLayout)); |
3275 | } |
3276 | |
3277 | PTR_MethodDesc IMD_GetWrappedMethodDesc() |
3278 | { |
3279 | LIMITED_METHOD_DAC_CONTRACT; |
3280 | |
3281 | _ASSERTE(IMD_IsWrapperStubWithInstantiations()); |
3282 | return RelativeFixupPointer<PTR_MethodDesc>::GetValueAtPtr(PTR_HOST_MEMBER_TADDR(InstantiatedMethodDesc, this, m_pWrappedMethodDesc)); |
3283 | } |
3284 | |
3285 | #ifndef DACCESS_COMPILE |
3286 | // Get the dictionary layout, if there is one |
3287 | DictionaryLayout* IMD_GetDictionaryLayout() |
3288 | { |
3289 | WRAPPER_NO_CONTRACT; |
3290 | if (IMD_IsWrapperStubWithInstantiations() && IMD_HasMethodInstantiation()) |
3291 | { |
3292 | InstantiatedMethodDesc* pIMD = IMD_GetWrappedMethodDesc()->AsInstantiatedMethodDesc(); |
3293 | return pIMD->m_pDictLayout.GetValueMaybeNull(); |
3294 | } |
3295 | else |
3296 | if (IMD_IsSharedByGenericMethodInstantiations()) |
3297 | return m_pDictLayout.GetValueMaybeNull(); |
3298 | else |
3299 | return NULL; |
3300 | } |
3301 | #endif // !DACCESS_COMPILE |
3302 | |
3303 | // Setup the IMD as shared code |
3304 | void SetupSharedMethodInstantiation(DWORD numGenericArgs, TypeHandle *pPerInstInfo, DictionaryLayout *pDL); |
3305 | |
3306 | // Setup the IMD as unshared code |
3307 | void SetupUnsharedMethodInstantiation(DWORD numGenericArgs, TypeHandle *pInst); |
3308 | |
3309 | // Setup the IMD as the special MethodDesc for a "generic" method |
3310 | void SetupGenericMethodDefinition(IMDInternalImport *pIMDII, LoaderAllocator* pAllocator, AllocMemTracker *pamTracker, |
3311 | Module *pModule, mdMethodDef tok); |
3312 | |
3313 | // Setup the IMD as a wrapper around another method desc |
3314 | void SetupWrapperStubWithInstantiations(MethodDesc* wrappedMD,DWORD numGenericArgs, TypeHandle *pGenericMethodInst); |
3315 | |
3316 | |
3317 | #ifdef EnC_SUPPORTED |
3318 | void SetupEnCAddedMethod() |
3319 | { |
3320 | LIMITED_METHOD_CONTRACT; |
3321 | m_wFlags2 = EnCAddedMethod; |
3322 | } |
3323 | #endif |
3324 | |
3325 | private: |
3326 | enum |
3327 | { |
3328 | KindMask = 0x07, |
3329 | GenericMethodDefinition = 0x00, |
3330 | UnsharedMethodInstantiation = 0x01, |
3331 | SharedMethodInstantiation = 0x02, |
3332 | WrapperStubWithInstantiations = 0x03, |
3333 | |
3334 | #ifdef EnC_SUPPORTED |
3335 | // Non-virtual method added through EditAndContinue. |
3336 | EnCAddedMethod = 0x07, |
3337 | #endif // EnC_SUPPORTED |
3338 | |
3339 | Unrestored = 0x08, |
3340 | |
3341 | #ifdef FEATURE_COMINTEROP |
3342 | HasComPlusCallInfo = 0x10, // this IMD contains an optional ComPlusCallInfo |
3343 | #endif // FEATURE_COMINTEROP |
3344 | }; |
3345 | |
3346 | friend class MethodDesc; // this fields are currently accessed by MethodDesc::Save/Restore etc. |
3347 | union { |
3348 | RelativePointer<PTR_DictionaryLayout> m_pDictLayout; //SharedMethodInstantiation |
3349 | |
3350 | RelativeFixupPointer<PTR_MethodDesc> m_pWrappedMethodDesc; // For WrapperStubWithInstantiations |
3351 | }; |
3352 | |
3353 | public: // <TODO>make private: JITinterface.cpp accesses through this </TODO> |
3354 | // Note we can't steal bits off m_pPerInstInfo as the JIT generates code to access through it!! |
3355 | |
3356 | // Type parameters to method (exact) |
3357 | // For non-unboxing instantiating stubs this is actually |
3358 | // a dictionary and further slots may hang off the end of the |
3359 | // instantiation. |
3360 | // |
3361 | // For generic method definitions that are not the typical method definition (e.g. C<int>.m<U>) |
3362 | // this field is null; to obtain the instantiation use LoadMethodInstantiation |
3363 | #if defined(FEATURE_NGEN_RELOCS_OPTIMIZATIONS) |
3364 | RelativePointer<PTR_Dictionary> m_pPerInstInfo; //SHARED |
3365 | #else |
3366 | PlainPointer<PTR_Dictionary> m_pPerInstInfo; //SHARED |
3367 | #endif |
3368 | |
3369 | private: |
3370 | WORD m_wFlags2; |
3371 | WORD m_wNumGenericArgs; |
3372 | |
3373 | public: |
3374 | static InstantiatedMethodDesc *FindOrCreateExactClassMethod(MethodTable *pExactMT, |
3375 | MethodDesc *pCanonicalMD); |
3376 | |
3377 | static InstantiatedMethodDesc* FindLoadedInstantiatedMethodDesc(MethodTable *pMT, |
3378 | mdMethodDef methodDef, |
3379 | Instantiation methodInst, |
3380 | BOOL getSharedNotStub); |
3381 | |
3382 | private: |
3383 | |
3384 | static InstantiatedMethodDesc *NewInstantiatedMethodDesc(MethodTable *pMT, |
3385 | MethodDesc* pGenericMDescInRepMT, |
3386 | MethodDesc* pSharedMDescForStub, |
3387 | Instantiation methodInst, |
3388 | BOOL getSharedNotStub); |
3389 | |
3390 | }; |
3391 | |
3392 | inline PTR_MethodTable MethodDesc::GetMethodTable_NoLogging() const |
3393 | { |
3394 | LIMITED_METHOD_DAC_CONTRACT; |
3395 | |
3396 | MethodDescChunk *pChunk = GetMethodDescChunk(); |
3397 | PREFIX_ASSUME(pChunk != NULL); |
3398 | return pChunk->GetMethodTable(); |
3399 | } |
3400 | |
3401 | inline PTR_MethodTable MethodDesc::GetMethodTable() const |
3402 | { |
3403 | LIMITED_METHOD_DAC_CONTRACT; |
3404 | |
3405 | g_IBCLogger.LogMethodDescAccess(this); |
3406 | return GetMethodTable_NoLogging(); |
3407 | } |
3408 | |
3409 | inline DPTR(RelativeFixupPointer<PTR_MethodTable>) MethodDesc::GetMethodTablePtr() const |
3410 | { |
3411 | LIMITED_METHOD_DAC_CONTRACT; |
3412 | |
3413 | MethodDescChunk *pChunk = GetMethodDescChunk(); |
3414 | PREFIX_ASSUME(pChunk != NULL); |
3415 | return pChunk->GetMethodTablePtr(); |
3416 | } |
3417 | |
3418 | inline MethodTable* MethodDesc::GetCanonicalMethodTable() |
3419 | { |
3420 | LIMITED_METHOD_DAC_CONTRACT; |
3421 | |
3422 | return GetMethodTable()->GetCanonicalMethodTable(); |
3423 | } |
3424 | |
3425 | inline mdMethodDef MethodDesc::GetMemberDef_NoLogging() const |
3426 | { |
3427 | LIMITED_METHOD_DAC_CONTRACT; |
3428 | |
3429 | MethodDescChunk *pChunk = GetMethodDescChunk(); |
3430 | PREFIX_ASSUME(pChunk != NULL); |
3431 | UINT16 tokrange = pChunk->GetTokRange(); |
3432 | |
3433 | UINT16 tokremainder = m_wFlags3AndTokenRemainder & enum_flag3_TokenRemainderMask; |
3434 | static_assert_no_msg(enum_flag3_TokenRemainderMask == METHOD_TOKEN_REMAINDER_MASK); |
3435 | |
3436 | return MergeToken(tokrange, tokremainder); |
3437 | } |
3438 | |
3439 | inline mdMethodDef MethodDesc::GetMemberDef() const |
3440 | { |
3441 | LIMITED_METHOD_DAC_CONTRACT; |
3442 | g_IBCLogger.LogMethodDescAccess(this); |
3443 | return GetMemberDef_NoLogging(); |
3444 | } |
3445 | |
3446 | // Set the offset of this method desc in a chunk table (which allows us |
3447 | // to work back to the method table/module pointer stored at the head of |
3448 | // the table. |
3449 | inline void MethodDesc::SetChunkIndex(MethodDescChunk * pChunk) |
3450 | { |
3451 | WRAPPER_NO_CONTRACT; |
3452 | |
3453 | // Calculate the offset (mod 8) from the chunk table header. |
3454 | SIZE_T offset = (BYTE*)this - (BYTE*)pChunk->GetFirstMethodDesc(); |
3455 | _ASSERTE((offset & ALIGNMENT_MASK) == 0); |
3456 | offset >>= ALIGNMENT_SHIFT; |
3457 | |
3458 | // Make sure that we did not overflow the BYTE |
3459 | _ASSERTE(offset == (BYTE)offset); |
3460 | m_chunkIndex = (BYTE)offset; |
3461 | |
3462 | // Make sure that the MethodDescChunk is setup correctly |
3463 | _ASSERTE(GetMethodDescChunk() == pChunk); |
3464 | } |
3465 | |
3466 | inline void MethodDesc::SetMemberDef(mdMethodDef mb) |
3467 | { |
3468 | WRAPPER_NO_CONTRACT; |
3469 | |
3470 | UINT16 tokrange; |
3471 | UINT16 tokremainder; |
3472 | SplitToken(mb, &tokrange, &tokremainder); |
3473 | |
3474 | _ASSERTE((tokremainder & ~enum_flag3_TokenRemainderMask) == 0); |
3475 | m_wFlags3AndTokenRemainder = (m_wFlags3AndTokenRemainder & ~enum_flag3_TokenRemainderMask) | tokremainder; |
3476 | |
3477 | if (GetMethodDescIndex() == 0) |
3478 | { |
3479 | GetMethodDescChunk()->SetTokenRange(tokrange); |
3480 | } |
3481 | |
3482 | #ifdef _DEBUG |
3483 | if (mb != 0) |
3484 | { |
3485 | _ASSERTE(GetMemberDef_NoLogging() == mb); |
3486 | } |
3487 | #endif |
3488 | } |
3489 | |
3490 | #ifdef _DEBUG |
3491 | |
3492 | inline BOOL MethodDesc::SanityCheck() |
3493 | { |
3494 | CONTRACTL |
3495 | { |
3496 | NOTHROW; |
3497 | GC_NOTRIGGER; |
3498 | SO_TOLERANT; |
3499 | MODE_ANY; |
3500 | SUPPORTS_DAC; |
3501 | } |
3502 | CONTRACTL_END; |
3503 | |
3504 | |
3505 | // Do a simple sanity test |
3506 | if (IsRestored()) |
3507 | { |
3508 | // If it looks good, do a more intensive sanity test. We don't care about the result, |
3509 | // we just want it to not AV. |
3510 | return GetMethodTable() == m_pDebugMethodTable.GetValue() && this->GetModule() != NULL; |
3511 | } |
3512 | |
3513 | return TRUE; |
3514 | } |
3515 | |
3516 | #endif // _DEBUG |
3517 | |
3518 | inline BOOL MethodDesc::IsEnCAddedMethod() |
3519 | { |
3520 | LIMITED_METHOD_DAC_CONTRACT; |
3521 | |
3522 | return (GetClassification() == mcInstantiated) && AsInstantiatedMethodDesc()->IMD_IsEnCAddedMethod(); |
3523 | } |
3524 | |
3525 | inline BOOL MethodDesc::HasNonVtableSlot() |
3526 | { |
3527 | LIMITED_METHOD_DAC_CONTRACT; |
3528 | |
3529 | return (m_wFlags & mdcHasNonVtableSlot) != 0; |
3530 | } |
3531 | |
3532 | inline Instantiation MethodDesc::GetMethodInstantiation() const |
3533 | { |
3534 | LIMITED_METHOD_DAC_CONTRACT; |
3535 | |
3536 | return |
3537 | (GetClassification() == mcInstantiated) |
3538 | ? AsInstantiatedMethodDesc()->IMD_GetMethodInstantiation() |
3539 | : Instantiation(); |
3540 | } |
3541 | |
3542 | inline Instantiation MethodDesc::GetClassInstantiation() const |
3543 | { |
3544 | LIMITED_METHOD_DAC_CONTRACT; |
3545 | |
3546 | return GetMethodTable()->GetInstantiation(); |
3547 | } |
3548 | |
3549 | inline BOOL MethodDesc::IsGenericMethodDefinition() const |
3550 | { |
3551 | LIMITED_METHOD_DAC_CONTRACT; |
3552 | |
3553 | g_IBCLogger.LogMethodDescAccess(this); |
3554 | return GetClassification() == mcInstantiated && AsInstantiatedMethodDesc()->IMD_IsGenericMethodDefinition(); |
3555 | } |
3556 | |
3557 | // True if the method descriptor is an instantiation of a generic method. |
3558 | inline BOOL MethodDesc::HasMethodInstantiation() const |
3559 | { |
3560 | LIMITED_METHOD_DAC_CONTRACT; |
3561 | |
3562 | return mcInstantiated == GetClassification() && AsInstantiatedMethodDesc()->IMD_HasMethodInstantiation(); |
3563 | } |
3564 | |
3565 | #if defined(FEATURE_GDBJIT) |
3566 | class CalledMethod |
3567 | { |
3568 | private: |
3569 | MethodDesc * m_pMD; |
3570 | void * m_CallAddr; |
3571 | CalledMethod * m_pNext; |
3572 | public: |
3573 | CalledMethod(MethodDesc *pMD, void * addr, CalledMethod * next) : m_pMD(pMD), m_CallAddr(addr), m_pNext(next) {} |
3574 | ~CalledMethod() {} |
3575 | MethodDesc * GetMethodDesc() { return m_pMD; } |
3576 | void * GetCallAddr() { return m_CallAddr; } |
3577 | CalledMethod * GetNext() { return m_pNext; } |
3578 | }; |
3579 | #endif |
3580 | |
3581 | #include "method.inl" |
3582 | |
3583 | #endif // !_METHOD_H |
3584 | |