Add BCAT delivery cache support (#1154)
* Initial bcat delivery cache support * Use LibHac 0.11.0 * Add option to open the BCAT savedata directory
This commit is contained in:
		
							parent
							
								
									23170da5a0
								
							
						
					
					
						commit
						7ab3fccd4d
					
				
					 9 changed files with 329 additions and 44 deletions
				
			
		|  | @ -0,0 +1,63 @@ | |||
| using LibHac; | ||||
| using LibHac.Bcat; | ||||
| using Ryujinx.Common; | ||||
| using System; | ||||
| using System.Runtime.InteropServices; | ||||
| 
 | ||||
| namespace Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator | ||||
| { | ||||
|     class IDeliveryCacheDirectoryService : IpcService, IDisposable | ||||
|     { | ||||
|         private LibHac.Bcat.Detail.Ipc.IDeliveryCacheDirectoryService _base; | ||||
| 
 | ||||
|         public IDeliveryCacheDirectoryService(LibHac.Bcat.Detail.Ipc.IDeliveryCacheDirectoryService baseService) | ||||
|         { | ||||
|             _base = baseService; | ||||
|         } | ||||
| 
 | ||||
|         [Command(0)] | ||||
|         // Open(nn::bcat::DirectoryName) | ||||
|         public ResultCode Open(ServiceCtx context) | ||||
|         { | ||||
|             DirectoryName directoryName = context.RequestData.ReadStruct<DirectoryName>(); | ||||
| 
 | ||||
|             Result result = _base.Open(ref directoryName); | ||||
| 
 | ||||
|             return (ResultCode)result.Value; | ||||
|         } | ||||
| 
 | ||||
|         [Command(1)] | ||||
|         // Read() -> (u32, buffer<nn::bcat::DeliveryCacheDirectoryEntry, 6>) | ||||
|         public ResultCode Read(ServiceCtx context) | ||||
|         { | ||||
|             long position = context.Request.ReceiveBuff[0].Position; | ||||
|             long size = context.Request.ReceiveBuff[0].Size; | ||||
| 
 | ||||
|             byte[] data = new byte[size]; | ||||
| 
 | ||||
|             Result result = _base.Read(out int entriesRead, MemoryMarshal.Cast<byte, DeliveryCacheDirectoryEntry>(data)); | ||||
| 
 | ||||
|             context.Memory.WriteBytes(position, data); | ||||
| 
 | ||||
|             context.ResponseData.Write(entriesRead); | ||||
| 
 | ||||
|             return (ResultCode)result.Value; | ||||
|         } | ||||
| 
 | ||||
|         [Command(2)] | ||||
|         // GetCount() -> u32 | ||||
|         public ResultCode GetCount(ServiceCtx context) | ||||
|         { | ||||
|             Result result = _base.GetCount(out int count); | ||||
| 
 | ||||
|             context.ResponseData.Write(count); | ||||
| 
 | ||||
|             return (ResultCode)result.Value; | ||||
|         } | ||||
| 
 | ||||
|         public void Dispose() | ||||
|         { | ||||
|             _base?.Dispose(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | @ -0,0 +1,76 @@ | |||
| using LibHac; | ||||
| using LibHac.Bcat; | ||||
| using Ryujinx.Common; | ||||
| using System; | ||||
| 
 | ||||
| namespace Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator | ||||
| { | ||||
|     class IDeliveryCacheFileService : IpcService, IDisposable | ||||
|     { | ||||
|         private LibHac.Bcat.Detail.Ipc.IDeliveryCacheFileService _base; | ||||
| 
 | ||||
|         public IDeliveryCacheFileService(LibHac.Bcat.Detail.Ipc.IDeliveryCacheFileService baseService) | ||||
|         { | ||||
|             _base = baseService; | ||||
|         } | ||||
| 
 | ||||
|         [Command(0)] | ||||
|         // Open(nn::bcat::DirectoryName, nn::bcat::FileName) | ||||
|         public ResultCode Open(ServiceCtx context) | ||||
|         { | ||||
|             DirectoryName directoryName = context.RequestData.ReadStruct<DirectoryName>(); | ||||
|             FileName fileName = context.RequestData.ReadStruct<FileName>(); | ||||
| 
 | ||||
|             Result result = _base.Open(ref directoryName, ref fileName); | ||||
| 
 | ||||
|             return (ResultCode)result.Value; | ||||
|         } | ||||
| 
 | ||||
|         [Command(1)] | ||||
|         // Read(u64) -> (u64, buffer<bytes, 6>) | ||||
|         public ResultCode Read(ServiceCtx context) | ||||
|         { | ||||
|             long position = context.Request.ReceiveBuff[0].Position; | ||||
|             long size = context.Request.ReceiveBuff[0].Size; | ||||
| 
 | ||||
|             long offset = context.RequestData.ReadInt64(); | ||||
| 
 | ||||
|             byte[] data = new byte[size]; | ||||
| 
 | ||||
|             Result result = _base.Read(out long bytesRead, offset, data); | ||||
| 
 | ||||
|             context.Memory.WriteBytes(position, data); | ||||
| 
 | ||||
|             context.ResponseData.Write(bytesRead); | ||||
| 
 | ||||
|             return (ResultCode)result.Value; | ||||
|         } | ||||
| 
 | ||||
|         [Command(2)] | ||||
|         // GetSize() -> u64 | ||||
|         public ResultCode GetSize(ServiceCtx context) | ||||
|         { | ||||
|             Result result = _base.GetSize(out long size); | ||||
| 
 | ||||
|             context.ResponseData.Write(size); | ||||
| 
 | ||||
|             return (ResultCode)result.Value; | ||||
|         } | ||||
| 
 | ||||
|         [Command(3)] | ||||
|         // GetDigest() -> nn::bcat::Digest | ||||
|         public ResultCode GetDigest(ServiceCtx context) | ||||
|         { | ||||
|             Result result = _base.GetDigest(out Digest digest); | ||||
| 
 | ||||
|             context.ResponseData.WriteStruct(digest); | ||||
| 
 | ||||
|             return (ResultCode)result.Value; | ||||
|         } | ||||
| 
 | ||||
|         public void Dispose() | ||||
|         { | ||||
|             _base?.Dispose(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | @ -1,47 +1,68 @@ | |||
| using Ryujinx.HLE.HOS.Services.Arp; | ||||
| using LibHac; | ||||
| using LibHac.Bcat; | ||||
| using System; | ||||
| using System.Text; | ||||
| using System.Runtime.InteropServices; | ||||
| 
 | ||||
| namespace Ryujinx.HLE.HOS.Services.Bcat.ServiceCreator | ||||
| { | ||||
|     class IDeliveryCacheStorageService : IpcService | ||||
|     class IDeliveryCacheStorageService : IpcService, IDisposable | ||||
|     { | ||||
|         private const int DeliveryCacheDirectoriesLimit    = 100; | ||||
|         private const int DeliveryCacheDirectoryNameLength = 32; | ||||
|         private LibHac.Bcat.Detail.Ipc.IDeliveryCacheStorageService _base; | ||||
| 
 | ||||
|         private string[] _deliveryCacheDirectories = new string[0]; | ||||
| 
 | ||||
|         public IDeliveryCacheStorageService(ServiceCtx context, ApplicationLaunchProperty applicationLaunchProperty) | ||||
|         public IDeliveryCacheStorageService(ServiceCtx context, LibHac.Bcat.Detail.Ipc.IDeliveryCacheStorageService baseService) | ||||
|         { | ||||
|             // TODO: Read directories.meta file from the save data (loaded in IServiceCreator) in _deliveryCacheDirectories. | ||||
|             _base = baseService; | ||||
|         } | ||||
| 
 | ||||
|         [Command(0)] | ||||
|         // CreateFileService() -> object<nn::bcat::detail::ipc::IDeliveryCacheFileService> | ||||
|         public ResultCode CreateFileService(ServiceCtx context) | ||||
|         { | ||||
|             Result result = _base.CreateFileService(out LibHac.Bcat.Detail.Ipc.IDeliveryCacheFileService service); | ||||
| 
 | ||||
|             if (result.IsSuccess()) | ||||
|             { | ||||
|                 MakeObject(context, new IDeliveryCacheFileService(service)); | ||||
|             } | ||||
| 
 | ||||
|             return (ResultCode)result.Value; | ||||
|         } | ||||
| 
 | ||||
|         [Command(1)] | ||||
|         // CreateDirectoryService() -> object<nn::bcat::detail::ipc::IDeliveryCacheDirectoryService> | ||||
|         public ResultCode CreateDirectoryService(ServiceCtx context) | ||||
|         { | ||||
|             Result result = _base.CreateDirectoryService(out LibHac.Bcat.Detail.Ipc.IDeliveryCacheDirectoryService service); | ||||
| 
 | ||||
|             if (result.IsSuccess()) | ||||
|             { | ||||
|                 MakeObject(context, new IDeliveryCacheDirectoryService(service)); | ||||
|             } | ||||
| 
 | ||||
|             return (ResultCode)result.Value; | ||||
|         } | ||||
| 
 | ||||
|         [Command(10)] | ||||
|         // EnumerateDeliveryCacheDirectory() -> (u32, buffer<nn::bcat::DirectoryName, 6>) | ||||
|         public ResultCode EnumerateDeliveryCacheDirectory(ServiceCtx context) | ||||
|         { | ||||
|             long outputPosition = context.Request.ReceiveBuff[0].Position; | ||||
|             long outputSize     = context.Request.ReceiveBuff[0].Size; | ||||
|             long position = context.Request.ReceiveBuff[0].Position; | ||||
|             long size = context.Request.ReceiveBuff[0].Size; | ||||
| 
 | ||||
|             for (int index = 0; index < _deliveryCacheDirectories.Length; index++) | ||||
|             { | ||||
|                 if (index == DeliveryCacheDirectoriesLimit - 1) | ||||
|                 { | ||||
|                     break; | ||||
|                 } | ||||
|             byte[] data = new byte[size]; | ||||
| 
 | ||||
|                 byte[] directoryNameBuffer = Encoding.ASCII.GetBytes(_deliveryCacheDirectories[index]); | ||||
|             Result result = _base.EnumerateDeliveryCacheDirectory(out int count, MemoryMarshal.Cast<byte, DirectoryName>(data)); | ||||
| 
 | ||||
|                 Array.Resize(ref directoryNameBuffer, DeliveryCacheDirectoryNameLength); | ||||
|             context.Memory.WriteBytes(position, data); | ||||
| 
 | ||||
|                 directoryNameBuffer[DeliveryCacheDirectoryNameLength - 1] = 0x00; | ||||
|                  | ||||
|                 context.Memory.WriteBytes(outputPosition + index * DeliveryCacheDirectoryNameLength, directoryNameBuffer); | ||||
|             } | ||||
|             context.ResponseData.Write(count); | ||||
| 
 | ||||
|             context.ResponseData.Write(_deliveryCacheDirectories.Length); | ||||
|             return (ResultCode)result.Value; | ||||
|         } | ||||
| 
 | ||||
|             return ResultCode.Success; | ||||
|         public void Dispose() | ||||
|         { | ||||
|             _base?.Dispose(); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| } | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Alex Barney
						Alex Barney