build_nvram_bin.py 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179
  1. #!/usr/bin/env python3
  2. #
  3. # Build Actions NVRAM config binary file
  4. #
  5. # Copyright (c) 2017 Actions Semiconductor Co., Ltd
  6. #
  7. # SPDX-License-Identifier: Apache-2.0
  8. #
  9. import os
  10. import sys
  11. import struct
  12. import argparse
  13. import zlib
  14. #import crc8
  15. NVRAM_REGION_SEG_MAGIC = 0x5253564E
  16. NVRAM_REGION_ITEM_MAGIC = 0x49
  17. NVRAM_REGION_SEG_VERSION = 0x1
  18. # for python2
  19. if sys.version_info < (3, 0):
  20. reload(sys)
  21. sys.setdefaultencoding('utf8')
  22. # private module
  23. from nvram_prop import *;
  24. NVRAM_WRITE_REGION_ALIGN_SIZE = 512
  25. NVRAM_SEG_HEADER_SIZE = 16
  26. NVRAM_ITEM_HEADER_SIZE = 8
  27. def calc_hash(key):
  28. hash = 0
  29. for byte in key:
  30. hash = hash + byte
  31. hash = hash ^ 0xa5
  32. hash = hash & 0xff
  33. return hash
  34. CRC8_MAXIM_TABLE = [
  35. 0x00, 0x5E, 0xBC, 0xE2, 0x61, 0x3F, 0xDD, 0x83,
  36. 0xC2, 0x9C, 0x7E, 0x20, 0xA3, 0xFD, 0x1F, 0x41,
  37. 0x00, 0x9D, 0x23, 0xBE, 0x46, 0xDB, 0x65, 0xF8,
  38. 0x8C, 0x11, 0xAF, 0x32, 0xCA, 0x57, 0xE9, 0x74
  39. ]
  40. def calc_crc8(data, length):
  41. """
  42. * Name: CRC-8/MAXIM x8+x5+x4+1
  43. * Poly: 0x31
  44. * Init: 0x00
  45. * Refin: True
  46. * Refout: True
  47. * Xorout: 0x00
  48. * Alias: DOW-CRC,CRC-8/IBUTTON,MSB first
  49. * Use: Maxim(Dallas)'s some devices,e.g. DS18B20
  50. """
  51. crc = 0
  52. for i in range(length):
  53. crc = data[i] ^ crc
  54. crc = CRC8_MAXIM_TABLE[crc & 0x0f] ^ CRC8_MAXIM_TABLE[16 + ((crc >> 4) & 0x0f)]
  55. return crc
  56. '''
  57. struct region_seg_header
  58. {
  59. u32_t magic;
  60. u8_t state;
  61. u8_t crc;
  62. u8_t version;
  63. u8_t reserved;
  64. u8_t seq_id;
  65. u8_t head_size;
  66. u16_t seg_size;
  67. u8_t reserved2[4];
  68. };
  69. struct nvram_item {
  70. u8_t magic;
  71. u8_t state;
  72. u8_t crc;
  73. u8_t hash;
  74. u8_t reserved;
  75. u8_t name_size;
  76. u16_t data_size;
  77. char data[0];
  78. };
  79. '''
  80. def gen_item(key, value):
  81. # append '\0' to string
  82. key_data = bytearray(key, 'utf8') + bytearray(1)
  83. key_data_len = len(key_data)
  84. value_data = bytearray(value, 'utf8') + bytearray(1)
  85. value_data_len = len(value_data)
  86. hash = calc_hash(key_data)
  87. total_len = NVRAM_ITEM_HEADER_SIZE + key_data_len + value_data_len
  88. # 16 byte aligned
  89. pad_data = bytearray(0)
  90. if total_len & 0xf:
  91. pad_len = 16 - (total_len & 0xf)
  92. pad_data = bytearray(0xff for i in range(pad_len))
  93. total_len = total_len + pad_len
  94. item_header = struct.pack('<BBBH', hash, 0x00, key_data_len, value_data_len)
  95. item = item_header + key_data + value_data
  96. crc = calc_crc8(item, len(item));
  97. item = struct.pack('<BBB', NVRAM_REGION_ITEM_MAGIC, 0xff, crc) + item + pad_data
  98. return (item, total_len)
  99. def build_nvram_region(filename, props):
  100. fr_file = open(filename,'wb+')
  101. # write NVRAM region header
  102. seg_header = struct.pack('<BxBBH4x', NVRAM_REGION_SEG_VERSION, 0x0, 16, 4096)
  103. crc = calc_crc8(bytearray(seg_header), len(seg_header));
  104. fr_info = struct.pack('<IBB', NVRAM_REGION_SEG_MAGIC, 0xff, crc) + seg_header
  105. fr_file.seek(0, 0)
  106. fr_file.write(fr_info);
  107. buildprops = props.get_all()
  108. # write NVRAM items
  109. data_len = 0;
  110. for key, value in buildprops.items():
  111. print('NVRAM: Property: ' + key + '=' + value);
  112. (item, item_len) = gen_item(key, value)
  113. data_len = data_len + item_len
  114. fr_file.write(item);
  115. # padding region aligned to NVRAM_WRITE_REGION_ALIGN_SIZE
  116. pos = fr_file.tell()
  117. pad_len = 0
  118. if (pos % NVRAM_WRITE_REGION_ALIGN_SIZE):
  119. pad_len = NVRAM_WRITE_REGION_ALIGN_SIZE - pos % NVRAM_WRITE_REGION_ALIGN_SIZE
  120. fr_file.write(bytearray(0xff for i in range(pad_len)));
  121. file_len = pos + pad_len
  122. fr_file.close()
  123. def main(argv):
  124. parser = argparse.ArgumentParser(
  125. description='Pack config files to NVRAM binary data',
  126. )
  127. parser.add_argument('-o', dest = 'output_file')
  128. parser.add_argument('input_files', nargs = '*')
  129. args = parser.parse_args();
  130. print('NVRAM: Build Factory NVRAM binary file')
  131. lines = []
  132. for input_file in args.input_files:
  133. if not os.path.isfile(input_file):
  134. continue
  135. print('NVRAM: Process property file: %s' %input_file)
  136. with open(input_file) as f:
  137. lines = lines + f.readlines()
  138. properties = PropFile(lines)
  139. # write the merged property file
  140. properties.write(os.path.join(os.path.dirname(args.output_file), 'nvram.prop'))
  141. print('NVRAM: Generate NVRAM file: %s.' %args.output_file)
  142. build_nvram_region(args.output_file, properties)
  143. if __name__ == "__main__":
  144. main(sys.argv)