Skip to content

Commit

Permalink
Added drag and drop
Browse files Browse the repository at this point in the history
  • Loading branch information
CMuncey authored May 2, 2018
1 parent b6325bd commit f5d28d1
Showing 1 changed file with 246 additions and 0 deletions.
246 changes: 246 additions & 0 deletions decompressor.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,246 @@
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdint.h>
#include "endian.h"
#include "crc.c"

#define UINTSIZE 0x01000000
#define COMPSIZE 0x02000000
#define DCMPSIZE 0x04000000

/* Structs */
typedef struct
{
uint32_t startV; /* Start Virtual Address */
uint32_t endV; /* End Virtual Address */
uint32_t startP; /* Start Physical Address */
uint32_t endP; /* End Phycical Address */
}
table_t;

/* Functions */
void decode(uint8_t*, uint8_t*, int32_t);
table_t getTabEnt(uint32_t);
void setTabEnt(uint32_t, table_t);
void loadROM(char*);
int32_t findTable();

/* Globals */
uint8_t* inROM;
uint8_t* outROM;
uint32_t* inTable;
uint32_t* outTable;

int main(int argc, char** argv)
{
FILE* outFile;
int32_t tabStart, tabSize, tabCount;
int32_t size, i;
table_t tab, tempTab;
char* name;

inROM = malloc(DCMPSIZE);
outROM = malloc(DCMPSIZE);

/* Load the ROM into inROM and outROM */
loadROM(argv[1]);

/* Find table offsets */
tabStart = findTable();
inTable = (uint32_t*)(inROM + tabStart);
outTable = (uint32_t*)(outROM + tabStart);
tab = getTabEnt(2);
tabSize = tab.endV - tab.startV;
tabCount = tabSize / 16;
i = 3;

/* Set everything past the table in outROM to 0 */
memset((uint8_t*)(outROM) + tab.endV, 0, DCMPSIZE - tab.endV);

while(i < tabCount)
{
tempTab = getTabEnt(i);
size = tempTab.endV - tempTab.startV;

/* Copy if decoded, decode if encoded */
if(tempTab.endP == 0x00000000)
memcpy((void*)outROM + tempTab.startV, (void*)inROM + tempTab.startP, size);
else
decode((void*)inROM + tempTab.startP, (void*)outROM + tempTab.startV, size);

/* Clean up outROM's table */
tempTab.startP = tempTab.startV;
tempTab.endP = 0x00000000;
setTabEnt(i, tempTab);

i++;
}

/* Write the new ROM */
i = 0;
size = strlen(argv[1]);
name = malloc(size + 7);
strcpy(name, argv[1]);
while(i < size)
{
if(name[i] == '.')
{
name[i] = '\0';
break;
}
i++;
}
strcat(name, "-decomp.z64");
outFile = fopen(name, "wb");
fwrite(outROM, sizeof(uint32_t), UINTSIZE, outFile);
fclose(outFile);
free(outROM);
free(inROM);

/* I have no idea what's going on with this. I think it's just Nintendo magic */
fix_crc(name);
free(name);

return(0);
}

int32_t findTable()
{
int32_t i;
uint32_t* temp;

i = 0;
temp = (uint32_t*)inROM;

while(i+4 < UINTSIZE)
{
/* Thsese values mark the begining of the file table */
if(htobe32(temp[i]) == 0x7A656C64)
{
if(htobe32(temp[i+1]) == 0x61407372)
{
if((htobe32(temp[i+2]) & 0xFF000000) == 0x64000000)
{
/* Search for the begining of the filetable */
i += 8;
while(htobe32(temp[i]) != 0x00001060)
i += 4;
return((i-4) * sizeof(uint32_t));
}
}
}

i += 4;
}

fprintf(stderr, "Error: Couldn't find table\n");
exit(1);
}

void loadROM(char* name)
{
uint32_t size;
FILE* romFile;

/* Open file, make sure it exists */
romFile = fopen(name, "rb");
if(romFile == NULL)
{
perror(name);
exit(1);
}
/* Find size of file */
fseek(romFile, 0, SEEK_END);
size = ftell(romFile);
fseek(romFile, 0, SEEK_SET);

/* If it's not the right size, exit */
if(size != COMPSIZE)
{
fprintf(stderr, "Error, %s is not the correct size", name);
exit(1);
}

/* Read to inROM, close romFile, and copy to outROM */
fread(inROM, sizeof(char), size, romFile);
fclose(romFile);
memcpy(outROM, inROM, size);
}

table_t getTabEnt(uint32_t i)
{
table_t tab;

/* First 32 bytes are VROM start address, next 32 are VROM end address */
/* Next 32 bytes are Physical start address, last 32 are Physical end address */
tab.startV = htobe32(inTable[i*4]);
tab.endV = htobe32(inTable[(i*4)+1]);
tab.startP = htobe32(inTable[(i*4)+2]);
tab.endP = htobe32(inTable[(i*4)+3]);

return(tab);
}

void setTabEnt(uint32_t i, table_t tab)
{
/* First 32 bytes are VROM start address, next 32 are VROM end address */
/* Next 32 bytes are Physical start address, last 32 are Physical end address */
outTable[i*4] = htobe32(tab.startV);
outTable[(i*4)+1] = htobe32(tab.endV);
outTable[(i*4)+2] = htobe32(tab.startP);
outTable[(i*4)+3] = htobe32(tab.endP);
}

void decode(uint8_t* source, uint8_t* decomp, int32_t decompSize)
{
uint32_t srcPlace = 0, dstPlace = 0;
uint32_t i, dist, copyPlace, numBytes;
uint8_t codeByte, byte1, byte2;
uint8_t bitCount = 0;

source += 0x10;
while(dstPlace < decompSize)
{
/* If there are no more bits to test, get a new byte */
if(!bitCount)
{
codeByte = source[srcPlace++];
bitCount = 8;
}

/* If bit 7 is a 1, just copy 1 byte from source to destination */
/* Else do some decoding */
if(codeByte & 0x80)
{
decomp[dstPlace++] = source[srcPlace++];
}
else
{
/* Get 2 bytes from source */
byte1 = source[srcPlace++];
byte2 = source[srcPlace++];

/* Calculate distance to move in destination */
/* And the number of bytes to copy */
dist = ((byte1 & 0xF) << 8) | byte2;
copyPlace = dstPlace - (dist + 1);
numBytes = byte1 >> 4;

/* Do more calculations on the number of bytes to copy */
if(!numBytes)
numBytes = source[srcPlace++] + 0x12;
else
numBytes += 2;

/* Copy data from a previous point in destination */
/* to current point in destination */
for(i = 0; i < numBytes; i++)
decomp[dstPlace++] = decomp[copyPlace++];
}

/* Set up for the next read cycle */
codeByte = codeByte << 1;
bitCount--;
}
}

0 comments on commit f5d28d1

Please sign in to comment.