mirror of
https://github.com/TwilitRealm/dusklight
synced 2026-06-22 06:56:31 -04:00
4df8ccc871
* Reorganize files into libs/{dolphin,JSystem,PowerPC_EABI_Support,revolution,TRK_MINNOW_DOLPHIN}
* Update configure.py and project.py for new libs structure
* Refactor `#include <dolphin/x.h>` -> `<x.h>`
* Remove `__REVOLUTION_SDK__` forwards from dolphin
* Fix dolphin/ references in revolution
* Wrap `#include <dolphin.h>` in `!__REVOLUTION_SDK__`
* Always build TRK against dolphin headers
* Resolve revolution SDK header resolution issues
161 lines
3.8 KiB
C
161 lines
3.8 KiB
C
#include <revolution/card.h>
|
|
|
|
#include "__card.h"
|
|
|
|
// prototypes
|
|
static void WriteCallback(s32 chan, s32 result);
|
|
static void EraseCallback(s32 chan, s32 result);
|
|
|
|
void* __CARDGetFatBlock(CARDControl* card) {
|
|
ASSERTLINE(57, card->currentFat);
|
|
return card->currentFat;
|
|
}
|
|
|
|
static void WriteCallback(s32 chan, s32 result) {
|
|
CARDControl* card;
|
|
CARDCallback callback;
|
|
u16* fat0;
|
|
u16* fat1;
|
|
|
|
card = &__CARDBlock[chan];
|
|
|
|
if (result >= 0) {
|
|
fat0 = (u16*)((u8*)card->workArea + 0x6000);
|
|
fat1 = (u16*)((u8*)card->workArea + 0x8000);
|
|
|
|
ASSERTLINE(82, card->currentFat);
|
|
if (card->currentFat == fat0) {
|
|
card->currentFat = fat1;
|
|
memcpy(fat1, fat0, 0x2000);
|
|
} else {
|
|
ASSERTLINE(90, card->currentFat == fat1);
|
|
card->currentFat = fat0;
|
|
memcpy(fat0, fat1, 0x2000);
|
|
}
|
|
}
|
|
|
|
if (!card->apiCallback)
|
|
__CARDPutControlBlock(card, result);
|
|
|
|
callback = card->eraseCallback;
|
|
if (callback) {
|
|
card->eraseCallback = NULL;
|
|
callback(chan, result);
|
|
}
|
|
}
|
|
|
|
static void EraseCallback(s32 chan, s32 result) {
|
|
CARDControl* card = &__CARDBlock[chan];
|
|
CARDCallback callback;
|
|
u16* fat;
|
|
u32 addr;
|
|
|
|
if (result < 0)
|
|
goto error;
|
|
|
|
fat = __CARDGetFatBlock(card);
|
|
addr = ((u32)fat - (u32)card->workArea) / CARD_SYSTEM_BLOCK_SIZE * card->sectorSize;
|
|
result = __CARDWrite(chan, addr, CARD_SYSTEM_BLOCK_SIZE, fat, WriteCallback);
|
|
if (result < 0)
|
|
goto error;
|
|
|
|
return;
|
|
|
|
error:
|
|
if (!card->apiCallback)
|
|
__CARDPutControlBlock(card, result);
|
|
|
|
callback = card->eraseCallback;
|
|
if (callback) {
|
|
card->eraseCallback = NULL;
|
|
callback(chan, result);
|
|
}
|
|
}
|
|
|
|
s32 __CARDAllocBlock(s32 chan, u32 cBlock, CARDCallback callback) {
|
|
CARDControl* card;
|
|
u16* fat;
|
|
u16 iBlock;
|
|
u16 startBlock;
|
|
u16 prevBlock;
|
|
u16 count;
|
|
|
|
ASSERTLINE(182, 0 < cBlock);
|
|
ASSERTLINE(183, 0 <= chan && chan < 2);
|
|
|
|
card = &__CARDBlock[chan];
|
|
if (!card->attached)
|
|
return CARD_RESULT_NOCARD;
|
|
|
|
fat = __CARDGetFatBlock(card);
|
|
if (fat[3] < cBlock)
|
|
return CARD_RESULT_INSSPACE;
|
|
|
|
fat[3] -= cBlock;
|
|
startBlock = 0xFFFF;
|
|
iBlock = fat[4];
|
|
count = 0;
|
|
while (0 < cBlock) {
|
|
if (card->cBlock - 5 < ++count)
|
|
return CARD_RESULT_BROKEN;
|
|
|
|
iBlock++;
|
|
if (!CARDIsValidBlockNo(card, iBlock))
|
|
iBlock = 5;
|
|
|
|
if (fat[iBlock] == 0x0000u) {
|
|
if (startBlock == 0xFFFF)
|
|
startBlock = iBlock;
|
|
else
|
|
fat[prevBlock] = iBlock;
|
|
prevBlock = iBlock;
|
|
fat[iBlock] = 0xFFFF;
|
|
--cBlock;
|
|
}
|
|
}
|
|
|
|
fat[4] = iBlock;
|
|
card->startBlock = startBlock;
|
|
return __CARDUpdateFatBlock(chan, fat, callback);
|
|
}
|
|
|
|
s32 __CARDFreeBlock(s32 chan, u16 nBlock, CARDCallback callback) {
|
|
CARDControl* card;
|
|
u16* fat;
|
|
u16 nextBlock;
|
|
|
|
ASSERTLINE(253, 0 <= chan && chan < 2);
|
|
|
|
card = &__CARDBlock[chan];
|
|
if (!card->attached)
|
|
return CARD_RESULT_NOCARD;
|
|
|
|
fat = __CARDGetFatBlock(card);
|
|
while (nBlock != 0xFFFF) {
|
|
if (!CARDIsValidBlockNo(card, nBlock))
|
|
return CARD_RESULT_BROKEN;
|
|
|
|
nextBlock = fat[nBlock];
|
|
fat[nBlock] = 0;
|
|
nBlock = nextBlock;
|
|
++fat[3];
|
|
}
|
|
|
|
return __CARDUpdateFatBlock(chan, fat, callback);
|
|
}
|
|
|
|
s32 __CARDUpdateFatBlock(s32 chan, u16* fat, CARDCallback callback) {
|
|
CARDControl* card;
|
|
u32 addr;
|
|
|
|
ASSERTLINE(295, 0 <= chan && chan < 2);
|
|
|
|
card = &__CARDBlock[chan];
|
|
++fat[2];
|
|
__CARDCheckSum(fat + 2, 0x1FFC, fat, fat + 1);
|
|
DCStoreRange(fat, 0x2000);
|
|
card->eraseCallback = callback;
|
|
addr = (((char*)fat - (char*)card->workArea) / 8192u) * card->sectorSize;
|
|
return __CARDEraseSector(chan, addr, EraseCallback);
|
|
}
|