Implement & match famicom_nesinfo

This commit is contained in:
Cuyler36
2024-01-22 16:40:00 -05:00
parent 7412daaecf
commit 0b157491f2
5 changed files with 886 additions and 24 deletions
+870 -24
View File
@@ -2,12 +2,15 @@
#include "_mem.h"
#include "dolphin/os.h"
#include "libultra/libultra.h"
#include "JSystem/JSystem.h"
#include "JSystem/JKernel/JKRFileLoader.h"
static u8 highscore_num = 0;
static u8 highscore_updated = 0;
static u8* highscore_flags = nullptr;
static u8 nesinfo_mcrd_cont_no = 0;
static u8* nesinfo_mcrd_game_name = nullptr;
static u8 nesinfo_mcrd_game_name[33];
static u8* nesinfo_ptr = nullptr;
static u32 nesinfo_tags_size = 0;
static u8* nesinfo_tags_start = nullptr;
@@ -25,7 +28,14 @@ static u32 nesinfo_expand_rom_size = 0;
static bool tcs_bad = false;
static bool ics_bad = false;
enum highscore_state { HIGHSCORE_STATE_UNSET, HIGHSCORE_STATE_SET, HIGHSCORE_STATE_2, HIGHSCORE_STATE_UNINITIALIZED };
// clang-format off
enum highscore_state {
HIGHSCORE_STATE_UNSET,
HIGHSCORE_STATE_SET,
HIGHSCORE_STATE_2,
HIGHSCORE_STATE_UNINITIALIZED
};
enum famicom_games {
FAMICOM_GAME_CLU_CLU_LAND,
@@ -33,7 +43,27 @@ enum famicom_games {
FAMICOM_GAME_DONKEY_KONG,
FAMICOM_GAME_DONKEY_KONG_JR_MATH,
FAMICOM_GAME_PINBALL,
FAMICOM_GAME_TENNIS,
FAMICOM_GAME_GOLF,
FAMICOM_GAME_PUNCHOUT,
FAMICOM_GAME_BASEBALL,
FAMICOM_GAME_CLU_CLU_LAND_D,
FAMICOM_GAME_DONKEY_KONG_3,
FAMICOM_GAME_DONKEY_KONG_JR,
FAMICOM_GAME_EXCITEBIKE,
FAMICOM_GAME_SOCCER,
FAMICOM_GAME_WARIOS_WOODS,
FAMICOM_GAME_ICE_CLIMBER,
FAMICOM_GAME_MARIO_BROS,
FAMICOM_GAME_SUPER_MARIO_BROS,
FAMICOM_GAME_LEGEND_OF_ZELDA_DISK,
FAMICOM_GAME_LEGEND_OF_ZELDA_ROM,
FAMICOM_GAME_GOMOKU_NARABE,
FAMICOM_GAME_MAHJONG,
FAMICOM_GAME_NUM,
FAMICOM_GAME_EXTERNAL = 31
};
#define END_TAG() 'E', 'N', 'D', 0
@@ -42,12 +72,17 @@ enum famicom_games {
#define GNO_TAG() 'G', 'N', 'O', sizeof(u8)
#define OFS_TAG() 'O', 'F', 'S', sizeof(u16)
#define HSC_TAG(l) 'H', 'S', 'C', (sizeof(u16) + l)
#define BBR_TAG() 'B', 'B', 'R', 4
#define QDS_TAG() 'Q', 'D', 'S', 5
#define SPE_TAG() 'S', 'P', 'E', 1
#define TCS_TAG() 'T', 'C', 'S', 2
#define ICS_TAG() 'I', 'C', 'S', 2
#define OFS_U16(ofs) ((u8)((u32)(ofs) >> 8), (u8)(ofs))
#define OFS_U16(ofs) (u8)((u32)(ofs) >> 8), (u8)(ofs)
#define OFS_U24(ofs) (u8)((u32)(ofs) >> 16), (u8)((u32)(ofs) >> 8), (u8)(ofs)
static u8 tags_table_default[] = { END_TAG() };
// clang-format off
static u8 tags_table_cluclu_land[] = {
GID_TAG(2), 'C', 'L',
GNM_TAG(12), 'C', 'L', 'U', ' ', 'C', 'L', 'U', ' ', 'L', 'A', 'N', 'D',
@@ -100,8 +135,275 @@ static u8 tags_table_pinball[] = {
GNM_TAG(7), 'P', 'I', 'N', 'B', 'A', 'L', 'L',
GNO_TAG(), FAMICOM_GAME_PINBALL,
OFS_TAG(), OFS_U16(0x0029),
HSC_TAG(2), OFS_U16(0x0507), 0, 0,
HSC_TAG(2), OFS_U16(0x0509), 0, 0,
HSC_TAG(5), OFS_U16(0x010E), 0, 0, 0, 0, 0,
HSC_TAG(5), OFS_U16(0x0113), 0, 0, 0, 0, 0,
END_TAG(),
};
static u8 tags_table_tennis[] = {
GID_TAG(2), 'T', 'E',
GNM_TAG(6), 'T', 'E', 'N', 'N', 'I', 'S',
GNO_TAG(), FAMICOM_GAME_TENNIS,
END_TAG(),
};
static u8 tags_table_golf[] = {
GID_TAG(2), 'G', 'F',
GNM_TAG(4), 'G', 'O', 'L', 'F',
GNO_TAG(), FAMICOM_GAME_GOLF,
OFS_TAG(), OFS_U16(0x0033),
HSC_TAG(1), OFS_U16(0x000E), 100,
END_TAG(),
};
static u8 tags_table_punchout[] = {
GID_TAG(2), 'P', 'T',
GNM_TAG(10), 'P', 'U', 'N', 'C', 'H', '-', 'O', 'U', 'T', '!',
GNO_TAG(), FAMICOM_GAME_PUNCHOUT,
OFS_TAG(), OFS_U16(0x0034),
HSC_TAG(10), OFS_U16(0x0110), 39, 39, 39, 39, 39, 39, 39, 39, 39, 39,
END_TAG(),
};
static u8 tags_table_baseball[] = {
GID_TAG(2), 'B', 'A',
GNM_TAG(6), 'B', 'A', 'S', 'E', 'B', 'A', 'L', 'L', // @BUG - should be GNM_TAG(8) ??
GNO_TAG(), FAMICOM_GAME_BASEBALL,
END_TAG(),
};
static u8 tags_table_cluclu_land_disk[] = {
GID_TAG(3), 'C', 'L', 'D',
GNM_TAG(14), 'C', 'L', 'U', ' ', 'C', 'L', 'U', ' ', 'L', 'A', 'N', 'D', ' ', 'D',
GNO_TAG(), FAMICOM_GAME_CLU_CLU_LAND_D,
OFS_TAG(), OFS_U16(0x003E),
QDS_TAG(), OFS_U24(0x00A170), OFS_U16(0x0050), // save 0x50 bytes starting at 0xA170
END_TAG(),
};
static u8 tags_table_donkey_kong_3[] = {
GID_TAG(2), 'D', 'T',
GNM_TAG(13), 'D', 'O', 'N', 'K', 'E', 'Y', ' ', 'K', 'O', 'N', 'G', ' ', '3',
GNO_TAG(), FAMICOM_GAME_DONKEY_KONG_3,
OFS_TAG(), OFS_U16(0x008E),
HSC_TAG(2), OFS_U16(0x008C), 0, 0,
HSC_TAG(2), OFS_U16(0x008E), 0, 0,
END_TAG(),
};
static u8 tags_table_donkey_kong_jr[] = {
GID_TAG(2), 'J', 'R',
GNM_TAG(14), 'D', 'O', 'N', 'K', 'E', 'Y', ' ', 'K', 'O', 'N', 'G', ' ', 'J', 'R',
GNO_TAG(), FAMICOM_GAME_DONKEY_KONG_JR,
OFS_TAG(), OFS_U16(0x0092),
HSC_TAG(2), OFS_U16(0x05D6), 0, 0,
HSC_TAG(2), OFS_U16(0x05D8), 0, 0,
END_TAG(),
};
static u8 tags_table_gomoku_narabe[] = {
GID_TAG(2), 'G', 'O',
GNM_TAG(13), 'G', 'O', 'M', 'O', 'K', 'U', ' ', 'N', 'A', 'R', 'A', 'B', 'E',
GNO_TAG(), FAMICOM_GAME_GOMOKU_NARABE,
END_TAG(),
};
static u8 tags_table_mahjong[] = {
GID_TAG(2), 'M', 'J',
GNM_TAG(6), 'M', 'A', 'H', 'J', 'O', 'N', 'G', // @BUG - should be GNM_TAG(7)
GNO_TAG(), FAMICOM_GAME_MAHJONG,
END_TAG(),
};
static u8 tags_table_wario_no_mori[] = {
GID_TAG(2), 'U', 'W',
GNM_TAG(13), 'W', 'A', 'R', 'I', 'O', '\'', 's', ' ', 'W', 'O', 'O', 'D', 'S',
GNO_TAG(), FAMICOM_GAME_WARIOS_WOODS,
OFS_TAG(), OFS_U16(0x00D8),
BBR_TAG(), OFS_U16(0x1B00), OFS_U16(0x0004), // 4 bytes from 0x1B00
BBR_TAG(), OFS_U16(0x1B10), OFS_U16(0x0034), // 34 bytes from 0x1B10
BBR_TAG(), OFS_U16(0x1B50), OFS_U16(0x0010), // 16 bytes from 0x1B50
BBR_TAG(), OFS_U16(0x1B67), OFS_U16(0x0003), // 3 bytes from 0x1B67
BBR_TAG(), OFS_U16(0x1B70), OFS_U16(0x0002), // 2 bytes from 0x1B70
BBR_TAG(), OFS_U16(0x1B74), OFS_U16(0x0002), // 2 bytes from 0x1B74
/* Was this data saved at a later date and therefore had to have its offset much further? */
OFS_TAG(), OFS_U16(0x0655),
BBR_TAG(), OFS_U16(0x1B6D), OFS_U16(0x0002), // 2 bytes from 0x1B6D
END_TAG(),
};
static u8 tags_table_ice_climber[] = {
GID_TAG(2), 'I', 'C',
GNM_TAG(11), 'I', 'C', 'E', ' ', 'C', 'L', 'I', 'M', 'B', 'E', 'R',
GNO_TAG(), FAMICOM_GAME_ICE_CLIMBER,
OFS_TAG(), OFS_U16(0x0128),
HSC_TAG(3), OFS_U16(0x87ED), 0, 0, 0,
END_TAG(),
};
static u8 tags_table_mario_bros[] = {
GID_TAG(2), 'M', 'A',
GNM_TAG(11), 'M', 'A', 'R', 'I', 'O', ' ', 'B', 'R', 'O', 'S', '.',
GNO_TAG(), FAMICOM_GAME_MARIO_BROS,
OFS_TAG(), OFS_U16(0x012B),
HSC_TAG(3), OFS_U16(0x0091), 0, 0, 0,
END_TAG(),
};
static u8 tags_table_super_mario_bros[] = {
GID_TAG(2), 'S', 'M',
GNM_TAG(16), 'S', 'U', 'P', 'E', 'R', ' ', 'M', 'A', 'R', 'I', 'O', ' ', 'B', 'R', 'O', 'S',
GNO_TAG(), FAMICOM_GAME_SUPER_MARIO_BROS,
OFS_TAG(), OFS_U16(0x0131),
HSC_TAG(6), OFS_U16(0x87D7), 0, 0, 0, 0, 0, 0,
HSC_TAG(1), OFS_U16(0x07FC), 0,
HSC_TAG(1), OFS_U16(0x87FD), 0,
END_TAG(),
};
static u8 tags_table_legend_of_zelda_disk[] = {
GID_TAG(2), 'Z', 'L',
GNM_TAG(15), 'L', 'E', 'G', 'E', 'N', 'D', ' ', 'O', 'F', ' ', 'Z', 'E', 'L', 'D', 'A',
GNO_TAG(), FAMICOM_GAME_LEGEND_OF_ZELDA_DISK,
OFS_TAG(), OFS_U16(0x0139),
QDS_TAG(), OFS_U24(0x01BAFD), OFS_U16(0x051C), // Save 0x51C bytes from 0x1BAFD
END_TAG(),
};
static u8 tags_table_legend_of_zelda_rom[] = {
GID_TAG(2), 'Z', 'L',
GNM_TAG(16), 'L', 'E', 'G', 'E', 'N', 'D', ' ', 'O', 'F', ' ', 'Z', 'E', 'L', 'D', 'A', '1',
GNO_TAG(), FAMICOM_GAME_LEGEND_OF_ZELDA_ROM,
OFS_TAG(), OFS_U16(0x0139),
BBR_TAG(), OFS_U16(0x0002), OFS_U16(0x051C), // Save 0x51C bytes from BBR+0x0002
SPE_TAG(), 1,
END_TAG(),
};
static u8 tags_table_soccer[] = {
GID_TAG(2), 'S', 'C',
GNM_TAG(6), 'S', 'O', 'C', 'C', 'E', 'R',
GNO_TAG(), FAMICOM_GAME_SOCCER,
END_TAG(),
};
static u8 tags_table_excite_bike[] = {
GID_TAG(2), 'E', 'B',
GNM_TAG(10), 'E', 'X', 'C', 'I', 'T', 'E', 'B', 'I', 'K', 'E',
GNO_TAG(), FAMICOM_GAME_EXCITEBIKE,
OFS_TAG(), OFS_U16(0x0096),
HSC_TAG(3), OFS_U16(0x8580), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x8583), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x8586), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x8589), 1, 12, 0,
HSC_TAG(3), OFS_U16(0x858C), 1, 6, 0,
HSC_TAG(3), OFS_U16(0x8590), 1, 24, 0,
HSC_TAG(3), OFS_U16(0x8593), 1, 22, 0,
HSC_TAG(3), OFS_U16(0x8596), 1, 10, 0,
HSC_TAG(3), OFS_U16(0x8599), 1, 20, 0,
HSC_TAG(3), OFS_U16(0x859C), 1, 6, 0,
HSC_TAG(3), OFS_U16(0x85A0), 10, 0, 0,
HSC_TAG(3), OFS_U16(0x85A3), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x85A6), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x85A9), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x85AC), 1, 12, 0,
HSC_TAG(3), OFS_U16(0x85AF), 1, 6, 0,
HSC_TAG(3), OFS_U16(0x85B3), 1, 24, 0,
HSC_TAG(3), OFS_U16(0x85B6), 1, 22, 0,
HSC_TAG(3), OFS_U16(0x85B9), 1, 10, 0,
HSC_TAG(3), OFS_U16(0x85BC), 1, 20, 0,
HSC_TAG(3), OFS_U16(0x85BF), 1, 6, 0,
HSC_TAG(3), OFS_U16(0x85C3), 10, 0, 0,
END_TAG(),
};
static u8 tags_table_excite_bike_design[] = {
GID_TAG(2), 'E', 'B',
GNM_TAG(10), 'E', 'X', 'C', 'I', 'T', 'E', 'B', 'I', 'K', 'E',
GNO_TAG(), FAMICOM_GAME_EXTERNAL,
OFS_TAG(), OFS_U16(0x0000),
HSC_TAG(3), OFS_U16(0x8580), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x8583), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x8586), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x8589), 1, 12, 0,
HSC_TAG(3), OFS_U16(0x858C), 1, 6, 0,
HSC_TAG(3), OFS_U16(0x8590), 1, 24, 0,
HSC_TAG(3), OFS_U16(0x8593), 1, 22, 0,
HSC_TAG(3), OFS_U16(0x8596), 1, 10, 0,
HSC_TAG(3), OFS_U16(0x8599), 1, 20, 0,
HSC_TAG(3), OFS_U16(0x859C), 1, 6, 0,
HSC_TAG(3), OFS_U16(0x85A0), 10, 0, 0,
HSC_TAG(3), OFS_U16(0x85A3), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x85A6), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x85A9), 1, 16, 0,
HSC_TAG(3), OFS_U16(0x85AC), 1, 12, 0,
HSC_TAG(3), OFS_U16(0x85AF), 1, 6, 0,
HSC_TAG(3), OFS_U16(0x85B3), 1, 24, 0,
HSC_TAG(3), OFS_U16(0x85B6), 1, 22, 0,
HSC_TAG(3), OFS_U16(0x85B9), 1, 10, 0,
HSC_TAG(3), OFS_U16(0x85BC), 1, 20, 0,
HSC_TAG(3), OFS_U16(0x85BF), 1, 6, 0,
HSC_TAG(3), OFS_U16(0x85C3), 10, 0, 0,
HSC_TAG(240), OFS_U16(0x86E0), 1, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 64, 9, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,
END_TAG(),
};
static u8 tags_table_nazo_no_murasamejo[] = {
GID_TAG(3), 'N', 'M', 'J',
GNM_TAG(16), 'N', 'A', 'Z', 'O', 'N', 'O', 'M', 'U', 'R', 'A', 'S', 'A', 'M', 'E', 'J', 'O',
GNO_TAG(), FAMICOM_GAME_EXTERNAL,
OFS_TAG(), OFS_U16(0x0000),
QDS_TAG(), OFS_U24(0x009AC7), OFS_U16(0x003E),
END_TAG(),
};
static u8 tags_table_super_mario_bros_2[] = {
TCS_TAG(), OFS_U16(0x0000),
ICS_TAG(), OFS_U16(0x0000),
GID_TAG(4), 'S', 'M', 'B', '2',
GNM_TAG(18), 'S', 'U', 'P', 'E', 'R', ' ', 'M', 'A', 'R', 'I', 'O', ' ', 'B', 'R', 'O', 'S', ' ', '2',
GNO_TAG(), FAMICOM_GAME_EXTERNAL,
OFS_TAG(), OFS_U16(0x0000),
QDS_TAG(), OFS_U24(0x01CC4E), OFS_U16(0x0001),
HSC_TAG(6), OFS_U16(0x87D7), 0, 0, 0, 0, 0, 0,
END_TAG(),
};
static u8 tags_table_marioopen_golf[] = {
TCS_TAG(),
ICS_TAG(),
GID_TAG(2), 'U', 'G',
GNM_TAG(14), 'M', 'A', 'R', 'I', 'O', 'O', 'P', 'E', 'N', ' ', 'G', 'O', 'L', 'F',
GNO_TAG(), FAMICOM_GAME_EXTERNAL,
OFS_TAG(), OFS_U16(0x0000),
BBR_TAG(), OFS_U16(0x0001), OFS_U16(0x0E0C),
END_TAG(),
};
static u8 tags_table_external_default[] = {
TCS_TAG(), OFS_U16(0x0000),
ICS_TAG(), OFS_U16(0x0000),
GNO_TAG(), FAMICOM_GAME_EXTERNAL,
END_TAG(),
};
static u8 tags_table_balloon_fight_jpn[] = {
TCS_TAG(), OFS_U16(0x0000),
ICS_TAG(), OFS_U16(0x0000),
GID_TAG(2), 'B', 'F',
GNM_TAG(13), 'B', 'A', 'L', 'L', 'O', 'O', 'N', ' ', 'F', 'I', 'G', 'H', 'T',
GNO_TAG(), FAMICOM_GAME_EXTERNAL,
OFS_TAG(), OFS_U16(0x0000),
HSC_TAG(5), OFS_U16(0x8629), 0, 0, 0, 1, 0,
HSC_TAG(5), OFS_U16(0x862E), 0, 0, 0, 1, 0,
HSC_TAG(5), OFS_U16(0x8633), 0, 0, 5, 2, 0,
END_TAG(),
};
@@ -112,15 +414,53 @@ typedef struct nesinfo_tag_s {
char* kanjiName;
} nesinfo_tag_info;
static nesinfo_tag_info tags_table_table[FAMICOM_INTERNAL_ROM_NUM];
static const nesinfo_tag_info tags_table_table[FAMICOM_INTERNAL_ROM_NUM] = {
{ tags_table_cluclu_land, sizeof(tags_table_cluclu_land), "Clu Clu Land", "Clu Clu Land" },
{ tags_table_balloon_fight, sizeof(tags_table_balloon_fight), "Balloon Fight", "Balloon Fight" },
{ tags_table_donkey_kong, sizeof(tags_table_donkey_kong), "Donkey Kong", "Donkey Kong" },
{ tags_table_sansu_asobi, sizeof(tags_table_sansu_asobi), "Donkey Jr Math", "Donkey Jr Math" },
{ tags_table_pinball, sizeof(tags_table_pinball), "Pinball", "Pinball" },
{ tags_table_tennis, sizeof(tags_table_tennis), "Tennis", "Tennis" },
{ tags_table_golf, sizeof(tags_table_golf), "Golf", "Golf" },
{ tags_table_punchout, sizeof(tags_table_punchout), "Punch Out", "Punch Out" },
{ tags_table_baseball, sizeof(tags_table_baseball), "Base Ball", "Base Ball" },
{ tags_table_cluclu_land_disk, sizeof(tags_table_cluclu_land_disk), "New Clu Clu Land", "New Clu Clu Land" },
{ tags_table_donkey_kong_3, sizeof(tags_table_donkey_kong_3), "Donkey Kong 3", "Donkey Kong 3" },
{ tags_table_donkey_kong_jr, sizeof(tags_table_donkey_kong_jr), "Donkey Kong Jr", "Donkey Kong Jr" },
{ tags_table_soccer, sizeof(tags_table_soccer), "Soccer", "Soccer" },
{ tags_table_excite_bike, sizeof(tags_table_excite_bike), "Excitebike", "Excitebike" },
{ tags_table_wario_no_mori, sizeof(tags_table_wario_no_mori), "Wario's Woods", "Wario's Woods" },
{ tags_table_ice_climber, sizeof(tags_table_ice_climber), "Ice Climber", "Ice Climber" },
{ tags_table_mario_bros, sizeof(tags_table_mario_bros), "Mario Bros", "Mario Bros" },
{ tags_table_super_mario_bros, sizeof(tags_table_super_mario_bros), "Super Mario Bros", "Super Mario Bros" },
{ tags_table_legend_of_zelda_rom, sizeof(tags_table_legend_of_zelda_rom), "Legend of Zelda", "Legend of Zelda" },
};
static u8 moriNameTable[FAMICOM_INTERNAL_ROM_NUM][16] = {
{ 'C', 'l', 'u', ' ', 'C', 'l', 'u', ' ', 'L', 'a', 'n', 'd', ' ', ' ', ' ', ' ', },
{ 'B', 'a', 'l', 'l', 'o', 'o', 'n', ' ', 'F', 'i', 'g', 'h', 't', ' ', ' ', ' ', },
{ 'D', 'o', 'n', 'k', 'e', 'y', ' ', 'K', 'o', 'n', 'g', ' ', ' ', ' ', ' ', ' ', },
{ 'D', 'o', 'n', 'k', 'e', 'y', ' ', 'J', 'r', ' ', 'M', 'a', 't', 'h', ' ', ' ', },
{ 'P', 'i', 'n', 'b', 'a', 'l', 'l', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', },
{ 'T', 'e', 'n', 'n', 'i', 's', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', },
{ 'G', 'o', 'l', 'f', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', },
{ 'P', 'u', 'n', 'c', 'h', ' ', 'O', 'u', 't', ' ', ' ', ' ', ' ', ' ', ' ', ' ', },
{ 'B', 'a', 's', 'e', ' ', 'B', 'a', 'l', 'l', ' ', ' ', ' ', ' ', ' ', ' ', ' ', },
{ 'N', 'e', 'w', ' ', 'C', 'l', 'u', ' ', 'C', 'l', 'u', ' ', 'L', 'a', 'n', 'd', },
{ 'D', 'o', 'n', 'k', 'e', 'y', ' ', 'K', 'o', 'n', 'g', ' ', '3', ' ', ' ', ' ', },
{ 'D', 'o', 'n', 'k', 'e', 'y', ' ', 'K', 'o', 'n', 'g', ' ', 'J', 'r', ' ', ' ', },
{ 'S', 'o', 'c', 'c', 'e', 'r', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', ' ', },
{ 'E', 'x', 'c', 'i', 't', 'e', 'b', 'i', 'k', 'e', ' ', ' ', ' ', ' ', ' ', ' ', },
{ 'W', 'a', 'r', 'i', 'o', '\'', 's', ' ', 'W', 'o', 'o', 'd', 's', ' ', ' ', ' ', },
{ 'I', 'c', 'e', ' ', 'C', 'l', 'i', 'm', 'b', 'e', 'r', ' ', ' ', ' ', ' ', ' ', },
{ 'M', 'a', 'r', 'i', 'o', ' ', 'B', 'r', 'o', 's', ' ', ' ', ' ', ' ', ' ', ' ', },
{ 'S', 'u', 'p', 'e', 'r', ' ', 'M', 'a', 'r', 'i', 'o', ' ', 'B', 'r', 'o', 's', },
{ 'L', 'e', 'g', 'e', 'n', 'd', ' ', 'o', 'f', ' ', 'Z', 'e', 'l', 'd', 'a', ' ', },
};
// clang-format on
static u8 moriNameTable[FAMICOM_INTERNAL_ROM_NUM][16] = {
// TODO
};
static void update_highscore_raw(u32 ofs, u32 size, u8* initial_value, u8* high_score, u8* state) {
static void update_highscore_raw(unsigned int ofs, unsigned int size, u8* initial_value, u8* high_score, u8* state) {
u8* current_score_p = &famicomCommon.sp->wram[ofs];
if (*state == HIGHSCORE_STATE_UNINITIALIZED) {
@@ -158,7 +498,7 @@ static u8 nesinfo_get_u8(u8* data) {
return (u8)(u32)data[0];
}
static void nesinfo_set_u16(u8* data, u16* value) {
static void nesinfo_set_u16(u8* data, u16 value) {
data[0] = (u8)(((u32)value >> 8) & 0xFF);
data[1] = (u8)((u32)value & 0xFF);
}
@@ -168,7 +508,7 @@ static u8* nesinfo_next_tag_raw(u8* data) {
memcmp(data, NESTAG_END, NESTAG_CMD_SIZE) == 0) {
return nullptr;
} else {
return &data[data[NESTAG_CMD_SIZE] + NESTAG_SIZE];
return data + data[NESTAG_CMD_SIZE] + NESTAG_SIZE;
}
}
@@ -180,11 +520,11 @@ static u8* nesinfo_next_tag(u8* data) {
}
if (memcmp(data, NESTAG_VEQ, NESTAG_CMD_SIZE) == 0) {
if (data[NESTAG_SIZE + 1] != 1) {
if (data[4] != 1) {
data = nesinfo_next_tag_raw(data);
}
} else if (memcmp(data, NESTAG_VNE, NESTAG_CMD_SIZE) == 0) {
if (data[NESTAG_SIZE + 1] == 1) {
if (data[4] == 1) {
data = nesinfo_next_tag_raw(data);
}
} else {
@@ -260,17 +600,523 @@ extern void nesinfo_tags_set(int rom_no) {
return;
}
if (rom_no < 0 || rom_no >= FAMICOM_INTERNAL_ROM_NUM) {
if (rom_no < 0) {
return;
}
nesinfo_tags_start = tags_table_table[rom_no].tags_table;
nesinfo_tags_end = nesinfo_tags_start + tags_table_table[rom_no].tags_size;
nesinfo_tags_size = (u32)(nesinfo_tags_end - nesinfo_tags_start);
nesinfo_tags_gameName = tags_table_table[rom_no].gameName;
nesinfo_tags_kanjiName = tags_table_table[rom_no].kanjiName;
u32 rom = rom_no;
if (rom >= FAMICOM_INTERNAL_ROM_NUM) {
return;
}
if (rom_no < FAMICOM_INTERNAL_ROM_NUM) {
nesinfo_tags_moriName = moriNameTable[rom_no];
nesinfo_tags_start = tags_table_table[rom].tags_table;
nesinfo_tags_end = nesinfo_tags_start + tags_table_table[rom].tags_size;
nesinfo_tags_size = (u32)(nesinfo_tags_end - nesinfo_tags_start);
nesinfo_tags_gameName = tags_table_table[rom].gameName;
nesinfo_tags_kanjiName = tags_table_table[rom].kanjiName;
if (rom < FAMICOM_INTERNAL_ROM_NUM) {
nesinfo_tags_moriName = moriNameTable[rom];
}
}
static int calcSum(u8* data, size_t len) {
int checksum = 0;
size_t i;
for (i = 0; i != len; i++) {
checksum += *data++;
}
return checksum;
}
static void special_zelda() {
u8* bbram_p = famicomCommon.bbramp;
int i;
bbram_p[0] = 0x00; // First byte in bbram is not used
bbram_p[1] = 0x5A; // Second byte must be 'Z' (0x5A)
memset(&bbram_p[0x52A], 0xFF, 3); // Reset 'IsSaveFileBCommitted'
bzero(&bbram_p[0x52D], 3); // Unused 3 bytes after save file?
memset(&bbram_p[0x51E], 0x5A, 3); // Reset save file open markers
memset(&bbram_p[0x521], 0xA5, 3); // Reset save file close markers
for (i = 0; i < 3; i++) {
/* Sum previous checksums plus death counter, quest number, and whatever 0x518 is */
u32 checksum = 0;
u16 chksum16;
u8 chksumUpper;
u8 embeddedChksumUpper;
u8 embeddedChksumLower;
checksum += (u16)calcSum(bbram_p + 2 + i * 8, 8); /* Checksum of name */
checksum += (u16)calcSum(bbram_p + 0x1A + i * 0x28, 0x28); /* Checksum of items */
checksum += (u16)calcSum(bbram_p + 0x92 + i * 0x180, 0x180); /* Checksum of world flags */
checksum += (u16)calcSum(bbram_p + 0x512 + i, 1); /* Checksum of 'SlotActive' */
checksum += bbram_p[0x515 + i];
checksum += bbram_p[0x518 + i];
checksum += bbram_p[0x51B + i];
chksum16 = (u16)checksum;
/* Compare against embedded file checksums */
chksumUpper = checksum >> 8;
if ((embeddedChksumUpper = bbram_p[0x524 + i * 2], embeddedChksumUpper == chksumUpper) &&
(embeddedChksumLower = bbram_p[0x525 + i * 2], embeddedChksumLower == (u8)checksum)) {
OSReport("チェックサムOK %d %04x\n", i, chksum16); // Checksum OK %d %04x\n
} else {
OSReport("チェックサムNG %d %04x %02x%02x\n", i, chksum16, bbram_p[0x524 + i * 2],
bbram_p[0x525 + i * 2]); // Checksum NG %d %04x %02x%02x\n
bbram_p[0x524 + i * 2] = chksumUpper;
bbram_p[0x525 + i * 2] = checksum;
}
}
}
extern void nesinfo_tag_process1(u8* save_data, int mode, u32* max_ofs_p) {
u8* data = nesinfo_tags_start;
int first_play = FALSE;
int cur_ofs = 0;
int max_ofs = 0;
if (mode == 0) {
highscore_num = 0;
}
OSReport("nesinfo_tag_process1 開始 mode = %d\n", mode); // nesinfo_tag_process1 start mode = %d\n
while (TRUE) {
OSReport("タグ=%.3s 長さ=%d\n", data, data[NESTAG_CMD_SIZE]); // Tag=%.3s Length=%d\n
data = nesinfo_next_tag(data);
if (data == nullptr) {
break;
}
if (memcmp(data, NESTAG_VEQ, NESTAG_CMD_SIZE) == 0) {
if (data[4] != 1) {
data = nesinfo_next_tag(data);
}
} else if (memcmp(data, NESTAG_VNE, NESTAG_CMD_SIZE) == 0) {
if (data[4] == 1) {
data = nesinfo_next_tag(data);
}
} else if (memcmp(data, NESTAG_GID, NESTAG_CMD_SIZE) == 0) {
print_stringn_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_GNM, NESTAG_CMD_SIZE) == 0) {
print_stringn_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_CPN, NESTAG_CMD_SIZE) == 0) {
memcpy(nesinfo_mcrd_game_name, &data[4], 16);
} else if (memcmp(data, NESTAG_OFS, NESTAG_CMD_SIZE) == 0) {
cur_ofs = nesinfo_get_u16(&data[4]);
} else if (memcmp(data, NESTAG_HSC, NESTAG_CMD_SIZE) == 0) {
size_t highscore_size = data[NESTAG_CMD_SIZE] - 2;
OSReport("ハイスコア: オフセット=%04x 初期値=",
nesinfo_get_u16(&data[4])); // High Score: Offset=%04x Default Value=
if (save_data != nullptr && mode == 1 && first_play) {
u16 ofs = nesinfo_get_u16(&data[4]); // unnecessary?
memcpy(save_data + 8 + cur_ofs, &data[6], highscore_size);
OSReport("初プレイでハイスコアデータがないので初期化しました\n"); // Since it's the first play and
// there's no high score data, it has
// been initialized.\n
}
cur_ofs += highscore_size;
if (mode == 0) {
highscore_num++;
}
print_hex_lf(&data[6], data[NESTAG_CMD_SIZE] - 2);
} else if (memcmp(data, NESTAG_GNO, NESTAG_CMD_SIZE) == 0) {
u32 bit = 1 << nesinfo_get_u8(&data[4]);
/* TODO: is save_data actually a struct? It has dynamic sized members in it. */
if (save_data != 0 && mode == 1 && ((*(u32*)&save_data[4]) & bit) == 0) {
(*(u32*)&save_data[4]) |= bit; // set first play bit
first_play = TRUE;
}
print_hex_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_BBR, NESTAG_CMD_SIZE) == 0) {
u16 bbram_ofs = nesinfo_get_u16(&data[4]);
u16 size = nesinfo_get_u16(&data[6]);
if (save_data != nullptr && mode == 1) {
if (first_play) {
OSReport("初プレイなので。クリアします\n"); // Since it's the first play, it will be cleared.\n
bzero(famicomCommon.bbramp + bbram_ofs, size);
} else {
OSReport("バッテリーバックアップのロード\n"); // Loading battery backup.\n
memcpy(famicomCommon.bbramp + bbram_ofs, save_data + 8 + cur_ofs, size);
}
}
cur_ofs += size;
print_hex_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_QDS, NESTAG_CMD_SIZE) == 0) {
u8 ofs_16_24 = nesinfo_get_u8(&data[4]);
u16 ofs_00_16 = nesinfo_get_u16(&data[5]);
u16 size = nesinfo_get_u16(&data[7]);
if (save_data != nullptr && mode == 1) {
if (first_play) {
OSReport("初プレイなのでディスクセーブエリアを保持します\n"); // Since it's the first play, the disk
// save area will be retained.\n
} else {
OSReport("ディスクセーブエリアをロード\n"); // Loading the disk save area.\n
memcpy(famicomCommon.nesromp + (ofs_16_24 << 16) + ofs_00_16, save_data + 8 + cur_ofs, size);
}
}
cur_ofs += size;
print_hex_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_SPE, NESTAG_CMD_SIZE) == 0) {
if (mode == 1) {
OSReport("特別プログラム発動!\n"); // Special program activated!\n
if (data[4] == 1 && first_play == FALSE) {
special_zelda(); // fixup zelda checksums
}
}
print_hex_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_ISZ, NESTAG_CMD_SIZE) == 0) {
print_hex_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_IFM, NESTAG_CMD_SIZE) == 0) {
print_hex_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_REM, NESTAG_CMD_SIZE) == 0) {
print_stringn_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_END, NESTAG_CMD_SIZE) == 0) {
break;
} else if (memcmp(data, NESTAG_TCS, NESTAG_CMD_SIZE) == 0) {
if (mode == 0 || mode == -1) {
u16 checksum = calc_check_sum2(nesinfo_tags_start, nesinfo_tags_size);
if (mode == -1) {
u16 chk = nesinfo_get_u16(&data[4]);
nesinfo_set_u16(&data[4], chk - checksum); // update tags checksum
} else {
if (checksum != 0) {
OSReport(VT_COL(VT_COLOR_YELLOW, VT_COLOR_BLACK) "BAD %04x\n" VT_RST, checksum);
tcs_bad = true;
break;
} else {
OSReport("OK\n");
}
}
}
} else if (memcmp(data, NESTAG_ICS, NESTAG_CMD_SIZE) == 0) {
if (mode == 0 || mode == -1) {
u16 checksum = calc_check_sum2(nesinfo_data_start, nesinfo_data_size);
if (mode == -1) {
nesinfo_set_u16(&data[4], checksum); // update image checksum
} else {
u16 chk = nesinfo_get_u16(&data[4]);
if (checksum != chk) {
ics_bad = true;
OSReport(VT_COL(VT_COLOR_YELLOW, VT_COLOR_BLACK) "BAD %04x %04x\n" VT_RST, checksum, chk);
} else {
OSReport("OK\n");
}
}
}
} else if (memcmp(data, NESTAG_ESZ, NESTAG_CMD_SIZE) == 0) {
nesinfo_expand_rom_size = nesinfo_get_u16(&data[4]) << 4;
} else if (memcmp(data, NESTAG_FIL, NESTAG_CMD_SIZE) == 0) {
// stubbed
} else if (memcmp(data, NESTAG_ROM, NESTAG_CMD_SIZE) == 0) {
u8 rom_id = data[4];
u8* rom_p = famicomCommon.nesromp;
OSReport("ロムデータ参照: %d\n", rom_id); // Referencing ROM data: %d\n
if (nesrom_filename_ptrs != nullptr && rom_p != nullptr) {
nesinfo_data_size = JKRFileLoader::readGlbResource(rom_p, 0x100000, nesrom_filename_ptrs[rom_id],
EXPAND_SWITCH_DECOMPRESS);
nesinfo_data_start = rom_p;
nesinfo_data_end = rom_p + nesinfo_data_size;
}
}
if (max_ofs < cur_ofs) {
max_ofs = cur_ofs;
}
}
OSReport("nesinfo_tag_process1 終了\n"); // nesinfo_tag_process1 finished\n
if (max_ofs_p != nullptr) {
*max_ofs_p = max_ofs;
}
}
extern void nesinfo_tag_process2() {
u8* data = nesinfo_tags_start;
OSReport("nesinfo_tag_process2 開始\n"); // nesinfo_tag_process2 start\n
while (TRUE) {
data = nesinfo_next_tag(data);
if (data == nullptr) {
break;
}
OSReport("タグ=%.3s 長さ=%d\n", data, data[NESTAG_CMD_SIZE]); // Tag=%.3s Length=%d\n
if (memcmp(data, NESTAG_VEQ, NESTAG_CMD_SIZE) == 0) {
if (data[4] != 1) {
data = nesinfo_next_tag(data);
}
} else if (memcmp(data, NESTAG_VNE, NESTAG_CMD_SIZE) == 0) {
if (data[4] == 1) {
data = nesinfo_next_tag(data);
}
} else if (memcmp(data, NESTAG_MOV, NESTAG_CMD_SIZE) == 0) {
u32 src = nesinfo_get_u16(&data[4]) << 4;
u32 dst = nesinfo_get_u16(&data[6]) << 4;
size_t size = nesinfo_get_u16(&data[8]) << 4;
if (nesinfo_rom_start != nullptr) {
memcpy(&nesinfo_rom_start[dst], &nesinfo_rom_start[src], size);
}
} else if (memcmp(data, NESTAG_NHD, NESTAG_CMD_SIZE) == 0) {
if (nesinfo_rom_start != nullptr) {
memcpy(nesinfo_rom_start, &data[4], data[NESTAG_CMD_SIZE]); // copy new header
}
} else if (memcmp(data, NESTAG_DIF, NESTAG_CMD_SIZE) == 0) {
u8* rom_p = nesinfo_rom_start;
if (rom_p != nullptr) {
int i;
size_t size = nesinfo_rom_start[4] * 0x4000 + nesinfo_rom_start[5] * 0x2000 + 16;
u8* rom_end_p = nesinfo_rom_start + size;
u8* dif_p = nesinfo_data_start;
while (rom_p < rom_end_p) {
rom_p += *dif_p++;
if (rom_p >= rom_end_p) {
break;
}
for (i = *dif_p++; i > 0; i--) {
*rom_p++ = *dif_p++;
}
if (rom_p >= rom_end_p) {
break;
}
}
}
} else if (memcmp(data, NESTAG_PAT, NESTAG_CMD_SIZE) == 0) {
int patch_type;
u8* next_tag_p;
u8* dst_p;
size_t patch_size;
u8* patch_data_p;
u32 patch_ofs;
next_tag_p = nesinfo_next_tag(data);
patch_data_p = &data[4];
while (patch_data_p < next_tag_p) {
patch_type = patch_data_p[0];
patch_size = patch_data_p[1];
patch_ofs = nesinfo_get_u16(&patch_data_p[2]);
patch_data_p += 4;
switch (patch_type) {
case 2: {
dst_p = &nesinfo_rom_start[patch_ofs];
break;
}
case 4:
case 5:
case 6:
case 7:
case 8:
break;
case 9: {
dst_p = &nesinfo_rom_start[patch_ofs << 4];
break;
}
case 3:
default: {
/* This gives us an effective patch application range of [0x80000000, 0x807FFFFF] */
if (patch_type >= 0x80 && patch_type <= 0xFF) {
dst_p = (u8*)((0x8000 + (patch_type - 0x80)) << 16) + patch_ofs;
} else {
dst_p = nullptr;
}
break;
}
}
if (dst_p != nullptr) {
while (patch_size-- > 0) {
*dst_p++ = *patch_data_p++;
}
} else {
patch_data_p += patch_size;
}
}
}
}
OSReport("nesinfo_tag_process2 終了\n"); // nesinfo_tag_process2 finished\n
}
extern void nesinfo_tag_process3(u8* save_data) {
u8* data = nesinfo_tags_start;
u32 cur_ofs = 0;
OSReport("nesinfo_tag_process3 開始\n"); // nesinfo_tag_process3 start\n
while (TRUE) {
data = nesinfo_next_tag(data);
if (data == nullptr) {
break;
}
OSReport("タグ=%.3s 長さ=%d\n", data, data[NESTAG_CMD_SIZE]); // Tag=%.3s Length=%d\n
if (memcmp(data, NESTAG_VEQ, NESTAG_CMD_SIZE) == 0) {
if (data[4] != 1) {
data = nesinfo_next_tag(data);
}
} else if (memcmp(data, NESTAG_VNE, NESTAG_CMD_SIZE) == 0) {
if (data[4] == 1) {
data = nesinfo_next_tag(data);
}
} else if (memcmp(data, NESTAG_OFS, NESTAG_CMD_SIZE) == 0) {
cur_ofs = nesinfo_get_u16(&data[4]);
} else if (memcmp(data, NESTAG_HSC, NESTAG_CMD_SIZE) == 0) {
cur_ofs = data[NESTAG_CMD_SIZE] + cur_ofs;
cur_ofs -= 2;
print_hex_lf(&data[6], data[NESTAG_CMD_SIZE] - 2);
} else if (memcmp(data, NESTAG_BBR, NESTAG_CMD_SIZE) == 0) {
u16 bbram_ofs = nesinfo_get_u16(&data[4]);
u16 size = nesinfo_get_u16(&data[6]);
if (save_data != nullptr) {
OSReport("バッテリーバックアップの保存\n"); // Saving battery backup.\n
memcpy(save_data + 8 + cur_ofs, famicomCommon.bbramp + bbram_ofs, size);
}
cur_ofs += size;
print_hex_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_QDS, NESTAG_CMD_SIZE) == 0) {
u8 ofs_16_24 = nesinfo_get_u8(&data[4]);
u16 ofs_00_16 = nesinfo_get_u16(&data[5]);
u16 size = nesinfo_get_u16(&data[7]);
if (save_data != nullptr) {
OSReport("ディスクセーブの保存\n"); // Saving the disk save.\n
memcpy(save_data + 8 + cur_ofs, famicomCommon.nesromp + (ofs_16_24 << 16) + ofs_00_16, size);
}
cur_ofs += size;
print_hex_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_SPE, NESTAG_CMD_SIZE) == 0) {
OSReport("特別プログラム発動!\n"); // Special program activated?\n
print_hex_lf(&data[4], data[NESTAG_CMD_SIZE]);
} else if (memcmp(data, NESTAG_END, NESTAG_CMD_SIZE) == 0) {
break;
}
}
OSReport("nesinfo_tag_process2 終了\n"); // nesinfo_tag_process2 finished\n
}
extern void nesinfo_update_highscore(u8* save_data, int mode) {
if (save_data != nullptr) {
u8* flags;
u8* data;
u32 cur_ofs;
cur_ofs = 0;
flags = highscore_flags;
data = nesinfo_tags_start;
while (TRUE) {
data = nesinfo_next_tag(data);
if (data == nullptr) {
break;
}
if (memcmp(data, NESTAG_VEQ, NESTAG_CMD_SIZE) == 0) {
if (data[4] != 1) {
data = nesinfo_next_tag(data);
}
} else if (memcmp(data, NESTAG_VNE, NESTAG_CMD_SIZE) == 0) {
if (data[4] == 1) {
data = nesinfo_next_tag(data);
}
} else if (memcmp(data, NESTAG_OFS, NESTAG_CMD_SIZE) == 0) {
cur_ofs = nesinfo_get_u16(&data[4]);
} else if (memcmp(data, NESTAG_HSC, NESTAG_CMD_SIZE) == 0) {
size_t highscore_size = data[NESTAG_CMD_SIZE] - 2;
u16 ofs_data = nesinfo_get_u16(&data[4]);
u32 ofs = ofs_data & 0x7FF;
u32 bit = (ofs_data >> 15) & 1;
u8* dst_p = &data[6];
if (save_data != nullptr) {
if (mode != 0) {
if (*flags == 1 && bit == 0) {
*flags = 0;
}
} else {
update_highscore_raw(ofs, highscore_size, dst_p, save_data + 8 + cur_ofs, flags);
}
}
cur_ofs += highscore_size;
flags++;
} else if (memcmp(data, NESTAG_BBR, NESTAG_CMD_SIZE) == 0) {
u16 bbram_ofs = nesinfo_get_u16(&data[4]);
u16 size = nesinfo_get_u16(&data[6]);
cur_ofs += size;
} else if (memcmp(data, NESTAG_QDS, NESTAG_CMD_SIZE) == 0) {
u8 ofs_16_24 = nesinfo_get_u8(&data[4]);
u16 ofs_00_16 = nesinfo_get_u16(&data[5]);
u16 size = nesinfo_get_u16(&data[7]);
cur_ofs += size;
} else if (memcmp(data, NESTAG_END, NESTAG_CMD_SIZE) == 0) {
break;
}
}
}
}
extern int nesinfo_get_highscore_num() {
return highscore_num;
}
extern u8* nesinfo_get_moriName() {
return nesinfo_tags_moriName;
}
extern void nesinfo_init() {
nesinfo_ptr = nullptr;
nesinfo_tags_size = 0;
nesinfo_tags_start = nullptr;
nesinfo_tags_end = nullptr;
nesinfo_data_size = 0;
nesinfo_data_start = nullptr;
nesinfo_data_end = nullptr;
nesinfo_rom_size = 0;
nesinfo_rom_start = nullptr;
nesinfo_rom_end = nullptr;
}
extern void highscore_setup_flags(u8* flags) {
highscore_flags = flags;
bzero(flags, highscore_num);
}