2021-03-26 00:16:08 +00:00
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
} ;
2022-12-24 17:30:39 +00:00
// 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.
2023-02-08 13:54:58 +00:00
string hash = Convert . ToHexString ( SHA256 . HashData ( BitConverter . GetBytes ( titleId ) ) ) . Remove ( 0x20 ) ;
2022-12-24 17:30:39 +00:00
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 ) ;
2021-03-26 00:16:08 +00:00
2022-12-24 17:30:39 +00:00
// TODO: Handle that using the FS service implementation and return the right error code instead of throwing exceptions.
Directory . CreateDirectory ( folderPath ) ;
2021-03-26 00:16:08 +00:00
2022-12-24 17:30:39 +00:00
while ( File . Exists ( filePath ) )
{
applicationAlbumEntry . AlbumFileDateTime . UniqueId + + ;
2021-03-26 00:16:08 +00:00
2022-12-24 17:30:39 +00:00
filePath = GenerateFilePath ( folderPath , applicationAlbumEntry , currentDateTime , hash ) ;
2021-03-26 00:16:08 +00:00
}
2022-12-24 17:30:39 +00:00
// NOTE: The saved JPEG file doesn't have the limitation in the extra EXIF data.
Image . LoadPixelData < Rgba32 > ( screenshotData , 1280 , 720 ) . SaveAsJpegAsync ( filePath ) ;
2021-03-26 00:16:08 +00:00
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 ) ;
}
}
}