2019-01-23 16:31:06 +01:00
|
|
|
# Copyright (c) 2017 Linaro Limited.
|
|
|
|
#
|
|
|
|
# SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
'''Runner for pyOCD .'''
|
|
|
|
|
|
|
|
import os
|
2021-03-16 14:02:48 +01:00
|
|
|
from os import path
|
2019-01-23 16:31:06 +01:00
|
|
|
|
2022-11-30 11:18:13 +01:00
|
|
|
from runners.core import ZephyrBinaryRunner, RunnerCaps, BuildConfiguration
|
2019-01-23 16:31:06 +01:00
|
|
|
|
|
|
|
DEFAULT_PYOCD_GDB_PORT = 3333
|
2019-09-30 15:04:05 +02:00
|
|
|
DEFAULT_PYOCD_TELNET_PORT = 4444
|
2019-01-23 16:31:06 +01:00
|
|
|
|
|
|
|
|
|
|
|
class PyOcdBinaryRunner(ZephyrBinaryRunner):
|
|
|
|
'''Runner front-end for pyOCD.'''
|
|
|
|
|
|
|
|
def __init__(self, cfg, target,
|
2019-01-30 16:42:22 +01:00
|
|
|
pyocd='pyocd',
|
2021-08-06 18:47:57 +02:00
|
|
|
dev_id=None, flash_addr=0x0, erase=False, flash_opts=None,
|
2019-09-30 15:04:05 +02:00
|
|
|
gdb_port=DEFAULT_PYOCD_GDB_PORT,
|
|
|
|
telnet_port=DEFAULT_PYOCD_TELNET_PORT, tui=False,
|
2021-03-16 14:02:48 +01:00
|
|
|
pyocd_config=None,
|
2021-08-06 18:47:57 +02:00
|
|
|
daparg=None, frequency=None, tool_opt=None):
|
2020-06-23 22:27:11 +02:00
|
|
|
super().__init__(cfg)
|
2019-01-23 16:31:06 +01:00
|
|
|
|
2021-03-16 14:02:48 +01:00
|
|
|
default = path.join(cfg.board_dir, 'support', 'pyocd.yaml')
|
|
|
|
if path.exists(default):
|
|
|
|
self.pyocd_config = default
|
|
|
|
else:
|
|
|
|
self.pyocd_config = None
|
|
|
|
|
|
|
|
|
2019-01-23 16:31:06 +01:00
|
|
|
self.target_args = ['-t', target]
|
2019-01-30 16:42:22 +01:00
|
|
|
self.pyocd = pyocd
|
2019-01-23 16:31:06 +01:00
|
|
|
self.flash_addr_args = ['-a', hex(flash_addr)] if flash_addr else []
|
2020-07-09 14:09:34 +02:00
|
|
|
self.erase = erase
|
2019-01-23 16:31:06 +01:00
|
|
|
self.gdb_cmd = [cfg.gdb] if cfg.gdb is not None else None
|
|
|
|
self.gdb_port = gdb_port
|
2019-09-30 15:04:05 +02:00
|
|
|
self.telnet_port = telnet_port
|
2019-01-23 16:31:06 +01:00
|
|
|
self.tui_args = ['-tui'] if tui else []
|
|
|
|
self.hex_name = cfg.hex_file
|
|
|
|
self.bin_name = cfg.bin_file
|
|
|
|
self.elf_name = cfg.elf_file
|
|
|
|
|
2021-03-16 14:02:48 +01:00
|
|
|
pyocd_config_args = []
|
|
|
|
|
|
|
|
if self.pyocd_config is not None:
|
|
|
|
pyocd_config_args = ['--config', self.pyocd_config]
|
|
|
|
|
|
|
|
self.pyocd_config_args = pyocd_config_args
|
|
|
|
|
2019-01-23 16:31:06 +01:00
|
|
|
board_args = []
|
2021-08-06 18:47:57 +02:00
|
|
|
if dev_id is not None:
|
|
|
|
board_args = ['-u', dev_id]
|
2019-01-23 16:31:06 +01:00
|
|
|
self.board_args = board_args
|
|
|
|
|
|
|
|
daparg_args = []
|
|
|
|
if daparg is not None:
|
|
|
|
daparg_args = ['-da', daparg]
|
|
|
|
self.daparg_args = daparg_args
|
|
|
|
|
|
|
|
frequency_args = []
|
|
|
|
if frequency is not None:
|
|
|
|
frequency_args = ['-f', frequency]
|
|
|
|
self.frequency_args = frequency_args
|
|
|
|
|
2022-08-02 20:48:07 +02:00
|
|
|
self.tool_opt_args = tool_opt or []
|
2019-11-24 10:10:07 +01:00
|
|
|
|
2019-01-30 16:42:22 +01:00
|
|
|
self.flash_extra = flash_opts if flash_opts else []
|
2019-01-23 16:31:06 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def name(cls):
|
|
|
|
return 'pyocd'
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def capabilities(cls):
|
|
|
|
return RunnerCaps(commands={'flash', 'debug', 'debugserver', 'attach'},
|
2022-08-02 20:48:07 +02:00
|
|
|
dev_id=True, flash_addr=True, erase=True,
|
|
|
|
tool_opt=True)
|
2021-08-06 18:47:57 +02:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def dev_id_help(cls) -> str:
|
|
|
|
return '''Device identifier. Use it to select the probe's unique ID
|
|
|
|
or substring thereof.'''
|
2019-01-23 16:31:06 +01:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def do_add_parser(cls, parser):
|
|
|
|
parser.add_argument('--target', required=True,
|
|
|
|
help='target override')
|
|
|
|
|
|
|
|
parser.add_argument('--daparg',
|
|
|
|
help='Additional -da arguments to pyocd tool')
|
2019-01-30 16:42:22 +01:00
|
|
|
parser.add_argument('--pyocd', default='pyocd',
|
|
|
|
help='path to pyocd tool, default is pyocd')
|
|
|
|
parser.add_argument('--flash-opt', default=[], action='append',
|
|
|
|
help='''Additional options for pyocd flash,
|
2019-06-25 19:33:27 +02:00
|
|
|
e.g. --flash-opt="-e=chip" to chip erase''')
|
2019-01-23 16:31:06 +01:00
|
|
|
parser.add_argument('--frequency',
|
|
|
|
help='SWD clock frequency in Hz')
|
|
|
|
parser.add_argument('--gdb-port', default=DEFAULT_PYOCD_GDB_PORT,
|
|
|
|
help='pyocd gdb port, defaults to {}'.format(
|
|
|
|
DEFAULT_PYOCD_GDB_PORT))
|
2019-09-30 15:04:05 +02:00
|
|
|
parser.add_argument('--telnet-port', default=DEFAULT_PYOCD_TELNET_PORT,
|
|
|
|
help='pyocd telnet port, defaults to {}'.format(
|
|
|
|
DEFAULT_PYOCD_TELNET_PORT))
|
2019-01-23 16:31:06 +01:00
|
|
|
parser.add_argument('--tui', default=False, action='store_true',
|
|
|
|
help='if given, GDB uses -tui')
|
2023-01-20 03:01:09 +01:00
|
|
|
parser.add_argument('--board-id', dest='dev_id',
|
|
|
|
help='obsolete synonym for -i/--dev-id')
|
2022-08-02 20:48:07 +02:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def tool_opt_help(cls) -> str:
|
|
|
|
return """Additional options for pyocd commander,
|
|
|
|
e.g. '--script=user.py'"""
|
2019-01-23 16:31:06 +01:00
|
|
|
|
|
|
|
@classmethod
|
2020-06-23 22:35:52 +02:00
|
|
|
def do_create(cls, cfg, args):
|
2019-01-23 16:31:06 +01:00
|
|
|
build_conf = BuildConfiguration(cfg.build_dir)
|
|
|
|
flash_addr = cls.get_flash_address(args, build_conf)
|
|
|
|
|
2019-06-03 05:33:41 +02:00
|
|
|
ret = PyOcdBinaryRunner(
|
2019-01-30 16:42:22 +01:00
|
|
|
cfg, args.target,
|
|
|
|
pyocd=args.pyocd,
|
2020-07-09 14:09:34 +02:00
|
|
|
flash_addr=flash_addr, erase=args.erase, flash_opts=args.flash_opt,
|
2019-09-30 15:04:05 +02:00
|
|
|
gdb_port=args.gdb_port, telnet_port=args.telnet_port, tui=args.tui,
|
2021-08-06 18:47:57 +02:00
|
|
|
dev_id=args.dev_id, daparg=args.daparg,
|
2019-11-24 10:10:07 +01:00
|
|
|
frequency=args.frequency,
|
|
|
|
tool_opt=args.tool_opt)
|
2019-01-23 16:31:06 +01:00
|
|
|
|
2019-06-03 05:33:41 +02:00
|
|
|
daparg = os.environ.get('PYOCD_DAPARG')
|
|
|
|
if not ret.daparg_args and daparg:
|
|
|
|
ret.logger.warning('PYOCD_DAPARG is deprecated; use --daparg')
|
|
|
|
ret.logger.debug('--daparg={} via PYOCD_DAPARG'.format(daparg))
|
|
|
|
ret.daparg_args = ['-da', daparg]
|
|
|
|
|
|
|
|
return ret
|
|
|
|
|
2019-01-23 16:31:06 +01:00
|
|
|
def port_args(self):
|
2019-09-30 15:04:05 +02:00
|
|
|
return ['-p', str(self.gdb_port), '-T', str(self.telnet_port)]
|
2019-01-23 16:31:06 +01:00
|
|
|
|
|
|
|
def do_run(self, command, **kwargs):
|
2019-06-03 00:18:39 +02:00
|
|
|
self.require(self.pyocd)
|
2019-01-23 16:31:06 +01:00
|
|
|
if command == 'flash':
|
|
|
|
self.flash(**kwargs)
|
|
|
|
else:
|
|
|
|
self.debug_debugserver(command, **kwargs)
|
|
|
|
|
|
|
|
def flash(self, **kwargs):
|
2021-02-03 15:12:08 +01:00
|
|
|
if self.hex_name is not None and os.path.isfile(self.hex_name):
|
2019-01-23 16:31:06 +01:00
|
|
|
fname = self.hex_name
|
2021-02-03 15:12:08 +01:00
|
|
|
elif self.bin_name is not None and os.path.isfile(self.bin_name):
|
2019-08-15 17:33:21 +02:00
|
|
|
self.logger.warning(
|
|
|
|
'hex file ({}) does not exist; falling back on .bin ({}). '.
|
|
|
|
format(self.hex_name, self.bin_name) +
|
|
|
|
'Consider enabling CONFIG_BUILD_OUTPUT_HEX.')
|
2019-01-23 16:31:06 +01:00
|
|
|
fname = self.bin_name
|
|
|
|
else:
|
|
|
|
raise ValueError(
|
2019-08-15 17:33:21 +02:00
|
|
|
'Cannot flash; no hex ({}) or bin ({}) files found. '.format(
|
2019-01-23 16:31:06 +01:00
|
|
|
self.hex_name, self.bin_name))
|
|
|
|
|
2020-07-09 14:09:34 +02:00
|
|
|
erase_method = 'chip' if self.erase else 'sector'
|
|
|
|
|
2019-01-30 16:42:22 +01:00
|
|
|
cmd = ([self.pyocd] +
|
|
|
|
['flash'] +
|
2021-03-16 14:02:48 +01:00
|
|
|
self.pyocd_config_args +
|
2020-07-09 14:09:34 +02:00
|
|
|
['-e', erase_method] +
|
2019-01-23 16:31:06 +01:00
|
|
|
self.flash_addr_args +
|
|
|
|
self.daparg_args +
|
|
|
|
self.target_args +
|
|
|
|
self.board_args +
|
|
|
|
self.frequency_args +
|
2019-11-24 10:10:07 +01:00
|
|
|
self.tool_opt_args +
|
2019-01-30 16:42:22 +01:00
|
|
|
self.flash_extra +
|
2019-01-23 16:31:06 +01:00
|
|
|
[fname])
|
|
|
|
|
2019-08-15 17:33:21 +02:00
|
|
|
self.logger.info('Flashing file: {}'.format(fname))
|
2019-01-23 16:31:06 +01:00
|
|
|
self.check_call(cmd)
|
|
|
|
|
2019-06-03 05:33:41 +02:00
|
|
|
def log_gdbserver_message(self):
|
|
|
|
self.logger.info('pyOCD GDB server running on port {}'.
|
|
|
|
format(self.gdb_port))
|
2019-01-23 16:31:06 +01:00
|
|
|
|
|
|
|
def debug_debugserver(self, command, **kwargs):
|
2019-01-30 16:42:22 +01:00
|
|
|
server_cmd = ([self.pyocd] +
|
|
|
|
['gdbserver'] +
|
2019-01-23 16:31:06 +01:00
|
|
|
self.daparg_args +
|
|
|
|
self.port_args() +
|
|
|
|
self.target_args +
|
|
|
|
self.board_args +
|
2019-11-24 10:10:07 +01:00
|
|
|
self.frequency_args +
|
|
|
|
self.tool_opt_args)
|
2019-01-23 16:31:06 +01:00
|
|
|
|
|
|
|
if command == 'debugserver':
|
2019-06-03 05:33:41 +02:00
|
|
|
self.log_gdbserver_message()
|
2019-01-23 16:31:06 +01:00
|
|
|
self.check_call(server_cmd)
|
|
|
|
else:
|
|
|
|
if self.gdb_cmd is None:
|
|
|
|
raise ValueError('Cannot debug; gdb is missing')
|
|
|
|
if self.elf_name is None:
|
|
|
|
raise ValueError('Cannot debug; elf is missing')
|
|
|
|
client_cmd = (self.gdb_cmd +
|
|
|
|
self.tui_args +
|
|
|
|
[self.elf_name] +
|
|
|
|
['-ex', 'target remote :{}'.format(self.gdb_port)])
|
|
|
|
if command == 'debug':
|
|
|
|
client_cmd += ['-ex', 'monitor halt',
|
|
|
|
'-ex', 'monitor reset',
|
|
|
|
'-ex', 'load']
|
|
|
|
|
2019-06-03 00:18:39 +02:00
|
|
|
self.require(client_cmd[0])
|
2019-06-03 05:33:41 +02:00
|
|
|
self.log_gdbserver_message()
|
2019-01-23 16:31:06 +01:00
|
|
|
self.run_server_and_client(server_cmd, client_cmd)
|