mirror of https://github.com/mgba-emu/mgba.git
GB MBC: Start work on TAMA6 RTC pages
This commit is contained in:
parent
1985713c3a
commit
0b50e7163e
|
@ -50,17 +50,12 @@ struct GBMBCHuC3SaveBuffer {
|
||||||
uint64_t latchedUnix;
|
uint64_t latchedUnix;
|
||||||
};
|
};
|
||||||
|
|
||||||
DECL_BITFIELD(GBMBCTAMA5SaveFlags, uint32_t);
|
|
||||||
DECL_BIT(GBMBCTAMA5SaveFlags, Disabled, 0);
|
|
||||||
|
|
||||||
struct GBMBCTAMA5SaveBuffer {
|
struct GBMBCTAMA5SaveBuffer {
|
||||||
uint8_t rtcTimerPage[0x8];
|
uint8_t rtcTimerPage[0x8];
|
||||||
uint8_t rtcAlarmPage[0x8];
|
uint8_t rtcAlarmPage[0x8];
|
||||||
uint8_t rtcFreePage0[0x8];
|
uint8_t rtcFreePage0[0x8];
|
||||||
uint8_t rtcFreePage1[0x8];
|
uint8_t rtcFreePage1[0x8];
|
||||||
uint64_t latchedUnix;
|
uint64_t latchedUnix;
|
||||||
GBMBCTAMA5SaveFlags flags;
|
|
||||||
uint32_t padding;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
void GBMBCRTCRead(struct GB* gb);
|
void GBMBCRTCRead(struct GB* gb);
|
||||||
|
|
|
@ -116,6 +116,15 @@ enum GBTAMA6RTCRegister {
|
||||||
GBTAMA6_RTC_PA0_MONTH_10 = 0xA,
|
GBTAMA6_RTC_PA0_MONTH_10 = 0xA,
|
||||||
GBTAMA6_RTC_PA0_YEAR_1 = 0xB,
|
GBTAMA6_RTC_PA0_YEAR_1 = 0xB,
|
||||||
GBTAMA6_RTC_PA0_YEAR_10 = 0xC,
|
GBTAMA6_RTC_PA0_YEAR_10 = 0xC,
|
||||||
|
GBTAMA6_RTC_PA1_MINUTE_1 = 0x2,
|
||||||
|
GBTAMA6_RTC_PA1_MINUTE_10 = 0x3,
|
||||||
|
GBTAMA6_RTC_PA1_HOUR_1 = 0x4,
|
||||||
|
GBTAMA6_RTC_PA1_HOUR_10 = 0x5,
|
||||||
|
GBTAMA6_RTC_PA1_WEEK = 0x6,
|
||||||
|
GBTAMA6_RTC_PA1_DAY_1 = 0x7,
|
||||||
|
GBTAMA6_RTC_PA1_DAY_10 = 0x8,
|
||||||
|
GBTAMA6_RTC_PA1_24_HOUR = 0xA,
|
||||||
|
GBTAMA6_RTC_PA1_LEAP_YEAR = 0xB,
|
||||||
GBTAMA6_RTC_PAGE = 0xD,
|
GBTAMA6_RTC_PAGE = 0xD,
|
||||||
GBTAMA6_RTC_TEST = 0xE,
|
GBTAMA6_RTC_TEST = 0xE,
|
||||||
GBTAMA6_RTC_RESET = 0xF,
|
GBTAMA6_RTC_RESET = 0xF,
|
||||||
|
@ -211,6 +220,9 @@ struct GBTAMA5State {
|
||||||
bool disabled;
|
bool disabled;
|
||||||
uint8_t registers[GBTAMA5_MAX];
|
uint8_t registers[GBTAMA5_MAX];
|
||||||
uint8_t rtcTimerPage[GBTAMA6_RTC_MAX];
|
uint8_t rtcTimerPage[GBTAMA6_RTC_MAX];
|
||||||
|
uint8_t rtcAlarmPage[GBTAMA6_RTC_MAX];
|
||||||
|
uint8_t rtcFreePage0[GBTAMA6_RTC_MAX];
|
||||||
|
uint8_t rtcFreePage1[GBTAMA6_RTC_MAX];
|
||||||
};
|
};
|
||||||
|
|
||||||
struct GBHuC3State {
|
struct GBHuC3State {
|
||||||
|
|
71
src/gb/mbc.c
71
src/gb/mbc.c
|
@ -455,6 +455,9 @@ void GBMBCInit(struct GB* gb) {
|
||||||
case GB_TAMA5:
|
case GB_TAMA5:
|
||||||
gb->memory.mbcWrite = _GBTAMA5;
|
gb->memory.mbcWrite = _GBTAMA5;
|
||||||
gb->memory.mbcRead = _GBTAMA5Read;
|
gb->memory.mbcRead = _GBTAMA5Read;
|
||||||
|
gb->memory.mbcState.tama5.rtcAlarmPage[GBTAMA6_RTC_PAGE] = 1;
|
||||||
|
gb->memory.mbcState.tama5.rtcFreePage0[GBTAMA6_RTC_PAGE] = 2;
|
||||||
|
gb->memory.mbcState.tama5.rtcFreePage1[GBTAMA6_RTC_PAGE] = 3;
|
||||||
gb->sramSize = 0x20;
|
gb->sramSize = 0x20;
|
||||||
break;
|
break;
|
||||||
case GB_MBC3_RTC:
|
case GB_MBC3_RTC:
|
||||||
|
@ -1694,11 +1697,19 @@ void _GBTAMA5(struct GB* gb, uint16_t address, uint8_t value) {
|
||||||
switch (address) {
|
switch (address) {
|
||||||
case GBTAMA6_DISABLE_TIMER:
|
case GBTAMA6_DISABLE_TIMER:
|
||||||
tama5->disabled = true;
|
tama5->disabled = true;
|
||||||
|
tama5->rtcTimerPage[GBTAMA6_RTC_PAGE] &= 0x7;
|
||||||
|
tama5->rtcAlarmPage[GBTAMA6_RTC_PAGE] &= 0x7;
|
||||||
|
tama5->rtcFreePage0[GBTAMA6_RTC_PAGE] &= 0x7;
|
||||||
|
tama5->rtcFreePage1[GBTAMA6_RTC_PAGE] &= 0x7;
|
||||||
break;
|
break;
|
||||||
case GBTAMA6_ENABLE_TIMER:
|
case GBTAMA6_ENABLE_TIMER:
|
||||||
tama5->disabled = false;
|
tama5->disabled = false;
|
||||||
tama5->rtcTimerPage[GBTAMA6_RTC_PA0_SECOND_1] = 0;
|
tama5->rtcTimerPage[GBTAMA6_RTC_PA0_SECOND_1] = 0;
|
||||||
tama5->rtcTimerPage[GBTAMA6_RTC_PA0_SECOND_10] = 0;
|
tama5->rtcTimerPage[GBTAMA6_RTC_PA0_SECOND_10] = 0;
|
||||||
|
tama5->rtcTimerPage[GBTAMA6_RTC_PAGE] |= 0x8;
|
||||||
|
tama5->rtcAlarmPage[GBTAMA6_RTC_PAGE] |= 0x8;
|
||||||
|
tama5->rtcFreePage0[GBTAMA6_RTC_PAGE] |= 0x8;
|
||||||
|
tama5->rtcFreePage1[GBTAMA6_RTC_PAGE] |= 0x8;
|
||||||
break;
|
break;
|
||||||
case GBTAMA6_MINUTE_WRITE:
|
case GBTAMA6_MINUTE_WRITE:
|
||||||
tama5->rtcTimerPage[GBTAMA6_RTC_PA0_MINUTE_1] = out & 0xF;
|
tama5->rtcTimerPage[GBTAMA6_RTC_PA0_MINUTE_1] = out & 0xF;
|
||||||
|
@ -1711,8 +1722,19 @@ void _GBTAMA5(struct GB* gb, uint16_t address, uint8_t value) {
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case 0x4: // RTC access
|
case 0x4: // RTC access
|
||||||
if (!(address & 1)) {
|
switch (tama5->registers[GBTAMA5_ADDR_LO]) {
|
||||||
|
case 0:
|
||||||
tama5->rtcTimerPage[out & 0xF] = out >> 4;
|
tama5->rtcTimerPage[out & 0xF] = out >> 4;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
tama5->rtcAlarmPage[out & 0xF] = out >> 4;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
tama5->rtcFreePage0[out & 0xF] = out >> 4;
|
||||||
|
break;
|
||||||
|
case 6:
|
||||||
|
tama5->rtcFreePage1[out & 0xF] = out >> 4;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
@ -1774,7 +1796,20 @@ uint8_t _GBTAMA5Read(struct GBMemory* memory, uint16_t address) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
_latchTAMA6Rtc(memory->rtc, tama5, &memory->rtcLastLatch);
|
_latchTAMA6Rtc(memory->rtc, tama5, &memory->rtcLastLatch);
|
||||||
value = tama5->rtcTimerPage[tama5->registers[GBTAMA5_WRITE_LO]];
|
switch (tama5->registers[GBTAMA5_ADDR_LO]) {
|
||||||
|
case 1:
|
||||||
|
value = tama5->rtcTimerPage[tama5->registers[GBTAMA5_WRITE_LO]];
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
value = tama5->rtcTimerPage[tama5->registers[GBTAMA5_WRITE_LO]];
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
value = tama5->rtcTimerPage[tama5->registers[GBTAMA5_WRITE_LO]];
|
||||||
|
break;
|
||||||
|
case 7:
|
||||||
|
value = tama5->rtcTimerPage[tama5->registers[GBTAMA5_WRITE_LO]];
|
||||||
|
break;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
mLOG(GB_MBC, STUB, "TAMA5 unknown read %s: %02X", tama5->reg == GBTAMA5_READ_HI ? "hi" : "lo", address);
|
mLOG(GB_MBC, STUB, "TAMA5 unknown read %s: %02X", tama5->reg == GBTAMA5_READ_HI ? "hi" : "lo", address);
|
||||||
|
@ -2219,6 +2254,7 @@ void GBMBCTAMA5Read(struct GB* gb) {
|
||||||
}
|
}
|
||||||
vf->seek(vf, gb->sramSize, SEEK_SET);
|
vf->seek(vf, gb->sramSize, SEEK_SET);
|
||||||
if (vf->read(vf, &buffer, sizeof(buffer)) < (ssize_t) sizeof(buffer)) {
|
if (vf->read(vf, &buffer, sizeof(buffer)) < (ssize_t) sizeof(buffer)) {
|
||||||
|
gb->memory.mbcState.tama5.disabled = false;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2226,12 +2262,24 @@ void GBMBCTAMA5Read(struct GB* gb) {
|
||||||
for (i = 0; i < 0x8; ++i) {
|
for (i = 0; i < 0x8; ++i) {
|
||||||
gb->memory.mbcState.tama5.rtcTimerPage[i * 2] = buffer.rtcTimerPage[i] & 0xF;
|
gb->memory.mbcState.tama5.rtcTimerPage[i * 2] = buffer.rtcTimerPage[i] & 0xF;
|
||||||
gb->memory.mbcState.tama5.rtcTimerPage[i * 2 + 1] = buffer.rtcTimerPage[i] >> 4;
|
gb->memory.mbcState.tama5.rtcTimerPage[i * 2 + 1] = buffer.rtcTimerPage[i] >> 4;
|
||||||
|
gb->memory.mbcState.tama5.rtcAlarmPage[i * 2] = buffer.rtcAlarmPage[i] & 0xF;
|
||||||
|
gb->memory.mbcState.tama5.rtcAlarmPage[i * 2 + 1] = buffer.rtcAlarmPage[i] >> 4;
|
||||||
|
gb->memory.mbcState.tama5.rtcFreePage0[i * 2] = buffer.rtcFreePage0[i] & 0xF;
|
||||||
|
gb->memory.mbcState.tama5.rtcFreePage0[i * 2 + 1] = buffer.rtcFreePage0[i] >> 4;
|
||||||
|
gb->memory.mbcState.tama5.rtcFreePage1[i * 2] = buffer.rtcFreePage1[i] & 0xF;
|
||||||
|
gb->memory.mbcState.tama5.rtcFreePage1[i * 2 + 1] = buffer.rtcFreePage1[i] >> 4;
|
||||||
}
|
}
|
||||||
LOAD_64LE(gb->memory.rtcLastLatch, 0, &buffer.latchedUnix);
|
LOAD_64LE(gb->memory.rtcLastLatch, 0, &buffer.latchedUnix);
|
||||||
|
|
||||||
uint32_t flags;
|
gb->memory.mbcState.tama5.disabled = !(gb->memory.mbcState.tama5.rtcTimerPage[GBTAMA6_RTC_PAGE] & 0x8);
|
||||||
LOAD_32LE(flags, 0, &buffer.flags);
|
|
||||||
gb->memory.mbcState.tama5.disabled = GBMBCTAMA5SaveFlagsGetDisabled(flags);
|
gb->memory.mbcState.tama5.rtcTimerPage[GBTAMA6_RTC_PAGE] &= 0xC;
|
||||||
|
gb->memory.mbcState.tama5.rtcAlarmPage[GBTAMA6_RTC_PAGE] &= 0xC;
|
||||||
|
gb->memory.mbcState.tama5.rtcAlarmPage[GBTAMA6_RTC_PAGE] |= 1;
|
||||||
|
gb->memory.mbcState.tama5.rtcFreePage0[GBTAMA6_RTC_PAGE] &= 0xC;
|
||||||
|
gb->memory.mbcState.tama5.rtcFreePage0[GBTAMA6_RTC_PAGE] |= 2;
|
||||||
|
gb->memory.mbcState.tama5.rtcFreePage1[GBTAMA6_RTC_PAGE] &= 0xC;
|
||||||
|
gb->memory.mbcState.tama5.rtcFreePage1[GBTAMA6_RTC_PAGE] |= 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
void GBMBCTAMA5Write(struct GB* gb) {
|
void GBMBCTAMA5Write(struct GB* gb) {
|
||||||
|
@ -2245,15 +2293,14 @@ void GBMBCTAMA5Write(struct GB* gb) {
|
||||||
for (i = 0; i < 8; ++i) {
|
for (i = 0; i < 8; ++i) {
|
||||||
buffer.rtcTimerPage[i] = gb->memory.mbcState.tama5.rtcTimerPage[i * 2] & 0xF;
|
buffer.rtcTimerPage[i] = gb->memory.mbcState.tama5.rtcTimerPage[i * 2] & 0xF;
|
||||||
buffer.rtcTimerPage[i] |= gb->memory.mbcState.tama5.rtcTimerPage[i * 2 + 1] << 4;
|
buffer.rtcTimerPage[i] |= gb->memory.mbcState.tama5.rtcTimerPage[i * 2 + 1] << 4;
|
||||||
buffer.rtcAlarmPage[i] = 0;
|
buffer.rtcAlarmPage[i] = gb->memory.mbcState.tama5.rtcAlarmPage[i * 2] & 0xF;
|
||||||
buffer.rtcFreePage0[i] = 0;
|
buffer.rtcAlarmPage[i] |= gb->memory.mbcState.tama5.rtcAlarmPage[i * 2 + 1] << 4;
|
||||||
buffer.rtcFreePage1[i] = 0;
|
buffer.rtcFreePage0[i] = gb->memory.mbcState.tama5.rtcFreePage0[i * 2] & 0xF;
|
||||||
|
buffer.rtcFreePage0[i] |= gb->memory.mbcState.tama5.rtcFreePage0[i * 2 + 1] << 4;
|
||||||
|
buffer.rtcFreePage1[i] = gb->memory.mbcState.tama5.rtcFreePage1[i * 2] & 0xF;
|
||||||
|
buffer.rtcFreePage1[i] |= gb->memory.mbcState.tama5.rtcFreePage1[i * 2 + 1] << 4;
|
||||||
}
|
}
|
||||||
STORE_64LE(gb->memory.rtcLastLatch, 0, &buffer.latchedUnix);
|
STORE_64LE(gb->memory.rtcLastLatch, 0, &buffer.latchedUnix);
|
||||||
|
|
||||||
uint32_t flags = 0;
|
|
||||||
flags = GBMBCTAMA5SaveFlagsSetDisabled(flags, gb->memory.mbcState.tama5.disabled);
|
|
||||||
STORE_32LE(flags, 0, &buffer.flags);
|
|
||||||
|
|
||||||
_appendSaveSuffix(gb, &buffer, sizeof(buffer));
|
_appendSaveSuffix(gb, &buffer, sizeof(buffer));
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue