RetroArch
|
rcheevos is a set of C code, or a library if you will, that tries to make it easier for emulators to process RetroAchievements data, providing support for achievements and leaderboards for their players.
Keep in mind that rcheevos does not provide HTTP network connections or JSON parsing. Clients must get data from RetroAchievements, parse the JSON payloads and pass the results down to rcheevos for processing. (TODO: document the server API and JSON schema.)
Not all structures defined by rcheevos can be created via the public API, but are exposed to allow interactions beyond just creation, destruction, and testing, such as the ones required by UI code that helps to create them.
Finally, rcheevos does not allocate or manage memory by itself. All structures that can be returned by it have a function to determine the number of bytes needed to hold the structure, and another one that actually builds the structure using a caller-provided buffer to bake it. However, calls to rcheevos may allocate and/or free memory as part of the Lua runtime, which is a dependency.
RetroAchievements has decided to support achievements written using the Lua language. The current expression-based implementation was too limiting already for the Nintendo 64, and was preventing the support of other systems.
rcheevos does not create or maintain a Lua state, you have to create your own state and provide it to rcheevos to be used when Lua-coded achievements are found.
Lua functions used in trigger operands receive two parameters: peek
, which is used to read from the emulated system's memory, and userdata
, which must be passed to peek
. peek
's signature is the same as its C counterpart:
An understanding about how achievements are developed may be useful, you can read more about it here.
There's only one thing that can be configured by users of rcheevos: RC_ALIGNMENT
. This macro holds the alignment of allocations made in the buffer provided to the parsing functions, and the default value is sizeof(void*)
.
If your platform will benefit from a different value, define a new value for it on your compiler flags before compiling the code. It has to be a power of 2, but no checking is done.
The functions that compute the amount of memory that something will take return the number of bytes, or a negative value from the following enumeration:
This enumeration uniquely identifies each of the supported platforms in RetroAchievements.
rc_operand_t
An operand is the leaf node of RetroAchievements expressions, and can hold one of the following:
The size
field, when applicable, holds one of these values:
The type
field is always valid, and holds one of these values:
RC_OPERAND_ADDRESS
, RC_OPERAND_DELTA
and RC_OPERAND_CONST
mean that the anonymous structure in the union is active. RC_OPERAND_FP
means that fp_value
is active. RC_OPERAND_LUA
means function_ref
is active.
rc_condition_t
A condition compares its two operands according to the defined operator. It also keeps track of other things to make it possible to code more advanced achievements.
type
can be one of these values:
oper
is the comparison operator to be used when comparing the two operands:
rc_condset_t
Condition sets are an ordered collection of conditions (rc_condition_t
), which are usually and'ed together to help build complex expressions for achievements.
rc_trigger_t
Triggers are the basic blocks of achievements and leaderboards. In fact, achievements are just triggers with some additional information like title, description, a badge, and some state, like whether it has already been awarded or not. All the logic to test if an achievement should be awarded is encapsulated in rc_trigger_t
.
The size in bytes of memory a trigger needs to be created is given by the rc_trigger_size
function:
The return value is the size needed for the trigger described by the memaddr
parameter, or a negative value with an error code.
Once the memory size is known, rc_parse_trigger
can be called to actually construct a trigger in the caller-provided buffer:
buffer
is the caller-allocated buffer, which must have enough space for the trigger. memaddr
describes the trigger, and must be the same one used to compute the trigger's size with rc_trigger_size
. L
must be a valid Lua state, and funcs_ndx
must be an index to the current Lua stack which contains a table which is a map of names to functions. This map is used to look for operands which are Lua functions.
Once the trigger is created, rc_test_trigger
can be called to test whether the trigger fires or not.
trigger
is the trigger to test. peek
is a callback used to read bytes from the emulated memory. ud
is an user-provided opaque value that is passed to peek
. L
is the Lua state in which context the Lua functions are looked for and called, if necessary.
rc_peek_t
's signature is:
where address
is the starting address to read from, num_bytes
the number of bytes to read (1, 2, or 4, little-endian), and ud
is the same value passed to rc_test_trigger
.
Addresses passed to
peek
do not map 1:1 to the emulated memory. (TODO: document the mapping frompeek
addresses to emulated memory for each supported system.)
Finally, rc_reset_trigger
can be used to reset the internal state of a trigger.
rc_term_t
A term is the leaf node of expressions used to compute values from operands. A term is evaluated by multiplying its two operands. invert
is used to invert the bits of the second operand of the term, when the unary operator ~
is used.
rc_expression_t
An expression is a collection of terms. All terms in the collection are added together to give the value of the expression.
rc_value_t
A value is a collection of expressions. It's used to give the value for a leaderboard, and it evaluates to value of the expression with the greatest value in the collection.
The size in bytes needed to create a value can be computed by rc_value_size
:
With the size at hand, the caller can allocate the necessary memory and pass it to rc_parse_value
to create the value:
buffer
, memaddr
, L
, and funcs_ndx
are the same as in rc_parse_trigger
.
To compute the value, use rc_evaluate_value
:
value
is the value to compute the value of, and peek
, ud
, and L
, are as in rc_test_trigger
.
rc_lboard_t
Leaderboards track a value over time, starting when a trigger is fired. The leaderboard can be canceled depending on the value of another trigger, and submitted to the RetroAchievements server depending on a third trigger.
The value submitted comes from the value
field. Values displayed to the player come from the progress
field unless it's NULL
, in which case it's the same as value
.
Leaderboards are created and parsed just the same as triggers and values:
A leaderboard can be evaluated with the rc_evaluate_lboard
function:
The function returns an action that must be performed by the caller, and value
contains the value to be used for that action when the function returns. The action can be one of:
The caller must keep track of these values and do the necessary actions:
RC_LBOARD_ACTIVE
and RC_LBOARD_INACTIVE
: just signal that the leaderboard didn't change its state.RC_LBOARD_STARTED
: indicates that the leaderboard has been started, so the caller can i.e. show a message for the player, and start showing its value in the UI.RC_LBOARD_CANCELED
: the leaderboard has been canceled, and the caller can inform the user and stop showing its value.RC_LBOARD_TRIGGERED
: the leaderboard has been finished, and the value must be submitted to the RetroAchievements server; the caller can also notify the player and stop showing the value in the UI.rc_reset_lboard
resets the leaderboard:
rcheevos includes helper functions to parse formatting strings from RetroAchievements, and format values according to them.
rc_parse_format
returns the format for the given string:
The returned value is one of:
RC_FORMAT_VALUE
is returned if format_str
doesn't contain a valid format.
rc_format_value
can be used to format the given value into the provided buffer:
buffer
receives value
formatted according to format
. No more than size
characters will be written to buffer
. 32 characters are enough to hold any valid value with any format.
rurl builds URLs to access many RetroAchievements web services. Its purpose it to just to free the developer from having to URL-encode parameters and build correct URL that are valid for the server.
rurl does not make HTTP requests.
All functions return 0
if successful, or -1
in case of errors. Errors are usually because the provided buffer is too small to hold the URL. If your buffer is large and you're still receiving errors, please open an issue.
All functions take a buffer
, where the URL will be written into, and size
with the size of the buffer. The other parameters are particular to the desired URL.