build_firmware.py 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212
  1. #!/usr/bin/env python3
  2. #
  3. # Build Actions SoC firmware (RAW/USB/OTA)
  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 time
  12. import struct
  13. import argparse
  14. import platform
  15. import subprocess
  16. import array
  17. import hashlib
  18. import shutil
  19. import zipfile
  20. import xml.etree.ElementTree as ET
  21. import zlib
  22. import json
  23. import traceback
  24. import csv
  25. from ctypes import *
  26. # private module
  27. from nvram_prop import *;
  28. CFG_TEST_GENERATE_OTA_FIRMWARES_COUNT = 3
  29. CFG_FIRMWARE_NEED_ENCYPT = False
  30. PRODUCTION_GLOBAL_BUFFER_DEFAULT_SIZE = 0x1000000
  31. PARTITION_ALIGNMENT = 0x1000
  32. class PARTITION_ENTRY(Structure): # import from ctypes
  33. _pack_ = 1
  34. _fields_ = [
  35. ("name", c_uint8 * 8),
  36. ("type", c_uint8),
  37. ("file_id", c_uint8),
  38. ("mirror_id", c_uint8, 4),
  39. ("storage_id", c_uint8, 4),
  40. ("flag", c_uint8),
  41. ("offset", c_uint32),
  42. ("size", c_uint32),
  43. ("entry_offs", c_uint32),
  44. ] # 64 bytes
  45. SIZEOF_PARTITION_ENTRY = 0x18
  46. class PARTITION_TABLE(Structure): # import from ctypes
  47. _pack_ = 1
  48. _fields_ = [
  49. ("magic", c_uint32),
  50. ("version", c_uint16),
  51. ("table_size", c_uint16),
  52. ("part_cnt", c_uint16),
  53. ("part_entry_size", c_uint16),
  54. ("reserved1", c_uint8 * 4),
  55. ("parts", PARTITION_ENTRY * 15),
  56. ("reserved2", c_uint8 * 4),
  57. ("table_crc", c_uint32),
  58. ] # 64 bytes
  59. SIZEOF_PARTITION_TABLE = 0x180
  60. PARTITION_TABLE_MAGIC = 0x54504341 #'ACPT'
  61. partition_type_table = {'RESERVED':0, 'BOOT':1, 'SYSTEM':2, 'RECOVERY':3, 'DATA':4, 'TEMP':5, 'SYS_PARAM':6}
  62. class FIRMWARE_VERSION(Structure): # import from ctypes
  63. _pack_ = 1
  64. _fields_ = [
  65. ("magic", c_uint32),
  66. ("version_code", c_uint32),
  67. ("version_res", c_uint32),
  68. ("sys_version_code",c_uint32),
  69. ("version_name", c_uint8 * 64),
  70. ("board_name", c_uint8 * 32),
  71. ("reserved", c_uint8 * 12),
  72. ("checksum", c_uint32),
  73. ] # 128 bytes
  74. SIZEOF_FIRMWARE_VERSION = 0x80
  75. FIRMWARE_VERSION_MAGIC = 0x52455646 #'FVER'
  76. class IMAGE_HEADER(Structure): # import from ctypes
  77. _pack_ = 1
  78. _fields_ = [
  79. ("magic", c_uint32),
  80. ("version", c_uint16),
  81. ("header_size", c_uint16),
  82. ("data_type", c_uint16),
  83. ("data_flag", c_uint16),
  84. ("data_offset", c_uint32),
  85. ("data_size", c_uint32),
  86. ("data_base_vaddr", c_uint32),
  87. ("data_entry_vaddr", c_uint32),
  88. ("data_checksum", c_uint32),
  89. ("reserved", c_uint8 * 28),
  90. ("header_crc", c_uint32),
  91. ] # 64 bytes
  92. SIZEOF_IMAGE_HEADER = 64
  93. IMAGE_HEADER_MAGIC = 0x4d494341 #'ACIM'
  94. MP_CARD_CFG_NAME="mp_card.cfg"
  95. FW_MAKER_EXT_CFG_NAME="fw_maker_ext.cfg"
  96. FW_BUILD_TIME_FILE_NAME="fw_build_time.bin";
  97. script_path = os.path.split(os.path.realpath(__file__))[0]
  98. soc_name = ''
  99. board_name = ''
  100. encrypt_fw = ''
  101. efuse_bin = ''
  102. # table for calculating CRC
  103. CRC16_TABLE = [
  104. 0x0000, 0x1189, 0x2312, 0x329b, 0x4624, 0x57ad, 0x6536, 0x74bf,
  105. 0x8c48, 0x9dc1, 0xaf5a, 0xbed3, 0xca6c, 0xdbe5, 0xe97e, 0xf8f7,
  106. 0x1081, 0x0108, 0x3393, 0x221a, 0x56a5, 0x472c, 0x75b7, 0x643e,
  107. 0x9cc9, 0x8d40, 0xbfdb, 0xae52, 0xdaed, 0xcb64, 0xf9ff, 0xe876,
  108. 0x2102, 0x308b, 0x0210, 0x1399, 0x6726, 0x76af, 0x4434, 0x55bd,
  109. 0xad4a, 0xbcc3, 0x8e58, 0x9fd1, 0xeb6e, 0xfae7, 0xc87c, 0xd9f5,
  110. 0x3183, 0x200a, 0x1291, 0x0318, 0x77a7, 0x662e, 0x54b5, 0x453c,
  111. 0xbdcb, 0xac42, 0x9ed9, 0x8f50, 0xfbef, 0xea66, 0xd8fd, 0xc974,
  112. 0x4204, 0x538d, 0x6116, 0x709f, 0x0420, 0x15a9, 0x2732, 0x36bb,
  113. 0xce4c, 0xdfc5, 0xed5e, 0xfcd7, 0x8868, 0x99e1, 0xab7a, 0xbaf3,
  114. 0x5285, 0x430c, 0x7197, 0x601e, 0x14a1, 0x0528, 0x37b3, 0x263a,
  115. 0xdecd, 0xcf44, 0xfddf, 0xec56, 0x98e9, 0x8960, 0xbbfb, 0xaa72,
  116. 0x6306, 0x728f, 0x4014, 0x519d, 0x2522, 0x34ab, 0x0630, 0x17b9,
  117. 0xef4e, 0xfec7, 0xcc5c, 0xddd5, 0xa96a, 0xb8e3, 0x8a78, 0x9bf1,
  118. 0x7387, 0x620e, 0x5095, 0x411c, 0x35a3, 0x242a, 0x16b1, 0x0738,
  119. 0xffcf, 0xee46, 0xdcdd, 0xcd54, 0xb9eb, 0xa862, 0x9af9, 0x8b70,
  120. 0x8408, 0x9581, 0xa71a, 0xb693, 0xc22c, 0xd3a5, 0xe13e, 0xf0b7,
  121. 0x0840, 0x19c9, 0x2b52, 0x3adb, 0x4e64, 0x5fed, 0x6d76, 0x7cff,
  122. 0x9489, 0x8500, 0xb79b, 0xa612, 0xd2ad, 0xc324, 0xf1bf, 0xe036,
  123. 0x18c1, 0x0948, 0x3bd3, 0x2a5a, 0x5ee5, 0x4f6c, 0x7df7, 0x6c7e,
  124. 0xa50a, 0xb483, 0x8618, 0x9791, 0xe32e, 0xf2a7, 0xc03c, 0xd1b5,
  125. 0x2942, 0x38cb, 0x0a50, 0x1bd9, 0x6f66, 0x7eef, 0x4c74, 0x5dfd,
  126. 0xb58b, 0xa402, 0x9699, 0x8710, 0xf3af, 0xe226, 0xd0bd, 0xc134,
  127. 0x39c3, 0x284a, 0x1ad1, 0x0b58, 0x7fe7, 0x6e6e, 0x5cf5, 0x4d7c,
  128. 0xc60c, 0xd785, 0xe51e, 0xf497, 0x8028, 0x91a1, 0xa33a, 0xb2b3,
  129. 0x4a44, 0x5bcd, 0x6956, 0x78df, 0x0c60, 0x1de9, 0x2f72, 0x3efb,
  130. 0xd68d, 0xc704, 0xf59f, 0xe416, 0x90a9, 0x8120, 0xb3bb, 0xa232,
  131. 0x5ac5, 0x4b4c, 0x79d7, 0x685e, 0x1ce1, 0x0d68, 0x3ff3, 0x2e7a,
  132. 0xe70e, 0xf687, 0xc41c, 0xd595, 0xa12a, 0xb0a3, 0x8238, 0x93b1,
  133. 0x6b46, 0x7acf, 0x4854, 0x59dd, 0x2d62, 0x3ceb, 0x0e70, 0x1ff9,
  134. 0xf78f, 0xe606, 0xd49d, 0xc514, 0xb1ab, 0xa022, 0x92b9, 0x8330,
  135. 0x7bc7, 0x6a4e, 0x58d5, 0x495c, 0x3de3, 0x2c6a, 0x1ef1, 0x0f78,
  136. ]
  137. def reflect(crc):
  138. """
  139. :type n: int
  140. :rtype: int
  141. """
  142. m = ['0' for i in range(16)]
  143. b = bin(crc)[2:]
  144. m[:len(b)] = b[::-1]
  145. return int(''.join(m) ,2)
  146. def _crc16(data, crc, table):
  147. """Calculate CRC16 using the given table.
  148. `data` - data for calculating CRC, must be bytes
  149. `crc` - initial value
  150. `table` - table for caclulating CRC (list of 256 integers)
  151. Return calculated value of CRC
  152. polynom : 0x1021
  153. order : 16
  154. crcinit : 0xffff
  155. crcxor : 0x0
  156. refin : 1
  157. refout : 1
  158. """
  159. crc = reflect(crc)
  160. for byte in data:
  161. crc = ((crc >> 8) & 0xff) ^ table[(crc & 0xff) ^ byte]
  162. crc = reflect(crc)
  163. # swap byte
  164. crc = ((crc >> 8) & 0xff) | ((crc & 0xff) << 8)
  165. return crc
  166. def crc16(data, crc=0xffff):
  167. """Calculate CRC16.
  168. `data` - data for calculating CRC, must be bytes
  169. `crc` - initial value
  170. Return calculated value of CRC
  171. """
  172. return _crc16(data, crc, CRC16_TABLE)
  173. def md5_file(filename):
  174. if os.path.isfile(filename):
  175. with open(filename, 'rb') as f:
  176. md5 = hashlib.md5()
  177. md5.update(f.read())
  178. hash = md5.hexdigest()
  179. return str(hash)
  180. return None
  181. def crc32_file(filename):
  182. if os.path.isfile(filename):
  183. with open(filename, 'rb') as f:
  184. crc = zlib.crc32(f.read(), 0) & 0xffffffff
  185. return crc
  186. return 0
  187. def pad_file(filename, align = 4, fillbyte = 0xff):
  188. with open(filename, 'ab') as f:
  189. filesize = f.tell()
  190. if (filesize % align):
  191. padsize = align - filesize & (align - 1)
  192. f.write(bytearray([fillbyte]*padsize))
  193. def new_file(filename, filesize, fillbyte = 0xff):
  194. with open(filename, 'wb') as f:
  195. f.write(bytearray([fillbyte]*filesize))
  196. def dd_file(input_file, output_file, count=None, seek=None, skip=None):
  197. inf = open(input_file, mode='rb')
  198. outf = open(output_file, mode='rb+')
  199. if skip is not None:
  200. inf.seek(skip)
  201. if seek is not None:
  202. outf.seek(seek)
  203. if count is None:
  204. count = os.path.getsize(input_file)
  205. outf.write(inf.read(count))
  206. inf.close
  207. outf.close
  208. def zip_dir(source_dir, output_filename):
  209. zf = zipfile.ZipFile(output_filename, 'w')
  210. pre_len = len(os.path.dirname(source_dir))
  211. for parent, dirnames, filenames in os.walk(source_dir):
  212. for filename in filenames:
  213. pathfile = os.path.join(parent, filename)
  214. arcname = pathfile[pre_len:].strip(os.path.sep)
  215. zf.write(pathfile, arcname)
  216. zf.close()
  217. def memcpy_n(cbuffer, bufsize, pylist):
  218. size = min(bufsize, len(pylist))
  219. for i in range(size):
  220. cbuffer[i]= ord(pylist[i])
  221. def c_struct_crc(c_struct, length):
  222. crc_buf = (c_byte * length)()
  223. memmove(addressof(crc_buf), addressof(c_struct), length)
  224. return zlib.crc32(crc_buf, 0) & 0xffffffff
  225. def align_down(data, alignment):
  226. return data // alignment * alignment
  227. def align_up(data, alignment):
  228. return align_down(data + alignment - 1, alignment)
  229. def run_cmd(cmd):
  230. """Echo and run the given command.
  231. Args:
  232. cmd: the command represented as a list of strings.
  233. Returns:
  234. A tuple of the output and the exit code.
  235. """
  236. # print("Running: ", " ".join(cmd))
  237. p = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
  238. output, _ = p.communicate()
  239. # print("%s" % (output.rstrip()))
  240. return (output, p.returncode)
  241. def panic(err_msg):
  242. print('\033[1;31;40m')
  243. print('FW: Error: %s\n' %err_msg)
  244. print('\033[0m')
  245. sys.exit(1)
  246. def print_notice(msg):
  247. print('\033[1;32;40m%s\033[0m' %msg)
  248. def cygpath(upath):
  249. cmd = ['cygpath', '-w', upath]
  250. (wpath, exit_code) = run_cmd(cmd)
  251. if (0 != exit_code):
  252. return upath
  253. return wpath.decode().strip()
  254. def is_windows():
  255. sysstr = platform.system()
  256. if (sysstr.startswith('Windows') or \
  257. sysstr.startswith('MSYS') or \
  258. sysstr.startswith('MINGW') or \
  259. sysstr.startswith('CYGWIN')):
  260. return True
  261. else:
  262. return False
  263. def is_msys():
  264. sysstr = platform.system()
  265. if (sysstr.startswith('MSYS') or \
  266. sysstr.startswith('MINGW') or \
  267. sysstr.startswith('CYGWIN')):
  268. return True
  269. else:
  270. return False
  271. def soc_is_andes():
  272. if soc_name == 'andes':
  273. return True
  274. else:
  275. return False
  276. class firmware(object):
  277. def __init__(self, cfg_file):
  278. self.crc_chunk_size = 32
  279. self.crc_full_chunk_size = self.crc_chunk_size + 2
  280. self.part_num = 0
  281. self.partitions = []
  282. self.disk_size = 0x400000 # 4MB by default
  283. self.fw_dir = os.path.dirname(cfg_file)
  284. self.bin_dir = os.path.join(self.fw_dir, 'bin')
  285. self.ota_dir = os.path.join(self.fw_dir, 'ota')
  286. self.orig_bin_dir = self.bin_dir + '_orig'
  287. self.orig_ota_dir = self.ota_dir + '_orig'
  288. self.att_dir = os.path.join(self.bin_dir, 'ATT')
  289. self.fw_version = {}
  290. self.boot_file_name = ''
  291. self.param_file_name = ''
  292. self.is_earphone_app = False
  293. self.upgrade_baudrate = 1000000
  294. self.atf_file_list = []
  295. if "bt_earphone" in os.path.abspath(self.fw_dir).split('\\'):
  296. self.is_earphone_app = True
  297. print("is earphone SDK")
  298. if not os.path.isdir(self.att_dir):
  299. os.mkdir(self.att_dir)
  300. if not os.path.isdir(self.ota_dir):
  301. os.mkdir(self.ota_dir)
  302. if not os.path.isdir(self.orig_ota_dir):
  303. os.mkdir(self.orig_ota_dir)
  304. # self.parse_config(cfg_file)
  305. def parse_config(self, cfg_file, pre_crc = False, pre_encrypt = False):
  306. print('FW: Parse config file: %s' %cfg_file)
  307. tree = ET.ElementTree(file=cfg_file)
  308. root = tree.getroot()
  309. if (root.tag != 'firmware'):
  310. sys.stderr.write('error: invalid firmware config file')
  311. sys.exit(1)
  312. disk_size_prop = root.find('disk_size')
  313. if disk_size_prop is not None:
  314. self.disk_size = int(disk_size_prop.text.strip(), 0)
  315. firmware_version = root.find('firmware_version')
  316. for prop in firmware_version:
  317. self.fw_version[prop.tag] = prop.text.strip()
  318. if 'version_name' in self.fw_version.keys():
  319. cur_time = time.strftime('%y%m%d%H%M',time.localtime(time.time()))
  320. version_name = self.fw_version['version_name'].replace('$(build_time)', cur_time)
  321. self.fw_version['version_name'] = version_name
  322. # gen fw build time file
  323. fw_build_time_file = os.path.join(self.fw_dir, FW_BUILD_TIME_FILE_NAME)
  324. with open(fw_build_time_file, 'w') as f:
  325. f.write(cur_time)
  326. part_list = root.find('partitions').findall('partition')
  327. for part in part_list:
  328. part_prop = {}
  329. for prop in part:
  330. part_prop[prop.tag] = prop.text.strip()
  331. if 'file_name' in part_prop.keys():
  332. bin_file = os.path.join(self.bin_dir, part_prop['file_name'])
  333. else:
  334. bin_file = None
  335. if bin_file and ('SYS_PARAM' != part_prop['type']) and not os.path.exists(bin_file):
  336. print_notice('partition %s ignored: cannot found bin %s' %(part_prop['name'], bin_file))
  337. else:
  338. self.partitions.append(part_prop)
  339. self.part_num = self.part_num + 1
  340. self.part_num = len(self.partitions);
  341. # print(self.part_num)
  342. # print(self.partitions)
  343. if (0 == self.part_num):
  344. panic('cannot found parition config')
  345. for part in self.partitions:
  346. if ('file_name' in part.keys()) and ('BOOT' == part['type']):
  347. self.boot_file_name = part['file_name']
  348. for part in self.partitions:
  349. if ('file_name' in part.keys()) and ('SYS_PARAM' == part['type']):
  350. self.param_file_name = part['file_name']
  351. param_file = os.path.join(self.bin_dir, self.param_file_name)
  352. self.generate_partition_table(param_file)
  353. self.generate_firmware_version(param_file)
  354. if os.path.isdir(self.orig_bin_dir):
  355. shutil.rmtree(self.orig_bin_dir)
  356. time.sleep(0.1)
  357. shutil.copytree(self.bin_dir, self.orig_bin_dir)
  358. if pre_crc:
  359. self.generate_crc_file()
  360. if pre_encrypt:
  361. if os.path.isfile(efuse_bin):
  362. self.encrypt_files()
  363. elif CFG_FIRMWARE_NEED_ENCYPT:
  364. panic('firmware encryt must be enabled')
  365. self.check_part_file_size()
  366. def check_part_file_size(self):
  367. csv_file = os.path.join(self.bin_dir, 'partition_size.csv')
  368. csv_f = open(csv_file, 'w', newline = '')
  369. csv_writer = csv.writer(csv_f, dialect = 'excel')
  370. csv_writer.writerow(['partition', 'size', 'real size', 'free'])
  371. for part in self.partitions:
  372. if not 'file_name' in part.keys():
  373. continue
  374. partfile = os.path.join(self.bin_dir, part['file_name']);
  375. if os.path.isfile(partfile):
  376. partfile_size = os.path.getsize(partfile)
  377. part_size = int(part['size'], 0);
  378. part_addr = int(part['address'], 0);
  379. file_addr = int(part['file_address'], 0);
  380. if part['type'] == 'SYSTEM':
  381. ksdfs_bin_file = os.path.join(self.bin_dir, "ksdfs.bin")
  382. if os.path.isfile(ksdfs_bin_file):
  383. partfile_size += os.path.getsize(ksdfs_bin_file)
  384. if file_addr < part_addr or file_addr >= (part_addr + part_size) :
  385. panic('partition %s: invalid file_address 0x%x' \
  386. %(part['name'], file_addr))
  387. part_file_max_size = part_size - (file_addr - part_addr)
  388. csv_writer.writerow([part['file_name'],
  389. "%.2f" %(part_file_max_size / 1024),
  390. "%.2f" %(partfile_size / 1024),
  391. "%.2f" %((part_file_max_size - partfile_size) / 1024)])
  392. print('partition %s: \'%s\' file size 0x%x(%d KB), max size 0x%x(%d KB)!' \
  393. %(part['name'], part['file_name'], partfile_size, partfile_size/1024, \
  394. part_file_max_size, part_file_max_size/1024))
  395. if partfile_size > part_file_max_size:
  396. csv_f.close()
  397. panic('partition %s: \'%s\' file size 0x%x is bigger than partition size 0x%x!' \
  398. %(part['name'], part['file_name'], partfile_size, part_size))
  399. csv_f.close()
  400. def boot_calc_checksum(self, data):
  401. s = sum(array.array('H',data))
  402. s = s & 0xffff
  403. return s
  404. def get_nvram_prop(self, name):
  405. prop_file = os.path.join(self.bin_dir, 'nvram.prop');
  406. if os.path.isfile(prop_file):
  407. with open(prop_file) as f:
  408. properties = PropFile(f.readlines())
  409. return properties.get(name)
  410. return ''
  411. def generate_crc_file(self):
  412. crc_files = []
  413. for part in self.partitions:
  414. if ('file_name' in part.keys()) and ('true' == part['enable_crc']):
  415. crc_files.append(part['file_name'])
  416. crc_files = list(set(crc_files))
  417. for crc_file in crc_files:
  418. self.add_crc(os.path.join(self.bin_dir, crc_file), self.crc_chunk_size)
  419. def encrypt_file(self, file_path, blk_size):
  420. #print ('FW: encrypt file %s' %(file_path))
  421. if not os.path.isfile(encrypt_fw):
  422. panic('Cannot found encrypt fw')
  423. if (is_windows()):
  424. # windows
  425. fw2x_path = script_path + '/utils/windows/fw2x.exe'
  426. else:
  427. if ('32bit' == platform.architecture()[0]):
  428. # linux x86
  429. fw2x_path = script_path + '/utils/linux-x86/fw2x/fw2x'
  430. else:
  431. # linux x86_64
  432. fw2x_path = script_path + '/utils/linux-x86_64/fw2x/fw2x'
  433. # extrace lfi.bin from btsys.fw
  434. fw2x_cmd = [fw2x_path, '-fsssfff', encrypt_fw, 'MakeEncryptBin',\
  435. 'Encrypt', str(blk_size), file_path, file_path, efuse_bin]
  436. (outmsg, exit_code) = run_cmd(fw2x_cmd)
  437. if exit_code != 0:
  438. print('FW2X: encrypt file %s error' %(file_path))
  439. print(outmsg)
  440. sys.exit(1)
  441. def encrypt_files(self):
  442. for part in self.partitions:
  443. if CFG_FIRMWARE_NEED_ENCYPT:
  444. if ('BOOT' == part['type'] or 'SYSTEM' == part['type'] or 'RECOVERY' == part['type']) or 'SYS_PARAM' == part['type'] \
  445. and ('true' != part['enable_encryption']):
  446. panic('BOOT/SYSTEM partition must enable encryption')
  447. sys.exit(3)
  448. if not 'file_name' in part.keys():
  449. continue
  450. if ('true' == part['enable_encryption']):
  451. print ('FW: encrypt file \'' + part['file_name'] + '\'')
  452. if ('true' == part['enable_crc']):
  453. self.encrypt_file(os.path.join(self.bin_dir, part['file_name']), self.crc_full_chunk_size)
  454. else:
  455. self.encrypt_file(os.path.join(self.bin_dir, part['file_name']), self.crc_chunk_size)
  456. def encrypt_param_file(self, param_file):
  457. for part in self.partitions:
  458. if ('file_name' in part.keys() and 'SYS_PARAM' == part['type']):
  459. if ('true' == part['enable_encryption']):
  460. if ('true' == part['enable_crc']):
  461. self.encrypt_file(param_file, self.crc_full_chunk_size)
  462. else:
  463. self.encrypt_file(param_file, self.crc_chunk_size)
  464. def generate_partition_table(self, param_file):
  465. part_tbl = PARTITION_TABLE()
  466. memset(addressof(part_tbl), 0, SIZEOF_PARTITION_TABLE)
  467. part_tbl.magic = PARTITION_TABLE_MAGIC
  468. part_tbl.ver = 0x0100
  469. part_tbl.table_size = SIZEOF_PARTITION_TABLE
  470. part_tbl.table_crc = 0x0
  471. part_tbl.part_entry_size = SIZEOF_PARTITION_ENTRY
  472. idx = 0
  473. for part in self.partitions:
  474. memcpy_n(part_tbl.parts[idx].name, 8, part['name'])
  475. part_tbl.parts[idx].type = partition_type_table[part['type']]
  476. part_tbl.parts[idx].file_id = int(part['file_id'])
  477. if 'mirror_id' in part.keys():
  478. part_tbl.parts[idx].mirror_id = int(part['mirror_id'])
  479. else:
  480. part_tbl.parts[idx].mirror_id = 0xf
  481. if 'storage_id' in part.keys():
  482. part_tbl.parts[idx].storage_id = int(part['storage_id'])
  483. else:
  484. part_tbl.parts[idx].storage_id = 0x0
  485. print('part [%d] mirror id %d' %(idx, part_tbl.parts[idx].mirror_id))
  486. part_tbl.parts[idx].flag = 0
  487. if 'enable_crc' in part.keys() and 'true' == part['enable_crc']:
  488. part_tbl.parts[idx].flag |= 0x1
  489. if 'enable_encryption' in part.keys() and 'true' == part['enable_encryption']:
  490. part_tbl.parts[idx].flag |= 0x2
  491. if 'enable_boot_check' in part.keys() and 'true' == part['enable_boot_check']:
  492. part_tbl.parts[idx].flag |= 0x4
  493. part_addr = int(part['address'], 0)
  494. part_size = int(part['size'], 0)
  495. file_addr = int(part['file_address'], 0)
  496. if part_addr & (PARTITION_ALIGNMENT - 1):
  497. panic('partition %s: unaligned part address 0x%x, need be aligned with 0x%x' \
  498. %(part['name'], part_addr, PARTITION_ALIGNMENT))
  499. if part_size & (PARTITION_ALIGNMENT - 1):
  500. panic('partition %s: unaligned part size 0x%x, need be aligned with 0x%x' \
  501. %(part['name'], part_size, PARTITION_ALIGNMENT))
  502. if file_addr < part_addr or file_addr >= (part_addr + part_size) :
  503. panic('partition %s: file_address 0x%x is not in part' \
  504. %(part['name'], file_addr))
  505. part_tbl.parts[idx].offset = part_addr
  506. part_tbl.parts[idx].size = part_size
  507. part_tbl.parts[idx].entry_offs = file_addr
  508. #print('%d: 0x%x, 0x%x' %(idx, part_tbl.parts[idx].offset, part_tbl.parts[idx].size))
  509. idx = idx + 1
  510. part_tbl.part_cnt = idx
  511. part_tbl.table_crc = c_struct_crc(part_tbl, SIZEOF_PARTITION_TABLE - 4)
  512. """
  513. with open('parttbl.bin', 'wb') as f:
  514. f.write(part_tbl)
  515. """
  516. with open(param_file, 'wb') as f:
  517. f.seek(0, 0)
  518. f.write(part_tbl)
  519. def generate_firmware_version(self, param_file):
  520. global board_name
  521. if 'version_code' not in self.fw_version:
  522. print('\033[1;31;40minfo: no version code\033[0m');
  523. self.fw_version['version_code'] = '0'
  524. if 'version_res' not in self.fw_version:
  525. print('\033[1;31;40minfo: no version res\033[0m');
  526. self.fw_version['version_res'] = '0'
  527. print(self.fw_version)
  528. fw_ver = FIRMWARE_VERSION()
  529. memset(addressof(fw_ver), 0, SIZEOF_FIRMWARE_VERSION)
  530. fw_ver.magic = FIRMWARE_VERSION_MAGIC
  531. fw_ver.version_code = int(self.fw_version['version_code'], 0)
  532. fw_ver.version_res = int(self.fw_version['version_res'], 0)
  533. version_name = self.fw_version['version_name']
  534. if len(version_name) > len(fw_ver.version_name):
  535. panic('max version_name is t' %(len(fw_ver.version_name)))
  536. memcpy_n(fw_ver.version_name, len(self.fw_version['version_name']), \
  537. self.fw_version['version_name'])
  538. memcpy_n(fw_ver.board_name, len(board_name), board_name)
  539. self.fw_version['board_name'] = board_name
  540. fw_ver.checksum = c_struct_crc(fw_ver, SIZEOF_FIRMWARE_VERSION - 4)
  541. #time.strftime('%y%m%d-%H%M%S',time.localtime(time.time()))
  542. """
  543. with open('fw.bin', 'wb') as f:
  544. f.write(fw_ver)
  545. """
  546. with open(param_file, 'rb+') as f:
  547. f.seek(SIZEOF_PARTITION_TABLE, 0)
  548. f.write(fw_ver)
  549. def generate_maker_ext_config(self, m_ext_cfg_file):
  550. with open(m_ext_cfg_file, 'a') as f:
  551. f.write('\n//nvram partition infor\n')
  552. nvram_partition_find = 0
  553. for part in self.partitions:
  554. if (('nvram_factory' == part['name'])):
  555. f.write('NVRAM_FACTORY_RO=' + str(part['address']) + ',' + str(part['size']) + ';\n')
  556. nvram_partition_find = 1;
  557. if (nvram_partition_find == 0):
  558. f.write('NVRAM_FACTORY_RO=0,0;\n')
  559. nvram_partition_find = 0
  560. for part in self.partitions:
  561. if (('nvram_user' == part['name'])):
  562. f.write('NVRAM_USER_RW=' + str(part['address']) + ',' + str(part['size']) + ';\n')
  563. nvram_partition_find = 1;
  564. if (nvram_partition_find == 0):
  565. f.write('NVRAM_USER_RW=0,0;\n')
  566. nvram_partition_find = 0
  567. for part in self.partitions:
  568. if (('nvram_factory_rw' == part['name'])):
  569. f.write('NVRAM_FACTORY_RW=' + str(part['address']) + ',' + str(part['size']) + ';\n')
  570. nvram_partition_find = 1;
  571. if (nvram_partition_find == 0):
  572. f.write('NVRAM_FACTORY_RW=0,0;\n')
  573. def generate_maker_config(self, mcfg_file):
  574. with open(mcfg_file, 'w') as f:
  575. f.write('SETPATH=\".\\";\n')
  576. f.write('SPI_STG_CAP=' + str(self.disk_size // 0x200) + ';\n')
  577. f.write('BASEFILE=\"afi.bin\";\n')
  578. """
  579. storage_id definition:
  580. - 0: NOR
  581. - 1: EMMC
  582. - 2: NAND
  583. INF_STORAGE_SOLUTION definition:
  584. - 0: NAND
  585. - 1: NOR
  586. - 2: MMC
  587. - 3: NOR + NAND
  588. - 4: NOR + EMMC
  589. """
  590. boot_storage_id = 0
  591. data_storage_id = 0
  592. # check boot storage id
  593. for part in self.partitions:
  594. if (('file_name' in part.keys()) and ('true' == part['enable_dfu']) and ('BOOT' == part['type'])):
  595. boot_storage_id = int(part['storage_id'], 0)
  596. # write normal partition first
  597. for part in self.partitions:
  598. if (('file_name' in part.keys()) and ('true' == part['enable_dfu']) and ('BOOT' != part['type'])):
  599. if (boot_storage_id == int(part['storage_id'], 0)) :
  600. f.write('WFILE=\"' + part['file_name'] + '\",' + part['file_address'])
  601. f.write(';\n')
  602. # nor boot ,can have extern data storage
  603. if (boot_storage_id == 0):
  604. for part in self.partitions:
  605. if (('file_name' in part.keys()) and ('true' == part['enable_dfu']) and ('BOOT' != part['type'])):
  606. if (boot_storage_id != int(part['storage_id'], 0) ) :
  607. f.write('EXTERN_DATA_FILE=\"' + part['file_name'] + '\",' + part['file_address'])
  608. f.write(';\n')
  609. data_storage_id = int(part['storage_id'], 0)
  610. # write boot partition
  611. for part in self.partitions:
  612. if (('file_name' in part.keys()) and ('true' == part['enable_dfu']) and ('BOOT' == part['type'])):
  613. if (boot_storage_id == int(part['storage_id'], 0)):
  614. f.write('WFILE=\"' + part['file_name'] + '\",' + part['file_address'])
  615. f.write(';\n')
  616. # check the current solution such as NOR-only, NAND-only, NOR + NAND
  617. # check the current solution such as NOR-only, EMMC-only NAND-only, NOR + NAND, NOR + EMMC
  618. if (boot_storage_id == 0):
  619. if (data_storage_id == 0):
  620. f.write('INF_TDK_STORAGE_TYPE=1' + ';\n') # NOR-only solution
  621. elif (data_storage_id == 1):
  622. f.write('INF_TDK_STORAGE_TYPE=4' + ';\n') # NOR + EMMC solution
  623. else :
  624. f.write('INF_TDK_STORAGE_TYPE=3' + ';\n') # NOR + NAND solution
  625. f.write('EXTERN_DATA_STG_CAP=0x80000' + ';\n')
  626. elif (boot_storage_id == 1): # EMMC-only solution
  627. f.write('INF_TDK_STORAGE_TYPE=2' + ';\n') # EMMC-only solution
  628. else: # NAND-only solution
  629. f.write('INF_TDK_STORAGE_TYPE=0' + ';\n') # NAND-only solution
  630. global_buffer_size = max(self.disk_size, 0)
  631. if global_buffer_size > PRODUCTION_GLOBAL_BUFFER_DEFAULT_SIZE:
  632. f.write('INF_GLOBAL_BUFFER_SIZE=' + str(global_buffer_size) + ';\n')
  633. # The length of 'VER' field is limited to 32 bytes
  634. if len(self.fw_version['version_name']) > 32:
  635. fw_ver_temp = self.fw_version['version_name'][0:31]
  636. else:
  637. fw_ver_temp = self.fw_version['version_name']
  638. f.write('VER=\"' + fw_ver_temp + '\"' + ';\n');
  639. #add mp card config
  640. if os.path.exists(os.path.join(self.bin_dir, MP_CARD_CFG_NAME)):
  641. f.write('#include \"' + MP_CARD_CFG_NAME + '\"\n')
  642. #add extern data config
  643. if os.path.exists(os.path.join(self.bin_dir, FW_MAKER_EXT_CFG_NAME)):
  644. f.write('#include \"' + FW_MAKER_EXT_CFG_NAME + '\"\n')
  645. def generate_raw_image(self, rawfw_name):
  646. print('FW: Build raw spinor image')
  647. rawfw_file = os.path.join(self.fw_dir, rawfw_name)
  648. if os.path.exists(rawfw_file):
  649. os.remove(rawfw_file)
  650. new_file(rawfw_file, self.disk_size, 0xff)
  651. for part in self.partitions:
  652. if ('file_name' in part.keys()) and ('true' == part['enable_raw']) and (int(part['storage_id'], 0) == 0 or int(part['storage_id'], 0) == 2):
  653. addr = int(part["address"], 16)
  654. srcfile = os.path.join(self.bin_dir, part['file_name'])
  655. dd_file(srcfile, rawfw_file, seek=addr)
  656. if not os.path.exists(rawfw_file):
  657. panic('Failed to generate SPINOR raw image')
  658. def generate_firmware(self, fw_name):
  659. print('FW: Build USB DFU image')
  660. maker_cfgfile = os.path.join(self.bin_dir, 'fw_maker.cfg')
  661. self.generate_maker_config(maker_cfgfile);
  662. maker_ext_cfgfile = os.path.join(self.bin_dir, FW_MAKER_EXT_CFG_NAME)
  663. self.generate_maker_ext_config(maker_ext_cfgfile);
  664. fw_file = os.path.join(self.fw_dir, fw_name);
  665. if (is_windows()):
  666. cmd = [script_path + '/utils/windows/maker.exe', '-c', maker_cfgfile, \
  667. '-o', fw_file, '-mf']
  668. else:
  669. if os.path.exists(fw_file):
  670. os.remove(fw_file)
  671. if ('32bit' == platform.architecture()[0]):
  672. # linux x86
  673. maker_path = script_path + '/utils/linux-x86/maker/PyMaker.pyo'
  674. else:
  675. # linux x86_64
  676. maker_path = script_path + '/utils/linux-x86_64/maker/PyMaker.pyo'
  677. cmd = ['python2', '-O', maker_path, '-c', maker_cfgfile, '-o', fw_file, '--mf', '1']
  678. (outmsg, exit_code) = run_cmd(cmd)
  679. if not os.path.exists(fw_file):
  680. panic('Maker error: %s' %(outmsg))
  681. print_notice('Build successfully!')
  682. if is_msys():
  683. print_notice('Firmware: ' + cygpath(fw_file))
  684. else:
  685. print_notice('Firmware: ' + fw_file)
  686. def generate_ota_xml(self, ota_dir):
  687. root = ET.Element('ota_firmware')
  688. root.text = '\n\t'
  689. root.tail = '\n'
  690. fw_ver = ET.SubElement(root, 'firmware_version')
  691. fw_ver.text = '\n\t\t'
  692. fw_ver.tail = '\n'
  693. for v in self.fw_version:
  694. type = ET.SubElement(fw_ver, v)
  695. type.text = self.fw_version[v]
  696. type.tail = '\n\t\t'
  697. parts = ET.SubElement(root, 'partitions')
  698. parts.text = '\n\t'
  699. parts.tail = '\n\t'
  700. ota_part = 0
  701. for part in self.partitions:
  702. if ('file_name' in part.keys()) and ('true' == part['enable_ota']):
  703. ota_part += 1
  704. # write part_num firstly for fast search
  705. part_num = ET.SubElement(parts, 'partitionsNum')
  706. part_num.text=str(ota_part)
  707. part_num.tail = '\n\t'
  708. for part in self.partitions:
  709. if ('file_name' in part.keys()) and ('true' == part['enable_ota']):
  710. partfile = os.path.join(ota_dir, part['file_name']);
  711. part_node = ET.SubElement(parts, 'partition')
  712. part_node.text = '\n\t\t'
  713. part_node.tail = '\n\t'
  714. type = ET.SubElement(part_node, 'type')
  715. type.text=part['type']
  716. type.tail = '\n\t\t'
  717. type = ET.SubElement(part_node, 'name')
  718. type.text=part['name']
  719. type.tail = '\n\t\t'
  720. type = ET.SubElement(part_node, 'file_id')
  721. type.text=part['file_id']
  722. type.tail = '\n\t\t'
  723. type = ET.SubElement(part_node, 'storage_id')
  724. type.text=part['storage_id']
  725. type.tail = '\n\t\t'
  726. url = ET.SubElement(part_node, 'file_name')
  727. url.text = part['file_name']
  728. url.tail = '\n\t\t'
  729. type = ET.SubElement(part_node, 'file_size')
  730. type.text=str(hex(os.path.getsize(partfile)))
  731. type.tail = '\n\t\t'
  732. crc = ET.SubElement(part_node, 'checksum')
  733. crc.text = str(hex(crc32_file(partfile)))
  734. crc.tail = '\n\t'
  735. tree = ET.ElementTree(root)
  736. tree.write(os.path.join(ota_dir, 'ota.xml'), xml_declaration=True, method="xml", encoding='UTF-8')
  737. def ota_get_file_seq(self, file_name):
  738. seq = 0
  739. #for part in self.partitions:
  740. for i, part in enumerate(self.partitions):
  741. if ('file_name' in part.keys()) and ('true' == part['enable_ota']):
  742. if os.path.basename(file_name) == part['file_name']:
  743. seq = i + 1;
  744. # boot is the last file in ota firmware
  745. if 'BOOT' == part['type']:
  746. seq = 0x10000
  747. if 'SYS_PARAM' == part['type']:
  748. seq = 0x10001
  749. return seq
  750. def generate_ota_image_internal(self, ota_file, ota_dir = '', ota_xml = ''):
  751. if os.path.exists(ota_file):
  752. os.remove(ota_file)
  753. if ota_xml == '':
  754. self.generate_ota_xml(ota_dir)
  755. else:
  756. shutil.copyfile(ota_xml, os.path.join(ota_dir, 'ota.xml'))
  757. # sort files in OTA firmware by upgrade sequence
  758. files = []
  759. for parent, dirnames, filenames in os.walk(ota_dir):
  760. for filename in filenames:
  761. files.append(os.path.join(parent, filename))
  762. files.sort(key=self.ota_get_file_seq)
  763. self.build_ota_image(ota_file, '', files)
  764. if not os.path.exists(ota_file):
  765. panic('Failed to generate OTA image')
  766. def build_ota_image(self, image_path, ota_file_dir, files = []):
  767. script_ota_path = os.path.join(script_path, 'build_ota_image.py')
  768. cmd = [sys.executable, script_ota_path, '-o', image_path]
  769. if files == []:
  770. for parent, dirnames, filenames in os.walk(ota_file_dir):
  771. for filename in filenames:
  772. cmd.append(os.path.join(parent, filename))
  773. else:
  774. cmd = cmd + files
  775. (outmsg, exit_code) = run_cmd(cmd)
  776. if exit_code !=0:
  777. print('make ota error')
  778. print(outmsg)
  779. sys.exit(1)
  780. def copy_ota_files(self, bin_dir, ota_dir):
  781. for part in self.partitions:
  782. if ('file_name' in part.keys()) and ('true' == part['enable_ota']):
  783. shutil.copyfile(os.path.join(bin_dir, part['file_name']), \
  784. os.path.join(ota_dir, part['file_name']))
  785. def generate_ota_image(self, ota_filename):
  786. print('FW: Build OTA image \'' + ota_filename + '\'')
  787. # generate OTA firmware with crc/randomizer
  788. self.copy_ota_files(self.bin_dir, self.ota_dir)
  789. self.generate_ota_image_internal(os.path.join(self.fw_dir, ota_filename), self.ota_dir)
  790. # generate OTA firmware without crc/randomizer
  791. self.copy_ota_files(self.orig_bin_dir, self.orig_ota_dir)
  792. self.generate_ota_image_internal(os.path.join(self.fw_dir, \
  793. ota_filename + '.orig'), self.orig_ota_dir, os.path.join(self.ota_dir, 'ota.xml'))
  794. def generate_att_image(self, att_filename, fw_filename):
  795. self.att_dir = os.path.join(self.fw_dir, 'att')
  796. if not os.path.isdir(self.att_dir):
  797. return
  798. print('ATT: Build ATT image')
  799. for part in self.partitions:
  800. if ('file_name' in part.keys())and ('true' == part['enable_dfu']):
  801. shutil.copyfile(os.path.join(self.bin_dir, part['file_name']), \
  802. os.path.join(self.att_dir, part['file_name']))
  803. maker_cfgfile = os.path.join(self.bin_dir, 'fw_maker.cfg')
  804. self.generate_maker_config(maker_cfgfile);
  805. temp_maker_cfgfile = os.path.join(self.bin_dir, 'fw_maker_temp.cfg')
  806. with open(temp_maker_cfgfile, 'wb') as f:
  807. with open(maker_cfgfile, 'rb') as f1:
  808. f1_data = f1.read()
  809. f.write(f1_data)
  810. mp_card_cfgfile = os.path.join(self.bin_dir, 'mp_card.cfg')
  811. with open(mp_card_cfgfile, 'rb') as f2:
  812. f2_data = f2.read()
  813. f.write(f2_data)
  814. maker_ext_cfgfile = os.path.join(self.bin_dir, 'fw_maker_ext.cfg')
  815. with open(maker_ext_cfgfile, 'rb') as f3:
  816. f3_data = f3.read()
  817. f.write(f3_data)
  818. shutil.copyfile(temp_maker_cfgfile, \
  819. os.path.join(self.att_dir, 'fw_maker.cfg'))
  820. shutil.copyfile(os.path.join(self.fw_dir, fw_filename),
  821. os.path.join(self.att_dir, 'attdfu.fw'))
  822. print('ATT: Generate ATT image %s' %(os.path.join(self.fw_dir, att_filename)))
  823. self.build_atf_image(os.path.join(self.fw_dir, att_filename), self.att_dir)
  824. def build_atf_image(self, image_path, atf_file_dir, files = []):
  825. script_atf_path = os.path.join(script_path, 'build_atf_image.py')
  826. cmd = [sys.executable, script_atf_path, '-o', image_path]
  827. for parent, dirnames, filenames in os.walk(atf_file_dir):
  828. for filename in filenames:
  829. files.append(os.path.join(parent, filename))
  830. cmd = cmd + files
  831. (outmsg, exit_code) = run_cmd(cmd)
  832. if exit_code !=0:
  833. print('make att error')
  834. print(outmsg)
  835. sys.exit(1)
  836. def add_crc(self, filename, chunk_size):
  837. with open(filename, 'rb') as f:
  838. filedata = f.read()
  839. length = len(filedata)
  840. i = 0
  841. with open(filename, 'wb') as f:
  842. while (length > 0):
  843. if (length < chunk_size):
  844. chunk = filedata[i : i + length] + bytearray(chunk_size - length)
  845. else:
  846. chunk = filedata[i : i + chunk_size]
  847. crc = crc16(bytearray(chunk), 0xffff)
  848. f.write(chunk + struct.pack('<H',crc))
  849. length -= chunk_size
  850. i += chunk_size
  851. def test_generate_ota_firmwares(self):
  852. if CFG_TEST_GENERATE_OTA_FIRMWARES_COUNT > 0:
  853. test_ota_fws = os.path.join(self.fw_dir, 'test_ota_fws')
  854. if os.path.exists(test_ota_fws):
  855. shutil.rmtree(test_ota_fws)
  856. time.sleep(0.1)
  857. os.mkdir(test_ota_fws)
  858. ota_temp_dir = os.path.join(test_ota_fws, 'temp_ota')
  859. shutil.copytree(self.ota_dir, ota_temp_dir)
  860. ota_temp_dir_orig = os.path.join(test_ota_fws, 'temp_ota_orig')
  861. shutil.copytree(self.orig_ota_dir, ota_temp_dir_orig)
  862. for i in range(CFG_TEST_GENERATE_OTA_FIRMWARES_COUNT):
  863. new_param_file = os.path.join(ota_temp_dir, self.param_file_name)
  864. orig_param_file = os.path.join(self.orig_bin_dir, self.param_file_name)
  865. shutil.copyfile(orig_param_file, new_param_file)
  866. # increase firmware version code
  867. version_code = int(self.fw_version['version_code'], 0) + 1
  868. self.fw_version['version_code'] = str(hex(version_code))
  869. # update fw version field in boot file
  870. self.generate_firmware_version(new_param_file)
  871. # copy the origin boot file to temp_ota_orig
  872. shutil.copyfile(new_param_file, os.path.join(ota_temp_dir_orig, self.param_file_name))
  873. self.encrypt_param_file(new_param_file)
  874. ota_fw_name = 'ota_' + str(i + 1) + '.bin'
  875. print('FW: Build test OTA image \'' + ota_fw_name + '\'')
  876. self.generate_ota_image_internal(os.path.join(test_ota_fws, ota_fw_name), ota_temp_dir)
  877. ota_fw_name = 'ota_' + str(i + 1) + '.bin.orig'
  878. print('FW: Build test origin OTA image \'' + ota_fw_name + '\'')
  879. self.generate_ota_image_internal(os.path.join(test_ota_fws, ota_fw_name), ota_temp_dir_orig, os.path.join(ota_temp_dir, 'ota.xml'))
  880. def generate_earphone_config(self, mcfg_file, upg_ini_file):
  881. with open(mcfg_file, 'w') as f, open(upg_ini_file, 'w') as f_u:
  882. fw_ver_str = json.dumps(self.fw_version)
  883. fw_ver_str = fw_ver_str.replace('\"', '\'')
  884. f.write('\nFW_VERSION="%s";\n\n' %fw_ver_str)
  885. f.write('\nSETPATH="..\\";\n')
  886. f.write('FIRMWARE_XML="firmware.xml";\n')
  887. f.write('\nSETPATH=".\\";\n')
  888. f.write('SPI_STG_CAP=%d;\n' %(self.disk_size // 0x200))
  889. f.write('BASEFILE="afi.bin";\n')
  890. f.write('UPGRADE_BAUDRATE=%d;\n' %self.upgrade_baudrate)
  891. f.write('PRODUCT_CONFIG="%s";\n' %os.path.basename(upg_ini_file))
  892. f.write('SYS_SDFS="ksdfs.bin";\n')
  893. f_u.write("version\n")
  894. f_u.write("baudrate = %d\n" %self.upgrade_baudrate)
  895. f_u.write("adfu_switch\n")
  896. # f_u.write("adfu_start\n")
  897. swrite_data_list = []
  898. # write dfu data
  899. for part in self.partitions:
  900. if ('file_name' in part.keys()) and ('true' == part['enable_dfu']):
  901. w_file_name = part['file_name']
  902. storage_id = int(part['storage_id'], 0)
  903. w_file_addr = int(part['file_address'], 0)
  904. if 'BOOT' == part['type']:
  905. info_tuple = (w_file_name, storage_id, w_file_addr, True)
  906. else:
  907. info_tuple = (w_file_name, storage_id, w_file_addr)
  908. swrite_data_list.append(info_tuple)
  909. swrite_data_list.sort(key = lambda x:(x[1], x[2]))
  910. prv_storage_id = -1
  911. for info_tuple in swrite_data_list:
  912. w_file_name = info_tuple[0]
  913. storage_id = info_tuple[1]
  914. w_file_addr = info_tuple[2]
  915. if storage_id != prv_storage_id:
  916. f.write("\n")
  917. f_u.write("\n")
  918. f_u.write("renew_flash_id %d\n" %storage_id)
  919. f_u.write("sinit %d\n" %storage_id)
  920. prv_storage_id = storage_id
  921. f_u.write("swrite 0x%08x 0 %s\n" %(w_file_addr, w_file_name))
  922. file_path = os.path.join(self.bin_dir, w_file_name)
  923. if os.path.getsize(file_path) > 0x400000:
  924. self.atf_file_list.append(file_path)
  925. else:
  926. f.write('WFILE="%s",0x%x;\n' %(w_file_name, w_file_addr))
  927. f_u.write("\nclear_history\n")
  928. f_u.write("disconnect reboot\n")
  929. #add extern data config
  930. f.write('\n#include "fw_product.cfg"\n')
  931. f.write('\nSETPATH=".\\config";\n')
  932. f.write('#include "fw_configuration.cfg"\n')
  933. def generate_earphone_firmware(self, fw_name):
  934. print('FW: Build earphone firmware image')
  935. maker_cfgfile = os.path.join(self.bin_dir, 'fw_maker.cfg')
  936. upg_cfgfile = os.path.join(self.bin_dir, 'product.ini')
  937. self.generate_earphone_config(maker_cfgfile, upg_cfgfile);
  938. upg_file = os.path.join(self.att_dir, 'upgrade.fw')
  939. fw_file = os.path.join(self.fw_dir, fw_name)
  940. if (is_windows()):
  941. cmd = [script_path + '/utils/windows/maker.exe', '-c', maker_cfgfile, \
  942. '-o', upg_file, '-mf']
  943. else:
  944. if ('32bit' == platform.architecture()[0]):
  945. # linux x86
  946. maker_path = script_path + '/utils/linux-x86/maker/PyMaker.pyo'
  947. else:
  948. # linux x86_64
  949. maker_path = script_path + '/utils/linux-x86_64/maker/PyMaker.pyo'
  950. cmd = ['python2', '-O', maker_path, '-c', maker_cfgfile, '-o', upg_file, '--mf', '1']
  951. (outmsg, exit_code) = run_cmd(cmd)
  952. if not os.path.exists(upg_file):
  953. panic('Maker error: %s' %(outmsg))
  954. self.build_atf_image(fw_file, self.att_dir, self.atf_file_list)
  955. print_notice('Build successfully!')
  956. def main(argv):
  957. global soc_name, board_name, encrypt_fw, efuse_bin
  958. parser = argparse.ArgumentParser(
  959. description='Build firmware',
  960. )
  961. parser.add_argument('-c', dest = 'fw_cfgfile')
  962. parser.add_argument('-e', dest = 'encrypt_fw')
  963. parser.add_argument('-ef', dest = 'efuse_bin')
  964. parser.add_argument('-b', dest = 'board_name')
  965. parser.add_argument('-s', dest = 'soc_name')
  966. parser.add_argument('-v', dest = 'fw_ver_file')
  967. args = parser.parse_args();
  968. fw_cfgfile = args.fw_cfgfile
  969. encrypt_fw = args.encrypt_fw
  970. efuse_bin = args.efuse_bin
  971. board_name = args.board_name
  972. date_stamp = time.strftime('%y%m%d',time.localtime(time.time()))
  973. fw_prefix = board_name + '_' + date_stamp
  974. soc_name = args.soc_name
  975. if (not os.path.isfile(fw_cfgfile)):
  976. print('firmware config file is not exist')
  977. sys.exit(1)
  978. try:
  979. fw = firmware(fw_cfgfile)
  980. fw.crc_chunk_size = 32
  981. fw.parse_config(fw_cfgfile)
  982. """
  983. if not fw.is_earphone_app:
  984. fw.generate_raw_image(fw_prefix + '_raw.bin')
  985. fw.generate_ota_image('ota.bin')
  986. fw.generate_firmware(fw_prefix + '.fw')
  987. #fw.generate_att_image(fw_prefix + '_att.fw', fw_prefix + '.fw')
  988. fw.test_generate_ota_firmwares()
  989. else:
  990. """
  991. fw.generate_earphone_firmware(fw_prefix + '.fw')
  992. except Exception as e:
  993. print('\033[1;31;40m')
  994. print('unknown exception, %s' %(e));
  995. print('\033[0m')
  996. # traceback.print_exc()
  997. sys.exit(2)
  998. if __name__ == '__main__':
  999. main(sys.argv[1:])