Skip to content

Commit 1771d63

Browse files
Convert pointers in a couple tables to relative (#78801)
`EmbeddedPointerIndirection` is used in: * Array of pointers to dispatch maps * Array of pointers to GC static infos * Array of pointers to eager constructors I'm also updating array of module initializers emission since the code to interpret this table at runtime is the same as eager constructors.
1 parent 3c74d65 commit 1771d63

File tree

4 files changed

+37
-14
lines changed

4 files changed

+37
-14
lines changed

src/coreclr/nativeaot/Common/src/Internal/Runtime/CompilerHelpers/StartupCodeHelpers.cs

+18-10
Original file line numberDiff line numberDiff line change
@@ -132,7 +132,7 @@ private static unsafe void InitializeGlobalTablesForModule(TypeManagerHandle typ
132132
IntPtr staticsSection = RuntimeImports.RhGetModuleSection(typeManager, ReadyToRunSectionType.GCStaticRegion, out length);
133133
if (staticsSection != IntPtr.Zero)
134134
{
135-
Debug.Assert(length % IntPtr.Size == 0);
135+
Debug.Assert(length % (MethodTable.SupportsRelativePointers ? sizeof(int) : sizeof(nint)) == 0);
136136

137137
object[] spine = InitializeStatics(staticsSection, length);
138138

@@ -170,31 +170,39 @@ internal static void RunModuleInitializers()
170170

171171
private static unsafe void RunInitializers(TypeManagerHandle typeManager, ReadyToRunSectionType section)
172172
{
173-
var initializers = (delegate*<void>*)RuntimeImports.RhGetModuleSection(typeManager, section, out int length);
174-
Debug.Assert(length % IntPtr.Size == 0);
175-
int count = length / IntPtr.Size;
176-
for (int i = 0; i < count; i++)
173+
var pInitializers = (byte*)RuntimeImports.RhGetModuleSection(typeManager, section, out int length);
174+
Debug.Assert(length % (MethodTable.SupportsRelativePointers ? sizeof(int) : sizeof(nint)) == 0);
175+
176+
for (byte* pCurrent = pInitializers;
177+
pCurrent < (pInitializers + length);
178+
pCurrent += MethodTable.SupportsRelativePointers ? sizeof(int) : sizeof(nint))
177179
{
178-
initializers[i]();
180+
var initializer = MethodTable.SupportsRelativePointers ? (delegate*<void>)ReadRelPtr32(pCurrent) : (delegate*<void>)pCurrent;
181+
initializer();
179182
}
183+
184+
static void* ReadRelPtr32(void* address)
185+
=> (byte*)address + *(int*)address;
180186
}
181187

182188
private static unsafe object[] InitializeStatics(IntPtr gcStaticRegionStart, int length)
183189
{
184-
IntPtr gcStaticRegionEnd = (IntPtr)((byte*)gcStaticRegionStart + length);
190+
byte* gcStaticRegionEnd = (byte*)gcStaticRegionStart + length;
185191

186-
object[] spine = new object[length / IntPtr.Size];
192+
object[] spine = new object[length / (MethodTable.SupportsRelativePointers ? sizeof(int) : sizeof(nint))];
187193

188194
ref object rawSpineData = ref Unsafe.As<byte, object>(ref Unsafe.As<RawArrayData>(spine).Data);
189195

190196
int currentBase = 0;
191-
for (IntPtr** block = (IntPtr**)gcStaticRegionStart; block < (IntPtr**)gcStaticRegionEnd; block++)
197+
for (byte* block = (byte*)gcStaticRegionStart;
198+
block < gcStaticRegionEnd;
199+
block += MethodTable.SupportsRelativePointers ? sizeof(int) : sizeof(nint))
192200
{
193201
// Gc Static regions can be shared by modules linked together during compilation. To ensure each
194202
// is initialized once, the static region pointer is stored with lowest bit set in the image.
195203
// The first time we initialize the static region its pointer is replaced with an object reference
196204
// whose lowest bit is no longer set.
197-
IntPtr* pBlock = *block;
205+
IntPtr* pBlock = MethodTable.SupportsRelativePointers ? (IntPtr*)ReadRelPtr32(block) : *(IntPtr**)block;
198206
nint blockAddr = MethodTable.SupportsRelativePointers ? (nint)ReadRelPtr32(pBlock) : *pBlock;
199207
if ((blockAddr & GCStaticRegionConstants.Uninitialized) == GCStaticRegionConstants.Uninitialized)
200208
{

src/coreclr/nativeaot/Common/src/Internal/Runtime/MethodTable.cs

+4-1
Original file line numberDiff line numberDiff line change
@@ -902,7 +902,10 @@ internal DispatchMap* DispatchMap
902902
return DynamicTemplateType->DispatchMap;
903903
}
904904

905-
return ((DispatchMap**)TypeManager.DispatchMap)[idxDispatchMap];
905+
if (SupportsRelativePointers)
906+
return (DispatchMap*)FollowRelativePointer((int*)TypeManager.DispatchMap + idxDispatchMap);
907+
else
908+
return ((DispatchMap**)TypeManager.DispatchMap)[idxDispatchMap];
906909
}
907910
}
908911

src/coreclr/tools/aot/ILCompiler.Compiler/Compiler/DependencyAnalysis/EmbeddedPointerIndirectionNode.cs

+10-2
Original file line numberDiff line numberDiff line change
@@ -30,8 +30,16 @@ protected internal EmbeddedPointerIndirectionNode(TTarget target)
3030

3131
public override void EncodeData(ref ObjectDataBuilder dataBuilder, NodeFactory factory, bool relocsOnly)
3232
{
33-
dataBuilder.RequireInitialPointerAlignment();
34-
dataBuilder.EmitPointerReloc(Target);
33+
if (factory.Target.SupportsRelativePointers)
34+
{
35+
dataBuilder.RequireInitialAlignment(sizeof(int));
36+
dataBuilder.EmitReloc(Target, RelocType.IMAGE_REL_BASED_RELPTR32);
37+
}
38+
else
39+
{
40+
dataBuilder.RequireInitialPointerAlignment();
41+
dataBuilder.EmitPointerReloc(Target);
42+
}
3543
}
3644

3745
// At minimum, Target needs to be reported as a static dependency by inheritors.

src/coreclr/tools/aot/ILCompiler.Compiler/Compiler/DependencyAnalysis/ModuleInitializerListNode.cs

+5-1
Original file line numberDiff line numberDiff line change
@@ -141,7 +141,11 @@ public override ObjectData GetData(NodeFactory factory, bool relocsOnly = false)
141141

142142
foreach (var module in sortedModules)
143143
{
144-
builder.EmitPointerReloc(factory.MethodEntrypoint(module.GetGlobalModuleType().GetStaticConstructor()));
144+
IMethodNode entrypoint = factory.MethodEntrypoint(module.GetGlobalModuleType().GetStaticConstructor());
145+
if (factory.Target.SupportsRelativePointers)
146+
builder.EmitReloc(entrypoint, RelocType.IMAGE_REL_BASED_RELPTR32);
147+
else
148+
builder.EmitPointerReloc(entrypoint);
145149
}
146150

147151
var result = builder.ToObjectData();

0 commit comments

Comments
 (0)