SvcSystem.cs 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. using ChocolArm64.Memory;
  2. using ChocolArm64.State;
  3. using Ryujinx.OsHle.Handles;
  4. using Ryujinx.OsHle.Ipc;
  5. using System;
  6. namespace Ryujinx.OsHle.Svc
  7. {
  8. partial class SvcHandler
  9. {
  10. private static void SvcCloseHandle(Switch Ns, ARegisters Registers, AMemory Memory)
  11. {
  12. int Handle = (int)Registers.X0;
  13. Ns.Os.CloseHandle(Handle);
  14. Registers.X0 = (int)SvcResult.Success;
  15. }
  16. private static void SvcResetSignal(Switch Ns, ARegisters Registers, AMemory Memory)
  17. {
  18. int Handle = (int)Registers.X0;
  19. //TODO: Implement events.
  20. Registers.X0 = (int)SvcResult.Success;
  21. }
  22. private static void SvcWaitSynchronization(Switch Ns, ARegisters Registers, AMemory Memory)
  23. {
  24. long HandlesPtr = (long)Registers.X0;
  25. int HandlesCount = (int)Registers.X2;
  26. long Timeout = (long)Registers.X3;
  27. //TODO: Implement events.
  28. Registers.X0 = (int)SvcResult.Success;
  29. }
  30. private static void SvcGetSystemTick(Switch Ns, ARegisters Registers, AMemory Memory)
  31. {
  32. Registers.X0 = (ulong)Registers.CntpctEl0;
  33. }
  34. private static void SvcConnectToNamedPort(Switch Ns, ARegisters Registers, AMemory Memory)
  35. {
  36. long StackPtr = (long)Registers.X0;
  37. long NamePtr = (long)Registers.X1;
  38. string Name = AMemoryHelper.ReadAsciiString(Memory, NamePtr, 8);
  39. //TODO: Validate that app has perms to access the service, and that the service
  40. //actually exists, return error codes otherwise.
  41. HSession Session = new HSession(Name);
  42. Registers.X1 = (ulong)Ns.Os.Handles.GenerateId(Session);
  43. Registers.X0 = (int)SvcResult.Success;
  44. }
  45. private static void SvcSendSyncRequest(Switch Ns, ARegisters Registers, AMemory Memory)
  46. {
  47. SendSyncRequest(Ns, Registers, Memory, false);
  48. }
  49. private static void SvcSendSyncRequestWithUserBuffer(Switch Ns, ARegisters Registers, AMemory Memory)
  50. {
  51. SendSyncRequest(Ns, Registers, Memory, true);
  52. }
  53. private static void SendSyncRequest(Switch Ns, ARegisters Registers, AMemory Memory, bool IsUser)
  54. {
  55. long CmdPtr = Registers.Tpidr;
  56. long Size = 0x100;
  57. int Handle = 0;
  58. if (IsUser)
  59. {
  60. CmdPtr = (long)Registers.X0;
  61. Size = (long)Registers.X1;
  62. Handle = (int)Registers.X2;
  63. }
  64. else
  65. {
  66. Handle = (int)Registers.X0;
  67. }
  68. byte[] CmdData = AMemoryHelper.ReadBytes(Memory, CmdPtr, (int)Size);
  69. HSession Session = Ns.Os.Handles.GetData<HSession>(Handle);
  70. IpcMessage Cmd = new IpcMessage(CmdData, CmdPtr, Session is HDomain);
  71. if (Session != null)
  72. {
  73. IpcHandler.IpcCall(Ns, Memory, Session, Cmd, CmdPtr, Handle);
  74. byte[] Response = AMemoryHelper.ReadBytes(Memory, CmdPtr, (int)Size);
  75. Registers.X0 = (int)SvcResult.Success;
  76. }
  77. else
  78. {
  79. Registers.X0 = (int)SvcResult.ErrBadIpcReq;
  80. }
  81. }
  82. private static void SvcBreak(Switch Ns, ARegisters Registers, AMemory Memory)
  83. {
  84. long Reason = (long)Registers.X0;
  85. long Unknown = (long)Registers.X1;
  86. long Info = (long)Registers.X2;
  87. throw new Exception($"SvcBreak: {Reason} {Unknown} {Info}");
  88. }
  89. private static void SvcOutputDebugString(Switch Ns, ARegisters Registers, AMemory Memory)
  90. {
  91. long Position = (long)Registers.X0;
  92. long Size = (long)Registers.X1;
  93. string Str = AMemoryHelper.ReadAsciiString(Memory, Position, (int)Size);
  94. Logging.Info($"SvcOutputDebugString: {Str}");
  95. Registers.X0 = (int)SvcResult.Success;
  96. }
  97. private static void SvcGetInfo(Switch Ns, ARegisters Registers, AMemory Memory)
  98. {
  99. long StackPtr = (long)Registers.X0;
  100. int InfoType = (int)Registers.X1;
  101. long Handle = (long)Registers.X2;
  102. int InfoId = (int)Registers.X3;
  103. //Fail for info not available on older Kernel versions.
  104. if (InfoType == 18 ||
  105. InfoType == 19)
  106. {
  107. Registers.X0 = (int)SvcResult.ErrBadInfo;
  108. return;
  109. }
  110. switch (InfoType)
  111. {
  112. case 2: Registers.X1 = GetMapRegionBaseAddr(); break;
  113. case 3: Registers.X1 = GetMapRegionSize(); break;
  114. case 4: Registers.X1 = GetHeapRegionBaseAddr(); break;
  115. case 5: Registers.X1 = GetHeapRegionSize(); break;
  116. case 6: Registers.X1 = GetTotalMem(Memory); break;
  117. case 7: Registers.X1 = GetUsedMem(Memory); break;
  118. case 11: Registers.X1 = GetRnd64(); break;
  119. case 12: Registers.X1 = GetAddrSpaceBaseAddr(); break;
  120. case 13: Registers.X1 = GetAddrSpaceSize(); break;
  121. case 14: Registers.X1 = GetMapRegionBaseAddr(); break;
  122. case 15: Registers.X1 = GetMapRegionSize(); break;
  123. default: throw new NotImplementedException($"SvcGetInfo: {InfoType} {Handle} {InfoId}");
  124. }
  125. Registers.X0 = (int)SvcResult.Success;
  126. }
  127. private static ulong GetTotalMem(AMemory Memory)
  128. {
  129. return (ulong)Memory.Manager.GetTotalMemorySize();
  130. }
  131. private static ulong GetUsedMem(AMemory Memory)
  132. {
  133. return (ulong)Memory.Manager.GetUsedMemorySize();
  134. }
  135. private static ulong GetRnd64()
  136. {
  137. return (ulong)Rng.Next() + ((ulong)Rng.Next() << 32);
  138. }
  139. private static ulong GetAddrSpaceBaseAddr()
  140. {
  141. return 0x08000000;
  142. }
  143. private static ulong GetAddrSpaceSize()
  144. {
  145. return AMemoryMgr.AddrSize - GetAddrSpaceBaseAddr();
  146. }
  147. private static ulong GetMapRegionBaseAddr()
  148. {
  149. return 0x80000000;
  150. }
  151. private static ulong GetMapRegionSize()
  152. {
  153. return 0x40000000;
  154. }
  155. private static ulong GetHeapRegionBaseAddr()
  156. {
  157. return GetMapRegionBaseAddr() + GetMapRegionSize();
  158. }
  159. private static ulong GetHeapRegionSize()
  160. {
  161. return 0x40000000;
  162. }
  163. }
  164. }