1 /***************************************************************************
3 * Open \______ \ ____ ____ | | _\_ |__ _______ ___
4 * Source | _// _ \_/ ___\| |/ /| __ \ / _ \ \/ /
5 * Jukebox | | ( <_> ) \___| < | \_\ ( <_> > < <
6 * Firmware |____|_ /\____/ \___ >__|_ \|___ /\____/__/\_ \
10 * Copyright (C) 2006 Dave Chapman
12 * This program is free software; you can redistribute it and/or
13 * modify it under the terms of the GNU General Public License
14 * as published by the Free Software Foundation; either version 2
15 * of the License, or (at your option) any later version.
17 * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
18 * KIND, either express or implied.
20 ****************************************************************************/
27 * CRC32 implementation taken from:
29 * efone - Distributed internet phone system.
31 * (c) 1999,2000 Krzysztof Dabrowski
32 * (c) 1999,2000 ElysiuM deeZine
34 * This program is free software; you can redistribute it and/or
35 * modify it under the terms of the GNU General Public License
36 * as published by the Free Software Foundation; either version
37 * 2 of the License, or (at your option) any later version.
41 /* based on implementation by Finn Yannick Jacobs */
43 /* crc_tab[] -- this crcTable is being build by chksum_crc32GenTab().
44 * so make sure, you call it before using the other
47 static unsigned int crc_tab
[256];
49 /* chksum_crc() -- to a given block, this one calculates the
50 * crc32-checksum until the length is
51 * reached. the crc32-checksum will be
54 static unsigned int chksum_crc32 (unsigned char *block
, unsigned int length
)
56 register unsigned long crc
;
60 for (i
= 0; i
< length
; i
++)
62 crc
= ((crc
>> 8) & 0x00FFFFFF) ^ crc_tab
[(crc
^ *block
++) & 0xFF];
67 /* chksum_crc32gentab() -- to a global crc_tab[256], this one will
68 * calculate the crcTable for crc32-checksums.
69 * it is generated to the polynom [..]
72 static void chksum_crc32gentab (void)
74 unsigned long crc
, poly
;
78 for (i
= 0; i
< 256; i
++)
81 for (j
= 8; j
> 0; j
--)
85 crc
= (crc
>> 1) ^ poly
;
96 static void int2le(unsigned int val
, unsigned char* addr
)
99 addr
[1] = (val
>> 8) & 0xff;
100 addr
[2] = (val
>> 16) & 0xff;
101 addr
[3] = (val
>> 24) & 0xff;
104 int mi4_encode(char *iname
, char *oname
, int version
, int magic
,
105 char *model
, char *type
)
111 unsigned int crc
= 0;
112 unsigned char *outbuf
;
114 file
= fopen(iname
, "rb");
119 fseek(file
,0,SEEK_END
);
120 length
= ftell(file
);
122 fseek(file
,0,SEEK_SET
);
124 /* Add 4 bytes to length (for magic), the 0x200 byte header and
125 then round to an even 0x400 bytes
127 mi4length
= (length
+4+0x200+0x3ff)&~0x3ff;
129 outbuf
= malloc(mi4length
);
132 printf("out of memory!\n");
136 /* Clear the buffer to zero */
137 memset(outbuf
, 0, mi4length
);
139 len
= fread(outbuf
+0x200, 1, length
, file
);
140 if(len
< (size_t)length
) {
146 /* We need to write some data into the actual image - before calculating
148 int2le(0x00000100, &outbuf
[0x2e0]); /* magic */
149 int2le(magic
, &outbuf
[0x2e4]); /* magic */
150 int2le(length
+4, &outbuf
[0x2e8]); /* length plus 0xaa55aa55 */
152 int2le(0xaa55aa55, &outbuf
[0x200+length
]); /* More Magic */
154 strncpy((char *)outbuf
+0x1f8, type
, 4); /* type of binary (RBBL, RBOS) */
155 strncpy((char *)outbuf
+0x1fc, model
, 4); /* 4 character model id */
157 /* Calculate CRC32 checksum */
158 chksum_crc32gentab ();
159 crc
= chksum_crc32 (outbuf
+0x200,mi4length
-0x200);
161 strncpy((char *)outbuf
, "PPOS", 4); /* Magic */
162 int2le(version
, &outbuf
[0x04]); /* .mi4 version */
163 int2le(length
+4, &outbuf
[0x08]); /* Length of firmware plus magic */
164 int2le(crc
, &outbuf
[0x0c]); /* CRC32 of mi4 file */
165 int2le(0x00000002, &outbuf
[0x10]); /* Encryption type: 2 = TEA */
166 int2le(mi4length
, &outbuf
[0x14]); /* Total .mi4 length */
167 int2le(mi4length
-0x200, &outbuf
[0x18]); /* Length of plaintext part */
169 /* v3 files require a dummy DSA signature */
170 if (version
== 0x00010301) {
174 file
= fopen(oname
, "wb");
180 len
= fwrite(outbuf
, 1, mi4length
, file
);
181 if(len
< (size_t)length
) {
188 fprintf(stderr
, "File encoded successfully\n" );