pyocd.py 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. # Copyright (c) 2017 Linaro Limited.
  2. #
  3. # SPDX-License-Identifier: Apache-2.0
  4. '''Runner for pyOCD .'''
  5. import os
  6. from os import path
  7. from runners.core import ZephyrBinaryRunner, RunnerCaps, \
  8. BuildConfiguration
  9. DEFAULT_PYOCD_GDB_PORT = 3333
  10. DEFAULT_PYOCD_TELNET_PORT = 4444
  11. class PyOcdBinaryRunner(ZephyrBinaryRunner):
  12. '''Runner front-end for pyOCD.'''
  13. def __init__(self, cfg, target,
  14. pyocd='pyocd',
  15. flash_addr=0x0, erase=False, flash_opts=None,
  16. gdb_port=DEFAULT_PYOCD_GDB_PORT,
  17. telnet_port=DEFAULT_PYOCD_TELNET_PORT, tui=False,
  18. pyocd_config=None,
  19. board_id=None, daparg=None, frequency=None, tool_opt=None):
  20. super().__init__(cfg)
  21. default = path.join(cfg.board_dir, 'support', 'pyocd.yaml')
  22. if path.exists(default):
  23. self.pyocd_config = default
  24. else:
  25. self.pyocd_config = None
  26. self.target_args = ['-t', target]
  27. self.pyocd = pyocd
  28. self.flash_addr_args = ['-a', hex(flash_addr)] if flash_addr else []
  29. self.erase = erase
  30. self.gdb_cmd = [cfg.gdb] if cfg.gdb is not None else None
  31. self.gdb_port = gdb_port
  32. self.telnet_port = telnet_port
  33. self.tui_args = ['-tui'] if tui else []
  34. self.hex_name = cfg.hex_file
  35. self.bin_name = cfg.bin_file
  36. self.elf_name = cfg.elf_file
  37. pyocd_config_args = []
  38. if self.pyocd_config is not None:
  39. pyocd_config_args = ['--config', self.pyocd_config]
  40. self.pyocd_config_args = pyocd_config_args
  41. board_args = []
  42. if board_id is not None:
  43. board_args = ['-u', board_id]
  44. self.board_args = board_args
  45. daparg_args = []
  46. if daparg is not None:
  47. daparg_args = ['-da', daparg]
  48. self.daparg_args = daparg_args
  49. frequency_args = []
  50. if frequency is not None:
  51. frequency_args = ['-f', frequency]
  52. self.frequency_args = frequency_args
  53. tool_opt_args = []
  54. if tool_opt is not None:
  55. tool_opt_args = [tool_opt]
  56. self.tool_opt_args = tool_opt_args
  57. self.flash_extra = flash_opts if flash_opts else []
  58. @classmethod
  59. def name(cls):
  60. return 'pyocd'
  61. @classmethod
  62. def capabilities(cls):
  63. return RunnerCaps(commands={'flash', 'debug', 'debugserver', 'attach'},
  64. flash_addr=True, erase=True)
  65. @classmethod
  66. def do_add_parser(cls, parser):
  67. parser.add_argument('--target', required=True,
  68. help='target override')
  69. parser.add_argument('--daparg',
  70. help='Additional -da arguments to pyocd tool')
  71. parser.add_argument('--pyocd', default='pyocd',
  72. help='path to pyocd tool, default is pyocd')
  73. parser.add_argument('--flash-opt', default=[], action='append',
  74. help='''Additional options for pyocd flash,
  75. e.g. --flash-opt="-e=chip" to chip erase''')
  76. parser.add_argument('--frequency',
  77. help='SWD clock frequency in Hz')
  78. parser.add_argument('--gdb-port', default=DEFAULT_PYOCD_GDB_PORT,
  79. help='pyocd gdb port, defaults to {}'.format(
  80. DEFAULT_PYOCD_GDB_PORT))
  81. parser.add_argument('--telnet-port', default=DEFAULT_PYOCD_TELNET_PORT,
  82. help='pyocd telnet port, defaults to {}'.format(
  83. DEFAULT_PYOCD_TELNET_PORT))
  84. parser.add_argument('--tui', default=False, action='store_true',
  85. help='if given, GDB uses -tui')
  86. parser.add_argument('--board-id',
  87. help='ID of board to flash, default is to prompt')
  88. parser.add_argument('--tool-opt',
  89. help='''Additional options for pyocd Commander,
  90. e.g. \'--script=user.py\' ''')
  91. @classmethod
  92. def do_create(cls, cfg, args):
  93. build_conf = BuildConfiguration(cfg.build_dir)
  94. flash_addr = cls.get_flash_address(args, build_conf)
  95. ret = PyOcdBinaryRunner(
  96. cfg, args.target,
  97. pyocd=args.pyocd,
  98. flash_addr=flash_addr, erase=args.erase, flash_opts=args.flash_opt,
  99. gdb_port=args.gdb_port, telnet_port=args.telnet_port, tui=args.tui,
  100. board_id=args.board_id, daparg=args.daparg,
  101. frequency=args.frequency,
  102. tool_opt=args.tool_opt)
  103. daparg = os.environ.get('PYOCD_DAPARG')
  104. if not ret.daparg_args and daparg:
  105. ret.logger.warning('PYOCD_DAPARG is deprecated; use --daparg')
  106. ret.logger.debug('--daparg={} via PYOCD_DAPARG'.format(daparg))
  107. ret.daparg_args = ['-da', daparg]
  108. return ret
  109. def port_args(self):
  110. return ['-p', str(self.gdb_port), '-T', str(self.telnet_port)]
  111. def do_run(self, command, **kwargs):
  112. self.require(self.pyocd)
  113. if command == 'flash':
  114. self.flash(**kwargs)
  115. else:
  116. self.debug_debugserver(command, **kwargs)
  117. def flash(self, **kwargs):
  118. if self.hex_name is not None and os.path.isfile(self.hex_name):
  119. fname = self.hex_name
  120. elif self.bin_name is not None and os.path.isfile(self.bin_name):
  121. self.logger.warning(
  122. 'hex file ({}) does not exist; falling back on .bin ({}). '.
  123. format(self.hex_name, self.bin_name) +
  124. 'Consider enabling CONFIG_BUILD_OUTPUT_HEX.')
  125. fname = self.bin_name
  126. else:
  127. raise ValueError(
  128. 'Cannot flash; no hex ({}) or bin ({}) files found. '.format(
  129. self.hex_name, self.bin_name))
  130. erase_method = 'chip' if self.erase else 'sector'
  131. cmd = ([self.pyocd] +
  132. ['flash'] +
  133. self.pyocd_config_args +
  134. ['-e', erase_method] +
  135. self.flash_addr_args +
  136. self.daparg_args +
  137. self.target_args +
  138. self.board_args +
  139. self.frequency_args +
  140. self.tool_opt_args +
  141. self.flash_extra +
  142. [fname])
  143. self.logger.info('Flashing file: {}'.format(fname))
  144. self.check_call(cmd)
  145. def log_gdbserver_message(self):
  146. self.logger.info('pyOCD GDB server running on port {}'.
  147. format(self.gdb_port))
  148. def debug_debugserver(self, command, **kwargs):
  149. server_cmd = ([self.pyocd] +
  150. ['gdbserver'] +
  151. self.daparg_args +
  152. self.port_args() +
  153. self.target_args +
  154. self.board_args +
  155. self.frequency_args +
  156. self.tool_opt_args)
  157. if command == 'debugserver':
  158. self.log_gdbserver_message()
  159. self.check_call(server_cmd)
  160. else:
  161. if self.gdb_cmd is None:
  162. raise ValueError('Cannot debug; gdb is missing')
  163. if self.elf_name is None:
  164. raise ValueError('Cannot debug; elf is missing')
  165. client_cmd = (self.gdb_cmd +
  166. self.tui_args +
  167. [self.elf_name] +
  168. ['-ex', 'target remote :{}'.format(self.gdb_port)])
  169. if command == 'debug':
  170. client_cmd += ['-ex', 'monitor halt',
  171. '-ex', 'monitor reset',
  172. '-ex', 'load']
  173. self.require(client_cmd[0])
  174. self.log_gdbserver_message()
  175. self.run_server_and_client(server_cmd, client_cmd)