IService.cs 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. using Ryujinx.Common.Logging;
  2. using Ryujinx.HLE.HOS.Services.Fatal.Types;
  3. using System;
  4. using System.Runtime.InteropServices;
  5. using System.Text;
  6. namespace Ryujinx.HLE.HOS.Services.Fatal
  7. {
  8. [Service("fatal:u")]
  9. class IService : IpcService
  10. {
  11. public IService(ServiceCtx context) { }
  12. [CommandHipc(0)]
  13. // ThrowFatal(u64 result_code, u64 pid)
  14. public ResultCode ThrowFatal(ServiceCtx context)
  15. {
  16. ResultCode resultCode = (ResultCode)context.RequestData.ReadUInt64();
  17. ulong pid = context.Request.HandleDesc.PId;
  18. return ThrowFatalWithCpuContextImpl(context, resultCode, pid, FatalPolicy.ErrorReportAndErrorScreen, null);
  19. }
  20. [CommandHipc(1)]
  21. // ThrowFatalWithPolicy(u64 result_code, u32 fatal_policy, u64 pid)
  22. public ResultCode ThrowFatalWithPolicy(ServiceCtx context)
  23. {
  24. ResultCode resultCode = (ResultCode)context.RequestData.ReadUInt64();
  25. FatalPolicy fatalPolicy = (FatalPolicy)context.RequestData.ReadUInt32();
  26. ulong pid = context.Request.HandleDesc.PId;
  27. return ThrowFatalWithCpuContextImpl(context, resultCode, pid, fatalPolicy, null);
  28. }
  29. [CommandHipc(2)]
  30. // ThrowFatalWithCpuContext(u64 result_code, u32 fatal_policy, u64 pid, buffer<bytes, 0x15> cpu_context)
  31. public ResultCode ThrowFatalWithCpuContext(ServiceCtx context)
  32. {
  33. ResultCode resultCode = (ResultCode)context.RequestData.ReadUInt64();
  34. FatalPolicy fatalPolicy = (FatalPolicy)context.RequestData.ReadUInt32();
  35. ulong pid = context.Request.HandleDesc.PId;
  36. ulong cpuContextPosition = context.Request.SendBuff[0].Position;
  37. ulong cpuContextSize = context.Request.SendBuff[0].Size;
  38. ReadOnlySpan<byte> cpuContextData = context.Memory.GetSpan(cpuContextPosition, (int)cpuContextSize);
  39. return ThrowFatalWithCpuContextImpl(context, resultCode, pid, fatalPolicy, cpuContextData);
  40. }
  41. private ResultCode ThrowFatalWithCpuContextImpl(ServiceCtx context, ResultCode resultCode, ulong pid, FatalPolicy fatalPolicy, ReadOnlySpan<byte> cpuContext)
  42. {
  43. StringBuilder errorReport = new StringBuilder();
  44. errorReport.AppendLine();
  45. errorReport.AppendLine("ErrorReport log:");
  46. errorReport.AppendLine($"\tTitleId: {context.Device.Application.TitleId:x16}");
  47. errorReport.AppendLine($"\tPid: {pid}");
  48. errorReport.AppendLine($"\tResultCode: {((int)resultCode & 0x1FF) + 2000}-{((int)resultCode >> 9) & 0x3FFF:d4}");
  49. errorReport.AppendLine($"\tFatalPolicy: {fatalPolicy}");
  50. if (cpuContext != null)
  51. {
  52. errorReport.AppendLine("CPU Context:");
  53. if (context.Device.Application.TitleIs64Bit)
  54. {
  55. CpuContext64 cpuContext64 = MemoryMarshal.Cast<byte, CpuContext64>(cpuContext)[0];
  56. errorReport.AppendLine($"\tStartAddress: 0x{cpuContext64.StartAddress:x16}");
  57. errorReport.AppendLine($"\tRegisterSetFlags: {cpuContext64.RegisterSetFlags}");
  58. if (cpuContext64.StackTraceSize > 0)
  59. {
  60. errorReport.AppendLine("\tStackTrace:");
  61. for (int i = 0; i < cpuContext64.StackTraceSize; i++)
  62. {
  63. errorReport.AppendLine($"\t\t0x{cpuContext64.StackTrace[i]:x16}");
  64. }
  65. }
  66. errorReport.AppendLine("\tRegisters:");
  67. for (int i = 0; i < cpuContext64.X.Length; i++)
  68. {
  69. errorReport.AppendLine($"\t\tX[{i:d2}]:\t0x{cpuContext64.X[i]:x16}");
  70. }
  71. errorReport.AppendLine();
  72. errorReport.AppendLine($"\t\tFP:\t0x{cpuContext64.FP:x16}");
  73. errorReport.AppendLine($"\t\tLR:\t0x{cpuContext64.LR:x16}");
  74. errorReport.AppendLine($"\t\tSP:\t0x{cpuContext64.SP:x16}");
  75. errorReport.AppendLine($"\t\tPC:\t0x{cpuContext64.PC:x16}");
  76. errorReport.AppendLine($"\t\tPState:\t0x{cpuContext64.PState:x16}");
  77. errorReport.AppendLine($"\t\tAfsr0:\t0x{cpuContext64.Afsr0:x16}");
  78. errorReport.AppendLine($"\t\tAfsr1:\t0x{cpuContext64.Afsr1:x16}");
  79. errorReport.AppendLine($"\t\tEsr:\t0x{cpuContext64.Esr:x16}");
  80. errorReport.AppendLine($"\t\tFar:\t0x{cpuContext64.Far:x16}");
  81. }
  82. else
  83. {
  84. CpuContext32 cpuContext32 = MemoryMarshal.Cast<byte, CpuContext32>(cpuContext)[0];
  85. errorReport.AppendLine($"\tStartAddress: 0x{cpuContext32.StartAddress:16}");
  86. errorReport.AppendLine($"\tRegisterSetFlags: {cpuContext32.RegisterSetFlags}");
  87. if (cpuContext32.StackTraceSize > 0)
  88. {
  89. errorReport.AppendLine("\tStackTrace:");
  90. for (int i = 0; i < cpuContext32.StackTraceSize; i++)
  91. {
  92. errorReport.AppendLine($"\t\t0x{cpuContext32.StackTrace[i]:x16}");
  93. }
  94. }
  95. errorReport.AppendLine("\tRegisters:");
  96. for (int i = 0; i < cpuContext32.X.Length; i++)
  97. {
  98. errorReport.AppendLine($"\t\tX[{i:d2}]:\t0x{cpuContext32.X[i]:x16}");
  99. }
  100. errorReport.AppendLine();
  101. errorReport.AppendLine($"\t\tFP:\t0x{cpuContext32.FP:x16}");
  102. errorReport.AppendLine($"\t\tFP:\t0x{cpuContext32.IP:x16}");
  103. errorReport.AppendLine($"\t\tSP:\t0x{cpuContext32.SP:x16}");
  104. errorReport.AppendLine($"\t\tLR:\t0x{cpuContext32.LR:x16}");
  105. errorReport.AppendLine($"\t\tPC:\t0x{cpuContext32.PC:x16}");
  106. errorReport.AppendLine($"\t\tPState:\t0x{cpuContext32.PState:x16}");
  107. errorReport.AppendLine($"\t\tAfsr0:\t0x{cpuContext32.Afsr0:x16}");
  108. errorReport.AppendLine($"\t\tAfsr1:\t0x{cpuContext32.Afsr1:x16}");
  109. errorReport.AppendLine($"\t\tEsr:\t0x{cpuContext32.Esr:x16}");
  110. errorReport.AppendLine($"\t\tFar:\t0x{cpuContext32.Far:x16}");
  111. }
  112. }
  113. Logger.Info?.Print(LogClass.ServiceFatal, errorReport.ToString());
  114. context.Device.System.KernelContext.Syscall.Break((ulong)resultCode);
  115. return ResultCode.Success;
  116. }
  117. }
  118. }