From ccb8c078aa0d948ad9a65d1b8ced760492ea48b3 Mon Sep 17 00:00:00 2001 From: Alex Barney Date: Tue, 18 May 2021 17:07:18 -0700 Subject: [PATCH] Write extra data when creating directory save data --- src/LibHac/Fs/Shim/PosixTime.cs | 18 ++++++ src/LibHac/FsSrv/SaveDataFileSystemService.cs | 2 +- .../FsSrv/SaveDataFileSystemServiceImpl.cs | 59 +++++++++++++++---- 3 files changed, 65 insertions(+), 14 deletions(-) create mode 100644 src/LibHac/Fs/Shim/PosixTime.cs diff --git a/src/LibHac/Fs/Shim/PosixTime.cs b/src/LibHac/Fs/Shim/PosixTime.cs new file mode 100644 index 00000000..ddc923bd --- /dev/null +++ b/src/LibHac/Fs/Shim/PosixTime.cs @@ -0,0 +1,18 @@ +using LibHac.FsSrv.Sf; + +namespace LibHac.Fs.Shim +{ + public static class PosixTimeShim + { + public static Result SetCurrentPosixTime(this FileSystemClient fs, Time.PosixTime currentPosixTime, + int timeDifferenceSeconds) + { + using ReferenceCountedDisposable fsProxy = fs.Impl.GetFileSystemProxyServiceObject(); + + Result rc = fsProxy.Target.SetCurrentPosixTimeWithTimeDifference(currentPosixTime.Value, + timeDifferenceSeconds); + fs.Impl.AbortIfNeeded(rc); + return rc; + } + } +} diff --git a/src/LibHac/FsSrv/SaveDataFileSystemService.cs b/src/LibHac/FsSrv/SaveDataFileSystemService.cs index 0929f45e..1662a649 100644 --- a/src/LibHac/FsSrv/SaveDataFileSystemService.cs +++ b/src/LibHac/FsSrv/SaveDataFileSystemService.cs @@ -984,7 +984,7 @@ namespace LibHac.FsSrv Result rc = RemoveSaveIndexerEntry(); if (rc.IsFailure()) return rc; - return Result.Success; + return ResultFs.TargetNotFound.LogConverted(saveFsResult); } if (ResultFs.TargetNotFound.Includes(saveFsResult)) diff --git a/src/LibHac/FsSrv/SaveDataFileSystemServiceImpl.cs b/src/LibHac/FsSrv/SaveDataFileSystemServiceImpl.cs index d8cda539..5ef851c4 100644 --- a/src/LibHac/FsSrv/SaveDataFileSystemServiceImpl.cs +++ b/src/LibHac/FsSrv/SaveDataFileSystemServiceImpl.cs @@ -134,15 +134,9 @@ namespace LibHac.FsSrv bool allowDirectorySaveData = IsAllowedDirectorySaveData2(spaceId, saveDataRootPath); - if (allowDirectorySaveData) - { - Span saveDirectoryPath = stackalloc byte[0x12]; - var sb = new U8StringBuilder(saveDirectoryPath); - sb.Append((byte)'/').AppendFormat(saveDataId, 'x', 16); - - rc = Utility.EnsureDirectory(saveDirectoryFs.Target, new U8Span(saveDirectoryPath)); - if (rc.IsFailure()) return rc; - } + // Note: When directory save data is allowed, Nintendo creates the save directory if it doesn't exist. + // This bypasses normal save data// creation, leaving the save with empty extra data. + // Instead, we return that the save doesn't exist if the directory is missing. // Note: Nintendo doesn't cache directory save data // if (!allowDirectorySaveData) @@ -345,10 +339,10 @@ namespace LibHac.FsSrv { // Use directory save data for now - ReferenceCountedDisposable fileSystem = null; + ReferenceCountedDisposable saveDirectoryFs = null; try { - Result rc = OpenSaveDataDirectoryFileSystem(out fileSystem, creationInfo.SpaceId, saveDataRootPath, + Result rc = OpenSaveDataDirectoryFileSystem(out saveDirectoryFs, creationInfo.SpaceId, saveDataRootPath, false); if (rc.IsFailure()) return rc; @@ -358,7 +352,46 @@ namespace LibHac.FsSrv if (_config.IsPseudoSaveData()) { - return Utility.EnsureDirectory(fileSystem.Target, new U8Span(saveDataPathBuffer)); + rc = Utility.EnsureDirectory(saveDirectoryFs.Target, new U8Span(saveDataPathBuffer)); + if (rc.IsFailure()) return rc; + + ReferenceCountedDisposable saveFs = null; + ReferenceCountedDisposable extraDataAccessor = null; + try + { + bool isJournalingSupported = SaveDataProperties.IsJournalingSupported(attribute.Type); + + rc = _config.SaveFsCreator.Create(out saveFs, out extraDataAccessor, _saveDataFsCacheManager, + ref saveDirectoryFs, creationInfo.SpaceId, saveDataId, allowDirectorySaveData: true, + useDeviceUniqueMac: false, isJournalingSupported, isMultiCommitSupported: false, + openReadOnly: false, openShared: false, _timeStampGetter); + if (rc.IsFailure()) return rc; + + var extraData = new SaveDataExtraData(); + extraData.Attribute = attribute; + extraData.OwnerId = creationInfo.OwnerId; + + rc = GetSaveDataCommitTimeStamp(out extraData.TimeStamp); + if (rc.IsFailure()) + extraData.TimeStamp = 0; + + extraData.CommitId = 0; + _config.GenerateRandomData(SpanHelpers.AsByteSpan(ref extraData.CommitId)).IgnoreResult(); + + extraData.Flags = creationInfo.Flags; + extraData.DataSize = creationInfo.Size; + extraData.JournalSize = creationInfo.JournalSize; + + rc = extraDataAccessor.Target.WriteExtraData(in extraData); + if (rc.IsFailure()) return rc; + + return extraDataAccessor.Target.CommitExtraData(true); + } + finally + { + saveFs?.Dispose(); + extraDataAccessor?.Dispose(); + } } else { @@ -367,7 +400,7 @@ namespace LibHac.FsSrv } finally { - fileSystem?.Dispose(); + saveDirectoryFs?.Dispose(); } }