2 * Copyright (C) 2014 Claudio Leite <leitec@staticky.com>
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License as published by
6 * the Free Software Foundation; either version 2 of the License, or
7 * (at your option) any later version.
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * General Public License for more details.
14 * You should have received a copy of the GNU General Public License
15 * along with this program; if not, write to the Free Software
16 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 * Builds a proper flash image for routers using some Gemtek
21 * OEM boards. These include the Airlink101 AR725W, the
22 * Asante SmartHub 600 (AWRT-600N), and Linksys WRT100/110.
24 * The resulting image is compatible with the factory firmware
25 * web upgrade and TFTP interface.
28 * gcc -O2 -o mkheader_gemtek mkheader_gemtek.c -lz
30 * Claudio Leite <leitec@staticky.com>
38 #include <zlib.h> /* for crc32() */
41 * The header is in little-endian format. In case
42 * we are on a BE host, we need to swap binary
46 # include <libkern/OSByteOrder.h>
47 # define le32 OSSwapHostToLittleInt32
49 # if defined(__linux__)
51 # if __BYTE_ORDER == __BIG_ENDIAN
52 # define CPU_BIG_ENDIAN
55 # include <sys/endian.h> /* BSD's should have this */
56 # if _BYTE_ORDER == _BIG_ENDIAN
57 # define CPU_BIG_ENDIAN
60 # ifdef CPU_BIG_ENDIAN
61 # define le32(x) (((x & 0xff000000) >> 24) | \
62 ((x & 0x00ff0000) >> 8) | \
63 ((x & 0x0000ff00) << 8) | \
64 ((x & 0x000000ff) << 24))
70 struct gemtek_header {
76 uint32_t fast_checksum;
81 #define HDRLEN sizeof(struct gemtek_header)
87 struct gemtek_header header;
90 struct machines mach_def[] = {
91 {"Airlink101 AR725W", "ar725w", 0x340000,
92 {"GMTK", "1003", le32(0x03000001), 0, 0,
93 0, "01\0\0", "EN\0\0"}},
94 {"Asante AWRT-600N", "awrt600n", 0x340000,
95 {"A600", "1005", le32(0x03000001), 0, 0,
96 0, "01\0\0", "EN\0\0"}},
97 {"Linksys WRT100", "wrt100", 0x320000,
98 {"GMTK", "1007", le32(0x03040001), 0, 0,
99 0, "2\0\0\0", "EN\0\0"}},
100 {"Linksys WRT110", "wrt110", 0x320000,
101 {"GMTK", "1007", le32(0x03040001), 0, 0,
102 0, "2\0\0\0", "EN\0\0"}},
107 main(int argc, char *argv[])
109 unsigned long res, flen;
110 struct gemtek_header my_hdr;
112 int image_type = -1, index;
117 fprintf(stderr, "mkheader_gemtek <uImage> <webflash image> [machine ID]\n");
118 fprintf(stderr, " where [machine ID] is one of:\n");
119 for (index = 0; mach_def[index].desc != 0; index++) {
120 fprintf(stderr, " %-10s %s", mach_def[index].id, mach_def[index].desc);
122 fprintf(stderr, " (default)\n");
124 fprintf(stderr, "\n");
131 for(index = 0; mach_def[index].id != 0; index++) {
132 if(strcmp(mach_def[index].id, argv[3]) == 0) {
138 if(image_type == -1) {
139 fprintf(stderr, "\nERROR: invalid machine type\n");
145 printf("Opening %s...\n", argv[1]);
147 f = fopen(argv[1], "r");
149 fprintf(stderr, "\nERROR: couldn't open input image\n");
153 fseek(f, 0, SEEK_END);
154 flen = (unsigned long) ftell(f);
156 printf(" %lu (0x%lX) bytes long\n", flen, flen);
158 if (flen > mach_def[image_type].maxsize) {
159 fprintf(stderr, "\nERROR: image exceeds maximum compatible size\n");
163 buf = malloc(flen + HDRLEN);
165 fprintf(stderr, "\nERROR: couldn't allocate buffer\n");
169 res = fread(buf + HDRLEN, 1, flen, f);
171 perror("Couldn't read entire file: fread()");
176 printf("\nCreating %s...\n", argv[2]);
178 memcpy(&my_hdr, &mach_def[image_type].header, HDRLEN);
180 printf(" Using %s magic\n", mach_def[image_type].desc);
182 my_hdr.imagesz = le32(flen + HDRLEN);
183 memcpy(my_hdr.lang, "EN", 2);
185 memcpy(buf, &my_hdr, HDRLEN);
187 crc = crc32(0, buf, flen + HDRLEN);
188 printf(" CRC32: %08X\n", crc);
190 my_hdr.checksum = le32(crc);
191 memcpy(buf, &my_hdr, HDRLEN);
193 printf(" Writing...\n");
195 f_out = fopen(argv[2], "w");
197 fprintf(stderr, "\nERROR: couldn't open output image\n");
201 fwrite(buf, 1, flen + HDRLEN, f_out);