2008-09-06 03:30:42 +00:00
|
|
|
/****************************************************************************
|
2009-07-22 02:05:49 +00:00
|
|
|
* FCE Ultra
|
2008-09-06 03:30:42 +00:00
|
|
|
* Nintendo Wii/Gamecube Port
|
|
|
|
*
|
2022-01-09 17:40:41 +01:00
|
|
|
* Tantric 2008-2022
|
2008-09-06 03:30:42 +00:00
|
|
|
*
|
2009-03-28 17:23:08 +00:00
|
|
|
* fceustate.cpp
|
2008-09-06 03:30:42 +00:00
|
|
|
*
|
|
|
|
* Memory Based Load/Save RAM Manager
|
|
|
|
*
|
|
|
|
* These are the battery-backed RAM (save data) routines, brought together
|
|
|
|
* as GCxxxxx
|
|
|
|
* The original file I/O is replaced with Memory Read/Writes to the
|
|
|
|
* savebuffer below
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <gccore.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <malloc.h>
|
|
|
|
#include <fat.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2008-11-17 00:09:31 +00:00
|
|
|
#include "fceugx.h"
|
2009-07-17 17:27:04 +00:00
|
|
|
#include "fceusupport.h"
|
2009-03-28 17:23:08 +00:00
|
|
|
#include "menu.h"
|
|
|
|
#include "filebrowser.h"
|
2008-09-06 03:30:42 +00:00
|
|
|
#include "fileop.h"
|
2017-08-29 12:07:19 -05:00
|
|
|
#include "pocketnes/goombasav.h"
|
2008-09-06 03:30:42 +00:00
|
|
|
|
2010-04-01 21:37:03 +00:00
|
|
|
static u32 WiiFCEU_GameSave(CartInfo *LocalHWInfo, int operation)
|
2008-09-06 03:30:42 +00:00
|
|
|
{
|
2008-12-30 00:05:57 +00:00
|
|
|
u32 offset = 0;
|
2008-12-22 08:38:49 +00:00
|
|
|
|
2008-09-06 03:30:42 +00:00
|
|
|
if(LocalHWInfo->battery && LocalHWInfo->SaveGame[0])
|
|
|
|
{
|
|
|
|
int x;
|
|
|
|
|
|
|
|
for(x=0;x<4;x++)
|
|
|
|
{
|
|
|
|
if(LocalHWInfo->SaveGame[x])
|
|
|
|
{
|
|
|
|
if(operation == 0) // save to file
|
2008-12-22 08:38:49 +00:00
|
|
|
memcpy(savebuffer+offset, LocalHWInfo->SaveGame[x], LocalHWInfo->SaveGameLen[x]);
|
2008-09-06 03:30:42 +00:00
|
|
|
else // load from file
|
2008-12-22 08:38:49 +00:00
|
|
|
memcpy(LocalHWInfo->SaveGame[x], savebuffer+offset, LocalHWInfo->SaveGameLen[x]);
|
|
|
|
offset += LocalHWInfo->SaveGameLen[x];
|
2008-09-06 03:30:42 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-12-22 08:38:49 +00:00
|
|
|
return offset;
|
2008-09-06 03:30:42 +00:00
|
|
|
}
|
|
|
|
|
2009-10-01 22:21:25 +00:00
|
|
|
bool SaveRAM (char * filepath, bool silent)
|
2008-09-06 03:30:42 +00:00
|
|
|
{
|
2008-11-12 08:40:09 +00:00
|
|
|
bool retval = false;
|
|
|
|
int datasize = 0;
|
|
|
|
int offset = 0;
|
2009-10-01 22:21:25 +00:00
|
|
|
int device;
|
|
|
|
|
|
|
|
if(!FindDevice(filepath, &device))
|
|
|
|
return 0;
|
2008-11-12 08:40:09 +00:00
|
|
|
|
2009-07-20 06:18:06 +00:00
|
|
|
if(GameInfo->type == GIT_FDS)
|
2008-10-03 18:11:32 +00:00
|
|
|
{
|
|
|
|
if(!silent)
|
2009-03-28 17:23:08 +00:00
|
|
|
InfoPrompt("RAM saving is not available for FDS games!");
|
2008-10-03 18:11:32 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2008-12-19 22:04:55 +00:00
|
|
|
AllocSaveBuffer ();
|
2008-09-06 03:30:42 +00:00
|
|
|
|
2008-10-01 05:25:38 +00:00
|
|
|
// save game save to savebuffer
|
2009-07-20 06:18:06 +00:00
|
|
|
if(GameInfo->type == GIT_CART)
|
2010-04-01 21:37:03 +00:00
|
|
|
datasize = WiiFCEU_GameSave(&iNESCart, 0);
|
2009-07-20 06:18:06 +00:00
|
|
|
else if(GameInfo->type == GIT_VSUNI)
|
2010-04-01 21:37:03 +00:00
|
|
|
datasize = WiiFCEU_GameSave(&UNIFCart, 0);
|
2008-09-06 03:30:42 +00:00
|
|
|
|
2017-08-29 12:07:19 -05:00
|
|
|
if (datasize)
|
|
|
|
{
|
|
|
|
// Check to see if this is a PocketNES save file
|
|
|
|
FILE* file = fopen(filepath, "rb");
|
|
|
|
if (file)
|
|
|
|
{
|
|
|
|
uint32 tag;
|
|
|
|
fread(&tag, sizeof(uint32), 1, file);
|
|
|
|
fclose(file);
|
|
|
|
|
|
|
|
if (goomba_is_sram(&tag))
|
|
|
|
{
|
2017-08-29 18:00:17 -05:00
|
|
|
void* gba_data = malloc(GOOMBA_COLOR_SRAM_SIZE);
|
|
|
|
|
|
|
|
file = fopen(filepath, "rb");
|
|
|
|
fread(gba_data, 1, GOOMBA_COLOR_SRAM_SIZE, file);
|
|
|
|
fclose(file);
|
|
|
|
|
|
|
|
void* cleaned = goomba_cleanup(gba_data);
|
|
|
|
if (!cleaned) {
|
|
|
|
ErrorPrompt(goomba_last_error());
|
|
|
|
} else if (cleaned != gba_data) {
|
|
|
|
memcpy(gba_data, cleaned, GOOMBA_COLOR_SRAM_SIZE);
|
|
|
|
free(cleaned);
|
|
|
|
}
|
2017-08-29 12:07:19 -05:00
|
|
|
|
|
|
|
// Look for just one save file. If there aren't any, or there is more than one, don't read any data.
|
2017-08-30 18:27:18 -05:00
|
|
|
const stateheader* sh1 = NULL;
|
|
|
|
const stateheader* sh2 = NULL;
|
2017-08-29 12:07:19 -05:00
|
|
|
|
2017-08-30 18:27:18 -05:00
|
|
|
const stateheader* sh = stateheader_first(gba_data);
|
2017-08-29 12:07:19 -05:00
|
|
|
while (sh && stateheader_plausible(sh)) {
|
|
|
|
if (little_endian_conv_16(sh->type) != GOOMBA_SRAMSAVE) {}
|
|
|
|
else if (sh1 == NULL) {
|
|
|
|
sh1 = sh;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
sh2 = sh;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
sh = stateheader_advance(sh);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sh1 == NULL)
|
|
|
|
{
|
|
|
|
ErrorPrompt("PocketNES save file has no SRAM.");
|
|
|
|
datasize = 0;
|
|
|
|
}
|
|
|
|
else if (sh2 != NULL)
|
|
|
|
{
|
|
|
|
ErrorPrompt("PocketNES save file has more than one SRAM.");
|
|
|
|
datasize = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-08-29 18:00:17 -05:00
|
|
|
char* newdata = goomba_new_sav(gba_data, sh1, savebuffer, datasize);
|
|
|
|
if (!newdata) {
|
|
|
|
ErrorPrompt(goomba_last_error());
|
|
|
|
datasize = 0;
|
|
|
|
} else {
|
|
|
|
memcpy(savebuffer, newdata, GOOMBA_COLOR_SRAM_SIZE);
|
|
|
|
datasize = GOOMBA_COLOR_SRAM_SIZE;
|
|
|
|
free(newdata);
|
|
|
|
}
|
2017-08-29 12:07:19 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-12 08:40:09 +00:00
|
|
|
if (datasize)
|
2008-09-06 03:30:42 +00:00
|
|
|
{
|
2009-10-01 22:21:25 +00:00
|
|
|
offset = SaveFile(filepath, datasize, silent);
|
2008-09-06 03:30:42 +00:00
|
|
|
|
|
|
|
if (offset > 0)
|
|
|
|
{
|
2009-03-28 17:23:08 +00:00
|
|
|
if (!silent)
|
|
|
|
InfoPrompt("Save successful");
|
2008-09-06 03:30:42 +00:00
|
|
|
retval = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2009-03-28 17:23:08 +00:00
|
|
|
if (!silent)
|
|
|
|
InfoPrompt("No data to save!");
|
2008-09-06 03:30:42 +00:00
|
|
|
}
|
2008-12-19 22:04:55 +00:00
|
|
|
FreeSaveBuffer ();
|
2008-09-06 03:30:42 +00:00
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2009-03-28 17:23:08 +00:00
|
|
|
bool
|
2009-10-01 22:21:25 +00:00
|
|
|
SaveRAMAuto (bool silent)
|
2008-09-06 03:30:42 +00:00
|
|
|
{
|
2008-11-12 08:40:09 +00:00
|
|
|
char filepath[1024];
|
2009-03-28 17:23:08 +00:00
|
|
|
|
2009-10-01 22:21:25 +00:00
|
|
|
if(!MakeFilePath(filepath, FILE_RAM, romFilename, 0))
|
2009-03-28 17:23:08 +00:00
|
|
|
return false;
|
|
|
|
|
2009-10-01 22:21:25 +00:00
|
|
|
return SaveRAM(filepath, silent);
|
2009-03-28 17:23:08 +00:00
|
|
|
}
|
|
|
|
|
2009-10-01 22:21:25 +00:00
|
|
|
bool LoadRAM (char * filepath, bool silent)
|
2009-03-28 17:23:08 +00:00
|
|
|
{
|
2008-11-12 08:40:09 +00:00
|
|
|
int offset = 0;
|
2008-12-19 22:04:55 +00:00
|
|
|
bool retval = false;
|
2009-10-01 22:21:25 +00:00
|
|
|
int device;
|
|
|
|
|
|
|
|
if(!FindDevice(filepath, &device))
|
|
|
|
return 0;
|
2008-11-12 08:40:09 +00:00
|
|
|
|
2009-07-20 06:18:06 +00:00
|
|
|
if(GameInfo->type == GIT_FDS) // RAM saves don't exist for FDS games
|
2008-10-03 18:11:32 +00:00
|
|
|
return false;
|
|
|
|
|
2008-12-19 22:04:55 +00:00
|
|
|
AllocSaveBuffer ();
|
2008-09-06 03:30:42 +00:00
|
|
|
|
2009-10-01 22:21:25 +00:00
|
|
|
offset = LoadFile(filepath, silent);
|
2008-09-06 03:30:42 +00:00
|
|
|
|
2017-08-29 12:07:19 -05:00
|
|
|
// Check to see if this is a PocketNES save file
|
|
|
|
if (goomba_is_sram(savebuffer))
|
|
|
|
{
|
2017-08-29 18:00:17 -05:00
|
|
|
void* cleaned = goomba_cleanup(savebuffer);
|
|
|
|
if (!cleaned) {
|
|
|
|
ErrorPrompt(goomba_last_error());
|
|
|
|
} else if (cleaned != savebuffer) {
|
|
|
|
memcpy(savebuffer, cleaned, GOOMBA_COLOR_SRAM_SIZE);
|
|
|
|
free(cleaned);
|
|
|
|
}
|
2017-08-29 17:02:03 -05:00
|
|
|
|
2017-08-29 12:07:19 -05:00
|
|
|
// Look for just one save file. If there aren't any, or there is more than one, don't read any data.
|
2017-08-30 18:27:18 -05:00
|
|
|
const stateheader* sh1 = NULL;
|
|
|
|
const stateheader* sh2 = NULL;
|
2017-08-29 12:07:19 -05:00
|
|
|
|
2017-08-30 18:27:18 -05:00
|
|
|
const stateheader* sh = stateheader_first(savebuffer);
|
2017-08-29 12:07:19 -05:00
|
|
|
while (sh && stateheader_plausible(sh)) {
|
2017-08-29 18:00:17 -05:00
|
|
|
if (little_endian_conv_16(sh->type) != GOOMBA_SRAMSAVE) { }
|
2017-08-29 12:07:19 -05:00
|
|
|
else if (sh1 == NULL) {
|
|
|
|
sh1 = sh;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
sh2 = sh;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
sh = stateheader_advance(sh);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sh1 == NULL)
|
|
|
|
{
|
|
|
|
ErrorPrompt("PocketNES save file has no SRAM.");
|
|
|
|
offset = 0;
|
|
|
|
}
|
|
|
|
else if (sh2 != NULL)
|
|
|
|
{
|
|
|
|
ErrorPrompt("PocketNES save file has more than one SRAM.");
|
|
|
|
offset = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
goomba_size_t len;
|
|
|
|
void* extracted = goomba_extract(savebuffer, sh1, &len);
|
|
|
|
if (!extracted)
|
|
|
|
ErrorPrompt(goomba_last_error());
|
|
|
|
else
|
|
|
|
{
|
|
|
|
memcpy(savebuffer, extracted, len);
|
|
|
|
offset = len;
|
|
|
|
free(extracted);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-09-06 03:30:42 +00:00
|
|
|
if (offset > 0)
|
|
|
|
{
|
2009-07-20 06:18:06 +00:00
|
|
|
if(GameInfo->type == GIT_CART)
|
2010-04-01 21:37:03 +00:00
|
|
|
WiiFCEU_GameSave(&iNESCart, 1);
|
2009-07-20 06:18:06 +00:00
|
|
|
else if(GameInfo->type == GIT_VSUNI)
|
2010-04-01 21:37:03 +00:00
|
|
|
WiiFCEU_GameSave(&UNIFCart, 1);
|
2008-10-01 05:25:38 +00:00
|
|
|
|
2008-09-06 03:30:42 +00:00
|
|
|
ResetNES();
|
2008-12-19 22:04:55 +00:00
|
|
|
retval = true;
|
2008-09-06 03:30:42 +00:00
|
|
|
}
|
2008-12-19 22:04:55 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
// if we reached here, nothing was done!
|
|
|
|
if(!silent)
|
2009-03-28 17:23:08 +00:00
|
|
|
InfoPrompt ("Save file not found");
|
2008-12-19 22:04:55 +00:00
|
|
|
}
|
|
|
|
FreeSaveBuffer ();
|
|
|
|
return retval;
|
2008-09-06 03:30:42 +00:00
|
|
|
}
|
2009-03-28 17:23:08 +00:00
|
|
|
|
|
|
|
bool
|
2009-10-01 22:21:25 +00:00
|
|
|
LoadRAMAuto (bool silent)
|
2009-03-28 17:23:08 +00:00
|
|
|
{
|
2009-04-09 07:49:28 +00:00
|
|
|
char filepath[MAXPATHLEN];
|
|
|
|
char filepath2[MAXPATHLEN];
|
2009-03-28 17:23:08 +00:00
|
|
|
|
2009-04-09 07:49:28 +00:00
|
|
|
// look for Auto save file
|
2009-10-01 22:21:25 +00:00
|
|
|
if(!MakeFilePath(filepath, FILE_RAM, romFilename, 0))
|
2009-03-28 17:23:08 +00:00
|
|
|
return false;
|
|
|
|
|
2009-10-01 22:21:25 +00:00
|
|
|
if (LoadRAM(filepath, silent))
|
2009-04-09 07:49:28 +00:00
|
|
|
return true;
|
|
|
|
|
2017-08-30 18:44:18 -05:00
|
|
|
if (!GCSettings.AppendAuto)
|
|
|
|
return false;
|
|
|
|
|
2009-04-09 07:49:28 +00:00
|
|
|
// look for file with no number or Auto appended
|
2009-10-01 22:21:25 +00:00
|
|
|
if(!MakeFilePath(filepath2, FILE_RAM, romFilename, -1))
|
2009-04-09 07:49:28 +00:00
|
|
|
return false;
|
|
|
|
|
2009-10-01 22:21:25 +00:00
|
|
|
if(LoadRAM(filepath2, silent))
|
2009-04-09 07:49:28 +00:00
|
|
|
{
|
|
|
|
// rename this file - append Auto
|
2009-10-01 22:21:25 +00:00
|
|
|
rename(filepath2, filepath); // rename file (to avoid duplicates)
|
2009-04-09 07:49:28 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2009-03-28 17:23:08 +00:00
|
|
|
}
|