From 087288310e7848cf8d8dc614a4c2467d2c2c020b Mon Sep 17 00:00:00 2001 From: Mikelle Date: Fri, 4 Aug 2023 14:18:00 -0700 Subject: [PATCH 1/9] Put HasNativeCodeReJITAware into GetFunctionAddress --- src/coreclr/debug/daccess/dacimpl.h | 11 ------ src/coreclr/debug/daccess/task.cpp | 4 +- src/coreclr/debug/di/breakpoint.cpp | 1 + src/coreclr/debug/ee/controller.cpp | 1 + src/coreclr/debug/ee/debugger.cpp | 49 +++--------------------- src/coreclr/debug/ee/debugger.h | 4 +- src/coreclr/debug/ee/functioninfo.cpp | 19 +++++----- src/coreclr/debug/inc/dbgipcevents.h | 1 + src/coreclr/vm/dbginterface.h | 7 +--- src/coreclr/vm/eedbginterfaceimpl.cpp | 27 +++++++++++++- src/coreclr/vm/encee.cpp | 4 +- src/coreclr/vm/method.cpp | 54 +++++++++++++-------------- src/coreclr/vm/method.hpp | 13 ++++--- 13 files changed, 85 insertions(+), 110 deletions(-) diff --git a/src/coreclr/debug/daccess/dacimpl.h b/src/coreclr/debug/daccess/dacimpl.h index ddf61370f416e..8de684a5dae9a 100644 --- a/src/coreclr/debug/daccess/dacimpl.h +++ b/src/coreclr/debug/daccess/dacimpl.h @@ -1253,17 +1253,6 @@ class ClrDataAccess /* [out] */ union STUB_BUF* outBuffer, /* [out] */ ULONG32* outFlags); - DebuggerJitInfo* GetDebuggerJitInfo(MethodDesc* methodDesc, - TADDR addr) - { - if (g_pDebugger) - { - return g_pDebugger->GetJitInfo(methodDesc, (PBYTE)addr, NULL); - } - - return NULL; - } - HRESULT GetMethodExtents(MethodDesc* methodDesc, METH_EXTENTS** extents); HRESULT GetMethodVarInfo(MethodDesc* methodDesc, diff --git a/src/coreclr/debug/daccess/task.cpp b/src/coreclr/debug/daccess/task.cpp index ea351e3a047b8..6354bae740990 100644 --- a/src/coreclr/debug/daccess/task.cpp +++ b/src/coreclr/debug/daccess/task.cpp @@ -5225,7 +5225,7 @@ EnumMethodInstances::Next(ClrDataAccess* dac, } } - if (!m_methodIter.Current()->HasNativeCodeReJITAware()) + if (!(g_pEEInterface->GetFunctionAddress(m_methodIter.Current()) != NULL)) { goto NextMethod; } @@ -5243,7 +5243,7 @@ EnumMethodInstances::CdStart(MethodDesc* methodDesc, CLRDATA_ENUM* handle) { if (!methodDesc->HasClassOrMethodInstantiation() && - !methodDesc->HasNativeCodeReJITAware()) + !(g_pEEInterface->GetFunctionAddress(methodDesc) != NULL)) { *handle = 0; return S_FALSE; diff --git a/src/coreclr/debug/di/breakpoint.cpp b/src/coreclr/debug/di/breakpoint.cpp index ad45df5c618ac..b08830d2f40c5 100644 --- a/src/coreclr/debug/di/breakpoint.cpp +++ b/src/coreclr/debug/di/breakpoint.cpp @@ -216,6 +216,7 @@ HRESULT CordbFunctionBreakpoint::Activate(BOOL fActivate) { pEvent->BreakpointData.nativeCodeMethodDescToken = (m_code.GetValue()->AsNativeCode())->GetVMNativeCodeMethodDescToken().ToLsPtr(); + pEvent->BreakpointData.codeStartAddress = (m_code.GetValue()->AsNativeCode())->GetAddress(); } // Note: we're sending a two-way event, so it blocks here diff --git a/src/coreclr/debug/ee/controller.cpp b/src/coreclr/debug/ee/controller.cpp index b624162f9bc77..725463f7fa5f4 100644 --- a/src/coreclr/debug/ee/controller.cpp +++ b/src/coreclr/debug/ee/controller.cpp @@ -1249,6 +1249,7 @@ bool DebuggerController::BindPatch(DebuggerControllerPatch *patch, } if (startAddr == NULL) { + _ASSERTE(startAddr == NULL); // Should not be trying to place patches on MethodDecs's for stubs. // These stubs will never get jitted. CONSISTENCY_CHECK_MSGF(!pMD->IsWrapperStub(), ("Can't place patch at stub md %p, %s::%s", diff --git a/src/coreclr/debug/ee/debugger.cpp b/src/coreclr/debug/ee/debugger.cpp index 073d758c81b87..fd41b9e798e25 100644 --- a/src/coreclr/debug/ee/debugger.cpp +++ b/src/coreclr/debug/ee/debugger.cpp @@ -2832,6 +2832,8 @@ HRESULT Debugger::GetILToNativeMapping(PCODE pNativeCodeStartAddress, ULONG32 cM } CONTRACTL_END; + _ASSERTE(pNativeCodeStartAddress != NULL); + #ifdef PROFILING_SUPPORTED // At this point, we're pulling in the debugger. if (!HasLazyData()) @@ -2998,6 +3000,7 @@ HRESULT Debugger::GetILToNativeMappingIntoArrays( _ASSERTE(pcMap != NULL); _ASSERTE(prguiILOffset != NULL); _ASSERTE(prguiNativeOffset != NULL); + _ASSERTE(pNativeCodeStartAddress != NULL); // Any caller of GetILToNativeMappingIntoArrays had better call // InitializeLazyDataIfNecessary first! @@ -5402,28 +5405,6 @@ void Debugger::ReleaseAllRuntimeThreads(AppDomain *pAppDomain) g_pEEInterface->ResumeFromDebug(pAppDomain); } -// Given a method, get's its EnC version number. 1 if the method is not EnCed. -// Note that MethodDescs are reused between versions so this will give us -// the most recent EnC number. -int Debugger::GetMethodEncNumber(MethodDesc * pMethod) -{ - CONTRACTL - { - THROWS; - GC_NOTRIGGER; - } - CONTRACTL_END; - - DebuggerJitInfo * dji = GetLatestJitInfoFromMethodDesc(pMethod); - if (dji == NULL) - { - // If there's no DJI, couldn't have been EnCed. - return 1; - } - return (int) dji->m_encVersion; -} - - bool Debugger::IsJMCMethod(Module* pModule, mdMethodDef tkMethod) { CONTRACTL @@ -6210,25 +6191,6 @@ void Debugger::LockAndSendEnCRemapCompleteEvent(MethodDesc *pMD) Thread *thread = g_pEEInterface->GetThread(); // Note that the debugger lock is reentrant, so we may or may not hold it already. SENDIPCEVENT_BEGIN(this, thread); - - EX_TRY - { - // Ensure the DJI for the latest version of this method has been pre-created. - // It's not clear whether this is necessary or not, but it shouldn't hurt since - // we're going to need to create it anyway since we'll be debugging inside it. - DebuggerJitInfo *dji = g_pDebugger->GetLatestJitInfoFromMethodDesc(pMD); - (void)dji; //prevent "unused variable" error from GCC - _ASSERTE( dji != NULL ); - } - EX_CATCH - { - // GetLatestJitInfo could throw on OOM, but the debugger isn't resiliant to OOM. - // I'm not aware of any other legitimate reason why it may throw, so we'll ASSERT - // if it fails. - _ASSERTE(!"Unexpected exception from Debugger::GetLatestJitInfoFromMethodDesc on EnC remap complete"); - } - EX_END_CATCH(RethrowTerminalExceptions); - // Send an EnC remap complete event to the Right Side. DebuggerIPCEvent* ipce = m_pRCThread->GetIPCEventSendBuffer(); InitIPCEvent(ipce, @@ -7856,6 +7818,7 @@ void Debugger::FirstChanceManagedExceptionCatcherFound(Thread *pThread, // Implements DebugInterface // Call by EE/exception. Must be on managed thread _ASSERTE(GetThreadNULLOk() != NULL); + _ASSERTE(pMethodAddr != NULL); // Quick check. if (!CORDebuggerAttached()) @@ -12616,7 +12579,7 @@ DWORD Debugger::GetThreadIdHelper(Thread *pThread) // does not own the memory provided via vars outparameter. //----------------------------------------------------------------------------- void Debugger::GetVarInfo(MethodDesc * fd, // [IN] method of interest - void *DebuggerVersionToken, // [IN] which edit version + CORDB_ADDRESS nativeCodeAddress, // [IN] which edit version SIZE_T * cVars, // [OUT] size of 'vars' const ICorDebugInfo::NativeVarInfo **vars // [OUT] map telling where local vars are stored ) @@ -12628,7 +12591,7 @@ void Debugger::GetVarInfo(MethodDesc * fd, // [IN] method of interest } CONTRACTL_END; - DebuggerJitInfo * ji = (DebuggerJitInfo *)DebuggerVersionToken; + DebuggerJitInfo * ji = g_pDebugger->GetJitInfo(fd, (const BYTE *)nativeCodeAddress); // If we didn't supply a DJI, then we're asking for the most recent version. if (ji == NULL) diff --git a/src/coreclr/debug/ee/debugger.h b/src/coreclr/debug/ee/debugger.h index 26edd26a96140..2c2440ddaf697 100644 --- a/src/coreclr/debug/ee/debugger.h +++ b/src/coreclr/debug/ee/debugger.h @@ -1933,8 +1933,6 @@ class Debugger : public DebugInterface bool IsJMCMethod(Module* pModule, mdMethodDef tkMethod); - int GetMethodEncNumber(MethodDesc * pMethod); - bool FirstChanceManagedException(Thread *pThread, SIZE_T currentIP, SIZE_T currentSP); @@ -1980,7 +1978,7 @@ class Debugger : public DebugInterface #endif // EnC_SUPPORTED void GetVarInfo(MethodDesc * fd, // [IN] method of interest - void *DebuggerVersionToken, // [IN] which edit version + CORDB_ADDRESS nativeCodeAddress, // [IN] which edit version SIZE_T * cVars, // [OUT] size of 'vars' const ICorDebugInfo::NativeVarInfo **vars // [OUT] map telling where local vars are stored ); diff --git a/src/coreclr/debug/ee/functioninfo.cpp b/src/coreclr/debug/ee/functioninfo.cpp index 76d4be3ab232f..0b314c0e33cd2 100644 --- a/src/coreclr/debug/ee/functioninfo.cpp +++ b/src/coreclr/debug/ee/functioninfo.cpp @@ -1576,16 +1576,17 @@ DebuggerJitInfo *DebuggerMethodInfo::FindOrCreateInitAndAddJitInfo(MethodDesc* f if (startAddr == NULL) { - // This will grab the start address for the current code version. + //Not any longer!!!!// This will grab the start address for the current code version. startAddr = g_pEEInterface->GetFunctionAddress(fd); - if (startAddr == NULL) - { - startAddr = fd->GetNativeCodeReJITAware(); - if (startAddr == NULL) - { - return NULL; - } - } + // startAddr = g_pEEInterface->GetNativeCodeReJITAware(fd); + // if (startAddr == NULL) + // { + // // startAddr = fd->GetFunctionAddress(); + // // if (startAddr == NULL) + // // { + // // return NULL; + // // } + // } } else { diff --git a/src/coreclr/debug/inc/dbgipcevents.h b/src/coreclr/debug/inc/dbgipcevents.h index 9fe1afd31a54b..e9643e50f480a 100644 --- a/src/coreclr/debug/inc/dbgipcevents.h +++ b/src/coreclr/debug/inc/dbgipcevents.h @@ -2011,6 +2011,7 @@ struct MSLAYOUT DebuggerIPCEvent SIZE_T offset; SIZE_T encVersion; LSPTR_METHODDESC nativeCodeMethodDescToken; // points to the MethodDesc if !isIL + CORDB_ADDRESS codeStartAddress; } BreakpointData; struct MSLAYOUT diff --git a/src/coreclr/vm/dbginterface.h b/src/coreclr/vm/dbginterface.h index daa57d25c86cf..85b9785bccbb9 100644 --- a/src/coreclr/vm/dbginterface.h +++ b/src/coreclr/vm/dbginterface.h @@ -203,7 +203,7 @@ class DebugInterface // Get debugger variable information for a specific version of a method virtual void GetVarInfo(MethodDesc * fd, // [IN] method of interest - void *DebuggerVersionToken, // [IN] which edit version + CORDB_ADDRESS nativeCodeAddress, // [IN] which edit version SIZE_T * cVars, // [OUT] size of 'vars' const ICorDebugInfo::NativeVarInfo **vars // [OUT] map telling where local vars are stored ) = 0; @@ -262,11 +262,6 @@ class DebugInterface virtual bool IsJMCMethod(Module* pModule, mdMethodDef tkMethod) = 0; - // Given a method, get's its EnC version number. 1 if the method is not EnCed. - // Note that MethodDescs are reused between versions so this will give us - // the most recent EnC number. - virtual int GetMethodEncNumber(MethodDesc * pMethod) = 0; - virtual void SendLogSwitchSetting (int iLevel, int iReason, _In_z_ LPCWSTR pLogSwitchName, diff --git a/src/coreclr/vm/eedbginterfaceimpl.cpp b/src/coreclr/vm/eedbginterfaceimpl.cpp index 792c608918a61..83825701257a6 100644 --- a/src/coreclr/vm/eedbginterfaceimpl.cpp +++ b/src/coreclr/vm/eedbginterfaceimpl.cpp @@ -631,7 +631,32 @@ PCODE EEDbgInterfaceImpl::GetFunctionAddress(MethodDesc *pFD) } CONTRACTL_END; - return pFD->GetNativeCode(); + //return pFD->GetNativeCode(); + WRAPPER_NO_CONTRACT; + SUPPORTS_DAC; + + PCODE pDefaultCode = pFD->GetNativeCode(); + if (pDefaultCode != NULL) + { + return pDefaultCode; + } + + { + Module *pModule = pFD->GetModule(); + CodeVersionManager *pCodeVersionManager = pModule->GetCodeVersionManager(); + CodeVersionManager::LockHolder codeVersioningLockHolder; + ILCodeVersion ilVersion = pCodeVersionManager->GetActiveILCodeVersion(PTR_MethodDesc(pFD)); + if (!ilVersion.IsDefaultVersion()) + { + NativeCodeVersion activeNativeCodeVersion = ilVersion.GetActiveNativeCodeVersion(PTR_MethodDesc(pFD)); + if (!activeNativeCodeVersion.IsNull()) + { + return activeNativeCodeVersion.GetNativeCode(); + } + } + + return NULL; + } } #ifndef DACCESS_COMPILE diff --git a/src/coreclr/vm/encee.cpp b/src/coreclr/vm/encee.cpp index 56500271314b5..4b5402a278ad7 100644 --- a/src/coreclr/vm/encee.cpp +++ b/src/coreclr/vm/encee.cpp @@ -804,8 +804,8 @@ NOINLINE void EditAndContinueModule::FixContextAndResume( // Get the var info which the codemanager will use for updating // enregistered variables correctly, or variables whose lifetimes differ // at the update point - g_pDebugInterface->GetVarInfo(pMD, oldDebuggerFuncHandle, &oldVarInfoCount, &pOldVarInfo); - g_pDebugInterface->GetVarInfo(pMD, NULL, &newVarInfoCount, &pNewVarInfo); + g_pDebugInterface->GetVarInfo(pMD, oldCodeInfo.GetCodeAddress(), &oldVarInfoCount, &pOldVarInfo); + g_pDebugInterface->GetVarInfo(pMD, newCodeInfo.GetCodeAddress(), &newVarInfoCount, &pNewVarInfo); #ifdef TARGET_X86 // save the frame pointer as FixContextForEnC might step on it. diff --git a/src/coreclr/vm/method.cpp b/src/coreclr/vm/method.cpp index 62b24e3dc091c..55334829daeec 100644 --- a/src/coreclr/vm/method.cpp +++ b/src/coreclr/vm/method.cpp @@ -935,33 +935,33 @@ PCODE MethodDesc::GetNativeCode() return GetStableEntryPoint(); } -PCODE MethodDesc::GetNativeCodeReJITAware() -{ - WRAPPER_NO_CONTRACT; - SUPPORTS_DAC; - - PCODE pDefaultCode = GetNativeCode(); - if (pDefaultCode != NULL) - { - return pDefaultCode; - } - - { - CodeVersionManager *pCodeVersionManager = GetCodeVersionManager(); - CodeVersionManager::LockHolder codeVersioningLockHolder; - ILCodeVersion ilVersion = pCodeVersionManager->GetActiveILCodeVersion(PTR_MethodDesc(this)); - if (!ilVersion.IsDefaultVersion()) - { - NativeCodeVersion activeNativeCodeVersion = ilVersion.GetActiveNativeCodeVersion(PTR_MethodDesc(this)); - if (!activeNativeCodeVersion.IsNull()) - { - return activeNativeCodeVersion.GetNativeCode(); - } - } - - return NULL; - } -} +// PCODE MethodDesc::GetNativeCodeReJITAware() +// { +// WRAPPER_NO_CONTRACT; +// SUPPORTS_DAC; + +// PCODE pDefaultCode = GetNativeCode(); +// if (pDefaultCode != NULL) +// { +// return pDefaultCode; +// } + +// { +// CodeVersionManager *pCodeVersionManager = GetCodeVersionManager(); +// CodeVersionManager::LockHolder codeVersioningLockHolder; +// ILCodeVersion ilVersion = pCodeVersionManager->GetActiveILCodeVersion(PTR_MethodDesc(this)); +// if (!ilVersion.IsDefaultVersion()) +// { +// NativeCodeVersion activeNativeCodeVersion = ilVersion.GetActiveNativeCodeVersion(PTR_MethodDesc(this)); +// if (!activeNativeCodeVersion.IsNull()) +// { +// return activeNativeCodeVersion.GetNativeCode(); +// } +// } + +// return NULL; +// } +// } //******************************************************************************* PTR_PCODE MethodDesc::GetAddrOfNativeCodeSlot() diff --git a/src/coreclr/vm/method.hpp b/src/coreclr/vm/method.hpp index 4b34045b57671..96371d76b3f56 100644 --- a/src/coreclr/vm/method.hpp +++ b/src/coreclr/vm/method.hpp @@ -1373,12 +1373,13 @@ class MethodDesc } // Perf warning: takes the CodeVersionManagerLock on every call - BOOL HasNativeCodeReJITAware() - { - LIMITED_METHOD_DAC_CONTRACT; + // BOOL HasNativeCodeReJITAware() + // { + // LIMITED_METHOD_DAC_CONTRACT; + // return GetFunctionAddress() != NULL; - return GetNativeCodeReJITAware() != NULL; - } + // // return GetNativeCodeReJITAware() != NULL; + // } BOOL SetNativeCodeInterlocked(PCODE addr, PCODE pExpected = NULL); @@ -1439,7 +1440,7 @@ class MethodDesc // Returns GetNativeCode() if it exists, but also checks to see if there // is a non-default IL code version and returns that. // Perf warning: takes the CodeVersionManagerLock on every call - PCODE GetNativeCodeReJITAware(); + //PCODE GetNativeCodeReJITAware(); #if defined(FEATURE_JIT_PITCHING) bool IsPitchable(); From 3fe7c024aaef4c4474ccddc62c7f53374519809f Mon Sep 17 00:00:00 2001 From: Mikelle Date: Fri, 4 Aug 2023 15:15:07 -0700 Subject: [PATCH 2/9] get rid of unused code --- src/coreclr/debug/ee/functioninfo.cpp | 10 ---------- src/coreclr/vm/method.cpp | 28 --------------------------- src/coreclr/vm/method.hpp | 14 -------------- 3 files changed, 52 deletions(-) diff --git a/src/coreclr/debug/ee/functioninfo.cpp b/src/coreclr/debug/ee/functioninfo.cpp index 0b314c0e33cd2..37b1730294071 100644 --- a/src/coreclr/debug/ee/functioninfo.cpp +++ b/src/coreclr/debug/ee/functioninfo.cpp @@ -1576,17 +1576,7 @@ DebuggerJitInfo *DebuggerMethodInfo::FindOrCreateInitAndAddJitInfo(MethodDesc* f if (startAddr == NULL) { - //Not any longer!!!!// This will grab the start address for the current code version. startAddr = g_pEEInterface->GetFunctionAddress(fd); - // startAddr = g_pEEInterface->GetNativeCodeReJITAware(fd); - // if (startAddr == NULL) - // { - // // startAddr = fd->GetFunctionAddress(); - // // if (startAddr == NULL) - // // { - // // return NULL; - // // } - // } } else { diff --git a/src/coreclr/vm/method.cpp b/src/coreclr/vm/method.cpp index 55334829daeec..c21ff6e4bac96 100644 --- a/src/coreclr/vm/method.cpp +++ b/src/coreclr/vm/method.cpp @@ -935,34 +935,6 @@ PCODE MethodDesc::GetNativeCode() return GetStableEntryPoint(); } -// PCODE MethodDesc::GetNativeCodeReJITAware() -// { -// WRAPPER_NO_CONTRACT; -// SUPPORTS_DAC; - -// PCODE pDefaultCode = GetNativeCode(); -// if (pDefaultCode != NULL) -// { -// return pDefaultCode; -// } - -// { -// CodeVersionManager *pCodeVersionManager = GetCodeVersionManager(); -// CodeVersionManager::LockHolder codeVersioningLockHolder; -// ILCodeVersion ilVersion = pCodeVersionManager->GetActiveILCodeVersion(PTR_MethodDesc(this)); -// if (!ilVersion.IsDefaultVersion()) -// { -// NativeCodeVersion activeNativeCodeVersion = ilVersion.GetActiveNativeCodeVersion(PTR_MethodDesc(this)); -// if (!activeNativeCodeVersion.IsNull()) -// { -// return activeNativeCodeVersion.GetNativeCode(); -// } -// } - -// return NULL; -// } -// } - //******************************************************************************* PTR_PCODE MethodDesc::GetAddrOfNativeCodeSlot() { diff --git a/src/coreclr/vm/method.hpp b/src/coreclr/vm/method.hpp index 96371d76b3f56..23c0a6a3b9c9c 100644 --- a/src/coreclr/vm/method.hpp +++ b/src/coreclr/vm/method.hpp @@ -1372,15 +1372,6 @@ class MethodDesc return GetNativeCode() != NULL; } - // Perf warning: takes the CodeVersionManagerLock on every call - // BOOL HasNativeCodeReJITAware() - // { - // LIMITED_METHOD_DAC_CONTRACT; - // return GetFunctionAddress() != NULL; - - // // return GetNativeCodeReJITAware() != NULL; - // } - BOOL SetNativeCodeInterlocked(PCODE addr, PCODE pExpected = NULL); PTR_PCODE GetAddrOfNativeCodeSlot(); @@ -1437,11 +1428,6 @@ class MethodDesc // Returns the address of the native code. PCODE GetNativeCode(); - // Returns GetNativeCode() if it exists, but also checks to see if there - // is a non-default IL code version and returns that. - // Perf warning: takes the CodeVersionManagerLock on every call - //PCODE GetNativeCodeReJITAware(); - #if defined(FEATURE_JIT_PITCHING) bool IsPitchable(); void PitchNativeCode(); From 8a9d0b1c89cdbb953e9386901910bed4184273b9 Mon Sep 17 00:00:00 2001 From: Mikelle Date: Wed, 9 Aug 2023 11:30:18 -0700 Subject: [PATCH 3/9] reverting GetFunctionAddress to previous behavior --- src/coreclr/debug/daccess/task.cpp | 4 ++-- src/coreclr/debug/di/breakpoint.cpp | 1 + src/coreclr/debug/ee/controller.cpp | 25 +++----------------- src/coreclr/debug/ee/debugger.cpp | 1 + src/coreclr/debug/ee/functioninfo.cpp | 11 +++++++-- src/coreclr/vm/eedbginterfaceimpl.cpp | 28 +---------------------- src/coreclr/vm/method.cpp | 33 ++++++++++++++++++++++++++- src/coreclr/vm/method.hpp | 13 +++++++++++ 8 files changed, 62 insertions(+), 54 deletions(-) diff --git a/src/coreclr/debug/daccess/task.cpp b/src/coreclr/debug/daccess/task.cpp index 6354bae740990..70a2406ee653d 100644 --- a/src/coreclr/debug/daccess/task.cpp +++ b/src/coreclr/debug/daccess/task.cpp @@ -5225,7 +5225,7 @@ EnumMethodInstances::Next(ClrDataAccess* dac, } } - if (!(g_pEEInterface->GetFunctionAddress(m_methodIter.Current()) != NULL)) + if (!m_methodIter.Current()->HasNativeCodeReJITAware()) { goto NextMethod; } @@ -5243,7 +5243,7 @@ EnumMethodInstances::CdStart(MethodDesc* methodDesc, CLRDATA_ENUM* handle) { if (!methodDesc->HasClassOrMethodInstantiation() && - !(g_pEEInterface->GetFunctionAddress(methodDesc) != NULL)) + !(methodDesc->HasNativeCodeReJITAware())) { *handle = 0; return S_FALSE; diff --git a/src/coreclr/debug/di/breakpoint.cpp b/src/coreclr/debug/di/breakpoint.cpp index b08830d2f40c5..568d7fc9fc66a 100644 --- a/src/coreclr/debug/di/breakpoint.cpp +++ b/src/coreclr/debug/di/breakpoint.cpp @@ -211,6 +211,7 @@ HRESULT CordbFunctionBreakpoint::Activate(BOOL fActivate) if (codeIsIL) { pEvent->BreakpointData.nativeCodeMethodDescToken = pEvent->BreakpointData.nativeCodeMethodDescToken.NullPtr(); + pEvent->BreakpointData.codeStartAddress = 0; } else { diff --git a/src/coreclr/debug/ee/controller.cpp b/src/coreclr/debug/ee/controller.cpp index 725463f7fa5f4..66a408c8a074a 100644 --- a/src/coreclr/debug/ee/controller.cpp +++ b/src/coreclr/debug/ee/controller.cpp @@ -1247,27 +1247,8 @@ bool DebuggerController::BindPatch(DebuggerControllerPatch *patch, startAddr = (CORDB_ADDRESS_TYPE *) CORDB_ADDRESS_TO_PTR(patch->GetDJI()->m_addrOfCode); _ASSERTE(startAddr != NULL); } - if (startAddr == NULL) - { - _ASSERTE(startAddr == NULL); - // Should not be trying to place patches on MethodDecs's for stubs. - // These stubs will never get jitted. - CONSISTENCY_CHECK_MSGF(!pMD->IsWrapperStub(), ("Can't place patch at stub md %p, %s::%s", - pMD, pMD->m_pszDebugClassName, pMD->m_pszDebugMethodName)); - - startAddr = (CORDB_ADDRESS_TYPE *)g_pEEInterface->GetFunctionAddress(pMD); - // - // Code is not available yet to patch. The prestub should - // notify us when it is executed. - // - if (startAddr == NULL) - { - LOG((LF_CORDB, LL_INFO10000, - "DC::BP: Patch at 0x%zx not bindable yet.\n", patch->offset)); - - return false; - } - } + //We should never be calling this function with both a NULL startAddr and a DJI that doesn't have code. + _ASSERTE(startAddr != NULL); } _ASSERTE(!g_pEEInterface->IsStub((const BYTE *)startAddr)); @@ -8656,7 +8637,7 @@ bool DebuggerFuncEvalComplete::SendEvent(Thread *thread, bool fIpChanged) // DebuggerEnCBreakpoint constructor - creates and activates a new EnC breakpoint // // Arguments: -// offset - native offset in the function to place the patch +// offset - IL offset in the function to place the patch // jitInfo - identifies the function in which the breakpoint is being placed // fTriggerType - breakpoint type: either REMAP_PENDING or REMAP_COMPLETE // pAppDomain - the breakpoint applies to the specified AppDomain only diff --git a/src/coreclr/debug/ee/debugger.cpp b/src/coreclr/debug/ee/debugger.cpp index fd41b9e798e25..8298914dabcd9 100644 --- a/src/coreclr/debug/ee/debugger.cpp +++ b/src/coreclr/debug/ee/debugger.cpp @@ -13982,6 +13982,7 @@ bool Debugger::GetILOffsetFromNative (MethodDesc *pFunc, const BYTE *pbAddr, DebuggerJitInfo *jitInfo = methodInfo->FindOrCreateInitAndAddJitInfo(pFunc, methodStartAddress); if (jitInfo == NULL) { + return false; } diff --git a/src/coreclr/debug/ee/functioninfo.cpp b/src/coreclr/debug/ee/functioninfo.cpp index 37b1730294071..96ba317b9b0ae 100644 --- a/src/coreclr/debug/ee/functioninfo.cpp +++ b/src/coreclr/debug/ee/functioninfo.cpp @@ -1565,7 +1565,6 @@ DebuggerJitInfo *DebuggerMethodInfo::FindOrCreateInitAndAddJitInfo(MethodDesc* f GC_NOTRIGGER; } CONTRACTL_END; - _ASSERTE(fd != NULL); // The debugger doesn't track Lightweight-codegen methods b/c they have no metadata. @@ -1577,6 +1576,14 @@ DebuggerJitInfo *DebuggerMethodInfo::FindOrCreateInitAndAddJitInfo(MethodDesc* f if (startAddr == NULL) { startAddr = g_pEEInterface->GetFunctionAddress(fd); + if (startAddr == NULL) + { + startAddr = fd->GetNativeCodeReJITAware(); + if (startAddr == NULL) + { + return NULL; + } + } } else { @@ -1639,7 +1646,7 @@ DebuggerJitInfo *DebuggerMethodInfo::CreateInitAndAddJitInfo(NativeCodeVersion n _ASSERTE(fd != NULL); - // May or may-not be jitted, that's why we passed in the start addr & size explicitly. + // May or may-not be jitted, that's why we passed in the start addr & size explicitly. _ASSERTE(startAddr != NULL); *jitInfoWasCreated = FALSE; diff --git a/src/coreclr/vm/eedbginterfaceimpl.cpp b/src/coreclr/vm/eedbginterfaceimpl.cpp index 83825701257a6..352a534d5c1a8 100644 --- a/src/coreclr/vm/eedbginterfaceimpl.cpp +++ b/src/coreclr/vm/eedbginterfaceimpl.cpp @@ -630,33 +630,7 @@ PCODE EEDbgInterfaceImpl::GetFunctionAddress(MethodDesc *pFD) SUPPORTS_DAC; } CONTRACTL_END; - - //return pFD->GetNativeCode(); - WRAPPER_NO_CONTRACT; - SUPPORTS_DAC; - - PCODE pDefaultCode = pFD->GetNativeCode(); - if (pDefaultCode != NULL) - { - return pDefaultCode; - } - - { - Module *pModule = pFD->GetModule(); - CodeVersionManager *pCodeVersionManager = pModule->GetCodeVersionManager(); - CodeVersionManager::LockHolder codeVersioningLockHolder; - ILCodeVersion ilVersion = pCodeVersionManager->GetActiveILCodeVersion(PTR_MethodDesc(pFD)); - if (!ilVersion.IsDefaultVersion()) - { - NativeCodeVersion activeNativeCodeVersion = ilVersion.GetActiveNativeCodeVersion(PTR_MethodDesc(pFD)); - if (!activeNativeCodeVersion.IsNull()) - { - return activeNativeCodeVersion.GetNativeCode(); - } - } - - return NULL; - } + return pFD->GetNativeCode(); } #ifndef DACCESS_COMPILE diff --git a/src/coreclr/vm/method.cpp b/src/coreclr/vm/method.cpp index c21ff6e4bac96..0756956ff2ceb 100644 --- a/src/coreclr/vm/method.cpp +++ b/src/coreclr/vm/method.cpp @@ -913,7 +913,6 @@ PCODE MethodDesc::GetNativeCode() WRAPPER_NO_CONTRACT; SUPPORTS_DAC; _ASSERTE(!IsDefaultInterfaceMethod() || HasNativeCodeSlot()); - if (HasNativeCodeSlot()) { // When profiler is enabled, profiler may ask to rejit a code even though we @@ -935,6 +934,38 @@ PCODE MethodDesc::GetNativeCode() return GetStableEntryPoint(); } +PCODE MethodDesc::GetNativeCodeReJITAware() +{ + WRAPPER_NO_CONTRACT; + SUPPORTS_DAC; + + PCODE pDefaultCode = GetNativeCode(); + if (pDefaultCode != NULL) + { + return pDefaultCode; + } + + else + { + CodeVersionManager *pCodeVersionManager = GetCodeVersionManager(); + CodeVersionManager::LockHolder codeVersioningLockHolder; + ILCodeVersionCollection ilVersionCollection = pCodeVersionManager->GetILCodeVersions(PTR_MethodDesc(this)); + for (ILCodeVersionIterator curIL = ilVersionCollection.Begin(), endIL = ilVersionCollection.End(); curIL != endIL; curIL++) + { + NativeCodeVersionCollection nativeCollection = curIL->GetNativeCodeVersions(PTR_MethodDesc(this)); + for (NativeCodeVersionIterator curNative = nativeCollection.Begin(), endNative = nativeCollection.End(); curNative != endNative; curNative++) + { + PCODE native = curNative->GetNativeCode(); + if(native != NULL) + { + return native; + } + } + } + return NULL; + } +} + //******************************************************************************* PTR_PCODE MethodDesc::GetAddrOfNativeCodeSlot() { diff --git a/src/coreclr/vm/method.hpp b/src/coreclr/vm/method.hpp index 23c0a6a3b9c9c..4b34045b57671 100644 --- a/src/coreclr/vm/method.hpp +++ b/src/coreclr/vm/method.hpp @@ -1372,6 +1372,14 @@ class MethodDesc return GetNativeCode() != NULL; } + // Perf warning: takes the CodeVersionManagerLock on every call + BOOL HasNativeCodeReJITAware() + { + LIMITED_METHOD_DAC_CONTRACT; + + return GetNativeCodeReJITAware() != NULL; + } + BOOL SetNativeCodeInterlocked(PCODE addr, PCODE pExpected = NULL); PTR_PCODE GetAddrOfNativeCodeSlot(); @@ -1428,6 +1436,11 @@ class MethodDesc // Returns the address of the native code. PCODE GetNativeCode(); + // Returns GetNativeCode() if it exists, but also checks to see if there + // is a non-default IL code version and returns that. + // Perf warning: takes the CodeVersionManagerLock on every call + PCODE GetNativeCodeReJITAware(); + #if defined(FEATURE_JIT_PITCHING) bool IsPitchable(); void PitchNativeCode(); From 49697d0c3a332c4551b87cfa762b12ff8b039f39 Mon Sep 17 00:00:00 2001 From: Mikelle Date: Thu, 10 Aug 2023 11:51:50 -0700 Subject: [PATCH 4/9] removing if statement --- src/coreclr/debug/ee/controller.cpp | 2 +- src/coreclr/debug/ee/debugger.cpp | 1 - src/coreclr/debug/ee/functioninfo.cpp | 8 ++------ 3 files changed, 3 insertions(+), 8 deletions(-) diff --git a/src/coreclr/debug/ee/controller.cpp b/src/coreclr/debug/ee/controller.cpp index 66a408c8a074a..ceeced2af066a 100644 --- a/src/coreclr/debug/ee/controller.cpp +++ b/src/coreclr/debug/ee/controller.cpp @@ -1247,7 +1247,7 @@ bool DebuggerController::BindPatch(DebuggerControllerPatch *patch, startAddr = (CORDB_ADDRESS_TYPE *) CORDB_ADDRESS_TO_PTR(patch->GetDJI()->m_addrOfCode); _ASSERTE(startAddr != NULL); } - //We should never be calling this function with both a NULL startAddr and a DJI that doesn't have code. + //We should never be calling this function with both a NULL startAddr and a DJI that doesn't have code. _ASSERTE(startAddr != NULL); } diff --git a/src/coreclr/debug/ee/debugger.cpp b/src/coreclr/debug/ee/debugger.cpp index 8298914dabcd9..fd41b9e798e25 100644 --- a/src/coreclr/debug/ee/debugger.cpp +++ b/src/coreclr/debug/ee/debugger.cpp @@ -13982,7 +13982,6 @@ bool Debugger::GetILOffsetFromNative (MethodDesc *pFunc, const BYTE *pbAddr, DebuggerJitInfo *jitInfo = methodInfo->FindOrCreateInitAndAddJitInfo(pFunc, methodStartAddress); if (jitInfo == NULL) { - return false; } diff --git a/src/coreclr/debug/ee/functioninfo.cpp b/src/coreclr/debug/ee/functioninfo.cpp index 96ba317b9b0ae..0ec7edd9dceaf 100644 --- a/src/coreclr/debug/ee/functioninfo.cpp +++ b/src/coreclr/debug/ee/functioninfo.cpp @@ -1566,7 +1566,6 @@ DebuggerJitInfo *DebuggerMethodInfo::FindOrCreateInitAndAddJitInfo(MethodDesc* f } CONTRACTL_END; _ASSERTE(fd != NULL); - // The debugger doesn't track Lightweight-codegen methods b/c they have no metadata. if (fd->IsDynamicMethod()) { @@ -1579,10 +1578,7 @@ DebuggerJitInfo *DebuggerMethodInfo::FindOrCreateInitAndAddJitInfo(MethodDesc* f if (startAddr == NULL) { startAddr = fd->GetNativeCodeReJITAware(); - if (startAddr == NULL) - { - return NULL; - } + _ASSERTE(startAddr != NULL); } } else @@ -1646,7 +1642,7 @@ DebuggerJitInfo *DebuggerMethodInfo::CreateInitAndAddJitInfo(NativeCodeVersion n _ASSERTE(fd != NULL); - // May or may-not be jitted, that's why we passed in the start addr & size explicitly. + // May or may-not be jitted, that's why we passed in the start addr & size explicitly. _ASSERTE(startAddr != NULL); *jitInfoWasCreated = FALSE; From d69d33d9942040e40c08adfb059c27ffad04ca48 Mon Sep 17 00:00:00 2001 From: Mikelle Date: Thu, 10 Aug 2023 15:55:25 -0700 Subject: [PATCH 5/9] clarify method names --- src/coreclr/debug/daccess/task.cpp | 4 ++-- src/coreclr/debug/ee/functioninfo.cpp | 2 +- src/coreclr/vm/method.cpp | 2 +- src/coreclr/vm/method.hpp | 6 +++--- 4 files changed, 7 insertions(+), 7 deletions(-) diff --git a/src/coreclr/debug/daccess/task.cpp b/src/coreclr/debug/daccess/task.cpp index 70a2406ee653d..4f9b76ec8f16c 100644 --- a/src/coreclr/debug/daccess/task.cpp +++ b/src/coreclr/debug/daccess/task.cpp @@ -5225,7 +5225,7 @@ EnumMethodInstances::Next(ClrDataAccess* dac, } } - if (!m_methodIter.Current()->HasNativeCodeReJITAware()) + if (!m_methodIter.Current()->HasNativeCodeAnyVersion()) { goto NextMethod; } @@ -5243,7 +5243,7 @@ EnumMethodInstances::CdStart(MethodDesc* methodDesc, CLRDATA_ENUM* handle) { if (!methodDesc->HasClassOrMethodInstantiation() && - !(methodDesc->HasNativeCodeReJITAware())) + !(methodDesc->HasNativeCodeAnyVersion())) { *handle = 0; return S_FALSE; diff --git a/src/coreclr/debug/ee/functioninfo.cpp b/src/coreclr/debug/ee/functioninfo.cpp index 0ec7edd9dceaf..3cbe0962d21a6 100644 --- a/src/coreclr/debug/ee/functioninfo.cpp +++ b/src/coreclr/debug/ee/functioninfo.cpp @@ -1577,7 +1577,7 @@ DebuggerJitInfo *DebuggerMethodInfo::FindOrCreateInitAndAddJitInfo(MethodDesc* f startAddr = g_pEEInterface->GetFunctionAddress(fd); if (startAddr == NULL) { - startAddr = fd->GetNativeCodeReJITAware(); + startAddr = fd->GetNativeCodeAnyVersion(); _ASSERTE(startAddr != NULL); } } diff --git a/src/coreclr/vm/method.cpp b/src/coreclr/vm/method.cpp index 0756956ff2ceb..29910d6cb4c1c 100644 --- a/src/coreclr/vm/method.cpp +++ b/src/coreclr/vm/method.cpp @@ -934,7 +934,7 @@ PCODE MethodDesc::GetNativeCode() return GetStableEntryPoint(); } -PCODE MethodDesc::GetNativeCodeReJITAware() +PCODE MethodDesc::GetNativeCodeAnyVersion() { WRAPPER_NO_CONTRACT; SUPPORTS_DAC; diff --git a/src/coreclr/vm/method.hpp b/src/coreclr/vm/method.hpp index 4b34045b57671..49f7481809f95 100644 --- a/src/coreclr/vm/method.hpp +++ b/src/coreclr/vm/method.hpp @@ -1373,11 +1373,11 @@ class MethodDesc } // Perf warning: takes the CodeVersionManagerLock on every call - BOOL HasNativeCodeReJITAware() + BOOL HasNativeCodeAnyVersion() { LIMITED_METHOD_DAC_CONTRACT; - return GetNativeCodeReJITAware() != NULL; + return GetNativeCodeAnyVersion() != NULL; } BOOL SetNativeCodeInterlocked(PCODE addr, PCODE pExpected = NULL); @@ -1439,7 +1439,7 @@ class MethodDesc // Returns GetNativeCode() if it exists, but also checks to see if there // is a non-default IL code version and returns that. // Perf warning: takes the CodeVersionManagerLock on every call - PCODE GetNativeCodeReJITAware(); + PCODE GetNativeCodeAnyVersion(); #if defined(FEATURE_JIT_PITCHING) bool IsPitchable(); From 86e84bdb65356b6c580a123d4c24a3dd5c35007e Mon Sep 17 00:00:00 2001 From: Mikelle Date: Thu, 10 Aug 2023 17:13:42 -0700 Subject: [PATCH 6/9] add comments --- src/coreclr/debug/ee/debugger.cpp | 5 +++++ src/coreclr/debug/ee/functioninfo.cpp | 1 + 2 files changed, 6 insertions(+) diff --git a/src/coreclr/debug/ee/debugger.cpp b/src/coreclr/debug/ee/debugger.cpp index fd41b9e798e25..d020d104e39fa 100644 --- a/src/coreclr/debug/ee/debugger.cpp +++ b/src/coreclr/debug/ee/debugger.cpp @@ -12915,6 +12915,11 @@ HRESULT Debugger::UpdateFunction(MethodDesc* pMD, SIZE_T encVersion) // For each offset in the IL->Native map, set a new EnC breakpoint on the // ones that we know could be remap points. + + // Depending on which DJI was picked, the code might compute different IL offsets. The JIT may not guarantee it produces + // the same set of sequence points for every generic instantiation. + // Inside ENCSequencePointHelper there is logic that skips IL offsets that map to the same native offset. + // Its possible that one version of the code maps two IL offsets to the same native offset but another version of the code maps them to different offsets. PTR_DebuggerILToNativeMap seqMap = pJitInfo->GetSequenceMap(); for (unsigned int i = 0; i < pJitInfo->GetSequenceMapCount(); i++) { diff --git a/src/coreclr/debug/ee/functioninfo.cpp b/src/coreclr/debug/ee/functioninfo.cpp index 3cbe0962d21a6..94c2bedfcb86c 100644 --- a/src/coreclr/debug/ee/functioninfo.cpp +++ b/src/coreclr/debug/ee/functioninfo.cpp @@ -1575,6 +1575,7 @@ DebuggerJitInfo *DebuggerMethodInfo::FindOrCreateInitAndAddJitInfo(MethodDesc* f if (startAddr == NULL) { startAddr = g_pEEInterface->GetFunctionAddress(fd); + //Test failure requires the below workaround, not sure how startAddr could be null if (startAddr == NULL) { startAddr = fd->GetNativeCodeAnyVersion(); From aaa2f7babc6bce8db5b9b99524b05119ae377f0b Mon Sep 17 00:00:00 2001 From: Mikelle Date: Fri, 11 Aug 2023 10:08:47 -0700 Subject: [PATCH 7/9] use BreakpointData.codeStartAddress --- src/coreclr/debug/ee/debugger.cpp | 2 +- src/coreclr/debug/ee/functioninfo.cpp | 7 +------ 2 files changed, 2 insertions(+), 7 deletions(-) diff --git a/src/coreclr/debug/ee/debugger.cpp b/src/coreclr/debug/ee/debugger.cpp index d020d104e39fa..e654d15340ac6 100644 --- a/src/coreclr/debug/ee/debugger.cpp +++ b/src/coreclr/debug/ee/debugger.cpp @@ -10452,7 +10452,7 @@ bool Debugger::HandleIPCEvent(DebuggerIPCEvent * pEvent) DebuggerJitInfo * pDJI = NULL; if ((pMethodDesc != NULL) && (pDMI != NULL)) { - pDJI = pDMI->FindOrCreateInitAndAddJitInfo(pMethodDesc, NULL /* startAddr */); + pDJI = pDMI->FindOrCreateInitAndAddJitInfo(pMethodDesc, pEvent->BreakpointData.codeStartAddress); } { diff --git a/src/coreclr/debug/ee/functioninfo.cpp b/src/coreclr/debug/ee/functioninfo.cpp index 94c2bedfcb86c..19910c6429a9c 100644 --- a/src/coreclr/debug/ee/functioninfo.cpp +++ b/src/coreclr/debug/ee/functioninfo.cpp @@ -1575,12 +1575,7 @@ DebuggerJitInfo *DebuggerMethodInfo::FindOrCreateInitAndAddJitInfo(MethodDesc* f if (startAddr == NULL) { startAddr = g_pEEInterface->GetFunctionAddress(fd); - //Test failure requires the below workaround, not sure how startAddr could be null - if (startAddr == NULL) - { - startAddr = fd->GetNativeCodeAnyVersion(); - _ASSERTE(startAddr != NULL); - } + _ASSERTE(startAddr != NULL); } else { From 31529d5f727f92c8d89f88cda565f832d8890d2d Mon Sep 17 00:00:00 2001 From: Mikelle Date: Fri, 11 Aug 2023 11:55:07 -0700 Subject: [PATCH 8/9] dac cast --- src/coreclr/debug/ee/debugger.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/coreclr/debug/ee/debugger.cpp b/src/coreclr/debug/ee/debugger.cpp index e654d15340ac6..7f86962e30af8 100644 --- a/src/coreclr/debug/ee/debugger.cpp +++ b/src/coreclr/debug/ee/debugger.cpp @@ -10452,7 +10452,7 @@ bool Debugger::HandleIPCEvent(DebuggerIPCEvent * pEvent) DebuggerJitInfo * pDJI = NULL; if ((pMethodDesc != NULL) && (pDMI != NULL)) { - pDJI = pDMI->FindOrCreateInitAndAddJitInfo(pMethodDesc, pEvent->BreakpointData.codeStartAddress); + pDJI = pDMI->FindOrCreateInitAndAddJitInfo(pMethodDesc, dac_cast(pEvent->BreakpointData.codeStartAddress)); } { From e4d8bfefb26fe51c24f39393d2ecbaec1ca93a1b Mon Sep 17 00:00:00 2001 From: Mikelle Date: Fri, 11 Aug 2023 16:05:21 -0700 Subject: [PATCH 9/9] address code review --- src/coreclr/debug/ee/debugger.cpp | 2 +- src/coreclr/vm/method.hpp | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/coreclr/debug/ee/debugger.cpp b/src/coreclr/debug/ee/debugger.cpp index 7f86962e30af8..52dcd2c79d4b5 100644 --- a/src/coreclr/debug/ee/debugger.cpp +++ b/src/coreclr/debug/ee/debugger.cpp @@ -10452,7 +10452,7 @@ bool Debugger::HandleIPCEvent(DebuggerIPCEvent * pEvent) DebuggerJitInfo * pDJI = NULL; if ((pMethodDesc != NULL) && (pDMI != NULL)) { - pDJI = pDMI->FindOrCreateInitAndAddJitInfo(pMethodDesc, dac_cast(pEvent->BreakpointData.codeStartAddress)); + pDJI = pDMI->FindOrCreateInitAndAddJitInfo(pMethodDesc, PINSTRToPCODE(dac_cast(pEvent->BreakpointData.codeStartAddress))); } { diff --git a/src/coreclr/vm/method.hpp b/src/coreclr/vm/method.hpp index 49f7481809f95..aaecfb20779f3 100644 --- a/src/coreclr/vm/method.hpp +++ b/src/coreclr/vm/method.hpp @@ -1437,7 +1437,7 @@ class MethodDesc PCODE GetNativeCode(); // Returns GetNativeCode() if it exists, but also checks to see if there - // is a non-default IL code version and returns that. + // is a non-default code version that is populated with a code body and returns that. // Perf warning: takes the CodeVersionManagerLock on every call PCODE GetNativeCodeAnyVersion();