382 lines
13 KiB
C
382 lines
13 KiB
C
#include "global.h"
|
|
#include "gpu_regs.h"
|
|
#include "multiboot.h"
|
|
#include "malloc.h"
|
|
#include "bg.h"
|
|
#include "graphics.h"
|
|
#include "main.h"
|
|
#include "sprite.h"
|
|
#include "task.h"
|
|
#include "scanline_effect.h"
|
|
#include "window.h"
|
|
#include "text.h"
|
|
#include "menu.h"
|
|
#include "m4a.h"
|
|
#include "constants/rgb.h"
|
|
|
|
enum {
|
|
WIN_TITLE, // "Berry Program Update" header on the first screen
|
|
WIN_MSG_BODY,
|
|
WIN_GAME_NAMES, // The labels under the GBA graphics on the link screen
|
|
WIN_TURN_OFF_TITLE, // "Ruby/Sapphire" at the top of the "turn off the power" screen
|
|
};
|
|
|
|
struct {
|
|
u8 state;
|
|
u8 curScene;
|
|
u16 timer;
|
|
struct MultiBootParam mb;
|
|
} static * sBerryFix;
|
|
|
|
static void BerryFix_Main(void);
|
|
static void BerryFix_GpuSet(void);
|
|
static int BerryFix_TrySetScene(int);
|
|
static void BerryFix_SetScene(int);
|
|
static void BerryFix_HideScene(void);
|
|
|
|
static const u8 sText_BerryProgramUpdate[] = _("Berry Program Update");
|
|
static const u8 sText_RubySapphire[] = _("Ruby/Sapphire");
|
|
static const u8 sText_Emerald[] = _("Emerald");
|
|
static const u8 sText_BerryProgramWillBeUpdatedPressA[] = _("The Berry Program on your POKéMON\n"
|
|
"Ruby/Sapphire Game Pak will be updated.\n"
|
|
"{COLOR RED}{SHADOW LIGHT_RED}Press the A Button.");
|
|
static const u8 sText_EnsureGBAConnectionMatches[] = _("Please ensure the connection of your\n"
|
|
"Game Boy Advance system matches this.\n"
|
|
"{COLOR RED}{SHADOW LIGHT_RED}YES: Press the A Button.\n"
|
|
"NO: Turn off the power and try again.");
|
|
static const u8 sText_TurnOffPowerHoldingStartSelect[] = _("Please turn on the power of POKéMON\n"
|
|
"Ruby/Sapphire while holding START and\n"
|
|
"SELECT simultaneously. Then, ensure\n"
|
|
"the picture above appears.");
|
|
static const u8 sText_TransmittingPleaseWait[] = _("Transmitting. Please wait.\n"
|
|
"{COLOR RED}{SHADOW LIGHT_RED}Please do not turn off the power or\n"
|
|
"unplug the Game Boy Advance Game\nLink Cable.");
|
|
static const u8 sText_PleaseFollowInstructionsOnScreen[] = _("Please follow the instructions on your\n"
|
|
"POKéMON Ruby/Sapphire screen.");
|
|
static const u8 sText_TransmissionFailureTryAgain[] = _("Transmission failure.\n"
|
|
"{COLOR RED}{SHADOW LIGHT_RED}Please try again.");
|
|
|
|
static const struct BgTemplate sBerryFixBgTemplates[] = {
|
|
{
|
|
.bg = 0,
|
|
.charBaseIndex = 0,
|
|
.mapBaseIndex = 30,
|
|
.screenSize = 0,
|
|
.paletteMode = 0,
|
|
.priority = 0,
|
|
.baseTile = 0
|
|
},
|
|
{
|
|
.bg = 1,
|
|
.charBaseIndex = 1,
|
|
.mapBaseIndex = 31,
|
|
.screenSize = 0,
|
|
.paletteMode = 0,
|
|
.priority = 1,
|
|
.baseTile = 0
|
|
}
|
|
};
|
|
|
|
static const struct WindowTemplate sBerryFixWindowTemplates[] = {
|
|
[WIN_TITLE] = {
|
|
.bg = 0,
|
|
.tilemapLeft = 2,
|
|
.tilemapTop = 4,
|
|
.width = 26,
|
|
.height = 2,
|
|
.paletteNum = 15,
|
|
.baseBlock = 1
|
|
},
|
|
[WIN_MSG_BODY] = {
|
|
.bg = 0,
|
|
.tilemapLeft = 1,
|
|
.tilemapTop = 11,
|
|
.width = 28,
|
|
.height = 8,
|
|
.paletteNum = 15,
|
|
.baseBlock = 53
|
|
},
|
|
[WIN_GAME_NAMES] = {
|
|
.bg = 0,
|
|
.tilemapLeft = 0,
|
|
.tilemapTop = 8,
|
|
.width = 30,
|
|
.height = 2,
|
|
.paletteNum = 15,
|
|
.baseBlock = 277
|
|
},
|
|
[WIN_TURN_OFF_TITLE] = {
|
|
.bg = 0,
|
|
.tilemapLeft = 8,
|
|
.tilemapTop = 0,
|
|
.width = 14,
|
|
.height = 2,
|
|
.paletteNum = 15,
|
|
.baseBlock = 337
|
|
},
|
|
DUMMY_WIN_TEMPLATE
|
|
};
|
|
|
|
static const u16 ALIGNED(4) sText_Pal[] = INCBIN_U16("graphics/berry_fix/text.gbapal");
|
|
static const u8 sBerryProgramTextColors[] = {TEXT_DYNAMIC_COLOR_1, TEXT_DYNAMIC_COLOR_2, TEXT_DYNAMIC_COLOR_3};
|
|
static const u8 sGameTitleTextColors[] = { TEXT_COLOR_TRANSPARENT, TEXT_DYNAMIC_COLOR_1, TEXT_DYNAMIC_COLOR_4};
|
|
|
|
enum {
|
|
SCENE_ENSURE_CONNECT,
|
|
SCENE_TURN_OFF_POWER,
|
|
SCENE_TRANSMITTING,
|
|
SCENE_FOLLOW_INSTRUCT,
|
|
SCENE_TRANSMIT_FAILED,
|
|
SCENE_BEGIN,
|
|
SCENE_NONE
|
|
};
|
|
|
|
static const u8 *const sBerryProgramTexts[] = {
|
|
[SCENE_ENSURE_CONNECT] = sText_EnsureGBAConnectionMatches,
|
|
[SCENE_TURN_OFF_POWER] = sText_TurnOffPowerHoldingStartSelect,
|
|
[SCENE_TRANSMITTING] = sText_TransmittingPleaseWait,
|
|
[SCENE_FOLLOW_INSTRUCT] = sText_PleaseFollowInstructionsOnScreen,
|
|
[SCENE_TRANSMIT_FAILED] = sText_TransmissionFailureTryAgain,
|
|
[SCENE_BEGIN] = sText_BerryProgramWillBeUpdatedPressA
|
|
};
|
|
|
|
static const struct {
|
|
const u32 *gfx;
|
|
const u32 *tilemap;
|
|
const u16 *palette;
|
|
} sBerryFixGraphics[] = {
|
|
[SCENE_ENSURE_CONNECT] = {
|
|
gBerryFixGbaConnect_Gfx,
|
|
gBerryFixGbaConnect_Tilemap,
|
|
gBerryFixGbaConnect_Pal
|
|
},
|
|
[SCENE_TURN_OFF_POWER] = {
|
|
gBerryFixGameboyLogo_Gfx,
|
|
gBerryFixGameboyLogo_Tilemap,
|
|
gBerryFixGameboyLogo_Pal
|
|
},
|
|
[SCENE_TRANSMITTING] = {
|
|
gBerryFixGbaTransfer_Gfx,
|
|
gBerryFixGbaTransfer_Tilemap,
|
|
gBerryFixGbaTransfer_Pal
|
|
},
|
|
[SCENE_FOLLOW_INSTRUCT] = {
|
|
gBerryFixGbaTransferHighlight_Gfx,
|
|
gBerryFixGbaTransferHighlight_Tilemap,
|
|
gBerryFixGbaTransferHighlight_Pal
|
|
},
|
|
[SCENE_TRANSMIT_FAILED] = {
|
|
gBerryFixGbaTransferError_Gfx,
|
|
gBerryFixGbaTransferError_Tilemap,
|
|
gBerryFixGbaTransferError_Pal
|
|
},
|
|
[SCENE_BEGIN] = {
|
|
gBerryFixWindow_Gfx,
|
|
gBerryFixWindow_Tilemap,
|
|
gBerryFixWindow_Pal
|
|
},
|
|
};
|
|
|
|
extern const u8 gMultiBootProgram_BerryGlitchFix_Start[0x3BF4];
|
|
extern const u8 gMultiBootProgram_BerryGlitchFix_End[];
|
|
|
|
enum {
|
|
MAINSTATE_INIT,
|
|
MAINSTATE_BEGIN,
|
|
MAINSTATE_CONNECT,
|
|
MAINSTATE_INIT_MULTIBOOT,
|
|
MAINSTATE_MULTIBOOT,
|
|
MAINSTATE_TRANSMIT,
|
|
MAINSTATE_EXIT,
|
|
MAINSTATE_FAILED,
|
|
};
|
|
|
|
void CB2_InitBerryFixProgram(void)
|
|
{
|
|
DisableInterrupts(0xFFFF); // all
|
|
EnableInterrupts(INTR_FLAG_VBLANK);
|
|
m4aSoundVSyncOff();
|
|
SetVBlankCallback(NULL);
|
|
ResetSpriteData();
|
|
ResetTasks();
|
|
ScanlineEffect_Stop();
|
|
SetGpuReg(REG_OFFSET_DISPCNT, 0);
|
|
sBerryFix = AllocZeroed(sizeof(*sBerryFix));
|
|
sBerryFix->state = MAINSTATE_INIT;
|
|
sBerryFix->curScene = SCENE_NONE;
|
|
SetMainCallback2(BerryFix_Main);
|
|
}
|
|
|
|
#define TryScene(sceneNum) BerryFix_TrySetScene(sceneNum) == (sceneNum)
|
|
|
|
static void BerryFix_Main(void)
|
|
{
|
|
switch (sBerryFix->state)
|
|
{
|
|
case MAINSTATE_INIT:
|
|
BerryFix_GpuSet();
|
|
sBerryFix->state = MAINSTATE_BEGIN;
|
|
break;
|
|
case MAINSTATE_BEGIN:
|
|
if (TryScene(SCENE_BEGIN) && JOY_NEW(A_BUTTON))
|
|
sBerryFix->state = MAINSTATE_CONNECT;
|
|
break;
|
|
case MAINSTATE_CONNECT:
|
|
if (TryScene(SCENE_ENSURE_CONNECT) && JOY_NEW(A_BUTTON))
|
|
sBerryFix->state = MAINSTATE_INIT_MULTIBOOT;
|
|
break;
|
|
case MAINSTATE_INIT_MULTIBOOT:
|
|
if (TryScene(SCENE_TURN_OFF_POWER))
|
|
{
|
|
sBerryFix->mb.masterp = gMultiBootProgram_BerryGlitchFix_Start;
|
|
sBerryFix->mb.server_type = 0;
|
|
MultiBootInit(&sBerryFix->mb);
|
|
sBerryFix->timer = 0;
|
|
sBerryFix->state = MAINSTATE_MULTIBOOT;
|
|
}
|
|
break;
|
|
case MAINSTATE_MULTIBOOT:
|
|
MultiBootMain(&sBerryFix->mb);
|
|
if (sBerryFix->mb.probe_count != 0 || (!(sBerryFix->mb.response_bit & 2) || !(sBerryFix->mb.client_bit & 2)))
|
|
{
|
|
sBerryFix->timer = 0;
|
|
}
|
|
else if (++sBerryFix->timer > 180)
|
|
{
|
|
MultiBootStartMaster(&sBerryFix->mb,
|
|
gMultiBootProgram_BerryGlitchFix_Start + ROM_HEADER_SIZE,
|
|
(u32)(gMultiBootProgram_BerryGlitchFix_End - (gMultiBootProgram_BerryGlitchFix_Start + ROM_HEADER_SIZE)),
|
|
4,
|
|
1);
|
|
sBerryFix->state = MAINSTATE_TRANSMIT;
|
|
}
|
|
break;
|
|
case MAINSTATE_TRANSMIT:
|
|
if (TryScene(SCENE_TRANSMITTING))
|
|
{
|
|
MultiBootMain(&sBerryFix->mb);
|
|
|
|
if (MultiBootCheckComplete(&sBerryFix->mb))
|
|
sBerryFix->state = MAINSTATE_EXIT;
|
|
else if (!(sBerryFix->mb.client_bit & 2))
|
|
sBerryFix->state = MAINSTATE_FAILED;
|
|
}
|
|
break;
|
|
case MAINSTATE_EXIT:
|
|
if (TryScene(SCENE_FOLLOW_INSTRUCT) && JOY_NEW(A_BUTTON))
|
|
DoSoftReset();
|
|
break;
|
|
case MAINSTATE_FAILED:
|
|
if (TryScene(SCENE_TRANSMIT_FAILED) && JOY_NEW(A_BUTTON))
|
|
sBerryFix->state = MAINSTATE_BEGIN;
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void BerryFix_GpuSet(void)
|
|
{
|
|
s32 width, left;
|
|
|
|
SetGpuReg(REG_OFFSET_BG0CNT, 0);
|
|
SetGpuReg(REG_OFFSET_BG1CNT, 0);
|
|
SetGpuReg(REG_OFFSET_BG0HOFS, 0);
|
|
SetGpuReg(REG_OFFSET_BG0VOFS, 0);
|
|
SetGpuReg(REG_OFFSET_BG1HOFS, 0);
|
|
SetGpuReg(REG_OFFSET_BG1VOFS, 0);
|
|
SetGpuReg(REG_OFFSET_BLDCNT, 0);
|
|
|
|
DmaFill32(3, 0, VRAM, VRAM_SIZE);
|
|
DmaFill32(3, 0, OAM, OAM_SIZE);
|
|
DmaFill32(3, 0, PLTT, PLTT_SIZE);
|
|
ResetBgsAndClearDma3BusyFlags(0);
|
|
|
|
InitBgsFromTemplates(0, sBerryFixBgTemplates, ARRAY_COUNT(sBerryFixBgTemplates));
|
|
ChangeBgX(0, 0, BG_COORD_SET);
|
|
ChangeBgY(0, 0, BG_COORD_SET);
|
|
ChangeBgX(1, 0, BG_COORD_SET);
|
|
ChangeBgY(1, 0, BG_COORD_SET);
|
|
InitWindows(sBerryFixWindowTemplates);
|
|
DeactivateAllTextPrinters();
|
|
|
|
DmaCopy32(3, sText_Pal, BG_PLTT + PLTT_OFFSET_4BPP(15), sizeof(sText_Pal));
|
|
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_OBJ_1D_MAP);
|
|
FillWindowPixelBuffer(WIN_GAME_NAMES, PIXEL_FILL(0));
|
|
FillWindowPixelBuffer(WIN_TURN_OFF_TITLE, PIXEL_FILL(0));
|
|
FillWindowPixelBuffer(WIN_TITLE, PIXEL_FILL(10));
|
|
|
|
width = GetStringWidth(FONT_SMALL, sText_Emerald, 0);
|
|
left = (120 - width) / 2;
|
|
AddTextPrinterParameterized3(WIN_GAME_NAMES, FONT_SMALL, left, 3, sGameTitleTextColors, TEXT_SKIP_DRAW, sText_Emerald);
|
|
|
|
width = GetStringWidth(FONT_SMALL, sText_RubySapphire, 0);
|
|
left = (120 - width) / 2 + 120;
|
|
AddTextPrinterParameterized3(WIN_GAME_NAMES, FONT_SMALL, left, 3, sGameTitleTextColors, TEXT_SKIP_DRAW, sText_RubySapphire);
|
|
|
|
width = GetStringWidth(FONT_SMALL, sText_RubySapphire, 0);
|
|
left = (112 - width) / 2;
|
|
AddTextPrinterParameterized3(WIN_TURN_OFF_TITLE, FONT_SMALL, left, 0, sGameTitleTextColors, TEXT_SKIP_DRAW, sText_RubySapphire);
|
|
|
|
width = GetStringWidth(FONT_NORMAL, sText_BerryProgramUpdate, 0);
|
|
left = (208 - width) / 2;
|
|
AddTextPrinterParameterized3(WIN_TITLE, FONT_NORMAL, left, 2, sBerryProgramTextColors, TEXT_SKIP_DRAW, sText_BerryProgramUpdate);
|
|
|
|
CopyWindowToVram(WIN_GAME_NAMES, COPYWIN_GFX);
|
|
CopyWindowToVram(WIN_TURN_OFF_TITLE, COPYWIN_GFX);
|
|
CopyWindowToVram(WIN_TITLE, COPYWIN_GFX);
|
|
}
|
|
|
|
static int BerryFix_TrySetScene(int scene)
|
|
{
|
|
if (sBerryFix->curScene == scene)
|
|
return scene;
|
|
|
|
if (sBerryFix->curScene == SCENE_NONE)
|
|
{
|
|
BerryFix_SetScene(scene);
|
|
sBerryFix->curScene = scene;
|
|
}
|
|
else
|
|
{
|
|
BerryFix_HideScene();
|
|
sBerryFix->curScene = SCENE_NONE;
|
|
}
|
|
return sBerryFix->curScene;
|
|
}
|
|
|
|
static void BerryFix_SetScene(int scene)
|
|
{
|
|
FillBgTilemapBufferRect_Palette0(0, 0, 0, 0, 32, 32);
|
|
FillWindowPixelBuffer(WIN_MSG_BODY, PIXEL_FILL(10));
|
|
AddTextPrinterParameterized3(WIN_MSG_BODY, FONT_NORMAL, 0, 0, sBerryProgramTextColors, TEXT_SKIP_DRAW, sBerryProgramTexts[scene]);
|
|
PutWindowTilemap(WIN_MSG_BODY);
|
|
CopyWindowToVram(WIN_MSG_BODY, COPYWIN_GFX);
|
|
switch (scene)
|
|
{
|
|
case SCENE_ENSURE_CONNECT:
|
|
case SCENE_TRANSMITTING:
|
|
case SCENE_FOLLOW_INSTRUCT:
|
|
case SCENE_TRANSMIT_FAILED:
|
|
PutWindowTilemap(WIN_GAME_NAMES);
|
|
break;
|
|
case SCENE_TURN_OFF_POWER:
|
|
PutWindowTilemap(WIN_TURN_OFF_TITLE);
|
|
break;
|
|
case SCENE_BEGIN:
|
|
PutWindowTilemap(WIN_TITLE);
|
|
break;
|
|
}
|
|
CopyBgTilemapBufferToVram(0);
|
|
LZ77UnCompVram(sBerryFixGraphics[scene].gfx, (void *)BG_CHAR_ADDR(1));
|
|
LZ77UnCompVram(sBerryFixGraphics[scene].tilemap, (void *)BG_SCREEN_ADDR(31));
|
|
// These palettes range in size from 32-48 colors, so the below is interpreting whatever
|
|
// follows the palette (by default, the corresponding tiles) as the remaining 80-96.
|
|
CpuCopy32(sBerryFixGraphics[scene].palette, (void *)BG_PLTT, PLTT_SIZEOF(128));
|
|
ShowBg(0);
|
|
ShowBg(1);
|
|
}
|
|
|
|
static void BerryFix_HideScene(void)
|
|
{
|
|
HideBg(0);
|
|
HideBg(1);
|
|
}
|