Update Registrar from 12.1.0 to 13.1.0

This commit is contained in:
Alex Barney 2021-12-22 18:59:01 -07:00
parent cf7062788f
commit 03ba6b01aa
3 changed files with 123 additions and 24 deletions

View file

@ -1,7 +1,7 @@
using System; using System;
using LibHac.Common; using LibHac.Common;
using LibHac.Diag;
using LibHac.Fs.Impl; using LibHac.Fs.Impl;
using LibHac.Util;
namespace LibHac.Fs.Fsa; namespace LibHac.Fs.Fsa;
@ -15,12 +15,93 @@ public interface ISaveDataAttributeGetter : IDisposable
Result GetSaveDataAttribute(out SaveDataAttribute attribute); Result GetSaveDataAttribute(out SaveDataAttribute attribute);
} }
public interface IUnmountHookInvoker : IDisposable
{
void Invoke();
}
/// <summary> /// <summary>
/// Contains functions for registering and unregistering mounted <see cref="IFileSystem"/>s. /// Contains functions for registering and unregistering mounted <see cref="IFileSystem"/>s.
/// </summary> /// </summary>
/// <remarks>Based on FS 12.1.0 (nnSdk 12.3.1)</remarks> /// <remarks>Based on FS 13.1.0 (nnSdk 13.4.0)</remarks>
public static class Registrar public static class Registrar
{ {
private class UnmountHookFileSystem : IFileSystem
{
private UniqueRef<IFileSystem> _fileSystem;
private UniqueRef<IUnmountHookInvoker> _unmountHookInvoker;
public UnmountHookFileSystem(ref UniqueRef<IFileSystem> fileSystem,
ref UniqueRef<IUnmountHookInvoker> unmountHookInvoker)
{
_fileSystem = new UniqueRef<IFileSystem>(ref fileSystem);
_unmountHookInvoker = new UniqueRef<IUnmountHookInvoker>(ref unmountHookInvoker);
}
public override void Dispose()
{
if (_unmountHookInvoker.HasValue)
_unmountHookInvoker.Get.Invoke();
_unmountHookInvoker.Destroy();
_fileSystem.Destroy();
base.Dispose();
}
protected override Result DoCreateFile(in Path path, long size, CreateFileOptions option) =>
_fileSystem.Get.CreateFile(in path, size, option);
protected override Result DoDeleteFile(in Path path) => _fileSystem.Get.DeleteFile(in path);
protected override Result DoCreateDirectory(in Path path) => _fileSystem.Get.CreateDirectory(in path);
protected override Result DoDeleteDirectory(in Path path) => _fileSystem.Get.DeleteDirectory(in path);
protected override Result DoDeleteDirectoryRecursively(in Path path) =>
_fileSystem.Get.DeleteDirectoryRecursively(in path);
protected override Result DoCleanDirectoryRecursively(in Path path) =>
_fileSystem.Get.CleanDirectoryRecursively(in path);
protected override Result DoRenameFile(in Path currentPath, in Path newPath) =>
_fileSystem.Get.RenameFile(in currentPath, in newPath);
protected override Result DoRenameDirectory(in Path currentPath, in Path newPath) =>
_fileSystem.Get.RenameDirectory(in currentPath, in newPath);
protected override Result DoGetEntryType(out DirectoryEntryType entryType, in Path path) =>
_fileSystem.Get.GetEntryType(out entryType, in path);
protected override Result DoGetFreeSpaceSize(out long freeSpace, in Path path) =>
_fileSystem.Get.GetFreeSpaceSize(out freeSpace, in path);
protected override Result DoGetTotalSpaceSize(out long totalSpace, in Path path) =>
_fileSystem.Get.GetTotalSpaceSize(out totalSpace, in path);
protected override Result DoOpenFile(ref UniqueRef<IFile> outFile, in Path path, OpenMode mode) =>
_fileSystem.Get.OpenFile(ref outFile, in path, mode);
protected override Result DoOpenDirectory(ref UniqueRef<IDirectory> outDirectory, in Path path,
OpenDirectoryMode mode) =>
_fileSystem.Get.OpenDirectory(ref outDirectory, in path, mode);
protected override Result DoCommit() => _fileSystem.Get.Commit();
protected override Result DoCommitProvisionally(long counter) =>
_fileSystem.Get.CommitProvisionally(counter);
protected override Result DoRollback() => _fileSystem.Get.Rollback();
protected override Result DoFlush() => _fileSystem.Get.Flush();
protected override Result DoGetFileTimeStampRaw(out FileTimeStampRaw timeStamp, in Path path) =>
_fileSystem.Get.GetFileTimeStampRaw(out timeStamp, in path);
protected override Result DoQueryEntry(Span<byte> outBuffer, ReadOnlySpan<byte> inBuffer, QueryId queryId,
in Path path) => _fileSystem.Get.QueryEntry(outBuffer, inBuffer, queryId, in path);
}
public static Result Register(this FileSystemClient fs, U8Span name, ref UniqueRef<IFileSystem> fileSystem) public static Result Register(this FileSystemClient fs, U8Span name, ref UniqueRef<IFileSystem> fileSystem)
{ {
using var attributeGetter = new UniqueRef<ISaveDataAttributeGetter>(); using var attributeGetter = new UniqueRef<ISaveDataAttributeGetter>();
@ -51,12 +132,28 @@ public static class Registrar
public static Result Register(this FileSystemClient fs, U8Span name, IMultiCommitTarget multiCommitTarget, public static Result Register(this FileSystemClient fs, U8Span name, IMultiCommitTarget multiCommitTarget,
ref UniqueRef<IFileSystem> fileSystem, ref UniqueRef<ICommonMountNameGenerator> mountNameGenerator, ref UniqueRef<IFileSystem> fileSystem, ref UniqueRef<ICommonMountNameGenerator> mountNameGenerator,
bool useDataCache, bool usePathCache) bool useDataCache, IStorage storageForPurgeFileDataCache, bool usePathCache)
{
using var unmountHookInvoker = new UniqueRef<IUnmountHookInvoker>();
using var attributeGetter = new UniqueRef<ISaveDataAttributeGetter>();
Result rc = Register(fs, name, multiCommitTarget, ref fileSystem, ref mountNameGenerator,
ref attributeGetter.Ref(), useDataCache, storageForPurgeFileDataCache, usePathCache,
ref unmountHookInvoker.Ref());
if (rc.IsFailure()) return rc.Miss();
return Result.Success;
}
public static Result Register(this FileSystemClient fs, U8Span name, IMultiCommitTarget multiCommitTarget,
ref UniqueRef<IFileSystem> fileSystem, ref UniqueRef<ICommonMountNameGenerator> mountNameGenerator,
bool useDataCache, IStorage storageForPurgeFileDataCache, bool usePathCache,
ref UniqueRef<IUnmountHookInvoker> unmountHook)
{ {
using var attributeGetter = new UniqueRef<ISaveDataAttributeGetter>(); using var attributeGetter = new UniqueRef<ISaveDataAttributeGetter>();
Result rc = Register(fs, name, multiCommitTarget, ref fileSystem, ref mountNameGenerator, Result rc = Register(fs, name, multiCommitTarget, ref fileSystem, ref mountNameGenerator,
ref attributeGetter.Ref(), useDataCache, usePathCache, new Optional<Ncm.DataId>()); ref attributeGetter.Ref(), useDataCache, storageForPurgeFileDataCache, usePathCache, ref unmountHook);
if (rc.IsFailure()) return rc.Miss(); if (rc.IsFailure()) return rc.Miss();
return Result.Success; return Result.Success;
@ -64,12 +161,14 @@ public static class Registrar
public static Result Register(this FileSystemClient fs, U8Span name, IMultiCommitTarget multiCommitTarget, public static Result Register(this FileSystemClient fs, U8Span name, IMultiCommitTarget multiCommitTarget,
ref UniqueRef<IFileSystem> fileSystem, ref UniqueRef<ICommonMountNameGenerator> mountNameGenerator, ref UniqueRef<IFileSystem> fileSystem, ref UniqueRef<ICommonMountNameGenerator> mountNameGenerator,
bool useDataCache, bool usePathCache, Optional<Ncm.DataId> dataId) ref UniqueRef<ISaveDataAttributeGetter> saveAttributeGetter, bool useDataCache,
IStorage storageForPurgeFileDataCache, bool usePathCache)
{ {
using var attributeGetter = new UniqueRef<ISaveDataAttributeGetter>(); using var unmountHookInvoker = new UniqueRef<IUnmountHookInvoker>();
Result rc = Register(fs, name, multiCommitTarget, ref fileSystem, ref mountNameGenerator, Result rc = Register(fs, name, multiCommitTarget, ref fileSystem, ref mountNameGenerator,
ref attributeGetter.Ref(), useDataCache, usePathCache, dataId); ref saveAttributeGetter, useDataCache, storageForPurgeFileDataCache, usePathCache,
ref unmountHookInvoker.Ref());
if (rc.IsFailure()) return rc.Miss(); if (rc.IsFailure()) return rc.Miss();
return Result.Success; return Result.Success;
@ -77,29 +176,29 @@ public static class Registrar
public static Result Register(this FileSystemClient fs, U8Span name, IMultiCommitTarget multiCommitTarget, public static Result Register(this FileSystemClient fs, U8Span name, IMultiCommitTarget multiCommitTarget,
ref UniqueRef<IFileSystem> fileSystem, ref UniqueRef<ICommonMountNameGenerator> mountNameGenerator, ref UniqueRef<IFileSystem> fileSystem, ref UniqueRef<ICommonMountNameGenerator> mountNameGenerator,
ref UniqueRef<ISaveDataAttributeGetter> saveAttributeGetter, bool useDataCache, bool usePathCache) ref UniqueRef<ISaveDataAttributeGetter> saveAttributeGetter, bool useDataCache,
IStorage storageForPurgeFileDataCache, bool usePathCache, ref UniqueRef<IUnmountHookInvoker> unmountHook)
{ {
Result rc = Register(fs, name, multiCommitTarget, ref fileSystem, ref mountNameGenerator, if (useDataCache)
ref saveAttributeGetter, useDataCache, usePathCache, new Optional<Ncm.DataId>()); Assert.SdkAssert(storageForPurgeFileDataCache is not null);
if (rc.IsFailure()) return rc.Miss();
return Result.Success; using (var unmountHookFileSystem =
new UniqueRef<UnmountHookFileSystem>(new UnmountHookFileSystem(ref fileSystem, ref unmountHook)))
{
fileSystem.Set(ref unmountHookFileSystem.Ref());
} }
public static Result Register(this FileSystemClient fs, U8Span name, IMultiCommitTarget multiCommitTarget, if (!fileSystem.HasValue)
ref UniqueRef<IFileSystem> fileSystem, ref UniqueRef<ICommonMountNameGenerator> mountNameGenerator, return ResultFs.AllocationMemoryFailedInRegisterB.Log();
ref UniqueRef<ISaveDataAttributeGetter> saveAttributeGetter, bool useDataCache, bool usePathCache,
Optional<Ncm.DataId> dataId)
{
using var accessor = new UniqueRef<FileSystemAccessor>(new FileSystemAccessor(fs.Hos, name, using var accessor = new UniqueRef<FileSystemAccessor>(new FileSystemAccessor(fs.Hos, name,
multiCommitTarget, ref fileSystem, ref mountNameGenerator, ref saveAttributeGetter)); multiCommitTarget, ref fileSystem, ref mountNameGenerator, ref saveAttributeGetter));
if (!accessor.HasValue) if (!accessor.HasValue)
return ResultFs.AllocationMemoryFailedInRegisterB.Log(); return ResultFs.AllocationMemoryFailedInRegisterB.Log();
// accessor.Get.SetFileDataCacheAttachable(useDataCache); accessor.Get.SetFileDataCacheAttachable(useDataCache, storageForPurgeFileDataCache);
accessor.Get.SetPathBasedFileDataCacheAttachable(usePathCache); accessor.Get.SetPathBasedFileDataCacheAttachable(usePathCache);
// accessor.Get.SetDataId(dataId);
Result rc = fs.Impl.Register(ref accessor.Ref()); Result rc = fs.Impl.Register(ref accessor.Ref());
if (rc.IsFailure()) return rc.Miss(); if (rc.IsFailure()) return rc.Miss();

View file

@ -50,7 +50,7 @@ public static class SaveData
using var mountNameGenerator = new UniqueRef<ICommonMountNameGenerator>(); using var mountNameGenerator = new UniqueRef<ICommonMountNameGenerator>();
rc = fs.Fs.Register(mountName, fileSystemAdapterRaw, ref fileSystemAdapter.Ref(), rc = fs.Fs.Register(mountName, fileSystemAdapterRaw, ref fileSystemAdapter.Ref(),
ref mountNameGenerator.Ref(), false, true); ref mountNameGenerator.Ref(), false, null, true);
if (rc.IsFailure()) return rc.Miss(); if (rc.IsFailure()) return rc.Miss();
return Result.Success; return Result.Success;

View file

@ -87,7 +87,7 @@ public static class SystemSaveData
{ {
using var mountNameGenerator = new UniqueRef<ICommonMountNameGenerator>(); using var mountNameGenerator = new UniqueRef<ICommonMountNameGenerator>();
return fs.Register(mountName, fileSystemAdapterRaw, ref fileSystemAdapter.Ref(), return fs.Register(mountName, fileSystemAdapterRaw, ref fileSystemAdapter.Ref(),
ref mountNameGenerator.Ref(), false, false); ref mountNameGenerator.Ref(), false, null, false);
} }
else else
{ {