mirror of https://github.com/zeldaret/mm
386 lines
13 KiB
C
386 lines
13 KiB
C
#include "z64vimode.h"
|
|
|
|
#include "stdbool.h"
|
|
#include "PR/viint.h"
|
|
|
|
#include "debug.h"
|
|
#include "macros.h"
|
|
#include "main.h"
|
|
#include "padutils.h"
|
|
#include "regs.h"
|
|
|
|
typedef struct {
|
|
/* 0x00 */ u32 burst;
|
|
/* 0x04 */ u32 vSync;
|
|
/* 0x08 */ u32 hSync;
|
|
/* 0x0C */ u32 leap;
|
|
/* 0x10 */ u32 hStart;
|
|
/* 0x14 */ u32 vStart;
|
|
/* 0x18 */ u32 vBurst;
|
|
} ViModeStruct; // size = 0x1C
|
|
|
|
void ViMode_LogPrint(OSViMode* osViMode) {
|
|
}
|
|
|
|
/**
|
|
* Configures the custom OSViMode for this ViMode
|
|
*
|
|
* @param viMode ViMode to configure the custom OSViMode for
|
|
* @param type Identifying type for the OSViMode
|
|
* @param tvType TV Type: NTSC, PAL, MPAL or FPAL
|
|
* @param loRes Boolean: true = low resolution, false = high resolution.
|
|
* Corresponds to "L" or "H" in libultra VI mode names
|
|
* @param antialiasOff Boolean: true = point-sampling, false = anti-aliasing.
|
|
* Corresponds to "P" or "A" in libultra VI mode names
|
|
* @param modeN Boolean: controls interlacing mode, different based on resolution.
|
|
* Corresponds to "N" or "F" in libultra VI mode names
|
|
* @param fb16Bit Bolean: true = 16-bit framebuffer, false = 32-bit framebuffer.
|
|
* Corresponds to "1" or "2" in libultra VI mode names
|
|
* @param width Screen width
|
|
* @param height Screen height
|
|
* @param leftAdjust Left edge adjustment
|
|
* @param rightAdjust Right edge adjustment
|
|
* @param upperAdjust Upper edge adjustment
|
|
* @param lowerAdjust Lower edge adjustment
|
|
*/
|
|
void ViMode_Configure(OSViMode* viMode, s32 type, s32 tvType, s32 loRes, s32 antialiasOff, s32 modeN, s32 fb16Bit,
|
|
s32 width, s32 height, s32 leftAdjust, s32 rightAdjust, s32 upperAdjust, s32 lowerAdjust) {
|
|
s32 hiRes;
|
|
s32 antialiasOn;
|
|
s32 modeF;
|
|
s32 fb32Bit;
|
|
s32 hiResDeflicker; // deflickered interlacing
|
|
s32 hiResInterlaced;
|
|
s32 loResDeinterlaced;
|
|
s32 loResInterlaced;
|
|
s32 modeLAN1; // L=(lo res) A=(antialias) N=(deinterlace) 1=(16-bit)
|
|
s32 modeLPN2; // L=(lo res) P=(point-sampled) N=(deinterlace) 2=(32-bit)
|
|
s32 modeHPN2; // H=(hi res) P=(point-sampled) N=(normal interlacing) 2=(32-bit)
|
|
s32 yScaleLo;
|
|
s32 yScaleHiEvenField;
|
|
s32 yScaleHiOddField;
|
|
|
|
hiRes = !loRes;
|
|
antialiasOn = !antialiasOff;
|
|
modeF = !modeN;
|
|
fb32Bit = !fb16Bit;
|
|
|
|
hiResDeflicker = hiRes && modeF;
|
|
hiResInterlaced = hiRes && modeN;
|
|
loResDeinterlaced = loRes && modeN;
|
|
loResInterlaced = loRes && modeF;
|
|
|
|
modeLAN1 = loRes && antialiasOn && modeN && fb16Bit;
|
|
modeLPN2 = loRes && antialiasOff && modeN && fb32Bit;
|
|
modeHPN2 = hiRes && antialiasOff && modeN && fb32Bit;
|
|
|
|
upperAdjust &= ~1;
|
|
lowerAdjust &= ~1;
|
|
|
|
yScaleLo = (hiResDeflicker ? 2 : 1) * (((SCREEN_HEIGHT << 11) / SCREEN_HEIGHT) / (loRes ? 1 : 2));
|
|
|
|
yScaleHiEvenField = modeF ? (loResInterlaced ? 0x1000000 : 0x2000000) : 0;
|
|
yScaleHiOddField = modeF ? (loResInterlaced ? 0x3000000 : 0x2000000) : 0;
|
|
|
|
viMode->type = type;
|
|
viMode->comRegs.ctrl = VI_CTRL_PIXEL_ADV_3 | VI_CTRL_GAMMA_ON | VI_CTRL_GAMMA_DITHER_ON |
|
|
(!loResDeinterlaced ? VI_CTRL_SERRATE_ON : 0) | (antialiasOn ? VI_CTRL_DIVOT_ON : 0) |
|
|
(fb32Bit ? VI_CTRL_TYPE_32 : VI_CTRL_TYPE_16);
|
|
|
|
if (modeLAN1) {
|
|
// Anti-aliased, fetch extra lines as-needed
|
|
viMode->comRegs.ctrl |= 0x100;
|
|
} else if (modeLPN2 | modeHPN2) {
|
|
// Point-sampled, resampling disabled
|
|
viMode->comRegs.ctrl |= 0x300;
|
|
} else {
|
|
if (antialiasOff) {
|
|
// Point-sampled, resampling enabled
|
|
viMode->comRegs.ctrl |= 0x200;
|
|
} else {
|
|
// Anti-aliased, always fetch extra lines
|
|
viMode->comRegs.ctrl |= 0;
|
|
}
|
|
}
|
|
|
|
viMode->comRegs.width = width * (hiResInterlaced ? 2 : 1);
|
|
|
|
if (tvType < 3) {
|
|
static const ViModeStruct D_801DF0C0[] = {
|
|
{
|
|
// OS_TV_PAL
|
|
BURST(58, 35, 4, 64),
|
|
VSYNC(624),
|
|
HSYNC(3177, 21),
|
|
LEAP(3183, 3182),
|
|
HSTART(128, 768),
|
|
START(95, 569),
|
|
BURST(107, 2, 9, 0),
|
|
},
|
|
{
|
|
// OS_TV_NTSC
|
|
BURST(57, 34, 5, 62),
|
|
VSYNC(524),
|
|
HSYNC(3093, 0),
|
|
LEAP(3093, 3093),
|
|
HSTART(108, 748),
|
|
START(37, 511),
|
|
BURST(4, 2, 14, 0),
|
|
},
|
|
{
|
|
// OS_TV_MPAL
|
|
BURST(57, 30, 5, 70),
|
|
VSYNC(524),
|
|
HSYNC(3088, 0),
|
|
LEAP(3100, 3100),
|
|
HSTART(108, 748),
|
|
START(37, 511),
|
|
BURST(4, 2, 14, 0),
|
|
},
|
|
};
|
|
const ViModeStruct* ptr = &D_801DF0C0[tvType];
|
|
|
|
viMode->comRegs.burst = ptr->burst;
|
|
viMode->comRegs.vSync = ptr->vSync;
|
|
viMode->comRegs.hSync = ptr->hSync;
|
|
viMode->comRegs.leap = ptr->leap;
|
|
viMode->comRegs.hStart = ptr->hStart;
|
|
viMode->fldRegs[0].vStart = ptr->vStart;
|
|
viMode->fldRegs[1].vStart = ptr->vStart;
|
|
viMode->fldRegs[0].vBurst = ptr->vBurst;
|
|
viMode->fldRegs[1].vBurst = ptr->vBurst;
|
|
} else {
|
|
_dbg_hungup("../z_vimode.c", 216);
|
|
}
|
|
|
|
viMode->comRegs.hStart += (leftAdjust << 16) + (s16)rightAdjust;
|
|
viMode->fldRegs[0].vStart += (upperAdjust << 16) + (s16)lowerAdjust;
|
|
viMode->fldRegs[1].vStart += (upperAdjust << 16) + (s16)lowerAdjust;
|
|
|
|
if (loResDeinterlaced) {
|
|
viMode->comRegs.vSync++;
|
|
if (tvType == OS_TV_MPAL) {
|
|
viMode->comRegs.hSync += HSYNC(1, 4);
|
|
viMode->comRegs.leap += LEAP(-4, -2);
|
|
}
|
|
} else {
|
|
viMode->fldRegs[0].vStart += START(-3, -2);
|
|
if (tvType == OS_TV_MPAL) {
|
|
viMode->fldRegs[0].vBurst += BURST(-2, -1, 12, -1);
|
|
} else if (tvType == OS_TV_PAL) {
|
|
viMode->fldRegs[1].vBurst += BURST(-2, -1, 2, 0);
|
|
}
|
|
}
|
|
|
|
viMode->comRegs.xScale = (SCREEN_WIDTH << 10) / SCREEN_WIDTH;
|
|
viMode->comRegs.vCurrent = 0;
|
|
|
|
viMode->fldRegs[0].origin = ORIGIN(width * 2 * (fb16Bit ? 1 : 2));
|
|
viMode->fldRegs[1].origin = ORIGIN(width * 2 * (fb16Bit ? 1 : 2) * (loRes ? 1 : 2));
|
|
|
|
viMode->fldRegs[0].yScale = yScaleLo | yScaleHiEvenField;
|
|
viMode->fldRegs[1].yScale = yScaleLo | yScaleHiOddField;
|
|
|
|
viMode->fldRegs[0].vIntr = 2;
|
|
viMode->fldRegs[1].vIntr = 2;
|
|
}
|
|
|
|
void ViMode_Save(ViMode* viMode) {
|
|
R_VI_MODE_EDIT_STATE = viMode->editState;
|
|
R_VI_MODE_EDIT_WIDTH = viMode->viWidth;
|
|
R_VI_MODE_EDIT_HEIGHT = viMode->viHeight;
|
|
R_VI_MODE_EDIT_ULY_ADJ = viMode->upperAdjust;
|
|
R_VI_MODE_EDIT_LRY_ADJ = viMode->lowerAdjust;
|
|
R_VI_MODE_EDIT_ULX_ADJ = viMode->leftAdjust;
|
|
R_VI_MODE_EDIT_LRX_ADJ = viMode->rightAdjust;
|
|
|
|
if (SREG(58) == 1) {
|
|
SREG(58) = 0;
|
|
switch (SREG(59)) {
|
|
case 1:
|
|
ViMode_LogPrint(&osViModePalLan1);
|
|
break;
|
|
|
|
case 2:
|
|
ViMode_LogPrint(&osViModeFpalLan1);
|
|
break;
|
|
|
|
default:
|
|
ViMode_LogPrint(&viMode->customViMode);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void ViMode_Load(ViMode* viMode) {
|
|
//! @bug This condition always fails as the lowest bit is masked out to 0
|
|
if ((R_VI_MODE_EDIT_WIDTH & ~3) == 1) {
|
|
R_VI_MODE_EDIT_WIDTH += 4;
|
|
}
|
|
|
|
viMode->editState = R_VI_MODE_EDIT_STATE;
|
|
viMode->viWidth = R_VI_MODE_EDIT_WIDTH & ~3;
|
|
viMode->viHeight = R_VI_MODE_EDIT_HEIGHT;
|
|
viMode->upperAdjust = R_VI_MODE_EDIT_ULY_ADJ;
|
|
viMode->lowerAdjust = R_VI_MODE_EDIT_LRY_ADJ;
|
|
viMode->leftAdjust = R_VI_MODE_EDIT_ULX_ADJ;
|
|
viMode->rightAdjust = R_VI_MODE_EDIT_LRX_ADJ;
|
|
}
|
|
|
|
void ViMode_Init(ViMode* viMode) {
|
|
viMode->editState = VI_MODE_EDIT_STATE_INACTIVE;
|
|
viMode->viWidth = SCREEN_WIDTH;
|
|
viMode->viHeight = SCREEN_HEIGHT;
|
|
viMode->leftAdjust = 0;
|
|
viMode->rightAdjust = 0;
|
|
viMode->upperAdjust = 0;
|
|
viMode->lowerAdjust = 0;
|
|
viMode->viFeatures = OS_VI_DITHER_FILTER_ON | OS_VI_GAMMA_OFF;
|
|
viMode->tvType = osTvType;
|
|
viMode->fb16Bit = true;
|
|
viMode->modeN = true;
|
|
viMode->antialiasOff = false;
|
|
viMode->loRes = true;
|
|
|
|
ViMode_Save(viMode);
|
|
}
|
|
|
|
void ViMode_Destroy(ViMode* viMode) {
|
|
}
|
|
|
|
void ViMode_ConfigureFeatures(ViMode* viMode, s32 viFeatures) {
|
|
u32 ctrl = viMode->customViMode.comRegs.ctrl;
|
|
|
|
if (viFeatures & OS_VI_GAMMA_ON) {
|
|
ctrl |= VI_CTRL_GAMMA_ON;
|
|
}
|
|
if (viFeatures & OS_VI_GAMMA_OFF) {
|
|
ctrl &= ~VI_CTRL_GAMMA_ON;
|
|
}
|
|
if (viFeatures & OS_VI_GAMMA_DITHER_ON) {
|
|
ctrl |= VI_CTRL_GAMMA_DITHER_ON;
|
|
}
|
|
if (viFeatures & OS_VI_GAMMA_DITHER_OFF) {
|
|
ctrl &= ~VI_CTRL_GAMMA_DITHER_ON;
|
|
}
|
|
if (viFeatures & OS_VI_DIVOT_ON) {
|
|
ctrl |= VI_CTRL_DIVOT_ON;
|
|
}
|
|
if (viFeatures & OS_VI_DIVOT_OFF) {
|
|
ctrl &= ~VI_CTRL_DIVOT_ON;
|
|
}
|
|
viMode->customViMode.comRegs.ctrl = ctrl;
|
|
}
|
|
|
|
/**
|
|
* Updates the custom VI mode with controller input and any edits made with the REG editor
|
|
* (through R_VI_MODE_EDIT_* entries)
|
|
*/
|
|
void ViMode_Update(ViMode* viMode, Input* input) {
|
|
ViMode_Load(viMode);
|
|
|
|
if ((viMode->editState == VI_MODE_EDIT_STATE_ACTIVE) || (viMode->editState == VI_MODE_EDIT_STATE_2) ||
|
|
(viMode->editState == VI_MODE_EDIT_STATE_3)) {
|
|
gScreenWidth = viMode->viWidth;
|
|
gScreenHeight = viMode->viHeight;
|
|
|
|
// Controls to reset the ViMode to defaults
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_START | BTN_CUP | BTN_CRIGHT)) {
|
|
ViMode_Init(viMode);
|
|
}
|
|
|
|
// Controls to adjust the screen dimensions (upper-left)
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_CUP)) {
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_DUP)) {
|
|
viMode->upperAdjust--;
|
|
}
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_DDOWN)) {
|
|
viMode->upperAdjust++;
|
|
}
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_DLEFT)) {
|
|
viMode->leftAdjust--;
|
|
}
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_DRIGHT)) {
|
|
viMode->leftAdjust++;
|
|
}
|
|
}
|
|
|
|
// Controls to adjust the screen dimensions (lower-right)
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_CRIGHT)) {
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_DUP)) {
|
|
viMode->lowerAdjust--;
|
|
}
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_DDOWN)) {
|
|
viMode->lowerAdjust++;
|
|
}
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_DLEFT)) {
|
|
viMode->rightAdjust--;
|
|
}
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_DRIGHT)) {
|
|
viMode->rightAdjust++;
|
|
}
|
|
}
|
|
|
|
// Controls to adjust key features
|
|
if (CHECK_BTN_ALL(input->cur.button, BTN_CDOWN)) {
|
|
if (CHECK_BTN_ALL(input->press.button, BTN_DUP)) {
|
|
viMode->loRes = !viMode->loRes;
|
|
}
|
|
if (CHECK_BTN_ALL(input->press.button, BTN_DDOWN)) {
|
|
viMode->antialiasOff = !viMode->antialiasOff;
|
|
}
|
|
if (CHECK_BTN_ALL(input->press.button, BTN_DLEFT)) {
|
|
viMode->modeN = !viMode->modeN;
|
|
}
|
|
if (CHECK_BTN_ALL(input->press.button, BTN_DRIGHT)) {
|
|
viMode->fb16Bit = !viMode->fb16Bit;
|
|
}
|
|
}
|
|
|
|
// Clamp adjustments
|
|
if (viMode->editState >= VI_MODE_EDIT_STATE_2) {
|
|
// Allow parts of the framebuffer to possibly be offscreen by a small margin
|
|
if (viMode->leftAdjust < -16) {
|
|
viMode->leftAdjust = -16;
|
|
}
|
|
if (viMode->upperAdjust < -50) {
|
|
viMode->upperAdjust = -50;
|
|
}
|
|
if (viMode->rightAdjust > 16) {
|
|
viMode->rightAdjust = 16;
|
|
}
|
|
if (viMode->lowerAdjust > 50) {
|
|
viMode->lowerAdjust = 50;
|
|
}
|
|
} else {
|
|
// Do not allow parts of the framebuffer to end up offscreen
|
|
if (viMode->leftAdjust < 0) {
|
|
viMode->leftAdjust = 0;
|
|
}
|
|
if (viMode->upperAdjust < 0) {
|
|
viMode->upperAdjust = 0;
|
|
}
|
|
if (viMode->rightAdjust > 0) {
|
|
viMode->rightAdjust = 0;
|
|
}
|
|
if (viMode->lowerAdjust > 0) {
|
|
viMode->lowerAdjust = 0;
|
|
}
|
|
}
|
|
|
|
ViMode_Configure(&viMode->customViMode, OS_VI_MPAL_LPN1, osTvType, viMode->loRes, viMode->antialiasOff,
|
|
viMode->modeN, viMode->fb16Bit, viMode->viWidth, viMode->viHeight, viMode->leftAdjust,
|
|
viMode->rightAdjust, viMode->upperAdjust, viMode->lowerAdjust);
|
|
ViMode_ConfigureFeatures(viMode, viMode->viFeatures);
|
|
|
|
if (viMode->editState == VI_MODE_EDIT_STATE_3) {
|
|
// Log comparison between the NTSC LAN1 mode and the custom mode
|
|
ViMode_LogPrint(&osViModeNtscLan1);
|
|
ViMode_LogPrint(&viMode->customViMode);
|
|
viMode->editState = VI_MODE_EDIT_STATE_2;
|
|
}
|
|
}
|
|
|
|
ViMode_Save(viMode);
|
|
}
|