mirror of
https://github.com/ekeeke/Genesis-Plus-GX.git
synced 2025-01-17 21:59:11 +01:00
1617 lines
47 KiB
C
1617 lines
47 KiB
C
/*
|
|
* cheats.c
|
|
*
|
|
* Genesis Plus GX Cheats menu
|
|
*
|
|
* Copyright Eke-Eke (2010-2022)
|
|
*
|
|
* Redistribution and use of this code or any derivative works are permitted
|
|
* provided that the following conditions are met:
|
|
*
|
|
* - Redistributions may not be sold, nor may they be used in a commercial
|
|
* product or activity.
|
|
*
|
|
* - Redistributions that are modified from the original source must include the
|
|
* complete source code, including the source code for all components used by a
|
|
* binary built from the modified sources. However, as a special exception, the
|
|
* source code distributed need not include anything that is normally distributed
|
|
* (in either source or binary form) with the major components (compiler, kernel,
|
|
* and so on) of the operating system on which the executable runs, unless that
|
|
* component itself accompanies the executable.
|
|
*
|
|
* - Redistributions must reproduce the above copyright notice, this list of
|
|
* conditions and the following disclaimer in the documentation and/or other
|
|
* materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
|
|
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
* POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
****************************************************************************************/
|
|
|
|
#include "shared.h"
|
|
#include "file_load.h"
|
|
#include "cheats.h"
|
|
#include "font.h"
|
|
#include "gui.h"
|
|
|
|
#define BG_COLOR_1 {0x49,0x49,0x49,0xff}
|
|
#define BG_COLOR_2 {0x66,0x66,0x66,0xff}
|
|
|
|
#define MAX_CHEATS (350)
|
|
#define MAX_DESC_LENGTH (63)
|
|
|
|
#ifdef HW_RVL
|
|
#include "Key_Minus_wii_png.h"
|
|
#include "Key_Plus_wii_png.h"
|
|
#else
|
|
#include "Key_R_gcn_png.h"
|
|
#include "Key_L_gcn_png.h"
|
|
#endif
|
|
#include "Key_DPAD_png.h"
|
|
|
|
typedef struct
|
|
{
|
|
char code[12];
|
|
char text[MAX_DESC_LENGTH];
|
|
u8 enable;
|
|
u16 data;
|
|
u16 old;
|
|
u32 address;
|
|
u8 *prev;
|
|
} CHEATENTRY;
|
|
|
|
static int string_offset = 0;
|
|
static int selection = 0;
|
|
static int offset = 0;
|
|
static int type = 0;
|
|
static int maxcheats = 0;
|
|
static int maxROMcheats = 0;
|
|
static int maxRAMcheats = 0;
|
|
|
|
static CHEATENTRY cheatlist[MAX_CHEATS];
|
|
static u8 cheatIndexes[MAX_CHEATS];
|
|
|
|
static void cheatmenu_cb(void);
|
|
|
|
static gui_image star;
|
|
static gui_image bar_over;
|
|
static gui_image key_switch;
|
|
static gui_image key_enable;
|
|
static gui_image key_delete;
|
|
|
|
/*****************************************************************************/
|
|
/* GUI Buttons data */
|
|
/*****************************************************************************/
|
|
static butn_data arrow_up_data =
|
|
{
|
|
{NULL,NULL},
|
|
{Button_up_png,Button_up_over_png}
|
|
};
|
|
|
|
static butn_data arrow_down_data =
|
|
{
|
|
{NULL,NULL},
|
|
{Button_down_png,Button_down_over_png}
|
|
};
|
|
static butn_data button_digit_data =
|
|
{
|
|
{NULL,NULL},
|
|
{Button_digit_png,Button_digit_over_png}
|
|
};
|
|
|
|
/*****************************************************************************/
|
|
/* GUI Arrows button */
|
|
/*****************************************************************************/
|
|
|
|
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_ACTIVE|BUTTON_OVER_SFX,{0,0,0,0},14,368,360,32};
|
|
|
|
|
|
/*****************************************************************************/
|
|
/* GUI helpers */
|
|
/*****************************************************************************/
|
|
static gui_item action_cancel =
|
|
{
|
|
NULL,Key_B_png,"","Exit",10,422,28,28
|
|
};
|
|
|
|
static gui_item action_select =
|
|
{
|
|
NULL,Key_A_png,"","Edit Entry",602,422,28,28
|
|
};
|
|
|
|
/*****************************************************************************/
|
|
/* GUI Background images */
|
|
/*****************************************************************************/
|
|
static gui_image bg_cheats[7] =
|
|
{
|
|
{NULL,Bg_layer_png,IMAGE_VISIBLE|IMAGE_REPEAT,0,0,640,480,255},
|
|
{NULL,Bg_overlay_png,IMAGE_VISIBLE|IMAGE_REPEAT,0,0,640,480,255},
|
|
{NULL,Banner_top_png,IMAGE_VISIBLE|IMAGE_SLIDE_TOP,0,0,640,108,255},
|
|
{NULL,Banner_bottom_png,IMAGE_VISIBLE|IMAGE_SLIDE_BOTTOM,0,380,640,100,255},
|
|
{NULL,Main_logo_png,IMAGE_VISIBLE|IMAGE_SLIDE_TOP,466,40,152,44,255},
|
|
{NULL,Frame_s1_png,IMAGE_VISIBLE,8,70,372,336,152},
|
|
{NULL,Frame_s1_png,IMAGE_SLIDE_RIGHT,411,109,372,296,76},
|
|
};
|
|
|
|
/*****************************************************************************/
|
|
/* Menu Items description */
|
|
/*****************************************************************************/
|
|
static gui_item items_cheats[30] =
|
|
{
|
|
{NULL,NULL,"","Edit Entry",0,0,0,0},
|
|
{NULL,NULL,"","Edit Entry",0,0,0,0},
|
|
{NULL,NULL,"","Edit Entry",0,0,0,0},
|
|
{NULL,NULL,"","Edit Entry",0,0,0,0},
|
|
{NULL,NULL,"","Edit Entry",0,0,0,0},
|
|
{NULL,NULL,"","Edit Entry",0,0,0,0},
|
|
{NULL,NULL,"","Edit Entry",0,0,0,0},
|
|
{NULL,NULL,"","Edit Entry",0,0,0,0},
|
|
{NULL,NULL,"","Edit Entry",0,0,0,0},
|
|
{NULL,NULL,"","Edit Entry",0,0,0,0},
|
|
{NULL,NULL,"0","Add Character" ,440,136,40,40},
|
|
{NULL,NULL,"1","Add Character" ,486,136,40,40},
|
|
{NULL,NULL,"2","Add Character" ,532,136,40,40},
|
|
{NULL,NULL,"3","Add Character" ,578,136,40,40},
|
|
{NULL,NULL,"4","Add Character" ,440,182,40,40},
|
|
{NULL,NULL,"5","Add Character" ,486,182,40,40},
|
|
{NULL,NULL,"6","Add Character" ,532,182,40,40},
|
|
{NULL,NULL,"7","Add Character" ,578,182,40,40},
|
|
{NULL,NULL,"8","Add Character" ,440,228,40,40},
|
|
{NULL,NULL,"9","Add Character" ,486,228,40,40},
|
|
{NULL,NULL,"A","Add Character" ,532,228,40,40},
|
|
{NULL,NULL,"B","Add Character" ,578,228,40,40},
|
|
{NULL,NULL,"C","Add Character" ,440,274,40,40},
|
|
{NULL,NULL,"D","Add Character" ,486,274,40,40},
|
|
{NULL,NULL,"E","Add Character" ,532,274,40,40},
|
|
{NULL,NULL,"F","Add Character" ,578,274,40,40},
|
|
{NULL,NULL,"del","Backspace" ,440,338,40,40},
|
|
{NULL,NULL,":","Add Separator" ,486,338,40,40},
|
|
{NULL,NULL,"+","Next Characters",532,338,40,40},
|
|
{NULL,NULL,"ok","Save Entry" ,578,338,40,40}
|
|
};
|
|
|
|
/*****************************************************************************/
|
|
/* Menu Buttons description */
|
|
/*****************************************************************************/
|
|
static gui_butn buttons_cheats[30] =
|
|
{
|
|
{NULL, BUTTON_VISIBLE|BUTTON_ACTIVE|BUTTON_SELECT_SFX|BUTTON_OVER_SFX,{1,0,0,0},15,108,358,26},
|
|
{NULL, BUTTON_VISIBLE|BUTTON_SELECT_SFX|BUTTON_OVER_SFX,{1,0,0,0},15,134,358,26},
|
|
{NULL, BUTTON_VISIBLE|BUTTON_SELECT_SFX|BUTTON_OVER_SFX,{1,0,0,0},15,160,358,26},
|
|
{NULL, BUTTON_VISIBLE|BUTTON_SELECT_SFX|BUTTON_OVER_SFX,{1,0,0,0},15,186,358,26},
|
|
{NULL, BUTTON_VISIBLE|BUTTON_SELECT_SFX|BUTTON_OVER_SFX,{1,0,0,0},15,212,358,26},
|
|
{NULL, BUTTON_VISIBLE|BUTTON_SELECT_SFX|BUTTON_OVER_SFX,{1,0,0,0},15,238,358,26},
|
|
{NULL, BUTTON_VISIBLE|BUTTON_SELECT_SFX|BUTTON_OVER_SFX,{1,0,0,0},15,264,358,26},
|
|
{NULL, BUTTON_VISIBLE|BUTTON_SELECT_SFX|BUTTON_OVER_SFX,{1,0,0,0},15,290,358,26},
|
|
{NULL, BUTTON_VISIBLE|BUTTON_SELECT_SFX|BUTTON_OVER_SFX,{1,0,0,0},15,316,358,26},
|
|
{NULL, BUTTON_VISIBLE|BUTTON_SELECT_SFX|BUTTON_OVER_SFX,{1,0,0,0},15,342,358,26},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{0,4,0,1},440,136,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{0,4,1,1},486,136,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{0,4,1,1},532,136,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{0,4,1,0},578,136,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,0,1},440,182,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,1,1},486,182,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,1,1},532,182,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,1,0},578,182,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,0,1},440,228,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,1,1},486,228,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,1,1},532,228,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,1,0},578,228,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,0,1},440,274,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,1,1},486,274,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,1,1},532,274,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,4,1,0},578,274,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,0,0,1},440,338,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,0,1,1},486,338,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,0,1,1},532,338,40,40},
|
|
{&button_digit_data ,BUTTON_ACTIVE|BUTTON_SLIDE_RIGHT|BUTTON_OVER_SFX,{4,0,1,0},578,338,40,40}
|
|
};
|
|
|
|
/*****************************************************************************/
|
|
/* Menu description */
|
|
/*****************************************************************************/
|
|
static gui_menu menu_cheats =
|
|
{
|
|
"Cheats Manager",
|
|
0,0,
|
|
30,30,7,0,
|
|
items_cheats,
|
|
buttons_cheats,
|
|
bg_cheats,
|
|
{&action_cancel, &action_select},
|
|
{&arrow_up,&arrow_down},
|
|
cheatmenu_cb
|
|
};
|
|
|
|
static char ggvalidchars[] = "ABCDEFGHJKLMNPRSTVWXYZ0123456789";
|
|
|
|
static char arvalidchars[] = "0123456789ABCDEF";
|
|
|
|
static u32 decode_cheat(char *string, int index)
|
|
{
|
|
char *p;
|
|
int i,n;
|
|
u32 len = 0;
|
|
u32 address = 0;
|
|
u16 data = 0;
|
|
u8 ref = 0;
|
|
|
|
/* 16-bit Game Genie code (ABCD-EFGH) */
|
|
if ((strlen(string) >= 9) && (string[4] == '-'))
|
|
{
|
|
/* 16-bit system only */
|
|
if ((system_hw & SYSTEM_PBC) != SYSTEM_MD)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
for (i = 0; i < 8; i++)
|
|
{
|
|
if (i == 4) string++;
|
|
p = strchr (ggvalidchars, *string++);
|
|
if (p == NULL) return 0;
|
|
n = p - ggvalidchars;
|
|
|
|
switch (i)
|
|
{
|
|
case 0:
|
|
data |= n << 3;
|
|
break;
|
|
|
|
case 1:
|
|
data |= n >> 2;
|
|
address |= (n & 3) << 14;
|
|
break;
|
|
|
|
case 2:
|
|
address |= n << 9;
|
|
break;
|
|
|
|
case 3:
|
|
address |= (n & 0xF) << 20 | (n >> 4) << 8;
|
|
break;
|
|
|
|
case 4:
|
|
data |= (n & 1) << 12;
|
|
address |= (n >> 1) << 16;
|
|
break;
|
|
|
|
case 5:
|
|
data |= (n & 1) << 15 | (n >> 1) << 8;
|
|
break;
|
|
|
|
case 6:
|
|
data |= (n >> 3) << 13;
|
|
address |= (n & 7) << 5;
|
|
break;
|
|
|
|
case 7:
|
|
address |= n;
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* code length */
|
|
len = 9;
|
|
}
|
|
|
|
/* 8-bit Game Genie code (DDA-AAA-XXX) */
|
|
else if ((strlen(string) >= 11) && (string[3] == '-') && (string[7] == '-'))
|
|
{
|
|
/* 8-bit system only */
|
|
if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
/* decode 8-bit data */
|
|
for (i=0; i<2; i++)
|
|
{
|
|
p = strchr (arvalidchars, *string++);
|
|
if (p == NULL) return 0;
|
|
n = (p - arvalidchars) & 0xF;
|
|
data |= (n << ((1 - i) * 4));
|
|
}
|
|
|
|
/* decode 16-bit address (low 12-bits) */
|
|
for (i=0; i<3; i++)
|
|
{
|
|
if (i==1) string++; /* skip separator */
|
|
p = strchr (arvalidchars, *string++);
|
|
if (p == NULL) return 0;
|
|
n = (p - arvalidchars) & 0xF;
|
|
address |= (n << ((2 - i) * 4));
|
|
}
|
|
|
|
/* decode 16-bit address (high 4-bits) */
|
|
p = strchr (arvalidchars, *string++);
|
|
if (p == NULL) return 0;
|
|
n = (p - arvalidchars) & 0xF;
|
|
n ^= 0xF; /* bits inversion */
|
|
address |= (n << 12);
|
|
|
|
/* RAM address are also supported */
|
|
if (address >= 0xC000)
|
|
{
|
|
/* convert to 24-bit Work RAM address */
|
|
address = 0xFF0000 | (address & 0x1FFF);
|
|
}
|
|
|
|
/* decode reference 8-bit data */
|
|
for (i=0; i<2; i++)
|
|
{
|
|
string++; /* skip separator and 2nd digit */
|
|
p = strchr (arvalidchars, *string++);
|
|
if (p == NULL) return 0;
|
|
n = (p - arvalidchars) & 0xF;
|
|
ref |= (n << ((1 - i) * 4));
|
|
}
|
|
ref = (ref >> 2) | ((ref & 0x03) << 6); /* 2-bit right rotation */
|
|
ref ^= 0xBA; /* XOR */
|
|
|
|
/* update old data value */
|
|
cheatlist[index].old = ref;
|
|
|
|
/* code length */
|
|
len = 11;
|
|
}
|
|
|
|
/* Action Replay code */
|
|
else if (string[6] == ':')
|
|
{
|
|
if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)
|
|
{
|
|
/* 16-bit code (AAAAAA:DDDD) */
|
|
if (strlen(string) < 11) return 0;
|
|
|
|
/* decode 24-bit address */
|
|
for (i=0; i<6; i++)
|
|
{
|
|
p = strchr (arvalidchars, *string++);
|
|
if (p == NULL) return 0;
|
|
n = (p - arvalidchars) & 0xF;
|
|
address |= (n << ((5 - i) * 4));
|
|
}
|
|
|
|
/* decode 16-bit data */
|
|
string++;
|
|
for (i=0; i<4; i++)
|
|
{
|
|
p = strchr (arvalidchars, *string++);
|
|
if (p == NULL) return 0;
|
|
n = (p - arvalidchars) & 0xF;
|
|
data |= (n << ((3 - i) * 4));
|
|
}
|
|
|
|
/* code length */
|
|
len = 11;
|
|
}
|
|
else
|
|
{
|
|
/* 8-bit code (xxAAAA:DD) */
|
|
if (strlen(string) < 9) return 0;
|
|
|
|
/* decode 16-bit address */
|
|
string+=2;
|
|
for (i=0; i<4; i++)
|
|
{
|
|
p = strchr (arvalidchars, *string++);
|
|
if (p == NULL) return 0;
|
|
n = (p - arvalidchars) & 0xF;
|
|
address |= (n << ((3 - i) * 4));
|
|
}
|
|
|
|
/* ROM addresses are not supported */
|
|
if (address < 0xC000) return 0;
|
|
|
|
/* convert to 24-bit Work RAM address */
|
|
address = 0xFF0000 | (address & 0x1FFF);
|
|
|
|
/* decode 8-bit data */
|
|
string++;
|
|
for (i=0; i<2; i++)
|
|
{
|
|
p = strchr (arvalidchars, *string++);
|
|
if (p == NULL) return 0;
|
|
n = (p - arvalidchars) & 0xF;
|
|
data |= (n << ((1 - i) * 4));
|
|
}
|
|
|
|
/* code length */
|
|
len = 9;
|
|
}
|
|
}
|
|
|
|
/* Valid code found ? */
|
|
if (len)
|
|
{
|
|
/* update cheat address & data values */
|
|
cheatlist[index].address = address;
|
|
cheatlist[index].data = data;
|
|
}
|
|
|
|
/* return code length (0 = invalid) */
|
|
return len;
|
|
}
|
|
|
|
static void apply_cheats(void)
|
|
{
|
|
u8 *ptr;
|
|
|
|
/* clear ROM&RAM patches counter */
|
|
maxROMcheats = maxRAMcheats = 0;
|
|
|
|
int i;
|
|
for (i = 0; i < maxcheats; i++)
|
|
{
|
|
if (cheatlist[i].enable)
|
|
{
|
|
/* detect Work RAM patch */
|
|
if (cheatlist[i].address >= 0xFF0000)
|
|
{
|
|
/* add RAM patch */
|
|
cheatIndexes[maxRAMcheats++] = i;
|
|
}
|
|
|
|
/* check if Mega-CD game is running */
|
|
else if ((system_hw == SYSTEM_MCD) && !scd.cartridge.boot)
|
|
{
|
|
/* detect PRG-RAM patch (Sub-CPU side) */
|
|
if (cheatlist[i].address < 0x80000)
|
|
{
|
|
/* add RAM patch */
|
|
cheatIndexes[maxRAMcheats++] = i;
|
|
}
|
|
|
|
/* detect Word-RAM patch (Main-CPU side)*/
|
|
else if ((cheatlist[i].address >= 0x200000) && (cheatlist[i].address < 0x240000))
|
|
{
|
|
/* add RAM patch */
|
|
cheatIndexes[maxRAMcheats++] = i;
|
|
}
|
|
}
|
|
|
|
/* detect cartridge ROM patch */
|
|
else if (cheatlist[i].address < cart.romsize)
|
|
{
|
|
if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)
|
|
{
|
|
/* patch ROM data */
|
|
cheatlist[i].old = *(u16 *)(cart.rom + (cheatlist[i].address & 0xFFFFFE));
|
|
*(u16 *)(cart.rom + (cheatlist[i].address & 0xFFFFFE)) = cheatlist[i].data;
|
|
}
|
|
else
|
|
{
|
|
/* add ROM patch */
|
|
maxROMcheats++;
|
|
cheatIndexes[MAX_CHEATS - maxROMcheats] = i;
|
|
|
|
/* get current banked ROM address */
|
|
ptr = &z80_readmap[(cheatlist[i].address) >> 10][cheatlist[i].address & 0x03FF];
|
|
|
|
/* check if reference matches original ROM data */
|
|
if (((u8)cheatlist[i].old) == *ptr)
|
|
{
|
|
/* patch data */
|
|
*ptr = cheatlist[i].data;
|
|
|
|
/* save patched ROM address */
|
|
cheatlist[i].prev = ptr;
|
|
}
|
|
else
|
|
{
|
|
/* no patched ROM address yet */
|
|
cheatlist[i].prev = NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void clear_cheats(void)
|
|
{
|
|
int i;
|
|
|
|
/* no ROM patches with Mega-CD games */
|
|
if ((system_hw == SYSTEM_MCD) && !scd.cartridge.boot)
|
|
return;
|
|
|
|
/* disable cheats in reversed order in case the same address is used by multiple ROM patches */
|
|
i = maxcheats;
|
|
while (i > 0)
|
|
{
|
|
if (cheatlist[i-1].enable)
|
|
{
|
|
if (cheatlist[i-1].address < cart.romsize)
|
|
{
|
|
if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)
|
|
{
|
|
/* restore original ROM data */
|
|
*(u16 *)(cart.rom + (cheatlist[i-1].address & 0xFFFFFE)) = cheatlist[i-1].old;
|
|
}
|
|
else
|
|
{
|
|
/* check if previous banked ROM address has been patched */
|
|
if (cheatlist[i-1].prev != NULL)
|
|
{
|
|
/* restore original data */
|
|
*cheatlist[i-1].prev = cheatlist[i-1].old;
|
|
|
|
/* no more patched ROM address */
|
|
cheatlist[i-1].prev = NULL;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
i--;
|
|
}
|
|
}
|
|
|
|
static void switch_chars(void)
|
|
{
|
|
int i;
|
|
gui_menu *m = &menu_cheats;
|
|
|
|
if (m->items[10].text[0] == '0')
|
|
{
|
|
m->items[10].text[0] = 'G';
|
|
m->items[11].text[0] = 'H';
|
|
m->items[12].text[0] = 'J';
|
|
m->items[13].text[0] = 'K';
|
|
m->items[14].text[0] = 'L';
|
|
m->items[15].text[0] = 'M';
|
|
m->items[16].text[0] = 'N';
|
|
m->items[17].text[0] = 'P';
|
|
m->items[18].text[0] = 'R';
|
|
m->items[19].text[0] = 'S';
|
|
m->items[20].text[0] = 'T';
|
|
m->items[21].text[0] = 'V';
|
|
m->items[22].text[0] = 'W';
|
|
m->items[23].text[0] = 'X';
|
|
m->items[24].text[0] = 'Y';
|
|
m->items[25].text[0] = 'Z';
|
|
}
|
|
else if (m->items[10].text[0] == 'G')
|
|
{
|
|
m->items[10].text[0] = '0';
|
|
m->items[11].text[0] = '1';
|
|
m->items[12].text[0] = '2';
|
|
m->items[13].text[0] = '3';
|
|
m->items[14].text[0] = '4';
|
|
m->items[15].text[0] = '5';
|
|
m->items[16].text[0] = '6';
|
|
m->items[17].text[0] = '7';
|
|
m->items[18].text[0] = '8';
|
|
m->items[19].text[0] = '9';
|
|
m->items[20].text[0] = 'A';
|
|
m->items[21].text[0] = 'B';
|
|
m->items[22].text[0] = 'C';
|
|
m->items[23].text[0] = 'D';
|
|
m->items[24].text[0] = 'E';
|
|
m->items[25].text[0] = 'F';
|
|
}
|
|
else if (m->items[10].text[0] == 'A')
|
|
{
|
|
m->items[10].text[0] = 'Q';
|
|
m->items[11].text[0] = 'R';
|
|
m->items[12].text[0] = 'S';
|
|
m->items[13].text[0] = 'T';
|
|
m->items[14].text[0] = 'U';
|
|
m->items[15].text[0] = 'V';
|
|
m->items[16].text[0] = 'W';
|
|
m->items[17].text[0] = 'X';
|
|
m->items[18].text[0] = 'Y';
|
|
m->items[19].text[0] = 'Z';
|
|
m->items[20].text[0] = '0';
|
|
m->items[21].text[0] = '1';
|
|
m->items[22].text[0] = '2';
|
|
m->items[23].text[0] = '3';
|
|
m->items[24].text[0] = '4';
|
|
m->items[25].text[0] = '5';
|
|
}
|
|
else if (m->items[10].text[0] == 'Q')
|
|
{
|
|
m->items[10].text[0] = '6';
|
|
m->items[11].text[0] = '7';
|
|
m->items[12].text[0] = '8';
|
|
m->items[13].text[0] = '9';
|
|
|
|
/* hide unused buttons */
|
|
for (i=14; i<26; i++)
|
|
{
|
|
m->buttons[i].state &= ~BUTTON_VISIBLE;
|
|
}
|
|
m->buttons[10].shift[1] = 16;
|
|
m->buttons[11].shift[1] = 16;
|
|
m->buttons[12].shift[1] = 16;
|
|
m->buttons[13].shift[1] = 16;
|
|
m->buttons[26].shift[0] = 16;
|
|
m->buttons[27].shift[0] = 16;
|
|
m->buttons[28].shift[0] = 16;
|
|
m->buttons[29].shift[0] = 16;
|
|
}
|
|
else if (m->items[10].text[0] == '6')
|
|
{
|
|
m->items[10].text[0] = 'A';
|
|
m->items[11].text[0] = 'B';
|
|
m->items[12].text[0] = 'C';
|
|
m->items[13].text[0] = 'D';
|
|
m->items[14].text[0] = 'E';
|
|
m->items[15].text[0] = 'F';
|
|
m->items[16].text[0] = 'G';
|
|
m->items[17].text[0] = 'H';
|
|
m->items[18].text[0] = 'I';
|
|
m->items[19].text[0] = 'J';
|
|
m->items[20].text[0] = 'K';
|
|
m->items[21].text[0] = 'L';
|
|
m->items[22].text[0] = 'M';
|
|
m->items[23].text[0] = 'N';
|
|
m->items[24].text[0] = 'O';
|
|
m->items[25].text[0] = 'P';
|
|
|
|
/* show previously unused buttons */
|
|
for (i=14; i<26; i++)
|
|
{
|
|
m->buttons[i].state |= BUTTON_VISIBLE;
|
|
}
|
|
m->buttons[10].shift[1] = 4;
|
|
m->buttons[11].shift[1] = 4;
|
|
m->buttons[12].shift[1] = 4;
|
|
m->buttons[13].shift[1] = 4;
|
|
m->buttons[26].shift[0] = 4;
|
|
m->buttons[27].shift[0] = 4;
|
|
m->buttons[28].shift[0] = 4;
|
|
m->buttons[29].shift[0] = 4;
|
|
}
|
|
}
|
|
|
|
static void cheatmenu_cb(void)
|
|
{
|
|
int i;
|
|
int yoffset = 108;
|
|
char temp[MAX_DESC_LENGTH];
|
|
|
|
/* Draw browser array */
|
|
gxDrawRectangle(15, 108, 358, 26, 127, (GXColor)BG_COLOR_1);
|
|
gxDrawRectangle(15, 134, 358, 26, 127, (GXColor)BG_COLOR_2);
|
|
gxDrawRectangle(15, 160, 358, 26, 127, (GXColor)BG_COLOR_1);
|
|
gxDrawRectangle(15, 186, 358, 26, 127, (GXColor)BG_COLOR_2);
|
|
gxDrawRectangle(15, 212, 358, 26, 127, (GXColor)BG_COLOR_1);
|
|
gxDrawRectangle(15, 238, 358, 26, 127, (GXColor)BG_COLOR_2);
|
|
gxDrawRectangle(15, 264, 358, 26, 127, (GXColor)BG_COLOR_1);
|
|
gxDrawRectangle(15, 290, 358, 26, 127, (GXColor)BG_COLOR_2);
|
|
gxDrawRectangle(15, 316, 358, 26, 127, (GXColor)BG_COLOR_1);
|
|
gxDrawRectangle(15, 342, 358, 26, 127, (GXColor)BG_COLOR_2);
|
|
|
|
/* Draw Cheat list */
|
|
for (i=0; ((offset + i) < maxcheats) && (i < 10); i++)
|
|
{
|
|
if (i == selection)
|
|
{
|
|
/* selection bar */
|
|
gxDrawTexture(bar_over.texture,16,yoffset+1,356,24,255);
|
|
|
|
/* cheat description need to be specifically handled */
|
|
if (type)
|
|
{
|
|
/* check if text is being edited */
|
|
if (menu_cheats.bg_images[6].state & IMAGE_VISIBLE)
|
|
{
|
|
/* adjust offset so that last characters are visible */
|
|
string_offset += FONT_writeCenter(cheatlist[offset + i].text+string_offset,16,40,366,yoffset+21,(GXColor)WHITE);
|
|
}
|
|
else
|
|
{
|
|
/* scroll text (speed = 1/10 frame) */
|
|
if ((string_offset/10) >= strlen(cheatlist[offset + i].text))
|
|
{
|
|
string_offset = 0;
|
|
}
|
|
|
|
if (string_offset)
|
|
{
|
|
sprintf(temp,"%s ",cheatlist[offset + i].text+string_offset/10);
|
|
strncat(temp, cheatlist[offset + i].text, string_offset/10);
|
|
}
|
|
else
|
|
{
|
|
strcpy(temp, cheatlist[offset + i].text);
|
|
}
|
|
|
|
if (FONT_writeCenter(temp,16,40,366,yoffset+21,(GXColor)WHITE))
|
|
{
|
|
/* scroll text if string does not fit */
|
|
string_offset ++;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
FONT_writeCenter(cheatlist[offset + i].code,18,40,366,yoffset+22,(GXColor)WHITE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (type)
|
|
{
|
|
FONT_writeCenter(cheatlist[offset + i].text,16,40,366,yoffset+21,(GXColor)WHITE);
|
|
}
|
|
else
|
|
{
|
|
FONT_writeCenter(cheatlist[offset + i].code,18,40,366,yoffset+22,(GXColor)WHITE);
|
|
}
|
|
}
|
|
|
|
/* draw cheat enable mark */
|
|
if (cheatlist[offset + i].enable)
|
|
{
|
|
gxDrawTexture(star.texture,20,yoffset+5,16,16,255);
|
|
}
|
|
|
|
yoffset += 26;
|
|
}
|
|
|
|
/* New Entry */
|
|
if (i < 10)
|
|
{
|
|
if (i == selection)
|
|
{
|
|
/* selection bar */
|
|
gxDrawTexture(bar_over.texture,16,yoffset+1,356,24,255);
|
|
|
|
/* check if new code is being edited */
|
|
if (menu_cheats.bg_images[6].state & IMAGE_VISIBLE)
|
|
{
|
|
FONT_writeCenter(cheatlist[offset + selection].code,18,40,366,yoffset+22,(GXColor)WHITE);
|
|
}
|
|
else
|
|
{
|
|
FONT_writeCenter("New Code",18,40,366,yoffset+22,(GXColor)WHITE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
FONT_writeCenter("New Code",18,40,366,yoffset+22,(GXColor)WHITE);
|
|
}
|
|
}
|
|
|
|
/* Extra helpers */
|
|
if (maxcheats && !(menu_cheats.bg_images[6].state & IMAGE_VISIBLE))
|
|
{
|
|
/* switch between cheat code & description preview */
|
|
#ifdef HW_RVL
|
|
gxDrawTexture(key_switch.texture,268,424,24,24,255);
|
|
FONT_write(type ? "View\nCode":"View\nText",16,300,436,640,(GXColor)WHITE);
|
|
#else
|
|
gxDrawTexture(key_switch.texture,272,424,24,24,255);
|
|
FONT_write(type ? "View\nCode":"View\nText",16,304,436,640,(GXColor)WHITE);
|
|
#endif
|
|
|
|
/* delete & enable cheats */
|
|
if ((offset + selection) < maxcheats)
|
|
{
|
|
#ifdef HW_RVL
|
|
gxDrawTexture(key_enable.texture,152,424,24,24,255);
|
|
gxDrawTexture(key_delete.texture,372,424,24,24,255);
|
|
FONT_write(cheatlist[offset + selection].enable ? "Disable\nCheat":"Enable\nCheat",16,184,436,640,(GXColor)WHITE);
|
|
FONT_write("Delete\nCheat",16,404,436,640,(GXColor)WHITE);
|
|
#else
|
|
gxDrawTexture(key_enable.texture,136,426,44,20,255);
|
|
gxDrawTexture(key_delete.texture,368,426,44,20,255);
|
|
FONT_write(cheatlist[offset + selection].enable ? "Disable\nCheat":"Enable\nCheat",16,188,436,640,(GXColor)WHITE);
|
|
FONT_write("Delete\nCheat",16,420,436,640,(GXColor)WHITE);
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* CheatMenu
|
|
*
|
|
* Manage cheats for the currently loaded game
|
|
*
|
|
****************************************************************************/
|
|
void CheatMenu(void)
|
|
{
|
|
int i, update = 0;
|
|
int digit_cnt = 0;
|
|
int max = 0;
|
|
char temp[MAXPATHLEN];
|
|
char *str = NULL;
|
|
gui_menu *m = &menu_cheats;
|
|
|
|
/* clear existing ROM patches */
|
|
clear_cheats();
|
|
|
|
/* reset scrolling */
|
|
string_offset = 0;
|
|
|
|
/* background overlay */
|
|
if (config.bg_overlay)
|
|
{
|
|
bg_cheats[1].state |= IMAGE_VISIBLE;
|
|
}
|
|
else
|
|
{
|
|
bg_cheats[1].state &= ~IMAGE_VISIBLE;
|
|
}
|
|
|
|
/* additional textures */
|
|
star.texture = gxTextureOpenPNG(Star_full_png,0);
|
|
bar_over.texture = gxTextureOpenPNG(Overlay_bar_png,0);
|
|
key_switch.texture = gxTextureOpenPNG(Key_DPAD_png,0);
|
|
#ifdef HW_RVL
|
|
key_enable.texture = gxTextureOpenPNG(Key_Plus_wii_png,0);
|
|
key_delete.texture = gxTextureOpenPNG(Key_Minus_wii_png,0);
|
|
#else
|
|
key_enable.texture = gxTextureOpenPNG(Key_L_gcn_png,0);
|
|
key_delete.texture = gxTextureOpenPNG(Key_R_gcn_png,0);
|
|
#endif
|
|
|
|
/* selected item */
|
|
m->selected = selection;
|
|
|
|
/* slide-in menu */
|
|
GUI_InitMenu(m);
|
|
GUI_DrawMenuFX(m,30,0);
|
|
|
|
/* lock background elements */
|
|
m->bg_images[2].state &= ~IMAGE_SLIDE_TOP;
|
|
m->bg_images[3].state &= ~IMAGE_SLIDE_BOTTOM;
|
|
m->bg_images[4].state &= ~IMAGE_SLIDE_TOP;
|
|
|
|
while (update != -1)
|
|
{
|
|
/* update arrows buttons */
|
|
if (offset > 0)
|
|
m->arrows[0]->state |= BUTTON_VISIBLE;
|
|
else
|
|
m->arrows[0]->state &= ~BUTTON_VISIBLE;
|
|
if (((offset + 10) < (maxcheats + 1)) && ((offset + 10) < MAX_CHEATS))
|
|
m->arrows[1]->state |= BUTTON_VISIBLE;
|
|
else
|
|
m->arrows[1]->state &= ~BUTTON_VISIBLE;
|
|
|
|
/* draw menu */
|
|
GUI_DrawMenu(m);
|
|
|
|
/* check if browsing cheats list */
|
|
if (!(m->bg_images[6].state & IMAGE_VISIBLE))
|
|
{
|
|
/* restore scrolling list settings */
|
|
m->offset = offset;
|
|
m->max_items = (maxcheats < MAX_CHEATS) ? (maxcheats + 1) : MAX_CHEATS;
|
|
m->max_buttons = 10;
|
|
m->helpers[1] = NULL;
|
|
|
|
/* check if arrow button is high-lighted */
|
|
if (m->selected >= 30)
|
|
{
|
|
/* adjust selected button index */
|
|
m->selected -= 20;
|
|
}
|
|
}
|
|
|
|
#ifdef HW_RVL
|
|
if (Shutdown)
|
|
{
|
|
/* close additional textures */
|
|
gxTextureClose(&star.texture);
|
|
gxTextureClose(&bar_over.texture);
|
|
gxTextureClose(&key_switch.texture);
|
|
gxTextureClose(&key_enable.texture);
|
|
gxTextureClose(&key_delete.texture);
|
|
|
|
/* restore default GUI settings */
|
|
m->max_items = m->max_buttons = 30;
|
|
m->helpers[1] = &action_select;
|
|
}
|
|
#endif
|
|
|
|
/* update menu */
|
|
update = GUI_UpdateMenu(m);
|
|
|
|
/* check if browsing cheats list */
|
|
if (!(m->bg_images[6].state & IMAGE_VISIBLE))
|
|
{
|
|
if (m->selected < 10)
|
|
{
|
|
/* update selected cheat */
|
|
if (selection != m->selected)
|
|
{
|
|
selection = m->selected;
|
|
string_offset = 0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* arrow button is selected */
|
|
m->selected += 20;
|
|
}
|
|
|
|
/* save cheats list offset */
|
|
if (offset != m->offset)
|
|
{
|
|
offset = m->offset;
|
|
string_offset = 0;
|
|
}
|
|
|
|
/* restore default GUI settings */
|
|
m->offset = 0;
|
|
m->max_items = m->max_buttons = 30;
|
|
m->helpers[1] = &action_select;
|
|
}
|
|
|
|
|
|
/* handle pressed buttons */
|
|
if (update > 0)
|
|
{
|
|
switch (m->selected)
|
|
{
|
|
case 0:
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
case 4:
|
|
case 5:
|
|
case 6:
|
|
case 7:
|
|
case 8:
|
|
case 9: /* Edit cheat */
|
|
{
|
|
if (type && ((selection + offset) != maxcheats))
|
|
{
|
|
/* cheat description */
|
|
str = cheatlist[offset + selection].text;
|
|
strcpy(temp, str);
|
|
max = MAX_DESC_LENGTH - 2;
|
|
digit_cnt = strlen(str);
|
|
if (digit_cnt <= max)
|
|
{
|
|
str[digit_cnt] = '*';
|
|
str[digit_cnt+1] = 0;
|
|
}
|
|
|
|
/* init specific characters */
|
|
m->items[10].text[0] = '6';
|
|
m->items[27].text[0] = ' ';
|
|
strcpy(m->items[27].comment,"Add White Space");
|
|
switch_chars();
|
|
}
|
|
else
|
|
{
|
|
/* cheat code */
|
|
str = cheatlist[offset + selection].code;
|
|
strcpy(temp, str);
|
|
if ((offset + selection) == maxcheats)
|
|
{
|
|
/* initialize code */
|
|
max = 0;
|
|
digit_cnt = 0;
|
|
str[0] = '*';
|
|
str[1] = 0;
|
|
}
|
|
else
|
|
{
|
|
/* code type */
|
|
if (str[6] == ':')
|
|
{
|
|
/* Action Replay code */
|
|
if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)
|
|
{
|
|
/* 16-bit code */
|
|
max = 10;
|
|
}
|
|
else
|
|
{
|
|
/* 8-bit code */
|
|
max = 8;
|
|
}
|
|
}
|
|
else if (str[4] == '-')
|
|
{
|
|
/* 16-bit Game Genie code */
|
|
max = 8;
|
|
}
|
|
else
|
|
{
|
|
/* 8-bit Game Genie code */
|
|
max = 10;
|
|
}
|
|
|
|
/* set cursor to end of code */
|
|
digit_cnt = max + 1;
|
|
}
|
|
|
|
/* init specific characters */
|
|
m->items[10].text[0] = 'G';
|
|
m->items[27].text[0] = ':';
|
|
strcpy(m->items[27].comment,"Add Code Separator");
|
|
switch_chars();
|
|
}
|
|
|
|
/* show digit buttons */
|
|
for (i=10; i<30; i++)
|
|
m->buttons[i].state |= BUTTON_VISIBLE;
|
|
|
|
/* show right window */
|
|
m->bg_images[6].state |= IMAGE_VISIBLE;
|
|
|
|
/* disable left buttons */
|
|
for (i=0; i<10; i++)
|
|
m->buttons[i].state &= ~BUTTON_ACTIVE;
|
|
|
|
/* disable arrow buttons */
|
|
m->arrows[0]->state &= ~BUTTON_ACTIVE;
|
|
m->arrows[1]->state &= ~BUTTON_ACTIVE;
|
|
|
|
/* slide in right window */
|
|
GUI_DrawMenuFX(m,20,0);
|
|
|
|
/* update helper */
|
|
strcpy(action_cancel.comment,"Cancel");
|
|
|
|
/* select first digit */
|
|
m->selected = 10;
|
|
|
|
/* reset scrolling */
|
|
string_offset = 0;
|
|
break;
|
|
}
|
|
|
|
case 26: /* Backspace */
|
|
{
|
|
if (digit_cnt > 0)
|
|
{
|
|
/* delete last character */
|
|
str[digit_cnt--] = 0;
|
|
|
|
/* code separator is being deleted */
|
|
if ((str[digit_cnt] == ':') || (str[digit_cnt] == '-'))
|
|
{
|
|
/* reset detected code type (except 8-bit Game Genie code using 2 separators) */
|
|
if (((system_hw & SYSTEM_PBC) == SYSTEM_MD) || (digit_cnt != 7))
|
|
{
|
|
max = 0;
|
|
}
|
|
}
|
|
|
|
/* edit mark */
|
|
str[digit_cnt] = '*';
|
|
|
|
/* update scroll value if necessary */
|
|
if (string_offset > 0)
|
|
string_offset--;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 27:
|
|
{
|
|
if (type && ((offset + selection) != maxcheats))
|
|
{
|
|
/* SPACE character */
|
|
if (digit_cnt <= max)
|
|
{
|
|
str[digit_cnt++] = ' ';
|
|
str[digit_cnt] = 0;
|
|
if (digit_cnt <= max)
|
|
{
|
|
str[digit_cnt] = '*';
|
|
str[digit_cnt+1] = 0;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* Separator character */
|
|
if ((system_hw & SYSTEM_PBC) == SYSTEM_MD)
|
|
{
|
|
/* 16-bit codes */
|
|
if (digit_cnt == 4)
|
|
{
|
|
/* Game Genie code */
|
|
max = 8;
|
|
str[4] = '-';
|
|
}
|
|
else if ((digit_cnt == 6) && (max != 8))
|
|
{
|
|
/* Action Replay code */
|
|
max = 10;
|
|
str[6] = ':';
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* 8-bit codes */
|
|
if (digit_cnt == 3)
|
|
{
|
|
/* Game Genie code */
|
|
max = 10;
|
|
str[3] = '-';
|
|
}
|
|
else if ((digit_cnt == 7) && (max == 10))
|
|
{
|
|
/* Game Genie code (last part) */
|
|
str[7] = '-';
|
|
}
|
|
else if ((digit_cnt == 6) && (max != 10))
|
|
{
|
|
/* Action Replay code */
|
|
max = 8;
|
|
str[6] = ':';
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
digit_cnt++;
|
|
str[digit_cnt] = '*';
|
|
str[digit_cnt+1] = 0;
|
|
}
|
|
break;
|
|
}
|
|
|
|
case 28: /* Next character set */
|
|
{
|
|
GUI_DrawMenuFX(m,40,1);
|
|
switch_chars();
|
|
GUI_DrawMenuFX(m,40,0);
|
|
break;
|
|
}
|
|
|
|
case 29: /* Validate entry */
|
|
{
|
|
/* finalize cheat description */
|
|
if (type && ((offset + selection) != maxcheats))
|
|
{
|
|
str[digit_cnt] = 0;
|
|
update = -1;
|
|
}
|
|
|
|
/* finalize cheat code edition */
|
|
else if (max && (digit_cnt > max))
|
|
{
|
|
/* check if cheat code is valid */
|
|
if (decode_cheat(cheatlist[offset + selection].code, offset + selection))
|
|
{
|
|
/* new cheat ? */
|
|
if ((offset + selection) == maxcheats)
|
|
{
|
|
/* increase cheat count */
|
|
maxcheats++;
|
|
|
|
/* enable cheat by default */
|
|
cheatlist[offset + selection].enable = 1;
|
|
|
|
/* no description by default */
|
|
strcpy(cheatlist[offset + selection].text,"No Description");
|
|
}
|
|
|
|
/* return to cheat selection */
|
|
update = -1;
|
|
}
|
|
else
|
|
{
|
|
GUI_WaitPrompt("Error", "Invalid Cheat Code");
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
|
|
default: /* Add Character */
|
|
{
|
|
/* force code separator if none has been set yet */
|
|
if ((max == 0) && (digit_cnt == 6))
|
|
break;
|
|
|
|
/* force 8-bit Game Genie code last separator */
|
|
if (((system_hw & SYSTEM_PBC) != SYSTEM_MD) && (max == 10) && (digit_cnt == 7))
|
|
break;
|
|
|
|
/* add character */
|
|
if ((digit_cnt <= max) || (max == 0))
|
|
{
|
|
str[digit_cnt++] = m->items[m->selected].text[0];
|
|
str[digit_cnt] = 0;
|
|
if ((digit_cnt <= max) || (max == 0))
|
|
{
|
|
str[digit_cnt] = '*';
|
|
str[digit_cnt+1] = 0;
|
|
}
|
|
|
|
if (string_offset > 0)
|
|
string_offset ++;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else if (update < 0)
|
|
{
|
|
/* cancel */
|
|
if (m->bg_images[6].state & IMAGE_VISIBLE)
|
|
{
|
|
/* Restore old entry */
|
|
strcpy(str, temp);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/* check other buttons pressed while browsing cheats list */
|
|
if (maxcheats && !(m->bg_images[6].state & IMAGE_VISIBLE))
|
|
{
|
|
if ((m_input.keys & PAD_BUTTON_LEFT) || (m_input.keys & PAD_BUTTON_RIGHT))
|
|
{
|
|
/* Switch between cheat code & description */
|
|
type ^= 1;
|
|
|
|
/* reset scrolling */
|
|
string_offset = 0;
|
|
}
|
|
else if ((offset + selection) < maxcheats)
|
|
{
|
|
/* Delete selected cheat code*/
|
|
if (m_input.keys & PAD_TRIGGER_R)
|
|
{
|
|
if (GUI_WaitConfirm("Warning","Delete Cheat Entry ?"))
|
|
{
|
|
/* shift cheat list up to selected entry */
|
|
for (i = offset + selection + 1; i < maxcheats; i++)
|
|
{
|
|
strcpy(cheatlist[i-1].text,cheatlist[i].text);
|
|
strcpy(cheatlist[i-1].code,cheatlist[i].code);
|
|
cheatlist[i-1].address = cheatlist[i].address;
|
|
cheatlist[i-1].data = cheatlist[i].data;
|
|
cheatlist[i-1].enable = cheatlist[i].enable;
|
|
}
|
|
|
|
/* clear last cheat */
|
|
cheatlist[maxcheats-1].text[0] = 0;
|
|
cheatlist[maxcheats-1].code[0] = 0;
|
|
cheatlist[maxcheats-1].address = 0;
|
|
cheatlist[maxcheats-1].data = 0;
|
|
cheatlist[maxcheats-1].enable = 0;
|
|
|
|
/* adjust scrolling list */
|
|
if (maxcheats < 10)
|
|
{
|
|
/* disable next button */
|
|
m->buttons[maxcheats].state &= ~BUTTON_ACTIVE;
|
|
m->buttons[maxcheats-1].shift[1] = 0;
|
|
}
|
|
else
|
|
{
|
|
/* scroll down cheat list if there is less than 10 visible entries */
|
|
if ((maxcheats < (offset + 10)) && (maxcheats < MAX_CHEATS))
|
|
{
|
|
offset--;
|
|
}
|
|
}
|
|
|
|
/* decrease cheat count */
|
|
maxcheats--;
|
|
|
|
/* reset scrolling */
|
|
string_offset = 0;
|
|
}
|
|
}
|
|
else if (m_input.keys & PAD_TRIGGER_L)
|
|
{
|
|
/* Enable/Disable selected cheat code */
|
|
cheatlist[offset + selection].enable ^= 1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (update < 0)
|
|
{
|
|
if (m->bg_images[6].state & IMAGE_VISIBLE)
|
|
{
|
|
/* slide out right window */
|
|
GUI_DrawMenuFX(m,20,1);
|
|
|
|
/* hide digit buttons */
|
|
for (i=10; i<30; i++)
|
|
m->buttons[i].state &= ~BUTTON_VISIBLE;
|
|
|
|
/* hide right window */
|
|
m->bg_images[6].state &= ~IMAGE_VISIBLE;
|
|
|
|
/* enable left buttons */
|
|
for (i=0; i<10; i++)
|
|
{
|
|
if (i < maxcheats)
|
|
{
|
|
menu_cheats.buttons[i].state |= BUTTON_ACTIVE;
|
|
menu_cheats.buttons[i].shift[1] = 1;
|
|
}
|
|
else if (i == maxcheats)
|
|
{
|
|
menu_cheats.buttons[i].state |= BUTTON_ACTIVE;
|
|
menu_cheats.buttons[i].shift[1] = 0;
|
|
}
|
|
else
|
|
{
|
|
menu_cheats.buttons[i].state &= ~BUTTON_ACTIVE;
|
|
menu_cheats.buttons[i].shift[1] = 0;
|
|
}
|
|
}
|
|
|
|
/* enable arrow buttons */
|
|
m->arrows[0]->state |= BUTTON_ACTIVE;
|
|
m->arrows[1]->state |= BUTTON_ACTIVE;
|
|
|
|
/* restore helper */
|
|
strcpy(action_cancel.comment,"Back");
|
|
|
|
/* select current cheat */
|
|
m->selected = selection;
|
|
|
|
/* stay in menu */
|
|
update = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* apply ROM patches */
|
|
apply_cheats();
|
|
|
|
/* save cheats to file */
|
|
snprintf(temp, MAXPATHLEN, "%s/cheats/%s.pat", DEFAULT_PATH, rom_filename);
|
|
|
|
if (maxcheats)
|
|
{
|
|
/* open file */
|
|
FILE *f = fopen(temp, "w");
|
|
|
|
/* write cheats */
|
|
if (f)
|
|
{
|
|
for (i=0; i<maxcheats; i++)
|
|
{
|
|
fprintf(f, "%s\t%s\n", cheatlist[i].code, cheatlist[i].text);
|
|
fprintf(f, "%s\n", cheatlist[i].enable ? "ON" : "OFF");
|
|
}
|
|
fclose(f);
|
|
}
|
|
}
|
|
|
|
/* unlock background elements */
|
|
m->bg_images[2].state |= IMAGE_SLIDE_TOP;
|
|
m->bg_images[3].state |= IMAGE_SLIDE_BOTTOM;
|
|
m->bg_images[4].state |= IMAGE_SLIDE_TOP;
|
|
|
|
/* leave menu */
|
|
GUI_DrawMenuFX(m,30,1);
|
|
GUI_DeleteMenu(m);
|
|
|
|
/* close additional textures */
|
|
gxTextureClose(&star.texture);
|
|
gxTextureClose(&bar_over.texture);
|
|
gxTextureClose(&key_switch.texture);
|
|
gxTextureClose(&key_enable.texture);
|
|
gxTextureClose(&key_delete.texture);
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* CheatLoad
|
|
*
|
|
* Load cheats from associated .pat file, called when loading a new game
|
|
* ROM patches are automatically applied.
|
|
* RAM patches are applied once per frame.
|
|
*
|
|
****************************************************************************/
|
|
void CheatLoad(void)
|
|
{
|
|
int len;
|
|
int cnt = 0;
|
|
char temp[MAXPATHLEN];
|
|
|
|
/* reset cheat count */
|
|
maxcheats = 0;
|
|
|
|
/* make cheat filename */
|
|
snprintf(temp, MAXPATHLEN, "%s/cheats/%s.pat", DEFAULT_PATH, rom_filename);
|
|
|
|
/* open file */
|
|
FILE *f = fopen(temp, "r");
|
|
if (f)
|
|
{
|
|
/* clear string */
|
|
memset(temp, 0, 256);
|
|
|
|
/* read cheats from file (one line per cheat) */
|
|
while (fgets(temp, 256, f) && (maxcheats < MAX_CHEATS) && (cnt < MAX_CHEATS))
|
|
{
|
|
/* remove CR & EOL chars */
|
|
if ((temp[strlen(temp) - 2] == 0x0d) || (temp[strlen(temp) - 2] == 0x0a))
|
|
temp[strlen(temp) - 2] = 0;
|
|
else
|
|
temp[strlen(temp) - 1] = 0;
|
|
|
|
/* check cheat validty */
|
|
len = decode_cheat(temp, maxcheats);
|
|
|
|
if (len)
|
|
{
|
|
/* copy cheat code */
|
|
strncpy(cheatlist[maxcheats].code, temp, len);
|
|
cheatlist[maxcheats].code[len] = 0;
|
|
len++;
|
|
|
|
/* jump TAB and SPACE characters */
|
|
while ((temp[len] == 0x20) || (temp[len] == 0x09))
|
|
len++;
|
|
|
|
/* copy cheat description */
|
|
strncpy(cheatlist[maxcheats].text, &temp[len], MAX_DESC_LENGTH - 1);
|
|
cheatlist[maxcheats].text[MAX_DESC_LENGTH - 1] = 0;
|
|
|
|
/* increment cheat count */
|
|
maxcheats++;
|
|
}
|
|
else if (!strcmp(temp,"ON") && config.autocheat)
|
|
{
|
|
/* enable flag */
|
|
cheatlist[cnt++].enable = 1;
|
|
}
|
|
else if (!strcmp(temp,"OFF") && config.autocheat)
|
|
{
|
|
/* disable flag */
|
|
cheatlist[cnt++].enable = 0;
|
|
}
|
|
}
|
|
|
|
/* by default, disable cheats that were not flagged */
|
|
while (cnt < maxcheats)
|
|
cheatlist[cnt++].enable = 0;
|
|
|
|
/* close file */
|
|
fclose(f);
|
|
}
|
|
|
|
/* apply ROM patches */
|
|
apply_cheats();
|
|
|
|
/* adjust menu buttons */
|
|
for (cnt=0; cnt<10; cnt++)
|
|
{
|
|
if (cnt < maxcheats)
|
|
{
|
|
menu_cheats.buttons[cnt].state |= BUTTON_ACTIVE;
|
|
menu_cheats.buttons[cnt].shift[1] = 1;
|
|
}
|
|
else if (cnt == maxcheats)
|
|
{
|
|
menu_cheats.buttons[cnt].state |= BUTTON_ACTIVE;
|
|
menu_cheats.buttons[cnt].shift[1] = 0;
|
|
}
|
|
else
|
|
{
|
|
menu_cheats.buttons[cnt].shift[1] = 0;
|
|
menu_cheats.buttons[cnt].state &= ~BUTTON_ACTIVE;
|
|
}
|
|
}
|
|
|
|
/* reset menu */
|
|
selection = offset = 0;
|
|
}
|
|
|
|
/****************************************************************************
|
|
* RAMCheatUpdate
|
|
*
|
|
* Apply RAM patches (this should be called once per frame)
|
|
*
|
|
****************************************************************************/
|
|
void RAMCheatUpdate(void)
|
|
{
|
|
uint8 *base;
|
|
uint32 mask;
|
|
int index, cnt = maxRAMcheats;
|
|
|
|
while (cnt)
|
|
{
|
|
/* get cheat index */
|
|
index = cheatIndexes[--cnt];
|
|
|
|
/* detect destination RAM */
|
|
switch ((cheatlist[index].address >> 20) & 0xf)
|
|
{
|
|
case 0x0: /* Mega-CD PRG-RAM (512 KB) */
|
|
base = scd.prg_ram;
|
|
mask = 0x7fffe;
|
|
break;
|
|
|
|
case 0x2: /* Mega-CD 2M Word-RAM (256 KB) */
|
|
base = scd.word_ram_2M;
|
|
mask = 0x3fffe;
|
|
break;
|
|
|
|
default: /* Work-RAM (64 KB) */
|
|
base = work_ram;
|
|
mask = 0xfffe;
|
|
break;
|
|
}
|
|
|
|
/* apply RAM patch */
|
|
if (cheatlist[index].data & 0xFF00)
|
|
{
|
|
/* word patch */
|
|
*(u16 *)(base + (cheatlist[index].address & mask)) = cheatlist[index].data;
|
|
}
|
|
else
|
|
{
|
|
/* byte patch */
|
|
mask |= 1;
|
|
base[cheatlist[index].address & mask] = cheatlist[index].data;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/****************************************************************************
|
|
* ROMCheatUpdate
|
|
*
|
|
* Apply ROM patches (this should be called each time banking is changed)
|
|
*
|
|
****************************************************************************/
|
|
void ROMCheatUpdate(void)
|
|
{
|
|
int index, cnt = maxROMcheats;
|
|
u8 *ptr;
|
|
|
|
while (cnt)
|
|
{
|
|
/* get cheat index */
|
|
index = cheatIndexes[MAX_CHEATS - cnt];
|
|
|
|
/* check if previous banked ROM address was patched */
|
|
if (cheatlist[index].prev != NULL)
|
|
{
|
|
/* restore original data */
|
|
*cheatlist[index].prev = cheatlist[index].old;
|
|
|
|
/* no more patched ROM address */
|
|
cheatlist[index].prev = NULL;
|
|
}
|
|
|
|
/* get current banked ROM address */
|
|
ptr = &z80_readmap[(cheatlist[index].address) >> 10][cheatlist[index].address & 0x03FF];
|
|
|
|
/* check if reference matches original ROM data */
|
|
if (((u8)cheatlist[index].old) == *ptr)
|
|
{
|
|
/* patch data */
|
|
*ptr = cheatlist[index].data;
|
|
|
|
/* save patched ROM address */
|
|
cheatlist[index].prev = ptr;
|
|
}
|
|
|
|
/* next ROM patch */
|
|
cnt--;
|
|
}
|
|
}
|