Merge branch 'master' into medusa

This commit is contained in:
Vicki Pfau 2017-04-22 02:14:56 -07:00
commit 154dd2e8e7
34 changed files with 244 additions and 82 deletions

View File

@ -49,6 +49,8 @@ Features:
- Savestates now contain any RTC override data - Savestates now contain any RTC override data
- Command line ability to override configuration values - Command line ability to override configuration values
- Add option to allow preloading the entire ROM before running - Add option to allow preloading the entire ROM before running
- GB: Video/audio channel enabling/disabling
- Add option to lock video to integer scaling
Bugfixes: Bugfixes:
- LR35902: Fix core never exiting with certain event patterns - LR35902: Fix core never exiting with certain event patterns
- GB Timer: Improve DIV reset behavior - GB Timer: Improve DIV reset behavior
@ -73,6 +75,11 @@ Bugfixes:
- GB: Fix flickering when screen is strobed quickly - GB: Fix flickering when screen is strobed quickly
- FFmpeg: Fix overflow and general issues with audio encoding - FFmpeg: Fix overflow and general issues with audio encoding
- Qt: Fix crash when changing audio settings after a game is closed - Qt: Fix crash when changing audio settings after a game is closed
- GBA BIOS: Fix ArcTan sign in HLE BIOS
- GBA BIOS: Fix ArcTan2 sign in HLE BIOS (fixes mgba.io/i/689)
- GBA Video: Don't update background scanline params in mode 0 (fixes mgba.io/i/377)
- Qt: Ensure CLI backend is attached when submitting commands (fixes mgba.io/i/662)
- Core: Fix crash with rewind if savestates shrink
Misc: Misc:
- SDL: Remove scancode key input - SDL: Remove scancode key input
- GBA Video: Clean up unused timers - GBA Video: Clean up unused timers
@ -126,6 +133,7 @@ Misc:
- Qt: Add .gb/.gbc files to the extension list in Info.plist - Qt: Add .gb/.gbc files to the extension list in Info.plist
- Feature: Make -l option explicit - Feature: Make -l option explicit
- Core: Ability to enumerate and modify video and audio channels - Core: Ability to enumerate and modify video and audio channels
- Debugger: Make attaching a backend idempotent
medusa alpha 1: (2017-04-08) medusa alpha 1: (2017-04-08)
Features: Features:

View File

@ -56,6 +56,11 @@ typedef intptr_t ssize_t;
#include <sys/time.h> #include <sys/time.h>
#endif #endif
#ifdef PSP2
// For PATH_MAX on modern toolchains
#include <sys/syslimits.h>
#endif
#ifndef SSIZE_MAX #ifndef SSIZE_MAX
#define SSIZE_MAX ((ssize_t) (SIZE_MAX >> 1)) #define SSIZE_MAX ((ssize_t) (SIZE_MAX >> 1))
#endif #endif

View File

@ -42,6 +42,7 @@ struct mCoreOptions {
int width; int width;
int height; int height;
bool lockAspectRatio; bool lockAspectRatio;
bool lockIntegerScaling;
bool resampleVideo; bool resampleVideo;
bool suspendScreensaver; bool suspendScreensaver;
char* shader; char* shader;

View File

@ -71,6 +71,10 @@ struct GBVideoRenderer {
uint8_t* vram; uint8_t* vram;
union GBOAM* oam; union GBOAM* oam;
struct mTileCache* cache; struct mTileCache* cache;
bool disableBG;
bool disableOBJ;
bool disableWIN;
}; };
DECL_BITFIELD(GBRegisterLCDC, uint8_t); DECL_BITFIELD(GBRegisterLCDC, uint8_t);

View File

@ -349,6 +349,9 @@ void mCoreConfigMap(const struct mCoreConfig* config, struct mCoreOptions* opts)
if (_lookupIntValue(config, "lockAspectRatio", &fakeBool)) { if (_lookupIntValue(config, "lockAspectRatio", &fakeBool)) {
opts->lockAspectRatio = fakeBool; opts->lockAspectRatio = fakeBool;
} }
if (_lookupIntValue(config, "lockIntegerScaling", &fakeBool)) {
opts->lockIntegerScaling = fakeBool;
}
if (_lookupIntValue(config, "resampleVideo", &fakeBool)) { if (_lookupIntValue(config, "resampleVideo", &fakeBool)) {
opts->resampleVideo = fakeBool; opts->resampleVideo = fakeBool;
} }
@ -396,6 +399,7 @@ void mCoreConfigLoadDefaults(struct mCoreConfig* config, const struct mCoreOptio
ConfigurationSetIntValue(&config->defaultsTable, 0, "volume", opts->volume); ConfigurationSetIntValue(&config->defaultsTable, 0, "volume", opts->volume);
ConfigurationSetIntValue(&config->defaultsTable, 0, "mute", opts->mute); ConfigurationSetIntValue(&config->defaultsTable, 0, "mute", opts->mute);
ConfigurationSetIntValue(&config->defaultsTable, 0, "lockAspectRatio", opts->lockAspectRatio); ConfigurationSetIntValue(&config->defaultsTable, 0, "lockAspectRatio", opts->lockAspectRatio);
ConfigurationSetIntValue(&config->defaultsTable, 0, "lockIntegerScaling", opts->lockIntegerScaling);
ConfigurationSetIntValue(&config->defaultsTable, 0, "resampleVideo", opts->resampleVideo); ConfigurationSetIntValue(&config->defaultsTable, 0, "resampleVideo", opts->resampleVideo);
ConfigurationSetIntValue(&config->defaultsTable, 0, "suspendScreensaver", opts->suspendScreensaver); ConfigurationSetIntValue(&config->defaultsTable, 0, "suspendScreensaver", opts->suspendScreensaver);
} }

View File

@ -50,6 +50,8 @@ void mCoreRewindAppend(struct mCoreRewindContext* context, struct mCore* core) {
if (size2 > size) { if (size2 > size) {
context->currentState->truncate(context->currentState, size2); context->currentState->truncate(context->currentState, size2);
size = size2; size = size2;
} else if (size > size2) {
nextState->truncate(nextState, size);
} }
void* current = context->currentState->map(context->currentState, size, MAP_READ); void* current = context->currentState->map(context->currentState, size, MAP_READ);
void* next = nextState->map(nextState, size, MAP_READ); void* next = nextState->map(nextState, size, MAP_READ);

View File

@ -751,6 +751,9 @@ void CLIDebuggerAttachSystem(struct CLIDebugger* debugger, struct CLIDebuggerSys
} }
void CLIDebuggerAttachBackend(struct CLIDebugger* debugger, struct CLIDebuggerBackend* backend) { void CLIDebuggerAttachBackend(struct CLIDebugger* debugger, struct CLIDebuggerBackend* backend) {
if (debugger->backend == backend) {
return;
}
if (debugger->backend && debugger->backend->deinit) { if (debugger->backend && debugger->backend->deinit) {
debugger->backend->deinit(debugger->backend); debugger->backend->deinit(debugger->backend);
} }

View File

@ -465,11 +465,13 @@ void DSAttachDebugger(struct DS* ds, struct mDebugger* debugger) {
} }
void DSDetachDebugger(struct DS* ds) { void DSDetachDebugger(struct DS* ds) {
ds->debugger = NULL; if (ds->debugger) {
ARMHotplugDetach(ds->ds7.cpu, CPU_COMPONENT_DEBUGGER); ARMHotplugDetach(ds->ds7.cpu, CPU_COMPONENT_DEBUGGER);
ARMHotplugDetach(ds->ds9.cpu, CPU_COMPONENT_DEBUGGER); ARMHotplugDetach(ds->ds9.cpu, CPU_COMPONENT_DEBUGGER);
}
ds->ds7.cpu->components[CPU_COMPONENT_DEBUGGER] = NULL; ds->ds7.cpu->components[CPU_COMPONENT_DEBUGGER] = NULL;
ds->ds9.cpu->components[CPU_COMPONENT_DEBUGGER] = NULL; ds->ds9.cpu->components[CPU_COMPONENT_DEBUGGER] = NULL;
ds->debugger = NULL;
} }
bool DSLoadROM(struct DS* ds, struct VFile* vf) { bool DSLoadROM(struct DS* ds, struct VFile* vf) {

View File

@ -640,8 +640,8 @@ static void _sample(struct mTiming* timing, void* user, uint32_t cyclesLate) {
audio->lastRight = sampleRight; audio->lastRight = sampleRight;
audio->clock += audio->sampleInterval; audio->clock += audio->sampleInterval;
if (audio->clock >= CLOCKS_PER_BLIP_FRAME) { if (audio->clock >= CLOCKS_PER_BLIP_FRAME) {
blip_end_frame(audio->left, audio->clock); blip_end_frame(audio->left, CLOCKS_PER_BLIP_FRAME);
blip_end_frame(audio->right, audio->clock); blip_end_frame(audio->right, CLOCKS_PER_BLIP_FRAME);
audio->clock -= CLOCKS_PER_BLIP_FRAME; audio->clock -= CLOCKS_PER_BLIP_FRAME;
} }
} }

View File

@ -629,7 +629,19 @@ static size_t _GBCoreListAudioChannels(const struct mCore* core, const struct mC
static void _GBCoreEnableVideoLayer(struct mCore* core, size_t id, bool enable) { static void _GBCoreEnableVideoLayer(struct mCore* core, size_t id, bool enable) {
struct GB* gb = core->board; struct GB* gb = core->board;
// TODO switch (id) {
case 0:
gb->video.renderer->disableBG = !enable;
break;
case 1:
gb->video.renderer->disableOBJ = !enable;
break;
case 2:
gb->video.renderer->disableWIN = !enable;
break;
default:
break;
}
} }
static void _GBCoreEnableAudioChannel(struct mCore* core, size_t id, bool enable) { static void _GBCoreEnableAudioChannel(struct mCore* core, size_t id, bool enable) {

View File

@ -49,6 +49,10 @@ void GBVideoSoftwareRendererCreate(struct GBVideoSoftwareRenderer* renderer) {
renderer->d.getPixels = GBVideoSoftwareRendererGetPixels; renderer->d.getPixels = GBVideoSoftwareRendererGetPixels;
renderer->d.putPixels = GBVideoSoftwareRendererPutPixels; renderer->d.putPixels = GBVideoSoftwareRendererPutPixels;
renderer->d.disableBG = false;
renderer->d.disableOBJ = false;
renderer->d.disableWIN = false;
renderer->temporaryBuffer = 0; renderer->temporaryBuffer = 0;
} }
@ -128,9 +132,12 @@ static void GBVideoSoftwareRendererDrawRange(struct GBVideoRenderer* renderer, i
if (GBRegisterLCDCIsTileMap(softwareRenderer->lcdc)) { if (GBRegisterLCDCIsTileMap(softwareRenderer->lcdc)) {
maps += GB_SIZE_MAP; maps += GB_SIZE_MAP;
} }
if (softwareRenderer->d.disableBG) {
memset(&softwareRenderer->row[startX], 0, endX - startX);
}
if (GBRegisterLCDCIsBgEnable(softwareRenderer->lcdc) || softwareRenderer->model >= GB_MODEL_CGB) { if (GBRegisterLCDCIsBgEnable(softwareRenderer->lcdc) || softwareRenderer->model >= GB_MODEL_CGB) {
if (GBRegisterLCDCIsWindow(softwareRenderer->lcdc) && softwareRenderer->wy <= y && endX >= softwareRenderer->wx - 7) { if (GBRegisterLCDCIsWindow(softwareRenderer->lcdc) && softwareRenderer->wy <= y && endX >= softwareRenderer->wx - 7) {
if (softwareRenderer->wx - 7 > 0) { if (softwareRenderer->wx - 7 > 0 && !softwareRenderer->d.disableBG) {
GBVideoSoftwareRendererDrawBackground(softwareRenderer, maps, startX, softwareRenderer->wx - 7, softwareRenderer->scx, softwareRenderer->scy + y); GBVideoSoftwareRendererDrawBackground(softwareRenderer, maps, startX, softwareRenderer->wx - 7, softwareRenderer->scx, softwareRenderer->scy + y);
} }
@ -138,15 +145,17 @@ static void GBVideoSoftwareRendererDrawRange(struct GBVideoRenderer* renderer, i
if (GBRegisterLCDCIsWindowTileMap(softwareRenderer->lcdc)) { if (GBRegisterLCDCIsWindowTileMap(softwareRenderer->lcdc)) {
maps += GB_SIZE_MAP; maps += GB_SIZE_MAP;
} }
if (!softwareRenderer->d.disableWIN) {
GBVideoSoftwareRendererDrawBackground(softwareRenderer, maps, softwareRenderer->wx - 7, endX, 7 - softwareRenderer->wx, softwareRenderer->currentWy); GBVideoSoftwareRendererDrawBackground(softwareRenderer, maps, softwareRenderer->wx - 7, endX, 7 - softwareRenderer->wx, softwareRenderer->currentWy);
} else { }
} else if (!softwareRenderer->d.disableBG) {
GBVideoSoftwareRendererDrawBackground(softwareRenderer, maps, startX, endX, softwareRenderer->scx, softwareRenderer->scy + y); GBVideoSoftwareRendererDrawBackground(softwareRenderer, maps, startX, endX, softwareRenderer->scx, softwareRenderer->scy + y);
} }
} else { } else if (!softwareRenderer->d.disableBG) {
memset(&softwareRenderer->row[startX], 0, endX - startX); memset(&softwareRenderer->row[startX], 0, endX - startX);
} }
if (GBRegisterLCDCIsObjEnable(softwareRenderer->lcdc)) { if (GBRegisterLCDCIsObjEnable(softwareRenderer->lcdc) && !softwareRenderer->d.disableOBJ) {
size_t i; size_t i;
for (i = 0; i < oamMax; ++i) { for (i = 0; i < oamMax; ++i) {
GBVideoSoftwareRendererDrawObj(softwareRenderer, &obj[i], startX, endX, y); GBVideoSoftwareRendererDrawObj(softwareRenderer, &obj[i], startX, endX, y);

View File

@ -297,8 +297,8 @@ static void _sample(struct mTiming* timing, void* user, uint32_t cyclesLate) {
audio->lastRight = sampleRight; audio->lastRight = sampleRight;
audio->clock += audio->sampleInterval; audio->clock += audio->sampleInterval;
if (audio->clock >= CLOCKS_PER_FRAME) { if (audio->clock >= CLOCKS_PER_FRAME) {
blip_end_frame(audio->psg.left, audio->clock); blip_end_frame(audio->psg.left, CLOCKS_PER_FRAME);
blip_end_frame(audio->psg.right, audio->clock); blip_end_frame(audio->psg.right, CLOCKS_PER_FRAME);
audio->clock -= CLOCKS_PER_FRAME; audio->clock -= CLOCKS_PER_FRAME;
} }
} }

View File

@ -271,7 +271,7 @@ static void _Div(struct GBA* gba, int32_t num, int32_t denom) {
} }
} }
static int16_t _ArcTan(int16_t i) { static int16_t _ArcTan(int32_t i, int32_t* r1, int32_t* r3) {
int32_t a = -((i * i) >> 14); int32_t a = -((i * i) >> 14);
int32_t b = ((0xA9 * a) >> 14) + 0x390; int32_t b = ((0xA9 * a) >> 14) + 0x390;
b = ((b * a) >> 14) + 0x91C; b = ((b * a) >> 14) + 0x91C;
@ -280,10 +280,16 @@ static int16_t _ArcTan(int16_t i) {
b = ((b * a) >> 14) + 0x2081; b = ((b * a) >> 14) + 0x2081;
b = ((b * a) >> 14) + 0x3651; b = ((b * a) >> 14) + 0x3651;
b = ((b * a) >> 14) + 0xA2F9; b = ((b * a) >> 14) + 0xA2F9;
if (r1) {
*r1 = a;
}
if (r3) {
*r3 = b;
}
return (i * b) >> 16; return (i * b) >> 16;
} }
static int16_t _ArcTan2(int16_t x, int16_t y) { static int16_t _ArcTan2(int32_t x, int32_t y, int32_t* r1) {
if (!y) { if (!y) {
if (x >= 0) { if (x >= 0) {
return 0; return 0;
@ -299,21 +305,21 @@ static int16_t _ArcTan2(int16_t x, int16_t y) {
if (y >= 0) { if (y >= 0) {
if (x >= 0) { if (x >= 0) {
if (x >= y) { if (x >= y) {
return _ArcTan((y << 14)/ x); return _ArcTan((y << 14) / x, r1, NULL);
} }
} else if (-x >= y) { } else if (-x >= y) {
return _ArcTan((y << 14) / x) + 0x8000; return _ArcTan((y << 14) / x, r1, NULL) + 0x8000;
} }
return 0x4000 - _ArcTan((x << 14) / y); return 0x4000 - _ArcTan((x << 14) / y, r1, NULL);
} else { } else {
if (x <= 0) { if (x <= 0) {
if (-x > -y) { if (-x > -y) {
return _ArcTan((y << 14) / x) + 0x8000; return _ArcTan((y << 14) / x, r1, NULL) + 0x8000;
} }
} else if (x >= -y) { } else if (x >= -y) {
return _ArcTan((y << 14) / x) + 0x10000; return _ArcTan((y << 14) / x, r1, NULL) + 0x10000;
} }
return 0xC000 - _ArcTan((x << 14) / y); return 0xC000 - _ArcTan((x << 14) / y, r1, NULL);
} }
} }
@ -356,10 +362,11 @@ void GBASwi16(struct ARMCore* cpu, int immediate) {
cpu->gprs[0] = sqrt((uint32_t) cpu->gprs[0]); cpu->gprs[0] = sqrt((uint32_t) cpu->gprs[0]);
break; break;
case 0x9: case 0x9:
cpu->gprs[0] = (uint16_t) _ArcTan(cpu->gprs[0]); cpu->gprs[0] = _ArcTan(cpu->gprs[0], &cpu->gprs[1], &cpu->gprs[3]);
break; break;
case 0xA: case 0xA:
cpu->gprs[0] = (uint16_t) _ArcTan2(cpu->gprs[0], cpu->gprs[1]); cpu->gprs[0] = (uint16_t) _ArcTan2(cpu->gprs[0], cpu->gprs[1], &cpu->gprs[1]);
cpu->gprs[3] = 0x170;
break; break;
case 0xB: case 0xB:
case 0xC: case 0xC:

View File

@ -268,6 +268,7 @@ static void GBAProcessEvents(struct ARMCore* cpu) {
} }
} }
#ifdef USE_DEBUGGERS
void GBAAttachDebugger(struct GBA* gba, struct mDebugger* debugger) { void GBAAttachDebugger(struct GBA* gba, struct mDebugger* debugger) {
gba->debugger = (struct ARMDebugger*) debugger->platform; gba->debugger = (struct ARMDebugger*) debugger->platform;
gba->debugger->setSoftwareBreakpoint = _setSoftwareBreakpoint; gba->debugger->setSoftwareBreakpoint = _setSoftwareBreakpoint;
@ -277,10 +278,13 @@ void GBAAttachDebugger(struct GBA* gba, struct mDebugger* debugger) {
} }
void GBADetachDebugger(struct GBA* gba) { void GBADetachDebugger(struct GBA* gba) {
gba->debugger = 0; if (gba->debugger) {
ARMHotplugDetach(gba->cpu, CPU_COMPONENT_DEBUGGER); ARMHotplugDetach(gba->cpu, CPU_COMPONENT_DEBUGGER);
gba->cpu->components[CPU_COMPONENT_DEBUGGER] = 0; }
gba->cpu->components[CPU_COMPONENT_DEBUGGER] = NULL;
gba->debugger = NULL;
} }
#endif
bool GBALoadMB(struct GBA* gba, struct VFile* vf) { bool GBALoadMB(struct GBA* gba, struct VFile* vf) {
GBAUnloadROM(gba); GBAUnloadROM(gba);

View File

@ -581,10 +581,12 @@ static void GBAVideoSoftwareRendererDrawScanline(struct GBAVideoRenderer* render
} }
} }
} }
if (GBARegisterDISPCNTGetMode(softwareRenderer->dispcnt) != 0) {
softwareRenderer->bg[2].sx += softwareRenderer->bg[2].dmx; softwareRenderer->bg[2].sx += softwareRenderer->bg[2].dmx;
softwareRenderer->bg[2].sy += softwareRenderer->bg[2].dmy; softwareRenderer->bg[2].sy += softwareRenderer->bg[2].dmy;
softwareRenderer->bg[3].sx += softwareRenderer->bg[3].dmx; softwareRenderer->bg[3].sx += softwareRenderer->bg[3].dmx;
softwareRenderer->bg[3].sy += softwareRenderer->bg[3].dmy; softwareRenderer->bg[3].sy += softwareRenderer->bg[3].dmy;
}
GBAVideoSoftwareRendererPostprocessBuffer(softwareRenderer); GBAVideoSoftwareRendererPostprocessBuffer(softwareRenderer);

View File

@ -67,6 +67,10 @@ static void mGLContextResized(struct VideoBackend* v, unsigned w, unsigned h) {
drawH = w * v->height / v->width; drawH = w * v->height / v->width;
} }
} }
if (v->lockIntegerScaling) {
drawW -= drawW % v->width;
drawH -= drawH % v->height;
}
glMatrixMode(GL_MODELVIEW); glMatrixMode(GL_MODELVIEW);
glLoadIdentity(); glLoadIdentity();
glClearColor(0, 0, 0, 0); glClearColor(0, 0, 0, 0);

View File

@ -171,6 +171,10 @@ static void mGLES2ContextResized(struct VideoBackend* v, unsigned w, unsigned h)
drawH = w * v->height / v->width; drawH = w * v->height / v->width;
} }
} }
if (v->lockIntegerScaling) {
drawW -= drawW % v->width;
drawH -= drawH % v->height;
}
glViewport(0, 0, v->width, v->height); glViewport(0, 0, v->width, v->height);
glClearColor(0.f, 0.f, 0.f, 1.f); glClearColor(0.f, 0.f, 0.f, 1.f);
glClear(GL_COLOR_BUFFER_BIT); glClear(GL_COLOR_BUFFER_BIT);

View File

@ -16,7 +16,6 @@
#include <psp2/display.h> #include <psp2/display.h>
#include <psp2/kernel/processmgr.h> #include <psp2/kernel/processmgr.h>
#include <psp2/kernel/threadmgr.h> #include <psp2/kernel/threadmgr.h>
#include <psp2/moduleinfo.h>
#include <psp2/power.h> #include <psp2/power.h>
#include <psp2/sysmodule.h> #include <psp2/sysmodule.h>
#include <psp2/touch.h> #include <psp2/touch.h>
@ -164,6 +163,7 @@ int main() {
mPSP2MapKey(&runner.params.keyMap, SCE_CTRL_RIGHT, GUI_INPUT_RIGHT); mPSP2MapKey(&runner.params.keyMap, SCE_CTRL_RIGHT, GUI_INPUT_RIGHT);
mPSP2MapKey(&runner.params.keyMap, SCE_CTRL_SQUARE, mGUI_INPUT_SCREEN_MODE); mPSP2MapKey(&runner.params.keyMap, SCE_CTRL_SQUARE, mGUI_INPUT_SCREEN_MODE);
scePowerSetArmClockFrequency(444);
mGUIRunloop(&runner); mGUIRunloop(&runner);
vita2d_fini(); vita2d_fini();

View File

@ -20,6 +20,7 @@
#include <mgba-util/memory.h> #include <mgba-util/memory.h>
#include <mgba-util/circle-buffer.h> #include <mgba-util/circle-buffer.h>
#include <mgba-util/math.h>
#include <mgba-util/ring-fifo.h> #include <mgba-util/ring-fifo.h>
#include <mgba-util/threading.h> #include <mgba-util/threading.h>
#include <mgba-util/vfs.h> #include <mgba-util/vfs.h>
@ -31,7 +32,6 @@
#include <psp2/gxm.h> #include <psp2/gxm.h>
#include <psp2/kernel/sysmem.h> #include <psp2/kernel/sysmem.h>
#include <psp2/motion.h> #include <psp2/motion.h>
#include <psp2/power.h>
#include <vita2d.h> #include <vita2d.h>
@ -63,8 +63,8 @@ bool frameLimiter = true;
extern const uint8_t _binary_backdrop_png_start[]; extern const uint8_t _binary_backdrop_png_start[];
static vita2d_texture* backdrop = 0; static vita2d_texture* backdrop = 0;
#define PSP2_SAMPLES 128 #define PSP2_SAMPLES 256
#define PSP2_AUDIO_BUFFER_SIZE (PSP2_SAMPLES * 40) #define PSP2_AUDIO_BUFFER_SIZE (PSP2_SAMPLES * 20)
static struct mPSP2AudioContext { static struct mPSP2AudioContext {
struct RingFIFO buffer; struct RingFIFO buffer;
@ -176,7 +176,6 @@ void mPSP2Setup(struct mGUIRunner* runner) {
mCoreConfigSetDefaultIntValue(&runner->config, "threadedVideo", 1); mCoreConfigSetDefaultIntValue(&runner->config, "threadedVideo", 1);
mCoreLoadForeignConfig(runner->core, &runner->config); mCoreLoadForeignConfig(runner->core, &runner->config);
scePowerSetArmClockFrequency(333);
mPSP2MapKey(&runner->core->inputMap, SCE_CTRL_CROSS, GBA_KEY_A); mPSP2MapKey(&runner->core->inputMap, SCE_CTRL_CROSS, GBA_KEY_A);
mPSP2MapKey(&runner->core->inputMap, SCE_CTRL_CIRCLE, GBA_KEY_B); mPSP2MapKey(&runner->core->inputMap, SCE_CTRL_CIRCLE, GBA_KEY_B);
mPSP2MapKey(&runner->core->inputMap, SCE_CTRL_START, GBA_KEY_START); mPSP2MapKey(&runner->core->inputMap, SCE_CTRL_START, GBA_KEY_START);
@ -193,8 +192,10 @@ void mPSP2Setup(struct mGUIRunner* runner) {
desc = (struct mInputAxis) { GBA_KEY_RIGHT, GBA_KEY_LEFT, 192, 64 }; desc = (struct mInputAxis) { GBA_KEY_RIGHT, GBA_KEY_LEFT, 192, 64 };
mInputBindAxis(&runner->core->inputMap, PSP2_INPUT, 1, &desc); mInputBindAxis(&runner->core->inputMap, PSP2_INPUT, 1, &desc);
tex = vita2d_create_empty_texture_format(256, 256, SCE_GXM_TEXTURE_FORMAT_X8U8U8U8_1BGR); unsigned width, height;
screenshot = vita2d_create_empty_texture_format(256, 256, SCE_GXM_TEXTURE_FORMAT_X8U8U8U8_1BGR); runner->core->desiredVideoDimensions(runner->core, &width, &height);
tex = vita2d_create_empty_texture_format(256, toPow2(height), SCE_GXM_TEXTURE_FORMAT_X8U8U8U8_1BGR);
screenshot = vita2d_create_empty_texture_format(256, toPow2(height), SCE_GXM_TEXTURE_FORMAT_X8U8U8U8_1BGR);
outputBuffer = vita2d_texture_get_datap(tex); outputBuffer = vita2d_texture_get_datap(tex);
runner->core->setVideoBuffer(runner->core, outputBuffer, 256); runner->core->setVideoBuffer(runner->core, outputBuffer, 256);
@ -219,7 +220,6 @@ void mPSP2Setup(struct mGUIRunner* runner) {
} }
void mPSP2LoadROM(struct mGUIRunner* runner) { void mPSP2LoadROM(struct mGUIRunner* runner) {
scePowerSetArmClockFrequency(444);
float rate = 60.0f / 1.001f; float rate = 60.0f / 1.001f;
sceDisplayGetRefreshRate(&rate); sceDisplayGetRefreshRate(&rate);
double ratio = GBAAudioCalculateRatio(1, rate, 1); double ratio = GBAAudioCalculateRatio(1, rate, 1);
@ -262,7 +262,6 @@ void mPSP2PrepareForFrame(struct mGUIRunner* runner) {
blip_clear(runner->core->getAudioChannel(runner->core, 1)); blip_clear(runner->core->getAudioChannel(runner->core, 1));
break; break;
} }
sceKernelDelayThread(400);
} }
blip_read_samples(runner->core->getAudioChannel(runner->core, 0), &samples[0].left, PSP2_SAMPLES, true); blip_read_samples(runner->core->getAudioChannel(runner->core, 0), &samples[0].left, PSP2_SAMPLES, true);
blip_read_samples(runner->core->getAudioChannel(runner->core, 1), &samples[0].right, PSP2_SAMPLES, true); blip_read_samples(runner->core->getAudioChannel(runner->core, 1), &samples[0].right, PSP2_SAMPLES, true);
@ -297,7 +296,6 @@ void mPSP2UnloadROM(struct mGUIRunner* runner) {
default: default:
break; break;
} }
scePowerSetArmClockFrequency(333);
} }
void mPSP2Paused(struct mGUIRunner* runner) { void mPSP2Paused(struct mGUIRunner* runner) {

View File

@ -2,8 +2,13 @@ find_program(PYTHON python)
configure_file(${CMAKE_CURRENT_SOURCE_DIR}/setup.py.in ${CMAKE_CURRENT_BINARY_DIR}/setup.py) configure_file(${CMAKE_CURRENT_SOURCE_DIR}/setup.py.in ${CMAKE_CURRENT_BINARY_DIR}/setup.py)
get_property(INCLUDE_DIRECTORIES DIRECTORY PROPERTY INCLUDE_DIRECTORIES)
set(INCLUDE_FLAGS)
foreach(DIR IN LISTS INCLUDE_DIRECTORIES)
list(APPEND INCLUDE_FLAGS "-I${DIR}")
endforeach()
add_custom_command(OUTPUT build/lib/${BINARY_NAME}/__init__.py add_custom_command(OUTPUT build/lib/${BINARY_NAME}/__init__.py
COMMAND BINDIR=${CMAKE_CURRENT_BINARY_DIR}/.. ${PYTHON} ${CMAKE_CURRENT_BINARY_DIR}/setup.py build --build-base ${CMAKE_CURRENT_BINARY_DIR} COMMAND BINDIR=${CMAKE_CURRENT_BINARY_DIR}/.. CPPFLAGS="${INCLUDE_FLAGS}" ${PYTHON} ${CMAKE_CURRENT_BINARY_DIR}/setup.py build --build-base ${CMAKE_CURRENT_BINARY_DIR}
WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR} WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
DEPENDS ${BINARY_NAME} DEPENDS ${BINARY_NAME}
DEPENDS ${CMAKE_CURRENT_BINARY_DIR}/setup.py DEPENDS ${CMAKE_CURRENT_BINARY_DIR}/setup.py

View File

@ -38,8 +38,16 @@ void DebuggerConsoleController::enterLine(const QString& line) {
m_cond.wakeOne(); m_cond.wakeOne();
} }
void DebuggerConsoleController::detach() {
m_lines.append(QString());
m_cond.wakeOne();
DebuggerController::detach();
}
void DebuggerConsoleController::attachInternal() { void DebuggerConsoleController::attachInternal() {
m_history.clear();
mCore* core = m_gameController->thread()->core; mCore* core = m_gameController->thread()->core;
CLIDebuggerAttachBackend(&m_cliDebugger, &m_backend.d);
CLIDebuggerAttachSystem(&m_cliDebugger, core->cliDebuggerSystem(core)); CLIDebuggerAttachSystem(&m_cliDebugger, core->cliDebuggerSystem(core));
} }
@ -60,6 +68,8 @@ void DebuggerConsoleController::init(struct CLIDebuggerBackend* be) {
void DebuggerConsoleController::deinit(struct CLIDebuggerBackend* be) { void DebuggerConsoleController::deinit(struct CLIDebuggerBackend* be) {
Backend* consoleBe = reinterpret_cast<Backend*>(be); Backend* consoleBe = reinterpret_cast<Backend*>(be);
DebuggerConsoleController* self = consoleBe->self; DebuggerConsoleController* self = consoleBe->self;
self->m_lines.append(QString());
self->m_cond.wakeOne();
} }
const char* DebuggerConsoleController::readLine(struct CLIDebuggerBackend* be, size_t* len) { const char* DebuggerConsoleController::readLine(struct CLIDebuggerBackend* be, size_t* len) {
@ -71,6 +81,9 @@ const char* DebuggerConsoleController::readLine(struct CLIDebuggerBackend* be, s
self->m_cond.wait(&self->m_mutex); self->m_cond.wait(&self->m_mutex);
} }
self->m_last = self->m_lines.takeFirst().toUtf8(); self->m_last = self->m_lines.takeFirst().toUtf8();
if (self->m_last.isEmpty()) {
self->m_last = "\n";
}
*len = self->m_last.size(); *len = self->m_last.size();
return self->m_last.constData(); return self->m_last.constData();
@ -87,6 +100,9 @@ const char* DebuggerConsoleController::historyLast(struct CLIDebuggerBackend* be
DebuggerConsoleController* self = consoleBe->self; DebuggerConsoleController* self = consoleBe->self;
GameController::Interrupter interrupter(self->m_gameController, true); GameController::Interrupter interrupter(self->m_gameController, true);
QMutexLocker lock(&self->m_mutex); QMutexLocker lock(&self->m_mutex);
if (self->m_history.isEmpty()) {
return "\n";
}
self->m_last = self->m_history.last().toUtf8(); self->m_last = self->m_history.last().toUtf8();
return self->m_last.constData(); return self->m_last.constData();
} }

View File

@ -30,6 +30,7 @@ signals:
public slots: public slots:
void enterLine(const QString&); void enterLine(const QString&);
virtual void detach() override;
protected: protected:
virtual void attachInternal() override; virtual void attachInternal() override;

View File

@ -54,6 +54,7 @@ Display* Display::create(QWidget* parent) {
Display::Display(QWidget* parent) Display::Display(QWidget* parent)
: QWidget(parent) : QWidget(parent)
, m_lockAspectRatio(false) , m_lockAspectRatio(false)
, m_lockIntegerScaling(false)
, m_filter(false) , m_filter(false)
{ {
setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding); setSizePolicy(QSizePolicy::MinimumExpanding, QSizePolicy::MinimumExpanding);
@ -78,6 +79,10 @@ QSize Display::viewportSize() {
ds.setHeight(s.width() * m_coreHeight / m_coreWidth); ds.setHeight(s.width() * m_coreHeight / m_coreWidth);
} }
} }
if (isIntegerScalingLocked()) {
ds.setWidth(ds.width() - ds.width() % m_coreWidth);
ds.setHeight(ds.height() - ds.height() % m_coreHeight);
}
return ds; return ds;
} }
@ -90,6 +95,10 @@ void Display::lockAspectRatio(bool lock) {
m_messagePainter.resize(size(), m_lockAspectRatio, devicePixelRatio()); m_messagePainter.resize(size(), m_lockAspectRatio, devicePixelRatio());
} }
void Display::lockIntegerScaling(bool lock) {
m_lockIntegerScaling = lock;
}
void Display::filter(bool filter) { void Display::filter(bool filter) {
m_filter = filter; m_filter = filter;
} }

View File

@ -38,6 +38,7 @@ public:
static void setDriver(Driver driver) { s_driver = driver; } static void setDriver(Driver driver) { s_driver = driver; }
bool isAspectRatioLocked() const { return m_lockAspectRatio; } bool isAspectRatioLocked() const { return m_lockAspectRatio; }
bool isIntegerScalingLocked() const { return m_lockIntegerScaling; }
bool isFiltered() const { return m_filter; } bool isFiltered() const { return m_filter; }
virtual bool isDrawing() const = 0; virtual bool isDrawing() const = 0;
@ -57,6 +58,7 @@ public slots:
virtual void unpauseDrawing() = 0; virtual void unpauseDrawing() = 0;
virtual void forceDraw() = 0; virtual void forceDraw() = 0;
virtual void lockAspectRatio(bool lock); virtual void lockAspectRatio(bool lock);
virtual void lockIntegerScaling(bool lock);
virtual void filter(bool filter); virtual void filter(bool filter);
virtual void framePosted(const uint32_t*) = 0; virtual void framePosted(const uint32_t*) = 0;
virtual void setShaders(struct VDir*) = 0; virtual void setShaders(struct VDir*) = 0;
@ -78,6 +80,7 @@ private:
MessagePainter m_messagePainter; MessagePainter m_messagePainter;
bool m_lockAspectRatio; bool m_lockAspectRatio;
bool m_lockIntegerScaling;
bool m_filter; bool m_filter;
QTimer m_mouseTimer; QTimer m_mouseTimer;
int m_coreWidth; int m_coreWidth;

View File

@ -73,6 +73,8 @@ void DisplayGL::startDrawing(mCoreThread* thread) {
mCoreSyncSetVideoSync(&m_context->sync, false); mCoreSyncSetVideoSync(&m_context->sync, false);
lockAspectRatio(isAspectRatioLocked()); lockAspectRatio(isAspectRatioLocked());
lockIntegerScaling(isIntegerScalingLocked());
unsigned width, height; unsigned width, height;
thread->core->desiredVideoDimensions(thread->core, &width, &height); thread->core->desiredVideoDimensions(thread->core, &width, &height);
setSystemDimensions(width, height); setSystemDimensions(width, height);
@ -140,6 +142,13 @@ void DisplayGL::lockAspectRatio(bool lock) {
} }
} }
void DisplayGL::lockIntegerScaling(bool lock) {
Display::lockIntegerScaling(lock);
if (m_drawThread) {
QMetaObject::invokeMethod(m_painter, "lockIntegerScaling", Q_ARG(bool, lock));
}
}
void DisplayGL::filter(bool filter) { void DisplayGL::filter(bool filter) {
Display::filter(filter); Display::filter(filter);
if (m_drawThread) { if (m_drawThread) {
@ -294,6 +303,13 @@ void PainterGL::lockAspectRatio(bool lock) {
} }
} }
void PainterGL::lockIntegerScaling(bool lock) {
m_backend->lockIntegerScaling = lock;
if (m_started && !m_active) {
forceDraw();
}
}
void PainterGL::filter(bool filter) { void PainterGL::filter(bool filter) {
m_backend->filter = filter; m_backend->filter = filter;
if (m_started && !m_active) { if (m_started && !m_active) {

View File

@ -55,6 +55,7 @@ public slots:
void unpauseDrawing() override; void unpauseDrawing() override;
void forceDraw() override; void forceDraw() override;
void lockAspectRatio(bool lock) override; void lockAspectRatio(bool lock) override;
void lockIntegerScaling(bool lock) override;
void filter(bool filter) override; void filter(bool filter) override;
void framePosted(const uint32_t*) override; void framePosted(const uint32_t*) override;
void setShaders(struct VDir*) override; void setShaders(struct VDir*) override;
@ -96,6 +97,7 @@ public slots:
void unpause(); void unpause();
void resize(const QSize& size); void resize(const QSize& size);
void lockAspectRatio(bool lock); void lockAspectRatio(bool lock);
void lockIntegerScaling(bool lock);
void filter(bool filter); void filter(bool filter);
void setShaders(struct VDir*); void setShaders(struct VDir*);

View File

@ -31,6 +31,11 @@ void DisplayQt::lockAspectRatio(bool lock) {
update(); update();
} }
void DisplayQt::lockIntegerScaling(bool lock) {
Display::lockIntegerScaling(lock);
update();
}
void DisplayQt::filter(bool filter) { void DisplayQt::filter(bool filter) {
Display::filter(filter); Display::filter(filter);
update(); update();

View File

@ -30,6 +30,7 @@ public slots:
void unpauseDrawing() override { m_isDrawing = true; } void unpauseDrawing() override { m_isDrawing = true; }
void forceDraw() override { update(); } void forceDraw() override { update(); }
void lockAspectRatio(bool lock) override; void lockAspectRatio(bool lock) override;
void lockIntegerScaling(bool lock) override;
void filter(bool filter) override; void filter(bool filter) override;
void framePosted(const uint32_t*) override; void framePosted(const uint32_t*) override;
void setShaders(struct VDir*) override {} void setShaders(struct VDir*) override {}

View File

@ -623,6 +623,7 @@ void GameController::closeGame() {
return; return;
} }
setDebugger(nullptr);
if (mCoreThreadIsPaused(&m_threadContext)) { if (mCoreThreadIsPaused(&m_threadContext)) {
mCoreThreadUnpause(&m_threadContext); mCoreThreadUnpause(&m_threadContext);
} }

View File

@ -177,6 +177,7 @@ void SettingsView::updateConfig() {
saveSetting("frameskip", m_ui.frameskip); saveSetting("frameskip", m_ui.frameskip);
saveSetting("fpsTarget", m_ui.fpsTarget); saveSetting("fpsTarget", m_ui.fpsTarget);
saveSetting("lockAspectRatio", m_ui.lockAspectRatio); saveSetting("lockAspectRatio", m_ui.lockAspectRatio);
saveSetting("lockIntegerScaling", m_ui.lockIntegerScaling);
saveSetting("volume", m_ui.volume); saveSetting("volume", m_ui.volume);
saveSetting("mute", m_ui.mute); saveSetting("mute", m_ui.mute);
saveSetting("rewindEnable", m_ui.rewind); saveSetting("rewindEnable", m_ui.rewind);
@ -260,6 +261,7 @@ void SettingsView::reloadConfig() {
loadSetting("frameskip", m_ui.frameskip); loadSetting("frameskip", m_ui.frameskip);
loadSetting("fpsTarget", m_ui.fpsTarget); loadSetting("fpsTarget", m_ui.fpsTarget);
loadSetting("lockAspectRatio", m_ui.lockAspectRatio); loadSetting("lockAspectRatio", m_ui.lockAspectRatio);
loadSetting("lockIntegerScaling", m_ui.lockIntegerScaling);
loadSetting("volume", m_ui.volume); loadSetting("volume", m_ui.volume);
loadSetting("mute", m_ui.mute); loadSetting("mute", m_ui.mute);
loadSetting("rewindEnable", m_ui.rewind); loadSetting("rewindEnable", m_ui.rewind);

View File

@ -7,7 +7,7 @@
<x>0</x> <x>0</x>
<y>0</y> <y>0</y>
<width>650</width> <width>650</width>
<height>450</height> <height>454</height>
</rect> </rect>
</property> </property>
<property name="sizePolicy"> <property name="sizePolicy">
@ -115,7 +115,7 @@
<property name="editable"> <property name="editable">
<bool>true</bool> <bool>true</bool>
</property> </property>
<property name="currentText" stdset="0"> <property name="currentText">
<string>1536</string> <string>1536</string>
</property> </property>
<property name="currentIndex"> <property name="currentIndex">
@ -181,7 +181,7 @@
<property name="editable"> <property name="editable">
<bool>true</bool> <bool>true</bool>
</property> </property>
<property name="currentText" stdset="0"> <property name="currentText">
<string>44100</string> <string>44100</string>
</property> </property>
<property name="currentIndex"> <property name="currentIndex">
@ -380,13 +380,20 @@
</property> </property>
</widget> </widget>
</item> </item>
<item row="11" column="1"> <item row="12" column="1">
<widget class="QCheckBox" name="resampleVideo"> <widget class="QCheckBox" name="resampleVideo">
<property name="text"> <property name="text">
<string>Bilinear filtering</string> <string>Bilinear filtering</string>
</property> </property>
</widget> </widget>
</item> </item>
<item row="11" column="1">
<widget class="QCheckBox" name="lockIntegerScaling">
<property name="text">
<string>Force integer scaling</string>
</property>
</widget>
</item>
</layout> </layout>
</widget> </widget>
<widget class="QWidget" name="interface_2"> <widget class="QWidget" name="interface_2">

View File

@ -493,6 +493,7 @@ void Window::selectPatch() {
void Window::openView(QWidget* widget) { void Window::openView(QWidget* widget) {
connect(this, SIGNAL(shutdown()), widget, SLOT(close())); connect(this, SIGNAL(shutdown()), widget, SLOT(close()));
connect(m_controller, SIGNAL(gameStopped(mCoreThread*)), widget, SLOT(close()));
widget->setAttribute(Qt::WA_DeleteOnClose); widget->setAttribute(Qt::WA_DeleteOnClose);
widget->show(); widget->show();
} }
@ -811,6 +812,39 @@ void Window::gameStarted(mCoreThread* context, const QString& fname) {
m_hitUnimplementedBiosCall = false; m_hitUnimplementedBiosCall = false;
m_fpsTimer.start(); m_fpsTimer.start();
m_focusCheck.start(); m_focusCheck.start();
m_controller->threadInterrupt();
if (m_controller->isLoaded()) {
mCore* core = m_controller->thread()->core;
const mCoreChannelInfo* videoLayers;
const mCoreChannelInfo* audioChannels;
size_t nVideo = core->listVideoLayers(core, &videoLayers);
size_t nAudio = core->listAudioChannels(core, &audioChannels);
if (nVideo) {
for (size_t i = 0; i < nVideo; ++i) {
QAction* action = new QAction(videoLayers[i].visibleName, m_videoLayers);
action->setCheckable(true);
action->setChecked(true);
connect(action, &QAction::triggered, [this, videoLayers, i](bool enable) {
m_controller->setVideoLayerEnabled(videoLayers[i].id, enable);
});
m_videoLayers->addAction(action);
}
}
if (nAudio) {
for (size_t i = 0; i < nAudio; ++i) {
QAction* action = new QAction(audioChannels[i].visibleName, m_audioChannels);
action->setCheckable(true);
action->setChecked(true);
connect(action, &QAction::triggered, [this, audioChannels, i](bool enable) {
m_controller->setAudioChannelEnabled(audioChannels[i].id, enable);
});
m_audioChannels->addAction(action);
}
}
}
m_controller->threadContinue();
} }
void Window::gameStopped() { void Window::gameStopped() {
@ -834,6 +868,9 @@ void Window::gameStopped() {
m_screenWidget->setMinimumSize(m_display->minimumSize()); m_screenWidget->setMinimumSize(m_display->minimumSize());
setMouseTracking(false); setMouseTracking(false);
m_videoLayers->clear();
m_audioChannels->clear();
m_fpsTimer.stop(); m_fpsTimer.stop();
m_focusCheck.stop(); m_focusCheck.stop();
} }
@ -1301,6 +1338,14 @@ void Window::setupMenu(QMenuBar* menubar) {
}, this); }, this);
m_config->updateOption("lockAspectRatio"); m_config->updateOption("lockAspectRatio");
ConfigOption* lockIntegerScaling = m_config->addOption("lockIntegerScaling");
lockIntegerScaling->addBoolean(tr("Force integer scaling"), avMenu);
lockIntegerScaling->connect([this](const QVariant& value) {
m_display->lockIntegerScaling(value.toBool());
m_screenWidget->setLockIntegerScaling(value.toBool());
}, this);
m_config->updateOption("lockIntegerScaling");
ConfigOption* resampleVideo = m_config->addOption("resampleVideo"); ConfigOption* resampleVideo = m_config->addOption("resampleVideo");
resampleVideo->addBoolean(tr("Bilinear filtering"), avMenu); resampleVideo->addBoolean(tr("Bilinear filtering"), avMenu);
resampleVideo->connect([this](const QVariant& value) { resampleVideo->connect([this](const QVariant& value) {
@ -1376,45 +1421,12 @@ void Window::setupMenu(QMenuBar* menubar) {
#endif #endif
avMenu->addSeparator(); avMenu->addSeparator();
QMenu* videoLayers = avMenu->addMenu(tr("Video layers"));
m_inputModel->addMenu(videoLayers, avMenu);
for (int i = 0; i < 4; ++i) { m_videoLayers = avMenu->addMenu(tr("Video layers"));
QAction* enableBg = new QAction(tr("Background %0").arg(i), videoLayers); m_inputModel->addMenu(m_videoLayers, avMenu);
enableBg->setCheckable(true);
enableBg->setChecked(true);
connect(enableBg, &QAction::triggered, [this, i](bool enable) { m_controller->setVideoLayerEnabled(i, enable); });
addControlledAction(videoLayers, enableBg, QString("enableBG%0").arg(i));
}
QAction* enableObj = new QAction(tr("OBJ (sprites)"), videoLayers); m_audioChannels = avMenu->addMenu(tr("Audio channels"));
enableObj->setCheckable(true); m_inputModel->addMenu(m_audioChannels, avMenu);
enableObj->setChecked(true);
connect(enableObj, &QAction::triggered, [this](bool enable) { m_controller->setVideoLayerEnabled(4, enable); });
addControlledAction(videoLayers, enableObj, "enableOBJ");
QMenu* audioChannels = avMenu->addMenu(tr("Audio channels"));
m_inputModel->addMenu(audioChannels, avMenu);
for (int i = 0; i < 4; ++i) {
QAction* enableCh = new QAction(tr("Channel %0").arg(i + 1), audioChannels);
enableCh->setCheckable(true);
enableCh->setChecked(true);
connect(enableCh, &QAction::triggered, [this, i](bool enable) { m_controller->setAudioChannelEnabled(i, enable); });
addControlledAction(audioChannels, enableCh, QString("enableCh%0").arg(i + 1));
}
QAction* enableChA = new QAction(tr("Channel A"), audioChannels);
enableChA->setCheckable(true);
enableChA->setChecked(true);
connect(enableChA, &QAction::triggered, [this, i](bool enable) { m_controller->setAudioChannelEnabled(4, enable); });
addControlledAction(audioChannels, enableChA, QString("enableChA"));
QAction* enableChB = new QAction(tr("Channel B"), audioChannels);
enableChB->setCheckable(true);
enableChB->setChecked(true);
connect(enableChB, &QAction::triggered, [this, i](bool enable) { m_controller->setAudioChannelEnabled(5, enable); });
addControlledAction(audioChannels, enableChB, QString("enableChB"));
QMenu* toolsMenu = menubar->addMenu(tr("&Tools")); QMenu* toolsMenu = menubar->addMenu(tr("&Tools"));
m_inputModel->addMenu(toolsMenu); m_inputModel->addMenu(toolsMenu);
@ -1657,6 +1669,10 @@ void WindowBackground::setCenteredAspectRatio(int width, int height) {
m_aspectHeight = height; m_aspectHeight = height;
} }
void WindowBackground::setLockIntegerScaling(bool lock) {
m_lockIntegerScaling = lock;
}
void WindowBackground::paintEvent(QPaintEvent*) { void WindowBackground::paintEvent(QPaintEvent*) {
const QPixmap* logo = pixmap(); const QPixmap* logo = pixmap();
if (!logo) { if (!logo) {
@ -1680,6 +1696,10 @@ void WindowBackground::paintEvent(QPaintEvent*) {
ds.setHeight(ds.width() * m_aspectHeight / m_aspectWidth); ds.setHeight(ds.width() * m_aspectHeight / m_aspectWidth);
} }
} }
if (m_lockIntegerScaling) {
ds.setWidth(ds.width() - ds.width() % m_aspectWidth);
ds.setHeight(ds.height() - ds.height() % m_aspectHeight);
}
QPoint origin = QPoint((s.width() - ds.width()) / 2, (s.height() - ds.height()) / 2); QPoint origin = QPoint((s.width() - ds.width()) / 2, (s.height() - ds.height()) / 2);
QRect full(origin, ds); QRect full(origin, ds);
painter.drawPixmap(full, *logo); painter.drawPixmap(full, *logo);

View File

@ -172,6 +172,8 @@ private:
QTimer m_fpsTimer; QTimer m_fpsTimer;
QList<QString> m_mruFiles; QList<QString> m_mruFiles;
QMenu* m_mruMenu; QMenu* m_mruMenu;
QMenu* m_videoLayers;
QMenu* m_audioChannels;
ShaderSelector* m_shaderView; ShaderSelector* m_shaderView;
bool m_fullscreenOnStart; bool m_fullscreenOnStart;
QTimer m_focusCheck; QTimer m_focusCheck;
@ -207,6 +209,7 @@ public:
virtual QSize sizeHint() const override; virtual QSize sizeHint() const override;
void setLockAspectRatio(int width, int height); void setLockAspectRatio(int width, int height);
void setCenteredAspectRatio(int width, int height); void setCenteredAspectRatio(int width, int height);
void setLockIntegerScaling(bool lock);
protected: protected:
virtual void paintEvent(QPaintEvent*) override; virtual void paintEvent(QPaintEvent*) override;
@ -216,6 +219,7 @@ private:
bool m_centered; bool m_centered;
int m_aspectWidth; int m_aspectWidth;
int m_aspectHeight; int m_aspectHeight;
bool m_lockIntegerScaling;
}; };
} }

View File

@ -35,6 +35,7 @@ struct VideoBackend {
bool filter; bool filter;
bool lockAspectRatio; bool lockAspectRatio;
bool lockIntegerScaling;
}; };
struct VideoShader { struct VideoShader {