SvcSystem.cs 6.8 KB

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