librecmc : Bump to v1.5.15
[librecmc/librecmc.git] / tools / firmware-utils / src / buffalo-tag.c
1 /*
2  *  Copyright (C) 2009-2011 Gabor Juhos <juhosg@openwrt.org>
3  *
4  *  This program is free software; you can redistribute it and/or modify it
5  *  under the terms of the GNU General Public License version 2 as published
6  *  by the Free Software Foundation.
7  *
8  */
9
10 #include <stdio.h>
11 #include <stdlib.h>
12 #include <stdint.h>
13 #include <string.h>
14 #include <libgen.h>
15 #include <getopt.h>     /* for getopt() */
16 #include <netinet/in.h>
17
18 #include "buffalo-lib.h"
19
20 #define ERR(fmt, ...) do { \
21         fflush(0); \
22         fprintf(stderr, "[%s] *** error: " fmt "\n", \
23                         progname, ## __VA_ARGS__ ); \
24 } while (0)
25
26 static char *region_table[] = {
27         "JP", "US", "EU", "AP", "TW", "KR"
28 };
29
30 #define MAX_INPUT_FILES 2
31
32 static char *progname;
33 static char *ifname[MAX_INPUT_FILES];
34 static ssize_t fsize[MAX_INPUT_FILES];
35 static int num_files;
36 static char *ofname;
37 static char *product;
38 static char *brand;
39 static char *language;
40 static char *hwver;
41 static char *platform;
42 static int flag;
43 static char *major;
44 static char *minor = "1.01";
45 static int skipcrc;
46 static uint32_t base1;
47 static uint32_t base2;
48 static char *region_code;
49 static uint32_t region_mask;
50 static int num_regions;
51 static int dhp;
52
53 void usage(int status)
54 {
55         FILE *stream = (status != EXIT_SUCCESS) ? stderr : stdout;
56
57         fprintf(stream, "Usage: %s [OPTIONS...]\n", progname);
58         fprintf(stream,
59 "\n"
60 "Options:\n"
61 "  -a <platform>   set platform to <platform>\n"
62 "  -b <brand>      set brand to <brand>\n"
63 "  -c <base1>\n"
64 "  -d <base2>\n"
65 "  -f <flag>       set flag to <flag>\n"
66 "  -i <file>       read input from the file <file>\n"
67 "  -I <file>       read input from the file <file> for DHP series\n"
68 "  -l <language>   set language to <language>\n"
69 "  -m <version>    set minor version to <version>\n"
70 "  -o <file>       write output to the file <file>\n"
71 "  -p <product>    set product to <product>\n"
72 "  -r <region>     set image region to <region>\n"
73 "                  valid regions: JP, US, EU, AP, TW, KR, M_\n"
74 "  -s              skip CRC calculation\n"
75 "  -v <version>    set major version to <version>\n"
76 "  -w <version>    set harwdware version to <version>\n"
77 "  -h              show this screen\n"
78         );
79
80         exit(status);
81 }
82
83 static int check_params(void)
84 {
85
86 #define CHECKSTR(_var, _name, _len)     do {            \
87         if ((_var) == NULL) {                           \
88                 ERR("no %s specified", (_name));        \
89                 return -1;                              \
90         }                                               \
91         if ((_len) > 0 &&                               \
92             strlen((_var)) > ((_len) - 1)) {            \
93                 ERR("%s is too long", (_name));         \
94                 return -1;                              \
95         }                                               \
96 } while (0)
97
98         if (num_files == 0)
99                 ERR("no input files specified");
100
101         CHECKSTR(ofname, "output file", 0);
102         CHECKSTR(brand, "brand", TAG_BRAND_LEN);
103         CHECKSTR(product, "product", TAG_PRODUCT_LEN);
104         CHECKSTR(platform, "platform", TAG_PLATFORM_LEN);
105         CHECKSTR(major, "major version", TAG_VERSION_LEN);
106         CHECKSTR(minor, "minor version", TAG_VERSION_LEN);
107         CHECKSTR(language, "language", TAG_LANGUAGE_LEN);
108
109         if (hwver)
110                 CHECKSTR(hwver, "hardware version", 2);
111
112         if (num_regions == 0) {
113                 ERR("no region code specified");
114                 return -1;
115         }
116
117         return 0;
118
119 #undef CHECKSTR
120 }
121
122 static int process_region(char *reg)
123 {
124         int i;
125
126         if (strlen(reg) != 2) {
127                 ERR("invalid region code '%s'", reg);
128                 return -1;
129         }
130
131         if (strcmp(reg, "M_") == 0) {
132                 region_code = reg;
133                 region_mask |= ~0;
134                 num_regions = 32;
135                 return 0;
136         }
137
138         for (i = 0; i < ARRAY_SIZE(region_table); i++)
139                 if (strcmp(reg, region_table[i]) == 0) {
140                         region_code = reg;
141                         region_mask |= 1 << i;
142                         num_regions++;
143                         return 0;
144                 }
145
146         ERR("unknown region code '%s'", reg);
147         return -1;
148 }
149
150 static int process_ifname(char *name)
151 {
152         if (num_files >= ARRAY_SIZE(ifname)) {
153                 ERR("too many input files specified");
154                 return -1;
155         }
156
157         ifname[num_files++] = name;
158         return 0;
159 }
160
161 static void fixup_tag(unsigned char *buf, ssize_t buflen)
162 {
163         struct buffalo_tag *tag = (struct buffalo_tag *) buf;
164
165         memset(tag, '\0', sizeof(*tag));
166
167         memcpy(tag->brand, brand, strlen(brand));
168         memcpy(tag->product, product, strlen(product));
169         memcpy(tag->platform, platform, strlen(platform));
170         memcpy(tag->ver_major, major, strlen(major));
171         memcpy(tag->ver_minor, minor, strlen(minor));
172         memcpy(tag->language, language, strlen(language));
173
174         if (num_regions > 1) {
175                 tag->region_code[0] = 'M';
176                 tag->region_code[1] = '_';
177                 tag->region_mask = htonl(region_mask);
178         } else {
179                 memcpy(tag->region_code, region_code, 2);
180         }
181
182         tag->len = htonl(buflen);
183         tag->data_len = htonl(fsize[0]);
184         tag->base1 = htonl(base1);
185         tag->base2 = htonl(base2);
186         tag->flag = flag;
187
188         if (hwver) {
189                 memcpy(tag->hwv, "hwv", 3);
190                 memcpy(tag->hwv_val, hwver, strlen(hwver));
191         }
192
193         if (!skipcrc)
194                 tag->crc = htonl(buffalo_crc(buf, buflen));
195 }
196
197 static void fixup_tag2(unsigned char *buf, ssize_t buflen)
198 {
199         struct buffalo_tag2 *tag = (struct buffalo_tag2 *) buf;
200
201         memset(tag, '\0', sizeof(*tag));
202
203         memcpy(tag->brand, brand, strlen(brand));
204         memcpy(tag->product, product, strlen(product));
205         memcpy(tag->platform, platform, strlen(platform));
206         memcpy(tag->ver_major, major, strlen(major));
207         memcpy(tag->ver_minor, minor, strlen(minor));
208         memcpy(tag->language, language, strlen(language));
209
210         if (num_regions > 1) {
211                 tag->region_code[0] = 'M';
212                 tag->region_code[1] = '_';
213                 tag->region_mask = htonl(region_mask);
214         } else {
215                 memcpy(tag->region_code, region_code, 2);
216         }
217
218         tag->total_len = htonl(buflen);
219         tag->len1 = htonl(fsize[0]);
220         tag->len2 = htonl(fsize[1]);
221         tag->flag = flag;
222
223         if (hwver) {
224                 memcpy(tag->hwv, "hwv", 3);
225                 memcpy(tag->hwv_val, hwver, strlen(hwver));
226         }
227
228         if (!skipcrc)
229                 tag->crc = htonl(buffalo_crc(buf, buflen));
230 }
231
232 static void fixup_tag3(unsigned char *buf, ssize_t totlen)
233 {
234         struct buffalo_tag3 *tag = (struct buffalo_tag3 *) buf;
235
236         memset(tag, '\0', sizeof(*tag));
237
238         memcpy(tag->brand, brand, strlen(brand));
239         memcpy(tag->product, product, strlen(product));
240         memcpy(tag->platform, platform, strlen(platform));
241         memcpy(tag->ver_major, major, strlen(major));
242         memcpy(tag->ver_minor, minor, strlen(minor));
243         memcpy(tag->language, language, strlen(language));
244
245         if (num_regions > 1) {
246                 tag->region_code[0] = 'M';
247                 tag->region_code[1] = '_';
248                 tag->region_mask = htonl(region_mask);
249         } else {
250                 memcpy(tag->region_code, region_code, 2);
251         }
252
253         tag->total_len = htonl(totlen);
254         tag->len1 = htonl(fsize[0]);
255         tag->base2 = htonl(base2);
256
257         if (hwver) {
258                 memcpy(tag->hwv, "hwv", 3);
259                 memcpy(tag->hwv_val, hwver, strlen(hwver));
260         }
261 }
262
263 static int tag_file(void)
264 {
265         unsigned char *buf;
266         ssize_t offset;
267         ssize_t hdrlen;
268         ssize_t buflen;
269         int err;
270         int ret = -1;
271         int i;
272
273         if (dhp)
274                 hdrlen = sizeof(struct buffalo_tag3);
275         else if (num_files == 1)
276                 hdrlen = sizeof(struct buffalo_tag);
277         else
278                 hdrlen = sizeof(struct buffalo_tag2);
279
280         buflen = hdrlen;
281
282         for (i = 0; i < num_files; i++) {
283                 fsize[i] = get_file_size(ifname[i]);
284                 if (fsize[i] < 0) {
285                         ERR("unable to get size of '%s'", ifname[i]);
286                         goto out;
287                 }
288                 buflen += fsize[i];
289         }
290
291         buf = malloc(buflen);
292         if (!buf) {
293                 ERR("no memory for buffer\n");
294                 goto out;
295         }
296
297         offset = hdrlen;
298         for (i = 0; i < num_files; i++) {
299                 err = read_file_to_buf(ifname[i], buf + offset, fsize[i]);
300                 if (err) {
301                         ERR("unable to read from file '%s'", ifname[i]);
302                         goto free_buf;
303                 }
304
305                 offset += fsize[i];
306         }
307
308         if (dhp)
309                 fixup_tag3(buf, fsize[0] + 200);
310         else if (num_files == 1)
311                 fixup_tag(buf, buflen);
312         else
313                 fixup_tag2(buf, buflen);
314
315         err = write_buf_to_file(ofname, buf, buflen);
316         if (err) {
317                 ERR("unable to write to file '%s'", ofname);
318                 goto free_buf;
319         }
320
321         ret = 0;
322
323 free_buf:
324         free(buf);
325 out:
326         return ret;
327 }
328
329 int main(int argc, char *argv[])
330 {
331         int res = EXIT_FAILURE;
332         int err;
333
334         progname = basename(argv[0]);
335
336         while ( 1 ) {
337                 int c;
338
339                 c = getopt(argc, argv, "a:b:c:d:f:hi:l:m:o:p:r:sv:w:I:");
340                 if (c == -1)
341                         break;
342
343                 switch (c) {
344                 case 'a':
345                         platform = optarg;
346                         break;
347                 case 'b':
348                         brand = optarg;
349                         break;
350                 case 'c':
351                         base1 = strtoul(optarg, NULL, 16);
352                         break;
353                 case 'd':
354                         base2 = strtoul(optarg, NULL, 16);
355                         break;
356                 case 'f':
357                         flag = strtoul(optarg, NULL, 2);
358                         break;
359                 case 'I':
360                         dhp = 1;
361                         /* FALLTHROUGH */
362                 case 'i':
363                         err = process_ifname(optarg);
364                         if (err)
365                                 goto out;
366                         break;
367                 case 'l':
368                         language = optarg;
369                         break;
370                 case 'm':
371                         minor = optarg;
372                         break;
373                 case 'o':
374                         ofname = optarg;
375                         break;
376                 case 'p':
377                         product = optarg;
378                         break;
379                 case 'r':
380                         err = process_region(optarg);
381                         if (err)
382                                 goto out;
383                         break;
384                 case 's':
385                         skipcrc = 1;
386                         break;
387                 case 'v':
388                         major = optarg;
389                         break;
390                 case 'w':
391                         hwver = optarg;
392                         break;
393                 case 'h':
394                         usage(EXIT_SUCCESS);
395                         break;
396                 default:
397                         usage(EXIT_FAILURE);
398                         break;
399                 }
400         }
401
402         err = check_params();
403         if (err)
404                 goto out;
405
406         err = tag_file();
407         if (err)
408                 goto out;
409
410         res = EXIT_SUCCESS;
411
412 out:
413         return res;
414 }