pokeemerald/src/link.c

2448 lines
59 KiB
C
Raw Normal View History

2017-09-30 09:56:08 -04:00
#include "global.h"
2017-11-12 23:16:51 -05:00
#include "m4a.h"
#include "malloc.h"
2017-11-12 23:58:05 -05:00
#include "reset_save_heap.h"
2017-11-08 19:58:59 -05:00
#include "save.h"
2017-11-10 20:55:03 -05:00
#include "bg.h"
#include "window.h"
2017-11-07 20:25:24 -05:00
#include "librfu.h"
2017-12-16 11:27:10 -05:00
#include "random.h"
2017-11-09 21:14:43 -05:00
#include "decompress.h"
2017-11-07 22:26:00 -05:00
#include "string_util.h"
#include "event_data.h"
2017-11-09 21:48:27 -05:00
#include "item_menu.h"
#include "overworld.h"
2017-11-07 20:25:24 -05:00
#include "gpu_regs.h"
#include "palette.h"
#include "task.h"
2018-02-03 09:42:19 -05:00
#include "scanline_effect.h"
2017-11-12 23:16:51 -05:00
#include "menu.h"
#include "text.h"
2017-11-12 23:33:03 -05:00
#include "strings.h"
2018-01-18 12:53:31 -05:00
#include "constants/songs.h"
2017-11-12 23:16:51 -05:00
#include "sound.h"
2017-11-10 19:06:06 -05:00
#include "trade.h"
2017-11-12 22:24:15 -05:00
#include "battle.h"
2017-11-07 20:25:24 -05:00
#include "link.h"
2018-01-20 17:28:10 -05:00
#include "link_rfu.h"
2019-04-04 17:05:46 -04:00
#include "constants/rgb.h"
2019-10-10 03:50:51 -04:00
#include "constants/trade.h"
2017-09-30 09:56:08 -04:00
struct BlockTransfer
{
u16 pos;
u16 size;
2017-11-10 19:37:38 -05:00
const u8 *src;
2017-09-30 09:56:08 -04:00
bool8 active;
u8 multiplayerId;
};
struct LinkTestBGInfo
{
u32 screenBaseBlock;
u32 paletteNum;
u32 dummy_8;
u32 dummy_C;
};
static struct BlockTransfer sBlockSend;
static struct BlockTransfer sBlockRecv[MAX_LINK_PLAYERS];
static u32 sBlockSendDelayCounter;
static bool32 sDummy1; // Never read
static bool8 sDummy2; // Never assigned, read in unused function
static u32 sPlayerDataExchangeStatus;
static bool32 sDummy3; // Never read
static u8 sLinkTestLastBlockSendPos;
static u8 sLinkTestLastBlockRecvPos[MAX_LINK_PLAYERS];
static u8 sNumVBlanksWithoutSerialIntr;
static bool8 sSendBufferEmpty;
static u16 sSendNonzeroCheck;
static u16 sRecvNonzeroCheck;
static u8 sChecksumAvailable;
static u8 sHandshakePlayerCount;
2017-09-30 09:56:08 -04:00
2017-11-14 08:31:37 -05:00
u16 gLinkPartnersHeldKeys[6];
2017-11-11 00:25:05 -05:00
u32 gLinkDebugSeed;
2017-11-11 00:05:44 -05:00
struct LinkPlayerBlock gLocalLinkPlayerBlock;
bool8 gLinkErrorOccurred;
2017-11-11 00:25:05 -05:00
u32 gLinkDebugFlags;
u32 gLinkFiller1;
2017-11-14 08:31:37 -05:00
bool8 gRemoteLinkPlayersNotReceived[MAX_LINK_PLAYERS];
u8 gBlockReceivedStatus[MAX_LINK_PLAYERS];
u32 gLinkFiller2;
2017-11-11 00:05:44 -05:00
u16 gLinkHeldKeys;
2017-11-13 01:46:22 -05:00
u16 gRecvCmds[MAX_RFU_PLAYERS][CMD_LENGTH];
2017-11-11 00:05:44 -05:00
u32 gLinkStatus;
bool8 gLinkDummy1; // Never read
bool8 gLinkDummy2; // Never read
2020-08-13 03:09:47 -04:00
bool8 gReadyToExitStandby[MAX_LINK_PLAYERS];
bool8 gReadyToCloseLink[MAX_LINK_PLAYERS];
u16 gReadyCloseLinkType; // Never read
2017-11-11 00:05:44 -05:00
u8 gSuppressLinkErrorMessage;
2017-11-12 23:58:05 -05:00
bool8 gWirelessCommType;
2017-11-12 20:57:21 -05:00
bool8 gSavedLinkPlayerCount;
2017-11-13 01:46:22 -05:00
u16 gSendCmd[CMD_LENGTH];
2017-11-12 21:31:08 -05:00
u8 gSavedMultiplayerId;
2017-11-08 19:29:37 -05:00
bool8 gReceivedRemoteLinkPlayers;
2017-11-11 00:05:44 -05:00
struct LinkTestBGInfo gLinkTestBGInfo;
void (*gLinkCallback)(void);
2017-11-13 01:46:22 -05:00
u8 gShouldAdvanceLinkState;
2017-11-11 00:05:44 -05:00
u16 gLinkTestBlockChecksums[MAX_LINK_PLAYERS];
u8 gBlockRequestType;
u32 gLinkFiller3;
u32 gLinkFiller4;
u32 gLinkFiller5;
2017-11-12 23:16:51 -05:00
u8 gLastSendQueueCount;
2017-11-11 00:05:44 -05:00
struct Link gLink;
2017-11-12 23:16:51 -05:00
u8 gLastRecvQueueCount;
2017-11-14 08:31:37 -05:00
u16 gLinkSavedIme;
2017-11-08 19:29:37 -05:00
2017-11-11 00:05:44 -05:00
EWRAM_DATA u8 gLinkTestDebugValuesEnabled = 0;
2017-11-10 20:27:02 -05:00
EWRAM_DATA u8 gUnknown_020223BD = 0;
2020-08-24 14:52:33 -04:00
EWRAM_DATA u32 gBerryBlenderKeySendAttempts = 0;
2017-11-10 22:34:23 -05:00
EWRAM_DATA u16 gBlockRecvBuffer[MAX_RFU_PLAYERS][BLOCK_BUFFER_SIZE / 2] = {};
2017-11-11 00:05:44 -05:00
EWRAM_DATA u8 gBlockSendBuffer[BLOCK_BUFFER_SIZE] = {};
EWRAM_DATA bool8 gLinkOpen = FALSE;
EWRAM_DATA u16 gLinkType = 0;
EWRAM_DATA u16 gLinkTimeOutCounter = 0;
EWRAM_DATA struct LinkPlayer gLocalLinkPlayer = {};
2017-11-10 22:34:23 -05:00
EWRAM_DATA struct LinkPlayer gLinkPlayers[MAX_RFU_PLAYERS] = {};
2017-11-12 20:57:21 -05:00
EWRAM_DATA struct LinkPlayer gSavedLinkPlayers[MAX_RFU_PLAYERS] = {};
2017-11-12 22:09:11 -05:00
EWRAM_DATA struct {
u32 status;
2017-11-12 23:16:51 -05:00
u8 lastRecvQueueCount;
u8 lastSendQueueCount;
u8 unk_06;
} sLinkErrorBuffer = {};
2020-08-13 03:09:47 -04:00
static EWRAM_DATA u16 sReadyCloseLinkAttempts = 0; // never read
2020-06-06 16:46:19 -04:00
static EWRAM_DATA void *sLinkErrorBgTilemapBuffer = NULL;
2017-11-07 22:14:40 -05:00
2017-09-30 09:56:08 -04:00
// Static ROM declarations
2017-11-14 08:31:37 -05:00
static void InitLocalLinkPlayer(void);
static void VBlankCB_LinkError(void);
2017-11-11 00:05:44 -05:00
static void CB2_LinkTest(void);
static void ProcessRecvCmds(u8 unused);
static void LinkCB_SendHeldKeys(void);
2017-11-11 00:05:44 -05:00
static void ResetBlockSend(void);
static bool32 InitBlockSend(const void *src, size_t size);
static void LinkCB_BlockSendBegin(void);
static void LinkCB_BlockSend(void);
static void LinkCB_BlockSendEnd(void);
static void SetBlockReceivedFlag(u8 who);
static u16 LinkTestCalcBlockChecksum(const u16 *src, u16 size);
static void LinkTest_prnthex(u32 pos, u8 a0, u8 a1, u8 a2);
static void LinkCB_RequestPlayerDataExchange(void);
2017-11-11 00:25:05 -05:00
static void Task_PrintTestData(u8 taskId);
2017-11-16 23:37:09 -05:00
2020-08-13 03:09:47 -04:00
static void LinkCB_ReadyCloseLink(void);
static void LinkCB_WaitCloseLink(void);
static void LinkCB_ReadyCloseLinkWithJP(void);
static void LinkCB_WaitCloseLinkWithJP(void);
static void LinkCB_Standby(void);
static void LinkCB_StandbyForAll(void);
2017-11-16 23:37:09 -05:00
2017-11-13 01:46:22 -05:00
static void CheckErrorStatus(void);
static void CB2_PrintErrorMessage(void);
static bool8 IsSioMultiMaster(void);
2020-06-06 16:46:19 -04:00
static void SetWirelessCommType0_Internal(void);
2017-11-14 08:31:37 -05:00
static void DisableSerial(void);
static void EnableSerial(void);
static void CheckMasterOrSlave(void);
static void InitTimer(void);
static void EnqueueSendCmd(u16 *sendCmd);
static void DequeueRecvCmds(u16 (*recvCmds)[CMD_LENGTH]);
2017-11-16 23:37:09 -05:00
2017-11-15 08:44:11 -05:00
static void StartTransfer(void);
static bool8 DoHandshake(void);
2017-11-16 21:38:06 -05:00
static void DoRecv(void);
static void DoSend(void);
static void StopTimer(void);
static void SendRecvDone(void);
2017-11-13 01:46:22 -05:00
2017-09-30 09:56:08 -04:00
// .rodata
2020-06-06 16:46:19 -04:00
static const u16 sWirelessLinkDisplayPal[] = INCBIN_U16("graphics/interface/wireless_link_display.gbapal");
static const u32 sWirelessLinkDisplayGfx[] = INCBIN_U32("graphics/interface/wireless_link_display.4bpp.lz");
static const u32 sWirelessLinkDisplayTilemap[] = INCBIN_U32("graphics/interface/wireless_link_display.bin.lz");
static const u16 sLinkTestDigitsPal[] = INCBIN_U16("graphics/interface/link_test_digits.gbapal");
static const u16 sLinkTestDigitsGfx[] = INCBIN_U16("graphics/interface/link_test_digits.4bpp");
static const u8 sUnusedTransparentWhite[] = _("{HIGHLIGHT TRANSPARENT}{COLOR WHITE}");
static const u16 s2BlankTilesGfx[] = INCBIN_U16("graphics/interface/blank_1x2.4bpp");
static const struct BlockRequest sBlockRequests[] = {
2017-11-11 00:05:44 -05:00
{gBlockSendBuffer, 200},
{gBlockSendBuffer, 200},
{gBlockSendBuffer, 100},
{gBlockSendBuffer, 220},
{gBlockSendBuffer, 40}
2017-11-07 21:45:44 -05:00
};
2020-06-06 16:46:19 -04:00
static const u8 sBGControlRegs[] = {
2017-11-07 21:45:44 -05:00
REG_OFFSET_BG0CNT,
REG_OFFSET_BG1CNT,
REG_OFFSET_BG2CNT,
REG_OFFSET_BG3CNT
};
2020-06-06 16:46:19 -04:00
static const char sASCIIGameFreakInc[] = "GameFreak inc.";
static const char sASCIITestPrint[] = "TEST PRINT\nP0\nP1\nP2\nP3";
static const struct BgTemplate sLinkErrorBgTemplates[] = {
2017-11-10 20:55:03 -05:00
{
.bg = 0,
.charBaseIndex = 2,
.mapBaseIndex = 31,
.priority = 0
}, {
.bg = 1,
.charBaseIndex = 0,
.mapBaseIndex = 8,
.priority = 1
}
};
2020-06-06 16:46:19 -04:00
static const struct WindowTemplate sLinkErrorWindowTemplates[] = {
{
.bg = 0,
.tilemapLeft = 0,
.tilemapTop = 0,
.width = 30,
.height = 5,
.paletteNum = 15,
.baseBlock = 0x002
}, {
.bg = 0,
.tilemapLeft = 0,
.tilemapTop = 6,
.width = 30,
.height = 7,
.paletteNum = 15,
.baseBlock = 0x098
}, {
.bg = 0,
.tilemapLeft = 0,
.tilemapTop = 13,
.width = 30,
.height = 7,
.paletteNum = 15,
.baseBlock = 0x16A
}, DUMMY_WIN_TEMPLATE
2017-11-10 20:55:03 -05:00
};
2019-12-10 13:48:20 -05:00
static const u8 sTextColors[] = { TEXT_COLOR_TRANSPARENT, TEXT_COLOR_WHITE, TEXT_COLOR_DARK_GREY };
2020-06-06 16:46:19 -04:00
static const u8 sUnused_082ED224[] = {0x00, 0xFF, 0xFE, 0xFF, 0x00};
2017-11-07 20:25:24 -05:00
2017-09-30 09:56:08 -04:00
// .text
2017-11-07 20:25:24 -05:00
2018-07-15 04:23:38 -07:00
bool8 IsWirelessAdapterConnected(void)
2017-11-07 20:25:24 -05:00
{
2020-05-30 04:09:21 -04:00
SetWirelessCommType1();
InitRFUAPI();
2020-02-16 13:49:36 -05:00
if (rfu_LMAN_REQBN_softReset_and_checkID() == 0x8001)
2017-11-07 20:25:24 -05:00
{
rfu_REQ_stopMode();
rfu_waitREQComplete();
return TRUE;
}
2020-06-06 16:46:19 -04:00
SetWirelessCommType0_Internal();
2017-11-11 00:05:44 -05:00
CloseLink();
2017-11-07 20:25:24 -05:00
RestoreSerialTimer3IntrHandlers();
return FALSE;
}
2017-11-11 00:05:44 -05:00
void Task_DestroySelf(u8 taskId)
2017-11-07 20:25:24 -05:00
{
DestroyTask(taskId);
}
2017-11-11 00:05:44 -05:00
static void InitLinkTestBG(u8 paletteNum, u8 bgNum, u8 screenBaseBlock, u8 charBaseBlock, u16 a4)
2017-11-07 20:25:24 -05:00
{
2020-06-06 16:46:19 -04:00
LoadPalette(sLinkTestDigitsPal, paletteNum * 16, 0x20);
DmaCopy16(3, sLinkTestDigitsGfx, (u16 *)BG_CHAR_ADDR(charBaseBlock) + (16 * a4), sizeof sLinkTestDigitsGfx);
2017-11-11 00:05:44 -05:00
gLinkTestBGInfo.screenBaseBlock = screenBaseBlock;
gLinkTestBGInfo.paletteNum = paletteNum;
gLinkTestBGInfo.dummy_8 = a4;
switch (bgNum)
2017-11-07 20:25:24 -05:00
{
case 1:
2017-11-11 00:05:44 -05:00
SetGpuReg(REG_OFFSET_BG1CNT, BGCNT_SCREENBASE(screenBaseBlock) | BGCNT_PRIORITY(1) | BGCNT_CHARBASE(charBaseBlock));
2017-11-07 20:25:24 -05:00
break;
case 2:
2017-11-11 00:05:44 -05:00
SetGpuReg(REG_OFFSET_BG2CNT, BGCNT_SCREENBASE(screenBaseBlock) | BGCNT_PRIORITY(1) | BGCNT_CHARBASE(charBaseBlock));
2017-11-07 20:25:24 -05:00
break;
case 3:
2017-11-11 00:05:44 -05:00
SetGpuReg(REG_OFFSET_BG3CNT, BGCNT_SCREENBASE(screenBaseBlock) | BGCNT_PRIORITY(1) | BGCNT_CHARBASE(charBaseBlock));
2017-11-07 20:25:24 -05:00
break;
}
2017-11-11 00:05:44 -05:00
SetGpuReg(REG_OFFSET_BG0HOFS + bgNum * 4, 0);
SetGpuReg(REG_OFFSET_BG0VOFS + bgNum * 4, 0);
2017-11-07 20:25:24 -05:00
}
2017-11-07 21:45:44 -05:00
2017-11-11 00:05:44 -05:00
void sub_80094EC(u8 paletteNum, u8 bgNum, u8 screenBaseBlock, u8 charBaseBlock)
2017-11-07 21:45:44 -05:00
{
2020-06-06 16:46:19 -04:00
LoadPalette(sLinkTestDigitsPal, paletteNum * 16, 0x20);
DmaCopy16(3, sLinkTestDigitsGfx, (u16 *)BG_CHAR_ADDR(charBaseBlock), sizeof sLinkTestDigitsGfx);
2017-11-11 00:05:44 -05:00
gLinkTestBGInfo.screenBaseBlock = screenBaseBlock;
gLinkTestBGInfo.paletteNum = paletteNum;
gLinkTestBGInfo.dummy_8 = 0;
2020-06-06 16:46:19 -04:00
SetGpuReg(sBGControlRegs[bgNum], BGCNT_SCREENBASE(screenBaseBlock) | BGCNT_CHARBASE(charBaseBlock));
2017-11-07 21:45:44 -05:00
}
2017-11-07 22:14:40 -05:00
2017-11-11 00:05:44 -05:00
void LinkTestScreen(void)
2017-11-07 22:14:40 -05:00
{
int i;
ResetSpriteData();
FreeAllSpritePalettes();
ResetTasks();
SetVBlankCallback(VBlankCB_LinkError);
2017-11-11 00:05:44 -05:00
ResetBlockSend();
2020-06-03 18:00:53 -04:00
gLinkType = LINKTYPE_TRADE;
2017-11-11 00:05:44 -05:00
OpenLink();
2017-11-07 22:14:40 -05:00
SeedRng(gMain.vblankCounter2);
2017-11-16 21:38:06 -05:00
for (i = 0; i < MAX_LINK_PLAYERS; i++)
2017-11-07 22:14:40 -05:00
{
gSaveBlock2Ptr->playerTrainerId[i] = Random() % 256;
}
2017-11-11 00:05:44 -05:00
InitLinkTestBG(0, 2, 4, 0, 0);
2017-11-07 22:14:40 -05:00
SetGpuReg(REG_OFFSET_DISPCNT, DISPCNT_MODE_0 | DISPCNT_OBJ_1D_MAP | DISPCNT_BG0_ON | DISPCNT_BG2_ON | DISPCNT_OBJ_ON);
2017-11-11 00:05:44 -05:00
CreateTask(Task_DestroySelf, 0);
2017-11-07 22:14:40 -05:00
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
sDummy3 = FALSE;
2017-11-11 00:05:44 -05:00
InitLocalLinkPlayer();
2017-11-11 00:25:05 -05:00
CreateTask(Task_PrintTestData, 0);
2017-11-11 00:05:44 -05:00
SetMainCallback2(CB2_LinkTest);
2017-11-07 22:14:40 -05:00
}
2017-11-07 22:26:00 -05:00
2019-03-01 01:18:58 -05:00
void SetLocalLinkPlayerId(u8 playerId)
2017-11-07 22:26:00 -05:00
{
gLocalLinkPlayer.id = playerId;
2017-11-07 22:26:00 -05:00
}
2017-11-14 08:31:37 -05:00
static void InitLocalLinkPlayer(void)
2017-11-07 22:26:00 -05:00
{
2017-11-11 00:05:44 -05:00
gLocalLinkPlayer.trainerId = gSaveBlock2Ptr->playerTrainerId[0] | (gSaveBlock2Ptr->playerTrainerId[1] << 8) | (gSaveBlock2Ptr->playerTrainerId[2] << 16) | (gSaveBlock2Ptr->playerTrainerId[3] << 24);
StringCopy(gLocalLinkPlayer.name, gSaveBlock2Ptr->playerName);
gLocalLinkPlayer.gender = gSaveBlock2Ptr->playerGender;
gLocalLinkPlayer.linkType = gLinkType;
gLocalLinkPlayer.language = gGameLanguage;
gLocalLinkPlayer.version = gGameVersion + 0x4000;
gLocalLinkPlayer.lp_field_2 = 0x8000;
2019-10-09 05:56:44 -04:00
gLocalLinkPlayer.progressFlags = IsNationalPokedexEnabled();
2019-01-02 21:12:43 +00:00
if (FlagGet(FLAG_IS_CHAMPION))
2017-11-07 22:26:00 -05:00
{
2019-10-09 05:56:44 -04:00
gLocalLinkPlayer.progressFlags |= 0x10;
2017-11-07 22:26:00 -05:00
}
}
2017-11-07 22:52:10 -05:00
static void VBlankCB_LinkError(void)
2017-11-07 22:52:10 -05:00
{
LoadOam();
ProcessSpriteCopyRequests();
TransferPlttBuffer();
}
2017-11-14 08:31:37 -05:00
static void InitLink(void)
2017-11-07 22:52:10 -05:00
{
int i;
for (i = 0; i < CMD_LENGTH; i++)
2017-11-07 22:52:10 -05:00
{
gSendCmd[i] = 0xEfff;
2017-11-07 22:52:10 -05:00
}
2017-11-11 00:05:44 -05:00
gLinkOpen = TRUE;
EnableSerial();
2017-11-07 22:52:10 -05:00
}
2017-11-14 08:31:37 -05:00
static void Task_TriggerHandshake(u8 taskId)
2017-11-07 22:52:10 -05:00
{
2017-11-16 21:38:06 -05:00
if (++gTasks[taskId].data[0] == 5)
2017-11-07 22:52:10 -05:00
{
2017-11-13 01:46:22 -05:00
gShouldAdvanceLinkState = 1;
2017-11-07 22:52:10 -05:00
DestroyTask(taskId);
}
}
2017-11-11 00:05:44 -05:00
void OpenLink(void)
2017-11-07 22:52:10 -05:00
{
int i;
2017-11-12 23:58:05 -05:00
if (!gWirelessCommType)
2017-11-07 22:52:10 -05:00
{
2017-11-11 00:05:44 -05:00
ResetSerial();
InitLink();
gLinkCallback = LinkCB_RequestPlayerDataExchange;
gLinkVSyncDisabled = FALSE;
gLinkErrorOccurred = FALSE;
gSuppressLinkErrorMessage = FALSE;
2017-11-07 22:52:10 -05:00
ResetBlockReceivedFlags();
2017-11-11 00:05:44 -05:00
ResetBlockSend();
sDummy1 = FALSE;
gLinkDummy2 = FALSE;
gLinkDummy1 = FALSE;
2020-08-13 03:09:47 -04:00
gReadyCloseLinkType = 0;
2017-11-11 00:05:44 -05:00
CreateTask(Task_TriggerHandshake, 2);
2017-11-07 22:52:10 -05:00
}
else
{
InitRFUAPI();
2017-11-07 22:52:10 -05:00
}
gReceivedRemoteLinkPlayers = 0;
2017-11-16 21:38:06 -05:00
for (i = 0; i < MAX_LINK_PLAYERS; i++)
2017-11-07 22:52:10 -05:00
{
2017-11-14 08:31:37 -05:00
gRemoteLinkPlayersNotReceived[i] = TRUE;
2020-08-13 03:09:47 -04:00
gReadyToCloseLink[i] = FALSE;
gReadyToExitStandby[i] = FALSE;
2017-11-07 22:52:10 -05:00
}
}
2017-11-08 19:29:37 -05:00
2017-11-11 00:05:44 -05:00
void CloseLink(void)
2017-11-08 19:29:37 -05:00
{
gReceivedRemoteLinkPlayers = FALSE;
2017-11-12 23:58:05 -05:00
if (gWirelessCommType)
2017-11-08 19:29:37 -05:00
{
2020-05-30 04:09:21 -04:00
LinkRfu_Shutdown();
2017-11-08 19:29:37 -05:00
}
2017-11-11 00:05:44 -05:00
gLinkOpen = FALSE;
DisableSerial();
2017-11-08 19:29:37 -05:00
}
2017-11-11 00:05:44 -05:00
static void TestBlockTransfer(u8 nothing, u8 is, u8 used)
2017-11-08 19:29:37 -05:00
{
u8 i;
u8 status;
2017-11-11 00:05:44 -05:00
if (sLinkTestLastBlockSendPos != sBlockSend.pos)
2017-11-08 19:29:37 -05:00
{
2017-11-11 00:05:44 -05:00
LinkTest_prnthex(sBlockSend.pos, 2, 3, 2);
sLinkTestLastBlockSendPos = sBlockSend.pos;
2017-11-08 19:29:37 -05:00
}
2017-11-16 21:38:06 -05:00
for (i = 0; i < MAX_LINK_PLAYERS; i++)
2017-11-08 19:29:37 -05:00
{
2017-11-11 00:05:44 -05:00
if (sLinkTestLastBlockRecvPos[i] != sBlockRecv[i].pos)
2017-11-08 19:29:37 -05:00
{
2017-11-11 00:05:44 -05:00
LinkTest_prnthex(sBlockRecv[i].pos, 2, i + 4, 2);
sLinkTestLastBlockRecvPos[i] = sBlockRecv[i].pos;
2017-11-08 19:29:37 -05:00
}
}
status = GetBlockReceivedStatus();
if (status == 0xF) // 0b1111
{
2017-11-16 21:38:06 -05:00
for (i = 0; i < MAX_LINK_PLAYERS; i++)
2017-11-08 19:29:37 -05:00
{
if ((status >> i) & 1)
{
2017-11-11 00:05:44 -05:00
gLinkTestBlockChecksums[i] = LinkTestCalcBlockChecksum(gBlockRecvBuffer[i], sBlockRecv[i].size);
2017-11-08 19:29:37 -05:00
ResetBlockReceivedFlag(i);
2017-11-11 00:05:44 -05:00
if (gLinkTestBlockChecksums[i] != 0x0342)
2017-11-08 19:29:37 -05:00
{
2017-11-11 00:05:44 -05:00
gLinkTestDebugValuesEnabled = FALSE;
gUnknown_020223BD = FALSE;
2017-11-08 19:29:37 -05:00
}
}
}
}
}
2017-11-08 19:58:59 -05:00
2017-11-14 08:31:37 -05:00
static void LinkTestProcessKeyInput(void)
2017-11-08 19:58:59 -05:00
{
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON))
2017-11-08 19:58:59 -05:00
{
2017-11-13 01:46:22 -05:00
gShouldAdvanceLinkState = 1;
2017-11-08 19:58:59 -05:00
}
2020-09-04 21:11:55 -04:00
if (JOY_HELD(B_BUTTON))
2017-11-08 19:58:59 -05:00
{
2017-11-11 00:05:44 -05:00
InitBlockSend(gHeap + 0x4000, 0x00002004);
2017-11-08 19:58:59 -05:00
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(L_BUTTON))
2017-11-08 19:58:59 -05:00
{
2019-04-04 17:05:46 -04:00
BeginNormalPaletteFade(0xFFFFFFFF, 0, 16, 0, RGB(2, 0, 0));
2017-11-08 19:58:59 -05:00
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(START_BUTTON))
2017-11-08 19:58:59 -05:00
{
2017-11-11 00:05:44 -05:00
SetSuppressLinkErrorMessage(TRUE);
2017-11-08 19:58:59 -05:00
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(R_BUTTON))
2017-11-08 19:58:59 -05:00
{
2019-11-24 16:58:40 -05:00
TrySavingData(SAVE_LINK);
2017-11-08 19:58:59 -05:00
}
2020-09-04 21:11:55 -04:00
if (JOY_NEW(SELECT_BUTTON))
2017-11-08 19:58:59 -05:00
{
2020-08-13 03:09:47 -04:00
SetCloseLinkCallback();
2017-11-08 19:58:59 -05:00
}
2017-11-11 00:05:44 -05:00
if (gLinkTestDebugValuesEnabled)
2017-11-08 19:58:59 -05:00
{
2017-11-11 00:05:44 -05:00
SetLinkDebugValues(gMain.vblankCounter2, gLinkCallback ? gLinkVSyncDisabled : gLinkVSyncDisabled | 0x10);
2017-11-08 19:58:59 -05:00
}
}
2017-11-08 20:11:29 -05:00
2017-11-11 00:05:44 -05:00
static void CB2_LinkTest(void)
2017-11-08 20:11:29 -05:00
{
2017-11-11 00:05:44 -05:00
LinkTestProcessKeyInput();
TestBlockTransfer(1, 1, 0);
2017-11-08 20:11:29 -05:00
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
}
2017-11-11 00:05:44 -05:00
u16 LinkMain2(const u16 *heldKeys)
2017-11-08 20:11:29 -05:00
{
u8 i;
2017-11-11 00:05:44 -05:00
if (!gLinkOpen)
2017-11-08 20:11:29 -05:00
{
return 0;
}
for (i = 0; i < CMD_LENGTH; i++)
2017-11-08 20:11:29 -05:00
{
2017-11-11 00:05:44 -05:00
gSendCmd[i] = 0;
2017-11-08 20:11:29 -05:00
}
2017-11-11 00:05:44 -05:00
gLinkHeldKeys = *heldKeys;
if (gLinkStatus & LINK_STAT_CONN_ESTABLISHED)
2017-11-08 20:11:29 -05:00
{
2017-11-11 00:05:44 -05:00
ProcessRecvCmds(SIO_MULTI_CNT->id);
if (gLinkCallback != NULL)
2017-11-08 20:11:29 -05:00
{
2017-11-11 00:05:44 -05:00
gLinkCallback();
2017-11-08 20:11:29 -05:00
}
2017-11-13 01:46:22 -05:00
CheckErrorStatus();
2017-11-08 20:11:29 -05:00
}
2017-11-11 00:05:44 -05:00
return gLinkStatus;
2017-11-08 20:11:29 -05:00
}
2017-11-08 20:48:26 -05:00
2017-11-14 08:31:37 -05:00
static void HandleReceiveRemoteLinkPlayer(u8 who)
2017-11-08 20:48:26 -05:00
{
int i;
int count;
count = 0;
2017-11-14 08:31:37 -05:00
gRemoteLinkPlayersNotReceived[who] = FALSE;
2017-11-16 21:38:06 -05:00
for (i = 0; i < GetLinkPlayerCount_2(); i++)
2017-11-08 20:48:26 -05:00
{
2017-11-14 08:31:37 -05:00
count += gRemoteLinkPlayersNotReceived[i];
2017-11-08 20:48:26 -05:00
}
if (count == 0 && gReceivedRemoteLinkPlayers == 0)
{
gReceivedRemoteLinkPlayers = 1;
}
}
2017-11-09 21:14:43 -05:00
2017-11-11 00:05:44 -05:00
static void ProcessRecvCmds(u8 unused)
2017-11-09 21:14:43 -05:00
{
u16 i;
2017-11-16 21:38:06 -05:00
for (i = 0; i < MAX_LINK_PLAYERS; i++)
2017-11-09 21:14:43 -05:00
{
2017-11-14 08:31:37 -05:00
gLinkPartnersHeldKeys[i] = 0;
2017-11-11 00:05:44 -05:00
if (gRecvCmds[i][0] == 0)
2017-11-09 21:14:43 -05:00
{
continue;
}
2017-11-11 00:05:44 -05:00
switch (gRecvCmds[i][0])
2017-11-09 21:14:43 -05:00
{
2017-11-14 08:31:37 -05:00
case LINKCMD_SEND_LINK_TYPE:
2017-11-09 21:14:43 -05:00
{
struct LinkPlayerBlock *block;
2017-11-11 00:05:44 -05:00
InitLocalLinkPlayer();
block = &gLocalLinkPlayerBlock;
block->linkPlayer = gLocalLinkPlayer;
2020-06-06 16:46:19 -04:00
memcpy(block->magic1, sASCIIGameFreakInc, sizeof(block->magic1) - 1);
memcpy(block->magic2, sASCIIGameFreakInc, sizeof(block->magic2) - 1);
2017-11-11 00:05:44 -05:00
InitBlockSend(block, sizeof(*block));
2017-11-09 21:14:43 -05:00
break;
}
2020-08-24 14:52:33 -04:00
case LINKCMD_BLENDER_SEND_KEYS:
2017-11-14 08:31:37 -05:00
gLinkPartnersHeldKeys[i] = gRecvCmds[i][1];
2017-11-09 21:14:43 -05:00
break;
2017-11-14 08:31:37 -05:00
case LINKCMD_0x5555:
gLinkDummy2 = TRUE;
2017-11-09 21:14:43 -05:00
break;
2017-11-14 08:31:37 -05:00
case LINKCMD_0x5566:
gLinkDummy2 = TRUE;
2017-11-09 21:14:43 -05:00
break;
2017-11-14 08:31:37 -05:00
case LINKCMD_INIT_BLOCK:
2017-11-09 21:14:43 -05:00
{
struct BlockTransfer *blockRecv;
2017-11-11 00:05:44 -05:00
blockRecv = &sBlockRecv[i];
2017-11-09 21:14:43 -05:00
blockRecv->pos = 0;
2017-11-11 00:05:44 -05:00
blockRecv->size = gRecvCmds[i][1];
blockRecv->multiplayerId = gRecvCmds[i][2];
2017-11-09 21:14:43 -05:00
break;
}
2017-11-14 08:31:37 -05:00
case LINKCMD_CONT_BLOCK:
2017-11-09 21:14:43 -05:00
{
2017-11-11 00:05:44 -05:00
if (sBlockRecv[i].size > BLOCK_BUFFER_SIZE)
2017-11-09 21:14:43 -05:00
{
u16 *buffer;
u16 j;
buffer = (u16 *)gDecompressionBuffer;
2017-11-16 21:38:06 -05:00
for (j = 0; j < CMD_LENGTH - 1; j++)
2017-11-09 21:14:43 -05:00
{
2017-11-11 00:05:44 -05:00
buffer[(sBlockRecv[i].pos / 2) + j] = gRecvCmds[i][j + 1];
2017-11-09 21:14:43 -05:00
}
}
else
{
u16 j;
2017-11-16 21:38:06 -05:00
for (j = 0; j < CMD_LENGTH - 1; j++)
2017-11-09 21:14:43 -05:00
{
2017-11-11 00:05:44 -05:00
gBlockRecvBuffer[i][(sBlockRecv[i].pos / 2) + j] = gRecvCmds[i][j + 1];
2017-11-09 21:14:43 -05:00
}
}
2017-11-11 00:05:44 -05:00
sBlockRecv[i].pos += (CMD_LENGTH - 1) * 2;
2017-11-09 21:14:43 -05:00
2017-11-11 00:05:44 -05:00
if (sBlockRecv[i].pos >= sBlockRecv[i].size)
2017-11-09 21:14:43 -05:00
{
2017-11-14 08:31:37 -05:00
if (gRemoteLinkPlayersNotReceived[i] == TRUE)
2017-11-09 21:14:43 -05:00
{
struct LinkPlayerBlock *block;
struct LinkPlayer *linkPlayer;
block = (struct LinkPlayerBlock *)&gBlockRecvBuffer[i];
linkPlayer = &gLinkPlayers[i];
*linkPlayer = block->linkPlayer;
if ((linkPlayer->version & 0xFF) == VERSION_RUBY || (linkPlayer->version & 0xFF) == VERSION_SAPPHIRE)
{
2019-10-09 05:56:44 -04:00
linkPlayer->progressFlagsCopy = 0;
linkPlayer->neverRead = 0;
linkPlayer->progressFlags = 0;
2017-11-09 21:14:43 -05:00
}
sub_800B524(linkPlayer);
2020-06-06 16:46:19 -04:00
if (strcmp(block->magic1, sASCIIGameFreakInc) != 0
|| strcmp(block->magic2, sASCIIGameFreakInc) != 0)
2017-11-09 21:14:43 -05:00
{
2017-11-11 00:05:44 -05:00
SetMainCallback2(CB2_LinkError);
2017-11-09 21:14:43 -05:00
}
else
{
2017-11-11 00:05:44 -05:00
HandleReceiveRemoteLinkPlayer(i);
2017-11-09 21:14:43 -05:00
}
}
else
{
2017-11-11 00:05:44 -05:00
SetBlockReceivedFlag(i);
2017-11-09 21:14:43 -05:00
}
}
}
break;
2020-08-13 03:09:47 -04:00
case LINKCMD_READY_CLOSE_LINK:
gReadyToCloseLink[i] = TRUE;
2017-11-09 21:14:43 -05:00
break;
2020-08-13 03:09:47 -04:00
case LINKCMD_READY_EXIT_STANDBY:
gReadyToExitStandby[i] = TRUE;
2017-11-09 21:14:43 -05:00
break;
2020-08-24 14:52:33 -04:00
case LINKCMD_BLENDER_NO_PBLOCK_SPACE:
SetBerryBlenderLinkCallback();
2017-11-09 21:14:43 -05:00
break;
2020-08-13 03:09:47 -04:00
case LINKCMD_SEND_BLOCK_REQ:
2020-06-06 16:46:19 -04:00
SendBlock(0, sBlockRequests[gRecvCmds[i][1]].address, sBlockRequests[gRecvCmds[i][1]].size);
2017-11-09 21:14:43 -05:00
break;
2020-08-24 14:52:33 -04:00
case LINKCMD_SEND_HELD_KEYS:
2017-11-14 08:31:37 -05:00
gLinkPartnersHeldKeys[i] = gRecvCmds[i][1];
2017-11-09 21:14:43 -05:00
break;
}
}
}
2017-11-09 21:48:27 -05:00
2017-11-14 08:31:37 -05:00
static void BuildSendCmd(u16 command)
2017-11-09 21:48:27 -05:00
{
switch (command)
{
2017-11-14 08:31:37 -05:00
case LINKCMD_SEND_LINK_TYPE:
gSendCmd[0] = LINKCMD_SEND_LINK_TYPE;
2017-11-11 00:05:44 -05:00
gSendCmd[1] = gLinkType;
2017-11-09 21:48:27 -05:00
break;
2020-08-13 03:09:47 -04:00
case LINKCMD_READY_EXIT_STANDBY:
gSendCmd[0] = LINKCMD_READY_EXIT_STANDBY;
2017-11-09 21:48:27 -05:00
break;
2020-08-24 14:52:33 -04:00
case LINKCMD_BLENDER_SEND_KEYS:
gSendCmd[0] = LINKCMD_BLENDER_SEND_KEYS;
2017-11-11 00:05:44 -05:00
gSendCmd[1] = gMain.heldKeys;
2017-11-09 21:48:27 -05:00
break;
2017-11-14 08:31:37 -05:00
case LINKCMD_0x5555:
gSendCmd[0] = LINKCMD_0x5555;
2017-11-09 21:48:27 -05:00
break;
2017-11-14 08:31:37 -05:00
case LINKCMD_0x6666:
gSendCmd[0] = LINKCMD_0x6666;
2017-11-11 00:05:44 -05:00
gSendCmd[1] = 0;
2017-11-09 21:48:27 -05:00
break;
2017-11-14 08:31:37 -05:00
case LINKCMD_0x7777:
2017-11-09 21:48:27 -05:00
{
u8 i;
2017-11-14 08:31:37 -05:00
gSendCmd[0] = LINKCMD_0x7777;
2017-11-16 21:38:06 -05:00
for (i = 0; i < 5; i++)
2017-11-09 21:48:27 -05:00
{
2017-11-11 00:05:44 -05:00
gSendCmd[i + 1] = 0xEE;
2017-11-09 21:48:27 -05:00
}
break;
}
2017-11-14 08:31:37 -05:00
case LINKCMD_INIT_BLOCK:
gSendCmd[0] = LINKCMD_INIT_BLOCK;
2017-11-11 00:05:44 -05:00
gSendCmd[1] = sBlockSend.size;
gSendCmd[2] = sBlockSend.multiplayerId + 0x80;
2017-11-09 21:48:27 -05:00
break;
2020-08-24 14:52:33 -04:00
case LINKCMD_BLENDER_NO_PBLOCK_SPACE:
gSendCmd[0] = LINKCMD_BLENDER_NO_PBLOCK_SPACE;
2017-11-09 21:48:27 -05:00
break;
2017-11-14 08:31:37 -05:00
case LINKCMD_0xAAAB:
gSendCmd[0] = LINKCMD_0xAAAB;
gSendCmd[1] = gSpecialVar_ItemId;
2017-11-09 21:48:27 -05:00
break;
2020-08-13 03:09:47 -04:00
case LINKCMD_SEND_BLOCK_REQ:
gSendCmd[0] = LINKCMD_SEND_BLOCK_REQ;
2017-11-11 00:05:44 -05:00
gSendCmd[1] = gBlockRequestType;
2017-11-09 21:48:27 -05:00
break;
2020-08-13 03:09:47 -04:00
case LINKCMD_READY_CLOSE_LINK:
gSendCmd[0] = LINKCMD_READY_CLOSE_LINK;
gSendCmd[1] = gReadyCloseLinkType;
2017-11-09 21:48:27 -05:00
break;
2017-11-14 08:31:37 -05:00
case LINKCMD_0x5566:
gSendCmd[0] = LINKCMD_0x5566;
2017-11-09 21:48:27 -05:00
break;
2020-08-24 14:52:33 -04:00
case LINKCMD_SEND_HELD_KEYS:
if (gHeldKeyCodeToSend == 0 || gLinkTransferringData)
2017-11-09 21:48:27 -05:00
{
break;
}
2020-08-24 14:52:33 -04:00
gSendCmd[0] = LINKCMD_SEND_HELD_KEYS;
gSendCmd[1] = gHeldKeyCodeToSend;
2017-11-09 21:48:27 -05:00
break;
}
}
2017-11-09 21:56:47 -05:00
void StartSendingKeysToLink(void)
2017-11-09 21:56:47 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType)
2020-06-06 16:46:19 -04:00
StartSendingKeysToRfu();
gLinkCallback = LinkCB_SendHeldKeys;
2017-11-09 21:56:47 -05:00
}
bool32 IsSendingKeysToLink(void)
2017-11-09 21:56:47 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType)
return IsSendingKeysToRfu();
if (gLinkCallback == LinkCB_SendHeldKeys)
2017-11-09 21:56:47 -05:00
return TRUE;
2017-11-09 21:56:47 -05:00
return FALSE;
}
static void LinkCB_SendHeldKeys(void)
2017-11-09 21:56:47 -05:00
{
if (gReceivedRemoteLinkPlayers == TRUE)
2020-08-24 14:52:33 -04:00
BuildSendCmd(LINKCMD_SEND_HELD_KEYS);
2017-11-09 21:56:47 -05:00
}
2017-11-11 00:05:44 -05:00
void ClearLinkCallback(void)
2017-11-09 21:56:47 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType)
2019-10-05 10:41:37 -04:00
ClearLinkRfuCallback();
2017-11-09 21:56:47 -05:00
else
2017-11-11 00:05:44 -05:00
gLinkCallback = NULL;
2017-11-09 21:56:47 -05:00
}
2017-11-11 00:05:44 -05:00
void ClearLinkCallback_2(void)
2017-11-09 21:56:47 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType)
2019-10-05 10:41:37 -04:00
ClearLinkRfuCallback();
2017-11-09 21:56:47 -05:00
else
2017-11-11 00:05:44 -05:00
gLinkCallback = NULL;
2017-11-09 21:56:47 -05:00
}
2017-11-10 08:37:50 -05:00
u8 GetLinkPlayerCount(void)
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType)
2020-06-06 16:46:19 -04:00
return Rfu_GetLinkPlayerCount();
2017-11-11 00:05:44 -05:00
return EXTRACT_PLAYER_COUNT(gLinkStatus);
2017-11-10 08:37:50 -05:00
}
static int AreAnyLinkPlayersUsingVersions(u32 version1, u32 version2)
2017-11-10 08:37:50 -05:00
{
int i;
u8 nPlayers;
nPlayers = GetLinkPlayerCount();
2017-11-16 21:38:06 -05:00
for (i = 0; i < nPlayers; i++)
2017-11-10 08:37:50 -05:00
{
if ((gLinkPlayers[i].version & 0xFF) == version1
|| (gLinkPlayers[i].version & 0xFF) == version2)
2017-11-10 08:37:50 -05:00
return 1;
}
return -1;
}
u32 LinkDummy_Return2(void)
2017-11-10 08:37:50 -05:00
{
return 2;
}
// Unused
static bool32 IsFullLinkGroupWithNoRS(void)
2017-11-10 08:37:50 -05:00
{
if (GetLinkPlayerCount() != MAX_LINK_PLAYERS || AreAnyLinkPlayersUsingVersions(VERSION_RUBY, VERSION_SAPPHIRE) < 0)
2017-11-10 08:37:50 -05:00
{
return FALSE;
}
return TRUE;
}
2018-05-19 12:32:44 +02:00
bool32 Link_AnyPartnersPlayingRubyOrSapphire(void)
2017-11-10 08:37:50 -05:00
{
if (AreAnyLinkPlayersUsingVersions(VERSION_RUBY, VERSION_SAPPHIRE) >= 0)
2017-11-10 08:37:50 -05:00
{
return TRUE;
}
return FALSE;
}
bool32 Link_AnyPartnersPlayingFRLG_JP(void)
2017-11-10 08:37:50 -05:00
{
int i;
i = AreAnyLinkPlayersUsingVersions(VERSION_FIRE_RED, VERSION_LEAF_GREEN);
2017-11-10 08:37:50 -05:00
if (i >= 0 && gLinkPlayers[i].language == LANGUAGE_JAPANESE)
{
return TRUE;
}
return FALSE;
}
2017-11-11 00:05:44 -05:00
void OpenLinkTimed(void)
2017-11-10 08:37:50 -05:00
{
2017-11-11 00:05:44 -05:00
sPlayerDataExchangeStatus = EXCHANGE_NOT_STARTED;
gLinkTimeOutCounter = 0;
OpenLink();
2017-11-10 08:37:50 -05:00
}
2017-11-10 19:06:06 -05:00
2020-06-03 18:00:53 -04:00
u8 GetLinkPlayerDataExchangeStatusTimed(int minPlayers, int maxPlayers)
2017-11-10 19:06:06 -05:00
{
int i;
int count;
u32 index;
2020-06-03 18:00:53 -04:00
u8 numPlayers;
2017-11-10 19:06:06 -05:00
u32 linkType1;
u32 linkType2;
count = 0;
if (gReceivedRemoteLinkPlayers == TRUE)
{
2020-06-03 18:00:53 -04:00
numPlayers = GetLinkPlayerCount_2();
if (minPlayers > numPlayers || numPlayers > maxPlayers)
2017-11-10 19:06:06 -05:00
{
2020-06-03 18:00:53 -04:00
sPlayerDataExchangeStatus = EXCHANGE_WRONG_NUM_PLAYERS;
return sPlayerDataExchangeStatus;
2017-11-10 19:06:06 -05:00
}
else
{
if (GetLinkPlayerCount() == 0)
{
2017-11-11 00:05:44 -05:00
gLinkErrorOccurred = TRUE;
CloseLink();
2017-11-10 19:06:06 -05:00
}
2017-11-16 21:38:06 -05:00
for (i = 0, index = 0; i < GetLinkPlayerCount(); index++, i++)
2017-11-10 19:06:06 -05:00
{
if (gLinkPlayers[index].linkType == gLinkPlayers[0].linkType)
{
2017-11-16 21:38:06 -05:00
count++;
2017-11-10 19:06:06 -05:00
}
}
if (count == GetLinkPlayerCount())
{
2020-06-03 18:00:53 -04:00
if (gLinkPlayers[0].linkType == LINKTYPE_TRADE_SETUP)
2017-11-10 19:06:06 -05:00
{
2019-10-10 03:50:51 -04:00
switch (GetGameProgressForLinkTrade())
2017-11-10 19:06:06 -05:00
{
2020-06-03 18:00:53 -04:00
case TRADE_PLAYER_NOT_READY:
sPlayerDataExchangeStatus = EXCHANGE_PLAYER_NOT_READY;
break;
case TRADE_PARTNER_NOT_READY:
sPlayerDataExchangeStatus = EXCHANGE_PARTNER_NOT_READY;
break;
case TRADE_BOTH_PLAYERS_READY:
sPlayerDataExchangeStatus = EXCHANGE_COMPLETE;
break;
2017-11-10 19:06:06 -05:00
}
}
else
{
2017-11-11 00:05:44 -05:00
sPlayerDataExchangeStatus = EXCHANGE_COMPLETE;
2017-11-10 19:06:06 -05:00
}
}
else
{
2020-06-03 18:00:53 -04:00
sPlayerDataExchangeStatus = EXCHANGE_DIFF_SELECTIONS;
2017-11-10 19:06:06 -05:00
linkType1 = gLinkPlayers[GetMultiplayerId()].linkType;
linkType2 = gLinkPlayers[GetMultiplayerId() ^ 1].linkType;
2020-06-03 18:00:53 -04:00
if ((linkType1 == LINKTYPE_BATTLE_TOWER_50 && linkType2 == LINKTYPE_BATTLE_TOWER_OPEN)
|| (linkType1 == LINKTYPE_BATTLE_TOWER_OPEN && linkType2 == LINKTYPE_BATTLE_TOWER_50))
2017-11-10 19:06:06 -05:00
{
2020-06-03 18:00:53 -04:00
// 3 below indicates partner made different level mode selection
// See BattleFrontier_BattleTowerLobby_EventScript_AbortLinkDifferentSelections
2017-11-10 19:06:06 -05:00
gSpecialVar_0x8005 = 3;
}
}
}
}
2017-11-16 21:38:06 -05:00
else if (++gLinkTimeOutCounter > 600)
2017-11-10 19:06:06 -05:00
{
2017-11-11 00:05:44 -05:00
sPlayerDataExchangeStatus = EXCHANGE_TIMED_OUT;
2017-11-10 19:06:06 -05:00
}
2017-11-11 00:05:44 -05:00
return sPlayerDataExchangeStatus;
2017-11-10 19:06:06 -05:00
}
2017-11-10 19:14:35 -05:00
2017-11-11 00:05:44 -05:00
bool8 IsLinkPlayerDataExchangeComplete(void)
2017-11-10 19:14:35 -05:00
{
u8 i;
u8 count;
bool8 retval;
count = 0;
2017-11-16 21:38:06 -05:00
for (i = 0; i < GetLinkPlayerCount(); i++)
2017-11-10 19:14:35 -05:00
{
if (gLinkPlayers[i].linkType == gLinkPlayers[0].linkType)
{
2017-11-16 21:38:06 -05:00
count++;
2017-11-10 19:14:35 -05:00
}
}
if (count == GetLinkPlayerCount())
{
retval = TRUE;
2017-11-11 00:05:44 -05:00
sPlayerDataExchangeStatus = EXCHANGE_COMPLETE;
2017-11-10 19:14:35 -05:00
}
else
{
retval = FALSE;
2020-06-03 18:00:53 -04:00
sPlayerDataExchangeStatus = EXCHANGE_DIFF_SELECTIONS;
2017-11-10 19:14:35 -05:00
}
return retval;
}
u32 GetLinkPlayerTrainerId(u8 who)
{
return gLinkPlayers[who].trainerId;
}
2017-11-11 00:05:44 -05:00
void ResetLinkPlayers(void)
2017-11-10 19:14:35 -05:00
{
int i;
2017-11-16 21:38:06 -05:00
for (i = 0; i <= MAX_LINK_PLAYERS; i++)
2017-11-10 19:14:35 -05:00
{
gLinkPlayers[i] = (struct LinkPlayer){};
}
}
2017-11-10 19:25:46 -05:00
2017-11-11 00:05:44 -05:00
static void ResetBlockSend(void)
2017-11-10 19:25:46 -05:00
{
2017-11-11 00:05:44 -05:00
sBlockSend.active = FALSE;
sBlockSend.pos = 0;
sBlockSend.size = 0;
sBlockSend.src = NULL;
2017-11-10 19:25:46 -05:00
}
2017-11-11 00:05:44 -05:00
static bool32 InitBlockSend(const void *src, size_t size)
2017-11-10 19:25:46 -05:00
{
2017-11-11 00:05:44 -05:00
if (sBlockSend.active)
2017-11-10 19:25:46 -05:00
{
return FALSE;
}
2017-11-11 00:05:44 -05:00
sBlockSend.multiplayerId = GetMultiplayerId();
sBlockSend.active = TRUE;
sBlockSend.size = size;
sBlockSend.pos = 0;
if (size > BLOCK_BUFFER_SIZE)
2017-11-10 19:25:46 -05:00
{
2017-11-11 00:05:44 -05:00
sBlockSend.src = src;
2017-11-10 19:25:46 -05:00
}
else
{
2017-11-11 00:05:44 -05:00
if (src != gBlockSendBuffer)
2017-11-10 19:25:46 -05:00
{
2017-11-11 00:05:44 -05:00
memcpy(gBlockSendBuffer, src, size);
2017-11-10 19:25:46 -05:00
}
2017-11-11 00:05:44 -05:00
sBlockSend.src = gBlockSendBuffer;
2017-11-10 19:25:46 -05:00
}
2017-11-14 08:31:37 -05:00
BuildSendCmd(LINKCMD_INIT_BLOCK);
2017-11-11 00:05:44 -05:00
gLinkCallback = LinkCB_BlockSendBegin;
sBlockSendDelayCounter = 0;
2017-11-10 19:25:46 -05:00
return TRUE;
}
2017-11-10 19:37:38 -05:00
2017-11-11 00:05:44 -05:00
static void LinkCB_BlockSendBegin(void)
2017-11-10 19:37:38 -05:00
{
2017-11-16 21:38:06 -05:00
if (++sBlockSendDelayCounter > 2)
2017-11-10 19:37:38 -05:00
{
2017-11-11 00:05:44 -05:00
gLinkCallback = LinkCB_BlockSend;
2017-11-10 19:37:38 -05:00
}
}
2017-11-11 00:05:44 -05:00
static void LinkCB_BlockSend(void)
2017-11-10 19:37:38 -05:00
{
int i;
const u8 *src;
2017-11-11 00:05:44 -05:00
src = sBlockSend.src;
2017-11-14 08:31:37 -05:00
gSendCmd[0] = LINKCMD_CONT_BLOCK;
for (i = 0; i < CMD_LENGTH - 1; i++)
2017-11-10 19:37:38 -05:00
{
2017-11-11 00:05:44 -05:00
gSendCmd[i + 1] = (src[sBlockSend.pos + i * 2 + 1] << 8) | src[sBlockSend.pos + i * 2];
2017-11-10 19:37:38 -05:00
}
2017-11-11 00:05:44 -05:00
sBlockSend.pos += 14;
if (sBlockSend.size <= sBlockSend.pos)
2017-11-10 19:37:38 -05:00
{
2017-11-11 00:05:44 -05:00
sBlockSend.active = FALSE;
gLinkCallback = LinkCB_BlockSendEnd;
2017-11-10 19:37:38 -05:00
}
}
2017-11-10 20:27:02 -05:00
2017-11-11 00:05:44 -05:00
static void LinkCB_BlockSendEnd(void)
2017-11-10 20:27:02 -05:00
{
2017-11-11 00:05:44 -05:00
gLinkCallback = NULL;
2017-11-10 20:27:02 -05:00
}
2020-08-24 14:52:33 -04:00
static void LinkCB_BerryBlenderSendHeldKeys(void)
2017-11-10 20:27:02 -05:00
{
GetMultiplayerId();
2020-08-24 14:52:33 -04:00
BuildSendCmd(LINKCMD_BLENDER_SEND_KEYS);
gBerryBlenderKeySendAttempts++;
2017-11-10 20:27:02 -05:00
}
2017-11-10 20:47:24 -05:00
2020-08-24 14:52:33 -04:00
void SetBerryBlenderLinkCallback(void)
2017-11-10 20:47:24 -05:00
{
2020-08-24 14:52:33 -04:00
gBerryBlenderKeySendAttempts = 0;
2017-11-12 23:58:05 -05:00
if (gWirelessCommType)
2017-11-10 20:47:24 -05:00
{
2020-08-24 14:52:33 -04:00
Rfu_SetBerryBlenderLinkCallback();
2017-11-10 20:47:24 -05:00
}
else
{
2020-08-24 14:52:33 -04:00
gLinkCallback = LinkCB_BerryBlenderSendHeldKeys;
2017-11-10 20:47:24 -05:00
}
}
2017-11-10 22:08:17 -05:00
2020-08-24 14:52:33 -04:00
// Unused
static u32 GetBerryBlenderKeySendAttempts(void)
2017-11-10 22:08:17 -05:00
{
2020-08-24 14:52:33 -04:00
return gBerryBlenderKeySendAttempts;
2017-11-10 22:08:17 -05:00
}
2020-08-24 14:52:33 -04:00
// Unused
static void SendBerryBlenderNoSpaceForPokeblocks(void)
2017-11-10 22:08:17 -05:00
{
2020-08-24 14:52:33 -04:00
BuildSendCmd(LINKCMD_BLENDER_NO_PBLOCK_SPACE);
2017-11-10 22:08:17 -05:00
}
u8 GetMultiplayerId(void)
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-10 22:08:17 -05:00
{
2020-06-06 16:46:19 -04:00
return Rfu_GetMultiplayerId();
2017-11-10 22:08:17 -05:00
}
return SIO_MULTI_CNT->id;
}
u8 bitmask_all_link_players_but_self(void)
{
u8 mpId;
mpId = GetMultiplayerId();
return ((1 << MAX_LINK_PLAYERS) - 1) ^ (1 << mpId);
}
bool8 SendBlock(u8 unused, const void *src, u16 size)
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-10 22:08:17 -05:00
{
2020-06-06 16:46:19 -04:00
return Rfu_InitBlockSend(src, size);
2017-11-10 22:08:17 -05:00
}
2017-11-11 00:05:44 -05:00
return InitBlockSend(src, size);
2017-11-10 22:08:17 -05:00
}
2020-08-13 03:09:47 -04:00
bool8 SendBlockRequest(u8 blockReqType)
2017-11-10 22:08:17 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-10 22:08:17 -05:00
{
2020-08-13 03:09:47 -04:00
return Rfu_SendBlockRequest(blockReqType);
2017-11-10 22:08:17 -05:00
}
2017-11-11 00:05:44 -05:00
if (gLinkCallback == NULL)
2017-11-10 22:08:17 -05:00
{
2020-08-13 03:09:47 -04:00
gBlockRequestType = blockReqType;
BuildSendCmd(LINKCMD_SEND_BLOCK_REQ);
2017-11-10 22:08:17 -05:00
return TRUE;
}
return FALSE;
}
2018-12-31 02:22:21 -06:00
bool8 IsLinkTaskFinished(void)
2017-11-10 22:08:17 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-10 22:08:17 -05:00
{
2019-10-09 05:56:44 -04:00
return IsLinkRfuTaskFinished();
2017-11-10 22:08:17 -05:00
}
2017-11-11 00:05:44 -05:00
return gLinkCallback == NULL;
2017-11-10 22:08:17 -05:00
}
u8 GetBlockReceivedStatus(void)
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-10 22:08:17 -05:00
{
2020-06-06 16:46:19 -04:00
return Rfu_GetBlockReceivedStatus();
2017-11-10 22:08:17 -05:00
}
2017-11-14 08:31:37 -05:00
return (gBlockReceivedStatus[3] << 3) | (gBlockReceivedStatus[2] << 2) | (gBlockReceivedStatus[1] << 1) | (gBlockReceivedStatus[0] << 0);
2017-11-10 22:08:17 -05:00
}
2017-11-11 00:05:44 -05:00
static void SetBlockReceivedFlag(u8 who)
2017-11-10 22:08:17 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-10 22:08:17 -05:00
{
2020-06-06 16:46:19 -04:00
Rfu_SetBlockReceivedFlag(who);
2017-11-10 22:08:17 -05:00
}
else
{
2017-11-14 08:31:37 -05:00
gBlockReceivedStatus[who] = TRUE;
2017-11-10 22:08:17 -05:00
}
}
2017-11-10 22:34:23 -05:00
void ResetBlockReceivedFlags(void)
{
int i;
2018-01-18 12:53:31 -05:00
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-10 22:34:23 -05:00
{
2017-11-16 21:38:06 -05:00
for (i = 0; i < MAX_RFU_PLAYERS; i++)
2017-11-10 22:34:23 -05:00
{
2020-06-06 16:46:19 -04:00
Rfu_ResetBlockReceivedFlag(i);
2017-11-10 22:34:23 -05:00
}
}
else
{
2017-11-16 21:38:06 -05:00
for (i = 0; i < MAX_LINK_PLAYERS; i++)
2017-11-10 22:34:23 -05:00
{
2017-11-14 08:31:37 -05:00
gBlockReceivedStatus[i] = FALSE;
2017-11-10 22:34:23 -05:00
}
}
}
void ResetBlockReceivedFlag(u8 who)
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-10 22:34:23 -05:00
{
2020-06-06 16:46:19 -04:00
Rfu_ResetBlockReceivedFlag(who);
2017-11-10 22:34:23 -05:00
}
2017-11-14 08:31:37 -05:00
else if (gBlockReceivedStatus[who])
2017-11-10 22:34:23 -05:00
{
2017-11-14 08:31:37 -05:00
gBlockReceivedStatus[who] = FALSE;
2017-11-10 22:34:23 -05:00
}
}
void CheckShouldAdvanceLinkState(void)
2017-11-10 22:34:23 -05:00
{
2017-11-11 00:05:44 -05:00
if ((gLinkStatus & LINK_STAT_MASTER) && EXTRACT_PLAYER_COUNT(gLinkStatus) > 1)
2017-11-10 22:34:23 -05:00
{
2017-11-13 01:46:22 -05:00
gShouldAdvanceLinkState = 1;
2017-11-10 22:34:23 -05:00
}
}
2017-11-11 00:05:44 -05:00
static u16 LinkTestCalcBlockChecksum(const u16 *src, u16 size)
2017-11-10 22:34:23 -05:00
{
u16 chksum;
u16 i;
chksum = 0;
2017-11-16 21:38:06 -05:00
for (i = 0; i < size / 2; i++)
2017-11-10 22:34:23 -05:00
{
2017-11-11 00:05:44 -05:00
chksum += src[i];
2017-11-10 22:34:23 -05:00
}
return chksum;
}
2017-11-11 00:05:44 -05:00
static void LinkTest_prnthexchar(char a0, u8 a1, u8 a2)
2017-11-10 22:34:23 -05:00
{
u16 *vAddr;
2017-11-11 00:05:44 -05:00
vAddr = (u16 *)BG_SCREEN_ADDR(gLinkTestBGInfo.screenBaseBlock);
vAddr[a2 * 32 + a1] = (gLinkTestBGInfo.paletteNum << 12) | (a0 + 1 + gLinkTestBGInfo.dummy_8);
2017-11-10 22:34:23 -05:00
}
2017-11-10 22:44:35 -05:00
2017-11-11 00:05:44 -05:00
static void LinkTest_prntchar(char a0, u8 a1, u8 a2)
2017-11-10 22:44:35 -05:00
{
u16 *vAddr;
2017-11-11 00:05:44 -05:00
vAddr = (u16 *)BG_SCREEN_ADDR(gLinkTestBGInfo.screenBaseBlock);
vAddr[a2 * 32 + a1] = (gLinkTestBGInfo.paletteNum << 12) | (a0 + gLinkTestBGInfo.dummy_8);
2017-11-10 22:44:35 -05:00
}
2017-11-11 00:05:44 -05:00
static void LinkTest_prnthex(u32 pos, u8 a0, u8 a1, u8 a2)
2017-11-10 22:44:35 -05:00
{
2017-11-10 23:12:21 -05:00
char sp[32 / 2];
2017-11-10 22:44:35 -05:00
int i;
2017-11-16 21:38:06 -05:00
for (i = 0; i < a2; i++)
2017-11-10 22:44:35 -05:00
{
2017-11-11 00:05:44 -05:00
sp[i] = pos & 0xf;
pos >>= 4;
2017-11-10 22:44:35 -05:00
}
2017-11-16 21:38:06 -05:00
for (i = a2 - 1; i >= 0; i--)
2017-11-10 22:44:35 -05:00
{
2017-11-11 00:05:44 -05:00
LinkTest_prnthexchar(sp[i], a0, a1);
2017-11-16 21:38:06 -05:00
a0++;
2017-11-10 22:44:35 -05:00
}
}
2017-11-10 22:54:40 -05:00
2017-11-11 00:05:44 -05:00
static void LinkTest_prntint(int a0, u8 a1, u8 a2, u8 a3)
2017-11-10 22:54:40 -05:00
{
2017-11-10 23:12:21 -05:00
char sp[32 / 2];
2017-11-10 22:54:40 -05:00
int sp10;
int i;
sp10 = -1;
if (a0 < 0)
{
sp10 = a1;
a0 = -a0;
}
2017-11-16 21:38:06 -05:00
for (i = 0; i < a3; i++)
2017-11-10 22:54:40 -05:00
{
sp[i] = a0 % 10;
a0 /= 10;
}
2017-11-16 21:38:06 -05:00
for (i = a3 - 1; i >= 0; i--)
2017-11-10 22:54:40 -05:00
{
2017-11-11 00:05:44 -05:00
LinkTest_prnthexchar(sp[i], a1, a2);
2017-11-16 21:38:06 -05:00
a1++;
2017-11-10 22:54:40 -05:00
}
if (sp10 != -1)
{
2017-11-11 00:05:44 -05:00
LinkTest_prnthexchar(*"\n", sp10, a2);
2017-11-10 23:12:21 -05:00
}
}
2017-11-11 00:05:44 -05:00
static void LinkTest_prntstr(const char *a0, u8 a1, u8 a2)
2017-11-10 23:12:21 -05:00
{
int r6;
int i;
int r5;
r5 = 0;
r6 = 0;
2017-11-16 21:38:06 -05:00
for (i = 0; a0[i] != 0; a0++)
2017-11-10 23:12:21 -05:00
{
if (a0[i] == *"\n")
{
2017-11-16 21:38:06 -05:00
r5++;
2017-11-10 23:12:21 -05:00
r6 = 0;
}
else
{
2017-11-11 00:05:44 -05:00
LinkTest_prntchar(a0[i], a1 + r6, a2 + r5);
2017-11-16 21:38:06 -05:00
r6++;
2017-11-10 23:12:21 -05:00
}
2017-11-10 22:54:40 -05:00
}
}
2017-11-11 00:05:44 -05:00
static void LinkCB_RequestPlayerDataExchange(void)
{
if (gLinkStatus & LINK_STAT_MASTER)
{
2017-11-14 08:31:37 -05:00
BuildSendCmd(LINKCMD_SEND_LINK_TYPE);
2017-11-11 00:05:44 -05:00
}
gLinkCallback = NULL;
}
2017-11-11 00:25:05 -05:00
static void Task_PrintTestData(u8 taskId)
2017-11-11 00:05:44 -05:00
{
char sp[32];
int i;
2020-06-06 16:46:19 -04:00
strcpy(sp, sASCIITestPrint);
2017-11-11 00:05:44 -05:00
LinkTest_prntstr(sp, 5, 2);
LinkTest_prnthex(gShouldAdvanceLinkState, 2, 1, 2);
LinkTest_prnthex(gLinkStatus, 15, 1, 8);
LinkTest_prnthex(gLink.state, 2, 10, 2);
LinkTest_prnthex(EXTRACT_PLAYER_COUNT(gLinkStatus), 15, 10, 2);
LinkTest_prnthex(GetMultiplayerId(), 15, 12, 2);
2017-11-12 23:16:51 -05:00
LinkTest_prnthex(gLastSendQueueCount, 25, 1, 2);
LinkTest_prnthex(gLastRecvQueueCount, 25, 2, 2);
2017-11-11 00:05:44 -05:00
LinkTest_prnthex(GetBlockReceivedStatus(), 15, 5, 2);
2017-11-11 00:25:05 -05:00
LinkTest_prnthex(gLinkDebugSeed, 2, 12, 8);
LinkTest_prnthex(gLinkDebugFlags, 2, 13, 8);
2017-11-13 01:46:22 -05:00
LinkTest_prnthex(GetSioMultiSI(), 25, 5, 1);
LinkTest_prnthex(IsSioMultiMaster(), 25, 6, 1);
LinkTest_prnthex(IsLinkConnectionEstablished(), 25, 7, 1);
LinkTest_prnthex(HasLinkErrorOccurred(), 25, 8, 1);
2017-11-11 00:05:44 -05:00
for (i = 0; i < MAX_LINK_PLAYERS; i++)
{
LinkTest_prnthex(gLinkTestBlockChecksums[i], 10, 4 + i, 4);
}
}
2017-11-11 00:25:05 -05:00
void SetLinkDebugValues(u32 seed, u32 flags)
{
gLinkDebugSeed = seed;
gLinkDebugFlags = flags;
}
2017-11-11 09:56:47 -05:00
u8 GetSavedLinkPlayerCountAsBitFlags(void)
2017-11-11 09:56:47 -05:00
{
int i;
u8 flags;
flags = 0;
2017-11-16 21:38:06 -05:00
for (i = 0; i < gSavedLinkPlayerCount; i++)
2017-11-11 09:56:47 -05:00
{
flags |= (1 << i);
}
return flags;
}
u8 GetLinkPlayerCountAsBitFlags(void)
2017-11-11 09:56:47 -05:00
{
int i;
u8 flags;
flags = 0;
2017-11-16 21:38:06 -05:00
for (i = 0; i < GetLinkPlayerCount(); i++)
2017-11-11 09:56:47 -05:00
{
flags |= (1 << i);
}
return flags;
}
void SaveLinkPlayers(u8 playerCount)
2017-11-11 09:56:47 -05:00
{
int i;
gSavedLinkPlayerCount = playerCount;
2017-11-12 21:31:08 -05:00
gSavedMultiplayerId = GetMultiplayerId();
2017-11-16 21:38:06 -05:00
for (i = 0; i < MAX_RFU_PLAYERS; i++)
2017-11-11 09:56:47 -05:00
{
2017-11-12 20:57:21 -05:00
gSavedLinkPlayers[i] = gLinkPlayers[i];
2017-11-11 09:56:47 -05:00
}
}
// The number of players when trading began. This is frequently compared against the
// current number of connected players to check if anyone dropped out.
u8 GetSavedPlayerCount(void)
2017-11-11 09:56:47 -05:00
{
2017-11-12 20:57:21 -05:00
return gSavedLinkPlayerCount;
2017-11-11 09:56:47 -05:00
}
// Unused
static u8 GetSavedMultiplayerId(void)
2017-11-11 09:56:47 -05:00
{
2017-11-12 21:31:08 -05:00
return gSavedMultiplayerId;
2017-11-11 09:56:47 -05:00
}
2017-11-12 20:57:21 -05:00
bool8 DoesLinkPlayerCountMatchSaved(void)
2017-11-12 20:57:21 -05:00
{
int i;
u32 count = 0;
2017-11-12 20:57:21 -05:00
2017-11-16 21:38:06 -05:00
for (i = 0; i < gSavedLinkPlayerCount; i++)
2017-11-12 20:57:21 -05:00
{
if (gLinkPlayers[i].trainerId == gSavedLinkPlayers[i].trainerId)
{
if (gLinkType == LINKTYPE_BATTLE_TOWER)
2017-11-12 20:57:21 -05:00
{
if (gLinkType == gLinkPlayers[i].linkType)
2017-11-16 21:38:06 -05:00
count++;
2017-11-12 20:57:21 -05:00
}
else
{
2017-11-16 21:38:06 -05:00
count++;
2017-11-12 20:57:21 -05:00
}
}
}
if (count == gSavedLinkPlayerCount)
{
2017-11-12 21:31:08 -05:00
if (GetLinkPlayerCount_2() == gSavedLinkPlayerCount)
2017-11-12 20:57:21 -05:00
{
return TRUE;
}
}
return FALSE;
}
2017-11-12 21:26:11 -05:00
void ClearSavedLinkPlayers(void)
2017-11-12 21:26:11 -05:00
{
int i;
// Clearly not what was meant to be written, but here it is anyway.
2017-11-16 21:38:06 -05:00
for (i = 0; i < 4; i++)
2017-11-12 21:26:11 -05:00
{
CpuSet(&gSavedLinkPlayers[i], NULL, sizeof(struct LinkPlayer));
}
}
void CheckLinkPlayersMatchSaved(void)
2017-11-12 21:26:11 -05:00
{
u8 i;
2017-11-16 21:38:06 -05:00
for (i = 0; i < gSavedLinkPlayerCount; i++)
2017-11-12 21:26:11 -05:00
{
if (gSavedLinkPlayers[i].trainerId != gLinkPlayers[i].trainerId
|| StringCompare(gSavedLinkPlayers[i].name, gLinkPlayers[i].name) != 0)
2017-11-12 21:26:11 -05:00
{
gLinkErrorOccurred = TRUE;
CloseLink();
SetMainCallback2(CB2_LinkError);
}
}
}
2017-11-12 21:31:08 -05:00
void ResetLinkPlayerCount(void)
2017-11-12 21:31:08 -05:00
{
gSavedLinkPlayerCount = 0;
gSavedMultiplayerId = 0;
}
u8 GetLinkPlayerCount_2(void)
{
return EXTRACT_PLAYER_COUNT(gLinkStatus);
}
2017-11-12 22:09:11 -05:00
bool8 IsLinkMaster(void)
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType)
2017-11-12 22:09:11 -05:00
{
return Rfu_IsMaster();
}
return EXTRACT_MASTER(gLinkStatus);
}
// Unused
static u8 GetDummy2(void)
2017-11-12 22:09:11 -05:00
{
return sDummy2;
2017-11-12 22:09:11 -05:00
}
2020-08-13 03:09:47 -04:00
void SetCloseLinkCallbackAndType(u16 type)
2017-11-12 22:09:11 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-12 22:09:11 -05:00
{
2020-08-13 03:09:47 -04:00
Rfu_SetCloseLinkCallback();
2017-11-12 22:09:11 -05:00
}
else
{
if (gLinkCallback == NULL)
{
2020-08-13 03:09:47 -04:00
gLinkCallback = LinkCB_ReadyCloseLink;
gLinkDummy1 = FALSE;
2020-08-13 03:09:47 -04:00
gReadyCloseLinkType = type;
2017-11-12 22:09:11 -05:00
}
}
}
2020-08-13 03:09:47 -04:00
void SetCloseLinkCallback(void)
2017-11-12 22:09:11 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-12 22:09:11 -05:00
{
2020-08-13 03:09:47 -04:00
Rfu_SetCloseLinkCallback();
2017-11-12 22:09:11 -05:00
}
else
{
if (gLinkCallback != NULL)
{
2020-08-13 03:09:47 -04:00
sReadyCloseLinkAttempts++;
2017-11-12 22:09:11 -05:00
}
else
{
2020-08-13 03:09:47 -04:00
gLinkCallback = LinkCB_ReadyCloseLink;
gLinkDummy1 = FALSE;
2020-08-13 03:09:47 -04:00
gReadyCloseLinkType = 0;
2017-11-12 22:09:11 -05:00
}
}
}
2017-11-12 22:24:15 -05:00
2020-08-13 03:09:47 -04:00
static void LinkCB_ReadyCloseLink(void)
2017-11-12 22:24:15 -05:00
{
2017-11-12 23:16:51 -05:00
if (gLastRecvQueueCount == 0)
2017-11-12 22:24:15 -05:00
{
2020-08-13 03:09:47 -04:00
BuildSendCmd(LINKCMD_READY_CLOSE_LINK);
gLinkCallback = LinkCB_WaitCloseLink;
2017-11-12 22:24:15 -05:00
}
}
2020-08-13 03:09:47 -04:00
static void LinkCB_WaitCloseLink(void)
2017-11-12 22:24:15 -05:00
{
int i;
unsigned count;
2020-08-13 03:09:47 -04:00
// Wait for all players to be ready
u8 linkPlayerCount = GetLinkPlayerCount();
2017-11-12 22:24:15 -05:00
count = 0;
2017-11-16 21:38:06 -05:00
for (i = 0; i < linkPlayerCount; i++)
2017-11-12 22:24:15 -05:00
{
2020-08-13 03:09:47 -04:00
if (gReadyToCloseLink[i])
2017-11-16 21:38:06 -05:00
count++;
2017-11-12 22:24:15 -05:00
}
2020-08-13 03:09:47 -04:00
2017-11-12 22:24:15 -05:00
if (count == linkPlayerCount)
{
2020-08-13 03:09:47 -04:00
// All ready, close link
2017-11-12 22:24:15 -05:00
gBattleTypeFlags &= ~BATTLE_TYPE_20;
gLinkVSyncDisabled = TRUE;
CloseLink();
gLinkCallback = NULL;
gLinkDummy1 = TRUE;
2017-11-12 22:24:15 -05:00
}
}
2020-08-13 03:09:47 -04:00
// Used instead of SetCloseLinkCallback when disconnecting from an attempt to link with a foreign game
void SetCloseLinkCallbackHandleJP(void)
2017-11-12 22:24:15 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-12 22:24:15 -05:00
{
2020-08-13 03:09:47 -04:00
Rfu_SetCloseLinkCallback();
2017-11-12 22:24:15 -05:00
}
else
{
if (gLinkCallback != NULL)
{
2020-08-13 03:09:47 -04:00
sReadyCloseLinkAttempts++;
2017-11-12 22:24:15 -05:00
}
else
{
2020-08-13 03:09:47 -04:00
gLinkCallback = LinkCB_ReadyCloseLinkWithJP;
gLinkDummy1 = FALSE;
2020-08-13 03:09:47 -04:00
gReadyCloseLinkType = 0;
2017-11-12 22:24:15 -05:00
}
}
}
2017-11-12 22:30:00 -05:00
2020-08-13 03:09:47 -04:00
static void LinkCB_ReadyCloseLinkWithJP(void)
2017-11-12 22:30:00 -05:00
{
2017-11-12 23:16:51 -05:00
if (gLastRecvQueueCount == 0)
2017-11-12 22:30:00 -05:00
{
2020-08-13 03:09:47 -04:00
BuildSendCmd(LINKCMD_READY_CLOSE_LINK);
gLinkCallback = LinkCB_WaitCloseLinkWithJP;
2017-11-12 22:30:00 -05:00
}
}
2020-08-13 03:09:47 -04:00
static void LinkCB_WaitCloseLinkWithJP(void)
2017-11-12 22:30:00 -05:00
{
int i;
unsigned count;
u8 linkPlayerCount;
linkPlayerCount = GetLinkPlayerCount();
count = 0;
2020-08-13 03:09:47 -04:00
// Wait for all non-foreign players to be ready
2017-11-16 21:38:06 -05:00
for (i = 0; i < linkPlayerCount; i++)
2017-11-12 22:30:00 -05:00
{
2020-08-13 03:09:47 -04:00
// Rather than communicate with the foreign game
// just assume they're ready to disconnect
2017-11-12 22:30:00 -05:00
if (gLinkPlayers[i].language == LANGUAGE_JAPANESE)
2017-11-16 21:38:06 -05:00
count++;
2020-08-13 03:09:47 -04:00
else if (gReadyToCloseLink[i])
2017-11-16 21:38:06 -05:00
count++;
2017-11-12 22:30:00 -05:00
}
2020-08-13 03:09:47 -04:00
2017-11-12 22:30:00 -05:00
if (count == linkPlayerCount)
{
2020-08-13 03:09:47 -04:00
// All ready, close link
2017-11-12 22:30:00 -05:00
gBattleTypeFlags &= ~BATTLE_TYPE_20;
gLinkVSyncDisabled = TRUE;
CloseLink();
gLinkCallback = NULL;
gLinkDummy1 = TRUE;
2017-11-12 22:44:20 -05:00
}
}
2020-08-13 03:09:47 -04:00
void SetLinkStandbyCallback(void)
2017-11-12 22:44:20 -05:00
{
2017-11-12 23:58:05 -05:00
if (gWirelessCommType == TRUE)
2017-11-12 22:44:20 -05:00
{
2020-08-13 03:09:47 -04:00
Rfu_SetLinkStandbyCallback();
2017-11-12 22:44:20 -05:00
}
else
{
if (gLinkCallback == NULL)
{
2020-08-13 03:09:47 -04:00
gLinkCallback = LinkCB_Standby;
2017-11-12 22:44:20 -05:00
}
gLinkDummy1 = FALSE;
2017-11-12 22:44:20 -05:00
}
}
2020-08-13 03:09:47 -04:00
static void LinkCB_Standby(void)
2017-11-12 22:44:20 -05:00
{
2017-11-12 23:16:51 -05:00
if (gLastRecvQueueCount == 0)
2017-11-12 22:44:20 -05:00
{
2020-08-13 03:09:47 -04:00
BuildSendCmd(LINKCMD_READY_EXIT_STANDBY);
gLinkCallback = LinkCB_StandbyForAll;
2017-11-12 22:44:20 -05:00
}
}
2020-08-13 03:09:47 -04:00
static void LinkCB_StandbyForAll(void)
2017-11-12 22:44:20 -05:00
{
u8 i;
2020-08-13 03:09:47 -04:00
u8 linkPlayerCount = GetLinkPlayerCount();
2017-11-16 21:38:06 -05:00
for (i = 0; i < linkPlayerCount; i++)
2017-11-12 22:44:20 -05:00
{
2020-08-13 03:09:47 -04:00
if (!gReadyToExitStandby[i])
2017-11-12 22:44:20 -05:00
break;
}
2020-08-13 03:09:47 -04:00
// If true, all players ready to exit standby
2017-11-12 22:44:20 -05:00
if (i == linkPlayerCount)
{
2017-11-16 21:38:06 -05:00
for (i = 0; i < MAX_LINK_PLAYERS; i++)
2020-08-13 03:09:47 -04:00
gReadyToExitStandby[i] = FALSE;
2017-11-12 22:44:20 -05:00
gLinkCallback = NULL;
2017-11-12 22:30:00 -05:00
}
}
2017-11-12 23:16:51 -05:00
2017-11-13 01:46:22 -05:00
static void CheckErrorStatus(void)
2017-11-12 23:16:51 -05:00
{
if (gLinkOpen && EXTRACT_LINK_ERRORS(gLinkStatus))
{
if (!gSuppressLinkErrorMessage)
{
sLinkErrorBuffer.status = gLinkStatus;
sLinkErrorBuffer.lastRecvQueueCount = gLastRecvQueueCount;
sLinkErrorBuffer.lastSendQueueCount = gLastSendQueueCount;
SetMainCallback2(CB2_LinkError);
}
gLinkErrorOccurred = TRUE;
CloseLink();
}
}
void BufferLinkErrorInfo(u32 status, u8 lastSendQueueCount, u8 lastRecvQueueCount, bool8 unk_06)
2017-11-12 23:16:51 -05:00
{
sLinkErrorBuffer.status = status;
sLinkErrorBuffer.lastSendQueueCount = lastSendQueueCount;
sLinkErrorBuffer.lastRecvQueueCount = lastRecvQueueCount;
sLinkErrorBuffer.unk_06 = unk_06;
}
void CB2_LinkError(void)
{
u8 *tilemapBuffer;
SetGpuReg(REG_OFFSET_DISPCNT, 0);
2018-01-18 12:53:31 -05:00
m4aMPlayStop(&gMPlayInfo_SE1);
m4aMPlayStop(&gMPlayInfo_SE2);
m4aMPlayStop(&gMPlayInfo_SE3);
2017-11-12 23:16:51 -05:00
InitHeap(gHeap, HEAP_SIZE);
ResetSpriteData();
FreeAllSpritePalettes();
ResetPaletteFadeControl();
FillPalette(0, 0, 2);
ResetTasks();
2018-01-21 23:20:13 -05:00
ScanlineEffect_Stop();
2017-11-12 23:58:05 -05:00
if (gWirelessCommType)
2017-11-12 23:16:51 -05:00
{
if (!sLinkErrorBuffer.unk_06)
{
2017-11-12 23:58:05 -05:00
gWirelessCommType = 3;
2017-11-12 23:16:51 -05:00
}
2020-06-06 16:46:19 -04:00
ResetLinkRfuGFLayer();
2017-11-12 23:16:51 -05:00
}
SetVBlankCallback(VBlankCB_LinkError);
2017-11-12 23:16:51 -05:00
ResetBgsAndClearDma3BusyFlags(0);
2020-06-06 16:46:19 -04:00
InitBgsFromTemplates(0, sLinkErrorBgTemplates, ARRAY_COUNT(sLinkErrorBgTemplates));
sLinkErrorBgTilemapBuffer = tilemapBuffer = malloc(0x800);
2017-11-12 23:16:51 -05:00
SetBgTilemapBuffer(1, tilemapBuffer);
2020-06-06 16:46:19 -04:00
if (InitWindows(sLinkErrorWindowTemplates))
2017-11-12 23:16:51 -05:00
{
DeactivateAllTextPrinters();
2020-05-14 01:37:09 -07:00
ResetTempTileDataBuffers();
2017-11-12 23:16:51 -05:00
SetGpuReg(REG_OFFSET_BLDCNT, 0);
SetGpuReg(REG_OFFSET_BLDALPHA, 0);
SetGpuReg(REG_OFFSET_BG0HOFS, 0);
SetGpuReg(REG_OFFSET_BG0VOFS, 0);
SetGpuReg(REG_OFFSET_BG1HOFS, 0);
SetGpuReg(REG_OFFSET_BG1VOFS, 0);
ClearGpuRegBits(REG_OFFSET_DISPCNT, DISPCNT_WIN0_ON | DISPCNT_WIN1_ON | DISPCNT_OBJWIN_ON);
LoadPalette(gUnknown_0860F074, 0xf0, 0x20);
gSoftResetDisabled = FALSE;
CreateTask(Task_DestroySelf, 0);
StopMapMusic();
gMain.callback1 = NULL;
RunTasks();
AnimateSprites();
BuildOamBuffer();
UpdatePaletteFade();
2017-11-13 01:46:22 -05:00
SetMainCallback2(CB2_PrintErrorMessage);
2017-11-12 23:16:51 -05:00
}
}
2017-11-12 23:33:03 -05:00
2017-11-13 00:15:31 -05:00
static void sub_800B080(void)
2017-11-12 23:33:03 -05:00
{
2020-06-06 16:46:19 -04:00
LoadBgTiles(0, s2BlankTilesGfx, 0x20, 0);
DecompressAndLoadBgGfxUsingHeap(1, sWirelessLinkDisplayGfx, FALSE, 0, 0);
CopyToBgTilemapBuffer(1, sWirelessLinkDisplayTilemap, 0, 0);
2017-11-12 23:33:03 -05:00
CopyBgTilemapBufferToVram(1);
2020-06-06 16:46:19 -04:00
LoadPalette(sWirelessLinkDisplayPal, 0, 0x20);
FillWindowPixelBuffer(0, PIXEL_FILL(0));
FillWindowPixelBuffer(2, PIXEL_FILL(0));
2019-12-10 13:48:20 -05:00
AddTextPrinterParameterized3(0, 3, 2, 6, sTextColors, 0, gText_CommErrorEllipsis);
AddTextPrinterParameterized3(2, 3, 2, 1, sTextColors, 0, gText_MoveCloserToLinkPartner);
2017-11-12 23:33:03 -05:00
PutWindowTilemap(0);
PutWindowTilemap(2);
CopyWindowToVram(0, 0);
CopyWindowToVram(2, 3);
}
2017-11-12 23:58:05 -05:00
2017-11-13 00:15:31 -05:00
static void sub_800B138(void)
2017-11-12 23:58:05 -05:00
{
2020-06-06 16:46:19 -04:00
LoadBgTiles(0, s2BlankTilesGfx, 0x20, 0);
FillWindowPixelBuffer(1, PIXEL_FILL(0));
FillWindowPixelBuffer(2, PIXEL_FILL(0));
2019-12-10 13:48:20 -05:00
AddTextPrinterParameterized3(1, 3, 2, 0, sTextColors, 0, gText_CommErrorCheckConnections);
2017-11-12 23:58:05 -05:00
PutWindowTilemap(1);
PutWindowTilemap(2);
CopyWindowToVram(1, 0);
CopyWindowToVram(2, 3);
}
2017-11-13 01:46:22 -05:00
static void CB2_PrintErrorMessage(void)
2017-11-12 23:58:05 -05:00
{
switch (gMain.state)
{
case 00:
if (sLinkErrorBuffer.unk_06)
{
sub_800B080();
}
else
{
sub_800B138();
}
break;
case 02:
ShowBg(0);
if (sLinkErrorBuffer.unk_06)
{
ShowBg(1);
}
break;
case 30:
PlaySE(SE_BOO);
break;
case 60:
PlaySE(SE_BOO);
break;
case 90:
PlaySE(SE_BOO);
break;
case 130:
if (gWirelessCommType == 2)
{
2019-12-10 13:48:20 -05:00
AddTextPrinterParameterized3(0, 3, 2, 20, sTextColors, 0, gText_ABtnTitleScreen);
2017-11-12 23:58:05 -05:00
}
else if (gWirelessCommType == 1)
{
2019-12-10 13:48:20 -05:00
AddTextPrinterParameterized3(0, 3, 2, 20, sTextColors, 0, gText_ABtnRegistrationCounter);
2017-11-12 23:58:05 -05:00
}
break;
}
if (gMain.state == 160)
{
if (gWirelessCommType == 1)
{
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON))
2017-11-12 23:58:05 -05:00
{
PlaySE(SE_PIN);
gWirelessCommType = 0;
sLinkErrorBuffer.unk_06 = 0;
sub_81700F8();
}
}
else if (gWirelessCommType == 2)
{
2020-09-04 21:11:55 -04:00
if (JOY_NEW(A_BUTTON))
2017-11-12 23:58:05 -05:00
{
rfu_REQ_stopMode();
rfu_waitREQComplete();
DoSoftReset();
}
}
}
if (gMain.state != 160)
{
2017-11-16 21:38:06 -05:00
gMain.state++;
2017-11-12 23:58:05 -05:00
}
}
2017-11-13 00:15:31 -05:00
2017-11-16 22:13:42 -05:00
// TODO: there might be a file boundary here, let's name it
2017-11-13 01:46:22 -05:00
bool8 GetSioMultiSI(void)
2017-11-13 00:15:31 -05:00
{
return (REG_SIOCNT & 0x04) != 0;
}
2017-11-13 01:46:22 -05:00
static bool8 IsSioMultiMaster(void)
2017-11-13 00:15:31 -05:00
{
return (REG_SIOCNT & 0x8) && !(REG_SIOCNT & 0x04);
}
2017-11-13 01:46:22 -05:00
bool8 IsLinkConnectionEstablished(void)
2017-11-13 00:15:31 -05:00
{
return EXTRACT_CONN_ESTABLISHED(gLinkStatus);
}
void SetSuppressLinkErrorMessage(bool8 flag)
{
gSuppressLinkErrorMessage = flag;
}
2017-11-13 01:46:22 -05:00
bool8 HasLinkErrorOccurred(void)
2017-11-13 00:15:31 -05:00
{
return gLinkErrorOccurred;
}
2017-11-13 00:31:27 -05:00
void sub_800B348(void)
{
struct LinkPlayerBlock *block;
InitLocalLinkPlayer();
block = &gLocalLinkPlayerBlock;
block->linkPlayer = gLocalLinkPlayer;
2020-06-06 16:46:19 -04:00
memcpy(block->magic1, sASCIIGameFreakInc, sizeof(block->magic1) - 1);
memcpy(block->magic2, sASCIIGameFreakInc, sizeof(block->magic2) - 1);
2017-11-13 00:31:27 -05:00
memcpy(gBlockSendBuffer, block, sizeof(*block));
}
2020-06-06 16:46:19 -04:00
void LinkPlayerFromBlock(u32 who)
2017-11-13 00:31:27 -05:00
{
2018-01-20 16:38:44 -05:00
u8 who_ = who;
2017-11-13 00:31:27 -05:00
struct LinkPlayerBlock *block;
struct LinkPlayer *player;
2018-01-20 16:38:44 -05:00
block = (struct LinkPlayerBlock *)gBlockRecvBuffer[who_];
player = &gLinkPlayers[who_];
2017-11-13 00:31:27 -05:00
*player = block->linkPlayer;
sub_800B524(player);
2020-06-06 16:46:19 -04:00
if (strcmp(block->magic1, sASCIIGameFreakInc) != 0 || strcmp(block->magic2, sASCIIGameFreakInc) != 0)
2017-11-13 00:31:27 -05:00
{
SetMainCallback2(CB2_LinkError);
}
}
2017-11-13 01:01:27 -05:00
bool8 HandleLinkConnection(void)
{
bool32 r4;
bool32 r5;
if (gWirelessCommType == 0)
{
2017-11-13 01:46:22 -05:00
gLinkStatus = LinkMain1(&gShouldAdvanceLinkState, gSendCmd, gRecvCmds);
2017-11-13 01:01:27 -05:00
LinkMain2(&gMain.heldKeys);
if ((gLinkStatus & LINK_STAT_RECEIVED_NOTHING) && sub_808766C() == TRUE)
{
return TRUE;
}
}
else
{
r4 = sub_8010EC0();
r5 = sub_8010F1C();
if (sub_808766C() == TRUE)
{
if (r4 == TRUE || IsRfuRecvQueueEmpty() || r5)
2017-11-13 01:01:27 -05:00
{
return TRUE;
}
}
}
return FALSE;
}
2017-11-13 01:04:31 -05:00
2020-05-30 04:09:21 -04:00
void SetWirelessCommType1(void)
2017-11-13 01:04:31 -05:00
{
if (gReceivedRemoteLinkPlayers == 0)
{
gWirelessCommType = 1;
}
}
2020-06-06 16:46:19 -04:00
static void SetWirelessCommType0_Internal(void)
2017-11-13 01:04:31 -05:00
{
if (gReceivedRemoteLinkPlayers == 0)
{
gWirelessCommType = 0;
}
}
2020-06-06 16:46:19 -04:00
void SetWirelessCommType0(void)
2017-11-13 01:04:31 -05:00
{
if (gReceivedRemoteLinkPlayers == 0)
{
gWirelessCommType = 0;
}
}
2017-11-13 01:12:34 -05:00
u32 GetLinkRecvQueueLength(void)
2017-11-13 01:12:34 -05:00
{
if (gWirelessCommType != 0)
{
return GetRfuRecvQueueLength();
2017-11-13 01:12:34 -05:00
}
return gLink.recvQueue.count;
}
2019-03-02 02:44:02 -05:00
bool32 sub_800B504(void)
2017-11-13 01:12:34 -05:00
{
if (GetLinkRecvQueueLength() > 2)
2017-11-13 01:12:34 -05:00
{
return TRUE;
}
return FALSE;
}
// Unused
u8 GetWirelessCommType(void)
2017-11-13 01:12:34 -05:00
{
return gWirelessCommType;
}
void sub_800B524(struct LinkPlayer *player)
{
2019-10-09 05:56:44 -04:00
player->progressFlagsCopy = player->progressFlags;
2017-11-13 01:12:34 -05:00
ConvertInternationalString(player->name, player->language);
}
2017-11-13 01:20:38 -05:00
2017-11-14 08:31:37 -05:00
static void DisableSerial(void)
2017-11-13 01:20:38 -05:00
{
DisableInterrupts(INTR_FLAG_TIMER3 | INTR_FLAG_SERIAL);
REG_SIOCNT = SIO_MULTI_MODE;
REG_TMCNT_H(3) = 0;
REG_IF = INTR_FLAG_TIMER3 | INTR_FLAG_SERIAL;
REG_SIOMLT_SEND = 0;
REG_SIOMLT_RECV = 0;
CpuFill32(0, &gLink, sizeof(gLink));
}
2017-11-13 01:24:28 -05:00
2017-11-14 08:31:37 -05:00
static void EnableSerial(void)
2017-11-13 01:24:28 -05:00
{
DisableInterrupts(INTR_FLAG_TIMER3 | INTR_FLAG_SERIAL);
REG_RCNT = 0;
REG_SIOCNT = SIO_MULTI_MODE;
REG_SIOCNT |= SIO_115200_BPS | SIO_INTR_ENABLE;
EnableInterrupts(INTR_FLAG_SERIAL);
REG_SIOMLT_SEND = 0;
CpuFill32(0, &gLink, sizeof(gLink));
2017-11-14 08:44:32 -05:00
sNumVBlanksWithoutSerialIntr = 0;
2017-11-14 08:31:37 -05:00
sSendNonzeroCheck = 0;
2017-11-16 21:38:06 -05:00
sRecvNonzeroCheck = 0;
sChecksumAvailable = 0;
2017-11-15 08:44:11 -05:00
sHandshakePlayerCount = 0;
2017-11-13 01:24:28 -05:00
gLastSendQueueCount = 0;
gLastRecvQueueCount = 0;
}
void ResetSerial(void)
{
EnableSerial();
DisableSerial();
}
2017-11-13 01:46:22 -05:00
2017-11-16 22:13:42 -05:00
// link_main1.c
2017-11-13 01:46:22 -05:00
u32 LinkMain1(u8 *shouldAdvanceLinkState, u16 *sendCmd, u16 (*recvCmds)[CMD_LENGTH])
{
u32 retVal;
u32 retVal2;
switch (gLink.state)
{
case LINK_STATE_START0:
DisableSerial();
gLink.state = 1;
break;
case LINK_STATE_START1:
if (*shouldAdvanceLinkState == 1)
{
EnableSerial();
gLink.state = 2;
}
break;
case LINK_STATE_HANDSHAKE:
switch (*shouldAdvanceLinkState)
{
default:
2017-11-14 08:31:37 -05:00
CheckMasterOrSlave();
2017-11-13 01:46:22 -05:00
break;
case 1:
2017-11-14 08:31:37 -05:00
if (gLink.isMaster == LINK_MASTER && gLink.playerCount > 1)
2017-11-13 01:46:22 -05:00
{
gLink.handshakeAsMaster = TRUE;
}
break;
case 2:
gLink.state = LINK_STATE_START0;
REG_SIOMLT_SEND = 0;
break;
}
break;
case LINK_STATE_INIT_TIMER:
2017-11-14 08:31:37 -05:00
InitTimer();
2017-11-13 01:46:22 -05:00
gLink.state = LINK_STATE_CONN_ESTABLISHED;
// fallthrough
case LINK_STATE_CONN_ESTABLISHED:
2017-11-14 08:31:37 -05:00
EnqueueSendCmd(sendCmd);
DequeueRecvCmds(recvCmds);
2017-11-13 01:46:22 -05:00
break;
}
*shouldAdvanceLinkState = 0;
retVal = gLink.localId;
2017-11-14 08:31:37 -05:00
retVal |= (gLink.playerCount << LINK_STAT_PLAYER_COUNT_SHIFT);
if (gLink.isMaster == LINK_MASTER)
2017-11-13 01:46:22 -05:00
{
2017-11-14 08:31:37 -05:00
retVal |= LINK_STAT_MASTER;
2017-11-13 01:46:22 -05:00
}
{
2017-11-14 08:31:37 -05:00
u32 receivedNothing = gLink.receivedNothing << LINK_STAT_RECEIVED_NOTHING_SHIFT;
u32 link_field_F = gLink.link_field_F << LINK_STAT_UNK_FLAG_9_SHIFT;
u32 hardwareError = gLink.hardwareError << LINK_STAT_ERROR_HARDWARE_SHIFT;
u32 badChecksum = gLink.badChecksum << LINK_STAT_ERROR_CHECKSUM_SHIFT;
u32 queueFull = gLink.queueFull << LINK_STAT_ERROR_QUEUE_FULL_SHIFT;
2017-11-13 01:46:22 -05:00
u32 val;
if (gLink.state == LINK_STATE_CONN_ESTABLISHED)
{
2017-11-14 08:31:37 -05:00
val = LINK_STAT_CONN_ESTABLISHED;
2017-11-13 01:46:22 -05:00
val |= receivedNothing;
val |= retVal;
val |= link_field_F;
val |= hardwareError;
val |= badChecksum;
val |= queueFull;
}
else
{
val = retVal;
val |= receivedNothing;
val |= link_field_F;
val |= hardwareError;
val |= badChecksum;
val |= queueFull;
}
retVal = val;
}
if (gLink.lag == LAG_MASTER)
2017-11-14 08:31:37 -05:00
{
retVal |= LINK_STAT_ERROR_LAG_MASTER;
}
2017-11-13 01:46:22 -05:00
2017-11-14 08:31:37 -05:00
if (gLink.localId >= MAX_LINK_PLAYERS)
{
retVal |= LINK_STAT_ERROR_INVALID_ID;
}
2017-11-13 01:46:22 -05:00
retVal2 = retVal;
if (gLink.lag == LAG_SLAVE)
2017-11-14 08:31:37 -05:00
{
retVal2 |= LINK_STAT_ERROR_LAG_SLAVE;
}
2017-11-13 01:46:22 -05:00
return retVal2;
}
2017-11-14 08:31:37 -05:00
static void CheckMasterOrSlave(void)
{
u32 terminals;
terminals = *(vu32 *)REG_ADDR_SIOCNT & (SIO_MULTI_SD | SIO_MULTI_SI);
if (terminals == SIO_MULTI_SD && gLink.localId == 0)
{
gLink.isMaster = LINK_MASTER;
}
else
{
gLink.isMaster = LINK_SLAVE;
}
}
static void InitTimer(void)
{
if (gLink.isMaster)
{
2017-11-16 21:38:06 -05:00
REG_TM3CNT_L = -197;
2017-11-14 08:31:37 -05:00
REG_TM3CNT_H = TIMER_64CLK | TIMER_INTR_ENABLE;
EnableInterrupts(INTR_FLAG_TIMER3);
}
}
static void EnqueueSendCmd(u16 *sendCmd)
{
u8 i;
u8 offset;
gLinkSavedIme = REG_IME;
REG_IME = 0;
if (gLink.sendQueue.count < QUEUE_CAPACITY)
{
offset = gLink.sendQueue.pos + gLink.sendQueue.count;
if (offset >= QUEUE_CAPACITY)
{
offset -= QUEUE_CAPACITY;
}
2017-11-16 21:38:06 -05:00
for (i = 0; i < CMD_LENGTH; i++)
2017-11-14 08:31:37 -05:00
{
sSendNonzeroCheck |= *sendCmd;
gLink.sendQueue.data[i][offset] = *sendCmd;
*sendCmd = 0;
sendCmd++;
}
}
else
{
gLink.queueFull = QUEUE_FULL_SEND;
}
if (sSendNonzeroCheck)
{
2017-11-16 21:38:06 -05:00
gLink.sendQueue.count++;
2017-11-14 08:31:37 -05:00
sSendNonzeroCheck = 0;
}
REG_IME = gLinkSavedIme;
gLastSendQueueCount = gLink.sendQueue.count;
}
static void DequeueRecvCmds(u16 (*recvCmds)[CMD_LENGTH])
{
u8 i;
u8 j;
gLinkSavedIme = REG_IME;
REG_IME = 0;
if (gLink.recvQueue.count == 0)
{
2017-11-16 21:38:06 -05:00
for (i = 0; i < gLink.playerCount; i++)
2017-11-14 08:31:37 -05:00
{
2017-11-16 21:38:06 -05:00
for (j = 0; j < CMD_LENGTH; j++)
2017-11-14 08:31:37 -05:00
{
recvCmds[i][j] = 0;
}
}
gLink.receivedNothing = TRUE;
}
else
{
2017-11-16 21:38:06 -05:00
for (i = 0; i < gLink.playerCount; i++)
2017-11-14 08:31:37 -05:00
{
2017-11-16 21:38:06 -05:00
for (j = 0; j < CMD_LENGTH; j++)
2017-11-14 08:31:37 -05:00
{
recvCmds[i][j] = gLink.recvQueue.data[i][j][gLink.recvQueue.pos];
}
}
2017-11-16 21:38:06 -05:00
gLink.recvQueue.count--;
gLink.recvQueue.pos++;
2017-11-14 08:31:37 -05:00
if (gLink.recvQueue.pos >= QUEUE_CAPACITY)
{
gLink.recvQueue.pos = 0;
}
gLink.receivedNothing = FALSE;
}
REG_IME = gLinkSavedIme;
}
2017-11-16 21:38:06 -05:00
// link_intr.c
2017-11-14 08:44:32 -05:00
void LinkVSync(void)
{
if (gLink.isMaster)
{
switch (gLink.state)
{
case LINK_STATE_CONN_ESTABLISHED:
if (gLink.serialIntrCounter < 9)
{
if (gLink.hardwareError != TRUE)
{
gLink.lag = LAG_MASTER;
}
else
{
StartTransfer();
}
}
else if (gLink.lag != LAG_MASTER)
{
gLink.serialIntrCounter = 0;
StartTransfer();
}
break;
case LINK_STATE_HANDSHAKE:
StartTransfer();
break;
}
}
else if (gLink.state == LINK_STATE_CONN_ESTABLISHED || gLink.state == LINK_STATE_HANDSHAKE)
{
2017-11-16 21:38:06 -05:00
if (++sNumVBlanksWithoutSerialIntr > 10)
2017-11-14 08:44:32 -05:00
{
if (gLink.state == LINK_STATE_CONN_ESTABLISHED)
{
gLink.lag = LAG_SLAVE;
}
if (gLink.state == LINK_STATE_HANDSHAKE)
{
gLink.playerCount = 0;
gLink.link_field_F = FALSE;
}
}
}
}
2017-11-15 08:44:11 -05:00
void Timer3Intr(void)
{
StopTimer();
StartTransfer();
}
void SerialCB(void)
{
gLink.localId = SIO_MULTI_CNT->id;
switch (gLink.state)
{
case LINK_STATE_CONN_ESTABLISHED:
gLink.hardwareError = SIO_MULTI_CNT->error;
DoRecv();
DoSend();
SendRecvDone();
break;
case LINK_STATE_HANDSHAKE:
if (DoHandshake())
{
if (gLink.isMaster)
{
gLink.state = LINK_STATE_INIT_TIMER;
gLink.serialIntrCounter = 8;
}
else
{
gLink.state = LINK_STATE_CONN_ESTABLISHED;
}
}
break;
}
2017-11-16 21:38:06 -05:00
gLink.serialIntrCounter++;
2017-11-15 08:44:11 -05:00
sNumVBlanksWithoutSerialIntr = 0;
if (gLink.serialIntrCounter == 8)
{
gLastRecvQueueCount = gLink.recvQueue.count;
}
}
static void StartTransfer(void)
{
REG_SIOCNT |= SIO_START;
}
static bool8 DoHandshake(void)
{
u8 i;
u8 playerCount;
u16 minRecv;
playerCount = 0;
minRecv = 0xFFFF;
2017-11-15 08:44:11 -05:00
if (gLink.handshakeAsMaster == TRUE)
{
REG_SIOMLT_SEND = MASTER_HANDSHAKE;
}
else
{
REG_SIOMLT_SEND = SLAVE_HANDSHAKE;
}
*(u64 *)gLink.tempRecvBuffer = REG_SIOMLT_RECV;
REG_SIOMLT_RECV = 0;
gLink.handshakeAsMaster = FALSE;
2017-11-16 21:38:06 -05:00
for (i = 0; i < 4; i++)
2017-11-15 08:44:11 -05:00
{
if ((gLink.tempRecvBuffer[i] & ~0x3) == SLAVE_HANDSHAKE || gLink.tempRecvBuffer[i] == MASTER_HANDSHAKE)
{
2017-11-16 21:38:06 -05:00
playerCount++;
2017-11-15 08:44:11 -05:00
if (minRecv > gLink.tempRecvBuffer[i] && gLink.tempRecvBuffer[i] != 0)
{
minRecv = gLink.tempRecvBuffer[i];
}
}
else
{
if (gLink.tempRecvBuffer[i] != 0xFFFF)
2017-11-15 08:44:11 -05:00
{
playerCount = 0;
}
break;
}
}
gLink.playerCount = playerCount;
if (gLink.playerCount > 1 && gLink.playerCount == sHandshakePlayerCount && gLink.tempRecvBuffer[0] == MASTER_HANDSHAKE)
{
return TRUE;
}
if (gLink.playerCount > 1)
{
gLink.link_field_F = (minRecv & 3) + 1;
}
else
{
gLink.link_field_F = 0;
}
sHandshakePlayerCount = gLink.playerCount;
return FALSE;
}
2017-11-16 21:38:06 -05:00
static void DoRecv(void)
{
u16 recv[4];
u8 i;
u8 index;
*(u64 *)recv = REG_SIOMLT_RECV;
if (gLink.sendCmdIndex == 0)
{
for (i = 0; i < gLink.playerCount; i++)
{
if (gLink.checksum != recv[i] && sChecksumAvailable)
{
gLink.badChecksum = TRUE;
}
}
gLink.checksum = 0;
sChecksumAvailable = TRUE;
}
else
{
index = gLink.recvQueue.pos + gLink.recvQueue.count;
if (index >= QUEUE_CAPACITY)
{
index -= QUEUE_CAPACITY;
}
if (gLink.recvQueue.count < QUEUE_CAPACITY)
{
for (i = 0; i < gLink.playerCount; i++)
{
gLink.checksum += recv[i];
sRecvNonzeroCheck |= recv[i];
gLink.recvQueue.data[i][gLink.recvCmdIndex][index] = recv[i];
}
}
else
{
gLink.queueFull = QUEUE_FULL_RECV;
}
gLink.recvCmdIndex++;
if (gLink.recvCmdIndex == CMD_LENGTH && sRecvNonzeroCheck)
{
gLink.recvQueue.count++;
sRecvNonzeroCheck = 0;
}
}
}
static void DoSend(void)
{
if (gLink.sendCmdIndex == CMD_LENGTH)
{
REG_SIOMLT_SEND = gLink.checksum;
if (!sSendBufferEmpty)
{
gLink.sendQueue.count--;
gLink.sendQueue.pos++;
if (gLink.sendQueue.pos >= QUEUE_CAPACITY)
{
gLink.sendQueue.pos = 0;
}
}
else
{
sSendBufferEmpty = FALSE;
}
}
else
{
if (!sSendBufferEmpty && gLink.sendQueue.count == 0)
{
sSendBufferEmpty = TRUE;
}
if (sSendBufferEmpty)
{
REG_SIOMLT_SEND = 0;
}
else
{
REG_SIOMLT_SEND = gLink.sendQueue.data[gLink.sendCmdIndex][gLink.sendQueue.pos];
}
gLink.sendCmdIndex++;
}
}
static void StopTimer(void)
{
if (gLink.isMaster)
{
REG_TM3CNT_H &= ~TIMER_ENABLE;
REG_TM3CNT_L = -197;
}
}
static void SendRecvDone(void)
{
if (gLink.recvCmdIndex == CMD_LENGTH)
{
gLink.sendCmdIndex = 0;
gLink.recvCmdIndex = 0;
}
else if (gLink.isMaster)
{
REG_TM3CNT_H |= TIMER_ENABLE;
}
}
2017-11-16 22:13:42 -05:00
void ResetSendBuffer(void)
{
u8 i;
u8 j;
gLink.sendQueue.count = 0;
gLink.sendQueue.pos = 0;
for (i = 0; i < CMD_LENGTH; i++)
{
for (j = 0; j < QUEUE_CAPACITY; j++)
{
gLink.sendQueue.data[i][j] = 0xEFFF;
}
}
}
void ResetRecvBuffer(void)
{
u8 i;
u8 j;
u8 k;
gLink.recvQueue.count = 0;
gLink.recvQueue.pos = 0;
for (i = 0; i < MAX_LINK_PLAYERS; i++)
{
for (j = 0; j < CMD_LENGTH; j++)
{
for (k = 0; k < QUEUE_CAPACITY; k++)
{
gLink.recvQueue.data[i][j][k] = 0xEFFF;
}
}
}
}