otp.py 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277
  1. #!/usr/bin/env python3
  2. import logging
  3. import argparse
  4. import subprocess
  5. import os
  6. import sys
  7. import re
  8. import struct
  9. import datetime
  10. OTP_MAGIC = 0xBABE
  11. OTP_VERSION = 0x02
  12. OTP_RESERVED = 0x00
  13. OTP_COLORS = {
  14. "unknown": 0x00,
  15. "black": 0x01,
  16. "white": 0x02,
  17. }
  18. OTP_REGIONS = {
  19. "unknown": 0x00,
  20. "eu_ru": 0x01,
  21. "us_ca_au": 0x02,
  22. "jp": 0x03,
  23. "world": 0x04,
  24. }
  25. OTP_DISPLAYS = {
  26. "unknown": 0x00,
  27. "erc": 0x01,
  28. "mgg": 0x02,
  29. }
  30. from flipper.app import App
  31. from flipper.utils.programmer_openocd import OpenOCDProgrammer, OpenOCDProgrammerResult
  32. class OTPException(Exception):
  33. def __init__(self, message: str, result: OpenOCDProgrammerResult):
  34. self.message = message
  35. self.result = result
  36. def get_exit_code(self) -> int:
  37. return int(self.result.value)
  38. class Main(App):
  39. def init(self):
  40. # SubParsers
  41. self.subparsers = self.parser.add_subparsers(help="sub-command help")
  42. # Generate All
  43. self.parser_generate_all = self.subparsers.add_parser(
  44. "generate", help="Generate OTP binary"
  45. )
  46. self._addFirstArgs(self.parser_generate_all)
  47. self._addSecondArgs(self.parser_generate_all)
  48. self.parser_generate_all.add_argument("file", help="Output file")
  49. self.parser_generate_all.set_defaults(func=self.generate_all)
  50. # Flash First
  51. self.parser_flash_first = self.subparsers.add_parser(
  52. "flash_first", help="Flash first block of OTP to device"
  53. )
  54. self._addArgsOpenOCD(self.parser_flash_first)
  55. self._addFirstArgs(self.parser_flash_first)
  56. self.parser_flash_first.set_defaults(func=self.flash_first)
  57. # Flash Second
  58. self.parser_flash_second = self.subparsers.add_parser(
  59. "flash_second", help="Flash second block of OTP to device"
  60. )
  61. self._addArgsOpenOCD(self.parser_flash_second)
  62. self._addSecondArgs(self.parser_flash_second)
  63. self.parser_flash_second.set_defaults(func=self.flash_second)
  64. # Flash All
  65. self.parser_flash_all = self.subparsers.add_parser(
  66. "flash_all", help="Flash OTP to device"
  67. )
  68. self._addArgsOpenOCD(self.parser_flash_all)
  69. self._addFirstArgs(self.parser_flash_all)
  70. self._addSecondArgs(self.parser_flash_all)
  71. self.parser_flash_all.set_defaults(func=self.flash_all)
  72. # logging
  73. self.logger = logging.getLogger()
  74. self.timestamp = datetime.datetime.now().timestamp()
  75. def _addArgsOpenOCD(self, parser):
  76. parser.add_argument(
  77. "--port-base", type=int, help="OpenOCD port base", default=3333
  78. )
  79. parser.add_argument(
  80. "--interface",
  81. type=str,
  82. help="OpenOCD interface",
  83. default="interface/cmsis-dap.cfg",
  84. )
  85. parser.add_argument(
  86. "--serial", type=str, help="OpenOCD interface serial number"
  87. )
  88. def _addFirstArgs(self, parser):
  89. parser.add_argument("--version", type=int, help="Version", required=True)
  90. parser.add_argument("--firmware", type=int, help="Firmware", required=True)
  91. parser.add_argument("--body", type=int, help="Body", required=True)
  92. parser.add_argument("--connect", type=int, help="Connect", required=True)
  93. parser.add_argument("--display", type=str, help="Display", required=True)
  94. def _addSecondArgs(self, parser):
  95. parser.add_argument("--color", type=str, help="Color", required=True)
  96. parser.add_argument("--region", type=str, help="Region", required=True)
  97. parser.add_argument("--name", type=str, help="Name", required=True)
  98. def _processFirstArgs(self):
  99. if self.args.display not in OTP_DISPLAYS:
  100. self.parser.error(f"Invalid display. Use one of {OTP_DISPLAYS.keys()}")
  101. self.args.display = OTP_DISPLAYS[self.args.display]
  102. def _processSecondArgs(self):
  103. if self.args.color not in OTP_COLORS:
  104. self.parser.error(f"Invalid color. Use one of {OTP_COLORS.keys()}")
  105. self.args.color = OTP_COLORS[self.args.color]
  106. if self.args.region not in OTP_REGIONS:
  107. self.parser.error(f"Invalid region. Use one of {OTP_REGIONS.keys()}")
  108. self.args.region = OTP_REGIONS[self.args.region]
  109. if len(self.args.name) > 8:
  110. self.parser.error("Name is too long. Max 8 symbols.")
  111. if re.match(r"^[a-zA-Z0-9.]+$", self.args.name) is None:
  112. self.parser.error(
  113. "Name contains incorrect symbols. Only a-zA-Z0-9 allowed."
  114. )
  115. def _packFirst(self):
  116. return struct.pack(
  117. "<" "HBBL" "BBBBBBH",
  118. OTP_MAGIC,
  119. OTP_VERSION,
  120. OTP_RESERVED,
  121. int(self.timestamp),
  122. self.args.version,
  123. self.args.firmware,
  124. self.args.body,
  125. self.args.connect,
  126. self.args.display,
  127. OTP_RESERVED,
  128. OTP_RESERVED,
  129. )
  130. def _packSecond(self):
  131. return struct.pack(
  132. "<" "BBHL" "8s",
  133. self.args.color,
  134. self.args.region,
  135. OTP_RESERVED,
  136. OTP_RESERVED,
  137. self.args.name.encode("ascii"),
  138. )
  139. def generate_all(self):
  140. self.logger.info(f"Generating OTP")
  141. self._processFirstArgs()
  142. self._processSecondArgs()
  143. with open(f"{self.args.file}_first.bin", "wb") as file:
  144. file.write(self._packFirst())
  145. with open(f"{self.args.file}_second.bin", "wb") as file:
  146. file.write(self._packSecond())
  147. self.logger.info(
  148. f"Generated files: {self.args.file}_first.bin and {self.args.file}_second.bin"
  149. )
  150. return 0
  151. def flash_first(self):
  152. self.logger.info(f"Flashing first block of OTP")
  153. self._processFirstArgs()
  154. filename = f"otp_unknown_first_{self.timestamp}.bin"
  155. try:
  156. self.logger.info(f"Packing binary data")
  157. with open(filename, "wb") as file:
  158. file.write(self._packFirst())
  159. self.logger.info(f"Flashing OTP")
  160. openocd = OpenOCDProgrammer(
  161. self.args.interface,
  162. self.args.port_base,
  163. self.args.serial,
  164. )
  165. programmer_result = openocd.otp_write(0x1FFF7000, filename)
  166. if programmer_result != OpenOCDProgrammerResult.Success:
  167. raise OTPException("Failed to flash OTP", programmer_result)
  168. self.logger.info(f"Flashed Successfully")
  169. except OTPException as e:
  170. self.logger.exception(e)
  171. return e.get_exit_code()
  172. finally:
  173. os.remove(filename)
  174. return 0
  175. def flash_second(self):
  176. self.logger.info(f"Flashing second block of OTP")
  177. self._processSecondArgs()
  178. filename = f"otp_{self.args.name}_second_{self.timestamp}.bin"
  179. try:
  180. self.logger.info(f"Packing binary data")
  181. with open(filename, "wb") as file:
  182. file.write(self._packSecond())
  183. self.logger.info(f"Flashing OTP")
  184. openocd = OpenOCDProgrammer(
  185. self.args.interface,
  186. self.args.port_base,
  187. self.args.serial,
  188. )
  189. programmer_result = openocd.otp_write(0x1FFF7010, filename)
  190. if programmer_result != OpenOCDProgrammerResult.Success:
  191. raise OTPException("Failed to flash OTP", programmer_result)
  192. self.logger.info(f"Flashed Successfully")
  193. except OTPException as e:
  194. self.logger.exception(e)
  195. return e.get_exit_code()
  196. finally:
  197. os.remove(filename)
  198. return 0
  199. def flash_all(self):
  200. self.logger.info(f"Flashing OTP")
  201. self._processFirstArgs()
  202. self._processSecondArgs()
  203. filename = f"otp_{self.args.name}_whole_{self.timestamp}.bin"
  204. try:
  205. self.logger.info(f"Packing binary data")
  206. with open(filename, "wb") as file:
  207. file.write(self._packFirst())
  208. file.write(self._packSecond())
  209. self.logger.info(f"Flashing OTP")
  210. openocd = OpenOCDProgrammer(
  211. self.args.interface,
  212. self.args.port_base,
  213. self.args.serial,
  214. )
  215. programmer_result = openocd.otp_write(0x1FFF7000, filename)
  216. if programmer_result != OpenOCDProgrammerResult.Success:
  217. raise OTPException("Failed to flash OTP", programmer_result)
  218. self.logger.info(f"Flashed Successfully")
  219. except OTPException as e:
  220. self.logger.exception(e)
  221. return e.get_exit_code()
  222. finally:
  223. os.remove(filename)
  224. return 0
  225. if __name__ == "__main__":
  226. Main()()