rockbox/tools/descramble.c
Daniel Stenberg 2acc0ac542 Updated our source code header to explicitly mention that we are GPL v2 or
later. We still need to hunt down snippets used that are not. 1324 modified
files...
http://www.rockbox.org/mail/archive/rockbox-dev-archive-2008-06/0060.shtml


git-svn-id: svn://svn.rockbox.org/rockbox/trunk@17847 a1c6a512-1295-4272-9138-f99709370657
2008-06-28 18:10:04 +00:00

295 lines
7.4 KiB
C

/***************************************************************************
* __________ __ ___.
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
* \/ \/ \/ \/ \/
* $Id$
*
* Copyright (C) 2002 by Björn Stenberg
*
* 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 "iriver.h"
#include "gigabeat.h"
int iaudio_decode(char *iname, char *oname);
unsigned int le2int(unsigned char* buf)
{
unsigned int res = (buf[3] << 24) | (buf[2] << 16) | (buf[1] << 8) | buf[0];
return res;
}
void int2le(unsigned int val, unsigned char* addr)
{
addr[0] = val & 0xFF;
addr[1] = (val >> 8) & 0xff;
addr[2] = (val >> 16) & 0xff;
addr[3] = (val >> 24) & 0xff;
}
void usage(void)
{
printf("usage: descramble [options] <input file> <output file>\n");
printf("options:\n"
"\t-fm Archos FM recorder format\n"
"\t-v2 Archos V2 recorder format\n"
"\t-mm Archos Multimedia format\n"
"\t-iriver iRiver format\n"
"\t-gigabeat Toshiba Gigabeat format\n"
"\t-iaudio iAudio format\n"
"\nNo option assumes Archos standard player/recorder format.\n");
exit(1);
}
int main (int argc, char** argv)
{
unsigned long length,i,slen;
unsigned char *inbuf,*outbuf;
char *iname = argv[1];
char *oname = argv[2];
unsigned char header[32];
int headerlen = 6;
int descramble = 1;
FILE* file;
if (argc < 3) {
usage();
}
if (!strcmp(argv[1], "-fm") || !strcmp(argv[1], "-v2")) {
headerlen = 24;
iname = argv[2];
oname = argv[3];
}
if (!strcmp(argv[1], "-mm")) {
headerlen = 16;
iname = argv[2];
oname = argv[3];
descramble = 0;
}
if(!strcmp(argv[1], "-iriver")) {
/* iRiver code dealt with in the iriver.c code */
iname = argv[2];
oname = argv[3];
return iriver_decode(iname, oname, FALSE, STRIP_NONE) ? -1 : 0;
}
if(!strcmp(argv[1], "-gigabeat")) {
iname = argv[2];
oname = argv[3];
gigabeat_code(iname, oname);
return 0;
}
if(!strcmp(argv[1], "-iaudio")) {
iname = argv[2];
oname = argv[3];
return iaudio_decode(iname, oname);
}
/* open file and check size */
file = fopen(iname,"rb");
if (!file) {
perror(iname);
return -1;
}
fseek(file,0,SEEK_END);
length = ftell(file) - headerlen; /* skip header */
fseek(file,0,SEEK_SET);
i = fread(header, 1, headerlen, file);
if ( !i ) {
perror(iname);
return -1;
}
inbuf = malloc(length);
outbuf = malloc(length);
if ( !inbuf || !outbuf ) {
printf("out of memory!\n");
return -1;
}
/* read file */
i=fread(inbuf,1,length,file);
if ( !i ) {
perror(iname);
return -1;
}
fclose(file);
if (descramble) {
/* descramble */
slen = length/4;
for (i = 0; i < length; i++) {
unsigned long addr = ((i % slen) << 2) + i/slen;
unsigned char data = inbuf[i];
data = ~((data >> 1) | ((data << 7) & 0x80)); /* poor man's ROR */
outbuf[addr] = data;
}
}
else {
void* tmpptr;
unsigned int j=0;
int stringlen = 32;
int unpackedsize;
unsigned char xorstring[32];
unpackedsize = header[4] | header[5] << 8;
unpackedsize |= header[6] << 16 | header[7] << 24;
length = header[8] | header[9] << 8;
length |= header[10] << 16 | header[11] << 24;
/* calculate the xor string used */
for (i=0; i<(unsigned long)stringlen; i++) {
int top=0, topchar=0, c;
int bytecount[256];
memset(bytecount, 0, sizeof(bytecount));
/* gather byte frequency statistics */
for (c=i; c<(int)length; c+=stringlen)
bytecount[inbuf[c]]++;
/* find the most frequent byte */
for (c=0; c<256; c++) {
if (bytecount[c] > top) {
top = bytecount[c];
topchar = c;
}
}
xorstring[i] = topchar;
}
printf("XOR string: %.*s\n", stringlen, xorstring);
/* xor the buffer */
for (i=0; i<length; i++)
outbuf[i] = inbuf[i] ^ xorstring[i & (stringlen-1)];
/* unpack */
tmpptr = realloc(inbuf, unpackedsize);
memset(tmpptr, 0, unpackedsize);
inbuf = outbuf;
outbuf = tmpptr;
for (i=0; i<length;) {
int bit;
int head = inbuf[i++];
for (bit=0; bit<8 && i<length; bit++) {
if (head & (1 << (bit))) {
outbuf[j++] = inbuf[i++];
}
else {
int x;
int byte1 = inbuf[i];
int byte2 = inbuf[i+1];
int count = (byte2 & 0x0f) + 3;
int src =
(j & 0xfffff000) + (byte1 | ((byte2 & 0xf0)<<4)) + 18;
if (src > (int)j)
src -= 0x1000;
for (x=0; x<count; x++)
outbuf[j++] = outbuf[src+x];
i += 2;
}
}
}
length = j;
}
/* write file */
file = fopen(oname,"wb");
if ( !file ) {
perror(argv[2]);
return -1;
}
if ( !fwrite(outbuf,length,1,file) ) {
perror(argv[2]);
return -1;
}
fclose(file);
free(inbuf);
free(outbuf);
return 0;
}
int iaudio_decode(char *iname, char *oname)
{
size_t len;
int length;
FILE *file;
char *outbuf;
int i;
unsigned char sum = 0;
unsigned char filesum;
file = fopen(iname, "rb");
if (!file) {
perror(iname);
return -1;
}
fseek(file,0,SEEK_END);
length = ftell(file);
fseek(file,0,SEEK_SET);
outbuf = malloc(length);
if ( !outbuf ) {
printf("out of memory!\n");
return -1;
}
len = fread(outbuf, 1, length, file);
if(len < (size_t)length) {
perror(iname);
return -2;
}
fclose(file);
for(i = 0; i < length-0x1030;i++)
sum += outbuf[0x1030 + i];
filesum = outbuf[0x102b];
if(filesum != sum) {
printf("Checksum mismatch!\n");
return -1;
}
file = fopen(oname, "wb");
if (!file) {
perror(oname);
return -3;
}
len = fwrite(outbuf+0x1030, 1, length-0x1030, file);
if(len < (size_t)length-0x1030) {
perror(oname);
return -4;
}
fclose(file);
return 0;
}