1 //===-- DebugLoc.cpp - Implement DebugLoc class ---------------------------===//
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
7 //===----------------------------------------------------------------------===//
9 #include "llvm/IR/DebugLoc.h"
10 #include "LLVMContextImpl.h"
11 #include "llvm/Config/llvm-config.h"
12 #include "llvm/IR/DebugInfo.h"
15 //===----------------------------------------------------------------------===//
16 // DebugLoc Implementation
17 //===----------------------------------------------------------------------===//
18 DebugLoc::DebugLoc(const DILocation
*L
) : Loc(const_cast<DILocation
*>(L
)) {}
19 DebugLoc::DebugLoc(const MDNode
*L
) : Loc(const_cast<MDNode
*>(L
)) {}
21 DILocation
*DebugLoc::get() const {
22 return cast_or_null
<DILocation
>(Loc
.get());
25 unsigned DebugLoc::getLine() const {
26 assert(get() && "Expected valid DebugLoc");
27 return get()->getLine();
30 unsigned DebugLoc::getCol() const {
31 assert(get() && "Expected valid DebugLoc");
32 return get()->getColumn();
35 MDNode
*DebugLoc::getScope() const {
36 assert(get() && "Expected valid DebugLoc");
37 return get()->getScope();
40 DILocation
*DebugLoc::getInlinedAt() const {
41 assert(get() && "Expected valid DebugLoc");
42 return get()->getInlinedAt();
45 MDNode
*DebugLoc::getInlinedAtScope() const {
46 return cast
<DILocation
>(Loc
)->getInlinedAtScope();
49 DebugLoc
DebugLoc::getFnDebugLoc() const {
50 // FIXME: Add a method on \a DILocation that does this work.
51 const MDNode
*Scope
= getInlinedAtScope();
52 if (auto *SP
= getDISubprogram(Scope
))
53 return DebugLoc::get(SP
->getScopeLine(), 0, SP
);
58 bool DebugLoc::isImplicitCode() const {
59 if (DILocation
*Loc
= get()) {
60 return Loc
->isImplicitCode();
65 void DebugLoc::setImplicitCode(bool ImplicitCode
) {
66 if (DILocation
*Loc
= get()) {
67 Loc
->setImplicitCode(ImplicitCode
);
71 DebugLoc
DebugLoc::get(unsigned Line
, unsigned Col
, const MDNode
*Scope
,
72 const MDNode
*InlinedAt
, bool ImplicitCode
) {
73 // If no scope is available, this is an unknown location.
77 return DILocation::get(Scope
->getContext(), Line
, Col
,
78 const_cast<MDNode
*>(Scope
),
79 const_cast<MDNode
*>(InlinedAt
), ImplicitCode
);
82 DebugLoc
DebugLoc::appendInlinedAt(DebugLoc DL
, DILocation
*InlinedAt
,
84 DenseMap
<const MDNode
*, MDNode
*> &Cache
,
86 SmallVector
<DILocation
*, 3> InlinedAtLocations
;
87 DILocation
*Last
= InlinedAt
;
88 DILocation
*CurInlinedAt
= DL
;
90 // Gather all the inlined-at nodes.
91 while (DILocation
*IA
= CurInlinedAt
->getInlinedAt()) {
92 // Skip any we've already built nodes for.
93 if (auto *Found
= Cache
[IA
]) {
94 Last
= cast
<DILocation
>(Found
);
98 if (ReplaceLast
&& !IA
->getInlinedAt())
100 InlinedAtLocations
.push_back(IA
);
104 // Starting from the top, rebuild the nodes to point to the new inlined-at
105 // location (then rebuilding the rest of the chain behind it) and update the
106 // map of already-constructed inlined-at nodes.
107 for (const DILocation
*MD
: reverse(InlinedAtLocations
))
108 Cache
[MD
] = Last
= DILocation::getDistinct(
109 Ctx
, MD
->getLine(), MD
->getColumn(), MD
->getScope(), Last
);
114 #if !defined(NDEBUG) || defined(LLVM_ENABLE_DUMP)
115 LLVM_DUMP_METHOD
void DebugLoc::dump() const { print(dbgs()); }
118 void DebugLoc::print(raw_ostream
&OS
) const {
122 // Print source line info.
123 auto *Scope
= cast
<DIScope
>(getScope());
124 OS
<< Scope
->getFilename();
125 OS
<< ':' << getLine();
127 OS
<< ':' << getCol();
129 if (DebugLoc InlinedAtDL
= getInlinedAt()) {
131 InlinedAtDL
.print(OS
);