From 5e2f992c798395357849bb0f9b2e540cf92beb2d Mon Sep 17 00:00:00 2001 From: Shardul555 Date: Fri, 20 Aug 2021 23:08:01 +0530 Subject: [PATCH] Achievement support for RetroPlayer --- src/cheevos/Cheevos.cpp | 56 ++++++++++++++++++++++++++++++++++++++++- src/cheevos/Cheevos.h | 16 ++++++++++++ 2 files changed, 71 insertions(+), 1 deletion(-) diff --git a/src/cheevos/Cheevos.cpp b/src/cheevos/Cheevos.cpp index e5486e96..72142b43 100644 --- a/src/cheevos/Cheevos.cpp +++ b/src/cheevos/Cheevos.cpp @@ -17,9 +17,13 @@ using namespace LIBRETRO; +constexpr int URL_SIZE = 512; + CCheevos::CCheevos() { rc_runtime_init(&m_runtime); + m_cheevo_id = 0; + m_url = NULL; } void CCheevos::Initialize() @@ -46,7 +50,9 @@ CCheevos& CCheevos::Get(void) bool CCheevos::GenerateHashFromFile(char* hash, int consoleID, const char* filePath) { - return rc_hash_generate_from_file(hash, consoleID, filePath) != 0; + int res = rc_hash_generate_from_file(hash, consoleID, filePath); + m_hash = hash; + return res != 0; } bool CCheevos::GetGameIDUrl(char* url, size_t size, const char* hash) @@ -63,6 +69,12 @@ bool CCheevos::GetPatchFileUrl(char* url, return rc_url_get_patch(url, size, username, token, gameID) == 0; } +void CCheevos::SetRetroAchievementsCredentials(const char* username, const char* token) +{ + m_username = username; + m_token = token; +} + bool CCheevos::PostRichPresenceUrl(char* url, size_t urlSize, char* postData, @@ -91,6 +103,48 @@ void CCheevos::EvaluateRichPresence(char* evaluation, size_t size) rc_evaluate_richpresence(m_richPresence, evaluation, size, PeekInternal, this, NULL); } +void CCheevos::ActivateAchievement(unsigned cheevo_id, const char* memaddr) +{ + rc_runtime_activate_achievement(&m_runtime, cheevo_id, memaddr, NULL, 0); + // it will return integer value 0 in case achivement is activated successfully. +} + +bool CCheevos::AwardAchievement( + char* url, size_t size, unsigned cheevo_id, int hardcore, const char* game_hash) +{ + return rc_url_award_cheevo(url, size, m_username.c_str(), m_token.c_str(), cheevo_id, 0, + game_hash) >= 0; +} + +void CCheevos::GetCheevo_URL_ID(void (*Callback)(const char* achievement_url, unsigned cheevo_id)) +{ + this->Callback = Callback; +} + +void CCheevos::DeactivateTriggeredAchievement(unsigned cheevo_id) +{ + rc_runtime_deactivate_achievement(&m_runtime, cheevo_id); + char url[URL_SIZE]; + if (AwardAchievement(url, URL_SIZE, cheevo_id, 0, m_hash.c_str())) + { + std::string achievement_url = url; + this->Callback(url, cheevo_id); + } +} + +void CCheevos::RuntimeEventHandler(const rc_runtime_event_t* runtime_event) +{ + if (runtime_event->type == RC_RUNTIME_EVENT_ACHIEVEMENT_TRIGGERED) + { + CCheevos::Get().DeactivateTriggeredAchievement(runtime_event->id); + } +} + +void CCheevos::TestCheevoStatusPerFrame() +{ + rc_runtime_do_frame(&m_runtime, &RuntimeEventHandler, PeekInternal, this, NULL); +} + unsigned int CCheevos::PeekInternal(unsigned address, unsigned num_bytes, void* ud) { CCheevos* cheevos = static_cast(ud); diff --git a/src/cheevos/Cheevos.h b/src/cheevos/Cheevos.h index f8e8167e..3310810d 100644 --- a/src/cheevos/Cheevos.h +++ b/src/cheevos/Cheevos.h @@ -30,6 +30,7 @@ class CCheevos void ResetRuntime(); bool GenerateHashFromFile(char* hash, int consoleID, const char* filePath); bool GetGameIDUrl(char* url, size_t size, const char* hash); + void SetRetroAchievementsCredentials(const char* username, const char* token); bool GetPatchFileUrl( char* url, size_t size, const char* username, const char* token, unsigned gameID); bool PostRichPresenceUrl(char* url, @@ -42,7 +43,13 @@ class CCheevos const char* richPresence); void EnableRichPresence(const char* script); void EvaluateRichPresence(char* evaluation, size_t size); + void ActivateAchievement(unsigned cheevo_id, const char* memaddr); + bool AwardAchievement( + char* url, size_t size, unsigned cheevo_id, int hardcore, const char* game_hash); + void DeactivateTriggeredAchievement(unsigned cheevo_id); + void TestCheevoStatusPerFrame(); unsigned int Peek(unsigned int address, unsigned int numBytes); + void GetCheevo_URL_ID(void (*Callback)(const char* achievement_url, unsigned cheevo_id)); private: const uint8_t* FixupFind(unsigned address, CMemoryMap& mmap, int consolecheevos); @@ -51,6 +58,9 @@ class CCheevos size_t Reduse(size_t addr, size_t mask); static unsigned int PeekInternal(unsigned address, unsigned num_bytes, void* ud); + static void RuntimeEventHandler(const rc_runtime_event_t* runtime_event); + + void (*Callback)(const char* achievement_url, unsigned cheevo_id); rc_runtime_t m_runtime; std::unordered_map m_addressFixups; @@ -59,5 +69,11 @@ class CCheevos rc_richpresence_t* m_richPresence = nullptr; std::string m_richPresenceScript; std::vector m_richPresenceBuffer; + + std::string m_hash; + std::string m_username; + std::string m_token; + char* m_url; + unsigned m_cheevo_id; }; }