LibHacHorizonManager.cs 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. using LibHac;
  2. using LibHac.Bcat;
  3. using LibHac.Common;
  4. using LibHac.Fs.Fsa;
  5. using LibHac.Fs.Shim;
  6. using LibHac.FsSrv.Impl;
  7. using LibHac.Loader;
  8. using LibHac.Ncm;
  9. using Ryujinx.HLE.FileSystem;
  10. using Ryujinx.HLE.HOS.Services.Arp;
  11. using System;
  12. using StorageId = LibHac.Ncm.StorageId;
  13. namespace Ryujinx.HLE.HOS
  14. {
  15. public class LibHacHorizonManager
  16. {
  17. private LibHac.Horizon Server { get; set; }
  18. public HorizonClient RyujinxClient { get; private set; }
  19. public HorizonClient ApplicationClient { get; private set; }
  20. public HorizonClient AccountClient { get; private set; }
  21. public HorizonClient AmClient { get; private set; }
  22. public HorizonClient BcatClient { get; private set; }
  23. public HorizonClient FsClient { get; private set; }
  24. public HorizonClient NsClient { get; private set; }
  25. public HorizonClient SdbClient { get; private set; }
  26. private SharedRef<LibHacIReader> _arpIReader;
  27. internal LibHacIReader ArpIReader => _arpIReader.Get;
  28. public LibHacHorizonManager()
  29. {
  30. InitializeServer();
  31. }
  32. private void InitializeServer()
  33. {
  34. Server = new LibHac.Horizon(new HorizonConfiguration());
  35. RyujinxClient = Server.CreatePrivilegedHorizonClient();
  36. }
  37. public void InitializeArpServer()
  38. {
  39. _arpIReader.Reset(new LibHacIReader());
  40. RyujinxClient.Sm.RegisterService(new LibHacArpServiceObject(ref _arpIReader), "arp:r").ThrowIfFailure();
  41. }
  42. public void InitializeBcatServer()
  43. {
  44. BcatClient = Server.CreateHorizonClient(new ProgramLocation(SystemProgramId.Bcat, StorageId.BuiltInSystem), BcatFsPermissions);
  45. _ = new BcatServer(BcatClient);
  46. }
  47. public void InitializeFsServer(VirtualFileSystem virtualFileSystem)
  48. {
  49. virtualFileSystem.InitializeFsServer(Server, out var fsClient);
  50. FsClient = fsClient;
  51. CleanSdCardDirectory();
  52. }
  53. public void InitializeSystemClients()
  54. {
  55. AccountClient = Server.CreateHorizonClient(new ProgramLocation(SystemProgramId.Account, StorageId.BuiltInSystem), AccountFsPermissions);
  56. AmClient = Server.CreateHorizonClient(new ProgramLocation(SystemProgramId.Am, StorageId.BuiltInSystem), AmFsPermissions);
  57. NsClient = Server.CreateHorizonClient(new ProgramLocation(SystemProgramId.Ns, StorageId.BuiltInSystem), NsFsPermissions);
  58. SdbClient = Server.CreateHorizonClient(new ProgramLocation(SystemProgramId.Sdb, StorageId.BuiltInSystem), SdbFacData, SdbFacDescriptor);
  59. }
  60. public void InitializeApplicationClient(ProgramId programId, in Npdm npdm)
  61. {
  62. ApplicationClient = Server.CreateHorizonClient(new ProgramLocation(programId, StorageId.BuiltInUser), npdm.FsAccessControlData, npdm.FsAccessControlDescriptor);
  63. }
  64. // This function was added to avoid errors that come from a user's keys or SD encryption seed changing.
  65. // Catching these errors and recreating the file ended up not working because of the different ways
  66. // applications respond to a file suddenly containing all zeros or having a length of zero.
  67. // Clearing the SD card save directory was determined to be the best option for the moment since
  68. // the saves on the SD card are meant as caches that can be deleted at any time.
  69. private void CleanSdCardDirectory()
  70. {
  71. Result rc = RyujinxClient.Fs.MountSdCard("sdcard".ToU8Span());
  72. if (rc.IsFailure()) return;
  73. try
  74. {
  75. RyujinxClient.Fs.CleanDirectoryRecursively("sdcard:/Nintendo/save".ToU8Span()).IgnoreResult();
  76. RyujinxClient.Fs.DeleteDirectoryRecursively("sdcard:/save".ToU8Span()).IgnoreResult();
  77. }
  78. finally
  79. {
  80. RyujinxClient.Fs.Unmount("sdcard".ToU8Span());
  81. }
  82. }
  83. private static AccessControlBits.Bits AccountFsPermissions => AccessControlBits.Bits.SystemSaveData |
  84. AccessControlBits.Bits.GameCard |
  85. AccessControlBits.Bits.SaveDataMeta |
  86. AccessControlBits.Bits.GetRightsId;
  87. private static AccessControlBits.Bits AmFsPermissions => AccessControlBits.Bits.SaveDataManagement |
  88. AccessControlBits.Bits.CreateSaveData |
  89. AccessControlBits.Bits.SystemData;
  90. private static AccessControlBits.Bits BcatFsPermissions => AccessControlBits.Bits.SystemSaveData;
  91. private static AccessControlBits.Bits NsFsPermissions => AccessControlBits.Bits.ApplicationInfo |
  92. AccessControlBits.Bits.SystemSaveData |
  93. AccessControlBits.Bits.GameCard |
  94. AccessControlBits.Bits.SaveDataManagement |
  95. AccessControlBits.Bits.ContentManager |
  96. AccessControlBits.Bits.ImageManager |
  97. AccessControlBits.Bits.SystemSaveDataManagement |
  98. AccessControlBits.Bits.SystemUpdate |
  99. AccessControlBits.Bits.SdCard |
  100. AccessControlBits.Bits.FormatSdCard |
  101. AccessControlBits.Bits.GetRightsId |
  102. AccessControlBits.Bits.RegisterProgramIndexMapInfo |
  103. AccessControlBits.Bits.MoveCacheStorage;
  104. // Sdb has save data access control info so we can't store just its access control bits
  105. private static ReadOnlySpan<byte> SdbFacData => new byte[]
  106. {
  107. 0x01, 0x00, 0x00, 0x00, 0x08, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1C, 0x00, 0x00, 0x00,
  108. 0x00, 0x00, 0x00, 0x00, 0x1C, 0x00, 0x00, 0x00, 0x18, 0x00, 0x00, 0x00, 0x02, 0x00, 0x00, 0x00,
  109. 0x03, 0x03, 0x00, 0x00, 0x1F, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x09, 0x10, 0x00, 0x00,
  110. 0x00, 0x00, 0x00, 0x01
  111. };
  112. private static ReadOnlySpan<byte> SdbFacDescriptor => new byte[]
  113. {
  114. 0x01, 0x00, 0x02, 0x00, 0x08, 0x00, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  115. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  116. 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1F, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
  117. 0x01, 0x09, 0x10, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01
  118. };
  119. }
  120. }