2019-10-27 17:50:50 -07:00
using ARMeilleure.Memory ;
2018-10-17 10:15:50 -07:00
using Ryujinx.Common.Logging ;
2019-09-18 17:45:11 -07:00
using Ryujinx.HLE.HOS.Services.Arp ;
2018-10-07 06:13:46 -07:00
using Ryujinx.HLE.Utilities ;
2018-02-24 20:34:16 -08:00
using System.Collections.Generic ;
2019-09-18 17:45:11 -07:00
namespace Ryujinx.HLE.HOS.Services.Account.Acc
2018-02-24 20:34:16 -08:00
{
2019-07-10 08:59:54 -07:00
[Service("acc:u0")]
2019-09-18 17:45:11 -07:00
class IAccountServiceForApplication : IpcService
2018-02-24 20:34:16 -08:00
{
2019-06-15 15:35:38 -07:00
private bool _userRegistrationRequestPermitted = false ;
private ApplicationLaunchProperty _applicationLaunchProperty ;
2019-09-18 17:45:11 -07:00
public IAccountServiceForApplication ( ServiceCtx context ) { }
2018-02-24 20:34:16 -08:00
2019-07-11 18:13:43 -07:00
[Command(0)]
2018-10-13 16:16:02 -07:00
// GetUserCount() -> i32
2019-07-14 12:04:38 -07:00
public ResultCode GetUserCount ( ServiceCtx context )
2018-04-17 09:41:14 -07:00
{
2019-06-15 15:35:38 -07:00
context . ResponseData . Write ( context . Device . System . State . Account . GetUserCount ( ) ) ;
2018-04-17 09:41:14 -07:00
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2018-04-17 09:41:14 -07:00
}
2018-06-10 17:46:42 -07:00
2019-07-11 18:13:43 -07:00
[Command(1)]
2018-10-13 16:16:02 -07:00
// GetUserExistence(nn::account::Uid) -> bool
2019-07-14 12:04:38 -07:00
public ResultCode GetUserExistence ( ServiceCtx context )
2018-06-09 21:36:07 -07:00
{
2019-06-15 15:35:38 -07:00
UInt128 userId = new UInt128 ( context . RequestData . ReadBytes ( 0x10 ) ) ;
if ( userId . IsNull )
{
2019-07-14 12:04:38 -07:00
return ResultCode . NullArgument ;
2019-06-15 15:35:38 -07:00
}
2018-06-09 21:36:07 -07:00
2019-06-15 15:35:38 -07:00
context . ResponseData . Write ( context . Device . System . State . Account . TryGetUser ( userId , out _ ) ) ;
2018-06-09 21:36:07 -07:00
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2018-06-09 21:36:07 -07:00
}
2018-04-17 09:41:14 -07:00
2019-07-11 18:13:43 -07:00
[Command(2)]
2018-10-13 16:16:02 -07:00
// ListAllUsers() -> array<nn::account::Uid, 0xa>
2019-07-14 12:04:38 -07:00
public ResultCode ListAllUsers ( ServiceCtx context )
2018-06-09 21:36:07 -07:00
{
2019-06-15 15:35:38 -07:00
return WriteUserList ( context , context . Device . System . State . Account . GetAllUsers ( ) ) ;
2018-06-09 21:36:07 -07:00
}
2018-06-10 17:46:42 -07:00
2019-07-11 18:13:43 -07:00
[Command(3)]
2018-10-13 16:16:02 -07:00
// ListOpenUsers() -> array<nn::account::Uid, 0xa>
2019-07-14 12:04:38 -07:00
public ResultCode ListOpenUsers ( ServiceCtx context )
2018-02-24 20:34:16 -08:00
{
2019-06-15 15:35:38 -07:00
return WriteUserList ( context , context . Device . System . State . Account . GetOpenedUsers ( ) ) ;
2018-08-14 15:02:42 -07:00
}
2019-07-14 12:04:38 -07:00
private ResultCode WriteUserList ( ServiceCtx context , IEnumerable < UserProfile > profiles )
2018-08-14 15:02:42 -07:00
{
2019-06-15 15:35:38 -07:00
if ( context . Request . RecvListBuff . Count = = 0 )
{
2019-07-14 12:04:38 -07:00
return ResultCode . InvalidInputBuffer ;
2019-06-15 15:35:38 -07:00
}
2018-12-06 03:16:24 -08:00
long outputPosition = context . Request . RecvListBuff [ 0 ] . Position ;
long outputSize = context . Request . RecvListBuff [ 0 ] . Size ;
2018-08-14 15:02:42 -07:00
2019-10-27 17:50:50 -07:00
MemoryHelper . FillWithZeros ( context . Memory , outputPosition , ( int ) outputSize ) ;
2019-06-15 15:35:38 -07:00
ulong offset = 0 ;
2018-08-14 15:02:42 -07:00
2019-06-15 15:35:38 -07:00
foreach ( UserProfile userProfile in profiles )
2018-08-14 15:02:42 -07:00
{
2019-06-15 15:35:38 -07:00
if ( offset + 0x10 > ( ulong ) outputSize )
2018-08-14 15:02:42 -07:00
{
break ;
}
2019-06-15 15:35:38 -07:00
context . Memory . WriteInt64 ( outputPosition + ( long ) offset , userProfile . UserId . Low ) ;
context . Memory . WriteInt64 ( outputPosition + ( long ) offset + 8 , userProfile . UserId . High ) ;
offset + = 0x10 ;
2018-08-14 15:02:42 -07:00
}
2018-04-16 17:24:42 -07:00
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2018-02-24 20:34:16 -08:00
}
2019-07-11 18:13:43 -07:00
[Command(4)]
2018-10-13 16:16:02 -07:00
// GetLastOpenedUser() -> nn::account::Uid
2019-07-14 12:04:38 -07:00
public ResultCode GetLastOpenedUser ( ServiceCtx context )
2018-06-02 15:46:09 -07:00
{
2019-06-15 15:35:38 -07:00
context . Device . System . State . Account . LastOpenedUser . UserId . Write ( context . ResponseData ) ;
2018-06-02 15:46:09 -07:00
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2018-06-02 15:46:09 -07:00
}
2019-07-11 18:13:43 -07:00
[Command(5)]
2018-10-13 16:16:02 -07:00
// GetProfile(nn::account::Uid) -> object<nn::account::profile::IProfile>
2019-07-14 12:04:38 -07:00
public ResultCode GetProfile ( ServiceCtx context )
2018-02-24 20:34:16 -08:00
{
2019-06-15 15:35:38 -07:00
UInt128 userId = new UInt128 ( context . RequestData . ReadBytes ( 0x10 ) ) ;
2018-08-14 15:02:42 -07:00
2019-06-15 15:35:38 -07:00
if ( ! context . Device . System . State . Account . TryGetUser ( userId , out UserProfile userProfile ) )
2018-08-14 15:02:42 -07:00
{
2019-06-15 15:35:38 -07:00
Logger . PrintWarning ( LogClass . ServiceAcc , $"User 0x{userId} not found!" ) ;
2018-08-14 15:02:42 -07:00
2019-07-14 12:04:38 -07:00
return ResultCode . UserNotFound ;
2018-08-14 15:02:42 -07:00
}
2019-06-15 15:35:38 -07:00
MakeObject ( context , new IProfile ( userProfile ) ) ;
// Doesn't occur in our case.
2019-09-04 09:09:20 -07:00
// return ResultCode.NullObject;
2018-02-24 20:34:16 -08:00
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2018-02-24 20:34:16 -08:00
}
2019-07-11 18:13:43 -07:00
[Command(50)]
2018-10-13 16:16:02 -07:00
// IsUserRegistrationRequestPermitted(u64, pid) -> bool
2019-07-14 12:04:38 -07:00
public ResultCode IsUserRegistrationRequestPermitted ( ServiceCtx context )
2018-10-13 16:16:02 -07:00
{
2019-06-15 15:35:38 -07:00
// The u64 argument seems to be unused by account.
context . ResponseData . Write ( _userRegistrationRequestPermitted ) ;
2018-10-13 16:16:02 -07:00
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2018-10-13 16:16:02 -07:00
}
2019-07-11 18:13:43 -07:00
[Command(51)]
2018-10-13 16:16:02 -07:00
// TrySelectUserWithoutInteraction(bool) -> nn::account::Uid
2019-07-14 12:04:38 -07:00
public ResultCode TrySelectUserWithoutInteraction ( ServiceCtx context )
2018-10-13 16:16:02 -07:00
{
2019-06-15 15:35:38 -07:00
if ( context . Device . System . State . Account . GetUserCount ( ) ! = 1 )
{
// Invalid UserId.
new UInt128 ( 0 , 0 ) . Write ( context . ResponseData ) ;
2018-10-13 16:16:02 -07:00
2019-06-15 15:35:38 -07:00
return 0 ;
}
bool baasCheck = context . RequestData . ReadBoolean ( ) ;
if ( baasCheck )
{
// This checks something related to baas (online), and then return an invalid UserId if the check in baas returns an error code.
// In our case, we can just log it for now.
2018-10-13 16:16:02 -07:00
2019-06-15 15:35:38 -07:00
Logger . PrintStub ( LogClass . ServiceAcc , new { baasCheck } ) ;
}
2018-10-13 16:16:02 -07:00
2019-06-15 15:35:38 -07:00
// As we returned an invalid UserId if there is more than one user earlier, now we can return only the first one.
context . Device . System . State . Account . GetFirst ( ) . UserId . Write ( context . ResponseData ) ;
2018-10-13 16:16:02 -07:00
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2018-10-13 16:16:02 -07:00
}
2019-07-11 18:13:43 -07:00
[Command(100)]
[Command(140)] // 6.0.0+
2018-10-13 16:16:02 -07:00
// InitializeApplicationInfo(u64, pid)
2019-06-15 15:35:38 -07:00
// Both calls (100, 140) use the same submethod, maybe there's something different further along when arp:r is called?
2019-07-14 12:04:38 -07:00
public ResultCode InitializeApplicationInfo ( ServiceCtx context )
2018-02-24 20:34:16 -08:00
{
2019-06-15 15:35:38 -07:00
if ( _applicationLaunchProperty ! = null )
{
2019-07-14 12:04:38 -07:00
return ResultCode . ApplicationLaunchPropertyAlreadyInit ;
2019-06-15 15:35:38 -07:00
}
// The u64 argument seems to be unused by account.
2018-12-06 03:16:24 -08:00
long unknown = context . RequestData . ReadInt64 ( ) ;
2018-10-13 16:16:02 -07:00
2019-06-15 15:35:38 -07:00
// TODO: Account actually calls nn::arp::detail::IReader::GetApplicationLaunchProperty() with the current PID and store the result (ApplicationLaunchProperty) internally.
// For now we can hardcode values, and fix it after GetApplicationLaunchProperty is implemented.
/ *
if ( nn : : arp : : detail : : IReader : : GetApplicationLaunchProperty ( ) = = 0xCC9D ) // InvalidProcessId
{
2019-09-10 02:55:28 -07:00
_applicationLaunchProperty = ApplicationLaunchProperty . Default ;
2019-06-15 15:35:38 -07:00
2019-07-14 12:04:38 -07:00
return ResultCode . InvalidArgument ;
2019-06-15 15:35:38 -07:00
}
else
* /
{
2019-09-10 02:55:28 -07:00
_applicationLaunchProperty = ApplicationLaunchProperty . GetByPid ( context ) ;
2019-06-15 15:35:38 -07:00
}
2019-01-10 16:11:46 -08:00
Logger . PrintStub ( LogClass . ServiceAcc , new { unknown } ) ;
2018-04-16 17:24:42 -07:00
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2018-02-24 20:34:16 -08:00
}
2019-07-11 18:13:43 -07:00
[Command(101)]
2019-06-15 15:35:38 -07:00
// GetBaasAccountManagerForApplication(nn::account::Uid) -> object<nn::account::baas::IManagerForApplication>
2019-07-14 12:04:38 -07:00
public ResultCode GetBaasAccountManagerForApplication ( ServiceCtx context )
2018-02-24 20:34:16 -08:00
{
2019-06-15 15:35:38 -07:00
UInt128 userId = new UInt128 ( context . RequestData . ReadBytes ( 0x10 ) ) ;
if ( userId . IsNull )
{
2019-07-14 12:04:38 -07:00
return ResultCode . NullArgument ;
2019-06-15 15:35:38 -07:00
}
if ( _applicationLaunchProperty = = null )
{
2019-07-14 12:04:38 -07:00
return ResultCode . InvalidArgument ;
2019-06-15 15:35:38 -07:00
}
MakeObject ( context , new IManagerForApplication ( userId , _applicationLaunchProperty ) ) ;
// Doesn't occur in our case.
2019-07-14 12:04:38 -07:00
// return ResultCode.NullObject;
2019-06-15 15:35:38 -07:00
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2019-06-15 15:35:38 -07:00
}
2019-07-11 18:13:43 -07:00
[Command(110)]
2019-06-15 15:35:38 -07:00
// StoreSaveDataThumbnail(nn::account::Uid, buffer<bytes, 5>)
2019-07-14 12:04:38 -07:00
public ResultCode StoreSaveDataThumbnail ( ServiceCtx context )
2019-06-15 15:35:38 -07:00
{
if ( _applicationLaunchProperty = = null )
{
2019-07-14 12:04:38 -07:00
return ResultCode . InvalidArgument ;
2019-06-15 15:35:38 -07:00
}
UInt128 userId = new UInt128 ( context . RequestData . ReadBytes ( 0x10 ) ) ;
if ( userId . IsNull )
{
2019-07-14 12:04:38 -07:00
return ResultCode . NullArgument ;
2019-06-15 15:35:38 -07:00
}
if ( context . Request . SendBuff . Count = = 0 )
{
2019-07-14 12:04:38 -07:00
return ResultCode . InvalidInputBuffer ;
2019-06-15 15:35:38 -07:00
}
long inputPosition = context . Request . SendBuff [ 0 ] . Position ;
long inputSize = context . Request . SendBuff [ 0 ] . Size ;
if ( inputSize ! = 0x24000 )
{
2019-07-14 12:04:38 -07:00
return ResultCode . InvalidInputBufferSize ;
2019-06-15 15:35:38 -07:00
}
byte [ ] thumbnailBuffer = context . Memory . ReadBytes ( inputPosition , inputSize ) ;
// TODO: Store thumbnailBuffer somewhere, in save data 0x8000000000000010 ?
Logger . PrintStub ( LogClass . ServiceAcc ) ;
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2019-06-15 15:35:38 -07:00
}
2019-07-11 18:13:43 -07:00
[Command(111)]
2019-06-15 15:35:38 -07:00
// ClearSaveDataThumbnail(nn::account::Uid)
2019-07-14 12:04:38 -07:00
public ResultCode ClearSaveDataThumbnail ( ServiceCtx context )
2019-06-15 15:35:38 -07:00
{
if ( _applicationLaunchProperty = = null )
{
2019-07-14 12:04:38 -07:00
return ResultCode . InvalidArgument ;
2019-06-15 15:35:38 -07:00
}
UInt128 userId = new UInt128 ( context . RequestData . ReadBytes ( 0x10 ) ) ;
if ( userId . IsNull )
{
2019-07-14 12:04:38 -07:00
return ResultCode . NullArgument ;
2019-06-15 15:35:38 -07:00
}
// TODO: Clear the Thumbnail somewhere, in save data 0x8000000000000010 ?
Logger . PrintStub ( LogClass . ServiceAcc ) ;
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2019-06-15 15:35:38 -07:00
}
2019-07-11 18:13:43 -07:00
[Command(150)] // 6.0.0+
2019-06-15 15:35:38 -07:00
// IsUserAccountSwitchLocked() -> bool
2019-07-14 12:04:38 -07:00
public ResultCode IsUserAccountSwitchLocked ( ServiceCtx context )
2019-06-15 15:35:38 -07:00
{
// TODO : Validate the following check.
/ *
if ( _applicationLaunchProperty ! = null )
{
2019-07-14 12:04:38 -07:00
return ResultCode . ApplicationLaunchPropertyAlreadyInit ;
2019-06-15 15:35:38 -07:00
}
* /
// Account actually calls nn::arp::detail::IReader::GetApplicationControlProperty() with the current PID and store the result (NACP File) internally.
// But since we use LibHac and we load one Application at a time, it's not necessary.
2019-07-10 10:20:01 -07:00
context . ResponseData . Write ( context . Device . System . ControlData . UserAccountSwitchLock ) ;
2018-10-13 16:16:02 -07:00
2019-06-15 15:35:38 -07:00
Logger . PrintStub ( LogClass . ServiceAcc ) ;
2018-02-24 20:34:16 -08:00
2019-07-14 12:04:38 -07:00
return ResultCode . Success ;
2018-02-24 20:34:16 -08:00
}
}
2018-04-17 09:41:14 -07:00
}