1# Copyright (c) 2019 Thomas Kupper <thomas.kupper@gmail.com> 2# 3# SPDX-License-Identifier: Apache-2.0 4 5'''Runner for flashing with stm32flash.''' 6 7import platform 8from os import path 9 10from runners.core import RunnerCaps, ZephyrBinaryRunner 11 12DEFAULT_DEVICE = '/dev/ttyUSB0' 13if platform.system() == 'Darwin': 14 DEFAULT_DEVICE = '/dev/tty.SLAB_USBtoUART' 15 16class Stm32flashBinaryRunner(ZephyrBinaryRunner): 17 '''Runner front-end for stm32flash.''' 18 19 def __init__(self, cfg, device, action='write', baud=57600, 20 force_binary=False, start_addr=0, exec_addr=None, 21 serial_mode='8e1', reset=False, verify=False): 22 super().__init__(cfg) 23 24 self.device = device 25 self.action = action 26 self.baud = baud 27 self.force_binary = force_binary 28 self.start_addr = start_addr 29 self.exec_addr = exec_addr 30 self.serial_mode = serial_mode 31 self.reset = reset 32 self.verify = verify 33 34 @classmethod 35 def name(cls): 36 return 'stm32flash' 37 38 @classmethod 39 def capabilities(cls): 40 return RunnerCaps(commands={'flash'}, reset=True) 41 42 @classmethod 43 def do_add_parser(cls, parser): 44 45 # required argument(s) 46 # none for now 47 48 # optional argument(s) 49 parser.add_argument('--device', default=DEFAULT_DEVICE, required=False, 50 help='serial port to flash, default \'' + DEFAULT_DEVICE + '\'') 51 52 parser.add_argument('--action', default='write', required=False, 53 choices=['erase', 'info', 'start', 'write'], 54 help='erase / get device info / start execution / write flash') 55 56 parser.add_argument('--baud-rate', default='57600', required=False, 57 choices=['1200', '1800', '2400', '4800', '9600', '19200', 58 '38400', '57600', '115200', '230400', '256000', '460800', 59 '500000', '576000', '921600', '1000000', '1500000', '2000000'], 60 help='serial baud rate, default \'57600\'') 61 62 parser.add_argument('--force-binary', required=False, action='store_true', 63 help='force the binary parser') 64 65 parser.add_argument('--start-addr', default=0, required=False, 66 help='specify start address for write operation, default \'0\'') 67 68 parser.add_argument('--execution-addr', default=None, required=False, 69 help='start execution at specified address, default \'0\' \ 70 which means start of flash') 71 72 parser.add_argument('--serial-mode', default='8e1', required=False, 73 help='serial port mode, default \'8e1\'') 74 75 parser.add_argument('--verify', default=False, required=False, action='store_true', 76 help='verify writes, default False') 77 78 parser.set_defaults(reset=False) 79 80 @classmethod 81 def do_create(cls, cfg, args): 82 return Stm32flashBinaryRunner(cfg, device=args.device, action=args.action, 83 baud=args.baud_rate, force_binary=args.force_binary, 84 start_addr=args.start_addr, exec_addr=args.execution_addr, 85 serial_mode=args.serial_mode, reset=args.reset, verify=args.verify) 86 87 def do_run(self, command, **kwargs): 88 self.require('stm32flash') 89 self.ensure_output('bin') 90 91 bin_name = self.cfg.bin_file 92 bin_size = path.getsize(bin_name) 93 94 cmd_flash = ['stm32flash', '-b', self.baud, 95 '-m', self.serial_mode] 96 97 action = self.action.lower() 98 99 if action == 'info': 100 # show device information and exit 101 msg_text = f"get device info from {self.device}" 102 103 elif action == 'erase': 104 # erase flash 105 #size_aligned = (int(bin_size) >> 12) + 1 << 12 106 size_aligned = (int(bin_size) & 0xfffff000) + 4096 107 msg_text = f"erase {size_aligned} bit starting at {self.start_addr}" 108 cmd_flash.extend([ 109 '-S', str(self.start_addr) + ":" + str(size_aligned), '-o']) 110 111 elif action == 'start': 112 # start execution 113 msg_text = f"start code execution at {self.exec_addr}" 114 if self.exec_addr: 115 if self.exec_addr == 0 or self.exec_addr.lower() == '0x0': 116 msg_text += " (flash start)" 117 else: 118 self.exec_addr = 0 119 cmd_flash.extend([ 120 '-g', str(self.exec_addr)]) 121 122 elif action == 'write': 123 # flash binary file 124 msg_text = f"write {bin_size} bytes starting at {self.start_addr}" 125 cmd_flash.extend([ 126 '-S', str(self.start_addr) + ":" + str(bin_size), 127 '-w', bin_name]) 128 129 if self.exec_addr: 130 cmd_flash.extend(['-g', self.exec_addr]) 131 132 if self.force_binary: 133 cmd_flash.extend(['-f']) 134 135 if self.reset: 136 cmd_flash.extend(['-R']) 137 138 if self.verify: 139 cmd_flash.extend(['-v']) 140 141 else: 142 msg_text = f"invalid action \'{action}\' passed!" 143 self.logger.error(f'Invalid action \'{action}\' passed!') 144 return -1 145 146 cmd_flash.extend([self.device]) 147 self.logger.info("Board: " + msg_text) 148 self.check_call(cmd_flash) 149 self.logger.info(f'Board: finished \'{action}\' .') 150