forked from len0rd/rockbox
		
	* update Makefile * add README * update svn properties * fix small mistakes git-svn-id: svn://svn.rockbox.org/rockbox/trunk@17926 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			242 lines
		
	
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			242 lines
		
	
	
	
		
			6.9 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2008 by Maurus Cuelenaere
 | |
|  *
 | |
|  * 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 software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/
 | |
| 
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <sys/types.h>
 | |
| #include <fcntl.h>
 | |
| #include <unistd.h>
 | |
| #include <sys/stat.h>
 | |
| #include <stdbool.h>
 | |
| #include <dirent.h>
 | |
| 
 | |
| #define VERSION "0.1"
 | |
| 
 | |
| static unsigned char* int2le(unsigned int val)
 | |
| {
 | |
|     static unsigned char addr[4];
 | |
|     addr[0] = val & 0xff;
 | |
|     addr[1] = (val >> 8) & 0xff;
 | |
|     addr[2] = (val >> 16) & 0xff;
 | |
|     addr[3] = (val >> 24) & 0xff;
 | |
|     return addr;
 | |
| }
 | |
| 
 | |
| static unsigned int le2int(unsigned char* buf)
 | |
| {
 | |
|    unsigned int res = (buf[3] << 24) | (buf[2] << 16) | (buf[1] << 8) | buf[0];
 | |
| 
 | |
|    return res;
 | |
| }
 | |
| 
 | |
| unsigned int _filesize(FILE* fd)
 | |
| {
 | |
|     unsigned int tmp, oldpos;
 | |
|     oldpos = ftell(fd);
 | |
|     fseek(fd, 0, SEEK_END);
 | |
|     tmp = ftell(fd);
 | |
|     fseek(fd, oldpos, SEEK_SET);
 | |
|     return tmp;
 | |
| }
 | |
| 
 | |
| static void print_usage(void)
 | |
| {
 | |
| #ifdef _WIN32
 | |
|     fprintf(stderr, "Usage: hxfreplace.exe [IN_FW] [OUT_FW] [BIN_FILE]\n\n");
 | |
|     fprintf(stderr, "Example: hxfreplace.exe VX747.HXF out.hxf ccpmp.bin\n\n");
 | |
| #else
 | |
|     fprintf(stderr, "Usage: HXFreplace [IN_FW] [OUT_FW] [BIN_FILE]\n\n");
 | |
|     fprintf(stderr, "Example: HXFreplace VX747.HXF out.hxf ccpmp.bin\n\n");
 | |
| #endif
 | |
| }
 | |
| 
 | |
| static int checksum(FILE *file)
 | |
| {
 | |
|     int oldpos = ftell(file);
 | |
|     int ret=0, i, filesize = _filesize(file)-0x40;
 | |
|     unsigned char *buf;
 | |
|     
 | |
|     buf = (unsigned char*)malloc(filesize);
 | |
|     
 | |
|     if(buf == NULL)
 | |
|     {
 | |
|         fseek(file, oldpos, SEEK_SET);
 | |
|         fprintf(stderr, "[ERR] Error while allocating memory\n");
 | |
|         return 0;
 | |
|     }
 | |
|     
 | |
|     fseek(file, 0x40, SEEK_SET);
 | |
|     if(fread(buf, filesize, 1, file) != 1)
 | |
|     {
 | |
|         free(buf);
 | |
|         fseek(file, oldpos, SEEK_SET);
 | |
|         fprintf(stderr, "[ERR] Error while reading from file\n");
 | |
|         return 0;
 | |
|     }
 | |
|     
 | |
|     fprintf(stderr, "[INFO] Computing checksum...");
 | |
|     
 | |
|     for(i = 0; i < filesize; i+=4)
 | |
|         ret += le2int(&buf[i]);
 | |
|     
 | |
|     free(buf);
 | |
|     fseek(file, oldpos, SEEK_SET);
 | |
|     
 | |
|     fprintf(stderr, " Done!\n");
 | |
|     return ret;
 | |
| }
 | |
| 
 | |
| int main(int argc, char *argv[])
 | |
| {
 | |
|     FILE *infile, *outfile, *fw;
 | |
|     
 | |
|     fprintf(stderr, "HXFreplace v" VERSION " - (C) 2008 Maurus Cuelenaere\n");
 | |
|     fprintf(stderr, "This is free software; see the source for copying conditions.  There is NO\n");
 | |
|     fprintf(stderr, "warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n\n");
 | |
|     
 | |
|     if(argc != 4)
 | |
|     {
 | |
|         print_usage();
 | |
|         return 1;
 | |
|     }
 | |
| 
 | |
| 	if((infile = fopen(argv[1], "rb")) == NULL)
 | |
|     {
 | |
| 		fprintf(stderr, "[ERR]  Cannot open %s\n", argv[1]);
 | |
| 		return 2;
 | |
| 	}
 | |
|     
 | |
|     if(fseek(infile, 0x40, SEEK_SET) != 0)
 | |
|     {
 | |
|         fprintf(stderr, "[ERR]  Cannot seek to 0x40\n");
 | |
|         fclose(infile);
 | |
|         return 3;
 | |
|     }
 | |
|     
 | |
|     fprintf(stderr, "[INFO] Searching for ccpmp.bin...\n");
 | |
|     
 | |
|     int found = -1;
 | |
|     int filenamesize;
 | |
|     char *filename;
 | |
|     unsigned char tmp[4];
 | |
|     
 | |
| #define READ(x, len) if(fread(x, len, 1, infile) != 1) \
 | |
|                      { \
 | |
|                         fprintf(stderr, "[ERR]  Cannot read from %s\n", argv[1]); \
 | |
|                         fclose(infile); \
 | |
|                         return 4; \
 | |
|                      }
 | |
|     while(found < 0)
 | |
|     {
 | |
|         READ(&tmp[0], 4);
 | |
|         filenamesize = le2int(tmp);
 | |
|         filename = (char*)malloc(filenamesize);
 | |
|         READ(filename, filenamesize);
 | |
|         if(strcmp(filename, "ccpmp.bin") == 0)
 | |
|             found = ftell(infile);
 | |
|         else
 | |
|         {
 | |
|             READ(&tmp[0], 4);
 | |
|             fseek(infile, le2int(tmp), SEEK_CUR);
 | |
|         }
 | |
|         free(filename);
 | |
|     }
 | |
|     
 | |
|     fprintf(stderr, "[INFO] Found ccpmp.bin at 0x%x\n", found);
 | |
|     
 | |
|     if((outfile = fopen(argv[2], "wb+")) == NULL)
 | |
|     {
 | |
|         fclose(infile);
 | |
|         fprintf(stderr, "[ERR]  Cannot open %s\n", argv[2]);
 | |
|         return 5;
 | |
|     }
 | |
|     
 | |
| #define WRITE(x, len) if(fwrite(x, len, 1, outfile) != 1) \
 | |
|                       { \
 | |
|                             fprintf(stderr, "[ERR]  Cannot write to %s\n", argv[2]); \
 | |
|                             fclose(outfile); \
 | |
|                             if(fw != NULL) \
 | |
|                                 fclose(fw); \
 | |
|                             return 5; \
 | |
|                        }
 | |
|                             
 | |
|     unsigned char* buffer;
 | |
|     
 | |
|     buffer = (unsigned char*)malloc(found);
 | |
|     fseek(infile, 0, SEEK_SET);
 | |
|     READ(buffer, found);
 | |
|     WRITE(buffer, found);
 | |
|     free(buffer);
 | |
|     
 | |
|     if((fw = fopen(argv[3], "rb")) == NULL)
 | |
|     {
 | |
|         fclose(infile);
 | |
|         fclose(outfile);
 | |
|         fprintf(stderr, "[ERR]  Cannot open %s\n", argv[3]);
 | |
|     }
 | |
|     
 | |
|     int fw_filesize = _filesize(fw);
 | |
|     
 | |
| #define READ2(x, len) if(fread(x, len, 1, fw) != 1) \
 | |
|                       { \
 | |
|                          fprintf(stderr, "[ERR]  Cannot read from %s\n", argv[3]); \
 | |
|                          fclose(infile); \
 | |
|                          fclose(outfile); \
 | |
|                          return 6; \
 | |
|                       }
 | |
|     buffer = (unsigned char*)malloc(fw_filesize);
 | |
|     READ2(buffer, fw_filesize);
 | |
|     fputc(0x20, outfile); /* Padding */
 | |
|     WRITE(int2le(fw_filesize), 4);
 | |
|     WRITE(buffer, fw_filesize);
 | |
|     free(buffer);
 | |
|     fclose(fw);
 | |
|     fw = NULL;
 | |
|     
 | |
|     fseek(infile, found+1, SEEK_SET);
 | |
|     READ(&tmp, 4);
 | |
|     if(fseek(infile, le2int(&tmp[0]), SEEK_CUR) != 0)
 | |
|     {
 | |
|         fprintf(stderr, "[INFO] Cannot seek into %s\n", argv[1]);
 | |
|         fclose(infile);
 | |
|         fclose(outfile);
 | |
|         return 7;
 | |
|     }
 | |
|     found = ftell(infile);
 | |
|     
 | |
|     int other_size = _filesize(infile) - found;
 | |
|     buffer = (unsigned char*)malloc(other_size);
 | |
|     READ(buffer, other_size);
 | |
|     WRITE(buffer, other_size);
 | |
|     free(buffer);
 | |
|     fclose(infile);
 | |
|     
 | |
|     fflush(outfile);
 | |
|     fseek(outfile, 0x14, SEEK_SET);
 | |
|     WRITE(int2le(_filesize(outfile)), 4);
 | |
|     WRITE(int2le(checksum(outfile)), 4);
 | |
|     fclose(outfile);
 | |
|     
 | |
|     fprintf(stderr, "[INFO] Done!\n");
 | |
|     
 | |
|     return 0;
 | |
| }
 |