2022-05-23 09:30:13 -04:00
|
|
|
# SPDX-FileCopyrightText: 2022 Espressif Systems (Shanghai) CO LTD
|
|
|
|
# SPDX-License-Identifier: Apache-2.0
|
2019-07-23 15:37:31 -04:00
|
|
|
import json
|
|
|
|
import os
|
2021-01-25 21:49:01 -05:00
|
|
|
import re
|
2019-07-23 15:37:31 -04:00
|
|
|
import shlex
|
2022-08-20 05:36:14 -04:00
|
|
|
import shutil
|
2021-01-25 21:49:01 -05:00
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
import threading
|
2019-07-23 15:37:31 -04:00
|
|
|
import time
|
2022-08-20 05:36:14 -04:00
|
|
|
from textwrap import indent
|
2019-07-23 15:37:31 -04:00
|
|
|
from threading import Thread
|
2022-06-03 08:46:56 -04:00
|
|
|
from typing import Any, Dict, List, Optional
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
from click.core import Context
|
2022-09-07 16:45:04 -04:00
|
|
|
from idf_py_actions.constants import OPENOCD_TAGET_CONFIG, OPENOCD_TAGET_CONFIG_DEFAULT
|
2019-07-23 15:37:31 -04:00
|
|
|
from idf_py_actions.errors import FatalError
|
2022-06-03 08:46:56 -04:00
|
|
|
from idf_py_actions.tools import PropertyDict, ensure_build_directory
|
2019-07-23 15:37:31 -04:00
|
|
|
|
|
|
|
PYTHON = sys.executable
|
2022-08-20 05:36:14 -04:00
|
|
|
ESP_ROM_INFO_FILE = 'roms.json'
|
|
|
|
GDBINIT_PYTHON_TEMPLATE = '''
|
|
|
|
# Add Python GDB extensions
|
|
|
|
python
|
|
|
|
import sys
|
|
|
|
sys.path = {sys_path}
|
|
|
|
import freertos_gdb
|
|
|
|
end
|
|
|
|
'''
|
|
|
|
GDBINIT_PYTHON_NOT_SUPPORTED = '''
|
|
|
|
# Python scripting is not supported in this copy of GDB.
|
|
|
|
# Please make sure that your Python distribution contains Python shared library.
|
|
|
|
'''
|
|
|
|
GDBINIT_BOOTLOADER_ADD_SYMBOLS = '''
|
|
|
|
# Load bootloader symbols
|
|
|
|
set confirm off
|
|
|
|
add-symbol-file {boot_elf}
|
|
|
|
set confirm on
|
|
|
|
'''
|
|
|
|
GDBINIT_BOOTLOADER_NOT_FOUND = '''
|
|
|
|
# Bootloader elf was not found
|
|
|
|
'''
|
|
|
|
GDBINIT_APP_ADD_SYMBOLS = '''
|
|
|
|
# Load application file
|
|
|
|
file {app_elf}
|
|
|
|
'''
|
|
|
|
GDBINIT_CONNECT = '''
|
|
|
|
# Connect to the default openocd-esp port and break on app_main()
|
|
|
|
target remote :3333
|
|
|
|
monitor reset halt
|
|
|
|
flushregs
|
|
|
|
thbreak app_main
|
|
|
|
continue
|
|
|
|
'''
|
|
|
|
GDBINIT_MAIN = '''
|
|
|
|
source {py_extensions}
|
|
|
|
source {symbols}
|
|
|
|
source {connect}
|
|
|
|
'''
|
2019-07-23 15:37:31 -04:00
|
|
|
|
|
|
|
|
2022-09-07 16:45:04 -04:00
|
|
|
def get_openocd_arguments(target: str) -> str:
|
|
|
|
default_args = OPENOCD_TAGET_CONFIG_DEFAULT.format(target=target)
|
|
|
|
return str(OPENOCD_TAGET_CONFIG.get(target, default_args))
|
|
|
|
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def action_extensions(base_actions: Dict, project_path: str) -> Dict:
|
2021-01-25 21:49:01 -05:00
|
|
|
OPENOCD_OUT_FILE = 'openocd_out.txt'
|
|
|
|
GDBGUI_OUT_FILE = 'gdbgui_out.txt'
|
2019-07-23 15:37:31 -04:00
|
|
|
# Internal dictionary of currently active processes, threads and their output files
|
2022-06-03 08:46:56 -04:00
|
|
|
processes: Dict = {'threads_to_join': [], 'openocd_issues': None}
|
2020-04-14 04:45:20 -04:00
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def _check_for_common_openocd_issues(file_name: str, print_all: bool=True) -> Any:
|
2021-01-25 21:49:01 -05:00
|
|
|
if processes['openocd_issues'] is not None:
|
|
|
|
return processes['openocd_issues']
|
2020-04-14 04:45:20 -04:00
|
|
|
try:
|
2021-01-25 21:49:01 -05:00
|
|
|
message = 'Please check JTAG connection!'
|
|
|
|
with open(file_name, 'r') as f:
|
2020-04-14 04:45:20 -04:00
|
|
|
content = f.read()
|
|
|
|
if print_all:
|
|
|
|
print(content)
|
2021-01-25 21:49:01 -05:00
|
|
|
if re.search(r'Address already in use', content):
|
|
|
|
message = ('Please check if another process uses the mentioned ports. OpenOCD already running, perhaps in the background?\n'
|
|
|
|
'Please list all processes to check if OpenOCD is already running; if so, terminate it before starting OpenOCD from idf.py')
|
2020-04-14 04:45:20 -04:00
|
|
|
finally:
|
2021-01-25 21:49:01 -05:00
|
|
|
processes['openocd_issues'] = message
|
2020-04-14 04:45:20 -04:00
|
|
|
return message
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def _check_openocd_errors(fail_if_openocd_failed: Dict, target: str, ctx: Context) -> None:
|
2019-07-23 15:37:31 -04:00
|
|
|
if fail_if_openocd_failed:
|
2021-01-25 21:49:01 -05:00
|
|
|
if 'openocd' in processes and processes['openocd'] is not None:
|
|
|
|
p = processes['openocd']
|
|
|
|
name = processes['openocd_outfile_name']
|
2019-07-23 15:37:31 -04:00
|
|
|
# watch OpenOCD (for 5x500ms) to check if it hasn't terminated or outputs an error
|
|
|
|
for _ in range(5):
|
|
|
|
if p.poll() is not None:
|
2021-01-25 21:49:01 -05:00
|
|
|
print('OpenOCD exited with {}'.format(p.poll()))
|
2019-07-23 15:37:31 -04:00
|
|
|
break
|
2021-01-25 21:49:01 -05:00
|
|
|
with open(name, 'r') as f:
|
2019-07-23 15:37:31 -04:00
|
|
|
content = f.read()
|
2021-01-25 21:49:01 -05:00
|
|
|
if re.search(r'no device found', content):
|
2019-07-23 15:37:31 -04:00
|
|
|
break
|
2021-01-25 21:49:01 -05:00
|
|
|
if re.search(r'Listening on port \d+ for gdb connections', content):
|
2019-07-23 15:37:31 -04:00
|
|
|
# expect OpenOCD has started successfully - stop watching
|
|
|
|
return
|
|
|
|
time.sleep(0.5)
|
|
|
|
else:
|
|
|
|
return
|
|
|
|
# OpenOCD exited or error message detected -> print possible output and terminate
|
2020-04-14 04:45:20 -04:00
|
|
|
raise FatalError('Action "{}" failed due to errors in OpenOCD:\n{}'.format(target, _check_for_common_openocd_issues(name)), ctx)
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def _terminate_async_target(target: str) -> None:
|
2019-07-23 15:37:31 -04:00
|
|
|
if target in processes and processes[target] is not None:
|
|
|
|
try:
|
2021-01-25 21:49:01 -05:00
|
|
|
if target + '_outfile' in processes:
|
|
|
|
processes[target + '_outfile'].close()
|
2019-07-23 15:37:31 -04:00
|
|
|
p = processes[target]
|
|
|
|
if p.poll() is None:
|
|
|
|
p.terminate()
|
|
|
|
# waiting 10x100ms for the process to terminate gracefully
|
|
|
|
for _ in range(10):
|
|
|
|
if p.poll() is not None:
|
|
|
|
break
|
|
|
|
time.sleep(0.1)
|
|
|
|
else:
|
|
|
|
p.kill()
|
2021-01-25 21:49:01 -05:00
|
|
|
if target + '_outfile_name' in processes:
|
|
|
|
if target == 'openocd':
|
|
|
|
print(_check_for_common_openocd_issues(processes[target + '_outfile_name'], print_all=False))
|
|
|
|
os.unlink(processes[target + '_outfile_name'])
|
2019-07-23 15:37:31 -04:00
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
2021-01-25 21:49:01 -05:00
|
|
|
print('Failed to close/kill {}'.format(target))
|
2019-07-23 15:37:31 -04:00
|
|
|
processes[target] = None # to indicate this has ended
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def is_gdb_with_python(gdb: str) -> bool:
|
2022-03-23 13:18:02 -04:00
|
|
|
# execute simple python command to check is it supported
|
|
|
|
return subprocess.run([gdb, '--batch-silent', '--ex', 'python import os'], stderr=subprocess.DEVNULL).returncode == 0
|
|
|
|
|
2022-08-20 05:36:14 -04:00
|
|
|
def get_normalized_path(path: str) -> str:
|
|
|
|
if os.name == 'nt':
|
|
|
|
return os.path.normpath(path).replace('\\','\\\\')
|
|
|
|
return path
|
|
|
|
|
|
|
|
def get_rom_if_condition_str(date_addr: int, date_str: str) -> str:
|
|
|
|
r = []
|
|
|
|
for i in range(0, len(date_str), 4):
|
|
|
|
value = hex(int.from_bytes(bytes(date_str[i:i + 4], 'utf-8'), 'little'))
|
|
|
|
r.append(f'(*(int*) {hex(date_addr + i)}) == {value}')
|
|
|
|
return 'if ' + ' && '.join(r)
|
|
|
|
|
|
|
|
def generate_gdbinit_rom_add_symbols(target: str) -> str:
|
|
|
|
base_ident = ' '
|
|
|
|
rom_elfs_dir = os.getenv('ESP_ROM_ELF_DIR')
|
|
|
|
if not rom_elfs_dir:
|
|
|
|
raise FatalError('ESP_ROM_ELF_DIR environment variable is not defined. Please try to run IDF "install" and "export" scripts.')
|
|
|
|
with open(os.path.join(os.path.dirname(os.path.realpath(__file__)), ESP_ROM_INFO_FILE), 'r') as f:
|
|
|
|
roms = json.load(f)
|
|
|
|
if target not in roms:
|
|
|
|
msg_body = f'Target "{target}" was not found in "{ESP_ROM_INFO_FILE}". Please check IDF integrity.'
|
|
|
|
if os.getenv('ESP_IDF_GDB_TESTING'):
|
|
|
|
raise FatalError(msg_body)
|
|
|
|
print(f'Warning: {msg_body}')
|
|
|
|
return f'# {msg_body}'
|
|
|
|
r = ['', f'# Load {target} ROM ELF symbols']
|
|
|
|
is_one_revision = len(roms[target]) == 1
|
|
|
|
if not is_one_revision:
|
|
|
|
r.append('define target hookpost-remote')
|
|
|
|
r.append('set confirm off')
|
|
|
|
# Workaround for reading ROM data on xtensa chips
|
|
|
|
# This should be deleted after the new openocd-esp release (newer than v0.11.0-esp32-20220706)
|
|
|
|
xtensa_chips = ['esp32', 'esp32s2', 'esp32s3']
|
|
|
|
if target in xtensa_chips:
|
|
|
|
r.append('monitor xtensa set_permissive 1')
|
|
|
|
# Since GDB does not have 'else if' statement than we use nested 'if..else' instead.
|
|
|
|
for i, k in enumerate(roms[target], 1):
|
|
|
|
indent_str = base_ident * i
|
|
|
|
rom_file = get_normalized_path(os.path.join(rom_elfs_dir, f'{target}_rev{k["rev"]}_rom.elf'))
|
|
|
|
build_date_addr = int(k['build_date_str_addr'], base=16)
|
|
|
|
r.append(indent(f'# if $_streq((char *) {hex(build_date_addr)}, "{k["build_date_str"]}")', indent_str))
|
|
|
|
r.append(indent(get_rom_if_condition_str(build_date_addr, k['build_date_str']), indent_str))
|
|
|
|
r.append(indent(f'add-symbol-file {rom_file}', indent_str + base_ident))
|
|
|
|
r.append(indent('else', indent_str))
|
|
|
|
if i == len(roms[target]):
|
|
|
|
# In case no one known ROM ELF fits - print error and exit with error code 1
|
|
|
|
indent_str += base_ident
|
|
|
|
msg_body = f'unknown {target} ROM revision.'
|
|
|
|
if os.getenv('ESP_IDF_GDB_TESTING'):
|
|
|
|
r.append(indent(f'echo Error: {msg_body}\\n', indent_str))
|
|
|
|
r.append(indent('quit 1', indent_str))
|
|
|
|
else:
|
|
|
|
r.append(indent(f'echo Warning: {msg_body}\\n', indent_str))
|
|
|
|
# Close 'else' operators
|
|
|
|
for i in range(len(roms[target]), 0, -1):
|
|
|
|
r.append(indent('end', base_ident * i))
|
|
|
|
if target in xtensa_chips:
|
|
|
|
r.append('monitor xtensa set_permissive 0')
|
|
|
|
r.append('set confirm on')
|
|
|
|
if not is_one_revision:
|
|
|
|
r.append('end')
|
|
|
|
r.append('')
|
|
|
|
return os.linesep.join(r)
|
|
|
|
raise FatalError(f'{ESP_ROM_INFO_FILE} file not found. Please check IDF integrity.')
|
|
|
|
|
|
|
|
def generate_gdbinit_files(gdb: str, gdbinit: Optional[str], project_desc: Dict[str, Any]) -> None:
|
|
|
|
app_elf = get_normalized_path(os.path.join(project_desc['build_dir'], project_desc['app_elf']))
|
|
|
|
if not os.path.exists(app_elf):
|
|
|
|
raise FatalError('ELF file not found. You need to build & flash the project before running debug targets')
|
|
|
|
|
|
|
|
# Recreate empty 'gdbinit' directory
|
|
|
|
gdbinit_dir = os.path.join(project_desc['build_dir'], 'gdbinit')
|
|
|
|
if os.path.isfile(gdbinit_dir):
|
|
|
|
os.remove(gdbinit_dir)
|
|
|
|
elif os.path.isdir(gdbinit_dir):
|
|
|
|
shutil.rmtree(gdbinit_dir)
|
|
|
|
os.mkdir(gdbinit_dir)
|
|
|
|
|
|
|
|
# Prepare gdbinit for Python GDB extensions import
|
|
|
|
py_extensions = os.path.join(gdbinit_dir, 'py_extensions')
|
|
|
|
with open(py_extensions, 'w') as f:
|
2022-03-23 13:18:02 -04:00
|
|
|
if is_gdb_with_python(gdb):
|
2022-08-20 05:36:14 -04:00
|
|
|
f.write(GDBINIT_PYTHON_TEMPLATE.format(sys_path=sys.path))
|
|
|
|
else:
|
|
|
|
f.write(GDBINIT_PYTHON_NOT_SUPPORTED)
|
|
|
|
|
|
|
|
# Prepare gdbinit for related ELFs symbols load
|
|
|
|
symbols = os.path.join(gdbinit_dir, 'symbols')
|
|
|
|
with open(symbols, 'w') as f:
|
|
|
|
boot_elf = get_normalized_path(project_desc['bootloader_elf']) if 'bootloader_elf' in project_desc else None
|
|
|
|
if boot_elf and os.path.exists(boot_elf):
|
|
|
|
f.write(GDBINIT_BOOTLOADER_ADD_SYMBOLS.format(boot_elf=boot_elf))
|
|
|
|
else:
|
|
|
|
f.write(GDBINIT_BOOTLOADER_NOT_FOUND)
|
|
|
|
f.write(generate_gdbinit_rom_add_symbols(project_desc['target']))
|
|
|
|
f.write(GDBINIT_APP_ADD_SYMBOLS.format(app_elf=app_elf))
|
|
|
|
|
|
|
|
# Generate the gdbinit for target connect if no custom gdbinit is present
|
|
|
|
if not gdbinit:
|
|
|
|
gdbinit = os.path.join(gdbinit_dir, 'connect')
|
|
|
|
with open(gdbinit, 'w') as f:
|
|
|
|
f.write(GDBINIT_CONNECT)
|
|
|
|
|
|
|
|
with open(os.path.join(gdbinit_dir, 'gdbinit'), 'w') as f:
|
|
|
|
f.write(GDBINIT_MAIN.format(py_extensions=py_extensions, symbols=symbols, connect=gdbinit))
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def debug_cleanup() -> None:
|
2021-01-25 21:49:01 -05:00
|
|
|
print('cleaning up debug targets')
|
|
|
|
for t in processes['threads_to_join']:
|
2019-07-23 15:37:31 -04:00
|
|
|
if threading.currentThread() != t:
|
|
|
|
t.join()
|
2021-01-25 21:49:01 -05:00
|
|
|
_terminate_async_target('openocd')
|
|
|
|
_terminate_async_target('gdbgui')
|
|
|
|
_terminate_async_target('gdb')
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def post_debug(action: str, ctx: Context, args: PropertyDict, **kwargs: str) -> None:
|
2019-07-23 15:37:31 -04:00
|
|
|
""" Deal with asynchronous targets, such as openocd running in background """
|
2021-01-25 21:49:01 -05:00
|
|
|
if kwargs['block'] == 1:
|
|
|
|
for target in ['openocd', 'gdbgui']:
|
2019-07-23 15:37:31 -04:00
|
|
|
if target in processes and processes[target] is not None:
|
|
|
|
break
|
|
|
|
else:
|
|
|
|
return
|
|
|
|
try:
|
|
|
|
p = processes[target]
|
2021-01-25 21:49:01 -05:00
|
|
|
name = processes[target + '_outfile_name']
|
2019-07-23 15:37:31 -04:00
|
|
|
pos = 0
|
|
|
|
while True:
|
2021-01-25 21:49:01 -05:00
|
|
|
with open(name, 'r') as f:
|
2019-07-23 15:37:31 -04:00
|
|
|
f.seek(pos)
|
|
|
|
for line in f:
|
|
|
|
print(line.rstrip())
|
|
|
|
pos = f.tell()
|
|
|
|
if p.poll() is not None:
|
|
|
|
print('"{}" exited with {}'.format(target, p.poll()))
|
|
|
|
break
|
|
|
|
time.sleep(0.5)
|
|
|
|
except KeyboardInterrupt:
|
2021-01-25 21:49:01 -05:00
|
|
|
print('Terminated -> exiting debug utility targets')
|
|
|
|
_terminate_async_target('openocd')
|
|
|
|
_terminate_async_target('gdbgui')
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def get_project_desc(args: PropertyDict, ctx: Context) -> Any:
|
2021-01-25 21:49:01 -05:00
|
|
|
desc_path = os.path.join(args.build_dir, 'project_description.json')
|
2019-07-23 15:37:31 -04:00
|
|
|
if not os.path.exists(desc_path):
|
|
|
|
ensure_build_directory(args, ctx.info_name)
|
2021-01-25 21:49:01 -05:00
|
|
|
with open(desc_path, 'r') as f:
|
2019-07-23 15:37:31 -04:00
|
|
|
project_desc = json.load(f)
|
|
|
|
return project_desc
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def openocd(action: str, ctx: Context, args: PropertyDict, openocd_scripts: Optional[str], openocd_commands: str) -> None:
|
2019-07-23 15:37:31 -04:00
|
|
|
"""
|
|
|
|
Execute openocd as external tool
|
|
|
|
"""
|
2021-01-25 21:49:01 -05:00
|
|
|
if os.getenv('OPENOCD_SCRIPTS') is None:
|
|
|
|
raise FatalError('OPENOCD_SCRIPTS not found in the environment: Please run export.sh/export.bat', ctx)
|
|
|
|
openocd_arguments = os.getenv('OPENOCD_COMMANDS') if openocd_commands is None else openocd_commands
|
2019-07-23 15:37:31 -04:00
|
|
|
project_desc = get_project_desc(args, ctx)
|
|
|
|
if openocd_arguments is None:
|
|
|
|
# use default value if commands not defined in the environment nor command line
|
2021-01-25 21:49:01 -05:00
|
|
|
target = project_desc['target']
|
2022-09-07 16:45:04 -04:00
|
|
|
openocd_arguments = get_openocd_arguments(target)
|
2019-07-23 15:37:31 -04:00
|
|
|
print('Note: OpenOCD cfg not found (via env variable OPENOCD_COMMANDS nor as a --openocd-commands argument)\n'
|
|
|
|
'OpenOCD arguments default to: "{}"'.format(openocd_arguments))
|
|
|
|
# script directory is taken from the environment by OpenOCD, update only if command line arguments to override
|
|
|
|
if openocd_scripts is not None:
|
2021-01-25 21:49:01 -05:00
|
|
|
openocd_arguments += ' -s {}'.format(openocd_scripts)
|
|
|
|
local_dir = project_desc['build_dir']
|
|
|
|
args = ['openocd'] + shlex.split(openocd_arguments)
|
2019-07-23 15:37:31 -04:00
|
|
|
openocd_out_name = os.path.join(local_dir, OPENOCD_OUT_FILE)
|
2021-01-25 21:49:01 -05:00
|
|
|
openocd_out = open(openocd_out_name, 'a+')
|
2019-07-23 15:37:31 -04:00
|
|
|
try:
|
|
|
|
process = subprocess.Popen(args, stdout=openocd_out, stderr=subprocess.STDOUT, bufsize=1)
|
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
2021-01-25 21:49:01 -05:00
|
|
|
raise FatalError('Error starting openocd. Please make sure it is installed and is present in executable paths', ctx)
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2021-01-25 21:49:01 -05:00
|
|
|
processes['openocd'] = process
|
|
|
|
processes['openocd_outfile'] = openocd_out
|
|
|
|
processes['openocd_outfile_name'] = openocd_out_name
|
|
|
|
print('OpenOCD started as a background task {}'.format(process.pid))
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2022-08-20 05:36:14 -04:00
|
|
|
def get_gdb_args(project_desc: Dict[str, Any]) -> List:
|
|
|
|
gdbinit = os.path.join(project_desc['build_dir'], 'gdbinit', 'gdbinit')
|
2021-10-25 22:41:13 -04:00
|
|
|
args = ['-x={}'.format(gdbinit)]
|
|
|
|
debug_prefix_gdbinit = project_desc.get('debug_prefix_map_gdbinit')
|
|
|
|
if debug_prefix_gdbinit:
|
|
|
|
args.append('-ix={}'.format(debug_prefix_gdbinit))
|
|
|
|
return args
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def gdbui(action: str, ctx: Context, args: PropertyDict, gdbgui_port: Optional[str], gdbinit: Optional[str], require_openocd: bool) -> None:
|
2019-07-23 15:37:31 -04:00
|
|
|
"""
|
|
|
|
Asynchronous GDB-UI target
|
|
|
|
"""
|
|
|
|
project_desc = get_project_desc(args, ctx)
|
2021-01-25 21:49:01 -05:00
|
|
|
local_dir = project_desc['build_dir']
|
|
|
|
gdb = project_desc['monitor_toolprefix'] + 'gdb'
|
2022-08-20 05:36:14 -04:00
|
|
|
generate_gdbinit_files(gdb, gdbinit, project_desc)
|
2021-10-25 22:41:13 -04:00
|
|
|
|
|
|
|
# this is a workaround for gdbgui
|
|
|
|
# gdbgui is using shlex.split for the --gdb-args option. When the input is:
|
|
|
|
# - '"-x=foo -x=bar"', would return ['foo bar']
|
|
|
|
# - '-x=foo', would return ['-x', 'foo'] and mess up the former option '--gdb-args'
|
|
|
|
# so for one item, use extra double quotes. for more items, use no extra double quotes.
|
2022-08-20 05:36:14 -04:00
|
|
|
gdb_args_list = get_gdb_args(project_desc)
|
2022-06-03 08:46:56 -04:00
|
|
|
gdb_args = '"{}"'.format(' '.join(gdb_args_list)) if len(gdb_args_list) == 1 else ' '.join(gdb_args_list)
|
2021-10-25 22:41:13 -04:00
|
|
|
args = ['gdbgui', '-g', gdb, '--gdb-args', gdb_args]
|
|
|
|
print(args)
|
|
|
|
|
2019-07-23 15:37:31 -04:00
|
|
|
if gdbgui_port is not None:
|
2021-01-25 21:49:01 -05:00
|
|
|
args += ['--port', gdbgui_port]
|
2019-07-23 15:37:31 -04:00
|
|
|
gdbgui_out_name = os.path.join(local_dir, GDBGUI_OUT_FILE)
|
2021-01-25 21:49:01 -05:00
|
|
|
gdbgui_out = open(gdbgui_out_name, 'a+')
|
2021-04-27 08:27:22 -04:00
|
|
|
env = os.environ.copy()
|
|
|
|
# The only known solution for https://github.com/cs01/gdbgui/issues/359 is to set the following environment
|
|
|
|
# variable. The greenlet package cannot be downgraded for compatibility with other requirements (gdbgui,
|
|
|
|
# pygdbmi).
|
|
|
|
env['PURE_PYTHON'] = '1'
|
2019-07-23 15:37:31 -04:00
|
|
|
try:
|
2021-04-27 08:27:22 -04:00
|
|
|
process = subprocess.Popen(args, stdout=gdbgui_out, stderr=subprocess.STDOUT, bufsize=1, env=env)
|
2022-11-10 11:06:44 -05:00
|
|
|
except (OSError, subprocess.CalledProcessError) as e:
|
2019-07-23 15:37:31 -04:00
|
|
|
print(e)
|
2022-11-10 11:06:44 -05:00
|
|
|
if sys.version_info[:2] >= (3, 11):
|
|
|
|
raise SystemExit('Unfortunately, gdbgui is supported only with Python 3.10 or older. '
|
|
|
|
'See: https://github.com/espressif/esp-idf/issues/10116. '
|
|
|
|
'Please use "idf.py gdb" or debug in Eclipse/Vscode instead.')
|
2021-09-16 10:48:03 -04:00
|
|
|
raise FatalError('Error starting gdbgui. Please make sure gdbgui has been installed with '
|
|
|
|
'"install.{sh,bat,ps1,fish} --enable-gdbgui" and can be started.', ctx)
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2021-01-25 21:49:01 -05:00
|
|
|
processes['gdbgui'] = process
|
|
|
|
processes['gdbgui_outfile'] = gdbgui_out
|
|
|
|
processes['gdbgui_outfile_name'] = gdbgui_out_name
|
|
|
|
print('gdbgui started as a background task {}'.format(process.pid))
|
2019-07-23 15:37:31 -04:00
|
|
|
_check_openocd_errors(fail_if_openocd_failed, action, ctx)
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def global_callback(ctx: Context, global_args: PropertyDict, tasks: List) -> None:
|
|
|
|
def move_to_front(task_name: str) -> None:
|
2019-07-23 15:37:31 -04:00
|
|
|
for index, task in enumerate(tasks):
|
|
|
|
if task.name == task_name:
|
|
|
|
tasks.insert(0, tasks.pop(index))
|
|
|
|
break
|
|
|
|
|
2021-01-25 21:49:01 -05:00
|
|
|
debug_targets = any([task.name in ('openocd', 'gdbgui') for task in tasks])
|
2019-07-23 15:37:31 -04:00
|
|
|
if debug_targets:
|
|
|
|
# Register the meta cleanup callback -> called on FatalError
|
2021-01-25 21:49:01 -05:00
|
|
|
ctx.meta['cleanup'] = debug_cleanup
|
|
|
|
move_to_front('gdbgui') # possibly 2nd
|
|
|
|
move_to_front('openocd') # always 1st
|
2019-07-23 15:37:31 -04:00
|
|
|
# followed by "monitor", "gdb" or "gdbtui" in any order
|
|
|
|
|
2021-01-25 21:49:01 -05:00
|
|
|
post_action = ctx.invoke(ctx.command.get_command(ctx, 'post_debug'))
|
|
|
|
if any([task.name in ('monitor', 'gdb', 'gdbtui') for task in tasks]):
|
|
|
|
post_action.action_args['block'] = 0
|
2019-07-23 15:37:31 -04:00
|
|
|
else:
|
2021-01-25 21:49:01 -05:00
|
|
|
post_action.action_args['block'] = 1
|
2019-07-23 15:37:31 -04:00
|
|
|
tasks.append(post_action) # always last
|
2021-01-25 21:49:01 -05:00
|
|
|
if any([task.name == 'openocd' for task in tasks]):
|
2019-07-23 15:37:31 -04:00
|
|
|
for task in tasks:
|
2021-01-25 21:49:01 -05:00
|
|
|
if task.name in ('gdb', 'gdbgui', 'gdbtui'):
|
|
|
|
task.action_args['require_openocd'] = True
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def run_gdb(gdb_args: List) -> int:
|
2019-07-23 15:37:31 -04:00
|
|
|
p = subprocess.Popen(gdb_args)
|
2021-01-25 21:49:01 -05:00
|
|
|
processes['gdb'] = p
|
2019-07-23 15:37:31 -04:00
|
|
|
return p.wait()
|
|
|
|
|
2022-06-03 08:46:56 -04:00
|
|
|
def gdbtui(action: str, ctx: Context, args: PropertyDict, gdbinit: str, require_openocd: bool) -> None:
|
2019-07-23 15:37:31 -04:00
|
|
|
"""
|
|
|
|
Synchronous GDB target with text ui mode
|
|
|
|
"""
|
2022-09-07 16:45:04 -04:00
|
|
|
gdb(action, ctx, args, False, 1, gdbinit, require_openocd)
|
2019-07-23 15:37:31 -04:00
|
|
|
|
2022-09-07 16:45:04 -04:00
|
|
|
def gdb(action: str, ctx: Context, args: PropertyDict, batch: bool, gdb_tui: Optional[int], gdbinit: Optional[str], require_openocd: bool) -> None:
|
2019-07-23 15:37:31 -04:00
|
|
|
"""
|
|
|
|
Synchronous GDB target
|
|
|
|
"""
|
|
|
|
watch_openocd = Thread(target=_check_openocd_errors, args=(fail_if_openocd_failed, action, ctx, ))
|
|
|
|
watch_openocd.start()
|
2021-01-25 21:49:01 -05:00
|
|
|
processes['threads_to_join'].append(watch_openocd)
|
2022-03-23 13:18:02 -04:00
|
|
|
project_desc = get_project_desc(args, ctx)
|
2021-01-25 21:49:01 -05:00
|
|
|
gdb = project_desc['monitor_toolprefix'] + 'gdb'
|
2022-08-20 05:36:14 -04:00
|
|
|
generate_gdbinit_files(gdb, gdbinit, project_desc)
|
|
|
|
args = [gdb, *get_gdb_args(project_desc)]
|
2019-07-23 15:37:31 -04:00
|
|
|
if gdb_tui is not None:
|
|
|
|
args += ['-tui']
|
2022-09-07 16:45:04 -04:00
|
|
|
if batch:
|
|
|
|
args += ['--batch']
|
2021-10-25 22:41:13 -04:00
|
|
|
t = Thread(target=run_gdb, args=(args,))
|
2019-07-23 15:37:31 -04:00
|
|
|
t.start()
|
|
|
|
while True:
|
|
|
|
try:
|
|
|
|
t.join()
|
|
|
|
break
|
|
|
|
except KeyboardInterrupt:
|
|
|
|
# Catching Keyboard interrupt, as this is used for breaking running program in gdb
|
|
|
|
continue
|
|
|
|
finally:
|
|
|
|
watch_openocd.join()
|
2020-09-17 09:08:54 -04:00
|
|
|
try:
|
2021-01-25 21:49:01 -05:00
|
|
|
processes['threads_to_join'].remove(watch_openocd)
|
2020-09-17 09:08:54 -04:00
|
|
|
except ValueError:
|
|
|
|
# Valid scenario: watch_openocd task won't be in the list if openocd not started from idf.py
|
|
|
|
pass
|
2019-07-23 15:37:31 -04:00
|
|
|
|
|
|
|
fail_if_openocd_failed = {
|
2021-01-25 21:49:01 -05:00
|
|
|
'names': ['--require-openocd', '--require_openocd'],
|
|
|
|
'help':
|
|
|
|
('Fail this target if openocd (this targets dependency) failed.\n'),
|
|
|
|
'is_flag': True,
|
|
|
|
'default': False,
|
2019-07-23 15:37:31 -04:00
|
|
|
}
|
2020-04-15 13:02:14 -04:00
|
|
|
gdbinit = {
|
2021-01-25 21:49:01 -05:00
|
|
|
'names': ['--gdbinit'],
|
|
|
|
'help': ('Specify the name of gdbinit file to use\n'),
|
|
|
|
'default': None,
|
2020-04-15 13:02:14 -04:00
|
|
|
}
|
2019-07-23 15:37:31 -04:00
|
|
|
debug_actions = {
|
2021-01-25 21:49:01 -05:00
|
|
|
'global_action_callbacks': [global_callback],
|
|
|
|
'actions': {
|
|
|
|
'openocd': {
|
|
|
|
'callback': openocd,
|
|
|
|
'help': 'Run openocd from current path',
|
|
|
|
'options': [
|
2019-07-23 15:37:31 -04:00
|
|
|
{
|
2021-01-25 21:49:01 -05:00
|
|
|
'names': ['--openocd-scripts', '--openocd_scripts'],
|
|
|
|
'help':
|
|
|
|
('Script directory for openocd cfg files.\n'),
|
|
|
|
'default':
|
2019-07-23 15:37:31 -04:00
|
|
|
None,
|
|
|
|
},
|
|
|
|
{
|
2021-01-25 21:49:01 -05:00
|
|
|
'names': ['--openocd-commands', '--openocd_commands'],
|
|
|
|
'help':
|
|
|
|
('Command line arguments for openocd.\n'),
|
|
|
|
'default': None,
|
2019-07-23 15:37:31 -04:00
|
|
|
}
|
|
|
|
],
|
2021-01-25 21:49:01 -05:00
|
|
|
'order_dependencies': ['all', 'flash'],
|
2019-07-23 15:37:31 -04:00
|
|
|
},
|
2021-01-25 21:49:01 -05:00
|
|
|
'gdb': {
|
|
|
|
'callback': gdb,
|
|
|
|
'help': 'Run the GDB.',
|
|
|
|
'options': [
|
2022-09-07 16:45:04 -04:00
|
|
|
{
|
|
|
|
'names': ['--batch'],
|
|
|
|
'help': ('exit after processing gdbinit.\n'),
|
|
|
|
'hidden': True,
|
|
|
|
'is_flag': True,
|
|
|
|
'default': False,
|
|
|
|
},
|
2019-07-23 15:37:31 -04:00
|
|
|
{
|
2021-01-25 21:49:01 -05:00
|
|
|
'names': ['--gdb-tui', '--gdb_tui'],
|
2022-09-07 16:45:04 -04:00
|
|
|
'help': ('run gdb in TUI mode\n'),
|
|
|
|
'default': None,
|
2020-04-15 13:02:14 -04:00
|
|
|
}, gdbinit, fail_if_openocd_failed
|
2019-07-23 15:37:31 -04:00
|
|
|
],
|
2021-01-25 21:49:01 -05:00
|
|
|
'order_dependencies': ['all', 'flash'],
|
2019-07-23 15:37:31 -04:00
|
|
|
},
|
2021-01-25 21:49:01 -05:00
|
|
|
'gdbgui': {
|
|
|
|
'callback': gdbui,
|
|
|
|
'help': 'GDB UI in default browser.',
|
|
|
|
'options': [
|
2019-07-23 15:37:31 -04:00
|
|
|
{
|
2021-01-25 21:49:01 -05:00
|
|
|
'names': ['--gdbgui-port', '--gdbgui_port'],
|
|
|
|
'help':
|
|
|
|
('The port on which gdbgui will be hosted. Default: 5000\n'),
|
|
|
|
'default':
|
2019-07-23 15:37:31 -04:00
|
|
|
None,
|
2020-04-15 13:02:14 -04:00
|
|
|
}, gdbinit, fail_if_openocd_failed
|
2019-07-23 15:37:31 -04:00
|
|
|
],
|
2021-01-25 21:49:01 -05:00
|
|
|
'order_dependencies': ['all', 'flash'],
|
2019-07-23 15:37:31 -04:00
|
|
|
},
|
2021-01-25 21:49:01 -05:00
|
|
|
'gdbtui': {
|
|
|
|
'callback': gdbtui,
|
|
|
|
'help': 'GDB TUI mode.',
|
|
|
|
'options': [gdbinit, fail_if_openocd_failed],
|
|
|
|
'order_dependencies': ['all', 'flash'],
|
2019-07-23 15:37:31 -04:00
|
|
|
},
|
2021-08-11 09:24:17 -04:00
|
|
|
'post-debug': {
|
|
|
|
'callback': post_debug,
|
|
|
|
'help': 'Utility target to read the output of async debug action and stop them.',
|
|
|
|
'options': [
|
|
|
|
{
|
|
|
|
'names': ['--block', '--block'],
|
|
|
|
'help':
|
|
|
|
('Set to 1 for blocking the console on the outputs of async debug actions\n'),
|
|
|
|
'default': 0,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
'order_dependencies': [],
|
|
|
|
},
|
2021-01-25 21:49:01 -05:00
|
|
|
'post_debug': {
|
|
|
|
'callback': post_debug,
|
2021-08-11 09:24:17 -04:00
|
|
|
'deprecated': {
|
2022-03-18 07:09:14 -04:00
|
|
|
'since': 'v4.4',
|
2021-08-11 09:24:17 -04:00
|
|
|
'removed': 'v5.0',
|
2022-03-18 07:09:14 -04:00
|
|
|
'exit_with_error': True,
|
|
|
|
'message': 'Have you wanted to run "post-debug" instead?',
|
2021-08-11 09:24:17 -04:00
|
|
|
},
|
|
|
|
'hidden': True,
|
2021-01-25 21:49:01 -05:00
|
|
|
'help': 'Utility target to read the output of async debug action and stop them.',
|
|
|
|
'options': [
|
2019-07-23 15:37:31 -04:00
|
|
|
{
|
2021-01-25 21:49:01 -05:00
|
|
|
'names': ['--block', '--block'],
|
|
|
|
'help':
|
|
|
|
('Set to 1 for blocking the console on the outputs of async debug actions\n'),
|
|
|
|
'default': 0,
|
2019-07-23 15:37:31 -04:00
|
|
|
},
|
|
|
|
],
|
2021-01-25 21:49:01 -05:00
|
|
|
'order_dependencies': [],
|
2019-07-23 15:37:31 -04:00
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
return debug_actions
|