initial commit
This commit is contained in:
commit
ff8389a76b
5
.gitignore
vendored
Normal file
5
.gitignore
vendored
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
.vs/
|
||||||
|
.vscode/
|
||||||
|
|
||||||
|
out/
|
||||||
|
*.json
|
10
.gitmodules
vendored
Normal file
10
.gitmodules
vendored
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
[submodule "vendor/spdlog"]
|
||||||
|
path = vendor/spdlog
|
||||||
|
url = https://github.com/gabime/spdlog
|
||||||
|
[submodule "vendor/glfw"]
|
||||||
|
path = vendor/glfw
|
||||||
|
url = https://github.com/glfw/glfw
|
||||||
|
[submodule "vendor/imgui"]
|
||||||
|
path = vendor/imgui
|
||||||
|
url = https://github.com/ocornut/imgui
|
||||||
|
branch = docking
|
28
CMakeLists.txt
Normal file
28
CMakeLists.txt
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
# CMakeList.txt : Top-level CMake project file, do global configuration
|
||||||
|
# and include sub-projects here.
|
||||||
|
#
|
||||||
|
cmake_minimum_required (VERSION 3.8)
|
||||||
|
|
||||||
|
project ("NES Emulator")
|
||||||
|
|
||||||
|
add_subdirectory("vendor/glfw")
|
||||||
|
add_subdirectory("vendor/glad")
|
||||||
|
add_subdirectory("vendor/spdlog")
|
||||||
|
|
||||||
|
set(IMGUI_SOURCES
|
||||||
|
${CMAKE_SOURCE_DIR}/vendor/imgui/imgui.cpp
|
||||||
|
${CMAKE_SOURCE_DIR}/vendor/imgui/imgui_demo.cpp
|
||||||
|
${CMAKE_SOURCE_DIR}/vendor/imgui/imgui_draw.cpp
|
||||||
|
${CMAKE_SOURCE_DIR}/vendor/imgui/imgui_tables.cpp
|
||||||
|
${CMAKE_SOURCE_DIR}/vendor/imgui/imgui_widgets.cpp
|
||||||
|
${CMAKE_SOURCE_DIR}/vendor/imgui/backends/imgui_impl_glfw.cpp
|
||||||
|
${CMAKE_SOURCE_DIR}/vendor/imgui/backends/imgui_impl_opengl3.cpp
|
||||||
|
)
|
||||||
|
|
||||||
|
set(IMGUI_INCLUDE
|
||||||
|
${CMAKE_SOURCE_DIR}/vendor
|
||||||
|
${CMAKE_SOURCE_DIR}/vendor/imgui
|
||||||
|
)
|
||||||
|
|
||||||
|
# Include sub-projects.
|
||||||
|
add_subdirectory ("src")
|
19
README.md
Normal file
19
README.md
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
# NESemu
|
||||||
|
|
||||||
|
NESemu is my fourth attempt at writing a working Nintendo NES Emulator.
|
||||||
|
|
||||||
|
This time I'm also writing a Debugger that will hopefully help me finish the project this time.
|
||||||
|
|
||||||
|

|
||||||
|
|
||||||
|
Also I'm writing it in C++ this time because jesus christ i hate C. My previous attempt is [here](https://github.com/Lauchmelder23/NESEmulator).
|
||||||
|
|
||||||
|
## Libraries
|
||||||
|
* [spdlog](https://github.com/gabime/spdlog) for logging
|
||||||
|
* [GLFW](https://github.com/glfw/glfw) for window and event handling
|
||||||
|
* [GLAD](https://glad.dav1d.de/) to load OpenGL
|
||||||
|
* [ImGui](https://github.com/ocornut/imgui) for the GUI
|
||||||
|
|
||||||
|
## Resources
|
||||||
|
* [Nesdev Wiki](https://wiki.nesdev.org/w/index.php) - Probably the most exhaustive NES documentation available on the internet
|
||||||
|
* [Masswerk](https://www.masswerk.at/6502/6502_instruction_set.html) - Exhaustive list of official and illegal instructions (there are some errors regarding the cycles of illegal opcodes)
|
BIN
imgs/debug_example.png
Normal file
BIN
imgs/debug_example.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 117 KiB |
BIN
roms/nestest.nes
Normal file
BIN
roms/nestest.nes
Normal file
Binary file not shown.
109
src/Application.cpp
Normal file
109
src/Application.cpp
Normal file
|
@ -0,0 +1,109 @@
|
||||||
|
#include "Application.hpp"
|
||||||
|
|
||||||
|
#include <stdexcept>
|
||||||
|
#include <glad/glad.h>
|
||||||
|
#include <GLFW/glfw3.h>
|
||||||
|
|
||||||
|
#include <imgui/backends/imgui_impl_glfw.h>
|
||||||
|
#include <imgui/backends/imgui_impl_opengl3.h>
|
||||||
|
#include <imgui/imgui.h>
|
||||||
|
|
||||||
|
#include "Log.hpp"
|
||||||
|
#include "Bus.hpp"
|
||||||
|
#include "Debugger.hpp"
|
||||||
|
#include "gfx/Window.hpp"
|
||||||
|
|
||||||
|
void Application::Launch()
|
||||||
|
{
|
||||||
|
glfwInit();
|
||||||
|
|
||||||
|
Application* app = nullptr;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
app = new Application;
|
||||||
|
}
|
||||||
|
catch (const std::runtime_error& err)
|
||||||
|
{
|
||||||
|
LOG_CORE_FATAL(err.what());
|
||||||
|
delete app;
|
||||||
|
glfwTerminate();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (app == nullptr)
|
||||||
|
{
|
||||||
|
LOG_CORE_ERROR("Application object is nullptr");
|
||||||
|
glfwTerminate();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
while (app->Update());
|
||||||
|
delete app;
|
||||||
|
glfwTerminate();
|
||||||
|
}
|
||||||
|
|
||||||
|
Application::Application() :
|
||||||
|
bus(nullptr), window(nullptr)
|
||||||
|
{
|
||||||
|
LOG_CORE_INFO("Creating window");
|
||||||
|
try
|
||||||
|
{
|
||||||
|
window = new Window(1280, 720, "NES Emulator");
|
||||||
|
}
|
||||||
|
catch (const std::runtime_error& err)
|
||||||
|
{
|
||||||
|
LOG_CORE_ERROR("Window creation failed");
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
|
||||||
|
LOG_CORE_INFO("Loading OpenGL API");
|
||||||
|
if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress))
|
||||||
|
throw std::runtime_error("Failed to set up OpenGL loader");
|
||||||
|
|
||||||
|
LOG_CORE_INFO("Setting up ImGui");
|
||||||
|
IMGUI_CHECKVERSION();
|
||||||
|
ImGui::CreateContext();
|
||||||
|
ImGuiIO& io = ImGui::GetIO();
|
||||||
|
io.ConfigFlags |= ImGuiConfigFlags_NavEnableKeyboard;
|
||||||
|
io.ConfigFlags |= ImGuiConfigFlags_DockingEnable;
|
||||||
|
io.ConfigFlags |= ImGuiConfigFlags_ViewportsEnable;
|
||||||
|
|
||||||
|
ImGui_ImplGlfw_InitForOpenGL(window->GetNativeWindow(), true);
|
||||||
|
ImGui_ImplOpenGL3_Init("#version 460 core");
|
||||||
|
|
||||||
|
ImGui::StyleColorsDark();
|
||||||
|
|
||||||
|
ImGuiStyle& style = ImGui::GetStyle();
|
||||||
|
if (io.ConfigFlags & ImGuiConfigFlags_ViewportsEnable)
|
||||||
|
{
|
||||||
|
style.WindowRounding = 0.0f;
|
||||||
|
style.Colors[ImGuiCol_WindowBg].w = 1.0f;
|
||||||
|
}
|
||||||
|
|
||||||
|
bus = new Bus;
|
||||||
|
debugger = new Debugger(bus);
|
||||||
|
}
|
||||||
|
|
||||||
|
Application::~Application()
|
||||||
|
{
|
||||||
|
delete debugger;
|
||||||
|
|
||||||
|
if(bus)
|
||||||
|
delete bus;
|
||||||
|
|
||||||
|
delete window;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Application::Update()
|
||||||
|
{
|
||||||
|
glfwPollEvents();
|
||||||
|
|
||||||
|
if (!debugger->Update())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
window->Begin();
|
||||||
|
debugger->Render();
|
||||||
|
window->End();
|
||||||
|
|
||||||
|
return !window->ShouldClose();
|
||||||
|
}
|
22
src/Application.hpp
Normal file
22
src/Application.hpp
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
class Bus;
|
||||||
|
class Window;
|
||||||
|
class Debugger;
|
||||||
|
|
||||||
|
class Application
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
static void Launch();
|
||||||
|
|
||||||
|
private:
|
||||||
|
Application();
|
||||||
|
~Application();
|
||||||
|
|
||||||
|
bool Update();
|
||||||
|
|
||||||
|
private:
|
||||||
|
Window* window;
|
||||||
|
Bus* bus;
|
||||||
|
Debugger* debugger;
|
||||||
|
};
|
142
src/Bus.cpp
Normal file
142
src/Bus.cpp
Normal file
|
@ -0,0 +1,142 @@
|
||||||
|
#include "Bus.hpp"
|
||||||
|
#include "Log.hpp"
|
||||||
|
|
||||||
|
#include <stdexcept>
|
||||||
|
|
||||||
|
Bus::Bus() :
|
||||||
|
cpu(this), ppu(this), cartridge(this)
|
||||||
|
{
|
||||||
|
LOG_CORE_INFO("Allocating RAM");
|
||||||
|
RAM = std::vector<Byte>(0x800);
|
||||||
|
|
||||||
|
LOG_CORE_INFO("Allocating VRAM");
|
||||||
|
VRAM = std::vector<Byte>(0x1000);
|
||||||
|
|
||||||
|
LOG_CORE_INFO("Inserting cartridge");
|
||||||
|
cartridge.Load("roms/nestest.nes");
|
||||||
|
|
||||||
|
LOG_CORE_INFO("Powering up CPU");
|
||||||
|
cpu.Powerup();
|
||||||
|
|
||||||
|
LOG_CORE_INFO("Powering up PPU");
|
||||||
|
ppu.Powerup();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Bus::Reboot()
|
||||||
|
{
|
||||||
|
cpu.Powerup();
|
||||||
|
ppu.Powerup();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Bus::Reset()
|
||||||
|
{
|
||||||
|
cpu.Reset();
|
||||||
|
ppu.Reset();
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t Bus::Tick()
|
||||||
|
{
|
||||||
|
uint8_t result = cpu.Tick();
|
||||||
|
|
||||||
|
// 3 ppu ticks per cpu tick
|
||||||
|
ppu.Tick();
|
||||||
|
ppu.Tick();
|
||||||
|
ppu.Tick();
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Bus::Instruction()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
while (Tick());
|
||||||
|
}
|
||||||
|
catch (const std::runtime_error& err)
|
||||||
|
{
|
||||||
|
LOG_CORE_FATAL("Fatal Bus error: {0}", err.what());
|
||||||
|
cpu.Halt();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Bus::Frame()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
while (!ppu.IsFrameDone())
|
||||||
|
Tick();
|
||||||
|
}
|
||||||
|
catch (const std::runtime_error& err)
|
||||||
|
{
|
||||||
|
LOG_CORE_FATAL("Fatal Bus error: {0}", err.what());
|
||||||
|
cpu.Halt();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
Byte Bus::ReadCPU(Word addr)
|
||||||
|
{
|
||||||
|
if (0x0000 <= addr && addr < 0x2000)
|
||||||
|
{
|
||||||
|
return RAM[addr & 0x7FF];
|
||||||
|
}
|
||||||
|
else if (0x2000 <= addr && addr < 0x4000)
|
||||||
|
{
|
||||||
|
return ppu.ReadRegister(addr & 0x7);
|
||||||
|
}
|
||||||
|
else if (0x8000 <= addr && addr <= 0xFFFF)
|
||||||
|
{
|
||||||
|
return cartridge.ReadCPU(addr);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Byte Bus::ReadPPU(Word addr)
|
||||||
|
{
|
||||||
|
addr &= 0x3FFF;
|
||||||
|
|
||||||
|
if (0x0000 <= addr && addr < 0x2000)
|
||||||
|
{
|
||||||
|
return cartridge.ReadPPU(addr);
|
||||||
|
}
|
||||||
|
else if(0x2000 <= addr && addr < 0x4000)
|
||||||
|
{
|
||||||
|
return VRAM[addr & 0xFFF];
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0x00;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Bus::WriteCPU(Word addr, Byte val)
|
||||||
|
{
|
||||||
|
if (0x0000 <= addr && addr < 0x2000)
|
||||||
|
{
|
||||||
|
RAM[addr & 0x7FF] = val;
|
||||||
|
}
|
||||||
|
else if (0x2000 <= addr && addr < 0x4000)
|
||||||
|
{
|
||||||
|
ppu.WriteRegister(addr & 0x7, val);
|
||||||
|
}
|
||||||
|
else if (0x8000 <= addr && addr <= 0xFFFF)
|
||||||
|
{
|
||||||
|
cartridge.WriteCPU(addr, val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Bus::WritePPU(Word addr, Byte val)
|
||||||
|
{
|
||||||
|
addr &= 0x3FFF;
|
||||||
|
|
||||||
|
if (0x0000 <= addr && addr < 0x2000)
|
||||||
|
{
|
||||||
|
cartridge.WritePPU(addr, val);
|
||||||
|
}
|
||||||
|
else if (0x2000 <= addr && addr < 0x4000)
|
||||||
|
{
|
||||||
|
VRAM[addr & 0xFFF] = val;
|
||||||
|
}
|
||||||
|
}
|
38
src/Bus.hpp
Normal file
38
src/Bus.hpp
Normal file
|
@ -0,0 +1,38 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include "Types.hpp"
|
||||||
|
#include "CPU.hpp"
|
||||||
|
#include "PPU.hpp"
|
||||||
|
#include "Cartridge.hpp"
|
||||||
|
|
||||||
|
class Bus
|
||||||
|
{
|
||||||
|
friend class Debugger;
|
||||||
|
friend class MemoryViewer;
|
||||||
|
friend class NametableViewer;
|
||||||
|
|
||||||
|
public:
|
||||||
|
Bus();
|
||||||
|
|
||||||
|
void Reboot();
|
||||||
|
void Reset();
|
||||||
|
|
||||||
|
uint8_t Tick();
|
||||||
|
bool Instruction();
|
||||||
|
bool Frame();
|
||||||
|
|
||||||
|
Byte ReadCPU(Word addr);
|
||||||
|
Byte ReadPPU(Word addr);
|
||||||
|
void WriteCPU(Word addr, Byte val);
|
||||||
|
void WritePPU(Word addr, Byte val);
|
||||||
|
|
||||||
|
inline void NMI() { cpu.NMI(); }
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::vector<Byte> RAM, VRAM;
|
||||||
|
CPU cpu;
|
||||||
|
PPU ppu;
|
||||||
|
Cartridge cartridge;
|
||||||
|
};
|
33
src/CMakeLists.txt
Normal file
33
src/CMakeLists.txt
Normal file
|
@ -0,0 +1,33 @@
|
||||||
|
add_executable(nesemu
|
||||||
|
"main.cpp"
|
||||||
|
"Application.cpp"
|
||||||
|
"Bus.cpp"
|
||||||
|
"CPU.cpp"
|
||||||
|
"Cartridge.cpp"
|
||||||
|
"mappers/Mapper000.cpp"
|
||||||
|
"Log.cpp"
|
||||||
|
"PPU.cpp"
|
||||||
|
"gfx/Window.cpp"
|
||||||
|
"debugger/CPUWatcher.cpp"
|
||||||
|
"debugger/Debugger.cpp" "debugger/PPUWatcher.cpp" "debugger/Disassembler.cpp" "debugger/MemoryViewer.cpp" "debugger/NametableViewer.cpp")
|
||||||
|
|
||||||
|
target_include_directories(nesemu PRIVATE
|
||||||
|
mappers
|
||||||
|
gfx
|
||||||
|
debugger
|
||||||
|
${IMGUI_INCLUDE}
|
||||||
|
)
|
||||||
|
|
||||||
|
target_sources(nesemu PRIVATE
|
||||||
|
${IMGUI_SOURCES}
|
||||||
|
)
|
||||||
|
|
||||||
|
target_link_libraries(nesemu
|
||||||
|
spdlog
|
||||||
|
glfw
|
||||||
|
glad
|
||||||
|
)
|
||||||
|
|
||||||
|
add_custom_command(TARGET nesemu POST_BUILD
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E copy_directory ${CMAKE_SOURCE_DIR}/roms $<TARGET_FILE_DIR:nesemu>/roms
|
||||||
|
)
|
1239
src/CPU.cpp
Normal file
1239
src/CPU.cpp
Normal file
File diff suppressed because it is too large
Load diff
191
src/CPU.hpp
Normal file
191
src/CPU.hpp
Normal file
|
@ -0,0 +1,191 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <functional>
|
||||||
|
#include <sstream>
|
||||||
|
#include <deque>
|
||||||
|
#include "Types.hpp"
|
||||||
|
|
||||||
|
class Bus;
|
||||||
|
|
||||||
|
using Operation = std::function<void(void)>;
|
||||||
|
using AddressingMode = std::function<void(void)>;
|
||||||
|
|
||||||
|
enum class Addressing
|
||||||
|
{
|
||||||
|
ABS,
|
||||||
|
ABX,
|
||||||
|
ABY,
|
||||||
|
ACC,
|
||||||
|
IDX,
|
||||||
|
IDY,
|
||||||
|
IMM,
|
||||||
|
IMP,
|
||||||
|
IND,
|
||||||
|
REL,
|
||||||
|
ZPG,
|
||||||
|
ZPX,
|
||||||
|
ZPY
|
||||||
|
};
|
||||||
|
|
||||||
|
union StatusFlag
|
||||||
|
{
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
Byte Carry : 1;
|
||||||
|
Byte Zero : 1;
|
||||||
|
Byte InterruptDisable : 1;
|
||||||
|
Byte Decimal : 1;
|
||||||
|
Byte Break : 1;
|
||||||
|
Byte NoEffect : 1;
|
||||||
|
Byte Overflow : 1;
|
||||||
|
Byte Negative : 1;
|
||||||
|
} Flag;
|
||||||
|
|
||||||
|
Word Raw;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct Instruction
|
||||||
|
{
|
||||||
|
Operation Operation = nullptr;
|
||||||
|
AddressingMode Mode = nullptr;
|
||||||
|
Addressing AddrType = Addressing::IMP;
|
||||||
|
uint8_t Size = 0;
|
||||||
|
uint8_t Cycles = 0;
|
||||||
|
char Mnemonic[5] = " XXX";
|
||||||
|
};
|
||||||
|
|
||||||
|
class CPU
|
||||||
|
{
|
||||||
|
friend class Debugger;
|
||||||
|
friend class CPUWatcher;
|
||||||
|
friend class Disassembler;
|
||||||
|
|
||||||
|
public:
|
||||||
|
CPU(Bus* bus);
|
||||||
|
|
||||||
|
uint8_t Tick();
|
||||||
|
void Powerup();
|
||||||
|
void Reset();
|
||||||
|
inline void Halt() { halted = true; }
|
||||||
|
|
||||||
|
void IRQ();
|
||||||
|
void NMI();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void CreateInstructionTable();
|
||||||
|
|
||||||
|
inline void Push(Byte val) { Write(0x0100 | (sp--), val); }
|
||||||
|
inline Byte Pop() { return Read(0x0100 | (++sp)); }
|
||||||
|
|
||||||
|
Byte Read(Word addr);
|
||||||
|
void Write(Word addr, Byte val);
|
||||||
|
|
||||||
|
private:
|
||||||
|
Address rawAddress;
|
||||||
|
Address absoluteAddress;
|
||||||
|
Byte relativeAddress;
|
||||||
|
Byte fetchedVal;
|
||||||
|
bool accumulatorAddressing = false;
|
||||||
|
void ABS();
|
||||||
|
void ABX();
|
||||||
|
void ABY();
|
||||||
|
void ACC();
|
||||||
|
void IDX();
|
||||||
|
void IDY();
|
||||||
|
void IMM();
|
||||||
|
void IMP();
|
||||||
|
void IND();
|
||||||
|
void REL();
|
||||||
|
void ZPG();
|
||||||
|
void ZPX();
|
||||||
|
void ZPY();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void ADC();
|
||||||
|
void AND();
|
||||||
|
void ASL();
|
||||||
|
void BCC();
|
||||||
|
void BCS();
|
||||||
|
void BEQ();
|
||||||
|
void BIT();
|
||||||
|
void BMI();
|
||||||
|
void BNE();
|
||||||
|
void BPL();
|
||||||
|
void BRK();
|
||||||
|
void BVC();
|
||||||
|
void BVS();
|
||||||
|
void CLC();
|
||||||
|
void CLD();
|
||||||
|
void CLI();
|
||||||
|
void CLV();
|
||||||
|
void CMP();
|
||||||
|
void CPX();
|
||||||
|
void CPY();
|
||||||
|
void DCP();
|
||||||
|
void DEC();
|
||||||
|
void DEX();
|
||||||
|
void DEY();
|
||||||
|
void EOR();
|
||||||
|
void INC();
|
||||||
|
void INX();
|
||||||
|
void INY();
|
||||||
|
void ISC();
|
||||||
|
void JMP();
|
||||||
|
void JSR();
|
||||||
|
void LAX();
|
||||||
|
void LDA();
|
||||||
|
void LDX();
|
||||||
|
void LDY();
|
||||||
|
void LSR();
|
||||||
|
void NOP();
|
||||||
|
void ORA();
|
||||||
|
void PHA();
|
||||||
|
void PHP();
|
||||||
|
void PLA();
|
||||||
|
void PLP();
|
||||||
|
void RLA();
|
||||||
|
void ROL();
|
||||||
|
void ROR();
|
||||||
|
void RRA();
|
||||||
|
void RTI();
|
||||||
|
void RTS();
|
||||||
|
void SAX();
|
||||||
|
void SBC();
|
||||||
|
void SEC();
|
||||||
|
void SED();
|
||||||
|
void SEI();
|
||||||
|
void SLO();
|
||||||
|
void SRE();
|
||||||
|
void STA();
|
||||||
|
void STX();
|
||||||
|
void STY();
|
||||||
|
void TAX();
|
||||||
|
void TAY();
|
||||||
|
void TSX();
|
||||||
|
void TXA();
|
||||||
|
void TXS();
|
||||||
|
void TYA();
|
||||||
|
|
||||||
|
private:
|
||||||
|
Byte acc;
|
||||||
|
Byte idx, idy;
|
||||||
|
Address pc;
|
||||||
|
Word sp;
|
||||||
|
|
||||||
|
StatusFlag status;
|
||||||
|
|
||||||
|
Instruction* currentInstruction = nullptr;
|
||||||
|
uint8_t remainingCycles = 0;
|
||||||
|
uint8_t additionalCycles = 0;
|
||||||
|
uint64_t totalCycles = 0;
|
||||||
|
std::deque<Word> pastPCs;
|
||||||
|
bool halted = false;
|
||||||
|
#ifndef NDEBUG
|
||||||
|
std::stringstream debugString;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
private:
|
||||||
|
std::array<Instruction, 256> InstructionTable;
|
||||||
|
Bus* bus;
|
||||||
|
};
|
59
src/Cartridge.cpp
Normal file
59
src/Cartridge.cpp
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
#include "Cartridge.hpp"
|
||||||
|
#include "Bus.hpp"
|
||||||
|
#include "Log.hpp"
|
||||||
|
|
||||||
|
#include <fstream>
|
||||||
|
|
||||||
|
#include "Mapper.hpp"
|
||||||
|
#include "mappers/Mapper000.hpp"
|
||||||
|
|
||||||
|
Cartridge::Cartridge(Bus* bus) :
|
||||||
|
bus(bus)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
Cartridge::~Cartridge()
|
||||||
|
{
|
||||||
|
delete mapper;
|
||||||
|
}
|
||||||
|
|
||||||
|
Byte Cartridge::ReadCPU(Word addr)
|
||||||
|
{
|
||||||
|
return mapper->ReadCPU(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
Byte Cartridge::ReadPPU(Word addr)
|
||||||
|
{
|
||||||
|
return mapper->ReadPPU(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cartridge::WriteCPU(Word addr, Byte val)
|
||||||
|
{
|
||||||
|
mapper->WriteCPU(addr, val);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cartridge::WritePPU(Word addr, Byte val)
|
||||||
|
{
|
||||||
|
mapper->WritePPU(addr, val);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cartridge::Load(std::string path)
|
||||||
|
{
|
||||||
|
std::ifstream file(path, std::ios::binary);
|
||||||
|
if (!file)
|
||||||
|
throw std::runtime_error("Failed to open file " + path);
|
||||||
|
|
||||||
|
LOG_CORE_INFO("Extracting header");
|
||||||
|
Header header;
|
||||||
|
file.read((char*)&header, sizeof(Header));
|
||||||
|
|
||||||
|
uint8_t mapperNumber = (header.MapperHi & 0xF0) | (header.MapperLo >> 4);
|
||||||
|
LOG_CORE_INFO("Cartridge requires Mapper {0:d}", mapperNumber);
|
||||||
|
switch (mapperNumber)
|
||||||
|
{
|
||||||
|
case 0: mapper = new Mapper000(header, file); break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
throw std::runtime_error("Unsupported mapper ID " + std::to_string(mapperNumber));
|
||||||
|
}
|
||||||
|
}
|
42
src/Cartridge.hpp
Normal file
42
src/Cartridge.hpp
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
#include "Types.hpp"
|
||||||
|
#include "Mapper.hpp"
|
||||||
|
|
||||||
|
class Bus;
|
||||||
|
|
||||||
|
struct Header
|
||||||
|
{
|
||||||
|
Byte Signature[4];
|
||||||
|
Byte PrgROM;
|
||||||
|
Byte ChrROM;
|
||||||
|
Byte MapperLo;
|
||||||
|
Byte MapperHi;
|
||||||
|
Byte PrgRAM;
|
||||||
|
Byte TV1;
|
||||||
|
Byte TV2;
|
||||||
|
Byte Padding[5];
|
||||||
|
};
|
||||||
|
|
||||||
|
class Cartridge
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
Cartridge(Bus* bus);
|
||||||
|
~Cartridge();
|
||||||
|
|
||||||
|
Byte ReadCPU(Word addr);
|
||||||
|
Byte ReadPPU(Word addr);
|
||||||
|
void WriteCPU(Word addr, Byte val);
|
||||||
|
void WritePPU(Word addr, Byte val);
|
||||||
|
|
||||||
|
void Load(std::string path);
|
||||||
|
|
||||||
|
inline Mapper* GetMapper() { return mapper; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
Mapper* mapper;
|
||||||
|
Bus* bus;
|
||||||
|
};
|
18
src/Log.cpp
Normal file
18
src/Log.cpp
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
#include "Log.hpp"
|
||||||
|
#include "spdlog/sinks/stdout_color_sinks.h"
|
||||||
|
|
||||||
|
std::shared_ptr<spdlog::logger> Log::coreLogger;
|
||||||
|
std::shared_ptr<spdlog::logger> Log::debugLogger;
|
||||||
|
|
||||||
|
void Log::Init()
|
||||||
|
{
|
||||||
|
debugLogger = spdlog::stdout_color_mt("DEBUG");
|
||||||
|
debugLogger->set_pattern("%^[%T] %n: %v%$");
|
||||||
|
debugLogger->set_level(spdlog::level::trace);
|
||||||
|
|
||||||
|
coreLogger = spdlog::stdout_color_mt("CORE");
|
||||||
|
coreLogger->set_pattern("%^[%T] %n: %v%$");
|
||||||
|
coreLogger->set_level(spdlog::level::info);
|
||||||
|
|
||||||
|
LOG_CORE_INFO("Logger initialized");
|
||||||
|
}
|
30
src/Log.hpp
Normal file
30
src/Log.hpp
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <spdlog/spdlog.h>
|
||||||
|
#include <spdlog/fmt/ostr.h>
|
||||||
|
|
||||||
|
#define FORCE_NO_DEBUG_LOG
|
||||||
|
|
||||||
|
class Log
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
static void Init();
|
||||||
|
inline static std::shared_ptr<spdlog::logger> GetCoreLogger() { return coreLogger; }
|
||||||
|
inline static std::shared_ptr<spdlog::logger> GetDebugLogger() { return debugLogger; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
static std::shared_ptr<spdlog::logger> coreLogger;
|
||||||
|
static std::shared_ptr<spdlog::logger> debugLogger;
|
||||||
|
};
|
||||||
|
|
||||||
|
#define LOG_CORE_TRACE(...) ::Log::GetCoreLogger()->trace(__VA_ARGS__)
|
||||||
|
#define LOG_CORE_INFO(...) ::Log::GetCoreLogger()->info(__VA_ARGS__)
|
||||||
|
#define LOG_CORE_WARN(...) ::Log::GetCoreLogger()->warn(__VA_ARGS__)
|
||||||
|
#define LOG_CORE_ERROR(...) ::Log::GetCoreLogger()->error(__VA_ARGS__)
|
||||||
|
#define LOG_CORE_FATAL(...) ::Log::GetCoreLogger()->critical(__VA_ARGS__)
|
||||||
|
|
||||||
|
#define LOG_DEBUG_TRACE(...) ::Log::GetDebugLogger()->trace(__VA_ARGS__)
|
||||||
|
#define LOG_DEBUG_INFO(...) ::Log::GetDebugLogger()->info(__VA_ARGS__)
|
||||||
|
#define LOG_DEBUG_WARN(...) ::Log::GetDebugLogger()->warn(__VA_ARGS__)
|
||||||
|
#define LOG_DEBUG_ERROR(...) ::Log::GetDebugLogger()->error(__VA_ARGS__)
|
||||||
|
#define LOG_DEBUG_FATAL(...) ::Log::GetDebugLogger()->critical(__VA_ARGS__)
|
22
src/Mapper.hpp
Normal file
22
src/Mapper.hpp
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
#include "Types.hpp"
|
||||||
|
|
||||||
|
class Mapper
|
||||||
|
{
|
||||||
|
friend class Disassembler;
|
||||||
|
|
||||||
|
public:
|
||||||
|
virtual Byte ReadCPU(Word addr) = 0;
|
||||||
|
virtual Byte ReadPPU(Word addr) = 0;
|
||||||
|
virtual void WriteCPU(Word addr, Byte val) = 0;
|
||||||
|
virtual void WritePPU(Word addr, Byte val) = 0;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
Mapper() = default;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
std::vector<Byte> PRG_ROM;
|
||||||
|
std::vector<Byte> CHR_ROM;
|
||||||
|
};
|
156
src/PPU.cpp
Normal file
156
src/PPU.cpp
Normal file
|
@ -0,0 +1,156 @@
|
||||||
|
#include "PPU.hpp"
|
||||||
|
#include "Log.hpp"
|
||||||
|
#include "Bus.hpp"
|
||||||
|
|
||||||
|
PPU::PPU(Bus* bus) :
|
||||||
|
bus(bus), ppuctrl{0}, ppustatus{0}
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::Powerup()
|
||||||
|
{
|
||||||
|
ppuctrl.Raw = 0b00000000;
|
||||||
|
ppumask.Raw = 0b00000000;
|
||||||
|
ppustatus.Raw = 0b10100000;
|
||||||
|
ppuscroll.x = 0x00;
|
||||||
|
ppuscroll.y = 0x00;
|
||||||
|
ppuaddr.Raw = 0x0000;
|
||||||
|
|
||||||
|
x = 0;
|
||||||
|
y = 0;
|
||||||
|
addressLatch = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::Reset()
|
||||||
|
{
|
||||||
|
ppuctrl.Raw = 0b00000000;
|
||||||
|
ppumask.Raw = 0b00000000;
|
||||||
|
ppuscroll.x = 0x00;
|
||||||
|
ppuscroll.y = 0x00;
|
||||||
|
|
||||||
|
x = 0;
|
||||||
|
y = 0;
|
||||||
|
addressLatch = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::Tick()
|
||||||
|
{
|
||||||
|
if (y == 241 && x == 1)
|
||||||
|
{
|
||||||
|
ppustatus.Flag.VBlankStarted = 1;
|
||||||
|
if (ppuctrl.Flag.VBlankNMI)
|
||||||
|
bus->NMI();
|
||||||
|
|
||||||
|
isFrameDone = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (y == 261 && x == 1)
|
||||||
|
ppustatus.Flag.VBlankStarted = 0;
|
||||||
|
|
||||||
|
x++;
|
||||||
|
if (x > 340)
|
||||||
|
{
|
||||||
|
x = 0;
|
||||||
|
y++;
|
||||||
|
if (y > 261)
|
||||||
|
y = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Byte PPU::ReadRegister(Byte id)
|
||||||
|
{
|
||||||
|
Byte returnVal = 0x00;
|
||||||
|
|
||||||
|
switch (id)
|
||||||
|
{
|
||||||
|
case 0:
|
||||||
|
returnVal = ppuctrl.Raw;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 1:
|
||||||
|
returnVal = ppumask.Raw;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
returnVal = ppustatus.Raw;
|
||||||
|
ppustatus.Flag.VBlankStarted = 0;
|
||||||
|
addressLatch = 0;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 5:
|
||||||
|
returnVal = 0x00;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 6:
|
||||||
|
returnVal = 0x00;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 7:
|
||||||
|
returnVal = bus->ReadPPU(ppuaddr.Raw);
|
||||||
|
ppuaddr.Raw += (ppuctrl.Flag.VRAMAddrIncrement ? 32 : 1);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
LOG_CORE_WARN("Tried to read unimplemented PPU register $20{0:02X}", (Word)id);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return returnVal;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::WriteRegister(Byte id, Byte val)
|
||||||
|
{
|
||||||
|
switch (id)
|
||||||
|
{
|
||||||
|
case 0:
|
||||||
|
ppuctrl.Raw = val;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 1:
|
||||||
|
ppumask.Raw = val;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
ppustatus.Raw = val;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 5:
|
||||||
|
if (addressLatch == 0)
|
||||||
|
ppuscroll.x = val;
|
||||||
|
else
|
||||||
|
ppuscroll.y = val;
|
||||||
|
|
||||||
|
addressLatch = 1 - addressLatch;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 6:
|
||||||
|
if (addressLatch == 0)
|
||||||
|
ppuaddr.Bytes.hi = val;
|
||||||
|
else
|
||||||
|
ppuaddr.Bytes.lo = val;
|
||||||
|
|
||||||
|
addressLatch = 1 - addressLatch;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 7:
|
||||||
|
bus->WritePPU(ppuaddr.Raw, val);
|
||||||
|
ppuaddr.Raw += (ppuctrl.Flag.VRAMAddrIncrement ? 32 : 1);
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
LOG_CORE_WARN("Tried to write unimplemented PPU register $20{0:02X}", (Word)id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ppustatus.Flag.Unused = val & 0x1F;
|
||||||
|
}
|
||||||
|
|
||||||
|
Byte PPU::Read(Word addr)
|
||||||
|
{
|
||||||
|
return bus->ReadPPU(addr);
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPU::Write(Word addr, Byte val)
|
||||||
|
{
|
||||||
|
bus->WritePPU(addr, val);
|
||||||
|
}
|
90
src/PPU.hpp
Normal file
90
src/PPU.hpp
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "Types.hpp"
|
||||||
|
|
||||||
|
class Bus;
|
||||||
|
|
||||||
|
class PPU
|
||||||
|
{
|
||||||
|
friend class PPUWatcher;
|
||||||
|
|
||||||
|
public:
|
||||||
|
PPU(Bus* bus);
|
||||||
|
|
||||||
|
void Powerup();
|
||||||
|
void Reset();
|
||||||
|
|
||||||
|
void Tick();
|
||||||
|
|
||||||
|
Byte ReadRegister(Byte id);
|
||||||
|
void WriteRegister(Byte id, Byte val);
|
||||||
|
|
||||||
|
inline bool IsFrameDone() { bool returnVal = isFrameDone; isFrameDone = false; return returnVal; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
Byte Read(Word addr);
|
||||||
|
void Write(Word addr, Byte val);
|
||||||
|
|
||||||
|
private: // Registers
|
||||||
|
|
||||||
|
union
|
||||||
|
{
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
Byte BaseNametableAddr : 2;
|
||||||
|
Byte VRAMAddrIncrement : 1;
|
||||||
|
Byte SpritePatternTableAddr : 1;
|
||||||
|
Byte BackgrPatternTableAddr : 1;
|
||||||
|
Byte SpriteSize : 1;
|
||||||
|
Byte MasterSlaveSelect : 1;
|
||||||
|
Byte VBlankNMI : 1;
|
||||||
|
} Flag;
|
||||||
|
|
||||||
|
Byte Raw;
|
||||||
|
} ppuctrl;
|
||||||
|
|
||||||
|
union
|
||||||
|
{
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
Byte Greyscale : 1;
|
||||||
|
Byte BackgroundOnLeft : 1;
|
||||||
|
Byte SpriteOnLeft : 1;
|
||||||
|
Byte ShowBackground : 1;
|
||||||
|
Byte ShowSprites : 1;
|
||||||
|
Byte EmphasizeRed : 1;
|
||||||
|
Byte EmphasizeGreen : 1;
|
||||||
|
Byte EmphasizeBlue : 1;
|
||||||
|
} Flag;
|
||||||
|
|
||||||
|
Byte Raw;
|
||||||
|
} ppumask;
|
||||||
|
|
||||||
|
union
|
||||||
|
{
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
Byte Unused : 5;
|
||||||
|
Byte SpriteOverflow : 1;
|
||||||
|
Byte SpriteZeroHit : 1;
|
||||||
|
Byte VBlankStarted : 1;
|
||||||
|
} Flag;
|
||||||
|
|
||||||
|
Byte Raw;
|
||||||
|
} ppustatus;
|
||||||
|
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
Byte x;
|
||||||
|
Byte y;
|
||||||
|
} ppuscroll;
|
||||||
|
|
||||||
|
Address ppuaddr;
|
||||||
|
|
||||||
|
uint16_t x, y;
|
||||||
|
Byte addressLatch = 0;
|
||||||
|
|
||||||
|
private:
|
||||||
|
bool isFrameDone = false;
|
||||||
|
Bus* bus;
|
||||||
|
};
|
17
src/Types.hpp
Normal file
17
src/Types.hpp
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
|
||||||
|
using Byte = uint8_t;
|
||||||
|
using Word = uint16_t;
|
||||||
|
|
||||||
|
union Address
|
||||||
|
{
|
||||||
|
struct
|
||||||
|
{
|
||||||
|
Byte lo;
|
||||||
|
Byte hi;
|
||||||
|
} Bytes;
|
||||||
|
|
||||||
|
Word Raw;
|
||||||
|
};
|
101
src/debugger/CPUWatcher.cpp
Normal file
101
src/debugger/CPUWatcher.cpp
Normal file
|
@ -0,0 +1,101 @@
|
||||||
|
#include "CPUWatcher.hpp"
|
||||||
|
|
||||||
|
#include <iomanip>
|
||||||
|
|
||||||
|
#include <imgui/imgui.h>
|
||||||
|
#include "../CPU.hpp"
|
||||||
|
|
||||||
|
CPUWatcher::CPUWatcher(CPU* cpu) :
|
||||||
|
DebugWindow("CPU Watch"), cpu(cpu)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void CPUWatcher::OnRender()
|
||||||
|
{
|
||||||
|
ImGui::SetNextWindowSize(ImVec2(400, 400), ImGuiCond_FirstUseEver);
|
||||||
|
if (!ImGui::Begin(title.c_str(), &isOpen))
|
||||||
|
{
|
||||||
|
ImGui::End();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::Text("Registers");
|
||||||
|
if (ImGui::BeginTable("Registers", 6))
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("A");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("X");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Y");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("PC");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("SP");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("P");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%02X", cpu->acc);
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%02X", cpu->idx);
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%02X", cpu->idy);
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%04X", cpu->pc);
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%02X", cpu->sp);
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%02X", cpu->status.Raw);
|
||||||
|
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
ImGui::Text("Status Flag Breakdown");
|
||||||
|
if (ImGui::BeginTable("Status", 8))
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("N");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("V");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("U");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("B");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("D");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("I");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Z");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("C");
|
||||||
|
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(cpu->status.Flag.Negative ? "1" : "-");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(cpu->status.Flag.Overflow ? "1" : "-");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(cpu->status.Flag.NoEffect ? "1" : "-");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(cpu->status.Flag.Break ? "1" : "-");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(cpu->status.Flag.Decimal ? "1" : "-");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(cpu->status.Flag.InterruptDisable ? "1" : "-");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(cpu->status.Flag.Zero ? "1" : "-");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(cpu->status.Flag.Carry ? "1" : "-");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
|
||||||
|
ImGui::Text("Halted: %s", cpu->halted ? "Yes" : "No");
|
||||||
|
|
||||||
|
ImGui::End();
|
||||||
|
}
|
17
src/debugger/CPUWatcher.hpp
Normal file
17
src/debugger/CPUWatcher.hpp
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "DebugWindow.hpp"
|
||||||
|
|
||||||
|
class CPU;
|
||||||
|
|
||||||
|
class CPUWatcher :
|
||||||
|
public DebugWindow
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
CPUWatcher(CPU* cpu);
|
||||||
|
|
||||||
|
virtual void OnRender() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
CPU* cpu;
|
||||||
|
};
|
19
src/debugger/DebugWindow.hpp
Normal file
19
src/debugger/DebugWindow.hpp
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
class DebugWindow
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
DebugWindow(const std::string& title) :
|
||||||
|
title(title)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual ~DebugWindow() = default;
|
||||||
|
virtual void OnRender() = 0;
|
||||||
|
|
||||||
|
public:
|
||||||
|
const std::string title;
|
||||||
|
bool isOpen = false;
|
||||||
|
};
|
127
src/debugger/Debugger.cpp
Normal file
127
src/debugger/Debugger.cpp
Normal file
|
@ -0,0 +1,127 @@
|
||||||
|
#include "Debugger.hpp"
|
||||||
|
|
||||||
|
#include <imgui/imgui.h>
|
||||||
|
#include <imgui/imgui_internal.h>
|
||||||
|
|
||||||
|
#include "../Bus.hpp"
|
||||||
|
#include "CPUWatcher.hpp"
|
||||||
|
#include "PPUWatcher.hpp"
|
||||||
|
#include "Disassembler.hpp"
|
||||||
|
#include "MemoryViewer.hpp"
|
||||||
|
#include "NametableViewer.hpp"
|
||||||
|
|
||||||
|
Debugger::Debugger(Bus* bus) :
|
||||||
|
bus(bus)
|
||||||
|
{
|
||||||
|
windows.push_back(new CPUWatcher(&bus->cpu));
|
||||||
|
windows.push_back(new PPUWatcher(&bus->ppu));
|
||||||
|
windows.push_back(new Disassembler(&bus->cpu));
|
||||||
|
windows.push_back(new MemoryViewer(bus));
|
||||||
|
windows.push_back(new NametableViewer(bus));
|
||||||
|
}
|
||||||
|
|
||||||
|
Debugger::~Debugger()
|
||||||
|
{
|
||||||
|
for (DebugWindow* window : windows)
|
||||||
|
delete window;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Debugger::Update()
|
||||||
|
{
|
||||||
|
if (running)
|
||||||
|
return bus->Frame();
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void Debugger::Render()
|
||||||
|
{
|
||||||
|
ImGui::SetNextWindowSize(ImVec2(400, 600), ImGuiCond_FirstUseEver);
|
||||||
|
if (!ImGui::Begin("Debugger", (bool*)0, ImGuiWindowFlags_MenuBar))
|
||||||
|
{
|
||||||
|
ImGui::End();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ImGui::BeginMenuBar())
|
||||||
|
{
|
||||||
|
if (ImGui::BeginMenu("Tools"))
|
||||||
|
{
|
||||||
|
for (DebugWindow* window : windows)
|
||||||
|
{
|
||||||
|
ImGui::MenuItem(window->title.c_str(), NULL, &window->isOpen);
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::EndMenu();
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::EndMenuBar();
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
|
||||||
|
if (ImGui::CollapsingHeader("Controls", ImGuiTreeNodeFlags_DefaultOpen))
|
||||||
|
{
|
||||||
|
ImGui::PushItemFlag(ImGuiItemFlags_Disabled, running);
|
||||||
|
if (running)
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_Button, ImVec4{ 0.7f, 0.7f, 0.7f, 0.7f });
|
||||||
|
|
||||||
|
if (ImGui::Button("Single Step"))
|
||||||
|
bus->Instruction();
|
||||||
|
|
||||||
|
ImGui::SameLine();
|
||||||
|
|
||||||
|
if (ImGui::Button("Single Frame"))
|
||||||
|
bus->Frame();
|
||||||
|
|
||||||
|
if (running)
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
ImGui::PopItemFlag();
|
||||||
|
|
||||||
|
ImGui::SameLine();
|
||||||
|
|
||||||
|
if (ImGui::Button(running ? "Pause" : "Run"))
|
||||||
|
running = !running;
|
||||||
|
|
||||||
|
ImGui::PushItemFlag(ImGuiItemFlags_Disabled, running);
|
||||||
|
if (running)
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_Button, ImVec4{ 0.7f, 0.7f, 0.7f, 0.7f });
|
||||||
|
|
||||||
|
if (ImGui::Button("Reset"))
|
||||||
|
{
|
||||||
|
bus->Reset();
|
||||||
|
if (overrideResetVector)
|
||||||
|
bus->cpu.pc.Raw = resetVector;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::SameLine();
|
||||||
|
|
||||||
|
if (ImGui::Button("Reboot"))
|
||||||
|
{
|
||||||
|
bus->Reboot();
|
||||||
|
if (overrideResetVector)
|
||||||
|
bus->cpu.pc.Raw = resetVector;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (running)
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
ImGui::PopItemFlag();
|
||||||
|
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
|
||||||
|
ImGui::Text("Override Reset Vector: ");
|
||||||
|
ImGui::SameLine();
|
||||||
|
if (ImGui::Button(overrideResetVector ? "Yes" : "No"))
|
||||||
|
overrideResetVector = !overrideResetVector;
|
||||||
|
|
||||||
|
ImGui::InputScalar("Reset Vector", ImGuiDataType_U16, &resetVector, (const void*)0, (const void*)0, "%04X", ImGuiInputTextFlags_CharsHexadecimal);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (DebugWindow* window : windows)
|
||||||
|
{
|
||||||
|
if (window->isOpen) window->OnRender();
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::End();
|
||||||
|
}
|
27
src/debugger/Debugger.hpp
Normal file
27
src/debugger/Debugger.hpp
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
#include "DebugWindow.hpp"
|
||||||
|
|
||||||
|
class Bus;
|
||||||
|
|
||||||
|
class Debugger
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
Debugger(Bus* bus);
|
||||||
|
~Debugger();
|
||||||
|
|
||||||
|
bool Update();
|
||||||
|
void Render();
|
||||||
|
|
||||||
|
public:
|
||||||
|
bool isOpen = true;
|
||||||
|
|
||||||
|
private:
|
||||||
|
Bus* bus;
|
||||||
|
bool running = false;
|
||||||
|
bool overrideResetVector = false;
|
||||||
|
uint16_t resetVector = 0x0000;
|
||||||
|
|
||||||
|
std::vector<DebugWindow*> windows;
|
||||||
|
};
|
177
src/debugger/Disassembler.cpp
Normal file
177
src/debugger/Disassembler.cpp
Normal file
|
@ -0,0 +1,177 @@
|
||||||
|
#include "Disassembler.hpp"
|
||||||
|
|
||||||
|
#include <iomanip>
|
||||||
|
#include <imgui/imgui.h>
|
||||||
|
#include "../Mapper.hpp"
|
||||||
|
#include "../CPU.hpp"
|
||||||
|
|
||||||
|
#define FORMAT std::setfill('0') << std::setw(4) << std::hex << std::uppercase
|
||||||
|
|
||||||
|
Disassembler::Disassembler(CPU* cpu) :
|
||||||
|
DebugWindow("Disassembler"), cpu(cpu)
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void Disassembler::OnRender()
|
||||||
|
{
|
||||||
|
static bool scrollCenter = true;
|
||||||
|
|
||||||
|
ImGui::SetNextWindowSize(ImVec2(400, 400), ImGuiCond_FirstUseEver);
|
||||||
|
|
||||||
|
if (!ImGui::Begin(title.c_str(), &isOpen))
|
||||||
|
{
|
||||||
|
ImGui::End();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string disassembly;
|
||||||
|
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_Text, ImVec4{ 0.8f, 0.8f, 0.8f, 1.0f });
|
||||||
|
if (cpu->pastPCs.size() < 50)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < 50 - cpu->pastPCs.size(); i++)
|
||||||
|
{
|
||||||
|
ImGui::Text("-");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (Word pc : cpu->pastPCs)
|
||||||
|
{
|
||||||
|
Disassemble(disassembly, pc);
|
||||||
|
ImGui::Text("- %s", disassembly.c_str());
|
||||||
|
}
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
|
||||||
|
uint16_t pc = cpu->pc.Raw;
|
||||||
|
Disassemble(disassembly, pc);
|
||||||
|
ImGui::Text("> %s", disassembly.c_str());
|
||||||
|
|
||||||
|
if (scrollCenter)
|
||||||
|
{
|
||||||
|
ImGui::SetScrollHereY(0.5f);
|
||||||
|
scrollCenter = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_Text, ImVec4{ 0.8f, 0.8f, 0.8f, 1.0f });
|
||||||
|
for (int i = 0; i < 50; i++)
|
||||||
|
{
|
||||||
|
Disassemble(disassembly, pc);
|
||||||
|
ImGui::Text("+ %s", disassembly.c_str());
|
||||||
|
}
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
|
||||||
|
ImGui::End();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Disassembler::Disassemble(std::string& target, uint16_t& pc)
|
||||||
|
{
|
||||||
|
std::stringstream ss;
|
||||||
|
ss << FORMAT << pc << ": ";
|
||||||
|
Instruction* currentInstr = &cpu->InstructionTable[cpu->Read(pc)];
|
||||||
|
|
||||||
|
for (int i = 0; i < currentInstr->Size; i++)
|
||||||
|
{
|
||||||
|
ss << FORMAT << std::setw(2) << (Word)cpu->Read(pc + i) << " ";
|
||||||
|
}
|
||||||
|
ss << std::string(15 - ss.str().size(), ' ') << currentInstr->Mnemonic << " ";
|
||||||
|
|
||||||
|
Address absoluteAddress;
|
||||||
|
switch (currentInstr->AddrType)
|
||||||
|
{
|
||||||
|
case Addressing::ACC:
|
||||||
|
ss << "A";
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Addressing::ABS:
|
||||||
|
{
|
||||||
|
absoluteAddress.Bytes.lo = cpu->Read(pc + 1);
|
||||||
|
absoluteAddress.Bytes.hi = cpu->Read(pc + 2);
|
||||||
|
|
||||||
|
ss << "$" << FORMAT << absoluteAddress.Raw;
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case Addressing::ABX:
|
||||||
|
{
|
||||||
|
absoluteAddress.Bytes.lo = cpu->Read(pc + 1);
|
||||||
|
absoluteAddress.Bytes.hi = cpu->Read(pc + 2);
|
||||||
|
|
||||||
|
ss << "$" << FORMAT << absoluteAddress.Raw << ",X";
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case Addressing::ABY:
|
||||||
|
{
|
||||||
|
absoluteAddress.Bytes.lo = cpu->Read(pc + 1);
|
||||||
|
absoluteAddress.Bytes.hi = cpu->Read(pc + 2);
|
||||||
|
|
||||||
|
ss << "$" << FORMAT << absoluteAddress.Raw << ",Y";
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case Addressing::IMM:
|
||||||
|
{
|
||||||
|
Word value = cpu->Read(pc + 1);
|
||||||
|
|
||||||
|
ss << "#$" << FORMAT << std::setw(2) << value;
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case Addressing::IMP:
|
||||||
|
break;
|
||||||
|
|
||||||
|
case Addressing::IND:
|
||||||
|
{
|
||||||
|
absoluteAddress.Bytes.lo = cpu->Read(pc + 1);
|
||||||
|
absoluteAddress.Bytes.hi = cpu->Read(pc + 2);
|
||||||
|
|
||||||
|
ss << "($" << FORMAT << absoluteAddress.Raw << ")";
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case Addressing::IDX:
|
||||||
|
{
|
||||||
|
absoluteAddress.Bytes.lo = cpu->Read(pc + 1);
|
||||||
|
|
||||||
|
ss << "($" << FORMAT << std::setw(2) << (Word)absoluteAddress.Bytes.lo << ",X)";
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case Addressing::IDY:
|
||||||
|
{
|
||||||
|
absoluteAddress.Bytes.lo = cpu->Read(pc + 1);
|
||||||
|
|
||||||
|
ss << "($" << FORMAT << std::setw(2) << (Word)absoluteAddress.Bytes.lo << "),Y";
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case Addressing::REL:
|
||||||
|
{
|
||||||
|
absoluteAddress.Bytes.lo = cpu->Read(pc + 1);
|
||||||
|
|
||||||
|
ss << "$" << FORMAT << std::setw(2) << (Word)absoluteAddress.Bytes.lo;
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case Addressing::ZPG:
|
||||||
|
{
|
||||||
|
absoluteAddress.Bytes.lo = cpu->Read(pc + 1);
|
||||||
|
|
||||||
|
ss << "$" << FORMAT << std::setw(2) << (Word)absoluteAddress.Bytes.lo;
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case Addressing::ZPX:
|
||||||
|
{
|
||||||
|
absoluteAddress.Bytes.lo = cpu->Read(pc + 1);
|
||||||
|
|
||||||
|
ss << "$" << FORMAT << std::setw(2) << (Word)absoluteAddress.Bytes.lo << ",X";
|
||||||
|
} break;
|
||||||
|
|
||||||
|
case Addressing::ZPY:
|
||||||
|
{
|
||||||
|
absoluteAddress.Bytes.lo = cpu->Read(pc + 1);
|
||||||
|
|
||||||
|
ss << "$" << FORMAT << std::setw(2) << (Word)absoluteAddress.Bytes.lo << ",Y";
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
|
||||||
|
pc += currentInstr->Size;
|
||||||
|
target = ss.str();
|
||||||
|
}
|
20
src/debugger/Disassembler.hpp
Normal file
20
src/debugger/Disassembler.hpp
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "DebugWindow.hpp"
|
||||||
|
|
||||||
|
class CPU;
|
||||||
|
|
||||||
|
class Disassembler :
|
||||||
|
public DebugWindow
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
Disassembler(CPU* cpu);
|
||||||
|
|
||||||
|
virtual void OnRender() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void Disassemble(std::string& target, uint16_t& pc);
|
||||||
|
|
||||||
|
private:
|
||||||
|
CPU* cpu;
|
||||||
|
};
|
78
src/debugger/MemoryViewer.cpp
Normal file
78
src/debugger/MemoryViewer.cpp
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
#include "MemoryViewer.hpp"
|
||||||
|
|
||||||
|
#include <imgui/imgui.h>
|
||||||
|
#include "../Bus.hpp"
|
||||||
|
|
||||||
|
MemoryViewer::MemoryViewer(Bus* bus) :
|
||||||
|
DebugWindow("Memory Viewer"), bus(bus)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void MemoryViewer::OnRender()
|
||||||
|
{
|
||||||
|
ImGui::SetNextWindowSize(ImVec2(400, 400), ImGuiCond_FirstUseEver);
|
||||||
|
|
||||||
|
if (!ImGui::Begin(title.c_str(), &isOpen))
|
||||||
|
{
|
||||||
|
ImGui::End();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::BeginTabBar("Pages");
|
||||||
|
|
||||||
|
for (Byte page = 0; page < 8; page++)
|
||||||
|
{
|
||||||
|
char title[7];
|
||||||
|
std::sprintf(title, "Page %d", page);
|
||||||
|
if (ImGui::BeginTabItem(title))
|
||||||
|
{
|
||||||
|
DrawPage(page);
|
||||||
|
ImGui::EndTabItem();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::EndTabBar();
|
||||||
|
ImGui::End();
|
||||||
|
}
|
||||||
|
|
||||||
|
void MemoryViewer::DrawPage(Byte page)
|
||||||
|
{
|
||||||
|
Word baseAddr = ((Word)page << 8);
|
||||||
|
if (ImGui::BeginTable("memorymap", 17))
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_Text, ImVec4(0.7f, 0.7f, 0.7f, 1.0f));
|
||||||
|
for (Byte header = 0x0; header < 0x10; header++)
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%X", header);
|
||||||
|
}
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
|
||||||
|
for (Byte hi = 0x0; hi <= 0xF; hi++)
|
||||||
|
{
|
||||||
|
Byte hiOffset = hi << 4;
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_Text, ImVec4(0.7f, 0.7f, 0.7f, 1.0f));
|
||||||
|
ImGui::Text("%04X", baseAddr | hiOffset);
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
|
||||||
|
for (Byte lo = 0x0; lo <= 0xF; lo++)
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
|
||||||
|
Byte entry = bus->RAM[baseAddr | hiOffset | lo];
|
||||||
|
|
||||||
|
if (entry == 0x00)
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_Text, ImVec4(0.7f, 0.7f, 0.7f, 1.0f));
|
||||||
|
|
||||||
|
ImGui::Text("%02X", entry);
|
||||||
|
|
||||||
|
if (entry == 0x00)
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
}
|
21
src/debugger/MemoryViewer.hpp
Normal file
21
src/debugger/MemoryViewer.hpp
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "DebugWindow.hpp"
|
||||||
|
#include "../Types.hpp"
|
||||||
|
|
||||||
|
class Bus;
|
||||||
|
|
||||||
|
class MemoryViewer :
|
||||||
|
public DebugWindow
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
MemoryViewer(Bus* bus);
|
||||||
|
|
||||||
|
virtual void OnRender() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void DrawPage(Byte page);
|
||||||
|
|
||||||
|
private:
|
||||||
|
Bus* bus;
|
||||||
|
};
|
78
src/debugger/NametableViewer.cpp
Normal file
78
src/debugger/NametableViewer.cpp
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
#include "NametableViewer.hpp"
|
||||||
|
|
||||||
|
#include <imgui/imgui.h>
|
||||||
|
#include "../Bus.hpp"
|
||||||
|
|
||||||
|
NametableViewer::NametableViewer(Bus* bus) :
|
||||||
|
DebugWindow("Nametable Viewer"), bus(bus)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void NametableViewer::OnRender()
|
||||||
|
{
|
||||||
|
ImGui::SetNextWindowSize(ImVec2(400, 400), ImGuiCond_FirstUseEver);
|
||||||
|
if (!ImGui::Begin(title.c_str(), &isOpen))
|
||||||
|
{
|
||||||
|
ImGui::End();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::BeginTabBar("Nametables");
|
||||||
|
for (uint8_t index = 0; index < 4; index++)
|
||||||
|
{
|
||||||
|
char baseAddress[6];
|
||||||
|
std::sprintf(baseAddress, "$2%X00", index * 4);
|
||||||
|
if (ImGui::BeginTabItem(baseAddress))
|
||||||
|
{
|
||||||
|
DisplayNametable(index);
|
||||||
|
ImGui::EndTabItem();
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
ImGui::EndTabBar();
|
||||||
|
|
||||||
|
ImGui::End();
|
||||||
|
}
|
||||||
|
|
||||||
|
void NametableViewer::DisplayNametable(uint8_t index)
|
||||||
|
{
|
||||||
|
Word baseAddr = 0x400 * index;
|
||||||
|
Word displayBaseAddr = 0x2000 + baseAddr;
|
||||||
|
if (ImGui::BeginTable("memorymap", 17))
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_Text, ImVec4(0.7f, 0.7f, 0.7f, 1.0f));
|
||||||
|
for (Byte header = 0x0; header < 0x10; header++)
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%X", header);
|
||||||
|
}
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
|
||||||
|
for (Word hi = 0x0; hi <= 0x3F; hi++)
|
||||||
|
{
|
||||||
|
Byte hiOffset = hi << 4;
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_Text, ImVec4(0.7f, 0.7f, 0.7f, 1.0f));
|
||||||
|
ImGui::Text("%04X", displayBaseAddr | hiOffset);
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
|
||||||
|
for (Byte lo = 0x0; lo <= 0xF; lo++)
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
|
||||||
|
Byte entry = bus->VRAM[baseAddr | hiOffset | lo];
|
||||||
|
|
||||||
|
if (entry == 0x00)
|
||||||
|
ImGui::PushStyleColor(ImGuiCol_Text, ImVec4(0.7f, 0.7f, 0.7f, 1.0f));
|
||||||
|
|
||||||
|
ImGui::Text("%02X", entry);
|
||||||
|
|
||||||
|
if (entry == 0x00)
|
||||||
|
ImGui::PopStyleColor();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
}
|
20
src/debugger/NametableViewer.hpp
Normal file
20
src/debugger/NametableViewer.hpp
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "DebugWindow.hpp"
|
||||||
|
|
||||||
|
class Bus;
|
||||||
|
|
||||||
|
class NametableViewer :
|
||||||
|
public DebugWindow
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
NametableViewer(Bus* bus);
|
||||||
|
|
||||||
|
virtual void OnRender() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
void DisplayNametable(uint8_t index);
|
||||||
|
|
||||||
|
private:
|
||||||
|
Bus* bus;
|
||||||
|
};
|
152
src/debugger/PPUWatcher.cpp
Normal file
152
src/debugger/PPUWatcher.cpp
Normal file
|
@ -0,0 +1,152 @@
|
||||||
|
#include "PPUWatcher.hpp"
|
||||||
|
|
||||||
|
#include <imgui/imgui.h>
|
||||||
|
#include "../PPU.hpp"
|
||||||
|
|
||||||
|
PPUWatcher::PPUWatcher(PPU* ppu) :
|
||||||
|
DebugWindow("PPU Watch"), ppu(ppu)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void PPUWatcher::OnRender()
|
||||||
|
{
|
||||||
|
ImGui::SetNextWindowSize(ImVec2(400, 400), ImGuiCond_FirstUseEver);
|
||||||
|
if (!ImGui::Begin(title.c_str(), &isOpen))
|
||||||
|
{
|
||||||
|
ImGui::End();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::Text("On Pixel (%d, %d)", ppu->x, ppu->y);
|
||||||
|
|
||||||
|
ImGui::Separator();
|
||||||
|
|
||||||
|
if (ImGui::CollapsingHeader("PPUCTRL"))
|
||||||
|
{
|
||||||
|
if (ImGui::BeginTable("ppuctrl", 2))
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Base Nametable Addr");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("$%04X", 0x2000 + 0x400 * ppu->ppuctrl.Flag.BaseNametableAddr);
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("VRAM Addr Increment");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%d", ppu->ppuctrl.Flag.VRAMAddrIncrement ? 32 : 1);
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Sprite Pattern Table Addr");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("$%04X", ppu->ppuctrl.Flag.SpritePatternTableAddr ? 0x1000 : 0x0000);
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Backgr Pattern Table Addr");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("$%04X", ppu->ppuctrl.Flag.BackgrPatternTableAddr ? 0x1000 : 0x0000);
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Master/Slave");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(ppu->ppuctrl.Flag.MasterSlaveSelect ? "Output to EXT" : "Read from EXT");
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("VBlank NMI Generation");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(ppu->ppuctrl.Flag.VBlankNMI ? "On" : "Off");
|
||||||
|
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ImGui::CollapsingHeader("PPUMASK"))
|
||||||
|
{
|
||||||
|
if (ImGui::BeginTable("ppumask", 2))
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Greyscale");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(ppu->ppumask.Flag.Greyscale ? "On" : "Off");
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Left Col Background");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(ppu->ppumask.Flag.BackgroundOnLeft ? "Show" : "Hide");
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Left Col Sprites");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(ppu->ppumask.Flag.SpriteOnLeft ? "Show" : "Hide");
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Show Background");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(ppu->ppumask.Flag.ShowBackground ? "On" : "Off");
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Show Sprites");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(ppu->ppumask.Flag.ShowSprites ? "On" : "Off");
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Emphasized Colors");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%s%s%s", ppu->ppumask.Flag.EmphasizeRed ? "R" : "-", ppu->ppumask.Flag.EmphasizeGreen ? "G" : "-", ppu->ppumask.Flag.EmphasizeBlue ? "B" : "-");
|
||||||
|
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ImGui::CollapsingHeader("PPUSTATUS"))
|
||||||
|
{
|
||||||
|
if (ImGui::BeginTable("ppustatus", 2))
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Lower 5 Bits");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%02X", ppu->ppustatus.Flag.Unused);
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Sprite Overflow");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(ppu->ppustatus.Flag.SpriteOverflow ? "Yes" : "No");
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Sprite 0 Hit");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(ppu->ppustatus.Flag.SpriteZeroHit ? "Yes" : "No");
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("VBlank Started");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text(ppu->ppustatus.Flag.VBlankStarted ? "Yes" : "No");
|
||||||
|
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ImGui::CollapsingHeader("PPUSCROLL & PPUADDR"))
|
||||||
|
{
|
||||||
|
if (ImGui::BeginTable("ppuscrolladdr", 2))
|
||||||
|
{
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Scroll X");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%d", ppu->ppuscroll.x);
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Scroll Y");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("%d", ppu->ppuscroll.x);
|
||||||
|
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("Address");
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::Text("$%04X", ppu->ppuaddr.Raw);
|
||||||
|
|
||||||
|
ImGui::EndTable();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ImGui::End();
|
||||||
|
}
|
17
src/debugger/PPUWatcher.hpp
Normal file
17
src/debugger/PPUWatcher.hpp
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "DebugWindow.hpp"
|
||||||
|
|
||||||
|
class PPU;
|
||||||
|
|
||||||
|
class PPUWatcher :
|
||||||
|
public DebugWindow
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
PPUWatcher(PPU* ppu);
|
||||||
|
|
||||||
|
virtual void OnRender() override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
PPU* ppu;
|
||||||
|
};
|
52
src/gfx/Window.cpp
Normal file
52
src/gfx/Window.cpp
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
#include "Window.hpp"
|
||||||
|
|
||||||
|
#include <stdexcept>
|
||||||
|
#include <imgui/backends/imgui_impl_glfw.h>
|
||||||
|
#include <imgui/backends/imgui_impl_opengl3.h>
|
||||||
|
#include <imgui/imgui.h>
|
||||||
|
|
||||||
|
Window::Window(uint16_t width, uint16_t height, const std::string& title) :
|
||||||
|
handle(nullptr)
|
||||||
|
{
|
||||||
|
handle = glfwCreateWindow(width, height, title.c_str(), nullptr, nullptr);
|
||||||
|
if (handle == nullptr)
|
||||||
|
{
|
||||||
|
const char* err;
|
||||||
|
int code = glfwGetError(&err);
|
||||||
|
throw std::runtime_error(std::string(err) + " (" + std::to_string(code) + ")");
|
||||||
|
}
|
||||||
|
|
||||||
|
glfwMakeContextCurrent(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
Window::~Window()
|
||||||
|
{
|
||||||
|
if(handle)
|
||||||
|
glfwDestroyWindow(handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Window::Begin()
|
||||||
|
{
|
||||||
|
glClearColor(0.0f, 0.0f, 0.0f, 0.0f);
|
||||||
|
glClear(GL_COLOR_BUFFER_BIT);
|
||||||
|
|
||||||
|
ImGui_ImplOpenGL3_NewFrame();
|
||||||
|
ImGui_ImplGlfw_NewFrame();
|
||||||
|
ImGui::NewFrame();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Window::End()
|
||||||
|
{
|
||||||
|
ImGui::Render();
|
||||||
|
ImGui_ImplOpenGL3_RenderDrawData(ImGui::GetDrawData());
|
||||||
|
|
||||||
|
if (ImGui::GetIO().ConfigFlags & ImGuiConfigFlags_ViewportsEnable)
|
||||||
|
{
|
||||||
|
GLFWwindow* backup_current_context = glfwGetCurrentContext();
|
||||||
|
ImGui::UpdatePlatformWindows();
|
||||||
|
ImGui::RenderPlatformWindowsDefault();
|
||||||
|
glfwMakeContextCurrent(backup_current_context);
|
||||||
|
}
|
||||||
|
|
||||||
|
glfwSwapBuffers(handle);
|
||||||
|
}
|
23
src/gfx/Window.hpp
Normal file
23
src/gfx/Window.hpp
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
#include <string>
|
||||||
|
|
||||||
|
#include <glad/glad.h>
|
||||||
|
#include <glfw/glfw3.h>
|
||||||
|
|
||||||
|
class Window
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
Window(uint16_t width, uint16_t height, const std::string& title);
|
||||||
|
~Window();
|
||||||
|
|
||||||
|
inline bool ShouldClose() { return glfwWindowShouldClose(handle); }
|
||||||
|
inline GLFWwindow* GetNativeWindow() { return handle; }
|
||||||
|
|
||||||
|
void Begin();
|
||||||
|
void End();
|
||||||
|
|
||||||
|
private:
|
||||||
|
GLFWwindow* handle;
|
||||||
|
};
|
10
src/main.cpp
Normal file
10
src/main.cpp
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
#include "Application.hpp"
|
||||||
|
#include "Log.hpp"
|
||||||
|
|
||||||
|
int main(int argc, char** argv)
|
||||||
|
{
|
||||||
|
Log::Init();
|
||||||
|
Application::Launch();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
40
src/mappers/Mapper000.cpp
Normal file
40
src/mappers/Mapper000.cpp
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
#include "Mapper000.hpp"
|
||||||
|
#include "../Log.hpp"
|
||||||
|
|
||||||
|
#include <fstream>
|
||||||
|
#include "../Cartridge.hpp"
|
||||||
|
|
||||||
|
Mapper000::Mapper000(Header& header, std::ifstream& ifs)
|
||||||
|
{
|
||||||
|
LOG_CORE_INFO("Allocating PRG ROM");
|
||||||
|
PRG_ROM = std::vector<Byte>(0x4000);
|
||||||
|
ifs.read((char*)PRG_ROM.data(), 0x4000);
|
||||||
|
|
||||||
|
LOG_CORE_INFO("Allocating CHR ROM");
|
||||||
|
CHR_ROM = std::vector<Byte>(0x2000);
|
||||||
|
ifs.read((char*)CHR_ROM.data(), 0x2000);
|
||||||
|
}
|
||||||
|
|
||||||
|
Byte Mapper000::ReadCPU(Word addr)
|
||||||
|
{
|
||||||
|
if (0x8000 <= addr && addr <= 0xFFFF)
|
||||||
|
{
|
||||||
|
return PRG_ROM[addr & 0x03FFF];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Byte Mapper000::ReadPPU(Word addr)
|
||||||
|
{
|
||||||
|
if (0x0000 <= addr && addr <= 0x1FFF)
|
||||||
|
{
|
||||||
|
return CHR_ROM[addr];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Mapper000::WriteCPU(Word addr, Byte val)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
void Mapper000::WritePPU(Word addr, Byte val)
|
||||||
|
{
|
||||||
|
}
|
19
src/mappers/Mapper000.hpp
Normal file
19
src/mappers/Mapper000.hpp
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
#include <fstream>
|
||||||
|
#include "../Mapper.hpp"
|
||||||
|
|
||||||
|
struct Header;
|
||||||
|
|
||||||
|
class Mapper000 :
|
||||||
|
public Mapper
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
Mapper000(Header& header, std::ifstream& ifs);
|
||||||
|
|
||||||
|
virtual Byte ReadCPU(Word addr) override;
|
||||||
|
virtual Byte ReadPPU(Word addr) override;
|
||||||
|
virtual void WriteCPU(Word addr, Byte val) override;
|
||||||
|
virtual void WritePPU(Word addr, Byte val) override;
|
||||||
|
};
|
9
vendor/glad/CMakeLists.txt
vendored
Normal file
9
vendor/glad/CMakeLists.txt
vendored
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
project(glad)
|
||||||
|
|
||||||
|
add_library(glad STATIC
|
||||||
|
"src/glad.c"
|
||||||
|
)
|
||||||
|
|
||||||
|
target_include_directories(glad PUBLIC
|
||||||
|
"include"
|
||||||
|
)
|
311
vendor/glad/include/KHR/khrplatform.h
vendored
Normal file
311
vendor/glad/include/KHR/khrplatform.h
vendored
Normal file
|
@ -0,0 +1,311 @@
|
||||||
|
#ifndef __khrplatform_h_
|
||||||
|
#define __khrplatform_h_
|
||||||
|
|
||||||
|
/*
|
||||||
|
** Copyright (c) 2008-2018 The Khronos Group Inc.
|
||||||
|
**
|
||||||
|
** Permission is hereby granted, free of charge, to any person obtaining a
|
||||||
|
** copy of this software and/or associated documentation files (the
|
||||||
|
** "Materials"), to deal in the Materials without restriction, including
|
||||||
|
** without limitation the rights to use, copy, modify, merge, publish,
|
||||||
|
** distribute, sublicense, and/or sell copies of the Materials, and to
|
||||||
|
** permit persons to whom the Materials are furnished to do so, subject to
|
||||||
|
** the following conditions:
|
||||||
|
**
|
||||||
|
** The above copyright notice and this permission notice shall be included
|
||||||
|
** in all copies or substantial portions of the Materials.
|
||||||
|
**
|
||||||
|
** THE MATERIALS ARE PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||||||
|
** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
|
||||||
|
** IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
|
||||||
|
** CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
|
||||||
|
** TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
|
||||||
|
** MATERIALS OR THE USE OR OTHER DEALINGS IN THE MATERIALS.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* Khronos platform-specific types and definitions.
|
||||||
|
*
|
||||||
|
* The master copy of khrplatform.h is maintained in the Khronos EGL
|
||||||
|
* Registry repository at https://github.com/KhronosGroup/EGL-Registry
|
||||||
|
* The last semantic modification to khrplatform.h was at commit ID:
|
||||||
|
* 67a3e0864c2d75ea5287b9f3d2eb74a745936692
|
||||||
|
*
|
||||||
|
* Adopters may modify this file to suit their platform. Adopters are
|
||||||
|
* encouraged to submit platform specific modifications to the Khronos
|
||||||
|
* group so that they can be included in future versions of this file.
|
||||||
|
* Please submit changes by filing pull requests or issues on
|
||||||
|
* the EGL Registry repository linked above.
|
||||||
|
*
|
||||||
|
*
|
||||||
|
* See the Implementer's Guidelines for information about where this file
|
||||||
|
* should be located on your system and for more details of its use:
|
||||||
|
* http://www.khronos.org/registry/implementers_guide.pdf
|
||||||
|
*
|
||||||
|
* This file should be included as
|
||||||
|
* #include <KHR/khrplatform.h>
|
||||||
|
* by Khronos client API header files that use its types and defines.
|
||||||
|
*
|
||||||
|
* The types in khrplatform.h should only be used to define API-specific types.
|
||||||
|
*
|
||||||
|
* Types defined in khrplatform.h:
|
||||||
|
* khronos_int8_t signed 8 bit
|
||||||
|
* khronos_uint8_t unsigned 8 bit
|
||||||
|
* khronos_int16_t signed 16 bit
|
||||||
|
* khronos_uint16_t unsigned 16 bit
|
||||||
|
* khronos_int32_t signed 32 bit
|
||||||
|
* khronos_uint32_t unsigned 32 bit
|
||||||
|
* khronos_int64_t signed 64 bit
|
||||||
|
* khronos_uint64_t unsigned 64 bit
|
||||||
|
* khronos_intptr_t signed same number of bits as a pointer
|
||||||
|
* khronos_uintptr_t unsigned same number of bits as a pointer
|
||||||
|
* khronos_ssize_t signed size
|
||||||
|
* khronos_usize_t unsigned size
|
||||||
|
* khronos_float_t signed 32 bit floating point
|
||||||
|
* khronos_time_ns_t unsigned 64 bit time in nanoseconds
|
||||||
|
* khronos_utime_nanoseconds_t unsigned time interval or absolute time in
|
||||||
|
* nanoseconds
|
||||||
|
* khronos_stime_nanoseconds_t signed time interval in nanoseconds
|
||||||
|
* khronos_boolean_enum_t enumerated boolean type. This should
|
||||||
|
* only be used as a base type when a client API's boolean type is
|
||||||
|
* an enum. Client APIs which use an integer or other type for
|
||||||
|
* booleans cannot use this as the base type for their boolean.
|
||||||
|
*
|
||||||
|
* Tokens defined in khrplatform.h:
|
||||||
|
*
|
||||||
|
* KHRONOS_FALSE, KHRONOS_TRUE Enumerated boolean false/true values.
|
||||||
|
*
|
||||||
|
* KHRONOS_SUPPORT_INT64 is 1 if 64 bit integers are supported; otherwise 0.
|
||||||
|
* KHRONOS_SUPPORT_FLOAT is 1 if floats are supported; otherwise 0.
|
||||||
|
*
|
||||||
|
* Calling convention macros defined in this file:
|
||||||
|
* KHRONOS_APICALL
|
||||||
|
* KHRONOS_APIENTRY
|
||||||
|
* KHRONOS_APIATTRIBUTES
|
||||||
|
*
|
||||||
|
* These may be used in function prototypes as:
|
||||||
|
*
|
||||||
|
* KHRONOS_APICALL void KHRONOS_APIENTRY funcname(
|
||||||
|
* int arg1,
|
||||||
|
* int arg2) KHRONOS_APIATTRIBUTES;
|
||||||
|
*/
|
||||||
|
|
||||||
|
#if defined(__SCITECH_SNAP__) && !defined(KHRONOS_STATIC)
|
||||||
|
# define KHRONOS_STATIC 1
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*-------------------------------------------------------------------------
|
||||||
|
* Definition of KHRONOS_APICALL
|
||||||
|
*-------------------------------------------------------------------------
|
||||||
|
* This precedes the return type of the function in the function prototype.
|
||||||
|
*/
|
||||||
|
#if defined(KHRONOS_STATIC)
|
||||||
|
/* If the preprocessor constant KHRONOS_STATIC is defined, make the
|
||||||
|
* header compatible with static linking. */
|
||||||
|
# define KHRONOS_APICALL
|
||||||
|
#elif defined(_WIN32)
|
||||||
|
# define KHRONOS_APICALL __declspec(dllimport)
|
||||||
|
#elif defined (__SYMBIAN32__)
|
||||||
|
# define KHRONOS_APICALL IMPORT_C
|
||||||
|
#elif defined(__ANDROID__)
|
||||||
|
# define KHRONOS_APICALL __attribute__((visibility("default")))
|
||||||
|
#else
|
||||||
|
# define KHRONOS_APICALL
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*-------------------------------------------------------------------------
|
||||||
|
* Definition of KHRONOS_APIENTRY
|
||||||
|
*-------------------------------------------------------------------------
|
||||||
|
* This follows the return type of the function and precedes the function
|
||||||
|
* name in the function prototype.
|
||||||
|
*/
|
||||||
|
#if defined(_WIN32) && !defined(_WIN32_WCE) && !defined(__SCITECH_SNAP__)
|
||||||
|
/* Win32 but not WinCE */
|
||||||
|
# define KHRONOS_APIENTRY __stdcall
|
||||||
|
#else
|
||||||
|
# define KHRONOS_APIENTRY
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*-------------------------------------------------------------------------
|
||||||
|
* Definition of KHRONOS_APIATTRIBUTES
|
||||||
|
*-------------------------------------------------------------------------
|
||||||
|
* This follows the closing parenthesis of the function prototype arguments.
|
||||||
|
*/
|
||||||
|
#if defined (__ARMCC_2__)
|
||||||
|
#define KHRONOS_APIATTRIBUTES __softfp
|
||||||
|
#else
|
||||||
|
#define KHRONOS_APIATTRIBUTES
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*-------------------------------------------------------------------------
|
||||||
|
* basic type definitions
|
||||||
|
*-----------------------------------------------------------------------*/
|
||||||
|
#if (defined(__STDC_VERSION__) && __STDC_VERSION__ >= 199901L) || defined(__GNUC__) || defined(__SCO__) || defined(__USLC__)
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Using <stdint.h>
|
||||||
|
*/
|
||||||
|
#include <stdint.h>
|
||||||
|
typedef int32_t khronos_int32_t;
|
||||||
|
typedef uint32_t khronos_uint32_t;
|
||||||
|
typedef int64_t khronos_int64_t;
|
||||||
|
typedef uint64_t khronos_uint64_t;
|
||||||
|
#define KHRONOS_SUPPORT_INT64 1
|
||||||
|
#define KHRONOS_SUPPORT_FLOAT 1
|
||||||
|
/*
|
||||||
|
* To support platform where unsigned long cannot be used interchangeably with
|
||||||
|
* inptr_t (e.g. CHERI-extended ISAs), we can use the stdint.h intptr_t.
|
||||||
|
* Ideally, we could just use (u)intptr_t everywhere, but this could result in
|
||||||
|
* ABI breakage if khronos_uintptr_t is changed from unsigned long to
|
||||||
|
* unsigned long long or similar (this results in different C++ name mangling).
|
||||||
|
* To avoid changes for existing platforms, we restrict usage of intptr_t to
|
||||||
|
* platforms where the size of a pointer is larger than the size of long.
|
||||||
|
*/
|
||||||
|
#if defined(__SIZEOF_LONG__) && defined(__SIZEOF_POINTER__)
|
||||||
|
#if __SIZEOF_POINTER__ > __SIZEOF_LONG__
|
||||||
|
#define KHRONOS_USE_INTPTR_T
|
||||||
|
#endif
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#elif defined(__VMS ) || defined(__sgi)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Using <inttypes.h>
|
||||||
|
*/
|
||||||
|
#include <inttypes.h>
|
||||||
|
typedef int32_t khronos_int32_t;
|
||||||
|
typedef uint32_t khronos_uint32_t;
|
||||||
|
typedef int64_t khronos_int64_t;
|
||||||
|
typedef uint64_t khronos_uint64_t;
|
||||||
|
#define KHRONOS_SUPPORT_INT64 1
|
||||||
|
#define KHRONOS_SUPPORT_FLOAT 1
|
||||||
|
|
||||||
|
#elif defined(_WIN32) && !defined(__SCITECH_SNAP__)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Win32
|
||||||
|
*/
|
||||||
|
typedef __int32 khronos_int32_t;
|
||||||
|
typedef unsigned __int32 khronos_uint32_t;
|
||||||
|
typedef __int64 khronos_int64_t;
|
||||||
|
typedef unsigned __int64 khronos_uint64_t;
|
||||||
|
#define KHRONOS_SUPPORT_INT64 1
|
||||||
|
#define KHRONOS_SUPPORT_FLOAT 1
|
||||||
|
|
||||||
|
#elif defined(__sun__) || defined(__digital__)
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Sun or Digital
|
||||||
|
*/
|
||||||
|
typedef int khronos_int32_t;
|
||||||
|
typedef unsigned int khronos_uint32_t;
|
||||||
|
#if defined(__arch64__) || defined(_LP64)
|
||||||
|
typedef long int khronos_int64_t;
|
||||||
|
typedef unsigned long int khronos_uint64_t;
|
||||||
|
#else
|
||||||
|
typedef long long int khronos_int64_t;
|
||||||
|
typedef unsigned long long int khronos_uint64_t;
|
||||||
|
#endif /* __arch64__ */
|
||||||
|
#define KHRONOS_SUPPORT_INT64 1
|
||||||
|
#define KHRONOS_SUPPORT_FLOAT 1
|
||||||
|
|
||||||
|
#elif 0
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Hypothetical platform with no float or int64 support
|
||||||
|
*/
|
||||||
|
typedef int khronos_int32_t;
|
||||||
|
typedef unsigned int khronos_uint32_t;
|
||||||
|
#define KHRONOS_SUPPORT_INT64 0
|
||||||
|
#define KHRONOS_SUPPORT_FLOAT 0
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Generic fallback
|
||||||
|
*/
|
||||||
|
#include <stdint.h>
|
||||||
|
typedef int32_t khronos_int32_t;
|
||||||
|
typedef uint32_t khronos_uint32_t;
|
||||||
|
typedef int64_t khronos_int64_t;
|
||||||
|
typedef uint64_t khronos_uint64_t;
|
||||||
|
#define KHRONOS_SUPPORT_INT64 1
|
||||||
|
#define KHRONOS_SUPPORT_FLOAT 1
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Types that are (so far) the same on all platforms
|
||||||
|
*/
|
||||||
|
typedef signed char khronos_int8_t;
|
||||||
|
typedef unsigned char khronos_uint8_t;
|
||||||
|
typedef signed short int khronos_int16_t;
|
||||||
|
typedef unsigned short int khronos_uint16_t;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Types that differ between LLP64 and LP64 architectures - in LLP64,
|
||||||
|
* pointers are 64 bits, but 'long' is still 32 bits. Win64 appears
|
||||||
|
* to be the only LLP64 architecture in current use.
|
||||||
|
*/
|
||||||
|
#ifdef KHRONOS_USE_INTPTR_T
|
||||||
|
typedef intptr_t khronos_intptr_t;
|
||||||
|
typedef uintptr_t khronos_uintptr_t;
|
||||||
|
#elif defined(_WIN64)
|
||||||
|
typedef signed long long int khronos_intptr_t;
|
||||||
|
typedef unsigned long long int khronos_uintptr_t;
|
||||||
|
#else
|
||||||
|
typedef signed long int khronos_intptr_t;
|
||||||
|
typedef unsigned long int khronos_uintptr_t;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if defined(_WIN64)
|
||||||
|
typedef signed long long int khronos_ssize_t;
|
||||||
|
typedef unsigned long long int khronos_usize_t;
|
||||||
|
#else
|
||||||
|
typedef signed long int khronos_ssize_t;
|
||||||
|
typedef unsigned long int khronos_usize_t;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if KHRONOS_SUPPORT_FLOAT
|
||||||
|
/*
|
||||||
|
* Float type
|
||||||
|
*/
|
||||||
|
typedef float khronos_float_t;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if KHRONOS_SUPPORT_INT64
|
||||||
|
/* Time types
|
||||||
|
*
|
||||||
|
* These types can be used to represent a time interval in nanoseconds or
|
||||||
|
* an absolute Unadjusted System Time. Unadjusted System Time is the number
|
||||||
|
* of nanoseconds since some arbitrary system event (e.g. since the last
|
||||||
|
* time the system booted). The Unadjusted System Time is an unsigned
|
||||||
|
* 64 bit value that wraps back to 0 every 584 years. Time intervals
|
||||||
|
* may be either signed or unsigned.
|
||||||
|
*/
|
||||||
|
typedef khronos_uint64_t khronos_utime_nanoseconds_t;
|
||||||
|
typedef khronos_int64_t khronos_stime_nanoseconds_t;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Dummy value used to pad enum types to 32 bits.
|
||||||
|
*/
|
||||||
|
#ifndef KHRONOS_MAX_ENUM
|
||||||
|
#define KHRONOS_MAX_ENUM 0x7FFFFFFF
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Enumerated boolean type
|
||||||
|
*
|
||||||
|
* Values other than zero should be considered to be true. Therefore
|
||||||
|
* comparisons should not be made against KHRONOS_TRUE.
|
||||||
|
*/
|
||||||
|
typedef enum {
|
||||||
|
KHRONOS_FALSE = 0,
|
||||||
|
KHRONOS_TRUE = 1,
|
||||||
|
KHRONOS_BOOLEAN_ENUM_FORCE_SIZE = KHRONOS_MAX_ENUM
|
||||||
|
} khronos_boolean_enum_t;
|
||||||
|
|
||||||
|
#endif /* __khrplatform_h_ */
|
3694
vendor/glad/include/glad/glad.h
vendored
Normal file
3694
vendor/glad/include/glad/glad.h
vendored
Normal file
File diff suppressed because it is too large
Load diff
1833
vendor/glad/src/glad.c
vendored
Normal file
1833
vendor/glad/src/glad.c
vendored
Normal file
File diff suppressed because it is too large
Load diff
1
vendor/glfw
vendored
Submodule
1
vendor/glfw
vendored
Submodule
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 97da62a027794d9ff0f4512268cb9a73a8fb5073
|
1
vendor/imgui
vendored
Submodule
1
vendor/imgui
vendored
Submodule
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 8639a2f9f8d6d53f4c7a221579de5871051153d9
|
1
vendor/spdlog
vendored
Submodule
1
vendor/spdlog
vendored
Submodule
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 2f2d04b3e840428a18942ca2d3d65203ec564647
|
Loading…
Reference in a new issue