
- Migrate set breakpoint requests to use typed RequestHandler - `SetBreakpointsRequestHandler` - `SetDataBreakpointsRequestHandler` - `SetFunctionBreakpointsRequestHandler` - `SetInstructionBreakpointsRequestHandler` - `DataBreakpointInfoRequestHandler` - Decouple JSON from lldb-dap `Breakpoint` classes
96 lines
3.7 KiB
C++
96 lines
3.7 KiB
C++
//===-- DataBreakpointInfoRequestHandler.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 "Protocol/ProtocolTypes.h"
|
|
#include "RequestHandler.h"
|
|
#include "lldb/API/SBMemoryRegionInfo.h"
|
|
#include "llvm/ADT/StringExtras.h"
|
|
#include <optional>
|
|
|
|
namespace lldb_dap {
|
|
|
|
/// Obtains information on a possible data breakpoint that could be set on an
|
|
/// expression or variable. Clients should only call this request if the
|
|
/// corresponding capability supportsDataBreakpoints is true.
|
|
llvm::Expected<protocol::DataBreakpointInfoResponseBody>
|
|
DataBreakpointInfoRequestHandler::Run(
|
|
const protocol::DataBreakpointInfoArguments &args) const {
|
|
protocol::DataBreakpointInfoResponseBody response;
|
|
lldb::SBFrame frame = dap.GetLLDBFrame(args.frameId.value_or(UINT64_MAX));
|
|
lldb::SBValue variable = dap.variables.FindVariable(
|
|
args.variablesReference.value_or(0), args.name);
|
|
std::string addr, size;
|
|
|
|
bool is_data_ok = true;
|
|
if (variable.IsValid()) {
|
|
lldb::addr_t load_addr = variable.GetLoadAddress();
|
|
size_t byte_size = variable.GetByteSize();
|
|
if (load_addr == LLDB_INVALID_ADDRESS) {
|
|
is_data_ok = false;
|
|
response.description = "does not exist in memory, its location is " +
|
|
std::string(variable.GetLocation());
|
|
} else if (byte_size == 0) {
|
|
is_data_ok = false;
|
|
response.description = "variable size is 0";
|
|
} else {
|
|
addr = llvm::utohexstr(load_addr);
|
|
size = llvm::utostr(byte_size);
|
|
}
|
|
} else if (args.variablesReference.value_or(0) == 0 && frame.IsValid()) {
|
|
lldb::SBValue value = frame.EvaluateExpression(args.name.c_str());
|
|
if (value.GetError().Fail()) {
|
|
lldb::SBError error = value.GetError();
|
|
const char *error_cstr = error.GetCString();
|
|
is_data_ok = false;
|
|
response.description = error_cstr && error_cstr[0]
|
|
? std::string(error_cstr)
|
|
: "evaluation failed";
|
|
} else {
|
|
uint64_t load_addr = value.GetValueAsUnsigned();
|
|
lldb::SBData data = value.GetPointeeData();
|
|
if (data.IsValid()) {
|
|
size = llvm::utostr(data.GetByteSize());
|
|
addr = llvm::utohexstr(load_addr);
|
|
lldb::SBMemoryRegionInfo region;
|
|
lldb::SBError err =
|
|
dap.target.GetProcess().GetMemoryRegionInfo(load_addr, region);
|
|
// Only lldb-server supports "qMemoryRegionInfo". So, don't fail this
|
|
// request if SBProcess::GetMemoryRegionInfo returns error.
|
|
if (err.Success()) {
|
|
if (!(region.IsReadable() || region.IsWritable())) {
|
|
is_data_ok = false;
|
|
response.description = "memory region for address " + addr +
|
|
" has no read or write permissions";
|
|
}
|
|
}
|
|
} else {
|
|
is_data_ok = false;
|
|
response.description =
|
|
"unable to get byte size for expression: " + args.name;
|
|
}
|
|
}
|
|
} else {
|
|
is_data_ok = false;
|
|
response.description = "variable not found: " + args.name;
|
|
}
|
|
|
|
if (is_data_ok) {
|
|
response.dataId = addr + "/" + size;
|
|
response.accessTypes = {protocol::eDataBreakpointAccessTypeRead,
|
|
protocol::eDataBreakpointAccessTypeWrite,
|
|
protocol::eDataBreakpointAccessTypeReadWrite};
|
|
response.description = size + " bytes at " + addr + " " + args.name;
|
|
}
|
|
|
|
return response;
|
|
}
|
|
|
|
} // namespace lldb_dap
|