FileSystemProxyHelper.cs 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. using LibHac;
  2. using LibHac.Fs;
  3. using LibHac.FsSystem;
  4. using LibHac.FsSystem.NcaUtils;
  5. using LibHac.Spl;
  6. using System.IO;
  7. namespace Ryujinx.HLE.HOS.Services.Fs.FileSystemProxy
  8. {
  9. static class FileSystemProxyHelper
  10. {
  11. public static ResultCode OpenNsp(ServiceCtx context, string pfsPath, out IFileSystem openedFileSystem)
  12. {
  13. openedFileSystem = null;
  14. try
  15. {
  16. LocalStorage storage = new LocalStorage(pfsPath, FileAccess.Read, FileMode.Open);
  17. PartitionFileSystem nsp = new PartitionFileSystem(storage);
  18. ImportTitleKeysFromNsp(nsp, context.Device.System.KeySet);
  19. openedFileSystem = new IFileSystem(nsp);
  20. }
  21. catch (HorizonResultException ex)
  22. {
  23. return (ResultCode)ex.ResultValue.Value;
  24. }
  25. return ResultCode.Success;
  26. }
  27. public static ResultCode OpenNcaFs(ServiceCtx context, string ncaPath, LibHac.Fs.IStorage ncaStorage, out IFileSystem openedFileSystem)
  28. {
  29. openedFileSystem = null;
  30. try
  31. {
  32. Nca nca = new Nca(context.Device.System.KeySet, ncaStorage);
  33. if (!nca.SectionExists(NcaSectionType.Data))
  34. {
  35. return ResultCode.PartitionNotFound;
  36. }
  37. LibHac.Fs.IFileSystem fileSystem = nca.OpenFileSystem(NcaSectionType.Data, context.Device.System.FsIntegrityCheckLevel);
  38. openedFileSystem = new IFileSystem(fileSystem);
  39. }
  40. catch (HorizonResultException ex)
  41. {
  42. return (ResultCode)ex.ResultValue.Value;
  43. }
  44. return ResultCode.Success;
  45. }
  46. public static ResultCode OpenFileSystemFromInternalFile(ServiceCtx context, string fullPath, out IFileSystem openedFileSystem)
  47. {
  48. openedFileSystem = null;
  49. DirectoryInfo archivePath = new DirectoryInfo(fullPath).Parent;
  50. while (string.IsNullOrWhiteSpace(archivePath.Extension))
  51. {
  52. archivePath = archivePath.Parent;
  53. }
  54. if (archivePath.Extension == ".nsp" && File.Exists(archivePath.FullName))
  55. {
  56. FileStream pfsFile = new FileStream(
  57. archivePath.FullName.TrimEnd(Path.DirectorySeparatorChar),
  58. FileMode.Open,
  59. FileAccess.Read);
  60. try
  61. {
  62. PartitionFileSystem nsp = new PartitionFileSystem(pfsFile.AsStorage());
  63. ImportTitleKeysFromNsp(nsp, context.Device.System.KeySet);
  64. string filename = fullPath.Replace(archivePath.FullName, string.Empty).TrimStart('\\');
  65. Result result = nsp.OpenFile(out LibHac.Fs.IFile ncaFile, filename, OpenMode.Read);
  66. if (result.IsFailure())
  67. {
  68. return (ResultCode)result.Value;
  69. }
  70. return OpenNcaFs(context, fullPath, ncaFile.AsStorage(), out openedFileSystem);
  71. }
  72. catch (HorizonResultException ex)
  73. {
  74. return (ResultCode)ex.ResultValue.Value;
  75. }
  76. }
  77. return ResultCode.PathDoesNotExist;
  78. }
  79. public static void ImportTitleKeysFromNsp(LibHac.Fs.IFileSystem nsp, Keyset keySet)
  80. {
  81. foreach (DirectoryEntryEx ticketEntry in nsp.EnumerateEntries("/", "*.tik"))
  82. {
  83. Result result = nsp.OpenFile(out LibHac.Fs.IFile ticketFile, ticketEntry.FullPath, OpenMode.Read);
  84. if (result.IsSuccess())
  85. {
  86. Ticket ticket = new Ticket(ticketFile.AsStream());
  87. keySet.ExternalKeySet.Add(new RightsId(ticket.RightsId), new AccessKey(ticket.GetTitleKey(keySet)));
  88. }
  89. }
  90. }
  91. public static Result ReadFsPath(out FsPath path, ServiceCtx context, int index = 0)
  92. {
  93. long position = context.Request.PtrBuff[index].Position;
  94. long size = context.Request.PtrBuff[index].Size;
  95. byte[] pathBytes = context.Memory.ReadBytes(position, size);
  96. return FsPath.FromSpan(out path, pathBytes);
  97. }
  98. }
  99. }