There’s Assembly.GetExecutingAssembly() and Assembly.GetCallingAssembly(). Note that GetCallingAssembly() has a Remark mentioning that depending on how JIT inlining behaves it may be possible that one method is (or is not) inlined into another and so GetCallingAssembly() returns varying results.
Now how is GetExecutingAssembly() different? JIT inlining could technically inline the code that calls GetExecutingAssembly() and so that code now belongs to a different assembly and depending on whether that happened GetExecutingAssembly() can just as well produce varying results.
Why doesn’t GetExecutingAssembly() description have remarks mentioning JIT inining similar to what GetCallingAssembly() description has?
The
GetExecutingAssemblymethod is not susceptible to JIT inlining because of the same reasonMethodBase.GetCurrentMethodis also not susceptible since they are implemented in a similar way.Both methods declare a local variable of a special enumeration
StackCrawlMarkand initialize it toStackCrawlMark.LookForMyCaller. This local variable has the side effect of preventing the method calling intoGetExecutingAssemblyorGetCurrentMethodfrom being inlined which will then guarantee correct results.This is supported by experimentation and also by the comment associated with this enumeration in the SSCLI20:
The reason
GetCallingAssemblyis susceptible is because you’re looking for the caller’s caller and the local variable only guarantees that the caller is not inlined, which means that the grandparent method can be inlined leading to an unexpected result.