From a32cb5cc5e4fa0f1590725036ca58a490d0f1164 Mon Sep 17 00:00:00 2001 From: Vicki Pfau Date: Mon, 22 Apr 2019 13:59:39 -0700 Subject: [PATCH 01/10] All: Fix several memory leaks --- CHANGES | 1 + src/core/cheats.c | 1 + src/debugger/test/parser.c | 28 ++++----- src/feature/gui/gui-runner.c | 2 + src/gb/test/core.c | 4 ++ src/gb/test/mbc.c | 1 + src/gb/test/memory.c | 1 + src/gb/test/rtc.c | 1 + src/gba/test/cheats.c | 90 ++++++++++++++--------------- src/platform/libretro/libretro.c | 1 + src/platform/openemu/mGBAGameCore.m | 1 + src/platform/sdl/main.c | 1 + src/platform/test/fuzz-main.c | 1 + 13 files changed, 74 insertions(+), 59 deletions(-) diff --git a/CHANGES b/CHANGES index 35422cef7..d20924cce 100644 --- a/CHANGES +++ b/CHANGES @@ -28,6 +28,7 @@ Other fixes: - Qt: Fix crash in sprite viewer magnification (fixes mgba.io/i/1362) - 3DS: Ensure core 2 can be used for threaded renderer (fixes mgba.io/i/1371) - GB Core: Fix toggling WIN and OBJ being swapped + - All: Fix several memory leaks Misc: - GBA Savedata: EEPROM performance fixes - GBA Savedata: Automatically map 1Mbit Flash files as 1Mbit Flash diff --git a/src/core/cheats.c b/src/core/cheats.c index d0ed72508..0abcbd537 100644 --- a/src/core/cheats.c +++ b/src/core/cheats.c @@ -90,6 +90,7 @@ void mCheatSetDeinit(struct mCheatSet* set) { if (set->name) { free(set->name); } + StringListDeinit(&set->lines); set->deinit(set); free(set); } diff --git a/src/debugger/test/parser.c b/src/debugger/test/parser.c index a2635f4fb..c1c2840d2 100644 --- a/src/debugger/test/parser.c +++ b/src/debugger/test/parser.c @@ -21,16 +21,16 @@ struct LPTest { struct ParseTree* tree = &lp->tree; \ parseLexedExpression(tree, &lp->lv) -M_TEST_SUITE_SETUP(Parser) { +static int parseSetup(void** state) { struct LPTest* lp = malloc(sizeof(struct LPTest)); LexVectorInit(&lp->lv, 0); *state = lp; return 0; } -M_TEST_SUITE_TEARDOWN(Parser) { +static int parseTeardown(void** state) { struct LPTest* lp = *state; - parseFree(&lp->tree); \ + parseFree(&lp->tree); lexFree(&lp->lv); LexVectorDeinit(&lp->lv); free(lp); @@ -135,14 +135,14 @@ M_TEST_DEFINE(parseUnaryChainedOperator) { assert_int_equal(tree->rhs->rhs->token.uintValue, 2); } -M_TEST_SUITE_DEFINE_SETUP_TEARDOWN(Parser, - cmocka_unit_test(parseEmpty), - cmocka_unit_test(parseInt), - cmocka_unit_test(parseLexError), - cmocka_unit_test(parseError), - cmocka_unit_test(parseSimpleExpression), - cmocka_unit_test(parseAddMultplyExpression), - cmocka_unit_test(parseParentheticalExpression), - cmocka_unit_test(parseParentheticalAddMultplyExpression), - cmocka_unit_test(parseIsolatedOperator), - cmocka_unit_test(parseUnaryChainedOperator)) +M_TEST_SUITE_DEFINE(Parser, + cmocka_unit_test_setup_teardown(parseEmpty, parseSetup, parseTeardown), + cmocka_unit_test_setup_teardown(parseInt, parseSetup, parseTeardown), + cmocka_unit_test_setup_teardown(parseLexError, parseSetup, parseTeardown), + cmocka_unit_test_setup_teardown(parseError, parseSetup, parseTeardown), + cmocka_unit_test_setup_teardown(parseSimpleExpression, parseSetup, parseTeardown), + cmocka_unit_test_setup_teardown(parseAddMultplyExpression, parseSetup, parseTeardown), + cmocka_unit_test_setup_teardown(parseParentheticalExpression, parseSetup, parseTeardown), + cmocka_unit_test_setup_teardown(parseParentheticalAddMultplyExpression, parseSetup, parseTeardown), + cmocka_unit_test_setup_teardown(parseIsolatedOperator, parseSetup, parseTeardown), + cmocka_unit_test_setup_teardown(parseUnaryChainedOperator, parseSetup, parseTeardown)) diff --git a/src/feature/gui/gui-runner.c b/src/feature/gui/gui-runner.c index 50c9dea42..128160cad 100644 --- a/src/feature/gui/gui-runner.c +++ b/src/feature/gui/gui-runner.c @@ -362,6 +362,7 @@ void mGUIRun(struct mGUIRunner* runner, const char* path) { found = mCoreLoadFile(runner->core, path); if (!found) { mLOG(GUI_RUNNER, WARN, "Failed to load %s!", path); + mCoreConfigDeinit(&runner->core->config); runner->core->deinit(runner->core); } } @@ -606,6 +607,7 @@ void mGUIRun(struct mGUIRunner* runner, const char* path) { } mInputMapDeinit(&runner->core->inputMap); mLOG(GUI_RUNNER, DEBUG, "Deinitializing core..."); + mCoreConfigDeinit(&runner->core->config); runner->core->deinit(runner->core); runner->core = NULL; diff --git a/src/gb/test/core.c b/src/gb/test/core.c index a8638012b..1d0b3416a 100644 --- a/src/gb/test/core.c +++ b/src/gb/test/core.c @@ -31,6 +31,7 @@ M_TEST_DEFINE(reset) { assert_true(core->init(core)); mCoreInitConfig(core, NULL); core->reset(core); + mCoreConfigDeinit(&core->config); core->deinit(core); } @@ -41,6 +42,7 @@ M_TEST_DEFINE(loadNullROM) { mCoreInitConfig(core, NULL); assert_false(core->loadROM(core, NULL)); core->reset(core); + mCoreConfigDeinit(&core->config); core->deinit(core); } @@ -51,7 +53,9 @@ M_TEST_DEFINE(isROM) { struct mCore* core = mCoreFindVF(vf); assert_non_null(core); assert_int_equal(core->platform(core), PLATFORM_GB); + vf->close(vf); assert_true(core->init(core)); + core->deinit(core); } diff --git a/src/gb/test/mbc.c b/src/gb/test/mbc.c index 5498d004d..7bcfcd248 100644 --- a/src/gb/test/mbc.c +++ b/src/gb/test/mbc.c @@ -27,6 +27,7 @@ M_TEST_SUITE_TEARDOWN(GBMBC) { return 0; } struct mCore* core = *state; + mCoreConfigDeinit(&core->config); core->deinit(core); return 0; } diff --git a/src/gb/test/memory.c b/src/gb/test/memory.c index 01a4b398f..ab64598ff 100644 --- a/src/gb/test/memory.c +++ b/src/gb/test/memory.c @@ -27,6 +27,7 @@ M_TEST_SUITE_TEARDOWN(GBMemory) { return 0; } struct mCore* core = *state; + mCoreConfigDeinit(&core->config); core->deinit(core); return 0; } diff --git a/src/gb/test/rtc.c b/src/gb/test/rtc.c index 71d06ced2..6ac03feb5 100644 --- a/src/gb/test/rtc.c +++ b/src/gb/test/rtc.c @@ -69,6 +69,7 @@ M_TEST_SUITE_TEARDOWN(GBRTC) { return 0; } struct GBRTCTest* test = *state; + mCoreConfigDeinit(&test->core->config); test->core->deinit(test->core); free(test); return 0; diff --git a/src/gba/test/cheats.c b/src/gba/test/cheats.c index a18e77c5e..6f4b8cb4c 100644 --- a/src/gba/test/cheats.c +++ b/src/gba/test/cheats.c @@ -13,7 +13,7 @@ #include "gba/cheats/parv3.h" #include "gba/cheats/gameshark.h" -M_TEST_SUITE_SETUP(GBACheats) { +static int cheatsSetup(void** state) { struct mCore* core = GBACoreCreate(); core->init(core); core->cheatDevice(core); @@ -21,7 +21,7 @@ M_TEST_SUITE_SETUP(GBACheats) { return 0; } -M_TEST_SUITE_TEARDOWN(GBACheats) { +static int cheatsTeardown(void** state) { if (!*state) { return 0; } @@ -36,7 +36,7 @@ M_TEST_DEFINE(createSet) { assert_non_null(device); struct mCheatSet* set = device->createSet(device, NULL); assert_non_null(set); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(addRawPARv3) { @@ -48,7 +48,7 @@ M_TEST_DEFINE(addRawPARv3) { GBACheatSetGameSharkVersion((struct GBACheatSet*) set, GBA_GS_PARV3_RAW); assert_true(set->addLine(set, "80000000 00000000", GBA_CHEAT_PRO_ACTION_REPLAY)); assert_false(set->addLine(set, "43000000 00000000", GBA_CHEAT_PRO_ACTION_REPLAY)); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3Assign) { @@ -72,7 +72,7 @@ M_TEST_DEFINE(doPARv3Assign) { assert_int_equal(core->rawRead16(core, 0x03000002, -1), 0x5678); assert_int_equal(core->rawRead32(core, 0x03000004, -1), 0x12345678); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3Slide1) { @@ -101,7 +101,7 @@ M_TEST_DEFINE(doPARv3Slide1) { assert_int_equal(core->rawRead8(core, 0x03000004, -1), 0); assert_int_equal(core->rawRead8(core, 0x03000005, -1), 0); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3Slide2) { @@ -130,7 +130,7 @@ M_TEST_DEFINE(doPARv3Slide2) { assert_int_equal(core->rawRead16(core, 0x03000008, -1), 0); assert_int_equal(core->rawRead16(core, 0x0300000A, -1), 0); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3Slide4) { @@ -159,7 +159,7 @@ M_TEST_DEFINE(doPARv3Slide4) { assert_int_equal(core->rawRead16(core, 0x03000010, -1), 0); assert_int_equal(core->rawRead16(core, 0x03000014, -1), 0); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3If1) { @@ -188,7 +188,7 @@ M_TEST_DEFINE(doPARv3If1) { assert_int_equal(core->rawRead8(core, 0x03000000, -1), 0x1); assert_int_equal(core->rawRead8(core, 0x03000001, -1), 0x11); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3If1x1) { @@ -243,7 +243,7 @@ M_TEST_DEFINE(doPARv3If1x1) { assert_int_equal(core->rawRead8(core, 0x03000002, -1), 0x21); assert_int_equal(core->rawRead8(core, 0x03000003, -1), 0x31); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3If2) { @@ -277,7 +277,7 @@ M_TEST_DEFINE(doPARv3If2) { assert_int_equal(core->rawRead8(core, 0x03000001, -1), 0x11); assert_int_equal(core->rawRead8(core, 0x03000002, -1), 0x21); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3If2x2) { @@ -345,7 +345,7 @@ M_TEST_DEFINE(doPARv3If2x2) { assert_int_equal(core->rawRead8(core, 0x03000003, -1), 0x31); assert_int_equal(core->rawRead8(core, 0x03000004, -1), 0x41); assert_int_equal(core->rawRead8(core, 0x03000005, -1), 0x51); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3If2Contain1) { @@ -392,7 +392,7 @@ M_TEST_DEFINE(doPARv3If2Contain1) { assert_int_equal(core->rawRead8(core, 0x03000000, -1), 0x1); assert_int_equal(core->rawRead8(core, 0x03000001, -1), 0x1); assert_int_equal(core->rawRead8(core, 0x03000002, -1), 0x21); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3IfX) { @@ -421,7 +421,7 @@ M_TEST_DEFINE(doPARv3IfX) { mCheatRefresh(device, set); assert_int_equal(core->rawRead8(core, 0x03000000, -1), 0x1); assert_int_equal(core->rawRead8(core, 0x03000001, -1), 0x11); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3IfXxX) { @@ -484,7 +484,7 @@ M_TEST_DEFINE(doPARv3IfXxX) { assert_int_equal(core->rawRead8(core, 0x03000002, -1), 0x21); assert_int_equal(core->rawRead8(core, 0x03000003, -1), 0x32); assert_int_equal(core->rawRead8(core, 0x03000004, -1), 0x41); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3IfXElse) { @@ -519,7 +519,7 @@ M_TEST_DEFINE(doPARv3IfXElse) { assert_int_equal(core->rawRead8(core, 0x03000000, -1), 0x1); assert_int_equal(core->rawRead8(core, 0x03000001, -1), 0x11); assert_int_equal(core->rawRead8(core, 0x03000002, -1), 0x22); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3IfXElsexX) { @@ -590,7 +590,7 @@ M_TEST_DEFINE(doPARv3IfXElsexX) { assert_int_equal(core->rawRead8(core, 0x03000003, -1), 0x32); assert_int_equal(core->rawRead8(core, 0x03000004, -1), 0x42); assert_int_equal(core->rawRead8(core, 0x03000005, -1), 0x51); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3IfXElsexXElse) { @@ -668,7 +668,7 @@ M_TEST_DEFINE(doPARv3IfXElsexXElse) { assert_int_equal(core->rawRead8(core, 0x03000004, -1), 0x42); assert_int_equal(core->rawRead8(core, 0x03000005, -1), 0x51); assert_int_equal(core->rawRead8(core, 0x03000006, -1), 0x62); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3IfXContain1) { @@ -730,7 +730,7 @@ M_TEST_DEFINE(doPARv3IfXContain1) { assert_int_equal(core->rawRead8(core, 0x03000002, -1), 0x21); assert_int_equal(core->rawRead8(core, 0x03000003, -1), 0x31); assert_int_equal(core->rawRead8(core, 0x03000004, -1), 0x41); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3IfXContain1Else) { @@ -800,7 +800,7 @@ M_TEST_DEFINE(doPARv3IfXContain1Else) { assert_int_equal(core->rawRead8(core, 0x03000003, -1), 0x31); assert_int_equal(core->rawRead8(core, 0x03000004, -1), 0x41); assert_int_equal(core->rawRead8(core, 0x03000005, -1), 0x52); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3IfXElseContain1) { @@ -870,7 +870,7 @@ M_TEST_DEFINE(doPARv3IfXElseContain1) { assert_int_equal(core->rawRead8(core, 0x03000003, -1), 0x32); assert_int_equal(core->rawRead8(core, 0x03000004, -1), 0x41); assert_int_equal(core->rawRead8(core, 0x03000005, -1), 0x52); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3IfXContain1ElseContain1) { @@ -1016,7 +1016,7 @@ M_TEST_DEFINE(doPARv3IfXContain1ElseContain1) { assert_int_equal(core->rawRead8(core, 0x03000006, -1), 0x62); assert_int_equal(core->rawRead8(core, 0x03000007, -1), 0x71); assert_int_equal(core->rawRead8(core, 0x03000008, -1), 0x82); - set->deinit(set); + mCheatSetDeinit(set); } M_TEST_DEFINE(doPARv3IfButton) { @@ -1046,28 +1046,28 @@ M_TEST_DEFINE(doPARv3IfButton) { core->rawWrite8(core, 0x03000000, -1, 0); mCheatRefresh(device, set); assert_int_equal(core->rawRead8(core, 0x03000000, -1), 0); - set->deinit(set); + mCheatSetDeinit(set); } -M_TEST_SUITE_DEFINE_SETUP_TEARDOWN(GBACheats, - cmocka_unit_test(createSet), - cmocka_unit_test(addRawPARv3), - cmocka_unit_test(doPARv3Assign), - cmocka_unit_test(doPARv3Slide1), - cmocka_unit_test(doPARv3Slide2), - cmocka_unit_test(doPARv3Slide4), - cmocka_unit_test(doPARv3If1), - cmocka_unit_test(doPARv3If1x1), - cmocka_unit_test(doPARv3If2), - cmocka_unit_test(doPARv3If2x2), - cmocka_unit_test(doPARv3If2Contain1), - cmocka_unit_test(doPARv3IfX), - cmocka_unit_test(doPARv3IfXxX), - cmocka_unit_test(doPARv3IfXElse), - cmocka_unit_test(doPARv3IfXElsexX), - cmocka_unit_test(doPARv3IfXElsexXElse), - cmocka_unit_test(doPARv3IfXContain1), - cmocka_unit_test(doPARv3IfXContain1Else), - cmocka_unit_test(doPARv3IfXElseContain1), - cmocka_unit_test(doPARv3IfXContain1ElseContain1), - cmocka_unit_test(doPARv3IfButton)) +M_TEST_SUITE_DEFINE(GBACheats, + cmocka_unit_test_setup_teardown(createSet, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(addRawPARv3, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3Assign, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3Slide1, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3Slide2, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3Slide4, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3If1, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3If1x1, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3If2, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3If2x2, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3If2Contain1, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3IfX, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3IfXxX, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3IfXElse, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3IfXElsexX, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3IfXElsexXElse, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3IfXContain1, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3IfXContain1Else, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3IfXElseContain1, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3IfXContain1ElseContain1, cheatsSetup, cheatsTeardown), + cmocka_unit_test_setup_teardown(doPARv3IfButton, cheatsSetup, cheatsTeardown)) diff --git a/src/platform/libretro/libretro.c b/src/platform/libretro/libretro.c index 6caa5d866..29dbc9033 100644 --- a/src/platform/libretro/libretro.c +++ b/src/platform/libretro/libretro.c @@ -562,6 +562,7 @@ void retro_unload_game(void) { if (!core) { return; } + mCoreConfigDeinit(&core->config); core->deinit(core); mappedMemoryFree(data, dataSize); data = 0; diff --git a/src/platform/openemu/mGBAGameCore.m b/src/platform/openemu/mGBAGameCore.m index 9b2b93b05..2612ddef4 100644 --- a/src/platform/openemu/mGBAGameCore.m +++ b/src/platform/openemu/mGBAGameCore.m @@ -76,6 +76,7 @@ - (void)dealloc { + mCoreConfigDeinit(&core->config); core->deinit(core); [cheatSets release]; free(outputBuffer); diff --git a/src/platform/sdl/main.c b/src/platform/sdl/main.c index 0da0698b9..450e1868a 100644 --- a/src/platform/sdl/main.c +++ b/src/platform/sdl/main.c @@ -143,6 +143,7 @@ int main(int argc, char** argv) { if (!mSDLInit(&renderer)) { freeArguments(&args); + mCoreConfigDeinit(&renderer.core->config); renderer.core->deinit(renderer.core); return 1; } diff --git a/src/platform/test/fuzz-main.c b/src/platform/test/fuzz-main.c index 044450099..b167a1443 100644 --- a/src/platform/test/fuzz-main.c +++ b/src/platform/test/fuzz-main.c @@ -165,6 +165,7 @@ loadError: if (outputBuffer) { free(outputBuffer); } + mCoreConfigDeinit(&core->config); core->deinit(core); return !cleanExit; From 6af4ddefdd3b242e38ac50d780b1e5a5d99e9539 Mon Sep 17 00:00:00 2001 From: Vicki Pfau Date: Fri, 26 Apr 2019 14:53:47 -0700 Subject: [PATCH 02/10] Qt: Cap window size to monitor size --- CHANGES | 1 + src/platform/qt/Window.cpp | 14 ++++++++++++-- 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/CHANGES b/CHANGES index d20924cce..4ea93c5c0 100644 --- a/CHANGES +++ b/CHANGES @@ -41,6 +41,7 @@ Misc: - Qt: Support switching webcams - Core: Add keysRead callback - Vita: Improved frame drawing speed + - Qt: Cap window size on start to monitor size 0.7.1: (2019-02-24) Bugfixes: diff --git a/src/platform/qt/Window.cpp b/src/platform/qt/Window.cpp index e669924bd..acf56669e 100644 --- a/src/platform/qt/Window.cpp +++ b/src/platform/qt/Window.cpp @@ -5,14 +5,15 @@ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #include "Window.h" -#include #include #include #include #include #include #include +#include #include +#include #ifdef USE_SQLITE3 #include "ArchiveInspector.h" @@ -200,6 +201,15 @@ void Window::argumentsPassed(mArguments* args) { void Window::resizeFrame(const QSize& size) { QSize newSize(size); + if (windowHandle()) { + QRect geom = windowHandle()->screen()->availableGeometry(); + if (newSize.width() > geom.width()) { + newSize.setWidth(geom.width()); + } + if (newSize.height() > geom.height()) { + newSize.setHeight(geom.height()); + } + } m_screenWidget->setSizeHint(newSize); newSize -= m_screenWidget->size(); newSize += this->size(); @@ -595,7 +605,7 @@ void Window::showEvent(QShowEvent* event) { m_wasOpened = true; resizeFrame(m_screenWidget->sizeHint()); QVariant windowPos = m_config->getQtOption("windowPos"); - QRect geom = QApplication::desktop()->availableGeometry(this); + QRect geom = windowHandle()->screen()->availableGeometry(); if (!windowPos.isNull() && geom.contains(windowPos.toPoint())) { move(windowPos.toPoint()); } else { From 223be49e7845327997a5258338cf6eb6190b475f Mon Sep 17 00:00:00 2001 From: Vicki Pfau Date: Sat, 27 Apr 2019 11:52:04 -0700 Subject: [PATCH 03/10] GBA BIOS: Add timings for HLE BIOS math functions (fixes #1396) --- CHANGES | 1 + src/gba/bios.c | 110 ++++++++++++++++++++++++++++++++++++++++++++----- 2 files changed, 100 insertions(+), 11 deletions(-) diff --git a/CHANGES b/CHANGES index 4ea93c5c0..6957c5484 100644 --- a/CHANGES +++ b/CHANGES @@ -42,6 +42,7 @@ Misc: - Core: Add keysRead callback - Vita: Improved frame drawing speed - Qt: Cap window size on start to monitor size + - GBA BIOS: Add timings for HLE BIOS math functions (fixes mgba.io/i/1396) 0.7.1: (2019-02-24) Bugfixes: diff --git a/src/gba/bios.c b/src/gba/bios.c index 186c1d1b6..a264354eb 100644 --- a/src/gba/bios.c +++ b/src/gba/bios.c @@ -10,6 +10,7 @@ #include #include #include +#include const uint32_t GBA_BIOS_CHECKSUM = 0xBAAE187F; const uint32_t GBA_DS_BIOS_CHECKSUM = 0xBAAE1880; @@ -22,6 +23,18 @@ static void _unRl(struct GBA* gba, int width); static void _unFilter(struct GBA* gba, int inwidth, int outwidth); static void _unBitPack(struct GBA* gba); +static int _mulWait(int32_t r) { + if ((r & 0xFFFFFF00) == 0xFFFFFF00 || !(r & 0xFFFFFF00)) { + return 1; + } else if ((r & 0xFFFF0000) == 0xFFFF0000 || !(r & 0xFFFF0000)) { + return 2; + } else if ((r & 0xFF000000) == 0xFF000000 || !(r & 0xFF000000)) { + return 3; + } else { + return 4; + } +} + static void _SoftReset(struct GBA* gba) { struct ARMCore* cpu = gba->cpu; ARMSetPrivilegeMode(cpu, MODE_IRQ); @@ -273,16 +286,30 @@ static void _Div(struct GBA* gba, int32_t num, int32_t denom) { cpu->gprs[1] = 0; cpu->gprs[3] = INT32_MIN; } + int loops = clz32(denom) - clz32(num); + if (loops < 1) { + loops = 1; + } + cpu->cycles += 4 /* prologue */ + 13 * loops + 7 /* epilogue */; } -static int16_t _ArcTan(int32_t i, int32_t* r1, int32_t* r3) { +static int16_t _ArcTan(int32_t i, int32_t* r1, int32_t* r3, int32_t* cycles) { + int currentCycles = 37; + currentCycles += _mulWait(i * i); int32_t a = -((i * i) >> 14); + currentCycles += _mulWait(0xA9 * a); int32_t b = ((0xA9 * a) >> 14) + 0x390; + currentCycles += _mulWait(b * a); b = ((b * a) >> 14) + 0x91C; + currentCycles += _mulWait(b * a); b = ((b * a) >> 14) + 0xFB6; + currentCycles += _mulWait(b * a); b = ((b * a) >> 14) + 0x16AA; + currentCycles += _mulWait(b * a); b = ((b * a) >> 14) + 0x2081; + currentCycles += _mulWait(b * a); b = ((b * a) >> 14) + 0x3651; + currentCycles += _mulWait(b * a); b = ((b * a) >> 14) + 0xA2F9; if (r1) { *r1 = a; @@ -290,10 +317,11 @@ static int16_t _ArcTan(int32_t i, int32_t* r1, int32_t* r3) { if (r3) { *r3 = b; } + *cycles += currentCycles; return (i * b) >> 16; } -static int16_t _ArcTan2(int32_t x, int32_t y, int32_t* r1) { +static int16_t _ArcTan2(int32_t x, int32_t y, int32_t* r1, int32_t* cycles) { if (!y) { if (x >= 0) { return 0; @@ -309,24 +337,77 @@ static int16_t _ArcTan2(int32_t x, int32_t y, int32_t* r1) { if (y >= 0) { if (x >= 0) { if (x >= y) { - return _ArcTan((y << 14) / x, r1, NULL); + return _ArcTan((y << 14) / x, r1, NULL, cycles); } } else if (-x >= y) { - return _ArcTan((y << 14) / x, r1, NULL) + 0x8000; + return _ArcTan((y << 14) / x, r1, NULL, cycles) + 0x8000; } - return 0x4000 - _ArcTan((x << 14) / y, r1, NULL); + return 0x4000 - _ArcTan((x << 14) / y, r1, NULL, cycles); } else { if (x <= 0) { if (-x > -y) { - return _ArcTan((y << 14) / x, r1, NULL) + 0x8000; + return _ArcTan((y << 14) / x, r1, NULL, cycles) + 0x8000; } } else if (x >= -y) { - return _ArcTan((y << 14) / x, r1, NULL) + 0x10000; + return _ArcTan((y << 14) / x, r1, NULL, cycles) + 0x10000; } - return 0xC000 - _ArcTan((x << 14) / y, r1, NULL); + return 0xC000 - _ArcTan((x << 14) / y, r1, NULL, cycles); } } +static int32_t _Sqrt(uint32_t x, int32_t* cycles) { + if (!x) { + *cycles += 53; + return 0; + } + int32_t currentCycles = 15; + uint32_t lower; + uint32_t upper = x; + uint32_t bound = 1; + while (bound < upper) { + upper >>= 1; + bound <<= 1; + currentCycles += 6; + } + while (true) { + currentCycles += 6; + upper = x; + uint32_t accum = 0; + lower = bound; + while (true) { + currentCycles += 5; + uint32_t oldLower = lower; + if (lower <= upper >> 1) { + lower <<= 1; + } + if (oldLower >= upper >> 1) { + break; + } + } + while (true) { + currentCycles += 8; + accum <<= 1; + if (upper >= lower) { + ++accum; + upper -= lower; + } + if (lower == bound) { + break; + } + lower >>= 1; + } + uint32_t oldBound = bound; + bound += accum; + bound >>= 1; + if (bound >= oldBound) { + bound = oldBound; + break; + } + } + *cycles += currentCycles; + return bound; +} + void GBASwi16(struct ARMCore* cpu, int immediate) { struct GBA* gba = (struct GBA*) cpu->master; mLOG(GBA_BIOS, DEBUG, "SWI: %02X r0: %08X r1: %08X r2: %08X r3: %08X", @@ -369,13 +450,13 @@ void GBASwi16(struct ARMCore* cpu, int immediate) { _Div(gba, cpu->gprs[1], cpu->gprs[0]); break; case 0x8: - cpu->gprs[0] = sqrt((uint32_t) cpu->gprs[0]); + cpu->gprs[0] = _Sqrt(cpu->gprs[0], &cpu->cycles); break; case 0x9: - cpu->gprs[0] = _ArcTan(cpu->gprs[0], &cpu->gprs[1], &cpu->gprs[3]); + cpu->gprs[0] = _ArcTan(cpu->gprs[0], &cpu->gprs[1], &cpu->gprs[3], &cpu->cycles); break; case 0xA: - cpu->gprs[0] = (uint16_t) _ArcTan2(cpu->gprs[0], cpu->gprs[1], &cpu->gprs[1]); + cpu->gprs[0] = (uint16_t) _ArcTan2(cpu->gprs[0], cpu->gprs[1], &cpu->gprs[1], &cpu->cycles); cpu->gprs[3] = 0x170; break; case 0xB: @@ -497,6 +578,13 @@ void GBASwi16(struct ARMCore* cpu, int immediate) { default: mLOG(GBA_BIOS, STUB, "Stub software interrupt: %02X", immediate); } + gba->cpu->cycles += 45 + cpu->memory.activeNonseqCycles16 /* 8 bit load for SWI # */; + // Return cycles + if (gba->cpu->executionMode == MODE_ARM) { + gba->cpu->cycles += cpu->memory.activeNonseqCycles32 + cpu->memory.activeSeqCycles32; + } else { + gba->cpu->cycles += cpu->memory.activeNonseqCycles16 + cpu->memory.activeSeqCycles16; + } gba->memory.biosPrefetch = 0xE3A02004; } From a898c1b7558023580efb0919c7cdd70c6eea7737 Mon Sep 17 00:00:00 2001 From: Vicki Pfau Date: Sun, 28 Apr 2019 13:13:04 -0700 Subject: [PATCH 04/10] Debugger: Make tracing compatible with breakpoints/watchpoints --- CHANGES | 1 + include/mgba/internal/debugger/cli-debugger.h | 2 + src/debugger/cli-debugger.c | 50 +++++++++++++------ src/gb/debugger/cli.c | 2 +- src/gba/debugger/cli.c | 2 +- 5 files changed, 41 insertions(+), 16 deletions(-) diff --git a/CHANGES b/CHANGES index 6957c5484..2ef909c91 100644 --- a/CHANGES +++ b/CHANGES @@ -43,6 +43,7 @@ Misc: - Vita: Improved frame drawing speed - Qt: Cap window size on start to monitor size - GBA BIOS: Add timings for HLE BIOS math functions (fixes mgba.io/i/1396) + - Debugger: Make tracing compatible with breakpoints/watchpoints 0.7.1: (2019-02-24) Bugfixes: diff --git a/include/mgba/internal/debugger/cli-debugger.h b/include/mgba/internal/debugger/cli-debugger.h index 7a03d1df0..60ec00fa4 100644 --- a/include/mgba/internal/debugger/cli-debugger.h +++ b/include/mgba/internal/debugger/cli-debugger.h @@ -74,6 +74,8 @@ struct CLIDebugger { struct CLIDebuggerSystem* system; struct CLIDebuggerBackend* backend; + + int traceRemaining; }; void CLIDebuggerCreate(struct CLIDebugger*); diff --git a/src/debugger/cli-debugger.c b/src/debugger/cli-debugger.c index 1cdfc20e8..d2a93aa30 100644 --- a/src/debugger/cli-debugger.c +++ b/src/debugger/cli-debugger.c @@ -29,6 +29,7 @@ const char* ERROR_OVERFLOW = "Arguments overflow"; const char* ERROR_INVALID_ARGS = "Invalid arguments"; static struct ParseTree* _parseTree(const char** string); +static bool _doTrace(struct CLIDebugger* debugger); #if !defined(NDEBUG) && !defined(_WIN32) static void _breakInto(struct CLIDebugger*, struct CLIDebugVector*); @@ -147,7 +148,7 @@ static void _breakInto(struct CLIDebugger* debugger, struct CLIDebugVector* dv) static void _continue(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { UNUSED(dv); - debugger->d.state = DEBUGGER_RUNNING; + debugger->d.state = debugger->traceRemaining != 0 ? DEBUGGER_CUSTOM : DEBUGGER_RUNNING; } static void _next(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { @@ -645,19 +646,31 @@ static void _trace(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { return; } + debugger->traceRemaining = dv->intValue; + if (debugger->traceRemaining == 0) { + return; + } + if (_doTrace(debugger)) { + debugger->d.state = DEBUGGER_CUSTOM; + } else { + debugger->system->printStatus(debugger->system); + } +} + +static bool _doTrace(struct CLIDebugger* debugger) { char trace[1024]; trace[sizeof(trace) - 1] = '\0'; - - int i; - for (i = 0; i < dv->intValue; ++i) { - debugger->d.core->step(debugger->d.core); - size_t traceSize = sizeof(trace) - 1; - debugger->d.platform->trace(debugger->d.platform, trace, &traceSize); - if (traceSize + 1 < sizeof(trace)) { - trace[traceSize + 1] = '\0'; - } - debugger->backend->printf(debugger->backend, "%s\n", trace); + debugger->d.core->step(debugger->d.core); + size_t traceSize = sizeof(trace) - 1; + debugger->d.platform->trace(debugger->d.platform, trace, &traceSize); + if (traceSize + 1 < sizeof(trace)) { + trace[traceSize + 1] = '\0'; } + debugger->backend->printf(debugger->backend, "%s\n", trace); + if (debugger->traceRemaining > 0) { + --debugger->traceRemaining; + } + return debugger->traceRemaining != 0; } static void _printStatus(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { @@ -889,6 +902,9 @@ static void _commandLine(struct mDebugger* debugger) { static void _reportEntry(struct mDebugger* debugger, enum mDebuggerEntryReason reason, struct mDebuggerEntryInfo* info) { struct CLIDebugger* cliDebugger = (struct CLIDebugger*) debugger; + if (cliDebugger->traceRemaining > 0) { + cliDebugger->traceRemaining = 0; + } switch (reason) { case DEBUGGER_ENTER_MANUAL: case DEBUGGER_ENTER_ATTACHED: @@ -923,6 +939,7 @@ static void _reportEntry(struct mDebugger* debugger, enum mDebuggerEntryReason r static void _cliDebuggerInit(struct mDebugger* debugger) { struct CLIDebugger* cliDebugger = (struct CLIDebugger*) debugger; + cliDebugger->traceRemaining = 0; cliDebugger->backend->init(cliDebugger->backend); } @@ -943,12 +960,17 @@ static void _cliDebuggerDeinit(struct mDebugger* debugger) { static void _cliDebuggerCustom(struct mDebugger* debugger) { struct CLIDebugger* cliDebugger = (struct CLIDebugger*) debugger; - bool retain = false; + bool retain = true; + enum mDebuggerState next = DEBUGGER_RUNNING; + if (cliDebugger->traceRemaining) { + retain = _doTrace(cliDebugger) && retain; + next = DEBUGGER_PAUSED; + } if (cliDebugger->system) { - retain = cliDebugger->system->custom(cliDebugger->system); + retain = cliDebugger->system->custom(cliDebugger->system) && retain; } if (!retain && debugger->state == DEBUGGER_CUSTOM) { - debugger->state = DEBUGGER_RUNNING; + debugger->state = next; } } diff --git a/src/gb/debugger/cli.c b/src/gb/debugger/cli.c index a051ea577..2d86b12ec 100644 --- a/src/gb/debugger/cli.c +++ b/src/gb/debugger/cli.c @@ -60,7 +60,7 @@ static bool _GBCLIDebuggerCustom(struct CLIDebuggerSystem* debugger) { gbDebugger->inVblank = GBRegisterSTATGetMode(((struct GB*) gbDebugger->core->board)->memory.io[REG_STAT]) == 1; return true; } - return false; + return true; } static void _frame(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { diff --git a/src/gba/debugger/cli.c b/src/gba/debugger/cli.c index 4eb13b0f3..c9f8bf2b7 100644 --- a/src/gba/debugger/cli.c +++ b/src/gba/debugger/cli.c @@ -59,7 +59,7 @@ static bool _GBACLIDebuggerCustom(struct CLIDebuggerSystem* debugger) { gbaDebugger->inVblank = GBARegisterDISPSTATGetInVblank(((struct GBA*) gbaDebugger->core->board)->memory.io[REG_DISPSTAT >> 1]); return true; } - return false; + return true; } static void _frame(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { From 1d6d4a5377ef0c9b1b1ed261e2ba852f082b0998 Mon Sep 17 00:00:00 2001 From: Vicki Pfau Date: Sun, 28 Apr 2019 13:27:10 -0700 Subject: [PATCH 05/10] Debugger: Add tracing to file --- CHANGES | 1 + include/mgba/internal/debugger/cli-debugger.h | 2 ++ src/debugger/cli-debugger.c | 27 ++++++++++++++++--- 3 files changed, 26 insertions(+), 4 deletions(-) diff --git a/CHANGES b/CHANGES index 2ef909c91..6d5c5c83a 100644 --- a/CHANGES +++ b/CHANGES @@ -6,6 +6,7 @@ Features: - Debugger: Add unary operators and memory dereferencing - GB: Expose platform information to CLI debugger - Support Discord Rich Presence + - Debugger: Add tracing to file Emulation fixes: - GBA: All IRQs have 7 cycle delay (fixes mgba.io/i/539, mgba.io/i/1208) - GBA: Reset now reloads multiboot ROMs diff --git a/include/mgba/internal/debugger/cli-debugger.h b/include/mgba/internal/debugger/cli-debugger.h index 60ec00fa4..0409ffe8e 100644 --- a/include/mgba/internal/debugger/cli-debugger.h +++ b/include/mgba/internal/debugger/cli-debugger.h @@ -17,6 +17,7 @@ extern const char* ERROR_OVERFLOW; extern const char* ERROR_INVALID_ARGS; struct CLIDebugger; +struct VFile; struct CLIDebugVector { struct CLIDebugVector* next; @@ -76,6 +77,7 @@ struct CLIDebugger { struct CLIDebuggerBackend* backend; int traceRemaining; + struct VFile* traceVf; }; void CLIDebuggerCreate(struct CLIDebugger*); diff --git a/src/debugger/cli-debugger.c b/src/debugger/cli-debugger.c index d2a93aa30..965e57f5a 100644 --- a/src/debugger/cli-debugger.c +++ b/src/debugger/cli-debugger.c @@ -11,6 +11,7 @@ #include #include #include +#include #if ENABLE_SCRIPTING #include @@ -100,7 +101,7 @@ static struct CLIDebuggerCommandSummary _debuggerCommands[] = { { "r/2", _readHalfword, "I", "Read a halfword from a specified offset" }, { "r/4", _readWord, "I", "Read a word from a specified offset" }, { "status", _printStatus, "", "Print the current status" }, - { "trace", _trace, "I", "Trace a fixed number of instructions" }, + { "trace", _trace, "Is", "Trace a number of instructions" }, { "w", _setReadWriteWatchpoint, "Is", "Set a watchpoint" }, { "w/1", _writeByte, "II", "Write a byte at a specified offset" }, { "w/2", _writeHalfword, "II", "Write a halfword at a specified offset" }, @@ -647,9 +648,16 @@ static void _trace(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { } debugger->traceRemaining = dv->intValue; + if (debugger->traceVf) { + debugger->traceVf->close(debugger->traceVf); + debugger->traceVf = NULL; + } if (debugger->traceRemaining == 0) { return; } + if (dv->next && dv->next->charValue) { + debugger->traceVf = VFileOpen(dv->next->charValue, O_CREAT | O_WRONLY | O_APPEND); + } if (_doTrace(debugger)) { debugger->d.state = DEBUGGER_CUSTOM; } else { @@ -661,12 +669,17 @@ static bool _doTrace(struct CLIDebugger* debugger) { char trace[1024]; trace[sizeof(trace) - 1] = '\0'; debugger->d.core->step(debugger->d.core); - size_t traceSize = sizeof(trace) - 1; + size_t traceSize = sizeof(trace) - 2; debugger->d.platform->trace(debugger->d.platform, trace, &traceSize); - if (traceSize + 1 < sizeof(trace)) { + if (traceSize + 1 <= sizeof(trace)) { + trace[traceSize] = '\n'; trace[traceSize + 1] = '\0'; } - debugger->backend->printf(debugger->backend, "%s\n", trace); + if (debugger->traceVf) { + debugger->traceVf->write(debugger->traceVf, trace, traceSize + 1); + } else { + debugger->backend->printf(debugger->backend, "%s", trace); + } if (debugger->traceRemaining > 0) { --debugger->traceRemaining; } @@ -940,11 +953,17 @@ static void _reportEntry(struct mDebugger* debugger, enum mDebuggerEntryReason r static void _cliDebuggerInit(struct mDebugger* debugger) { struct CLIDebugger* cliDebugger = (struct CLIDebugger*) debugger; cliDebugger->traceRemaining = 0; + cliDebugger->traceVf = NULL; cliDebugger->backend->init(cliDebugger->backend); } static void _cliDebuggerDeinit(struct mDebugger* debugger) { struct CLIDebugger* cliDebugger = (struct CLIDebugger*) debugger; + if (cliDebugger->traceVf) { + cliDebugger->traceVf->close(cliDebugger->traceVf); + cliDebugger->traceVf = NULL; + } + if (cliDebugger->system) { if (cliDebugger->system->deinit) { cliDebugger->system->deinit(cliDebugger->system); From 1deff07aa1b0d3d4b6b15e875fbeb0cc8eff0926 Mon Sep 17 00:00:00 2001 From: Vicki Pfau Date: Sun, 28 Apr 2019 13:37:51 -0700 Subject: [PATCH 06/10] Debugger: Print breakpoint/watchpoint number when inserting --- CHANGES | 1 + include/mgba/internal/debugger/cli-debugger.h | 2 ++ src/arm/debugger/cli-debugger.c | 14 ++++++++++---- src/debugger/cli-debugger.c | 12 ++++++++++-- 4 files changed, 23 insertions(+), 6 deletions(-) diff --git a/CHANGES b/CHANGES index 6d5c5c83a..a0af8727d 100644 --- a/CHANGES +++ b/CHANGES @@ -45,6 +45,7 @@ Misc: - Qt: Cap window size on start to monitor size - GBA BIOS: Add timings for HLE BIOS math functions (fixes mgba.io/i/1396) - Debugger: Make tracing compatible with breakpoints/watchpoints + - Debugger: Print breakpoint/watchpoint number when inserting 0.7.1: (2019-02-24) Bugfixes: diff --git a/include/mgba/internal/debugger/cli-debugger.h b/include/mgba/internal/debugger/cli-debugger.h index 0409ffe8e..35a1a936a 100644 --- a/include/mgba/internal/debugger/cli-debugger.h +++ b/include/mgba/internal/debugger/cli-debugger.h @@ -15,6 +15,8 @@ CXX_GUARD_START extern const char* ERROR_MISSING_ARGS; extern const char* ERROR_OVERFLOW; extern const char* ERROR_INVALID_ARGS; +extern const char* INFO_BREAKPOINT_ADDED; +extern const char* INFO_WATCHPOINT_ADDED; struct CLIDebugger; struct VFile; diff --git a/src/arm/debugger/cli-debugger.c b/src/arm/debugger/cli-debugger.c index fe1265a11..d458ad4aa 100644 --- a/src/arm/debugger/cli-debugger.c +++ b/src/arm/debugger/cli-debugger.c @@ -147,21 +147,27 @@ static void _printStatus(struct CLIDebuggerSystem* debugger) { static void _setBreakpointARM(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { struct CLIDebuggerBackend* be = debugger->backend; if (!dv || dv->type != CLIDV_INT_TYPE) { - be->printf(be, "%s\n", ERROR_MISSING_ARGS); + be->printf(be, "%s", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; - ARMDebuggerSetSoftwareBreakpoint(debugger->d.platform, address, MODE_ARM); + ssize_t id = ARMDebuggerSetSoftwareBreakpoint(debugger->d.platform, address, MODE_ARM); + if (id > 0) { + debugger->backend->printf(debugger->backend, INFO_BREAKPOINT_ADDED, id); + } } static void _setBreakpointThumb(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { struct CLIDebuggerBackend* be = debugger->backend; if (!dv || dv->type != CLIDV_INT_TYPE) { - be->printf(be, "%s\n", ERROR_MISSING_ARGS); + be->printf(be, "%s", ERROR_MISSING_ARGS); return; } uint32_t address = dv->intValue; - ARMDebuggerSetSoftwareBreakpoint(debugger->d.platform, address, MODE_THUMB); + ssize_t id = ARMDebuggerSetSoftwareBreakpoint(debugger->d.platform, address, MODE_THUMB); + if (id > 0) { + debugger->backend->printf(debugger->backend, INFO_BREAKPOINT_ADDED, id); + } } void ARMCLIDebuggerCreate(struct CLIDebuggerSystem* debugger) { diff --git a/src/debugger/cli-debugger.c b/src/debugger/cli-debugger.c index 965e57f5a..64d431297 100644 --- a/src/debugger/cli-debugger.c +++ b/src/debugger/cli-debugger.c @@ -28,6 +28,8 @@ const char* ERROR_MISSING_ARGS = "Arguments missing"; // TODO: share const char* ERROR_OVERFLOW = "Arguments overflow"; const char* ERROR_INVALID_ARGS = "Invalid arguments"; +const char* INFO_BREAKPOINT_ADDED = "Added breakpoint #%" PRIz "i\n"; +const char* INFO_WATCHPOINT_ADDED = "Added watchpoint #%" PRIz "i\n"; static struct ParseTree* _parseTree(const char** string); static bool _doTrace(struct CLIDebugger* debugger); @@ -553,7 +555,10 @@ static void _setBreakpoint(struct CLIDebugger* debugger, struct CLIDebugVector* return; } } - debugger->d.platform->setBreakpoint(debugger->d.platform, &breakpoint); + ssize_t id = debugger->d.platform->setBreakpoint(debugger->d.platform, &breakpoint); + if (id > 0) { + debugger->backend->printf(debugger->backend, INFO_BREAKPOINT_ADDED, id); + } } static void _setWatchpoint(struct CLIDebugger* debugger, struct CLIDebugVector* dv, enum mWatchpointType type) { @@ -579,7 +584,10 @@ static void _setWatchpoint(struct CLIDebugger* debugger, struct CLIDebugVector* return; } } - debugger->d.platform->setWatchpoint(debugger->d.platform, &watchpoint); + ssize_t id = debugger->d.platform->setWatchpoint(debugger->d.platform, &watchpoint); + if (id > 0) { + debugger->backend->printf(debugger->backend, INFO_WATCHPOINT_ADDED, id); + } } static void _setReadWriteWatchpoint(struct CLIDebugger* debugger, struct CLIDebugVector* dv) { From 19a42a387a4a9316e5626c8079bed4d989d9c3ac Mon Sep 17 00:00:00 2001 From: Vicki Pfau Date: Mon, 29 Apr 2019 12:16:41 -0700 Subject: [PATCH 07/10] LR35902: Fix trailing whitespace in disassembly --- CHANGES | 1 + src/lr35902/decoder.c | 15 +++++++++------ 2 files changed, 10 insertions(+), 6 deletions(-) diff --git a/CHANGES b/CHANGES index a0af8727d..37d9bdb00 100644 --- a/CHANGES +++ b/CHANGES @@ -30,6 +30,7 @@ Other fixes: - 3DS: Ensure core 2 can be used for threaded renderer (fixes mgba.io/i/1371) - GB Core: Fix toggling WIN and OBJ being swapped - All: Fix several memory leaks + - LR35902: Fix trailing whitespace in disassembly Misc: - GBA Savedata: EEPROM performance fixes - GBA Savedata: Automatically map 1Mbit Flash files as 1Mbit Flash diff --git a/src/lr35902/decoder.c b/src/lr35902/decoder.c index 089b52c69..af7a605e6 100644 --- a/src/lr35902/decoder.c +++ b/src/lr35902/decoder.c @@ -504,6 +504,9 @@ static int _decodeOperand(struct LR35902Operand op, uint16_t pc, char* buffer, i return 0; } + strncpy(buffer, " ", blen - 1); + ADVANCE(1); + if (op.flags & LR35902_OP_FLAG_MEMORY) { strncpy(buffer, "[", blen - 1); ADVANCE(1); @@ -545,16 +548,16 @@ int LR35902Disassemble(struct LR35902InstructionInfo* info, uint16_t pc, char* b int total = 0; const char* cond = _lr35902Conditions[info->condition]; - written = snprintf(buffer, blen - 1, "%s ", mnemonic); + written = snprintf(buffer, blen - 1, "%s", mnemonic); ADVANCE(written); if (cond) { - written = snprintf(buffer, blen - 1, "%s", cond); + written = snprintf(buffer, blen - 1, " %s", cond); ADVANCE(written); if (info->op1.reg || info->op1.immediate || info->op2.reg || info->op2.immediate) { - strncpy(buffer, ", ", blen - 1); - ADVANCE(2); + strncpy(buffer, ",", blen - 1); + ADVANCE(1); } } @@ -565,8 +568,8 @@ int LR35902Disassemble(struct LR35902InstructionInfo* info, uint16_t pc, char* b if (info->op2.reg || (!info->op1.immediate && info->opcodeSize > 1 && info->opcode[0] != 0xCB)) { if (written) { - strncpy(buffer, ", ", blen - 1); - ADVANCE(2); + strncpy(buffer, ",", blen - 1); + ADVANCE(1); } written = _decodeOperand(info->op2, pc, buffer, blen); ADVANCE(written); From fbe375fab992cb4bb2fbc005ed26bc488ccb3389 Mon Sep 17 00:00:00 2001 From: Vicki Pfau Date: Tue, 30 Apr 2019 17:44:56 -0700 Subject: [PATCH 08/10] Core: Add support for caching bitmapped modes --- CHANGES | 1 + include/mgba/core/bitmap-cache.h | 64 +++++++++++ include/mgba/core/cache-set.h | 5 +- src/core/bitmap-cache.c | 182 +++++++++++++++++++++++++++++++ src/core/cache-set.c | 20 +++- src/gb/renderers/cache-set.c | 5 +- src/gba/renderers/cache-set.c | 52 +++++++-- src/platform/qt/MapView.cpp | 43 ++++++-- src/platform/qt/MapView.h | 1 + 9 files changed, 347 insertions(+), 26 deletions(-) create mode 100644 include/mgba/core/bitmap-cache.h create mode 100644 src/core/bitmap-cache.c diff --git a/CHANGES b/CHANGES index 37d9bdb00..76c9d07ee 100644 --- a/CHANGES +++ b/CHANGES @@ -7,6 +7,7 @@ Features: - GB: Expose platform information to CLI debugger - Support Discord Rich Presence - Debugger: Add tracing to file + - Map viewer supports bitmapped GBA modes Emulation fixes: - GBA: All IRQs have 7 cycle delay (fixes mgba.io/i/539, mgba.io/i/1208) - GBA: Reset now reloads multiboot ROMs diff --git a/include/mgba/core/bitmap-cache.h b/include/mgba/core/bitmap-cache.h new file mode 100644 index 000000000..d8bd416e9 --- /dev/null +++ b/include/mgba/core/bitmap-cache.h @@ -0,0 +1,64 @@ +/* Copyright (c) 2013-2019 Jeffrey Pfau + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ +#ifndef M_BITMAP_CACHE_H +#define M_BITMAP_CACHE_H + +#include + +CXX_GUARD_START + +#include + +DECL_BITFIELD(mBitmapCacheConfiguration, uint32_t); +DECL_BIT(mBitmapCacheConfiguration, ShouldStore, 0); + +DECL_BITFIELD(mBitmapCacheSystemInfo, uint32_t); +DECL_BITS(mBitmapCacheSystemInfo, EntryBPP, 0, 3); +DECL_BIT(mBitmapCacheSystemInfo, UsesPalette, 3); +DECL_BITS(mBitmapCacheSystemInfo, Width, 4, 10); +DECL_BITS(mBitmapCacheSystemInfo, Height, 14, 10); +DECL_BITS(mBitmapCacheSystemInfo, Buffers, 24, 2); + +struct mBitmapCacheEntry { + uint32_t paletteVersion; + uint32_t vramVersion; + uint8_t vramClean; +}; + +struct mBitmapCache { + color_t* cache; + struct mBitmapCacheEntry* status; + + uint32_t globalPaletteVersion; + + uint8_t* vram; + color_t* palette; + + uint32_t bitsSize; + uint32_t bitsStart[2]; + uint32_t stride; + uint8_t buffer; + + mBitmapCacheConfiguration config; + mBitmapCacheSystemInfo sysConfig; + + void* context; +}; + +void mBitmapCacheInit(struct mBitmapCache* cache); +void mBitmapCacheDeinit(struct mBitmapCache* cache); +void mBitmapCacheConfigure(struct mBitmapCache* cache, mBitmapCacheConfiguration config); +void mBitmapCacheConfigureSystem(struct mBitmapCache* cache, mBitmapCacheSystemInfo config); +void mBitmapCacheWriteVRAM(struct mBitmapCache* cache, uint32_t address); +void mBitmapCacheWritePalette(struct mBitmapCache* cache, uint32_t entry, color_t color); + +void mBitmapCacheCleanRow(struct mBitmapCache* cache, struct mBitmapCacheEntry* entry, unsigned y); +bool mBitmapCacheCheckRow(struct mBitmapCache* cache, const struct mBitmapCacheEntry* entry, unsigned y); +const color_t* mBitmapCacheGetRow(struct mBitmapCache* cache, unsigned y); + +CXX_GUARD_END + +#endif diff --git a/include/mgba/core/cache-set.h b/include/mgba/core/cache-set.h index 8fed166a9..5749c0e64 100644 --- a/include/mgba/core/cache-set.h +++ b/include/mgba/core/cache-set.h @@ -10,19 +10,22 @@ CXX_GUARD_START +#include #include #include #include DECLARE_VECTOR(mMapCacheSet, struct mMapCache); +DECLARE_VECTOR(mBitmapCacheSet, struct mBitmapCache); DECLARE_VECTOR(mTileCacheSet, struct mTileCache); struct mCacheSet { struct mMapCacheSet maps; + struct mBitmapCacheSet bitmaps; struct mTileCacheSet tiles; }; -void mCacheSetInit(struct mCacheSet*, size_t nMaps, size_t nTiles); +void mCacheSetInit(struct mCacheSet*, size_t nMaps, size_t nBitmaps, size_t nTiles); void mCacheSetDeinit(struct mCacheSet*); void mCacheSetAssignVRAM(struct mCacheSet*, void* vram); diff --git a/src/core/bitmap-cache.c b/src/core/bitmap-cache.c new file mode 100644 index 000000000..c86768e45 --- /dev/null +++ b/src/core/bitmap-cache.c @@ -0,0 +1,182 @@ +/* Copyright (c) 2013-2019 Jeffrey Pfau + * + * This Source Code Form is subject to the terms of the Mozilla Public + * License, v. 2.0. If a copy of the MPL was not distributed with this + * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ +#include + +#include + +void mBitmapCacheInit(struct mBitmapCache* cache) { + // TODO: Reconfigurable cache for space savings + cache->cache = NULL; + cache->config = mBitmapCacheConfigurationFillShouldStore(0); + cache->status = NULL; + cache->palette = NULL; + cache->buffer = 0; +} + +static void _freeCache(struct mBitmapCache* cache) { + size_t size = mBitmapCacheSystemInfoGetHeight(cache->sysConfig) * mBitmapCacheSystemInfoGetBuffers(cache->sysConfig); + mappedMemoryFree(cache->cache, mBitmapCacheSystemInfoGetWidth(cache->sysConfig) * size * sizeof(color_t)); + mappedMemoryFree(cache->status, size * sizeof(*cache->status)); + if (cache->palette) { + free(cache->palette); + } + cache->cache = NULL; + cache->status = NULL; + cache->palette = NULL; +} + +static void _redoCacheSize(struct mBitmapCache* cache) { + if (!mBitmapCacheConfigurationIsShouldStore(cache->config)) { + return; + } + + size_t size = mBitmapCacheSystemInfoGetHeight(cache->sysConfig) * mBitmapCacheSystemInfoGetBuffers(cache->sysConfig); + cache->cache = anonymousMemoryMap(mBitmapCacheSystemInfoGetWidth(cache->sysConfig) * size * sizeof(color_t)); + cache->status = anonymousMemoryMap(size * sizeof(*cache->status)); + if (mBitmapCacheSystemInfoIsUsesPalette(cache->sysConfig)) { + cache->palette = malloc((1 << (1 << mBitmapCacheSystemInfoGetEntryBPP(cache->sysConfig))) * sizeof(color_t)); + } else { + cache->palette = NULL; + } +} + +void mBitmapCacheConfigure(struct mBitmapCache* cache, mBitmapCacheConfiguration config) { + if (config == cache->config) { + return; + } + _freeCache(cache); + cache->config = config; + _redoCacheSize(cache); +} + +void mBitmapCacheConfigureSystem(struct mBitmapCache* cache, mBitmapCacheSystemInfo config) { + if (config == cache->sysConfig) { + return; + } + _freeCache(cache); + cache->sysConfig = config; + _redoCacheSize(cache); + + size_t stride = mBitmapCacheSystemInfoGetWidth(cache->sysConfig); + size_t size = stride * mBitmapCacheSystemInfoGetHeight(cache->sysConfig); + size_t bpe = mBitmapCacheSystemInfoGetEntryBPP(cache->sysConfig); + if (bpe > 3) { + size <<= bpe - 3; + stride <<= bpe - 3; + } else { + size >>= 3 - bpe; + stride >>= 3 - bpe; + } + cache->bitsSize = size; + cache->stride = stride; +} + +void mBitmapCacheDeinit(struct mBitmapCache* cache) { + _freeCache(cache); +} + +void mBitmapCacheWriteVRAM(struct mBitmapCache* cache, uint32_t address) { + size_t i; + for (i = 0; i < mBitmapCacheSystemInfoGetBuffers(cache->sysConfig); ++i) { + if (address < cache->bitsStart[i]) { + continue; + } + uint32_t offset = address - cache->bitsStart[i]; + if (offset >= cache->bitsSize) { + continue; + } + offset /= cache->stride; + offset *= mBitmapCacheSystemInfoGetBuffers(cache->sysConfig); + offset += cache->buffer; + cache->status[offset].vramClean = 0; + ++cache->status[offset].vramVersion; + } +} + +void mBitmapCacheWritePalette(struct mBitmapCache* cache, uint32_t entry, color_t color) { + if (!mBitmapCacheSystemInfoIsUsesPalette(cache->sysConfig)) { + return; + } + size_t maxEntry = 1 << (1 << mBitmapCacheSystemInfoGetEntryBPP(cache->sysConfig)); + if (entry >= maxEntry) { + return; + } + cache->palette[entry] = color; + ++cache->globalPaletteVersion; +} + +uint32_t _lookupEntry8(void* vram, uint32_t offset) { + return ((uint8_t*) vram)[offset]; +} + +uint32_t _lookupEntry15(void* vram, uint32_t offset) { + return mColorFrom555(((uint16_t*) vram)[offset]); +} + +void mBitmapCacheCleanRow(struct mBitmapCache* cache, struct mBitmapCacheEntry* entry, unsigned y) { + color_t* row = &cache->cache[(cache->buffer * mBitmapCacheSystemInfoGetHeight(cache->sysConfig) + y) * mBitmapCacheSystemInfoGetWidth(cache->sysConfig)]; + size_t location = cache->buffer + mBitmapCacheSystemInfoGetBuffers(cache->sysConfig) * y; + struct mBitmapCacheEntry* status = &cache->status[location]; + struct mBitmapCacheEntry desiredStatus = { + .paletteVersion = cache->globalPaletteVersion, + .vramVersion = entry->vramVersion, + .vramClean = 1 + }; + + if (entry) { + entry[location] = desiredStatus; + } + + if (!mBitmapCacheConfigurationIsShouldStore(cache->config) || !memcmp(status, &desiredStatus, sizeof(*entry))) { + return; + } + + size_t offset = cache->bitsStart[cache->buffer] + y * mBitmapCacheSystemInfoGetWidth(cache->sysConfig); + void* vram; + int bpe = mBitmapCacheSystemInfoGetEntryBPP(cache->sysConfig); + uint32_t (*lookupEntry)(void*, uint32_t); + switch (bpe) { + case 3: + lookupEntry = _lookupEntry8; + vram = &cache->vram[offset]; + break; + case 4: + lookupEntry = _lookupEntry15; + vram = &cache->vram[offset << 1]; + break; + default: + abort(); + break; + } + + size_t x; + if (mBitmapCacheSystemInfoIsUsesPalette(cache->sysConfig)) { + for (x = 0; x < mBitmapCacheSystemInfoGetWidth(cache->sysConfig); ++x) { + row[x] = cache->palette[lookupEntry(vram, x)]; + } + } else { + for (x = 0; x < mBitmapCacheSystemInfoGetWidth(cache->sysConfig); ++x) { + row[x] = lookupEntry(vram, x); + } + } + *status = desiredStatus; +} + +bool mBitmapCacheCheckRow(struct mBitmapCache* cache, const struct mBitmapCacheEntry* entry, unsigned y) { + size_t location = cache->buffer + mBitmapCacheSystemInfoGetBuffers(cache->sysConfig) * y; + struct mBitmapCacheEntry desiredStatus = { + .paletteVersion = cache->globalPaletteVersion, + .vramVersion = entry->vramVersion, + .vramClean = 1 + }; + + return memcmp(&entry[location], &desiredStatus, sizeof(*entry)) == 0; +} + +const color_t* mBitmapCacheGetRow(struct mBitmapCache* cache, unsigned y) { + color_t* row = &cache->cache[(cache->buffer * mBitmapCacheSystemInfoGetHeight(cache->sysConfig) + y) * mBitmapCacheSystemInfoGetWidth(cache->sysConfig)]; + return row; +} \ No newline at end of file diff --git a/src/core/cache-set.c b/src/core/cache-set.c index 7561981e3..a638c786e 100644 --- a/src/core/cache-set.c +++ b/src/core/cache-set.c @@ -6,11 +6,14 @@ #include DEFINE_VECTOR(mMapCacheSet, struct mMapCache); +DEFINE_VECTOR(mBitmapCacheSet, struct mBitmapCache); DEFINE_VECTOR(mTileCacheSet, struct mTileCache); -void mCacheSetInit(struct mCacheSet* cache, size_t nMaps, size_t nTiles) { +void mCacheSetInit(struct mCacheSet* cache, size_t nMaps, size_t nBitmaps, size_t nTiles) { mMapCacheSetInit(&cache->maps, nMaps); mMapCacheSetResize(&cache->maps, nMaps); + mBitmapCacheSetInit(&cache->bitmaps, nBitmaps); + mBitmapCacheSetResize(&cache->bitmaps, nBitmaps); mTileCacheSetInit(&cache->tiles, nTiles); mTileCacheSetResize(&cache->tiles, nTiles); @@ -18,6 +21,9 @@ void mCacheSetInit(struct mCacheSet* cache, size_t nMaps, size_t nTiles) { for (i = 0; i < nMaps; ++i) { mMapCacheInit(mMapCacheSetGetPointer(&cache->maps, i)); } + for (i = 0; i < nBitmaps; ++i) { + mBitmapCacheInit(mBitmapCacheSetGetPointer(&cache->bitmaps, i)); + } for (i = 0; i < nTiles; ++i) { mTileCacheInit(mTileCacheSetGetPointer(&cache->tiles, i)); } @@ -28,6 +34,9 @@ void mCacheSetDeinit(struct mCacheSet* cache) { for (i = 0; i < mMapCacheSetSize(&cache->maps); ++i) { mMapCacheDeinit(mMapCacheSetGetPointer(&cache->maps, i)); } + for (i = 0; i < mBitmapCacheSetSize(&cache->bitmaps); ++i) { + mBitmapCacheDeinit(mBitmapCacheSetGetPointer(&cache->bitmaps, i)); + } for (i = 0; i < mTileCacheSetSize(&cache->tiles); ++i) { mTileCacheDeinit(mTileCacheSetGetPointer(&cache->tiles, i)); } @@ -38,6 +47,9 @@ void mCacheSetAssignVRAM(struct mCacheSet* cache, void* vram) { for (i = 0; i < mMapCacheSetSize(&cache->maps); ++i) { mMapCacheSetGetPointer(&cache->maps, i)->vram = vram; } + for (i = 0; i < mBitmapCacheSetSize(&cache->bitmaps); ++i) { + mBitmapCacheSetGetPointer(&cache->bitmaps, i)->vram = vram; + } for (i = 0; i < mTileCacheSetSize(&cache->tiles); ++i) { struct mTileCache* tileCache = mTileCacheSetGetPointer(&cache->tiles, i); tileCache->vram = (void*) ((uintptr_t) vram + tileCache->tileBase); @@ -49,6 +61,9 @@ void mCacheSetWriteVRAM(struct mCacheSet* cache, uint32_t address) { for (i = 0; i < mMapCacheSetSize(&cache->maps); ++i) { mMapCacheWriteVRAM(mMapCacheSetGetPointer(&cache->maps, i), address); } + for (i = 0; i < mBitmapCacheSetSize(&cache->bitmaps); ++i) { + mBitmapCacheWriteVRAM(mBitmapCacheSetGetPointer(&cache->bitmaps, i), address); + } for (i = 0; i < mTileCacheSetSize(&cache->tiles); ++i) { mTileCacheWriteVRAM(mTileCacheSetGetPointer(&cache->tiles, i), address); } @@ -56,6 +71,9 @@ void mCacheSetWriteVRAM(struct mCacheSet* cache, uint32_t address) { void mCacheSetWritePalette(struct mCacheSet* cache, uint32_t entry, color_t color) { size_t i; + for (i = 0; i < mBitmapCacheSetSize(&cache->bitmaps); ++i) { + mBitmapCacheWritePalette(mBitmapCacheSetGetPointer(&cache->bitmaps, i), entry, color); + } for (i = 0; i < mTileCacheSetSize(&cache->tiles); ++i) { mTileCacheWritePalette(mTileCacheSetGetPointer(&cache->tiles, i), entry, color); } diff --git a/src/gb/renderers/cache-set.c b/src/gb/renderers/cache-set.c index 741d043cd..9703c09c1 100644 --- a/src/gb/renderers/cache-set.c +++ b/src/gb/renderers/cache-set.c @@ -11,16 +11,13 @@ #include void GBVideoCacheInit(struct mCacheSet* cache) { - mCacheSetInit(cache, 2, 1); + mCacheSetInit(cache, 2, 0, 1); mTileCacheConfiguration config = 0; config = mTileCacheSystemInfoSetPaletteBPP(config, 1); // 2^(2^1) = 4 entries config = mTileCacheSystemInfoSetPaletteCount(config, 4); // 16 palettes config = mTileCacheSystemInfoSetMaxTiles(config, 1024); - mTileCacheInit(mTileCacheSetGetPointer(&cache->tiles, 0)); mTileCacheConfigureSystem(mTileCacheSetGetPointer(&cache->tiles, 0), config, 0, 0); - mMapCacheInit(mMapCacheSetGetPointer(&cache->maps, 0)); - mMapCacheInit(mMapCacheSetGetPointer(&cache->maps, 1)); mMapCacheSetGetPointer(&cache->maps, 0)->tileCache = mTileCacheSetGetPointer(&cache->tiles, 0); mMapCacheSetGetPointer(&cache->maps, 1)->tileCache = mTileCacheSetGetPointer(&cache->tiles, 0); } diff --git a/src/gba/renderers/cache-set.c b/src/gba/renderers/cache-set.c index dae92b675..71a4b8617 100644 --- a/src/gba/renderers/cache-set.c +++ b/src/gba/renderers/cache-set.c @@ -11,35 +11,45 @@ #include void GBAVideoCacheInit(struct mCacheSet* cache) { - mCacheSetInit(cache, 4, 4); + mCacheSetInit(cache, 4, 2, 4); mTileCacheSystemInfo sysconfig = 0; mTileCacheConfiguration config = mTileCacheConfigurationFillShouldStore(0); sysconfig = mTileCacheSystemInfoSetPaletteBPP(sysconfig, 2); // 2^(2^2) = 16 entries sysconfig = mTileCacheSystemInfoSetPaletteCount(sysconfig, 4); // 16 palettes sysconfig = mTileCacheSystemInfoSetMaxTiles(sysconfig, 2048); - mTileCacheInit(mTileCacheSetGetPointer(&cache->tiles, 0)); mTileCacheConfigure(mTileCacheSetGetPointer(&cache->tiles, 0), config); mTileCacheConfigureSystem(mTileCacheSetGetPointer(&cache->tiles, 0), sysconfig, 0, 0); sysconfig = mTileCacheSystemInfoSetMaxTiles(sysconfig, 1024); - mTileCacheInit(mTileCacheSetGetPointer(&cache->tiles, 2)); mTileCacheConfigure(mTileCacheSetGetPointer(&cache->tiles, 2), config); mTileCacheConfigureSystem(mTileCacheSetGetPointer(&cache->tiles, 2), sysconfig, 0x10000, 0x100); sysconfig = mTileCacheSystemInfoSetPaletteBPP(sysconfig, 3); // 2^(2^3) = 256 entries sysconfig = mTileCacheSystemInfoSetPaletteCount(sysconfig, 0); // 1 palettes sysconfig = mTileCacheSystemInfoSetMaxTiles(sysconfig, 2048); - mTileCacheInit(mTileCacheSetGetPointer(&cache->tiles, 1)); mTileCacheConfigure(mTileCacheSetGetPointer(&cache->tiles, 1), config); mTileCacheConfigureSystem(mTileCacheSetGetPointer(&cache->tiles, 1), sysconfig, 0, 0); sysconfig = mTileCacheSystemInfoSetMaxTiles(sysconfig, 1024); - mTileCacheInit(mTileCacheSetGetPointer(&cache->tiles, 3)); mTileCacheConfigure(mTileCacheSetGetPointer(&cache->tiles, 3), config); mTileCacheConfigureSystem(mTileCacheSetGetPointer(&cache->tiles, 3), sysconfig, 0x10000, 0x100); - mMapCacheInit(mMapCacheSetGetPointer(&cache->maps, 0)); - mMapCacheInit(mMapCacheSetGetPointer(&cache->maps, 1)); - mMapCacheInit(mMapCacheSetGetPointer(&cache->maps, 2)); - mMapCacheInit(mMapCacheSetGetPointer(&cache->maps, 3)); + mBitmapCacheSystemInfo bitConfig; + bitConfig = mBitmapCacheSystemInfoSetEntryBPP(0, 4); + bitConfig = mBitmapCacheSystemInfoClearUsesPalette(bitConfig); + bitConfig = mBitmapCacheSystemInfoSetHeight(bitConfig, 160); + bitConfig = mBitmapCacheSystemInfoSetWidth(bitConfig, 240); + bitConfig = mBitmapCacheSystemInfoSetBuffers(bitConfig, 1); + mBitmapCacheConfigureSystem(mBitmapCacheSetGetPointer(&cache->bitmaps, 0), bitConfig); + mBitmapCacheSetGetPointer(&cache->bitmaps, 0)->bitsStart[0] = 0; + mBitmapCacheSetGetPointer(&cache->bitmaps, 0)->bitsStart[1] = 0xA000; + + bitConfig = mBitmapCacheSystemInfoSetEntryBPP(0, 3); + bitConfig = mBitmapCacheSystemInfoFillUsesPalette(bitConfig); + bitConfig = mBitmapCacheSystemInfoSetHeight(bitConfig, 160); + bitConfig = mBitmapCacheSystemInfoSetWidth(bitConfig, 240); + bitConfig = mBitmapCacheSystemInfoSetBuffers(bitConfig, 2); + mBitmapCacheConfigureSystem(mBitmapCacheSetGetPointer(&cache->bitmaps, 1), bitConfig); + mBitmapCacheSetGetPointer(&cache->bitmaps, 1)->bitsStart[0] = 0; + mBitmapCacheSetGetPointer(&cache->bitmaps, 1)->bitsStart[1] = 0xA000; } void GBAVideoCacheAssociate(struct mCacheSet* cache, struct GBAVideo* video) { @@ -77,6 +87,8 @@ static void mapParser2(struct mMapCache* cache, struct mMapCacheEntry* entry, vo } static void GBAVideoCacheWriteDISPCNT(struct mCacheSet* cache, uint16_t value) { + mBitmapCacheSetGetPointer(&cache->bitmaps, 1)->buffer = GBARegisterDISPCNTGetFrameSelect(value); + switch (GBARegisterDISPCNTGetMode(value)) { case 0: default: @@ -110,6 +122,28 @@ static void GBAVideoCacheWriteDISPCNT(struct mCacheSet* cache, uint16_t value) { mMapCacheSetGetPointer(&cache->maps, 3)->tileCache = mTileCacheSetGetPointer(&cache->tiles, 1); break; } + + mBitmapCacheSystemInfo bitConfig; + switch (GBARegisterDISPCNTGetMode(value)) { + case 3: + bitConfig = mBitmapCacheSystemInfoSetEntryBPP(0, 4); + bitConfig = mBitmapCacheSystemInfoClearUsesPalette(bitConfig); + bitConfig = mBitmapCacheSystemInfoSetHeight(bitConfig, 160); + bitConfig = mBitmapCacheSystemInfoSetWidth(bitConfig, 240); + bitConfig = mBitmapCacheSystemInfoSetBuffers(bitConfig, 1); + mBitmapCacheConfigureSystem(mBitmapCacheSetGetPointer(&cache->bitmaps, 0), bitConfig); + mBitmapCacheSetGetPointer(&cache->bitmaps, 0)->buffer = 0; + break; + case 5: + bitConfig = mBitmapCacheSystemInfoSetEntryBPP(0, 4); + bitConfig = mBitmapCacheSystemInfoClearUsesPalette(bitConfig); + bitConfig = mBitmapCacheSystemInfoSetHeight(bitConfig, 128); + bitConfig = mBitmapCacheSystemInfoSetWidth(bitConfig, 160); + bitConfig = mBitmapCacheSystemInfoSetBuffers(bitConfig, 2); + mBitmapCacheConfigureSystem(mBitmapCacheSetGetPointer(&cache->bitmaps, 0), bitConfig); + mBitmapCacheSetGetPointer(&cache->bitmaps, 0)->buffer = GBARegisterDISPCNTGetFrameSelect(value); + break; + } } static void GBAVideoCacheWriteBGCNT(struct mCacheSet* cache, size_t bg, uint16_t value) { diff --git a/src/platform/qt/MapView.cpp b/src/platform/qt/MapView.cpp index e4d970e5f..a0232c50b 100644 --- a/src/platform/qt/MapView.cpp +++ b/src/platform/qt/MapView.cpp @@ -12,7 +12,10 @@ #include #include #ifdef M_CORE_GBA +#include +#include #include +#include #endif #ifdef M_CORE_GB #include @@ -139,17 +142,36 @@ bool MapView::eventFilter(QObject* obj, QEvent* event) { void MapView::updateTilesGBA(bool force) { { CoreController::Interrupter interrupter(m_controller); - mMapCache* mapCache = mMapCacheSetGetPointer(&m_cacheSet->maps, m_map); - int tilesW = 1 << mMapCacheSystemInfoGetTilesWide(mapCache->sysConfig); - int tilesH = 1 << mMapCacheSystemInfoGetTilesHigh(mapCache->sysConfig); - m_rawMap = QImage(QSize(tilesW * 8, tilesH * 8), QImage::Format_ARGB32); - uchar* bgBits = m_rawMap.bits(); - for (int j = 0; j < tilesH; ++j) { - for (int i = 0; i < tilesW; ++i) { - mMapCacheCleanTile(mapCache, m_mapStatus, i, j); + int bitmap = -1; + if (m_controller->platform() == PLATFORM_GBA) { + int mode = GBARegisterDISPCNTGetMode(static_cast(m_controller->thread()->core->board)->memory.io[REG_DISPCNT]); + if (m_map == 2 && mode > 2) { + bitmap = mode == 4 ? 1 : 0; } - for (int i = 0; i < 8; ++i) { - memcpy(static_cast(&bgBits[tilesW * 32 * (i + j * 8)]), mMapCacheGetRow(mapCache, i + j * 8), tilesW * 32); + } + if (bitmap >= 0) { + mBitmapCache* bitmapCache = mBitmapCacheSetGetPointer(&m_cacheSet->bitmaps, bitmap); + int width = mBitmapCacheSystemInfoGetWidth(bitmapCache->sysConfig); + int height = mBitmapCacheSystemInfoGetHeight(bitmapCache->sysConfig); + m_rawMap = QImage(QSize(width, height), QImage::Format_ARGB32); + uchar* bgBits = m_rawMap.bits(); + for (int j = 0; j < height; ++j) { + mBitmapCacheCleanRow(bitmapCache, m_bitmapStatus, j); + memcpy(static_cast(&bgBits[width * j * 4]), mBitmapCacheGetRow(bitmapCache, j), width * 4); + } + } else { + mMapCache* mapCache = mMapCacheSetGetPointer(&m_cacheSet->maps, m_map); + int tilesW = 1 << mMapCacheSystemInfoGetTilesWide(mapCache->sysConfig); + int tilesH = 1 << mMapCacheSystemInfoGetTilesHigh(mapCache->sysConfig); + m_rawMap = QImage(QSize(tilesW * 8, tilesH * 8), QImage::Format_ARGB32); + uchar* bgBits = m_rawMap.bits(); + for (int j = 0; j < tilesH; ++j) { + for (int i = 0; i < tilesW; ++i) { + mMapCacheCleanTile(mapCache, m_mapStatus, i, j); + } + for (int i = 0; i < 8; ++i) { + memcpy(static_cast(&bgBits[tilesW * 32 * (i + j * 8)]), mMapCacheGetRow(mapCache, i + j * 8), tilesW * 32); + } } } } @@ -181,7 +203,6 @@ void MapView::exportMap() { png_structp png = PNGWriteOpen(vf); png_infop info = PNGWriteHeaderA(png, m_rawMap.width(), m_rawMap.height()); - mMapCache* mapCache = mMapCacheSetGetPointer(&m_cacheSet->maps, m_map); QImage map = m_rawMap.rgbSwapped(); PNGWritePixelsA(png, map.width(), map.height(), map.bytesPerLine() / 4, static_cast(map.constBits())); PNGWriteClose(png, info); diff --git a/src/platform/qt/MapView.h b/src/platform/qt/MapView.h index c8aa0867a..4b39dcd7a 100644 --- a/src/platform/qt/MapView.h +++ b/src/platform/qt/MapView.h @@ -45,6 +45,7 @@ private: std::shared_ptr m_controller; mMapCacheEntry m_mapStatus[128 * 128] = {}; // TODO: Correct size + mBitmapCacheEntry m_bitmapStatus[512 * 2] = {}; // TODO: Correct size int m_map = 0; QImage m_rawMap; int m_boundary; From 0f9c4bbe60d4c30e10ac0d15d5c653297c51e034 Mon Sep 17 00:00:00 2001 From: Vicki Pfau Date: Tue, 30 Apr 2019 17:45:38 -0700 Subject: [PATCH 09/10] GBA Memory: Fix writing to OBJ memory in modes 3 and 5 --- CHANGES | 1 + src/gba/memory.c | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/CHANGES b/CHANGES index 76c9d07ee..d5d08b3e5 100644 --- a/CHANGES +++ b/CHANGES @@ -20,6 +20,7 @@ Emulation fixes: - GB Video: Fix window y changing mid-window (fixes mgba.io/i/1345) - GB Video: Fix more window edge cases (fixes mgba.io/i/1346) - GB Timer: Fix timing adjustments when writing to TAC (fixes mgba.io/i/1340) + - GBA Memory: Fix writing to OBJ memory in modes 3 and 5 Other fixes: - Qt: More app metadata fixes - Qt: Fix load recent from archive (fixes mgba.io/i/1325) diff --git a/src/gba/memory.c b/src/gba/memory.c index 32d707c60..46aecae5b 100644 --- a/src/gba/memory.c +++ b/src/gba/memory.c @@ -938,7 +938,7 @@ void GBAStore8(struct ARMCore* cpu, uint32_t address, int8_t value, int* cycleCo GBAStore16(cpu, address & ~1, ((uint8_t) value) | ((uint8_t) value << 8), cycleCounter); break; case REGION_VRAM: - if ((address & 0x0001FFFF) >= ((GBARegisterDISPCNTGetMode(gba->memory.io[REG_DISPCNT >> 1]) == 4) ? 0x00014000 : 0x00010000)) { + if ((address & 0x0001FFFF) >= ((GBARegisterDISPCNTGetMode(gba->memory.io[REG_DISPCNT >> 1]) >= 3) ? 0x00014000 : 0x00010000)) { // TODO: check BG mode mLOG(GBA_MEM, GAME_ERROR, "Cannot Store8 to OBJ: 0x%08X", address); break; From 37a564da4c34ec84f95a1224e5f5fc1161176184 Mon Sep 17 00:00:00 2001 From: Vicki Pfau Date: Tue, 30 Apr 2019 18:16:44 -0700 Subject: [PATCH 10/10] Qt: Fix adjusting magnification in tile viewer when not fitting to window --- CHANGES | 1 + src/platform/qt/TileView.cpp | 5 ++++- 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/CHANGES b/CHANGES index d5d08b3e5..049717040 100644 --- a/CHANGES +++ b/CHANGES @@ -33,6 +33,7 @@ Other fixes: - GB Core: Fix toggling WIN and OBJ being swapped - All: Fix several memory leaks - LR35902: Fix trailing whitespace in disassembly + - Qt: Fix adjusting magnification in tile viewer when not fitting to window Misc: - GBA Savedata: EEPROM performance fixes - GBA Savedata: Automatically map 1Mbit Flash files as 1Mbit Flash diff --git a/src/platform/qt/TileView.cpp b/src/platform/qt/TileView.cpp index bc7381f7a..01673c9df 100644 --- a/src/platform/qt/TileView.cpp +++ b/src/platform/qt/TileView.cpp @@ -65,7 +65,10 @@ TileView::TileView(std::shared_ptr controller, QWidget* parent) } updateTiles(true); }); - connect(m_ui.magnification, static_cast(&QSpinBox::valueChanged), [this]() { + connect(m_ui.magnification, static_cast(&QSpinBox::valueChanged), [this](int mag) { + if (!m_ui.tileFit->isChecked()) { + m_ui.tiles->setMinimumSize(mag * 8 * m_ui.tilesPerRow->value(), m_ui.tiles->minimumSize().height()); + } updateTiles(true); });