extern const char SCOREDAT_ROUTE_NONE[]; extern char scoredat_name_default[]; extern const char SCOREDAT_MAGIC_0[]; extern const char SCOREDAT_MAGIC_1[]; extern const char SCOREDAT_FN_EASY_0[]; extern const char SCOREDAT_FN_EASY_1[]; extern const char SCOREDAT_FN_NORMAL_0[]; extern const char SCOREDAT_FN_NORMAL_1[]; extern const char SCOREDAT_FN_HARD_0[]; extern const char SCOREDAT_FN_HARD_1[]; extern const char SCOREDAT_FN_LUNATIC_0[]; extern const char SCOREDAT_FN_LUNATIC_1[]; #define scoredat_fn(buf, inst) \ switch(rank) { \ case RANK_EASY: strcpy(fn, SCOREDAT_FN_EASY_##inst); break; \ case RANK_NORMAL: strcpy(fn, SCOREDAT_FN_NORMAL_##inst); break; \ case RANK_HARD: strcpy(fn, SCOREDAT_FN_HARD_##inst); break; \ case RANK_LUNATIC: strcpy(fn, SCOREDAT_FN_LUNATIC_##inst); break; \ } int8_t scoredat_name_byte_encode(int8_t byte) { return (byte + SCOREDAT_NAME_KEY); } int8_t scoredat_name_byte_decode(int8_t byte) { return (byte + (0x100 - SCOREDAT_NAME_KEY)); } #pragma warn -rch void scoredat_recreate() { int i; int16_t stage; const char *route; int32_t score; scoredat_declare(); // Will be name-encoded, and therefore modified in the .data section! char *name; const char *magic; char fn[16]; route = SCOREDAT_ROUTE_NONE; name = scoredat_name_default; magic = SCOREDAT_MAGIC_0; scoredat_fn(fn, 0); scoredat_cli(); if(scoredat_create(fn) == 0) { goto end; return; } scoredat_write(magic, sizeof(SCOREDAT_MAGIC) - 1); stage = SCOREDAT_PLACES; score = (SCOREDAT_PLACES * 100); for(i = 0; i < SCOREDAT_NAME_BYTES; i++) { name[i] = scoredat_name_byte_encode(name[i]); } for(i = 0; i < SCOREDAT_PLACES; i++) { scoredat_write(name, SCOREDAT_NAME_BYTES); } for(i = 0; i < SCOREDAT_PLACES; i++) { scoredat_write(&score, sizeof(score)); score = score - 100; } for(i = 0; i < SCOREDAT_PLACES; i++) { scoredat_write(&stage, sizeof(stage)); stage = stage - 1; } for(i = 0; i < SCOREDAT_PLACES; i++) { scoredat_write(route, sizeof(twobyte_t)); } scoredat_close(); end: scoredat_sti(); } int scoredat_load() { union { char magic[sizeof(SCOREDAT_MAGIC) - 1]; char space[50]; } buf; char fn[16]; scoredat_declare(); scoredat_fn(fn, 1); if(!scoredat_exist(fn)) { scoredat_recreate(); } scoredat_cli(); if(scoredat_ropen(fn) == 0) { scoredat_error(SCOREDAT_ERROR_NOT_FOUND); scoredat_sti(); return 1; } scoredat_read(buf.magic, sizeof(buf.magic)); // Who cares about the last three bytes anyway, right. if(memcmp(buf.magic, SCOREDAT_MAGIC_1, 4)) { scoredat_close(); scoredat_error(SCOREDAT_ERROR_INVALID); scoredat_sti(); return 1; } scoredat_names = new int8_t[SCOREDAT_NAMES_SIZE]; scoredat_stages = new int16_t[SCOREDAT_PLACES]; scoredat_routes = new int8_t[SCOREDAT_ROUTE_LEN * SCOREDAT_PLACES]; scoredat_score = new int32_t[SCOREDAT_PLACES]; scoredat_read(scoredat_names, SCOREDAT_NAMES_SIZE); scoredat_read(scoredat_score, sizeof(int32_t) * SCOREDAT_PLACES); scoredat_read(scoredat_stages, sizeof(int16_t) * SCOREDAT_PLACES); scoredat_read(scoredat_routes, SCOREDAT_ROUTE_LEN * SCOREDAT_PLACES); scoredat_close(); for(int i = 0; i < SCOREDAT_NAMES_SIZE; i++) { scoredat_names[i] = scoredat_name_byte_decode(scoredat_names[i]); } scoredat_sti(); return 0; }