otp.py 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  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 = 0x01
  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. }
  24. BOARD_RESERVED = 0x0000
  25. class Main:
  26. def __init__(self):
  27. # command args
  28. self.parser = argparse.ArgumentParser()
  29. self.parser.add_argument("-d", "--debug", action="store_true", help="Debug")
  30. self.subparsers = self.parser.add_subparsers(help="sub-command help")
  31. # Generate
  32. self.parser_generate = self.subparsers.add_parser(
  33. "generate", help="Generate OTP binary"
  34. )
  35. self._add_args(self.parser_generate)
  36. self.parser_generate.add_argument("file", help="Output file")
  37. self.parser_generate.set_defaults(func=self.generate)
  38. # Flash
  39. self.parser_flash = self.subparsers.add_parser(
  40. "flash", help="Flash OTP to device"
  41. )
  42. self._add_args(self.parser_flash)
  43. self.parser_flash.add_argument(
  44. "--port", type=str, help="Port to connect: swd or usb1", default="swd"
  45. )
  46. self.parser_flash.set_defaults(func=self.flash)
  47. # logging
  48. self.logger = logging.getLogger()
  49. self.timestamp = datetime.datetime.now().timestamp()
  50. def __call__(self):
  51. self.args = self.parser.parse_args()
  52. if "func" not in self.args:
  53. self.parser.error("Choose something to do")
  54. # configure log output
  55. self.log_level = logging.DEBUG if self.args.debug else logging.INFO
  56. self.logger.setLevel(self.log_level)
  57. self.handler = logging.StreamHandler(sys.stdout)
  58. self.handler.setLevel(self.log_level)
  59. self.formatter = logging.Formatter("%(asctime)s [%(levelname)s] %(message)s")
  60. self.handler.setFormatter(self.formatter)
  61. self.logger.addHandler(self.handler)
  62. # execute requested function
  63. self.args.func()
  64. def _add_args(self, parser):
  65. parser.add_argument("--version", type=int, help="Version", default=10)
  66. parser.add_argument("--firmware", type=int, help="Firmware", default=6)
  67. parser.add_argument("--body", type=int, help="Body", default=8)
  68. parser.add_argument("--connect", type=int, help="Connect", default=5)
  69. parser.add_argument("--color", type=str, help="Color", default="unknown")
  70. parser.add_argument("--region", type=str, help="Region", default="unknown")
  71. parser.add_argument("--name", type=str, help="Name", required=True)
  72. def _process_args(self):
  73. if self.args.color not in OTP_COLORS:
  74. self.parser.error(f"Invalid color. Use one of {OTP_COLORS.keys()}")
  75. self.args.color = OTP_COLORS[self.args.color]
  76. if self.args.region not in OTP_REGIONS:
  77. self.parser.error(f"Invalid region. Use one of {OTP_REGIONS.keys()}")
  78. self.args.region = OTP_REGIONS[self.args.region]
  79. if len(self.args.name) > 8:
  80. self.parser.error("Name is too long. Max 8 symbols.")
  81. if re.match(r"[a-zA-Z0-9]+", self.args.name) is None:
  82. self.parser.error(
  83. "Name contains incorrect symbols. Only a-zA-Z0-9 allowed."
  84. )
  85. def _pack_struct(self):
  86. return struct.pack(
  87. "<" "HBBL" "BBBBBBH" "8s",
  88. OTP_MAGIC,
  89. OTP_VERSION,
  90. OTP_RESERVED,
  91. int(self.timestamp),
  92. self.args.version,
  93. self.args.firmware,
  94. self.args.body,
  95. self.args.connect,
  96. self.args.color,
  97. self.args.region,
  98. BOARD_RESERVED,
  99. self.args.name.encode("ascii"),
  100. )
  101. def generate(self):
  102. self.logger.debug(f"Generating OTP")
  103. self._process_args()
  104. data = self._pack_struct()
  105. open(self.args.file, "wb").write(data)
  106. def flash(self):
  107. self.logger.debug(f"Flashing OTP")
  108. self._process_args()
  109. data = self._pack_struct()
  110. filename = f"otp_{self.args.name}_{self.timestamp}.bin"
  111. open(filename, "wb").write(data)
  112. try:
  113. output = subprocess.check_output(
  114. [
  115. "STM32_Programmer_CLI",
  116. "-q",
  117. "-c",
  118. f"port={self.args.port}",
  119. "-d",
  120. filename,
  121. "0x1FFF7000",
  122. ]
  123. )
  124. assert output
  125. self.logger.info(f"Success")
  126. except subprocess.CalledProcessError as e:
  127. self.logger.error(e.output.decode())
  128. self.logger.error(f"Failed to call STM32_Programmer_CLI")
  129. return
  130. except Exception as e:
  131. self.logger.error(f"Failed to call STM32_Programmer_CLI")
  132. self.logger.exception(e)
  133. return
  134. # reboot
  135. subprocess.check_output(
  136. [
  137. "STM32_Programmer_CLI",
  138. "-q",
  139. "-c",
  140. f"port={self.args.port}",
  141. ]
  142. )
  143. os.remove(filename)
  144. if __name__ == "__main__":
  145. Main()()