mii: Implement IDatabaseService commands using MiiManager
Since the MiiManager was designed around the IPC interface, this is quite easy. Only functions that were clearly defined were implemented.master
parent
daf5b8c61b
commit
e25a7891e9
|
|
@ -310,6 +310,8 @@ add_library(core STATIC
|
||||||
hle/service/mig/mig.h
|
hle/service/mig/mig.h
|
||||||
hle/service/mii/mii.cpp
|
hle/service/mii/mii.cpp
|
||||||
hle/service/mii/mii.h
|
hle/service/mii/mii.h
|
||||||
|
hle/service/mii/mii_manager.cpp
|
||||||
|
hle/service/mii/mii_manager.h
|
||||||
hle/service/mm/mm_u.cpp
|
hle/service/mm/mm_u.cpp
|
||||||
hle/service/mm/mm_u.h
|
hle/service/mm/mm_u.h
|
||||||
hle/service/ncm/ncm.cpp
|
hle/service/ncm/ncm.cpp
|
||||||
|
|
|
||||||
|
|
@ -5,9 +5,11 @@
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
|
||||||
#include "common/logging/log.h"
|
#include "common/logging/log.h"
|
||||||
|
#include "common/string_util.h"
|
||||||
#include "core/hle/ipc_helpers.h"
|
#include "core/hle/ipc_helpers.h"
|
||||||
#include "core/hle/kernel/hle_ipc.h"
|
#include "core/hle/kernel/hle_ipc.h"
|
||||||
#include "core/hle/service/mii/mii.h"
|
#include "core/hle/service/mii/mii.h"
|
||||||
|
#include "core/hle/service/mii/mii_manager.h"
|
||||||
#include "core/hle/service/service.h"
|
#include "core/hle/service/service.h"
|
||||||
#include "core/hle/service/sm/sm.h"
|
#include "core/hle/service/sm/sm.h"
|
||||||
|
|
||||||
|
|
@ -18,28 +20,28 @@ public:
|
||||||
explicit IDatabaseService() : ServiceFramework{"IDatabaseService"} {
|
explicit IDatabaseService() : ServiceFramework{"IDatabaseService"} {
|
||||||
// clang-format off
|
// clang-format off
|
||||||
static const FunctionInfo functions[] = {
|
static const FunctionInfo functions[] = {
|
||||||
{0, nullptr, "IsUpdated"},
|
{0, &IDatabaseService::IsUpdated, "IsUpdated"},
|
||||||
{1, nullptr, "IsFullDatabase"},
|
{1, &IDatabaseService::IsFullDatabase, "IsFullDatabase"},
|
||||||
{2, nullptr, "GetCount"},
|
{2, &IDatabaseService::GetCount, "GetCount"},
|
||||||
{3, nullptr, "Get"},
|
{3, &IDatabaseService::Get, "Get"},
|
||||||
{4, nullptr, "Get1"},
|
{4, &IDatabaseService::Get1, "Get1"},
|
||||||
{5, nullptr, "UpdateLatest"},
|
{5, nullptr, "UpdateLatest"},
|
||||||
{6, nullptr, "BuildRandom"},
|
{6, &IDatabaseService::BuildRandom, "BuildRandom"},
|
||||||
{7, nullptr, "BuildDefault"},
|
{7, &IDatabaseService::BuildDefault, "BuildDefault"},
|
||||||
{8, nullptr, "Get2"},
|
{8, &IDatabaseService::Get2, "Get2"},
|
||||||
{9, nullptr, "Get3"},
|
{9, &IDatabaseService::Get3, "Get3"},
|
||||||
{10, nullptr, "UpdateLatest1"},
|
{10, nullptr, "UpdateLatest1"},
|
||||||
{11, nullptr, "FindIndex"},
|
{11, &IDatabaseService::FindIndex, "FindIndex"},
|
||||||
{12, nullptr, "Move"},
|
{12, &IDatabaseService::Move, "Move"},
|
||||||
{13, nullptr, "AddOrReplace"},
|
{13, &IDatabaseService::AddOrReplace, "AddOrReplace"},
|
||||||
{14, nullptr, "Delete"},
|
{14, &IDatabaseService::Delete, "Delete"},
|
||||||
{15, nullptr, "DestroyFile"},
|
{15, nullptr, "DestroyFile"},
|
||||||
{16, nullptr, "DeleteFile"},
|
{16, nullptr, "DeleteFile"},
|
||||||
{17, nullptr, "Format"},
|
{17, &IDatabaseService::Format, "Format"},
|
||||||
{18, nullptr, "Import"},
|
{18, nullptr, "Import"},
|
||||||
{19, nullptr, "Export"},
|
{19, nullptr, "Export"},
|
||||||
{20, nullptr, "IsBrokenDatabaseWithClearFlag"},
|
{20, nullptr, "IsBrokenDatabaseWithClearFlag"},
|
||||||
{21, nullptr, "GetIndex"},
|
{21, &IDatabaseService::GetIndex, "GetIndex"},
|
||||||
{22, nullptr, "SetInterfaceVersion"},
|
{22, nullptr, "SetInterfaceVersion"},
|
||||||
{23, nullptr, "Convert"},
|
{23, nullptr, "Convert"},
|
||||||
};
|
};
|
||||||
|
|
@ -47,6 +49,231 @@ public:
|
||||||
|
|
||||||
RegisterHandlers(functions);
|
RegisterHandlers(functions);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
template <typename OutType>
|
||||||
|
std::vector<u8> SerializeArray(OutType (MiiManager::*getter)(u32) const, u32 offset,
|
||||||
|
u32 requested_size, u32& read_size) {
|
||||||
|
read_size = std::min(requested_size, db.Size() - offset);
|
||||||
|
|
||||||
|
std::vector<u8> out(read_size * sizeof(OutType));
|
||||||
|
|
||||||
|
for (u32 i = 0; i < read_size; ++i) {
|
||||||
|
const auto obj = (db.*getter)(offset + i);
|
||||||
|
std::memcpy(out.data() + i * sizeof(OutType), &obj, sizeof(OutType));
|
||||||
|
}
|
||||||
|
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
|
void IsUpdated(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto unknown{rp.PopRaw<u32>()};
|
||||||
|
|
||||||
|
LOG_WARNING(Service_Mii, "(STUBBED) called with unknown={:08X}", unknown);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.Push(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
void IsFullDatabase(Kernel::HLERequestContext& ctx) {
|
||||||
|
LOG_DEBUG(Service_Mii, "called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.Push(db.Full());
|
||||||
|
}
|
||||||
|
|
||||||
|
void GetCount(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto unknown{rp.PopRaw<u32>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with unknown={:08X}", unknown);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.Push<u32>(db.Size());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gets Miis from database at offset and index in format MiiInfoElement
|
||||||
|
void Get(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto size{rp.PopRaw<u32>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with size={:08X}, offset={:08X}", size, offsets[0]);
|
||||||
|
|
||||||
|
u32 read_size{};
|
||||||
|
ctx.WriteBuffer(SerializeArray(&MiiManager::GetInfoElement, offsets[0], size, read_size));
|
||||||
|
offsets[0] += read_size;
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.Push<u32>(read_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gets Miis from database at offset and index in format MiiInfo
|
||||||
|
void Get1(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto size{rp.PopRaw<u32>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with size={:08X}, offset={:08X}", size, offsets[1]);
|
||||||
|
|
||||||
|
u32 read_size{};
|
||||||
|
ctx.WriteBuffer(SerializeArray(&MiiManager::GetInfo, offsets[1], size, read_size));
|
||||||
|
offsets[1] += read_size;
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.Push<u32>(read_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void BuildRandom(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto random_params{rp.PopRaw<RandomParameters>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with param_1={:08X}, param_2={:08X}, param_3={:08X}",
|
||||||
|
random_params.unknown_1, random_params.unknown_2, random_params.unknown_3);
|
||||||
|
|
||||||
|
const auto info = db.CreateRandom(random_params);
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2 + sizeof(MiiInfo) / sizeof(u32)};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.PushRaw<MiiInfo>(info);
|
||||||
|
}
|
||||||
|
|
||||||
|
void BuildDefault(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto index{rp.PopRaw<u32>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with index={:08X}", index);
|
||||||
|
|
||||||
|
const auto info = db.CreateDefault(index);
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2 + sizeof(MiiInfo) / sizeof(u32)};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.PushRaw<MiiInfo>(info);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gets Miis from database at offset and index in format MiiStoreDataElement
|
||||||
|
void Get2(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto size{rp.PopRaw<u32>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with size={:08X}, offset={:08X}", size, offsets[2]);
|
||||||
|
|
||||||
|
u32 read_size{};
|
||||||
|
ctx.WriteBuffer(
|
||||||
|
SerializeArray(&MiiManager::GetStoreDataElement, offsets[2], size, read_size));
|
||||||
|
offsets[2] += read_size;
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.Push<u32>(read_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Gets Miis from database at offset and index in format MiiStoreData
|
||||||
|
void Get3(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto size{rp.PopRaw<u32>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with size={:08X}, offset={:08X}", size, offsets[3]);
|
||||||
|
|
||||||
|
u32 read_size{};
|
||||||
|
ctx.WriteBuffer(SerializeArray(&MiiManager::GetStoreData, offsets[3], size, read_size));
|
||||||
|
offsets[3] += read_size;
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.Push<u32>(read_size);
|
||||||
|
}
|
||||||
|
|
||||||
|
void FindIndex(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto uuid{rp.PopRaw<Common::UUID>()};
|
||||||
|
const auto unknown{rp.PopRaw<bool>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with uuid={}, unknown={}", uuid.FormatSwitch(), unknown);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
|
||||||
|
const auto index = db.IndexOf(uuid);
|
||||||
|
if (index > MAX_MIIS) {
|
||||||
|
// TODO(DarkLordZach): Find a better error code
|
||||||
|
rb.Push(ResultCode(-1));
|
||||||
|
rb.Push(index);
|
||||||
|
} else {
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.Push(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Move(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto uuid{rp.PopRaw<Common::UUID>()};
|
||||||
|
const auto index{rp.PopRaw<u32>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with uuid={}, index={:08X}", uuid.FormatSwitch(), index);
|
||||||
|
|
||||||
|
const auto success = db.Move(uuid, index);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
// TODO(DarkLordZach): Find a better error code
|
||||||
|
rb.Push(success ? RESULT_SUCCESS : ResultCode(-1));
|
||||||
|
}
|
||||||
|
|
||||||
|
void AddOrReplace(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto data{rp.PopRaw<MiiStoreData>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with Mii data uuid={}, name={}", data.uuid.FormatSwitch(),
|
||||||
|
Common::UTF16ToUTF8(data.Name()));
|
||||||
|
|
||||||
|
const auto success = db.AddOrReplace(data);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
// TODO(DarkLordZach): Find a better error code
|
||||||
|
rb.Push(success ? RESULT_SUCCESS : ResultCode(-1));
|
||||||
|
}
|
||||||
|
|
||||||
|
void Delete(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto uuid{rp.PopRaw<Common::UUID>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with uuid={}", uuid.FormatSwitch());
|
||||||
|
|
||||||
|
const auto success = db.Remove(uuid);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
// TODO(DarkLordZach): Find a better error code
|
||||||
|
rb.Push(success ? RESULT_SUCCESS : ResultCode(-1));
|
||||||
|
}
|
||||||
|
|
||||||
|
void Format(Kernel::HLERequestContext& ctx) {
|
||||||
|
LOG_DEBUG(Service_Mii, "called");
|
||||||
|
|
||||||
|
db.Clear();
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
}
|
||||||
|
|
||||||
|
void GetIndex(Kernel::HLERequestContext& ctx) {
|
||||||
|
IPC::RequestParser rp{ctx};
|
||||||
|
const auto info{rp.PopRaw<MiiInfo>()};
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_Mii, "called with Mii info uuid={}, name={}", info.uuid.FormatSwitch(),
|
||||||
|
Common::UTF16ToUTF8(info.Name()));
|
||||||
|
|
||||||
|
const auto index = db.IndexOf(info);
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 2};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.Push(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
MiiManager db;
|
||||||
|
|
||||||
|
// Last read offsets of Get functions
|
||||||
|
std::array<u32, 4> offsets{};
|
||||||
};
|
};
|
||||||
|
|
||||||
class MiiDBModule final : public ServiceFramework<MiiDBModule> {
|
class MiiDBModule final : public ServiceFramework<MiiDBModule> {
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue