aboutsummaryrefslogtreecommitdiff
path: root/lldb/tools/lldb-dap/Handler/SourceRequestHandler.cpp
blob: 755ad206abe26aaaa5b8c6a02fb13f1b812d1172 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
//===-- SourceRequestHandler.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 "Handler/RequestHandler.h"
#include "LLDBUtils.h"
#include "Protocol/ProtocolRequests.h"
#include "Protocol/ProtocolTypes.h"
#include "lldb/API/SBAddress.h"
#include "lldb/API/SBExecutionContext.h"
#include "lldb/API/SBFrame.h"
#include "lldb/API/SBInstructionList.h"
#include "lldb/API/SBProcess.h"
#include "lldb/API/SBStream.h"
#include "lldb/API/SBSymbol.h"
#include "lldb/API/SBTarget.h"
#include "lldb/API/SBThread.h"
#include "lldb/lldb-types.h"
#include "llvm/Support/Error.h"

namespace lldb_dap {

/// Source request; value of command field is 'source'. The request retrieves
/// the source code for a given source reference.
llvm::Expected<protocol::SourceResponseBody>
SourceRequestHandler::Run(const protocol::SourceArguments &args) const {

  uint32_t source_ref =
      args.source->sourceReference.value_or(args.sourceReference);
  const std::optional<lldb::addr_t> source_addr_opt =
      dap.GetSourceReferenceAddress(source_ref);

  if (!source_addr_opt)
    return llvm::make_error<DAPError>(
        llvm::formatv("Unknown source reference {}", source_ref));

  lldb::SBAddress address(*source_addr_opt, dap.target);
  if (!address.IsValid())
    return llvm::make_error<DAPError>("source not found");

  lldb::SBSymbol symbol = address.GetSymbol();
  lldb::SBInstructionList insts;

  if (symbol.IsValid()) {
    insts = symbol.GetInstructions(dap.target);
  } else {
    // No valid symbol, just return the disassembly.
    insts = dap.target.ReadInstructions(
        address, dap.k_number_of_assembly_lines_for_nodebug);
  }

  if (!insts || insts.GetSize() == 0)
    return llvm::make_error<DAPError>(
        llvm::formatv("no instruction source for address {}",
                      address.GetLoadAddress(dap.target)));

  lldb::SBStream stream;
  lldb::SBExecutionContext exe_ctx(dap.target);
  insts.GetDescription(stream, exe_ctx);
  return protocol::SourceResponseBody{/*content=*/stream.GetData(),
                                      /*mimeType=*/
                                      "text/x-lldb.disassembly"};
}

} // namespace lldb_dap