SvcSystem.cs 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  1. using ChocolArm64.Memory;
  2. using ChocolArm64.State;
  3. using Ryujinx.Core.OsHle.Exceptions;
  4. using Ryujinx.Core.OsHle.Handles;
  5. using Ryujinx.Core.OsHle.Ipc;
  6. using Ryujinx.Core.OsHle.IpcServices;
  7. using System;
  8. using System.Threading;
  9. using static Ryujinx.Core.OsHle.ErrorCode;
  10. namespace Ryujinx.Core.OsHle.Svc
  11. {
  12. partial class SvcHandler
  13. {
  14. private const int AllowedCpuIdBitmask = 0b1111;
  15. private const bool EnableProcessDebugging = false;
  16. private void SvcExitProcess(AThreadState ThreadState)
  17. {
  18. Ns.Os.ExitProcess(ThreadState.ProcessId);
  19. }
  20. private void SvcClearEvent(AThreadState ThreadState)
  21. {
  22. int Handle = (int)ThreadState.X0;
  23. //TODO: Implement events.
  24. ThreadState.X0 = 0;
  25. }
  26. private void SvcCloseHandle(AThreadState ThreadState)
  27. {
  28. int Handle = (int)ThreadState.X0;
  29. Ns.Os.CloseHandle(Handle);
  30. ThreadState.X0 = 0;
  31. }
  32. private void SvcResetSignal(AThreadState ThreadState)
  33. {
  34. int Handle = (int)ThreadState.X0;
  35. //TODO: Implement events.
  36. ThreadState.X0 = 0;
  37. }
  38. private void SvcWaitSynchronization(AThreadState ThreadState)
  39. {
  40. long HandlesPtr = (long)ThreadState.X0;
  41. int HandlesCount = (int)ThreadState.X2;
  42. long Timeout = (long)ThreadState.X3;
  43. //TODO: Implement events.
  44. HThread CurrThread = Process.GetThread(ThreadState.Tpidr);
  45. Process.Scheduler.Suspend(CurrThread.ProcessorId);
  46. Process.Scheduler.Resume(CurrThread);
  47. ThreadState.X0 = 0;
  48. }
  49. private void SvcGetSystemTick(AThreadState ThreadState)
  50. {
  51. ThreadState.X0 = (ulong)ThreadState.CntpctEl0;
  52. }
  53. private void SvcConnectToNamedPort(AThreadState ThreadState)
  54. {
  55. long StackPtr = (long)ThreadState.X0;
  56. long NamePtr = (long)ThreadState.X1;
  57. string Name = AMemoryHelper.ReadAsciiString(Memory, NamePtr, 8);
  58. //TODO: Validate that app has perms to access the service, and that the service
  59. //actually exists, return error codes otherwise.
  60. HSession Session = new HSession(ServiceFactory.MakeService(Name));
  61. ThreadState.X1 = (ulong)Ns.Os.Handles.GenerateId(Session);
  62. ThreadState.X0 = 0;
  63. }
  64. private void SvcSendSyncRequest(AThreadState ThreadState)
  65. {
  66. SendSyncRequest(ThreadState, false);
  67. }
  68. private void SvcSendSyncRequestWithUserBuffer(AThreadState ThreadState)
  69. {
  70. SendSyncRequest(ThreadState, true);
  71. }
  72. private void SendSyncRequest(AThreadState ThreadState, bool UserBuffer)
  73. {
  74. long CmdPtr = ThreadState.Tpidr;
  75. long Size = 0x100;
  76. int Handle = 0;
  77. if (UserBuffer)
  78. {
  79. CmdPtr = (long)ThreadState.X0;
  80. Size = (long)ThreadState.X1;
  81. Handle = (int)ThreadState.X2;
  82. }
  83. else
  84. {
  85. Handle = (int)ThreadState.X0;
  86. }
  87. HThread CurrThread = Process.GetThread(ThreadState.Tpidr);
  88. Process.Scheduler.Suspend(CurrThread.ProcessorId);
  89. byte[] CmdData = AMemoryHelper.ReadBytes(Memory, CmdPtr, (int)Size);
  90. HSession Session = Ns.Os.Handles.GetData<HSession>(Handle);
  91. IpcMessage Cmd = new IpcMessage(CmdData, CmdPtr, Session is HDomain);
  92. if (Session != null)
  93. {
  94. IpcHandler.IpcCall(Ns, Memory, Session, Cmd, ThreadState.ThreadId, CmdPtr, Handle);
  95. byte[] Response = AMemoryHelper.ReadBytes(Memory, CmdPtr, (int)Size);
  96. ThreadState.X0 = 0;
  97. }
  98. else
  99. {
  100. ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidIpcReq);
  101. }
  102. Thread.Yield();
  103. Process.Scheduler.Resume(CurrThread);
  104. }
  105. private void SvcBreak(AThreadState ThreadState)
  106. {
  107. long Reason = (long)ThreadState.X0;
  108. long Unknown = (long)ThreadState.X1;
  109. long Info = (long)ThreadState.X2;
  110. throw new GuestBrokeExecutionException();
  111. }
  112. private void SvcOutputDebugString(AThreadState ThreadState)
  113. {
  114. long Position = (long)ThreadState.X0;
  115. long Size = (long)ThreadState.X1;
  116. string Str = AMemoryHelper.ReadAsciiString(Memory, Position, (int)Size);
  117. Logging.Info($"SvcOutputDebugString: {Str}");
  118. ThreadState.X0 = 0;
  119. }
  120. private void SvcGetInfo(AThreadState ThreadState)
  121. {
  122. long StackPtr = (long)ThreadState.X0;
  123. int InfoType = (int)ThreadState.X1;
  124. long Handle = (long)ThreadState.X2;
  125. int InfoId = (int)ThreadState.X3;
  126. //Fail for info not available on older Kernel versions.
  127. if (InfoType == 18 ||
  128. InfoType == 19)
  129. {
  130. ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidInfo);
  131. return;
  132. }
  133. switch (InfoType)
  134. {
  135. case 0:
  136. ThreadState.X1 = AllowedCpuIdBitmask;
  137. break;
  138. case 2:
  139. ThreadState.X1 = MemoryRegions.MapRegionAddress;
  140. break;
  141. case 3:
  142. ThreadState.X1 = MemoryRegions.MapRegionSize;
  143. break;
  144. case 4:
  145. ThreadState.X1 = MemoryRegions.HeapRegionAddress;
  146. break;
  147. case 5:
  148. ThreadState.X1 = MemoryRegions.HeapRegionSize;
  149. break;
  150. case 6:
  151. ThreadState.X1 = MemoryRegions.TotalMemoryAvailable;
  152. break;
  153. case 7:
  154. ThreadState.X1 = MemoryRegions.TotalMemoryUsed + CurrentHeapSize;
  155. break;
  156. case 8:
  157. ThreadState.X1 = EnableProcessDebugging ? 1 : 0;
  158. break;
  159. case 11:
  160. ThreadState.X1 = (ulong)Rng.Next() + ((ulong)Rng.Next() << 32);
  161. break;
  162. case 12:
  163. ThreadState.X1 = MemoryRegions.AddrSpaceStart;
  164. break;
  165. case 13:
  166. ThreadState.X1 = MemoryRegions.AddrSpaceSize;
  167. break;
  168. case 14:
  169. ThreadState.X1 = MemoryRegions.MapRegionAddress;
  170. break;
  171. case 15:
  172. ThreadState.X1 = MemoryRegions.MapRegionSize;
  173. break;
  174. default: throw new NotImplementedException($"SvcGetInfo: {InfoType} {Handle} {InfoId}");
  175. }
  176. ThreadState.X0 = 0;
  177. }
  178. }
  179. }