2020-01-21 14:23:11 -08:00
|
|
|
using LibHac;
|
2020-05-14 23:16:46 -07:00
|
|
|
using LibHac.Common;
|
2020-01-21 14:23:11 -08:00
|
|
|
using LibHac.Fs;
|
|
|
|
using LibHac.FsService;
|
|
|
|
using LibHac.FsSystem;
|
2020-05-14 23:16:46 -07:00
|
|
|
using LibHac.Spl;
|
2018-11-18 11:37:41 -08:00
|
|
|
using Ryujinx.HLE.FileSystem.Content;
|
2018-09-08 15:04:26 -07:00
|
|
|
using Ryujinx.HLE.HOS;
|
2018-02-04 15:08:20 -08:00
|
|
|
using System;
|
|
|
|
using System.IO;
|
|
|
|
|
2018-09-08 15:04:26 -07:00
|
|
|
namespace Ryujinx.HLE.FileSystem
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2019-09-02 09:03:57 -07:00
|
|
|
public class VirtualFileSystem : IDisposable
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2020-01-05 03:49:44 -08:00
|
|
|
public const string BasePath = "Ryujinx";
|
|
|
|
public const string NandPath = "bis";
|
|
|
|
public const string SdCardPath = "sdcard";
|
2018-09-08 15:04:26 -07:00
|
|
|
public const string SystemPath = "system";
|
|
|
|
|
2018-11-18 11:37:41 -08:00
|
|
|
public static string SafeNandPath = Path.Combine(NandPath, "safe");
|
2018-09-08 15:04:26 -07:00
|
|
|
public static string SystemNandPath = Path.Combine(NandPath, "system");
|
|
|
|
public static string UserNandPath = Path.Combine(NandPath, "user");
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2020-01-24 08:01:21 -08:00
|
|
|
private static bool _isInitialized = false;
|
|
|
|
|
2020-01-21 14:23:11 -08:00
|
|
|
public Keyset KeySet { get; private set; }
|
|
|
|
public FileSystemServer FsServer { get; private set; }
|
|
|
|
public FileSystemClient FsClient { get; private set; }
|
|
|
|
public EmulatedGameCard GameCard { get; private set; }
|
2020-03-03 06:07:06 -08:00
|
|
|
public EmulatedSdCard SdCard { get; private set; }
|
2020-01-21 14:23:11 -08:00
|
|
|
|
2020-01-24 08:01:21 -08:00
|
|
|
private VirtualFileSystem()
|
2020-01-21 14:23:11 -08:00
|
|
|
{
|
|
|
|
Reload();
|
|
|
|
}
|
|
|
|
|
2018-02-04 15:08:20 -08:00
|
|
|
public Stream RomFs { get; private set; }
|
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
public void LoadRomFs(string fileName)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-12-06 03:16:24 -08:00
|
|
|
RomFs = new FileStream(fileName, FileMode.Open, FileAccess.Read);
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
public void SetRomFs(Stream romfsStream)
|
2018-09-08 11:33:27 -07:00
|
|
|
{
|
|
|
|
RomFs?.Close();
|
2018-12-06 03:16:24 -08:00
|
|
|
RomFs = romfsStream;
|
2018-09-08 11:33:27 -07:00
|
|
|
}
|
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
public string GetFullPath(string basePath, string fileName)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-12-06 03:16:24 -08:00
|
|
|
if (fileName.StartsWith("//"))
|
2018-03-06 12:27:50 -08:00
|
|
|
{
|
2018-12-06 03:16:24 -08:00
|
|
|
fileName = fileName.Substring(2);
|
2018-03-06 12:27:50 -08:00
|
|
|
}
|
2018-12-06 03:16:24 -08:00
|
|
|
else if (fileName.StartsWith('/'))
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-12-06 03:16:24 -08:00
|
|
|
fileName = fileName.Substring(1);
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
2018-03-06 12:27:50 -08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
string fullPath = Path.GetFullPath(Path.Combine(basePath, fileName));
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
if (!fullPath.StartsWith(GetBasePath()))
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
return fullPath;
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
|
2019-09-08 14:33:40 -07:00
|
|
|
public string GetSdCardPath() => MakeFullPath(SdCardPath);
|
2018-02-06 15:28:32 -08:00
|
|
|
|
2019-09-08 14:33:40 -07:00
|
|
|
public string GetNandPath() => MakeFullPath(NandPath);
|
2018-02-06 15:28:32 -08:00
|
|
|
|
2019-09-08 14:33:40 -07:00
|
|
|
public string GetSystemPath() => MakeFullPath(SystemPath);
|
2018-08-04 14:38:49 -07:00
|
|
|
|
2019-09-08 14:33:40 -07:00
|
|
|
internal string GetSavePath(ServiceCtx context, SaveInfo saveInfo, bool isDirectory = true)
|
2018-09-08 15:04:26 -07:00
|
|
|
{
|
2019-09-08 14:33:40 -07:00
|
|
|
string saveUserPath = "";
|
|
|
|
string baseSavePath = NandPath;
|
|
|
|
ulong currentTitleId = saveInfo.TitleId;
|
|
|
|
|
|
|
|
switch (saveInfo.SaveSpaceId)
|
|
|
|
{
|
|
|
|
case SaveSpaceId.NandUser: baseSavePath = UserNandPath; break;
|
|
|
|
case SaveSpaceId.NandSystem: baseSavePath = SystemNandPath; break;
|
|
|
|
case SaveSpaceId.SdCard: baseSavePath = Path.Combine(SdCardPath, "Nintendo"); break;
|
|
|
|
}
|
|
|
|
|
|
|
|
baseSavePath = Path.Combine(baseSavePath, "save");
|
|
|
|
|
|
|
|
if (saveInfo.TitleId == 0 && saveInfo.SaveDataType == SaveDataType.SaveData)
|
|
|
|
{
|
|
|
|
currentTitleId = context.Process.TitleId;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (saveInfo.SaveSpaceId == SaveSpaceId.NandUser)
|
|
|
|
{
|
|
|
|
saveUserPath = saveInfo.UserId.IsNull ? "savecommon" : saveInfo.UserId.ToString();
|
|
|
|
}
|
|
|
|
|
|
|
|
string savePath = Path.Combine(baseSavePath,
|
|
|
|
saveInfo.SaveId.ToString("x16"),
|
|
|
|
saveUserPath,
|
|
|
|
saveInfo.SaveDataType == SaveDataType.SaveData ? currentTitleId.ToString("x16") : string.Empty);
|
|
|
|
|
|
|
|
return MakeFullPath(savePath, isDirectory);
|
2018-09-08 15:04:26 -07:00
|
|
|
}
|
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
public string GetFullPartitionPath(string partitionPath)
|
2018-11-18 11:37:41 -08:00
|
|
|
{
|
2019-09-08 14:33:40 -07:00
|
|
|
return MakeFullPath(partitionPath);
|
2018-11-18 11:37:41 -08:00
|
|
|
}
|
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
public string SwitchPathToSystemPath(string switchPath)
|
2018-07-17 12:14:27 -07:00
|
|
|
{
|
2018-12-06 03:16:24 -08:00
|
|
|
string[] parts = switchPath.Split(":");
|
2018-11-18 11:37:41 -08:00
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
if (parts.Length != 2)
|
2018-07-17 12:14:27 -07:00
|
|
|
{
|
|
|
|
return null;
|
|
|
|
}
|
2018-11-18 16:20:17 -08:00
|
|
|
|
2019-09-08 14:33:40 -07:00
|
|
|
return GetFullPath(MakeFullPath(parts[0]), parts[1]);
|
2018-07-17 12:14:27 -07:00
|
|
|
}
|
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
public string SystemPathToSwitchPath(string systemPath)
|
2018-07-17 12:14:27 -07:00
|
|
|
{
|
2018-12-06 03:16:24 -08:00
|
|
|
string baseSystemPath = GetBasePath() + Path.DirectorySeparatorChar;
|
2018-11-18 11:37:41 -08:00
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
if (systemPath.StartsWith(baseSystemPath))
|
2018-07-17 12:14:27 -07:00
|
|
|
{
|
2018-12-06 03:16:24 -08:00
|
|
|
string rawPath = systemPath.Replace(baseSystemPath, "");
|
|
|
|
int firstSeparatorOffset = rawPath.IndexOf(Path.DirectorySeparatorChar);
|
2018-11-18 11:37:41 -08:00
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
if (firstSeparatorOffset == -1)
|
2018-07-17 12:14:27 -07:00
|
|
|
{
|
2018-12-06 03:16:24 -08:00
|
|
|
return $"{rawPath}:/";
|
2018-07-17 12:14:27 -07:00
|
|
|
}
|
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
string basePath = rawPath.Substring(0, firstSeparatorOffset);
|
|
|
|
string fileName = rawPath.Substring(firstSeparatorOffset + 1);
|
2018-11-18 11:37:41 -08:00
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
return $"{basePath}:/{fileName}";
|
2018-07-17 12:14:27 -07:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2019-09-08 14:33:40 -07:00
|
|
|
private string MakeFullPath(string path, bool isDirectory = true)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-11-18 11:37:41 -08:00
|
|
|
// Handles Common Switch Content Paths
|
2019-09-08 14:33:40 -07:00
|
|
|
switch (path)
|
2018-11-18 11:37:41 -08:00
|
|
|
{
|
|
|
|
case ContentPath.SdCard:
|
|
|
|
case "@Sdcard":
|
2019-09-08 14:33:40 -07:00
|
|
|
path = SdCardPath;
|
2018-11-18 11:37:41 -08:00
|
|
|
break;
|
|
|
|
case ContentPath.User:
|
2019-09-08 14:33:40 -07:00
|
|
|
path = UserNandPath;
|
2018-11-18 11:37:41 -08:00
|
|
|
break;
|
|
|
|
case ContentPath.System:
|
2019-09-08 14:33:40 -07:00
|
|
|
path = SystemNandPath;
|
2018-11-18 11:37:41 -08:00
|
|
|
break;
|
|
|
|
case ContentPath.SdCardContent:
|
2019-09-08 14:33:40 -07:00
|
|
|
path = Path.Combine(SdCardPath, "Nintendo", "Contents");
|
2018-11-18 11:37:41 -08:00
|
|
|
break;
|
|
|
|
case ContentPath.UserContent:
|
2019-09-08 14:33:40 -07:00
|
|
|
path = Path.Combine(UserNandPath, "Contents");
|
2018-11-18 11:37:41 -08:00
|
|
|
break;
|
|
|
|
case ContentPath.SystemContent:
|
2019-09-08 14:33:40 -07:00
|
|
|
path = Path.Combine(SystemNandPath, "Contents");
|
2018-11-18 11:37:41 -08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-09-08 14:33:40 -07:00
|
|
|
string fullPath = Path.Combine(GetBasePath(), path);
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2019-09-08 14:33:40 -07:00
|
|
|
if (isDirectory)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2019-09-08 14:33:40 -07:00
|
|
|
if (!Directory.Exists(fullPath))
|
|
|
|
{
|
|
|
|
Directory.CreateDirectory(fullPath);
|
|
|
|
}
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
return fullPath;
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
|
2018-02-21 13:56:52 -08:00
|
|
|
public DriveInfo GetDrive()
|
|
|
|
{
|
|
|
|
return new DriveInfo(Path.GetPathRoot(GetBasePath()));
|
|
|
|
}
|
|
|
|
|
|
|
|
public string GetBasePath()
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-12-06 03:16:24 -08:00
|
|
|
string appDataPath = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData);
|
2018-02-27 15:45:07 -08:00
|
|
|
|
2018-12-06 03:16:24 -08:00
|
|
|
return Path.Combine(appDataPath, BasePath);
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
|
2020-01-21 14:23:11 -08:00
|
|
|
public void Reload()
|
|
|
|
{
|
|
|
|
ReloadKeySet();
|
|
|
|
|
|
|
|
LocalFileSystem serverBaseFs = new LocalFileSystem(GetBasePath());
|
|
|
|
|
|
|
|
DefaultFsServerObjects fsServerObjects = DefaultFsServerObjects.GetDefaultEmulatedCreators(serverBaseFs, KeySet);
|
|
|
|
|
|
|
|
GameCard = fsServerObjects.GameCard;
|
2020-03-03 06:07:06 -08:00
|
|
|
SdCard = fsServerObjects.SdCard;
|
2020-01-21 14:23:11 -08:00
|
|
|
|
2020-03-09 15:34:35 -07:00
|
|
|
SdCard.SetSdCardInsertionStatus(true);
|
|
|
|
|
2020-01-21 14:23:11 -08:00
|
|
|
FileSystemServerConfig fsServerConfig = new FileSystemServerConfig
|
|
|
|
{
|
|
|
|
FsCreators = fsServerObjects.FsCreators,
|
|
|
|
DeviceOperator = fsServerObjects.DeviceOperator,
|
|
|
|
ExternalKeySet = KeySet.ExternalKeySet
|
|
|
|
};
|
|
|
|
|
|
|
|
FsServer = new FileSystemServer(fsServerConfig);
|
|
|
|
FsClient = FsServer.CreateFileSystemClient();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private void ReloadKeySet()
|
|
|
|
{
|
|
|
|
string keyFile = null;
|
|
|
|
string titleKeyFile = null;
|
|
|
|
string consoleKeyFile = null;
|
|
|
|
|
|
|
|
string home = Environment.GetFolderPath(Environment.SpecialFolder.UserProfile);
|
|
|
|
|
|
|
|
LoadSetAtPath(Path.Combine(home, ".switch"));
|
|
|
|
LoadSetAtPath(GetSystemPath());
|
|
|
|
|
|
|
|
void LoadSetAtPath(string basePath)
|
|
|
|
{
|
|
|
|
string localKeyFile = Path.Combine(basePath, "prod.keys");
|
|
|
|
string localTitleKeyFile = Path.Combine(basePath, "title.keys");
|
|
|
|
string localConsoleKeyFile = Path.Combine(basePath, "console.keys");
|
|
|
|
|
|
|
|
if (File.Exists(localKeyFile))
|
|
|
|
{
|
|
|
|
keyFile = localKeyFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (File.Exists(localTitleKeyFile))
|
|
|
|
{
|
|
|
|
titleKeyFile = localTitleKeyFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (File.Exists(localConsoleKeyFile))
|
|
|
|
{
|
|
|
|
consoleKeyFile = localConsoleKeyFile;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
KeySet = ExternalKeyReader.ReadKeyFile(keyFile, titleKeyFile, consoleKeyFile);
|
|
|
|
}
|
|
|
|
|
2020-05-14 23:16:46 -07:00
|
|
|
public void ImportTickets(IFileSystem fs)
|
|
|
|
{
|
|
|
|
foreach (DirectoryEntryEx ticketEntry in fs.EnumerateEntries("/", "*.tik"))
|
|
|
|
{
|
|
|
|
Result result = fs.OpenFile(out IFile ticketFile, ticketEntry.FullPath.ToU8Span(), OpenMode.Read);
|
|
|
|
|
|
|
|
if (result.IsSuccess())
|
|
|
|
{
|
|
|
|
Ticket ticket = new Ticket(ticketFile.AsStream());
|
|
|
|
|
|
|
|
KeySet.ExternalKeySet.Add(new RightsId(ticket.RightsId), new AccessKey(ticket.GetTitleKey(KeySet)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-21 14:23:11 -08:00
|
|
|
public void Unload()
|
|
|
|
{
|
|
|
|
RomFs?.Dispose();
|
|
|
|
}
|
|
|
|
|
2018-02-04 15:08:20 -08:00
|
|
|
public void Dispose()
|
|
|
|
{
|
|
|
|
Dispose(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected virtual void Dispose(bool disposing)
|
|
|
|
{
|
2018-02-20 02:54:00 -08:00
|
|
|
if (disposing)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2020-01-21 14:23:11 -08:00
|
|
|
Unload();
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
}
|
2020-01-24 08:01:21 -08:00
|
|
|
|
|
|
|
public static VirtualFileSystem CreateInstance()
|
|
|
|
{
|
|
|
|
if (_isInitialized)
|
|
|
|
{
|
2020-05-14 23:16:46 -07:00
|
|
|
throw new InvalidOperationException($"VirtualFileSystem can only be instantiated once!");
|
2020-01-24 08:01:21 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
_isInitialized = true;
|
|
|
|
|
|
|
|
return new VirtualFileSystem();
|
|
|
|
}
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
}
|