buffalo-tag.c 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  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. #include <stdio.h>
  10. #include <stdlib.h>
  11. #include <stdint.h>
  12. #include <string.h>
  13. #include <libgen.h>
  14. #include <getopt.h> /* for getopt() */
  15. #include <netinet/in.h>
  16. #include "buffalo-lib.h"
  17. #define ERR(fmt, ...) do { \
  18. fflush(0); \
  19. fprintf(stderr, "[%s] *** error: " fmt "\n", \
  20. progname, ## __VA_ARGS__ ); \
  21. } while (0)
  22. static char *region_table[] = {
  23. "JP", "US", "EU", "AP", "TW", "KR"
  24. };
  25. #define MAX_INPUT_FILES 2
  26. static char *progname;
  27. static char *ifname[MAX_INPUT_FILES];
  28. static ssize_t fsize[MAX_INPUT_FILES];
  29. static int num_files;
  30. static char *ofname;
  31. static char *product;
  32. static char *brand;
  33. static char *language;
  34. static char *hwver;
  35. static char *platform;
  36. static int flag;
  37. static char *major;
  38. static char *minor = "1.01";
  39. static int skipcrc;
  40. static uint32_t base1;
  41. static uint32_t base2;
  42. static char *region_code;
  43. static uint32_t region_mask;
  44. static int num_regions;
  45. void usage(int status)
  46. {
  47. FILE *stream = (status != EXIT_SUCCESS) ? stderr : stdout;
  48. fprintf(stream, "Usage: %s [OPTIONS...]\n", progname);
  49. fprintf(stream,
  50. "\n"
  51. "Options:\n"
  52. " -a <platform> set platform to <platform>\n"
  53. " -b <brand> set brand to <brand>\n"
  54. " -c <base1>\n"
  55. " -d <base2>\n"
  56. " -f <flag> set flag to <flag>\n"
  57. " -i <file> read input from the file <file>\n"
  58. " -l <language> set language to <language>\n"
  59. " -m <version> set minor version to <version>\n"
  60. " -o <file> write output to the file <file>\n"
  61. " -p <product> set product to <product>\n"
  62. " -r <region> set image region to <region>\n"
  63. " valid regions: JP, US, EU, AP, TW, KR, M_\n"
  64. " -s skip CRC calculation\n"
  65. " -v <version> set major version to <version>\n"
  66. " -w <version> set harwdware version to <version>\n"
  67. " -h show this screen\n"
  68. );
  69. exit(status);
  70. }
  71. static int check_params(void)
  72. {
  73. #define CHECKSTR(_var, _name, _len) do { \
  74. if ((_var) == NULL) { \
  75. ERR("no %s specified", (_name)); \
  76. return -1; \
  77. } \
  78. if ((_len) > 0 && \
  79. strlen((_var)) > ((_len) - 1)) { \
  80. ERR("%s is too long", (_name)); \
  81. return -1; \
  82. } \
  83. } while (0)
  84. if (num_files == 0)
  85. ERR("no input files specified");
  86. CHECKSTR(ofname, "output file", 0);
  87. CHECKSTR(brand, "brand", TAG_BRAND_LEN);
  88. CHECKSTR(product, "product", TAG_PRODUCT_LEN);
  89. CHECKSTR(platform, "platform", TAG_PLATFORM_LEN);
  90. CHECKSTR(major, "major version", TAG_VERSION_LEN);
  91. CHECKSTR(minor, "minor version", TAG_VERSION_LEN);
  92. CHECKSTR(language, "language", TAG_LANGUAGE_LEN);
  93. if (hwver)
  94. CHECKSTR(hwver, "hardware version", 2);
  95. if (num_regions == 0) {
  96. ERR("no region code specified");
  97. return -1;
  98. }
  99. return 0;
  100. #undef CHECKSTR
  101. }
  102. static int process_region(char *reg)
  103. {
  104. int i;
  105. if (strlen(reg) != 2) {
  106. ERR("invalid region code '%s'", reg);
  107. return -1;
  108. }
  109. if (strcmp(reg, "M_") == 0) {
  110. region_code = reg;
  111. region_mask |= ~0;
  112. num_regions = 32;
  113. return 0;
  114. }
  115. for (i = 0; i < ARRAY_SIZE(region_table); i++)
  116. if (strcmp(reg, region_table[i]) == 0) {
  117. region_code = reg;
  118. region_mask |= 1 << i;
  119. num_regions++;
  120. return 0;
  121. }
  122. ERR("unknown region code '%s'", reg);
  123. return -1;
  124. }
  125. static int process_ifname(char *name)
  126. {
  127. if (num_files >= ARRAY_SIZE(ifname)) {
  128. ERR("too many input files specified");
  129. return -1;
  130. }
  131. ifname[num_files++] = name;
  132. return 0;
  133. }
  134. static void fixup_tag(unsigned char *buf, ssize_t buflen)
  135. {
  136. struct buffalo_tag *tag = (struct buffalo_tag *) buf;
  137. memset(tag, '\0', sizeof(*tag));
  138. memcpy(tag->brand, brand, strlen(brand));
  139. memcpy(tag->product, product, strlen(product));
  140. memcpy(tag->platform, platform, strlen(platform));
  141. memcpy(tag->ver_major, major, strlen(major));
  142. memcpy(tag->ver_minor, minor, strlen(minor));
  143. memcpy(tag->language, language, strlen(language));
  144. if (num_regions > 1) {
  145. tag->region_code[0] = 'M';
  146. tag->region_code[1] = '_';
  147. tag->region_mask = htonl(region_mask);
  148. } else {
  149. memcpy(tag->region_code, region_code, 2);
  150. }
  151. tag->len = htonl(buflen);
  152. tag->data_len = htonl(fsize[0]);
  153. tag->base1 = htonl(base1);
  154. tag->base2 = htonl(base2);
  155. tag->flag = flag;
  156. if (hwver) {
  157. memcpy(tag->hwv, "hwv", 3);
  158. memcpy(tag->hwv_val, hwver, strlen(hwver));
  159. }
  160. if (!skipcrc)
  161. tag->crc = htonl(buffalo_crc(buf, buflen));
  162. }
  163. static void fixup_tag2(unsigned char *buf, ssize_t buflen)
  164. {
  165. struct buffalo_tag2 *tag = (struct buffalo_tag2 *) buf;
  166. memset(tag, '\0', sizeof(*tag));
  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. if (num_regions > 1) {
  174. tag->region_code[0] = 'M';
  175. tag->region_code[1] = '_';
  176. tag->region_mask = htonl(region_mask);
  177. } else {
  178. memcpy(tag->region_code, region_code, 2);
  179. }
  180. tag->total_len = htonl(buflen);
  181. tag->len1 = htonl(fsize[0]);
  182. tag->len2 = htonl(fsize[1]);
  183. tag->flag = flag;
  184. if (hwver) {
  185. memcpy(tag->hwv, "hwv", 3);
  186. memcpy(tag->hwv_val, hwver, strlen(hwver));
  187. }
  188. if (!skipcrc)
  189. tag->crc = htonl(buffalo_crc(buf, buflen));
  190. }
  191. static int tag_file(void)
  192. {
  193. unsigned char *buf;
  194. ssize_t offset;
  195. ssize_t hdrlen;
  196. ssize_t buflen;
  197. int err;
  198. int ret = -1;
  199. int i;
  200. if (num_files == 1)
  201. hdrlen = sizeof(struct buffalo_tag);
  202. else
  203. hdrlen = sizeof(struct buffalo_tag2);
  204. buflen = hdrlen;
  205. for (i = 0; i < num_files; i++) {
  206. fsize[i] = get_file_size(ifname[i]);
  207. if (fsize[i] < 0) {
  208. ERR("unable to get size of '%s'", ifname[i]);
  209. goto out;
  210. }
  211. buflen += fsize[i];
  212. }
  213. buf = malloc(buflen);
  214. if (!buf) {
  215. ERR("no memory for buffer\n");
  216. goto out;
  217. }
  218. offset = hdrlen;
  219. for (i = 0; i < num_files; i++) {
  220. err = read_file_to_buf(ifname[i], buf + offset, fsize[i]);
  221. if (err) {
  222. ERR("unable to read from file '%s'", ifname[i]);
  223. goto free_buf;
  224. }
  225. offset += fsize[i];
  226. }
  227. if (num_files == 1)
  228. fixup_tag(buf, buflen);
  229. else
  230. fixup_tag2(buf, buflen);
  231. err = write_buf_to_file(ofname, buf, buflen);
  232. if (err) {
  233. ERR("unable to write to file '%s'", ofname);
  234. goto free_buf;
  235. }
  236. ret = 0;
  237. free_buf:
  238. free(buf);
  239. out:
  240. return ret;
  241. }
  242. int main(int argc, char *argv[])
  243. {
  244. int res = EXIT_FAILURE;
  245. int err;
  246. progname = basename(argv[0]);
  247. while ( 1 ) {
  248. int c;
  249. c = getopt(argc, argv, "a:b:c:d:f:hi:l:m:o:p:r:sv:w:");
  250. if (c == -1)
  251. break;
  252. switch (c) {
  253. case 'a':
  254. platform = optarg;
  255. break;
  256. case 'b':
  257. brand = optarg;
  258. break;
  259. case 'c':
  260. base1 = strtoul(optarg, NULL, 16);
  261. break;
  262. case 'd':
  263. base2 = strtoul(optarg, NULL, 16);
  264. break;
  265. case 'f':
  266. flag = strtoul(optarg, NULL, 2);
  267. break;
  268. case 'i':
  269. err = process_ifname(optarg);
  270. if (err)
  271. goto out;
  272. break;
  273. case 'l':
  274. language = optarg;
  275. break;
  276. case 'm':
  277. minor = optarg;
  278. break;
  279. case 'o':
  280. ofname = optarg;
  281. break;
  282. case 'p':
  283. product = optarg;
  284. break;
  285. case 'r':
  286. err = process_region(optarg);
  287. if (err)
  288. goto out;
  289. break;
  290. case 's':
  291. skipcrc = 1;
  292. break;
  293. case 'v':
  294. major = optarg;
  295. break;
  296. case 'w':
  297. hwver = optarg;
  298. break;
  299. case 'h':
  300. usage(EXIT_SUCCESS);
  301. break;
  302. default:
  303. usage(EXIT_FAILURE);
  304. break;
  305. }
  306. }
  307. err = check_params();
  308. if (err)
  309. goto out;
  310. err = tag_file();
  311. if (err)
  312. goto out;
  313. res = EXIT_SUCCESS;
  314. out:
  315. return res;
  316. }