1 //===-- DebugLoc.cpp - Implement DebugLoc class ---------------------------===//
3 // The LLVM Compiler Infrastructure
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
8 //===----------------------------------------------------------------------===//
10 #include "llvm/IR/DebugLoc.h"
11 #include "LLVMContextImpl.h"
12 #include "llvm/Config/llvm-config.h"
13 #include "llvm/IR/DebugInfo.h"
16 //===----------------------------------------------------------------------===//
17 // DebugLoc Implementation
18 //===----------------------------------------------------------------------===//
19 DebugLoc::DebugLoc(const DILocation
*L
) : Loc(const_cast<DILocation
*>(L
)) {}
20 DebugLoc::DebugLoc(const MDNode
*L
) : Loc(const_cast<MDNode
*>(L
)) {}
22 DILocation
*DebugLoc::get() const {
23 return cast_or_null
<DILocation
>(Loc
.get());
26 unsigned DebugLoc::getLine() const {
27 assert(get() && "Expected valid DebugLoc");
28 return get()->getLine();
31 unsigned DebugLoc::getCol() const {
32 assert(get() && "Expected valid DebugLoc");
33 return get()->getColumn();
36 MDNode
*DebugLoc::getScope() const {
37 assert(get() && "Expected valid DebugLoc");
38 return get()->getScope();
41 DILocation
*DebugLoc::getInlinedAt() const {
42 assert(get() && "Expected valid DebugLoc");
43 return get()->getInlinedAt();
46 MDNode
*DebugLoc::getInlinedAtScope() const {
47 return cast
<DILocation
>(Loc
)->getInlinedAtScope();
50 DebugLoc
DebugLoc::getFnDebugLoc() const {
51 // FIXME: Add a method on \a DILocation that does this work.
52 const MDNode
*Scope
= getInlinedAtScope();
53 if (auto *SP
= getDISubprogram(Scope
))
54 return DebugLoc::get(SP
->getScopeLine(), 0, SP
);
59 bool DebugLoc::isImplicitCode() const {
60 if (DILocation
*Loc
= get()) {
61 return Loc
->isImplicitCode();
66 void DebugLoc::setImplicitCode(bool ImplicitCode
) {
67 if (DILocation
*Loc
= get()) {
68 Loc
->setImplicitCode(ImplicitCode
);
72 DebugLoc
DebugLoc::get(unsigned Line
, unsigned Col
, const MDNode
*Scope
,
73 const MDNode
*InlinedAt
, bool ImplicitCode
) {
74 // If no scope is available, this is an unknown location.
78 return DILocation::get(Scope
->getContext(), Line
, Col
,
79 const_cast<MDNode
*>(Scope
),
80 const_cast<MDNode
*>(InlinedAt
), ImplicitCode
);
83 DebugLoc
DebugLoc::appendInlinedAt(DebugLoc DL
, DILocation
*InlinedAt
,
85 DenseMap
<const MDNode
*, MDNode
*> &Cache
,
87 SmallVector
<DILocation
*, 3> InlinedAtLocations
;
88 DILocation
*Last
= InlinedAt
;
89 DILocation
*CurInlinedAt
= DL
;
91 // Gather all the inlined-at nodes.
92 while (DILocation
*IA
= CurInlinedAt
->getInlinedAt()) {
93 // Skip any we've already built nodes for.
94 if (auto *Found
= Cache
[IA
]) {
95 Last
= cast
<DILocation
>(Found
);
99 if (ReplaceLast
&& !IA
->getInlinedAt())
101 InlinedAtLocations
.push_back(IA
);
105 // Starting from the top, rebuild the nodes to point to the new inlined-at
106 // location (then rebuilding the rest of the chain behind it) and update the
107 // map of already-constructed inlined-at nodes.
108 for (const DILocation
*MD
: reverse(InlinedAtLocations
))
109 Cache
[MD
] = Last
= DILocation::getDistinct(
110 Ctx
, MD
->getLine(), MD
->getColumn(), MD
->getScope(), Last
);
115 #if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
116 LLVM_DUMP_METHOD
void DebugLoc::dump() const { print(dbgs()); }
119 void DebugLoc::print(raw_ostream
&OS
) const {
123 // Print source line info.
124 auto *Scope
= cast
<DIScope
>(getScope());
125 OS
<< Scope
->getFilename();
126 OS
<< ':' << getLine();
128 OS
<< ':' << getCol();
130 if (DebugLoc InlinedAtDL
= getInlinedAt()) {
132 InlinedAtDL
.print(OS
);