SvcSystem.cs 7.4 KB

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