sovereignx/src/bg.c

534 lines
13 KiB
C
Raw Normal View History

2017-09-05 03:18:52 +01:00
#include "global.h"
#include "dma3.h"
#define DISPCNT_ALL_BG_AND_MODE_BITS 0x0F07
#define SECONDARY_TILESET_BASE_TILE 0x4000
#define SECONDARY_TILESET_BASE_BLOCK 0x800
struct BgControl {
struct BgConfig {
u16 visible:1;
u16 unknown_1:1;
u16 screenSize:2;
u16 priority:2;
u16 mosaic:1;
u16 wraparound:1;
u16 tilesBaseBlock:2;
u16 tileMapBaseBlock:5;
u16 paletteMode:1;
u8 unknown_2;
u8 unknown_3;
} configs[4];
u16 bgVisibilityAndMode;
};
struct BgTemplate {
u32 bg:2;
u32 tilesBaseBlock:2;
u32 tileMapBaseBlock:5;
u32 screenSize:2;
u32 paletteMode:1;
u32 priority:2;
u32 unk_1:10;
};
struct BgConfig2 {
u32 unk_1:10;
u32 unk_2:4;
u32 unk_3:18;
u32 unk_4;
u32 unk_5;
u32 unk_6;
};
extern struct BgControl gUnknown_030008E0; // gGpuBgConfigs
extern struct BgConfig2 gUnknown_030008F8[4]; // gGpuBgConfigs2
extern u32 gUnknown_03000938[4];
extern u32 gUnneededFireRedVariable;
extern struct BgConfig gZeroedBgControlStruct;
extern bool32 IsInvalidBgDuplicate(u8 bg);
void ResetBgControlStructs();
extern void SetTextModeAndHideBgs();
extern bool8 IsInvalidBg(u8 bg);
extern void SetGpuReg(u8 regOffset, u16 value);
extern u16 GetGpuReg(u8 regOffset);
void ResetBgs(void)
{
ResetBgControlStructs();
gUnknown_030008E0.bgVisibilityAndMode = 0;
SetTextModeAndHideBgs();
}
void SetBgModeInternal(u8 bgMode)
{
gUnknown_030008E0.bgVisibilityAndMode &= 0xFFF8;
gUnknown_030008E0.bgVisibilityAndMode |= bgMode;
}
u8 GetBgMode(void)
{
return gUnknown_030008E0.bgVisibilityAndMode & 0x7;
}
void ResetBgControlStructs(void)
{
struct BgConfig* bgConfigs = &gUnknown_030008E0.configs[0];
struct BgConfig zeroedConfig = gZeroedBgControlStruct;
int i;
for (i = 0; i < 4; i++)
{
bgConfigs[i] = zeroedConfig;
}
}
void Unused_ResetBgControlStruct(u8 bg)
{
if (IsInvalidBg(bg) == FALSE)
{
gUnknown_030008E0.configs[bg] = gZeroedBgControlStruct;
}
}
void SetBgControlAttributes(u8 bg, u8 tilesBaseBlock, u8 tileMapBaseBlock, u8 screenSize, u8 paletteMode, u8 priority, u8 mosaic, u8 wraparound)
{
if (IsInvalidBg(bg) == FALSE)
{
if (tilesBaseBlock != 0xFF)
{
gUnknown_030008E0.configs[bg].tilesBaseBlock = tilesBaseBlock & 0x3;
}
if (tileMapBaseBlock != 0xFF)
{
gUnknown_030008E0.configs[bg].tileMapBaseBlock = tileMapBaseBlock & 0x1F;
}
if (screenSize != 0xFF)
{
gUnknown_030008E0.configs[bg].screenSize = screenSize & 0x3;
}
if (paletteMode != 0xFF)
{
gUnknown_030008E0.configs[bg].paletteMode = paletteMode;
}
if (priority != 0xFF)
{
gUnknown_030008E0.configs[bg].priority = priority & 0x3;
}
if (mosaic != 0xFF)
{
gUnknown_030008E0.configs[bg].mosaic = mosaic & 0x1;
}
if (wraparound != 0xFF)
{
gUnknown_030008E0.configs[bg].wraparound = wraparound;
}
gUnknown_030008E0.configs[bg].unknown_2 = 0;
gUnknown_030008E0.configs[bg].unknown_3 = 0;
gUnknown_030008E0.configs[bg].visible = 1;
}
}
u16 GetBgControlAttribute(u8 bg, u8 attributeId)
{
if (IsInvalidBg(bg) == FALSE && gUnknown_030008E0.configs[bg].visible != FALSE)
{
switch (attributeId)
{
case 1:
return gUnknown_030008E0.configs[bg].visible;
case 2:
return gUnknown_030008E0.configs[bg].tilesBaseBlock;
case 3:
return gUnknown_030008E0.configs[bg].tileMapBaseBlock;
case 4:
return gUnknown_030008E0.configs[bg].screenSize;
case 5:
return gUnknown_030008E0.configs[bg].paletteMode;
case 6:
return gUnknown_030008E0.configs[bg].priority;
case 7:
return gUnknown_030008E0.configs[bg].mosaic;
case 8:
return gUnknown_030008E0.configs[bg].wraparound;
}
}
return 0xFF;
}
u8 LoadBgVram(u8 bg, void *src, u16 size, u16 destOffset, u8 mode)
{
u16 offset;
s8 cursor;
if (IsInvalidBg(bg) == FALSE && gUnknown_030008E0.configs[bg].visible != FALSE)
{
switch (mode)
{
case 0x1:
offset = gUnknown_030008E0.configs[bg].tilesBaseBlock * SECONDARY_TILESET_BASE_TILE;
break;
case 0x2:
offset = gUnknown_030008E0.configs[bg].tileMapBaseBlock * SECONDARY_TILESET_BASE_BLOCK;
break;
default:
cursor = -1;
goto end;
}
offset = destOffset + offset;
cursor = RequestDma3Copy(src, (void*)(offset + BG_VRAM), size, 0);
if (cursor == -1)
{
return -1;
}
}
else
{
return -1;
}
end:
return cursor;
}
void ShowBgInternal(u8 bg)
{
u16 value;
if (IsInvalidBg(bg) == FALSE && gUnknown_030008E0.configs[bg].visible != FALSE)
{
value = gUnknown_030008E0.configs[bg].priority |
(gUnknown_030008E0.configs[bg].tilesBaseBlock << 2) |
(gUnknown_030008E0.configs[bg].mosaic << 6) |
(gUnknown_030008E0.configs[bg].paletteMode << 7) |
(gUnknown_030008E0.configs[bg].tileMapBaseBlock << 8) |
(gUnknown_030008E0.configs[bg].wraparound << 13) |
(gUnknown_030008E0.configs[bg].screenSize << 14);
SetGpuReg((bg << 1) + 0x8, value);
gUnknown_030008E0.bgVisibilityAndMode |= 1 << (bg + 8);
gUnknown_030008E0.bgVisibilityAndMode &= DISPCNT_ALL_BG_AND_MODE_BITS;
}
}
void HideBgInternal(u8 bg)
{
if (IsInvalidBg(bg) == FALSE)
{
gUnknown_030008E0.bgVisibilityAndMode &= ~(1 << (bg + 8));
gUnknown_030008E0.bgVisibilityAndMode &= DISPCNT_ALL_BG_AND_MODE_BITS;
}
}
void SyncBgVisibilityAndMode()
{
SetGpuReg(0, (GetGpuReg(0) & ~DISPCNT_ALL_BG_AND_MODE_BITS) | gUnknown_030008E0.bgVisibilityAndMode);
}
void SetTextModeAndHideBgs()
{
SetGpuReg(0, GetGpuReg(0) & ~DISPCNT_ALL_BG_AND_MODE_BITS);
}
void SetBgAffineInternal(u8 bg, u32 srcCenterX, u32 srcCenterY, u16 dispCenterX, u16 dispCenterY, u16 scaleX, u16 scaleY, u16 rotationAngle)
{
struct BgAffineSrcData src;
struct BgAffineDstData dest;
switch (gUnknown_030008E0.bgVisibilityAndMode & 0x7)
{
case 1:
if (bg != 2)
return;
break;
case 2:
if (bg < 2 || bg > 3)
return;
break;
case 0:
default:
return;
}
src.texX = srcCenterX;
src.texY = srcCenterY;
src.scrX = dispCenterX;
src.scrY = dispCenterY;
src.sx = scaleX;
src.sy = scaleY;
src.alpha = rotationAngle;
BgAffineSet(&src, &dest, 1);
SetGpuReg(REG_OFFSET_BG2PA, dest.pa);
SetGpuReg(REG_OFFSET_BG2PB, dest.pb);
SetGpuReg(REG_OFFSET_BG2PC, dest.pc);
SetGpuReg(REG_OFFSET_BG2PD, dest.pd);
SetGpuReg(REG_OFFSET_BG2PA, dest.pa);
SetGpuReg(REG_OFFSET_BG2X_L, (s16)(dest.dx));
SetGpuReg(REG_OFFSET_BG2X_H, (s16)(dest.dx >> 16));
SetGpuReg(REG_OFFSET_BG2Y_L, (s16)(dest.dy));
SetGpuReg(REG_OFFSET_BG2Y_H, (s16)(dest.dy >> 16));
}
bool8 IsInvalidBg(u8 bg)
{
if (bg > 3)
return TRUE;
return FALSE;
}
int DummiedOutFireRedLeafGreenTileAllocFunc(int a1, int a2, int a3, int a4)
{
return 0;
}
void ResetBgsAndClearDma3BusyFlags(u32 leftoverFireRedLeafGreenVariable)
{
int i;
ResetBgs();
for (i = 0; i < 4; i++)
{
gUnknown_03000938[i] = 0;
}
gUnneededFireRedVariable = leftoverFireRedLeafGreenVariable;
}
void InitBgsFromTemplates(u8 bgMode, struct BgTemplate *templates, u8 numTemplates)
{
int i;
u8 bg;
SetBgModeInternal(bgMode);
ResetBgControlStructs();
for (i = 0; i < numTemplates; i++)
{
bg = templates[i].bg;
if (bg < 4) {
SetBgControlAttributes(bg,
templates[i].tilesBaseBlock,
templates[i].tileMapBaseBlock,
templates[i].screenSize,
templates[i].paletteMode,
templates[i].priority,
0,
0);
gUnknown_030008F8[bg].unk_1 = templates[i].unk_1;
gUnknown_030008F8[bg].unk_2 = 0;
gUnknown_030008F8[bg].unk_3 = 0;
gUnknown_030008F8[bg].unk_4 = 0;
gUnknown_030008F8[bg].unk_5 = 0;
gUnknown_030008F8[bg].unk_6 = 0;
}
}
}
void InitBgFromTemplate(struct BgTemplate *template)
{
u8 bg = template->bg;
if (bg < 4)
{
SetBgControlAttributes(bg,
template->tilesBaseBlock,
template->tileMapBaseBlock,
template->screenSize,
template->paletteMode,
template->priority,
0,
0);
gUnknown_030008F8[bg].unk_1 = template->unk_1;
gUnknown_030008F8[bg].unk_2 = 0;
gUnknown_030008F8[bg].unk_3 = 0;
gUnknown_030008F8[bg].unk_4 = 0;
gUnknown_030008F8[bg].unk_5 = 0;
gUnknown_030008F8[bg].unk_6 = 0;
}
}
void SetBgMode(u8 bgMode)
{
SetBgModeInternal(bgMode);
}
u16 LoadBgTiles(u8 bg, void *src, u16 size, u16 destOffset)
{
u16 unk;
u8 cursor;
if (GetBgControlAttribute(bg, 5) == 0)
{
unk = (gUnknown_030008F8[bg].unk_1 + destOffset) * 0x20;
}
else
{
unk = (gUnknown_030008F8[bg].unk_1 + destOffset) * 0x40;
}
cursor = LoadBgVram(bg, src, size, unk, DISPCNT_MODE_1);
if (cursor == 0xFF)
{
return -1;
}
gUnknown_03000938[cursor >> 5] |= (1 << (cursor & 0x1F));
if (gUnneededFireRedVariable == 1)
{
DummiedOutFireRedLeafGreenTileAllocFunc(bg, unk >> 5, size >> 5, 1);
}
return cursor;
}
u16 LoadBgTilemap(u8 bg, void *src, u16 size, u16 destOffset)
{
u8 cursor;
cursor = LoadBgVram(bg, src, size, destOffset * 2, DISPCNT_MODE_2);
if (cursor == 0xFF)
{
return -1;
}
gUnknown_03000938[cursor >> 5] |= (1 << (cursor & 0x1F));
return cursor;
}
#ifdef NONMATCHING
u16 Unused_LoadBgPalette(u8 bg, void *src, u16 size, u16 destOffset)
{
u16 unk_1;
s8 cursor;
int cursor2;
u32* ptr;
if (IsInvalidBgDuplicate(bg) == FALSE)
{
unk_1 = (gUnknown_030008F8[bg].unk_2 * 0x20) + (destOffset * 2);
cursor = RequestDma3Copy(src, (void*)(unk_1 + BG_PLTT), size, 0);
cursor2 = cursor;
}
else
{
return -1;
}
if (cursor == -1)
{
return -1;
}
ptr = gUnknown_03000938;
if (cursor < 0)
{
cursor2 += 0x1F;
}
cursor2 >>= 5;
ptr[cursor2] |= (1 << (s8)(cursor - (cursor2 << 5)));
return (u8)cursor;
}
#else
__attribute__((naked))
u8 Unused_LoadBgPalette(u8 bg, void *src, u16 size, u16 destOffset)
{
asm("push {r4-r7,lr}\n\
add r7, r1, #0\n\
lsl r0, #24\n\
lsr r4, r0, #24\n\
lsl r2, #16\n\
lsr r6, r2, #16\n\
lsl r3, #16\n\
lsr r5, r3, #16\n\
add r0, r4, #0\n\
bl IsInvalidBgDuplicate\n\
cmp r0, #0\n\
bne _08001A98\n\
ldr r1, =gUnknown_030008F8\n\
lsl r0, r4, #4\n\
add r0, r1\n\
ldrb r1, [r0, #0x1]\n\
lsl r1, #26\n\
lsr r1, #28\n\
lsl r1, #5\n\
lsl r0, r5, #1\n\
add r1, r0\n\
lsl r1, #16\n\
lsr r1, #16\n\
mov r0, #0xA0\n\
lsl r0, #19\n\
add r1, r0\n\
add r0, r7, #0\n\
add r2, r6, #0\n\
mov r3, #0\n\
bl RequestDma3Copy\n\
lsl r3, r0, #24\n\
asr r1, r3, #24\n\
mov r0, #0x1\n\
neg r0, r0\n\
cmp r1, r0\n\
bne _08001AA4\n\
_08001A98:\n\
ldr r0, =0x0000ffff\n\
b _08001AC8\n\
.pool\n\
_08001AA4:\n\
ldr r4, =gUnknown_03000938\n\
add r0, r1, #0\n\
cmp r1, #0\n\
bge _08001AAE\n\
add r0, #0x1F\n\
_08001AAE:\n\
asr r0, #5\n\
lsl r2, r0, #2\n\
add r2, r4\n\
lsl r0, #5\n\
sub r0, r1, r0\n\
lsl r0, #24\n\
asr r0, #24\n\
mov r1, #0x1\n\
lsl r1, r0\n\
ldr r0, [r2]\n\
orr r0, r1\n\
str r0, [r2]\n\
lsr r0, r3, #24\n\
_08001AC8:\n\
pop {r4-r7}\n\
pop {r1}\n\
bx r1\n\
.pool");
}
#endif // NONMATCHING