|
|
@ -2,6 +2,7 @@
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#include <random>
|
|
|
|
#include <boost/optional.hpp>
|
|
|
|
#include <boost/optional.hpp>
|
|
|
|
#include "core/hle/service/acc/profile_manager.h"
|
|
|
|
#include "core/hle/service/acc/profile_manager.h"
|
|
|
|
#include "core/settings.h"
|
|
|
|
#include "core/settings.h"
|
|
|
@ -12,6 +13,15 @@ constexpr ResultCode ERROR_TOO_MANY_USERS(ErrorModule::Account, -1);
|
|
|
|
constexpr ResultCode ERROR_USER_ALREADY_EXISTS(ErrorModule::Account, -2);
|
|
|
|
constexpr ResultCode ERROR_USER_ALREADY_EXISTS(ErrorModule::Account, -2);
|
|
|
|
constexpr ResultCode ERROR_ARGUMENT_IS_NULL(ErrorModule::Account, 20);
|
|
|
|
constexpr ResultCode ERROR_ARGUMENT_IS_NULL(ErrorModule::Account, 20);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const UUID& UUID::Generate() {
|
|
|
|
|
|
|
|
std::random_device device;
|
|
|
|
|
|
|
|
std::mt19937 gen(device());
|
|
|
|
|
|
|
|
std::uniform_int_distribution<u64> distribution(1, std::numeric_limits<u64>::max());
|
|
|
|
|
|
|
|
uuid[0] = distribution(gen);
|
|
|
|
|
|
|
|
uuid[1] = distribution(gen);
|
|
|
|
|
|
|
|
return *this;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ProfileManager::ProfileManager() {
|
|
|
|
ProfileManager::ProfileManager() {
|
|
|
|
// TODO(ogniK): Create the default user we have for now until loading/saving users is added
|
|
|
|
// TODO(ogniK): Create the default user we have for now until loading/saving users is added
|
|
|
|
auto user_uuid = UUID{1, 0};
|
|
|
|
auto user_uuid = UUID{1, 0};
|
|
|
@ -25,7 +35,7 @@ boost::optional<size_t> ProfileManager::AddToProfiles(const ProfileInfo& user) {
|
|
|
|
if (user_count >= MAX_USERS) {
|
|
|
|
if (user_count >= MAX_USERS) {
|
|
|
|
return boost::none;
|
|
|
|
return boost::none;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
profiles[user_count] = std::move(user);
|
|
|
|
profiles[user_count] = user;
|
|
|
|
return user_count++;
|
|
|
|
return user_count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -43,7 +53,7 @@ bool ProfileManager::RemoveProfileAtIndex(size_t index) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Helper function to register a user to the system
|
|
|
|
/// Helper function to register a user to the system
|
|
|
|
ResultCode ProfileManager::AddUser(ProfileInfo user) {
|
|
|
|
ResultCode ProfileManager::AddUser(const ProfileInfo& user) {
|
|
|
|
if (AddToProfiles(user) == boost::none) {
|
|
|
|
if (AddToProfiles(user) == boost::none) {
|
|
|
|
return ERROR_TOO_MANY_USERS;
|
|
|
|
return ERROR_TOO_MANY_USERS;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -52,7 +62,7 @@ ResultCode ProfileManager::AddUser(ProfileInfo user) {
|
|
|
|
|
|
|
|
|
|
|
|
/// Create a new user on the system. If the uuid of the user already exists, the user is not
|
|
|
|
/// Create a new user on the system. If the uuid of the user already exists, the user is not
|
|
|
|
/// created.
|
|
|
|
/// created.
|
|
|
|
ResultCode ProfileManager::CreateNewUser(UUID uuid, std::array<u8, 0x20>& username) {
|
|
|
|
ResultCode ProfileManager::CreateNewUser(UUID uuid, const ProfileUsername& username) {
|
|
|
|
if (user_count == MAX_USERS) {
|
|
|
|
if (user_count == MAX_USERS) {
|
|
|
|
return ERROR_TOO_MANY_USERS;
|
|
|
|
return ERROR_TOO_MANY_USERS;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -67,7 +77,7 @@ ResultCode ProfileManager::CreateNewUser(UUID uuid, std::array<u8, 0x20>& userna
|
|
|
|
return ERROR_USER_ALREADY_EXISTS;
|
|
|
|
return ERROR_USER_ALREADY_EXISTS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ProfileInfo profile;
|
|
|
|
ProfileInfo profile;
|
|
|
|
profile.user_uuid = std::move(uuid);
|
|
|
|
profile.user_uuid = uuid;
|
|
|
|
profile.username = username;
|
|
|
|
profile.username = username;
|
|
|
|
profile.data = {};
|
|
|
|
profile.data = {};
|
|
|
|
profile.creation_time = 0x0;
|
|
|
|
profile.creation_time = 0x0;
|
|
|
@ -79,7 +89,7 @@ ResultCode ProfileManager::CreateNewUser(UUID uuid, std::array<u8, 0x20>& userna
|
|
|
|
/// specifically by allowing an std::string for the username. This is required specifically since
|
|
|
|
/// specifically by allowing an std::string for the username. This is required specifically since
|
|
|
|
/// we're loading a string straight from the config
|
|
|
|
/// we're loading a string straight from the config
|
|
|
|
ResultCode ProfileManager::CreateNewUser(UUID uuid, const std::string& username) {
|
|
|
|
ResultCode ProfileManager::CreateNewUser(UUID uuid, const std::string& username) {
|
|
|
|
std::array<u8, 0x20> username_output;
|
|
|
|
ProfileUsername username_output;
|
|
|
|
if (username.size() > username_output.size()) {
|
|
|
|
if (username.size() > username_output.size()) {
|
|
|
|
std::copy_n(username.begin(), username_output.size(), username_output.begin());
|
|
|
|
std::copy_n(username.begin(), username_output.size(), username_output.begin());
|
|
|
|
} else {
|
|
|
|
} else {
|
|
|
@ -102,7 +112,7 @@ boost::optional<size_t> ProfileManager::GetUserIndex(const UUID& uuid) const {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Returns a users profile index based on their profile
|
|
|
|
/// Returns a users profile index based on their profile
|
|
|
|
boost::optional<size_t> ProfileManager::GetUserIndex(ProfileInfo user) const {
|
|
|
|
boost::optional<size_t> ProfileManager::GetUserIndex(const ProfileInfo& user) const {
|
|
|
|
return GetUserIndex(user.user_uuid);
|
|
|
|
return GetUserIndex(user.user_uuid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -125,7 +135,7 @@ bool ProfileManager::GetProfileBase(UUID uuid, ProfileBase& profile) const {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Returns the data structure used by the switch when GetProfileBase is called on acc:*
|
|
|
|
/// Returns the data structure used by the switch when GetProfileBase is called on acc:*
|
|
|
|
bool ProfileManager::GetProfileBase(ProfileInfo user, ProfileBase& profile) const {
|
|
|
|
bool ProfileManager::GetProfileBase(const ProfileInfo& user, ProfileBase& profile) const {
|
|
|
|
return GetProfileBase(user.user_uuid, profile);
|
|
|
|
return GetProfileBase(user.user_uuid, profile);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -168,8 +178,8 @@ void ProfileManager::CloseUser(UUID uuid) {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Gets all valid user ids on the system
|
|
|
|
/// Gets all valid user ids on the system
|
|
|
|
std::array<UUID, MAX_USERS> ProfileManager::GetAllUsers() const {
|
|
|
|
UserIDArray ProfileManager::GetAllUsers() const {
|
|
|
|
std::array<UUID, MAX_USERS> output;
|
|
|
|
UserIDArray output;
|
|
|
|
std::transform(profiles.begin(), profiles.end(), output.begin(),
|
|
|
|
std::transform(profiles.begin(), profiles.end(), output.begin(),
|
|
|
|
[](const ProfileInfo& p) { return p.user_uuid; });
|
|
|
|
[](const ProfileInfo& p) { return p.user_uuid; });
|
|
|
|
return output;
|
|
|
|
return output;
|
|
|
@ -177,8 +187,8 @@ std::array<UUID, MAX_USERS> ProfileManager::GetAllUsers() const {
|
|
|
|
|
|
|
|
|
|
|
|
/// Get all the open users on the system and zero out the rest of the data. This is specifically
|
|
|
|
/// Get all the open users on the system and zero out the rest of the data. This is specifically
|
|
|
|
/// needed for GetOpenUsers and we need to ensure the rest of the output buffer is zero'd out
|
|
|
|
/// needed for GetOpenUsers and we need to ensure the rest of the output buffer is zero'd out
|
|
|
|
std::array<UUID, MAX_USERS> ProfileManager::GetOpenUsers() const {
|
|
|
|
UserIDArray ProfileManager::GetOpenUsers() const {
|
|
|
|
std::array<UUID, MAX_USERS> output;
|
|
|
|
UserIDArray output;
|
|
|
|
std::transform(profiles.begin(), profiles.end(), output.begin(), [](const ProfileInfo& p) {
|
|
|
|
std::transform(profiles.begin(), profiles.end(), output.begin(), [](const ProfileInfo& p) {
|
|
|
|
if (p.is_open)
|
|
|
|
if (p.is_open)
|
|
|
|
return p.user_uuid;
|
|
|
|
return p.user_uuid;
|
|
|
@ -195,9 +205,9 @@ UUID ProfileManager::GetLastOpenedUser() const {
|
|
|
|
|
|
|
|
|
|
|
|
/// Return the users profile base and the unknown arbitary data.
|
|
|
|
/// Return the users profile base and the unknown arbitary data.
|
|
|
|
bool ProfileManager::GetProfileBaseAndData(boost::optional<size_t> index, ProfileBase& profile,
|
|
|
|
bool ProfileManager::GetProfileBaseAndData(boost::optional<size_t> index, ProfileBase& profile,
|
|
|
|
std::array<u8, MAX_DATA>& data) const {
|
|
|
|
ProfileData& data) const {
|
|
|
|
if (GetProfileBase(index, profile)) {
|
|
|
|
if (GetProfileBase(index, profile)) {
|
|
|
|
std::memcpy(data.data(), profiles[index.get()].data.data(), MAX_DATA);
|
|
|
|
data = profiles[index.get()].data;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
@ -205,14 +215,14 @@ bool ProfileManager::GetProfileBaseAndData(boost::optional<size_t> index, Profil
|
|
|
|
|
|
|
|
|
|
|
|
/// Return the users profile base and the unknown arbitary data.
|
|
|
|
/// Return the users profile base and the unknown arbitary data.
|
|
|
|
bool ProfileManager::GetProfileBaseAndData(UUID uuid, ProfileBase& profile,
|
|
|
|
bool ProfileManager::GetProfileBaseAndData(UUID uuid, ProfileBase& profile,
|
|
|
|
std::array<u8, MAX_DATA>& data) const {
|
|
|
|
ProfileData& data) const {
|
|
|
|
auto idx = GetUserIndex(uuid);
|
|
|
|
auto idx = GetUserIndex(uuid);
|
|
|
|
return GetProfileBaseAndData(idx, profile, data);
|
|
|
|
return GetProfileBaseAndData(idx, profile, data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// Return the users profile base and the unknown arbitary data.
|
|
|
|
/// Return the users profile base and the unknown arbitary data.
|
|
|
|
bool ProfileManager::GetProfileBaseAndData(ProfileInfo user, ProfileBase& profile,
|
|
|
|
bool ProfileManager::GetProfileBaseAndData(const ProfileInfo& user, ProfileBase& profile,
|
|
|
|
std::array<u8, MAX_DATA>& data) const {
|
|
|
|
ProfileData& data) const {
|
|
|
|
return GetProfileBaseAndData(user.user_uuid, profile, data);
|
|
|
|
return GetProfileBaseAndData(user.user_uuid, profile, data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|