
This adds a new 'CapabilitiesEventBody' type for having a well structured type for the event and updates the 'restart' request to dynamically set their capabilities.
144 lines
5.2 KiB
C++
144 lines
5.2 KiB
C++
//===-- RestartRequestHandler.cpp -----------------------------------------===//
|
|
//
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "DAP.h"
|
|
#include "EventHelper.h"
|
|
#include "JSONUtils.h"
|
|
#include "LLDBUtils.h"
|
|
#include "Protocol/ProtocolRequests.h"
|
|
#include "RequestHandler.h"
|
|
#include "llvm/Support/JSON.h"
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
namespace lldb_dap {
|
|
|
|
// "RestartRequest": {
|
|
// "allOf": [ { "$ref": "#/definitions/Request" }, {
|
|
// "type": "object",
|
|
// "description": "Restarts a debug session. Clients should only call this
|
|
// request if the corresponding capability `supportsRestartRequest` is
|
|
// true.\nIf the capability is missing or has the value false, a typical
|
|
// client emulates `restart` by terminating the debug adapter first and then
|
|
// launching it anew.",
|
|
// "properties": {
|
|
// "command": {
|
|
// "type": "string",
|
|
// "enum": [ "restart" ]
|
|
// },
|
|
// "arguments": {
|
|
// "$ref": "#/definitions/RestartArguments"
|
|
// }
|
|
// },
|
|
// "required": [ "command" ]
|
|
// }]
|
|
// },
|
|
// "RestartArguments": {
|
|
// "type": "object",
|
|
// "description": "Arguments for `restart` request.",
|
|
// "properties": {
|
|
// "arguments": {
|
|
// "oneOf": [
|
|
// { "$ref": "#/definitions/LaunchRequestArguments" },
|
|
// { "$ref": "#/definitions/AttachRequestArguments" }
|
|
// ],
|
|
// "description": "The latest version of the `launch` or `attach`
|
|
// configuration."
|
|
// }
|
|
// }
|
|
// },
|
|
// "RestartResponse": {
|
|
// "allOf": [ { "$ref": "#/definitions/Response" }, {
|
|
// "type": "object",
|
|
// "description": "Response to `restart` request. This is just an
|
|
// acknowledgement, so no body field is required."
|
|
// }]
|
|
// },
|
|
void RestartRequestHandler::operator()(
|
|
const llvm::json::Object &request) const {
|
|
llvm::json::Object response;
|
|
FillResponse(request, response);
|
|
if (!dap.target.GetProcess().IsValid()) {
|
|
response["success"] = llvm::json::Value(false);
|
|
EmplaceSafeString(response, "message",
|
|
"Restart request received but no process was launched.");
|
|
dap.SendJSON(llvm::json::Value(std::move(response)));
|
|
return;
|
|
}
|
|
|
|
const llvm::json::Object *arguments = request.getObject("arguments");
|
|
if (arguments) {
|
|
// The optional `arguments` field in RestartRequest can contain an updated
|
|
// version of the launch arguments. If there's one, use it.
|
|
if (const llvm::json::Value *restart_arguments =
|
|
arguments->get("arguments")) {
|
|
protocol::LaunchRequestArguments updated_arguments;
|
|
llvm::json::Path::Root root;
|
|
if (!fromJSON(*restart_arguments, updated_arguments, root)) {
|
|
response["success"] = llvm::json::Value(false);
|
|
EmplaceSafeString(
|
|
response, "message",
|
|
llvm::formatv("Failed to parse updated launch arguments: {0}",
|
|
llvm::toString(root.getError()))
|
|
.str());
|
|
dap.SendJSON(llvm::json::Value(std::move(response)));
|
|
return;
|
|
}
|
|
dap.last_launch_request = updated_arguments;
|
|
// Update DAP configuration based on the latest copy of the launch
|
|
// arguments.
|
|
dap.SetConfiguration(updated_arguments.configuration, false);
|
|
dap.ConfigureSourceMaps();
|
|
}
|
|
}
|
|
|
|
// Keep track of the old PID so when we get a "process exited" event from the
|
|
// killed process we can detect it and not shut down the whole session.
|
|
lldb::SBProcess process = dap.target.GetProcess();
|
|
dap.restarting_process_id = process.GetProcessID();
|
|
|
|
// Stop the current process if necessary. The logic here is similar to
|
|
// CommandObjectProcessLaunchOrAttach::StopProcessIfNecessary, except that
|
|
// we don't ask the user for confirmation.
|
|
if (process.IsValid()) {
|
|
ScopeSyncMode scope_sync_mode(dap.debugger);
|
|
lldb::StateType state = process.GetState();
|
|
if (state != lldb::eStateConnected) {
|
|
process.Kill();
|
|
}
|
|
// Clear the list of thread ids to avoid sending "thread exited" events
|
|
// for threads of the process we are terminating.
|
|
dap.thread_ids.clear();
|
|
}
|
|
|
|
// FIXME: Should we run 'preRunCommands'?
|
|
// FIXME: Should we add a 'preRestartCommands'?
|
|
if (llvm::Error err = LaunchProcess(*dap.last_launch_request)) {
|
|
response["success"] = llvm::json::Value(false);
|
|
EmplaceSafeString(response, "message", llvm::toString(std::move(err)));
|
|
dap.SendJSON(llvm::json::Value(std::move(response)));
|
|
return;
|
|
}
|
|
|
|
// This is normally done after receiving a "configuration done" request.
|
|
// Because we're restarting, configuration has already happened so we can
|
|
// continue the process right away.
|
|
if (dap.stop_at_entry) {
|
|
if (llvm::Error err = SendThreadStoppedEvent(dap, /*on_entry=*/true)) {
|
|
EmplaceSafeString(response, "message", llvm::toString(std::move(err)));
|
|
dap.SendJSON(llvm::json::Value(std::move(response)));
|
|
return;
|
|
}
|
|
} else {
|
|
dap.target.GetProcess().Continue();
|
|
}
|
|
|
|
dap.SendJSON(llvm::json::Value(std::move(response)));
|
|
}
|
|
|
|
} // namespace lldb_dap
|