Executable.cs 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174
  1. using ChocolArm64.Memory;
  2. using Ryujinx.HLE.HOS;
  3. using Ryujinx.HLE.HOS.Kernel;
  4. using Ryujinx.HLE.Loaders.Executables;
  5. using Ryujinx.HLE.Utilities;
  6. using System.Collections.Generic;
  7. using System.IO;
  8. namespace Ryujinx.HLE.Loaders
  9. {
  10. class Executable
  11. {
  12. private List<ElfDyn> Dynamic;
  13. private Dictionary<long, string> m_SymbolTable;
  14. public IReadOnlyDictionary<long, string> SymbolTable => m_SymbolTable;
  15. public string Name { get; private set; }
  16. public string FilePath { get; private set; }
  17. public long ImageBase { get; private set; }
  18. public long ImageEnd { get; private set; }
  19. private AMemory Memory;
  20. private KMemoryManager MemoryManager;
  21. public Executable(IExecutable Exe, KMemoryManager MemoryManager, AMemory Memory, long ImageBase)
  22. {
  23. Dynamic = new List<ElfDyn>();
  24. m_SymbolTable = new Dictionary<long, string>();
  25. FilePath = Exe.FilePath;
  26. if (FilePath != null)
  27. {
  28. Name = Path.GetFileNameWithoutExtension(FilePath.Replace(Homebrew.TemporaryNroSuffix, ""));
  29. }
  30. this.Memory = Memory;
  31. this.MemoryManager = MemoryManager;
  32. this.ImageBase = ImageBase;
  33. this.ImageEnd = ImageBase;
  34. long TextPosition = ImageBase + (uint)Exe.TextOffset;
  35. long ROPosition = ImageBase + (uint)Exe.ROOffset;
  36. long DataPosition = ImageBase + (uint)Exe.DataOffset;
  37. long TextSize = (uint)IntUtils.AlignUp(Exe.Text.Length, KMemoryManager.PageSize);
  38. long ROSize = (uint)IntUtils.AlignUp(Exe.RO.Length, KMemoryManager.PageSize);
  39. long DataSize = (uint)IntUtils.AlignUp(Exe.Data.Length, KMemoryManager.PageSize);
  40. long DataAndBssSize = (uint)IntUtils.AlignUp(Exe.BssSize, KMemoryManager.PageSize) + DataSize;
  41. ImageEnd = DataPosition + DataAndBssSize;
  42. MemoryManager.HleMapProcessCode(TextPosition, TextSize + ROSize + DataAndBssSize);
  43. MemoryManager.SetProcessMemoryPermission(ROPosition, ROSize, MemoryPermission.Read);
  44. MemoryManager.SetProcessMemoryPermission(DataPosition, DataAndBssSize, MemoryPermission.ReadAndWrite);
  45. Memory.WriteBytes(TextPosition, Exe.Text);
  46. Memory.WriteBytes(ROPosition, Exe.RO);
  47. Memory.WriteBytes(DataPosition, Exe.Data);
  48. if (Exe.Mod0Offset == 0)
  49. {
  50. return;
  51. }
  52. long Mod0Offset = ImageBase + Exe.Mod0Offset;
  53. int Mod0Magic = Memory.ReadInt32(Mod0Offset + 0x0);
  54. long DynamicOffset = Memory.ReadInt32(Mod0Offset + 0x4) + Mod0Offset;
  55. long BssStartOffset = Memory.ReadInt32(Mod0Offset + 0x8) + Mod0Offset;
  56. long BssEndOffset = Memory.ReadInt32(Mod0Offset + 0xc) + Mod0Offset;
  57. long EhHdrStartOffset = Memory.ReadInt32(Mod0Offset + 0x10) + Mod0Offset;
  58. long EhHdrEndOffset = Memory.ReadInt32(Mod0Offset + 0x14) + Mod0Offset;
  59. long ModObjOffset = Memory.ReadInt32(Mod0Offset + 0x18) + Mod0Offset;
  60. while (true)
  61. {
  62. long TagVal = Memory.ReadInt64(DynamicOffset + 0);
  63. long Value = Memory.ReadInt64(DynamicOffset + 8);
  64. DynamicOffset += 0x10;
  65. ElfDynTag Tag = (ElfDynTag)TagVal;
  66. if (Tag == ElfDynTag.DT_NULL)
  67. {
  68. break;
  69. }
  70. Dynamic.Add(new ElfDyn(Tag, Value));
  71. }
  72. long StrTblAddr = ImageBase + GetFirstValue(ElfDynTag.DT_STRTAB);
  73. long SymTblAddr = ImageBase + GetFirstValue(ElfDynTag.DT_SYMTAB);
  74. long SymEntSize = GetFirstValue(ElfDynTag.DT_SYMENT);
  75. while ((ulong)SymTblAddr < (ulong)StrTblAddr)
  76. {
  77. ElfSym Sym = GetSymbol(SymTblAddr, StrTblAddr);
  78. m_SymbolTable.TryAdd(Sym.Value, Sym.Name);
  79. SymTblAddr += SymEntSize;
  80. }
  81. }
  82. private ElfRel GetRelocation(long Position)
  83. {
  84. long Offset = Memory.ReadInt64(Position + 0);
  85. long Info = Memory.ReadInt64(Position + 8);
  86. long Addend = Memory.ReadInt64(Position + 16);
  87. int RelType = (int)(Info >> 0);
  88. int SymIdx = (int)(Info >> 32);
  89. ElfSym Symbol = GetSymbol(SymIdx);
  90. return new ElfRel(Offset, Addend, Symbol, (ElfRelType)RelType);
  91. }
  92. private ElfSym GetSymbol(int Index)
  93. {
  94. long StrTblAddr = ImageBase + GetFirstValue(ElfDynTag.DT_STRTAB);
  95. long SymTblAddr = ImageBase + GetFirstValue(ElfDynTag.DT_SYMTAB);
  96. long SymEntSize = GetFirstValue(ElfDynTag.DT_SYMENT);
  97. long Position = SymTblAddr + Index * SymEntSize;
  98. return GetSymbol(Position, StrTblAddr);
  99. }
  100. private ElfSym GetSymbol(long Position, long StrTblAddr)
  101. {
  102. int NameIndex = Memory.ReadInt32(Position + 0);
  103. int Info = Memory.ReadByte(Position + 4);
  104. int Other = Memory.ReadByte(Position + 5);
  105. int SHIdx = Memory.ReadInt16(Position + 6);
  106. long Value = Memory.ReadInt64(Position + 8);
  107. long Size = Memory.ReadInt64(Position + 16);
  108. string Name = string.Empty;
  109. for (int Chr; (Chr = Memory.ReadByte(StrTblAddr + NameIndex++)) != 0;)
  110. {
  111. Name += (char)Chr;
  112. }
  113. return new ElfSym(Name, Info, Other, SHIdx, Value, Size);
  114. }
  115. private long GetFirstValue(ElfDynTag Tag)
  116. {
  117. foreach (ElfDyn Entry in Dynamic)
  118. {
  119. if (Entry.Tag == Tag)
  120. {
  121. return Entry.Value;
  122. }
  123. }
  124. return 0;
  125. }
  126. }
  127. }