build_boot_image.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  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 array
  13. import argparse
  14. import zlib
  15. import configparser
  16. from ctypes import *;
  17. BINARY_BLOCK_SIZE_ALIGN = 512
  18. class BOOT_PARAMETERS(Structure): # import from ctypes
  19. _pack_ = 1
  20. _fields_ = [
  21. ("magic", c_uint32),
  22. ("uart_baudrate", c_uint32),
  23. ("uart_id", c_uint8),
  24. ("uart_mfp", c_uint8),
  25. ("jtag_groud", c_uint8),
  26. ("psram_mfp", c_uint8),
  27. ("adfu_txrx", c_uint8),
  28. ("adfu_gpio", c_uint8),
  29. ("reserved", c_uint8 * 14),
  30. ("checksum", c_uint32),
  31. ] # 32 bytes
  32. SIZEOF_BOOT_PARAMETERS_VERSION = 0x20
  33. BOOT_PARAMETERS_MAGIC = 0x52415042 #'BPAR'
  34. BOOT_PARAMETERS_OFFSET = 0x30
  35. def c_struct_crc(c_struct, length):
  36. crc_buf = (c_byte * length)()
  37. memmove(addressof(crc_buf), addressof(c_struct), length)
  38. return zlib.crc32(crc_buf, 0) & 0xffffffff
  39. def generate_boot_parameters(boot_name, boot_ini_name, out_ini_bin):
  40. if os.path.exists(boot_ini_name):
  41. boot_param = BOOT_PARAMETERS()
  42. memset(addressof(boot_param), 0, SIZEOF_BOOT_PARAMETERS_VERSION)
  43. boot_param.magic = BOOT_PARAMETERS_MAGIC
  44. configer = configparser.ConfigParser()
  45. configer.read(boot_ini_name)
  46. uart_baudrate = configer.get("serial config", "uart_baudrate")
  47. if uart_baudrate != None:
  48. boot_param.uart_baudrate = int(uart_baudrate, 0)
  49. print('uart id %s' %(uart_baudrate))
  50. uart_id = configer.get("serial config", "uart_id")
  51. if uart_id != None:
  52. boot_param.uart_id = int(uart_id, 0)
  53. print('uart id %s' %(uart_id))
  54. uart_mfp = configer.get("serial config", "uart_mfp")
  55. if uart_mfp != None:
  56. boot_param.uart_mfp = int(uart_mfp, 0)
  57. print('uart id %s' %(uart_mfp))
  58. jtag_groud = configer.get("jtag config", "jtag_groud")
  59. if jtag_groud != None:
  60. boot_param.jtag_groud = int(jtag_groud, 0)
  61. print('jtag groud %s' %(jtag_groud))
  62. psram_mfp = configer.get("psram_config", "psram_mfp")
  63. if psram_mfp != None:
  64. boot_param.psram_mfp = int(psram_mfp, 0)
  65. print('psram mfp %s' %(psram_mfp))
  66. if configer.has_option("adfu config", "adfu_txrx") :
  67. adfu_txrx = configer.get("adfu config", "adfu_txrx")
  68. boot_param.adfu_txrx = int(adfu_txrx, 0)
  69. print('adfu_txrx %s' %(adfu_txrx))
  70. if configer.has_option("adfu config", "adfu_gpio") :
  71. adfu_gpio = configer.get("adfu config", "adfu_gpio")
  72. boot_param.adfu_gpio = int(adfu_gpio, 0)
  73. print('adfu_gpio %s' %(adfu_gpio))
  74. boot_param.checksum = c_struct_crc(boot_param, SIZEOF_BOOT_PARAMETERS_VERSION - 4)
  75. if out_ini_bin is None :
  76. with open(boot_name, 'rb+') as f:
  77. f.seek(BOOT_PARAMETERS_OFFSET, 0)
  78. f.write(boot_param)
  79. f.close()
  80. else :
  81. with open(out_ini_bin, 'wb+') as f:
  82. f.write(boot_param)
  83. f.close()
  84. def boot_padding(filename, align = BINARY_BLOCK_SIZE_ALIGN):
  85. fsize = os.path.getsize(filename)
  86. if fsize % align:
  87. padding_size = align - (fsize % align)
  88. print('fsize %d, padding_size %d' %(fsize, padding_size))
  89. with open(filename, 'rb+') as f:
  90. f.seek(fsize, 0)
  91. buf = (c_byte * padding_size)();
  92. f.write(buf)
  93. f.close()
  94. def boot_append_nand_id_table(filename, id_table_name):
  95. if not os.path.exists(id_table_name):
  96. return None
  97. fsize = os.path.getsize(filename)
  98. id_table_size = os.path.getsize(id_table_name)
  99. if id_table_size != 0:
  100. print('padding nand id table size %d' %(id_table_size))
  101. with open(id_table_name, 'rb') as f:
  102. read_buf = f.read()
  103. f.close()
  104. with open(filename, 'rb+') as f:
  105. f.seek(fsize, 0)
  106. f.write(read_buf)
  107. f.close()
  108. def boot_calc_checksum(data):
  109. s = sum(array.array('H',data))
  110. s = s & 0xffff
  111. return s
  112. def lark_boot_post_build(boot_name, boot_ini_name, nand_id_name):
  113. boot_len = os.path.getsize(boot_name)
  114. print('lark boot loader origin length %d.' %boot_len)
  115. generate_boot_parameters(boot_name, boot_ini_name, None)
  116. boot_append_nand_id_table(boot_name, nand_id_name)
  117. boot_padding(boot_name)
  118. boot_len_new = os.path.getsize(boot_name)
  119. print('boot loader new length %d.' %boot_len_new)
  120. with open(boot_name, 'rb+') as f:
  121. f.seek(0, 0)
  122. data = f.read(32)
  123. h_magic0,h_magic1,h_load_addr,h_name,h_version, \
  124. h_header_size,h_header_chksm,h_data_chksm,\
  125. h_body_size,h_tail_size \
  126. = struct.unpack('III4sHHHHII',data)
  127. if(h_magic0 != 0x48544341 or h_magic1 != 0x41435448):
  128. print('boot loader header check fail.')
  129. return
  130. h_body_size = boot_len_new - h_header_size
  131. f.seek(0x18, 0)
  132. f.write(struct.pack('<I', h_body_size))
  133. # use tail_size to record origin boot size
  134. h_tail_size = boot_len
  135. f.seek(0x1c, 0)
  136. f.write(struct.pack('<I', h_tail_size))
  137. f.seek(h_header_size, 0)
  138. data = f.read(h_body_size)
  139. checksum = boot_calc_checksum(data)
  140. checksum = 0xffff - checksum
  141. f.seek(0x16, 0)
  142. f.write(struct.pack('<H', checksum))
  143. f.seek(0x14, 0)
  144. f.write(struct.pack('<H', 0))
  145. f.seek(0x0, 0)
  146. data = f.read(h_header_size)
  147. checksum = boot_calc_checksum(data)
  148. checksum = 0xffff - checksum
  149. f.seek(0x14, 0)
  150. f.write(struct.pack('<H', checksum))
  151. f.close()
  152. print('boot loader add cksum pass.')
  153. """
  154. ******lark use tlv*******************
  155. """
  156. """
  157. #define IMAGE_TLV_BOOTINI 0x1000 /* INI*/
  158. #define IMAGE_TLV_NANDID 0x2000 /* nand id */
  159. """
  160. IMAGE_TLV_BOOTINI=0x1000
  161. IMAGE_TLV_NANDID=0x2000
  162. IMAGE_TLV_INFO_MAGIC=0x5935
  163. IMAGE_TLV_PROT_INFO_MAGIC=0x593a
  164. def img_header_check(img_bin_f):
  165. """
  166. check img header, and add h_img_size
  167. """
  168. img_len = os.path.getsize(img_bin_f)
  169. print('img origin length %d' %(img_len))
  170. if img_len % 4:
  171. boot_padding(img_bin_f, 4)
  172. img_len = os.path.getsize(img_bin_f)
  173. print('img align 4 length %d' %(img_len))
  174. with open(img_bin_f, 'rb+') as f:
  175. f.seek(0, 0)
  176. data = f.read(48)
  177. h_magic0,h_magic1,h_load_addr,h_name,h_run_addr,h_img_size,h_img_chksum, h_hdr_chksum, \
  178. h_header_size,h_ptlv_size,h_tlv_size,h_version,h_flags \
  179. = struct.unpack('III8sIIIIHHHHI',data)
  180. if(h_magic0 != 0x48544341 or h_magic1 != 0x41435448):
  181. print('magic header check fail.')
  182. sys.exit(1)
  183. print('img h_header_size %d.' %h_header_size)
  184. h_img_size = img_len - h_header_size
  185. f.seek(0x18, 0)
  186. f.write(struct.pack('<I', h_img_size))
  187. print('img h_img_size %d.' %h_img_size)
  188. f.seek(0x26, 0)
  189. f.write(struct.pack('<h', 0))
  190. f.close()
  191. def Add_data_to_tlv(img_bin_f, tlv_data_bin_f, tlv_type):
  192. """
  193. add data tlv
  194. """
  195. if not os.path.exists(tlv_data_bin_f):
  196. return 0
  197. img_len = os.path.getsize(img_bin_f)
  198. tlv_len = os.path.getsize(tlv_data_bin_f)
  199. print('tlv data length %d, img_len=%d' %(tlv_len,img_len))
  200. with open(tlv_data_bin_f, 'rb') as f:
  201. tlv_data = f.read()
  202. f.close()
  203. with open(img_bin_f, 'rb+') as f:
  204. f.seek(0, 0)
  205. data = f.read(48)
  206. h_magic0,h_magic1,h_load_addr,h_name,h_run_addr,h_img_size,h_img_chksum, h_hdr_chksum, \
  207. h_header_size,h_ptlv_size,h_tlv_size,h_version,h_flags \
  208. = struct.unpack('III8sIIIIHHHHI',data)
  209. if(h_magic0 != 0x48544341 or h_magic1 != 0x41435448):
  210. print('magic header check fail.')
  211. sys.exit(1)
  212. plv_start = h_header_size + h_img_size + h_ptlv_size
  213. print('plv_start %d,header_size=%d, img_len=%d ptlv_size=%d' %(plv_start, h_header_size, h_img_size, h_ptlv_size))
  214. f.seek(plv_start, 0)
  215. tlv_all_size = tlv_len + 4
  216. tlv_tol_size = 0
  217. if(plv_start == img_len):#not tlv header
  218. print('add tlv magic')
  219. tlv_tol_size = tlv_all_size
  220. f.write(struct.pack('<h', IMAGE_TLV_INFO_MAGIC))
  221. f.write(struct.pack('<h', tlv_tol_size))
  222. else :
  223. data1 = f.read(4)
  224. tlv_magic, tlv_tol_size= struct.unpack('HH',data1)
  225. if(tlv_magic != IMAGE_TLV_INFO_MAGIC):
  226. print('tlv magic check fail.')
  227. sys.exit(1)
  228. print('old tlv_tol_size %d' %(tlv_tol_size))
  229. new_tlv_start = plv_start + tlv_tol_size + 4
  230. tlv_tol_size = tlv_tol_size + tlv_all_size
  231. f.seek(plv_start+2, 0)
  232. f.write(struct.pack('<h', tlv_tol_size))
  233. f.seek(new_tlv_start, 0)
  234. print('tlv_tol_size %d' %(tlv_tol_size))
  235. f.write(struct.pack('<h', tlv_type))
  236. f.write(struct.pack('<h', tlv_len))
  237. f.write(tlv_data)
  238. f.close()
  239. tlv_tol_size = tlv_tol_size + 4
  240. #if(tlv_tol_size > h_tlv_size):
  241. # print('tlv_tol_size %d > header tlv size %d'%(tlv_tol_size, h_tlv_size))
  242. # sys.exit(1)
  243. return tlv_tol_size
  244. return 0
  245. def image_calc_checksum(data):
  246. s = sum(array.array('I',data))
  247. s = s & 0xffffffff
  248. return s
  249. def image_add_cksum(filename, tlv_size):
  250. with open(filename, 'rb+') as f:
  251. f.seek(0, 0)
  252. data = f.read(48)
  253. h_magic0,h_magic1,h_load_addr,h_name,h_run_addr,h_img_size,h_img_chksum, h_hdr_chksum, \
  254. h_header_size,h_ptlv_size,h_tlv_size,h_version,h_flags \
  255. = struct.unpack('III8sIIIIHHHHI',data)
  256. if(h_magic0 != 0x48544341 or h_magic1 != 0x41435448):
  257. print('magic header check fail.')
  258. sys.exit(1)
  259. print('img header_size %d, img size=%d, ptlv_size=%d' %(h_header_size,h_img_size, h_ptlv_size))
  260. f.seek(h_header_size, 0)
  261. data = f.read(h_img_size)
  262. checksum = image_calc_checksum(data)
  263. checksum = 0xffffffff - checksum
  264. f.seek(0x1c, 0)
  265. f.write(struct.pack('<I', checksum))
  266. print('img checksum 0x%x.' %checksum)
  267. f.seek(0x28, 0)
  268. f.write(struct.pack('<H', tlv_size))
  269. f.seek(0x0, 0)
  270. data = f.read(h_header_size)
  271. checksum = image_calc_checksum(data)
  272. checksum = 0xffffffff - checksum
  273. f.seek(0x20, 0)
  274. f.write(struct.pack('<I', checksum))
  275. print('header checksum 0x%x.' %checksum)
  276. f.close()
  277. print('boot loader add cksum pass.')
  278. def lark_boot_post_build(boot_name, boot_ini_name, nand_id_name):
  279. print('lark boot add ini & nand id to tlv')
  280. img_header_check(boot_name)
  281. boot_ini_f = os.path.join(os.path.dirname(boot_name), 'bootini.bin')
  282. generate_boot_parameters(boot_name, boot_ini_name, boot_ini_f)
  283. tlv_size = Add_data_to_tlv(boot_name, boot_ini_f,IMAGE_TLV_BOOTINI)
  284. tmp_size = Add_data_to_tlv(boot_name, nand_id_name,IMAGE_TLV_NANDID)
  285. if (tmp_size != 0) :
  286. tlv_size = tmp_size
  287. image_add_cksum(boot_name, tlv_size)
  288. boot_padding(boot_name)
  289. def pearlriver_boot_post_build(boot_name, boot_ini_name):
  290. print('pearlriver boot add ini')
  291. img_header_check(boot_name)
  292. boot_ini_f = os.path.join(os.path.dirname(boot_name), 'bootini.bin')
  293. generate_boot_parameters(boot_name, boot_ini_name, boot_ini_f)
  294. tlv_size = Add_data_to_tlv(boot_name, boot_ini_f,IMAGE_TLV_BOOTINI)
  295. image_add_cksum(boot_name, tlv_size)
  296. boot_padding(boot_name)
  297. def boot_post_build(boot_name, boot_ini_name, nand_id_name):
  298. with open(boot_name, 'rb+') as f:
  299. f.seek(0, 0)
  300. data = f.read(16)
  301. f.close()
  302. h_magic0,h_magic1,h_load_addr,h_name = struct.unpack('III4s',data)
  303. if(h_magic0 != 0x48544341 or h_magic1 != 0x41435448):
  304. print('boot loader header check fail.')
  305. return
  306. print('boot name =%s'% h_name)
  307. if(h_name == b'boot'):
  308. lark_boot_post_build(boot_name, boot_ini_name, nand_id_name)
  309. elif(h_name == b'brec'):
  310. pearlriver_boot_post_build(boot_name, boot_ini_name)
  311. else:
  312. lark_boot_post_build(boot_name, boot_ini_name, nand_id_name)
  313. def main(argv):
  314. boot_post_build(argv[1], argv[2], argv[3])
  315. if __name__ == "__main__":
  316. main(sys.argv)