mirror of
https://github.com/libretro/RetroArch.git
synced 2024-12-18 22:57:06 +00:00
2977 lines
88 KiB
C
2977 lines
88 KiB
C
/* RetroArch - A frontend for libretro.
|
|
* Copyright (C) 2015-2016 - Andre Leiradella
|
|
*
|
|
* RetroArch is free software: you can redistribute it and/or modify it under the terms
|
|
* of the GNU General Public License as published by the Free Software Found-
|
|
* ation, either version 3 of the License, or (at your option) any later version.
|
|
*
|
|
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
|
|
* PURPOSE. See the GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along with RetroArch.
|
|
* If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include <ctype.h>
|
|
|
|
#include <file/file_path.h>
|
|
#include <string/stdstring.h>
|
|
#include <streams/interface_stream.h>
|
|
#include <streams/file_stream.h>
|
|
#include <features/features_cpu.h>
|
|
#include <formats/cdfs.h>
|
|
#include <compat/strl.h>
|
|
#include <rhash.h>
|
|
#include <retro_miscellaneous.h>
|
|
#include <retro_math.h>
|
|
#include <net/net_http.h>
|
|
#include <libretro.h>
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "../config.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_MENU
|
|
#include "../menu/menu_driver.h"
|
|
#include "../menu/menu_entries.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_GFX_WIDGETS
|
|
#include "../gfx/gfx_widgets.h"
|
|
#endif
|
|
|
|
#ifdef HAVE_THREADS
|
|
#include <rthreads/rthreads.h>
|
|
#endif
|
|
|
|
#ifdef HAVE_DISCORD
|
|
#include "../discord/discord.h"
|
|
#endif
|
|
|
|
#include "badges.h"
|
|
#include "cheevos.h"
|
|
#include "fixup.h"
|
|
#include "parser.h"
|
|
#include "hash.h"
|
|
#include "util.h"
|
|
|
|
#include "../file_path_special.h"
|
|
#include "../paths.h"
|
|
#include "../command.h"
|
|
#include "../dynamic.h"
|
|
#include "../configuration.h"
|
|
#include "../performance_counters.h"
|
|
#include "../msg_hash.h"
|
|
#include "../retroarch.h"
|
|
#include "../core.h"
|
|
#include "../version.h"
|
|
|
|
#include "../frontend/frontend_driver.h"
|
|
#include "../network/net_http_special.h"
|
|
#include "../tasks/tasks_internal.h"
|
|
|
|
#include "../deps/rcheevos/include/rcheevos.h"
|
|
#include "../deps/rcheevos/include/rurl.h"
|
|
|
|
/* Define this macro to prevent cheevos from being deactivated. */
|
|
#undef CHEEVOS_DONT_DEACTIVATE
|
|
|
|
/* Define this macro to dump all cheevos' addresses. */
|
|
#undef CHEEVOS_DUMP_ADDRS
|
|
|
|
/* Define this macro to remove HTTP timeouts. */
|
|
#undef CHEEVOS_NO_TIMEOUT
|
|
|
|
/* Define this macro to load a JSON file from disk instead of downloading
|
|
* from retroachievements.org. */
|
|
#undef CHEEVOS_JSON_OVERRIDE
|
|
|
|
/* Define this macro with a string to save the JSON file to disk with
|
|
* that name. */
|
|
#undef CHEEVOS_SAVE_JSON
|
|
|
|
/* Define this macro to log URLs. */
|
|
#undef CHEEVOS_LOG_URLS
|
|
|
|
/* Define this macro to have the password and token logged. THIS WILL DISCLOSE
|
|
* THE USER'S PASSWORD, TAKE CARE! */
|
|
#undef CHEEVOS_LOG_PASSWORD
|
|
|
|
/* Define this macro to log downloaded badge images. */
|
|
#undef CHEEVOS_LOG_BADGES
|
|
|
|
/* Number of usecs to wait between posting rich presence to the site. */
|
|
/* Keep consistent with SERVER_PING_FREQUENCY from RAIntegration. */
|
|
#define CHEEVOS_PING_FREQUENCY 2 * 60 * 1000000
|
|
|
|
typedef struct
|
|
{
|
|
rc_trigger_t* trigger;
|
|
const rcheevos_racheevo_t* info;
|
|
int active;
|
|
int last;
|
|
} rcheevos_cheevo_t;
|
|
|
|
typedef struct
|
|
{
|
|
rc_lboard_t* lboard;
|
|
const rcheevos_ralboard_t* info;
|
|
bool active;
|
|
int last_value;
|
|
int format;
|
|
} rcheevos_lboard_t;
|
|
|
|
typedef struct
|
|
{
|
|
rc_richpresence_t* richpresence;
|
|
char evaluation[256];
|
|
retro_time_t last_update;
|
|
} rcheevos_richpresence_t;
|
|
|
|
enum rcheevos_async_io_type
|
|
{
|
|
CHEEVOS_ASYNC_RICHPRESENCE,
|
|
CHEEVOS_ASYNC_AWARD_ACHIEVEMENT
|
|
};
|
|
|
|
typedef struct rcheevos_async_io_request
|
|
{
|
|
int id;
|
|
int value;
|
|
int attempt_count;
|
|
char type;
|
|
char hardcore;
|
|
char user_agent[256];
|
|
} rcheevos_async_io_request;
|
|
|
|
typedef struct
|
|
{
|
|
retro_task_t* task;
|
|
#ifdef HAVE_THREADS
|
|
slock_t* task_lock;
|
|
#endif
|
|
|
|
bool core_supports;
|
|
bool invalid_peek_address;
|
|
|
|
rcheevos_rapatchdata_t patchdata;
|
|
rcheevos_cheevo_t* core;
|
|
rcheevos_cheevo_t* unofficial;
|
|
rcheevos_lboard_t* lboards;
|
|
rcheevos_richpresence_t richpresence;
|
|
|
|
rcheevos_fixups_t fixups;
|
|
|
|
char token[32];
|
|
char hash[33];
|
|
} rcheevos_locals_t;
|
|
|
|
typedef struct
|
|
{
|
|
int label;
|
|
const char* name;
|
|
const uint32_t* ext_hashes;
|
|
} rcheevos_finder_t;
|
|
|
|
typedef struct
|
|
{
|
|
uint8_t id[4]; /* NES^Z */
|
|
uint8_t rom_size;
|
|
uint8_t vrom_size;
|
|
uint8_t rom_type;
|
|
uint8_t rom_type2;
|
|
uint8_t reserve[8];
|
|
} rcheevos_nes_header_t;
|
|
|
|
static rcheevos_locals_t rcheevos_locals =
|
|
{
|
|
NULL, /* task */
|
|
#ifdef HAVE_THREADS
|
|
NULL, /* task_lock */
|
|
#endif
|
|
true, /* core_supports */
|
|
false,/* invalid_peek_address */
|
|
{0}, /* patchdata */
|
|
NULL, /* core */
|
|
NULL, /* unofficial */
|
|
NULL, /* lboards */
|
|
{0}, /* rich presence */
|
|
{0}, /* fixups */
|
|
{0}, /* token */
|
|
"N/A",/* hash */
|
|
};
|
|
|
|
bool rcheevos_loaded = false;
|
|
bool rcheevos_hardcore_active = false;
|
|
bool rcheevos_hardcore_paused = false;
|
|
bool rcheevos_state_loaded_flag = false;
|
|
int rcheevos_cheats_are_enabled = 0;
|
|
int rcheevos_cheats_were_enabled = 0;
|
|
char rcheevos_user_agent_prefix[128] = "";
|
|
|
|
#ifdef HAVE_THREADS
|
|
#define CHEEVOS_LOCK(l) do { slock_lock(l); } while (0)
|
|
#define CHEEVOS_UNLOCK(l) do { slock_unlock(l); } while (0)
|
|
#else
|
|
#define CHEEVOS_LOCK(l)
|
|
#define CHEEVOS_UNLOCK(l)
|
|
#endif
|
|
|
|
#define CHEEVOS_MB(x) ((x) * 1024 * 1024)
|
|
|
|
/*****************************************************************************
|
|
Supporting functions.
|
|
*****************************************************************************/
|
|
|
|
#ifndef CHEEVOS_VERBOSE
|
|
|
|
void rcheevos_log(const char *fmt, ...)
|
|
{
|
|
(void)fmt;
|
|
}
|
|
|
|
#endif
|
|
|
|
static void rcheevos_get_user_agent(char* buffer)
|
|
{
|
|
struct retro_system_info *system = runloop_get_libretro_system_info();
|
|
const char* scan;
|
|
char* ptr;
|
|
|
|
if (!rcheevos_user_agent_prefix[0])
|
|
{
|
|
const frontend_ctx_driver_t *frontend = frontend_get_ptr();
|
|
int major, minor;
|
|
char tmp[64];
|
|
|
|
ptr = rcheevos_user_agent_prefix + sprintf(rcheevos_user_agent_prefix, "RetroArch/%s", PACKAGE_VERSION);
|
|
|
|
if (frontend && frontend->get_os)
|
|
{
|
|
frontend->get_os(tmp, sizeof(tmp), &major, &minor);
|
|
ptr += sprintf(ptr, " (%s %d.%d)", tmp, major, minor);
|
|
}
|
|
}
|
|
|
|
ptr = buffer + sprintf(buffer, "%s", rcheevos_user_agent_prefix);
|
|
|
|
if (system && !string_is_empty(system->library_name))
|
|
{
|
|
const char* path = path_get(RARCH_PATH_CORE);
|
|
if (!string_is_empty(path))
|
|
{
|
|
sprintf(ptr, " %s", path_basename(path));
|
|
path_remove_extension(ptr);
|
|
ptr += strlen(ptr);
|
|
}
|
|
else
|
|
{
|
|
*ptr++ = ' ';
|
|
|
|
scan = system->library_name;
|
|
while (*scan)
|
|
{
|
|
if (*scan == ' ')
|
|
{
|
|
*ptr++ = '_';
|
|
++scan;
|
|
}
|
|
else
|
|
{
|
|
*ptr++ = *scan++;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (system->library_version)
|
|
{
|
|
*ptr++ = '/';
|
|
|
|
scan = system->library_version;
|
|
while (*scan)
|
|
{
|
|
if (*scan == ' ')
|
|
{
|
|
*ptr++ = '_';
|
|
++scan;
|
|
}
|
|
else
|
|
{
|
|
*ptr++ = *scan++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
*ptr = '\0';
|
|
}
|
|
|
|
static void rcheevos_log_url(const char* format, const char* url)
|
|
{
|
|
#ifdef CHEEVOS_LOG_URLS
|
|
#ifdef CHEEVOS_LOG_PASSWORD
|
|
CHEEVOS_LOG(format, url);
|
|
#else
|
|
char copy[256];
|
|
char* aux = NULL;
|
|
char* next = NULL;
|
|
|
|
if (!string_is_empty(url))
|
|
strlcpy(copy, url, sizeof(copy));
|
|
|
|
aux = strstr(copy, "?p=");
|
|
|
|
if (!aux)
|
|
aux = strstr(copy, "&p=");
|
|
|
|
if (aux)
|
|
{
|
|
aux += 3;
|
|
next = strchr(aux, '&');
|
|
|
|
if (next)
|
|
{
|
|
do
|
|
{
|
|
*aux++ = *next++;
|
|
} while (next[-1] != 0);
|
|
}
|
|
else
|
|
*aux = 0;
|
|
}
|
|
|
|
aux = strstr(copy, "?t=");
|
|
|
|
if (!aux)
|
|
aux = strstr(copy, "&t=");
|
|
|
|
if (aux)
|
|
{
|
|
aux += 3;
|
|
next = strchr(aux, '&');
|
|
|
|
if (next)
|
|
{
|
|
do
|
|
{
|
|
*aux++ = *next++;
|
|
} while (next[-1] != 0);
|
|
}
|
|
else
|
|
*aux = 0;
|
|
}
|
|
|
|
CHEEVOS_LOG(format, copy);
|
|
#endif
|
|
#else
|
|
(void)format;
|
|
(void)url;
|
|
#endif
|
|
}
|
|
|
|
static retro_time_t rcheevos_async_send_rich_presence(rcheevos_async_io_request* request);
|
|
static void rcheevos_async_award_achievement(rcheevos_async_io_request* request);
|
|
|
|
static void rcheevos_async_task_handler(retro_task_t* task)
|
|
{
|
|
rcheevos_async_io_request* request = (rcheevos_async_io_request*)task->user_data;
|
|
|
|
switch (request->type)
|
|
{
|
|
case CHEEVOS_ASYNC_RICHPRESENCE:
|
|
if (request->id == (int)rcheevos_locals.patchdata.game_id)
|
|
{
|
|
/* update the task to fire again in two minutes */
|
|
task->when = rcheevos_async_send_rich_presence(request);
|
|
}
|
|
else
|
|
{
|
|
/* game changed; stop the recurring task - a new one will be scheduled for the next game */
|
|
task_set_finished(task, 1);
|
|
free(request);
|
|
}
|
|
break;
|
|
|
|
case CHEEVOS_ASYNC_AWARD_ACHIEVEMENT:
|
|
rcheevos_async_award_achievement(request);
|
|
task_set_finished(task, 1);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void rcheevos_async_schedule(rcheevos_async_io_request* request, retro_time_t delay)
|
|
{
|
|
retro_task_t* task = task_init();
|
|
task->when = cpu_features_get_time_usec() + delay;
|
|
task->handler = rcheevos_async_task_handler;
|
|
task->user_data = request;
|
|
task->progress = -1;
|
|
task_queue_push(task);
|
|
}
|
|
|
|
static const char* rcheevos_rc_error(int ret)
|
|
{
|
|
switch (ret)
|
|
{
|
|
case RC_OK: return "Ok";
|
|
case RC_INVALID_LUA_OPERAND: return "Invalid Lua operand";
|
|
case RC_INVALID_MEMORY_OPERAND: return "Invalid memory operand";
|
|
case RC_INVALID_CONST_OPERAND: return "Invalid constant operand";
|
|
case RC_INVALID_FP_OPERAND: return "Invalid floating-point operand";
|
|
case RC_INVALID_CONDITION_TYPE: return "Invalid condition type";
|
|
case RC_INVALID_OPERATOR: return "Invalid operator";
|
|
case RC_INVALID_REQUIRED_HITS: return "Invalid required hits";
|
|
case RC_DUPLICATED_START: return "Duplicated start condition";
|
|
case RC_DUPLICATED_CANCEL: return "Duplicated cancel condition";
|
|
case RC_DUPLICATED_SUBMIT: return "Duplicated submit condition";
|
|
case RC_DUPLICATED_VALUE: return "Duplicated value expression";
|
|
case RC_DUPLICATED_PROGRESS: return "Duplicated progress expression";
|
|
case RC_MISSING_START: return "Missing start condition";
|
|
case RC_MISSING_CANCEL: return "Missing cancel condition";
|
|
case RC_MISSING_SUBMIT: return "Missing submit condition";
|
|
case RC_MISSING_VALUE: return "Missing value expression";
|
|
case RC_INVALID_LBOARD_FIELD: return "Invalid field in leaderboard";
|
|
case RC_MISSING_DISPLAY_STRING: return "Missing display string";
|
|
case RC_OUT_OF_MEMORY: return "Out of memory";
|
|
case RC_INVALID_VALUE_FLAG: return "Invalid flag in value expression";
|
|
case RC_MISSING_VALUE_MEASURED: return "Missing measured flag in value expression";
|
|
case RC_MULTIPLE_MEASURED: return "Multiple measured targets";
|
|
case RC_INVALID_MEASURED_TARGET: return "Invalid measured target";
|
|
|
|
default: return "Unknown error";
|
|
}
|
|
}
|
|
|
|
static int rcheevos_parse(const char* json)
|
|
{
|
|
char buffer[256];
|
|
settings_t *settings = config_get_ptr();
|
|
int res = 0;
|
|
int i = 0;
|
|
unsigned j = 0;
|
|
unsigned count = 0;
|
|
rcheevos_cheevo_t* cheevo = NULL;
|
|
rcheevos_lboard_t* lboard = NULL;
|
|
rcheevos_racheevo_t* rac = NULL;
|
|
|
|
rcheevos_fixup_init(&rcheevos_locals.fixups);
|
|
|
|
res = rcheevos_get_patchdata(json, &rcheevos_locals.patchdata);
|
|
|
|
if (res != 0)
|
|
{
|
|
RARCH_ERR(RCHEEVOS_TAG "Error parsing cheevos");
|
|
return -1;
|
|
}
|
|
|
|
if ( rcheevos_locals.patchdata.core_count == 0
|
|
&& rcheevos_locals.patchdata.unofficial_count == 0
|
|
&& rcheevos_locals.patchdata.lboard_count == 0)
|
|
{
|
|
rcheevos_locals.core = NULL;
|
|
rcheevos_locals.unofficial = NULL;
|
|
rcheevos_locals.lboards = NULL;
|
|
rcheevos_locals.richpresence.richpresence = NULL;
|
|
rcheevos_free_patchdata(&rcheevos_locals.patchdata);
|
|
return 0;
|
|
}
|
|
|
|
/* Achievement memory accesses are 0-based, regardless of where the memory is accessed by the
|
|
* emulated code. As such, address 0 should always be accessible and serves as an indicator that
|
|
* other addresses will also be accessible. Individual achievements will be "Unsupported" if
|
|
* they contain addresses that cannot be resolved. This check gives the user immediate feedback
|
|
* if the core they're trying to use will disable all achievements as "Unsupported".
|
|
*/
|
|
if (!rcheevos_patch_address(0, rcheevos_locals.patchdata.console_id))
|
|
{
|
|
int delay_judgment = 0;
|
|
|
|
rarch_system_info_t* system = runloop_get_system_info();
|
|
if (system->mmaps.num_descriptors == 0)
|
|
{
|
|
/* Special case: the mupen64plus-nx core doesn't initialize the RAM immediately. To avoid a race
|
|
* condition - if the core says there's SYSTEM_RAM, but the pointer is NULL, proceed. If the memory
|
|
* isn't exposed when the achievements start processing, they'll be marked "Unsupported" individually.
|
|
*/
|
|
retro_ctx_memory_info_t meminfo;
|
|
meminfo.id = RETRO_MEMORY_SYSTEM_RAM;
|
|
core_get_memory(&meminfo);
|
|
|
|
delay_judgment |= (meminfo.size > 0);
|
|
}
|
|
else
|
|
{
|
|
/* Special case: the sameboy core exposes the RAM at $8000, but not the ROM at $0000. NES and
|
|
* Gameboy achievements do attempt to map the entire bus, and it's unlikely that an achievement
|
|
* will reference the ROM data, so if the RAM is still present, allow the core to load. If any
|
|
* achievements do reference the ROM data, they'll be marked "Unsupported" individually.
|
|
*/
|
|
delay_judgment |= (rcheevos_patch_address(0x8000, rcheevos_locals.patchdata.console_id) != NULL);
|
|
}
|
|
|
|
if (!delay_judgment)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "No memory exposed by core\n");
|
|
|
|
if (settings->bools.cheevos_verbose_enable)
|
|
runloop_msg_queue_push("Cannot activate achievements using this core.", 0, 4 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_WARNING);
|
|
|
|
goto error;
|
|
}
|
|
}
|
|
|
|
/* Allocate memory. */
|
|
rcheevos_locals.core = (rcheevos_cheevo_t*)
|
|
calloc(rcheevos_locals.patchdata.core_count, sizeof(rcheevos_cheevo_t));
|
|
|
|
rcheevos_locals.unofficial = (rcheevos_cheevo_t*)
|
|
calloc(rcheevos_locals.patchdata.unofficial_count, sizeof(rcheevos_cheevo_t));
|
|
|
|
rcheevos_locals.lboards = (rcheevos_lboard_t*)
|
|
calloc(rcheevos_locals.patchdata.lboard_count, sizeof(rcheevos_lboard_t));
|
|
|
|
if ( !rcheevos_locals.core
|
|
|| !rcheevos_locals.unofficial
|
|
|| !rcheevos_locals.lboards)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "Error allocating memory for cheevos");
|
|
goto error;
|
|
}
|
|
|
|
/* Initialize. */
|
|
for (i = 0; i < 2; i++)
|
|
{
|
|
if (i == 0)
|
|
{
|
|
cheevo = rcheevos_locals.core;
|
|
rac = rcheevos_locals.patchdata.core;
|
|
count = rcheevos_locals.patchdata.core_count;
|
|
}
|
|
else
|
|
{
|
|
cheevo = rcheevos_locals.unofficial;
|
|
rac = rcheevos_locals.patchdata.unofficial;
|
|
count = rcheevos_locals.patchdata.unofficial_count;
|
|
}
|
|
|
|
for (j = 0; j < count; j++, cheevo++, rac++)
|
|
{
|
|
cheevo->info = rac;
|
|
res = rc_trigger_size(cheevo->info->memaddr);
|
|
|
|
if (res < 0)
|
|
{
|
|
snprintf(buffer, sizeof(buffer), "Error in achievement %d \"%s\": %s",
|
|
cheevo->info->id, cheevo->info->title, rcheevos_rc_error(res));
|
|
|
|
if (settings->bools.cheevos_verbose_enable)
|
|
runloop_msg_queue_push(buffer, 0, 4 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "%s: mem %s\n", buffer, cheevo->info->memaddr);
|
|
cheevo->trigger = NULL;
|
|
cheevo->active = 0;
|
|
cheevo->last = 1;
|
|
continue;
|
|
}
|
|
|
|
cheevo->trigger = (rc_trigger_t*)calloc(1, res);
|
|
|
|
if (!cheevo->trigger)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "Error allocating memory for cheevos");
|
|
goto error;
|
|
}
|
|
|
|
rc_parse_trigger(cheevo->trigger, cheevo->info->memaddr, NULL, 0);
|
|
cheevo->active = RCHEEVOS_ACTIVE_SOFTCORE | RCHEEVOS_ACTIVE_HARDCORE;
|
|
cheevo->last = 1;
|
|
}
|
|
}
|
|
|
|
lboard = rcheevos_locals.lboards;
|
|
count = rcheevos_locals.patchdata.lboard_count;
|
|
|
|
for (j = 0; j < count; j++, lboard++)
|
|
{
|
|
lboard->info = rcheevos_locals.patchdata.lboards + j;
|
|
res = rc_lboard_size(lboard->info->mem);
|
|
|
|
if (res < 0)
|
|
{
|
|
snprintf(buffer, sizeof(buffer), "Error in leaderboard %d \"%s\": %s",
|
|
lboard->info->id, lboard->info->title, rcheevos_rc_error(res));
|
|
|
|
if (settings->bools.cheevos_verbose_enable)
|
|
runloop_msg_queue_push(buffer, 0, 4 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "%s mem: %s\n", buffer, lboard->info->mem);
|
|
lboard->lboard = NULL;
|
|
continue;
|
|
}
|
|
|
|
lboard->lboard = (rc_lboard_t*)calloc(1, res);
|
|
|
|
if (!lboard->lboard)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "Error allocating memory for cheevos");
|
|
goto error;
|
|
}
|
|
|
|
rc_parse_lboard(lboard->lboard,
|
|
lboard->info->mem, NULL, 0);
|
|
lboard->active = false;
|
|
lboard->last_value = 0;
|
|
lboard->format = rc_parse_format(lboard->info->format);
|
|
}
|
|
|
|
if (rcheevos_locals.patchdata.richpresence_script)
|
|
{
|
|
int buffer_size = rc_richpresence_size(rcheevos_locals.patchdata.richpresence_script);
|
|
if (buffer_size <= 0)
|
|
{
|
|
snprintf(buffer, sizeof(buffer), "Error in rich presence: %s", rcheevos_rc_error(buffer_size));
|
|
|
|
if (settings->bools.cheevos_verbose_enable)
|
|
runloop_msg_queue_push(buffer, 0, 4 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "%s\n", buffer);
|
|
rcheevos_locals.richpresence.richpresence = NULL;
|
|
}
|
|
else
|
|
{
|
|
char *buffer = (char*)malloc(buffer_size);
|
|
rcheevos_locals.richpresence.richpresence = rc_parse_richpresence(buffer, rcheevos_locals.patchdata.richpresence_script, NULL, 0);
|
|
}
|
|
|
|
rcheevos_locals.richpresence.evaluation[0] = '\0';
|
|
}
|
|
|
|
if (!rcheevos_locals.richpresence.richpresence && rcheevos_locals.patchdata.title)
|
|
{
|
|
snprintf(rcheevos_locals.richpresence.evaluation, sizeof(rcheevos_locals.richpresence.evaluation),
|
|
"Playing %s", rcheevos_locals.patchdata.title);
|
|
}
|
|
|
|
/* schedule the first rich presence call in 30 seconds */
|
|
{
|
|
rcheevos_async_io_request* request = (rcheevos_async_io_request*)calloc(1, sizeof(rcheevos_async_io_request));
|
|
request->id = rcheevos_locals.patchdata.game_id;
|
|
request->type = CHEEVOS_ASYNC_RICHPRESENCE;
|
|
rcheevos_async_schedule(request, CHEEVOS_PING_FREQUENCY / 4);
|
|
}
|
|
|
|
return 0;
|
|
|
|
error:
|
|
CHEEVOS_FREE(rcheevos_locals.core);
|
|
CHEEVOS_FREE(rcheevos_locals.unofficial);
|
|
CHEEVOS_FREE(rcheevos_locals.lboards);
|
|
rcheevos_free_patchdata(&rcheevos_locals.patchdata);
|
|
rcheevos_fixup_destroy(&rcheevos_locals.fixups);
|
|
return -1;
|
|
}
|
|
|
|
/*****************************************************************************
|
|
Test all the achievements (call once per frame).
|
|
*****************************************************************************/
|
|
|
|
static void rcheevos_award_task_callback(retro_task_t* task, void* task_data, void* user_data, const char* error)
|
|
{
|
|
rcheevos_async_io_request* request = (rcheevos_async_io_request*)user_data;
|
|
|
|
if (!error)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Awarded achievement %u\n", request->id);
|
|
free(request);
|
|
}
|
|
else
|
|
{
|
|
/* double the wait between each attempt until we hit a maximum delay of two minutes
|
|
* 250ms -> 500ms -> 1s -> 2s -> 4s -> 8s -> 16s -> 32s -> 64s -> 120s -> 120s... */
|
|
retro_time_t retry_delay = (request->attempt_count > 8) ? (120 * 1000 * 1000) : ((250 * 1000) << request->attempt_count);
|
|
|
|
request->attempt_count++;
|
|
rcheevos_async_schedule(request, retry_delay);
|
|
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "Error awarding achievement %u: %s\n", request->id, error);
|
|
}
|
|
}
|
|
|
|
static void rcheevos_async_award_achievement(rcheevos_async_io_request* request)
|
|
{
|
|
char buffer[256];
|
|
settings_t *settings = config_get_ptr();
|
|
int ret = rc_url_award_cheevo(buffer, sizeof(buffer), settings->arrays.cheevos_username, rcheevos_locals.token, request->id, request->hardcore);
|
|
|
|
if (ret != 0)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "Buffer too small to create URL\n");
|
|
free(request);
|
|
return;
|
|
}
|
|
|
|
rcheevos_log_url(RCHEEVOS_TAG "rc_url_award_cheevo: %s\n", buffer);
|
|
task_push_http_transfer_with_user_agent(buffer, true, NULL, request->user_agent, rcheevos_award_task_callback, request);
|
|
}
|
|
|
|
static void rcheevos_award(rcheevos_cheevo_t* cheevo, int mode)
|
|
{
|
|
char buffer[256];
|
|
settings_t *settings = config_get_ptr();
|
|
buffer[0] = 0;
|
|
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "awarding cheevo %u: %s (%s)\n",
|
|
cheevo->info->id, cheevo->info->title, cheevo->info->description);
|
|
|
|
/* Deactivates the cheevo. */
|
|
cheevo->active &= ~mode;
|
|
|
|
if (mode == RCHEEVOS_ACTIVE_HARDCORE)
|
|
cheevo->active &= ~RCHEEVOS_ACTIVE_SOFTCORE;
|
|
|
|
/* Show the OSD message. */
|
|
{
|
|
#if defined(HAVE_GFX_WIDGETS)
|
|
bool widgets_ready = gfx_widgets_ready();
|
|
if (widgets_ready)
|
|
gfx_widgets_push_achievement(cheevo->info->title, cheevo->info->badge);
|
|
else
|
|
#endif
|
|
{
|
|
snprintf(buffer, sizeof(buffer), "Achievement Unlocked: %s", cheevo->info->title);
|
|
runloop_msg_queue_push(buffer, 0, 2 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
runloop_msg_queue_push(cheevo->info->description, 0, 3 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
}
|
|
}
|
|
|
|
/* Start the award task. */
|
|
{
|
|
rcheevos_async_io_request* request = (rcheevos_async_io_request*)calloc(1, sizeof(rcheevos_async_io_request));
|
|
request->type = CHEEVOS_ASYNC_AWARD_ACHIEVEMENT;
|
|
request->id = cheevo->info->id;
|
|
request->hardcore = ((mode & RCHEEVOS_ACTIVE_HARDCORE) != 0) ? 1 : 0;
|
|
rcheevos_get_user_agent(request->user_agent);
|
|
rcheevos_async_award_achievement(request);
|
|
}
|
|
|
|
/* Take a screenshot of the achievement. */
|
|
if (settings && settings->bools.cheevos_auto_screenshot)
|
|
{
|
|
char shotname[8192];
|
|
|
|
snprintf(shotname, sizeof(shotname), "%s/%s-cheevo-%u",
|
|
settings->paths.directory_screenshot,
|
|
path_basename(path_get(RARCH_PATH_BASENAME)),
|
|
cheevo->info->id);
|
|
shotname[sizeof(shotname) - 1] = '\0';
|
|
|
|
if (take_screenshot(settings->paths.directory_screenshot,
|
|
shotname, true,
|
|
video_driver_cached_frame_has_valid_framebuffer(), false, true))
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "got a screenshot for cheevo %u\n", cheevo->info->id);
|
|
else
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "failed to get screenshot for cheevo %u\n", cheevo->info->id);
|
|
}
|
|
}
|
|
|
|
static unsigned rcheevos_peek(unsigned address, unsigned num_bytes, void* ud)
|
|
{
|
|
const uint8_t* data = rcheevos_fixup_find(&rcheevos_locals.fixups,
|
|
address, rcheevos_locals.patchdata.console_id);
|
|
unsigned value = 0;
|
|
|
|
if (data)
|
|
{
|
|
switch (num_bytes)
|
|
{
|
|
case 4: value |= data[2] << 16 | data[3] << 24;
|
|
case 2: value |= data[1] << 8;
|
|
case 1: value |= data[0];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
rcheevos_locals.invalid_peek_address = true;
|
|
}
|
|
|
|
return value;
|
|
}
|
|
|
|
static int rcheevos_has_indirect_memref(const rc_memref_value_t* memrefs)
|
|
{
|
|
const rc_memref_value_t* memref = memrefs;
|
|
while (memref != NULL)
|
|
{
|
|
if (memref->memref.is_indirect)
|
|
return 1;
|
|
|
|
memref = memref->next;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static void rcheevos_test_cheevo_set(bool official)
|
|
{
|
|
settings_t *settings = config_get_ptr();
|
|
int mode = RCHEEVOS_ACTIVE_SOFTCORE;
|
|
rcheevos_cheevo_t* cheevo;
|
|
int i, count;
|
|
|
|
if (settings && settings->bools.cheevos_hardcore_mode_enable && !rcheevos_hardcore_paused)
|
|
mode = RCHEEVOS_ACTIVE_HARDCORE;
|
|
|
|
if (official)
|
|
{
|
|
cheevo = rcheevos_locals.core;
|
|
count = rcheevos_locals.patchdata.core_count;
|
|
}
|
|
else
|
|
{
|
|
cheevo = rcheevos_locals.unofficial;
|
|
count = rcheevos_locals.patchdata.unofficial_count;
|
|
}
|
|
|
|
rcheevos_locals.invalid_peek_address = false;
|
|
|
|
for (i = 0; i < count; i++, cheevo++)
|
|
{
|
|
/* Check if the achievement is active for the current mode. */
|
|
if (cheevo->active & mode)
|
|
{
|
|
int valid = rc_test_trigger(cheevo->trigger, rcheevos_peek, NULL, NULL);
|
|
|
|
/* trigger must be false for at least one frame before it can trigger. if last is true, the trigger hasn't yet been false. */
|
|
if (cheevo->last)
|
|
{
|
|
/* if the we're still waiting for the trigger to stabilize, check to see if an error occurred */
|
|
if (rcheevos_locals.invalid_peek_address)
|
|
{
|
|
/* reset the flag for the next achievement */
|
|
rcheevos_locals.invalid_peek_address = false;
|
|
|
|
if (rcheevos_has_indirect_memref(cheevo->trigger->memrefs))
|
|
{
|
|
/* ignore bad addresses possibly generated by AddAddress */
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Ignoring invalid address in achievement with AddAddress: %s\n", cheevo->info->title);
|
|
}
|
|
else
|
|
{
|
|
/* could not map one or more addresses - disable the achievement */
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "Achievement disabled (invalid address): %s\n", cheevo->info->title);
|
|
cheevo->active = 0;
|
|
|
|
/* clear out the trigger so it shows up as 'Unsupported' in the menu */
|
|
CHEEVOS_FREE(cheevo->trigger);
|
|
cheevo->trigger = NULL;
|
|
|
|
continue;
|
|
}
|
|
}
|
|
|
|
/* no error, reset any hit counts for the next check */
|
|
rc_reset_trigger(cheevo->trigger);
|
|
}
|
|
else if (valid)
|
|
rcheevos_award(cheevo, mode);
|
|
|
|
cheevo->last = valid;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void rcheevos_lboard_submit_task(retro_task_t *task, void* task_data, void* user_data,
|
|
const char* error)
|
|
{
|
|
int ret;
|
|
MD5_CTX ctx;
|
|
uint8_t hash[16];
|
|
char signature[64];
|
|
char buffer[256];
|
|
char user_agent[256];
|
|
const rcheevos_lboard_t* lboard = (const rcheevos_lboard_t*)user_data;
|
|
settings_t *settings = config_get_ptr();
|
|
const char *cheevos_username = settings->arrays.cheevos_username;
|
|
|
|
if (!error)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Submitted leaderboard %u\n", lboard->info->id);
|
|
return;
|
|
}
|
|
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "Error submitting leaderboard %u: %s\n", lboard->info->id, error);
|
|
|
|
/* Try again. */
|
|
|
|
/* Evaluate the signature. */
|
|
snprintf(signature, sizeof(signature), "%u%s%u", lboard->info->id,
|
|
cheevos_username, lboard->info->id);
|
|
|
|
MD5_Init(&ctx);
|
|
MD5_Update(&ctx, (void*)signature, strlen(signature));
|
|
MD5_Final(hash, &ctx);
|
|
|
|
/* Start the request. */
|
|
ret = rc_url_submit_lboard(buffer, sizeof(buffer), cheevos_username,
|
|
rcheevos_locals.token, lboard->info->id, lboard->last_value, hash);
|
|
|
|
if (ret != 0)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "Buffer to small to create URL\n");
|
|
return;
|
|
}
|
|
|
|
rcheevos_get_user_agent(user_agent);
|
|
|
|
rcheevos_log_url(RCHEEVOS_TAG "rc_url_submit_lboard: %s\n", buffer);
|
|
task_push_http_transfer_with_user_agent(buffer, true, NULL, user_agent, rcheevos_lboard_submit_task, user_data);
|
|
}
|
|
|
|
static void rcheevos_lboard_submit(rcheevos_lboard_t* lboard)
|
|
{
|
|
char buffer[256];
|
|
char value[16];
|
|
|
|
/* Deactivate the leaderboard. */
|
|
lboard->active = 0;
|
|
|
|
/* Failsafe for improper leaderboards. */
|
|
if (lboard->last_value == 0)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "Leaderboard %s tried to submit 0\n", lboard->info->title);
|
|
runloop_msg_queue_push("Leaderboard attempt cancelled!", 0, 2 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
return;
|
|
}
|
|
|
|
/* Show the OSD message. */
|
|
rc_format_value(value, sizeof(value), lboard->last_value, lboard->format);
|
|
|
|
snprintf(buffer, sizeof(buffer), "Submitted %s for %s",
|
|
value, lboard->info->title);
|
|
runloop_msg_queue_push(buffer, 0, 2 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
|
|
/* Start the submit task. */
|
|
rcheevos_lboard_submit_task(NULL, NULL, lboard, "no error, first try");
|
|
}
|
|
|
|
static void rcheevos_test_leaderboards(void)
|
|
{
|
|
rcheevos_lboard_t* lboard = rcheevos_locals.lboards;
|
|
unsigned i;
|
|
|
|
rcheevos_locals.invalid_peek_address = false;
|
|
|
|
for (i = 0; i < rcheevos_locals.patchdata.lboard_count; i++, lboard++)
|
|
{
|
|
if (!lboard->lboard)
|
|
continue;
|
|
|
|
switch (rc_evaluate_lboard(lboard->lboard, &lboard->last_value, rcheevos_peek, NULL, NULL))
|
|
{
|
|
default:
|
|
case RC_LBOARD_INACTIVE:
|
|
break;
|
|
|
|
case RC_LBOARD_ACTIVE:
|
|
/* this is where we would update the onscreen tracker */
|
|
break;
|
|
|
|
case RC_LBOARD_TRIGGERED:
|
|
rcheevos_lboard_submit(lboard);
|
|
break;
|
|
|
|
case RC_LBOARD_CANCELED:
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Cancel leaderboard %s\n", lboard->info->title);
|
|
lboard->active = 0;
|
|
runloop_msg_queue_push("Leaderboard attempt cancelled!",
|
|
0, 2 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
break;
|
|
}
|
|
|
|
case RC_LBOARD_STARTED:
|
|
{
|
|
char buffer[256];
|
|
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Leaderboard started: %s\n", lboard->info->title);
|
|
lboard->active = 1;
|
|
|
|
snprintf(buffer, sizeof(buffer),
|
|
"Leaderboard Active: %s", lboard->info->title);
|
|
runloop_msg_queue_push(buffer, 0, 2 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
runloop_msg_queue_push(lboard->info->description, 0, 3 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (rcheevos_locals.invalid_peek_address)
|
|
{
|
|
/* reset the flag for the next leaderboard */
|
|
rcheevos_locals.invalid_peek_address = false;
|
|
|
|
if (!rcheevos_has_indirect_memref(lboard->lboard->memrefs))
|
|
{
|
|
/* disable the leaderboard */
|
|
CHEEVOS_FREE(lboard->lboard);
|
|
lboard->lboard = NULL;
|
|
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Leaderboard disabled (invalid address): %s\n", lboard->info->title);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
const char* rcheevos_get_richpresence(void)
|
|
{
|
|
if (!rcheevos_locals.richpresence.richpresence)
|
|
return NULL;
|
|
else
|
|
return rcheevos_locals.richpresence.evaluation;
|
|
}
|
|
|
|
static retro_time_t rcheevos_async_send_rich_presence(rcheevos_async_io_request* request)
|
|
{
|
|
settings_t *settings = config_get_ptr();
|
|
const char *cheevos_username = settings->arrays.cheevos_username;
|
|
bool cheevos_richpresence_enable = settings->bools.cheevos_richpresence_enable;
|
|
|
|
if (cheevos_richpresence_enable && rcheevos_locals.richpresence.richpresence)
|
|
{
|
|
rc_evaluate_richpresence(rcheevos_locals.richpresence.richpresence,
|
|
rcheevos_locals.richpresence.evaluation,
|
|
sizeof(rcheevos_locals.richpresence.evaluation), rcheevos_peek, NULL, NULL);
|
|
}
|
|
|
|
{
|
|
char url[256], post_data[1024];
|
|
|
|
snprintf(url, sizeof(url),
|
|
"http://retroachievements.org/dorequest.php?r=ping&u=%s&t=%s",
|
|
cheevos_username, rcheevos_locals.token);
|
|
|
|
if (rcheevos_locals.richpresence.evaluation[0])
|
|
{
|
|
char* tmp = NULL;
|
|
net_http_urlencode(&tmp, rcheevos_locals.richpresence.evaluation);
|
|
snprintf(post_data, sizeof(post_data), "g=%u&m=%s", rcheevos_locals.patchdata.game_id, tmp);
|
|
CHEEVOS_FREE(tmp);
|
|
|
|
#ifdef HAVE_DISCORD
|
|
if (settings->bools.discord_enable)
|
|
discord_update(DISCORD_PRESENCE_RETROACHIEVEMENTS, false);
|
|
#endif
|
|
}
|
|
else
|
|
snprintf(post_data, sizeof(post_data), "g=%u", rcheevos_locals.patchdata.game_id);
|
|
|
|
rcheevos_get_user_agent(request->user_agent);
|
|
task_push_http_post_transfer_with_user_agent(url, post_data, true, "POST", request->user_agent, NULL, NULL);
|
|
}
|
|
|
|
/* Update rich presence every two minutes */
|
|
if (settings->bools.cheevos_richpresence_enable)
|
|
return cpu_features_get_time_usec() + CHEEVOS_PING_FREQUENCY;
|
|
|
|
/* Send ping every four minutes */
|
|
return cpu_features_get_time_usec() + CHEEVOS_PING_FREQUENCY * 2;
|
|
}
|
|
|
|
void rcheevos_reset_game(void)
|
|
{
|
|
unsigned i;
|
|
rcheevos_lboard_t* lboard;
|
|
rcheevos_cheevo_t* cheevo = rcheevos_locals.core;
|
|
|
|
for (i = 0; i < rcheevos_locals.patchdata.core_count; i++, cheevo++)
|
|
{
|
|
if (cheevo->trigger)
|
|
rc_reset_trigger(cheevo->trigger);
|
|
cheevo->last = 1;
|
|
}
|
|
|
|
cheevo = rcheevos_locals.unofficial;
|
|
|
|
for (i = 0; i < rcheevos_locals.patchdata.unofficial_count;
|
|
i++, cheevo++)
|
|
{
|
|
if (cheevo->trigger)
|
|
rc_reset_trigger(cheevo->trigger);
|
|
cheevo->last = 1;
|
|
}
|
|
|
|
lboard = rcheevos_locals.lboards;
|
|
for (i = 0; i < rcheevos_locals.patchdata.lboard_count;
|
|
i++, lboard++)
|
|
{
|
|
if (lboard->lboard)
|
|
rc_reset_lboard(lboard->lboard);
|
|
|
|
if (lboard->active)
|
|
{
|
|
lboard->active = 0;
|
|
|
|
/* This ensures the leaderboard won't restart
|
|
* until the start trigger is false for at
|
|
* least one frame */
|
|
if (lboard->lboard)
|
|
lboard->lboard->submitted = 1;
|
|
}
|
|
}
|
|
|
|
rcheevos_locals.richpresence.last_update = cpu_features_get_time_usec();
|
|
}
|
|
|
|
#ifdef HAVE_MENU
|
|
static void rcheevos_append_menu_achievement(menu_displaylist_info_t* info, size_t idx, enum msg_hash_enums enum_idx, rcheevos_cheevo_t* cheevo)
|
|
{
|
|
bool active = false;
|
|
|
|
if (!cheevo->trigger)
|
|
{
|
|
enum_idx = MENU_ENUM_LABEL_CHEEVOS_UNSUPPORTED_ENTRY;
|
|
active = true; /* not really, but forces the badge to appear disabled */
|
|
}
|
|
else if (!(cheevo->active & RCHEEVOS_ACTIVE_HARDCORE))
|
|
enum_idx = MENU_ENUM_LABEL_CHEEVOS_UNLOCKED_ENTRY_HARDCORE;
|
|
else if (!(cheevo->active & RCHEEVOS_ACTIVE_SOFTCORE))
|
|
enum_idx = MENU_ENUM_LABEL_CHEEVOS_UNLOCKED_ENTRY;
|
|
else
|
|
{
|
|
/* use enum passed in - either "Locked" for core or "Unofficial" for unofficial */
|
|
active = true;
|
|
}
|
|
|
|
menu_entries_append_enum(info->list, cheevo->info->title,
|
|
cheevo->info->description, enum_idx,
|
|
MENU_SETTINGS_CHEEVOS_START + idx, 0, 0);
|
|
|
|
set_badge_info(&badges_ctx, idx, cheevo->info->badge, active);
|
|
}
|
|
#endif
|
|
|
|
void rcheevos_populate_menu(void* data)
|
|
{
|
|
#ifdef HAVE_MENU
|
|
int i = 0;
|
|
int count = 0;
|
|
rcheevos_cheevo_t* cheevo = NULL;
|
|
menu_displaylist_info_t* info = (menu_displaylist_info_t*)data;
|
|
settings_t* settings = config_get_ptr();
|
|
bool cheevos_enable = settings->bools.cheevos_enable;
|
|
bool cheevos_hardcore_mode_enable = settings->bools.cheevos_hardcore_mode_enable;
|
|
bool cheevos_test_unofficial = settings->bools.cheevos_test_unofficial;
|
|
|
|
if ( cheevos_enable
|
|
&& cheevos_hardcore_mode_enable
|
|
&& rcheevos_loaded)
|
|
{
|
|
if (!rcheevos_hardcore_paused)
|
|
menu_entries_append_enum(info->list,
|
|
msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_PAUSE),
|
|
msg_hash_to_str(MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE),
|
|
MENU_ENUM_LABEL_ACHIEVEMENT_PAUSE,
|
|
MENU_SETTING_ACTION_PAUSE_ACHIEVEMENTS, 0, 0);
|
|
else
|
|
menu_entries_append_enum(info->list,
|
|
msg_hash_to_str(MENU_ENUM_LABEL_VALUE_ACHIEVEMENT_RESUME),
|
|
msg_hash_to_str(MENU_ENUM_LABEL_ACHIEVEMENT_RESUME),
|
|
MENU_ENUM_LABEL_ACHIEVEMENT_RESUME,
|
|
MENU_SETTING_ACTION_RESUME_ACHIEVEMENTS, 0, 0);
|
|
}
|
|
|
|
cheevo = rcheevos_locals.core;
|
|
for (count = rcheevos_locals.patchdata.core_count; count > 0; count--)
|
|
{
|
|
rcheevos_append_menu_achievement(info, i++, MENU_ENUM_LABEL_CHEEVOS_LOCKED_ENTRY, cheevo++);
|
|
}
|
|
|
|
if (cheevos_test_unofficial)
|
|
{
|
|
cheevo = rcheevos_locals.unofficial;
|
|
for (count = rcheevos_locals.patchdata.unofficial_count; count > 0; count--)
|
|
{
|
|
rcheevos_append_menu_achievement(info, i++, MENU_ENUM_LABEL_CHEEVOS_UNOFFICIAL_ENTRY, cheevo++);
|
|
}
|
|
}
|
|
|
|
if (i == 0)
|
|
{
|
|
menu_entries_append_enum(info->list,
|
|
msg_hash_to_str(MENU_ENUM_LABEL_VALUE_NO_ACHIEVEMENTS_TO_DISPLAY),
|
|
msg_hash_to_str(MENU_ENUM_LABEL_NO_ACHIEVEMENTS_TO_DISPLAY),
|
|
MENU_ENUM_LABEL_NO_ACHIEVEMENTS_TO_DISPLAY,
|
|
FILE_TYPE_NONE, 0, 0);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
bool rcheevos_get_description(rcheevos_ctx_desc_t* desc)
|
|
{
|
|
unsigned idx;
|
|
const rcheevos_cheevo_t* cheevo;
|
|
|
|
if (!desc)
|
|
return false;
|
|
|
|
*desc->s = 0;
|
|
|
|
if (rcheevos_loaded)
|
|
{
|
|
idx = desc->idx;
|
|
|
|
if (idx < rcheevos_locals.patchdata.core_count)
|
|
cheevo = rcheevos_locals.core + idx;
|
|
else
|
|
{
|
|
idx -= rcheevos_locals.patchdata.core_count;
|
|
|
|
if (idx < rcheevos_locals.patchdata.unofficial_count)
|
|
cheevo = rcheevos_locals.unofficial + idx;
|
|
else
|
|
return true;
|
|
}
|
|
|
|
strlcpy(desc->s, cheevo->info->description, desc->len);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool rcheevos_apply_cheats(bool* data_bool)
|
|
{
|
|
rcheevos_cheats_are_enabled = *data_bool;
|
|
rcheevos_cheats_were_enabled |= rcheevos_cheats_are_enabled;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool rcheevos_unload(void)
|
|
{
|
|
bool running = false;
|
|
unsigned i = 0, count = 0;
|
|
|
|
CHEEVOS_LOCK(rcheevos_locals.task_lock);
|
|
running = rcheevos_locals.task != NULL;
|
|
CHEEVOS_UNLOCK(rcheevos_locals.task_lock);
|
|
|
|
if (running)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Asked the load thread to terminate\n");
|
|
task_queue_cancel_task(rcheevos_locals.task);
|
|
|
|
#ifdef HAVE_THREADS
|
|
do
|
|
{
|
|
CHEEVOS_LOCK(rcheevos_locals.task_lock);
|
|
running = rcheevos_locals.task != NULL;
|
|
CHEEVOS_UNLOCK(rcheevos_locals.task_lock);
|
|
}
|
|
while (running);
|
|
#endif
|
|
}
|
|
|
|
if (rcheevos_loaded)
|
|
{
|
|
for (i = 0, count = rcheevos_locals.patchdata.core_count; i < count; i++)
|
|
{
|
|
CHEEVOS_FREE(rcheevos_locals.core[i].trigger);
|
|
}
|
|
|
|
for (i = 0, count = rcheevos_locals.patchdata.unofficial_count; i < count; i++)
|
|
{
|
|
CHEEVOS_FREE(rcheevos_locals.unofficial[i].trigger);
|
|
}
|
|
|
|
for (i = 0, count = rcheevos_locals.patchdata.lboard_count; i < count; i++)
|
|
{
|
|
CHEEVOS_FREE(rcheevos_locals.lboards[i].lboard);
|
|
}
|
|
|
|
CHEEVOS_FREE(rcheevos_locals.core);
|
|
CHEEVOS_FREE(rcheevos_locals.unofficial);
|
|
CHEEVOS_FREE(rcheevos_locals.lboards);
|
|
CHEEVOS_FREE(rcheevos_locals.richpresence.richpresence);
|
|
rcheevos_free_patchdata(&rcheevos_locals.patchdata);
|
|
rcheevos_fixup_destroy(&rcheevos_locals.fixups);
|
|
|
|
rcheevos_locals.core = NULL;
|
|
rcheevos_locals.unofficial = NULL;
|
|
rcheevos_locals.lboards = NULL;
|
|
rcheevos_locals.richpresence.richpresence = NULL;
|
|
|
|
rcheevos_loaded = false;
|
|
rcheevos_hardcore_active = false;
|
|
rcheevos_hardcore_paused = false;
|
|
rcheevos_state_loaded_flag = false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool rcheevos_toggle_hardcore_mode(void)
|
|
{
|
|
settings_t *settings = config_get_ptr();
|
|
bool cheevos_hardcore_mode_enable = settings->bools.cheevos_hardcore_mode_enable;
|
|
bool rewind_enable = settings->bools.rewind_enable;
|
|
|
|
/* reset and deinit rewind to avoid cheat the score */
|
|
if (cheevos_hardcore_mode_enable
|
|
&& !rcheevos_hardcore_paused)
|
|
{
|
|
const char *msg = msg_hash_to_str(
|
|
MSG_CHEEVOS_HARDCORE_MODE_ENABLE);
|
|
|
|
/* reset the state loaded flag in case it was set */
|
|
rcheevos_state_loaded_flag = false;
|
|
|
|
/* send reset core cmd to avoid any user
|
|
* savestate previusly loaded. */
|
|
command_event(CMD_EVENT_RESET, NULL);
|
|
|
|
if (rewind_enable)
|
|
command_event(CMD_EVENT_REWIND_DEINIT, NULL);
|
|
|
|
CHEEVOS_LOG("%s\n", msg);
|
|
runloop_msg_queue_push(msg, 0, 3 * 60, true, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
}
|
|
else
|
|
{
|
|
if (rewind_enable)
|
|
command_event(CMD_EVENT_REWIND_INIT, NULL);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void rcheevos_test(void)
|
|
{
|
|
settings_t *settings = config_get_ptr();
|
|
|
|
rcheevos_test_cheevo_set(true);
|
|
|
|
if (settings)
|
|
{
|
|
if (settings->bools.cheevos_test_unofficial)
|
|
rcheevos_test_cheevo_set(false);
|
|
|
|
if (settings->bools.cheevos_hardcore_mode_enable &&
|
|
settings->bools.cheevos_leaderboards_enable &&
|
|
!rcheevos_hardcore_paused)
|
|
rcheevos_test_leaderboards();
|
|
}
|
|
}
|
|
|
|
bool rcheevos_set_cheats(void)
|
|
{
|
|
rcheevos_cheats_were_enabled = rcheevos_cheats_are_enabled;
|
|
return true;
|
|
}
|
|
|
|
void rcheevos_set_support_cheevos(bool state)
|
|
{
|
|
rcheevos_locals.core_supports = state;
|
|
}
|
|
|
|
bool rcheevos_get_support_cheevos(void)
|
|
{
|
|
return rcheevos_locals.core_supports;
|
|
}
|
|
|
|
int rcheevos_get_console(void)
|
|
{
|
|
return rcheevos_locals.patchdata.console_id;
|
|
}
|
|
|
|
const char* rcheevos_get_hash(void)
|
|
{
|
|
return rcheevos_locals.hash;
|
|
}
|
|
|
|
static void rcheevos_unlock_cb(unsigned id, void* userdata)
|
|
{
|
|
int i = 0;
|
|
unsigned j = 0, count = 0;
|
|
rcheevos_cheevo_t* cheevo = NULL;
|
|
|
|
for (i = 0; i < 2; i++)
|
|
{
|
|
if (i == 0)
|
|
{
|
|
cheevo = rcheevos_locals.core;
|
|
count = rcheevos_locals.patchdata.core_count;
|
|
}
|
|
else
|
|
{
|
|
cheevo = rcheevos_locals.unofficial;
|
|
count = rcheevos_locals.patchdata.unofficial_count;
|
|
}
|
|
|
|
for (j = 0; j < count; j++, cheevo++)
|
|
{
|
|
if (cheevo->info->id == id)
|
|
{
|
|
#ifndef CHEEVOS_DONT_DEACTIVATE
|
|
cheevo->active &= ~*(unsigned*)userdata;
|
|
#endif
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "cheevo %u deactivated (%s): %s\n", id,
|
|
(*(unsigned*)userdata) == RCHEEVOS_ACTIVE_HARDCORE ? "hardcore" : "softcore",
|
|
cheevo->info->title);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#include "coro.h"
|
|
|
|
/* Uncomment the following two lines to debug rcheevos_iterate, this will
|
|
* disable the coroutine yielding.
|
|
*
|
|
* The code is very easy to understand. It's meant to be like BASIC:
|
|
* CORO_GOTO will jump execution to another label, CORO_GOSUB will
|
|
* call another label, and CORO_RET will return from a CORO_GOSUB.
|
|
*
|
|
* This coroutine code is inspired in a very old pure C implementation
|
|
* that runs everywhere:
|
|
*
|
|
* https://www.chiark.greenend.org.uk/~sgtatham/coroutines.html
|
|
*/
|
|
/*#undef CORO_YIELD
|
|
#define CORO_YIELD()*/
|
|
|
|
typedef struct
|
|
{
|
|
/* variables used in the co-routine */
|
|
char badge_name[16];
|
|
char url[256];
|
|
char badge_basepath[PATH_MAX_LENGTH];
|
|
char badge_fullpath[PATH_MAX_LENGTH];
|
|
unsigned char last_hash[16];
|
|
unsigned char hash[16];
|
|
unsigned ext_hash;
|
|
unsigned gameid;
|
|
unsigned i;
|
|
unsigned j;
|
|
unsigned k;
|
|
size_t bytes;
|
|
size_t count;
|
|
size_t offset;
|
|
size_t len;
|
|
size_t size;
|
|
MD5_CTX md5;
|
|
rcheevos_nes_header_t header;
|
|
retro_time_t t0;
|
|
struct retro_system_info sysinfo;
|
|
void *data;
|
|
char *json;
|
|
const char *path;
|
|
const char *ext;
|
|
intfstream_t *stream;
|
|
rcheevos_cheevo_t *cheevo;
|
|
settings_t *settings;
|
|
struct http_connection_t *conn;
|
|
struct http_t *http;
|
|
const rcheevos_cheevo_t *cheevo_end;
|
|
cdfs_track_t *track;
|
|
cdfs_file_t cdfp;
|
|
|
|
/* co-routine required fields */
|
|
CORO_FIELDS
|
|
} rcheevos_coro_t;
|
|
|
|
enum
|
|
{
|
|
/* Negative values because CORO_SUB generates positive values */
|
|
RCHEEVOS_GENERIC_MD5 = -1,
|
|
RCHEEVOS_SNES_MD5 = -2,
|
|
RCHEEVOS_LYNX_MD5 = -3,
|
|
RCHEEVOS_NES_MD5 = -4,
|
|
RCHEEVOS_PSX_MD5 = -5,
|
|
RCHEEVOS_ARCADE_MD5 = -6,
|
|
RCHEEVOS_EVAL_MD5 = -7,
|
|
RCHEEVOS_SEGACD_MD5 = -8,
|
|
RCHEEVOS_GET_GAMEID = -9,
|
|
RCHEEVOS_GET_CHEEVOS = -10,
|
|
RCHEEVOS_GET_BADGES = -11,
|
|
RCHEEVOS_LOGIN = -12,
|
|
RCHEEVOS_HTTP_GET = -13,
|
|
RCHEEVOS_DEACTIVATE = -14,
|
|
RCHEEVOS_PLAYING = -15,
|
|
RCHEEVOS_DELAY = -16,
|
|
RCHEEVOS_PCE_CD_MD5 = -17,
|
|
RCHEEVOS_NDS_MD5 = -18,
|
|
RCHEEVOS_BUFFER_FILE = -19
|
|
};
|
|
|
|
static int rcheevos_prepare_hash_psx(rcheevos_coro_t* coro)
|
|
{
|
|
char buffer[2048];
|
|
char exe_name_buffer[64];
|
|
size_t exe_name_size;
|
|
const char* exe_name = NULL;
|
|
char* scan = NULL;
|
|
int success = 0;
|
|
size_t to_read = 0;
|
|
|
|
/* find the data track - it should be the first one */
|
|
coro->track = cdfs_open_data_track(coro->path);
|
|
|
|
if (!coro->track)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "could not open CD\n");
|
|
return false;
|
|
}
|
|
|
|
/* open the SYSTEM.CNF file and find the BOOT= record */
|
|
if (cdfs_open_file(&coro->cdfp, coro->track, "SYSTEM.CNF"))
|
|
{
|
|
cdfs_read_file(&coro->cdfp, buffer, sizeof(buffer));
|
|
|
|
for (scan = buffer; scan < &buffer[sizeof(buffer)] && *scan; ++scan)
|
|
{
|
|
if (strncmp(scan, "BOOT", 4) == 0)
|
|
{
|
|
exe_name = scan + 4;
|
|
while (isspace(*exe_name))
|
|
++exe_name;
|
|
|
|
if (*exe_name == '=')
|
|
{
|
|
++exe_name;
|
|
while (isspace(*exe_name))
|
|
++exe_name;
|
|
|
|
if (strncmp(exe_name, "cdrom:", 6) == 0)
|
|
exe_name += 6;
|
|
if (*exe_name == '\\')
|
|
++exe_name;
|
|
break;
|
|
}
|
|
}
|
|
|
|
while (*scan && *scan != '\n')
|
|
++scan;
|
|
}
|
|
|
|
cdfs_close_file(&coro->cdfp);
|
|
|
|
if (exe_name)
|
|
{
|
|
scan = (char*)exe_name;
|
|
while (!isspace(*scan) && *scan != ';')
|
|
++scan;
|
|
*scan = '\0';
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* no SYSTEM.CNF, check for a PSX.EXE */
|
|
exe_name = "PSX.EXE";
|
|
}
|
|
|
|
if (!exe_name || !cdfs_open_file(&coro->cdfp, coro->track, exe_name))
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "could not locate primary executable\n");
|
|
}
|
|
else
|
|
{
|
|
/* store the exe name, we're about to overwrite buffer */
|
|
strlcpy(exe_name_buffer, exe_name, sizeof(exe_name_buffer));
|
|
exe_name_buffer[sizeof(exe_name_buffer) - 1] = '\0';
|
|
exe_name_size = strlen(exe_name_buffer);
|
|
|
|
/* read the first sector of the executable */
|
|
cdfs_read_file(&coro->cdfp, buffer, sizeof(buffer));
|
|
|
|
/* the PSX-E header specifies the executable size as a 4-byte value 28 bytes into the header, which doesn't
|
|
* include the header itself. We want to include the header in the hash, so append another 2048 to that value.
|
|
* ASSERT: this results in the same value as coro->cdfp->size */
|
|
coro->count = 2048 + (((uint8_t)buffer[28 + 3] << 24) | ((uint8_t)buffer[28 + 2] << 16) |
|
|
((uint8_t)buffer[28 + 1] << 8) | (uint8_t)buffer[28]);
|
|
|
|
if (coro->count <= CHEEVOS_MB(16)) /* sanity check */
|
|
{
|
|
/* there's a few games that use a singular engine and only differ via their data files.
|
|
* luckily, they have unique serial numbers, and use the serial number as the boot file in the
|
|
* standard way. include the boot executable name in the hash */
|
|
coro->count += exe_name_size;
|
|
|
|
free(coro->data);
|
|
coro->data = (uint8_t*)malloc(coro->count);
|
|
memcpy(coro->data, exe_name_buffer, exe_name_size);
|
|
coro->len = exe_name_size;
|
|
|
|
memcpy((uint8_t*)coro->data + coro->len, buffer, sizeof(buffer));
|
|
coro->len += sizeof(buffer);
|
|
|
|
while (coro->len < coro->count)
|
|
{
|
|
to_read = coro->count - coro->len;
|
|
if (to_read > 2048)
|
|
to_read = 2048;
|
|
|
|
cdfs_read_file(&coro->cdfp, (uint8_t*)coro->data + coro->len, to_read);
|
|
|
|
coro->len += to_read;
|
|
};
|
|
|
|
success = 1;
|
|
}
|
|
|
|
cdfs_close_file(&coro->cdfp);
|
|
}
|
|
|
|
cdfs_close_track(coro->track);
|
|
coro->track = NULL;
|
|
|
|
return success;
|
|
}
|
|
|
|
static int rcheevos_prepare_hash_nintendo_ds(rcheevos_coro_t* coro)
|
|
{
|
|
unsigned char header[512];
|
|
int success = 0;
|
|
intfstream_t *stream = intfstream_open_file(
|
|
coro->path, RETRO_VFS_FILE_ACCESS_READ,
|
|
RETRO_VFS_FILE_ACCESS_HINT_NONE);
|
|
|
|
if (stream)
|
|
{
|
|
if (intfstream_read(stream, header, sizeof(header)) == 512)
|
|
{
|
|
unsigned int hash_size, arm9_size, arm9_addr, arm7_size, arm7_addr, icon_addr;
|
|
int offset = 0;
|
|
|
|
if (header[0] == 0x2E && header[1] == 0x00 && header[2] == 0x00 && header[3] == 0xEA &&
|
|
header[0xB0] == 0x44 && header[0xB1] == 0x46 && header[0xB2] == 0x96 && header[0xB3] == 0x00)
|
|
{
|
|
/* SuperCard header detected, ignore it */
|
|
offset = 512;
|
|
intfstream_seek(stream, offset, RETRO_VFS_SEEK_POSITION_START);
|
|
intfstream_read(stream, header, sizeof(header));
|
|
}
|
|
|
|
arm9_addr = header[0x20] | (header[0x21] << 8) | (header[0x22] << 16) | (header[0x23] << 24);
|
|
arm9_size = header[0x2C] | (header[0x2D] << 8) | (header[0x2E] << 16) | (header[0x2F] << 24);
|
|
arm7_addr = header[0x30] | (header[0x31] << 8) | (header[0x32] << 16) | (header[0x33] << 24);
|
|
arm7_size = header[0x3C] | (header[0x3D] << 8) | (header[0x3E] << 16) | (header[0x3F] << 24);
|
|
icon_addr = header[0x68] | (header[0x69] << 8) | (header[0x6A] << 16) | (header[0x6B] << 24);
|
|
|
|
hash_size = 0x160 + arm9_size + arm7_size + 0xA00;
|
|
if (hash_size > 16 * 1024 * 1024)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "arm9 code size (%u) + arm7 code size (%u) exceeds 16MB", arm9_size, arm7_size);
|
|
}
|
|
else
|
|
{
|
|
if (coro->data)
|
|
free(coro->data);
|
|
|
|
coro->data = malloc(hash_size);
|
|
if (!coro->data)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "failed to allocate %u bytes", hash_size);
|
|
intfstream_close(stream);
|
|
CORO_STOP();
|
|
}
|
|
else
|
|
{
|
|
uint8_t* hash_ptr = (uint8_t*)coro->data;
|
|
|
|
memcpy(hash_ptr, header, 0x160);
|
|
hash_ptr += 0x160;
|
|
|
|
intfstream_seek(stream, arm9_addr + offset, RETRO_VFS_SEEK_POSITION_START);
|
|
intfstream_read(stream, hash_ptr, arm9_size);
|
|
hash_ptr += arm9_size;
|
|
|
|
intfstream_seek(stream, arm7_addr + offset, RETRO_VFS_SEEK_POSITION_START);
|
|
intfstream_read(stream, hash_ptr, arm7_size);
|
|
hash_ptr += arm7_size;
|
|
|
|
intfstream_seek(stream, icon_addr + offset, RETRO_VFS_SEEK_POSITION_START);
|
|
intfstream_read(stream, hash_ptr, 0xA00);
|
|
|
|
coro->len = hash_size;
|
|
success = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
intfstream_close(stream);
|
|
}
|
|
|
|
return success;
|
|
}
|
|
|
|
static int rcheevos_iterate(rcheevos_coro_t* coro)
|
|
{
|
|
char buffer[2048];
|
|
const int snes_header_len = 0x200;
|
|
const int lynx_header_len = 0x40;
|
|
ssize_t num_read = 0;
|
|
size_t to_read = 4096;
|
|
uint8_t* ptr = NULL;
|
|
const char* end = NULL;
|
|
|
|
static const uint32_t snes_exts[] =
|
|
{
|
|
0x0b88aa88U, /* smc */
|
|
0x0b8872bbU, /* fig */
|
|
0x0b88a9a1U, /* sfc */
|
|
0x0b887623U, /* gd3 */
|
|
0x0b887627U, /* gd7 */
|
|
0x0b886bf3U, /* dx2 */
|
|
0x0b886312U, /* bsx */
|
|
0x0b88abd2U, /* swc */
|
|
0
|
|
};
|
|
|
|
static const uint32_t nes_exts[] =
|
|
{
|
|
0x0b88944bU, /* nes */
|
|
0
|
|
};
|
|
|
|
static const uint32_t lynx_exts[] =
|
|
{
|
|
0x0b888cf7U, /* lnx */
|
|
0
|
|
};
|
|
|
|
static const uint32_t psx_exts[] =
|
|
{
|
|
0x0b886782U, /* cue */
|
|
0x0b88899aU, /* m3u */
|
|
/*0x0b88af0bU,* toc */
|
|
/*0x0b88652fU,* ccd */
|
|
/*0x0b889c67U,* pbp */
|
|
0x0b8865d4U, /* chd */
|
|
0
|
|
};
|
|
|
|
static const uint32_t segacd_exts[] =
|
|
{
|
|
0x0b886782U, /* cue */
|
|
0x0b8880d0U, /* iso */
|
|
0x0b8865d4U, /* chd */
|
|
0
|
|
};
|
|
|
|
static const uint32_t pce_cd_exts[] =
|
|
{
|
|
0x0b886782U, /* cue */
|
|
0x0b8865d4U, /* chd */
|
|
0
|
|
};
|
|
|
|
static const uint32_t arcade_exts[] =
|
|
{
|
|
0x0b88c7d8U, /* zip */
|
|
0
|
|
};
|
|
|
|
static const uint32_t nds_exts[] =
|
|
{
|
|
0x00b88942aU, /* nds */
|
|
0
|
|
};
|
|
|
|
static rcheevos_finder_t finders[] =
|
|
{
|
|
{RCHEEVOS_SNES_MD5, "SNES (discards header)", snes_exts},
|
|
{RCHEEVOS_LYNX_MD5, "Atari Lynx (discards header)", lynx_exts},
|
|
{RCHEEVOS_NES_MD5, "NES (discards header)", nes_exts},
|
|
{RCHEEVOS_NDS_MD5, "Nintendo DS (main executables)", nds_exts},
|
|
{RCHEEVOS_PSX_MD5, "Playstation (main executable)", psx_exts},
|
|
{RCHEEVOS_PCE_CD_MD5, "PC Engine CD (boot sector)", pce_cd_exts},
|
|
{RCHEEVOS_SEGACD_MD5, "Sega CD/Saturn (first sector)", segacd_exts},
|
|
{RCHEEVOS_ARCADE_MD5, "Arcade (filename)", arcade_exts},
|
|
{RCHEEVOS_GENERIC_MD5, "Generic (plain content)", NULL}
|
|
};
|
|
|
|
CORO_ENTER();
|
|
|
|
coro->settings = config_get_ptr();
|
|
|
|
/* Bail out if cheevos are disabled.
|
|
* But set the above anyways,
|
|
* command_read_ram needs it. */
|
|
if (!coro->settings->bools.cheevos_enable)
|
|
CORO_STOP();
|
|
|
|
/* Use the selected file's extension to determine which method to use */
|
|
for (coro->i = 0; coro->i < ARRAY_SIZE(finders); coro->i++)
|
|
{
|
|
if (finders[coro->i].ext_hashes)
|
|
{
|
|
for (coro->j = 0; finders[coro->i].ext_hashes[coro->j]; coro->j++)
|
|
{
|
|
if (finders[coro->i].ext_hashes[coro->j] == coro->ext_hash)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "testing %s\n", finders[coro->i].name);
|
|
CORO_GOSUB(finders[coro->i].label);
|
|
|
|
if (coro->gameid != 0)
|
|
goto found;
|
|
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Use the extensions supported by the core as a hint to what method we should use. */
|
|
core_get_system_info(&coro->sysinfo);
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "no method for file extension, trying core supported extensions: %s\n", coro->sysinfo.valid_extensions);
|
|
for (coro->i = 0; coro->i < ARRAY_SIZE(finders); coro->i++)
|
|
{
|
|
if (finders[coro->i].ext_hashes)
|
|
{
|
|
for (coro->j = 0; finders[coro->i].ext_hashes[coro->j]; coro->j++)
|
|
{
|
|
if (finders[coro->i].ext_hashes[coro->j] == coro->ext_hash)
|
|
break;
|
|
}
|
|
|
|
/* did we already check this one? */
|
|
if (finders[coro->i].ext_hashes[coro->j] == coro->ext_hash)
|
|
continue;
|
|
|
|
coro->ext = coro->sysinfo.valid_extensions;
|
|
|
|
while (coro->ext)
|
|
{
|
|
unsigned hash;
|
|
end = strchr(coro->ext, '|');
|
|
|
|
if (end)
|
|
{
|
|
hash = rcheevos_djb2(coro->ext, end - coro->ext);
|
|
coro->ext = end + 1;
|
|
}
|
|
else
|
|
{
|
|
hash = rcheevos_djb2(coro->ext, strlen(coro->ext));
|
|
coro->ext = NULL;
|
|
}
|
|
|
|
for (coro->j = 0; finders[coro->i].ext_hashes[coro->j]; coro->j++)
|
|
{
|
|
if (finders[coro->i].ext_hashes[coro->j] == hash)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "testing %s\n", finders[coro->i].name);
|
|
CORO_GOSUB(finders[coro->i].label);
|
|
|
|
if (coro->gameid != 0)
|
|
goto found;
|
|
|
|
coro->ext = NULL; /* force next finder */
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Try hashing methods not specifically tied to a file extension */
|
|
for (coro->i = 0; coro->i < ARRAY_SIZE(finders); coro->i++)
|
|
{
|
|
if (finders[coro->i].ext_hashes)
|
|
continue;
|
|
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "testing %s\n", finders[coro->i].name);
|
|
CORO_GOSUB(finders[coro->i].label);
|
|
|
|
if (coro->gameid != 0)
|
|
goto found;
|
|
}
|
|
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "this game doesn't feature achievements\n");
|
|
strcpy(rcheevos_locals.hash, "N/A");
|
|
rcheevos_hardcore_paused = true;
|
|
CORO_STOP();
|
|
|
|
found:
|
|
|
|
#ifdef CHEEVOS_JSON_OVERRIDE
|
|
{
|
|
size_t size = 0;
|
|
FILE *file = fopen(CHEEVOS_JSON_OVERRIDE, "rb");
|
|
|
|
fseek(file, 0, SEEK_END);
|
|
size = ftell(file);
|
|
fseek(file, 0, SEEK_SET);
|
|
|
|
coro->json = (char*)malloc(size + 1);
|
|
fread((void*)coro->json, 1, size, file);
|
|
|
|
fclose(file);
|
|
coro->json[size] = 0;
|
|
}
|
|
#else
|
|
CORO_GOSUB(RCHEEVOS_GET_CHEEVOS);
|
|
|
|
if (!coro->json)
|
|
{
|
|
runloop_msg_queue_push("Error loading achievements.", 0, 5 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "error loading achievements\n");
|
|
CORO_STOP();
|
|
}
|
|
#endif
|
|
|
|
#ifdef CHEEVOS_SAVE_JSON
|
|
{
|
|
FILE *file = fopen(CHEEVOS_SAVE_JSON, "w");
|
|
fwrite((void*)coro->json, 1, strlen(coro->json), file);
|
|
fclose(file);
|
|
}
|
|
#endif
|
|
if (rcheevos_parse(coro->json))
|
|
{
|
|
CHEEVOS_FREE(coro->json);
|
|
CORO_STOP();
|
|
}
|
|
|
|
CHEEVOS_FREE(coro->json);
|
|
|
|
if ( rcheevos_locals.patchdata.core_count == 0
|
|
&& rcheevos_locals.patchdata.unofficial_count == 0
|
|
&& rcheevos_locals.patchdata.lboard_count == 0)
|
|
{
|
|
runloop_msg_queue_push(
|
|
"This game has no achievements.",
|
|
0, 5 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
|
|
rcheevos_hardcore_paused = true;
|
|
|
|
CORO_STOP();
|
|
}
|
|
|
|
rcheevos_loaded = true;
|
|
|
|
/*
|
|
* Inputs: CHEEVOS_VAR_GAMEID
|
|
* Outputs:
|
|
*/
|
|
CORO_GOSUB(RCHEEVOS_DEACTIVATE);
|
|
|
|
/*
|
|
* Inputs: CHEEVOS_VAR_GAMEID
|
|
* Outputs:
|
|
*/
|
|
CORO_GOSUB(RCHEEVOS_PLAYING);
|
|
|
|
if (coro->settings->bools.cheevos_verbose_enable && rcheevos_locals.patchdata.core_count > 0)
|
|
{
|
|
char msg[256];
|
|
int mode = RCHEEVOS_ACTIVE_SOFTCORE;
|
|
const rcheevos_cheevo_t* cheevo = rcheevos_locals.core;
|
|
const rcheevos_cheevo_t* end = cheevo + rcheevos_locals.patchdata.core_count;
|
|
int number_of_unlocked = rcheevos_locals.patchdata.core_count;
|
|
int number_of_unsupported = 0;
|
|
|
|
if (coro->settings->bools.cheevos_hardcore_mode_enable && !rcheevos_hardcore_paused)
|
|
mode = RCHEEVOS_ACTIVE_HARDCORE;
|
|
|
|
for (; cheevo < end; cheevo++)
|
|
{
|
|
if (!cheevo->trigger)
|
|
number_of_unsupported++;
|
|
else if (cheevo->active & mode)
|
|
number_of_unlocked--;
|
|
}
|
|
|
|
if (!number_of_unsupported)
|
|
{
|
|
snprintf(msg, sizeof(msg),
|
|
"You have %d of %d achievements unlocked.",
|
|
number_of_unlocked, rcheevos_locals.patchdata.core_count);
|
|
}
|
|
else
|
|
{
|
|
snprintf(msg, sizeof(msg),
|
|
"You have %d of %d achievements unlocked (%d unsupported).",
|
|
number_of_unlocked - number_of_unsupported, rcheevos_locals.patchdata.core_count, number_of_unsupported);
|
|
}
|
|
|
|
msg[sizeof(msg) - 1] = 0;
|
|
runloop_msg_queue_push(msg, 0, 6 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
}
|
|
|
|
CORO_GOSUB(RCHEEVOS_GET_BADGES);
|
|
CORO_STOP();
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Loads a file into memory
|
|
* Input coro->path
|
|
* Output coro->data, coro->len
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_BUFFER_FILE)
|
|
if (!coro->data)
|
|
{
|
|
coro->stream = intfstream_open_file(
|
|
coro->path,
|
|
RETRO_VFS_FILE_ACCESS_READ,
|
|
RETRO_VFS_FILE_ACCESS_HINT_NONE);
|
|
|
|
if (!coro->stream)
|
|
CORO_STOP();
|
|
|
|
CORO_YIELD();
|
|
coro->len = 0;
|
|
coro->count = intfstream_get_size(coro->stream);
|
|
|
|
/* size limit */
|
|
if (coro->count > CHEEVOS_MB(64))
|
|
coro->count = CHEEVOS_MB(64);
|
|
|
|
coro->data = malloc(coro->count);
|
|
|
|
if (!coro->data)
|
|
{
|
|
intfstream_close(coro->stream);
|
|
CHEEVOS_FREE(coro->stream);
|
|
CORO_STOP();
|
|
}
|
|
|
|
for (;;)
|
|
{
|
|
ptr = (uint8_t*)coro->data + coro->len;
|
|
to_read = 8192;
|
|
|
|
if (to_read > coro->count)
|
|
to_read = coro->count;
|
|
|
|
num_read = intfstream_read(coro->stream, (void*)ptr, to_read);
|
|
if (num_read <= 0)
|
|
break;
|
|
|
|
coro->len += num_read;
|
|
coro->count -= num_read;
|
|
|
|
if (coro->count == 0)
|
|
break;
|
|
|
|
CORO_YIELD();
|
|
}
|
|
|
|
intfstream_close(coro->stream);
|
|
CHEEVOS_FREE(coro->stream);
|
|
}
|
|
CORO_RET();
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Tries to identify a SNES game
|
|
* Input coro->path or coro->data+coro->len
|
|
* Output coro->gameid
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_SNES_MD5)
|
|
CORO_GOSUB(RCHEEVOS_BUFFER_FILE);
|
|
|
|
/* Checks for the existence of a headered SNES file.
|
|
Unheadered files fall back to RCHEEVOS_GENERIC_MD5. */
|
|
if (coro->len < 0x2000 || coro->len % 0x2000 != snes_header_len)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "could not locate SNES header\n", coro->gameid);
|
|
coro->gameid = 0;
|
|
CORO_RET();
|
|
}
|
|
|
|
coro->offset = snes_header_len;
|
|
coro->count = 0;
|
|
|
|
CORO_GOSUB(RCHEEVOS_EVAL_MD5);
|
|
CORO_GOTO(RCHEEVOS_GET_GAMEID);
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Tries to identify an Atari Lynx game
|
|
* Input coro->path or coro->data+coro->len
|
|
* Output coro->gameid
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_LYNX_MD5)
|
|
CORO_GOSUB(RCHEEVOS_BUFFER_FILE);
|
|
|
|
/* Checks for the existence of a headered Lynx file.
|
|
Unheadered files fall back to RCHEEVOS_GENERIC_MD5. */
|
|
if (coro->len <= (unsigned)lynx_header_len ||
|
|
memcmp("LYNX", (void *)coro->data, 5) != 0)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "could not locate LYNX header\n", coro->gameid);
|
|
coro->gameid = 0;
|
|
CORO_RET();
|
|
}
|
|
|
|
coro->offset = lynx_header_len;
|
|
coro->count = coro->len - lynx_header_len;
|
|
|
|
CORO_GOSUB(RCHEEVOS_EVAL_MD5);
|
|
CORO_GOTO(RCHEEVOS_GET_GAMEID);
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Tries to identify a NES game
|
|
* Input coro->path or coro->data+coro->len
|
|
* Output coro->gameid
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_NES_MD5)
|
|
CORO_GOSUB(RCHEEVOS_BUFFER_FILE);
|
|
|
|
/* Checks for the existence of a headered NES file.
|
|
Unheadered files fall back to RCHEEVOS_GENERIC_MD5. */
|
|
if (coro->len < sizeof(coro->header))
|
|
{
|
|
coro->gameid = 0;
|
|
CORO_RET();
|
|
}
|
|
|
|
memcpy((void*)&coro->header, coro->data,
|
|
sizeof(coro->header));
|
|
|
|
if ( coro->header.id[0] != 'N'
|
|
|| coro->header.id[1] != 'E'
|
|
|| coro->header.id[2] != 'S'
|
|
|| coro->header.id[3] != 0x1a)
|
|
{
|
|
coro->gameid = 0;
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "could not locate NES header\n", coro->gameid);
|
|
CORO_RET();
|
|
}
|
|
|
|
coro->offset = sizeof(coro->header);
|
|
coro->count = coro->len - coro->offset;
|
|
|
|
CORO_GOSUB(RCHEEVOS_EVAL_MD5);
|
|
CORO_GOTO(RCHEEVOS_GET_GAMEID);
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Tries to identify a Sega CD game
|
|
* Input coro->path, coro->len
|
|
* Output coro->gameid
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_SEGACD_MD5)
|
|
{
|
|
/* ignore bin files less than 16MB - they're probably a ROM, not a CD */
|
|
if (coro->ext_hash == 0x0b8861beU)
|
|
{
|
|
to_read = coro->len;
|
|
if (to_read == 0)
|
|
{
|
|
coro->stream = intfstream_open_file(coro->path,
|
|
RETRO_VFS_FILE_ACCESS_READ, RETRO_VFS_FILE_ACCESS_HINT_NONE);
|
|
if (coro->stream)
|
|
{
|
|
to_read = intfstream_get_size(coro->stream);
|
|
intfstream_close(coro->stream);
|
|
CHEEVOS_FREE(coro->stream);
|
|
}
|
|
}
|
|
|
|
if (to_read < CHEEVOS_MB(16))
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "ignoring small BIN file - assuming not CD\n", coro->gameid);
|
|
coro->gameid = 0;
|
|
CORO_RET();
|
|
}
|
|
}
|
|
|
|
/* find the data track - it should be the first one */
|
|
coro->track = cdfs_open_data_track(coro->path);
|
|
if (coro->track)
|
|
{
|
|
/* open the raw CD */
|
|
if (cdfs_open_file(&coro->cdfp, coro->track, NULL))
|
|
{
|
|
coro->count = 512;
|
|
free(coro->data);
|
|
coro->data = (uint8_t*)malloc(coro->count);
|
|
cdfs_read_file(&coro->cdfp, coro->data, coro->count);
|
|
coro->len = coro->count;
|
|
|
|
cdfs_close_file(&coro->cdfp);
|
|
|
|
cdfs_close_track(coro->track);
|
|
coro->track = NULL;
|
|
|
|
CORO_GOSUB(RCHEEVOS_EVAL_MD5);
|
|
CORO_GOTO(RCHEEVOS_GET_GAMEID);
|
|
}
|
|
|
|
cdfs_close_track(coro->track);
|
|
coro->track = NULL;
|
|
}
|
|
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "could not open CD\n", coro->gameid);
|
|
coro->gameid = 0;
|
|
CORO_RET();
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Tries to identify a PC Engine CD game
|
|
* Input coro->path
|
|
* Output coro->gameid
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_PCE_CD_MD5)
|
|
{
|
|
/* find the data track - it should be the second one */
|
|
coro->track = cdfs_open_data_track(coro->path);
|
|
if (coro->track)
|
|
{
|
|
/* open the raw CD */
|
|
if (cdfs_open_file(&coro->cdfp, coro->track, NULL))
|
|
{
|
|
/* the PC-Engine uses the second sector to specify boot information and program name.
|
|
* the string "PC Engine CD-ROM SYSTEM" should exist at 32 bytes into the sector
|
|
* http://shu.sheldows.com/shu/download/pcedocs/pce_cdrom.html
|
|
*/
|
|
cdfs_seek_sector(&coro->cdfp, 1);
|
|
cdfs_read_file(&coro->cdfp, buffer, 128);
|
|
|
|
if (strncmp("PC Engine CD-ROM SYSTEM", (const char*)& buffer[32], 23) != 0)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "not a PC Engine CD\n", coro->gameid);
|
|
|
|
cdfs_close_track(coro->track);
|
|
coro->track = NULL;
|
|
|
|
coro->gameid = 0;
|
|
CORO_RET();
|
|
}
|
|
|
|
{
|
|
/* the first three bytes specify the sector of the program data, and the fourth byte
|
|
* is the number of sectors.
|
|
*/
|
|
const unsigned int first_sector = buffer[0] * 65536 + buffer[1] * 256 + buffer[2];
|
|
cdfs_seek_sector(&coro->cdfp, first_sector);
|
|
|
|
to_read = buffer[3] * 2048;
|
|
}
|
|
|
|
coro->count = to_read + 22;
|
|
free(coro->data);
|
|
coro->data = (uint8_t*)malloc(coro->count);
|
|
memcpy(coro->data, &buffer[106], 22);
|
|
|
|
cdfs_read_file(&coro->cdfp, ((uint8_t*)coro->data) + 22, to_read);
|
|
coro->len = coro->count;
|
|
|
|
cdfs_close_file(&coro->cdfp);
|
|
|
|
cdfs_close_track(coro->track);
|
|
coro->track = NULL;
|
|
|
|
CORO_GOSUB(RCHEEVOS_EVAL_MD5);
|
|
CORO_GOTO(RCHEEVOS_GET_GAMEID);
|
|
}
|
|
|
|
cdfs_close_track(coro->track);
|
|
coro->track = NULL;
|
|
}
|
|
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "could not open CD\n", coro->gameid);
|
|
coro->gameid = 0;
|
|
CORO_RET();
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Tries to identify a Playstation game
|
|
* Input coro->path
|
|
* Output coro->gameid
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_PSX_MD5)
|
|
{
|
|
if (rcheevos_prepare_hash_psx(coro))
|
|
{
|
|
CORO_GOSUB(RCHEEVOS_EVAL_MD5);
|
|
CORO_GOTO(RCHEEVOS_GET_GAMEID);
|
|
}
|
|
|
|
coro->gameid = 0;
|
|
CORO_RET();
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Tries to identify a Nintendo DS game
|
|
* Input coro->path
|
|
* Output coro->gameid
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_NDS_MD5)
|
|
{
|
|
if (rcheevos_prepare_hash_nintendo_ds(coro))
|
|
{
|
|
CORO_GOSUB(RCHEEVOS_EVAL_MD5);
|
|
CORO_GOTO(RCHEEVOS_GET_GAMEID);
|
|
}
|
|
|
|
coro->gameid = 0;
|
|
CORO_RET();
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Tries to identify a game by examining the entire file (no special processing)
|
|
* Input coro->path or coro->data+coro->len
|
|
* Output coro->gameid
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_GENERIC_MD5)
|
|
CORO_GOSUB(RCHEEVOS_BUFFER_FILE);
|
|
|
|
coro->offset = 0;
|
|
coro->count = 0;
|
|
|
|
CORO_GOSUB(RCHEEVOS_EVAL_MD5);
|
|
|
|
if (coro->count == 0)
|
|
{
|
|
coro->gameid = 0;
|
|
CORO_RET();
|
|
}
|
|
|
|
CORO_GOTO(RCHEEVOS_GET_GAMEID);
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Tries to identify an arcade game based on its filename (with no extension).
|
|
* An arcade game "rom" is a zip file containing many ROMs.
|
|
* Input coro->path
|
|
* Output coro->gameid
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_ARCADE_MD5)
|
|
if (!string_is_empty(coro->path))
|
|
{
|
|
char base_noext[PATH_MAX_LENGTH];
|
|
fill_pathname_base_noext(base_noext, coro->path, sizeof(base_noext));
|
|
|
|
MD5_Init(&coro->md5);
|
|
MD5_Update(&coro->md5, (void*)base_noext, strlen(base_noext));
|
|
MD5_Final(coro->hash, &coro->md5);
|
|
|
|
CORO_GOTO(RCHEEVOS_GET_GAMEID);
|
|
}
|
|
CORO_RET();
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Evaluates the CHEEVOS_VAR_MD5 hash
|
|
* Inputs coro->data, coro->count, coro->offset, coro->len
|
|
* Outputs coro->hash
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_EVAL_MD5)
|
|
|
|
if (coro->count == 0)
|
|
coro->count = coro->len;
|
|
|
|
if (coro->len - coro->offset < coro->count)
|
|
coro->count = coro->len - coro->offset;
|
|
|
|
/* size limit */
|
|
if (coro->count > CHEEVOS_MB(64))
|
|
coro->count = CHEEVOS_MB(64);
|
|
|
|
MD5_Init(&coro->md5);
|
|
MD5_Update(&coro->md5,
|
|
(void*)((uint8_t*)coro->data + coro->offset),
|
|
coro->count);
|
|
MD5_Final(coro->hash, &coro->md5);
|
|
|
|
CORO_RET();
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Gets the achievements from Retro Achievements
|
|
* Inputs coro->hash
|
|
* Outputs coro->gameid
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_GET_GAMEID)
|
|
|
|
{
|
|
int size;
|
|
|
|
if (memcmp(coro->last_hash, coro->hash, sizeof(coro->hash)) == 0)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "hash did not change, returning %u\n", coro->gameid);
|
|
CORO_RET();
|
|
}
|
|
memcpy(coro->last_hash, coro->hash, sizeof(coro->hash));
|
|
|
|
size = rc_url_get_gameid(coro->url, sizeof(coro->url), coro->hash);
|
|
|
|
if (size < 0)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "buffer too small to create URL\n");
|
|
CORO_RET();
|
|
}
|
|
|
|
sprintf(rcheevos_locals.hash, "%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x",
|
|
coro->hash[0], coro->hash[1], coro->hash[2], coro->hash[3],
|
|
coro->hash[4], coro->hash[5], coro->hash[6], coro->hash[7],
|
|
coro->hash[8], coro->hash[9], coro->hash[10], coro->hash[11],
|
|
coro->hash[12], coro->hash[13], coro->hash[14], coro->hash[15]);
|
|
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "checking %s\n", rcheevos_locals.hash);
|
|
rcheevos_log_url(RCHEEVOS_TAG "rc_url_get_gameid: %s\n", coro->url);
|
|
CORO_GOSUB(RCHEEVOS_HTTP_GET);
|
|
|
|
if (!coro->json)
|
|
CORO_RET();
|
|
|
|
coro->gameid = chevos_get_gameid(coro->json);
|
|
|
|
CHEEVOS_FREE(coro->json);
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "got game id %u\n", coro->gameid);
|
|
CORO_RET();
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Gets the achievements from Retro Achievements
|
|
* Inputs CHEEVOS_VAR_GAMEID
|
|
* Outputs CHEEVOS_VAR_JSON
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_GET_CHEEVOS)
|
|
{
|
|
int ret;
|
|
|
|
CORO_GOSUB(RCHEEVOS_LOGIN);
|
|
|
|
ret = rc_url_get_patch(coro->url, sizeof(coro->url), coro->settings->arrays.cheevos_username, rcheevos_locals.token, coro->gameid);
|
|
|
|
if (ret < 0)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "buffer too small to create URL\n");
|
|
CORO_STOP();
|
|
}
|
|
|
|
rcheevos_log_url(RCHEEVOS_TAG "rc_url_get_patch: %s\n", coro->url);
|
|
CORO_GOSUB(RCHEEVOS_HTTP_GET);
|
|
|
|
if (!coro->json)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "error getting achievements for game id %u\n", coro->gameid);
|
|
CORO_STOP();
|
|
}
|
|
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "got achievements for game id %u\n", coro->gameid);
|
|
CORO_RET();
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Gets the achievements from Retro Achievements
|
|
* Inputs CHEEVOS_VAR_GAMEID
|
|
* Outputs CHEEVOS_VAR_JSON
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_GET_BADGES)
|
|
|
|
badges_ctx = new_badges_ctx;
|
|
|
|
#if defined(HAVE_GFX_WIDGETS)
|
|
if (false) /* we always want badges if menu widgets are enabled */
|
|
#endif
|
|
{
|
|
settings_t *settings = config_get_ptr();
|
|
if (!(
|
|
string_is_equal(settings->arrays.menu_driver, "xmb") ||
|
|
string_is_equal(settings->arrays.menu_driver, "ozone")
|
|
) ||
|
|
!settings->bools.cheevos_badges_enable)
|
|
CORO_RET();
|
|
}
|
|
|
|
for (coro->i = 0; coro->i < 2; coro->i++)
|
|
{
|
|
if (coro->i == 0)
|
|
{
|
|
coro->cheevo = rcheevos_locals.core;
|
|
coro->cheevo_end = coro->cheevo + rcheevos_locals.patchdata.core_count;
|
|
}
|
|
else
|
|
{
|
|
coro->cheevo = rcheevos_locals.unofficial;
|
|
coro->cheevo_end = coro->cheevo + rcheevos_locals.patchdata.unofficial_count;
|
|
}
|
|
|
|
for (; coro->cheevo < coro->cheevo_end; coro->cheevo++)
|
|
{
|
|
for (coro->j = 0 ; coro->j < 2; coro->j++)
|
|
{
|
|
coro->badge_fullpath[0] = '\0';
|
|
fill_pathname_application_special(
|
|
coro->badge_fullpath,
|
|
sizeof(coro->badge_fullpath),
|
|
APPLICATION_SPECIAL_DIRECTORY_THUMBNAILS_CHEEVOS_BADGES);
|
|
|
|
if (!path_is_directory(coro->badge_fullpath))
|
|
path_mkdir(coro->badge_fullpath);
|
|
CORO_YIELD();
|
|
|
|
if (!coro->cheevo->info->badge || !coro->cheevo->info->badge[0])
|
|
continue;
|
|
|
|
if (coro->j == 0)
|
|
snprintf(coro->badge_name,
|
|
sizeof(coro->badge_name),
|
|
"%s.png", coro->cheevo->info->badge);
|
|
else
|
|
snprintf(coro->badge_name,
|
|
sizeof(coro->badge_name),
|
|
"%s_lock.png", coro->cheevo->info->badge);
|
|
|
|
fill_pathname_join(
|
|
coro->badge_fullpath,
|
|
coro->badge_fullpath,
|
|
coro->badge_name,
|
|
sizeof(coro->badge_fullpath));
|
|
|
|
if (!badge_exists(coro->badge_fullpath))
|
|
{
|
|
#ifdef CHEEVOS_LOG_BADGES
|
|
CHEEVOS_LOG(
|
|
RCHEEVOS_TAG "downloading badge %s\n",
|
|
coro->badge_fullpath);
|
|
#endif
|
|
snprintf(coro->url,
|
|
sizeof(coro->url),
|
|
"http://i.retroachievements.org/Badge/%s",
|
|
coro->badge_name);
|
|
|
|
CORO_GOSUB(RCHEEVOS_HTTP_GET);
|
|
|
|
if (coro->json)
|
|
{
|
|
if (!filestream_write_file(coro->badge_fullpath,
|
|
coro->json, coro->k))
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "error writing badge %s\n", coro->badge_fullpath);
|
|
else
|
|
{
|
|
CHEEVOS_FREE(coro->json);
|
|
coro->json = NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
CORO_RET();
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Logs in the user at Retro Achievements
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_LOGIN)
|
|
{
|
|
int ret;
|
|
char tok[256];
|
|
|
|
if (rcheevos_locals.token[0])
|
|
CORO_RET();
|
|
|
|
if (string_is_empty(coro->settings->arrays.cheevos_username))
|
|
{
|
|
runloop_msg_queue_push(
|
|
"Missing RetroAchievements account information.",
|
|
0, 5 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
runloop_msg_queue_push(
|
|
"Please fill in your account information in Settings.",
|
|
0, 5 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "login info not informed\n");
|
|
CORO_STOP();
|
|
}
|
|
|
|
if (string_is_empty(coro->settings->arrays.cheevos_token))
|
|
ret = rc_url_login_with_password(coro->url, sizeof(coro->url),
|
|
coro->settings->arrays.cheevos_username,
|
|
coro->settings->arrays.cheevos_password);
|
|
else
|
|
ret = rc_url_login_with_token(coro->url, sizeof(coro->url),
|
|
coro->settings->arrays.cheevos_username,
|
|
coro->settings->arrays.cheevos_token);
|
|
|
|
if (ret < 0)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "buffer too small to create URL\n");
|
|
CORO_STOP();
|
|
}
|
|
|
|
rcheevos_log_url(RCHEEVOS_TAG "rc_url_login_with_password: %s\n", coro->url);
|
|
CORO_GOSUB(RCHEEVOS_HTTP_GET);
|
|
|
|
if (!coro->json)
|
|
{
|
|
runloop_msg_queue_push("RetroAchievements: Error contacting server.", 0, 5 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "error getting user token\n");
|
|
|
|
CORO_STOP();
|
|
}
|
|
|
|
ret = rcheevos_get_token(coro->json, tok, sizeof(tok));
|
|
|
|
if (ret != 0)
|
|
{
|
|
char msg[512];
|
|
snprintf(msg, sizeof(msg),
|
|
"RetroAchievements: %s",
|
|
tok);
|
|
runloop_msg_queue_push(msg, 0, 5 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
*coro->settings->arrays.cheevos_token = 0;
|
|
|
|
CHEEVOS_FREE(coro->json);
|
|
CORO_STOP();
|
|
}
|
|
|
|
CHEEVOS_FREE(coro->json);
|
|
|
|
if (coro->settings->bools.cheevos_verbose_enable)
|
|
{
|
|
char msg[256];
|
|
snprintf(msg, sizeof(msg),
|
|
"RetroAchievements: Logged in as \"%s\".",
|
|
coro->settings->arrays.cheevos_username);
|
|
msg[sizeof(msg) - 1] = 0;
|
|
runloop_msg_queue_push(msg, 0, 3 * 60, false, NULL, MESSAGE_QUEUE_ICON_DEFAULT, MESSAGE_QUEUE_CATEGORY_INFO);
|
|
}
|
|
|
|
strlcpy(rcheevos_locals.token, tok,
|
|
sizeof(rcheevos_locals.token));
|
|
|
|
/* Save token to config and clear pass on success */
|
|
strlcpy(coro->settings->arrays.cheevos_token, tok,
|
|
sizeof(coro->settings->arrays.cheevos_token));
|
|
|
|
*coro->settings->arrays.cheevos_password = 0;
|
|
CORO_RET();
|
|
}
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Pauses execution for five seconds
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_DELAY)
|
|
|
|
{
|
|
retro_time_t t1;
|
|
coro->t0 = cpu_features_get_time_usec();
|
|
|
|
do
|
|
{
|
|
CORO_YIELD();
|
|
t1 = cpu_features_get_time_usec();
|
|
}while ((t1 - coro->t0) < 3000000);
|
|
}
|
|
|
|
CORO_RET();
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Makes a HTTP GET request
|
|
* Inputs CHEEVOS_VAR_URL
|
|
* Outputs CHEEVOS_VAR_JSON
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_HTTP_GET)
|
|
|
|
for (coro->k = 0; coro->k < 5; coro->k++)
|
|
{
|
|
if (coro->k != 0)
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Retrying HTTP request: %u of 5\n", coro->k + 1);
|
|
|
|
coro->json = NULL;
|
|
coro->conn = net_http_connection_new(
|
|
coro->url, "GET", NULL);
|
|
|
|
if (!coro->conn)
|
|
{
|
|
CORO_GOSUB(RCHEEVOS_DELAY);
|
|
continue;
|
|
}
|
|
|
|
/* Don't bother with timeouts here, it's just a string scan. */
|
|
while (!net_http_connection_iterate(coro->conn)) {}
|
|
|
|
/* Error finishing the connection descriptor. */
|
|
if (!net_http_connection_done(coro->conn))
|
|
{
|
|
net_http_connection_free(coro->conn);
|
|
continue;
|
|
}
|
|
|
|
rcheevos_get_user_agent(buffer);
|
|
net_http_connection_set_user_agent(coro->conn, buffer);
|
|
|
|
coro->http = net_http_new(coro->conn);
|
|
|
|
/* Error connecting to the endpoint. */
|
|
if (!coro->http)
|
|
{
|
|
net_http_connection_free(coro->conn);
|
|
CORO_GOSUB(RCHEEVOS_DELAY);
|
|
continue;
|
|
}
|
|
|
|
while (!net_http_update(coro->http, NULL, NULL))
|
|
CORO_YIELD();
|
|
|
|
{
|
|
size_t length;
|
|
uint8_t *data = net_http_data(coro->http,
|
|
&length, false);
|
|
|
|
if (data)
|
|
{
|
|
coro->json = (char*)malloc(length + 1);
|
|
|
|
if (coro->json)
|
|
{
|
|
memcpy((void*)coro->json, (void*)data, length);
|
|
CHEEVOS_FREE(data);
|
|
coro->json[length] = 0;
|
|
}
|
|
|
|
coro->k = (unsigned)length;
|
|
net_http_delete(coro->http);
|
|
net_http_connection_free(coro->conn);
|
|
CORO_RET();
|
|
}
|
|
}
|
|
|
|
net_http_delete(coro->http);
|
|
net_http_connection_free(coro->conn);
|
|
}
|
|
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Couldn't connect to server after 5 tries\n");
|
|
CORO_RET();
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Deactivates the achievements already awarded
|
|
* Inputs CHEEVOS_VAR_GAMEID
|
|
* Outputs
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_DEACTIVATE)
|
|
|
|
CORO_GOSUB(RCHEEVOS_LOGIN);
|
|
{
|
|
int ret;
|
|
unsigned mode;
|
|
|
|
/* Two calls - one for softcore and one for hardcore */
|
|
for (coro->i = 0; coro->i < 2; coro->i++)
|
|
{
|
|
ret = rc_url_get_unlock_list(coro->url, sizeof(coro->url),
|
|
coro->settings->arrays.cheevos_username,
|
|
rcheevos_locals.token, coro->gameid, coro->i);
|
|
|
|
if (ret < 0)
|
|
{
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "buffer too small to create URL\n");
|
|
CORO_STOP();
|
|
}
|
|
|
|
rcheevos_log_url(RCHEEVOS_TAG "rc_url_get_unlock_list: %s\n", coro->url);
|
|
CORO_GOSUB(RCHEEVOS_HTTP_GET);
|
|
|
|
if (coro->json)
|
|
{
|
|
mode = coro->i == 0 ? RCHEEVOS_ACTIVE_SOFTCORE : RCHEEVOS_ACTIVE_HARDCORE;
|
|
rcheevos_deactivate_unlocks(coro->json, rcheevos_unlock_cb, &mode);
|
|
CHEEVOS_FREE(coro->json);
|
|
}
|
|
else
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "error retrieving list of unlocked achievements in softcore mode\n");
|
|
}
|
|
}
|
|
|
|
CORO_RET();
|
|
|
|
|
|
/**************************************************************************
|
|
* Info Posts the "playing" activity to Retro Achievements
|
|
* Inputs CHEEVOS_VAR_GAMEID
|
|
* Outputs
|
|
*************************************************************************/
|
|
CORO_SUB(RCHEEVOS_PLAYING)
|
|
|
|
snprintf(
|
|
coro->url, sizeof(coro->url),
|
|
"http://retroachievements.org/dorequest.php?r=postactivity&u=%s&t=%s&a=3&m=%u",
|
|
coro->settings->arrays.cheevos_username,
|
|
rcheevos_locals.token, coro->gameid
|
|
);
|
|
|
|
coro->url[sizeof(coro->url) - 1] = 0;
|
|
rcheevos_log_url(RCHEEVOS_TAG "url to post the 'playing' activity: %s\n", coro->url);
|
|
|
|
CORO_GOSUB(RCHEEVOS_HTTP_GET);
|
|
|
|
if (coro->json)
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "posted playing activity\n");
|
|
CHEEVOS_FREE(coro->json);
|
|
}
|
|
else
|
|
CHEEVOS_ERR(RCHEEVOS_TAG "error posting playing activity\n");
|
|
|
|
CORO_RET();
|
|
|
|
CORO_LEAVE();
|
|
}
|
|
|
|
static void rcheevos_task_handler(retro_task_t *task)
|
|
{
|
|
rcheevos_coro_t *coro = (rcheevos_coro_t*)task->state;
|
|
|
|
if (!coro)
|
|
return;
|
|
|
|
if (!rcheevos_iterate(coro) || task_get_cancelled(task))
|
|
{
|
|
task_set_finished(task, true);
|
|
|
|
CHEEVOS_LOCK(rcheevos_locals.task_lock);
|
|
rcheevos_locals.task = NULL;
|
|
CHEEVOS_UNLOCK(rcheevos_locals.task_lock);
|
|
|
|
if (task_get_cancelled(task))
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Load task cancelled\n");
|
|
}
|
|
else
|
|
{
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "Load task finished\n");
|
|
}
|
|
|
|
CHEEVOS_FREE(coro->data);
|
|
CHEEVOS_FREE(coro->path);
|
|
CHEEVOS_FREE(coro);
|
|
}
|
|
}
|
|
|
|
bool rcheevos_load(const void *data)
|
|
{
|
|
char buffer[32];
|
|
retro_task_t *task = NULL;
|
|
const struct retro_game_info *info = NULL;
|
|
rcheevos_coro_t *coro = NULL;
|
|
|
|
rcheevos_loaded = false;
|
|
rcheevos_hardcore_paused = false;
|
|
|
|
if (!rcheevos_locals.core_supports || !data)
|
|
{
|
|
rcheevos_hardcore_paused = true;
|
|
return false;
|
|
}
|
|
|
|
coro = (rcheevos_coro_t*)calloc(1, sizeof(*coro));
|
|
|
|
if (!coro)
|
|
return false;
|
|
|
|
task = task_init();
|
|
|
|
if (!task)
|
|
{
|
|
CHEEVOS_FREE(coro);
|
|
return false;
|
|
}
|
|
|
|
CORO_SETUP();
|
|
|
|
info = (const struct retro_game_info*)data;
|
|
strlcpy(buffer, path_get_extension(info->path), sizeof(buffer));
|
|
|
|
if (info->data)
|
|
{
|
|
coro->len = info->size;
|
|
|
|
/* size limit */
|
|
if (coro->len > CHEEVOS_MB(64))
|
|
coro->len = CHEEVOS_MB(64);
|
|
|
|
coro->data = malloc(coro->len);
|
|
|
|
if (!coro->data)
|
|
{
|
|
CHEEVOS_FREE(task);
|
|
CHEEVOS_FREE(coro);
|
|
return false;
|
|
}
|
|
|
|
memcpy(coro->data, info->data, coro->len);
|
|
coro->path = NULL;
|
|
}
|
|
else
|
|
{
|
|
coro->data = NULL;
|
|
coro->path = strdup(info->path);
|
|
|
|
/* if we're looking at an m3u file, get the first disc from the playlist */
|
|
if (string_is_equal_noncase(path_get_extension(coro->path), "m3u"))
|
|
{
|
|
intfstream_t* m3u_stream = intfstream_open_file(coro->path, RETRO_VFS_FILE_ACCESS_READ, RETRO_VFS_FILE_ACCESS_HINT_NONE);
|
|
if (m3u_stream)
|
|
{
|
|
char m3u_contents[1024];
|
|
char disc_path[PATH_MAX_LENGTH];
|
|
char* tmp;
|
|
int64_t num_read = intfstream_read(
|
|
m3u_stream, m3u_contents, sizeof(m3u_contents) - 1);
|
|
|
|
intfstream_close(m3u_stream);
|
|
m3u_contents[num_read] = '\0';
|
|
|
|
tmp = m3u_contents;
|
|
while (*tmp && *tmp != '\n')
|
|
++tmp;
|
|
if (tmp > buffer && tmp[-1] == '\r')
|
|
--tmp;
|
|
*tmp = '\0';
|
|
|
|
fill_pathname_basedir(disc_path, coro->path, sizeof(disc_path));
|
|
strlcat(disc_path, m3u_contents, sizeof(disc_path));
|
|
|
|
free((void*)coro->path);
|
|
coro->path = strdup(disc_path);
|
|
|
|
strlcpy(buffer, path_get_extension(disc_path), sizeof(buffer));
|
|
}
|
|
}
|
|
}
|
|
|
|
buffer[sizeof(buffer) - 1] = '\0';
|
|
string_to_lower(buffer);
|
|
coro->ext_hash = rcheevos_djb2(buffer, strlen(buffer));
|
|
CHEEVOS_LOG(RCHEEVOS_TAG "ext_hash %08x ('%s')\n", coro->ext_hash, buffer);
|
|
|
|
task->handler = rcheevos_task_handler;
|
|
task->state = (void*)coro;
|
|
task->mute = true;
|
|
task->callback = NULL;
|
|
task->user_data = NULL;
|
|
task->progress = 0;
|
|
task->title = NULL;
|
|
|
|
#ifdef HAVE_THREADS
|
|
if (!rcheevos_locals.task_lock)
|
|
rcheevos_locals.task_lock = slock_new();
|
|
#endif
|
|
|
|
CHEEVOS_LOCK(rcheevos_locals.task_lock);
|
|
rcheevos_locals.task = task;
|
|
CHEEVOS_UNLOCK(rcheevos_locals.task_lock);
|
|
|
|
task_queue_push(task);
|
|
|
|
return true;
|
|
}
|