2020-09-20 03:45:46 +00:00
|
|
|
using LibHac.Ns;
|
2019-07-04 15:14:17 +00:00
|
|
|
using Ryujinx.Common;
|
2018-10-17 17:15:50 +00:00
|
|
|
using Ryujinx.Common.Logging;
|
2020-08-02 00:20:44 +00:00
|
|
|
using Ryujinx.Common.Memory;
|
|
|
|
using Ryujinx.Common.Utilities;
|
2020-09-20 03:45:46 +00:00
|
|
|
using Ryujinx.HLE.HOS.Ipc;
|
|
|
|
using Ryujinx.HLE.HOS.Kernel.Threading;
|
2019-09-19 00:45:11 +00:00
|
|
|
using Ryujinx.HLE.HOS.Services.Account.Acc;
|
|
|
|
using Ryujinx.HLE.HOS.Services.Friend.ServiceCreator.FriendService;
|
2023-01-04 22:15:45 +00:00
|
|
|
using Ryujinx.Horizon.Common;
|
2020-08-02 00:20:44 +00:00
|
|
|
using System;
|
2019-07-04 15:14:17 +00:00
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
|
2019-09-19 00:45:11 +00:00
|
|
|
namespace Ryujinx.HLE.HOS.Services.Friend.ServiceCreator
|
2018-02-10 00:14:55 +00:00
|
|
|
{
|
2018-03-19 18:58:46 +00:00
|
|
|
class IFriendService : IpcService
|
2018-02-10 00:14:55 +00:00
|
|
|
{
|
2019-07-04 15:14:17 +00:00
|
|
|
private FriendServicePermissionLevel _permissionLevel;
|
2020-09-20 03:45:46 +00:00
|
|
|
private KEvent _completionEvent;
|
2019-07-04 15:14:17 +00:00
|
|
|
|
|
|
|
public IFriendService(FriendServicePermissionLevel permissionLevel)
|
2018-02-10 00:14:55 +00:00
|
|
|
{
|
2019-07-04 15:14:17 +00:00
|
|
|
_permissionLevel = permissionLevel;
|
2018-02-10 00:14:55 +00:00
|
|
|
}
|
2018-08-14 22:02:42 +00:00
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(0)]
|
2020-09-20 03:45:46 +00:00
|
|
|
// GetCompletionEvent() -> handle<copy>
|
|
|
|
public ResultCode GetCompletionEvent(ServiceCtx context)
|
|
|
|
{
|
|
|
|
if (_completionEvent == null)
|
|
|
|
{
|
|
|
|
_completionEvent = new KEvent(context.Device.System.KernelContext);
|
|
|
|
}
|
|
|
|
|
2023-01-04 22:15:45 +00:00
|
|
|
if (context.Process.HandleTable.GenerateHandle(_completionEvent.ReadableEvent, out int completionEventHandle) != Result.Success)
|
2020-09-20 03:45:46 +00:00
|
|
|
{
|
|
|
|
throw new InvalidOperationException("Out of handles!");
|
|
|
|
}
|
|
|
|
|
|
|
|
context.Response.HandleDesc = IpcHandleDesc.MakeCopy(completionEventHandle);
|
|
|
|
|
|
|
|
return ResultCode.Success;
|
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(1)]
|
2022-11-23 15:25:49 +00:00
|
|
|
// nn::friends::Cancel()
|
|
|
|
public ResultCode Cancel(ServiceCtx context)
|
|
|
|
{
|
|
|
|
// TODO: Original service sets an internal field to 1 here. Determine usage.
|
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceFriend);
|
|
|
|
|
|
|
|
return ResultCode.Success;
|
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(10100)]
|
2020-02-02 03:24:17 +00:00
|
|
|
// nn::friends::GetFriendListIds(int offset, nn::account::Uid userId, nn::friends::detail::ipc::SizedFriendFilter friendFilter, ulong pidPlaceHolder, pid)
|
2019-07-12 01:13:43 +00:00
|
|
|
// -> int outCount, array<nn::account::NetworkServiceAccountId, 0xa>
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode GetFriendListIds(ServiceCtx context)
|
2018-09-25 22:59:29 +00:00
|
|
|
{
|
2019-07-04 15:14:17 +00:00
|
|
|
int offset = context.RequestData.ReadInt32();
|
|
|
|
|
|
|
|
// Padding
|
|
|
|
context.RequestData.ReadInt32();
|
|
|
|
|
2020-02-02 03:24:17 +00:00
|
|
|
UserId userId = context.RequestData.ReadStruct<UserId>();
|
2019-07-04 15:14:17 +00:00
|
|
|
FriendFilter filter = context.RequestData.ReadStruct<FriendFilter>();
|
2018-09-25 22:59:29 +00:00
|
|
|
|
2019-07-04 15:14:17 +00:00
|
|
|
// Pid placeholder
|
|
|
|
context.RequestData.ReadInt64();
|
2018-09-25 22:59:29 +00:00
|
|
|
|
2020-02-02 03:24:17 +00:00
|
|
|
if (userId.IsNull)
|
2018-09-25 22:59:29 +00:00
|
|
|
{
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.InvalidArgument;
|
2019-07-04 15:14:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// There are no friends online, so we return 0 because the nn::account::NetworkServiceAccountId array is empty.
|
|
|
|
context.ResponseData.Write(0);
|
2018-09-25 22:59:29 +00:00
|
|
|
|
2020-08-03 23:32:53 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceFriend, new
|
2019-07-04 15:14:17 +00:00
|
|
|
{
|
2020-02-02 03:24:17 +00:00
|
|
|
UserId = userId.ToString(),
|
2019-07-04 15:14:17 +00:00
|
|
|
offset,
|
|
|
|
filter.PresenceStatus,
|
|
|
|
filter.IsFavoriteOnly,
|
|
|
|
filter.IsSameAppPresenceOnly,
|
|
|
|
filter.IsSameAppPlayedOnly,
|
|
|
|
filter.IsArbitraryAppPlayedOnly,
|
|
|
|
filter.PresenceGroupId,
|
|
|
|
});
|
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2019-07-04 15:14:17 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(10101)]
|
2020-02-02 03:24:17 +00:00
|
|
|
// nn::friends::GetFriendList(int offset, nn::account::Uid userId, nn::friends::detail::ipc::SizedFriendFilter friendFilter, ulong pidPlaceHolder, pid)
|
2019-07-12 01:13:43 +00:00
|
|
|
// -> int outCount, array<nn::friends::detail::FriendImpl, 0x6>
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode GetFriendList(ServiceCtx context)
|
2019-07-04 15:14:17 +00:00
|
|
|
{
|
|
|
|
int offset = context.RequestData.ReadInt32();
|
|
|
|
|
|
|
|
// Padding
|
|
|
|
context.RequestData.ReadInt32();
|
|
|
|
|
vi: Unify resolutions values and accurate implementation of them. (#2640)
* vi: Unify resolutions values and accurate implementation of them.
To continue what was made in #2618, I've REd `vi` service a bit. Now values and checks related to displays are more accurate.
- `am` GetDefaultDisplayResolution / GetDefaultDisplayResolutionChangeEvent have more informations on what the service does.
- `vi:u/vi:m/vi:s` GetDisplayService are now accurate.
- `IApplicationDisplay` GetRelayService, GetSystemDisplayService, GetManagerDisplayService, GetIndirectDisplayTransactionService, ListDisplays, OpenDisplay, OpenDefaultDisplay, CloseDisplay, GetDisplayResolution are now properly implemented.
- Some other calls are cleaned or have extra checks accordingly to RE.
Additionnaly, `IFriendService` have some wrong aligned things, and `pm:info` service placeholder was missing.
* just use _openedDisplayInfo.Remove()
* use context.Memory.Fill()
* fix some casting
* remove unneeded comment
* cleanup
* uses TryAdd
* displayId > ulong
* GetDisplayResolution > ulong
* UL
2021-09-19 10:57:39 +00:00
|
|
|
UserId userId = context.RequestData.ReadStruct<UserId>();
|
2019-07-04 15:14:17 +00:00
|
|
|
FriendFilter filter = context.RequestData.ReadStruct<FriendFilter>();
|
|
|
|
|
|
|
|
// Pid placeholder
|
|
|
|
context.RequestData.ReadInt64();
|
|
|
|
|
2020-02-02 03:24:17 +00:00
|
|
|
if (userId.IsNull)
|
2019-07-04 15:14:17 +00:00
|
|
|
{
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.InvalidArgument;
|
2019-07-04 15:14:17 +00:00
|
|
|
}
|
2018-09-25 22:59:29 +00:00
|
|
|
|
|
|
|
// There are no friends online, so we return 0 because the nn::account::NetworkServiceAccountId array is empty.
|
2018-12-06 11:16:24 +00:00
|
|
|
context.ResponseData.Write(0);
|
|
|
|
|
2020-08-03 23:32:53 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceFriend, new {
|
2020-02-02 03:24:17 +00:00
|
|
|
UserId = userId.ToString(),
|
2019-07-04 15:14:17 +00:00
|
|
|
offset,
|
2019-01-11 00:11:46 +00:00
|
|
|
filter.PresenceStatus,
|
|
|
|
filter.IsFavoriteOnly,
|
|
|
|
filter.IsSameAppPresenceOnly,
|
|
|
|
filter.IsSameAppPlayedOnly,
|
|
|
|
filter.IsArbitraryAppPlayedOnly,
|
|
|
|
filter.PresenceGroupId,
|
|
|
|
});
|
2018-09-25 22:59:29 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-09-25 22:59:29 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(10120)] // 10.0.0+
|
2021-12-30 10:42:22 +00:00
|
|
|
// nn::friends::IsFriendListCacheAvailable(nn::account::Uid userId) -> bool
|
|
|
|
public ResultCode IsFriendListCacheAvailable(ServiceCtx context)
|
|
|
|
{
|
|
|
|
UserId userId = context.RequestData.ReadStruct<UserId>();
|
|
|
|
|
|
|
|
if (userId.IsNull)
|
|
|
|
{
|
|
|
|
return ResultCode.InvalidArgument;
|
|
|
|
}
|
|
|
|
|
2022-08-11 21:07:37 +00:00
|
|
|
// TODO: Service mount the friends:/ system savedata and try to load friend.cache file, returns true if exists, false otherwise.
|
2021-12-30 10:42:22 +00:00
|
|
|
// NOTE: If no cache is available, guest then calls nn::friends::EnsureFriendListAvailable, we can avoid that by faking the cache check.
|
|
|
|
context.ResponseData.Write(true);
|
|
|
|
|
|
|
|
// TODO: Since we don't support friend features, it's fine to stub it for now.
|
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceFriend, new { UserId = userId.ToString() });
|
|
|
|
|
|
|
|
return ResultCode.Success;
|
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(10121)] // 10.0.0+
|
2021-12-30 10:42:22 +00:00
|
|
|
// nn::friends::EnsureFriendListAvailable(nn::account::Uid userId)
|
|
|
|
public ResultCode EnsureFriendListAvailable(ServiceCtx context)
|
|
|
|
{
|
|
|
|
UserId userId = context.RequestData.ReadStruct<UserId>();
|
|
|
|
|
|
|
|
if (userId.IsNull)
|
|
|
|
{
|
|
|
|
return ResultCode.InvalidArgument;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Service mount the friends:/ system savedata and create a friend.cache file for the given user id.
|
|
|
|
// Since we don't support friend features, it's fine to stub it for now.
|
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceFriend, new { UserId = userId.ToString() });
|
|
|
|
|
|
|
|
return ResultCode.Success;
|
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(10400)]
|
2020-03-23 21:19:45 +00:00
|
|
|
// nn::friends::GetBlockedUserListIds(int offset, nn::account::Uid userId) -> (u32, buffer<nn::account::NetworkServiceAccountId, 0xa>)
|
|
|
|
public ResultCode GetBlockedUserListIds(ServiceCtx context)
|
|
|
|
{
|
|
|
|
int offset = context.RequestData.ReadInt32();
|
|
|
|
|
|
|
|
// Padding
|
|
|
|
context.RequestData.ReadInt32();
|
|
|
|
|
|
|
|
UserId userId = context.RequestData.ReadStruct<UserId>();
|
|
|
|
|
|
|
|
// There are no friends blocked, so we return 0 because the nn::account::NetworkServiceAccountId array is empty.
|
|
|
|
context.ResponseData.Write(0);
|
|
|
|
|
2020-08-03 23:32:53 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceFriend, new { offset, UserId = userId.ToString() });
|
2020-03-23 21:19:45 +00:00
|
|
|
|
|
|
|
return ResultCode.Success;
|
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(10600)]
|
2020-02-02 03:24:17 +00:00
|
|
|
// nn::friends::DeclareOpenOnlinePlaySession(nn::account::Uid userId)
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode DeclareOpenOnlinePlaySession(ServiceCtx context)
|
2019-04-25 13:03:00 +00:00
|
|
|
{
|
2020-02-02 03:24:17 +00:00
|
|
|
UserId userId = context.RequestData.ReadStruct<UserId>();
|
2019-07-04 15:14:17 +00:00
|
|
|
|
2020-02-02 03:24:17 +00:00
|
|
|
if (userId.IsNull)
|
2019-07-04 15:14:17 +00:00
|
|
|
{
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.InvalidArgument;
|
2019-07-04 15:14:17 +00:00
|
|
|
}
|
2019-04-25 13:03:00 +00:00
|
|
|
|
2021-04-23 20:26:31 +00:00
|
|
|
context.Device.System.AccountManager.OpenUserOnlinePlay(userId);
|
2022-08-11 21:07:37 +00:00
|
|
|
|
2021-04-23 20:26:31 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceFriend, new { UserId = userId.ToString() });
|
2019-04-25 13:03:00 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2019-04-25 13:03:00 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(10601)]
|
2020-02-02 03:24:17 +00:00
|
|
|
// nn::friends::DeclareCloseOnlinePlaySession(nn::account::Uid userId)
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode DeclareCloseOnlinePlaySession(ServiceCtx context)
|
2018-08-14 22:02:42 +00:00
|
|
|
{
|
2020-02-02 03:24:17 +00:00
|
|
|
UserId userId = context.RequestData.ReadStruct<UserId>();
|
2019-07-04 15:14:17 +00:00
|
|
|
|
2020-02-02 03:24:17 +00:00
|
|
|
if (userId.IsNull)
|
2019-07-04 15:14:17 +00:00
|
|
|
{
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.InvalidArgument;
|
2019-07-04 15:14:17 +00:00
|
|
|
}
|
2018-08-14 22:02:42 +00:00
|
|
|
|
2021-04-23 20:26:31 +00:00
|
|
|
context.Device.System.AccountManager.CloseUserOnlinePlay(userId);
|
2018-08-14 22:02:42 +00:00
|
|
|
|
2021-04-23 20:26:31 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceFriend, new { UserId = userId.ToString() });
|
2018-09-25 22:59:29 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-08-14 22:02:42 +00:00
|
|
|
}
|
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(10610)]
|
2019-07-04 15:14:17 +00:00
|
|
|
// nn::friends::UpdateUserPresence(nn::account::Uid, u64, pid, buffer<nn::friends::detail::UserPresenceImpl, 0x19>)
|
2019-07-14 19:04:38 +00:00
|
|
|
public ResultCode UpdateUserPresence(ServiceCtx context)
|
2018-08-14 22:02:42 +00:00
|
|
|
{
|
2020-02-02 03:24:17 +00:00
|
|
|
UserId uuid = context.RequestData.ReadStruct<UserId>();
|
2018-08-14 22:02:42 +00:00
|
|
|
|
2019-07-04 15:14:17 +00:00
|
|
|
// Pid placeholder
|
|
|
|
context.RequestData.ReadInt64();
|
2018-09-25 22:59:29 +00:00
|
|
|
|
2021-04-24 10:16:01 +00:00
|
|
|
ulong position = context.Request.PtrBuff[0].Position;
|
|
|
|
ulong size = context.Request.PtrBuff[0].Size;
|
2018-09-25 22:59:29 +00:00
|
|
|
|
2022-11-24 14:26:29 +00:00
|
|
|
ReadOnlySpan<UserPresence> userPresenceInputArray = MemoryMarshal.Cast<byte, UserPresence>(context.Memory.GetSpan(position, (int)size));
|
2018-09-25 22:59:29 +00:00
|
|
|
|
2019-07-04 15:14:17 +00:00
|
|
|
if (uuid.IsNull)
|
|
|
|
{
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.InvalidArgument;
|
2019-07-04 15:14:17 +00:00
|
|
|
}
|
|
|
|
|
2022-11-24 14:26:29 +00:00
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceFriend, new { UserId = uuid.ToString(), userPresenceInputArray = userPresenceInputArray.ToArray() });
|
2018-08-14 22:02:42 +00:00
|
|
|
|
2019-07-14 19:04:38 +00:00
|
|
|
return ResultCode.Success;
|
2018-08-14 22:02:42 +00:00
|
|
|
}
|
2020-08-02 00:20:44 +00:00
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(10700)]
|
2020-08-02 00:20:44 +00:00
|
|
|
// nn::friends::GetPlayHistoryRegistrationKey(b8 unknown, nn::account::Uid) -> buffer<nn::friends::PlayHistoryRegistrationKey, 0x1a>
|
|
|
|
public ResultCode GetPlayHistoryRegistrationKey(ServiceCtx context)
|
|
|
|
{
|
|
|
|
bool unknownBool = context.RequestData.ReadBoolean();
|
|
|
|
UserId userId = context.RequestData.ReadStruct<UserId>();
|
|
|
|
|
2021-04-24 10:16:01 +00:00
|
|
|
context.Response.PtrBuff[0] = context.Response.PtrBuff[0].WithSize(0x40UL);
|
2020-12-12 03:06:20 +00:00
|
|
|
|
vi: Unify resolutions values and accurate implementation of them. (#2640)
* vi: Unify resolutions values and accurate implementation of them.
To continue what was made in #2618, I've REd `vi` service a bit. Now values and checks related to displays are more accurate.
- `am` GetDefaultDisplayResolution / GetDefaultDisplayResolutionChangeEvent have more informations on what the service does.
- `vi:u/vi:m/vi:s` GetDisplayService are now accurate.
- `IApplicationDisplay` GetRelayService, GetSystemDisplayService, GetManagerDisplayService, GetIndirectDisplayTransactionService, ListDisplays, OpenDisplay, OpenDefaultDisplay, CloseDisplay, GetDisplayResolution are now properly implemented.
- Some other calls are cleaned or have extra checks accordingly to RE.
Additionnaly, `IFriendService` have some wrong aligned things, and `pm:info` service placeholder was missing.
* just use _openedDisplayInfo.Remove()
* use context.Memory.Fill()
* fix some casting
* remove unneeded comment
* cleanup
* uses TryAdd
* displayId > ulong
* GetDisplayResolution > ulong
* UL
2021-09-19 10:57:39 +00:00
|
|
|
ulong bufferPosition = context.Request.RecvListBuff[0].Position;
|
2020-08-02 00:20:44 +00:00
|
|
|
|
|
|
|
if (userId.IsNull)
|
|
|
|
{
|
|
|
|
return ResultCode.InvalidArgument;
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: Calls nn::friends::detail::service::core::PlayHistoryManager::GetInstance and stores the instance.
|
|
|
|
|
|
|
|
byte[] randomBytes = new byte[8];
|
|
|
|
|
2022-11-16 18:34:18 +00:00
|
|
|
Random.Shared.NextBytes(randomBytes);
|
2020-08-02 00:20:44 +00:00
|
|
|
|
|
|
|
// NOTE: Calls nn::friends::detail::service::core::UuidManager::GetInstance and stores the instance.
|
|
|
|
// Then call nn::friends::detail::service::core::AccountStorageManager::GetInstance and store the instance.
|
|
|
|
// Then it checks if an Uuid is already stored for the UserId, if not it generates a random Uuid.
|
|
|
|
// And store it in the savedata 8000000000000080 in the friends:/uid.bin file.
|
|
|
|
|
|
|
|
Array16<byte> randomGuid = new Array16<byte>();
|
|
|
|
|
2022-08-11 21:07:37 +00:00
|
|
|
Guid.NewGuid().ToByteArray().AsSpan().CopyTo(randomGuid.AsSpan());
|
2020-08-02 00:20:44 +00:00
|
|
|
|
|
|
|
PlayHistoryRegistrationKey playHistoryRegistrationKey = new PlayHistoryRegistrationKey
|
|
|
|
{
|
|
|
|
Type = 0x101,
|
|
|
|
KeyIndex = (byte)(randomBytes[0] & 7),
|
|
|
|
UserIdBool = 0, // TODO: Find it.
|
|
|
|
UnknownBool = (byte)(unknownBool ? 1 : 0), // TODO: Find it.
|
|
|
|
Reserved = new Array11<byte>(),
|
|
|
|
Uuid = randomGuid
|
|
|
|
};
|
|
|
|
|
|
|
|
ReadOnlySpan<byte> playHistoryRegistrationKeyBuffer = SpanHelpers.AsByteSpan(ref playHistoryRegistrationKey);
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
NOTE: The service uses the KeyIndex to get a random key from a keys buffer (since the key index is stored in the returned buffer).
|
|
|
|
We currently don't support play history and online services so we can use a blank key for now.
|
|
|
|
Code for reference:
|
|
|
|
|
|
|
|
byte[] hmacKey = new byte[0x20];
|
|
|
|
|
|
|
|
HMACSHA256 hmacSha256 = new HMACSHA256(hmacKey);
|
|
|
|
byte[] hmacHash = hmacSha256.ComputeHash(playHistoryRegistrationKeyBuffer);
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2021-04-24 10:16:01 +00:00
|
|
|
context.Memory.Write(bufferPosition, playHistoryRegistrationKeyBuffer);
|
|
|
|
context.Memory.Write(bufferPosition + 0x20, new byte[0x20]); // HmacHash
|
2020-08-02 00:20:44 +00:00
|
|
|
|
|
|
|
return ResultCode.Success;
|
|
|
|
}
|
2020-09-20 03:45:46 +00:00
|
|
|
|
2023-04-14 23:00:34 +00:00
|
|
|
[CommandCmif(10702)]
|
2020-09-20 03:45:46 +00:00
|
|
|
// nn::friends::AddPlayHistory(nn::account::Uid, u64, pid, buffer<nn::friends::PlayHistoryRegistrationKey, 0x19>, buffer<nn::friends::InAppScreenName, 0x19>, buffer<nn::friends::InAppScreenName, 0x19>)
|
|
|
|
public ResultCode AddPlayHistory(ServiceCtx context)
|
|
|
|
{
|
|
|
|
UserId userId = context.RequestData.ReadStruct<UserId>();
|
|
|
|
|
|
|
|
// Pid placeholder
|
|
|
|
context.RequestData.ReadInt64();
|
2022-02-09 20:18:07 +00:00
|
|
|
ulong pid = context.Request.HandleDesc.PId;
|
2020-09-20 03:45:46 +00:00
|
|
|
|
2021-04-24 10:16:01 +00:00
|
|
|
ulong playHistoryRegistrationKeyPosition = context.Request.PtrBuff[0].Position;
|
|
|
|
ulong PlayHistoryRegistrationKeySize = context.Request.PtrBuff[0].Size;
|
2020-09-20 03:45:46 +00:00
|
|
|
|
2021-04-24 10:16:01 +00:00
|
|
|
ulong inAppScreenName1Position = context.Request.PtrBuff[1].Position;
|
|
|
|
ulong inAppScreenName1Size = context.Request.PtrBuff[1].Size;
|
2020-09-20 03:45:46 +00:00
|
|
|
|
2021-04-24 10:16:01 +00:00
|
|
|
ulong inAppScreenName2Position = context.Request.PtrBuff[2].Position;
|
|
|
|
ulong inAppScreenName2Size = context.Request.PtrBuff[2].Size;
|
2020-09-20 03:45:46 +00:00
|
|
|
|
|
|
|
if (userId.IsNull || inAppScreenName1Size > 0x48 || inAppScreenName2Size > 0x48)
|
|
|
|
{
|
|
|
|
return ResultCode.InvalidArgument;
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Call nn::arp::GetApplicationControlProperty here when implemented.
|
2023-03-31 19:16:46 +00:00
|
|
|
ApplicationControlProperty controlProperty = context.Device.Processes.ActiveApplication.ApplicationControlProperties;
|
2020-09-20 03:45:46 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
|
|
|
|
NOTE: The service calls nn::friends::detail::service::core::PlayHistoryManager to store informations using the registration key computed in GetPlayHistoryRegistrationKey.
|
|
|
|
Then calls nn::friends::detail::service::core::FriendListManager to update informations on the friend list.
|
|
|
|
We currently don't support play history and online services so it's fine to do nothing.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
Logger.Stub?.PrintStub(LogClass.ServiceFriend);
|
|
|
|
|
|
|
|
return ResultCode.Success;
|
|
|
|
}
|
2018-02-10 00:14:55 +00:00
|
|
|
}
|
2023-04-14 23:00:34 +00:00
|
|
|
}
|