2002-03-28 14:34:13 +00:00
|
|
|
|
/***************************************************************************
|
|
|
|
|
* __________ __ ___.
|
|
|
|
|
* Open \______ \ ____ ____ | | _\_ |__ _______ ___
|
|
|
|
|
* Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
|
|
|
|
|
* Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
|
|
|
|
|
* Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
|
|
|
|
|
* \/ \/ \/ \/ \/
|
|
|
|
|
* $Id$
|
|
|
|
|
*
|
|
|
|
|
* Copyright (C) 2002 by Bj<EFBFBD>rn Stenberg
|
|
|
|
|
*
|
|
|
|
|
* All files in this archive are subject to the GNU General Public License.
|
|
|
|
|
* See the file COPYING in the source tree root for full license agreement.
|
|
|
|
|
*
|
|
|
|
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
|
|
|
|
* KIND, either express or implied.
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
2005-01-24 09:01:48 +00:00
|
|
|
|
#include "iriver.h"
|
|
|
|
|
|
|
|
|
|
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=X Archos Multimedia format (X values: A=JBMM, B=AV1xx, C=AV3xx)\n"
|
|
|
|
|
"\t-iriver iRiver format\n"
|
|
|
|
|
"\nNo option assumes Archos standard player/recorder format.\n");
|
|
|
|
|
exit(1);
|
|
|
|
|
}
|
|
|
|
|
|
2002-03-28 14:34:13 +00:00
|
|
|
|
int main (int argc, char** argv)
|
|
|
|
|
{
|
|
|
|
|
unsigned long length,i,slen;
|
|
|
|
|
unsigned char *inbuf,*outbuf;
|
2002-12-19 15:00:16 +00:00
|
|
|
|
unsigned char *iname = argv[1];
|
|
|
|
|
unsigned char *oname = argv[2];
|
2004-01-08 00:22:27 +00:00
|
|
|
|
unsigned char header[32];
|
2002-12-19 15:00:16 +00:00
|
|
|
|
int headerlen = 6;
|
2004-01-08 00:22:27 +00:00
|
|
|
|
int descramble = 1;
|
2002-03-28 14:34:13 +00:00
|
|
|
|
FILE* file;
|
|
|
|
|
|
|
|
|
|
if (argc < 3) {
|
2005-01-24 09:01:48 +00:00
|
|
|
|
usage();
|
2002-03-28 14:34:13 +00:00
|
|
|
|
}
|
2002-12-19 15:00:16 +00:00
|
|
|
|
|
2003-10-27 18:04:28 +00:00
|
|
|
|
if (!strcmp(argv[1], "-fm") || !strcmp(argv[1], "-v2")) {
|
2002-12-19 15:00:16 +00:00
|
|
|
|
headerlen = 24;
|
|
|
|
|
iname = argv[2];
|
|
|
|
|
oname = argv[3];
|
|
|
|
|
}
|
2004-01-08 00:22:27 +00:00
|
|
|
|
|
|
|
|
|
if (!strcmp(argv[1], "-mm")) {
|
|
|
|
|
headerlen = 16;
|
|
|
|
|
iname = argv[2];
|
|
|
|
|
oname = argv[3];
|
|
|
|
|
descramble = 0;
|
|
|
|
|
}
|
2005-01-24 09:01:48 +00:00
|
|
|
|
|
|
|
|
|
if(!strcmp(argv[1], "-iriver")) {
|
|
|
|
|
/* iRiver code dealt with in the iriver.c code */
|
|
|
|
|
iname = argv[2];
|
|
|
|
|
oname = argv[3];
|
|
|
|
|
iriver_decode(iname, oname, FALSE, STRIP_NONE);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
2002-03-28 14:34:13 +00:00
|
|
|
|
|
|
|
|
|
/* open file and check size */
|
2002-12-19 15:00:16 +00:00
|
|
|
|
file = fopen(iname,"rb");
|
2002-03-28 14:34:13 +00:00
|
|
|
|
if (!file) {
|
2002-12-19 15:00:16 +00:00
|
|
|
|
perror(oname);
|
2002-03-28 14:34:13 +00:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
fseek(file,0,SEEK_END);
|
2002-12-19 15:00:16 +00:00
|
|
|
|
length = ftell(file) - headerlen; /* skip header */
|
2004-01-08 00:22:27 +00:00
|
|
|
|
fseek(file,0,SEEK_SET);
|
|
|
|
|
i = fread(header, 1, headerlen, file);
|
|
|
|
|
if ( !i ) {
|
|
|
|
|
perror(iname);
|
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
2002-03-28 14:34:13 +00:00
|
|
|
|
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 ) {
|
2002-12-19 15:00:16 +00:00
|
|
|
|
perror(iname);
|
2002-03-28 14:34:13 +00:00
|
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
fclose(file);
|
|
|
|
|
|
2004-01-08 00:22:27 +00:00
|
|
|
|
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;
|
|
|
|
|
}
|
2002-03-28 14:34:13 +00:00
|
|
|
|
}
|
2004-01-08 00:22:27 +00:00
|
|
|
|
else {
|
|
|
|
|
void* tmpptr;
|
|
|
|
|
unsigned int j=0;
|
|
|
|
|
int stringlen = 32;
|
|
|
|
|
int unpackedsize;
|
|
|
|
|
unsigned char xorstring[32];
|
|
|
|
|
|
2004-01-26 07:51:35 +00:00
|
|
|
|
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;
|
2004-01-08 00:22:27 +00:00
|
|
|
|
|
|
|
|
|
/* calculate the xor string used */
|
|
|
|
|
for (i=0; i<stringlen; i++) {
|
|
|
|
|
int top=0, topchar=0, c;
|
|
|
|
|
int bytecount[256];
|
|
|
|
|
memset(bytecount, 0, sizeof(bytecount));
|
|
|
|
|
|
|
|
|
|
/* gather byte frequency statistics */
|
|
|
|
|
for (c=i; c<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 > j)
|
|
|
|
|
src -= 0x1000;
|
|
|
|
|
|
|
|
|
|
for (x=0; x<count; x++)
|
|
|
|
|
outbuf[j++] = outbuf[src+x];
|
|
|
|
|
i += 2;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
length = j;
|
|
|
|
|
}
|
|
|
|
|
|
2002-03-28 14:34:13 +00:00
|
|
|
|
/* write file */
|
2002-12-19 15:00:16 +00:00
|
|
|
|
file = fopen(oname,"wb");
|
2002-03-28 14:34:13 +00:00
|
|
|
|
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;
|
|
|
|
|
}
|