mirror of https://github.com/mgba-emu/mgba.git
Wii: 240p support
This commit is contained in:
parent
7643e45212
commit
d9964ee727
1
CHANGES
1
CHANGES
|
@ -8,6 +8,7 @@ Features:
|
||||||
- Libretro: Memory map and achievement support (leiradel)
|
- Libretro: Memory map and achievement support (leiradel)
|
||||||
- GUI: Add UI control remapping
|
- GUI: Add UI control remapping
|
||||||
- GUI: Add fast-forward
|
- GUI: Add fast-forward
|
||||||
|
- Wii: 240p support
|
||||||
Bugfixes:
|
Bugfixes:
|
||||||
- SDL: Fix axes being mapped wrong
|
- SDL: Fix axes being mapped wrong
|
||||||
- GBA Memory: Fix mirror on non-overdumped Classic NES games
|
- GBA Memory: Fix mirror on non-overdumped Classic NES games
|
||||||
|
|
|
@ -40,14 +40,23 @@ static enum ScreenMode {
|
||||||
SM_MAX
|
SM_MAX
|
||||||
} screenMode = SM_PA;
|
} screenMode = SM_PA;
|
||||||
|
|
||||||
enum FilterMode {
|
static enum FilterMode {
|
||||||
FM_NEAREST,
|
FM_NEAREST,
|
||||||
FM_LINEAR,
|
FM_LINEAR,
|
||||||
FM_MAX
|
FM_MAX
|
||||||
} filterMode = FM_NEAREST;
|
} filterMode = FM_NEAREST;
|
||||||
|
|
||||||
|
static enum VideoMode {
|
||||||
|
VM_AUTODETECT,
|
||||||
|
VM_480i,
|
||||||
|
VM_480p,
|
||||||
|
VM_240p,
|
||||||
|
// TODO: PAL support
|
||||||
|
VM_MAX
|
||||||
|
} videoMode = VM_AUTODETECT;
|
||||||
|
|
||||||
#define SAMPLES 1024
|
#define SAMPLES 1024
|
||||||
#define GUI_SCALE 1.35
|
#define GUI_SCALE 1.35f
|
||||||
|
|
||||||
static void _retraceCallback(u32 count);
|
static void _retraceCallback(u32 count);
|
||||||
|
|
||||||
|
@ -81,6 +90,8 @@ static void* outputBuffer;
|
||||||
static struct mRumble rumble;
|
static struct mRumble rumble;
|
||||||
static struct mRotationSource rotation;
|
static struct mRotationSource rotation;
|
||||||
static GXRModeObj* vmode;
|
static GXRModeObj* vmode;
|
||||||
|
static float wAdjust;
|
||||||
|
static float hAdjust;
|
||||||
static Mtx model, view, modelview;
|
static Mtx model, view, modelview;
|
||||||
static uint16_t* texmem;
|
static uint16_t* texmem;
|
||||||
static GXTexObj tex;
|
static GXTexObj tex;
|
||||||
|
@ -102,7 +113,33 @@ static volatile int currentAudioBuffer = 0;
|
||||||
|
|
||||||
static struct GUIFont* font;
|
static struct GUIFont* font;
|
||||||
|
|
||||||
static void reconfigureScreen(struct mCore* core, GXRModeObj* vmode) {
|
static void reconfigureScreen(struct mGUIRunner* runner) {
|
||||||
|
if (runner) {
|
||||||
|
unsigned mode;
|
||||||
|
if (mCoreConfigGetUIntValue(&runner->config, "videoMode", &mode) && mode < VM_MAX) {
|
||||||
|
videoMode = mode;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
wAdjust = 1.f;
|
||||||
|
hAdjust = 1.f;
|
||||||
|
|
||||||
|
switch (videoMode) {
|
||||||
|
case VM_AUTODETECT:
|
||||||
|
default:
|
||||||
|
vmode = VIDEO_GetPreferredMode(0);
|
||||||
|
break;
|
||||||
|
case VM_480i:
|
||||||
|
vmode = &TVNtsc480Int;
|
||||||
|
break;
|
||||||
|
case VM_480p:
|
||||||
|
vmode = &TVNtsc480Prog;
|
||||||
|
break;
|
||||||
|
case VM_240p:
|
||||||
|
vmode = &TVNtsc240Ds;
|
||||||
|
wAdjust = 0.5f;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
free(framebuffer[0]);
|
free(framebuffer[0]);
|
||||||
free(framebuffer[1]);
|
free(framebuffer[1]);
|
||||||
|
|
||||||
|
@ -128,17 +165,21 @@ static void reconfigureScreen(struct mCore* core, GXRModeObj* vmode) {
|
||||||
GX_SetCopyFilter(vmode->aa, vmode->sample_pattern, GX_TRUE, vmode->vfilter);
|
GX_SetCopyFilter(vmode->aa, vmode->sample_pattern, GX_TRUE, vmode->vfilter);
|
||||||
GX_SetFieldMode(vmode->field_rendering, ((vmode->viHeight == 2 * vmode->xfbHeight) ? GX_ENABLE : GX_DISABLE));
|
GX_SetFieldMode(vmode->field_rendering, ((vmode->viHeight == 2 * vmode->xfbHeight) ? GX_ENABLE : GX_DISABLE));
|
||||||
|
|
||||||
if (core) {
|
if (runner) {
|
||||||
core->desiredVideoDimensions(core, &corew, &coreh);
|
runner->params.width = vmode->fbWidth * GUI_SCALE;
|
||||||
int hfactor = vmode->fbWidth / corew;
|
runner->params.height = vmode->efbHeight * GUI_SCALE;
|
||||||
int vfactor = vmode->efbHeight / coreh;
|
if (runner->core) {
|
||||||
if (hfactor > vfactor) {
|
runner->core->desiredVideoDimensions(runner->core, &corew, &coreh);
|
||||||
scaleFactor = vfactor;
|
int hfactor = vmode->fbWidth / (corew * wAdjust);
|
||||||
} else {
|
int vfactor = vmode->efbHeight / (coreh * hAdjust);
|
||||||
scaleFactor = hfactor;
|
if (hfactor > vfactor) {
|
||||||
|
scaleFactor = vfactor;
|
||||||
|
} else {
|
||||||
|
scaleFactor = hfactor;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
}
|
||||||
|
|
||||||
int main(int argc, char* argv[]) {
|
int main(int argc, char* argv[]) {
|
||||||
VIDEO_Init();
|
VIDEO_Init();
|
||||||
|
@ -155,16 +196,12 @@ int main(int argc, char* argv[]) {
|
||||||
#error This pixel format is unsupported. Please use -DCOLOR_16-BIT -DCOLOR_5_6_5
|
#error This pixel format is unsupported. Please use -DCOLOR_16-BIT -DCOLOR_5_6_5
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
vmode = VIDEO_GetPreferredMode(0);
|
|
||||||
|
|
||||||
GXColor bg = { 0, 0, 0, 0xFF };
|
GXColor bg = { 0, 0, 0, 0xFF };
|
||||||
void* fifo = memalign(32, 0x40000);
|
void* fifo = memalign(32, 0x40000);
|
||||||
memset(fifo, 0, 0x40000);
|
memset(fifo, 0, 0x40000);
|
||||||
GX_Init(fifo, 0x40000);
|
GX_Init(fifo, 0x40000);
|
||||||
GX_SetCopyClear(bg, 0x00FFFFFF);
|
GX_SetCopyClear(bg, 0x00FFFFFF);
|
||||||
|
|
||||||
reconfigureScreen(NULL, vmode);
|
|
||||||
|
|
||||||
GX_SetCullMode(GX_CULL_NONE);
|
GX_SetCullMode(GX_CULL_NONE);
|
||||||
GX_SetDispCopyGamma(GX_GM_1_0);
|
GX_SetDispCopyGamma(GX_GM_1_0);
|
||||||
|
|
||||||
|
@ -215,7 +252,7 @@ int main(int argc, char* argv[]) {
|
||||||
|
|
||||||
struct mGUIRunner runner = {
|
struct mGUIRunner runner = {
|
||||||
.params = {
|
.params = {
|
||||||
vmode->fbWidth * GUI_SCALE, vmode->efbHeight * GUI_SCALE,
|
720, 480,
|
||||||
font, "",
|
font, "",
|
||||||
_drawStart, _drawEnd,
|
_drawStart, _drawEnd,
|
||||||
_pollInput, _pollCursor,
|
_pollInput, _pollCursor,
|
||||||
|
@ -332,6 +369,19 @@ int main(int argc, char* argv[]) {
|
||||||
{ .id = 0 }
|
{ .id = 0 }
|
||||||
},
|
},
|
||||||
.configExtra = (struct GUIMenuItem[]) {
|
.configExtra = (struct GUIMenuItem[]) {
|
||||||
|
{
|
||||||
|
.title = "Video mode",
|
||||||
|
.data = "videoMode",
|
||||||
|
.submenu = 0,
|
||||||
|
.state = 0,
|
||||||
|
.validStates = (const char*[]) {
|
||||||
|
"Autodetect (recommended)",
|
||||||
|
"480i",
|
||||||
|
"480p",
|
||||||
|
"240p",
|
||||||
|
},
|
||||||
|
.nStates = 4
|
||||||
|
},
|
||||||
{
|
{
|
||||||
.title = "Screen mode",
|
.title = "Screen mode",
|
||||||
.data = "screenMode",
|
.data = "screenMode",
|
||||||
|
@ -369,6 +419,7 @@ int main(int argc, char* argv[]) {
|
||||||
.pollGameInput = _pollGameInput
|
.pollGameInput = _pollGameInput
|
||||||
};
|
};
|
||||||
mGUIInit(&runner, "wii");
|
mGUIInit(&runner, "wii");
|
||||||
|
reconfigureScreen(&runner);
|
||||||
|
|
||||||
_mapKey(&runner.params.keyMap, GCN1_INPUT, PAD_BUTTON_A, GUI_INPUT_SELECT);
|
_mapKey(&runner.params.keyMap, GCN1_INPUT, PAD_BUTTON_A, GUI_INPUT_SELECT);
|
||||||
_mapKey(&runner.params.keyMap, GCN1_INPUT, PAD_BUTTON_B, GUI_INPUT_BACK);
|
_mapKey(&runner.params.keyMap, GCN1_INPUT, PAD_BUTTON_B, GUI_INPUT_BACK);
|
||||||
|
@ -517,8 +568,8 @@ static enum GUICursorState _pollCursor(unsigned* x, unsigned* y) {
|
||||||
|
|
||||||
void _reproj(int w, int h) {
|
void _reproj(int w, int h) {
|
||||||
Mtx44 proj;
|
Mtx44 proj;
|
||||||
int top = (vmode->efbHeight - h) / 2;
|
int top = (vmode->efbHeight * hAdjust - h) / 2;
|
||||||
int left = (vmode->fbWidth - w) / 2;
|
int left = (vmode->fbWidth * wAdjust - w) / 2;
|
||||||
guOrtho(proj, -top, top + h, -left, left + w, 0, 300);
|
guOrtho(proj, -top, top + h, -left, left + w, 0, 300);
|
||||||
GX_LoadProjectionMtx(proj, GX_ORTHOGRAPHIC);
|
GX_LoadProjectionMtx(proj, GX_ORTHOGRAPHIC);
|
||||||
}
|
}
|
||||||
|
@ -606,7 +657,7 @@ void _gameUnloaded(struct mGUIRunner* runner) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void _gameLoaded(struct mGUIRunner* runner) {
|
void _gameLoaded(struct mGUIRunner* runner) {
|
||||||
reconfigureScreen(runner->core, vmode);
|
reconfigureScreen(runner);
|
||||||
if (runner->core->platform(runner->core) == PLATFORM_GBA && ((struct GBA*) runner->core->board)->memory.hw.devices & HW_GYRO) {
|
if (runner->core->platform(runner->core) == PLATFORM_GBA && ((struct GBA*) runner->core->board)->memory.hw.devices & HW_GYRO) {
|
||||||
int i;
|
int i;
|
||||||
for (i = 0; i < 6; ++i) {
|
for (i = 0; i < 6; ++i) {
|
||||||
|
@ -627,6 +678,11 @@ void _unpaused(struct mGUIRunner* runner) {
|
||||||
_CPU_ISR_Restore(level);
|
_CPU_ISR_Restore(level);
|
||||||
|
|
||||||
unsigned mode;
|
unsigned mode;
|
||||||
|
if (mCoreConfigGetUIntValue(&runner->config, "videoMode", &mode) && mode < VM_MAX) {
|
||||||
|
if (mode != videoMode) {
|
||||||
|
reconfigureScreen(runner);
|
||||||
|
}
|
||||||
|
}
|
||||||
if (mCoreConfigGetUIntValue(&runner->config, "screenMode", &mode) && mode < SM_MAX) {
|
if (mCoreConfigGetUIntValue(&runner->config, "screenMode", &mode) && mode < SM_MAX) {
|
||||||
screenMode = mode;
|
screenMode = mode;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue