Transferpak: Add support for RAM saving, and RTC saving (Visual Boy Advance format)
Remove UpdateRTC() till I can get a better idea if the RTC code works properly.
This commit is contained in:
parent
74d3825870
commit
258a5a0eb8
|
@ -38,11 +38,8 @@ bool load_file(const char* filename, void** buffer, size_t* size)
|
||||||
FILE* fd;
|
FILE* fd;
|
||||||
size_t l_size;
|
size_t l_size;
|
||||||
void* l_buffer;
|
void* l_buffer;
|
||||||
int err;
|
|
||||||
bool ret;
|
|
||||||
|
|
||||||
/* open file */
|
/* open file */
|
||||||
ret = false;
|
|
||||||
fd = fopen(filename, "rb");
|
fd = fopen(filename, "rb");
|
||||||
if (fd == NULL)
|
if (fd == NULL)
|
||||||
{
|
{
|
||||||
|
@ -50,51 +47,48 @@ bool load_file(const char* filename, void** buffer, size_t* size)
|
||||||
}
|
}
|
||||||
|
|
||||||
/* obtain file size */
|
/* obtain file size */
|
||||||
ret = false;
|
if (fseek(fd, 0, SEEK_END) != 0)
|
||||||
err = fseek(fd, 0, SEEK_END);
|
|
||||||
if (err != 0)
|
|
||||||
{
|
{
|
||||||
goto close_file;
|
fclose(fd);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
err = ftell(fd);
|
l_size = (size_t)ftell(fd);
|
||||||
if (err == -1)
|
if (l_size == -1)
|
||||||
{
|
{
|
||||||
goto close_file;
|
fclose(fd);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
l_size = (size_t)err;
|
|
||||||
|
|
||||||
err = fseek(fd, 0, SEEK_SET);
|
if (fseek(fd, 0, SEEK_SET) != 0)
|
||||||
if (err != 0)
|
|
||||||
{
|
{
|
||||||
goto close_file;
|
fclose(fd);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* allocate buffer */
|
/* allocate buffer */
|
||||||
l_buffer = malloc(l_size);
|
l_buffer = malloc(l_size);
|
||||||
if (l_buffer == NULL)
|
if (l_buffer == NULL)
|
||||||
{
|
{
|
||||||
goto close_file;
|
fclose(fd);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* copy file content to buffer */
|
/* copy file content to buffer */
|
||||||
ret = false;
|
if (fread(l_buffer, 1, l_size, fd) != l_size)
|
||||||
err = fread(l_buffer, 1, l_size, fd);
|
|
||||||
if (err != l_size)
|
|
||||||
{
|
{
|
||||||
free(l_buffer);
|
free(l_buffer);
|
||||||
goto close_file;
|
fclose(fd);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* commit buffer,size */
|
/* commit buffer,size */
|
||||||
ret = true;
|
|
||||||
*buffer = l_buffer;
|
*buffer = l_buffer;
|
||||||
*size = l_size;
|
*size = l_size;
|
||||||
|
|
||||||
/* close file */
|
/* close file */
|
||||||
close_file:
|
|
||||||
fclose(fd);
|
fclose(fd);
|
||||||
return ret;
|
return true;
|
||||||
}
|
}
|
||||||
//--------------------------------------------------------------------------------------
|
//--------------------------------------------------------------------------------------
|
||||||
|
|
||||||
|
@ -407,7 +401,6 @@ static void write_gb_cart_mbc3(struct gb_cart* gb_cart, uint16_t address, const
|
||||||
//Implement RTC timer / latch
|
//Implement RTC timer / latch
|
||||||
if (gb_cart->rtc_latch == 0 && data[0] == 1)
|
if (gb_cart->rtc_latch == 0 && data[0] == 1)
|
||||||
{
|
{
|
||||||
correctRTC(gb_cart);
|
|
||||||
gb_cart->rtc_latch_second = gb_cart->rtc_second;
|
gb_cart->rtc_latch_second = gb_cart->rtc_second;
|
||||||
gb_cart->rtc_latch_minute = gb_cart->rtc_minute;
|
gb_cart->rtc_latch_minute = gb_cart->rtc_minute;
|
||||||
gb_cart->rtc_latch_hour = gb_cart->rtc_hour;
|
gb_cart->rtc_latch_hour = gb_cart->rtc_hour;
|
||||||
|
@ -475,6 +468,8 @@ static void write_gb_cart_mbc4(struct gb_cart* gb_cart, uint16_t address, const
|
||||||
|
|
||||||
static void read_gb_cart_mbc5(struct gb_cart* gb_cart, uint16_t address, uint8_t* data)
|
static void read_gb_cart_mbc5(struct gb_cart* gb_cart, uint16_t address, uint8_t* data)
|
||||||
{
|
{
|
||||||
|
size_t offset;
|
||||||
|
|
||||||
if ((address < 0x4000)) //Rom Bank 0
|
if ((address < 0x4000)) //Rom Bank 0
|
||||||
{
|
{
|
||||||
memcpy(data, &gb_cart->rom[address], 0x20);
|
memcpy(data, &gb_cart->rom[address], 0x20);
|
||||||
|
@ -760,43 +755,8 @@ static const struct parsed_cart_type* parse_cart_type(uint8_t cart_type)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void correctRTC(struct gb_cart* gb_cart)
|
bool GBCart::init_gb_cart(struct gb_cart* gb_cart, const char* gb_file)
|
||||||
{
|
{
|
||||||
time_t now, dif;
|
|
||||||
int days;
|
|
||||||
|
|
||||||
now = time(NULL);
|
|
||||||
dif = now - gb_cart->rtc_last_time;
|
|
||||||
|
|
||||||
gb_cart->rtc_second += (BYTE)(dif % 60);
|
|
||||||
dif /= 60;
|
|
||||||
gb_cart->rtc_minute += (BYTE)(dif % 60);
|
|
||||||
dif /= 60;
|
|
||||||
gb_cart->rtc_hour += (BYTE)(dif % 24);
|
|
||||||
dif /= 24;
|
|
||||||
|
|
||||||
days = (int)(gb_cart->rtc_day + ((gb_cart->rtc_day_carry & 1) << 8) + dif);
|
|
||||||
gb_cart->rtc_day = (days & 0xFF);
|
|
||||||
|
|
||||||
if (days > 255)
|
|
||||||
{
|
|
||||||
if (days > 511)
|
|
||||||
{
|
|
||||||
days &= 511;
|
|
||||||
gb_cart->rtc_day_carry |= 0x80;
|
|
||||||
}
|
|
||||||
if (days > 255)
|
|
||||||
{
|
|
||||||
gb_cart->rtc_day_carry = (gb_cart->rtc_day_carry & 0xFE) | (days > 255 ? 1 : 0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
gb_cart->rtc_last_time = now;
|
|
||||||
}
|
|
||||||
|
|
||||||
int GBCart::init_gb_cart(struct gb_cart* gb_cart, const char* gb_file)
|
|
||||||
{
|
|
||||||
int err;
|
|
||||||
const struct parsed_cart_type* type;
|
const struct parsed_cart_type* type;
|
||||||
uint8_t* rom = NULL;
|
uint8_t* rom = NULL;
|
||||||
size_t rom_size = 0;
|
size_t rom_size = 0;
|
||||||
|
@ -804,16 +764,15 @@ int GBCart::init_gb_cart(struct gb_cart* gb_cart, const char* gb_file)
|
||||||
size_t ram_size = 0;
|
size_t ram_size = 0;
|
||||||
|
|
||||||
/* load GB cart ROM */
|
/* load GB cart ROM */
|
||||||
err = load_file(gb_file, (void**)&rom, &rom_size);
|
if (!load_file(gb_file, (void**)&rom, &rom_size))
|
||||||
if (err == 0)
|
|
||||||
{
|
{
|
||||||
return err;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (rom_size < 0x8000)
|
if (rom_size < 0x8000)
|
||||||
{
|
{
|
||||||
err = 0;
|
free(rom);
|
||||||
goto free_rom;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* get and parse cart type */
|
/* get and parse cart type */
|
||||||
|
@ -821,8 +780,8 @@ int GBCart::init_gb_cart(struct gb_cart* gb_cart, const char* gb_file)
|
||||||
type = parse_cart_type(cart_type);
|
type = parse_cart_type(cart_type);
|
||||||
if (type == NULL)
|
if (type == NULL)
|
||||||
{
|
{
|
||||||
err = 0;
|
free(rom);
|
||||||
goto free_rom;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* load ram (if present) */
|
/* load ram (if present) */
|
||||||
|
@ -840,21 +799,26 @@ int GBCart::init_gb_cart(struct gb_cart* gb_cart, const char* gb_file)
|
||||||
|
|
||||||
if (ram_size != 0)
|
if (ram_size != 0)
|
||||||
{
|
{
|
||||||
ram = (uint8_t*)malloc(ram_size);
|
if (type->extra_devices & GED_RTC)
|
||||||
if (ram == NULL)
|
|
||||||
{
|
{
|
||||||
err = 0;
|
ram_size += 0x30;
|
||||||
goto free_rom;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
read_from_file("C:/Users/death/Desktop/pokemonsilver.sav", ram, ram_size + ((type->extra_devices & GED_RTC ) ? 0x30 : 0x00));
|
ram = (uint8_t*)malloc(ram_size );
|
||||||
|
if (ram == NULL)
|
||||||
|
{
|
||||||
|
free(rom);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
read_from_file("C:/Users/death/Desktop/pokemonsilver.sav", ram, ram_size );
|
||||||
}
|
}
|
||||||
|
|
||||||
//If we have RTC we need to load in the data, we assume the save will use the VBA-M format
|
//If we have RTC we need to load in the data, we assume the save will use the VBA-M format
|
||||||
if (type->extra_devices & GED_RTC)
|
if (type->extra_devices & GED_RTC)
|
||||||
{
|
{
|
||||||
gbCartRTC rtc;
|
gbCartRTC rtc;
|
||||||
memcpy(&rtc, &ram[ram_size], 0x30);
|
memcpy(&rtc, &ram[ram_size-0x30], 0x30);
|
||||||
|
|
||||||
gb_cart->rtc_second = rtc.second;
|
gb_cart->rtc_second = rtc.second;
|
||||||
gb_cart->rtc_minute = rtc.minute;
|
gb_cart->rtc_minute = rtc.minute;
|
||||||
|
@ -867,8 +831,6 @@ int GBCart::init_gb_cart(struct gb_cart* gb_cart, const char* gb_file)
|
||||||
gb_cart->rtc_latch_day = rtc.latch_day;
|
gb_cart->rtc_latch_day = rtc.latch_day;
|
||||||
gb_cart->rtc_latch_day_carry = rtc.latch_day_carry;
|
gb_cart->rtc_latch_day_carry = rtc.latch_day_carry;
|
||||||
gb_cart->rtc_last_time = rtc.mapperLastTime;
|
gb_cart->rtc_last_time = rtc.mapperLastTime;
|
||||||
|
|
||||||
correctRTC(gb_cart);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -882,11 +844,37 @@ int GBCart::init_gb_cart(struct gb_cart* gb_cart, const char* gb_file)
|
||||||
gb_cart->has_rtc = (type->extra_devices & GED_RTC) ? 1 : 0;
|
gb_cart->has_rtc = (type->extra_devices & GED_RTC) ? 1 : 0;
|
||||||
gb_cart->read_gb_cart = type->read_gb_cart;
|
gb_cart->read_gb_cart = type->read_gb_cart;
|
||||||
gb_cart->write_gb_cart = type->write_gb_cart;
|
gb_cart->write_gb_cart = type->write_gb_cart;
|
||||||
return 1;
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
free_rom:
|
void GBCart::save_gb_cart(struct gb_cart* gb_cart)
|
||||||
free(rom);
|
{
|
||||||
return err;
|
FILE *fRAM = fopen("C:/Users/death/Desktop/pokemonsilver.sav", "wb");
|
||||||
|
|
||||||
|
if (gb_cart->has_rtc)
|
||||||
|
{
|
||||||
|
fwrite(gb_cart->ram, 1, gb_cart->ram_size-0x30, fRAM);
|
||||||
|
|
||||||
|
gbCartRTC rtc;
|
||||||
|
rtc.second = gb_cart->rtc_second;
|
||||||
|
rtc.minute = gb_cart->rtc_minute;
|
||||||
|
rtc.hour = gb_cart->rtc_hour;
|
||||||
|
rtc.day = gb_cart->rtc_day;
|
||||||
|
rtc.day_carry = gb_cart->rtc_day_carry;
|
||||||
|
rtc.latch_second = gb_cart->rtc_latch_second;
|
||||||
|
rtc.latch_minute = gb_cart->rtc_latch_minute;
|
||||||
|
rtc.latch_hour = gb_cart->rtc_latch_hour;
|
||||||
|
rtc.latch_day = gb_cart->rtc_latch_day;
|
||||||
|
rtc.latch_day_carry = gb_cart->rtc_latch_day_carry;
|
||||||
|
rtc.mapperLastTime = gb_cart->rtc_last_time;
|
||||||
|
fwrite(&rtc, 1, 0x30, fRAM);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
fwrite(gb_cart->ram, 1, gb_cart->ram_size, fRAM);
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose(fRAM);
|
||||||
}
|
}
|
||||||
|
|
||||||
void GBCart::release_gb_cart(struct gb_cart* gb_cart)
|
void GBCart::release_gb_cart(struct gb_cart* gb_cart)
|
||||||
|
|
|
@ -63,8 +63,9 @@ struct gbCartRTC {
|
||||||
class GBCart
|
class GBCart
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
static int init_gb_cart(struct gb_cart* gb_cart, const char* gb_file);
|
static bool init_gb_cart(struct gb_cart* gb_cart, const char* gb_file);
|
||||||
static void release_gb_cart(struct gb_cart* gb_cart);
|
static void release_gb_cart(struct gb_cart* gb_cart);
|
||||||
|
static void save_gb_cart(struct gb_cart* gb_cart);
|
||||||
|
|
||||||
static void read_gb_cart(struct gb_cart* gb_cart, uint16_t address, uint8_t* data);
|
static void read_gb_cart(struct gb_cart* gb_cart, uint16_t address, uint8_t* data);
|
||||||
static void write_gb_cart(struct gb_cart* gb_cart, uint16_t address, const uint8_t* data);
|
static void write_gb_cart(struct gb_cart* gb_cart, uint16_t address, const uint8_t* data);
|
||||||
|
|
|
@ -22,7 +22,7 @@ uint16_t gb_cart_address(unsigned int bank, uint16_t address)
|
||||||
void Transferpak::Init()
|
void Transferpak::Init()
|
||||||
{
|
{
|
||||||
memset(&tpak, 0, sizeof(tpak));
|
memset(&tpak, 0, sizeof(tpak));
|
||||||
tpak.access_mode = (GBCart::init_gb_cart(&tpak.gb_cart, "C:/Users/death/Desktop/pokemonsilver.gbc") == 0) ? CART_NOT_INSERTED : CART_ACCESS_MODE_0;
|
tpak.access_mode = (!GBCart::init_gb_cart(&tpak.gb_cart, "C:/Users/death/Desktop/pokemonsilver.gbc")) ? CART_NOT_INSERTED : CART_ACCESS_MODE_0;
|
||||||
|
|
||||||
tpak.access_mode_changed = 0x44;
|
tpak.access_mode_changed = 0x44;
|
||||||
}
|
}
|
||||||
|
@ -113,6 +113,7 @@ void Transferpak::WriteTo(uint16_t address, uint8_t * data)
|
||||||
if (tpak.enabled)
|
if (tpak.enabled)
|
||||||
{
|
{
|
||||||
GBCart::write_gb_cart(&tpak.gb_cart, gb_cart_address(tpak.bank, address), data);
|
GBCart::write_gb_cart(&tpak.gb_cart, gb_cart_address(tpak.bank, address), data);
|
||||||
|
GBCart::save_gb_cart(&tpak.gb_cart);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
Loading…
Reference in New Issue