diff --git a/CMakeLists.txt b/CMakeLists.txt
index 6960c96f..e726bfbd 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -200,6 +200,7 @@ SET(SRC_GBA
src/gba/Cheats.cpp
src/gba/CheatSearch.cpp
src/gba/EEprom.cpp
+ src/gba/ereader.cpp
src/gba/Flash.cpp
src/gba/GBA.cpp
src/gba/GBAGfx.cpp
diff --git a/project/vc2008_mfc/VBA2008.vcproj b/project/vc2008_mfc/VBA2008.vcproj
index 46e483a8..f7200b0d 100644
--- a/project/vc2008_mfc/VBA2008.vcproj
+++ b/project/vc2008_mfc/VBA2008.vcproj
@@ -430,6 +430,14 @@
RelativePath="..\..\src\gba\agbprint.h"
>
+
+
+
+
diff --git a/project/vs2010_mfc/VBA2010.sln b/project/vs2010_mfc/VBA2010.sln
index e5211571..59df2772 100644
--- a/project/vs2010_mfc/VBA2010.sln
+++ b/project/vs2010_mfc/VBA2010.sln
@@ -6,7 +6,7 @@ Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "VisualBoyAdvance-M", "VBA20
EndProject
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "zlib", "..\..\..\dependencies\zlib\project\vs2010\zlib.vcxproj", "{3E03C179-8251-46E4-81F4-466F114BAC63}"
EndProject
-Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "SFML_Network", "..\..\..\dependencies\SFML\build\vc2010\sfml-network.vcxproj", "{823DDC98-42D5-4A38-88CF-9DC06C788AE4}"
+Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "SFML", "..\..\..\dependencies\SFML\build\vc2010\sfml-network.vcxproj", "{823DDC98-42D5-4A38-88CF-9DC06C788AE4}"
EndProject
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "libpng", "..\..\..\dependencies\libpng\project\vs2010\libpng.vcxproj", "{0008960E-E0DD-41A6-8265-00B31DDB4C21}"
EndProject
diff --git a/project/vs2010_mfc/VBA2010.vcxproj b/project/vs2010_mfc/VBA2010.vcxproj
index 9ca2dec7..16733b6b 100644
--- a/project/vs2010_mfc/VBA2010.vcxproj
+++ b/project/vs2010_mfc/VBA2010.vcxproj
@@ -110,7 +110,7 @@
AnySuitable
Speed
..\..\src;..\..\fex;..\..\..\dependencies\libpng;..\..\..\dependencies\msvc;..\..\..\dependencies\SFML\include;..\..\..\dependencies\SubWCRev;..\..\..\dependencies\zlib;%(AdditionalIncludeDirectories)
- WIN32;_WINDOWS;NDEBUG;GBA_LOGGING;OEMRESOURCE;MMX;ASM;FINAL_VERSION;BKPT_SUPPORT;_CRT_SECURE_NO_DEPRECATE;HAS_FILE_EXTRACTOR;%(PreprocessorDefinitions)
+ WIN32;_WINDOWS;NO_OAL;NDEBUG;GBA_LOGGING;OEMRESOURCE;MMX;ASM;FINAL_VERSION;BKPT_SUPPORT;_CRT_SECURE_NO_DEPRECATE;HAS_FILE_EXTRACTOR;%(PreprocessorDefinitions)
MultiThreaded
StreamingSIMDExtensions
Fast
@@ -166,6 +166,7 @@
+
@@ -296,6 +297,7 @@
+
diff --git a/project/vs2010_mfc/VBA2010.vcxproj.filters b/project/vs2010_mfc/VBA2010.vcxproj.filters
index 70ddff39..42e79aec 100644
--- a/project/vs2010_mfc/VBA2010.vcxproj.filters
+++ b/project/vs2010_mfc/VBA2010.vcxproj.filters
@@ -109,6 +109,9 @@
Core\GBA
+
+ Core\GBA
+
Core\GBA
@@ -487,6 +490,9 @@
Core\GBA
+
+ Core\GBA
+
Core\GBA
diff --git a/project/vs2013_mfc/VBA2013.vcxproj b/project/vs2013_mfc/VBA2013.vcxproj
index 70d3d60d..fa2ffe80 100644
--- a/project/vs2013_mfc/VBA2013.vcxproj
+++ b/project/vs2013_mfc/VBA2013.vcxproj
@@ -176,6 +176,7 @@
+
@@ -303,6 +304,7 @@
+
diff --git a/project/vs2013_mfc/VBA2013.vcxproj.filters b/project/vs2013_mfc/VBA2013.vcxproj.filters
index 70ddff39..d4a40f95 100644
--- a/project/vs2013_mfc/VBA2013.vcxproj.filters
+++ b/project/vs2013_mfc/VBA2013.vcxproj.filters
@@ -118,6 +118,9 @@
Core\GBA
+
+ Core\GBA
+
Core\GBA
@@ -487,6 +490,9 @@
Core\GBA
+
+ Core\GBA
+
Core\GBA
diff --git a/src/gba/GBA.cpp b/src/gba/GBA.cpp
index aebffcdb..6009dbbf 100644
--- a/src/gba/GBA.cpp
+++ b/src/gba/GBA.cpp
@@ -20,6 +20,7 @@
#include "../common/Port.h"
#include "../System.h"
#include "agbprint.h"
+#include "ereader.h"
#include "GBALink.h"
#ifdef PROFILING
@@ -159,6 +160,9 @@ u8 memoryWaitSeq32[16] =
u8 biosProtected[4];
+const char *loadDotCodeFile;
+const char *saveDotCodeFile;
+
#ifdef WORDS_BIGENDIAN
bool cpuBiosSwapped = false;
#endif
@@ -1616,6 +1620,26 @@ void doMirroring (bool b)
}
}
+const char* GetLoadDotCodeFile()
+{
+ return loadDotCodeFile;
+}
+
+const char* GetSaveDotCodeFile()
+{
+ return saveDotCodeFile;
+}
+
+void SetLoadDotCodeFile(const char *szFile)
+{
+ loadDotCodeFile = szFile;
+}
+
+void SetSaveDotCodeFile(const char *szFile)
+{
+ saveDotCodeFile = szFile;
+}
+
void CPUUpdateRender()
{
switch(DISPCNT & 7) {
@@ -1931,8 +1955,19 @@ void CPUSoftwareInterrupt(int comment)
VCOUNT);
}
#endif
- CPUSoftwareInterrupt();
- return;
+ if ((comment & 0xF8) != 0xE0)
+ {
+ CPUSoftwareInterrupt();
+ return;
+ }
+ else
+ {
+ if (CheckEReaderRegion())
+ BIOS_EReader_ScanCard(comment);
+ else
+ CPUSoftwareInterrupt();
+ return;
+ }
}
// This would be correct, but it causes problems if uncommented
// else {
@@ -2160,6 +2195,17 @@ void CPUSoftwareInterrupt(int comment)
case 0x1F:
BIOS_MidiKey2Freq();
break;
+ case 0xE0:
+ case 0xE1:
+ case 0xE2:
+ case 0xE3:
+ case 0xE4:
+ case 0xE5:
+ case 0xE6:
+ case 0xE7:
+ if (CheckEReaderRegion())
+ BIOS_EReader_ScanCard(comment);
+ break;
case 0x2A:
BIOS_SndDriverJmpTableCopy();
// let it go, because we don't really emulate this function
@@ -3271,6 +3317,18 @@ void CPUReset()
break;
}
}
+ switch (CheckEReaderRegion())
+ {
+ case 1: //US
+ EReaderWriteMemory(0x8009134, 0x46C0DFE0);
+ break;
+ case 2:
+ EReaderWriteMemory(0x8008A8C, 0x46C0DFE0);
+ break;
+ case 3:
+ EReaderWriteMemory(0x80091A8, 0x46C0DFE0);
+ break;
+ }
rtcReset();
// clean registers
memset(®[0], 0, sizeof(reg));
diff --git a/src/gba/GBA.h b/src/gba/GBA.h
index 7da9df11..5feca67c 100644
--- a/src/gba/GBA.h
+++ b/src/gba/GBA.h
@@ -114,6 +114,11 @@ extern void cpuProfil(profile_segment *seg);
extern void cpuEnableProfiling(int hz);
#endif
+const char* GetLoadDotCodeFile();
+const char* GetSaveDotCodeFile();
+void SetLoadDotCodeFile(const char *szFile);
+void SetSaveDotCodeFile(const char *szFile);
+
extern struct EmulatedSystem GBASystem;
#define R13_IRQ 18
diff --git a/src/gba/ereader.cpp b/src/gba/ereader.cpp
new file mode 100644
index 00000000..64197a92
--- /dev/null
+++ b/src/gba/ereader.cpp
@@ -0,0 +1,670 @@
+#include
+#include
+#include
+#include
+
+#include "GBA.h"
+#include "GBAinline.h"
+#include "Globals.h"
+#include "ereader.h"
+
+
+//#include "win32/stdafx.h"
+//#include "win32/vba.h"
+//#include "win32/MainWnd.h"
+//#include "win32/FileDlg.h"
+//#include "win32/WinResUtil.h"
+//#include "win32/MainWnd.h"
+
+
+int eReaderSaveRAW = 1;
+
+char US_Ereader[19] = "CARDE READERPSAE01";
+char JAP_Ereader[19] = "CARDE READERPEAJ01";
+char JAP_Ereader_plus[19] = "CARDEREADER+PSAJ01";
+char rom_info[19];
+
+char Signature[0x29] = "E-Reader Dotcode -Created- by CaitSith2";
+
+
+unsigned char ShortDotCodeHeader[0x30] = {
+ 0x00, 0x30, 0x01, 0x01,
+ 0x00, 0x01, 0x05, 0x10,
+ 0x00, 0x00, 0x10, 0x12, //Constant data
+
+ 0x00, 0x00, //Header First 2 bytes
+
+ 0x02, 0x00, //Constant data
+
+ 0x00, 0x00, //Header Second 2 bytes
+
+ 0x10, 0x47, 0xEF, //Global Checksum 1
+
+ 0x19, 0x00, 0x00, 0x00, 0x08, 0x4E, 0x49,
+ 0x4E, 0x54, 0x45, 0x4E, 0x44, 0x4F, 0x00, 0x22,
+ 0x00, 0x09, //Constant data
+
+ 0x00, 0x00, //Header, last 8 bytes
+ 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00,
+ 0x00, //Header Checksum
+ 0x57 //Global Checksum 2
+};
+
+unsigned char LongDotCodeHeader[0x30] = {
+ 0x00, 0x30, 0x01, 0x02,
+ 0x00, 0x01, 0x08, 0x10,
+ 0x00, 0x00, 0x10, 0x12, //Constant Data
+
+ 0x00, 0x00, //Header, first 2 bytes
+
+ 0x01, 0x00, //Constant data
+
+ 0x00, 0x00, //Header, second 2 bytes
+ 0x10, 0x9A, 0x99, //Global Checksum 1
+
+ 0x19, 0x00, 0x00, 0x00, 0x08, 0x4E, 0x49,
+ 0x4E, 0x54, 0x45, 0x4E, 0x44, 0x4F, 0x00, 0x22,
+ 0x00, 0x09, //Constant data
+
+
+ 0x00, 0x00, //Header, last 8 bytes
+ 0x00, 0x00, 0x00, 0x00,
+ 0x00, 0x00,
+ 0x00, //Header Checksum
+ 0x57 //Global Checksum 2
+};
+
+unsigned char shortheader[0x18] = {
+ 0x00, 0x02, 0x00, 0x01, 0x40, 0x10, 0x00, 0x1C,
+ 0x10, 0x6F, 0x40, 0xDA, 0x39, 0x25, 0x8E, 0xE0,
+ 0x7B, 0xB5, 0x98, 0xB6, 0x5B, 0xCF, 0x7F, 0x72
+};
+unsigned char longheader[0x18] = {
+ 0x00, 0x03, 0x00, 0x19, 0x40, 0x10, 0x00, 0x2C,
+ 0x0E, 0x88, 0xED, 0x82, 0x50, 0x67, 0xFB, 0xD1,
+ 0x43, 0xEE, 0x03, 0xC6, 0xC6, 0x2B, 0x2C, 0x93
+};
+
+unsigned char dotcodeheader[0x48];
+unsigned char dotcodedata[0xB38];
+unsigned char dotcodetemp[0xB00];
+int dotcodepointer;
+int dotcodeinterleave;
+int decodestate;
+
+u32 GFpow;
+
+unsigned char *DotCodeData;
+char filebuffer[2048];
+
+int dotcodesize;
+
+int CheckEReaderRegion(void) //US = 1, JAP = 2, JAP+ = 3
+{
+ int i;
+ for(i=0;i<18;i++)
+ rom_info[i] = rom[0xA0+i];
+ rom_info[i] = 0;
+
+ if (std::tolower(*rom_info, std::locale()) == std::tolower(*US_Ereader, std::locale()))
+ return 1;
+ if (std::tolower(*rom_info, std::locale()) == std::tolower(*JAP_Ereader, std::locale()))
+ return 2;
+ if (std::tolower(*rom_info, std::locale()) == std::tolower(*JAP_Ereader_plus, std::locale()))
+ return 3;
+
+ return 0;
+}
+
+int LoadDotCodeData(int size, u32* DCdata, unsigned long MEM1, unsigned long MEM2, int loadraw)
+{
+ u32 temp1;
+ int i,j;
+
+ unsigned char scanmap[28];
+ int scantotal=0;
+
+ for(i=0;i<28;i++)
+ scanmap[i] = 0;
+
+ unsigned char longdotcodescan[28] = {
+ 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1,
+ 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1,
+ 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1,
+ 0xF1, 0xF2, 0xB1, 0xB1
+ };
+ unsigned char shortdotcodescan[18] = {
+ 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1,
+ 0xF1, 0xF1, 0xF1, 0xF1, 0xF1, 0xF1,
+ 0xF1, 0xF1, 0xF1, 0xF2, 0xB1, 0xB1
+ };
+
+ temp1 = CPUReadMemory(MEM1-4);
+ for(i=0;i<0x60;i+=4)
+ CPUWriteMemory((MEM2-8)+i,0);
+ for(i=0;i<0x1860;i+=4)
+ CPUWriteMemory(temp1+i,0);
+ if(DCdata != NULL)
+ {
+ if(size==0xB60)
+ {
+ for(i=0;i<28;i++)
+ {
+ for(j=0,scantotal=0;j<0x68;j+=4)
+ {
+ scantotal += DCdata[((i*0x68)+j)>>2];
+ }
+ if(scantotal)
+ scanmap[i] = longdotcodescan[i];
+ }
+ for(i=0;i>2]);
+ }
+ }
+ else if (size==0x750)
+ {
+ for(i=0;i<18;i++)
+ {
+ if((DCdata[0] == 0x02011394) && (DCdata[1] == 0x0203E110) && (i == 0))
+ continue;
+ for(j=0,scantotal=0;j<0x68;j+=4)
+ {
+ scantotal += DCdata[((i*0x68)+j)>>2];
+ }
+ if(scantotal)
+ scanmap[i] = longdotcodescan[i];
+ }
+ for(i=0;i>2]);
+ }
+ }
+ }
+ CPUWriteMemory(MEM2-8,0x1860);
+ CPUWriteMemory(MEM2-4,temp1);
+
+
+ if(size==0xB60)
+ {
+ if(loadraw)
+ {
+ for(i=0;i<28;i++)
+ CPUWriteByte(MEM2+0x18+i,scanmap[i]);
+ }
+ else
+ {
+ CPUWriteMemory(MEM2+0x18,0xF1F1F1F1);
+ CPUWriteMemory(MEM2+0x18+4,0xF1F1F1F1);
+ CPUWriteMemory(MEM2+0x18+8,0xF1F1F1F1);
+ CPUWriteMemory(MEM2+0x18+12,0xF1F1F1F1);
+ CPUWriteMemory(MEM2+0x18+16,0xF1F1F1F1);
+ CPUWriteMemory(MEM2+0x18+20,0xF1F1F1F1);
+ CPUWriteMemory(MEM2+0x18+24,0xB1B1F2F1);
+ }
+ CPUWriteMemory(MEM2+0x40,0x19);
+ CPUWriteMemory(MEM2+0x44,0x34);
+ }
+ else if (size==0x750)
+ {
+ if(loadraw)
+ {
+ for(i=0;i<18;i++)
+ CPUWriteByte(MEM2+i,scanmap[i]);
+ }
+ else
+ {
+ CPUWriteMemory(MEM2,0xF1F1F1F1);
+ CPUWriteMemory(MEM2+4,0xF1F1F1F1);
+ CPUWriteMemory(MEM2+8,0xF1F1F1F1);
+ CPUWriteMemory(MEM2+12,0xF2F1F1F1);
+ CPUWriteMemory(MEM2+16,0xB1B1);
+ }
+ CPUWriteMemory(MEM2+0x40,0x01);
+ CPUWriteMemory(MEM2+0x44,0x12);
+ }
+ CPUWriteMemory(MEM2+0x48,0x3C);
+ CPUWriteMemory(MEM2+0x4C,MEM2);
+
+ return 0;
+}
+
+void EReaderWriteMemory(u32 address, u32 value)
+{
+ switch(address >> 24)
+ {
+ case 2:
+ WRITE32LE(((u32 *)&workRAM[address & 0x3FFFF]), value);
+ break;
+ case 3:
+ WRITE32LE(((u32 *)&internalRAM[address & 0x7FFF]), value);
+ break;
+ default:
+ WRITE32LE(((u32 *)&rom[address & 0x1FFFFFF]), value);
+ //rom[address & 0x1FFFFFF] = data;
+ break;
+ }
+}
+
+
+void BIOS_EReader_ScanCard(int swi_num)
+{
+
+ int i,j,k;
+ int dotcodetype;
+
+ int global1, global2;
+
+ FILE *f;
+
+ //Open dotcode bin/raw
+
+ if(swi_num == 0xE0)
+ {
+ dotcodepointer = 0;
+ dotcodeinterleave = 0;
+ decodestate = 0;
+
+ const char* loadDotCodeFile = GetLoadDotCodeFile();
+
+ if (loadDotCodeFile == 0)
+ {
+ reg[0].I = 0x301;
+ return;
+ }
+ f=fopen(loadDotCodeFile,"rb");
+ //f=fopen(filebuffer,"rb");
+ //f=fopen("dotcode4.raw","rb");
+ if(f==NULL)
+ {
+ reg[0].I = 0x303;
+ return;
+ }
+ fseek(f,0,SEEK_END);
+ i=ftell(f);
+ fseek(f,0,SEEK_SET);
+ if((i==0xB60) || (i==0x750))
+ {
+ dotcodetype = 0;
+ }
+ else if ((i==0x81C) || (i==0x51C))
+ {
+ dotcodetype = 1;
+ }
+ else
+ {
+ fclose(f);
+ reg[0].I = 0x303;
+ return;
+ }
+ DotCodeData = (unsigned char*)malloc(i);
+ if(DotCodeData == NULL)
+ {
+ reg[0].I = 0x303;
+ return;
+ }
+ fread(DotCodeData,1,i,f);
+ fclose(f);
+
+ if(dotcodetype == 0)
+ {
+
+ switch(CheckEReaderRegion())
+ {
+ case 1: //US
+ LoadDotCodeData(i,(u32 *)DotCodeData,0x2032D14,0x2028B28,1);
+ EReaderWriteMemory(0x80091BA,0x46C0DFE2);
+ break;
+ case 2:
+ LoadDotCodeData(i,(u32 *)DotCodeData,0x2006EC4,0x2002478,1);
+ EReaderWriteMemory(0x8008B12,0x46C0DFE2);
+ break;
+ case 3:
+ LoadDotCodeData(i,(u32 *)DotCodeData,0x202F8A4,0x2031034,1);
+ EReaderWriteMemory(0x800922E,0x46C0DFE2);
+ break;
+ }
+ reg[0].I = 0;
+ free(DotCodeData);
+ }
+ else
+ {
+ //dotcodesize = i;
+ if(i==0x81C)
+ dotcodesize = 0xB60;
+ else
+ dotcodesize = 0x750;
+
+
+ switch(CheckEReaderRegion())
+ {
+ case 1: //US
+ LoadDotCodeData(dotcodesize,(u32 *)NULL,0x2032D14,0x2028B28,0);
+ EReaderWriteMemory(0x80091BA,0x46C0DFE1);
+ break;
+ case 2:
+ LoadDotCodeData(dotcodesize,(u32 *)NULL,0x2006EC4,0x2002478,0);
+ EReaderWriteMemory(0x8008B12,0x46C0DFE1);
+ break;
+ case 3:
+ LoadDotCodeData(dotcodesize,(u32 *)NULL,0x202F8A4,0x2031034,0);
+ EReaderWriteMemory(0x800922E,0x46C0DFE1);
+ break;
+ }
+ reg[0].I = 0;
+ dotcodesize = i;
+ }
+ }
+ else if (swi_num == 0xE1)
+ {
+
+ switch(CheckEReaderRegion())
+ {
+ case 1: //US
+ EReaderWriteMemory(0x80091BA,0xF8A5F03B);
+ EReaderWriteMemory(0x3002F7C,0xEFE40000); //Beginning of Reed-Solomon decoder
+ EReaderWriteMemory(0x3003144,0xCA00002F); //Fix required to Correct 16 "Erasures"
+ EReaderWriteMemory(0x300338C,0xEFE50000); //End of Reed-Solomon decoder
+ GFpow = 0x3000A6C;
+ break;
+ case 2:
+ EReaderWriteMemory(0x8008B12,0xFB0BF035);
+ EReaderWriteMemory(0x3002F88,0xEFE40000);
+ EReaderWriteMemory(0x3003150,0xCA00002F);
+ EReaderWriteMemory(0x3003398,0xEFE50000);
+ GFpow = 0x3000A78;
+ break;
+ case 3:
+ EReaderWriteMemory(0x800922E,0xF94BF04B);
+ EReaderWriteMemory(0x3002F7C,0xEFE40000);
+ EReaderWriteMemory(0x3003144,0xCA00002F);
+ EReaderWriteMemory(0x300338C,0xEFE50000);
+ GFpow = 0x3000A6C;
+ break;
+ }
+ armNextPC -= 2;
+ reg[15].I -= 2;
+
+ for(i=0,j=0;i<12;i++)
+ j ^= DotCodeData[i];
+ if(dotcodesize == 0x81C)
+ {
+ LongDotCodeHeader[0x2E] = j;
+ LongDotCodeHeader[0x0D] = DotCodeData[0];
+ LongDotCodeHeader[0x0C] = DotCodeData[1];
+ LongDotCodeHeader[0x11] = DotCodeData[2];
+ LongDotCodeHeader[0x10] = DotCodeData[3];
+
+ LongDotCodeHeader[0x26] = DotCodeData[4];
+ LongDotCodeHeader[0x27] = DotCodeData[5];
+ LongDotCodeHeader[0x28] = DotCodeData[6];
+ LongDotCodeHeader[0x29] = DotCodeData[7];
+ LongDotCodeHeader[0x2A] = DotCodeData[8];
+ LongDotCodeHeader[0x2B] = DotCodeData[9];
+ LongDotCodeHeader[0x2C] = DotCodeData[10];
+ LongDotCodeHeader[0x2D] = DotCodeData[11];
+
+
+ LongDotCodeHeader[0x12] = 0x10; //calculate Global Checksum 1
+ LongDotCodeHeader[0x02] = 1; //Do not calculate Global Checksum 2
+
+ for(i=0x0C,j=0;i<0x81C;i++)
+ {
+ if(i&1)
+ j += DotCodeData[i];
+ else
+ j += (DotCodeData[i] << 8);
+ }
+ j &= 0xFFFF;
+ j ^= 0xFFFF;
+ LongDotCodeHeader[0x13] = (j & 0xFF00) >> 8;
+ LongDotCodeHeader[0x14] = (j & 0x00FF);
+
+ for(i=0,j=0;i<0x2F;i++)
+ j+=LongDotCodeHeader[i];
+ j &= 0xFF;
+ for(i=1,global2=0;i<0x2C;i++)
+ {
+ for(k=0,global1=0;k<0x30;k++)
+ {
+ global1 ^= DotCodeData[((i-1)*0x30)+k+0x0C];
+ }
+ global2 += global1;
+ }
+ global2 += j;
+ global2 &= 0xFF;
+ global2 ^= 0xFF;
+ LongDotCodeHeader[0x2F] = global2;
+
+
+ }
+ else
+ {
+ ShortDotCodeHeader[0x2E] = j;
+ ShortDotCodeHeader[0x0D] = DotCodeData[0];
+ ShortDotCodeHeader[0x0C] = DotCodeData[1];
+ ShortDotCodeHeader[0x11] = DotCodeData[2];
+ ShortDotCodeHeader[0x10] = DotCodeData[3];
+
+ ShortDotCodeHeader[0x26] = DotCodeData[4];
+ ShortDotCodeHeader[0x27] = DotCodeData[5];
+ ShortDotCodeHeader[0x28] = DotCodeData[6];
+ ShortDotCodeHeader[0x29] = DotCodeData[7];
+ ShortDotCodeHeader[0x2A] = DotCodeData[8];
+ ShortDotCodeHeader[0x2B] = DotCodeData[9];
+ ShortDotCodeHeader[0x2C] = DotCodeData[10];
+ ShortDotCodeHeader[0x2D] = DotCodeData[11];
+
+ ShortDotCodeHeader[0x12] = 0x10; //calculate Global Checksum 1
+ ShortDotCodeHeader[0x02] = 1; //Do not calculate Global Checksum 2
+
+ for(i=0x0C,j=0;i<0x51C;i++)
+ {
+ if(i&1)
+ j += DotCodeData[i];
+ else
+ j += (DotCodeData[i] << 8);
+ }
+ j &= 0xFFFF;
+ j ^= 0xFFFF;
+ ShortDotCodeHeader[0x13] = (j & 0xFF00) >> 8;
+ ShortDotCodeHeader[0x14] = (j & 0x00FF);
+
+ for(i=0,j=0;i<0x2F;i++)
+ j+=ShortDotCodeHeader[i];
+ j &= 0xFF;
+ for(i=1,global2=0;i<0x1C;i++)
+ {
+ for(k=0,global1=0;k<0x30;k++)
+ {
+ global1 ^= DotCodeData[((i-1)*0x30)+k+0x0C];
+ }
+ global2 += global1;
+ }
+ global2 += j;
+ global2 &= 0xFF;
+ global2 ^= 0xFF;
+ ShortDotCodeHeader[0x2F] = global2;
+ }
+
+ }
+ else if (swi_num == 0xE2) //Header
+ {
+ switch(CheckEReaderRegion())
+ {
+ case 1: //US
+ EReaderWriteMemory(0x80091BA,0xF8A5F03B);
+ EReaderWriteMemory(0x300338C,0xEFE30000);
+ GFpow = 0x3000A6C;
+ break;
+ case 2:
+ EReaderWriteMemory(0x8008B12,0xFB0BF035);
+ EReaderWriteMemory(0x3003398,0xEFE30000);
+ GFpow = 0x3000A78;
+ break;
+ case 3:
+ EReaderWriteMemory(0x800922E,0xF94BF04B);
+ EReaderWriteMemory(0x300338C,0xEFE30000);
+ GFpow = 0x3000A6C;
+ break;
+ }
+ armNextPC -= 2;
+ reg[15].I -= 2;
+ }
+ else if ((swi_num == 0xE3) || (swi_num == 0xE5)) //Dotcode data
+ {
+ if((reg[0].I >= 0) && (reg[0].I <= 0x10))
+ {
+ if(decodestate == 0)
+ {
+ for(i=0x17;i>=0;i--)
+ {
+ if((0x17 - i) < 8)
+ j=CPUReadByte(GFpow + CPUReadByte(GFpow+0x200+i));
+ else
+ j=CPUReadByte(GFpow + CPUReadByte(GFpow+0x200+i)) ^ 0xFF;
+
+ dotcodeheader[(0x17-i)] = j;
+ dotcodeheader[(0x17-i)+0x18] = j;
+ dotcodeheader[(0x17-i)+0x30] = j;
+ }
+ for(i=0;i<28;i++)
+ for(j=0;j<2;j++)
+ dotcodedata[(i*0x68)+j] = dotcodeheader[(i*2)+j];
+ dotcodeinterleave = dotcodeheader[7];
+ decodestate = 1;
+ }
+ else
+ {
+ for(i=0x3F;i>=0;i--)
+ {
+ if((0x3F - i) < 0x30)
+ j=CPUReadByte(GFpow + CPUReadByte(GFpow+0x200+i));
+ else
+ j=CPUReadByte(GFpow + CPUReadByte(GFpow+0x200+i)) ^ 0xFF;
+ dotcodetemp[((0x3F-i)*dotcodeinterleave)+dotcodepointer] = j;
+ }
+ dotcodepointer++;
+
+ if((dotcodepointer == dotcodeinterleave))
+ {
+ switch(dotcodeinterleave)
+ {
+ case 0x1C:
+ j = 0x724;
+ k = 0x750 - j;
+ break;
+ case 0x2C:
+ j = 0xB38;
+ k = 0xB60 - j;
+ break;
+ }
+ dotcodepointer = 0;
+ for(i=2;iSetCheck(agbPrintIsEnabled());
}
+void MainWnd::OnOptionsEmulatorEreader()
+{
+ eReaderSaveRAW = !eReaderSaveRAW;
+}
+
+void MainWnd::OnUpdateOptionsEmulatorEreader(CCmdUI* pCmdUI)
+{
+ pCmdUI->SetCheck(eReaderSaveRAW == 1);
+}
+
void MainWnd::OnOptionsEmulatorRealtimeclock()
{
theApp.winRtcEnable = !theApp.winRtcEnable;
diff --git a/src/win32/VBA.cpp b/src/win32/VBA.cpp
index 4912c621..179df3ed 100644
--- a/src/win32/VBA.cpp
+++ b/src/win32/VBA.cpp
@@ -25,6 +25,7 @@
#include "../System.h"
#include "../gba/agbprint.h"
+#include "../gba/ereader.h"
#include "../gba/cheatSearch.h"
#include "../gba/GBA.h"
#include "../gba/Globals.h"
@@ -1413,6 +1414,10 @@ void VBA::loadSettings()
{
CString buffer;
+ eReaderSaveRAW = regQueryDwordValue("eReaderSaveRAW", 1);
+ if (eReaderSaveRAW < 0 || eReaderSaveRAW > 1)
+ eReaderSaveRAW = 1;
+
lastFullscreen = (VIDEO_SIZE)regQueryDwordValue("lastFullscreen", VIDEO_1024x768);
languageOption = regQueryDwordValue("language", 1);
@@ -2512,6 +2517,8 @@ void VBA::movieReadNext()
void VBA::saveSettings()
{
+ regSetDwordValue("eReaderSaveRAW", eReaderSaveRAW);
+
regSetDwordValue("language", languageOption);
regSetStringValue("languageName", languageName);
diff --git a/src/win32/VBA.h b/src/win32/VBA.h
index 00537d6e..fb6d1fe9 100644
--- a/src/win32/VBA.h
+++ b/src/win32/VBA.h
@@ -206,6 +206,9 @@ class VBA : public CWinApp
CString filename;
CString dir;
+ CString loadDotCodeFile;
+ CString saveDotCodeFile;
+
CString wndClass;
int linkTimeout;
diff --git a/src/win32/VBA.rc b/src/win32/VBA.rc
index 16d25850..be520a08 100644
--- a/src/win32/VBA.rc
+++ b/src/win32/VBA.rc
@@ -1600,7 +1600,10 @@ BEGIN
MENUITEM "Open GBC...", ID_FILE_OPEN_GBC
MENUITEM "Open GB...", ID_FILE_OPEN_GB
MENUITEM "Close", ID_FILE_CLOSE
- MENUITEM SEPARATOR
+ MENUITEM SEPARATOR
+ MENUITEM "Load E-Reader DotCode...", ID_FILE_LOAD_DOTCODE
+ MENUITEM "Save E-Reader DotCode...", ID_FILE_SAVE_DOTCODE
+ MENUITEM SEPARATOR
MENUITEM "&Link with...", ID_OPTIONS_LINK_OPTIONS
MENUITEM SEPARATOR
POPUP "Recent"
@@ -1872,6 +1875,7 @@ BEGIN
MENUITEM "Auto-apply IPS/UPS/PPF", ID_OPTIONS_EMULATOR_AUTOMATICALLYAPPLYPATCHFILES
MENUITEM "Pause when inactive", ID_OPTIONS_EMULATOR_PAUSEWHENINACTIVE
MENUITEM "AGB Print", ID_OPTIONS_EMULATOR_AGBPRINT
+ MENUITEM "E-Reader", ID_OPTIONS_EMULATOR_EREADER
MENUITEM "Real Time Clock", ID_OPTIONS_EMULATOR_REALTIMECLOCK
MENUITEM "&Game Overrides...", ID_OPTIONS_EMULATOR_GAMEOVERRIDES
POPUP "Show speed"
diff --git a/src/win32/resource.h b/src/win32/resource.h
index 951345e0..74bd2097 100644
--- a/src/win32/resource.h
+++ b/src/win32/resource.h
@@ -779,6 +779,7 @@
#define ID_OPTIONS_VIDEO_RENDEROPTIONS_GLTRIANGLE 40239
#define ID_OPTIONS_VIDEO_RENDEROPTIONS_GLQUADS 40240
#define ID_OPTIONS_EMULATOR_AGBPRINT 40247
+#define ID_OPTIONS_EMULATOR_EREADER 40280
#define ID_OPTIONS_EMULATOR_REALTIMECLOCK 40248
#define ID_OPTIONS_GAMEBOY_SGB2 40249
#define ID_SYSTEM_MINIMIZE 40250
@@ -874,6 +875,8 @@
#define ID_AUDIO_CORE_SETTINGS 40365
#define ID_FILE_OPEN_GBA 40366
#define ID_OPTIONS_VIDEO_LAYERS_RESET 40367
+#define ID_FILE_LOAD_DOTCODE 40368
+#define ID_FILE_SAVE_DOTCODE 40369
#define ID_LOADGAME_DONOTCHANGECHEATLIST 40371
#define ID_OPTIONS_EMULATOR_SAVETYPE_DETECTNOW 40372
#define ID_FILE_PAUSE 40373