Skip to content

Commit

Permalink
Fix FnPtrTypeDesc::GetManagedClassObject
Browse files Browse the repository at this point in the history
The code was missing support for allocating RuntimeType objects on frozen heap

Fixes dotnet#82252
  • Loading branch information
jkotas committed Feb 18, 2023
1 parent 473e278 commit 13f4e0e
Show file tree
Hide file tree
Showing 3 changed files with 28 additions and 31 deletions.
23 changes: 3 additions & 20 deletions src/coreclr/vm/typedesc.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -522,26 +522,9 @@ OBJECTREF FnPtrTypeDesc::GetManagedClassObject()
}
CONTRACTL_END;

if (m_hExposedClassObject == NULL) {
REFLECTCLASSBASEREF refClass = NULL;
GCPROTECT_BEGIN(refClass);
refClass = (REFLECTCLASSBASEREF) AllocateObject(g_pRuntimeTypeClass);

LoaderAllocator *pLoaderAllocator = GetLoaderAllocator();
TypeHandle th = TypeHandle(this);
((ReflectClassBaseObject*)OBJECTREFToObject(refClass))->SetType(th);
((ReflectClassBaseObject*)OBJECTREFToObject(refClass))->SetKeepAlive(pLoaderAllocator->GetExposedObject());

// Let all threads fight over who wins using InterlockedCompareExchange.
// Only the winner can set m_hExposedClassObject from NULL.
LOADERHANDLE hExposedClassObject = pLoaderAllocator->AllocateHandle(refClass);

if (InterlockedCompareExchangeT(&m_hExposedClassObject, hExposedClassObject, static_cast<LOADERHANDLE>(NULL)))
{
pLoaderAllocator->FreeHandle(hExposedClassObject);
}

GCPROTECT_END();
if (m_hExposedClassObject == NULL)
{
TypeHandle(this).AllocateManagedClassObject(&m_hExposedClassObject);
}
return GetManagedClassObjectIfExists();
}
Expand Down
33 changes: 24 additions & 9 deletions src/coreclr/vm/typedesc.h
Original file line number Diff line number Diff line change
Expand Up @@ -540,6 +540,7 @@ class FnPtrTypeDesc : public TypeDesc
void EnumMemoryRegions(CLRDataEnumMemoryFlags flags);
#endif //DACCESS_COMPILE

#ifndef DACCESS_COMPILE
OBJECTREF GetManagedClassObject();

OBJECTREF GetManagedClassObjectIfExists()
Expand All @@ -552,23 +553,37 @@ class FnPtrTypeDesc : public TypeDesc
}
CONTRACTL_END;

OBJECTREF objRet = NULL;
GET_LOADERHANDLE_VALUE_FAST(GetLoaderAllocator(), m_hExposedClassObject, &objRet);
return objRet;
const RUNTIMETYPEHANDLE handle = m_hExposedClassObject;

OBJECTREF retVal;
if (!TypeHandle::GetManagedClassObjectFromHandleFast(handle, &retVal) &&
!GetLoaderAllocator()->GetHandleValueFastPhase2(handle, &retVal))
{
return NULL;
}

COMPILER_ASSUME(retVal != NULL);
return retVal;
}

OBJECTREF GetManagedClassObjectFast()
{
LIMITED_METHOD_CONTRACT;

OBJECTREF objRet = NULL;
LoaderAllocator::GetHandleValueFast(m_hExposedClassObject, &objRet);
return objRet;
}
OBJECTREF objRef;
if (!TypeHandle::GetManagedClassObjectFromHandleFast(m_hExposedClassObject, &objRef))
{
return FALSE;
}
COMPILER_ASSUME(objRef != NULL);
return objRef;
}
#endif

protected:
// Handle back to the internal reflection Type object
LOADERHANDLE m_hExposedClassObject;
// Non-unloadable context: internal RuntimeType object handle
// Unloadable context: slot index in LoaderAllocator's pinned table
RUNTIMETYPEHANDLE m_hExposedClassObject;

// Number of arguments
DWORD m_NumArgs;
Expand Down
3 changes: 1 addition & 2 deletions src/coreclr/vm/typehandle.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -349,8 +349,7 @@ bool TypeHandle::IsManagedClassObjectPinned() const
{
LIMITED_METHOD_DAC_CONTRACT;

// Function pointers are always mapped to typeof(IntPtr)
return !GetLoaderAllocator()->CanUnload() || IsFnPtrType();
return !GetLoaderAllocator()->CanUnload();
}

void TypeHandle::AllocateManagedClassObject(RUNTIMETYPEHANDLE* pDest)
Expand Down

0 comments on commit 13f4e0e

Please sign in to comment.