From c08d352c637af95f23ad3f9eb62436b2865500a4 Mon Sep 17 00:00:00 2001 From: Amy Date: Sat, 25 Aug 2018 20:55:29 -0700 Subject: [PATCH] R2RDump - Fix errors caused by ExportedTypes rids in AvailableTypes section (#19647) * Fix error caused by exportedType in AvailableTypes section * TypeDef and ExportedType can both have the same rid * Use HashSet instead of Dictionary --- src/tools/r2rdump/R2RReader.cs | 64 ++++++++++++++++++++++++++++++---- src/tools/r2rdump/README.md | 4 +++ 2 files changed, 62 insertions(+), 6 deletions(-) diff --git a/src/tools/r2rdump/R2RReader.cs b/src/tools/r2rdump/R2RReader.cs index 2a6c410eaa..549415f669 100644 --- a/src/tools/r2rdump/R2RReader.cs +++ b/src/tools/r2rdump/R2RReader.cs @@ -360,6 +360,9 @@ namespace R2RDump { return; } + + HashSet added = new HashSet(); + R2RSection availableTypesSection = R2RHeader.Sections[R2RSection.SectionType.READYTORUN_SECTION_AVAILABLE_TYPES]; int availableTypesOffset = GetOffset(availableTypesSection.RelativeVirtualAddress); NativeParser parser = new NativeParser(Image, (uint)availableTypesOffset); @@ -370,8 +373,28 @@ namespace R2RDump { uint rid = curParser.GetUnsigned(); rid = rid >> 1; + if (added.Contains(rid)) + continue; + TypeDefinitionHandle typeDefHandle = MetadataTokens.TypeDefinitionHandle((int)rid); - AvailableTypes.Add(GetTypeDefFullName(MetadataReader, typeDefHandle)); + string typeDefName = GetTypeDefFullName(MetadataReader, typeDefHandle); + ExportedTypeHandle exportedTypeHandle = MetadataTokens.ExportedTypeHandle((int)rid); + string exportedTypeName = GetExportedTypeFullName(MetadataReader, exportedTypeHandle); + if (typeDefName == null && exportedTypeName == null) + { + R2RDump.WriteWarning($"AvailableType with rid {rid} is not a TypeDef or ExportedType"); + } + if (typeDefName != null) + { + AvailableTypes.Add(typeDefName); + added.Add(rid); + } + if (exportedTypeName != null) + { + AvailableTypes.Add("exported " + exportedTypeName); + added.Add(rid); + } + curParser = allEntriesEnum.GetNext(); } } @@ -510,17 +533,46 @@ namespace R2RDump /// public static string GetTypeDefFullName(MetadataReader mdReader, TypeDefinitionHandle handle) { - TypeDefinition typeDef; + string typeNamespace = ""; string typeStr = ""; do { - typeDef = mdReader.GetTypeDefinition(handle); - typeStr = "." + mdReader.GetString(typeDef.Name) + typeStr; - handle = typeDef.GetDeclaringType(); + try + { + TypeDefinition typeDef = mdReader.GetTypeDefinition(handle); + typeStr = "." + mdReader.GetString(typeDef.Name) + typeStr; + handle = typeDef.GetDeclaringType(); + if (handle.IsNil) + typeNamespace = mdReader.GetString(typeDef.Namespace); + } + catch (BadImageFormatException) + { + return null; + } } while (!handle.IsNil); - return mdReader.GetString(typeDef.Namespace) + typeStr; + return typeNamespace + typeStr; + } + + /// + /// Get the full name of an ExportedType, including namespace + /// + public static string GetExportedTypeFullName(MetadataReader mdReader, ExportedTypeHandle handle) + { + string typeNamespace = ""; + string typeStr = ""; + try + { + ExportedType exportedType = mdReader.GetExportedType(handle); + typeStr = "." + mdReader.GetString(exportedType.Name) + typeStr; + typeNamespace = mdReader.GetString(exportedType.Namespace); + } + catch (BadImageFormatException) + { + return null; + } + return typeNamespace + typeStr; } /// diff --git a/src/tools/r2rdump/README.md b/src/tools/r2rdump/README.md index d83b68dfdd..2d6a6d694e 100644 --- a/src/tools/r2rdump/README.md +++ b/src/tools/r2rdump/README.md @@ -37,6 +37,10 @@ dotnet R2RDump.dll --in <path to ReadyToRun image> ![R2RFormat](R2RFormat.png) +### System.Reflection.Metadata + +Used for getting method and type signatures from tokens (see: http://jilc.sourceforge.net/ecma_p2_cil.shtml) + ### READYTORUN_SECTION_COMPILER_IDENTIFIER A string describing the compiler. -- 2.34.1