copro.py 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. import logging
  2. import json
  3. from io import BytesIO
  4. import tarfile
  5. import xml.etree.ElementTree as ET
  6. from flipper.utils import *
  7. from flipper.assets.coprobin import CoproBinary, get_stack_type
  8. CUBE_COPRO_PATH = "Projects/STM32WB_Copro_Wireless_Binaries"
  9. MANIFEST_TEMPLATE = {
  10. "manifest": {"version": 0, "timestamp": 0},
  11. "copro": {
  12. "fus": {"version": {"major": 1, "minor": 2, "sub": 0}, "files": []},
  13. "radio": {
  14. "version": {},
  15. "files": [],
  16. },
  17. },
  18. }
  19. class Copro:
  20. def __init__(self, mcu):
  21. self.mcu = mcu
  22. self.version = None
  23. self.cube_dir = None
  24. self.mcu_copro = None
  25. self.logger = logging.getLogger(self.__class__.__name__)
  26. def loadCubeInfo(self, cube_dir, cube_version):
  27. if not os.path.isdir(cube_dir):
  28. raise Exception(f'"{cube_dir}" doesn\'t exists')
  29. self.cube_dir = cube_dir
  30. self.mcu_copro = os.path.join(self.cube_dir, CUBE_COPRO_PATH, self.mcu)
  31. if not os.path.isdir(self.mcu_copro):
  32. raise Exception(f'"{self.mcu_copro}" doesn\'t exists')
  33. cube_manifest_file = os.path.join(self.cube_dir, "package.xml")
  34. cube_manifest = ET.parse(cube_manifest_file)
  35. cube_package = cube_manifest.find("PackDescription")
  36. if not cube_package:
  37. raise Exception(f"Unknown Cube manifest format")
  38. cube_version = cube_package.get("Patch") or cube_package.get("Release")
  39. if not cube_version or not cube_version.startswith("FW.WB"):
  40. raise Exception(f"Incorrect Cube package or version info")
  41. cube_version = cube_version.replace("FW.WB.", "", 1)
  42. if cube_version != cube_version:
  43. raise Exception(f"Unsupported cube version")
  44. self.version = cube_version
  45. @staticmethod
  46. def _getFileName(name):
  47. return os.path.join("core2_firmware", name)
  48. def addFile(self, array, filename, **kwargs):
  49. source_file = os.path.join(self.mcu_copro, filename)
  50. self.output_tar.add(source_file, arcname=self._getFileName(filename))
  51. array.append({"name": filename, "sha256": file_sha256(source_file), **kwargs})
  52. def bundle(self, output_file, stack_file_name, stack_type, stack_addr=None):
  53. self.output_tar = tarfile.open(output_file, "w:gz")
  54. stack_file = os.path.join(self.mcu_copro, stack_file_name)
  55. # Form Manifest
  56. manifest = dict(MANIFEST_TEMPLATE)
  57. manifest["manifest"]["timestamp"] = timestamp()
  58. copro_bin = CoproBinary(stack_file)
  59. self.logger.info(f"Bundling {copro_bin.img_sig.get_version()}")
  60. stack_type_code = get_stack_type(stack_type)
  61. manifest["copro"]["radio"]["version"].update(
  62. {
  63. "type": stack_type_code,
  64. "major": copro_bin.img_sig.version_major,
  65. "minor": copro_bin.img_sig.version_minor,
  66. "sub": copro_bin.img_sig.version_sub,
  67. "branch": copro_bin.img_sig.version_branch,
  68. "release": copro_bin.img_sig.version_build,
  69. }
  70. )
  71. if not stack_addr:
  72. stack_addr = copro_bin.get_flash_load_addr()
  73. self.logger.info(f"Using guessed flash address 0x{stack_addr:x}")
  74. # Old FUS Update
  75. self.addFile(
  76. manifest["copro"]["fus"]["files"],
  77. "stm32wb5x_FUS_fw_for_fus_0_5_3.bin",
  78. condition="==0.5.3",
  79. address="0x080EC000",
  80. )
  81. # New FUS Update
  82. self.addFile(
  83. manifest["copro"]["fus"]["files"],
  84. "stm32wb5x_FUS_fw.bin",
  85. condition=">0.5.3",
  86. address="0x080EC000",
  87. )
  88. # BLE Full Stack
  89. self.addFile(
  90. manifest["copro"]["radio"]["files"],
  91. stack_file_name,
  92. address=f"0x{stack_addr:X}",
  93. )
  94. # Save manifest
  95. manifest_data = json.dumps(manifest, indent=4).encode("utf-8")
  96. info = tarfile.TarInfo(self._getFileName("Manifest.json"))
  97. info.size = len(manifest_data)
  98. self.output_tar.addfile(info, BytesIO(manifest_data))
  99. self.output_tar.close()