1
0
Fork 0
mirror of https://github.com/zeldaret/oot.git synced 2024-12-04 16:55:56 +00:00
oot/src/boot/yaz0.c

108 lines
2.6 KiB
C
Raw Normal View History

2020-03-17 04:31:30 +00:00
#include <z64.h>
#include <global.h>
u8 sYaz0DataBuffer[0x400];
u32 sYaz0CurDataEnd;
u32 sYaz0CurRomStart;
u32 sYaz0CurSize;
u32 sYaz0MaxPtr;
2020-03-22 21:19:43 +00:00
void* Yaz0_FirstDMA() {
2020-03-17 04:31:30 +00:00
u32 pad0;
u32 pad1;
u32 dmaSize;
u32 curSize;
sYaz0MaxPtr = sYaz0CurDataEnd - 0x19;
2020-03-22 21:19:43 +00:00
2020-03-17 04:31:30 +00:00
curSize = sYaz0CurDataEnd - (u32)sYaz0DataBuffer;
2020-03-22 21:19:43 +00:00
dmaSize = (curSize > sYaz0CurSize) ? sYaz0CurSize : curSize;
2020-03-17 04:31:30 +00:00
DmaMgr_DMARomToRam(sYaz0CurRomStart, sYaz0DataBuffer, dmaSize);
sYaz0CurRomStart += dmaSize;
sYaz0CurSize -= dmaSize;
return sYaz0DataBuffer;
}
2020-03-22 21:19:43 +00:00
void* Yaz0_NextDMA(void* curSrcPos) {
2020-03-17 04:31:30 +00:00
u8* dst;
u32 restSize;
u32 dmaSize;
restSize = sYaz0CurDataEnd - (u32)curSrcPos;
2020-03-22 21:19:43 +00:00
dst = (restSize & 7) ? (sYaz0DataBuffer - (restSize & 7)) + 8 : sYaz0DataBuffer;
2020-03-17 04:31:30 +00:00
bcopy(curSrcPos, dst, restSize);
dmaSize = (sYaz0CurDataEnd - (u32)dst) - restSize;
2020-03-22 21:19:43 +00:00
if (sYaz0CurSize < dmaSize) {
2020-03-17 04:31:30 +00:00
dmaSize = sYaz0CurSize;
2020-03-22 21:19:43 +00:00
}
2020-03-17 04:31:30 +00:00
2020-03-22 21:19:43 +00:00
if (dmaSize != 0) {
2020-03-17 04:31:30 +00:00
DmaMgr_DMARomToRam(sYaz0CurRomStart, (u32)dst + restSize, dmaSize);
sYaz0CurRomStart += dmaSize;
sYaz0CurSize -= dmaSize;
2020-03-22 21:19:43 +00:00
if (!sYaz0CurSize) {
2020-03-17 04:31:30 +00:00
sYaz0MaxPtr = (u32)dst + restSize + dmaSize;
2020-03-22 21:19:43 +00:00
}
2020-03-17 04:31:30 +00:00
}
return dst;
}
2020-03-22 21:19:43 +00:00
void Yaz0_DecompressImpl(Yaz0Header* hdr, u8* dst) {
2020-03-17 04:31:30 +00:00
u32 bitIdx;
u8* src;
u8* dstEnd;
u32 chunkHeader;
u32 nibble;
u8* backPtr;
s32 chunkSize;
u32 off;
bitIdx = 0;
2020-03-23 10:35:43 +00:00
src = (u8*)hdr->data;
2020-03-17 04:31:30 +00:00
dstEnd = dst + hdr->decSize;
2020-03-22 21:19:43 +00:00
do {
if (bitIdx == 0) {
2020-03-23 10:35:43 +00:00
if ((sYaz0MaxPtr < (u32)src) && (sYaz0CurSize != 0)) {
2020-03-17 04:31:30 +00:00
src = Yaz0_NextDMA(src);
2020-03-23 10:35:43 +00:00
}
2020-03-17 04:31:30 +00:00
chunkHeader = *src++;
bitIdx = 8;
}
2020-03-23 10:35:43 +00:00
if (chunkHeader & (1 << 7)) { // uncompressed
*dst = *src;
dst++;
src++;
2020-03-22 21:19:43 +00:00
} else { // compressed
2020-03-23 10:35:43 +00:00
off = ((*src & 0xF) << 8 | *(src + 1));
2020-03-17 04:31:30 +00:00
nibble = *src >> 4;
2020-03-23 10:35:43 +00:00
backPtr = dst - off;
2020-03-17 04:31:30 +00:00
src += 2;
2020-03-22 21:19:43 +00:00
chunkSize = (nibble == 0) // N = chunkSize; B = back offset
? *src++ + 0x12 // 3 bytes 0B BB NN
: nibble + 2; // 2 bytes NB BB
2020-03-17 04:31:30 +00:00
2020-03-23 10:35:43 +00:00
do {
2020-03-17 04:31:30 +00:00
*dst++ = *(backPtr++ - 1);
2020-03-23 10:35:43 +00:00
chunkSize--;
} while (chunkSize != 0);
2020-03-17 04:31:30 +00:00
}
2020-03-23 10:35:43 +00:00
chunkHeader <<= 1;
2020-03-17 04:31:30 +00:00
bitIdx--;
} while (dst != dstEnd);
}
2020-03-22 21:19:43 +00:00
void Yaz0_Decompress(u32 romStart, void* dst, u32 size) {
2020-03-17 04:31:30 +00:00
sYaz0CurRomStart = romStart;
sYaz0CurSize = size;
sYaz0CurDataEnd = sYaz0DataBuffer + sizeof(sYaz0DataBuffer);
Yaz0_DecompressImpl(Yaz0_FirstDMA(), dst);
}