Big restructuring in order to switch to CMake
authorslack <slack@codemaniacs.com>
Tue, 20 Jan 2009 17:39:00 +0000 (18:39 +0100)
committerslack <slack@codemaniacs.com>
Tue, 20 Jan 2009 17:46:54 +0000 (18:46 +0100)
- GameBoy core is now a library
- test_core.cc is now the main wenboi program
- CMakeLists.txt everywhere

Still more changes to come, as I'd like to remove SDL
dependencies from the core, abstracting the video driver and
creating a dummy one for wendi, and a SDL one based on the
existing code.

It would be nice too if class GameBoy had a cleaner
interface with less public stuff :P

34 files changed:
CMakeLists.txt [new file with mode: 0644]
Makefile [deleted file]
common/Logger.h [moved from Logger.h with 100% similarity]
common/Singleton.h [moved from Singleton.h with 100% similarity]
common/sized_types.h [moved from sized_types.h with 100% similarity]
common/toString.h [moved from wendi/disasm.h with 78% similarity]
core/CMakeLists.txt [new file with mode: 0644]
core/GBMemory.cc [moved from GBMemory.cc with 99% similarity]
core/GBMemory.h [moved from GBMemory.h with 99% similarity]
core/GBRom.cc [moved from GBRom.cc with 99% similarity]
core/GBRom.h [moved from GBRom.h with 98% similarity]
core/GBVideo.cc [moved from GBVideo.cc with 99% similarity]
core/GBVideo.h [moved from GBVideo.h with 100% similarity]
core/GameBoy.cc [moved from gbcore.cc with 72% similarity]
core/GameBoy.h [moved from gbcore.h with 95% similarity]
core/Instruction.h [moved from wendi/Instruction.h with 98% similarity]
core/MBC.cc [moved from MBC.cc with 97% similarity]
core/MBC.h [moved from MBC.h with 96% similarity]
core/MBC1.cc [moved from MBC1.cc with 98% similarity]
core/MBC1.h [moved from MBC1.h with 100% similarity]
core/NoMBC.cc [moved from NoMBC.cc with 98% similarity]
core/NoMBC.h [moved from NoMBC.h with 100% similarity]
core/disasm_macros.h [moved from wendi/disasm_macros.h with 88% similarity]
core/opcodes.h [moved from opcodes.h with 99% similarity]
core/util.cc [moved from util.cc with 100% similarity]
core/util.h [moved from util.h with 96% similarity]
wenboi/CMakeLists.txt [new file with mode: 0644]
wenboi/wenboi.cc [moved from tests/test_core.cc with 98% similarity]
wendi/CMakeLists.txt [new file with mode: 0644]
wendi/CodeBlock.h
wendi/disasm.cc [deleted file]
wendi/disassembly_output.h
wendi/output_graph.cc
wendi/wendi.cc

diff --git a/CMakeLists.txt b/CMakeLists.txt
new file mode 100644 (file)
index 0000000..fd3c3e7
--- /dev/null
@@ -0,0 +1,8 @@
+PROJECT(wenboi)
+
+CMAKE_MINIMUM_REQUIRED(VERSION 2.6.0)
+FIND_PACKAGE(SDL REQUIRED)
+
+ADD_SUBDIRECTORY(core)
+ADD_SUBDIRECTORY(wenboi)
+ADD_SUBDIRECTORY(wendi)
diff --git a/Makefile b/Makefile
deleted file mode 100644 (file)
index 61f2172..0000000
--- a/Makefile
+++ /dev/null
@@ -1,59 +0,0 @@
-#CXXFLAGS=-pg -O3 -g -Wall -Weffc++ -Wstrict-null-sentinel -Wold-style-cast 
-CXXFLAGS=-pg -g -Wall -Weffc++ -Wold-style-cast \
-        -Woverloaded-virtual $(shell sdl-config --cflags)
-LDFLAGS=-pg -g $(shell sdl-config --libs)
-
-all: tests
-
-tests: tests/test_gbrom tests/test_core wendi/wendi
-
-util.o: util.cc util.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-
-GBVideo.o: GBVideo.cc GBVideo.h Logger.h util.h gbcore.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-
-GBMemory.o: GBMemory.cc GBMemory.h Logger.h MBC.h gbcore.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-
-MBC.o: MBC.cc MBC.h Logger.h NoMBC.h MBC1.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-
-NoMBC.o: NoMBC.cc NoMBC.h Logger.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-
-MBC1.o: MBC1.cc MBC1.h Logger.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-
-gbcore.o: gbcore.cc gbcore.h opcodes.h \
-       GBRom.h Logger.h MBC.h GBMemory.h GBVideo.h util.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-               
-tests/test_gbrom: GBRom.cc GBRom.h 
-       g++ $(CXXFLAGS)  -DTEST_GBROM -o $@ GBRom.cc $(LDFLAGS)
-
-tests/test_core: tests/test_core.cc gbcore.o MBC.o GBMemory.o GBRom.o \
-       GBVideo.o util.o NoMBC.o MBC1.o wendi/disasm.o
-       g++ $(CXXFLAGS) -o $@ $^ $(LDFLAGS)
-
-wendi/CodeBlock.o: wendi/CodeBlock.cc wendi/CodeBlock.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-
-wendi/disasm.o: wendi/disasm.cc wendi/disasm.h wendi/Instruction.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-
-wendi/output_txt.o: wendi/output_txt.cc wendi/output_txt.h wendi/disassembly_output.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-
-wendi/output_graph.o: wendi/output_graph.cc wendi/output_graph.h wendi/disassembly_output.h Logger.h
-       g++ $(CXXFLAGS) -c -o $@ $<
-
-wendi/wendi: wendi/wendi.cc wendi/CodeBlock.o wendi/disasm.o \
-       wendi/output_txt.o wendi/output_graph.o gbcore.o MBC.o \
-       GBMemory.o GBRom.o GBVideo.o util.o NoMBC.o MBC1.o
-       g++ $(CXXFLAGS) -o $@ $^ $(LDFLAGS)
-
-clean:
-       rm -f *.o wendi/*.o wendi/wendi tests/test_gbrom tests/test_core
-
-.PHONY: clean tests all
similarity index 100%
rename from Logger.h
rename to common/Logger.h
similarity index 100%
rename from Singleton.h
rename to common/Singleton.h
similarity index 100%
rename from sized_types.h
rename to common/sized_types.h
similarity index 78%
rename from wendi/disasm.h
rename to common/toString.h
index 617435409ad2cabf31307984405f8be7a1c9decb..bf47b03aecfc57ac8433e8d449473669569b1891 100644 (file)
 
     You should have received a copy of the GNU General Public License
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
-*/
-#ifndef DISASM_H
-#define DISASM_H
-
-#include "../gbcore.h"
-#include "Instruction.h"
+*/ 
+#ifndef TOSTRING_H
+#define TOSTRING_H
 
 #include <string>
 #include <sstream>
 
 template <class T> 
-std::string ToString(const T &object)
+std::string toString(const T &object)
 {
        std::ostringstream os;
        os << object;
        return(os.str());
 }
 
-Instruction disassemble_opcode(GameBoy &gb, u16 addr);
-std::string get_port_name(int port);
-
 #endif
+
diff --git a/core/CMakeLists.txt b/core/CMakeLists.txt
new file mode 100644 (file)
index 0000000..982988b
--- /dev/null
@@ -0,0 +1,6 @@
+FIND_PACKAGE(SDL REQUIRED)
+
+ADD_LIBRARY(wenboicore STATIC GameBoy.cc GBMemory.cc GBRom.cc GBVideo.cc MBC.cc NoMBC.cc MBC1.cc util.cc) 
+TARGET_LINK_LIBRARIES(wenboicore ${SDL_LIBRARY})
+INCLUDE_DIRECTORIES(${SDL_INCLUDE_DIR})
+
similarity index 99%
rename from GBMemory.cc
rename to core/GBMemory.cc
index a1216a70bbf716bcc9a4a87c83ffd05e88d66531..19c29e68a1fef527cc1987d22fee00dba44c390a 100644 (file)
@@ -16,9 +16,9 @@
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
 */ 
 #include "GBMemory.h"
+#include "GameBoy.h"
 #include "MBC.h"
-#include "gbcore.h"
-#include "Logger.h"
+#include "../common/Logger.h"
 #include <iostream>
 #include <sstream>
 #include <iomanip>
similarity index 99%
rename from GBMemory.h
rename to core/GBMemory.h
index 4f2a911b962a2dd390b743e02349495821791301..def434a871a04faac181093574b0f718213ef13c 100644 (file)
@@ -18,7 +18,7 @@
 #ifndef GBMEMORY_H
 #define GBMEMORY_H
 
-#include "sized_types.h"
+#include "../common/sized_types.h"
 #include <map>
 
 class GameBoy;
similarity index 99%
rename from GBRom.cc
rename to core/GBRom.cc
index 23c568f1ffd63354be23f8527c89e1f0c986c886..0c12e4967bf7e70f5f09c62f648eed7d3ca4ada2 100644 (file)
--- a/GBRom.cc
@@ -16,7 +16,7 @@
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
 */ 
 #include "GBRom.h"
-#include "Logger.h"
+#include "../common/Logger.h"
 #include <fstream>
 #include <iostream>
 #include <iomanip>
similarity index 98%
rename from GBRom.h
rename to core/GBRom.h
index d2771e5ba1a0e32262913f9dd216daf991d7ada3..053395bd5b2a348138006c8a97cbf335ac01d5b9 100644 (file)
--- a/GBRom.h
@@ -18,7 +18,7 @@
 #ifndef GBROM_H
 #define GBROM_H
 
-#include "sized_types.h"
+#include "../common/sized_types.h"
 #include <string>
 
 namespace cartridge_types {
similarity index 99%
rename from GBVideo.cc
rename to core/GBVideo.cc
index 2a6d4b13f79f7714af47b5daf546f1cb2d76b565..2fb2793824d492aa7b9e9038c02cfa370568c7fb 100644 (file)
@@ -16,8 +16,8 @@
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
 */ 
 #include "GBVideo.h"
-#include "gbcore.h"
-#include "Logger.h"
+#include "GameBoy.h"
+#include "../common/Logger.h"
 #include "util.h"
 #include <iostream>
 #include <algorithm>
similarity index 100%
rename from GBVideo.h
rename to core/GBVideo.h
similarity index 72%
rename from gbcore.cc
rename to core/GameBoy.cc
index 16c80426e0c04df2ce1879301b6bc54bd25d031a..a0d7045781b61039f66210b485300ccaff705fe5 100644 (file)
--- a/gbcore.cc
     You should have received a copy of the GNU General Public License
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
 */ 
-#include "gbcore.h"
 
+#include "GameBoy.h"
 #include "GBRom.h"
 #include "MBC.h"
-#include "Logger.h"
 #include "util.h"
-#include "wendi/disasm.h"
+#include "../common/Logger.h"
+#include "../common/toString.h"
+
 #include <sstream>
 #include <iomanip>
 #include <string>
@@ -1549,7 +1550,7 @@ GameBoy::run_status GameBoy::run()
 
 std::string GameBoy::status_string()
 {
-        Instruction ins(disassemble_opcode(*this, regs.PC));
+        Instruction ins(disassemble_opcode(regs.PC));
 
        std::ostringstream result;
        result << "t = " << std::dec << cycle_count << 
@@ -1604,3 +1605,536 @@ void GameBoy::update_JOYP()
        memory.high[GBMemory::I_JOYP]=JOYP;
 }
 
+
+//////////////////////////////////////////////////
+// OPCODE DISASSEMBLER
+//////////////////////////////////////////////////
+
+#include "disasm_macros.h"
+
+std::string GameBoy::get_port_name(int port)
+{
+       std::string port_name;
+
+       switch (port)
+       {
+               case 0x00: port_name = "JOYP"; break;
+               case 0x01: port_name = "SB  "; break;
+               case 0x02: port_name = "SC  "; break;
+               case 0x04: port_name = "DIV "; break;
+               case 0x05: port_name = "TIMA"; break;
+               case 0x06: port_name = "TMA "; break;
+               case 0x07: port_name = "TAC "; break;
+               case 0x10: port_name = "CH1_ENT";          break;
+               case 0x11: port_name = "CH1_WAVE";         break;
+               case 0x12: port_name = "CH1_ENV";          break;
+               case 0x13: port_name = "CH1_FREQ_LO";      break;
+               case 0x14: port_name = "CH1_FREQ_HI_KICK"; break;
+               case 0x16: port_name = "CH2_WAVE";         break;
+               case 0x17: port_name = "CH2_ENV";          break;
+               case 0x18: port_name = "CH2_FREQ_LO";      break;
+               case 0x19: port_name = "CH2_FREQ_HI_KICK"; break;
+               case 0x1A: port_name = "CH3_ONOFF";        break;
+               case 0x1C: port_name = "CH3_VOLUME";       break;
+               case 0x1D: port_name = "CH3_FREQ_LO";      break;
+               case 0x1E: port_name = "CH3_FREQ_HI_KICK"; break;
+               case 0x21: port_name = "CH4_ENV";          break;
+               case 0x22: port_name = "CH4_POLY";         break;
+               case 0x23: port_name = "CH4_KICK";         break;
+               case 0x24: port_name = "SND_VIN";          break;
+               case 0x25: port_name = "SND_STEREO";       break;
+               case 0x26: port_name = "SND_STAT";         break;
+               case 0x40: port_name = "LCDC"; break;
+               case 0x41: port_name = "STAT"; break;
+               case 0x42: port_name = "SCY "; break; 
+               case 0x43: port_name = "SCX "; break; 
+               case 0x44: port_name = "LY  "; break; 
+               case 0x45: port_name = "LYC "; break; 
+               case 0x4A: port_name = "WY  "; break; 
+               case 0x4B: port_name = "WX  "; break; 
+               case 0x47: port_name = "BGP "; break; 
+               case 0x48: port_name = "OBP0"; break; 
+               case 0x49: port_name = "OBP1"; break; 
+               case 0x46: port_name = "DMA "; break;
+               case 0x0F: port_name = "IF  "; break;
+               case 0xFF: port_name = "IE  "; break;
+               default:
+                                  if (port >= 0x80 && port <= 0xFE) 
+                                          port_name = "HRAM";
+                                  else if (port >= 0x30 && port <= 0x3F)
+                                          port_name = "Wave Pattern RAM";
+
+       }
+       return port_name;
+}
+
+
+Instruction GameBoy::disassemble_opcode(u16 addr)
+{
+       u8 opcode, sub_opcode=0xFF;
+       std::ostringstream result;
+       Instruction::InstructionType ins_type;
+       std::string                  opcode_str;
+       Instruction::Operand         op1, op2;
+
+       u16 PC = addr;
+       opcode = this->memory.read(PC++, GBMemory::DONT_WATCH);
+
+       result << std::hex << std::uppercase << std::setfill('0');
+
+       ins_type = Instruction::LOAD;
+
+       switch(opcode)
+       {
+               // LD n, nn
+               dis_for_each_register(0x3E, 0x06, 0x0E, 0x16, 0x1E, 0x26, 0x2E, "LD", Instruction::LOAD, dis_reg_inm)
+
+               // LD r1,r2
+               dis_for_each_register(0x7F, 0x78, 0x79, 0x7A, 0x7B, 0x7C, 0x7D, "LD", Instruction::LOAD, dis_A_reg)
+               dis_for_each_register(0x47, 0x40, 0x41, 0x42, 0x43, 0x44, 0x45, "LD", Instruction::LOAD, dis_B_reg)
+               dis_for_each_register(0x4F, 0x48, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, "LD", Instruction::LOAD, dis_C_reg)
+               dis_for_each_register(0x57, 0x50, 0x51, 0x52, 0x53, 0x54, 0x55, "LD", Instruction::LOAD, dis_D_reg)
+               dis_for_each_register(0x5F, 0x58, 0x59, 0x5A, 0x5B, 0x5C, 0x5D, "LD", Instruction::LOAD, dis_E_reg)
+               dis_for_each_register(0x67, 0x60, 0x61, 0x62, 0x63, 0x64, 0x65, "LD", Instruction::LOAD, dis_H_reg)
+               dis_for_each_register(0x6F, 0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, "LD", Instruction::LOAD, dis_L_reg)
+               
+               // LD reg, (HL)
+               dis_for_each_register(0x7E, 0x46, 0x4E, 0x56, 0x5E, 0x66, 0x6E, "LD", Instruction::LOAD, dis_reg__HL_)
+
+               // LD (HL), reg
+               dis_for_each_register(0x77, 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, "LD", Instruction::LOAD, dis__HL__reg)
+       
+               dis__reg16__inm(0x36, "LD", Instruction::LOAD, HL)
+               
+               dis_reg__reg16_(0x0A, "LD", Instruction::LOAD, A, BC)
+               dis_reg__reg16_(0x1A, "LD", Instruction::LOAD, A, DE)
+               dis_reg__inm_(0xFA, "LD", Instruction::LOAD, A)
+
+               dis__reg16__reg(0x02, "LD", Instruction::LOAD, BC, A)
+               dis__reg16__reg(0x12, "LD", Instruction::LOAD, DE, A)
+               dis__inm__reg(0xEA, "LD", Instruction::LOAD, A)
+
+               // LD A, (C)
+               case 0xF2:
+                       result << "LD A, (0xFF00+C)";
+                       opcode_str = "LDH";
+                       ins_type = Instruction::LOAD; 
+                       op1.str="A";
+                       op2.str="C";
+                       op1.type=Instruction::REG;
+                       op1.reg =Instruction::A;
+                       op2.type=Instruction::MEM_INDIRECT;
+                       op2.reg =Instruction::C;
+                       break;
+
+               // LD (C), A
+               case 0xE2:
+                       result << "LD (0xFF00+C), A";
+                       opcode_str = "LDH";
+                       ins_type = Instruction::LOAD; 
+                       op1.str="C";
+                       op2.str="A";
+                       op1.type=Instruction::MEM_INDIRECT;
+                       op1.reg =Instruction::C;
+                       op2.type=Instruction::REG;
+                       op2.reg =Instruction::A;
+                       break;
+
+               // LD A, (HLD); LD A, (HL-); LDD A,(HL);
+               case 0x3A:
+                       result << "LD A, (HL-)";
+                       opcode_str = "LDD";
+                       ins_type = Instruction::LOAD; 
+                       op1.str="A";
+                       op2.str="(HL)";
+                       op1.type=Instruction::REG;
+                       op1.reg =Instruction::A;
+                       op2.type=Instruction::MEM_INDIRECT;
+                       op2.reg =Instruction::HL;
+                       break;
+               // LD (HLD), A; LD (HL-), A; LDD (HL), A;
+               case 0x32:
+                       result << "LD (HL-), A";
+                       opcode_str = "LDD";
+                       ins_type = Instruction::LOAD; 
+                       op1.str="(HL)";
+                       op2.str="A";
+                       op1.type=Instruction::MEM_INDIRECT;
+                       op1.reg =Instruction::HL;
+                       op2.type=Instruction::REG;
+                       op2.reg =Instruction::A;
+                       break;
+               // LD A, (HLI); LD A, (HL+); LDI A, (HL);
+               case 0x2A:
+                       result << "LD A, (HL+)";
+                       opcode_str = "LDI";
+                       ins_type = Instruction::LOAD; 
+                       op1.str="A";
+                       op2.str="(HL)";
+                       op1.type=Instruction::REG;
+                       op1.reg =Instruction::A;
+                       op2.type=Instruction::MEM_INDIRECT;
+                       op2.reg =Instruction::HL;
+                       break;
+               // LD (HLI), A; LD (HL+), A; LDI (HL), A;
+               case 0x22:
+                       result << "LD (HL+), A";
+                       opcode_str = "LDI";
+                       ins_type = Instruction::LOAD; 
+                       op1.str="(HL)";
+                       op2.str="A";
+                       op1.type=Instruction::MEM_INDIRECT;
+                       op1.reg =Instruction::HL;
+                       op2.type=Instruction::REG;
+                       op2.reg =Instruction::A;
+                       break;
+
+               // LDH (n), A
+               case 0xE0: {
+                       int port = int(this->memory.read(PC++, GBMemory::DONT_WATCH));
+                       
+                       result << "LD (0xFF" << 
+                                       std::setw(2) << port << "), A" << "\t[" << get_port_name(port) << "]";
+                       opcode_str = "LDH";
+                       ins_type = Instruction::LOAD; 
+                       op1.str=std::string("(") + toString(port) + ")";
+                       op2.str="A";
+                       op1.type=Instruction::MEM_DIRECT;
+                       op1.val =0xFF00+port;
+                       op2.type=Instruction::REG;
+                       op2.reg =Instruction::A;
+                       break;
+               }
+               // LDH A, (n)
+               case 0xF0: {
+                       int port = int(this->memory.read(PC++, GBMemory::DONT_WATCH));
+                       result << "LD A, (0xFF" << 
+                                       std::setw(2) << port << ")" << "\t[" << get_port_name(port) << "]";
+                       opcode_str = "LDH";
+                       ins_type = Instruction::LOAD; 
+                       op1.str="A";
+                       op2.str=std::string("(") + toString(port) + ")";
+                       op1.type=Instruction::REG;
+                       op1.reg =Instruction::A;
+                       op2.type=Instruction::MEM_DIRECT;
+                       op2.val =0xFF00+port;
+                       break;
+               }
+
+               dis_reg16_inm(0x01, "LD", Instruction::LOAD, BC)
+               dis_reg16_inm(0x11, "LD", Instruction::LOAD, DE)
+               dis_reg16_inm(0x21, "LD", Instruction::LOAD, HL)
+               dis_reg16_inm(0x31, "LD", Instruction::LOAD, SP)
+               
+               // LD SP, HL
+               case 0xF9:
+                       result << "LD SP, HL";
+                       opcode_str = "LD";
+                       ins_type = Instruction::LOAD; 
+                       op1.str = "SP";
+                       op2.str = "HL";
+                       op1.type = Instruction::REG;
+                       op1.reg  = Instruction::SP;
+                       op2.type = Instruction::REG;
+                       op2.reg  = Instruction::HL;
+
+               // LD HL, SP+n
+               // LDHL SP, n
+               case 0xF8: {
+                       int n = int(this->memory.read(PC++, GBMemory::DONT_WATCH));
+                       result << "LD HL, SP + 0x"<< std::setw(2) << n;
+                       opcode_str = "LD HL, SP+";
+                       ins_type = Instruction::LOAD; 
+                       op1.str = toString(n);
+                       op2.str = "";
+                       op1.type = Instruction::INM8;
+                       op1.val = n;
+                       break; 
+               }
+
+               // LD (nn), SP
+               dis__inm__reg16(0x08, "LD", Instruction::LOAD, SP)
+
+               // PUSH nn
+               dis_reg16(0xF5, "PUSH", Instruction::OTHER, AF)
+               dis_reg16(0xC5, "PUSH", Instruction::OTHER, BC)
+               dis_reg16(0xD5, "PUSH", Instruction::OTHER, DE)
+               dis_reg16(0xE5, "PUSH", Instruction::OTHER, HL)
+
+               // POP nn
+               dis_reg16(0xF1, "POP", Instruction::OTHER, AF)
+               dis_reg16(0xC1, "POP", Instruction::OTHER, BC)
+               dis_reg16(0xD1, "POP", Instruction::OTHER, DE)
+               dis_reg16(0xE1, "POP", Instruction::OTHER, HL)
+
+               // 8-bit ALU
+               // ADD A,reg
+               dis_for_each_register(0x87, 0x80, 0x81, 0x82, 0x83, 0x84, 0x85, "ADD", Instruction::ALU, dis_A_reg)
+
+               dis_reg__reg16_(0x86, "ADD", Instruction::ALU, A, HL)
+               dis_reg_inm(0xC6, "ADD", Instruction::ALU, A)
+               
+               // ADC A, n
+               dis_for_each_register(0x8F, 0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, "ADC", Instruction::ALU, dis_A_reg)
+               
+               dis_reg__reg16_(0x8E, "ADC", Instruction::ALU, A, HL)
+               dis_reg_inm(0xCE, "ADC", Instruction::ALU, A)
+
+               // SUB n
+               dis_for_each_register(0x97, 0x90, 0x91, 0x92, 0x93, 0x94, 0x95, "SUB", Instruction::ALU, dis_reg)
+               
+               dis__reg16_(0x96, "SUB", Instruction::ALU, HL)
+               dis_inm8(0xD6, "SUB", Instruction::ALU) 
+               
+               // SBC n
+               dis_for_each_register(0x9F, 0x98, 0x99, 0x9A, 0x9B, 0x9C, 0x9D, "SBC", Instruction::ALU, dis_reg)
+
+               dis__reg16_(0x9E, "SBC", Instruction::ALU, HL)
+               dis_reg_inm(0xDE, "SBC", Instruction::ALU, A)
+
+               // AND n
+               dis_for_each_register(0xA7, 0xA0, 0xA1, 0xA2, 0xA3, 0xA4, 0xA5, "AND", Instruction::ALU, dis_reg)
+
+               dis__reg16_(0xA6, "AND", Instruction::ALU, HL)
+               dis_inm8(0xE6, "AND", Instruction::ALU)
+               
+               // OR n
+               dis_for_each_register(0xB7, 0xB0, 0xB1, 0xB2, 0xB3, 0xB4, 0xB5, "OR", Instruction::ALU, dis_reg)
+               
+               dis__reg16_(0xB6, "OR", Instruction::ALU, HL)
+               dis_inm8(0xF6, "OR", Instruction::ALU)
+
+               // XOR n
+               dis_for_each_register(0xAF, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAD, "XOR", Instruction::ALU, dis_reg)
+
+               dis__reg16_(0xAE, "XOR", Instruction::ALU, HL)
+               dis_inm8(0xEE, "XOR", Instruction::ALU)
+               
+               // CP n
+               dis_for_each_register(0xBF, 0xB8, 0xB9, 0xBA, 0xBB, 0xBC, 0xBD, "CP", Instruction::ALU, dis_reg)
+               
+               dis__reg16_(0xBE, "CP", Instruction::ALU, HL)
+               dis_inm8(0xFE, "CP", Instruction::ALU)
+
+               // INC n
+               dis_for_each_register(0x3C, 0x04, 0x0C, 0x14, 0x1C, 0x24, 0x2C, "INC", Instruction::ALU, dis_reg)
+
+               dis__reg16_(0x34, "INC", Instruction::ALU, HL)
+               
+               // DEC n
+               dis_for_each_register(0x3D, 0x05, 0x0D, 0x15, 0x1D, 0x25, 0x2D, "DEC", Instruction::ALU, dis_reg)
+
+               dis__reg16_(0x35, "DEC", Instruction::ALU, HL)
+               
+               // 16-bit ALU
+               // ADD HL, n
+               dis_for_each_register16(0x09, 0x19, 0x29, 0x39, "ADD", Instruction::ALU, dis_HL_reg16)
+
+               // ADD SP, #
+               dis_reg16_inm8(0xE8, "ADD", Instruction::ALU, SP)
+               
+               // INC nn
+               dis_for_each_register16(0x03, 0x13, 0x23, 0x33, "INC", Instruction::ALU, dis_reg16)
+
+               // DEC nn
+               dis_for_each_register16(0x0B, 0x1B, 0x2B, 0x3B, "DEC", Instruction::ALU, dis_reg16)
+
+               // Miscellaneous instructions
+               case 0xCB: {
+                       sub_opcode = this->memory.read(PC++, GBMemory::DONT_WATCH);
+                       switch(sub_opcode)
+                       {
+                               // SWAP n
+                               dis_for_each_register(0x37, 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, "SWAP", Instruction::ALU, dis_reg)
+
+                               // SWAP (HL)
+                               dis__reg16_(0x36, "SWAP", Instruction::ALU, HL)
+
+                               // RLC n
+                               dis_for_each_register(0x07, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, "RLC", Instruction::ALU, dis_reg)
+
+                               // RLC (HL)
+                               dis__reg16_(0x06, "RLC", Instruction::ALU, HL)
+
+                               // RL n (through carry)
+                               dis_for_each_register(0x17, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, "RL", Instruction::ALU, dis_reg)
+
+                               // RL (HL) (through carry)
+                               dis__reg16_(0x16, "RL", Instruction::ALU, HL)
+
+                               // RRC n
+                               dis_for_each_register(0x0F, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, "RRC", Instruction::ALU, dis_reg)
+
+                               // RRC (HL)
+                               dis__reg16_(0x0E, "RRC", Instruction::ALU, HL)
+
+                               // RR n (through carry)
+                               dis_for_each_register(0x1F, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, "RR", Instruction::ALU, dis_reg)
+
+                               // RR (HL) (through carry)
+                               dis__reg16_(0x1E, "RR", Instruction::ALU, HL)
+
+                               // SLA n
+                               dis_for_each_register(0x27, 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, "SLA", Instruction::ALU, dis_reg)
+
+                               // SLA (HL)
+                               dis__reg16_(0x26, "SLA", Instruction::ALU, HL)
+
+                               // SRA n
+                               dis_for_each_register(0x2F, 0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, "SRA", Instruction::ALU, dis_reg)
+                               
+                               // SRA (HL)
+                               dis__reg16_(0x2E, "SRA", Instruction::ALU, HL)
+
+                               // SRL n
+                               dis_for_each_register(0x3F, 0x38, 0x39, 0x3A, 0x3B, 0x3C, 0x3D, "SRA", Instruction::ALU, dis_reg)
+                               
+                               // SRL (HL)
+                               dis__reg16_(0x3E, "SRA", Instruction::ALU, HL)
+
+                               default: {
+                                       int bit_op = sub_opcode >> 6;
+                                       int reg = sub_opcode & 7;
+                                       int b   = (sub_opcode >> 3) & 7;
+                                       const char *bit_ops[4]={"Unknown", "BIT", "RES", "SET"};
+                                       const char *regs[8]={"B","C","D","E","H","L","(HL)", "A"};
+                                       result << bit_ops[bit_op] << " " << b << ", " << regs[reg];
+                                       opcode_str = bit_ops[bit_op];
+                                       ins_type = Instruction::ALU;
+                                       op1.str = toString(b);
+                                       op1.type = Instruction::INM8;
+                                       op1.val = b;
+                                       op2.str = regs[reg];
+                                       switch(reg)
+                                       {
+                                               case 0:
+                                                       op2.type = Instruction::REG;
+                                                       op2.reg  = Instruction::B;
+                                                       break;
+                                               case 1:
+                                                       op2.type = Instruction::REG;
+                                                       op2.reg  = Instruction::C;
+                                                       break;
+                                               case 2:
+                                                       op2.type = Instruction::REG;
+                                                       op2.reg  = Instruction::D;
+                                                       break;
+                                               case 3:
+                                                       op2.type = Instruction::REG;
+                                                       op2.reg  = Instruction::E;
+                                                       break;
+                                               case 4:
+                                                       op2.type = Instruction::REG;
+                                                       op2.reg  = Instruction::H;
+                                                       break;
+                                               case 5:
+                                                       op2.type = Instruction::REG;
+                                                       op2.reg  = Instruction::L;
+                                                       break;
+                                               case 6:
+                                                       op2.type = Instruction::MEM_INDIRECT;
+                                                       op2.reg  = Instruction::HL;
+                                                       break;
+                                               case 7:
+                                                       op2.type = Instruction::REG;
+                                                       op2.reg  = Instruction::A;
+                                                       break;
+                                       }
+                                       break;
+                               }
+                               
+                       }
+                       break;
+               }
+
+               dis(0x27, "DAA", Instruction::ALU)
+
+               dis(0x2F, "CPL", Instruction::ALU)
+
+               dis(0x3F, "CCF", Instruction::ALU)
+
+               dis(0x37, "SCF", Instruction::ALU)
+               dis(0x00, "NOP", Instruction::OTHER)
+               dis(0x76, "HALT", Instruction::OTHER)
+
+               dis(0x10, "STOP", Instruction::OTHER)
+               
+               dis(0xF3, "DI", Instruction::OTHER)
+
+               dis(0xFB, "EI", Instruction::OTHER)
+                          
+               // Rotates and shifts
+               dis(0x07, "RLCA", Instruction::ALU)
+
+               dis(0x17, "RLA", Instruction::ALU)
+               
+               dis(0x0F, "RRCA", Instruction::ALU)
+
+               dis(0x1F, "RRA", Instruction::ALU)
+               
+               // Jumps
+               dis_inm16(0xC3, "JP", Instruction::UNCONDITIONAL_JUMP)
+               // JP cc, nn
+               dis_inm16(0xC2, "JP NZ", Instruction::CONDITIONAL_JUMP)
+               dis_inm16(0xCA, "JP Z",  Instruction::CONDITIONAL_JUMP)
+               dis_inm16(0xD2, "JP NC", Instruction::CONDITIONAL_JUMP)
+               dis_inm16(0xDA, "JP C",  Instruction::CONDITIONAL_JUMP)
+               case 0xE9:
+                       result << "JP (HL)";
+                       opcode_str = "JP";
+                       ins_type = Instruction::UNCONDITIONAL_JUMP; 
+                       op1.str="(HL)";
+                       op2.str="";
+                       op1.type = Instruction::MEM_INDIRECT;
+                       op1.reg = Instruction::HL;
+                       op2.type = Instruction::NONE;
+                       break;
+               
+               dis_JR(0x18, "JR", Instruction::UNCONDITIONAL_JUMP)
+               dis_JR(0x20, "JR NZ", Instruction::CONDITIONAL_JUMP)
+               dis_JR(0x28, "JR Z",  Instruction::CONDITIONAL_JUMP)
+               dis_JR(0x30, "JR NC", Instruction::CONDITIONAL_JUMP)
+               dis_JR(0x38, "JR C",  Instruction::CONDITIONAL_JUMP)
+
+               // Calls
+               dis_inm16(0xCD, "CALL", Instruction::CALL)
+               // CALL cc, nn
+               dis_inm16(0xC4, "CALL NZ", Instruction::CALL)
+               dis_inm16(0xCC, "CALL Z", Instruction::CALL)
+               dis_inm16(0xD4, "CALL NC", Instruction::CALL)
+               dis_inm16(0xDC, "CALL C", Instruction::CALL)
+
+               // Restarts
+               dis(0xC7, "RST 0x00", Instruction::RESET)
+               dis(0xCF, "RST 0x08", Instruction::RESET)
+               dis(0xD7, "RST 0x10", Instruction::RESET)
+               dis(0xDF, "RST 0x18", Instruction::RESET)
+               dis(0xE7, "RST 0x20", Instruction::RESET)
+               dis(0xEF, "RST 0x28", Instruction::RESET)
+               dis(0xF7, "RST 0x30", Instruction::RESET)
+               dis(0xFF, "RST 0x38", Instruction::RESET)
+
+               // Returns
+               dis(0xC9, "RET", Instruction::UNCONDITIONAL_RET)
+               // RET cc
+               dis(0xC0, "RET NZ", Instruction::CONDITIONAL_RET)
+               dis(0xC8, "RET Z",  Instruction::CONDITIONAL_RET)
+               dis(0xD0, "RET NC", Instruction::CONDITIONAL_RET)
+               dis(0xD8, "RET C",  Instruction::CONDITIONAL_RET)
+
+               dis(0xD9, "RETI", Instruction::UNCONDITIONAL_RET)
+
+               default:
+                       std::ostringstream errmsg;
+                       errmsg << "Unknown opcode 0x";
+                       errmsg << std::hex << std::setw(2) << std::setfill('0') << opcode;
+                       errmsg << " at 0x" << std::hex << std::setw(4) << PC-1;
+                       logger.trace(errmsg.str());
+                       break;
+
+       } // end switch
+
+       return Instruction(PC-addr, ins_type, opcode, sub_opcode,
+                       result.str(), opcode_str, op1, op2);
+}
+
similarity index 95%
rename from gbcore.h
rename to core/GameBoy.h
index 8d7b328e0108de9d9898b6e56fc1ef6221ccf8ab..27381e238d2b02177cc50b8b64635fd9fe6e0e98 100644 (file)
--- a/gbcore.h
 #ifndef GBCORE_H
 #define GBCORE_H
 
-#include "sized_types.h"
-#include "GBMemory.h"
-#include "GBVideo.h"
-#include "Logger.h"
 #include <iomanip>
 #include <string>
 #include <map>
 
+#include "GBMemory.h"
+#include "GBVideo.h"
+#include "Instruction.h"
+#include "../common/sized_types.h"
+#include "../common/Logger.h"
+
 union GBRom;
 
 class GameBoy
@@ -149,6 +151,8 @@ class GameBoy
        void disable_breakpoint(int id);
 
        std::string status_string();
+       Instruction disassemble_opcode(u16 addr);
+       static std::string get_port_name(int port);
 
        // prevent object copying
        private:
similarity index 98%
rename from wendi/Instruction.h
rename to core/Instruction.h
index f7d53164242b424069ec163e60865457134e43d4..21e48f88739c306ef125c3469838e6b1870ea68c 100644 (file)
@@ -18,7 +18,7 @@
 #ifndef INSTRUCTION_H
 #define INSTRUCTION_H
 
-#include "../sized_types.h"
+#include "../common/sized_types.h"
 
 struct Instruction
 {
similarity index 97%
rename from MBC.cc
rename to core/MBC.cc
index 07af1b5f3d9d1f4026a04c76335b5470d836f8e4..689bb0e4f844eb7ecc65c7fe70aa6f3ee643c61b 100644 (file)
--- a/MBC.cc
@@ -19,7 +19,7 @@
 #include "NoMBC.h"
 #include "MBC1.h"
 
-#include "Logger.h"
+#include "../common/Logger.h"
 
 MBC *create_MBC(GBRom *rom)
 {
similarity index 96%
rename from MBC.h
rename to core/MBC.h
index b6786772a7dcccc97933b2e56526d7cdaba9eae4..c265589f2ecbad0a5a0fbb7564b868547545472f 100644 (file)
--- a/MBC.h
@@ -18,7 +18,7 @@
 #ifndef MBC_H
 #define MBC_H
 
-#include "sized_types.h"
+#include "../common/sized_types.h"
 #include "GBRom.h"
 
 class MBC
similarity index 98%
rename from MBC1.cc
rename to core/MBC1.cc
index dfea2bc40832cad7a7c8fd245c752b46d9d56277..f3a48b03b5e48784671d8925b598f56f19d302f3 100644 (file)
--- a/MBC1.cc
@@ -16,7 +16,7 @@
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
 */ 
 #include "MBC1.h"
-#include "Logger.h"
+#include "../common/Logger.h"
 
 #include <iomanip>
 #include <cassert>
similarity index 100%
rename from MBC1.h
rename to core/MBC1.h
similarity index 98%
rename from NoMBC.cc
rename to core/NoMBC.cc
index 1bdbd0b4bb6d149688e6d8602c578267fc4d7e5f..2bf5c8ec8df2a56c05dd6f0dac1aeedb78eb259b 100644 (file)
--- a/NoMBC.cc
@@ -16,7 +16,7 @@
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
 */ 
 #include "NoMBC.h"
-#include "Logger.h"
+#include "../common/Logger.h"
 
 #include <iomanip>
 
similarity index 100%
rename from NoMBC.h
rename to core/NoMBC.h
similarity index 88%
rename from wendi/disasm_macros.h
rename to core/disasm_macros.h
index 2721e5308cdb9bc9db2c17a2ab7fe97b3aaefafc..54a203efbecd2c28344d893335847a5a4429e35a 100644 (file)
@@ -16,8 +16,7 @@
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
 */
 
-// Macros to avoid unnecesary repetition in disasm.cc (like opcodes.h for
-// gbcore)
+// Macros to avoid unnecesary repetition in GameBoy.cc (disassembler)
 
 #ifndef DISASM_MACROS_H
 #define DISASM_MACROS_H
 // OP inm8
 #define dis_inm8(opcode, name, itype) \
        case opcode: {\
-               int inm = int(gb.memory.read(PC++, GBMemory::DONT_WATCH)); \
+               int inm = int(this->memory.read(PC++, GBMemory::DONT_WATCH)); \
                result << name << " 0x" << std::setw(2) << inm; \
                opcode_str = name; \
                ins_type = itype; \
-               op1.str  = ToString(inm); \
+               op1.str  = toString(inm); \
                op2.str  = ""; \
                op1.type = Instruction::INM8; \
                op1.val  = inm; \
 // OP inm16
 #define dis_inm16(opcode, name, itype) \
        case opcode: {\
-               int inm = int(gb.memory.read16(PC, GBMemory::DONT_WATCH)); \
+               int inm = int(this->memory.read16(PC, GBMemory::DONT_WATCH)); \
         PC += 2; \
                result << name << " 0x" << std::setw(4) << inm; \
                opcode_str = name; \
                ins_type = itype; \
-               op1.str  = ToString(inm); \
+               op1.str  = toString(inm); \
                op2.str  = ""; \
                op1.type = Instruction::INM16; \
                op1.val  = inm; \
 
 #define dis_reg_inm(opcode, name, itype, reg8) \
        case opcode: {\
-               int inm = int(gb.memory.read(PC++, GBMemory::DONT_WATCH)); \
+               int inm = int(this->memory.read(PC++, GBMemory::DONT_WATCH)); \
                result << name << " " << #reg8 << ", 0x" << std::setw(2) << inm; \
                opcode_str = name; \
                ins_type = itype; \
                op1.str  = #reg8; \
-               op2.str  = ToString(inm); \
+               op2.str  = toString(inm); \
                op1.type = Instruction::REG; \
                op1.reg  = Instruction::reg8; \
                op2.type = Instruction::INM8; \
 
 #define dis_reg16_inm(opcode, name, itype, reg16) \
        case opcode: {\
-               int inm = int(gb.memory.read16(PC, GBMemory::DONT_WATCH)); \
+               int inm = int(this->memory.read16(PC, GBMemory::DONT_WATCH)); \
                PC += 2; \
                result << name << " " << #reg16 << ", 0x" << std::setw(4) << inm; \
                opcode_str = name; \
                ins_type = itype; \
                op1.str  = #reg16; \
-               op2.str  = ToString(inm); \
+               op2.str  = toString(inm); \
                op1.type = Instruction::REG; \
                op1.reg  = Instruction::reg16; \
                op2.type = Instruction::INM8; \
 
 #define dis_reg16_inm8(opcode, name, itype, reg16) \
        case opcode: {\
-               int inm = int(gb.memory.read(PC++, GBMemory::DONT_WATCH)); \
+               int inm = int(this->memory.read(PC++, GBMemory::DONT_WATCH)); \
                result << name << " " << #reg16 << ", 0x" << std::setw(2) << inm; \
                opcode_str = name; \
                ins_type = itype; \
                op1.str  = #reg16; \
-               op2.str  = ToString(inm); \
+               op2.str  = toString(inm); \
                op1.type = Instruction::REG; \
                op1.reg  = Instruction::reg16; \
                op2.type = Instruction::INM8; \
 // OP reg, (inm)
 #define dis_reg__inm_(opcode, name, itype, reg8) \
        case opcode: {\
-               int inm = int(gb.memory.read16(PC, GBMemory::DONT_WATCH)); \
+               int inm = int(this->memory.read16(PC, GBMemory::DONT_WATCH)); \
         PC += 2; \
                result << name << " " << #reg8 << ", (0x" << \
                                std::setw(4) << inm << ")"; \
                opcode_str = name; \
                ins_type = itype; \
                op1.str  = #reg8; \
-               op2.str  = std::string("(") + ToString(inm) + ")"; \
+               op2.str  = std::string("(") + toString(inm) + ")"; \
                op1.type = Instruction::REG; \
                op1.reg  = Instruction::reg8; \
                op2.type = Instruction::MEM_DIRECT; \
 // OP (inm), reg
 #define dis__inm__reg(opcode, name, itype, reg8) \
        case opcode: {\
-               int inm = int(gb.memory.read16(PC, GBMemory::DONT_WATCH)); \
+               int inm = int(this->memory.read16(PC, GBMemory::DONT_WATCH)); \
         PC += 2; \
                result << name << " (0x" << \
                                std::setw(4) << inm << "), " << #reg8; \
                opcode_str = name; \
                ins_type = itype; \
-               op1.str  = std::string("(") + ToString(inm) + ")"; \
+               op1.str  = std::string("(") + toString(inm) + ")"; \
                op2.str  = #reg8; \
                op1.type = Instruction::MEM_DIRECT; \
                op1.val  = inm; \
 // OP (inm), reg16
 #define dis__inm__reg16(opcode, name, itype, reg16) \
        case opcode: {\
-               int inm = int(gb.memory.read16(PC, GBMemory::DONT_WATCH)); \
+               int inm = int(this->memory.read16(PC, GBMemory::DONT_WATCH)); \
         PC += 2; \
                result << name << " (0x" << \
                                std::setw(4) << inm << "), " << #reg16; \
                opcode_str = name; \
                ins_type = itype; \
-               op1.str  = std::string("(") + ToString(inm) + ")"; \
+               op1.str  = std::string("(") + toString(inm) + ")"; \
                op2.str  = #reg16; \
                op1.type = Instruction::MEM_DIRECT; \
                op1.val  = inm; \
 // OP (reg16), inm
 #define dis__reg16__inm(opcode, name, itype, reg16) \
        case opcode: {\
-               int inm = int(gb.memory.read(PC++, GBMemory::DONT_WATCH)); \
+               int inm = int(this->memory.read(PC++, GBMemory::DONT_WATCH)); \
                result << name << " (" << #reg16 << "), 0x" << \
                                std::setw(2) << inm; \
                opcode_str = name; \
                ins_type = itype; \
                op1.str  = std::string("(") + #reg16 + ")"; \
-               op2.str  = ToString(inm); \
+               op2.str  = toString(inm); \
                op1.type = Instruction::MEM_INDIRECT; \
                op1.reg  = Instruction::reg16; \
                op2.type = Instruction::INM8; \
 // Special routine for JR
 #define dis_JR(opcode, name, itype) \
        case opcode: { \
-               s8 offset = gb.memory.read(PC++); \
+               s8 offset = this->memory.read(PC++); \
                result << name << " " << std::dec << int(offset) << "\t[0x" \
                                 << std::hex << std::setw(2) << int(PC+offset) << "]"; \
                opcode_str = name; \
                ins_type = itype; \
-               op1.str  = ToString(int(offset)); \
-               op2.str  = ToString(int(PC+offset)); \
+               op1.str  = toString(int(offset)); \
+               op2.str  = toString(int(PC+offset)); \
                op1.type = Instruction::INM8; \
                op1.val  = offset; \
                op2.type = Instruction::NONE; \
similarity index 99%
rename from opcodes.h
rename to core/opcodes.h
index f2507d11a6d27a2a2915dc16d1550d56b49dd559..6ed1d2c5f83ba64292c770711eeaa1713f72b664 100644 (file)
--- a/opcodes.h
@@ -16,7 +16,7 @@
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
 */ 
 
-// Macro definitions to avoid unnecesary repetition in gbcore.cc
+// Macro definitions to avoid unnecesary repetition in GameBoy.cc
 
 #ifndef OPCODES_H
 #define OPCODES_H
similarity index 100%
rename from util.cc
rename to core/util.cc
similarity index 96%
rename from util.h
rename to core/util.h
index 576587804f85ed7afd1d8f0dee0316eada9457a5..b06b4d71b19d0499301bf55ffc4b57178163a56b 100644 (file)
--- a/util.h
@@ -18,7 +18,7 @@
 #ifndef UTIL_H
 #define UTIL_H
 
-#include "sized_types.h"
+#include "../common/sized_types.h"
 
 uint32 set_bit(uint32 val, uint32 pos);
 uint32 reset_bit(uint32 val, uint32 pos);
diff --git a/wenboi/CMakeLists.txt b/wenboi/CMakeLists.txt
new file mode 100644 (file)
index 0000000..de03cf0
--- /dev/null
@@ -0,0 +1,4 @@
+ADD_EXECUTABLE(wenboi wenboi.cc)
+TARGET_LINK_LIBRARIES(wenboi wenboicore ${SDL_LIBRARY})
+INCLUDE_DIRECTORIES(${SDL_INCLUDE_DIR})
+
similarity index 98%
rename from tests/test_core.cc
rename to wenboi/wenboi.cc
index c6b60d10ebbfd5e23aad93a6f910411b768e0b6f..e6c979b799b1a9d91f516bf7e70100c1d2d407d4 100644 (file)
@@ -15,9 +15,8 @@
     You should have received a copy of the GNU General Public License
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
 */
-#include "../gbcore.h"
-#include "../Logger.h"
-#include "../wendi/disasm.h"
+#include "../core/GameBoy.h"
+#include "../common/Logger.h"
 #include <iostream>
 #include <iomanip>
 #include <sstream>
@@ -181,7 +180,7 @@ int main(int argc, char **argv)
                        pos = start;
                        while (pos < end)
                        {
-                               Instruction ins(disassemble_opcode(gb, pos));
+                               Instruction ins(gb.disassemble_opcode(pos));
                                cout << "0x" << std::hex << std::setw(4) << std::setfill('0') << 
                                                pos << "\t";
                                for (int i=0; i<ins.length; i++)
diff --git a/wendi/CMakeLists.txt b/wendi/CMakeLists.txt
new file mode 100644 (file)
index 0000000..94314d9
--- /dev/null
@@ -0,0 +1,4 @@
+ADD_EXECUTABLE(wendi wendi.cc CodeBlock.cc output_graph.cc output_txt.cc)
+TARGET_LINK_LIBRARIES(wendi ${SDL_LIBRARY} wenboicore)
+INCLUDE_DIRECTORIES(${SDL_INCLUDE_DIR})
+
index 6539cd96c50b9a80ef265443c626c2b59e8cdfc3..2b484443246034149180bad4d3b14eb6b43fd392 100644 (file)
@@ -22,8 +22,8 @@
 #include <utility>
 #include <string>
 #include <list>
-#include "Instruction.h"
-#include "../sized_types.h"
+#include "../core/Instruction.h"
+#include "../common/sized_types.h"
 
 typedef u16 address;
 
diff --git a/wendi/disasm.cc b/wendi/disasm.cc
deleted file mode 100644 (file)
index 888e385..0000000
+++ /dev/null
@@ -1,549 +0,0 @@
-/*
-    Copyright 2008 Jorge Gorbe Moya <slack@codemaniacs.com>
-
-    This file is part of wenboi 
-
-    wenboi is free software: you can redistribute it and/or modify it under the
-    terms of the GNU General Public License version 3 only, as published by the
-    Free Software Foundation.
-
-    wenboi 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 wenboi.  If not, see <http://www.gnu.org/licenses/>.
-*/
-#include <iomanip>
-
-#include "../Logger.h"
-#include "disasm.h"
-#include "disasm_macros.h"
-
-std::string get_port_name(int port)
-{
-       std::string port_name;
-
-       switch (port)
-       {
-               case 0x00: port_name = "JOYP"; break;
-               case 0x01: port_name = "SB  "; break;
-               case 0x02: port_name = "SC  "; break;
-               case 0x04: port_name = "DIV "; break;
-               case 0x05: port_name = "TIMA"; break;
-               case 0x06: port_name = "TMA "; break;
-               case 0x07: port_name = "TAC "; break;
-               case 0x10: port_name = "CH1_ENT";          break;
-               case 0x11: port_name = "CH1_WAVE";         break;
-               case 0x12: port_name = "CH1_ENV";          break;
-               case 0x13: port_name = "CH1_FREQ_LO";      break;
-               case 0x14: port_name = "CH1_FREQ_HI_KICK"; break;
-               case 0x16: port_name = "CH2_WAVE";         break;
-               case 0x17: port_name = "CH2_ENV";          break;
-               case 0x18: port_name = "CH2_FREQ_LO";      break;
-               case 0x19: port_name = "CH2_FREQ_HI_KICK"; break;
-               case 0x1A: port_name = "CH3_ONOFF";        break;
-               case 0x1C: port_name = "CH3_VOLUME";       break;
-               case 0x1D: port_name = "CH3_FREQ_LO";      break;
-               case 0x1E: port_name = "CH3_FREQ_HI_KICK"; break;
-               case 0x21: port_name = "CH4_ENV";          break;
-               case 0x22: port_name = "CH4_POLY";         break;
-               case 0x23: port_name = "CH4_KICK";         break;
-               case 0x24: port_name = "SND_VIN";          break;
-               case 0x25: port_name = "SND_STEREO";       break;
-               case 0x26: port_name = "SND_STAT";         break;
-               case 0x40: port_name = "LCDC"; break;
-               case 0x41: port_name = "STAT"; break;
-               case 0x42: port_name = "SCY "; break; 
-               case 0x43: port_name = "SCX "; break; 
-               case 0x44: port_name = "LY  "; break; 
-               case 0x45: port_name = "LYC "; break; 
-               case 0x4A: port_name = "WY  "; break; 
-               case 0x4B: port_name = "WX  "; break; 
-               case 0x47: port_name = "BGP "; break; 
-               case 0x48: port_name = "OBP0"; break; 
-               case 0x49: port_name = "OBP1"; break; 
-               case 0x46: port_name = "DMA "; break;
-               case 0x0F: port_name = "IF  "; break;
-               case 0xFF: port_name = "IE  "; break;
-               default:
-                                  if (port >= 0x80 && port <= 0xFE) 
-                                          port_name = "HRAM";
-                                  else if (port >= 0x30 && port <= 0x3F)
-                                          port_name = "Wave Pattern RAM";
-
-       }
-       return port_name;
-}
-
-
-Instruction disassemble_opcode(GameBoy &gb, u16 addr)
-{
-       u8 opcode, sub_opcode=0xFF;
-       std::ostringstream result;
-       Instruction::InstructionType ins_type;
-       std::string                  opcode_str;
-       Instruction::Operand         op1, op2;
-
-       u16 PC = addr;
-       opcode = gb.memory.read(PC++, GBMemory::DONT_WATCH);
-
-       result << std::hex << std::uppercase << std::setfill('0');
-
-       ins_type = Instruction::LOAD;
-
-       switch(opcode)
-       {
-               // LD n, nn
-               dis_for_each_register(0x3E, 0x06, 0x0E, 0x16, 0x1E, 0x26, 0x2E, "LD", Instruction::LOAD, dis_reg_inm)
-
-               // LD r1,r2
-               dis_for_each_register(0x7F, 0x78, 0x79, 0x7A, 0x7B, 0x7C, 0x7D, "LD", Instruction::LOAD, dis_A_reg)
-               dis_for_each_register(0x47, 0x40, 0x41, 0x42, 0x43, 0x44, 0x45, "LD", Instruction::LOAD, dis_B_reg)
-               dis_for_each_register(0x4F, 0x48, 0x49, 0x4A, 0x4B, 0x4C, 0x4D, "LD", Instruction::LOAD, dis_C_reg)
-               dis_for_each_register(0x57, 0x50, 0x51, 0x52, 0x53, 0x54, 0x55, "LD", Instruction::LOAD, dis_D_reg)
-               dis_for_each_register(0x5F, 0x58, 0x59, 0x5A, 0x5B, 0x5C, 0x5D, "LD", Instruction::LOAD, dis_E_reg)
-               dis_for_each_register(0x67, 0x60, 0x61, 0x62, 0x63, 0x64, 0x65, "LD", Instruction::LOAD, dis_H_reg)
-               dis_for_each_register(0x6F, 0x68, 0x69, 0x6A, 0x6B, 0x6C, 0x6D, "LD", Instruction::LOAD, dis_L_reg)
-               
-               // LD reg, (HL)
-               dis_for_each_register(0x7E, 0x46, 0x4E, 0x56, 0x5E, 0x66, 0x6E, "LD", Instruction::LOAD, dis_reg__HL_)
-
-               // LD (HL), reg
-               dis_for_each_register(0x77, 0x70, 0x71, 0x72, 0x73, 0x74, 0x75, "LD", Instruction::LOAD, dis__HL__reg)
-       
-               dis__reg16__inm(0x36, "LD", Instruction::LOAD, HL)
-               
-               dis_reg__reg16_(0x0A, "LD", Instruction::LOAD, A, BC)
-               dis_reg__reg16_(0x1A, "LD", Instruction::LOAD, A, DE)
-               dis_reg__inm_(0xFA, "LD", Instruction::LOAD, A)
-
-               dis__reg16__reg(0x02, "LD", Instruction::LOAD, BC, A)
-               dis__reg16__reg(0x12, "LD", Instruction::LOAD, DE, A)
-               dis__inm__reg(0xEA, "LD", Instruction::LOAD, A)
-
-               // LD A, (C)
-               case 0xF2:
-                       result << "LD A, (0xFF00+C)";
-                       opcode_str = "LDH";
-                       ins_type = Instruction::LOAD; 
-                       op1.str="A";
-                       op2.str="C";
-                       op1.type=Instruction::REG;
-                       op1.reg =Instruction::A;
-                       op2.type=Instruction::MEM_INDIRECT;
-                       op2.reg =Instruction::C;
-                       break;
-
-               // LD (C), A
-               case 0xE2:
-                       result << "LD (0xFF00+C), A";
-                       opcode_str = "LDH";
-                       ins_type = Instruction::LOAD; 
-                       op1.str="C";
-                       op2.str="A";
-                       op1.type=Instruction::MEM_INDIRECT;
-                       op1.reg =Instruction::C;
-                       op2.type=Instruction::REG;
-                       op2.reg =Instruction::A;
-                       break;
-
-               // LD A, (HLD); LD A, (HL-); LDD A,(HL);
-               case 0x3A:
-                       result << "LD A, (HL-)";
-                       opcode_str = "LDD";
-                       ins_type = Instruction::LOAD; 
-                       op1.str="A";
-                       op2.str="(HL)";
-                       op1.type=Instruction::REG;
-                       op1.reg =Instruction::A;
-                       op2.type=Instruction::MEM_INDIRECT;
-                       op2.reg =Instruction::HL;
-                       break;
-               // LD (HLD), A; LD (HL-), A; LDD (HL), A;
-               case 0x32:
-                       result << "LD (HL-), A";
-                       opcode_str = "LDD";
-                       ins_type = Instruction::LOAD; 
-                       op1.str="(HL)";
-                       op2.str="A";
-                       op1.type=Instruction::MEM_INDIRECT;
-                       op1.reg =Instruction::HL;
-                       op2.type=Instruction::REG;
-                       op2.reg =Instruction::A;
-                       break;
-               // LD A, (HLI); LD A, (HL+); LDI A, (HL);
-               case 0x2A:
-                       result << "LD A, (HL+)";
-                       opcode_str = "LDI";
-                       ins_type = Instruction::LOAD; 
-                       op1.str="A";
-                       op2.str="(HL)";
-                       op1.type=Instruction::REG;
-                       op1.reg =Instruction::A;
-                       op2.type=Instruction::MEM_INDIRECT;
-                       op2.reg =Instruction::HL;
-                       break;
-               // LD (HLI), A; LD (HL+), A; LDI (HL), A;
-               case 0x22:
-                       result << "LD (HL+), A";
-                       opcode_str = "LDI";
-                       ins_type = Instruction::LOAD; 
-                       op1.str="(HL)";
-                       op2.str="A";
-                       op1.type=Instruction::MEM_INDIRECT;
-                       op1.reg =Instruction::HL;
-                       op2.type=Instruction::REG;
-                       op2.reg =Instruction::A;
-                       break;
-
-               // LDH (n), A
-               case 0xE0: {
-                       int port = int(gb.memory.read(PC++, GBMemory::DONT_WATCH));
-                       
-                       result << "LD (0xFF" << 
-                                       std::setw(2) << port << "), A" << "\t[" << get_port_name(port) << "]";
-                       opcode_str = "LDH";
-                       ins_type = Instruction::LOAD; 
-                       op1.str=std::string("(") + ToString(port) + ")";
-                       op2.str="A";
-                       op1.type=Instruction::MEM_DIRECT;
-                       op1.val =0xFF00+port;
-                       op2.type=Instruction::REG;
-                       op2.reg =Instruction::A;
-                       break;
-               }
-               // LDH A, (n)
-               case 0xF0: {
-                       int port = int(gb.memory.read(PC++, GBMemory::DONT_WATCH));
-                       result << "LD A, (0xFF" << 
-                                       std::setw(2) << port << ")" << "\t[" << get_port_name(port) << "]";
-                       opcode_str = "LDH";
-                       ins_type = Instruction::LOAD; 
-                       op1.str="A";
-                       op2.str=std::string("(") + ToString(port) + ")";
-                       op1.type=Instruction::REG;
-                       op1.reg =Instruction::A;
-                       op2.type=Instruction::MEM_DIRECT;
-                       op2.val =0xFF00+port;
-                       break;
-               }
-
-               dis_reg16_inm(0x01, "LD", Instruction::LOAD, BC)
-               dis_reg16_inm(0x11, "LD", Instruction::LOAD, DE)
-               dis_reg16_inm(0x21, "LD", Instruction::LOAD, HL)
-               dis_reg16_inm(0x31, "LD", Instruction::LOAD, SP)
-               
-               // LD SP, HL
-               case 0xF9:
-                       result << "LD SP, HL";
-                       opcode_str = "LD";
-                       ins_type = Instruction::LOAD; 
-                       op1.str = "SP";
-                       op2.str = "HL";
-                       op1.type = Instruction::REG;
-                       op1.reg  = Instruction::SP;
-                       op2.type = Instruction::REG;
-                       op2.reg  = Instruction::HL;
-
-               // LD HL, SP+n
-               // LDHL SP, n
-               case 0xF8: {
-                       int n = int(gb.memory.read(PC++, GBMemory::DONT_WATCH));
-                       result << "LD HL, SP + 0x"<< std::setw(2) << n;
-                       opcode_str = "LD HL, SP+";
-                       ins_type = Instruction::LOAD; 
-                       op1.str = ToString(n);
-                       op2.str = "";
-                       op1.type = Instruction::INM8;
-                       op1.val = n;
-                       break; 
-               }
-
-               // LD (nn), SP
-               dis__inm__reg16(0x08, "LD", Instruction::LOAD, SP)
-
-               // PUSH nn
-               dis_reg16(0xF5, "PUSH", Instruction::OTHER, AF)
-               dis_reg16(0xC5, "PUSH", Instruction::OTHER, BC)
-               dis_reg16(0xD5, "PUSH", Instruction::OTHER, DE)
-               dis_reg16(0xE5, "PUSH", Instruction::OTHER, HL)
-
-               // POP nn
-               dis_reg16(0xF1, "POP", Instruction::OTHER, AF)
-               dis_reg16(0xC1, "POP", Instruction::OTHER, BC)
-               dis_reg16(0xD1, "POP", Instruction::OTHER, DE)
-               dis_reg16(0xE1, "POP", Instruction::OTHER, HL)
-
-               // 8-bit ALU
-               // ADD A,reg
-               dis_for_each_register(0x87, 0x80, 0x81, 0x82, 0x83, 0x84, 0x85, "ADD", Instruction::ALU, dis_A_reg)
-
-               dis_reg__reg16_(0x86, "ADD", Instruction::ALU, A, HL)
-               dis_reg_inm(0xC6, "ADD", Instruction::ALU, A)
-               
-               // ADC A, n
-               dis_for_each_register(0x8F, 0x88, 0x89, 0x8A, 0x8B, 0x8C, 0x8D, "ADC", Instruction::ALU, dis_A_reg)
-               
-               dis_reg__reg16_(0x8E, "ADC", Instruction::ALU, A, HL)
-               dis_reg_inm(0xCE, "ADC", Instruction::ALU, A)
-
-               // SUB n
-               dis_for_each_register(0x97, 0x90, 0x91, 0x92, 0x93, 0x94, 0x95, "SUB", Instruction::ALU, dis_reg)
-               
-               dis__reg16_(0x96, "SUB", Instruction::ALU, HL)
-               dis_inm8(0xD6, "SUB", Instruction::ALU) 
-               
-               // SBC n
-               dis_for_each_register(0x9F, 0x98, 0x99, 0x9A, 0x9B, 0x9C, 0x9D, "SBC", Instruction::ALU, dis_reg)
-
-               dis__reg16_(0x9E, "SBC", Instruction::ALU, HL)
-               dis_reg_inm(0xDE, "SBC", Instruction::ALU, A)
-
-               // AND n
-               dis_for_each_register(0xA7, 0xA0, 0xA1, 0xA2, 0xA3, 0xA4, 0xA5, "AND", Instruction::ALU, dis_reg)
-
-               dis__reg16_(0xA6, "AND", Instruction::ALU, HL)
-               dis_inm8(0xE6, "AND", Instruction::ALU)
-               
-               // OR n
-               dis_for_each_register(0xB7, 0xB0, 0xB1, 0xB2, 0xB3, 0xB4, 0xB5, "OR", Instruction::ALU, dis_reg)
-               
-               dis__reg16_(0xB6, "OR", Instruction::ALU, HL)
-               dis_inm8(0xF6, "OR", Instruction::ALU)
-
-               // XOR n
-               dis_for_each_register(0xAF, 0xA8, 0xA9, 0xAA, 0xAB, 0xAC, 0xAD, "XOR", Instruction::ALU, dis_reg)
-
-               dis__reg16_(0xAE, "XOR", Instruction::ALU, HL)
-               dis_inm8(0xEE, "XOR", Instruction::ALU)
-               
-               // CP n
-               dis_for_each_register(0xBF, 0xB8, 0xB9, 0xBA, 0xBB, 0xBC, 0xBD, "CP", Instruction::ALU, dis_reg)
-               
-               dis__reg16_(0xBE, "CP", Instruction::ALU, HL)
-               dis_inm8(0xFE, "CP", Instruction::ALU)
-
-               // INC n
-               dis_for_each_register(0x3C, 0x04, 0x0C, 0x14, 0x1C, 0x24, 0x2C, "INC", Instruction::ALU, dis_reg)
-
-               dis__reg16_(0x34, "INC", Instruction::ALU, HL)
-               
-               // DEC n
-               dis_for_each_register(0x3D, 0x05, 0x0D, 0x15, 0x1D, 0x25, 0x2D, "DEC", Instruction::ALU, dis_reg)
-
-               dis__reg16_(0x35, "DEC", Instruction::ALU, HL)
-               
-               // 16-bit ALU
-               // ADD HL, n
-               dis_for_each_register16(0x09, 0x19, 0x29, 0x39, "ADD", Instruction::ALU, dis_HL_reg16)
-
-               // ADD SP, #
-               dis_reg16_inm8(0xE8, "ADD", Instruction::ALU, SP)
-               
-               // INC nn
-               dis_for_each_register16(0x03, 0x13, 0x23, 0x33, "INC", Instruction::ALU, dis_reg16)
-
-               // DEC nn
-               dis_for_each_register16(0x0B, 0x1B, 0x2B, 0x3B, "DEC", Instruction::ALU, dis_reg16)
-
-               // Miscellaneous instructions
-               case 0xCB: {
-                       sub_opcode = gb.memory.read(PC++, GBMemory::DONT_WATCH);
-                       switch(sub_opcode)
-                       {
-                               // SWAP n
-                               dis_for_each_register(0x37, 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, "SWAP", Instruction::ALU, dis_reg)
-
-                               // SWAP (HL)
-                               dis__reg16_(0x36, "SWAP", Instruction::ALU, HL)
-
-                               // RLC n
-                               dis_for_each_register(0x07, 0x00, 0x01, 0x02, 0x03, 0x04, 0x05, "RLC", Instruction::ALU, dis_reg)
-
-                               // RLC (HL)
-                               dis__reg16_(0x06, "RLC", Instruction::ALU, HL)
-
-                               // RL n (through carry)
-                               dis_for_each_register(0x17, 0x10, 0x11, 0x12, 0x13, 0x14, 0x15, "RL", Instruction::ALU, dis_reg)
-
-                               // RL (HL) (through carry)
-                               dis__reg16_(0x16, "RL", Instruction::ALU, HL)
-
-                               // RRC n
-                               dis_for_each_register(0x0F, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, "RRC", Instruction::ALU, dis_reg)
-
-                               // RRC (HL)
-                               dis__reg16_(0x0E, "RRC", Instruction::ALU, HL)
-
-                               // RR n (through carry)
-                               dis_for_each_register(0x1F, 0x18, 0x19, 0x1A, 0x1B, 0x1C, 0x1D, "RR", Instruction::ALU, dis_reg)
-
-                               // RR (HL) (through carry)
-                               dis__reg16_(0x1E, "RR", Instruction::ALU, HL)
-
-                               // SLA n
-                               dis_for_each_register(0x27, 0x20, 0x21, 0x22, 0x23, 0x24, 0x25, "SLA", Instruction::ALU, dis_reg)
-
-                               // SLA (HL)
-                               dis__reg16_(0x26, "SLA", Instruction::ALU, HL)
-
-                               // SRA n
-                               dis_for_each_register(0x2F, 0x28, 0x29, 0x2A, 0x2B, 0x2C, 0x2D, "SRA", Instruction::ALU, dis_reg)
-                               
-                               // SRA (HL)
-                               dis__reg16_(0x2E, "SRA", Instruction::ALU, HL)
-
-                               // SRL n
-                               dis_for_each_register(0x3F, 0x38, 0x39, 0x3A, 0x3B, 0x3C, 0x3D, "SRA", Instruction::ALU, dis_reg)
-                               
-                               // SRL (HL)
-                               dis__reg16_(0x3E, "SRA", Instruction::ALU, HL)
-
-                               default: {
-                                       int bit_op = sub_opcode >> 6;
-                                       int reg = sub_opcode & 7;
-                                       int b   = (sub_opcode >> 3) & 7;
-                                       const char *bit_ops[4]={"Unknown", "BIT", "RES", "SET"};
-                                       const char *regs[8]={"B","C","D","E","H","L","(HL)", "A"};
-                                       result << bit_ops[bit_op] << " " << b << ", " << regs[reg];
-                                       opcode_str = bit_ops[bit_op];
-                                       ins_type = Instruction::ALU;
-                                       op1.str = ToString(b);
-                                       op1.type = Instruction::INM8;
-                                       op1.val = b;
-                                       op2.str = regs[reg];
-                                       switch(reg)
-                                       {
-                                               case 0:
-                                                       op2.type = Instruction::REG;
-                                                       op2.reg  = Instruction::B;
-                                                       break;
-                                               case 1:
-                                                       op2.type = Instruction::REG;
-                                                       op2.reg  = Instruction::C;
-                                                       break;
-                                               case 2:
-                                                       op2.type = Instruction::REG;
-                                                       op2.reg  = Instruction::D;
-                                                       break;
-                                               case 3:
-                                                       op2.type = Instruction::REG;
-                                                       op2.reg  = Instruction::E;
-                                                       break;
-                                               case 4:
-                                                       op2.type = Instruction::REG;
-                                                       op2.reg  = Instruction::H;
-                                                       break;
-                                               case 5:
-                                                       op2.type = Instruction::REG;
-                                                       op2.reg  = Instruction::L;
-                                                       break;
-                                               case 6:
-                                                       op2.type = Instruction::MEM_INDIRECT;
-                                                       op2.reg  = Instruction::HL;
-                                                       break;
-                                               case 7:
-                                                       op2.type = Instruction::REG;
-                                                       op2.reg  = Instruction::A;
-                                                       break;
-                                       }
-                                       break;
-                               }
-                               
-                       }
-                       break;
-               }
-
-               dis(0x27, "DAA", Instruction::ALU)
-
-               dis(0x2F, "CPL", Instruction::ALU)
-
-               dis(0x3F, "CCF", Instruction::ALU)
-
-               dis(0x37, "SCF", Instruction::ALU)
-               dis(0x00, "NOP", Instruction::OTHER)
-               dis(0x76, "HALT", Instruction::OTHER)
-
-               dis(0x10, "STOP", Instruction::OTHER)
-               
-               dis(0xF3, "DI", Instruction::OTHER)
-
-               dis(0xFB, "EI", Instruction::OTHER)
-                          
-               // Rotates and shifts
-               dis(0x07, "RLCA", Instruction::ALU)
-
-               dis(0x17, "RLA", Instruction::ALU)
-               
-               dis(0x0F, "RRCA", Instruction::ALU)
-
-               dis(0x1F, "RRA", Instruction::ALU)
-               
-               // Jumps
-               dis_inm16(0xC3, "JP", Instruction::UNCONDITIONAL_JUMP)
-               // JP cc, nn
-               dis_inm16(0xC2, "JP NZ", Instruction::CONDITIONAL_JUMP)
-               dis_inm16(0xCA, "JP Z",  Instruction::CONDITIONAL_JUMP)
-               dis_inm16(0xD2, "JP NC", Instruction::CONDITIONAL_JUMP)
-               dis_inm16(0xDA, "JP C",  Instruction::CONDITIONAL_JUMP)
-               case 0xE9:
-                       result << "JP (HL)";
-                       opcode_str = "JP";
-                       ins_type = Instruction::UNCONDITIONAL_JUMP; 
-                       op1.str="(HL)";
-                       op2.str="";
-                       op1.type = Instruction::MEM_INDIRECT;
-                       op1.reg = Instruction::HL;
-                       op2.type = Instruction::NONE;
-                       break;
-               
-               dis_JR(0x18, "JR", Instruction::UNCONDITIONAL_JUMP)
-               dis_JR(0x20, "JR NZ", Instruction::CONDITIONAL_JUMP)
-               dis_JR(0x28, "JR Z",  Instruction::CONDITIONAL_JUMP)
-               dis_JR(0x30, "JR NC", Instruction::CONDITIONAL_JUMP)
-               dis_JR(0x38, "JR C",  Instruction::CONDITIONAL_JUMP)
-
-               // Calls
-               dis_inm16(0xCD, "CALL", Instruction::CALL)
-               // CALL cc, nn
-               dis_inm16(0xC4, "CALL NZ", Instruction::CALL)
-               dis_inm16(0xCC, "CALL Z", Instruction::CALL)
-               dis_inm16(0xD4, "CALL NC", Instruction::CALL)
-               dis_inm16(0xDC, "CALL C", Instruction::CALL)
-
-               // Restarts
-               dis(0xC7, "RST 0x00", Instruction::RESET)
-               dis(0xCF, "RST 0x08", Instruction::RESET)
-               dis(0xD7, "RST 0x10", Instruction::RESET)
-               dis(0xDF, "RST 0x18", Instruction::RESET)
-               dis(0xE7, "RST 0x20", Instruction::RESET)
-               dis(0xEF, "RST 0x28", Instruction::RESET)
-               dis(0xF7, "RST 0x30", Instruction::RESET)
-               dis(0xFF, "RST 0x38", Instruction::RESET)
-
-               // Returns
-               dis(0xC9, "RET", Instruction::UNCONDITIONAL_RET)
-               // RET cc
-               dis(0xC0, "RET NZ", Instruction::CONDITIONAL_RET)
-               dis(0xC8, "RET Z",  Instruction::CONDITIONAL_RET)
-               dis(0xD0, "RET NC", Instruction::CONDITIONAL_RET)
-               dis(0xD8, "RET C",  Instruction::CONDITIONAL_RET)
-
-               dis(0xD9, "RETI", Instruction::UNCONDITIONAL_RET)
-
-               default:
-                       std::ostringstream errmsg;
-                       errmsg << "Unknown opcode 0x";
-                       errmsg << std::hex << std::setw(2) << std::setfill('0') << opcode;
-                       errmsg << " at 0x" << std::hex << std::setw(4) << PC-1;
-                       logger.trace(errmsg.str());
-                       break;
-
-       } // end switch
-
-       return Instruction(PC-addr, ins_type, opcode, sub_opcode,
-                       result.str(), opcode_str, op1, op2);
-}
-
index 0ab845bc156f8181cf0a8ade95b331da5fee1359..bf3e6ee575f3647cb7cd4afa2992eb744b1d09ba 100644 (file)
@@ -18,7 +18,7 @@
 #ifndef DISASSEMBLY_OUTPUT_H
 #define DISASSEMBLY_OUTPUT_H
 
-#include "../gbcore.h"
+#include "../core/GameBoy.h"
 #include "CodeBlock.h"
 #include <vector>
 
index 517f890727a7c85f4a611c187ddf1dd894d8b414..27a88631d11b3aa7c4d547d47fc826a0ce5e5c9f 100644 (file)
@@ -16,9 +16,9 @@
     along with wenboi.  If not, see <http://www.gnu.org/licenses/>.
 */
 #include "output_graph.h"
+#include "../common/Logger.h"
 #include <sstream>
 #include <iomanip>
-#include "../Logger.h"
 
 using std::vector;
 using std::string;
index 78b87c99e7e71c896f403cd96e21bb4e03529265..fc425826ae7b2898c5959210ff8cdbcce771a39d 100644 (file)
 
 // wendi, the WENboi DIsassembler :)
 
-#include "../gbcore.h"
-#include "../Logger.h"
+#include "../core/GameBoy.h"
+#include "../common/Logger.h"
 #include "disassembly_output.h"
 #include "output_txt.h"
 #include "output_graph.h"
 #include "CodeBlock.h"
-#include "disasm.h"
 
 #include <vector>
 #include <utility>
@@ -332,7 +331,7 @@ int main(int argc, char **argv)
                bool block_end = false;
                while(!block_end)
                {
-                       Instruction ins(disassemble_opcode(gb, addr));
+                       Instruction ins(gb.disassemble_opcode(addr));
                        block.add_instruction(ins.all, ins.length);
 
                        if (is_jump(ins))