implemented new save manager menu (multiple slots, snapshots, slot deletion...)

miscellaneous improvement to GUI engine
This commit is contained in:
ekeeke31 2010-05-06 12:59:43 +00:00
parent e596250c89
commit 44073ca30e
26 changed files with 1590 additions and 1237 deletions

View File

@ -14,7 +14,7 @@ of samples per frame and keeping PSG & FM chips in sync.
* fixed YM2612 context saving/loading. * fixed YM2612 context saving/loading.
* fixed YM2612 state on reset. * fixed YM2612 state on reset.
* removed outdated & less accurate Gens YM2612 core * removed outdated & less accurate Gens YM2612 core
* added preliminar emulation of YM2612 DAC original resolution (configurable). * added configurable YM2612 DAC resolution emulation.
* added configurable & faster FIR resampler (thanks to Blargg & AamirM), dropped libsamplerate support. * added configurable & faster FIR resampler (thanks to Blargg & AamirM), dropped libsamplerate support.
* added configurable Low-Pass filtering * added configurable Low-Pass filtering
* added configurable 3-Band Equalizer (thanks to Neil C). * added configurable 3-Band Equalizer (thanks to Neil C).
@ -25,7 +25,9 @@ of samples per frame and keeping PSG & FM chips in sync.
--------------- ---------------
* added support for CRAM writes during horizontal blanking (Striker, Zero the Kamikaze Squirrel,...) * added support for CRAM writes during horizontal blanking (Striker, Zero the Kamikaze Squirrel,...)
* added support for 2-Cell vertical scrolling in Interlaced 2 mode * added support for 2-Cell vertical scrolling in Interlaced 2 mode
* fixed left-most column vertical scrolling when horizontally scrolled (Gynoug, F1)
* fixed VBLANK transition line checks * fixed VBLANK transition line checks
* improved VDP FIFO timings accuracy: fixes Sol Deace intro
* improved sprite masking accuracy (thanks to Nemesis for his sprite test program) * improved sprite masking accuracy (thanks to Nemesis for his sprite test program)
* improved HBLANK flag timing accuracy: fixes Mega Turrican (Sky level) * improved HBLANK flag timing accuracy: fixes Mega Turrican (Sky level)
* improved sprites processing accuracy: fixes (un)masked sprites in Mickey Mania (3D level), Sonic 2 (VS mode). * improved sprites processing accuracy: fixes (un)masked sprites in Mickey Mania (3D level), Sonic 2 (VS mode).
@ -48,9 +50,9 @@ of samples per frame and keeping PSG & FM chips in sync.
* added Action Replay hardware emulation (Action replay ROM is now fully supported). * added Action Replay hardware emulation (Action replay ROM is now fully supported).
* added S&K "Lock-On" hardware emulation (you can "lock" any games to Sonic & Knuckles). * added S&K "Lock-On" hardware emulation (you can "lock" any games to Sonic & Knuckles).
* added Cartridge "hot swap" feature. * added Cartridge "hot swap" feature.
* fixed Backup Memory support in some games that were actually using serial EEPROM. * added EEPROM support in some games requiring it.
* fixed Realtec mapper emulation: fixes missing sound in Balloon Boy / Funny World. * fixed Realtec mapper emulation: fixes missing sound in Balloon Boy / Funny World.
* fixed lightgun auto detection: fixes cursor position in Lethal Enforcers II. * fixed lightgun auto-detection: fixes default cursor position in Lethal Enforcers II.
* lots of code cleanup, bugfixes & optimization. * lots of code cleanup, bugfixes & optimization.
@ -63,7 +65,14 @@ of samples per frame and keeping PSG & FM chips in sync.
* improved reset button behavior, now works more like the real Genesis reset button. * improved reset button behavior, now works more like the real Genesis reset button.
* improved lightgun cursors layout. * improved lightgun cursors layout.
* fixed stability issues and memory leaks. * fixed stability issues and memory leaks.
* compiled with devkitPPC r19 & libOGC 1.8.1 (SDHC support, new DVDX, etc)
[Wii only]
---------------
* added USB2 support through Hermes cIOS/mload (cIOS 202 is required)
* added Video Hardware Gamma control
* added Video Hardware "Trap Filter" control
* improved Mouse emulation through the Wiimote
* compiled with devkitPPC r21 & libOGC 1.8.3 (SDHC support, DVDX v2 support)

View File

@ -43,7 +43,7 @@ void sram_init()
{ {
memset (&sram, 0, sizeof (T_SRAM)); memset (&sram, 0, sizeof (T_SRAM));
memset (&sram.sram[0], 0xFF, 0x10000); memset (&sram.sram[0], 0xFF, 0x10000);
sram.crc = crc32 (0, &sram.sram[0], 0x10000); sram.crc = crc32(0, &sram.sram[0], 0x10000);
if ((READ_BYTE(cart.rom,0x1b0) == 0x52) && (READ_BYTE(cart.rom,0x1b1) == 0x41)) if ((READ_BYTE(cart.rom,0x1b0) == 0x52) && (READ_BYTE(cart.rom,0x1b1) == 0x41))
{ {

View File

@ -79,7 +79,7 @@ void config_default(void)
config.mg = 1.0; config.mg = 1.0;
config.hg = 1.0; config.hg = 1.0;
config.rolloff = 0.995; config.rolloff = 0.995;
config.dac_bits = 14; config.dac_bits = 14;
/* system options */ /* system options */
config.region_detect = 0; config.region_detect = 0;
@ -116,11 +116,12 @@ void config_default(void)
/* menu options */ /* menu options */
#ifdef HW_RVL #ifdef HW_RVL
config.sram_auto = 0; config.s_auto = 1;
#else #else
config.sram_auto = -1; config.s_auto = 0;
#endif #endif
config.state_auto = -1; config.s_default = 1;
config.s_device = 0;
config.bg_color = 0; config.bg_color = 0;
config.bg_overlay = 0; config.bg_overlay = 0;
config.screen_w = 658; config.screen_w = 658;

View File

@ -70,8 +70,9 @@ typedef struct
uint16 pad_keymap[4][MAX_KEYS]; uint16 pad_keymap[4][MAX_KEYS];
uint32 wpad_keymap[4*3][MAX_KEYS]; uint32 wpad_keymap[4*3][MAX_KEYS];
t_input_config input[MAX_INPUTS]; t_input_config input[MAX_INPUTS];
int8 sram_auto; uint8 s_auto;
int8 state_auto; uint8 s_default;
uint8 s_device;
int8 bg_color; int8 bg_color;
int8 bg_overlay; int8 bg_overlay;
int16 screen_w; int16 screen_w;

View File

@ -227,8 +227,8 @@ int FAT_Open(int type)
int max = 0; int max = 0;
char root[10] = ""; char root[10] = "";
/* FAT header */
#ifdef HW_RVL #ifdef HW_RVL
/* FAT header */
if (type == TYPE_SD) sprintf (root, "sd:"); if (type == TYPE_SD) sprintf (root, "sd:");
else if (type == TYPE_USB) sprintf (root, "usb:"); else if (type == TYPE_USB) sprintf (root, "usb:");
#endif #endif

View File

@ -1,637 +0,0 @@
/*
* file_mem.c
*
* FAT and Memory Card SRAM/Savestate files managment
*
* Softdev (2006)
* Eke-Eke (2007,2008,2009)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
***************************************************************************/
#include "shared.h"
#include "file_mem.h"
#include "file_fat.h"
#include "dvd.h"
#include "gui.h"
#include "filesel.h"
#include "saveicon.h"
/* Global ROM filename */
char rom_filename[MAXJOLIET];
/* Support for MemCards */
/**
* libOGC System Work Area
*/
static u8 SysArea[CARD_WORKAREA] ATTRIBUTE_ALIGN (32);
/**
* DMA Transfer Area.
* Must be 32-byte aligned.
* 64k SRAM + 2k Icon
*/
static u8 savebuffer[STATE_SIZE] ATTRIBUTE_ALIGN (32);
/****************************************************************************
* SDCARD Access functions
*
* We use the same buffer as for Memory Card manager
* Function returns TRUE on success.
*****************************************************************************/
static int FAT_ManageFile(char *filename, u8 direction, u8 filetype)
{
char fname[MAXPATHLEN];
int done = 0;
int filesize;
/* build complete SDCARD filename */
sprintf (fname, "%s/saves/%s", DEFAULT_PATH, filename);
/* open file */
FILE *fp = fopen(fname, direction ? "rb" : "wb");
if (fp == NULL)
{
GUI_WaitPrompt("Error","Unable to open file !");
return 0;
}
switch (direction)
{
case 0: /* SAVING */
if (filetype) /* SRAM */
{
memcpy(savebuffer, sram.sram, 0x10000);
sram.crc = crc32 (0, sram.sram, 0x10000);
filesize = 0x10000;
}
else filesize = state_save(savebuffer); /* STATE */
/* write buffer (2k blocks) */
while (filesize > FATCHUNK)
{
fwrite(savebuffer + done, FATCHUNK, 1, fp);
filesize -= FATCHUNK;
done += FATCHUNK;
}
done += fwrite(savebuffer + done, filesize, 1, fp);
fclose(fp);
if (done < filesize)
{
GUI_WaitPrompt("Error","Unable to write file !");
return 0;
}
GUI_MsgBoxClose();
return 1;
case 1: /* LOADING */
/* read size */
fseek(fp , 0 , SEEK_END);
filesize = ftell (fp);
fseek(fp, 0, SEEK_SET);
/* read into buffer (2k blocks) */
while (filesize > FATCHUNK)
{
fread(savebuffer + done, FATCHUNK, 1, fp);
filesize -= FATCHUNK;
done += FATCHUNK;
}
done += fread(savebuffer + done, filesize, 1, fp);
fclose(fp);
if (done < filesize)
{
GUI_WaitPrompt("Error","Unable to read file !");
return 0;
}
if (filetype) /* SRAM */
{
memcpy(sram.sram, savebuffer, done);
sram.crc = crc32 (0, sram.sram, 0x10000);
}
else
{
/* STATE */
if (!state_load(savebuffer))
{
GUI_WaitPrompt("Error","File version is not compatible !");
return 0;
}
}
GUI_MsgBoxClose();
return 1;
}
return 0;
}
/****************************************************************************
* MountTheCard
*
* libOGC provides the CARD_Mount function, and it should be all you need.
* However, experience with previous emulators has taught me that you are
* better off doing a little bit more than that!
*
* Function returns TRUE on success.
*****************************************************************************/
static int MountTheCard (u8 slot)
{
int tries = 0;
int CardError;
#if defined(HW_DOL)
*(unsigned long *) (0xCC006800) |= 1 << 13; /*** Disable Encryption ***/
uselessinquiry ();
#elif defined(HW_RVL)
*(unsigned long *) (0xCD006800) |= 1 << 13; /*** Disable Encryption ***/
#endif
while (tries < 10)
{
VIDEO_WaitVSync ();
CardError = CARD_Mount (slot, SysArea, NULL); /*** Don't need or want a callback ***/
if (CardError == 0)
return 1;
else
EXI_ProbeReset ();
tries++;
}
return 0;
}
/****************************************************************************
* CardFileExists
*
* Wrapper to search through the files on the card.
* Returns TRUE if found.
****************************************************************************/
static int CardFileExists (char *filename, u8 slot)
{
card_dir CardDir;
int CardError = CARD_FindFirst (slot, &CardDir, TRUE);
while (CardError >= 0)
{
CardError = CARD_FindNext (&CardDir);
if (strcmp ((char *) CardDir.filename, filename) == 0)
return 1;
}
return 0;
}
/****************************************************************************
* FILE autoload (SRAM/FreezeState or Config File)
*
*
*****************************************************************************/
void memfile_autoload(s8 autosram, s8 autostate)
{
/* this should be transparent to the user */
SILENT = 1;
/* SRAM */
if (autosram != -1)
ManageSRAM(1,autosram);
/* STATE */
if (autostate != -1)
ManageState(1,autostate);
SILENT = 0;
}
void memfile_autosave(s8 autosram, s8 autostate)
{
int crccheck = crc32 (0, sram.sram, 0x10000);
/* this should be transparent to the user */
SILENT = 1;
/* SRAM */
if ((autosram != -1) && (crccheck != sram.crc))
ManageSRAM(0, autosram);
/* STATE */
if (autostate != -1)
ManageState(0,autostate);
SILENT = 0;
}
/****************************************************************************
* ManageSRAM
*
* Here is the main SRAM Management stuff.
* The output file contains an icon (2K), 64 bytes comment and the SRAM (64k).
* As memcards are allocated in blocks of 8k or more, you have a around
* 6k bytes to save/load any other data you wish without altering any of the
* main save / load code.
*
* direction == 0 save, 1 load.
****************************************************************************/
int ManageSRAM (u8 direction, u8 device)
{
if (!cart.romsize)
return 0;
char filename[MAXJOLIET];
if (direction)
GUI_MsgBoxOpen("Information","Loading SRAM ...",1);
else
GUI_MsgBoxOpen("Information","Saving SRAM ...",1);
/* clean buffer */
memset(savebuffer, 0, STATE_SIZE);
if (device == 0)
{
/* FAT support */
sprintf (filename, "%s.srm", rom_filename);
return FAT_ManageFile(filename,direction,1);
}
/* Memory CARD support */
char action[80];
int CardError;
unsigned int SectorSize = 0;
int blocks;
char comment[2][32] = { {"Genesis Plus 1.2a"}, {"SRAM Save"} };
int outbytes = 0;
int sbo;
unsigned long inzipped,outzipped;
card_file CardFile;
card_stat CardStatus;
/* First, build a filename */
sprintf (filename, "MD-%04X.srm", realchecksum);
strcpy (comment[1], filename);
/* set MCARD slot nr. */
u8 CARDSLOT = device - 1;
/* Saving */
if (direction == 0)
{
/*** Build the output buffer ***/
memcpy (&savebuffer, &icon, 2048);
memcpy (&savebuffer[2048], &comment[0], 64);
inzipped = 0x10000;
outzipped = 0x12000;
compress2 ((Bytef *) &savebuffer[2112+sizeof(outzipped)], &outzipped, (Bytef *) &sram.sram, inzipped, 9);
memcpy(&savebuffer[2112], &outzipped, sizeof(outzipped));
}
outbytes = 2048 + 64 + outzipped + sizeof(outzipped);
/*** Initialise the CARD system ***/
memset (&SysArea, 0, CARD_WORKAREA);
CARD_Init ("GENP", "00");
/*** Attempt to mount the card ***/
CardError = MountTheCard (CARDSLOT);
if (CardError)
{
/*** Retrieve the sector size ***/
CardError = CARD_GetSectorSize (CARDSLOT, &SectorSize);
if (SectorSize)
{
switch (direction)
{
case 0: /*** Saving ***/
/*** Determine number of blocks on this card ***/
blocks = (outbytes / SectorSize) * SectorSize;
if (outbytes % SectorSize)
blocks += SectorSize;
/*** Check if a previous save exists ***/
if (CardFileExists (filename,CARDSLOT))
{
CardError = CARD_Open (CARDSLOT, filename, &CardFile);
if (CardError)
{
sprintf (action, "Unable to open file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount (CARDSLOT);
return 0;
}
int size = CardFile.len;
CARD_Close (&CardFile);
if (size < blocks)
{
/* new size is bigger: check if there is enough space left */
CardError = CARD_Create (CARDSLOT, "TEMP", blocks-size, &CardFile);
if (CardError)
{
sprintf (action, "Unable to create temporary file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount (CARDSLOT);
return 0;
}
CARD_Close (&CardFile);
CARD_Delete(CARDSLOT, "TEMP");
}
/* always delete existing slot */
CARD_Delete(CARDSLOT, filename);
}
/*** Create a new slot ***/
CardError = CARD_Create (CARDSLOT, filename, blocks, &CardFile);
if (CardError)
{
sprintf (action, "Unable to create new file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount (CARDSLOT);
return 0;
}
/*** Continue and save ***/
CARD_GetStatus (CARDSLOT, CardFile.filenum, &CardStatus);
CardStatus.icon_addr = 0x0;
CardStatus.icon_fmt = 2;
CardStatus.icon_speed = 1;
CardStatus.comment_addr = 2048;
CARD_SetStatus (CARDSLOT, CardFile.filenum, &CardStatus);
/*** And write the blocks out ***/
sbo = 0;
while (outbytes > 0)
{
CardError = CARD_Write (&CardFile, &savebuffer[sbo], SectorSize, sbo);
outbytes -= SectorSize;
sbo += SectorSize;
}
CARD_Close (&CardFile);
CARD_Unmount (CARDSLOT);
sram.crc = crc32 (0, &sram.sram[0], 0x10000);
GUI_MsgBoxClose();
return 1;
default: /*** Loading ***/
memset (&CardFile, 0, sizeof (CardFile));
CardError = CARD_Open (CARDSLOT, filename, &CardFile);
if (CardError)
{
sprintf (action, "Unable to open file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount (CARDSLOT);
return 0;
}
blocks = CardFile.len;
if (blocks < SectorSize)
blocks = SectorSize;
if (blocks % SectorSize)
blocks++;
/*** Just read the file back in ***/
sbo = 0;
while (blocks > 0)
{
CARD_Read (&CardFile, &savebuffer[sbo], SectorSize, sbo);
sbo += SectorSize;
blocks -= SectorSize;
}
CARD_Close (&CardFile);
CARD_Unmount (CARDSLOT);
/*** update SRAM ***/
memcpy(&inzipped,&savebuffer[2112],sizeof(inzipped));
outzipped = 0x10000;
uncompress ((Bytef *) &sram.sram, &outzipped, (Bytef *) &savebuffer[2112+sizeof(inzipped)], inzipped);
sram.crc = crc32 (0, &sram.sram[0], 0x10000);
GUI_MsgBoxClose();
return 1;
}
}
GUI_WaitPrompt("Error","Invalid sector size");
return 0;
}
GUI_WaitPrompt("Error","Unable to mount memory card");
return 0;
}
/****************************************************************************
* ManageState
*
* Here is the main Freeze File Management stuff.
* The output file contains an icon (2K), 64 bytes comment and the STATE (~128k)
*
* direction == 0 save, 1 load.
****************************************************************************/
int ManageState (u8 direction, u8 device)
{
if (!cart.romsize)
return 0;
char filename[MAXJOLIET];
if (direction)
GUI_MsgBoxOpen("Information","Loading State ...",1);
else
GUI_MsgBoxOpen("Information","Saving State ...",1);
/* clean buffer */
memset(savebuffer, 0, STATE_SIZE);
if (device == 0)
{
/* FAT support */
sprintf (filename, "%s.gpz", rom_filename);
return FAT_ManageFile(filename,direction,0);
}
/* Memory CARD support */
char action[80];
int CardError;
unsigned int SectorSize;
int blocks;
char comment[2][32] = { {"Genesis Plus 1.2a [FRZ]"}, {"Freeze State"} };
int outbytes = 0;
int sbo;
int state_size = 0;
card_file CardFile;
card_stat CardStatus;
/* First, build a filename */
sprintf (filename, "MD-%04X.gpz", realchecksum);
strcpy (comment[1], filename);
/* set MCARD slot nr. */
u8 CARDSLOT = device - 1;
/* Saving */
if (direction == 0)
{
/* Build the output buffer */
memcpy (&savebuffer, &icon, 2048);
memcpy (&savebuffer[2048], &comment[0], 64);
state_size = state_save(&savebuffer[2112]);
}
outbytes = 2048 + 64 + state_size;
/*** Initialise the CARD system ***/
memset (&SysArea, 0, CARD_WORKAREA);
CARD_Init ("GENP", "00");
/*** Attempt to mount the card ***/
CardError = MountTheCard (CARDSLOT);
if (CardError)
{
/*** Retrieve the sector size ***/
CardError = CARD_GetSectorSize (CARDSLOT, &SectorSize);
if (SectorSize)
{
switch (direction)
{
case 0: /*** Saving ***/
/*** Determine number of blocks on this card ***/
blocks = (outbytes / SectorSize) * SectorSize;
if (outbytes % SectorSize)
blocks += SectorSize;
/*** Check if a previous save exists ***/
if (CardFileExists (filename, CARDSLOT))
{
CardError = CARD_Open (CARDSLOT, filename, &CardFile);
if (CardError)
{
sprintf (action, "Unable to open file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount (CARDSLOT);
return 0;
}
int size = CardFile.len;
CARD_Close (&CardFile);
if (size < blocks)
{
/* new size is bigger: check if there is enough space left */
CardError = CARD_Create (CARDSLOT, "TEMP", blocks-size, &CardFile);
if (CardError)
{
sprintf (action, "Unable to create temporary file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount (CARDSLOT);
return 0;
}
CARD_Close (&CardFile);
CARD_Delete(CARDSLOT, "TEMP");
}
/* always delete existing slot */
CARD_Delete(CARDSLOT, filename);
}
/*** Create a new slot ***/
CardError = CARD_Create (CARDSLOT, filename, blocks, &CardFile);
if (CardError)
{
sprintf (action, "Unable to create new file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount (CARDSLOT);
return 0;
}
/*** Continue and save ***/
CARD_GetStatus (CARDSLOT, CardFile.filenum, &CardStatus);
CardStatus.icon_addr = 0x0;
CardStatus.icon_fmt = 2;
CardStatus.icon_speed = 1;
CardStatus.comment_addr = 2048;
CARD_SetStatus (CARDSLOT, CardFile.filenum, &CardStatus);
/*** And write the blocks out ***/
sbo = 0;
while (outbytes > 0)
{
CardError = CARD_Write (&CardFile, &savebuffer[sbo], SectorSize, sbo);
outbytes -= SectorSize;
sbo += SectorSize;
}
CARD_Close (&CardFile);
CARD_Unmount (CARDSLOT);
GUI_MsgBoxClose();
return 1;
default: /*** Loading ***/
memset (&CardFile, 0, sizeof (CardFile));
CardError = CARD_Open (CARDSLOT, filename, &CardFile);
if (CardError)
{
sprintf (action, "Unable to open file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount (CARDSLOT);
return 0;
}
blocks = CardFile.len;
if (blocks < SectorSize)
blocks = SectorSize;
if (blocks % SectorSize)
blocks++;
/*** Just read the file back in ***/
sbo = 0;
while (blocks > 0)
{
CARD_Read (&CardFile, &savebuffer[sbo], SectorSize, sbo);
sbo += SectorSize;
blocks -= SectorSize;
}
CARD_Close (&CardFile);
CARD_Unmount (CARDSLOT);
/*** Load State ***/
state_load(&savebuffer[2112]);
GUI_MsgBoxClose();
return 1;
}
}
GUI_WaitPrompt("Error","Invalid sector size");
return 0;
}
GUI_WaitPrompt("Error","Unable to mount memory card !");
return 0;
}

View File

@ -0,0 +1,569 @@
/*
* file_slot.c
*
* FAT and Memory Card SRAM/State slots managment
*
* Softdev (2006)
* Eke-Eke (2007,2008,2009)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
***************************************************************************/
#include "shared.h"
#include "file_slot.h"
#include "file_fat.h"
#include "dvd.h"
#include "gui.h"
#include "filesel.h"
#include "saveicon.h"
/**
* libOGC CARD System Work Area
*/
static u8 SysArea[CARD_WORKAREA] ATTRIBUTE_ALIGN (32);
/**
* DMA Transfer Area.
* Must be 32-byte aligned.
* 64k SRAM + 2k Icon
*/
static u8 savebuffer[STATE_SIZE] ATTRIBUTE_ALIGN (32);
/****************************************************************************
* CardMount
*
* libOGC provides the CARD_Mount function, and it should be all you need.
* However, experience with previous emulators has taught me that you are
* better off doing a little bit more than that!
*
*****************************************************************************/
static int CardMount(int slot)
{
int tries = 0;
int CardError;
#if defined(HW_DOL)
*(unsigned long *) (0xCC006800) |= 1 << 13; /*** Disable Encryption ***/
uselessinquiry ();
#elif defined(HW_RVL)
*(unsigned long *) (0xCD006800) |= 1 << 13; /*** Disable Encryption ***/
#endif
while (tries < 10)
{
VIDEO_WaitVSync ();
CardError = CARD_Mount (slot, SysArea, NULL); /*** Don't need or want a callback ***/
if (CardError == 0)
return 1;
else
EXI_ProbeReset ();
tries++;
}
return 0;
}
/****************************************************************************
* CardFileExists
*
* Wrapper to search through the files on the card.
****************************************************************************/
static int CardFileExists (char *filename, int slot)
{
card_dir CardDir;
int CardError = CARD_FindFirst (slot, &CardDir, TRUE);
while (CardError >= 0)
{
CardError = CARD_FindNext (&CardDir);
if (strcmp ((char *) CardDir.filename, filename) == 0)
return 1;
}
return 0;
}
/****************************************************************************
* Slot Management
*
*
****************************************************************************/
void slot_autoload(int slot, int device)
{
if (!cart.romsize)
return;
SILENT = 1;
slot_load(slot, device);
SILENT = 0;
}
void slot_autosave(int slot, int device)
{
if (!cart.romsize)
return;
/* only save if SRAM changed */
if (!slot && (crc32(0, &sram.sram[0], 0x10000) == sram.crc))
return;
SILENT = 1;
slot_save(slot, device);
SILENT = 0;
}
void slot_autodetect(int slot, int device, t_slot *ptr)
{
if (!ptr)
return;
char filename[MAXPATHLEN];
memset(ptr,0,sizeof(t_slot));
if (device == 0)
{
/* FAT support */
if (slot > 0)
sprintf (filename,"%s/saves/%s.gp%d", DEFAULT_PATH, rom_filename, slot - 1);
else
sprintf (filename,"%s/saves/%s.srm", DEFAULT_PATH, rom_filename);
/* Open file */
FILE *fp = fopen(filename, "rb");
if (fp)
{
/* Retrieve date & close */
struct stat filestat;
stat(filename, &filestat);
struct tm *timeinfo = localtime(&filestat.st_mtime);
ptr->year = 1900 + timeinfo->tm_year;
ptr->month = timeinfo->tm_mon;
ptr->day = timeinfo->tm_mday;
ptr->hour = timeinfo->tm_hour;
ptr->min = timeinfo->tm_min;
fclose(fp);
ptr->valid = 1;
}
}
else
{
/* Memory Card support */
if (slot > 0)
sprintf(filename,"MD-%04X.gp%d", realchecksum, slot - 1);
else
sprintf(filename,"MD-%04X.srm", realchecksum);
/* Initialise the CARD system */
memset(&SysArea, 0, CARD_WORKAREA);
CARD_Init("GENP", "00");
/* CARD slot */
device--;
/* Mount CARD */
if (CardMount(device))
{
/* Open file */
card_file CardFile;
if (CARD_Open(device, filename, &CardFile))
{
/* Retrieve date & close */
card_stat CardStatus;
CARD_GetStatus(device, CardFile.filenum, &CardStatus);
time_t rawtime = CardStatus.time;
struct tm *timeinfo = localtime(&rawtime);
ptr->year = 1900 + timeinfo->tm_year;
ptr->month = timeinfo->tm_mon;
ptr->day = timeinfo->tm_mday;
ptr->hour = timeinfo->tm_hour;
ptr->min = timeinfo->tm_min;
CARD_Close(&CardFile);
ptr->valid = 1;
}
CARD_Unmount(device);
}
}
}
int slot_delete(int slot, int device)
{
char filename[MAXPATHLEN];
int ret = 0;
if (device == 0)
{
/* FAT support */
if (slot > 0)
{
/* remove screenshot */
sprintf(filename,"%s/saves/%s__%d.png", DEFAULT_PATH, rom_filename, slot - 1);
remove(filename);
sprintf (filename,"%s/saves/%s.gp%d", DEFAULT_PATH, rom_filename, slot - 1);
}
else
sprintf (filename,"%s/saves/%s.srm", DEFAULT_PATH, rom_filename);
/* Delete file */
ret = remove(filename);
}
else
{
/* Memory Card support */
if (slot > 0)
sprintf(filename,"MD-%04X.gp%d", realchecksum, slot - 1);
else
sprintf(filename,"MD-%04X.srm", realchecksum);
/* Initialise the CARD system */
memset(&SysArea, 0, CARD_WORKAREA);
CARD_Init("GENP", "00");
/* CARD slot */
device--;
/* Mount CARD */
if (CardMount(device))
{
/* Open file */
if (CardFileExists(filename, device))
ret = CARD_Delete(device,filename);
CARD_Unmount(device);
}
}
return ret;
}
int slot_load(int slot, int device)
{
char filename[MAXPATHLEN];
int filesize, done = 0;
int offset = device ? 2112 : 0;
if (slot > 0)
GUI_MsgBoxOpen("Information","Loading State ...",1);
else
GUI_MsgBoxOpen("Information","Loading SRAM ...",1);
/* clean buffer */
memset(savebuffer, 0, STATE_SIZE);
if (device == 0)
{
/* FAT support */
if (slot > 0)
sprintf (filename,"%s/saves/%s.gp%d", DEFAULT_PATH, rom_filename, slot - 1);
else
sprintf (filename,"%s/saves/%s.srm", DEFAULT_PATH, rom_filename);
/* Open file */
FILE *fp = fopen(filename, "rb");
if (fp)
{
/* Read size */
fseek(fp , 0 , SEEK_END);
filesize = ftell (fp);
fseek(fp, 0, SEEK_SET);
/* Read into buffer (2k blocks) */
while (filesize > FATCHUNK)
{
fread(savebuffer + done, FATCHUNK, 1, fp);
done += FATCHUNK;
filesize -= FATCHUNK;
}
/* Read remaining bytes */
fread(savebuffer + done, filesize, 1, fp);
done += filesize;
fclose(fp);
}
else
{
GUI_WaitPrompt("Error","Unable to open file !");
return 0;
}
}
else
{
/* Memory Card support */
if (index > 0)
sprintf(filename, "MD-%04X.gp%d", realchecksum, slot - 1);
else
sprintf(filename, "MD-%04X.srm", realchecksum);
/* Initialise the CARD system */
char action[80];
memset(&SysArea, 0, CARD_WORKAREA);
CARD_Init("GENP", "00");
/* CARD slot */
device--;
/* Attempt to mount the card */
if (CardMount(device))
{
/* Retrieve the sector size */
u32 SectorSize = 0;
int CardError = CARD_GetSectorSize(device, &SectorSize);
if (SectorSize)
{
/* Open file */
card_file CardFile;
CardError = CARD_Open(device, filename, &CardFile);
if (CardError)
{
sprintf(action, "Unable to open file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount(device);
return 0;
}
/* Get file size */
filesize = CardFile.len;
if (filesize % SectorSize)
filesize = ((filesize / SectorSize) + 1) * SectorSize;
/* Read file sectors */
while (filesize > 0)
{
CARD_Read(&CardFile, &savebuffer[done], SectorSize, done);
done += SectorSize;
filesize -= SectorSize;
}
CARD_Close(&CardFile);
CARD_Unmount(device);
}
else
{
sprintf(action, "Invalid sector size (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount(device);
return 0;
}
}
else
{
GUI_WaitPrompt("Error","Unable to mount memory card");
return 0;
}
}
if (slot > 0)
{
/* Load state */
if (!state_load(&savebuffer[offset]))
{
GUI_WaitPrompt("Error","Version is not compatible !");
return 0;
}
}
else
{
/* Load SRAM & update CRC */
memcpy(sram.sram, &savebuffer[offset], 0x10000);
sram.crc = crc32(0, &sram.sram[0], 0x10000);
}
GUI_MsgBoxClose();
return 1;
}
int slot_save(int slot, int device)
{
char filename[MAXPATHLEN];
int filesize, done = 0;
int offset = device ? 2112 : 0;
/* clean buffer */
memset(savebuffer, 0, STATE_SIZE);
if (slot > 0)
{
GUI_MsgBoxOpen("Information","Saving State ...",1);
filesize = state_save(&savebuffer[offset]);
}
else
{
GUI_MsgBoxOpen("Information","Saving SRAM ...",1);
memcpy(&savebuffer[offset], sram.sram, 0x10000);
sram.crc = crc32(0, &sram.sram[0], 0x10000);
filesize = 0x10000;
}
if (device == 0)
{
/* FAT support */
if (slot > 0)
sprintf(filename, "%s/saves/%s.gp%d", DEFAULT_PATH, rom_filename, slot - 1);
else
sprintf(filename, "%s/saves/%s.srm", DEFAULT_PATH, rom_filename);
/* Open file */
FILE *fp = fopen(filename, "wb");
if (fp)
{
/* Read into buffer (2k blocks) */
while (filesize > FATCHUNK)
{
fwrite(savebuffer + done, FATCHUNK, 1, fp);
done += FATCHUNK;
filesize -= FATCHUNK;
}
/* Write remaining bytes */
fwrite(savebuffer + done, filesize, 1, fp);
done += filesize;
fclose(fp);
if (slot)
{
/* save screenshot */
sprintf(filename,"%s/saves/%s__%d.png", DEFAULT_PATH, rom_filename, slot - 1);
gxSaveScreenshot(filename);
}
}
else
{
GUI_WaitPrompt("Error","Unable to open file !");
return 0;
}
}
else
{
/* Memory Card support */
if (index > 0)
sprintf(filename, "MD-%04X.gp%d", realchecksum, slot - 1);
else
sprintf(filename, "MD-%04X.srm", realchecksum);
/* Initialise the CARD system */
char action[80];
memset(&SysArea, 0, CARD_WORKAREA);
CARD_Init("GENP", "00");
/* CARD slot */
device--;
/* Attempt to mount the card */
if (CardMount(device))
{
/* Retrieve the sector size */
u32 SectorSize = 0;
int CardError = CARD_GetSectorSize(device, &SectorSize);
if (SectorSize)
{
/* Build the output buffer */
char comment[2][32] = { {"Genesis Plus GX"}, {"SRAM Save"} };
strcpy (comment[1], filename);
memcpy (&savebuffer[0], &icon, 2048);
memcpy (&savebuffer[2048], &comment[0], 64);
/* Adjust file size */
filesize += 2112;
if (filesize % SectorSize)
filesize = ((filesize / SectorSize) + 1) * SectorSize;
/* Check if file already exists */
card_file CardFile;
if (CardFileExists(filename,device))
{
CardError = CARD_Open(device, filename, &CardFile);
if (CardError)
{
sprintf(action, "Unable to open file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount(device);
return 0;
}
int size = filesize - CardFile.len;
CARD_Close(&CardFile);
memset(&CardFile,0,sizeof(CardFile));
if (size > 0)
{
/* new file is bigger: check if there is enough space left */
CardError = CARD_Create(device, "TEMP", size, &CardFile);
if (CardError)
{
sprintf(action, "Unable to create temporary file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount(device);
return 0;
}
CARD_Close(&CardFile);
CARD_Delete(device, "TEMP");
memset(&CardFile,0,sizeof(CardFile));
}
/* delete previously existing slot */
CARD_Delete(device, filename);
}
/* Create a new slot */
CardError = CARD_Create(device, filename, filesize, &CardFile);
if (CardError)
{
sprintf(action, "Unable to create file (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount(device);
return 0;
}
/* Get current time */
time_t rawtime;
time(&rawtime);
/* Update file informations */
card_stat CardStatus;
CARD_GetStatus(device, CardFile.filenum, &CardStatus);
CardStatus.icon_addr = 0x0;
CardStatus.icon_fmt = 2;
CardStatus.icon_speed = 1;
CardStatus.comment_addr = 2048;
CardStatus.time = rawtime;
CARD_SetStatus(device, CardFile.filenum, &CardStatus);
/* Write file sectors */
while (filesize > 0)
{
CARD_Write(&CardFile, &savebuffer[done], SectorSize, done);
filesize -= SectorSize;
done += SectorSize;
}
/* Close file */
CARD_Close(&CardFile);
CARD_Unmount(device);
}
else
{
sprintf(action, "Invalid sector size (%d)", CardError);
GUI_WaitPrompt("Error",action);
CARD_Unmount(device);
return 0;
}
}
else
{
GUI_WaitPrompt("Error","Unable to mount memory card");
return 0;
}
}
GUI_MsgBoxClose();
return 1;
}

View File

@ -1,5 +1,5 @@
/* /*
* file_mem.c * file_slot.c
* *
* FAT and Memory Card SRAM/Savestate files managment * FAT and Memory Card SRAM/Savestate files managment
* *
@ -22,14 +22,24 @@
* *
***************************************************************************/ ***************************************************************************/
#ifndef _FILE_MEM_H #ifndef _FILE_SLOT_H
#define _FILE_MEM_H #define _FILE_SLOT_H
extern int ManageSRAM(u8 direction, u8 device); typedef struct
extern int ManageState(u8 direction, u8 device); {
extern void memfile_autosave(s8 autosram, s8 autostate); int valid;
extern void memfile_autoload(s8 autosram, s8 autostate); u16 year;
u8 month;
u8 day;
u8 hour;
u8 min;
} t_slot;
extern char rom_filename[256]; extern void slot_autoload(int slot, int device);
extern void slot_autosave(int slot, int device);
extern void slot_autodetect(int slot, int device, t_slot *ptr);
extern int slot_delete(int slot, int device);
extern int slot_load(int slot, int device);
extern int slot_save(int slot, int device);
#endif #endif

View File

@ -23,11 +23,13 @@
********************************************************************************/ ********************************************************************************/
#include "shared.h" #include "shared.h"
#include "filesel.h"
#include "menu.h"
#include "font.h" #include "font.h"
#include "gui.h" #include "gui.h"
#include "file_dvd.h" #include "file_dvd.h"
#include "file_fat.h" #include "file_fat.h"
#include "filesel.h" #include "file_slot.h"
#ifdef HW_RVL #ifdef HW_RVL
#include <wiiuse/wpad.h> #include <wiiuse/wpad.h>
@ -43,6 +45,8 @@ static int selection = 0;
static int old_selection = 0; static int old_selection = 0;
static int old_offset = 0; static int old_offset = 0;
static int maxfiles = 0; static int maxfiles = 0;
static int string_offset = 0;
static void selector_cb(void);
/*****************************************************************************/ /*****************************************************************************/
/* GUI Buttons data */ /* GUI Buttons data */
@ -63,8 +67,8 @@ static butn_data arrow_down_data =
/* GUI Arrows button */ /* GUI Arrows button */
/*****************************************************************************/ /*****************************************************************************/
static gui_butn arrow_up = {&arrow_up_data,BUTTON_VISIBLE|BUTTON_OVER_SFX,{0,0,0,0},14,76,360,32}; static gui_butn arrow_up = {&arrow_up_data,BUTTON_VISIBLE|BUTTON_ACTIVE|BUTTON_OVER_SFX,{0,0,0,0},14,76,360,32};
static gui_butn arrow_down = {&arrow_down_data,BUTTON_VISIBLE|BUTTON_OVER_SFX,{0,0,0,0},14,368,360,32}; static gui_butn arrow_down = {&arrow_down_data,BUTTON_VISIBLE|BUTTON_ACTIVE|BUTTON_OVER_SFX,{0,0,0,0},14,368,360,32};
/*****************************************************************************/ /*****************************************************************************/
/* GUI helpers */ /* GUI helpers */
@ -108,7 +112,8 @@ static gui_menu menu_browser =
NULL, NULL,
bg_filesel, bg_filesel,
{&action_cancel, &action_select}, {&action_cancel, &action_select},
{&arrow_up,&arrow_down} {&arrow_up,&arrow_down},
selector_cb
}; };
/*************************************************************************** /***************************************************************************
@ -146,36 +151,11 @@ int FileSortCallback(const void *f1, const void *f2)
* ROM size is returned * ROM size is returned
* *
****************************************************************************/ ****************************************************************************/
int FileSelector(unsigned char *buffer, bool useFAT) static void selector_cb(void)
{ {
short p; int i;
int ret,i,yoffset;
int size = 0;
int go_up = 0;
int string_offset = 0;
int old = -1;
char text[MAXPATHLEN]; char text[MAXPATHLEN];
char fname[MAXPATHLEN]; int yoffset = PAGEOFFSET;
FILE *xml,*snap;
#ifdef HW_RVL
int x,y;
gui_butn *button;
#endif
/* Background Settings */
if (config.bg_color == 0)
bg_filesel[0].data = Bg_main_2_png;
else
bg_filesel[0].data = Bg_main_png;
if (config.bg_overlay)
bg_filesel[1].state |= IMAGE_VISIBLE;
else
bg_filesel[1].state &= ~IMAGE_VISIBLE;
/* Initialize Menu */
gui_menu *m = &menu_browser;
GUI_InitMenu(m);
/* Initialize directory icon */ /* Initialize directory icon */
gui_image dir_icon; gui_image dir_icon;
@ -193,6 +173,91 @@ int FileSelector(unsigned char *buffer, bool useFAT)
bar_over.x = 22; bar_over.x = 22;
bar_over.y = -(bar_over.h - dir_icon.h)/2; bar_over.y = -(bar_over.h - dir_icon.h)/2;
/* Draw Files list */
for (i = offset; i < (offset + PAGESIZE) && (i < maxfiles); i++)
{
if (i == selection)
{
/* selection bar */
gxDrawTexture(bar_over.texture,bar_over.x,yoffset+bar_over.y,bar_over.w,bar_over.h,255);
/* scrolling text */
if ((string_offset/10) >= strlen(filelist[i].filename))
string_offset = 0;
sprintf(text, "%s ",filelist[i].filename + string_offset/10);
strncat(text, filelist[i].filename, string_offset/10);
if (filelist[i].flags)
{
/* directory icon */
gxDrawTexture(dir_icon.texture,dir_icon.x-1,yoffset-1,dir_icon.w+2,dir_icon.h+2,255);
if (FONT_write(text,18,dir_icon.x+dir_icon.w+6,yoffset+16,bar_over.w-dir_icon.w-14,(GXColor)WHITE))
{
/* string is too large -> scroll text */
string_offset ++;
}
}
else
{
if (FONT_write(text,18,dir_icon.x,yoffset+16,bar_over.w-8,(GXColor)WHITE))
{
/* text scrolling */
string_offset ++;
}
}
}
else
{
if (filelist[i].flags)
{
/* directory icon */
gxDrawTexture(dir_icon.texture,dir_icon.x,yoffset,dir_icon.w,dir_icon.h,255);
FONT_write(filelist[i].filename,16,dir_icon.x+dir_icon.w+6,yoffset+16,bar_over.w-dir_icon.w-14,(GXColor)WHITE);
}
else
{
FONT_write(filelist[i].filename,16,dir_icon.x,yoffset+16,bar_over.w-8,(GXColor)WHITE);
}
}
yoffset += 22;
}
gxTextureClose(&bar_over.texture);
gxTextureClose(&dir_icon.texture);
}
int FileSelector(unsigned char *buffer, bool useFAT)
{
short p;
int ret;
int size = 0;
int go_up = 0;
int old = -1;
char fname[MAXPATHLEN];
char text[MAXPATHLEN];
FILE *xml,*snap;
#ifdef HW_RVL
int x,y,i,yoffset;
gui_butn *button;
#endif
/* Background Settings */
if (config.bg_color == 0)
bg_filesel[0].data = Bg_main_2_png;
else
bg_filesel[0].data = Bg_main_png;
if (config.bg_overlay)
bg_filesel[1].state |= IMAGE_VISIBLE;
else
bg_filesel[1].state &= ~IMAGE_VISIBLE;
/* Initialize Menu */
gui_menu *m = &menu_browser;
GUI_InitMenu(m);
string_offset = 0;
while (1) while (1)
{ {
/* ROM file snapshot/database */ /* ROM file snapshot/database */
@ -210,7 +275,8 @@ int FileSelector(unsigned char *buffer, bool useFAT)
{ {
/* get ROM filename without extension */ /* get ROM filename without extension */
sprintf (text, "%s", filelist[selection].filename); sprintf (text, "%s", filelist[selection].filename);
if (strlen(text) >= 4) text[strlen(text) - 4] = 0; if (strlen(text) >= 4)
text[strlen(text) - 4] = 0;
/* ROM database informations */ /* ROM database informations */
sprintf (fname, "%s/db/%s.xml", DEFAULT_PATH, text); sprintf (fname, "%s/db/%s.xml", DEFAULT_PATH, text);
@ -227,7 +293,8 @@ int FileSelector(unsigned char *buffer, bool useFAT)
if (snap) if (snap)
{ {
bg_filesel[8].texture = gxTextureOpenPNG(0,snap); bg_filesel[8].texture = gxTextureOpenPNG(0,snap);
if (bg_filesel[8].texture) bg_filesel[8].state |= IMAGE_VISIBLE; if (bg_filesel[8].texture)
bg_filesel[8].state |= IMAGE_VISIBLE;
fclose(snap); fclose(snap);
} }
} }
@ -256,60 +323,10 @@ int FileSelector(unsigned char *buffer, bool useFAT)
/* Draw menu*/ /* Draw menu*/
GUI_DrawMenu(m); GUI_DrawMenu(m);
/* Draw Files list */
yoffset = PAGEOFFSET;
for (i = offset; i < (offset + PAGESIZE) && (i < maxfiles); i++)
{
if (i == selection)
{
/* scrolling text */
if ((string_offset/10) >= strlen(filelist[i].filename)) string_offset = 0;
sprintf(text, "%s ",filelist[i].filename + string_offset/10);
strncat(text, filelist[i].filename, string_offset/10);
gxDrawTexture(bar_over.texture,bar_over.x,yoffset+bar_over.y,bar_over.w,bar_over.h,255);
if (filelist[i].flags)
{
/* directory icon */
gxDrawTexture(dir_icon.texture,dir_icon.x-1,yoffset-1,dir_icon.w+2,dir_icon.h+2,255);
if (FONT_write(text,18,dir_icon.x+dir_icon.w+6,yoffset+16,bar_over.w-dir_icon.w-14,(GXColor)WHITE))
{
/* text scrolling */
string_offset ++;
}
}
else
{
if (FONT_write(text,18,dir_icon.x,yoffset+16,bar_over.w-8,(GXColor)WHITE))
{
/* text scrolling */
string_offset ++;
}
}
}
else
{
if (filelist[i].flags)
{
/* directory icon */
gxDrawTexture(dir_icon.texture,dir_icon.x,yoffset,dir_icon.w,dir_icon.h,255);
FONT_write(filelist[i].filename,16,dir_icon.x+dir_icon.w+6,yoffset+16,bar_over.w-dir_icon.w-14,(GXColor)WHITE);
}
else
{
FONT_write(filelist[i].filename,16,dir_icon.x,yoffset+16,bar_over.w-8,(GXColor)WHITE);
}
}
yoffset += 22;
}
#ifdef HW_RVL #ifdef HW_RVL
if (Shutdown) if (Shutdown)
{ {
gxTextureClose(&w_pointer); gxTextureClose(&w_pointer);
gxTextureClose(&bar_over.texture);
gxTextureClose(&dir_icon.texture);
GUI_DeleteMenu(m); GUI_DeleteMenu(m);
GUI_FadeOut(); GUI_FadeOut();
shutdown(); shutdown();
@ -371,8 +388,10 @@ int FileSelector(unsigned char *buffer, bool useFAT)
if (p & PAD_BUTTON_DOWN) if (p & PAD_BUTTON_DOWN)
{ {
selection++; selection++;
if (selection == maxfiles) selection = offset = 0; if (selection == maxfiles)
if ((selection - offset) >= PAGESIZE) offset += PAGESIZE; selection = offset = 0;
if ((selection - offset) >= PAGESIZE)
offset += PAGESIZE;
} }
/* highlight previous item */ /* highlight previous item */
@ -384,8 +403,10 @@ int FileSelector(unsigned char *buffer, bool useFAT)
selection = maxfiles - 1; selection = maxfiles - 1;
offset = selection - PAGESIZE + 1; offset = selection - PAGESIZE + 1;
} }
if (selection < offset) offset -= PAGESIZE; if (selection < offset)
if (offset < 0) offset = 0; offset -= PAGESIZE;
if (offset < 0)
offset = 0;
} }
/* go back one page */ /* go back one page */
@ -397,24 +418,26 @@ int FileSelector(unsigned char *buffer, bool useFAT)
selection = maxfiles - 1; selection = maxfiles - 1;
offset = selection - PAGESIZE + 1; offset = selection - PAGESIZE + 1;
} }
if (selection < offset) offset -= PAGESIZE; if (selection < offset)
if (offset < 0) offset = 0; offset -= PAGESIZE;
if (offset < 0)
offset = 0;
} }
/* go forward one page */ /* go forward one page */
else if (p & PAD_TRIGGER_R) else if (p & PAD_TRIGGER_R)
{ {
selection += PAGESIZE; selection += PAGESIZE;
if (selection > maxfiles - 1) selection = offset = 0; if (selection > maxfiles - 1)
if ((selection - offset) >= PAGESIZE) offset += PAGESIZE; selection = offset = 0;
if ((selection - offset) >= PAGESIZE)
offset += PAGESIZE;
} }
/* quit */ /* quit */
else if (p & PAD_TRIGGER_Z) else if (p & PAD_TRIGGER_Z)
{ {
GUI_DeleteMenu(m); GUI_DeleteMenu(m);
gxTextureClose(&bar_over.texture);
gxTextureClose(&dir_icon.texture);
return 0; return 0;
} }
@ -442,14 +465,18 @@ int FileSelector(unsigned char *buffer, bool useFAT)
selection = maxfiles - 1; selection = maxfiles - 1;
offset = selection - PAGESIZE + 1; offset = selection - PAGESIZE + 1;
} }
if (selection < offset) offset -= PAGESIZE; if (selection < offset)
if (offset < 0) offset = 0; offset -= PAGESIZE;
if (offset < 0)
offset = 0;
} }
else if (m->selected == (m->max_buttons+1)) /* down arrow */ else if (m->selected == (m->max_buttons+1)) /* down arrow */
{ {
selection++; selection++;
if (selection == maxfiles) selection = offset = 0; if (selection == maxfiles)
if ((selection - offset) >= PAGESIZE) offset += PAGESIZE; selection = offset = 0;
if ((selection - offset) >= PAGESIZE)
offset += PAGESIZE;
} }
} }
#endif #endif
@ -498,8 +525,6 @@ int FileSelector(unsigned char *buffer, bool useFAT)
else else
{ {
GUI_DeleteMenu(m); GUI_DeleteMenu(m);
gxTextureClose(&bar_over.texture);
gxTextureClose(&dir_icon.texture);
return 0; return 0;
} }
} }
@ -511,8 +536,6 @@ int FileSelector(unsigned char *buffer, bool useFAT)
if (go_up) if (go_up)
{ {
GUI_DeleteMenu(m); GUI_DeleteMenu(m);
gxTextureClose(&bar_over.texture);
gxTextureClose(&dir_icon.texture);
return 0; return 0;
} }
else else
@ -529,16 +552,18 @@ int FileSelector(unsigned char *buffer, bool useFAT)
/* Reload emulation */ /* Reload emulation */
if (size) if (size)
{ {
memfile_autosave(-1,config.state_auto); if (config.s_auto & 2)
slot_autosave(config.s_default,config.s_device);
reloadrom(size,filelist[selection].filename); reloadrom(size,filelist[selection].filename);
memfile_autoload(config.sram_auto,config.state_auto); if (config.s_auto & 1)
slot_autoload(0,config.s_device);
if (config.s_auto & 2)
slot_autoload(config.s_default,config.s_device);
} }
/* Exit */ /* Exit */
GUI_MsgBoxClose(); GUI_MsgBoxClose();
GUI_DeleteMenu(m); GUI_DeleteMenu(m);
gxTextureClose(&bar_over.texture);
gxTextureClose(&dir_icon.texture);
return size; return size;
} }

View File

@ -24,7 +24,7 @@
#include "shared.h" #include "shared.h"
#include "font.h" #include "font.h"
#include "gui.h" #include "menu.h"
#define _SHIFTR(v, s, w) \ #define _SHIFTR(v, s, w) \
((u32)(((u32)(v) >> (s)) & ((0x01 << (w)) - 1))) ((u32)(((u32)(v) >> (s)) & ((0x01 << (w)) - 1)))

View File

@ -28,8 +28,6 @@
#define MAXCODES 8 #define MAXCODES 8
extern char menutitle[60];
typedef struct typedef struct
{ {
int address; int address;
@ -168,7 +166,7 @@ void DrawGGCodes ()
unsigned char c[2] = { 0, 0 }; unsigned char c[2] = { 0, 0 };
gxClearScreen ((GXColor)BLACK); gxClearScreen ((GXColor)BLACK);
WriteCentre (134, menutitle); WriteCentre (134, "Game Genie Entry");
for (i = 0; i < MAXCODES; i++) for (i = 0; i < MAXCODES; i++)
{ {
@ -340,7 +338,6 @@ void GGSelectLine ()
void GetGGEntries () void GetGGEntries ()
{ {
editing = 0; editing = 0;
strcpy (menutitle, "Game Genie Entry");
GGSelectLine (); GGSelectLine ();
/* Apply Game Genie patches */ /* Apply Game Genie patches */

View File

@ -219,20 +219,26 @@ void GUI_DrawMenu(gui_menu *menu)
if (button->state & BUTTON_VISIBLE) if (button->state & BUTTON_VISIBLE)
{ {
/* draw button + items */ /* item select (text or image) */
item = &menu->items[menu->offset +i]; item = (menu->items) ? (&menu->items[menu->offset + i]) : NULL;
if ((i == menu->selected) && (button->state & BUTTON_ACTIVE))
{
if (button->data) gxDrawTexture(button->data->texture[1],button->x-4,button->y-4,button->w+8,button->h+8,255);
if (item->texture) /* draw button + items */
if ((i == menu->selected) || (button->state & BUTTON_SELECTED))
{
if (button->data)
gxDrawTexture(button->data->texture[1],button->x-4,button->y-4,button->w+8,button->h+8,255);
if (item)
{ {
gxDrawTexture(item->texture, item->x-4,item->y-4,item->w+8,item->h+8,255); if (item->texture)
FONT_writeCenter(item->text,18,button->x+4,item->x-4,button->y+(button->h - 36)/2+18,(GXColor)DARK_GREY); {
} gxDrawTexture(item->texture, item->x-4,item->y-4,item->w+8,item->h+8,255);
else FONT_writeCenter(item->text,18,button->x+4,item->x-4,button->y+(button->h - 36)/2+18,(GXColor)DARK_GREY);
{ }
FONT_writeCenter(item->text,18,item->x+2,item->x+item->w+2,button->y+(button->h-18)/2+18,(GXColor)DARK_GREY); else
{
FONT_writeCenter(item->text,18,item->x+2,item->x+item->w+2,button->y+(button->h-18)/2+18,(GXColor)DARK_GREY);
}
} }
} }
else else
@ -240,14 +246,17 @@ void GUI_DrawMenu(gui_menu *menu)
if (button->data) if (button->data)
gxDrawTexture(button->data->texture[0],button->x,button->y,button->w, button->h,255); gxDrawTexture(button->data->texture[0],button->x,button->y,button->w, button->h,255);
if (item->texture) if (item)
{ {
gxDrawTexture(item->texture,item->x,item->y,item->w,item->h,255); if (item->texture)
FONT_writeCenter(item->text,16,button->x+8,item->x,button->y+(button->h - 32)/2+16,(GXColor)DARK_GREY); {
} gxDrawTexture(item->texture,item->x,item->y,item->w,item->h,255);
else FONT_writeCenter(item->text,16,button->x+8,item->x,button->y+(button->h - 32)/2+16,(GXColor)DARK_GREY);
{ }
FONT_writeCenter(item->text,16,item->x,item->x+item->w,button->y+(button->h - 16)/2+16,(GXColor)DARK_GREY); else
{
FONT_writeCenter(item->text,16,item->x,item->x+item->w,button->y+(button->h - 16)/2+16,(GXColor)DARK_GREY);
}
} }
} }
} }
@ -290,6 +299,9 @@ void GUI_DrawMenu(gui_menu *menu)
FONT_alignRight(item->comment,16,item->x-6,item->y+(item->h-16)/2+16,(GXColor)WHITE); FONT_alignRight(item->comment,16,item->x-6,item->y+(item->h-16)/2+16,(GXColor)WHITE);
} }
} }
if (menu->cb)
menu->cb();
} }
/* Draw Menu with transitions effects */ /* Draw Menu with transitions effects */
@ -321,7 +333,6 @@ void GUI_DrawMenuFX(gui_menu *menu, u8 speed, u8 out)
max_offset = temp; max_offset = temp;
} }
if (image->state & IMAGE_SLIDE_TOP) if (image->state & IMAGE_SLIDE_TOP)
{ {
temp = image->y + image->h; temp = image->y + image->h;
@ -410,7 +421,7 @@ void GUI_DrawMenuFX(gui_menu *menu, u8 speed, u8 out)
} }
/* menu title */ /* menu title */
if (menu->bg_images[2].state & IMAGE_SLIDE_TOP) if ((menu->bg_images[2].state & IMAGE_SLIDE_TOP) || (menu->bg_images[3].state & IMAGE_SLIDE_TOP))
FONT_write(menu->title, 22,10,out ? (56 + temp - max_offset) : (56 -temp),640,(GXColor)WHITE); FONT_write(menu->title, 22,10,out ? (56 + temp - max_offset) : (56 -temp),640,(GXColor)WHITE);
else else
FONT_write(menu->title, 22,10,56,640,(GXColor)WHITE); FONT_write(menu->title, 22,10,56,640,(GXColor)WHITE);
@ -450,24 +461,72 @@ void GUI_DrawMenuFX(gui_menu *menu, u8 speed, u8 out)
text_color.a = 255; text_color.a = 255;
} }
/* draw button + items */ /* item select (text or image) */
item = &menu->items[menu->offset + i]; item = (menu->items) ? (&menu->items[menu->offset + i]) : NULL;
/* draw button + items */
if (button->data) if (button->data)
gxDrawTexture(button->data->texture[0],button->x+xoffset,button->y+yoffset,button->w, button->h,item_alpha); gxDrawTexture(button->data->texture[0],button->x+xoffset,button->y+yoffset,button->w, button->h,item_alpha);
if (item->texture) if (item)
{ {
gxDrawTexture(item->texture,item->x+xoffset,item->y+yoffset,item->w,item->h,item_alpha); if (item->texture)
FONT_writeCenter(item->text,16,button->x+xoffset+8,item->x+xoffset,button->y+yoffset+(button->h - 32)/2+16,text_color); {
} gxDrawTexture(item->texture,item->x+xoffset,item->y+yoffset,item->w,item->h,item_alpha);
else FONT_writeCenter(item->text,16,button->x+xoffset+8,item->x+xoffset,button->y+yoffset+(button->h - 32)/2+16,text_color);
{ }
FONT_writeCenter(item->text,16,item->x+xoffset,item->x+item->w+xoffset,button->y+yoffset+(button->h - 16)/2+16,text_color); else
{
FONT_writeCenter(item->text,16,item->x+xoffset,item->x+item->w+xoffset,button->y+yoffset+(button->h - 16)/2+16,text_color);
}
} }
} }
} }
/* draw arrow */
for (i=0; i<2; i++)
{
button = menu->arrows[i];
if (button)
{
if (button->state & BUTTON_VISIBLE)
{
if (menu->selected == (menu->max_buttons + i))
gxDrawTexture(button->data->texture[1],button->x-2,button->y-2,button->w+4,button->h+4,255);
else
gxDrawTexture(button->data->texture[0],button->x,button->y,button->w, button->h,255);
}
}
}
if (!(menu->bg_images[3].state & IMAGE_SLIDE_BOTTOM) && !(menu->bg_images[4].state & IMAGE_SLIDE_BOTTOM))
{
/* left comment */
item = menu->helpers[0];
if (item)
{
if (item->data && strlen(item->comment))
{
gxDrawTexture(item->texture,item->x,item->y,item->w,item->h,255);
FONT_write(item->comment,16,item->x+item->w+6,item->y+(item->h-16)/2 + 16,640,(GXColor)WHITE);
}
}
/* right comment */
item = menu->helpers[1];
if (item)
{
if (item->data && strlen(item->comment))
{
gxDrawTexture(item->texture,item->x,item->y,item->w,item->h,255);
FONT_alignRight(item->comment,16,item->x-6,item->y+(item->h-16)/2+16,(GXColor)WHITE);
}
}
}
if (menu->cb)
menu->cb();
/* update offset */ /* update offset */
temp -= speed; temp -= speed;
@ -478,6 +537,7 @@ void GUI_DrawMenuFX(gui_menu *menu, u8 speed, u8 out)
else if (alpha < 0) else if (alpha < 0)
alpha = 0; alpha = 0;
/* copy EFB to XFB */ /* copy EFB to XFB */
gxSetScreen(); gxSetScreen();
} }
@ -517,6 +577,7 @@ void GUI_SlideMenuTitle(gui_menu *m, int title_offset)
GUI_DrawMenu(m); GUI_DrawMenu(m);
#ifdef HW_RVL #ifdef HW_RVL
/* keep pointer active */
if (m_input.ir.valid) if (m_input.ir.valid)
{ {
/* get cursor position */ /* get cursor position */
@ -554,11 +615,6 @@ void GUI_SlideMenuTitle(gui_menu *m, int title_offset)
} }
} }
} }
else
{
/* reinitialize selection */
if (m->selected >= m->max_buttons) m->selected = 0;
}
#endif #endif
gxSetScreen(); gxSetScreen();
usleep(6000); usleep(6000);
@ -600,10 +656,13 @@ int GUI_UpdateMenu(gui_menu *menu)
for (i=0; i<max_buttons; i++) for (i=0; i<max_buttons; i++)
{ {
button = &menu->buttons[i]; button = &menu->buttons[i];
if ((button->state & BUTTON_ACTIVE)&&(x>=button->x)&&(x<=(button->x+button->w))&&(y>=button->y)&&(y<=(button->y+button->h))) if ((button->state & BUTTON_ACTIVE) && (button->state & BUTTON_VISIBLE))
{ {
selected = i; if((x>=button->x)&&(x<=(button->x+button->w))&&(y>=button->y)&&(y<=(button->y+button->h)))
break; {
selected = i;
break;
}
} }
} }
@ -612,7 +671,7 @@ int GUI_UpdateMenu(gui_menu *menu)
button = menu->arrows[i]; button = menu->arrows[i];
if (button) if (button)
{ {
if (button->state & BUTTON_VISIBLE) if ((button->state & BUTTON_ACTIVE) && (button->state & BUTTON_VISIBLE))
{ {
if ((x<=(button->x+button->w))&&(y>=button->y)&&(y<=(button->y+button->h))) if ((x<=(button->x+button->w))&&(y>=button->y)&&(y<=(button->y+button->h)))
{ {
@ -626,7 +685,12 @@ int GUI_UpdateMenu(gui_menu *menu)
else else
{ {
/* reinitialize selection */ /* reinitialize selection */
if (selected >= menu->max_buttons) selected = 0; if (selected >= menu->max_buttons)
{
selected = 0;
while (!(menu->buttons[selected].state & BUTTON_ACTIVE))
selected++;
}
} }
#endif #endif
@ -683,9 +747,12 @@ int GUI_UpdateMenu(gui_menu *menu)
if (p & PAD_BUTTON_A) if (p & PAD_BUTTON_A)
{ {
if (selected < max_buttons) ret = 1; /* menu clicked */ if (selected < max_buttons)
else if (selected == max_buttons) menu->offset --; /* up arrow */ ret = 1; /* menu clicked */
else if (selected == (max_buttons+1))menu->offset ++; /* down arrow */ else if (selected == max_buttons)
menu->offset --; /* up arrow */
else if (selected == (max_buttons+1))
menu->offset ++; /* down arrow */
} }
else if ((p & PAD_BUTTON_B) || (p & PAD_TRIGGER_Z)) else if ((p & PAD_BUTTON_B) || (p & PAD_TRIGGER_Z))
{ {
@ -739,14 +806,18 @@ int GUI_UpdateMenu(gui_menu *menu)
button = menu->arrows[0]; button = menu->arrows[0];
if (button) if (button)
{ {
if (menu->offset > 0) button->state |= BUTTON_VISIBLE; if (menu->offset > 0)
else button->state &= ~BUTTON_VISIBLE; button->state |= BUTTON_VISIBLE;
else
button->state &= ~BUTTON_VISIBLE;
} }
button = menu->arrows[1]; button = menu->arrows[1];
if (button) if (button)
{ {
if ((menu->offset + max_buttons) < max_items) button->state |= BUTTON_VISIBLE; if ((menu->offset + max_buttons) < max_items)
else button->state &= ~BUTTON_VISIBLE; button->state |= BUTTON_VISIBLE;
else
button->state &= ~BUTTON_VISIBLE;
} }
if (ret > 0) if (ret > 0)
@ -778,9 +849,12 @@ int GUI_RunMenu(gui_menu *menu)
update = GUI_UpdateMenu(menu); update = GUI_UpdateMenu(menu);
} }
if (update == 2) return (-2-menu->offset-menu->selected); if (update == 2)
else if (update == 1) return (menu->offset + menu->selected); return (-2-menu->offset-menu->selected);
else return -1; else if (update == 1)
return (menu->offset + menu->selected);
else
return -1;
} }
#if 0 #if 0
@ -917,16 +991,16 @@ void GUI_TextWindow(gui_menu *parent, char *title, char *items[], u8 nb_items, u
/* check for valid buttons */ /* check for valid buttons */
selected = -1; selected = -1;
if ((x>=xwindow)&&(x<=(xwindow+window->width))&&(y>=ypos+i*(20 + h))&&(y<=(ypos+i*(20+h)+h))) if ((x>=xwindow)&&(x<=(xwindow+window->width))&&(y>=ypos+i*(20 + h))&&(y<=(ypos+i*(20+h)+h)))
{ {
selected = i; selected = i;
break; break;
}
} }
} }
else else
{ {
/* reinitialize selection */ /* reinitialize selection */
if (selected == -1) selected = 0; if (selected == -1)
selected = 0;
} }
#endif #endif
@ -936,11 +1010,13 @@ void GUI_TextWindow(gui_menu *parent, char *title, char *items[], u8 nb_items, u
/* update selection */ /* update selection */
if (p&PAD_BUTTON_UP) if (p&PAD_BUTTON_UP)
{ {
if (selected > 0) selected --; if (selected > 0)
selected --;
} }
else if (p&PAD_BUTTON_DOWN) else if (p&PAD_BUTTON_DOWN)
{ {
if (selected < (nb_items -1)) selected ++; if (selected < (nb_items -1))
selected ++;
} }
/* sound fx */ /* sound fx */
@ -996,7 +1072,8 @@ void GUI_TextWindow(gui_menu *parent, char *title, char *items[], u8 nb_items, u
} }
/* restore helper comment */ /* restore helper comment */
if (parent->helpers[1]) parent->helpers[1]->data = Key_A_png; if (parent->helpers[1])
parent->helpers[1]->data = Key_A_png;
/* final position */ /* final position */
GUI_DrawMenu(parent); GUI_DrawMenu(parent);
@ -1047,7 +1124,8 @@ int GUI_OptionWindow(gui_menu *parent, char *title, char *items[], u8 nb_items)
int yoffset = ywindow + window->height; int yoffset = ywindow + window->height;
/* disable helper comment */ /* disable helper comment */
if (parent->helpers[1]) parent->helpers[1]->data = 0; if (parent->helpers[1])
parent->helpers[1]->data = 0;
/* slide in */ /* slide in */
while (yoffset > 0) while (yoffset > 0)
@ -1143,7 +1221,8 @@ int GUI_OptionWindow(gui_menu *parent, char *title, char *items[], u8 nb_items)
else else
{ {
/* reinitialize selection */ /* reinitialize selection */
if (selected == -1) selected = 0; if (selected == -1)
selected = 0;
} }
#endif #endif
@ -1153,11 +1232,13 @@ int GUI_OptionWindow(gui_menu *parent, char *title, char *items[], u8 nb_items)
/* update selection */ /* update selection */
if (p&PAD_BUTTON_UP) if (p&PAD_BUTTON_UP)
{ {
if (selected > 0) selected --; if (selected > 0)
selected --;
} }
else if (p&PAD_BUTTON_DOWN) else if (p&PAD_BUTTON_DOWN)
{ {
if (selected < (nb_items -1)) selected ++; if (selected < (nb_items -1))
selected ++;
} }
/* sound fx */ /* sound fx */
@ -1213,7 +1294,8 @@ int GUI_OptionWindow(gui_menu *parent, char *title, char *items[], u8 nb_items)
} }
/* restore helper comment */ /* restore helper comment */
if (parent->helpers[1]) parent->helpers[1]->data = Key_A_png; if (parent->helpers[1])
parent->helpers[1]->data = Key_A_png;
/* final position */ /* final position */
GUI_DrawMenu(parent); GUI_DrawMenu(parent);
@ -1298,14 +1380,18 @@ void GUI_OptionBox(gui_menu *parent, optioncallback cb, char *title, void *optio
if (type) if (type)
{ {
/* integer type */ /* integer type */
if (*(s16 *)option < 0) sprintf(msg,"-%d",abs(*(s16 *)option)); if (*(s16 *)option < 0)
else sprintf(msg,"%d",abs(*(s16 *)option)); sprintf(msg,"-%d",abs(*(s16 *)option));
else
sprintf(msg,"%d",abs(*(s16 *)option));
} }
else else
{ {
/* float type */ /* float type */
if (*(float *)option < 0.0) sprintf(msg,"-%1.2f",fabs(*(float *)option)); if (*(float *)option < 0.0)
else sprintf(msg,"%1.2f",fabs(*(float *)option)); sprintf(msg,"-%1.2f",fabs(*(float *)option));
else
sprintf(msg,"%1.2f",fabs(*(float *)option));
} }
/* draw option text */ /* draw option text */
@ -1385,13 +1471,15 @@ void GUI_OptionBox(gui_menu *parent, optioncallback cb, char *title, void *optio
{ {
/* integer type */ /* integer type */
*(s16 *)option -= (s16)step; *(s16 *)option -= (s16)step;
if (*(s16 *)option < (s16)min) *(s16 *)option = (s16)max; if (*(s16 *)option < (s16)min)
*(s16 *)option = (s16)max;
} }
else else
{ {
/* float type */ /* float type */
*(float *)option -= step; *(float *)option -= step;
if (*(float *)option < min) *(float *)option = max; if (*(float *)option < min)
*(float *)option = max;
} }
modified = 1; modified = 1;
@ -1403,13 +1491,15 @@ void GUI_OptionBox(gui_menu *parent, optioncallback cb, char *title, void *optio
{ {
/* integer type */ /* integer type */
*(s16 *)option += (s16)step; *(s16 *)option += (s16)step;
if (*(s16 *)option > (s16)max) *(s16 *)option = (s16)min; if (*(s16 *)option > (s16)max)
*(s16 *)option = (s16)min;
} }
else else
{ {
/* float type */ /* float type */
*(float *)option += step; *(float *)option += step;
if (*(float *)option > max) *(float *)option = min; if (*(float *)option > max)
*(float *)option = min;
} }
modified = 1; modified = 1;
@ -1428,7 +1518,8 @@ void GUI_OptionBox(gui_menu *parent, optioncallback cb, char *title, void *optio
((int)config.sfx_volume * 255) / 100,((int)config.sfx_volume * 255) / 100,NULL); ((int)config.sfx_volume * 255) / 100,((int)config.sfx_volume * 255) / 100,NULL);
/* option callback */ /* option callback */
if (cb) cb(); if (cb)
cb();
} }
} }
@ -1529,11 +1620,15 @@ void GUI_OptionBox2(gui_menu *parent, char *text_1, char *text_2, s16 *option_1,
gxDrawTexture(window,xwindow,ywindow,window->width,window->height,225); gxDrawTexture(window,xwindow,ywindow,window->width,window->height,225);
/* draw options text */ /* draw options text */
if (*option_1 < 0) sprintf(msg,"%s: -%02d",text_1,abs(*option_1)); if (*option_1 < 0)
else sprintf(msg,"%s: +%02d",text_1,abs(*option_1)); sprintf(msg,"%s: -%02d",text_1,abs(*option_1));
else
sprintf(msg,"%s: +%02d",text_1,abs(*option_1));
FONT_writeCenter(msg,24,xwindow,xwindow+window->width,240,(GXColor)WHITE); FONT_writeCenter(msg,24,xwindow,xwindow+window->width,240,(GXColor)WHITE);
if (*option_2 < 0) sprintf(msg,"%s: -%02d",text_2,abs(*option_2)); if (*option_2 < 0)
else sprintf(msg,"%s: +%02d",text_2,abs(*option_2)); sprintf(msg,"%s: -%02d",text_2,abs(*option_2));
else
sprintf(msg,"%s: +%02d",text_2,abs(*option_2));
FONT_writeCenter(msg,24,xwindow,xwindow+window->width,264,(GXColor)WHITE); FONT_writeCenter(msg,24,xwindow,xwindow+window->width,264,(GXColor)WHITE);
/* update inputs */ /* update inputs */
@ -1727,9 +1822,7 @@ static void *MsgBox_Thread(void *arg)
/* draw throbber */ /* draw throbber */
if (message_box.throbber) if (message_box.throbber)
{
gxDrawTextureRotate(message_box.throbber,166+(message_box.window->width-message_box.throbber->width)/2,160+message_box.window->height-message_box.throbber->height-20,message_box.throbber->width,message_box.throbber->height,(message_box.progress * 360.0) / 100.0, 255); gxDrawTextureRotate(message_box.throbber,166+(message_box.window->width-message_box.throbber->width)/2,160+message_box.window->height-message_box.throbber->height-20,message_box.throbber->width,message_box.throbber->height,(message_box.progress * 360.0) / 100.0, 255);
}
/* draw exit message */ /* draw exit message */
if (message_box.buttonA) if (message_box.buttonA)
@ -1738,13 +1831,16 @@ static void *MsgBox_Thread(void *arg)
{ {
FONT_write("OK",18,220+message_box.buttonA->width+6,288,640,(GXColor)WHITE); FONT_write("OK",18,220+message_box.buttonA->width+6,288,640,(GXColor)WHITE);
FONT_alignRight("CANCEL",18,166+message_box.window->width-(220-166),288,(GXColor)WHITE); FONT_alignRight("CANCEL",18,166+message_box.window->width-(220-166),288,(GXColor)WHITE);
if (message_box.buttonA) gxDrawTexture(message_box.buttonA, 220, 288-18+(18-message_box.buttonA->height)/2,message_box.buttonA->width, message_box.buttonA->height,255); if (message_box.buttonA)
if (message_box.buttonB) gxDrawTexture(message_box.buttonB, 328, 288-18+(18-message_box.buttonB->height)/2,message_box.buttonB->width, message_box.buttonB->height,255); gxDrawTexture(message_box.buttonA, 220, 288-18+(18-message_box.buttonA->height)/2,message_box.buttonA->width, message_box.buttonA->height,255);
if (message_box.buttonB)
gxDrawTexture(message_box.buttonB, 328, 288-18+(18-message_box.buttonB->height)/2,message_box.buttonB->width, message_box.buttonB->height,255);
} }
else else
{ {
FONT_writeCenter("Press to continue.",18,166,166+message_box.window->width,248+22,(GXColor)WHITE); FONT_writeCenter("Press to continue.",18,166,166+message_box.window->width,248+22,(GXColor)WHITE);
if (message_box.buttonA) gxDrawTexture(message_box.buttonA, 166+116, 248+4+(18-message_box.buttonA->height)/2,message_box.buttonA->width, message_box.buttonA->height,255); if (message_box.buttonA)
gxDrawTexture(message_box.buttonA, 166+116, 248+4+(18-message_box.buttonA->height)/2,message_box.buttonA->width, message_box.buttonA->height,255);
} }
} }
@ -1753,7 +1849,8 @@ static void *MsgBox_Thread(void *arg)
/* update progression */ /* update progression */
message_box.progress++; message_box.progress++;
if (message_box.progress > 100) message_box.progress = 0; if (message_box.progress > 100)
message_box.progress = 0;
} }
else else
{ {
@ -1767,14 +1864,17 @@ static void *MsgBox_Thread(void *arg)
/* update current Message Box */ /* update current Message Box */
void GUI_MsgBoxUpdate(char *title, char *msg) void GUI_MsgBoxUpdate(char *title, char *msg)
{ {
if (title) strncpy(message_box.title,title,64); if (title)
if (msg) strncpy(message_box.msg,msg,64); strncpy(message_box.title,title,64);
if (msg)
strncpy(message_box.msg,msg,64);
} }
/* setup current Message Box */ /* setup current Message Box */
void GUI_MsgBoxOpen(char *title, char *msg, bool throbber) void GUI_MsgBoxOpen(char *title, char *msg, bool throbber)
{ {
if (SILENT) return; if (SILENT)
return;
/* clear unused textures */ /* clear unused textures */
gxTextureClose(&message_box.buttonA); gxTextureClose(&message_box.buttonA);
@ -1790,7 +1890,8 @@ void GUI_MsgBoxOpen(char *title, char *msg, bool throbber)
/* initialize default textures */ /* initialize default textures */
message_box.window = gxTextureOpenPNG(Frame_s4_png,0); message_box.window = gxTextureOpenPNG(Frame_s4_png,0);
message_box.top = gxTextureOpenPNG(Frame_s4_title_png,0); message_box.top = gxTextureOpenPNG(Frame_s4_title_png,0);
if (throbber) message_box.throbber = gxTextureOpenPNG(Frame_throbber_png,0); if (throbber)
message_box.throbber = gxTextureOpenPNG(Frame_throbber_png,0);
/* window position */ /* window position */
int xwindow = 166; int xwindow = 166;
@ -1798,8 +1899,10 @@ void GUI_MsgBoxOpen(char *title, char *msg, bool throbber)
int ypos = 248; int ypos = 248;
/* disable helper comments */ /* disable helper comments */
if (message_box.parent->helpers[0]) message_box.parent->helpers[0]->data = 0; if (message_box.parent->helpers[0])
if (message_box.parent->helpers[1]) message_box.parent->helpers[1]->data = 0; message_box.parent->helpers[0]->data = 0;
if (message_box.parent->helpers[1])
message_box.parent->helpers[1]->data = 0;
/* slide in */ /* slide in */
int yoffset = ywindow + message_box.window->height; int yoffset = ywindow + message_box.window->height;
@ -1813,10 +1916,12 @@ void GUI_MsgBoxOpen(char *title, char *msg, bool throbber)
gxDrawTexture(message_box.top,xwindow,ywindow-yoffset,message_box.top->width,message_box.top->height,255); gxDrawTexture(message_box.top,xwindow,ywindow-yoffset,message_box.top->width,message_box.top->height,255);
/* draw title */ /* draw title */
if (title) FONT_writeCenter(title,20,xwindow,xwindow+message_box.window->width,ywindow+(message_box.top->height-20)/2+20-yoffset,(GXColor)WHITE); if (title)
FONT_writeCenter(title,20,xwindow,xwindow+message_box.window->width,ywindow+(message_box.top->height-20)/2+20-yoffset,(GXColor)WHITE);
/* draw box message */ /* draw box message */
if (msg) FONT_writeCenter(msg,18,xwindow,xwindow+message_box.window->width,ypos-yoffset,(GXColor)WHITE); if (msg)
FONT_writeCenter(msg,18,xwindow,xwindow+message_box.window->width,ypos-yoffset,(GXColor)WHITE);
/* update display */ /* update display */
gxSetScreen(); gxSetScreen();
@ -1825,6 +1930,16 @@ void GUI_MsgBoxOpen(char *title, char *msg, bool throbber)
yoffset -= 60; yoffset -= 60;
} }
/* Final position */
GUI_DrawMenu(message_box.parent);
gxDrawTexture(message_box.window,xwindow,ywindow,message_box.window->width,message_box.window->height,230);
gxDrawTexture(message_box.top,xwindow,ywindow,message_box.top->width,message_box.top->height,255);
if (title)
FONT_writeCenter(title,20,xwindow,xwindow+message_box.window->width,ywindow+(message_box.top->height-20)/2+20,(GXColor)WHITE);
if (msg)
FONT_writeCenter(msg,18,xwindow,xwindow+message_box.window->width,ypos,(GXColor)WHITE);
gxSetScreen();
/* resume LWP thread for MessageBox refresh */ /* resume LWP thread for MessageBox refresh */
message_box.progress = 0; message_box.progress = 0;
message_box.refresh = TRUE; message_box.refresh = TRUE;
@ -1873,8 +1988,10 @@ void GUI_MsgBoxClose(void)
} }
/* restore helper comment */ /* restore helper comment */
if (message_box.parent->helpers[0]) message_box.parent->helpers[0]->data = Key_B_png; if (message_box.parent->helpers[0])
if (message_box.parent->helpers[1]) message_box.parent->helpers[1]->data = Key_A_png; message_box.parent->helpers[0]->data = Key_B_png;
if (message_box.parent->helpers[1])
message_box.parent->helpers[1]->data = Key_A_png;
/* final position */ /* final position */
GUI_DrawMenu(message_box.parent); GUI_DrawMenu(message_box.parent);
@ -1891,7 +2008,8 @@ void GUI_MsgBoxClose(void)
void GUI_WaitPrompt(char *title, char *msg) void GUI_WaitPrompt(char *title, char *msg)
{ {
if (SILENT) return; if (SILENT)
return;
/* update message box */ /* update message box */
GUI_MsgBoxOpen(title, msg, 0); GUI_MsgBoxOpen(title, msg, 0);
@ -1900,8 +2018,10 @@ void GUI_WaitPrompt(char *title, char *msg)
message_box.buttonA = gxTextureOpenPNG(Key_A_png,0); message_box.buttonA = gxTextureOpenPNG(Key_A_png,0);
/* wait for button A */ /* wait for button A */
while (m_input.keys & PAD_BUTTON_A) VIDEO_WaitVSync(); while (m_input.keys & PAD_BUTTON_A)
while (!(m_input.keys & PAD_BUTTON_A)) VIDEO_WaitVSync(); VIDEO_WaitVSync();
while (!(m_input.keys & PAD_BUTTON_A))
VIDEO_WaitVSync();
/* always close message box */ /* always close message box */
GUI_MsgBoxClose(); GUI_MsgBoxClose();
@ -1920,7 +2040,8 @@ int GUI_ConfirmPrompt(char *msg)
/* wait for button A or button B*/ /* wait for button A or button B*/
s16 p = 0; s16 p = 0;
while (m_input.keys) VIDEO_WaitVSync(); while (m_input.keys)
VIDEO_WaitVSync();
while (!(p & (PAD_BUTTON_A | PAD_BUTTON_B))) while (!(p & (PAD_BUTTON_A | PAD_BUTTON_B)))
{ {
VIDEO_WaitVSync(); VIDEO_WaitVSync();
@ -1928,8 +2049,10 @@ int GUI_ConfirmPrompt(char *msg)
} }
/* return user choice */ /* return user choice */
if (p & PAD_BUTTON_A) return 1; if (p & PAD_BUTTON_A)
else return 0; return 1;
else
return 0;
} }
return 1; return 1;

View File

@ -33,7 +33,8 @@
/*****************************************************************************/ /*****************************************************************************/
#define BUTTON_VISIBLE 0x01 #define BUTTON_VISIBLE 0x01
#define BUTTON_ACTIVE 0x02 #define BUTTON_ACTIVE 0x02
#define BUTTON_OVER_SFX 0x04 #define BUTTON_SELECTED 0x04
#define BUTTON_OVER_SFX 0x08
#define BUTTON_SELECT_SFX 0x10 #define BUTTON_SELECT_SFX 0x10
#define BUTTON_FADE 0x20 #define BUTTON_FADE 0x20
#define BUTTON_SLIDE_LEFT 0x40 #define BUTTON_SLIDE_LEFT 0x40
@ -118,6 +119,7 @@ typedef struct
gui_image *bg_images; /* background images */ gui_image *bg_images; /* background images */
gui_item *helpers[2]; /* left & right key comments */ gui_item *helpers[2]; /* left & right key comments */
gui_butn *arrows[2]; /* arrows buttons */ gui_butn *arrows[2]; /* arrows buttons */
void (*cb)(void); /* specific draw callback */
} gui_menu; } gui_menu;
typedef struct typedef struct
@ -146,16 +148,10 @@ struct t_input_menu
/* Optionbox callback */ /* Optionbox callback */
typedef void (*optioncallback)(void); typedef void (*optioncallback)(void);
/* Generic textures*/
/* PNG images */ #ifdef HW_RVL
extern gx_texture *w_pointer;
/* Intro */ #endif
extern const u8 Bg_intro_c1_png[];
extern const u8 Bg_intro_c2_png[];
extern const u8 Bg_intro_c3_png[];
extern const u8 Bg_intro_c4_png[];
extern const u8 Bg_intro_c5_png[];
extern const u8 Bg_credits_png[];
/* Generic backgrounds */ /* Generic backgrounds */
extern const u8 Bg_main_png[]; extern const u8 Bg_main_png[];
@ -164,6 +160,9 @@ extern const u8 Bg_overlay_png[];
extern const u8 Banner_main_png[]; extern const u8 Banner_main_png[];
extern const u8 Banner_bottom_png[]; extern const u8 Banner_bottom_png[];
extern const u8 Banner_top_png[]; extern const u8 Banner_top_png[];
extern const u8 Banner_main_2_png[];
extern const u8 Banner_bottom_2_png[];
extern const u8 Banner_top_2_png[];
extern const u8 Main_logo_png[]; extern const u8 Main_logo_png[];
/* Generic frames */ /* Generic frames */
@ -175,44 +174,6 @@ extern const u8 Frame_s1_title_png[];
extern const u8 Frame_s4_title_png[]; extern const u8 Frame_s4_title_png[];
extern const u8 Frame_throbber_png[]; extern const u8 Frame_throbber_png[];
/* ROM Browser */
extern const u8 Overlay_bar_png[];
extern const u8 Browser_dir_png[];
extern const u8 Star_full_png[];
extern const u8 Star_empty_png[];
extern const u8 Snap_empty_png[];
extern const u8 Snap_frame_png[];
/* Main menu */
extern const u8 Main_load_png[];
extern const u8 Main_options_png[];
extern const u8 Main_quit_png[];
extern const u8 Main_file_png[];
extern const u8 Main_reset_png[];
extern const u8 Main_ggenie_png[];
extern const u8 Main_showinfo_png[];
extern const u8 Main_takeshot_png[];
#ifdef HW_RVL
extern const u8 Main_play_wii_png[];
#else
extern const u8 Main_play_gcn_png[];
#endif
/* Options menu */
extern const u8 Option_menu_png[];
extern const u8 Option_ctrl_png[];
extern const u8 Option_sound_png[];
extern const u8 Option_video_png[];
extern const u8 Option_system_png[];
/* Load ROM menu */
extern const u8 Load_recent_png[];
extern const u8 Load_sd_png[];
extern const u8 Load_dvd_png[];
#ifdef HW_RVL
extern const u8 Load_usb_png[];
#endif
/* Generic Buttons */ /* Generic Buttons */
extern const u8 Button_text_png[]; extern const u8 Button_text_png[];
extern const u8 Button_text_over_png[]; extern const u8 Button_text_over_png[];
@ -227,29 +188,6 @@ extern const u8 Button_down_over_png[];
extern const u8 Button_arrow_png[]; extern const u8 Button_arrow_png[];
extern const u8 Button_arrow_over_png[]; extern const u8 Button_arrow_over_png[];
/* Controller Settings */
extern const u8 Ctrl_4wayplay_png[];
extern const u8 Ctrl_gamepad_png[];
extern const u8 Ctrl_justifiers_png[];
extern const u8 Ctrl_menacer_png[];
extern const u8 Ctrl_mouse_png[];
extern const u8 Ctrl_none_png[];
extern const u8 Ctrl_teamplayer_png[];
extern const u8 Ctrl_pad3b_png[];
extern const u8 Ctrl_pad6b_png[];
extern const u8 Ctrl_config_png[];
extern const u8 Ctrl_player_png[];
extern const u8 Ctrl_player_over_png[];
extern const u8 Ctrl_player_none_png[];
extern const u8 ctrl_option_off_png[];
extern const u8 ctrl_option_on_png[];
extern const u8 ctrl_gamecube_png[];
#ifdef HW_RVL
extern const u8 ctrl_classic_png[];
extern const u8 ctrl_nunchuk_png[];
extern const u8 ctrl_wiimote_png[];
#endif
/* Generic images*/ /* Generic images*/
#ifdef HW_RVL #ifdef HW_RVL
#define Key_A_png Key_A_wii_png #define Key_A_png Key_A_wii_png
@ -262,6 +200,8 @@ extern const u8 Key_B_wii_png[];
#define Key_B_png Key_B_gcn_png #define Key_B_png Key_B_gcn_png
extern const u8 Key_A_gcn_png[]; extern const u8 Key_A_gcn_png[];
extern const u8 Key_B_gcn_png[]; extern const u8 Key_B_gcn_png[];
extern const u8 Star_empty_png[];
extern const u8 Star_full_png[];
#endif #endif
/* Generic Sounds */ /* Generic Sounds */
@ -272,11 +212,6 @@ extern const u32 button_select_pcm_size;
extern const u32 button_over_pcm_size; extern const u32 button_over_pcm_size;
extern const u32 intro_pcm_size; extern const u32 intro_pcm_size;
/* Generic textures*/
#ifdef HW_RVL
extern gx_texture *w_pointer;
#endif
extern u8 SILENT; extern u8 SILENT;
extern void GUI_InitMenu(gui_menu *menu); extern void GUI_InitMenu(gui_menu *menu);

View File

@ -24,6 +24,7 @@
#include "shared.h" #include "shared.h"
#include "font.h" #include "font.h"
#include "menu.h"
#include "gui.h" #include "gui.h"
/* /*

File diff suppressed because it is too large Load Diff

110
source/gx/gui/menu.h Normal file
View File

@ -0,0 +1,110 @@
/****************************************************************************
* menu.c
*
* Genesis Plus GX menus
*
* Eke-Eke (2009)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*
***************************************************************************/
#ifndef _MENU_H
#define _MENU_H
/* PNG images */
/* Intro */
extern const u8 Bg_intro_c1_png[];
extern const u8 Bg_intro_c2_png[];
extern const u8 Bg_intro_c3_png[];
extern const u8 Bg_intro_c4_png[];
extern const u8 Bg_intro_c5_png[];
extern const u8 Bg_credits_png[];
/* ROM Browser */
extern const u8 Overlay_bar_png[];
extern const u8 Browser_dir_png[];
extern const u8 Star_full_png[];
extern const u8 Star_empty_png[];
extern const u8 Snap_empty_png[];
extern const u8 Snap_frame_png[];
/* Main menu */
extern const u8 Main_load_png[];
extern const u8 Main_options_png[];
extern const u8 Main_quit_png[];
extern const u8 Main_file_png[];
extern const u8 Main_reset_png[];
extern const u8 Main_ggenie_png[];
extern const u8 Main_showinfo_png[];
extern const u8 Main_takeshot_png[];
#ifdef HW_RVL
extern const u8 Main_play_wii_png[];
#else
extern const u8 Main_play_gcn_png[];
#endif
/* Options menu */
extern const u8 Option_menu_png[];
extern const u8 Option_ctrl_png[];
extern const u8 Option_sound_png[];
extern const u8 Option_video_png[];
extern const u8 Option_system_png[];
/* Load ROM menu */
extern const u8 Load_recent_png[];
extern const u8 Load_sd_png[];
extern const u8 Load_dvd_png[];
#ifdef HW_RVL
extern const u8 Load_usb_png[];
#endif
/* Save Manager menu */
extern const u8 Button_load_png[];
extern const u8 Button_load_over_png[];
extern const u8 Button_save_png[];
extern const u8 Button_save_over_png[];
extern const u8 Button_special_png[];
extern const u8 Button_special_over_png[];
extern const u8 Button_delete_png[];
extern const u8 Button_delete_over_png[];
/* Controller Settings */
extern const u8 Ctrl_4wayplay_png[];
extern const u8 Ctrl_gamepad_png[];
extern const u8 Ctrl_justifiers_png[];
extern const u8 Ctrl_menacer_png[];
extern const u8 Ctrl_mouse_png[];
extern const u8 Ctrl_none_png[];
extern const u8 Ctrl_teamplayer_png[];
extern const u8 Ctrl_pad3b_png[];
extern const u8 Ctrl_pad6b_png[];
extern const u8 Ctrl_config_png[];
extern const u8 Ctrl_player_png[];
extern const u8 Ctrl_player_over_png[];
extern const u8 Ctrl_player_none_png[];
extern const u8 ctrl_option_off_png[];
extern const u8 ctrl_option_on_png[];
extern const u8 ctrl_gamecube_png[];
#ifdef HW_RVL
extern const u8 ctrl_classic_png[];
extern const u8 ctrl_nunchuk_png[];
extern const u8 ctrl_wiimote_png[];
#endif
extern void MainMenu (void);
#endif

View File

@ -826,7 +826,6 @@ void gxDrawScreenshot(u8 alpha)
void gxCopyScreenshot(gx_texture *texture) void gxCopyScreenshot(gx_texture *texture)
{ {
/* retrieve gamescreen texture */ /* retrieve gamescreen texture */
gxClearScreen((GXColor)BLACK);
GXTexObj texobj; GXTexObj texobj;
GX_InitTexObj(&texobj, texturemem, vwidth, vheight, GX_TF_RGB565, GX_CLAMP, GX_CLAMP, GX_FALSE); GX_InitTexObj(&texobj, texturemem, vwidth, vheight, GX_TF_RGB565, GX_CLAMP, GX_CLAMP, GX_FALSE);
GX_LoadTexObj(&texobj, GX_TEXMAP0); GX_LoadTexObj(&texobj, GX_TEXMAP0);

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.5 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 4.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 2.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 233 B

View File

@ -28,6 +28,7 @@
#include "history.h" #include "history.h"
#include "aram.h" #include "aram.h"
#include "dvd.h" #include "dvd.h"
#include "file_slot.h"
#ifdef HW_RVL #ifdef HW_RVL
#include "usb2storage.h" #include "usb2storage.h"
@ -35,7 +36,6 @@
#endif #endif
#include <fat.h> #include <fat.h>
#include <ogc/cast.h>
#ifdef HW_RVL #ifdef HW_RVL
#include <wiiuse/wpad.h> #include <wiiuse/wpad.h>
@ -196,7 +196,8 @@ void reloadrom (int size, char *name)
void shutdown(void) void shutdown(void)
{ {
/* system shutdown */ /* system shutdown */
memfile_autosave(-1,config.state_auto); if (config.s_auto & 2)
slot_autosave(config.s_default,config.s_device);
system_shutdown(); system_shutdown();
audio_shutdown(); audio_shutdown();
free(cart.rom); free(cart.rom);

View File

@ -24,7 +24,6 @@
#include "gx_audio.h" #include "gx_audio.h"
#include "gx_video.h" #include "gx_video.h"
#include "config.h" #include "config.h"
#include "file_mem.h"
#define DEFAULT_PATH "/genplus" #define DEFAULT_PATH "/genplus"
#define GG_ROM "/genplus/ggenie.bin" #define GG_ROM "/genplus/ggenie.bin"
@ -34,9 +33,9 @@
#define SK_UPMEM "/genplus/sk2chip.bin" #define SK_UPMEM "/genplus/sk2chip.bin"
#ifdef HW_RVL #ifdef HW_RVL
#define VERSION "version 1.4.xW" #define VERSION "version 1.4.0W"
#else #else
#define VERSION "version 1.4.xG" #define VERSION "version 1.4.0G"
#endif #endif
#define osd_input_Update() gx_input_UpdateEmu() #define osd_input_Update() gx_input_UpdateEmu()

View File

@ -60,6 +60,8 @@
uint16 peripherals; uint16 peripherals;
uint16 realchecksum; uint16 realchecksum;
ROMINFO rominfo; ROMINFO rominfo;
char rom_filename[256];
/*************************************************************************** /***************************************************************************
* Genesis ROM Manufacturers * Genesis ROM Manufacturers

View File

@ -64,6 +64,7 @@ extern COMPANYINFO companyinfo[MAXCOMPANY];
extern PERIPHERALINFO peripheralinfo[14]; extern PERIPHERALINFO peripheralinfo[14];
extern uint16 realchecksum; extern uint16 realchecksum;
extern uint16 peripherals; extern uint16 peripherals;
extern char rom_filename[256];
/* Function prototypes */ /* Function prototypes */
extern int load_rom(char *filename); extern int load_rom(char *filename);