From 6055b83ed75bda57ffbb7574f14fe2c871f6a303 Mon Sep 17 00:00:00 2001 From: slack Date: Fri, 13 Mar 2009 00:21:59 +0100 Subject: [PATCH] Qt window for emulation control. Coming soon: Debug GUI :) --- CMakeLists.txt | 2 + common/Logger.h | 5 +- core/GBMemory.h | 3 + core/GBVideo.cc | 1 + core/GBVideo.h | 2 +- core/GameBoy.cc | 24 ++++++- core/GameBoy.h | 23 ++++--- qtboi/QtBoiEmuThread.cc | 87 +++++++++++++++++-------- qtboi/QtBoiEmuThread.h | 12 +++- qtboi/QtBoiMainWindow.cc | 134 +++++++++++++++++++-------------------- qtboi/QtBoiMainWindow.h | 47 +++++++------- wendi/wendi.cc | 3 +- 12 files changed, 205 insertions(+), 138 deletions(-) diff --git a/CMakeLists.txt b/CMakeLists.txt index 201bfd2..2640704 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -3,3 +3,5 @@ PROJECT(wenboi) CMAKE_MINIMUM_REQUIRED(VERSION 2.6.0) ADD_SUBDIRECTORY(core) +ADD_SUBDIRECTORY(qtboi) +ADD_SUBDIRECTORY(wendi) diff --git a/common/Logger.h b/common/Logger.h index f402619..c1fc48d 100644 --- a/common/Logger.h +++ b/common/Logger.h @@ -19,6 +19,7 @@ #define LOGGER_H #include +#include #include #include #include @@ -46,8 +47,6 @@ class Logger: public Singleton out << "Starting log at " << static_cast(log_start_time) << std::endl; } - //void setOutput(std::ostream os) { out = os; } - void log(log_level level, std::string str) { if (level <= current_log_level) @@ -104,7 +103,7 @@ class Logger: public Singleton void set_log_level (log_level level) { current_log_level = level; } private: - std::ostream& out; + std::ostream &out; log_level current_log_level; time_t log_start_time; }; diff --git a/core/GBMemory.h b/core/GBMemory.h index 10a9592..37a7cdb 100644 --- a/core/GBMemory.h +++ b/core/GBMemory.h @@ -88,9 +88,12 @@ class GBMemory watchpoint_oldvalue(0), watchpoint_newvalue(0) {} + + void init(MBC *mbc) { this->mbc = mbc; } + enum WatchpointControl { WATCH=0, diff --git a/core/GBVideo.cc b/core/GBVideo.cc index f4e112b..1aa19ed 100644 --- a/core/GBVideo.cc +++ b/core/GBVideo.cc @@ -18,6 +18,7 @@ #include #include #include +#include #include "GBVideo.h" #include "GameBoy.h" diff --git a/core/GBVideo.h b/core/GBVideo.h index 0d0c0d5..34197b9 100644 --- a/core/GBVideo.h +++ b/core/GBVideo.h @@ -103,7 +103,7 @@ class GBVideo void set_display_mode(DisplayMode mode) { display_mode = mode; } // status queries - u32 get_frames_rendered() { return frames_rendered; } + u32 get_frames_rendered() const { return frames_rendered; } // prevent object copying private: diff --git a/core/GameBoy.cc b/core/GameBoy.cc index ad53ce1..d634d13 100644 --- a/core/GameBoy.cc +++ b/core/GameBoy.cc @@ -30,7 +30,7 @@ bool GameBoy::is_pad[NUM_CONTROLS]={false, false, false, false, true, true, true, true}; -GameBoy::GameBoy(std::string rom_name, GameBoyType type): +GameBoy::GameBoy(GameBoyType type): gameboy_type(type), memory(this), video(this), @@ -46,6 +46,16 @@ GameBoy::GameBoy(std::string rom_name, GameBoyType type): breakpoints(), last_breakpoint_id(0) { +} + +GameBoy::~GameBoy() +{ + free_ROM(); +} + +void GameBoy::load_ROM(std::string rom_name) +{ + free_ROM(); logger.info("GameBoy init"); rom = read_gbrom(rom_name); @@ -55,6 +65,13 @@ GameBoy::GameBoy(std::string rom_name, GameBoyType type): reset(); } + +void GameBoy::free_ROM() +{ + ::operator delete(reinterpret_cast(rom)); + ::operator delete(reinterpret_cast(memory.mbc)); +} + void GameBoy::reset() { logger.info("GameBoy reset"); @@ -62,6 +79,9 @@ void GameBoy::reset() HALT = 0; cycle_count = 0; cycles_until_next_instruction = 0; + + for (int i=0; irun_cycle(); - if (gb->video.get_frames_rendered() > frameCount) + status = gb.run_cycle(); + if (gb.video.get_frames_rendered() > frameCount) { - frameCount = gb->video.get_frames_rendered(); - emit redraw(gb->video.get_screen_buffer()); + frameCount = gb.video.get_frames_rendered(); + emit redraw(gb.video.get_screen_buffer()); } } break; case STEP: do { - status = gb->run_cycle(); - if (gb->video.get_frames_rendered() > frameCount) + status = gb.run_cycle(); + if (gb.video.get_frames_rendered() > frameCount) { - frameCount = gb->video.get_frames_rendered(); - emit redraw(gb->video.get_screen_buffer()); + frameCount = gb.video.get_frames_rendered(); + emit redraw(gb.video.get_screen_buffer()); } } while(status == GameBoy::WAIT); // do nothing + isPaused = true; break; } - print_run_result(*gb, status); - isPaused=true; - emit emulationPaused(); + print_run_result(gb, status); + if (isPaused) + { + emit emulationPaused(); + } } } + + cout << "Exiting emulation thread" << endl; + } diff --git a/qtboi/QtBoiEmuThread.h b/qtboi/QtBoiEmuThread.h index cb5e804..bb9b854 100644 --- a/qtboi/QtBoiEmuThread.h +++ b/qtboi/QtBoiEmuThread.h @@ -13,13 +13,17 @@ class QtBoiEmuThread: public QThread GameBoy::run_status status; bool isPaused; int frameCount; - QtBoiEmuThread(QObject *parent, QString romName); + + QtBoiEmuThread(QObject *parent); ~QtBoiEmuThread(); + + void loadROM(QString romName); public slots: + void reset(); void run(); void stop(); - void toggle_paused(); + void pause(); void cont(); void step(); @@ -28,10 +32,12 @@ class QtBoiEmuThread: public QThread void redraw(const uchar *buffer); private: + QString romName; enum RunningMode { RUN, STEP }; - GameBoy *gb; RunningMode runningMode; bool quitRequested; + bool resetRequested; + bool romLoaded; }; diff --git a/qtboi/QtBoiMainWindow.cc b/qtboi/QtBoiMainWindow.cc index d54e075..71820da 100644 --- a/qtboi/QtBoiMainWindow.cc +++ b/qtboi/QtBoiMainWindow.cc @@ -10,93 +10,89 @@ #include QtBoiMainWindow::QtBoiMainWindow(QWidget *parent) - :QMainWindow(parent), emuThread(0) + :QMainWindow(parent), emuThread(0) { - screen = new QImage(160, 144, QImage::Format_Indexed8); - screen->setNumColors(8); - screen->setColor(7, qRgb(0,0,0)); - screen->setColor(6, qRgb(36,36,36)); - screen->setColor(5, qRgb(73,73,73)); - screen->setColor(4, qRgb(109,109,109)); - screen->setColor(3, qRgb(146,146,146)); - screen->setColor(2, qRgb(182,182,182)); - screen->setColor(1, qRgb(219,219,219)); - screen->setColor(0, qRgb(255,255,255)); - - createMenu(); - resize(640,480); - centralWindow = new QLabel(this); - setCentralWidget(centralWindow); - + screen = new QImage(160, 144, QImage::Format_Indexed8); + screen->setNumColors(7); + screen->setColor(6, qRgb(0,0,0)); + screen->setColor(5, qRgb(42,42,42)); + screen->setColor(4, qRgb(85,85,85)); + screen->setColor(3, qRgb(127,127,127)); + screen->setColor(2, qRgb(170,170,170)); + screen->setColor(1, qRgb(212,212,212)); + screen->setColor(0, qRgb(255,255,255)); + + createMenu(); + resize(640,480); + centralWindow = new QLabel(this); + setCentralWidget(centralWindow); + + emuThread = new QtBoiEmuThread(this); + emuThread->start(); + + connect(emulatorCont, SIGNAL(triggered()), emuThread, SLOT(cont())); + connect(emulatorStop, SIGNAL(triggered()), emuThread, SLOT(stop())); + connect(emulatorPause, SIGNAL(triggered()), emuThread, SLOT(pause())); + connect(emulatorStep, SIGNAL(triggered()), emuThread, SLOT(step())); + connect(emulatorReset, SIGNAL(triggered()), emuThread, SLOT(reset())); + connect(emuThread, SIGNAL(redraw(const uchar*)), this, SLOT(onRedraw(const uchar*))); } QtBoiMainWindow::~QtBoiMainWindow() { - if (emuThread) { - emuThread->stop(); - delete emuThread; - } + if (emuThread) { + emuThread->stop(); + emuThread->wait(); + delete emuThread; + } } void QtBoiMainWindow::createMenu() { - loadROM = new QAction("&Load ROM...", this); - quit = new QAction("&Quit", this); - emulatorPause = new QAction("&Toggle pause", this); - emulatorCont = new QAction("&Go", this); - emulatorStop = new QAction("&Stop", this); - emulatorStep = new QAction("St&ep", this); - - QMenu *file; - file = menuBar()->addMenu("&File"); - file->addAction(loadROM); - file->addAction(quit); - - QMenu *emulator; - emulator = menuBar()->addMenu("&Emulator"); - emulator->addAction(emulatorCont); - emulator->addAction(emulatorPause); - emulator->addAction(emulatorStop); - - QMenu *debug; - debug = menuBar()->addMenu("&Debug"); - debug->addAction(emulatorStep); - - connect(quit, SIGNAL(triggered()), qApp, SLOT(quit())); - connect(loadROM, SIGNAL(triggered()), this, SLOT(onLoadROM())); + loadROM = new QAction("&Load ROM...", this); + quit = new QAction("&Quit", this); + emulatorPause = new QAction("&Pause", this); + emulatorCont = new QAction("&Go", this); + emulatorStop = new QAction("&Stop", this); + emulatorStep = new QAction("St&ep", this); + emulatorReset = new QAction("&Reset", this); + + + QMenu *file; + file = menuBar()->addMenu("&File"); + file->addAction(loadROM); + file->addAction(quit); + + QMenu *emulator; + emulator = menuBar()->addMenu("&Emulator"); + emulator->addAction(emulatorCont); + emulator->addAction(emulatorPause); + emulator->addAction(emulatorStop); + emulator->addAction(emulatorReset); + + QMenu *debug; + debug = menuBar()->addMenu("&Debug"); + debug->addAction(emulatorStep); + + connect(quit, SIGNAL(triggered()), qApp, SLOT(quit())); + connect(loadROM, SIGNAL(triggered()), this, SLOT(onLoadROM())); } void QtBoiMainWindow::onLoadROM() { - QString filename = QFileDialog::getOpenFileName(this, tr("Load ROM"), "../roms", tr("GameBoy ROMs (*.gb *.gbc)")); - if (filename == "") return; - - std::cout << filename.toStdString() << std::endl; - - if (emuThread) { - emuThread->stop(); - emuThread->wait(); - delete emuThread; - } - - emuThread = new QtBoiEmuThread(0, filename); - - connect(emulatorCont, SIGNAL(triggered()), emuThread, SLOT(cont()), Qt::DirectConnection); - connect(emulatorStop, SIGNAL(triggered()), emuThread, SLOT(stop()), Qt::DirectConnection); - connect(emulatorPause, SIGNAL(triggered()), emuThread, SLOT(toggle_paused()), Qt::DirectConnection); - connect(emulatorStep, SIGNAL(triggered()), emuThread, SLOT(step()), Qt::DirectConnection); - connect(emuThread, SIGNAL(redraw(const uchar*)), this, SLOT(onRedraw(const uchar*))); - - emuThread->start(); + QString filename = QFileDialog::getOpenFileName(this, tr("Load ROM"), "../roms", tr("GameBoy ROMs (*.gb *.gbc)")); + if (filename == "") return; + + emuThread->loadROM(filename); + } void QtBoiMainWindow::onRedraw(const uchar *buffer) { - uchar *pixels = screen->bits(); - memcpy(pixels, buffer, 160*144); - //centralWindow->setPixmap(QPixmap::fromImage(screen->scaled(320,288))); - centralWindow->setPixmap(QPixmap::fromImage(*screen)); + uchar *pixels = screen->bits(); + memcpy(pixels, buffer, 160*144); + centralWindow->setPixmap(QPixmap::fromImage(screen->scaled(320,288))); } diff --git a/qtboi/QtBoiMainWindow.h b/qtboi/QtBoiMainWindow.h index 8794aea..1382339 100644 --- a/qtboi/QtBoiMainWindow.h +++ b/qtboi/QtBoiMainWindow.h @@ -10,29 +10,30 @@ class QtBoiMainWindow: public QMainWindow { - Q_OBJECT - - public: - QtBoiMainWindow(QWidget *parent=0); - ~QtBoiMainWindow(); - - public slots: - void onLoadROM(); - void onRedraw(const uchar *buffer); - - private: - void createMenu(); - QtBoiEmuThread *emuThread; - - QLabel *centralWindow; - QImage *screen; - - QAction *loadROM; - QAction *quit; - QAction *emulatorPause; - QAction *emulatorCont; - QAction *emulatorStop; - QAction *emulatorStep; + Q_OBJECT + + public: + QtBoiMainWindow(QWidget *parent=0); + ~QtBoiMainWindow(); + + public slots: + void onLoadROM(); + void onRedraw(const uchar *buffer); + + private: + void createMenu(); + QtBoiEmuThread *emuThread; + + QLabel *centralWindow; + QImage *screen; + + QAction *loadROM; + QAction *quit; + QAction *emulatorPause; + QAction *emulatorCont; + QAction *emulatorStop; + QAction *emulatorStep; + QAction *emulatorReset; }; diff --git a/wendi/wendi.cc b/wendi/wendi.cc index fc42582..2b82c42 100644 --- a/wendi/wendi.cc +++ b/wendi/wendi.cc @@ -256,7 +256,8 @@ void new_block_start(address dst, address src, Instruction::InstructionType type int main(int argc, char **argv) { logger.set_log_level(Logger::TRACE); - GameBoy gb(argv[1]); + GameBoy gb; + gb.load_ROM(argv[1]); list pending; // pending holds empty CodeBlocks -- 2.34.1