build.py 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773
  1. #!/usr/bin/env python
  2. #
  3. # Copyright (c) 2020 Actions Semiconductor Co., Ltd
  4. #
  5. # SPDX-License-Identifier: Apache-2.0
  6. #
  7. import os
  8. import sys
  9. import glob
  10. import time
  11. import argparse
  12. import platform
  13. import shutil
  14. import subprocess
  15. import re
  16. import xml.etree.ElementTree as ET
  17. sdk_arch = "arm"
  18. sdk_root = os.getcwd()
  19. sdk_root_parent = os.path.dirname(sdk_root)
  20. #sdk_mcuboot_dir=os.path.join(sdk_root_parent, "bootloader", "mcuboot")
  21. #sdk_mcuboot_imgtool = os.path.join(sdk_mcuboot_dir, "scripts", "imgtool.py")
  22. #sdk_mcuboot_key= os.path.join(sdk_mcuboot_dir, "root-rsa-2048.pem")
  23. #sdk_mcuboot_app_path=os.path.join(sdk_mcuboot_dir, "boot", "zephyr")
  24. sdk_boards_dir = os.path.join(sdk_root, "boards", sdk_arch)
  25. #sdk_application_dir = os.path.join(sdk_root_parent, "application")
  26. sdk_application_dir=""
  27. sdk_app_dir = os.path.join(sdk_root_parent, "application")
  28. sdk_samples_dir = os.path.join(sdk_root, "samples")
  29. sdk_tests_dir = os.path.join(sdk_root, "tests")
  30. sdk_script_dir = os.path.join(sdk_root, "tools")
  31. sdk_script_prebuilt_dir = os.path.join(sdk_script_dir, "prebuilt")
  32. build_config_file = os.path.join(sdk_root, ".build_config")
  33. application = ""
  34. sub_app = ""
  35. board_conf = ""
  36. app_conf = ""
  37. def is_windows():
  38. sysstr = platform.system()
  39. if (sysstr.startswith('Windows') or \
  40. sysstr.startswith('MSYS') or \
  41. sysstr.startswith('MINGW') or \
  42. sysstr.startswith('CYGWIN')):
  43. return True
  44. else:
  45. return False
  46. def to_int(str):
  47. try:
  48. int(str)
  49. return int(str)
  50. except ValueError:
  51. try:
  52. float(str)
  53. return int(float(str))
  54. except ValueError:
  55. return False
  56. def read_choice( tip, path):
  57. print("")
  58. print(" %s" %(tip))
  59. print("")
  60. dirs = next(os.walk(path))[1]
  61. i = 0
  62. for file in dirs:
  63. file_path = os.path.join(path, file)
  64. if os.path.isdir(file_path):
  65. i += 1
  66. print(" %d. %s" %(i, file))
  67. print("")
  68. input_prompt = "Which would you like? [" + dirs[0] + "] "
  69. str = ""
  70. try:
  71. str = input(input_prompt)
  72. except Exception as ex:
  73. print("")
  74. if to_int(str) != False:
  75. j = to_int(str)
  76. else:
  77. j = 1
  78. if j > i:
  79. j = i
  80. j -= 1
  81. return dirs[j]
  82. def run_cmd(cmd):
  83. """Echo and run the given command.
  84. Args:
  85. cmd: the command represented as a list of strings.
  86. Returns:
  87. A tuple of the output and the exit code.
  88. """
  89. # print("Running: ", " ".join(cmd))
  90. p = subprocess.Popen(cmd, stdout=subprocess.PIPE, stderr=subprocess.STDOUT)
  91. output, _ = p.communicate()
  92. #print("%s" % (output.rstrip()))
  93. return (output, p.returncode)
  94. def build_nvram_bin(out_dir, board_dir, app_dir, app_cfg_dir):
  95. nvram_path = os.path.join(out_dir, "nvram.bin")
  96. app_cfg_path = os.path.join(app_cfg_dir, "nvram.prop")
  97. if os.path.exists(app_cfg_path) == False:
  98. app_cfg_path = os.path.join(app_dir, "nvram.prop")
  99. board_cnf_path = os.path.join(board_dir, "nvram.prop")
  100. print("\n build_nvram \n")
  101. if not os.path.isfile(board_cnf_path):
  102. print("\n board nvram %s not exists\n" %(board_cnf_path))
  103. return
  104. if not os.path.isfile(app_cfg_path):
  105. print("\n app nvram %s not exists\n" %(app_cfg_path))
  106. return
  107. script_nvram_path = os.path.join(sdk_script_dir, 'build_nvram_bin.py')
  108. cmd = ['python', '-B', script_nvram_path, '-o', nvram_path, app_cfg_path, board_cnf_path]
  109. (outmsg, exit_code) = run_cmd(cmd)
  110. if exit_code !=0:
  111. print('make build_nvram_bin error')
  112. print(outmsg.decode('utf-8'))
  113. sys.exit(1)
  114. print("\n build_nvram finshed\n\n")
  115. def dir_cp(out_dir, in_dir):
  116. print(" cp dir %s to dir %s" %(in_dir, out_dir))
  117. if os.path.exists(in_dir) == True:
  118. if os.path.exists(out_dir) == True:
  119. for parent, dirnames, filenames in os.walk(in_dir):
  120. for filename in filenames:
  121. pathfile = os.path.join(parent, filename)
  122. shutil.copyfile(pathfile, os.path.join(out_dir, filename))
  123. else:
  124. shutil.copytree(in_dir, out_dir)
  125. def dir_tree_cp(out_dir, in_dir):
  126. # keep directory structure
  127. print(" cp dir %s to dir %s" %(in_dir, out_dir))
  128. if os.path.exists(in_dir):
  129. if os.path.exists(out_dir):
  130. for parent, dirnames, filenames in os.walk(in_dir):
  131. for filename in filenames:
  132. copy_to_dir = os.path.join(out_dir, os.path.relpath(parent, in_dir))
  133. if not os.path.isdir(copy_to_dir):
  134. os.mkdir(copy_to_dir)
  135. pathfile = os.path.join(parent, filename)
  136. shutil.copyfile(pathfile, os.path.join(copy_to_dir, filename))
  137. else:
  138. shutil.copytree(in_dir, out_dir)
  139. def check_cp(out_dir, in_dir, name):
  140. in_cp = os.path.join(in_dir, name)
  141. out_cp = os.path.join(out_dir, name)
  142. if os.path.exists(in_cp):
  143. if os.path.isfile(in_cp) == True:
  144. print(" cp file %s to dir %s" %(in_cp,out_cp))
  145. shutil.copyfile(in_cp, out_cp)
  146. else :
  147. dir_tree_cp(out_dir, in_cp)
  148. def build_sdfs_bin_name(out_dir, board_dir, app_dir, app_cfg_dir, sdfs_name):
  149. board_sdfs_dir = os.path.join(board_dir, sdfs_name)
  150. app_sdfs_dir = os.path.join(app_dir, sdfs_name)
  151. app_cfg_sdfs_dir = os.path.join(app_cfg_dir, sdfs_name)
  152. #print("\n build_sdfs : sdfs_dir %s, board dfs %s\n\n" %(sdfs_dir, board_sdfs_dir))
  153. if (os.path.exists(board_sdfs_dir) == False) \
  154. and (os.path.exists(app_sdfs_dir) == False) \
  155. and (os.path.exists(app_cfg_sdfs_dir) == False):
  156. print("\n build_sdfs : %s not exists\n\n" %(sdfs_name))
  157. return
  158. sdfs_dir = os.path.join(out_dir, sdfs_name)
  159. if os.path.exists(sdfs_dir) == True:
  160. shutil.rmtree(sdfs_dir)
  161. time.sleep(0.1)
  162. if os.path.exists(board_sdfs_dir) == True:
  163. dir_cp(sdfs_dir, board_sdfs_dir)
  164. if os.path.exists(app_sdfs_dir) == True:
  165. dir_cp(sdfs_dir, app_sdfs_dir)
  166. if os.path.exists(app_cfg_sdfs_dir) == True:
  167. dir_cp(sdfs_dir, app_cfg_sdfs_dir)
  168. print("\n build_sdfs : sdfs_dir %s, sdfs_name %s\n\n" %(sdfs_dir, sdfs_name))
  169. if (sdfs_name == 'fonts') or (sdfs_name == 'res') or (sdfs_name == 'watch'):
  170. txtdir = os.path.join(sdfs_dir, "txt")
  171. if os.path.exists(txtdir) == True:
  172. shutil.rmtree(txtdir)
  173. for infile in glob.glob(os.path.join(sdfs_dir, '*.txt')):
  174. print(" remove file : %s\n" %(infile))
  175. os.remove(infile)
  176. if (sdfs_name == 'ksdfs') or (sdfs_name == 'sdfs'):
  177. ksym_bin = os.path.join(out_dir, "ksym.bin")
  178. if os.path.exists(ksym_bin):
  179. shutil.move(ksym_bin, os.path.join(sdfs_dir, "ksym.bin"))
  180. script_sdfs_path = os.path.join(sdk_script_dir, 'build_sdfs.py')
  181. sdfs_bin_path = os.path.join(out_dir, sdfs_name+".bin")
  182. cmd = ['python', '-B', script_sdfs_path, '-o', sdfs_bin_path, '-d', sdfs_dir]
  183. (outmsg, exit_code) = run_cmd(cmd)
  184. if exit_code !=0:
  185. print('make build_sdfs_bin error')
  186. print(outmsg.decode('utf-8'))
  187. sys.exit(1)
  188. print("\n build_sdfs %s finshed\n\n" %(sdfs_name))
  189. def build_sdfs_bin_bydir(out_dir, board_dir, app_dir, app_cfg_dir):
  190. bpath = os.path.join(board_dir, "fs_sdfs");
  191. apath = os.path.join(app_cfg_dir, "fs_sdfs");
  192. if os.path.exists(apath) == False:
  193. apath = os.path.join(app_dir, "fs_sdfs");
  194. if (os.path.exists(bpath) == False) and (os.path.exists(apath) == False):
  195. print('not fs_sdfs')
  196. return
  197. all_dirs = {}
  198. if os.path.exists(bpath) == True:
  199. for file in os.listdir(bpath):
  200. print("board list %s ---\n" %(file))
  201. if os.path.isdir(os.path.join(bpath,file)):
  202. all_dirs[file] = file
  203. if os.path.exists(apath) == True:
  204. for file in os.listdir(apath):
  205. print("app list %s ---\n" %(file))
  206. if os.path.isdir(os.path.join(apath,file)):
  207. all_dirs[file] = file
  208. for dir in all_dirs.keys():
  209. print("--build_sdfs %s ---\n" %(dir))
  210. build_sdfs_bin_name(out_dir, bpath, apath, app_cfg_dir, dir)
  211. def build_sdfs_bin(out_dir, board_dir, app_dir, app_cfg_dir):
  212. build_sdfs_bin_name(out_dir, board_dir, app_dir, app_cfg_dir,"ksdfs")
  213. build_sdfs_bin_name(out_dir, board_dir, app_dir, app_cfg_dir,"sdfs")
  214. build_sdfs_bin_name(out_dir, board_dir, app_dir, app_cfg_dir,"fatfs")
  215. build_sdfs_bin_name(out_dir, board_dir, app_dir, app_cfg_dir,"res")
  216. build_sdfs_bin_name(out_dir, board_dir, app_dir, app_cfg_dir,"fonts")
  217. build_sdfs_bin_name(out_dir, board_dir, app_dir, app_cfg_dir,"watch")
  218. build_sdfs_bin_bydir(out_dir, board_dir, app_dir, app_cfg_dir)
  219. def file_link(filename, file_add):
  220. if not os.path.exists(filename):
  221. return
  222. if not os.path.exists(file_add):
  223. return
  224. print("file %s, link %s \n" %(filename, file_add))
  225. with open(file_add, 'rb') as fa:
  226. file_data = fa.read()
  227. fa.close()
  228. fsize = os.path.getsize(filename)
  229. with open(filename, 'rb+') as f:
  230. f.seek(fsize, 0)
  231. f.write(file_data)
  232. f.close()
  233. def is_config_KALLSYMS(cfg_path):
  234. bret = False
  235. print("\n cfg_path =%s\n" %cfg_path)
  236. fp = open(cfg_path,"r")
  237. lines = fp.readlines()
  238. for elem in lines:
  239. if elem[0] != '#':
  240. _c = elem.strip().split("=")
  241. if _c[0] == "CONFIG_KALLSYMS":
  242. print("CONFIG_KALLSYMS");
  243. bret = True
  244. break;
  245. fp.close
  246. print("\nCONFIG_KALLSYMS=%d\n" %bret )
  247. return bret
  248. def build_datafs_bin(out_dir, app_dir, app_cfg_dir, fw_cfgfile):
  249. script_datafs_path = os.path.join(sdk_script_dir, 'build_datafs.py')
  250. tree = ET.ElementTree(file=fw_cfgfile)
  251. root = tree.getroot()
  252. if (root.tag != 'firmware'):
  253. sys.stderr.write('error: invalid firmware config file')
  254. sys.exit(1)
  255. part_list = root.find('partitions').findall('partition')
  256. for part in part_list:
  257. part_prop = {}
  258. for prop in part:
  259. part_prop[prop.tag] = prop.text.strip()
  260. if ('storage_id' in part_prop.keys()) and (part_prop['name'] == 'udisk'):
  261. app_datafs_dir = os.path.join(app_cfg_dir, part_prop['name'])
  262. if os.path.exists(app_datafs_dir) == False:
  263. app_datafs_dir = os.path.join(app_dir, part_prop['name'])
  264. if os.path.exists(app_datafs_dir) == False:
  265. print("\n build_datafs : app config datafs %s not exists\n\n" %(app_datafs_dir))
  266. continue
  267. datafs_cap = int(part_prop['size'], 0)
  268. datafs_bin_path = os.path.join(out_dir, part_prop['file_name'])
  269. print('DATAFS (%s) capacity => 0x%x' %(datafs_bin_path, datafs_cap))
  270. cmd = ['python', '-B', script_datafs_path, '-o', datafs_bin_path, '-s', str(datafs_cap), '-d', app_datafs_dir]
  271. (outmsg, exit_code) = run_cmd(cmd)
  272. if exit_code !=0:
  273. print('make build_datafs_bin %s error' %(datafs_bin_path))
  274. print(outmsg.decode('utf-8'))
  275. sys.exit(1)
  276. print("\n build_datafs finshed\n\n")
  277. def read_littlefs_block_size_config(cfg_path):
  278. block_size = 4096
  279. fp = open(cfg_path,"r")
  280. lines = fp.readlines()
  281. for elem in lines:
  282. if elem[0] != '#':
  283. _c = elem.strip().split("=")
  284. if _c[0] == "CONFIG_FS_LITTLEFS_BLOCK_SIZE":
  285. block_size = int(_c[1].strip('"'), 0)
  286. break;
  287. fp.close
  288. return block_size
  289. def build_littlefs_bin(out_dir, app_dir, app_cfg_dir, fw_cfgfile):
  290. script_littlefs_path = os.path.join(sdk_script_dir, 'build_littlefs.py')
  291. OS_OUTDIR = os.path.join(out_dir, "../../zephyr")
  292. tree = ET.ElementTree(file=fw_cfgfile)
  293. root = tree.getroot()
  294. if (root.tag != 'firmware'):
  295. sys.stderr.write('error: invalid firmware config file')
  296. sys.exit(1)
  297. part_list = root.find('partitions').findall('partition')
  298. for part in part_list:
  299. part_prop = {}
  300. for prop in part:
  301. part_prop[prop.tag] = prop.text.strip()
  302. if ('storage_id' in part_prop.keys()) and (part_prop['name'] == 'littlefs'):
  303. app_littlefs_dir = os.path.join(app_cfg_dir, part_prop['name'])
  304. if os.path.exists(app_littlefs_dir) == False:
  305. app_littlefs_dir = os.path.join(app_dir, part_prop['name'])
  306. if os.path.exists(app_littlefs_dir) == False:
  307. print("\n build_datafs : app config littlefs %s not exists\n\n" %(app_littlefs_dir))
  308. continue
  309. littlefs_cap = int(part_prop['size'], 0)
  310. littlefs_bin_path = os.path.join(out_dir, part_prop['file_name'])
  311. blocksize = read_littlefs_block_size_config(os.path.join(OS_OUTDIR, ".config"))
  312. print('LITTLEFS (%s) capacity => 0x%x' %(littlefs_bin_path, littlefs_cap))
  313. cmd = ['python', '-B', script_littlefs_path, '-o', littlefs_bin_path, '-s', str(littlefs_cap), '-d', app_littlefs_dir, '-b', str(blocksize)]
  314. (outmsg, exit_code) = run_cmd(cmd)
  315. if exit_code !=0:
  316. print('make build_littlefs_bin %s error' %(littlefs_bin_path))
  317. print(outmsg.decode('utf-8'))
  318. sys.exit(1)
  319. print('\n build_littlefs finshed \n\n')
  320. def read_val_by_config(cfg_path, cfg_name):
  321. val = "0"
  322. fp = open(cfg_path,"r")
  323. lines = fp.readlines()
  324. for elem in lines:
  325. if elem[0] != '#':
  326. _c = elem.strip().split("=")
  327. if _c[0] == cfg_name:
  328. val = _c[1].strip('"')
  329. break;
  330. fp.close
  331. return val
  332. def read_soc_by_config(cfg_path):
  333. soc = "lark"
  334. fp = open(cfg_path,"r")
  335. lines = fp.readlines()
  336. for elem in lines:
  337. if elem[0] != '#':
  338. _c = elem.strip().split("=")
  339. if _c[0] == "CONFIG_SOC_SERIES":
  340. soc = _c[1].strip('"')
  341. break;
  342. fp.close
  343. return soc
  344. def image_add_checksum(fw_dir_bin, img_bin):
  345. img_bin_file = os.path.join(fw_dir_bin, img_bin)
  346. if os.path.exists(img_bin_file) == False:
  347. return
  348. print("\nFW: Post build %s\n"%img_bin)
  349. script_firmware_path = os.path.join(sdk_script_dir, 'pack_app_image.py')
  350. cmd = ['python', '-B', script_firmware_path, img_bin_file]
  351. print("\n build cmd : %s\n\n" %(cmd))
  352. (outmsg, exit_code) = run_cmd(cmd)
  353. if exit_code !=0:
  354. print("pack %s error\n"%img_bin)
  355. print(outmsg)
  356. sys.exit(1)
  357. def build_firmware(board, out_dir, board_dir, app_dir, app_cfg_dir):
  358. print("\n build_firmware : out %s, board %s, app_dir %s cfg_dir %s\n\n" %(out_dir, board_dir, app_dir, app_cfg_dir))
  359. res_files = ["fonts", "res", "watch"]
  360. full_files = ["full","firmware","boot"]
  361. FW_DIR = os.path.join(out_dir, "_firmware")
  362. OS_OUTDIR = os.path.join(out_dir, "zephyr")
  363. if os.path.exists(FW_DIR) == False:
  364. os.mkdir(FW_DIR)
  365. FW_DIR_BIN = os.path.join(FW_DIR, "bin")
  366. if os.path.exists(FW_DIR_BIN) == False:
  367. os.mkdir(FW_DIR_BIN)
  368. if is_config_KALLSYMS(os.path.join(OS_OUTDIR, ".config")):
  369. print("cpy ksym.bin- to fw dir-");
  370. check_cp(FW_DIR_BIN , OS_OUTDIR, "ksym.bin")
  371. build_nvram_bin(FW_DIR_BIN, board_dir, app_dir, app_cfg_dir)
  372. build_sdfs_bin(FW_DIR_BIN, board_dir, app_dir, app_cfg_dir)
  373. soc = read_soc_by_config(os.path.join(OS_OUTDIR, ".config"))
  374. print("\n build_firmware : soc name= %s\n" %(soc))
  375. dir_cp(FW_DIR_BIN, os.path.join(sdk_script_prebuilt_dir, soc, "common", "bin"))
  376. dir_cp(FW_DIR_BIN, os.path.join(sdk_script_prebuilt_dir, soc, "bin"))
  377. dir_tree_cp(FW_DIR_BIN, os.path.join(sdk_script_prebuilt_dir, soc, "prebuild"))
  378. fw_cfgfile = os.path.join(FW_DIR, "firmware.xml")
  379. #check copy firmware.xml
  380. check_cp(FW_DIR, board_dir, "firmware.xml")
  381. check_cp(FW_DIR, app_dir, "firmware.xml")
  382. check_cp(FW_DIR, app_cfg_dir, "firmware.xml")
  383. #check copy firmware_debug.xml
  384. check_cp(FW_DIR, app_cfg_dir, "firmware_debug.xml")
  385. if not os.path.exists(fw_cfgfile):
  386. print("failed to find out firmware.xml")
  387. sys.exit(1)
  388. #check copy firmware_res.xml
  389. #res_cfgfile = os.path.join(FW_DIR, "firmware_res.xml")
  390. #check_cp(FW_DIR, app_dir, "firmware_res.xml")
  391. #check_cp(FW_DIR, app_cfg_dir, "firmware_res.xml")
  392. for res_file in res_files:
  393. res_cfgfile = os.path.join(FW_DIR, "ota_"+ res_file +".xml")
  394. check_cp(FW_DIR, app_dir, "ota_"+ res_file +".xml")
  395. check_cp(FW_DIR, app_cfg_dir, "ota_"+ res_file +".xml")
  396. #full
  397. for full_file in full_files:
  398. res_cfgfile = os.path.join(FW_DIR, "ota_"+ full_file +".xml")
  399. check_cp(FW_DIR, app_dir, "ota_"+ full_file +".xml")
  400. check_cp(FW_DIR, app_cfg_dir, "ota_"+ full_file +".xml")
  401. # check override files
  402. check_cp(FW_DIR_BIN, board_dir, "adfus_u.bin")
  403. check_cp(FW_DIR_BIN, board_dir, "adfus.bin")
  404. check_cp(FW_DIR_BIN, board_dir, "mbrec.bin")
  405. check_cp(FW_DIR_BIN, board_dir, "afi.bin")
  406. check_cp(FW_DIR_BIN, board_dir, "bootloader.ini")
  407. check_cp(FW_DIR_BIN, board_dir, "nand_id.bin")
  408. check_cp(FW_DIR_BIN, board_dir, "recovery.bin")
  409. check_cp(FW_DIR_BIN, board_dir, "fw_product.cfg")
  410. check_cp(FW_DIR_BIN, app_dir, "prebuild")
  411. check_cp(FW_DIR_BIN, app_cfg_dir, "prebuild")
  412. check_cp(FW_DIR_BIN, app_cfg_dir, "E_CHECK.FW")
  413. check_cp(FW_DIR_BIN, app_cfg_dir, "fwimage.cfg")
  414. ota_app_path = os.path.join(FW_DIR_BIN, "ota_app.bin")
  415. if os.path.exists(ota_app_path) == True:
  416. os.remove(ota_app_path)
  417. check_cp(FW_DIR_BIN, board_dir, "ota_app.bin")
  418. # signed img
  419. sign_dir =os.path.join(board_dir, "sign-img")
  420. if os.path.exists(sign_dir) == True:
  421. check_cp(FW_DIR_BIN, sign_dir, "mbrec.bin")
  422. print("\nFW: Post build mbrec.bin\n")
  423. script_firmware_path = os.path.join(sdk_script_dir, 'build_boot_image.py')
  424. cmd = ['python', '-B', script_firmware_path, os.path.join(FW_DIR_BIN, "mbrec.bin"), \
  425. os.path.join(FW_DIR_BIN, "bootloader.ini"), \
  426. os.path.join(FW_DIR_BIN, "nand_id.bin")]
  427. print("\n build cmd : %s\n\n" %(cmd))
  428. (outmsg, exit_code) = run_cmd(cmd)
  429. if exit_code !=0:
  430. print('make build_boot_image error')
  431. print(outmsg.decode('utf-8'))
  432. sys.exit(1)
  433. build_datafs_bin(FW_DIR_BIN, app_dir, app_cfg_dir, fw_cfgfile)
  434. build_littlefs_bin(FW_DIR_BIN, app_dir, app_cfg_dir, fw_cfgfile)
  435. #copy zephyr.bin
  436. app_bin_src_path = os.path.join(OS_OUTDIR, "zephyr.bin")
  437. if os.path.exists(app_bin_src_path) == False:
  438. print("\n app.bin not eixt=%s\n\n" %(app_bin_src_path))
  439. return
  440. #app_bin_dst_path=os.path.join(FW_DIR_BIN, "zephyr.bin")
  441. app_bin_dst_path=os.path.join(FW_DIR_BIN, "app.bin")
  442. shutil.copyfile(app_bin_src_path, app_bin_dst_path)
  443. # write checksum
  444. for res_file in res_files:
  445. script_firmware_path = os.path.join(sdk_script_dir, 'pack_res_chksum.py')
  446. cmd = ['python', '-B', script_firmware_path, '-b', app_bin_dst_path, '-r', res_file + ".bin"]
  447. print("\n build cmd : %s\n\n" %(cmd))
  448. (outmsg, exit_code) = run_cmd(cmd)
  449. if exit_code !=0:
  450. print('pack app error')
  451. print(outmsg.decode('utf-8'))
  452. sys.exit(1)
  453. image_add_checksum(FW_DIR_BIN, "recovery.bin");
  454. image_add_checksum(FW_DIR_BIN, "app.bin");
  455. image_add_checksum(FW_DIR_BIN, "ota_app.bin");
  456. # signed img
  457. if os.path.exists(sign_dir) == True:
  458. check_cp(FW_DIR_BIN, sign_dir, "recovery.bin")
  459. check_cp(FW_DIR_BIN, sign_dir, "app.bin")
  460. check_cp(FW_DIR_BIN, sign_dir, "ota_app.bin")
  461. print("\n--board %s ==--\n\n" %( board))
  462. script_firmware_path = os.path.join(sdk_script_dir, 'build_firmware.py')
  463. cmd = ['python', '-B', script_firmware_path, '-c', fw_cfgfile, \
  464. '-e', os.path.join(FW_DIR_BIN, "encrypt.bin"),\
  465. '-ef', os.path.join(FW_DIR_BIN, "E_CHECK.FW"),\
  466. '-s', "lark",\
  467. '-b', board]
  468. print("\n build cmd : %s\n\n" %(cmd))
  469. (outmsg, exit_code) = run_cmd(cmd)
  470. if exit_code !=0:
  471. print('make build_firmware error\n')
  472. print(outmsg.decode('utf-8'))
  473. sys.exit(1)
  474. app_bin_dst_path=os.path.join(FW_DIR_BIN, "app.bin")
  475. shutil.copyfile(app_bin_src_path, app_bin_dst_path)
  476. image_add_checksum(FW_DIR_BIN, "app.bin");
  477. # signed img
  478. if os.path.exists(sign_dir) == True:
  479. check_cp(FW_DIR_BIN, sign_dir, "app.bin")
  480. fw_cfgfile = os.path.join(FW_DIR, "firmware_debug.xml")
  481. if os.path.exists(fw_cfgfile):
  482. cmd = ['python', '-B', script_firmware_path, '-c', fw_cfgfile, \
  483. '-e', os.path.join(FW_DIR_BIN, "encrypt.bin"),\
  484. '-ef', os.path.join(FW_DIR_BIN, "E_CHECK.FW"),\
  485. '-s', "lark",\
  486. '-b', board,\
  487. '-x', "debug"]
  488. print("\n build cmd : %s\n\n" %(cmd))
  489. (outmsg, exit_code) = run_cmd(cmd)
  490. if exit_code !=0:
  491. print('make build_firmware_debug error\n')
  492. print(outmsg.decode('utf-8'))
  493. sys.exit(1)
  494. for res_file in res_files:
  495. ota_xml_path = os.path.join(FW_DIR,"ota_"+ res_file +".xml")
  496. if os.path.exists(ota_xml_path) == True:
  497. os.remove(ota_xml_path)
  498. for full_file in full_files:
  499. ota_xml_path = os.path.join(FW_DIR,"ota_"+ full_file +".xml")
  500. if os.path.exists(ota_xml_path) == True:
  501. os.remove(ota_xml_path)
  502. def build_zephyr_app_by_keil(board, out_dir, app_dir, app_cfg_p, libname, silent):
  503. if (is_windows()):
  504. os.chdir(sdk_root)
  505. build_args = "ninja2mdk.cmd " + board
  506. build_args += " " + os.path.relpath(app_dir)
  507. build_args += " " + app_cfg_p
  508. if libname is not None:
  509. build_args += " " + libname
  510. print("\n bulid cmd:%s \n\n" %(build_args))
  511. ret = os.system(build_args)
  512. if ret != 0:
  513. print("\n bulid error\n")
  514. sys.exit(1)
  515. print("\n Warning: please build using Keil-MDK! \n\n")
  516. if silent == False:
  517. mdk_dir = os.path.join(out_dir, "mdk_clang")
  518. os.startfile(os.path.join(mdk_dir, "mdk_clang.uvprojx"))
  519. sys.exit(0)
  520. else:
  521. print("\n Error: please build on windows! \n\n")
  522. sys.exit(1)
  523. def build_zephyr_app_by_gcc(board, out_dir, app_dir, app_cfg_p, libname, silent):
  524. build_args = "west build -p auto -b " + board # +" -t ram_report" #统计ram使用情况
  525. build_args += " -d " + out_dir + " " + app_dir
  526. if app_cfg_p != ".":
  527. cfg_file = os.path.join(app_cfg_p, "prj.conf")
  528. if os.path.exists(os.path.join(app_dir,cfg_file)) == True:
  529. if (is_windows()):
  530. cfg_file = cfg_file.replace('\\', '/')
  531. build_args += " -- " + " -DCONF_FILE=" + cfg_file
  532. os.chdir(sdk_root)
  533. print("\n bulid cmd:%s \n\n" %(build_args))
  534. ret = os.system(build_args)
  535. if ret != 0:
  536. print("\n bulid error\n")
  537. sys.exit(1)
  538. def build_zephyr_menuconfig(out_dir):
  539. build_args = "west build -d " + out_dir
  540. build_args += " -t menuconfig"
  541. print("\n bulid cmd:%s \n\n" %(build_args))
  542. ret = os.system(build_args)
  543. if ret != 0:
  544. print("\n bulid error\n")
  545. sys.exit(1)
  546. def main(argv):
  547. parser = argparse.ArgumentParser()
  548. parser.add_argument('-n', dest="cfgdel", help="remove cur build_config", action="store_true", required=False)
  549. parser.add_argument('-c', dest="clear", help="remove board out dir and build_config", action="store_true", required=False)
  550. parser.add_argument('-m', dest='menuconfig', help="do sdk Configuration", action="store_true", default=False)
  551. parser.add_argument('-p', dest='pack', help="pack firmware only", action="store_true", default=False)
  552. parser.add_argument('-l', dest="libname", help="build lib", required=False)
  553. parser.add_argument('-s', dest="silent", help="silent mode", action="store_true", required=False)
  554. parser.add_argument('-t', dest="sample", help="build sample, default build applicaction", action="store_true", required=False)
  555. parser.add_argument('-u', dest="test", help="build test, default build applicaction", action="store_true", required=False)
  556. parser.add_argument('-k', dest="bkeil", help="build by keil, default by gcc ", action="store_true", required=False)
  557. # aem add begin
  558. parser.add_argument('-a', dest="application", help="Select application", required=False)
  559. parser.add_argument('-b', dest="boards", help="Select boards configuration", required=False)
  560. # aem add end
  561. print("build")
  562. args = parser.parse_args()
  563. if args.sample == True:
  564. sdk_application_dir = sdk_samples_dir
  565. elif args.test == True:
  566. sdk_application_dir = sdk_tests_dir
  567. else:
  568. sdk_application_dir = sdk_app_dir
  569. application = ""
  570. sub_app = ""
  571. board_conf = ""
  572. app_conf = ""
  573. if args.cfgdel == True:
  574. if os.path.exists(build_config_file) == True:
  575. os.remove(build_config_file)
  576. #sys.exit(0)
  577. if os.path.exists(build_config_file) == True:
  578. fp = open(build_config_file,"r")
  579. lines = fp.readlines()
  580. for elem in lines:
  581. if elem[0] != '#':
  582. _c = elem.strip().split("=")
  583. if _c[0] == "APPLICATION":
  584. application = _c[1]
  585. elif _c[0] == "BOARD":
  586. board_conf = _c[1]
  587. elif _c[0] == "SUB_APP":
  588. sub_app = _c[1]
  589. elif _c[0] == "APP_CONF":
  590. app_conf = _c[1]
  591. fp.close
  592. if os.path.exists(build_config_file) == False:
  593. # aem add begin
  594. #board_conf = read_choice("Select boards configuration:", sdk_boards_dir)
  595. #application = read_choice("Select application:", sdk_application_dir)
  596. if args.boards:
  597. board_conf = args.boards
  598. else:
  599. board_conf = read_choice("Select boards configuration:", sdk_boards_dir)
  600. if args.application:
  601. application = args.application
  602. else:
  603. application = read_choice("Select application:", sdk_application_dir)
  604. # aem add end
  605. sub_app = ""
  606. app_cfg_dir=os.path.join(sdk_application_dir, application)
  607. if os.path.exists(os.path.join(app_cfg_dir, "CMakeLists.txt")) == False:
  608. sub_app = read_choice("Select "+ application+" sub app:", os.path.join(sdk_application_dir, application))
  609. app_cfg_dir=os.path.join(app_cfg_dir, sub_app)
  610. if os.path.exists(os.path.join(app_cfg_dir, "CMakeLists.txt")) == False:
  611. print("application %s not have CMakeLists.txt\n\n" %(app_cfg_dir))
  612. sys.exit(0)
  613. app_conf = ""
  614. if os.path.exists(os.path.join(app_cfg_dir, "app_conf")) == True:
  615. app_conf = read_choice("Select application conf:", os.path.join(app_cfg_dir, "app_conf"))
  616. app_cfg_dir = os.path.join(app_cfg_dir, "app_conf", app_conf);
  617. fp = open(build_config_file,"w")
  618. fp.write("# Build config\n")
  619. fp.write("BOARD=" + board_conf + "\n")
  620. fp.write("APPLICATION=" + application + "\n")
  621. fp.write("SUB_APP=" + sub_app + "\n")
  622. fp.write("APP_CONF=" + app_conf + "\n")
  623. fp.close()
  624. print("\n--== Build application %s (sub %s) (app_conf %s) board %s ==--\n\n" %(application, sub_app, app_conf,
  625. board_conf))
  626. if os.path.exists(os.path.join(sdk_application_dir, application, "CMakeLists.txt")) == True:
  627. application_path = os.path.join(sdk_application_dir, application)
  628. else:
  629. application_path = os.path.join(sdk_application_dir, application, sub_app)
  630. if os.path.exists(application_path) == False:
  631. print("\nNo application at %s \n\n" %(sdk_application_dir))
  632. sys.exit(1)
  633. if os.path.exists(os.path.join(application_path, "app_conf")) == True:
  634. application_cfg_reldir=os.path.join("app_conf", app_conf)
  635. elif os.path.exists(os.path.join(application_path, "boards")) == True:
  636. application_cfg_reldir =os.path.join("boards", board_conf)
  637. else :
  638. application_cfg_reldir = "."
  639. print(" application cfg dir %s, \n\n" %(application_cfg_reldir))
  640. sdk_out = os.path.join(application_path, "outdir")
  641. sdk_build = os.path.join(sdk_out, board_conf)
  642. if args.clear == True:
  643. print("\n Clean build out=%s \n\n" %(sdk_out))
  644. if os.path.exists(sdk_out) == True:
  645. shutil.rmtree(sdk_out, True)
  646. time.sleep(0.01)
  647. sys.exit(0)
  648. if args.menuconfig == True:
  649. if os.path.exists(sdk_build) == True:
  650. build_zephyr_menuconfig(sdk_build)
  651. else:
  652. print("please build")
  653. sys.exit(1)
  654. print("\n sdk_build=%s \n\n" %(sdk_build))
  655. if os.path.exists(sdk_build) == False:
  656. os.makedirs(sdk_build)
  657. board_conf_path = os.path.join(sdk_boards_dir, board_conf)
  658. if os.path.exists(board_conf_path) == False:
  659. print("\nNo board at %s \n\n" %(board_conf_dir))
  660. sys.exit(1)
  661. #build_zephyr_app(board_conf, sdk_build_boot, sdk_mcuboot_app_path)
  662. if args.pack == False:
  663. if args.bkeil == True:
  664. build_zephyr_app_by_keil(board_conf, sdk_build, application_path, application_cfg_reldir, args.libname, args.silent)
  665. else:
  666. build_zephyr_app_by_gcc(board_conf, sdk_build, application_path, application_cfg_reldir, args.libname, args.silent)
  667. build_firmware(board_conf, sdk_build, board_conf_path, application_path, os.path.join(application_path, application_cfg_reldir))
  668. print("build finished")
  669. if __name__ == "__main__":
  670. main(sys.argv)