mirror of
https://github.com/pmret/papermario.git
synced 2024-11-13 22:43:00 +01:00
230 lines
5.3 KiB
C
230 lines
5.3 KiB
C
|
/* snesrc - SNES Recompiler
|
||
|
*
|
||
|
* Mar 23, 2010: addition by spinout to actually fix CRC if it is incorrect
|
||
|
*
|
||
|
* Copyright notice for this file:
|
||
|
* Copyright (C) 2005 Parasyte
|
||
|
*
|
||
|
* Based on uCON64's N64 checksum algorithm by Andreas Sterbenz
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or modify
|
||
|
* it under the terms of the GNU General Public License as published by
|
||
|
* the Free Software Foundation; either version 2 of the License, or
|
||
|
* (at your option) any later version.
|
||
|
*
|
||
|
* This program is distributed in the hope that it will be useful,
|
||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||
|
* GNU General Public License for more details.
|
||
|
*
|
||
|
* You should have received a copy of the GNU General Public License
|
||
|
* along with this program; if not, write to the Free Software
|
||
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||
|
*/
|
||
|
|
||
|
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
|
||
|
#define ROL(i, b) (((i) << (b)) | ((i) >> (32 - (b))))
|
||
|
#define BYTES2LONG(b) ( (b)[0] << 24 | \
|
||
|
(b)[1] << 16 | \
|
||
|
(b)[2] << 8 | \
|
||
|
(b)[3] )
|
||
|
|
||
|
#define N64_HEADER_SIZE 0x40
|
||
|
#define N64_BC_SIZE (0x1000 - N64_HEADER_SIZE)
|
||
|
|
||
|
#define N64_CRC1 0x10
|
||
|
#define N64_CRC2 0x14
|
||
|
|
||
|
#define CHECKSUM_START 0x00001000
|
||
|
#define CHECKSUM_LENGTH 0x00100000
|
||
|
#define CHECKSUM_CIC6102 0xF8CA4DDC
|
||
|
#define CHECKSUM_CIC6103 0xA3886759
|
||
|
#define CHECKSUM_CIC6105 0xDF26F436
|
||
|
#define CHECKSUM_CIC6106 0x1FEA617A
|
||
|
|
||
|
#define Write32(Buffer, Offset, Value)\
|
||
|
Buffer[Offset] = (Value & 0xFF000000) >> 24;\
|
||
|
Buffer[Offset + 1] = (Value & 0x00FF0000) >> 16;\
|
||
|
Buffer[Offset + 2] = (Value & 0x0000FF00) >> 8;\
|
||
|
Buffer[Offset + 3] = (Value & 0x000000FF);\
|
||
|
|
||
|
unsigned int crc_table[256];
|
||
|
|
||
|
void gen_table() {
|
||
|
unsigned int crc, poly;
|
||
|
int i, j;
|
||
|
|
||
|
poly = 0xEDB88320;
|
||
|
for (i = 0; i < 256; i++) {
|
||
|
crc = i;
|
||
|
for (j = 8; j > 0; j--) {
|
||
|
if (crc & 1) crc = (crc >> 1) ^ poly;
|
||
|
else crc >>= 1;
|
||
|
}
|
||
|
crc_table[i] = crc;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
unsigned int crc32(unsigned char *data, int len) {
|
||
|
unsigned int crc = ~0;
|
||
|
int i;
|
||
|
|
||
|
for (i = 0; i < len; i++) {
|
||
|
crc = (crc >> 8) ^ crc_table[(crc ^ data[i]) & 0xFF];
|
||
|
}
|
||
|
|
||
|
return ~crc;
|
||
|
}
|
||
|
|
||
|
|
||
|
int N64GetCIC(unsigned char *data) {
|
||
|
switch (crc32(&data[N64_HEADER_SIZE], N64_BC_SIZE)) {
|
||
|
case 0x6170A4A1: return 6101;
|
||
|
case 0x90BB6CB5: return 6102;
|
||
|
case 0x0B050EE0: return 6103;
|
||
|
case 0x98BC2C86: return 6105;
|
||
|
case 0xACC8580A: return 6106;
|
||
|
}
|
||
|
|
||
|
return 6105;
|
||
|
}
|
||
|
|
||
|
int N64CalcCRC(unsigned int *crc, unsigned char *data) {
|
||
|
int bootcode, i;
|
||
|
unsigned int seed;
|
||
|
|
||
|
unsigned int t1, t2, t3;
|
||
|
unsigned int t4, t5, t6;
|
||
|
unsigned int r, d;
|
||
|
|
||
|
|
||
|
switch ((bootcode = N64GetCIC(data))) {
|
||
|
case 6101:
|
||
|
case 6102:
|
||
|
seed = CHECKSUM_CIC6102;
|
||
|
break;
|
||
|
case 6103:
|
||
|
seed = CHECKSUM_CIC6103;
|
||
|
break;
|
||
|
case 6105:
|
||
|
seed = CHECKSUM_CIC6105;
|
||
|
break;
|
||
|
case 6106:
|
||
|
seed = CHECKSUM_CIC6106;
|
||
|
break;
|
||
|
default:
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
t1 = t2 = t3 = t4 = t5 = t6 = seed;
|
||
|
|
||
|
i = CHECKSUM_START;
|
||
|
while (i < (CHECKSUM_START + CHECKSUM_LENGTH)) {
|
||
|
d = BYTES2LONG(&data[i]);
|
||
|
if ((t6 + d) < t6) t4++;
|
||
|
t6 += d;
|
||
|
t3 ^= d;
|
||
|
r = ROL(d, (d & 0x1F));
|
||
|
t5 += r;
|
||
|
if (t2 > d) t2 ^= r;
|
||
|
else t2 ^= t6 ^ d;
|
||
|
|
||
|
if (bootcode == 6105) t1 += BYTES2LONG(&data[N64_HEADER_SIZE + 0x0710 + (i & 0xFF)]) ^ d;
|
||
|
else t1 += t5 ^ d;
|
||
|
|
||
|
i += 4;
|
||
|
}
|
||
|
if (bootcode == 6103) {
|
||
|
crc[0] = (t6 ^ t4) + t3;
|
||
|
crc[1] = (t5 ^ t2) + t1;
|
||
|
}
|
||
|
else if (bootcode == 6106) {
|
||
|
crc[0] = (t6 * t4) + t3;
|
||
|
crc[1] = (t5 * t2) + t1;
|
||
|
}
|
||
|
else {
|
||
|
crc[0] = t6 ^ t4 ^ t3;
|
||
|
crc[1] = t5 ^ t2 ^ t1;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int main(int argc, char **argv) {
|
||
|
FILE *fin;
|
||
|
int cic;
|
||
|
unsigned int crc[2];
|
||
|
unsigned char *buffer;
|
||
|
|
||
|
//Init CRC algorithm
|
||
|
gen_table();
|
||
|
|
||
|
//Check args
|
||
|
if (argc != 2) {
|
||
|
printf("Usage: n64sums <infile>\n");
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
//Open file
|
||
|
if (!(fin = fopen(argv[1], "r+b"))) {
|
||
|
printf("Unable to open \"%s\" in mode \"%s\"\n", argv[1], "r+b");
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
//Allocate memory
|
||
|
if (!(buffer = (unsigned char*)malloc((CHECKSUM_START + CHECKSUM_LENGTH)))) {
|
||
|
printf("Unable to allocate %d bytes of memory\n", (CHECKSUM_START + CHECKSUM_LENGTH));
|
||
|
fclose(fin);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
//Read data
|
||
|
if (fread(buffer, 1, (CHECKSUM_START + CHECKSUM_LENGTH), fin) != (CHECKSUM_START + CHECKSUM_LENGTH)) {
|
||
|
printf("Unable to read %d bytes of data (invalid N64 image?)\n", (CHECKSUM_START + CHECKSUM_LENGTH));
|
||
|
fclose(fin);
|
||
|
free(buffer);
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
//Check CIC BootChip
|
||
|
cic = N64GetCIC(buffer);
|
||
|
printf("BootChip: ");
|
||
|
printf((cic ? "CIC-NUS-%d\n" : "Unknown\n"), cic);
|
||
|
|
||
|
//Calculate CRC
|
||
|
if (N64CalcCRC(crc, buffer)) {
|
||
|
printf("Unable to calculate CRC\n");
|
||
|
}
|
||
|
else {
|
||
|
printf("CRC 1: 0x%08X ", BYTES2LONG(&buffer[N64_CRC1]));
|
||
|
printf("Calculated: 0x%08X ", crc[0]);
|
||
|
if (crc[0] == BYTES2LONG(&buffer[N64_CRC1]))
|
||
|
printf("(Good)\n");
|
||
|
else{
|
||
|
Write32(buffer, N64_CRC1, crc[0]);
|
||
|
fseek(fin, N64_CRC1, SEEK_SET);
|
||
|
fwrite(&buffer[N64_CRC1], 1, 4, fin);
|
||
|
printf("(Bad, fixed)\n");
|
||
|
}
|
||
|
|
||
|
printf("CRC 2: 0x%08X ", BYTES2LONG(&buffer[N64_CRC2]));
|
||
|
printf("Calculated: 0x%08X ", crc[1]);
|
||
|
if (crc[1] == BYTES2LONG(&buffer[N64_CRC2]))
|
||
|
printf("(Good)\n");
|
||
|
else{
|
||
|
Write32(buffer, N64_CRC2, crc[1]);
|
||
|
fseek(fin, N64_CRC2, SEEK_SET);
|
||
|
fwrite(&buffer[N64_CRC2], 1, 4, fin);
|
||
|
printf("(Bad, fixed)\n");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
fclose(fin);
|
||
|
free(buffer);
|
||
|
|
||
|
return 0;
|
||
|
}
|