5d08e9b495
* hos: Cleanup the project Since a lot of changes has been done on the HOS project, there are some leftover here and there, or class just used in one service, things at wrong places, and more. This PR fixes that, additionnally to that, I've realigned some vars because I though it make the code more readable. * Address gdkchan feedback * addresses Thog feedback * Revert ElfSymbol
164 lines
No EOL
5.5 KiB
C#
164 lines
No EOL
5.5 KiB
C#
using Ryujinx.Common;
|
|
using Ryujinx.HLE.HOS;
|
|
using Ryujinx.HLE.HOS.Ipc;
|
|
using Ryujinx.HLE.HOS.Services;
|
|
using System;
|
|
using System.Diagnostics;
|
|
using System.Linq;
|
|
using System.Reflection;
|
|
using System.Runtime.Serialization;
|
|
using System.Text;
|
|
|
|
namespace Ryujinx.HLE.Exceptions
|
|
{
|
|
[Serializable]
|
|
internal class ServiceNotImplementedException : Exception
|
|
{
|
|
public IpcService Service { get; }
|
|
public ServiceCtx Context { get; }
|
|
public IpcMessage Request { get; }
|
|
|
|
public ServiceNotImplementedException(IpcService service, ServiceCtx context)
|
|
: this(service, context, "The service call is not implemented.") { }
|
|
|
|
public ServiceNotImplementedException(IpcService service, ServiceCtx context, string message) : base(message)
|
|
{
|
|
Service = service;
|
|
Context = context;
|
|
Request = context.Request;
|
|
}
|
|
|
|
public ServiceNotImplementedException(IpcService service, ServiceCtx context, string message, Exception inner) : base(message, inner)
|
|
{
|
|
Service = service;
|
|
Context = context;
|
|
Request = context.Request;
|
|
}
|
|
|
|
protected ServiceNotImplementedException(SerializationInfo info, StreamingContext context) : base(info, context) { }
|
|
|
|
public override string Message
|
|
{
|
|
get
|
|
{
|
|
return base.Message + Environment.NewLine + Environment.NewLine + BuildMessage();
|
|
}
|
|
}
|
|
|
|
private string BuildMessage()
|
|
{
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
// Print the IPC command details (service name, command ID, and handler)
|
|
(Type callingType, MethodBase callingMethod) = WalkStackTrace(new StackTrace(this));
|
|
|
|
if (callingType != null && callingMethod != null)
|
|
{
|
|
// If the type is past 0xF, we are using TIPC
|
|
var ipcCommands = Request.Type > IpcMessageType.TipcCloseSession ? Service.TipcCommands : Service.HipcCommands;
|
|
|
|
// Find the handler for the method called
|
|
var ipcHandler = ipcCommands.FirstOrDefault(x => x.Value == callingMethod);
|
|
var ipcCommandId = ipcHandler.Key;
|
|
var ipcMethod = ipcHandler.Value;
|
|
|
|
if (ipcMethod != null)
|
|
{
|
|
sb.AppendLine($"Service Command: {Service.GetType().FullName}: {ipcCommandId} ({ipcMethod.Name})");
|
|
sb.AppendLine();
|
|
}
|
|
}
|
|
|
|
sb.AppendLine("Guest Stack Trace:");
|
|
sb.AppendLine(Context.Thread.GetGuestStackTrace());
|
|
|
|
// Print buffer information
|
|
if (Request.PtrBuff.Count > 0 ||
|
|
Request.SendBuff.Count > 0 ||
|
|
Request.ReceiveBuff.Count > 0 ||
|
|
Request.ExchangeBuff.Count > 0 ||
|
|
Request.RecvListBuff.Count > 0)
|
|
{
|
|
sb.AppendLine("Buffer Information:");
|
|
|
|
if (Request.PtrBuff.Count > 0)
|
|
{
|
|
sb.AppendLine("\tPtrBuff:");
|
|
|
|
foreach (var buff in Request.PtrBuff)
|
|
{
|
|
sb.AppendLine($"\t[{buff.Index}] Position: 0x{buff.Position:x16} Size: 0x{buff.Size:x16}");
|
|
}
|
|
}
|
|
|
|
if (Request.SendBuff.Count > 0)
|
|
{
|
|
sb.AppendLine("\tSendBuff:");
|
|
|
|
foreach (var buff in Request.SendBuff)
|
|
{
|
|
sb.AppendLine($"\tPosition: 0x{buff.Position:x16} Size: 0x{buff.Size:x16} Flags: {buff.Flags}");
|
|
}
|
|
}
|
|
|
|
if (Request.ReceiveBuff.Count > 0)
|
|
{
|
|
sb.AppendLine("\tReceiveBuff:");
|
|
|
|
foreach (var buff in Request.ReceiveBuff)
|
|
{
|
|
sb.AppendLine($"\tPosition: 0x{buff.Position:x16} Size: 0x{buff.Size:x16} Flags: {buff.Flags}");
|
|
}
|
|
}
|
|
|
|
if (Request.ExchangeBuff.Count > 0)
|
|
{
|
|
sb.AppendLine("\tExchangeBuff:");
|
|
|
|
foreach (var buff in Request.ExchangeBuff)
|
|
{
|
|
sb.AppendLine($"\tPosition: 0x{buff.Position:x16} Size: 0x{buff.Size:x16} Flags: {buff.Flags}");
|
|
}
|
|
}
|
|
|
|
if (Request.RecvListBuff.Count > 0)
|
|
{
|
|
sb.AppendLine("\tRecvListBuff:");
|
|
|
|
foreach (var buff in Request.RecvListBuff)
|
|
{
|
|
sb.AppendLine($"\tPosition: 0x{buff.Position:x16} Size: 0x{buff.Size:x16}");
|
|
}
|
|
}
|
|
|
|
sb.AppendLine();
|
|
}
|
|
|
|
sb.AppendLine("Raw Request Data:");
|
|
sb.Append(HexUtils.HexTable(Request.RawData));
|
|
|
|
return sb.ToString();
|
|
}
|
|
|
|
private static (Type, MethodBase) WalkStackTrace(StackTrace trace)
|
|
{
|
|
int i = 0;
|
|
|
|
StackFrame frame;
|
|
|
|
// Find the IIpcService method that threw this exception
|
|
while ((frame = trace.GetFrame(i++)) != null)
|
|
{
|
|
var method = frame.GetMethod();
|
|
var declType = method.DeclaringType;
|
|
|
|
if (typeof(IpcService).IsAssignableFrom(declType))
|
|
{
|
|
return (declType, method);
|
|
}
|
|
}
|
|
|
|
return (null, null);
|
|
}
|
|
}
|
|
} |