GBA.Unity/Assets/emulator/saving/Flash.cs
2024-08-16 14:51:15 +08:00

223 lines
7.2 KiB
C#

using System;
using static Util;
namespace OptimeGBA
{
public enum FlashState
{
InitialState,
PreCommand0,
PreCommand1,
FullErase,
EraseSector,
PrepareWriteByte,
SetBank,
}
public enum FlashStateSecondary
{
Ready,
PrepareEraseCommand
}
public enum FlashSize
{
Flash512k,
Flash1m
}
public unsafe sealed class Flash : SaveProvider
{
Gba Gba;
FlashState State = FlashState.InitialState;
FlashStateSecondary StateSecondary = FlashStateSecondary.Ready;
FlashSize Size;
bool IdentificationMode = false;
bool Bank1 = false;
bool PrepareSetBank = false;
bool PrepareWrite = false;
byte[] Memory;
public Flash(Gba gba, FlashSize size)
{
Gba = gba;
Size = size;
switch (size)
{
case FlashSize.Flash1m:
Memory = new byte[131072];
break;
case FlashSize.Flash512k:
Memory = new byte[65536];
break;
}
}
public override byte Read8(uint addr)
{
byte val = 0;
addr -= 0xE000000;
if (Bank1) addr += 0x10000;
if (addr < Memory.Length)
{
val = Memory[addr];
}
if (IdentificationMode)
{
// Return correct IDs in identification mode
if (Size == FlashSize.Flash1m)
{
switch (addr)
{
case 0x0: return 0x62;
case 0x1: return 0x13;
}
}
else
{
switch (addr)
{
case 0x0: return 0x32;
case 0x1: return 0x1B;
}
}
}
// Debug.Log("Flash.Read8 addr:" + HexN(addr, 8) + " val:" + HexN(val, 2));
// Gba.Arm7.Error("read");
return val;
}
public override void Write8(uint addr, byte val)
{
if (PrepareSetBank && addr == 0xE000000)
{
if (Size == FlashSize.Flash1m)
{
Bank1 = (val & 1) != 0 ? true : false;
}
PrepareSetBank = false;
return;
}
if (PrepareWrite)
{
addr -= 0xE000000;
if (Bank1) addr += 0x10000;
if (addr < Memory.Length)
{
// Writes can only clear bits
Memory[addr] &= val;
Dirty = true;
}
PrepareWrite = false;
return;
}
switch (State)
{
case FlashState.InitialState:
if (addr == 0xE005555 && val == 0xAA)
{
// Debug.Log("pre-command 0 sent");
State = FlashState.PreCommand0;
}
break;
case FlashState.PreCommand0:
if (addr == 0xE002AAA && val == 0x55)
{
// Debug.Log("pre-command 1 sent, ready to receive commands");
State = FlashState.PreCommand1;
}
break;
case FlashState.PreCommand1:
switch (StateSecondary)
{
case FlashStateSecondary.Ready:
if (addr == 0xE005555)
{
switch (val)
{
case 0x90:
// Debug.Log("enter identification mode");
IdentificationMode = true;
break;
case 0xF0:
// Debug.Log("exit identification mode");
// Gba.Arm7.Error("here");
IdentificationMode = false;
break;
case 0x80:
// Debug.Log("preparing to erase");
StateSecondary = FlashStateSecondary.PrepareEraseCommand;
break;
case 0xB0:
// Debug.Log("preparing to set bank");
PrepareSetBank = true;
break;
case 0xA0:
// Debug.Log("preparing to write");
PrepareWrite = true;
break;
}
}
break;
case FlashStateSecondary.PrepareEraseCommand:
switch (val)
{
// Erase everything
case 0x10:
if (addr == 0xE005555)
{
for (uint i = 0; i < Memory.Length; i++)
{
Memory[i] = 0xFF;
}
Dirty = true;
}
StateSecondary = FlashStateSecondary.Ready;
break;
// Erase 4 KB
case 0x30:
uint page = addr & 0xF000;
if (Bank1) page += 0x10000;
for (uint i = 0; i < 0x1000; i++)
{
if (i < Memory.Length)
{
Memory[page + i] = 0xFF;
}
}
Dirty = true;
StateSecondary = FlashStateSecondary.Ready;
break;
}
break;
}
State = FlashState.InitialState;
break;
}
}
public override byte[] GetSave()
{
return Memory;
}
public override void LoadSave(byte[] save)
{
for (uint i = 0; i < save.Length && i < Memory.Length; i++)
{
Memory[i] = save[i];
}
}
}
}