Duckstation/src/frontend-common/cheevos.cpp
2021-03-08 01:53:26 +10:00

1176 lines
33 KiB
C++

#include "cheevos.h"
#include "common/cd_image.h"
#include "common/file_system.h"
#include "common/log.h"
#include "common/md5_digest.h"
#include "common/string.h"
#include "common/string_util.h"
#include "common/timestamp.h"
#include "common_host_interface.h"
#include "core/bios.h"
#include "core/bus.h"
#include "core/cpu_core.h"
#include "core/host_display.h"
#include "core/system.h"
#include "fullscreen_ui.h"
#include "http_downloader.h"
#include "imgui_fullscreen.h"
#include "rapidjson/document.h"
#include "rc_url.h"
#include "rcheevos.h"
#include "scmversion/scmversion.h"
#include <algorithm>
#include <cstdarg>
#include <functional>
#include <string>
#include <vector>
Log_SetChannel(Cheevos);
namespace Cheevos {
enum : s32
{
HTTP_OK = FrontendCommon::HTTPDownloader::HTTP_OK,
// Number of seconds between rich presence pings. RAIntegration uses 2 minutes.
RICH_PRESENCE_PING_FREQUENCY = 2 * 60,
NO_RICH_PRESENCE_PING_FREQUENCY = RICH_PRESENCE_PING_FREQUENCY * 2
};
static void CheevosEventHandler(const rc_runtime_event_t* runtime_event);
static unsigned CheevosPeek(unsigned address, unsigned num_bytes, void* ud);
static void ActivateLockedAchievements();
static bool ActivateAchievement(Achievement* achievement);
static void DeactivateAchievement(Achievement* achievement);
static void SendPing();
static void SendPlaying();
/// Uses a temporarily (second) CD image to resolve the hash.
static void GameChanged();
bool g_active = false;
bool g_challenge_mode = false;
u32 g_game_id = 0;
static bool s_logged_in = false;
static bool s_test_mode = false;
static bool s_use_first_disc_from_playlist = true;
static bool s_rich_presence_enabled = false;
static rc_runtime_t s_rcheevos_runtime;
static std::unique_ptr<FrontendCommon::HTTPDownloader> s_http_downloader;
static std::string s_username;
static std::string s_login_token;
static std::string s_game_path;
static std::string s_game_hash;
static std::string s_game_title;
static std::string s_game_developer;
static std::string s_game_publisher;
static std::string s_game_release_date;
static std::string s_game_icon;
static std::vector<Achievement> s_achievements;
static bool s_has_rich_presence = false;
static std::string s_rich_presence_string;
static Common::Timer s_last_ping_time;
static u32 s_total_image_downloads;
static u32 s_completed_image_downloads;
static bool s_image_download_progress_active;
static ALWAYS_INLINE CommonHostInterface* GetHostInterface()
{
return static_cast<CommonHostInterface*>(g_host_interface);
}
static void FormattedError(const char* format, ...)
{
std::va_list ap;
va_start(ap, format);
SmallString str;
str.AppendString("Cheevos Error: ");
str.AppendFormattedStringVA(format, ap);
va_end(ap);
GetHostInterface()->AddOSDMessage(str.GetCharArray(), 10.0f);
Log_ErrorPrint(str.GetCharArray());
}
static std::string GetErrorFromResponseJSON(const rapidjson::Document& doc)
{
if (doc.HasMember("Error") && doc["Error"].IsString())
return doc["Error"].GetString();
return "";
}
static void LogFailedResponseJSON(const FrontendCommon::HTTPDownloader::Request::Data& data)
{
const std::string str_data(reinterpret_cast<const char*>(data.data()), data.size());
Log_ErrorPrintf("API call failed. Response JSON was:\n%s", str_data.c_str());
}
static bool ParseResponseJSON(const char* request_type, s32 status_code,
const FrontendCommon::HTTPDownloader::Request::Data& data, rapidjson::Document& doc,
const char* success_field = "Success")
{
if (status_code != HTTP_OK || data.empty())
{
FormattedError("%s failed: empty response", request_type);
LogFailedResponseJSON(data);
return false;
}
doc.Parse(reinterpret_cast<const char*>(data.data()), data.size());
if (doc.HasParseError())
{
FormattedError("%s failed: parse error at offset %zu: %u", request_type, doc.GetErrorOffset(),
static_cast<unsigned>(doc.GetParseError()));
LogFailedResponseJSON(data);
return false;
}
if (success_field && (!doc.HasMember(success_field) || !doc[success_field].GetBool()))
{
const std::string error(GetErrorFromResponseJSON(doc));
FormattedError("%s failed: Server returned an error: %s", request_type, error.c_str());
LogFailedResponseJSON(data);
return false;
}
return true;
}
template<typename T>
static std::string GetOptionalString(const T& value, const char* key)
{
if (!value.HasMember(key) || !value[key].IsString())
return std::string();
return value[key].GetString();
}
template<typename T>
static u32 GetOptionalUInt(const T& value, const char* key)
{
if (!value.HasMember(key) || !value[key].IsUint())
return 0;
return value[key].GetUint();
}
static Achievement* GetAchievementByID(u32 id)
{
for (Achievement& ach : s_achievements)
{
if (ach.id == id)
return &ach;
}
return nullptr;
}
static void ClearGameInfo()
{
const bool had_game = (g_game_id != 0);
s_has_rich_presence = false;
while (!s_achievements.empty())
{
Achievement& ach = s_achievements.back();
DeactivateAchievement(&ach);
s_achievements.pop_back();
}
std::string().swap(s_game_title);
std::string().swap(s_game_developer);
std::string().swap(s_game_publisher);
std::string().swap(s_game_release_date);
std::string().swap(s_game_icon);
s_rich_presence_string.clear();
g_game_id = 0;
if (had_game)
GetHostInterface()->OnAchievementsRefreshed();
}
static void ClearGamePath()
{
std::string().swap(s_game_path);
std::string().swap(s_game_hash);
}
static std::string GetUserAgent()
{
return StringUtil::StdStringFromFormat("DuckStation %s", g_scm_tag_str);
}
bool Initialize(bool test_mode, bool use_first_disc_from_playlist, bool enable_rich_presence, bool challenge_mode)
{
s_http_downloader = FrontendCommon::HTTPDownloader::Create();
if (!s_http_downloader)
{
Log_ErrorPrint("Failed to create HTTP downloader, cannot use cheevos");
return false;
}
s_http_downloader->SetUserAgent(GetUserAgent());
g_active = true;
g_challenge_mode = challenge_mode;
s_test_mode = test_mode;
s_use_first_disc_from_playlist = use_first_disc_from_playlist;
s_rich_presence_enabled = enable_rich_presence;
rc_runtime_init(&s_rcheevos_runtime);
s_last_ping_time.Reset();
s_username = GetHostInterface()->GetStringSettingValue("Cheevos", "Username");
s_login_token = GetHostInterface()->GetStringSettingValue("Cheevos", "Token");
s_logged_in = (!s_username.empty() && !s_login_token.empty());
if (IsLoggedIn() && System::IsValid())
GameChanged();
return true;
}
void Reset()
{
if (!g_active)
return;
Log_DevPrint("Resetting rcheevos state...");
rc_runtime_reset(&s_rcheevos_runtime);
}
void Shutdown()
{
if (!g_active)
return;
Assert(!s_image_download_progress_active);
s_http_downloader->WaitForAllRequests();
ClearGameInfo();
ClearGamePath();
std::string().swap(s_username);
std::string().swap(s_login_token);
s_logged_in = false;
GetHostInterface()->OnAchievementsRefreshed();
g_active = false;
rc_runtime_destroy(&s_rcheevos_runtime);
s_http_downloader.reset();
}
void Update()
{
s_http_downloader->PollRequests();
if (HasActiveGame())
{
rc_runtime_do_frame(&s_rcheevos_runtime, &CheevosEventHandler, &CheevosPeek, nullptr, nullptr);
if (!s_test_mode)
{
const s32 ping_frequency =
s_rich_presence_enabled ? RICH_PRESENCE_PING_FREQUENCY : NO_RICH_PRESENCE_PING_FREQUENCY;
if (static_cast<s32>(s_last_ping_time.GetTimeSeconds()) >= ping_frequency)
SendPing();
}
}
}
bool IsLoggedIn()
{
return s_logged_in;
}
bool IsTestModeActive()
{
return s_test_mode;
}
bool IsUsingFirstDiscFromPlaylist()
{
return s_use_first_disc_from_playlist;
}
bool IsRichPresenceEnabled()
{
return s_rich_presence_enabled;
}
const std::string& GetUsername()
{
return s_username;
}
const std::string& GetRichPresenceString()
{
return s_rich_presence_string;
}
static void LoginASyncCallback(s32 status_code, const FrontendCommon::HTTPDownloader::Request::Data& data)
{
if (GetHostInterface()->IsFullscreenUIEnabled())
ImGuiFullscreen::CloseBackgroundProgressDialog("cheevos_async_login");
rapidjson::Document doc;
if (!ParseResponseJSON("Login", status_code, data, doc))
return;
if (!doc["User"].IsString() || !doc["Token"].IsString())
{
FormattedError("Login failed. Please check your user name and password, and try again.");
return;
}
std::string username = doc["User"].GetString();
std::string login_token = doc["Token"].GetString();
// save to config
{
std::lock_guard<std::recursive_mutex> guard(GetHostInterface()->GetSettingsLock());
GetHostInterface()->GetSettingsInterface()->SetStringValue("Cheevos", "Username", username.c_str());
GetHostInterface()->GetSettingsInterface()->SetStringValue("Cheevos", "Token", login_token.c_str());
GetHostInterface()->GetSettingsInterface()->SetStringValue(
"Cheevos", "LoginTimestamp", TinyString::FromFormat("%" PRIu64, Timestamp::Now().AsUnixTimestamp()));
GetHostInterface()->GetSettingsInterface()->Save();
}
if (GetHostInterface()->IsFullscreenUIEnabled())
{
GetHostInterface()->ReportFormattedMessage(
GetHostInterface()->TranslateString("Cheevos", "Logged into RetroAchievements using username '%s'."),
username.c_str());
}
if (g_active)
{
s_username = std::move(username);
s_login_token = std::move(login_token);
s_logged_in = true;
// If we have a game running, set it up.
if (System::IsValid())
GameChanged();
}
}
static void SendLogin(const char* username, const char* password, FrontendCommon::HTTPDownloader* http_downloader)
{
char url[768] = {};
int res = rc_url_login_with_password(url, sizeof(url), username, password);
Assert(res == 0);
http_downloader->CreateRequest(url, LoginASyncCallback);
}
bool LoginAsync(const char* username, const char* password)
{
s_http_downloader->WaitForAllRequests();
if (s_logged_in || std::strlen(username) == 0 || std::strlen(password) == 0)
return false;
if (GetHostInterface()->IsFullscreenUIEnabled())
{
ImGuiFullscreen::OpenBackgroundProgressDialog(
"cheevos_async_login", GetHostInterface()->TranslateStdString("Cheevos", "Logging in to RetroAchivements..."), 0,
1, 0);
}
SendLogin(username, password, s_http_downloader.get());
return true;
}
bool Login(const char* username, const char* password)
{
if (g_active)
{
if (!LoginAsync(username, password))
return false;
s_http_downloader->WaitForAllRequests();
return IsLoggedIn();
}
// create a temporary downloader if we're not initialized
Assert(!g_active);
std::unique_ptr<FrontendCommon::HTTPDownloader> http_downloader = FrontendCommon::HTTPDownloader::Create();
if (!http_downloader)
return false;
http_downloader->SetUserAgent(GetUserAgent());
SendLogin(username, password, http_downloader.get());
http_downloader->WaitForAllRequests();
return !GetHostInterface()->GetStringSettingValue("Cheevos", "Token").empty();
}
void Logout()
{
if (g_active)
{
s_http_downloader->WaitForAllRequests();
if (s_logged_in)
{
ClearGameInfo();
std::string().swap(s_username);
std::string().swap(s_login_token);
s_logged_in = false;
GetHostInterface()->OnAchievementsRefreshed();
}
}
// remove from config
std::lock_guard<std::recursive_mutex> guard(GetHostInterface()->GetSettingsLock());
{
GetHostInterface()->GetSettingsInterface()->DeleteValue("Cheevos", "Username");
GetHostInterface()->GetSettingsInterface()->DeleteValue("Cheevos", "Token");
GetHostInterface()->GetSettingsInterface()->Save();
}
}
static void UpdateImageDownloadProgress()
{
static const char* str_id = "cheevo_image_download";
if (s_completed_image_downloads >= s_total_image_downloads)
{
s_completed_image_downloads = 0;
s_total_image_downloads = 0;
if (s_image_download_progress_active)
{
ImGuiFullscreen::CloseBackgroundProgressDialog(str_id);
s_image_download_progress_active = false;
}
return;
}
if (!GetHostInterface()->IsFullscreenUIEnabled())
return;
std::string message(g_host_interface->TranslateStdString("Cheevos", "Downloading achievement resources..."));
if (!s_image_download_progress_active)
{
ImGuiFullscreen::OpenBackgroundProgressDialog(str_id, std::move(message), 0,
static_cast<s32>(s_total_image_downloads),
static_cast<s32>(s_completed_image_downloads));
s_image_download_progress_active = true;
}
else
{
ImGuiFullscreen::UpdateBackgroundProgressDialog(str_id, std::move(message), 0,
static_cast<s32>(s_total_image_downloads),
static_cast<s32>(s_completed_image_downloads));
}
}
static void DownloadImage(std::string url, std::string cache_filename)
{
auto callback = [cache_filename](s32 status_code, const FrontendCommon::HTTPDownloader::Request::Data& data) {
s_completed_image_downloads++;
UpdateImageDownloadProgress();
if (status_code != HTTP_OK)
return;
if (!FileSystem::WriteBinaryFile(cache_filename.c_str(), data.data(), data.size()))
{
Log_ErrorPrintf("Failed to write badge image to '%s'", cache_filename.c_str());
return;
}
FullscreenUI::InvalidateCachedTexture(cache_filename);
UpdateImageDownloadProgress();
};
s_total_image_downloads++;
UpdateImageDownloadProgress();
s_http_downloader->CreateRequest(std::move(url), std::move(callback));
}
static std::string GetBadgeImageFilename(const char* badge_name, bool locked, bool cache_path)
{
if (!cache_path)
{
return StringUtil::StdStringFromFormat("%s%s.png", badge_name, locked ? "_lock" : "");
}
else
{
// well, this comes from the internet.... :)
SmallString clean_name(badge_name);
FileSystem::SanitizeFileName(clean_name);
return GetHostInterface()->GetUserDirectoryRelativePath("cache" FS_OSPATH_SEPARATOR_STR
"achievement_badge" FS_OSPATH_SEPARATOR_STR "%s%s.png",
clean_name.GetCharArray(), locked ? "_lock" : "");
}
}
static std::string ResolveBadgePath(const char* badge_name, bool locked)
{
char url[256];
// unlocked image
std::string cache_path(GetBadgeImageFilename(badge_name, locked, true));
if (FileSystem::FileExists(cache_path.c_str()))
return cache_path;
std::string badge_name_with_extension(GetBadgeImageFilename(badge_name, locked, false));
int res = rc_url_get_badge_image(url, sizeof(url), badge_name_with_extension.c_str());
Assert(res == 0);
DownloadImage(url, cache_path);
return cache_path;
}
static void DisplayAchievementSummary()
{
std::string title = s_game_title;
if (g_challenge_mode)
title += GetHostInterface()->TranslateString("Cheevos", " (Hardcore Mode)");
std::string summary;
if (GetAchievementCount() > 0)
{
summary = StringUtil::StdStringFromFormat(
GetHostInterface()->TranslateString("Cheevos", "You have earned %u of %u achievements, and %u of %u points."),
GetUnlockedAchiementCount(), GetAchievementCount(), GetCurrentPointsForGame(), GetMaximumPointsForGame());
}
else
{
summary = GetHostInterface()->TranslateString("Cheevos", "This game has no achievements.");
}
ImGuiFullscreen::AddNotification(10.0f, std::move(title), std::move(summary), s_game_icon);
}
static void GetUserUnlocksCallback(s32 status_code, const FrontendCommon::HTTPDownloader::Request::Data& data)
{
rapidjson::Document doc;
if (!ParseResponseJSON("Get User Unlocks", status_code, data, doc))
{
ClearGameInfo();
return;
}
// verify game id for sanity
const u32 game_id = GetOptionalUInt(doc, "GameID");
if (game_id != g_game_id)
{
FormattedError("GameID from user unlocks doesn't match (got %u expected %u)", game_id, g_game_id);
ClearGameInfo();
return;
}
// flag achievements as unlocked
if (doc.HasMember("UserUnlocks") && doc["UserUnlocks"].IsArray())
{
for (const auto& value : doc["UserUnlocks"].GetArray())
{
if (!value.IsUint())
continue;
const u32 achievement_id = value.GetUint();
Achievement* cheevo = GetAchievementByID(achievement_id);
if (!cheevo)
{
Log_ErrorPrintf("Server returned unknown achievement %u", achievement_id);
continue;
}
cheevo->locked = false;
}
}
// start scanning for locked achievements
ActivateLockedAchievements();
DisplayAchievementSummary();
SendPlaying();
SendPing();
GetHostInterface()->OnAchievementsRefreshed();
}
static void GetUserUnlocks()
{
char url[256];
int res = rc_url_get_unlock_list(url, sizeof(url), s_username.c_str(), s_login_token.c_str(), g_game_id,
static_cast<int>(g_challenge_mode));
Assert(res == 0);
s_http_downloader->CreateRequest(url, GetUserUnlocksCallback);
}
static void GetPatchesCallback(s32 status_code, const FrontendCommon::HTTPDownloader::Request::Data& data)
{
ClearGameInfo();
rapidjson::Document doc;
if (!ParseResponseJSON("Get Patches", status_code, data, doc))
return;
if (!doc.HasMember("PatchData") || !doc["PatchData"].IsObject())
{
FormattedError("No patch data returned from server.");
return;
}
// parse info
const auto patch_data(doc["PatchData"].GetObject());
if (!patch_data["ID"].IsUint())
{
FormattedError("Patch data is missing game ID");
return;
}
g_game_id = GetOptionalUInt(patch_data, "ID");
s_game_title = GetOptionalString(patch_data, "Title");
s_game_developer = GetOptionalString(patch_data, "Developer");
s_game_publisher = GetOptionalString(patch_data, "Publisher");
s_game_release_date = GetOptionalString(patch_data, "Released");
// try for a icon
std::string icon_name(GetOptionalString(patch_data, "ImageIcon"));
if (!icon_name.empty())
{
s_game_icon = GetHostInterface()->GetUserDirectoryRelativePath(
"cache" FS_OSPATH_SEPARATOR_STR "achievement_gameicon" FS_OSPATH_SEPARATOR_STR "%u.png", g_game_id);
if (!FileSystem::FileExists(s_game_icon.c_str()))
{
// for some reason rurl doesn't have this :(
std::string icon_url(StringUtil::StdStringFromFormat("http://i.retroachievements.org%s", icon_name.c_str()));
DownloadImage(std::move(icon_url), s_game_icon);
}
}
// parse achievements
if (patch_data.HasMember("Achievements") && patch_data["Achievements"].IsArray())
{
const auto achievements(patch_data["Achievements"].GetArray());
for (const auto& achievement : achievements)
{
if (!achievement.HasMember("ID") || !achievement["ID"].IsNumber() || !achievement.HasMember("Flags") ||
!achievement["Flags"].IsNumber() || !achievement.HasMember("MemAddr") || !achievement["MemAddr"].IsString() ||
!achievement.HasMember("Title") || !achievement["Title"].IsString())
{
continue;
}
const u32 id = achievement["ID"].GetUint();
const u32 category = achievement["Flags"].GetUint();
const char* memaddr = achievement["MemAddr"].GetString();
std::string title = achievement["Title"].GetString();
std::string description = GetOptionalString(achievement, "Description");
std::string badge_name = GetOptionalString(achievement, "BadgeName");
const u32 points = GetOptionalUInt(achievement, "Points");
// Skip local and unofficial achievements for now.
if (static_cast<AchievementCategory>(category) == AchievementCategory::Local ||
static_cast<AchievementCategory>(category) == AchievementCategory::Unofficial)
{
Log_WarningPrintf("Skipping unofficial achievement %u (%s)", id, title.c_str());
continue;
}
if (GetAchievementByID(id))
{
Log_ErrorPrintf("Achievement %u already exists", id);
continue;
}
Achievement cheevo;
cheevo.id = id;
cheevo.memaddr = memaddr;
cheevo.title = std::move(title);
cheevo.description = std::move(description);
cheevo.locked = true;
cheevo.active = false;
cheevo.points = points;
if (!badge_name.empty())
{
cheevo.locked_badge_path = ResolveBadgePath(badge_name.c_str(), true);
cheevo.unlocked_badge_path = ResolveBadgePath(badge_name.c_str(), false);
}
s_achievements.push_back(std::move(cheevo));
}
}
// parse rich presence
if (s_rich_presence_enabled && patch_data.HasMember("RichPresencePatch") &&
patch_data["RichPresencePatch"].IsString())
{
const char* patch = patch_data["RichPresencePatch"].GetString();
int res = rc_runtime_activate_richpresence(&s_rcheevos_runtime, patch, nullptr, 0);
if (res == RC_OK)
s_has_rich_presence = true;
else
Log_WarningPrintf("Failed to activate rich presence: %s", rc_error_str(res));
}
Log_InfoPrintf("Game Title: %s", s_game_title.c_str());
Log_InfoPrintf("Game Developer: %s", s_game_developer.c_str());
Log_InfoPrintf("Game Publisher: %s", s_game_publisher.c_str());
Log_InfoPrintf("Achievements: %zu", s_achievements.size());
if (!s_achievements.empty() || s_has_rich_presence)
{
if (!s_test_mode)
{
GetUserUnlocks();
}
else
{
ActivateLockedAchievements();
DisplayAchievementSummary();
GetHostInterface()->OnAchievementsRefreshed();
}
}
else
{
DisplayAchievementSummary();
ClearGameInfo();
return;
}
}
static void GetPatches(u32 game_id)
{
char url[256] = {};
int res = rc_url_get_patch(url, sizeof(url), s_username.c_str(), s_login_token.c_str(), game_id);
Assert(res == 0);
s_http_downloader->CreateRequest(url, GetPatchesCallback);
}
static std::string GetGameHash(CDImage* cdi)
{
std::string executable_name;
std::vector<u8> executable_data;
if (!System::ReadExecutableFromImage(cdi, &executable_name, &executable_data))
return {};
BIOS::PSEXEHeader header;
if (executable_data.size() >= sizeof(header))
std::memcpy(&header, executable_data.data(), sizeof(header));
if (!BIOS::IsValidPSExeHeader(header, static_cast<u32>(executable_data.size())))
{
Log_ErrorPrintf("PS-EXE header is invalid in '%s' (%zu bytes)", executable_name.c_str(), executable_data.size());
return {};
}
// See rcheevos hash.c - rc_hash_psx().
const u32 MAX_HASH_SIZE = 64 * 1024 * 1024;
const u32 hash_size = std::min<u32>(sizeof(header) + header.file_size, MAX_HASH_SIZE);
Assert(hash_size <= executable_data.size());
MD5Digest digest;
digest.Update(executable_name.c_str(), static_cast<u32>(executable_name.size()));
if (hash_size > 0)
digest.Update(executable_data.data(), hash_size);
u8 hash[16];
digest.Final(hash);
std::string hash_str(StringUtil::StdStringFromFormat(
"%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x", hash[0], hash[1], hash[2], hash[3], hash[4],
hash[5], hash[6], hash[7], hash[8], hash[9], hash[10], hash[11], hash[12], hash[13], hash[14], hash[15]));
Log_InfoPrintf("Hash for '%s' (%zu bytes, %u bytes hashed): %s", executable_name.c_str(), executable_data.size(),
hash_size, hash_str.c_str());
return hash_str;
}
static void GetGameIdCallback(s32 status_code, const FrontendCommon::HTTPDownloader::Request::Data& data)
{
rapidjson::Document doc;
if (!ParseResponseJSON("Get Game ID", status_code, data, doc))
return;
const u32 game_id = (doc.HasMember("GameID") && doc["GameID"].IsUint()) ? doc["GameID"].GetUint() : 0;
Log_InfoPrintf("Server returned GameID %u", game_id);
if (game_id != 0)
GetPatches(game_id);
}
void GameChanged()
{
Assert(System::IsValid());
const std::string& path = System::GetRunningPath();
if (path.empty() || s_game_path == path)
return;
std::unique_ptr<CDImage> cdi = CDImage::Open(path.c_str());
if (!cdi)
{
Log_ErrorPrintf("Failed to open temporary CD image '%s'", path.c_str());
ClearGameInfo();
return;
}
GameChanged(path, cdi.get());
}
void GameChanged(const std::string& path, CDImage* image)
{
if (s_game_path == path)
return;
std::string game_hash;
if (image)
{
game_hash = GetGameHash(image);
if (s_game_hash == game_hash)
{
// only the path has changed - different format/save state/etc.
Log_InfoPrintf("Detected path change from '%s' to '%s'", s_game_path.c_str(), path.c_str());
s_game_path = path;
return;
}
}
s_http_downloader->WaitForAllRequests();
const u32 playlist_count = System::GetMediaPlaylistCount();
if (playlist_count > 1 && s_use_first_disc_from_playlist)
{
// have to pass the path in, because the image isn't owned by the system yet
const u32 playlist_index = System::GetMediaPlaylistIndexForPath(path);
if (playlist_index > 0 && playlist_index < playlist_count)
{
const std::string& first_disc_path(System::GetMediaPlaylistPath(0));
std::unique_ptr<CDImage> first_disc_image(CDImage::Open(first_disc_path.c_str()));
if (first_disc_image)
{
Log_InfoPrintf("Using first disc '%s' from playlist (currently '%s')", first_disc_path.c_str(), path.c_str());
GameChanged(first_disc_path, first_disc_image.get());
return;
}
else
{
Log_ErrorPrintf("Failed to open first disc '%s' from playlist", first_disc_path.c_str());
return;
}
}
}
ClearGameInfo();
ClearGamePath();
s_game_path = path;
s_game_hash = std::move(game_hash);
if (!image)
return;
if (s_game_hash.empty())
{
GetHostInterface()->AddOSDMessage(GetHostInterface()->TranslateStdString(
"OSDMessage", "Failed to read executable from disc. Achievements disabled."),
10.0f);
return;
}
char url[256];
int res = rc_url_get_gameid(url, sizeof(url), s_game_hash.c_str());
Assert(res == 0);
s_http_downloader->CreateRequest(url, GetGameIdCallback);
}
static void SendPlayingCallback(s32 status_code, const FrontendCommon::HTTPDownloader::Request::Data& data)
{
rapidjson::Document doc;
if (!ParseResponseJSON("Post Activity", status_code, data, doc))
return;
Log_InfoPrintf("Playing game updated to %u (%s)", g_game_id, s_game_title.c_str());
}
void SendPlaying()
{
if (!HasActiveGame())
return;
char url[256];
int res = rc_url_post_playing(url, sizeof(url), s_username.c_str(), s_login_token.c_str(), g_game_id);
Assert(res == 0);
s_http_downloader->CreateRequest(url, SendPlayingCallback);
}
static void UpdateRichPresence()
{
char buffer[512];
int res = rc_runtime_get_richpresence(&s_rcheevos_runtime, buffer, sizeof(buffer), CheevosPeek, nullptr, nullptr);
if (res <= 0)
{
const bool had_rich_presence = !s_rich_presence_string.empty();
s_rich_presence_string.clear();
if (had_rich_presence)
GetHostInterface()->OnAchievementsRefreshed();
return;
}
if (s_rich_presence_string == buffer)
return;
s_rich_presence_string.assign(buffer);
GetHostInterface()->OnAchievementsRefreshed();
}
static void SendPingCallback(s32 status_code, const FrontendCommon::HTTPDownloader::Request::Data& data)
{
rapidjson::Document doc;
if (!ParseResponseJSON("Ping", status_code, data, doc))
return;
}
void SendPing()
{
if (!HasActiveGame())
return;
if (s_has_rich_presence)
UpdateRichPresence();
char url[256];
char post_data[512];
int res = rc_url_ping(url, sizeof(url), post_data, sizeof(post_data), s_username.c_str(), s_login_token.c_str(),
g_game_id, s_rich_presence_string.c_str());
Assert(res == 0);
s_http_downloader->CreatePostRequest(url, post_data, SendPingCallback);
s_last_ping_time.Reset();
}
const std::string& GetGameTitle()
{
return s_game_title;
}
const std::string& GetGameDeveloper()
{
return s_game_developer;
}
const std::string& GetGamePublisher()
{
return s_game_publisher;
}
const std::string& GetGameReleaseDate()
{
return s_game_release_date;
}
const std::string& GetGameIcon()
{
return s_game_icon;
}
bool EnumerateAchievements(std::function<bool(const Achievement&)> callback)
{
for (const Achievement& cheevo : s_achievements)
{
if (!callback(cheevo))
return false;
}
return true;
}
u32 GetUnlockedAchiementCount()
{
u32 count = 0;
for (const Achievement& cheevo : s_achievements)
{
if (!cheevo.locked)
count++;
}
return count;
}
u32 GetAchievementCount()
{
return static_cast<u32>(s_achievements.size());
}
u32 GetMaximumPointsForGame()
{
u32 points = 0;
for (const Achievement& cheevo : s_achievements)
points += cheevo.points;
return points;
}
u32 GetCurrentPointsForGame()
{
u32 points = 0;
for (const Achievement& cheevo : s_achievements)
{
if (!cheevo.locked)
points += cheevo.points;
}
return points;
}
void ActivateLockedAchievements()
{
for (Achievement& cheevo : s_achievements)
{
if (cheevo.locked)
ActivateAchievement(&cheevo);
}
}
bool ActivateAchievement(Achievement* achievement)
{
if (achievement->active)
return true;
const int err =
rc_runtime_activate_achievement(&s_rcheevos_runtime, achievement->id, achievement->memaddr.c_str(), nullptr, 0);
if (err != RC_OK)
{
Log_ErrorPrintf("Achievement %u memaddr parse error: %s", achievement->id, rc_error_str(err));
return false;
}
achievement->active = true;
Log_DevPrintf("Activated achievement %s (%u)", achievement->title.c_str(), achievement->id);
return true;
}
void DeactivateAchievement(Achievement* achievement)
{
if (!achievement->active)
return;
rc_runtime_deactivate_achievement(&s_rcheevos_runtime, achievement->id);
achievement->active = false;
Log_DevPrintf("Deactivated achievement %s (%u)", achievement->title.c_str(), achievement->id);
}
static void UnlockAchievementCallback(s32 status_code, const FrontendCommon::HTTPDownloader::Request::Data& data)
{
rapidjson::Document doc;
if (!ParseResponseJSON("Award Cheevo", status_code, data, doc))
return;
// we don't really need to do anything here
}
void UnlockAchievement(u32 achievement_id, bool add_notification /* = true*/)
{
Achievement* achievement = GetAchievementByID(achievement_id);
if (!achievement)
{
Log_ErrorPrintf("Attempting to unlock unknown achievement %u", achievement_id);
return;
}
else if (!achievement->locked)
{
Log_WarningPrintf("Achievement %u for game %u is already unlocked", achievement_id, g_game_id);
return;
}
achievement->locked = false;
DeactivateAchievement(achievement);
Log_InfoPrintf("Achievement %s (%u) for game %u unlocked", achievement->title.c_str(), achievement_id, g_game_id);
ImGuiFullscreen::AddNotification(15.0f, achievement->title, achievement->description,
achievement->unlocked_badge_path);
if (s_test_mode)
{
Log_WarningPrintf("Skipping sending achievement %u unlock to server because of test mode.", achievement_id);
return;
}
char url[256];
rc_url_award_cheevo(url, sizeof(url), s_username.c_str(), s_login_token.c_str(), achievement_id,
static_cast<int>(g_challenge_mode), s_game_hash.c_str());
s_http_downloader->CreateRequest(url, UnlockAchievementCallback);
}
void CheevosEventHandler(const rc_runtime_event_t* runtime_event)
{
static const char* events[] = {"RC_RUNTIME_EVENT_ACHIEVEMENT_ACTIVATED", "RC_RUNTIME_EVENT_ACHIEVEMENT_PAUSED",
"RC_RUNTIME_EVENT_ACHIEVEMENT_RESET", "RC_RUNTIME_EVENT_ACHIEVEMENT_TRIGGERED",
"RC_RUNTIME_EVENT_ACHIEVEMENT_PRIMED", "RC_RUNTIME_EVENT_LBOARD_STARTED",
"RC_RUNTIME_EVENT_LBOARD_CANCELED", "RC_RUNTIME_EVENT_LBOARD_UPDATED",
"RC_RUNTIME_EVENT_LBOARD_TRIGGERED", "RC_RUNTIME_EVENT_ACHIEVEMENT_DISABLED",
"RC_RUNTIME_EVENT_LBOARD_DISABLED"};
const char* event_text =
((unsigned)runtime_event->type >= countof(events)) ? "unknown" : events[(unsigned)runtime_event->type];
Log_DevPrintf("Cheevos Event %s for %u", event_text, runtime_event->id);
if (runtime_event->type == RC_RUNTIME_EVENT_ACHIEVEMENT_TRIGGERED)
UnlockAchievement(runtime_event->id);
}
// from cheats.cpp - do we want to move this somewhere else?
template<typename T>
static T DoMemoryRead(PhysicalMemoryAddress address)
{
T result;
if ((address & CPU::DCACHE_LOCATION_MASK) == CPU::DCACHE_LOCATION &&
(address & CPU::DCACHE_OFFSET_MASK) < CPU::DCACHE_SIZE)
{
std::memcpy(&result, &CPU::g_state.dcache[address & CPU::DCACHE_OFFSET_MASK], sizeof(result));
return result;
}
address &= CPU::PHYSICAL_MEMORY_ADDRESS_MASK;
if (address < Bus::RAM_MIRROR_END)
{
std::memcpy(&result, &Bus::g_ram[address & Bus::RAM_MASK], sizeof(result));
return result;
}
if (address >= Bus::BIOS_BASE && address < (Bus::BIOS_BASE + Bus::BIOS_SIZE))
{
std::memcpy(&result, &Bus::g_bios[address & Bus::BIOS_MASK], sizeof(result));
return result;
}
result = static_cast<T>(0);
return result;
}
unsigned CheevosPeek(unsigned address, unsigned num_bytes, void* ud)
{
switch (num_bytes)
{
case 1:
return ZeroExtend32(DoMemoryRead<u8>(address));
case 2:
return ZeroExtend32(DoMemoryRead<u16>(address));
case 4:
return ZeroExtend32(DoMemoryRead<u32>(address));
default:
return 0;
}
}
} // namespace Cheevos