HleProcessDebugger.cs 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. using ChocolArm64.Memory;
  2. using ChocolArm64.State;
  3. using Ryujinx.Common.Logging;
  4. using Ryujinx.HLE.HOS.Diagnostics.Demangler;
  5. using Ryujinx.HLE.HOS.Kernel.Memory;
  6. using Ryujinx.HLE.Loaders.Elf;
  7. using System.Collections.Generic;
  8. using System.Linq;
  9. using System.Text;
  10. using System.Threading;
  11. namespace Ryujinx.HLE.HOS.Kernel.Process
  12. {
  13. class HleProcessDebugger
  14. {
  15. private const int Mod0 = 'M' << 0 | 'O' << 8 | 'D' << 16 | '0' << 24;
  16. private KProcess _owner;
  17. private class Image
  18. {
  19. public long BaseAddress { get; private set; }
  20. public ElfSymbol[] Symbols { get; private set; }
  21. public Image(long baseAddress, ElfSymbol[] symbols)
  22. {
  23. BaseAddress = baseAddress;
  24. Symbols = symbols;
  25. }
  26. }
  27. private List<Image> _images;
  28. private int _loaded;
  29. public HleProcessDebugger(KProcess owner)
  30. {
  31. _owner = owner;
  32. _images = new List<Image>();
  33. }
  34. public void PrintGuestStackTrace(CpuThreadState threadState)
  35. {
  36. EnsureLoaded();
  37. StringBuilder trace = new StringBuilder();
  38. trace.AppendLine("Guest stack trace:");
  39. void AppendTrace(long address)
  40. {
  41. Image image = GetImage(address, out int imageIndex);
  42. if (image == null || !TryGetSubName(image, address, out string subName))
  43. {
  44. subName = $"Sub{address:x16}";
  45. }
  46. else if (subName.StartsWith("_Z"))
  47. {
  48. subName = Demangler.Parse(subName);
  49. }
  50. if (image != null)
  51. {
  52. long offset = address - image.BaseAddress;
  53. string imageName = GetGuessedNsoNameFromIndex(imageIndex);
  54. string imageNameAndOffset = $"[{_owner.Name}] {imageName}:0x{offset:x8}";
  55. trace.AppendLine($" {imageNameAndOffset} {subName}");
  56. }
  57. else
  58. {
  59. trace.AppendLine($" [{_owner.Name}] ??? {subName}");
  60. }
  61. }
  62. //TODO: ARM32.
  63. long framePointer = (long)threadState.X29;
  64. while (framePointer != 0)
  65. {
  66. if ((framePointer & 7) != 0 ||
  67. !_owner.CpuMemory.IsMapped(framePointer) ||
  68. !_owner.CpuMemory.IsMapped(framePointer + 8))
  69. {
  70. break;
  71. }
  72. //Note: This is the return address, we need to subtract one instruction
  73. //worth of bytes to get the branch instruction address.
  74. AppendTrace(_owner.CpuMemory.ReadInt64(framePointer + 8) - 4);
  75. framePointer = _owner.CpuMemory.ReadInt64(framePointer);
  76. }
  77. Logger.PrintInfo(LogClass.Cpu, trace.ToString());
  78. }
  79. private bool TryGetSubName(Image image, long address, out string name)
  80. {
  81. address -= image.BaseAddress;
  82. int left = 0;
  83. int right = image.Symbols.Length - 1;
  84. while (left <= right)
  85. {
  86. int size = right - left;
  87. int middle = left + (size >> 1);
  88. ElfSymbol symbol = image.Symbols[middle];
  89. long endAddr = symbol.Value + symbol.Size;
  90. if ((ulong)address >= (ulong)symbol.Value && (ulong)address < (ulong)endAddr)
  91. {
  92. name = symbol.Name;
  93. return true;
  94. }
  95. if ((ulong)address < (ulong)symbol.Value)
  96. {
  97. right = middle - 1;
  98. }
  99. else
  100. {
  101. left = middle + 1;
  102. }
  103. }
  104. name = null;
  105. return false;
  106. }
  107. private Image GetImage(long address, out int index)
  108. {
  109. lock (_images)
  110. {
  111. for (index = _images.Count - 1; index >= 0; index--)
  112. {
  113. if ((ulong)address >= (ulong)_images[index].BaseAddress)
  114. {
  115. return _images[index];
  116. }
  117. }
  118. }
  119. return null;
  120. }
  121. private string GetGuessedNsoNameFromIndex(int index)
  122. {
  123. if ((uint)index > 11)
  124. {
  125. return "???";
  126. }
  127. if (index == 0)
  128. {
  129. return "rtld";
  130. }
  131. else if (index == 1)
  132. {
  133. return "main";
  134. }
  135. else if (index == GetImagesCount() - 1)
  136. {
  137. return "sdk";
  138. }
  139. else
  140. {
  141. return "subsdk" + (index - 2);
  142. }
  143. }
  144. private int GetImagesCount()
  145. {
  146. lock (_images)
  147. {
  148. return _images.Count;
  149. }
  150. }
  151. private void EnsureLoaded()
  152. {
  153. if (Interlocked.CompareExchange(ref _loaded, 1, 0) == 0)
  154. {
  155. ScanMemoryForTextSegments();
  156. }
  157. }
  158. private void ScanMemoryForTextSegments()
  159. {
  160. ulong oldAddress = 0;
  161. ulong address = 0;
  162. while (address >= oldAddress)
  163. {
  164. KMemoryInfo info = _owner.MemoryManager.QueryMemory(address);
  165. if (info.State == MemoryState.Reserved)
  166. {
  167. break;
  168. }
  169. if (info.State == MemoryState.CodeStatic && info.Permission == MemoryPermission.ReadAndExecute)
  170. {
  171. LoadMod0Symbols(_owner.CpuMemory, (long)info.Address);
  172. }
  173. oldAddress = address;
  174. address = info.Address + info.Size;
  175. }
  176. }
  177. private void LoadMod0Symbols(MemoryManager memory, long textOffset)
  178. {
  179. long mod0Offset = textOffset + memory.ReadUInt32(textOffset + 4);
  180. if (mod0Offset < textOffset || !memory.IsMapped(mod0Offset) || (mod0Offset & 3) != 0)
  181. {
  182. return;
  183. }
  184. Dictionary<ElfDynamicTag, long> dynamic = new Dictionary<ElfDynamicTag, long>();
  185. int mod0Magic = memory.ReadInt32(mod0Offset + 0x0);
  186. if (mod0Magic != Mod0)
  187. {
  188. return;
  189. }
  190. long dynamicOffset = memory.ReadInt32(mod0Offset + 0x4) + mod0Offset;
  191. long bssStartOffset = memory.ReadInt32(mod0Offset + 0x8) + mod0Offset;
  192. long bssEndOffset = memory.ReadInt32(mod0Offset + 0xc) + mod0Offset;
  193. long ehHdrStartOffset = memory.ReadInt32(mod0Offset + 0x10) + mod0Offset;
  194. long ehHdrEndOffset = memory.ReadInt32(mod0Offset + 0x14) + mod0Offset;
  195. long modObjOffset = memory.ReadInt32(mod0Offset + 0x18) + mod0Offset;
  196. //TODO: Elf32.
  197. while (true)
  198. {
  199. long tagVal = memory.ReadInt64(dynamicOffset + 0);
  200. long value = memory.ReadInt64(dynamicOffset + 8);
  201. dynamicOffset += 0x10;
  202. ElfDynamicTag tag = (ElfDynamicTag)tagVal;
  203. if (tag == ElfDynamicTag.DT_NULL)
  204. {
  205. break;
  206. }
  207. dynamic[tag] = value;
  208. }
  209. if (!dynamic.TryGetValue(ElfDynamicTag.DT_STRTAB, out long strTab) ||
  210. !dynamic.TryGetValue(ElfDynamicTag.DT_SYMTAB, out long symTab) ||
  211. !dynamic.TryGetValue(ElfDynamicTag.DT_SYMENT, out long symEntSize))
  212. {
  213. return;
  214. }
  215. long strTblAddr = textOffset + strTab;
  216. long symTblAddr = textOffset + symTab;
  217. List<ElfSymbol> symbols = new List<ElfSymbol>();
  218. while ((ulong)symTblAddr < (ulong)strTblAddr)
  219. {
  220. ElfSymbol sym = GetSymbol(memory, symTblAddr, strTblAddr);
  221. symbols.Add(sym);
  222. symTblAddr += symEntSize;
  223. }
  224. lock (_images)
  225. {
  226. _images.Add(new Image(textOffset, symbols.OrderBy(x => x.Value).ToArray()));
  227. }
  228. }
  229. private ElfSymbol GetSymbol(MemoryManager memory, long address, long strTblAddr)
  230. {
  231. int nameIndex = memory.ReadInt32(address + 0);
  232. int info = memory.ReadByte (address + 4);
  233. int other = memory.ReadByte (address + 5);
  234. int shIdx = memory.ReadInt16(address + 6);
  235. long value = memory.ReadInt64(address + 8);
  236. long size = memory.ReadInt64(address + 16);
  237. string name = string.Empty;
  238. for (int chr; (chr = memory.ReadByte(strTblAddr + nameIndex++)) != 0;)
  239. {
  240. name += (char)chr;
  241. }
  242. return new ElfSymbol(name, info, other, shIdx, value, size);
  243. }
  244. }
  245. }