pokeemerald/src/main.c

435 lines
9.7 KiB
C
Raw Normal View History

2017-02-03 16:44:06 -08:00
#include "global.h"
2017-11-13 01:01:27 -05:00
#include "crt0.h"
#include "malloc.h"
2017-11-08 19:29:37 -05:00
#include "link.h"
2017-11-13 01:01:27 -05:00
#include "link_rfu.h"
#include "librfu.h"
#include "m4a.h"
2017-11-13 01:01:27 -05:00
#include "bg.h"
#include "rtc.h"
2018-02-03 09:42:19 -05:00
#include "scanline_effect.h"
2017-11-13 01:01:27 -05:00
#include "overworld.h"
#include "play_time.h"
#include "random.h"
#include "dma3.h"
2017-02-03 16:44:06 -08:00
#include "gba/flash_internal.h"
2017-11-13 01:01:27 -05:00
#include "load_save.h"
#include "gpu_regs.h"
#include "agb_flash.h"
#include "sound.h"
2017-09-05 20:13:34 +02:00
#include "battle.h"
2017-11-13 01:01:27 -05:00
#include "battle_controllers.h"
#include "text.h"
#include "intro.h"
#include "main.h"
2019-01-13 13:15:23 +01:00
#include "trainer_hill.h"
2021-03-01 14:19:41 -05:00
#include "constants/rgb.h"
2017-02-03 16:44:06 -08:00
static void VBlankIntr(void);
static void HBlankIntr(void);
static void VCountIntr(void);
static void SerialIntr(void);
static void IntrDummy(void);
const u8 gGameVersion = GAME_VERSION;
2017-02-03 16:44:06 -08:00
const u8 gGameLanguage = GAME_LANGUAGE; // English
const char BuildDateTime[] = "2005 02 21 11:10";
const IntrFunc gIntrTableTemplate[] =
{
VCountIntr, // V-count interrupt
SerialIntr, // Serial interrupt
Timer3Intr, // Timer 3 interrupt
HBlankIntr, // H-blank interrupt
VBlankIntr, // V-blank interrupt
IntrDummy, // Timer 0 interrupt
IntrDummy, // Timer 1 interrupt
IntrDummy, // Timer 2 interrupt
IntrDummy, // DMA 0 interrupt
IntrDummy, // DMA 1 interrupt
IntrDummy, // DMA 2 interrupt
IntrDummy, // DMA 3 interrupt
IntrDummy, // Key interrupt
IntrDummy, // Game Pak interrupt
};
#define INTR_COUNT ((int)(sizeof(gIntrTableTemplate)/sizeof(IntrFunc)))
2017-09-30 09:32:46 -04:00
static u16 gUnknown_03000000;
2017-02-03 16:44:06 -08:00
2017-11-07 23:53:30 -05:00
u16 gKeyRepeatStartDelay;
2017-11-11 00:05:44 -05:00
bool8 gLinkTransferringData;
2017-11-07 23:53:30 -05:00
struct Main gMain;
u16 gKeyRepeatContinueDelay;
bool8 gSoftResetDisabled;
IntrFunc gIntrTable[INTR_COUNT];
2017-11-11 00:05:44 -05:00
u8 gLinkVSyncDisabled;
2017-11-07 23:53:30 -05:00
u32 IntrMain_Buffer[0x200];
2018-11-28 10:14:32 -06:00
s8 gPcmDmaCounter;
2017-02-03 16:44:06 -08:00
2017-11-07 23:53:30 -05:00
static EWRAM_DATA u16 gTrainerId = 0;
2017-02-03 16:44:06 -08:00
2017-11-07 23:53:30 -05:00
//EWRAM_DATA void (**gFlashTimerIntrFunc)(void) = NULL;
2017-02-03 16:44:06 -08:00
static void UpdateLinkAndCallCallbacks(void);
static void InitMainCallbacks(void);
static void CallCallbacks(void);
static void SeedRngWithRtc(void);
2017-02-03 16:44:06 -08:00
static void ReadKeys(void);
void InitIntrHandlers(void);
2017-02-03 16:44:06 -08:00
static void WaitForVBlank(void);
void EnableVCountIntrAtLine150(void);
2017-02-03 16:44:06 -08:00
#define B_START_SELECT (B_BUTTON | START_BUTTON | SELECT_BUTTON)
2017-02-03 16:44:06 -08:00
void AgbMain()
{
// Modern compilers are liberal with the stack on entry to this function,
// so RegisterRamReset may crash if it resets IWRAM.
#if !MODERN
RegisterRamReset(RESET_ALL);
#endif //MODERN
2021-03-01 13:23:03 -05:00
*(vu16 *)BG_PLTT = RGB_WHITE; // Set the backdrop to white on startup
2017-02-03 16:44:06 -08:00
InitGpuRegManager();
REG_WAITCNT = WAITCNT_PREFETCH_ENABLE | WAITCNT_WS0_S_1 | WAITCNT_WS0_N_3;
InitKeys();
InitIntrHandlers();
m4aSoundInit();
EnableVCountIntrAtLine150();
2020-06-06 16:46:19 -04:00
InitRFU();
2017-02-03 16:44:06 -08:00
RtcInit();
CheckForFlashMemory();
InitMainCallbacks();
InitMapMusic();
#ifdef BUGFIX
SeedRngWithRtc(); // see comment at SeedRngWithRtc definition below
#endif
2017-02-03 16:44:06 -08:00
ClearDma3Requests();
ResetBgs();
SetDefaultFontsPointer();
InitHeap(gHeap, HEAP_SIZE);
2017-02-03 16:44:06 -08:00
gSoftResetDisabled = FALSE;
if (gFlashMemoryPresent != TRUE)
SetMainCallback2(NULL);
2017-11-11 00:05:44 -05:00
gLinkTransferringData = FALSE;
2017-02-03 16:44:06 -08:00
gUnknown_03000000 = 0xFC0;
for (;;)
{
ReadKeys();
if (gSoftResetDisabled == FALSE
&& (gMain.heldKeysRaw & A_BUTTON)
&& (gMain.heldKeysRaw & B_START_SELECT) == B_START_SELECT)
2017-02-03 16:44:06 -08:00
{
rfu_REQ_stopMode();
rfu_waitREQComplete();
DoSoftReset();
}
2021-03-29 18:32:22 -04:00
if (Overworld_SendKeysToLinkIsRunning() == TRUE)
2017-02-03 16:44:06 -08:00
{
2017-11-11 00:05:44 -05:00
gLinkTransferringData = TRUE;
2017-02-03 16:44:06 -08:00
UpdateLinkAndCallCallbacks();
2017-11-11 00:05:44 -05:00
gLinkTransferringData = FALSE;
2017-02-03 16:44:06 -08:00
}
else
{
2017-11-11 00:05:44 -05:00
gLinkTransferringData = FALSE;
2017-02-03 16:44:06 -08:00
UpdateLinkAndCallCallbacks();
2021-03-29 18:32:22 -04:00
if (Overworld_RecvKeysFromLinkIsRunning() == TRUE)
2017-02-03 16:44:06 -08:00
{
gMain.newKeys = 0;
2017-09-01 19:43:26 +02:00
ClearSpriteCopyRequests();
2017-11-11 00:05:44 -05:00
gLinkTransferringData = TRUE;
2017-02-03 16:44:06 -08:00
UpdateLinkAndCallCallbacks();
2017-11-11 00:05:44 -05:00
gLinkTransferringData = FALSE;
2017-02-03 16:44:06 -08:00
}
}
PlayTimeCounter_Update();
MapMusicMain();
WaitForVBlank();
}
}
static void UpdateLinkAndCallCallbacks(void)
{
if (!HandleLinkConnection())
CallCallbacks();
}
static void InitMainCallbacks(void)
{
gMain.vblankCounter1 = 0;
2019-03-01 01:49:11 -05:00
gTrainerHillVBlankCounter = NULL;
2017-02-03 16:44:06 -08:00
gMain.vblankCounter2 = 0;
gMain.callback1 = NULL;
2018-02-15 16:54:34 -06:00
SetMainCallback2(CB2_InitCopyrightScreenAfterBootup);
2017-09-04 18:26:39 +02:00
gSaveBlock2Ptr = &gSaveblock2;
2017-09-06 18:39:03 +02:00
gPokemonStoragePtr = &gPokemonStorage;
2017-02-03 16:44:06 -08:00
}
static void CallCallbacks(void)
{
if (gMain.callback1)
gMain.callback1();
if (gMain.callback2)
gMain.callback2();
}
void SetMainCallback2(MainCallback callback)
{
gMain.callback2 = callback;
gMain.state = 0;
}
void StartTimer1(void)
{
REG_TM1CNT_H = 0x80;
}
void SeedRngAndSetTrainerId(void)
{
u16 val = REG_TM1CNT_L;
SeedRng(val);
REG_TM1CNT_H = 0;
gTrainerId = val;
}
2017-09-03 22:50:17 +02:00
u16 GetGeneratedTrainerIdLower(void)
2017-02-03 16:44:06 -08:00
{
return gTrainerId;
}
void EnableVCountIntrAtLine150(void)
{
2017-02-17 10:56:19 -08:00
u16 gpuReg = (GetGpuReg(REG_OFFSET_DISPSTAT) & 0xFF) | (150 << 8);
SetGpuReg(REG_OFFSET_DISPSTAT, gpuReg | DISPSTAT_VCOUNT_INTR);
EnableInterrupts(INTR_FLAG_VCOUNT);
}
// FRLG commented this out to remove RTC, however Emerald didn't undo this!
#ifdef BUGFIX
static void SeedRngWithRtc(void)
{
u32 seed = RtcGetMinuteCount();
seed = (seed >> 16) ^ (seed & 0xFFFF);
SeedRng(seed);
}
#endif
2017-02-03 16:44:06 -08:00
void InitKeys(void)
{
gKeyRepeatContinueDelay = 5;
gKeyRepeatStartDelay = 40;
2017-02-17 10:52:03 -08:00
2017-02-03 16:44:06 -08:00
gMain.heldKeys = 0;
gMain.newKeys = 0;
gMain.newAndRepeatedKeys = 0;
gMain.heldKeysRaw = 0;
gMain.newKeysRaw = 0;
}
static void ReadKeys(void)
{
u16 keyInput = REG_KEYINPUT ^ KEYS_MASK;
gMain.newKeysRaw = keyInput & ~gMain.heldKeysRaw;
gMain.newKeys = gMain.newKeysRaw;
gMain.newAndRepeatedKeys = gMain.newKeysRaw;
// BUG: Key repeat won't work when pressing L using L=A button mode
// because it compares the raw key input with the remapped held keys.
// Note that newAndRepeatedKeys is never remapped either.
if (keyInput != 0 && gMain.heldKeys == keyInput)
{
gMain.keyRepeatCounter--;
if (gMain.keyRepeatCounter == 0)
2017-02-03 16:44:06 -08:00
{
gMain.newAndRepeatedKeys = keyInput;
gMain.keyRepeatCounter = gKeyRepeatContinueDelay;
}
}
else
{
// If there is no input or the input has changed, reset the counter.
gMain.keyRepeatCounter = gKeyRepeatStartDelay;
}
gMain.heldKeysRaw = keyInput;
gMain.heldKeys = gMain.heldKeysRaw;
// Remap L to A if the L=A option is enabled.
2020-07-14 11:13:03 +02:00
if (gSaveBlock2Ptr->optionsButtonMode == OPTIONS_BUTTON_MODE_L_EQUALS_A)
2017-02-03 16:44:06 -08:00
{
if (JOY_NEW(L_BUTTON))
2017-02-03 16:44:06 -08:00
gMain.newKeys |= A_BUTTON;
if (JOY_HELD(L_BUTTON))
2017-02-03 16:44:06 -08:00
gMain.heldKeys |= A_BUTTON;
}
if (gMain.newKeys & gMain.watchedKeysMask)
gMain.watchedKeysPressed = TRUE;
}
void InitIntrHandlers(void)
2017-02-03 16:44:06 -08:00
{
int i;
for (i = 0; i < INTR_COUNT; i++)
gIntrTable[i] = gIntrTableTemplate[i];
DmaCopy32(3, IntrMain, IntrMain_Buffer, sizeof(IntrMain_Buffer));
INTR_VECTOR = IntrMain_Buffer;
SetVBlankCallback(NULL);
SetHBlankCallback(NULL);
SetSerialCallback(NULL);
REG_IME = 1;
2017-02-17 10:52:03 -08:00
EnableInterrupts(INTR_FLAG_VBLANK);
2017-02-03 16:44:06 -08:00
}
void SetVBlankCallback(IntrCallback callback)
{
gMain.vblankCallback = callback;
}
void SetHBlankCallback(IntrCallback callback)
{
gMain.hblankCallback = callback;
}
void SetVCountCallback(IntrCallback callback)
{
gMain.vcountCallback = callback;
}
void RestoreSerialTimer3IntrHandlers(void)
{
gIntrTable[1] = SerialIntr;
gIntrTable[2] = Timer3Intr;
}
2017-02-03 16:44:06 -08:00
void SetSerialCallback(IntrCallback callback)
{
gMain.serialCallback = callback;
}
static void VBlankIntr(void)
{
2017-11-14 08:44:32 -05:00
if (gWirelessCommType != 0)
RfuVSync();
else if (gLinkVSyncDisabled == FALSE)
2017-11-14 08:44:32 -05:00
LinkVSync();
2017-02-03 16:44:06 -08:00
2017-02-17 10:52:03 -08:00
gMain.vblankCounter1++;
2019-03-01 01:49:11 -05:00
if (gTrainerHillVBlankCounter && *gTrainerHillVBlankCounter < 0xFFFFFFFF)
(*gTrainerHillVBlankCounter)++;
2017-02-03 16:44:06 -08:00
if (gMain.vblankCallback)
gMain.vblankCallback();
gMain.vblankCounter2++;
2017-02-17 10:52:03 -08:00
CopyBufferedValuesToGpuRegs();
ProcessDma3Requests();
2017-02-03 16:44:06 -08:00
gPcmDmaCounter = gSoundInfo.pcmDmaCounter;
m4aSoundMain();
TryReceiveLinkBattleData();
2017-09-05 20:13:34 +02:00
if (!gMain.inBattle || !(gBattleTypeFlags & (BATTLE_TYPE_LINK | BATTLE_TYPE_FRONTIER | BATTLE_TYPE_RECORDED)))
Random();
2017-02-03 16:44:06 -08:00
2020-06-07 17:37:09 -04:00
UpdateWirelessStatusIndicatorSprite();
2017-02-17 10:52:03 -08:00
INTR_CHECK |= INTR_FLAG_VBLANK;
2017-02-03 16:44:06 -08:00
gMain.intrCheck |= INTR_FLAG_VBLANK;
}
2017-09-04 18:26:39 +02:00
void InitFlashTimer(void)
2017-02-03 16:44:06 -08:00
{
SetFlashTimerIntr(2, gIntrTable + 0x7);
2017-02-03 16:44:06 -08:00
}
static void HBlankIntr(void)
{
if (gMain.hblankCallback)
gMain.hblankCallback();
INTR_CHECK |= INTR_FLAG_HBLANK;
gMain.intrCheck |= INTR_FLAG_HBLANK;
}
static void VCountIntr(void)
{
if (gMain.vcountCallback)
gMain.vcountCallback();
m4aSoundVSync();
2017-02-17 10:52:03 -08:00
INTR_CHECK |= INTR_FLAG_VCOUNT;
2017-02-03 16:44:06 -08:00
gMain.intrCheck |= INTR_FLAG_VCOUNT;
}
static void SerialIntr(void)
{
if (gMain.serialCallback)
gMain.serialCallback();
2017-02-17 10:52:03 -08:00
INTR_CHECK |= INTR_FLAG_SERIAL;
2017-02-03 16:44:06 -08:00
gMain.intrCheck |= INTR_FLAG_SERIAL;
}
static void IntrDummy(void)
{}
static void WaitForVBlank(void)
{
gMain.intrCheck &= ~INTR_FLAG_VBLANK;
2017-02-17 10:52:03 -08:00
2017-02-17 10:58:42 -08:00
while (!(gMain.intrCheck & INTR_FLAG_VBLANK))
;
}
2019-03-01 01:49:11 -05:00
void SetTrainerHillVBlankCounter(u32 *counter)
{
2019-03-01 01:49:11 -05:00
gTrainerHillVBlankCounter = counter;
}
2019-03-01 01:49:11 -05:00
void ClearTrainerHillVBlankCounter(void)
{
2019-03-01 01:49:11 -05:00
gTrainerHillVBlankCounter = NULL;
2017-02-03 16:44:06 -08:00
}
void DoSoftReset(void)
{
REG_IME = 0;
m4aSoundVSyncOff();
2018-01-16 18:59:06 +01:00
ScanlineEffect_Stop();
2017-02-03 16:44:06 -08:00
DmaStop(1);
DmaStop(2);
DmaStop(3);
SiiRtcProtect();
SoftReset(RESET_ALL);
}
void ClearPokemonCrySongs(void)
{
CpuFill16(0, gPokemonCrySongs, MAX_POKEMON_CRIES * sizeof(struct PokemonCrySong));
}