zyimage.c 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. /*
  2. * Copyright (C) 2014 Soul Trace <S-trace@list.ru>
  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. #define _POSIX_SOURCE
  10. #define _POSIX_C_SOURCE 199309L /* getopt */
  11. #include <stdio.h>
  12. #include <stdlib.h>
  13. #include <sys/types.h>
  14. #include <string.h>
  15. #include <unistd.h>
  16. #define szbuf 32768
  17. u_int32_t crc_tab[256];
  18. u_int32_t chksum_crc32 (FILE *f)
  19. {
  20. register unsigned long crc;
  21. unsigned long i, j;
  22. char *buffer = malloc(szbuf);
  23. char *buf;
  24. crc = 0xFFFFFFFF;
  25. while (!feof(f))
  26. {
  27. j = fread(buffer, 1, szbuf, f);
  28. buf = buffer;
  29. for (i = 0; i < j; i++)
  30. crc = ((crc >> 8) & 0x00FFFFFF) ^ crc_tab[(crc ^ *buf++) & 0xFF];
  31. }
  32. free(buffer);
  33. return crc;
  34. }
  35. void chksum_crc32gentab ()
  36. {
  37. unsigned long crc, poly;
  38. int i, j;
  39. poly = 0xEDB88320L;
  40. for (i = 0; i < 256; i++)
  41. {
  42. crc = i;
  43. for (j = 8; j > 0; j--)
  44. {
  45. if (crc & 1)
  46. crc = (crc >> 1) ^ poly;
  47. else
  48. crc >>= 1;
  49. }
  50. crc_tab[i] = crc;
  51. }
  52. }
  53. void usage(char *progname)
  54. {
  55. printf("Usage: %s [ -v Version ] [ -d Device_ID ] <input file>\n", progname);
  56. exit(1);
  57. }
  58. int main(int argc, char *argv[]) {
  59. struct signature
  60. {
  61. const char magic[4];
  62. unsigned int device_id;
  63. char firmware_version[48];
  64. unsigned int crc32;
  65. }
  66. sign =
  67. {
  68. { 'Z', 'N', 'B', 'G' },
  69. 1,
  70. { "V.1.0.0(1.0.0)" },
  71. 0
  72. };
  73. FILE *f;
  74. struct signature oldsign;
  75. char *filename;
  76. static const char *optString;
  77. int opt;
  78. if (argc < 1)
  79. usage(argv[0]);
  80. optString = "v:d:h";
  81. opt = getopt( argc, argv, optString );
  82. while( opt != -1 ) {
  83. switch( opt ) {
  84. case 'v':
  85. if (optarg == NULL)
  86. usage(argv[0]);
  87. strncpy(sign.firmware_version, optarg, sizeof(sign.firmware_version)-1);
  88. sign.firmware_version[sizeof(sign.firmware_version)-1]='\0'; /* Make sure that string is terminated correctly */
  89. break;
  90. case 'd':
  91. sign.device_id = atoi(optarg);
  92. if (sign.device_id == 0)
  93. sign.device_id = (int)strtol(optarg, NULL, 16);
  94. break;
  95. case '?':
  96. case 'h':
  97. usage(argv[0]);
  98. break;
  99. default:
  100. break;
  101. }
  102. opt = getopt( argc, argv, optString );
  103. }
  104. chksum_crc32gentab();
  105. filename=argv[optind];
  106. if (access(filename, W_OK) || access(filename, R_OK))
  107. {
  108. printf("Not open input file %s\n", filename);
  109. exit(1);
  110. }
  111. f = fopen(argv[optind], "r+");
  112. if (f != NULL)
  113. {
  114. fseek(f, sizeof(sign)*-1, SEEK_END);
  115. fread(&oldsign, sizeof(oldsign), 1, f);
  116. if (strncmp(oldsign.magic,"ZNBG", sizeof(oldsign.magic)) == 0 )
  117. {
  118. printf("Image is already signed as:\nDevice ID: 0x%08x\nFirmware version: %s\nImage CRC32: 0x%x\n", oldsign.device_id, oldsign.firmware_version, oldsign.crc32);
  119. exit(0);
  120. }
  121. fseek(f, 0, SEEK_SET);
  122. sign.crc32 = chksum_crc32(f);
  123. fwrite(&sign, sizeof(sign), 1, f);
  124. fclose(f);
  125. printf("Image signed as:\nDevice ID: 0x%08x\nFirmware version: %s\nImage CRC32: 0x%x\n", sign.device_id, sign.firmware_version, sign.crc32);
  126. }
  127. return 0;
  128. }