mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-07 11:33:44 +00:00
Identify when a lexical scope is split in to multiple instruction ranges. Emit such ranges using DW_AT_ranges.
This patch fixes bug (PR6894) introduced by previous version of this patch. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@102454 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
ed33b13a10
commit
eac9c07fde
@ -32,6 +32,7 @@
|
|||||||
#include "llvm/Analysis/DebugInfo.h"
|
#include "llvm/Analysis/DebugInfo.h"
|
||||||
#include "llvm/ADT/STLExtras.h"
|
#include "llvm/ADT/STLExtras.h"
|
||||||
#include "llvm/ADT/StringExtras.h"
|
#include "llvm/ADT/StringExtras.h"
|
||||||
|
#include "llvm/Support/CommandLine.h"
|
||||||
#include "llvm/Support/Debug.h"
|
#include "llvm/Support/Debug.h"
|
||||||
#include "llvm/Support/ErrorHandling.h"
|
#include "llvm/Support/ErrorHandling.h"
|
||||||
#include "llvm/Support/ValueHandle.h"
|
#include "llvm/Support/ValueHandle.h"
|
||||||
@ -40,6 +41,12 @@
|
|||||||
#include "llvm/System/Path.h"
|
#include "llvm/System/Path.h"
|
||||||
using namespace llvm;
|
using namespace llvm;
|
||||||
|
|
||||||
|
static cl::opt<bool> PrintDbgScope("print-dbgscope", cl::Hidden,
|
||||||
|
cl::desc("Print DbgScope information for each machine instruction"));
|
||||||
|
|
||||||
|
static cl::opt<bool> DisableDebugInfoPrinting("disable-debug-info-print", cl::Hidden,
|
||||||
|
cl::desc("Disable debug info printing"));
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
const char *DWARFGroupName = "DWARF Emission";
|
const char *DWARFGroupName = "DWARF Emission";
|
||||||
const char *DbgTimerName = "DWARF Debug Writer";
|
const char *DbgTimerName = "DWARF Debug Writer";
|
||||||
@ -184,6 +191,12 @@ public:
|
|||||||
DIE *getDIE() const { return TheDIE; }
|
DIE *getDIE() const { return TheDIE; }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
/// DbgRange - This is used to track range of instructions with identical
|
||||||
|
/// debug info scope.
|
||||||
|
///
|
||||||
|
typedef std::pair<const MachineInstr *, const MachineInstr *> DbgRange;
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
/// DbgScope - This class is used to track scope information.
|
/// DbgScope - This class is used to track scope information.
|
||||||
///
|
///
|
||||||
@ -195,17 +208,19 @@ class DbgScope {
|
|||||||
bool AbstractScope; // Abstract Scope
|
bool AbstractScope; // Abstract Scope
|
||||||
const MachineInstr *LastInsn; // Last instruction of this scope.
|
const MachineInstr *LastInsn; // Last instruction of this scope.
|
||||||
const MachineInstr *FirstInsn; // First instruction of this scope.
|
const MachineInstr *FirstInsn; // First instruction of this scope.
|
||||||
|
unsigned DFSIn, DFSOut;
|
||||||
// Scopes defined in scope. Contents not owned.
|
// Scopes defined in scope. Contents not owned.
|
||||||
SmallVector<DbgScope *, 4> Scopes;
|
SmallVector<DbgScope *, 4> Scopes;
|
||||||
// Variables declared in scope. Contents owned.
|
// Variables declared in scope. Contents owned.
|
||||||
SmallVector<DbgVariable *, 8> Variables;
|
SmallVector<DbgVariable *, 8> Variables;
|
||||||
|
SmallVector<DbgRange, 4> Ranges;
|
||||||
// Private state for dump()
|
// Private state for dump()
|
||||||
mutable unsigned IndentLevel;
|
mutable unsigned IndentLevel;
|
||||||
public:
|
public:
|
||||||
DbgScope(DbgScope *P, DIDescriptor D, MDNode *I = 0)
|
DbgScope(DbgScope *P, DIDescriptor D, MDNode *I = 0)
|
||||||
: Parent(P), Desc(D), InlinedAtLocation(I), AbstractScope(false),
|
: Parent(P), Desc(D), InlinedAtLocation(I), AbstractScope(false),
|
||||||
LastInsn(0), FirstInsn(0), IndentLevel(0) {}
|
LastInsn(0), FirstInsn(0),
|
||||||
|
DFSIn(0), DFSOut(0), IndentLevel(0) {}
|
||||||
virtual ~DbgScope();
|
virtual ~DbgScope();
|
||||||
|
|
||||||
// Accessors.
|
// Accessors.
|
||||||
@ -216,12 +231,55 @@ public:
|
|||||||
MDNode *getScopeNode() const { return Desc.getNode(); }
|
MDNode *getScopeNode() const { return Desc.getNode(); }
|
||||||
const SmallVector<DbgScope *, 4> &getScopes() { return Scopes; }
|
const SmallVector<DbgScope *, 4> &getScopes() { return Scopes; }
|
||||||
const SmallVector<DbgVariable *, 8> &getVariables() { return Variables; }
|
const SmallVector<DbgVariable *, 8> &getVariables() { return Variables; }
|
||||||
void setLastInsn(const MachineInstr *MI) { LastInsn = MI; }
|
const SmallVector<DbgRange, 4> &getRanges() { return Ranges; }
|
||||||
const MachineInstr *getLastInsn() { return LastInsn; }
|
|
||||||
void setFirstInsn(const MachineInstr *MI) { FirstInsn = MI; }
|
/// openInsnRange - This scope covers instruction range starting from MI.
|
||||||
|
void openInsnRange(const MachineInstr *MI) {
|
||||||
|
if (!FirstInsn)
|
||||||
|
FirstInsn = MI;
|
||||||
|
|
||||||
|
if (Parent)
|
||||||
|
Parent->openInsnRange(MI);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// extendInsnRange - Extend the current instruction range covered by
|
||||||
|
/// this scope.
|
||||||
|
void extendInsnRange(const MachineInstr *MI) {
|
||||||
|
assert (FirstInsn && "MI Range is not open!");
|
||||||
|
LastInsn = MI;
|
||||||
|
if (Parent)
|
||||||
|
Parent->extendInsnRange(MI);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// closeInsnRange - Create a range based on FirstInsn and LastInsn collected
|
||||||
|
/// until now. This is used when a new scope is encountered while walking
|
||||||
|
/// machine instructions.
|
||||||
|
void closeInsnRange(DbgScope *NewScope = NULL) {
|
||||||
|
assert (LastInsn && "Last insn missing!");
|
||||||
|
Ranges.push_back(DbgRange(FirstInsn, LastInsn));
|
||||||
|
FirstInsn = NULL;
|
||||||
|
LastInsn = NULL;
|
||||||
|
// If Parent dominates NewScope then do not close Parent's instruction
|
||||||
|
// range.
|
||||||
|
if (Parent && (!NewScope || !Parent->dominates(NewScope)))
|
||||||
|
Parent->closeInsnRange(NewScope);
|
||||||
|
}
|
||||||
|
|
||||||
void setAbstractScope() { AbstractScope = true; }
|
void setAbstractScope() { AbstractScope = true; }
|
||||||
bool isAbstractScope() const { return AbstractScope; }
|
bool isAbstractScope() const { return AbstractScope; }
|
||||||
const MachineInstr *getFirstInsn() { return FirstInsn; }
|
|
||||||
|
// Depth First Search support to walk and mainpluate DbgScope hierarchy.
|
||||||
|
unsigned getDFSOut() const { return DFSOut; }
|
||||||
|
void setDFSOut(unsigned O) { DFSOut = O; }
|
||||||
|
unsigned getDFSIn() const { return DFSIn; }
|
||||||
|
void setDFSIn(unsigned I) { DFSIn = I; }
|
||||||
|
bool dominates(const DbgScope *S) {
|
||||||
|
if (S == this)
|
||||||
|
return true;
|
||||||
|
if (DFSIn < S->getDFSIn() && DFSOut > S->getDFSOut())
|
||||||
|
return true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
/// addScope - Add a scope to the scope.
|
/// addScope - Add a scope to the scope.
|
||||||
///
|
///
|
||||||
@ -231,43 +289,6 @@ public:
|
|||||||
///
|
///
|
||||||
void addVariable(DbgVariable *V) { Variables.push_back(V); }
|
void addVariable(DbgVariable *V) { Variables.push_back(V); }
|
||||||
|
|
||||||
void fixInstructionMarkers(DenseMap<const MachineInstr *,
|
|
||||||
unsigned> &MIIndexMap) {
|
|
||||||
assert(getFirstInsn() && "First instruction is missing!");
|
|
||||||
|
|
||||||
// Use the end of last child scope as end of this scope.
|
|
||||||
const SmallVector<DbgScope *, 4> &Scopes = getScopes();
|
|
||||||
const MachineInstr *LastInsn = getFirstInsn();
|
|
||||||
unsigned LIndex = 0;
|
|
||||||
if (Scopes.empty()) {
|
|
||||||
assert(getLastInsn() && "Inner most scope does not have last insn!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
for (SmallVector<DbgScope *, 4>::const_iterator SI = Scopes.begin(),
|
|
||||||
SE = Scopes.end(); SI != SE; ++SI) {
|
|
||||||
DbgScope *DS = *SI;
|
|
||||||
DS->fixInstructionMarkers(MIIndexMap);
|
|
||||||
const MachineInstr *DSLastInsn = DS->getLastInsn();
|
|
||||||
unsigned DSI = MIIndexMap[DSLastInsn];
|
|
||||||
if (DSI > LIndex) {
|
|
||||||
LastInsn = DSLastInsn;
|
|
||||||
LIndex = DSI;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
unsigned CurrentLastInsnIndex = 0;
|
|
||||||
if (const MachineInstr *CL = getLastInsn())
|
|
||||||
CurrentLastInsnIndex = MIIndexMap[CL];
|
|
||||||
unsigned FIndex = MIIndexMap[getFirstInsn()];
|
|
||||||
|
|
||||||
// Set LastInsn as the last instruction for this scope only if
|
|
||||||
// it follows
|
|
||||||
// 1) this scope's first instruction and
|
|
||||||
// 2) current last instruction for this scope, if any.
|
|
||||||
if (LIndex >= CurrentLastInsnIndex && LIndex >= FIndex)
|
|
||||||
setLastInsn(LastInsn);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifndef NDEBUG
|
#ifndef NDEBUG
|
||||||
void dump() const;
|
void dump() const;
|
||||||
#endif
|
#endif
|
||||||
@ -308,7 +329,8 @@ DwarfDebug::DwarfDebug(AsmPrinter *A, Module *M)
|
|||||||
|
|
||||||
DwarfFrameSectionSym = DwarfInfoSectionSym = DwarfAbbrevSectionSym = 0;
|
DwarfFrameSectionSym = DwarfInfoSectionSym = DwarfAbbrevSectionSym = 0;
|
||||||
DwarfStrSectionSym = TextSectionSym = 0;
|
DwarfStrSectionSym = TextSectionSym = 0;
|
||||||
|
DwarfDebugRangeSectionSym = 0;
|
||||||
|
FunctionBeginSym = 0;
|
||||||
if (TimePassesIsEnabled) {
|
if (TimePassesIsEnabled) {
|
||||||
NamedRegionTimer T(DbgTimerName, DWARFGroupName);
|
NamedRegionTimer T(DbgTimerName, DWARFGroupName);
|
||||||
beginModule(M);
|
beginModule(M);
|
||||||
@ -1257,56 +1279,6 @@ DIE *DwarfDebug::createSubprogramDIE(const DISubprogram &SP, bool MakeDecl) {
|
|||||||
return SPDie;
|
return SPDie;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// getUpdatedDbgScope - Find DbgScope assicated with the instruction.
|
|
||||||
/// Update scope hierarchy. Create abstract scope if required.
|
|
||||||
DbgScope *DwarfDebug::getUpdatedDbgScope(MDNode *N, const MachineInstr *MI,
|
|
||||||
MDNode *InlinedAt) {
|
|
||||||
assert(N && "Invalid Scope encoding!");
|
|
||||||
assert(MI && "Missing machine instruction!");
|
|
||||||
bool isAConcreteScope = InlinedAt != 0;
|
|
||||||
|
|
||||||
DbgScope *NScope = NULL;
|
|
||||||
|
|
||||||
if (InlinedAt)
|
|
||||||
NScope = DbgScopeMap.lookup(InlinedAt);
|
|
||||||
else
|
|
||||||
NScope = DbgScopeMap.lookup(N);
|
|
||||||
assert(NScope && "Unable to find working scope!");
|
|
||||||
|
|
||||||
if (NScope->getFirstInsn())
|
|
||||||
return NScope;
|
|
||||||
|
|
||||||
DbgScope *Parent = NULL;
|
|
||||||
if (isAConcreteScope) {
|
|
||||||
DILocation IL(InlinedAt);
|
|
||||||
Parent = getUpdatedDbgScope(IL.getScope().getNode(), MI,
|
|
||||||
IL.getOrigLocation().getNode());
|
|
||||||
assert(Parent && "Unable to find Parent scope!");
|
|
||||||
NScope->setParent(Parent);
|
|
||||||
Parent->addScope(NScope);
|
|
||||||
} else if (DIDescriptor(N).isLexicalBlock()) {
|
|
||||||
DILexicalBlock DB(N);
|
|
||||||
Parent = getUpdatedDbgScope(DB.getContext().getNode(), MI, InlinedAt);
|
|
||||||
NScope->setParent(Parent);
|
|
||||||
Parent->addScope(NScope);
|
|
||||||
}
|
|
||||||
|
|
||||||
NScope->setFirstInsn(MI);
|
|
||||||
|
|
||||||
if (!Parent && !InlinedAt) {
|
|
||||||
StringRef SPName = DISubprogram(N).getLinkageName();
|
|
||||||
if (SPName == Asm->MF->getFunction()->getName())
|
|
||||||
CurrentFnDbgScope = NScope;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isAConcreteScope) {
|
|
||||||
ConcreteScopes[InlinedAt] = NScope;
|
|
||||||
getOrCreateAbstractScope(N);
|
|
||||||
}
|
|
||||||
|
|
||||||
return NScope;
|
|
||||||
}
|
|
||||||
|
|
||||||
DbgScope *DwarfDebug::getOrCreateAbstractScope(MDNode *N) {
|
DbgScope *DwarfDebug::getOrCreateAbstractScope(MDNode *N) {
|
||||||
assert(N && "Invalid Scope encoding!");
|
assert(N && "Invalid Scope encoding!");
|
||||||
|
|
||||||
@ -1404,22 +1376,41 @@ DIE *DwarfDebug::updateSubprogramScopeDIE(MDNode *SPNode) {
|
|||||||
/// for this scope and attach DW_AT_low_pc/DW_AT_high_pc labels.
|
/// for this scope and attach DW_AT_low_pc/DW_AT_high_pc labels.
|
||||||
DIE *DwarfDebug::constructLexicalScopeDIE(DbgScope *Scope) {
|
DIE *DwarfDebug::constructLexicalScopeDIE(DbgScope *Scope) {
|
||||||
|
|
||||||
MCSymbol *Start = InsnBeforeLabelMap.lookup(Scope->getFirstInsn());
|
DIE *ScopeDIE = new DIE(dwarf::DW_TAG_lexical_block);
|
||||||
MCSymbol *End = InsnAfterLabelMap.lookup(Scope->getLastInsn());
|
if (Scope->isAbstractScope())
|
||||||
|
return ScopeDIE;
|
||||||
|
|
||||||
|
const SmallVector<DbgRange, 4> &Ranges = Scope->getRanges();
|
||||||
|
if (Ranges.empty())
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
SmallVector<DbgRange, 4>::const_iterator RI = Ranges.begin();
|
||||||
|
if (Ranges.size() > 1) {
|
||||||
|
// .debug_range section has not been laid out yet. Emit offset in
|
||||||
|
// .debug_range as a uint, size 4, for now. emitDIE will handle
|
||||||
|
// DW_AT_ranges appropriately.
|
||||||
|
addUInt(ScopeDIE, dwarf::DW_AT_ranges, dwarf::DW_FORM_data4,
|
||||||
|
DebugRangeSymbols.size() * Asm->getTargetData().getPointerSize());
|
||||||
|
for (SmallVector<DbgRange, 4>::const_iterator RI = Ranges.begin(),
|
||||||
|
RE = Ranges.end(); RI != RE; ++RI) {
|
||||||
|
DebugRangeSymbols.push_back(LabelsBeforeInsn.lookup(RI->first));
|
||||||
|
DebugRangeSymbols.push_back(LabelsAfterInsn.lookup(RI->second));
|
||||||
|
}
|
||||||
|
DebugRangeSymbols.push_back(NULL);
|
||||||
|
DebugRangeSymbols.push_back(NULL);
|
||||||
|
return ScopeDIE;
|
||||||
|
}
|
||||||
|
|
||||||
|
MCSymbol *Start = LabelsBeforeInsn.lookup(RI->first);
|
||||||
|
MCSymbol *End = LabelsAfterInsn.lookup(RI->second);
|
||||||
|
|
||||||
if (Start == 0 || End == 0) return 0;
|
if (Start == 0 || End == 0) return 0;
|
||||||
|
|
||||||
assert(Start->isDefined() && "Invalid starting label for an inlined scope!");
|
assert(Start->isDefined() && "Invalid starting label for an inlined scope!");
|
||||||
assert(End->isDefined() && "Invalid end label for an inlined scope!");
|
assert(End->isDefined() && "Invalid end label for an inlined scope!");
|
||||||
|
|
||||||
DIE *ScopeDIE = new DIE(dwarf::DW_TAG_lexical_block);
|
addLabel(ScopeDIE, dwarf::DW_AT_low_pc, dwarf::DW_FORM_addr, Start);
|
||||||
if (Scope->isAbstractScope())
|
addLabel(ScopeDIE, dwarf::DW_AT_high_pc, dwarf::DW_FORM_addr, End);
|
||||||
return ScopeDIE;
|
|
||||||
|
|
||||||
addLabel(ScopeDIE, dwarf::DW_AT_low_pc, dwarf::DW_FORM_addr,
|
|
||||||
Start ? Start : Asm->GetTempSymbol("func_begin",
|
|
||||||
Asm->getFunctionNumber()));
|
|
||||||
addLabel(ScopeDIE, dwarf::DW_AT_high_pc, dwarf::DW_FORM_addr,
|
|
||||||
End ? End : Asm->GetTempSymbol("func_end",Asm->getFunctionNumber()));
|
|
||||||
|
|
||||||
return ScopeDIE;
|
return ScopeDIE;
|
||||||
}
|
}
|
||||||
@ -1428,14 +1419,28 @@ DIE *DwarfDebug::constructLexicalScopeDIE(DbgScope *Scope) {
|
|||||||
/// a function. Construct DIE to represent this concrete inlined copy
|
/// a function. Construct DIE to represent this concrete inlined copy
|
||||||
/// of the function.
|
/// of the function.
|
||||||
DIE *DwarfDebug::constructInlinedScopeDIE(DbgScope *Scope) {
|
DIE *DwarfDebug::constructInlinedScopeDIE(DbgScope *Scope) {
|
||||||
MCSymbol *StartLabel = InsnBeforeLabelMap.lookup(Scope->getFirstInsn());
|
|
||||||
MCSymbol *EndLabel = InsnAfterLabelMap.lookup(Scope->getLastInsn());
|
|
||||||
if (StartLabel == 0 || EndLabel == 0) return 0;
|
|
||||||
|
|
||||||
|
const SmallVector<DbgRange, 4> &Ranges = Scope->getRanges();
|
||||||
|
assert (Ranges.empty() == false
|
||||||
|
&& "DbgScope does not have instruction markers!");
|
||||||
|
|
||||||
|
// FIXME : .debug_inlined section specification does not clearly state how
|
||||||
|
// to emit inlined scope that is split into multiple instruction ranges.
|
||||||
|
// For now, use first instruction range and emit low_pc/high_pc pair and
|
||||||
|
// corresponding .debug_inlined section entry for this pair.
|
||||||
|
SmallVector<DbgRange, 4>::const_iterator RI = Ranges.begin();
|
||||||
|
MCSymbol *StartLabel = LabelsBeforeInsn.lookup(RI->first);
|
||||||
|
MCSymbol *EndLabel = LabelsAfterInsn.lookup(RI->second);
|
||||||
|
|
||||||
|
if (StartLabel == 0 || EndLabel == 0) {
|
||||||
|
assert (0 && "Unexpected Start and End labels for a inlined scope!");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
assert(StartLabel->isDefined() &&
|
assert(StartLabel->isDefined() &&
|
||||||
"Invalid starting label for an inlined scope!");
|
"Invalid starting label for an inlined scope!");
|
||||||
assert(EndLabel->isDefined() &&
|
assert(EndLabel->isDefined() &&
|
||||||
"Invalid end label for an inlined scope!");
|
"Invalid end label for an inlined scope!");
|
||||||
|
|
||||||
if (!Scope->getScopeNode())
|
if (!Scope->getScopeNode())
|
||||||
return NULL;
|
return NULL;
|
||||||
DIScope DS(Scope->getScopeNode());
|
DIScope DS(Scope->getScopeNode());
|
||||||
@ -1849,6 +1854,9 @@ void DwarfDebug::constructSubprogramDIE(MDNode *N) {
|
|||||||
/// content. Create global DIEs and emit initial debug info sections.
|
/// content. Create global DIEs and emit initial debug info sections.
|
||||||
/// This is inovked by the target AsmPrinter.
|
/// This is inovked by the target AsmPrinter.
|
||||||
void DwarfDebug::beginModule(Module *M) {
|
void DwarfDebug::beginModule(Module *M) {
|
||||||
|
if (DisableDebugInfoPrinting)
|
||||||
|
return;
|
||||||
|
|
||||||
DebugInfoFinder DbgFinder;
|
DebugInfoFinder DbgFinder;
|
||||||
DbgFinder.processModule(*M);
|
DbgFinder.processModule(*M);
|
||||||
|
|
||||||
@ -2111,10 +2119,6 @@ void DwarfDebug::beginScope(const MachineInstr *MI) {
|
|||||||
if (DL.isUnknown())
|
if (DL.isUnknown())
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Check and update last known location info.
|
|
||||||
if (DL == PrevInstLoc)
|
|
||||||
return;
|
|
||||||
|
|
||||||
MDNode *Scope = DL.getScope(Asm->MF->getFunction()->getContext());
|
MDNode *Scope = DL.getScope(Asm->MF->getFunction()->getContext());
|
||||||
|
|
||||||
// FIXME: Should only verify each scope once!
|
// FIXME: Should only verify each scope once!
|
||||||
@ -2126,65 +2130,174 @@ void DwarfDebug::beginScope(const MachineInstr *MI) {
|
|||||||
DenseMap<const MachineInstr *, DbgVariable *>::iterator DI
|
DenseMap<const MachineInstr *, DbgVariable *>::iterator DI
|
||||||
= DbgValueStartMap.find(MI);
|
= DbgValueStartMap.find(MI);
|
||||||
if (DI != DbgValueStartMap.end()) {
|
if (DI != DbgValueStartMap.end()) {
|
||||||
MCSymbol *Label = recordSourceLine(DL.getLine(), DL.getCol(), Scope);
|
MCSymbol *Label = NULL;
|
||||||
|
if (DL == PrevInstLoc)
|
||||||
|
Label = PrevLabel;
|
||||||
|
else {
|
||||||
|
Label = recordSourceLine(DL.getLine(), DL.getCol(), Scope);
|
||||||
PrevInstLoc = DL;
|
PrevInstLoc = DL;
|
||||||
|
PrevLabel = Label;
|
||||||
|
}
|
||||||
DI->second->setDbgValueLabel(Label);
|
DI->second->setDbgValueLabel(Label);
|
||||||
}
|
}
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Emit a label to indicate location change. This is used for line
|
// Emit a label to indicate location change. This is used for line
|
||||||
// table even if this instruction does start a new scope.
|
// table even if this instruction does not start a new scope.
|
||||||
MCSymbol *Label = recordSourceLine(DL.getLine(), DL.getCol(), Scope);
|
MCSymbol *Label = NULL;
|
||||||
|
if (DL == PrevInstLoc)
|
||||||
|
Label = PrevLabel;
|
||||||
|
else {
|
||||||
|
Label = recordSourceLine(DL.getLine(), DL.getCol(), Scope);
|
||||||
PrevInstLoc = DL;
|
PrevInstLoc = DL;
|
||||||
|
PrevLabel = Label;
|
||||||
|
}
|
||||||
|
|
||||||
// If this instruction begins a scope then note down corresponding label.
|
// If this instruction begins a scope then note down corresponding label.
|
||||||
if (InsnsBeginScopeSet.count(MI) != 0)
|
if (InsnsBeginScopeSet.count(MI) != 0)
|
||||||
InsnBeforeLabelMap[MI] = Label;
|
LabelsBeforeInsn[MI] = Label;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// endScope - Process end of a scope.
|
/// endScope - Process end of a scope.
|
||||||
void DwarfDebug::endScope(const MachineInstr *MI) {
|
void DwarfDebug::endScope(const MachineInstr *MI) {
|
||||||
// Ignore DBG_VALUE instruction.
|
|
||||||
if (MI->isDebugValue())
|
|
||||||
return;
|
|
||||||
|
|
||||||
// Check location.
|
|
||||||
DebugLoc DL = MI->getDebugLoc();
|
|
||||||
if (DL.isUnknown())
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (InsnsEndScopeSet.count(MI) != 0) {
|
if (InsnsEndScopeSet.count(MI) != 0) {
|
||||||
// Emit a label if this instruction ends a scope.
|
// Emit a label if this instruction ends a scope.
|
||||||
MCSymbol *Label = MMI->getContext().CreateTempSymbol();
|
MCSymbol *Label = MMI->getContext().CreateTempSymbol();
|
||||||
Asm->OutStreamer.EmitLabel(Label);
|
Asm->OutStreamer.EmitLabel(Label);
|
||||||
InsnAfterLabelMap[MI] = Label;
|
LabelsAfterInsn[MI] = Label;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// createDbgScope - Create DbgScope for the scope.
|
/// getOrCreateDbgScope - Create DbgScope for the scope.
|
||||||
void DwarfDebug::createDbgScope(MDNode *Scope, MDNode *InlinedAt) {
|
DbgScope *DwarfDebug::getOrCreateDbgScope(MDNode *Scope, MDNode *InlinedAt) {
|
||||||
if (!InlinedAt) {
|
if (!InlinedAt) {
|
||||||
DbgScope *WScope = DbgScopeMap.lookup(Scope);
|
DbgScope *WScope = DbgScopeMap.lookup(Scope);
|
||||||
if (WScope)
|
if (WScope)
|
||||||
return;
|
return WScope;
|
||||||
WScope = new DbgScope(NULL, DIDescriptor(Scope), NULL);
|
WScope = new DbgScope(NULL, DIDescriptor(Scope), NULL);
|
||||||
DbgScopeMap.insert(std::make_pair(Scope, WScope));
|
DbgScopeMap.insert(std::make_pair(Scope, WScope));
|
||||||
if (DIDescriptor(Scope).isLexicalBlock())
|
if (DIDescriptor(Scope).isLexicalBlock()) {
|
||||||
createDbgScope(DILexicalBlock(Scope).getContext().getNode(), NULL);
|
DbgScope *Parent =
|
||||||
return;
|
getOrCreateDbgScope(DILexicalBlock(Scope).getContext().getNode(), NULL);
|
||||||
|
WScope->setParent(Parent);
|
||||||
|
Parent->addScope(WScope);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!WScope->getParent()) {
|
||||||
|
StringRef SPName = DISubprogram(Scope).getLinkageName();
|
||||||
|
if (SPName == Asm->MF->getFunction()->getName())
|
||||||
|
CurrentFnDbgScope = WScope;
|
||||||
|
}
|
||||||
|
|
||||||
|
return WScope;
|
||||||
}
|
}
|
||||||
|
|
||||||
DbgScope *WScope = DbgScopeMap.lookup(InlinedAt);
|
DbgScope *WScope = DbgScopeMap.lookup(InlinedAt);
|
||||||
if (WScope)
|
if (WScope)
|
||||||
return;
|
return WScope;
|
||||||
|
|
||||||
WScope = new DbgScope(NULL, DIDescriptor(Scope), InlinedAt);
|
WScope = new DbgScope(NULL, DIDescriptor(Scope), InlinedAt);
|
||||||
DbgScopeMap.insert(std::make_pair(InlinedAt, WScope));
|
DbgScopeMap.insert(std::make_pair(InlinedAt, WScope));
|
||||||
DILocation DL(InlinedAt);
|
DILocation DL(InlinedAt);
|
||||||
createDbgScope(DL.getScope().getNode(), DL.getOrigLocation().getNode());
|
DbgScope *Parent =
|
||||||
|
getOrCreateDbgScope(DL.getScope().getNode(), DL.getOrigLocation().getNode());
|
||||||
|
WScope->setParent(Parent);
|
||||||
|
Parent->addScope(WScope);
|
||||||
|
|
||||||
|
ConcreteScopes[InlinedAt] = WScope;
|
||||||
|
getOrCreateAbstractScope(Scope);
|
||||||
|
|
||||||
|
return WScope;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// hasValidLocation - Return true if debug location entry attached with
|
||||||
|
/// machine instruction encodes valid location info.
|
||||||
|
static bool hasValidLocation(LLVMContext &Ctx,
|
||||||
|
const MachineInstr *MInsn,
|
||||||
|
MDNode *&Scope, MDNode *&InlinedAt) {
|
||||||
|
if (MInsn->isDebugValue())
|
||||||
|
return false;
|
||||||
|
DebugLoc DL = MInsn->getDebugLoc();
|
||||||
|
if (DL.isUnknown()) return false;
|
||||||
|
|
||||||
|
MDNode *S = DL.getScope(Ctx);
|
||||||
|
|
||||||
|
// There is no need to create another DIE for compile unit. For all
|
||||||
|
// other scopes, create one DbgScope now. This will be translated
|
||||||
|
// into a scope DIE at the end.
|
||||||
|
if (DIScope(S).isCompileUnit()) return false;
|
||||||
|
|
||||||
|
Scope = S;
|
||||||
|
InlinedAt = DL.getInlinedAt(Ctx);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// calculateDominanceGraph - Calculate dominance graph for DbgScope
|
||||||
|
/// hierarchy.
|
||||||
|
static void calculateDominanceGraph(DbgScope *Scope) {
|
||||||
|
assert (Scope && "Unable to calculate scop edominance graph!");
|
||||||
|
SmallVector<DbgScope *, 4> WorkStack;
|
||||||
|
WorkStack.push_back(Scope);
|
||||||
|
unsigned Counter = 0;
|
||||||
|
while (!WorkStack.empty()) {
|
||||||
|
DbgScope *WS = WorkStack.back();
|
||||||
|
const SmallVector<DbgScope *, 4> &Children = WS->getScopes();
|
||||||
|
bool visitedChildren = false;
|
||||||
|
for (SmallVector<DbgScope *, 4>::const_iterator SI = Children.begin(),
|
||||||
|
SE = Children.end(); SI != SE; ++SI) {
|
||||||
|
DbgScope *ChildScope = *SI;
|
||||||
|
if (!ChildScope->getDFSOut()) {
|
||||||
|
WorkStack.push_back(ChildScope);
|
||||||
|
visitedChildren = true;
|
||||||
|
ChildScope->setDFSIn(++Counter);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!visitedChildren) {
|
||||||
|
WorkStack.pop_back();
|
||||||
|
WS->setDFSOut(++Counter);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// printDbgScopeInfo - Print DbgScope info for each machine instruction.
|
||||||
|
static
|
||||||
|
void printDbgScopeInfo(LLVMContext &Ctx, const MachineFunction *MF,
|
||||||
|
DenseMap<const MachineInstr *, DbgScope *> &MI2ScopeMap)
|
||||||
|
{
|
||||||
|
#ifndef NDEBUG
|
||||||
|
unsigned PrevDFSIn = 0;
|
||||||
|
for (MachineFunction::const_iterator I = MF->begin(), E = MF->end();
|
||||||
|
I != E; ++I) {
|
||||||
|
for (MachineBasicBlock::const_iterator II = I->begin(), IE = I->end();
|
||||||
|
II != IE; ++II) {
|
||||||
|
const MachineInstr *MInsn = II;
|
||||||
|
MDNode *Scope = NULL;
|
||||||
|
MDNode *InlinedAt = NULL;
|
||||||
|
|
||||||
|
// Check if instruction has valid location information.
|
||||||
|
if (hasValidLocation(Ctx, MInsn, Scope, InlinedAt)) {
|
||||||
|
dbgs() << " [ ";
|
||||||
|
if (InlinedAt)
|
||||||
|
dbgs() << "*";
|
||||||
|
DenseMap<const MachineInstr *, DbgScope *>::iterator DI =
|
||||||
|
MI2ScopeMap.find(MInsn);
|
||||||
|
if (DI != MI2ScopeMap.end()) {
|
||||||
|
DbgScope *S = DI->second;
|
||||||
|
dbgs() << S->getDFSIn();
|
||||||
|
PrevDFSIn = S->getDFSIn();
|
||||||
|
} else
|
||||||
|
dbgs() << PrevDFSIn;
|
||||||
|
} else
|
||||||
|
dbgs() << " [ x" << PrevDFSIn;
|
||||||
|
dbgs() << " ]";
|
||||||
|
MInsn->dump();
|
||||||
|
}
|
||||||
|
dbgs() << "\n";
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
/// extractScopeInformation - Scan machine instructions in this function
|
/// extractScopeInformation - Scan machine instructions in this function
|
||||||
/// and collect DbgScopes. Return true, if at least one scope was found.
|
/// and collect DbgScopes. Return true, if at least one scope was found.
|
||||||
bool DwarfDebug::extractScopeInformation() {
|
bool DwarfDebug::extractScopeInformation() {
|
||||||
@ -2193,75 +2306,95 @@ bool DwarfDebug::extractScopeInformation() {
|
|||||||
if (!DbgScopeMap.empty())
|
if (!DbgScopeMap.empty())
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
DenseMap<const MachineInstr *, unsigned> MIIndexMap;
|
|
||||||
unsigned MIIndex = 0;
|
|
||||||
LLVMContext &Ctx = Asm->MF->getFunction()->getContext();
|
|
||||||
|
|
||||||
// Scan each instruction and create scopes. First build working set of scopes.
|
// Scan each instruction and create scopes. First build working set of scopes.
|
||||||
|
LLVMContext &Ctx = Asm->MF->getFunction()->getContext();
|
||||||
|
SmallVector<DbgRange, 4> MIRanges;
|
||||||
|
DenseMap<const MachineInstr *, DbgScope *> MI2ScopeMap;
|
||||||
|
MDNode *PrevScope = NULL;
|
||||||
|
MDNode *PrevInlinedAt = NULL;
|
||||||
|
const MachineInstr *RangeBeginMI = NULL;
|
||||||
|
const MachineInstr *PrevMI = NULL;
|
||||||
for (MachineFunction::const_iterator I = Asm->MF->begin(), E = Asm->MF->end();
|
for (MachineFunction::const_iterator I = Asm->MF->begin(), E = Asm->MF->end();
|
||||||
I != E; ++I) {
|
I != E; ++I) {
|
||||||
for (MachineBasicBlock::const_iterator II = I->begin(), IE = I->end();
|
for (MachineBasicBlock::const_iterator II = I->begin(), IE = I->end();
|
||||||
II != IE; ++II) {
|
II != IE; ++II) {
|
||||||
const MachineInstr *MInsn = II;
|
const MachineInstr *MInsn = II;
|
||||||
// FIXME : Remove DBG_VALUE check.
|
MDNode *Scope = NULL;
|
||||||
if (MInsn->isDebugValue()) continue;
|
MDNode *InlinedAt = NULL;
|
||||||
MIIndexMap[MInsn] = MIIndex++;
|
|
||||||
|
|
||||||
DebugLoc DL = MInsn->getDebugLoc();
|
// Check if instruction has valid location information.
|
||||||
if (DL.isUnknown()) continue;
|
if (!hasValidLocation(Ctx, MInsn, Scope, InlinedAt)) {
|
||||||
|
PrevMI = MInsn;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
MDNode *Scope = DL.getScope(Ctx);
|
// If scope has not changed then skip this instruction.
|
||||||
|
if (Scope == PrevScope && PrevInlinedAt == InlinedAt) {
|
||||||
|
PrevMI = MInsn;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
// There is no need to create another DIE for compile unit. For all
|
if (RangeBeginMI) {
|
||||||
// other scopes, create one DbgScope now. This will be translated
|
// If we have alread seen a beginning of a instruction range and
|
||||||
// into a scope DIE at the end.
|
// current instruction scope does not match scope of first instruction
|
||||||
if (DIScope(Scope).isCompileUnit()) continue;
|
// in this range then create a new instruction range.
|
||||||
createDbgScope(Scope, DL.getInlinedAt(Ctx));
|
DbgRange R(RangeBeginMI, PrevMI);
|
||||||
|
MI2ScopeMap[RangeBeginMI] = getOrCreateDbgScope(PrevScope, PrevInlinedAt);
|
||||||
|
MIRanges.push_back(R);
|
||||||
|
}
|
||||||
|
|
||||||
|
// This is a beginning of a new instruction range.
|
||||||
|
RangeBeginMI = MInsn;
|
||||||
|
|
||||||
|
// Reset previous markers.
|
||||||
|
PrevMI = MInsn;
|
||||||
|
PrevScope = Scope;
|
||||||
|
PrevInlinedAt = InlinedAt;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Create last instruction range.
|
||||||
// Build scope hierarchy using working set of scopes.
|
if (RangeBeginMI && PrevMI && PrevScope) {
|
||||||
for (MachineFunction::const_iterator I = Asm->MF->begin(), E = Asm->MF->end();
|
DbgRange R(RangeBeginMI, PrevMI);
|
||||||
I != E; ++I) {
|
MIRanges.push_back(R);
|
||||||
for (MachineBasicBlock::const_iterator II = I->begin(), IE = I->end();
|
MI2ScopeMap[RangeBeginMI] = getOrCreateDbgScope(PrevScope, PrevInlinedAt);
|
||||||
II != IE; ++II) {
|
|
||||||
const MachineInstr *MInsn = II;
|
|
||||||
// FIXME : Remove DBG_VALUE check.
|
|
||||||
if (MInsn->isDebugValue()) continue;
|
|
||||||
DebugLoc DL = MInsn->getDebugLoc();
|
|
||||||
if (DL.isUnknown()) continue;
|
|
||||||
|
|
||||||
MDNode *Scope = DL.getScope(Ctx);
|
|
||||||
if (Scope == 0) continue;
|
|
||||||
|
|
||||||
// There is no need to create another DIE for compile unit. For all
|
|
||||||
// other scopes, create one DbgScope now. This will be translated
|
|
||||||
// into a scope DIE at the end.
|
|
||||||
if (DIScope(Scope).isCompileUnit()) continue;
|
|
||||||
DbgScope *DScope = getUpdatedDbgScope(Scope, MInsn, DL.getInlinedAt(Ctx));
|
|
||||||
DScope->setLastInsn(MInsn);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!CurrentFnDbgScope)
|
if (!CurrentFnDbgScope)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
CurrentFnDbgScope->fixInstructionMarkers(MIIndexMap);
|
calculateDominanceGraph(CurrentFnDbgScope);
|
||||||
|
if (PrintDbgScope)
|
||||||
|
printDbgScopeInfo(Ctx, Asm->MF, MI2ScopeMap);
|
||||||
|
|
||||||
|
// Find ranges of instructions covered by each DbgScope;
|
||||||
|
DbgScope *PrevDbgScope = NULL;
|
||||||
|
for (SmallVector<DbgRange, 4>::const_iterator RI = MIRanges.begin(),
|
||||||
|
RE = MIRanges.end(); RI != RE; ++RI) {
|
||||||
|
const DbgRange &R = *RI;
|
||||||
|
DbgScope *S = MI2ScopeMap.lookup(R.first);
|
||||||
|
assert (S && "Lost DbgScope for a machine instruction!");
|
||||||
|
if (PrevDbgScope && !PrevDbgScope->dominates(S))
|
||||||
|
PrevDbgScope->closeInsnRange(S);
|
||||||
|
S->openInsnRange(R.first);
|
||||||
|
S->extendInsnRange(R.second);
|
||||||
|
PrevDbgScope = S;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (PrevDbgScope)
|
||||||
|
PrevDbgScope->closeInsnRange();
|
||||||
|
|
||||||
identifyScopeMarkers();
|
identifyScopeMarkers();
|
||||||
|
|
||||||
return !DbgScopeMap.empty();
|
return !DbgScopeMap.empty();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// identifyScopeMarkers() - Indentify instructions that are marking
|
/// identifyScopeMarkers() -
|
||||||
/// beginning of or end of a scope.
|
/// Each DbgScope has first instruction and last instruction to mark beginning
|
||||||
|
/// and end of a scope respectively. Create an inverse map that list scopes
|
||||||
|
/// starts (and ends) with an instruction. One instruction may start (or end)
|
||||||
|
/// multiple scopes. Ignore scopes that are not reachable.
|
||||||
void DwarfDebug::identifyScopeMarkers() {
|
void DwarfDebug::identifyScopeMarkers() {
|
||||||
|
|
||||||
// Each scope has first instruction and last instruction to mark beginning
|
|
||||||
// and end of a scope respectively. Create an inverse map that list scopes
|
|
||||||
// starts (and ends) with an instruction. One instruction may start (or end)
|
|
||||||
// multiple scopes. Ignore scopes that are not reachable.
|
|
||||||
SmallVector<DbgScope *, 4> WorkList;
|
SmallVector<DbgScope *, 4> WorkList;
|
||||||
WorkList.push_back(CurrentFnDbgScope);
|
WorkList.push_back(CurrentFnDbgScope);
|
||||||
while (!WorkList.empty()) {
|
while (!WorkList.empty()) {
|
||||||
@ -2275,11 +2408,17 @@ void DwarfDebug::identifyScopeMarkers() {
|
|||||||
|
|
||||||
if (S->isAbstractScope())
|
if (S->isAbstractScope())
|
||||||
continue;
|
continue;
|
||||||
assert(S->getFirstInsn() && "DbgScope does not have first instruction!");
|
|
||||||
InsnsBeginScopeSet.insert(S->getFirstInsn());
|
|
||||||
|
|
||||||
assert(S->getLastInsn() && "DbgScope does not have last instruction!");
|
const SmallVector<DbgRange, 4> &Ranges = S->getRanges();
|
||||||
InsnsEndScopeSet.insert(S->getLastInsn());
|
if (Ranges.empty())
|
||||||
|
continue;
|
||||||
|
for (SmallVector<DbgRange, 4>::const_iterator RI = Ranges.begin(),
|
||||||
|
RE = Ranges.end(); RI != RE; ++RI) {
|
||||||
|
assert(RI->first && "DbgRange does not have first instruction!");
|
||||||
|
assert(RI->second && "DbgRange does not have second instruction!");
|
||||||
|
InsnsBeginScopeSet.insert(RI->first);
|
||||||
|
InsnsEndScopeSet.insert(RI->second);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2306,9 +2445,10 @@ void DwarfDebug::beginFunction(const MachineFunction *MF) {
|
|||||||
|
|
||||||
collectVariableInfo();
|
collectVariableInfo();
|
||||||
|
|
||||||
|
FunctionBeginSym = Asm->GetTempSymbol("func_begin",
|
||||||
|
Asm->getFunctionNumber());
|
||||||
// Assumes in correct section after the entry point.
|
// Assumes in correct section after the entry point.
|
||||||
Asm->OutStreamer.EmitLabel(Asm->GetTempSymbol("func_begin",
|
Asm->OutStreamer.EmitLabel(FunctionBeginSym);
|
||||||
Asm->getFunctionNumber()));
|
|
||||||
|
|
||||||
// Emit label for the implicitly defined dbg.stoppoint at the start of the
|
// Emit label for the implicitly defined dbg.stoppoint at the start of the
|
||||||
// function.
|
// function.
|
||||||
@ -2372,8 +2512,8 @@ void DwarfDebug::endFunction(const MachineFunction *MF) {
|
|||||||
DeleteContainerSeconds(AbstractScopes);
|
DeleteContainerSeconds(AbstractScopes);
|
||||||
AbstractScopesList.clear();
|
AbstractScopesList.clear();
|
||||||
AbstractVariables.clear();
|
AbstractVariables.clear();
|
||||||
InsnBeforeLabelMap.clear();
|
LabelsBeforeInsn.clear();
|
||||||
InsnAfterLabelMap.clear();
|
LabelsAfterInsn.clear();
|
||||||
Lines.clear();
|
Lines.clear();
|
||||||
PrevLabel = NULL;
|
PrevLabel = NULL;
|
||||||
}
|
}
|
||||||
@ -3088,14 +3228,14 @@ void DwarfDebug::emitDebugRanges() {
|
|||||||
// Start the dwarf ranges section.
|
// Start the dwarf ranges section.
|
||||||
Asm->OutStreamer.SwitchSection(
|
Asm->OutStreamer.SwitchSection(
|
||||||
Asm->getObjFileLowering().getDwarfRangesSection());
|
Asm->getObjFileLowering().getDwarfRangesSection());
|
||||||
for (SmallVector<const MCSymbol *, 8>::const_iterator I = DebugRangeSymbols.begin(),
|
unsigned char Size = Asm->getTargetData().getPointerSize();
|
||||||
E = DebugRangeSymbols.end(); I != E; ++I) {
|
for (SmallVector<const MCSymbol *, 8>::iterator
|
||||||
|
I = DebugRangeSymbols.begin(), E = DebugRangeSymbols.end();
|
||||||
|
I != E; ++I) {
|
||||||
if (*I)
|
if (*I)
|
||||||
Asm->EmitLabelDifference(*I, TextSectionSym,
|
Asm->OutStreamer.EmitSymbolValue(const_cast<MCSymbol*>(*I), Size, 0);
|
||||||
Asm->getTargetData().getPointerSize());
|
|
||||||
else
|
else
|
||||||
Asm->OutStreamer.EmitIntValue(0, Asm->getTargetData().getPointerSize(),
|
Asm->OutStreamer.EmitIntValue(0, Size, /*addrspace*/0);
|
||||||
/*addrspace*/0);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -188,13 +188,13 @@ class DwarfDebug {
|
|||||||
DenseMap<MDNode*, SmallVector<InlineInfoLabels, 4> > InlineInfo;
|
DenseMap<MDNode*, SmallVector<InlineInfoLabels, 4> > InlineInfo;
|
||||||
SmallVector<MDNode *, 4> InlinedSPNodes;
|
SmallVector<MDNode *, 4> InlinedSPNodes;
|
||||||
|
|
||||||
/// InsnBeforeLabelMap - Maps instruction with label emitted before
|
/// LabelsBeforeInsn - Maps instruction with label emitted before
|
||||||
/// instruction.
|
/// instruction.
|
||||||
DenseMap<const MachineInstr *, MCSymbol *> InsnBeforeLabelMap;
|
DenseMap<const MachineInstr *, MCSymbol *> LabelsBeforeInsn;
|
||||||
|
|
||||||
/// InsnAfterLabelMap - Maps instruction with label emitted after
|
/// LabelsAfterInsn - Maps instruction with label emitted after
|
||||||
/// instruction.
|
/// instruction.
|
||||||
DenseMap<const MachineInstr *, MCSymbol *> InsnAfterLabelMap;
|
DenseMap<const MachineInstr *, MCSymbol *> LabelsAfterInsn;
|
||||||
|
|
||||||
SmallVector<const MCSymbol *, 8> DebugRangeSymbols;
|
SmallVector<const MCSymbol *, 8> DebugRangeSymbols;
|
||||||
|
|
||||||
@ -219,6 +219,7 @@ class DwarfDebug {
|
|||||||
MCSymbol *DwarfFrameSectionSym, *DwarfInfoSectionSym, *DwarfAbbrevSectionSym;
|
MCSymbol *DwarfFrameSectionSym, *DwarfInfoSectionSym, *DwarfAbbrevSectionSym;
|
||||||
MCSymbol *DwarfStrSectionSym, *TextSectionSym, *DwarfDebugRangeSectionSym;
|
MCSymbol *DwarfStrSectionSym, *TextSectionSym, *DwarfDebugRangeSectionSym;
|
||||||
|
|
||||||
|
MCSymbol *FunctionBeginSym;
|
||||||
private:
|
private:
|
||||||
|
|
||||||
/// getSourceDirectoryAndFileIds - Return the directory and file ids that
|
/// getSourceDirectoryAndFileIds - Return the directory and file ids that
|
||||||
@ -368,13 +369,8 @@ private:
|
|||||||
/// createSubprogramDIE - Create new DIE using SP.
|
/// createSubprogramDIE - Create new DIE using SP.
|
||||||
DIE *createSubprogramDIE(const DISubprogram &SP, bool MakeDecl = false);
|
DIE *createSubprogramDIE(const DISubprogram &SP, bool MakeDecl = false);
|
||||||
|
|
||||||
/// getUpdatedDbgScope - Find or create DbgScope assicated with
|
/// getOrCreateDbgScope - Create DbgScope for the scope.
|
||||||
/// the instruction. Initialize scope and update scope hierarchy.
|
DbgScope *getOrCreateDbgScope(MDNode *Scope, MDNode *InlinedAt);
|
||||||
DbgScope *getUpdatedDbgScope(MDNode *N, const MachineInstr *MI,
|
|
||||||
MDNode *InlinedAt);
|
|
||||||
|
|
||||||
/// createDbgScope - Create DbgScope for the scope.
|
|
||||||
void createDbgScope(MDNode *Scope, MDNode *InlinedAt);
|
|
||||||
|
|
||||||
DbgScope *getOrCreateAbstractScope(MDNode *N);
|
DbgScope *getOrCreateAbstractScope(MDNode *N);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user