3249f8ff41
* Use source generated json serializers in order to improve code trimming * Use strongly typed github releases model to fetch updates instead of raw Newtonsoft.Json parsing * Use separate model for LogEventArgs serialization * Make dynamic object formatter static. Fix string builder pooling. * Do not inherit json version of LogEventArgs from EventArgs * Fix extra space in object formatting * Write log json directly to stream instead of using buffer writer * Rebase fixes * Rebase fixes * Rebase fixes * Enforce block-scoped namespaces in the solution. Convert style for existing code * Apply suggestions from code review Co-authored-by: TSRBerry <20988865+TSRBerry@users.noreply.github.com> * Rebase indent fix * Fix indent * Delete unnecessary json properties * Rebase fix * Remove overridden json property names as they are handled in the options * Apply suggestions from code review Co-authored-by: TSRBerry <20988865+TSRBerry@users.noreply.github.com> * Use default json options in github api calls * Indentation and spacing fixes * Fix json serialization * Fix missing JsonConverter for config enums * Add double \n\n after the whole string, not inside join --------- Co-authored-by: TSRBerry <20988865+TSRBerry@users.noreply.github.com>
180 lines
No EOL
7.7 KiB
C#
180 lines
No EOL
7.7 KiB
C#
using LibHac.Common;
|
|
using LibHac.Fs;
|
|
using LibHac.Fs.Fsa;
|
|
using LibHac.FsSystem;
|
|
using LibHac.Tools.Fs;
|
|
using LibHac.Tools.FsSystem;
|
|
using LibHac.Tools.FsSystem.NcaUtils;
|
|
using Ryujinx.Common.Configuration;
|
|
using Ryujinx.Common.Logging;
|
|
using Ryujinx.Common.Utilities;
|
|
using System;
|
|
using System.Collections.Generic;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
|
|
namespace Ryujinx.HLE.Loaders.Processes.Extensions
|
|
{
|
|
public static class PartitionFileSystemExtensions
|
|
{
|
|
private static readonly DownloadableContentJsonSerializerContext ContentSerializerContext = new(JsonHelper.GetDefaultSerializerOptions());
|
|
private static readonly TitleUpdateMetadataJsonSerializerContext TitleSerializerContext = new(JsonHelper.GetDefaultSerializerOptions());
|
|
|
|
internal static (bool, ProcessResult) TryLoad(this PartitionFileSystem partitionFileSystem, Switch device, string path, out string errorMessage)
|
|
{
|
|
errorMessage = null;
|
|
|
|
// Load required NCAs.
|
|
Nca mainNca = null;
|
|
Nca patchNca = null;
|
|
Nca controlNca = null;
|
|
|
|
try
|
|
{
|
|
device.Configuration.VirtualFileSystem.ImportTickets(partitionFileSystem);
|
|
|
|
// TODO: To support multi-games container, this should use CNMT NCA instead.
|
|
foreach (DirectoryEntryEx fileEntry in partitionFileSystem.EnumerateEntries("/", "*.nca"))
|
|
{
|
|
Nca nca = partitionFileSystem.GetNca(device, fileEntry.FullPath);
|
|
|
|
if (nca.GetProgramIndex() != device.Configuration.UserChannelPersistence.Index)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (nca.IsPatch())
|
|
{
|
|
patchNca = nca;
|
|
}
|
|
else if (nca.IsProgram())
|
|
{
|
|
mainNca = nca;
|
|
}
|
|
else if (nca.IsControl())
|
|
{
|
|
controlNca = nca;
|
|
}
|
|
}
|
|
|
|
ProcessLoaderHelper.RegisterProgramMapInfo(device, partitionFileSystem).ThrowIfFailure();
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
errorMessage = $"Unable to load: {ex.Message}";
|
|
|
|
return (false, ProcessResult.Failed);
|
|
}
|
|
|
|
if (mainNca != null)
|
|
{
|
|
if (mainNca.Header.ContentType != NcaContentType.Program)
|
|
{
|
|
errorMessage = "Selected NCA file is not a \"Program\" NCA";
|
|
|
|
return (false, ProcessResult.Failed);
|
|
}
|
|
|
|
// Load Update NCAs.
|
|
Nca updatePatchNca = null;
|
|
Nca updateControlNca = null;
|
|
|
|
if (ulong.TryParse(mainNca.Header.TitleId.ToString("x16"), NumberStyles.HexNumber, CultureInfo.InvariantCulture, out ulong titleIdBase))
|
|
{
|
|
// Clear the program index part.
|
|
titleIdBase &= ~0xFUL;
|
|
|
|
// Load update information if exists.
|
|
string titleUpdateMetadataPath = System.IO.Path.Combine(AppDataManager.GamesDirPath, titleIdBase.ToString("x16"), "updates.json");
|
|
if (File.Exists(titleUpdateMetadataPath))
|
|
{
|
|
string updatePath = JsonHelper.DeserializeFromFile(titleUpdateMetadataPath, TitleSerializerContext.TitleUpdateMetadata).Selected;
|
|
if (File.Exists(updatePath))
|
|
{
|
|
PartitionFileSystem updatePartitionFileSystem = new(new FileStream(updatePath, FileMode.Open, FileAccess.Read).AsStorage());
|
|
|
|
device.Configuration.VirtualFileSystem.ImportTickets(updatePartitionFileSystem);
|
|
|
|
// TODO: This should use CNMT NCA instead.
|
|
foreach (DirectoryEntryEx fileEntry in updatePartitionFileSystem.EnumerateEntries("/", "*.nca"))
|
|
{
|
|
Nca nca = updatePartitionFileSystem.GetNca(device, fileEntry.FullPath);
|
|
|
|
if (nca.GetProgramIndex() != device.Configuration.UserChannelPersistence.Index)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if ($"{nca.Header.TitleId.ToString("x16")[..^3]}000" != titleIdBase.ToString("x16"))
|
|
{
|
|
break;
|
|
}
|
|
|
|
if (nca.IsProgram())
|
|
{
|
|
updatePatchNca = nca;
|
|
}
|
|
else if (nca.IsControl())
|
|
{
|
|
updateControlNca = nca;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (updatePatchNca != null)
|
|
{
|
|
patchNca = updatePatchNca;
|
|
}
|
|
|
|
if (updateControlNca != null)
|
|
{
|
|
controlNca = updateControlNca;
|
|
}
|
|
|
|
// Load contained DownloadableContents.
|
|
// TODO: If we want to support multi-processes in future, we shouldn't clear AddOnContent data here.
|
|
device.Configuration.ContentManager.ClearAocData();
|
|
device.Configuration.ContentManager.AddAocData(partitionFileSystem, path, mainNca.Header.TitleId, device.Configuration.FsIntegrityCheckLevel);
|
|
|
|
// Load DownloadableContents.
|
|
string addOnContentMetadataPath = System.IO.Path.Combine(AppDataManager.GamesDirPath, mainNca.Header.TitleId.ToString("x16"), "dlc.json");
|
|
if (File.Exists(addOnContentMetadataPath))
|
|
{
|
|
List<DownloadableContentContainer> dlcContainerList = JsonHelper.DeserializeFromFile(addOnContentMetadataPath, ContentSerializerContext.ListDownloadableContentContainer);
|
|
|
|
foreach (DownloadableContentContainer downloadableContentContainer in dlcContainerList)
|
|
{
|
|
foreach (DownloadableContentNca downloadableContentNca in downloadableContentContainer.DownloadableContentNcaList)
|
|
{
|
|
if (File.Exists(downloadableContentContainer.ContainerPath) && downloadableContentNca.Enabled)
|
|
{
|
|
device.Configuration.ContentManager.AddAocItem(downloadableContentNca.TitleId, downloadableContentContainer.ContainerPath, downloadableContentNca.FullPath);
|
|
}
|
|
else
|
|
{
|
|
Logger.Warning?.Print(LogClass.Application, $"Cannot find AddOnContent file {downloadableContentContainer.ContainerPath}. It may have been moved or renamed.");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return (true, mainNca.Load(device, patchNca, controlNca));
|
|
}
|
|
|
|
errorMessage = "Unable to load: Could not find Main NCA";
|
|
|
|
return (false, ProcessResult.Failed);
|
|
}
|
|
|
|
public static Nca GetNca(this IFileSystem fileSystem, Switch device, string path)
|
|
{
|
|
using var ncaFile = new UniqueRef<IFile>();
|
|
|
|
fileSystem.OpenFile(ref ncaFile.Ref, path.ToU8Span(), OpenMode.Read).ThrowIfFailure();
|
|
|
|
return new Nca(device.Configuration.VirtualFileSystem.KeySet, ncaFile.Release().AsStorage());
|
|
}
|
|
}
|
|
} |