2023-04-06 16:04:50 +02:00
//***********************************************************
// SEGA MASTER SYSTEM / MARK III / GAME GEAR / SG-1000 MODULE
//***********************************************************
2024-03-02 11:26:35 -05:00
# ifdef ENABLE_SMS
2022-06-16 17:57:00 +02:00
2023-04-06 16:04:50 +02:00
//******************************************
// Menus
//******************************************
// Adapters menu
static const char SMSAdapterItem1 [ ] PROGMEM = " SMS/MarkIII raphnet " ;
static const char SMSAdapterItem2 [ ] PROGMEM = " SMS Retrode " ;
static const char SMSAdapterItem3 [ ] PROGMEM = " SMS Retron3in1 " ;
static const char SMSAdapterItem4 [ ] PROGMEM = " GameGear Retrode " ;
static const char SMSAdapterItem5 [ ] PROGMEM = " GameGear Retron3in1 " ;
static const char SMSAdapterItem6 [ ] PROGMEM = " SG-1000 raphnet " ;
static const char * const SMSAdapterMenu [ ] PROGMEM = { SMSAdapterItem1 , SMSAdapterItem2 , SMSAdapterItem3 , SMSAdapterItem4 , SMSAdapterItem5 , SMSAdapterItem6 } ;
// Operations menu
2024-07-29 13:08:59 -07:00
static const char SMSOperationMenuItem4 [ ] PROGMEM = " Set ROM Size " ;
static const char * const SMSOperationMenu [ ] PROGMEM = { FSTRING_READ_ROM , FSTRING_READ_SAVE , FSTRING_WRITE_SAVE , SMSOperationMenuItem4 , FSTRING_RESET } ;
// SG Operations menu
static const char * const SGOperationsMenu [ ] PROGMEM = { FSTRING_READ_ROM , FSTRING_RESET } ;
2023-04-06 16:04:50 +02:00
// Rom sizes menu
static const char SMSRomSizeItem1 [ ] PROGMEM = " 8 KB " ;
static const char SMSRomSizeItem2 [ ] PROGMEM = " 16 KB " ;
static const char SMSRomSizeItem3 [ ] PROGMEM = " 24 KB " ;
static const char SMSRomSizeItem4 [ ] PROGMEM = " 32 KB " ;
2023-06-26 12:04:00 +02:00
static const char SMSRomSizeItem5 [ ] PROGMEM = " 40 KB " ; //SG-1000 40k mapping not yet supported
static const char SMSRomSizeItem6 [ ] PROGMEM = " 48 KB " ; //SG-1000 40k mapping not yet supported
2023-04-06 16:04:50 +02:00
static const char SMSRomSizeItem7 [ ] PROGMEM = " 64 KB " ;
static const char SMSRomSizeItem8 [ ] PROGMEM = " 128 KB " ;
static const char SMSRomSizeItem9 [ ] PROGMEM = " 256 KB " ;
static const char SMSRomSizeItem10 [ ] PROGMEM = " 512 KB " ;
static const char SMSRomSizeItem11 [ ] PROGMEM = " 1024 KB " ;
2023-06-26 12:04:00 +02:00
static const char * const SG1RomSizeMenu [ ] PROGMEM = { SMSRomSizeItem1 , SMSRomSizeItem2 , SMSRomSizeItem3 , SMSRomSizeItem4 } ; // Rom sizes for SG-1000
static const char * const SMSRomSizeMenu [ ] PROGMEM = { SMSRomSizeItem4 , SMSRomSizeItem7 , SMSRomSizeItem8 , SMSRomSizeItem9 , SMSRomSizeItem10 , SMSRomSizeItem11 } ; // Rom sizes for SMS and GG
2023-04-06 16:04:50 +02:00
// Init systems
2023-06-26 12:04:00 +02:00
static bool system_sms = false ; // SMS or MarkIII
static bool system_gg = false ; // GameGear
static bool system_sg1000 = false ; // SG-1000
2023-04-06 16:04:50 +02:00
// Init adapters
2023-06-26 12:04:00 +02:00
static bool adapter_raphnet = false ; // raphet adapater (SMS-to-MD or MIII-to-MD)
static bool adapter_retrode = false ; // Retrode adapter (SMS-to-MD or GG-to-MD)
static bool adapter_retron = false ; // Retron 3in1 adapter (SMS-to-MD or GG-to-MD)
2023-04-06 16:04:50 +02:00
2024-07-29 13:08:59 -07:00
// Manual ROM Size Selection Flag
bool manRomSizeSelected = false ;
2023-04-06 16:04:50 +02:00
//*********************************************************
// Main menu with systems/adapters setups to choose from
//*********************************************************
void smsMenu ( ) {
unsigned char SMSSetup ;
convertPgm ( SMSAdapterMenu , 6 ) ;
SMSSetup = question_box ( F ( " Select your setup " ) , menuOptions , 6 , 0 ) ;
2022-06-16 17:57:00 +02:00
2023-04-06 16:04:50 +02:00
switch ( SMSSetup ) {
2022-06-16 17:57:00 +02:00
case 0 :
2023-04-06 16:04:50 +02:00
// SMS or MarkIII with raphnet adapter
system_sms = true ;
adapter_raphnet = true ;
2022-06-16 17:57:00 +02:00
break ;
2023-06-26 12:04:00 +02:00
case 1 :
2023-04-06 16:04:50 +02:00
// SMS with Retrode adapter
system_sms = true ;
adapter_retrode = true ;
2022-06-16 17:57:00 +02:00
break ;
case 2 :
2023-04-06 16:04:50 +02:00
// SMS with Retron 3in1 adapter
system_sms = true ;
adapter_retron = true ;
2022-06-16 17:57:00 +02:00
break ;
case 3 :
2023-04-06 16:04:50 +02:00
// GameGear with Retrode adapter
system_gg = true ;
adapter_retrode = true ;
2022-06-16 17:57:00 +02:00
break ;
case 4 :
2023-04-06 16:04:50 +02:00
// GameGear with Retron 3in1 adapter
system_gg = true ;
adapter_retron = true ;
break ;
case 5 :
// SG-1000 with raphnet adapter
system_sg1000 = true ;
adapter_raphnet = true ;
2022-06-16 17:57:00 +02:00
break ;
}
2024-07-29 13:08:59 -07:00
for ( ; ; ) operationsMenu ( ) ;
2022-06-16 17:57:00 +02:00
}
2023-04-06 16:04:50 +02:00
//****************************************************
// Create custom menu depending on selected setup
//****************************************************
2024-07-29 13:08:59 -07:00
void operationsMenu ( ) {
2023-04-06 16:04:50 +02:00
unsigned char SMSOperation = ' 3 ' ;
2024-07-29 13:08:59 -07:00
if ( system_sg1000 ) {
convertPgm ( SGOperationsMenu , 2 ) ;
SMSOperation = question_box ( FS ( SMSAdapterItem6 ) , menuOptions , 2 , 0 ) ;
switch ( SMSOperation ) {
case 0 : // Read ROM
mode = CORE_SMS ;
setup_SMS ( ) ;
readROM_SMS ( ) ;
break ;
default :
case 1 :
// Reset
resetArduino ( ) ;
break ;
2023-04-06 16:04:50 +02:00
}
2024-07-29 13:08:59 -07:00
} else {
convertPgm ( SMSOperationMenu , 5 ) ;
if ( system_sms ) {
if ( adapter_raphnet ) {
SMSOperation = question_box ( FS ( SMSAdapterItem1 ) , menuOptions , 5 , 0 ) ;
} else if ( adapter_retrode ) {
SMSOperation = question_box ( FS ( SMSAdapterItem2 ) , menuOptions , 5 , 0 ) ;
} else if ( adapter_retron ) {
SMSOperation = question_box ( FS ( SMSAdapterItem3 ) , menuOptions , 5 , 0 ) ;
}
} else if ( system_gg ) {
if ( adapter_retrode ) {
SMSOperation = question_box ( FS ( SMSAdapterItem4 ) , menuOptions , 5 , 0 ) ;
} else if ( adapter_retron ) {
SMSOperation = question_box ( FS ( SMSAdapterItem5 ) , menuOptions , 5 , 0 ) ;
}
2023-04-06 16:04:50 +02:00
}
2022-06-16 17:57:00 +02:00
2024-07-29 13:08:59 -07:00
switch ( SMSOperation ) {
case 0 : // Read ROM
mode = CORE_SMS ;
setup_SMS ( ) ;
readROM_SMS ( ) ;
break ;
case 1 : // Read SRAM
mode = CORE_SMS ;
setup_SMS ( ) ;
readSRAM_SMS ( ) ;
break ;
case 2 : // Write SRAM
mode = CORE_SMS ;
setup_SMS ( ) ;
writeSRAM_SMS ( ) ;
break ;
case 3 : // Select ROM Size
manual_selectRomSize ( ) ;
break ;
default :
case 4 :
// Reset
resetArduino ( ) ;
break ;
}
2022-06-16 17:57:00 +02:00
}
2023-04-06 16:04:50 +02:00
display_Update ( ) ;
wait ( ) ;
2022-06-16 17:57:00 +02:00
}
2023-04-06 16:04:50 +02:00
//********************************
// Setup I/O
//********************************
2022-06-16 17:57:00 +02:00
void setup_SMS ( ) {
2023-06-26 15:25:54 -04:00
// Request 5V
2023-06-26 12:04:00 +02:00
setVoltage ( VOLTS_SET_5V ) ;
2022-06-16 17:57:00 +02:00
// Set Address Pins to Output
//A0-A7
DDRF = 0xFF ;
//A8-A14
DDRK = 0xFF ;
//A15
DDRH | = ( 1 < < 3 ) ;
2023-04-06 16:04:50 +02:00
//For Retrode adapter
if ( adapter_retrode ) {
2022-06-16 17:57:00 +02:00
PORTH & = ~ ( ( 1 < < 0 ) | ( 1 < < 3 ) | ( 1 < < 5 ) ) ;
PORTL & = ~ ( 1 < < 1 ) ;
DDRH & = ~ ( ( 1 < < 0 ) | ( 1 < < 5 ) ) ;
DDRL & = ~ ( ( 1 < < 1 ) ) ;
// Set Control Pins to Output OE(PH6)
DDRH | = ( 1 < < 6 ) ;
// WR(PL5) and RD(PL6)
DDRL | = ( 1 < < 5 ) | ( 1 < < 6 ) ;
// Setting OE(PH6) HIGH
PORTH | = ( 1 < < 6 ) ;
// Setting WR(PL5) and RD(PL6) HIGH
PORTL | = ( 1 < < 5 ) | ( 1 < < 6 ) ;
2023-06-26 12:04:00 +02:00
}
2023-04-06 16:04:50 +02:00
// For Raphnet and Retron adapters
else {
2022-06-16 17:57:00 +02:00
// Set Control Pins to Output RST(PH0) WR(PH5) OE(PH6)
DDRH | = ( 1 < < 0 ) | ( 1 < < 5 ) | ( 1 < < 6 ) ;
// CE(PL1)
DDRL | = ( 1 < < 1 ) ;
// Setting RST(PH0) WR(PH5) OE(PH6) HIGH
PORTH | = ( 1 < < 0 ) | ( 1 < < 5 ) | ( 1 < < 6 ) ;
// CE(PL1)
PORTL | = ( 1 < < 1 ) ;
}
2023-04-06 16:04:50 +02:00
if ( ! system_sg1000 ) {
// SMS/GG ROM has 16KB banks which can be mapped to one of three slots via register writes
// Register Slot Address space
// $fffd 0 $0000-$3fff
// $fffe 1 $4000-$7fff
// $ffff 2 $8000-$bfff
// Disable sram
writeByte_SMS ( 0xFFFC , 0 ) ;
// Map first 3 banks so we can read-out the header info
writeByte_SMS ( 0xFFFD , 0 ) ;
writeByte_SMS ( 0xFFFE , 1 ) ;
writeByte_SMS ( 0xFFFF , 2 ) ;
}
2022-06-16 17:57:00 +02:00
delay ( 400 ) ;
2024-07-29 13:08:59 -07:00
// Read and print cart info only if ROM size not manually selected
if ( manRomSizeSelected = = false ) {
getCartInfo_SMS ( ) ;
}
2022-06-16 17:57:00 +02:00
}
2023-04-06 16:04:50 +02:00
//*****************************************
2023-06-26 12:04:00 +02:00
// Low level functions
2023-04-06 16:04:50 +02:00
//*****************************************
2022-06-16 17:57:00 +02:00
void writeByte_SMS ( word myAddress , byte myData ) {
2023-04-06 16:04:50 +02:00
if ( adapter_retrode & & system_gg ) {
2022-06-16 17:57:00 +02:00
// Set Data Pins (D8-D15) to Output
DDRA = 0xFF ;
} else {
// Set Data Pins (D0-D7) to Output
DDRC = 0xFF ;
}
// Set address
PORTF = myAddress & 0xFF ;
PORTK = ( myAddress > > 8 ) & 0xFF ;
2023-04-06 16:04:50 +02:00
if ( ! adapter_retrode ) {
2023-11-16 22:27:01 +00:00
// A15/M0-7(PH3) and OE(PH6) are connected
2022-06-16 17:57:00 +02:00
PORTH = ( PORTH & 0 b11110111 ) | ( ( myAddress > > 12 ) & 0 b00001000 ) ;
}
// Output data
2023-04-06 16:04:50 +02:00
if ( adapter_retrode & & system_gg ) {
2022-06-16 17:57:00 +02:00
PORTA = myData ;
} else {
PORTC = myData ;
}
// Arduino running at 16Mhz -> one nop = 62.5ns
// Wait till output is stable
2022-10-13 09:49:03 +02:00
__asm__ ( " nop \n \t "
" nop \n \t "
" nop \n \t "
" nop \n \t " ) ;
2022-06-16 17:57:00 +02:00
2023-04-06 16:04:50 +02:00
if ( adapter_retrode ) {
2022-06-16 17:57:00 +02:00
// Switch WR(PL5) and OE/CE(PH6) to LOW
PORTL & = ~ ( 1 < < 5 ) ;
PORTH & = ~ ( 1 < < 6 ) ;
} else {
// Switch CE(PL1) and WR(PH5) to LOW
PORTL & = ~ ( 1 < < 1 ) ;
PORTH & = ~ ( 1 < < 5 ) ;
}
// Leave WR low for at least 60ns
2022-10-13 09:49:03 +02:00
__asm__ ( " nop \n \t "
" nop \n \t "
" nop \n \t "
" nop \n \t " ) ;
2022-06-16 17:57:00 +02:00
2023-04-06 16:04:50 +02:00
if ( adapter_retrode ) {
2022-06-16 17:57:00 +02:00
// Switch WR(PL5) and OE/CE(PH6) to HIGH
PORTH | = ( 1 < < 6 ) ;
PORTL | = ( 1 < < 5 ) ;
} else {
// Switch CE(PL1) and WR(PH5) to HIGH
PORTH | = ( 1 < < 5 ) ;
PORTL | = ( 1 < < 1 ) ;
}
// Leave WR high for at least 50ns
2022-10-13 09:49:03 +02:00
__asm__ ( " nop \n \t "
" nop \n \t "
" nop \n \t "
" nop \n \t " ) ;
2022-06-16 17:57:00 +02:00
2023-04-06 16:04:50 +02:00
if ( adapter_retrode & & system_gg ) {
2022-06-16 17:57:00 +02:00
// Set Data Pins (D8-D15) to Input
DDRA = 0x00 ;
} else {
// Set Data Pins (D0-D7) to Input
DDRC = 0x00 ;
}
}
byte readByte_SMS ( word myAddress ) {
2023-04-06 16:04:50 +02:00
if ( adapter_retrode & & system_gg ) {
2022-06-16 17:57:00 +02:00
// Set Data Pins (D8-D15) to Input
DDRA = 0x00 ;
} else {
// Set Data Pins (D0-D7) to Input
DDRC = 0x00 ;
}
// Set Address
PORTF = myAddress & 0xFF ;
PORTK = ( myAddress > > 8 ) & 0xFF ;
2023-04-06 16:04:50 +02:00
if ( ! adapter_retrode ) {
2023-11-16 22:27:01 +00:00
// A15/M0-7(PH3) and OE(PH6) are connected
2022-06-16 17:57:00 +02:00
PORTH = ( PORTH & 0 b11110111 ) | ( ( myAddress > > 12 ) & 0 b00001000 ) ;
}
2022-10-13 09:49:03 +02:00
__asm__ ( " nop \n \t "
" nop \n \t "
" nop \n \t "
" nop \n \t " ) ;
2022-06-16 17:57:00 +02:00
2023-04-06 16:04:50 +02:00
if ( adapter_retrode ) {
2022-06-16 17:57:00 +02:00
// Switch RD(PL6) and OE(PH6) to LOW
PORTL & = ~ ( 1 < < 6 ) ;
PORTH & = ~ ( 1 < < 6 ) ;
} else {
// Switch CE(PL1) and OE(PH6) to LOW
PORTL & = ~ ( 1 < < 1 ) ;
PORTH & = ~ ( 1 < < 6 ) ;
}
2022-10-13 09:49:03 +02:00
__asm__ ( " nop \n \t "
" nop \n \t "
" nop \n \t "
" nop \n \t " ) ;
2022-06-16 17:57:00 +02:00
// Read
2023-04-06 16:04:50 +02:00
byte tempByte = ( adapter_retrode & & system_gg ) ? PINA : PINC ;
2022-06-16 17:57:00 +02:00
2023-04-06 16:04:50 +02:00
if ( adapter_retrode ) {
2022-06-16 17:57:00 +02:00
// Switch RD(PL6) and OE(PH6) to HIGH
PORTH | = ( 1 < < 6 ) ;
PORTL | = ( 1 < < 6 ) ;
} else {
// Switch CE(PL1) and OE(PH6) to HIGH
PORTH | = ( 1 < < 6 ) ;
PORTL | = ( 1 < < 1 ) ;
}
2022-10-13 09:49:03 +02:00
__asm__ ( " nop \n \t "
" nop \n \t "
" nop \n \t "
" nop \n \t " ) ;
2022-06-16 17:57:00 +02:00
return tempByte ;
}
byte readNibble ( byte data , byte number ) {
2023-04-06 16:04:50 +02:00
return ( ( data > > ( number * 4 ) ) & 0xF ) ;
2022-06-16 17:57:00 +02:00
}
2023-04-06 16:04:50 +02:00
//*****************************************
// Cartridges functions
//*****************************************
2023-06-26 12:04:00 +02:00
void getCartInfo_SMS ( ) {
2024-07-29 13:08:59 -07:00
byte cartNib = readNibble ( readByte_SMS ( 0x7FFF ) , 0 ) ;
// Get rom size from header
// Note: Common for this value to be smaller than the actual value.
// Normally used for BIOS checksum calculations on export hardware (non JP).
switch ( cartNib ) {
2022-06-16 17:57:00 +02:00
case 0xa :
2024-07-29 13:08:59 -07:00
cartSize = 8192 ; // 8 KiB
2022-06-16 17:57:00 +02:00
break ;
case 0xb :
2024-07-29 13:08:59 -07:00
cartSize = 16384 ; // 16 KiB
2022-06-16 17:57:00 +02:00
break ;
case 0xc :
2024-07-29 13:08:59 -07:00
cartSize = 32768 ; // 32 KiB
2022-06-16 17:57:00 +02:00
break ;
case 0xd :
2024-07-29 13:08:59 -07:00
cartSize = 49152 ; // 48 KiB
2022-06-16 17:57:00 +02:00
break ;
case 0xe :
2024-07-29 13:08:59 -07:00
cartSize = 65536 ; // 64 KiB
2022-06-16 17:57:00 +02:00
break ;
case 0xf :
2024-07-29 13:08:59 -07:00
cartSize = 131072 ; // 128 KiB
2022-06-16 17:57:00 +02:00
break ;
case 0x0 :
2024-07-29 13:08:59 -07:00
cartSize = 262144 ; // 256 KiB
2022-06-16 17:57:00 +02:00
break ;
case 0x1 :
case 0x2 :
2024-07-29 13:08:59 -07:00
cartSize = 524288 ; // 512 KiB
2022-06-16 17:57:00 +02:00
break ;
case 0x3 :
// 0x3 is (only?) used in The Pro Yakyuu '91 (Game Gear)
2024-07-29 13:08:59 -07:00
cartSize = 131072 ; // 128 KiB
2022-06-16 17:57:00 +02:00
break ;
default :
2024-07-29 13:08:59 -07:00
cartSize = 49152 ; // 48 KiB
2022-06-16 17:57:00 +02:00
// LED Error
2024-05-26 22:43:41 +02:00
rgbLed ( blue_color ) ;
2022-06-16 17:57:00 +02:00
break ;
}
2023-04-06 16:04:50 +02:00
// Get rom name (expecting "TMR SEGA" string)
2022-06-16 17:57:00 +02:00
for ( byte i = 0 ; i < 8 ; i + + ) {
2023-04-06 16:04:50 +02:00
romName [ i ] = char ( readByte_SMS ( 0x7FF0 + i ) ) ;
2022-06-16 17:57:00 +02:00
}
romName [ 8 ] = ' \0 ' ;
2023-04-06 16:04:50 +02:00
// Attempt to detect cart size by checking if "TMR SEGA" is mirrored
2024-07-29 13:08:59 -07:00
// Based on https://www.raphnet.net/electronique/sms_cartridge_programmer/index_en.php#6
// Note: Logic does not work on US CloudMaster (256K) and Penquin Land (128K) SMS carts.
// Both detect as 512 KB based on the logic below.
if ( strcmp ( romName , " TMR SEGA " ) = = 0 ) {
byte bank = 1 ;
char romNameBuf [ 9 ] ;
while ( bank < 64 ) { // Total number of possible banks: 1 MiB / 16 (KiB/Bank)
bank + + ; // Increment the bank
writeByte_SMS ( 0xfffe , bank ) ; // Load bank into slot 1
for ( byte i = 0 ; i < 8 ; i + + ) {
romNameBuf [ i ] = char ( readByte_SMS ( 0x7FF0 + i ) ) ;
}
romNameBuf [ 8 ] = ' \0 ' ;
2023-04-06 16:04:50 +02:00
2024-07-29 13:08:59 -07:00
// Debug info:
// print_Msg(F("Bank: "));
// println_Msg(bank);
// print_Msg(F("Parsed ROM name: "));
// println_Msg(romNameBuf);
// println_Msg(FS(FSTRING_EMPTY));
2023-04-06 16:04:50 +02:00
2024-07-29 13:08:59 -07:00
if ( strcmp ( romNameBuf , romName ) = = 0 ) {
break ;
}
2022-06-16 17:57:00 +02:00
}
2024-07-29 13:08:59 -07:00
if ( bank > 2 ) { // 32 KiB is the smallest SMS/GG ROM size
cartSize = ( bank - 1 ) * 16384UL ;
2022-06-16 17:57:00 +02:00
}
2023-04-06 16:04:50 +02:00
2024-07-29 13:08:59 -07:00
// Debug info:
// print_Msg(F("Calculated ROM Size: "));
// println_Msg(cartSize);
2022-06-16 17:57:00 +02:00
2024-07-29 13:08:59 -07:00
// Reset Bank Slot 1
writeByte_SMS ( 0xFFFE , 1 ) ;
2022-06-16 17:57:00 +02:00
2024-07-29 13:08:59 -07:00
// Display header info
display_Clear ( ) ;
if ( system_sms ) {
println_Msg ( F ( " SMS Header: " ) ) ;
} else {
println_Msg ( F ( " GG Header: " ) ) ;
2022-06-16 17:57:00 +02:00
}
2024-07-29 13:08:59 -07:00
} else { // romName != "TMR SEGA"
// Fix for "Fantasy Zone (J) (V1.0)" that has not the normal header, but "COPYRIGHT SEGAPRG. BY T.ASAI".
char headerFZ [ 29 ] ;
if ( strcmp ( romName , " G. BY T.A " ) ! = 0 ) {
for ( byte i = 0 ; i < 28 ; i + + ) {
headerFZ [ i ] = char ( readByte_SMS ( 0x7FE0 + i ) ) ;
2023-04-06 16:04:50 +02:00
}
2024-07-29 13:08:59 -07:00
headerFZ [ 28 ] = ' \0 ' ;
if ( strcmp ( headerFZ , " COPYRIGHT SEGAPRG. BY T.ASAI " ) = = 0 ) {
strcpy ( romName , " TMR SEGA " ) ;
cartSize = 131072 ; // 128 KiB
2023-04-06 16:04:50 +02:00
}
2022-06-16 17:57:00 +02:00
}
2024-07-29 13:08:59 -07:00
manual_selectRomSize ( ) ;
2023-04-06 16:04:50 +02:00
// Display cart info
2022-06-16 17:57:00 +02:00
display_Clear ( ) ;
2023-04-06 16:04:50 +02:00
println_Msg ( F ( " SMS/GG header not found " ) ) ;
2022-06-16 17:57:00 +02:00
}
2024-07-29 13:08:59 -07:00
println_Msg ( FS ( FSTRING_SPACE ) ) ;
print_Msg ( FS ( FSTRING_NAME ) ) ;
println_Msg ( romName ) ;
print_Msg ( cartSize / 1024 ) ;
println_Msg ( F ( " KB " ) ) ;
println_Msg ( FS ( FSTRING_SPACE ) ) ;
2022-06-16 17:57:00 +02:00
2023-06-26 12:04:00 +02:00
// Wait for user input
2024-03-02 11:26:35 -05:00
# if (defined(ENABLE_LCD) || defined(ENABLE_OLED))
2023-06-26 12:04:00 +02:00
// Prints string out of the common strings array either with or without newline
print_STR ( press_button_STR , 1 ) ;
display_Update ( ) ;
wait ( ) ;
# endif
2023-04-06 16:04:50 +02:00
2022-06-16 17:57:00 +02:00
// Turn off LED
2024-05-26 22:43:41 +02:00
rgbLed ( black_color ) ;
2022-06-16 17:57:00 +02:00
}
2024-07-29 13:08:59 -07:00
void manual_selectRomSize ( ) {
// Set rom size manually
unsigned char SMSRomSize ;
if ( system_sg1000 ) {
// Rom sizes for SG-1000
convertPgm ( SG1RomSizeMenu , 4 ) ;
SMSRomSize = question_box ( F ( " Select ROM size " ) , menuOptions , 4 , 0 ) ;
switch ( SMSRomSize ) {
case 0 :
cartSize = 8192 ; // 8 KiB
break ;
case 1 :
cartSize = 16384 ; // 16 KiB
break ;
case 2 :
cartSize = 24576 ; // 24 KiB
break ;
case 3 :
cartSize = 32768 ; // 32 KiB
break ;
//case 4:
// cartSize = 40960; // 40KB
// break;
//case 5:
// cartSize = 49152; // 48KB
// break;
}
} else {
// Rom sizes for SMS and GG
convertPgm ( SMSRomSizeMenu , 6 ) ;
SMSRomSize = question_box ( F ( " Select ROM size " ) , menuOptions , 6 , 0 ) ;
switch ( SMSRomSize ) {
case 0 :
cartSize = 32768 ; // 32 KiB
break ;
case 1 :
cartSize = 65536 ; // 64 KiB
break ;
case 2 :
cartSize = 131072 ; // 128 KiB
break ;
case 3 :
cartSize = 262144 ; // 256 KiB
break ;
case 4 :
cartSize = 524288 ; // 512 KiB
break ;
case 5 :
cartSize = 1048576 ; // 1 MiB
break ;
}
}
strcpy ( romName , " UNKNOWN " ) ; // Use default UNKNOWN rom name upon manual selection
manRomSizeSelected = true ; // This ensures manually selected value is used upon read from the menu
// Display info
display_Clear ( ) ;
print_Msg ( F ( " Selected Size: " ) ) ;
print_Msg ( cartSize / 1024 ) ;
println_Msg ( F ( " KB " ) ) ;
print_STR ( press_button_STR , 1 ) ;
}
2023-04-06 16:04:50 +02:00
//******************************************
// Read ROM and save it to the SD card
//******************************************
2022-06-16 17:57:00 +02:00
void readROM_SMS ( ) {
2022-10-27 22:25:34 +02:00
// Get name, add extension depending on the system and convert to char array for sd lib
2023-04-06 16:04:50 +02:00
if ( system_sms ) {
2024-05-12 15:37:11 +02:00
createFolder ( " SMS " , " ROM " , romName , " sms " ) ;
2023-04-06 16:04:50 +02:00
} else if ( system_gg ) {
2024-05-12 15:37:11 +02:00
createFolder ( " GG " , " ROM " , romName , " gg " ) ;
2023-04-06 16:04:50 +02:00
} else {
2024-05-12 15:37:11 +02:00
createFolder ( " SG1000 " , " ROM " , romName , " sg " ) ;
2022-07-25 13:16:15 +02:00
}
2022-10-27 22:25:34 +02:00
2024-05-25 14:25:35 +02:00
printAndIncrementFolder ( true ) ;
2022-06-16 17:57:00 +02:00
2024-07-29 13:08:59 -07:00
print_Msg ( F ( " ROM Size: " ) ) ;
print_Msg ( cartSize / 1024 ) ;
println_Msg ( F ( " KB " ) ) ;
2022-06-16 17:57:00 +02:00
// Open file on sd card
if ( ! myFile . open ( fileName , O_RDWR | O_CREAT ) ) {
2022-10-30 02:21:01 +00:00
print_FatalError ( sd_error_STR ) ;
2022-06-16 17:57:00 +02:00
}
2024-07-29 13:08:59 -07:00
// Set default bank size to 16 KiB
word bankSize = 16384 ;
2022-06-16 17:57:00 +02:00
2024-07-29 13:08:59 -07:00
// For carts not using mappers (SG1000 or SMS/GG 32 KiB)
if ( ( system_sg1000 ) | | ( cartSize = = 32768 ) ) {
2022-06-16 17:57:00 +02:00
bankSize = cartSize ;
}
2023-04-06 16:04:50 +02:00
// Initialize progress bar
2022-06-16 17:57:00 +02:00
uint32_t processedProgressBar = 0 ;
uint32_t totalProgressBar = ( uint32_t ) ( cartSize ) ;
draw_progressbar ( 0 , totalProgressBar ) ;
for ( byte currBank = 0x0 ; currBank < ( cartSize / bankSize ) ; currBank + + ) {
// Write current 16KB bank to slot 2 register 0xFFFF
2023-04-06 16:04:50 +02:00
if ( ! system_sg1000 ) {
2022-06-16 17:57:00 +02:00
writeByte_SMS ( 0xFFFF , currBank ) ;
}
// Blink led
blinkLED ( ) ;
2022-09-09 00:21:16 +02:00
2022-06-16 17:57:00 +02:00
// Read 16KB from slot 2 which starts at 0x8000
for ( word currBuffer = 0 ; currBuffer < bankSize ; currBuffer + = 512 ) {
// Fill SD buffer
for ( int currByte = 0 ; currByte < 512 ; currByte + + ) {
2024-07-29 13:08:59 -07:00
sdBuffer [ currByte ] = readByte_SMS ( ( ( system_sg1000 ) | | ( cartSize = = 32768 ) ? 0 : 0x8000 ) + currBuffer + currByte ) ;
2022-06-16 17:57:00 +02:00
}
// hexdump for debugging:
// if (currBank == 0 && currBuffer == 0) {
// for (word xi = 0; xi < 0x100; xi++) {
// if (xi%16==0) {
// print_Msg_PaddedHex16(xi);
2024-03-02 11:26:35 -05:00
// print_Msg(FS(FSTRING_SPACE));
2022-06-16 17:57:00 +02:00
// }
// print_Msg_PaddedHexByte(sdBuffer[xi]);
// if (xi>0&&((xi+1)%16)==0) {
2024-03-02 11:26:35 -05:00
// println_Msg(FS(FSTRING_EMPTY));
2022-06-16 17:57:00 +02:00
// } else {
2024-03-02 11:26:35 -05:00
// print_Msg(FS(FSTRING_SPACE));
2022-06-16 17:57:00 +02:00
// }
// }
// }
myFile . write ( sdBuffer , 512 ) ;
}
2023-04-06 16:04:50 +02:00
// Update progress bar
2022-06-16 17:57:00 +02:00
processedProgressBar + = bankSize ;
draw_progressbar ( processedProgressBar , totalProgressBar ) ;
}
2023-04-06 16:04:50 +02:00
// Close file
2022-06-16 17:57:00 +02:00
myFile . close ( ) ;
2023-04-06 16:04:50 +02:00
// Compare dump checksum with database values
if ( system_sms ) {
compareCRC ( " sms.txt " , 0 , 1 , 0 ) ;
} else if ( system_gg ) {
compareCRC ( " gg.txt " , 0 , 1 , 0 ) ;
} else {
compareCRC ( " sg1000.txt " , 0 , 1 , 0 ) ;
}
2024-03-02 11:26:35 -05:00
# ifdef ENABLE_GLOBAL_LOG
2023-06-26 12:04:00 +02:00
save_log ( ) ;
# endif
2023-04-06 16:04:50 +02:00
print_STR ( press_button_STR , 1 ) ;
2022-06-16 17:57:00 +02:00
}
2023-04-06 16:04:50 +02:00
//******************************************
// Read SRAM and save to the SD card
///*****************************************
2022-06-16 17:57:00 +02:00
void readSRAM_SMS ( ) {
// Get name, add extension and convert to char array for sd lib
2024-05-12 15:37:11 +02:00
const char * system ;
2022-06-16 17:57:00 +02:00
2023-04-06 16:04:50 +02:00
if ( system_gg ) {
2024-05-12 15:37:11 +02:00
system = " GG " ;
2022-10-13 09:49:03 +02:00
} else {
2024-05-12 15:37:11 +02:00
system = " SMS " ;
2022-07-25 13:16:15 +02:00
}
2024-05-26 22:20:47 +02:00
createFolderAndOpenFile ( system , " SAVE " , romName , " sav " ) ;
2022-06-16 17:57:00 +02:00
2024-07-29 13:08:59 -07:00
// Write the whole 32 KiB
// When there is only 8 KiB of SRAM, the contents should be duplicated
word bankSize = 16384 ;
2024-05-26 22:20:47 +02:00
for ( byte currBank = 0x0 ; currBank < 2 ; currBank + + ) {
writeByte_SMS ( 0xFFFC , 0x08 | ( currBank < < 2 ) ) ;
// Blink led
blinkLED ( ) ;
2022-06-16 17:57:00 +02:00
2024-07-29 13:08:59 -07:00
// Read 16 KiB from slot 2 which starts at 0x8000
2024-05-26 22:20:47 +02:00
for ( word currBuffer = 0 ; currBuffer < bankSize ; currBuffer + = 512 ) {
// Fill SD buffer
for ( int currByte = 0 ; currByte < 512 ; currByte + + ) {
sdBuffer [ currByte ] = readByte_SMS ( 0x8000 + currBuffer + currByte ) ;
2022-06-16 17:57:00 +02:00
}
2024-05-26 22:20:47 +02:00
myFile . write ( sdBuffer , 512 ) ;
2022-06-16 17:57:00 +02:00
}
}
2024-05-26 22:20:47 +02:00
// Close file
myFile . close ( ) ;
print_STR ( press_button_STR , 1 ) ;
display_Update ( ) ;
2022-06-16 17:57:00 +02:00
}
2023-04-06 16:04:50 +02:00
//**********************************************
// Read file from SD card and write it to SRAM
//**********************************************
2022-06-16 17:57:00 +02:00
void writeSRAM_SMS ( ) {
if ( false ) {
2022-10-30 02:21:01 +00:00
print_Error ( F ( " DISABLED " ) ) ;
2022-10-13 09:49:03 +02:00
} else {
2024-06-16 10:55:50 +02:00
fileBrowser ( FS ( FSTRING_SELECT_FILE ) ) ;
2022-06-16 17:57:00 +02:00
sd . chdir ( ) ;
sprintf ( filePath , " %s/%s " , filePath , fileName ) ;
display_Clear ( ) ;
println_Msg ( F ( " Restoring from " ) ) ;
println_Msg ( filePath ) ;
display_Update ( ) ;
if ( myFile . open ( filePath , O_READ ) ) {
// Get SRAM size from file, with a maximum of 32KB
uint32_t sramSize = myFile . fileSize ( ) ;
if ( sramSize > ( ( uint32_t ) 32 * ( uint32_t ) 1024 ) ) {
sramSize = ( uint32_t ) 32 * ( uint32_t ) 1024 ;
}
print_Msg ( F ( " sramSize: " ) ) ;
print_Msg ( sramSize ) ;
2024-03-02 11:26:35 -05:00
println_Msg ( FS ( FSTRING_EMPTY ) ) ;
2024-07-29 13:08:59 -07:00
word bankSize = 16384 ;
2022-06-16 17:57:00 +02:00
for ( word address = 0x0 ; address < sramSize ; address + = 512 ) {
byte currBank = address > = bankSize ? 1 : 0 ;
word page_address = address - ( currBank * bankSize ) ;
writeByte_SMS ( 0xFFFC , 0x08 | ( currBank < < 2 ) ) ;
// Blink led
blinkLED ( ) ;
myFile . read ( sdBuffer , 512 ) ;
for ( int x = 0 ; x < 512 ; x + + ) {
writeByte_SMS ( 0x8000 + page_address + x , sdBuffer [ x ] ) ;
}
}
2023-04-06 16:04:50 +02:00
// Close file
myFile . close ( ) ;
print_STR ( press_button_STR , 1 ) ;
2022-06-16 17:57:00 +02:00
display_Update ( ) ;
2022-10-13 09:49:03 +02:00
} else {
2022-10-30 02:21:01 +00:00
print_FatalError ( sd_error_STR ) ;
2022-06-16 17:57:00 +02:00
}
}
2022-10-13 09:49:03 +02:00
sd . chdir ( ) ; // root
filePath [ 0 ] = ' \0 ' ; // Reset filePath
2022-06-16 17:57:00 +02:00
}
# endif
//******************************************
// End of File
2023-04-06 16:04:50 +02:00
//******************************************