176 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			176 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|   | /* snesrc - SNES Recompiler
 | ||
|  |  * | ||
|  |  * 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
 | ||
|  | 
 | ||
|  | 
 | ||
|  | unsigned int crc_table[256]; | ||
|  | 
 | ||
|  | void gen_table()  | ||
|  | { | ||
|  |     uint32_t crc, poly; | ||
|  |     int32_t  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)  | ||
|  | { | ||
|  |     uint32_t crc, i; | ||
|  | 
 | ||
|  |     crc = ~0; | ||
|  |     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 0; | ||
|  | } | ||
|  | 
 | ||
|  | int N64CalcCRC(unsigned int *crc, unsigned char *data)  | ||
|  | { | ||
|  |     int32_t bootcode, i; | ||
|  |     uint32_t seed, r, d; | ||
|  |     uint32_t t1, t2, t3; | ||
|  |     uint32_t t4, t5, t6; | ||
|  | 
 | ||
|  |     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 0; | ||
|  |     } | ||
|  | 
 | ||
|  |     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 1; | ||
|  | } | ||
|  | 
 | ||
|  | void fix_crc (uint8_t* buffer) | ||
|  | { | ||
|  |     uint8_t CRC1[4]; | ||
|  |     uint8_t CRC2[4]; | ||
|  |     uint32_t crc[2]; | ||
|  |     uint32_t i; | ||
|  |      | ||
|  |     gen_table(); | ||
|  | 
 | ||
|  |     /* If the CRC calc was successful, do stuff */ | ||
|  |     if(N64CalcCRC(crc, buffer)) | ||
|  |     { | ||
|  |         for(i = 0; i < 4; i++) | ||
|  |         { | ||
|  |             CRC1[i] = (crc[0] >> (24-8*i))&0xFF; | ||
|  |             CRC2[i] = (crc[1] >> (24-8*i))&0xFF; | ||
|  |         } | ||
|  |          | ||
|  |         /* If the CRC1 changed, update it */ | ||
|  |         if(crc[0] != BYTES2LONG(&buffer[N64_CRC1])) | ||
|  |             memcpy(buffer + N64_CRC1, CRC1, 4); | ||
|  |          | ||
|  |         /* If the CRC2 changed, update it */ | ||
|  |         if (crc[1] != BYTES2LONG(&buffer[N64_CRC2])) | ||
|  |             memcpy(buffer + N64_CRC2, CRC2, 4); | ||
|  |     } | ||
|  | } |