aboutsummaryrefslogtreecommitdiff
path: root/src/Ryujinx.HLE/HOS/Services/Caps/CaptureManager.cs
diff options
context:
space:
mode:
Diffstat (limited to 'src/Ryujinx.HLE/HOS/Services/Caps/CaptureManager.cs')
-rw-r--r--src/Ryujinx.HLE/HOS/Services/Caps/CaptureManager.cs134
1 files changed, 134 insertions, 0 deletions
diff --git a/src/Ryujinx.HLE/HOS/Services/Caps/CaptureManager.cs b/src/Ryujinx.HLE/HOS/Services/Caps/CaptureManager.cs
new file mode 100644
index 00000000..6320fe28
--- /dev/null
+++ b/src/Ryujinx.HLE/HOS/Services/Caps/CaptureManager.cs
@@ -0,0 +1,134 @@
+using Ryujinx.Common.Memory;
+using Ryujinx.HLE.HOS.Services.Caps.Types;
+using SixLabors.ImageSharp;
+using SixLabors.ImageSharp.PixelFormats;
+using System;
+using System.IO;
+using System.Runtime.CompilerServices;
+using System.Security.Cryptography;
+
+namespace Ryujinx.HLE.HOS.Services.Caps
+{
+ class CaptureManager
+ {
+ private string _sdCardPath;
+
+ private uint _shimLibraryVersion;
+
+ public CaptureManager(Switch device)
+ {
+ _sdCardPath = device.FileSystem.GetSdCardPath();
+ }
+
+ public ResultCode SetShimLibraryVersion(ServiceCtx context)
+ {
+ ulong shimLibraryVersion = context.RequestData.ReadUInt64();
+ ulong appletResourceUserId = context.RequestData.ReadUInt64();
+
+ // TODO: Service checks if the pid is present in an internal list and returns ResultCode.BlacklistedPid if it is.
+ // The list contents needs to be determined.
+
+ ResultCode resultCode = ResultCode.OutOfRange;
+
+ if (shimLibraryVersion != 0)
+ {
+ if (_shimLibraryVersion == shimLibraryVersion)
+ {
+ resultCode = ResultCode.Success;
+ }
+ else if (_shimLibraryVersion != 0)
+ {
+ resultCode = ResultCode.ShimLibraryVersionAlreadySet;
+ }
+ else if (shimLibraryVersion == 1)
+ {
+ resultCode = ResultCode.Success;
+
+ _shimLibraryVersion = 1;
+ }
+ }
+
+ return resultCode;
+ }
+
+ public ResultCode SaveScreenShot(byte[] screenshotData, ulong appletResourceUserId, ulong titleId, out ApplicationAlbumEntry applicationAlbumEntry)
+ {
+ applicationAlbumEntry = default;
+
+ if (screenshotData.Length == 0)
+ {
+ return ResultCode.NullInputBuffer;
+ }
+
+ /*
+ // NOTE: On our current implementation, appletResourceUserId starts at 0, disable it for now.
+ if (appletResourceUserId == 0)
+ {
+ return ResultCode.InvalidArgument;
+ }
+ */
+
+ /*
+ // Doesn't occur in our case.
+ if (applicationAlbumEntry == null)
+ {
+ return ResultCode.NullOutputBuffer;
+ }
+ */
+
+ if (screenshotData.Length >= 0x384000)
+ {
+ DateTime currentDateTime = DateTime.Now;
+
+ applicationAlbumEntry = new ApplicationAlbumEntry()
+ {
+ Size = (ulong)Unsafe.SizeOf<ApplicationAlbumEntry>(),
+ TitleId = titleId,
+ AlbumFileDateTime = new AlbumFileDateTime()
+ {
+ Year = (ushort)currentDateTime.Year,
+ Month = (byte)currentDateTime.Month,
+ Day = (byte)currentDateTime.Day,
+ Hour = (byte)currentDateTime.Hour,
+ Minute = (byte)currentDateTime.Minute,
+ Second = (byte)currentDateTime.Second,
+ UniqueId = 0
+ },
+ AlbumStorage = AlbumStorage.Sd,
+ ContentType = ContentType.Screenshot,
+ Padding = new Array5<byte>(),
+ Unknown0x1f = 1
+ };
+
+ // NOTE: The hex hash is a HMAC-SHA256 (first 32 bytes) using a hardcoded secret key over the titleId, we can simulate it by hashing the titleId instead.
+ string hash = Convert.ToHexString(SHA256.HashData(BitConverter.GetBytes(titleId))).Remove(0x20);
+ string folderPath = Path.Combine(_sdCardPath, "Nintendo", "Album", currentDateTime.Year.ToString("00"), currentDateTime.Month.ToString("00"), currentDateTime.Day.ToString("00"));
+ string filePath = GenerateFilePath(folderPath, applicationAlbumEntry, currentDateTime, hash);
+
+ // TODO: Handle that using the FS service implementation and return the right error code instead of throwing exceptions.
+ Directory.CreateDirectory(folderPath);
+
+ while (File.Exists(filePath))
+ {
+ applicationAlbumEntry.AlbumFileDateTime.UniqueId++;
+
+ filePath = GenerateFilePath(folderPath, applicationAlbumEntry, currentDateTime, hash);
+ }
+
+ // NOTE: The saved JPEG file doesn't have the limitation in the extra EXIF data.
+ Image.LoadPixelData<Rgba32>(screenshotData, 1280, 720).SaveAsJpegAsync(filePath);
+
+ return ResultCode.Success;
+ }
+
+ return ResultCode.NullInputBuffer;
+ }
+
+ private string GenerateFilePath(string folderPath, ApplicationAlbumEntry applicationAlbumEntry, DateTime currentDateTime, string hash)
+ {
+ string fileName = $"{currentDateTime:yyyyMMddHHmmss}{applicationAlbumEntry.AlbumFileDateTime.UniqueId:00}-{hash}.jpg";
+
+ return Path.Combine(folderPath, fileName);
+ }
+ }
+} \ No newline at end of file