Executable.cs 5.7 KB

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