[Kernel] Add --profile_directory flag, print achievements to log

Also added more error checking, and changed XELOG statements that use %ws to use %s & xe::to_string instead, seems to_string handles some non-latin strings better than %ws does.
This commit is contained in:
emoose 2018-11-17 17:36:19 +00:00
parent 66fca40b04
commit be8d85b168
No known key found for this signature in database
GPG Key ID: 3735C67912F5FF97
1 changed files with 63 additions and 17 deletions

View File

@ -11,6 +11,9 @@
#include "xenia/kernel/kernel_state.h" #include "xenia/kernel/kernel_state.h"
#include "xenia/kernel/util/shim_utils.h" #include "xenia/kernel/util/shim_utils.h"
#include <gflags/gflags.h>
#include "xenia/kernel/xam/user_profile.h" #include "xenia/kernel/xam/user_profile.h"
#include "xenia/base/clock.h" #include "xenia/base/clock.h"
#include "xenia/base/filesystem.h" #include "xenia/base/filesystem.h"
@ -21,6 +24,9 @@ namespace xe {
namespace kernel { namespace kernel {
namespace xam { namespace xam {
DEFINE_string(profile_directory, "Content\\Profile\\",
"The directory to store profile data inside");
constexpr uint32_t kDashboardID = 0xFFFE07D1; constexpr uint32_t kDashboardID = 0xFFFE07D1;
UserProfile::UserProfile() { UserProfile::UserProfile() {
@ -97,8 +103,11 @@ UserProfile::UserProfile() {
} }
void UserProfile::LoadGpdFiles() { void UserProfile::LoadGpdFiles() {
auto mmap_ = XELOGI("Loading profile GPDs from path %s", FLAGS_profile_directory.c_str());
MappedMemory::Open(L"profile\\FFFE07D1.gpd", MappedMemory::Mode::kRead);
auto mmap_ = MappedMemory::Open(
xe::to_wstring(FLAGS_profile_directory) + L"FFFE07D1.gpd",
MappedMemory::Mode::kRead);
if (!mmap_) { if (!mmap_) {
XELOGW( XELOGW(
"Failed to open dash GPD (FFFE07D1.gpd) for reading, using blank one"); "Failed to open dash GPD (FFFE07D1.gpd) for reading, using blank one");
@ -113,20 +122,29 @@ void UserProfile::LoadGpdFiles() {
for (auto title : titles) { for (auto title : titles) {
wchar_t fname[256]; wchar_t fname[256];
_swprintf(fname, L"profile\\%X.gpd", title.title_id); _swprintf(fname, L"%X.gpd", title.title_id);
mmap_ = MappedMemory::Open(fname, MappedMemory::Mode::kRead); mmap_ = MappedMemory::Open(xe::to_wstring(FLAGS_profile_directory) + fname,
MappedMemory::Mode::kRead);
if (!mmap_) { if (!mmap_) {
XELOGE("Failed to open GPD for title %X (%ws)!", title.title_id, XELOGE("Failed to open GPD for title %X (%s)!", title.title_id,
title.title_name.c_str()); xe::to_string(title.title_name).c_str());
continue; continue;
} }
util::GpdFile title_gpd; util::GpdFile title_gpd;
title_gpd.Read(mmap_->data(), mmap_->size()); bool result = title_gpd.Read(mmap_->data(), mmap_->size());
mmap_->Close(); mmap_->Close();
if (!result) {
XELOGE("Failed to read GPD for title %X (%s)!", title.title_id,
xe::to_string(title.title_name).c_str());
continue;
}
title_gpds_[title.title_id] = title_gpd; title_gpds_[title.title_id] = title_gpd;
} }
XELOGI("Loaded %d profile GPDs", title_gpds_.size() + 1);
} }
util::GpdFile* UserProfile::SetTitleSpaData(const util::SpaFile& spa_data) { util::GpdFile* UserProfile::SetTitleSpaData(const util::SpaFile& spa_data) {
@ -142,6 +160,8 @@ util::GpdFile* UserProfile::SetTitleSpaData(const util::SpaFile& spa_data) {
if (gpd != title_gpds_.end()) { if (gpd != title_gpds_.end()) {
auto& title_gpd = (*gpd).second; auto& title_gpd = (*gpd).second;
XELOGI("Loaded existing GPD for title %X", spa_title);
bool always_update_title = false; bool always_update_title = false;
if (!dash_gpd_.GetTitle(spa_title, &title_info)) { if (!dash_gpd_.GetTitle(spa_title, &title_info)) {
assert_always(); assert_always();
@ -175,9 +195,9 @@ util::GpdFile* UserProfile::SetTitleSpaData(const util::SpaFile& spa_data) {
// If it doesn't exist in GPD, add it to that too // If it doesn't exist in GPD, add it to that too
if (!ach_exists) { if (!ach_exists) {
XELOGD( XELOGD(
"Adding new achievement %d (%ws) from SPA (wasn't inside existing " "Adding new achievement %d (%s) from SPA (wasn't inside existing "
"GPD)", "GPD)",
ach.id, ach.label.c_str()); ach.id, xe::to_string(ach.label).c_str());
ach_updated = true; ach_updated = true;
title_gpd.UpdateAchievement(ach); title_gpd.UpdateAchievement(ach);
@ -194,7 +214,7 @@ util::GpdFile* UserProfile::SetTitleSpaData(const util::SpaFile& spa_data) {
UpdateGpd(kDashboardID, dash_gpd_); UpdateGpd(kDashboardID, dash_gpd_);
} else { } else {
// GPD not found... have to create it! // GPD not found... have to create it!
XELOGD("Creating new GPD for title %X", spa_title); XELOGI("Creating new GPD for title %X", spa_title);
title_info.title_name = xe::to_wstring(spa_data.GetTitleName()); title_info.title_name = xe::to_wstring(spa_data.GetTitleName());
title_info.title_id = spa_title; title_info.title_id = spa_title;
@ -249,8 +269,30 @@ util::GpdFile* UserProfile::SetTitleSpaData(const util::SpaFile& spa_data) {
curr_gpd_ = &title_gpds_[spa_title]; curr_gpd_ = &title_gpds_[spa_title];
curr_title_id_ = spa_title; curr_title_id_ = spa_title;
// Print achievement list to log, ATM there's no other way for users to see
// achievement status...
std::vector<util::XdbfAchievement> achievements;
if (curr_gpd_->GetAchievements(&achievements)) {
XELOGI("Achievement list:");
for (auto ach : achievements) {
// TODO: use ach.unachieved_desc for locked achievements?
// depends on XdbfAchievementFlags::kShowUnachieved afaik
XELOGI("%d - %s - %s - %d GS - %s", ach.id,
xe::to_string(ach.label).c_str(),
xe::to_string(ach.description).c_str(), ach.gamerscore,
ach.IsUnlocked() ? "unlocked" : "locked");
}
XELOGI("Unlocked achievements: %d/%d, gamerscore: %d/%d\r\n",
title_info.achievements_earned, title_info.achievements_possible,
title_info.gamerscore_earned, title_info.gamerscore_total);
}
return curr_gpd_; return curr_gpd_;
} }
util::GpdFile* UserProfile::GetTitleGpd(uint32_t title_id) { util::GpdFile* UserProfile::GetTitleGpd(uint32_t title_id) {
if (!title_id) { if (!title_id) {
return curr_gpd_; return curr_gpd_;
@ -279,7 +321,6 @@ bool UserProfile::UpdateTitleGpd() {
} }
bool UserProfile::UpdateAllGpds() { bool UserProfile::UpdateAllGpds() {
// TODO: optimize so we only have to update the current title?
for (const auto& pair : title_gpds_) { for (const auto& pair : title_gpds_) {
auto gpd = pair.second; auto gpd = pair.second;
bool result = UpdateGpd(pair.first, gpd); bool result = UpdateGpd(pair.first, gpd);
@ -301,16 +342,17 @@ bool UserProfile::UpdateGpd(uint32_t title_id, util::GpdFile& gpd_data) {
return false; return false;
} }
if (!filesystem::PathExists(L"profile\\")) { if (!filesystem::PathExists(xe::to_wstring(FLAGS_profile_directory))) {
filesystem::CreateFolder(L"profile\\"); filesystem::CreateFolder(xe::to_wstring(FLAGS_profile_directory));
} }
wchar_t fname[256]; wchar_t fname[256];
_swprintf(fname, L"profile\\%X.gpd", title_id); _swprintf(fname, L"%X.gpd", title_id);
filesystem::CreateFile(fname); filesystem::CreateFile(xe::to_wstring(FLAGS_profile_directory) + fname);
auto mmap_ = auto mmap_ =
MappedMemory::Open(fname, MappedMemory::Mode::kReadWrite, 0, gpd_length); MappedMemory::Open(xe::to_wstring(FLAGS_profile_directory) + fname,
MappedMemory::Mode::kReadWrite, 0, gpd_length);
if (!mmap_) { if (!mmap_) {
XELOGE("Failed to open %X.gpd for writing!", title_id); XELOGE("Failed to open %X.gpd for writing!", title_id);
return false; return false;
@ -327,8 +369,8 @@ bool UserProfile::UpdateGpd(uint32_t title_id, util::GpdFile& gpd_data) {
util::XdbfTitlePlayed title_info; util::XdbfTitlePlayed title_info;
if (dash_gpd_.GetTitle(title_id, &title_info)) { if (dash_gpd_.GetTitle(title_id, &title_info)) {
std::vector<util::XdbfAchievement> gpd_achievements; std::vector<util::XdbfAchievement> gpd_achievements;
// TODO: let user choose locale?
gpd_data.GetAchievements(&gpd_achievements); gpd_data.GetAchievements(&gpd_achievements);
uint32_t num_ach_total = 0; uint32_t num_ach_total = 0;
uint32_t num_ach_earned = 0; uint32_t num_ach_earned = 0;
uint32_t gamerscore_total = 0; uint32_t gamerscore_total = 0;
@ -342,6 +384,7 @@ bool UserProfile::UpdateGpd(uint32_t title_id, util::GpdFile& gpd_data) {
} }
} }
// Only update dash GPD if something has changed
if (num_ach_total != title_info.achievements_possible || if (num_ach_total != title_info.achievements_possible ||
num_ach_earned != title_info.achievements_earned || num_ach_earned != title_info.achievements_earned ||
gamerscore_total != title_info.gamerscore_total || gamerscore_total != title_info.gamerscore_total ||
@ -353,6 +396,9 @@ bool UserProfile::UpdateGpd(uint32_t title_id, util::GpdFile& gpd_data) {
dash_gpd_.UpdateTitle(title_info); dash_gpd_.UpdateTitle(title_info);
UpdateGpd(kDashboardID, dash_gpd_); UpdateGpd(kDashboardID, dash_gpd_);
// TODO: update gamerscore/achievements earned/titles played settings
// in dashboard GPD
} }
} }
} }