 21c9c04f9f
			
		
	
	
		21c9c04f9f
		
			
		
	
	
	
	
		
			
			* Update LibHac * Add IMultiCommitManager * Updates * Delete NuGet.Config * Add command version
		
			
				
	
	
		
			511 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			511 lines
		
	
	
	
		
			15 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using LibHac;
 | |
| using LibHac.Common;
 | |
| using LibHac.Fs;
 | |
| using LibHac.Fs.Shim;
 | |
| using LibHac.Ncm;
 | |
| using Ryujinx.HLE.HOS.Services.Mii.Types;
 | |
| using System.Runtime.CompilerServices;
 | |
| 
 | |
| namespace Ryujinx.HLE.HOS.Services.Mii
 | |
| {
 | |
|     class MiiDatabaseManager
 | |
|     {
 | |
|         private static bool IsTestModeEnabled = false;
 | |
|         private static uint MountCounter      = 0;
 | |
| 
 | |
|         private const ulong  DatabaseTestSaveDataId = 0x8000000000000031;
 | |
|         private const ulong  DatabaseSaveDataId     = 0x8000000000000030;
 | |
|         private const ulong  NsTitleId              = 0x010000000000001F;
 | |
|         private const ulong  SdbTitleId             = 0x0100000000000039;
 | |
| 
 | |
|         private static U8String DatabasePath = new U8String("mii:/MiiDatabase.dat");
 | |
|         private static U8String MountName    = new U8String("mii");
 | |
| 
 | |
|         private NintendoFigurineDatabase _database;
 | |
|         private bool                     _isDirty;
 | |
| 
 | |
|         private FileSystemClient _filesystemClient;
 | |
| 
 | |
|         protected ulong UpdateCounter { get; private set; }
 | |
| 
 | |
|         public MiiDatabaseManager()
 | |
|         {
 | |
|             _database     = new NintendoFigurineDatabase();
 | |
|             _isDirty      = false;
 | |
|             UpdateCounter = 0;
 | |
|         }
 | |
| 
 | |
|         private void ResetDatabase()
 | |
|         {
 | |
|             _database = new NintendoFigurineDatabase();
 | |
|             _database.Format();
 | |
|         }
 | |
| 
 | |
|         private void MarkDirty(DatabaseSessionMetadata metadata)
 | |
|         {
 | |
|             _isDirty = true;
 | |
| 
 | |
|             UpdateCounter++;
 | |
| 
 | |
|             metadata.UpdateCounter = UpdateCounter;
 | |
|         }
 | |
| 
 | |
|         private bool GetAtVirtualIndex(int index, out int realIndex, out StoreData storeData)
 | |
|         {
 | |
|             realIndex = -1;
 | |
|             storeData = new StoreData();
 | |
| 
 | |
|             int virtualIndex = 0;
 | |
| 
 | |
|             for (int i = 0; i < _database.Length; i++)
 | |
|             {
 | |
|                 StoreData tmp = _database.Get(i);
 | |
| 
 | |
|                 if (!tmp.IsSpecial())
 | |
|                 {
 | |
|                     if (index == virtualIndex)
 | |
|                     {
 | |
|                         realIndex = i;
 | |
|                         storeData = tmp;
 | |
| 
 | |
|                         return true;
 | |
|                     }
 | |
| 
 | |
|                     virtualIndex++;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         private int ConvertRealIndexToVirtualIndex(int realIndex)
 | |
|         {
 | |
|             int virtualIndex = 0;
 | |
| 
 | |
|             for (int i = 0; i < realIndex; i++)
 | |
|             {
 | |
|                 StoreData tmp = _database.Get(i);
 | |
| 
 | |
|                 if (!tmp.IsSpecial())
 | |
|                 {
 | |
|                     virtualIndex++;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return virtualIndex;
 | |
|         }
 | |
| 
 | |
|         public void InitializeDatabase(Switch device)
 | |
|         {
 | |
|             _filesystemClient = device.FileSystem.FsClient;
 | |
| 
 | |
|             // Ensure we have valid data in the database
 | |
|             _database.Format();
 | |
| 
 | |
|             MountSave();
 | |
|         }
 | |
| 
 | |
|         private Result MountSave()
 | |
|         {
 | |
|             Result result = Result.Success;
 | |
| 
 | |
|             if (MountCounter == 0)
 | |
|             {
 | |
|                 ulong targetSaveDataId;
 | |
|                 ulong targetTitleId;
 | |
| 
 | |
|                 if (IsTestModeEnabled)
 | |
|                 {
 | |
|                     targetSaveDataId = DatabaseTestSaveDataId;
 | |
|                     targetTitleId    = SdbTitleId;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     targetSaveDataId = DatabaseSaveDataId;
 | |
| 
 | |
|                     // Nintendo use NS TitleID when creating the production save even on sdb, let's follow that behaviour.
 | |
|                     targetTitleId = NsTitleId;
 | |
|                 }
 | |
| 
 | |
|                 U8Span mountName = new U8Span(MountName);
 | |
| 
 | |
|                 result = _filesystemClient.MountSystemSaveData(mountName, SaveDataSpaceId.System, targetSaveDataId);
 | |
| 
 | |
|                 if (result.IsFailure())
 | |
|                 {
 | |
|                     if (ResultFs.TargetNotFound.Includes(result))
 | |
|                     {
 | |
|                         // TODO: We're currently always specifying the owner ID because FS doesn't have a way of
 | |
|                         // knowing which process called it
 | |
|                         result = _filesystemClient.CreateSystemSaveData(targetSaveDataId, new TitleId(targetTitleId), 0x10000, 0x10000, SaveDataFlags.KeepAfterResettingSystemSaveDataWithoutUserSaveData);
 | |
|                         if (result.IsFailure()) return result;
 | |
| 
 | |
|                         result = _filesystemClient.MountSystemSaveData(mountName, SaveDataSpaceId.System, targetSaveDataId);
 | |
|                         if (result.IsFailure()) return result;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (result == Result.Success)
 | |
|                 {
 | |
|                     MountCounter++;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
| 
 | |
| 
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         public ResultCode DeleteFile()
 | |
|         {
 | |
|             ResultCode result = (ResultCode)_filesystemClient.DeleteFile(DatabasePath).Value;
 | |
| 
 | |
|             _filesystemClient.Commit(MountName);
 | |
| 
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         public ResultCode LoadFromFile(out bool isBroken)
 | |
|         {
 | |
|             isBroken = false;
 | |
| 
 | |
|             if (MountCounter == 0)
 | |
|             {
 | |
|                 return ResultCode.InvalidArgument;
 | |
|             }
 | |
| 
 | |
|             UpdateCounter++;
 | |
| 
 | |
|             ResetDatabase();
 | |
| 
 | |
|             Result result = _filesystemClient.OpenFile(out FileHandle handle, DatabasePath, OpenMode.Read);
 | |
| 
 | |
|             if (result.IsSuccess())
 | |
|             {
 | |
|                 result = _filesystemClient.GetFileSize(out long fileSize, handle);
 | |
|                 if (result.IsSuccess())
 | |
|                 {
 | |
|                     if (fileSize == Unsafe.SizeOf<NintendoFigurineDatabase>())
 | |
|                     {
 | |
|                         result = _filesystemClient.ReadFile(handle, 0, _database.AsSpan());
 | |
| 
 | |
|                         if (result.IsSuccess())
 | |
|                         {
 | |
|                             if (_database.Verify() != ResultCode.Success)
 | |
|                             {
 | |
|                                 ResetDatabase();
 | |
| 
 | |
|                                 isBroken = true;
 | |
|                             }
 | |
|                             else
 | |
|                             {
 | |
|                                 isBroken = _database.FixDatabase();
 | |
|                             }
 | |
|                         }
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         isBroken = true;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 _filesystemClient.CloseFile(handle);
 | |
| 
 | |
|                 return (ResultCode)result.Value;
 | |
|             }
 | |
|             else if (ResultFs.PathNotFound.Includes(result))
 | |
|             {
 | |
|                 return (ResultCode)ForceSaveDatabase().Value;
 | |
|             }
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         private Result ForceSaveDatabase()
 | |
|         {
 | |
|             Result result = _filesystemClient.CreateFile(DatabasePath, Unsafe.SizeOf<NintendoFigurineDatabase>());
 | |
| 
 | |
|             if (result.IsSuccess() || ResultFs.PathAlreadyExists.Includes(result))
 | |
|             {
 | |
|                 result = _filesystemClient.OpenFile(out FileHandle handle, DatabasePath, OpenMode.Write);
 | |
| 
 | |
|                 if (result.IsSuccess())
 | |
|                 {
 | |
|                     result = _filesystemClient.GetFileSize(out long fileSize, handle);
 | |
| 
 | |
|                     if (result.IsSuccess())
 | |
|                     {
 | |
|                         // If the size doesn't match, recreate the file
 | |
|                         if (fileSize != Unsafe.SizeOf<NintendoFigurineDatabase>())
 | |
|                         {
 | |
|                             _filesystemClient.CloseFile(handle);
 | |
| 
 | |
|                             result = _filesystemClient.DeleteFile(DatabasePath);
 | |
| 
 | |
|                             if (result.IsSuccess())
 | |
|                             {
 | |
|                                 result = _filesystemClient.CreateFile(DatabasePath, Unsafe.SizeOf<NintendoFigurineDatabase>());
 | |
| 
 | |
|                                 if (result.IsSuccess())
 | |
|                                 {
 | |
|                                     result = _filesystemClient.OpenFile(out handle, DatabasePath, OpenMode.Write);
 | |
|                                 }
 | |
|                             }
 | |
| 
 | |
|                             if (result.IsFailure())
 | |
|                             {
 | |
|                                 return result;
 | |
|                             }
 | |
|                         }
 | |
| 
 | |
|                         result = _filesystemClient.WriteFile(handle, 0, _database.AsReadOnlySpan(), WriteOption.Flush);
 | |
|                     }
 | |
| 
 | |
|                     _filesystemClient.CloseFile(handle);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (result.IsSuccess())
 | |
|             {
 | |
|                 _isDirty = false;
 | |
| 
 | |
|                 result = _filesystemClient.Commit(MountName);
 | |
|             }
 | |
| 
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         public DatabaseSessionMetadata CreateSessionMetadata(SpecialMiiKeyCode miiKeyCode)
 | |
|         {
 | |
|             return new DatabaseSessionMetadata(UpdateCounter, miiKeyCode);
 | |
|         }
 | |
| 
 | |
|         public void SetInterfaceVersion(DatabaseSessionMetadata metadata, uint interfaceVersion)
 | |
|         {
 | |
|             metadata.InterfaceVersion = interfaceVersion;
 | |
|         }
 | |
| 
 | |
|         public bool IsUpdated(DatabaseSessionMetadata metadata)
 | |
|         {
 | |
|             bool result = metadata.UpdateCounter != UpdateCounter;
 | |
| 
 | |
|             metadata.UpdateCounter = UpdateCounter;
 | |
| 
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         public int GetCount(DatabaseSessionMetadata metadata)
 | |
|         {
 | |
|             if (!metadata.MiiKeyCode.IsEnabledSpecialMii())
 | |
|             {
 | |
|                 int count = 0;
 | |
| 
 | |
|                 for (int i = 0; i < _database.Length; i++)
 | |
|                 {
 | |
|                     StoreData tmp = _database.Get(i);
 | |
| 
 | |
|                     if (!tmp.IsSpecial())
 | |
|                     {
 | |
|                         count++;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 return count;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return _database.Length;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void Get(DatabaseSessionMetadata metadata, int index, out StoreData storeData)
 | |
|         {
 | |
|             if (!metadata.MiiKeyCode.IsEnabledSpecialMii())
 | |
|             {
 | |
|                 if (GetAtVirtualIndex(index, out int realIndex, out _))
 | |
|                 {
 | |
|                     index = realIndex;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     index = 0;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             storeData = _database.Get(index);
 | |
|         }
 | |
| 
 | |
|         public ResultCode FindIndex(DatabaseSessionMetadata metadata, out int index, CreateId createId)
 | |
|         {
 | |
|             return FindIndex(out index, createId, metadata.MiiKeyCode.IsEnabledSpecialMii());
 | |
|         }
 | |
| 
 | |
|         public ResultCode FindIndex(out int index, CreateId createId, bool isSpecial)
 | |
|         {
 | |
|             if (_database.GetIndexByCreatorId(out int realIndex, createId))
 | |
|             {
 | |
|                 if (isSpecial)
 | |
|                 {
 | |
|                     index = realIndex;
 | |
| 
 | |
|                     return ResultCode.Success;
 | |
|                 }
 | |
| 
 | |
|                 StoreData storeData = _database.Get(realIndex);
 | |
| 
 | |
|                 if (!storeData.IsSpecial())
 | |
|                 {
 | |
|                     if (realIndex < 1)
 | |
|                     {
 | |
|                         index = 0;
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         index = ConvertRealIndexToVirtualIndex(realIndex);
 | |
|                     }
 | |
| 
 | |
|                     return ResultCode.Success;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             index = -1;
 | |
| 
 | |
|             return ResultCode.NotFound;
 | |
|         }
 | |
| 
 | |
|         public ResultCode Move(DatabaseSessionMetadata metadata, int newIndex, CreateId createId)
 | |
|         {
 | |
|             if (!metadata.MiiKeyCode.IsEnabledSpecialMii())
 | |
|             {
 | |
|                 if (GetAtVirtualIndex(newIndex, out int realIndex, out _))
 | |
|                 {
 | |
|                     newIndex = realIndex;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     newIndex = 0;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (_database.GetIndexByCreatorId(out int oldIndex, createId))
 | |
|             {
 | |
|                 StoreData realStoreData = _database.Get(oldIndex);
 | |
| 
 | |
|                 if (!metadata.MiiKeyCode.IsEnabledSpecialMii() && realStoreData.IsSpecial())
 | |
|                 {
 | |
|                     return ResultCode.InvalidOperationOnSpecialMii;
 | |
|                 }
 | |
| 
 | |
|                 ResultCode result = _database.Move(newIndex, oldIndex);
 | |
| 
 | |
|                 if (result == ResultCode.Success)
 | |
|                 {
 | |
|                     MarkDirty(metadata);
 | |
|                 }
 | |
| 
 | |
|                 return result;
 | |
|             }
 | |
| 
 | |
|             return ResultCode.NotFound;
 | |
|         }
 | |
| 
 | |
|         public ResultCode AddOrReplace(DatabaseSessionMetadata metadata, StoreData storeData)
 | |
|         {
 | |
|             if (!storeData.IsValid())
 | |
|             {
 | |
|                 return ResultCode.InvalidStoreData;
 | |
|             }
 | |
| 
 | |
|             if (!metadata.MiiKeyCode.IsEnabledSpecialMii() && !storeData.IsSpecial())
 | |
|             {
 | |
|                 if (_database.GetIndexByCreatorId(out int index, storeData.CreateId))
 | |
|                 {
 | |
|                     StoreData oldStoreData = _database.Get(index);
 | |
| 
 | |
|                     if (oldStoreData.IsSpecial())
 | |
|                     {
 | |
|                         return ResultCode.InvalidOperationOnSpecialMii;
 | |
|                     }
 | |
| 
 | |
|                     _database.Replace(index, storeData);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     if (_database.IsFull())
 | |
|                     {
 | |
|                         return ResultCode.DatabaseFull;
 | |
|                     }
 | |
| 
 | |
|                     _database.Add(storeData);
 | |
|                 }
 | |
| 
 | |
|                 MarkDirty(metadata);
 | |
| 
 | |
|                 return ResultCode.Success;
 | |
|             }
 | |
| 
 | |
|             return ResultCode.InvalidOperationOnSpecialMii;
 | |
|         }
 | |
| 
 | |
|         public ResultCode Delete(DatabaseSessionMetadata metadata, CreateId createId)
 | |
|         {
 | |
|             if (!_database.GetIndexByCreatorId(out int index, createId))
 | |
|             {
 | |
|                 return ResultCode.NotFound;
 | |
|             }
 | |
| 
 | |
|             if (!metadata.MiiKeyCode.IsEnabledSpecialMii())
 | |
|             {
 | |
|                 StoreData storeData = _database.Get(index);
 | |
| 
 | |
|                 if (storeData.IsSpecial())
 | |
|                 {
 | |
|                     return ResultCode.InvalidOperationOnSpecialMii;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             _database.Delete(index);
 | |
| 
 | |
|             MarkDirty(metadata);
 | |
| 
 | |
|             return ResultCode.Success;
 | |
|         }
 | |
| 
 | |
|         public ResultCode DestroyFile(DatabaseSessionMetadata metadata)
 | |
|         {
 | |
|             _database.CorruptDatabase();
 | |
| 
 | |
|             MarkDirty(metadata);
 | |
| 
 | |
|             ResultCode result = SaveDatabase();
 | |
| 
 | |
|             ResetDatabase();
 | |
| 
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         public ResultCode SaveDatabase()
 | |
|         {
 | |
|             if (_isDirty)
 | |
|             {
 | |
|                 return (ResultCode)ForceSaveDatabase().Value;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 return ResultCode.NotUpdated;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void FormatDatabase(DatabaseSessionMetadata metadata)
 | |
|         {
 | |
|             _database.Format();
 | |
| 
 | |
|             MarkDirty(metadata);
 | |
|         }
 | |
| 
 | |
|         public bool IsFullDatabase()
 | |
|         {
 | |
|             return _database.IsFull();
 | |
|         }
 | |
|     }
 | |
| }
 |