2019-04-19 19:56:16 +03:00
|
|
|
# Copyright (c) 2014-present PlatformIO <contact@platformio.org>
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
|
|
|
|
2021-03-17 17:42:11 +02:00
|
|
|
# pylint: disable=too-many-arguments, too-many-locals
|
|
|
|
# pylint: disable=too-many-branches, too-many-statements
|
2019-04-19 19:56:16 +03:00
|
|
|
|
2021-03-17 17:42:11 +02:00
|
|
|
import asyncio
|
2019-04-19 19:56:16 +03:00
|
|
|
import os
|
|
|
|
|
|
|
|
import click
|
|
|
|
|
2020-08-22 17:48:49 +03:00
|
|
|
from platformio import app, exception, fs, proc
|
2020-10-26 18:24:46 +02:00
|
|
|
from platformio.commands.platform import platform_install as cmd_platform_install
|
2021-03-17 17:42:11 +02:00
|
|
|
from platformio.compat import WINDOWS
|
|
|
|
from platformio.debug import helpers
|
|
|
|
from platformio.debug.exception import DebugInvalidOptionsError
|
|
|
|
from platformio.debug.process.client import DebugClientProcess
|
2020-10-26 18:24:46 +02:00
|
|
|
from platformio.platform.exception import UnknownPlatform
|
|
|
|
from platformio.platform.factory import PlatformFactory
|
2019-05-10 15:45:52 +03:00
|
|
|
from platformio.project.config import ProjectConfig
|
2019-11-28 16:15:54 +02:00
|
|
|
from platformio.project.exception import ProjectEnvsNotAvailableError
|
2019-09-23 23:13:48 +03:00
|
|
|
from platformio.project.helpers import is_platformio_project, load_project_ide_data
|
|
|
|
|
|
|
|
|
|
|
|
@click.command(
|
|
|
|
"debug",
|
|
|
|
context_settings=dict(ignore_unknown_options=True),
|
2020-08-28 14:08:26 +03:00
|
|
|
short_help="Unified debugger",
|
2019-09-23 23:13:48 +03:00
|
|
|
)
|
|
|
|
@click.option(
|
|
|
|
"-d",
|
|
|
|
"--project-dir",
|
|
|
|
default=os.getcwd,
|
|
|
|
type=click.Path(
|
|
|
|
exists=True, file_okay=False, dir_okay=True, writable=True, resolve_path=True
|
|
|
|
),
|
|
|
|
)
|
|
|
|
@click.option(
|
|
|
|
"-c",
|
|
|
|
"--project-conf",
|
|
|
|
type=click.Path(
|
|
|
|
exists=True, file_okay=True, dir_okay=False, readable=True, resolve_path=True
|
|
|
|
),
|
|
|
|
)
|
2019-04-19 19:56:16 +03:00
|
|
|
@click.option("--environment", "-e", metavar="<environment>")
|
|
|
|
@click.option("--verbose", "-v", is_flag=True)
|
|
|
|
@click.option("--interface", type=click.Choice(["gdb"]))
|
|
|
|
@click.argument("__unprocessed", nargs=-1, type=click.UNPROCESSED)
|
|
|
|
@click.pass_context
|
2019-09-23 23:13:48 +03:00
|
|
|
def cli(ctx, project_dir, project_conf, environment, verbose, interface, __unprocessed):
|
2019-09-27 14:13:53 +03:00
|
|
|
app.set_session_var("custom_project_conf", project_conf)
|
|
|
|
|
2019-06-27 14:57:05 +03:00
|
|
|
# use env variables from Eclipse or CLion
|
|
|
|
for sysenv in ("CWD", "PWD", "PLATFORMIO_PROJECT_DIR"):
|
|
|
|
if is_platformio_project(project_dir):
|
|
|
|
break
|
|
|
|
if os.getenv(sysenv):
|
|
|
|
project_dir = os.getenv(sysenv)
|
2019-04-19 19:56:16 +03:00
|
|
|
|
2019-08-12 19:44:37 +03:00
|
|
|
with fs.cd(project_dir):
|
2019-09-27 14:13:53 +03:00
|
|
|
config = ProjectConfig.get_instance(project_conf)
|
2019-05-10 15:45:52 +03:00
|
|
|
config.validate(envs=[environment] if environment else None)
|
|
|
|
|
|
|
|
env_name = environment or helpers.get_default_debug_env(config)
|
|
|
|
env_options = config.items(env=env_name, as_dict=True)
|
2019-04-19 19:56:16 +03:00
|
|
|
if not set(env_options.keys()) >= set(["platform", "board"]):
|
2019-11-28 16:15:54 +02:00
|
|
|
raise ProjectEnvsNotAvailableError()
|
2020-10-26 18:24:46 +02:00
|
|
|
|
|
|
|
try:
|
|
|
|
platform = PlatformFactory.new(env_options["platform"])
|
|
|
|
except UnknownPlatform:
|
|
|
|
ctx.invoke(
|
|
|
|
cmd_platform_install,
|
|
|
|
platforms=[env_options["platform"]],
|
|
|
|
skip_default_package=True,
|
|
|
|
)
|
|
|
|
platform = PlatformFactory.new(env_options["platform"])
|
|
|
|
|
|
|
|
debug_options = helpers.configure_initial_debug_options(platform, env_options)
|
2019-04-19 19:56:16 +03:00
|
|
|
assert debug_options
|
|
|
|
|
|
|
|
if not interface:
|
2019-09-23 23:13:48 +03:00
|
|
|
return helpers.predebug_project(ctx, project_dir, env_name, False, verbose)
|
2019-04-19 19:56:16 +03:00
|
|
|
|
2020-10-26 18:24:46 +02:00
|
|
|
ide_data = load_project_ide_data(project_dir, env_name)
|
|
|
|
if not ide_data:
|
|
|
|
raise DebugInvalidOptionsError("Could not load a build configuration")
|
2019-04-19 19:56:16 +03:00
|
|
|
|
|
|
|
if "--version" in __unprocessed:
|
2020-10-26 18:24:46 +02:00
|
|
|
result = proc.exec_command([ide_data["gdb_path"], "--version"])
|
2019-09-23 23:13:48 +03:00
|
|
|
if result["returncode"] == 0:
|
|
|
|
return click.echo(result["out"])
|
|
|
|
raise exception.PlatformioException("\n".join([result["out"], result["err"]]))
|
2019-04-19 19:56:16 +03:00
|
|
|
|
2019-06-11 20:30:06 +03:00
|
|
|
try:
|
2019-08-12 19:44:37 +03:00
|
|
|
fs.ensure_udev_rules()
|
2019-06-11 20:30:06 +03:00
|
|
|
except exception.InvalidUdevRules as e:
|
2019-11-06 22:30:58 +02:00
|
|
|
click.echo(
|
|
|
|
helpers.escape_gdbmi_stream("~", str(e) + "\n")
|
|
|
|
if helpers.is_gdbmi_mode()
|
|
|
|
else str(e) + "\n",
|
|
|
|
nl=False,
|
|
|
|
)
|
2019-06-11 20:30:06 +03:00
|
|
|
|
2020-10-26 18:24:46 +02:00
|
|
|
try:
|
|
|
|
debug_options = platform.configure_debug_options(debug_options, ide_data)
|
|
|
|
except NotImplementedError:
|
|
|
|
# legacy for ESP32 dev-platform <=2.0.0
|
|
|
|
debug_options["load_cmds"] = helpers.configure_esp32_load_cmds(
|
|
|
|
debug_options, ide_data
|
|
|
|
)
|
2019-04-19 19:56:16 +03:00
|
|
|
|
|
|
|
rebuild_prog = False
|
2019-09-23 23:13:48 +03:00
|
|
|
preload = debug_options["load_cmds"] == ["preload"]
|
|
|
|
load_mode = debug_options["load_mode"]
|
2019-04-19 19:56:16 +03:00
|
|
|
if load_mode == "always":
|
2020-10-26 18:24:46 +02:00
|
|
|
rebuild_prog = preload or not helpers.has_debug_symbols(ide_data["prog_path"])
|
2019-04-19 19:56:16 +03:00
|
|
|
elif load_mode == "modified":
|
2019-09-23 23:13:48 +03:00
|
|
|
rebuild_prog = helpers.is_prog_obsolete(
|
2020-10-26 18:24:46 +02:00
|
|
|
ide_data["prog_path"]
|
|
|
|
) or not helpers.has_debug_symbols(ide_data["prog_path"])
|
2019-04-19 19:56:16 +03:00
|
|
|
else:
|
2021-03-17 17:42:11 +02:00
|
|
|
rebuild_prog = not os.path.isfile(ide_data["prog_path"])
|
2019-04-19 19:56:16 +03:00
|
|
|
|
|
|
|
if preload or (not rebuild_prog and load_mode != "always"):
|
|
|
|
# don't load firmware through debug server
|
2019-09-23 23:13:48 +03:00
|
|
|
debug_options["load_cmds"] = []
|
2019-04-19 19:56:16 +03:00
|
|
|
|
|
|
|
if rebuild_prog:
|
2019-11-06 22:30:58 +02:00
|
|
|
if helpers.is_gdbmi_mode():
|
|
|
|
click.echo(
|
|
|
|
helpers.escape_gdbmi_stream(
|
|
|
|
"~", "Preparing firmware for debugging...\n"
|
|
|
|
),
|
|
|
|
nl=False,
|
|
|
|
)
|
|
|
|
stream = helpers.GDBMIConsoleStream()
|
2020-08-22 17:48:49 +03:00
|
|
|
with proc.capture_std_streams(stream):
|
2019-09-23 23:13:48 +03:00
|
|
|
helpers.predebug_project(ctx, project_dir, env_name, preload, verbose)
|
2019-11-06 22:30:58 +02:00
|
|
|
stream.close()
|
2019-04-19 19:56:16 +03:00
|
|
|
else:
|
|
|
|
click.echo("Preparing firmware for debugging...")
|
2019-09-23 23:13:48 +03:00
|
|
|
helpers.predebug_project(ctx, project_dir, env_name, preload, verbose)
|
2019-04-19 19:56:16 +03:00
|
|
|
|
|
|
|
# save SHA sum of newly created prog
|
|
|
|
if load_mode == "modified":
|
2020-10-26 18:24:46 +02:00
|
|
|
helpers.is_prog_obsolete(ide_data["prog_path"])
|
2019-04-19 19:56:16 +03:00
|
|
|
|
2021-03-17 17:42:11 +02:00
|
|
|
if not os.path.isfile(ide_data["prog_path"]):
|
2019-11-28 16:15:54 +02:00
|
|
|
raise DebugInvalidOptionsError("Program/firmware is missed")
|
2019-04-19 19:56:16 +03:00
|
|
|
|
2021-03-17 17:42:11 +02:00
|
|
|
loop = asyncio.ProactorEventLoop() if WINDOWS else asyncio.get_event_loop()
|
|
|
|
asyncio.set_event_loop(loop)
|
|
|
|
client = DebugClientProcess(project_dir, __unprocessed, debug_options, env_options)
|
|
|
|
coro = client.run(ide_data["gdb_path"], ide_data["prog_path"])
|
|
|
|
loop.run_until_complete(coro)
|
|
|
|
if WINDOWS:
|
|
|
|
# an issue with asyncio executor and STIDIN, it cannot be closed gracefully
|
|
|
|
os._exit(0) # pylint: disable=protected-access
|
|
|
|
loop.close()
|
2019-04-19 19:56:16 +03:00
|
|
|
|
|
|
|
return True
|