mirror of https://github.com/pmret/papermario.git
				
				
				
			
		
			
				
	
	
		
			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;
 | |
| }
 |