2020-03-20 17:40:56 +00:00
|
|
|
#define SCOREDAT_MAGIC "HISCORE"
|
|
|
|
#define SCOREDAT_PLACES 10
|
|
|
|
#define SCOREDAT_NAME_KANJI 8
|
|
|
|
// Actually creates slightly different assembly compared to sizeof() on a
|
|
|
|
// int8_t array!
|
|
|
|
#define SCOREDAT_NAME_BYTES (SCOREDAT_NAME_KANJI * 2)
|
2020-05-18 19:29:12 +00:00
|
|
|
#define SCOREDAT_NAMES_SIZE (SCOREDAT_PLACES * SCOREDAT_NAME_BYTES)
|
2020-05-14 18:52:40 +00:00
|
|
|
#define SCOREDAT_ROUTE_LEN 2
|
2020-03-20 17:40:56 +00:00
|
|
|
|
|
|
|
#define SCOREDAT_CLEARED 40
|
|
|
|
#define SCOREDAT_CLEARED_MAKAI (SCOREDAT_CLEARED + 10)
|
|
|
|
#define SCOREDAT_CLEARED_JIGOKU (SCOREDAT_CLEARED + 20)
|
|
|
|
|
|
|
|
#define SCOREDAT_NAME_KEY 0x9C
|
|
|
|
|
|
|
|
// Encodes or decodes a single name byte.
|
|
|
|
int8_t scoredat_name_byte_encode(int8_t byte);
|
|
|
|
int8_t scoredat_name_byte_decode(int8_t byte);
|
|
|
|
|
|
|
|
// On-disk structure of the REYHI*.DAT files.
|
|
|
|
// For reference, never actually used by the game itself
|
|
|
|
struct scoredat_t {
|
|
|
|
// Not null-terminated!
|
|
|
|
char magic[sizeof(SCOREDAT_MAGIC) - 1];
|
2021-09-06 18:16:39 +00:00
|
|
|
|
|
|
|
// Exclusively used to store full-width Shift-JIS code points.
|
|
|
|
// Not null-terminated.
|
|
|
|
int16_t name[SCOREDAT_PLACES][SCOREDAT_NAME_KANJI];
|
|
|
|
|
2020-03-20 17:40:56 +00:00
|
|
|
uint32_t points[SCOREDAT_PLACES];
|
|
|
|
int16_t stage[SCOREDAT_PLACES];
|
|
|
|
twobyte_t route[SCOREDAT_PLACES];
|
|
|
|
};
|
|
|
|
|
2020-05-18 19:29:12 +00:00
|
|
|
extern int8_t* scoredat_names; // Yeah, technically a scoredat_name_t.
|
2020-03-20 17:40:56 +00:00
|
|
|
extern int16_t* scoredat_stages;
|
2020-05-22 17:27:50 +00:00
|
|
|
extern int32_t* scoredat_points;
|
|
|
|
extern int8_t* scoredat_routes; // Yeah, technically a twobyte_t.
|
2020-03-20 17:40:56 +00:00
|
|
|
|
2020-05-14 18:52:40 +00:00
|
|
|
// Byte-wise access to [scoredat_routes].
|
|
|
|
inline int8_t& scoredat_route_byte(int place, int byte)
|
|
|
|
{
|
2020-05-22 17:27:50 +00:00
|
|
|
if(byte == 0) {
|
|
|
|
return scoredat_routes[place * SCOREDAT_ROUTE_LEN];
|
|
|
|
}
|
|
|
|
return scoredat_routes[(place * SCOREDAT_ROUTE_LEN) + byte];
|
2020-05-14 18:52:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Null-terminated version of scoredat_name_t, used internally.
|
2021-09-06 18:16:39 +00:00
|
|
|
typedef StupidBytewiseWrapperAround<struct {
|
2020-05-14 18:52:40 +00:00
|
|
|
int16_t codepoint[SCOREDAT_NAME_KANJI + 1];
|
2021-09-06 18:16:39 +00:00
|
|
|
}> scoredat_name_z_t;
|
2020-05-14 18:52:40 +00:00
|
|
|
|
2020-03-20 19:24:39 +00:00
|
|
|
// Loads the score file for the current [rank], recreating it if necessary.
|
|
|
|
// Returns 0 on success, 1 on failure.
|
|
|
|
int scoredat_load();
|
|
|
|
|
|
|
|
// Returns the high score for the difficulty previously loaded by
|
|
|
|
// scoredat_load().
|
2020-03-20 17:40:56 +00:00
|
|
|
uint32_t scoredat_hiscore_get();
|
|
|
|
|
|
|
|
// Sets [str] to the null-terminated name at the given [place] for the
|
2020-03-20 19:24:39 +00:00
|
|
|
// difficulty previously loaded by scoredat_load().
|
2020-03-20 17:40:56 +00:00
|
|
|
void scoredat_name_get(int place, char str[SCOREDAT_NAME_BYTES + 1]);
|