mirror of
https://github.com/afska/gba-link-connection.git
synced 2026-03-21 17:44:21 -05:00
130 lines
3.5 KiB
C++
130 lines
3.5 KiB
C++
// (0) Include the header
|
|
#include "../../../lib/LinkSPI.hpp"
|
|
|
|
#include <tonc.h>
|
|
#include <string>
|
|
#include "../../_lib/interrupt.h"
|
|
|
|
void log(std::string text);
|
|
void wait(u32 verticalLines);
|
|
inline void VBLANK() {}
|
|
|
|
// (1) Create a LinkSPI instance
|
|
LinkSPI* linkSPI = new LinkSPI();
|
|
|
|
void init() {
|
|
REG_DISPCNT = DCNT_MODE0 | DCNT_BG0;
|
|
tte_init_se_default(0, BG_CBB(0) | BG_SBB(31));
|
|
|
|
// (2) Add the interrupt service routines
|
|
interrupt_init();
|
|
interrupt_set_handler(INTR_VBLANK, VBLANK);
|
|
interrupt_enable(INTR_VBLANK);
|
|
interrupt_set_handler(INTR_SERIAL, LINK_SPI_ISR_SERIAL);
|
|
interrupt_enable(INTR_SERIAL);
|
|
}
|
|
|
|
int main() {
|
|
init();
|
|
|
|
bool firstTransfer = false;
|
|
bool async = false;
|
|
u32 counter = 0;
|
|
|
|
while (true) {
|
|
std::string output = "LinkSPI_demo (v7.0.0)\n\n";
|
|
u16 keys = ~REG_KEYS & KEY_ANY;
|
|
|
|
if (!linkSPI->isActive()) {
|
|
firstTransfer = true;
|
|
output += "START: Set as Master\n";
|
|
output += "SELECT: Set as Slave\n";
|
|
output += "\n(stop: press L+R)\n";
|
|
output += "(hold A on start for async)\n";
|
|
output += "(hold B on start for waitMode)\n";
|
|
output += "(hold UP for 8-bit mode)\n";
|
|
output +=
|
|
"\n\n\n\n\n\n\n\n\n[!] to test this demo...\n "
|
|
"...use a GBC Link Cable!";
|
|
|
|
if ((keys & KEY_START) | (keys & KEY_SELECT)) {
|
|
// (3) Initialize the library
|
|
linkSPI->activate((keys & KEY_START) ? LinkSPI::Mode::MASTER_256KBPS
|
|
: LinkSPI::Mode::SLAVE,
|
|
(keys & KEY_UP) ? LinkSPI::DataSize::SIZE_8BIT
|
|
: LinkSPI::DataSize::SIZE_32BIT);
|
|
linkSPI->setWaitModeActive(keys &
|
|
KEY_B); // see `waitMode` in README.md
|
|
if (keys & KEY_A)
|
|
async = true;
|
|
}
|
|
} else {
|
|
// Title
|
|
auto modeName =
|
|
linkSPI->getMode() == LinkSPI::Mode::SLAVE ? "[slave]" : "[master]";
|
|
output += std::string(modeName) + "\n";
|
|
if (firstTransfer) {
|
|
log(output + "Waiting...");
|
|
firstTransfer = false;
|
|
}
|
|
|
|
if (!async) {
|
|
// (4)/(5) Exchange 32-bit data with the other end
|
|
u32 remoteKeys = linkSPI->transfer(keys, []() {
|
|
u16 keys = ~REG_KEYS & KEY_ANY;
|
|
return (keys & KEY_L) && (keys & KEY_R);
|
|
});
|
|
output += "> " + std::to_string(keys) + "\n";
|
|
output += "< " + std::to_string(remoteKeys) + "\n";
|
|
} else {
|
|
// (6) Exchange data asynchronously
|
|
if (keys != 0 &&
|
|
linkSPI->getAsyncState() == LinkSPI::AsyncState::IDLE) {
|
|
counter++;
|
|
linkSPI->transferAsync(counter, []() {
|
|
u16 keys = ~REG_KEYS & KEY_ANY;
|
|
return (keys & KEY_L) && (keys & KEY_R);
|
|
});
|
|
log(output + ">> " + std::to_string(counter));
|
|
wait(228 * 60);
|
|
}
|
|
if (linkSPI->getAsyncState() == LinkSPI::AsyncState::READY) {
|
|
log(output + "<< " + std::to_string(linkSPI->getAsyncData()));
|
|
wait(228 * 60);
|
|
}
|
|
}
|
|
|
|
// Cancel
|
|
if ((keys & KEY_L) && (keys & KEY_R)) {
|
|
linkSPI->deactivate();
|
|
async = false;
|
|
counter = 0;
|
|
}
|
|
}
|
|
|
|
// Print
|
|
VBlankIntrWait();
|
|
log(output);
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void log(std::string text) {
|
|
tte_erase_screen();
|
|
tte_write("#{P:0,0}");
|
|
tte_write(text.c_str());
|
|
}
|
|
|
|
void wait(u32 verticalLines) {
|
|
u32 count = 0;
|
|
u32 vCount = REG_VCOUNT;
|
|
|
|
while (count < verticalLines) {
|
|
if (REG_VCOUNT != vCount) {
|
|
count++;
|
|
vCount = REG_VCOUNT;
|
|
}
|
|
};
|
|
}
|