/* 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; }