163 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
			
		
		
	
	
			163 lines
		
	
	
		
			5.2 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable File
		
	
	
	
	
#!/usr/bin/env python3
 | 
						|
 | 
						|
import logging
 | 
						|
import argparse
 | 
						|
import subprocess
 | 
						|
import os
 | 
						|
import sys
 | 
						|
import re
 | 
						|
import struct
 | 
						|
import datetime
 | 
						|
 | 
						|
OTP_MAGIC = 0xBABE
 | 
						|
OTP_VERSION = 0x01
 | 
						|
OTP_RESERVED = 0x00
 | 
						|
 | 
						|
OTP_COLORS = {
 | 
						|
    "unknown": 0x00,
 | 
						|
    "black": 0x01,
 | 
						|
    "white": 0x02,
 | 
						|
}
 | 
						|
OTP_REGIONS = {
 | 
						|
    "unknown": 0x00,
 | 
						|
    "eu_ru": 0x01,
 | 
						|
    "us_ca_au": 0x02,
 | 
						|
    "jp": 0x03,
 | 
						|
}
 | 
						|
 | 
						|
BOARD_RESERVED = 0x0000
 | 
						|
 | 
						|
 | 
						|
class Main:
 | 
						|
    def __init__(self):
 | 
						|
        # command args
 | 
						|
        self.parser = argparse.ArgumentParser()
 | 
						|
        self.parser.add_argument("-d", "--debug", action="store_true", help="Debug")
 | 
						|
        self.subparsers = self.parser.add_subparsers(help="sub-command help")
 | 
						|
        # Generate
 | 
						|
        self.parser_generate = self.subparsers.add_parser(
 | 
						|
            "generate", help="Generate OTP binary"
 | 
						|
        )
 | 
						|
        self._add_args(self.parser_generate)
 | 
						|
        self.parser_generate.add_argument("file", help="Output file")
 | 
						|
        self.parser_generate.set_defaults(func=self.generate)
 | 
						|
        # Flash
 | 
						|
        self.parser_flash = self.subparsers.add_parser(
 | 
						|
            "flash", help="Flash OTP to device"
 | 
						|
        )
 | 
						|
        self._add_args(self.parser_flash)
 | 
						|
        self.parser_flash.add_argument(
 | 
						|
            "--port", type=str, help="Port to connect: swd or usb1", default="swd"
 | 
						|
        )
 | 
						|
        self.parser_flash.set_defaults(func=self.flash)
 | 
						|
        # logging
 | 
						|
        self.logger = logging.getLogger()
 | 
						|
        self.timestamp = datetime.datetime.now().timestamp()
 | 
						|
 | 
						|
    def __call__(self):
 | 
						|
        self.args = self.parser.parse_args()
 | 
						|
        if "func" not in self.args:
 | 
						|
            self.parser.error("Choose something to do")
 | 
						|
        # configure log output
 | 
						|
        self.log_level = logging.DEBUG if self.args.debug else logging.INFO
 | 
						|
        self.logger.setLevel(self.log_level)
 | 
						|
        self.handler = logging.StreamHandler(sys.stdout)
 | 
						|
        self.handler.setLevel(self.log_level)
 | 
						|
        self.formatter = logging.Formatter("%(asctime)s [%(levelname)s] %(message)s")
 | 
						|
        self.handler.setFormatter(self.formatter)
 | 
						|
        self.logger.addHandler(self.handler)
 | 
						|
        # execute requested function
 | 
						|
        self.args.func()
 | 
						|
 | 
						|
    def _add_args(self, parser):
 | 
						|
        parser.add_argument("--version", type=int, help="Version", default=10)
 | 
						|
        parser.add_argument("--firmware", type=int, help="Firmware", default=6)
 | 
						|
        parser.add_argument("--body", type=int, help="Body", default=8)
 | 
						|
        parser.add_argument("--connect", type=int, help="Connect", default=5)
 | 
						|
        parser.add_argument("--color", type=str, help="Color", default="unknown")
 | 
						|
        parser.add_argument("--region", type=str, help="Region", default="unknown")
 | 
						|
        parser.add_argument("--name", type=str, help="Name", required=True)
 | 
						|
 | 
						|
    def _process_args(self):
 | 
						|
        if self.args.color not in OTP_COLORS:
 | 
						|
            self.parser.error(f"Invalid color. Use one of {OTP_COLORS.keys()}")
 | 
						|
        self.args.color = OTP_COLORS[self.args.color]
 | 
						|
 | 
						|
        if self.args.region not in OTP_REGIONS:
 | 
						|
            self.parser.error(f"Invalid region. Use one of {OTP_REGIONS.keys()}")
 | 
						|
        self.args.region = OTP_REGIONS[self.args.region]
 | 
						|
 | 
						|
        if len(self.args.name) > 8:
 | 
						|
            self.parser.error("Name is too long. Max 8 symbols.")
 | 
						|
        if re.match(r"^[a-zA-Z0-9.]+$", self.args.name) is None:
 | 
						|
            self.parser.error(
 | 
						|
                "Name contains incorrect symbols. Only a-zA-Z0-9 allowed."
 | 
						|
            )
 | 
						|
 | 
						|
    def _pack_struct(self):
 | 
						|
        return struct.pack(
 | 
						|
            "<" "HBBL" "BBBBBBH" "8s",
 | 
						|
            OTP_MAGIC,
 | 
						|
            OTP_VERSION,
 | 
						|
            OTP_RESERVED,
 | 
						|
            int(self.timestamp),
 | 
						|
            self.args.version,
 | 
						|
            self.args.firmware,
 | 
						|
            self.args.body,
 | 
						|
            self.args.connect,
 | 
						|
            self.args.color,
 | 
						|
            self.args.region,
 | 
						|
            BOARD_RESERVED,
 | 
						|
            self.args.name.encode("ascii"),
 | 
						|
        )
 | 
						|
 | 
						|
    def generate(self):
 | 
						|
        self.logger.debug(f"Generating OTP")
 | 
						|
        self._process_args()
 | 
						|
        data = self._pack_struct()
 | 
						|
        open(self.args.file, "wb").write(data)
 | 
						|
 | 
						|
    def flash(self):
 | 
						|
        self.logger.debug(f"Flashing OTP")
 | 
						|
        self._process_args()
 | 
						|
        data = self._pack_struct()
 | 
						|
        filename = f"otp_{self.args.name}_{self.timestamp}.bin"
 | 
						|
        open(filename, "wb").write(data)
 | 
						|
 | 
						|
        try:
 | 
						|
            output = subprocess.check_output(
 | 
						|
                [
 | 
						|
                    "STM32_Programmer_CLI",
 | 
						|
                    "-q",
 | 
						|
                    "-c",
 | 
						|
                    f"port={self.args.port}",
 | 
						|
                    "-d",
 | 
						|
                    filename,
 | 
						|
                    "0x1FFF7000",
 | 
						|
                ]
 | 
						|
            )
 | 
						|
            assert output
 | 
						|
            self.logger.info(f"Success")
 | 
						|
        except subprocess.CalledProcessError as e:
 | 
						|
            self.logger.error(e.output.decode())
 | 
						|
            self.logger.error(f"Failed to call STM32_Programmer_CLI")
 | 
						|
            return
 | 
						|
        except Exception as e:
 | 
						|
            self.logger.error(f"Failed to call STM32_Programmer_CLI")
 | 
						|
            self.logger.exception(e)
 | 
						|
            return
 | 
						|
        # reboot
 | 
						|
        subprocess.check_output(
 | 
						|
            [
 | 
						|
                "STM32_Programmer_CLI",
 | 
						|
                "-q",
 | 
						|
                "-c",
 | 
						|
                f"port={self.args.port}",
 | 
						|
            ]
 | 
						|
        )
 | 
						|
        os.remove(filename)
 | 
						|
 | 
						|
 | 
						|
if __name__ == "__main__":
 | 
						|
    Main()()
 |