mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-01 00:33:09 +00:00
Change how dbg_value sdnodes are converted into machine instructions. Their placement should be determined by the relative order of incoming llvm instructions. The scheduler will now use the SDNode ordering information to determine where to insert them. A dbg_value instruction is inserted after the instruction with the last highest source order and before the instruction with the next highest source order. It will optimize the placement by inserting right after the instruction that produces the value if they have consecutive order numbers.
Here is a theoretical example that illustrates why the placement is important. tmp1 = store tmp1 -> x ... tmp2 = add ... ... call ... store tmp2 -> x Now mem2reg comes along: tmp1 = dbg_value (tmp1 -> x) ... tmp2 = add ... ... call ... dbg_value (tmp2 -> x) When the debugger examine the value of x after the add instruction but before the call, it should have the value of tmp1. Furthermore, for dbg_value's that reference constants, they should not be emitted at the beginning of the block (since they do not have "producers"). This patch also cleans up how SDISel manages DbgValue nodes. It allow a SDNode to be referenced by multiple SDDbgValue nodes. When a SDNode is deleted, it uses the information to find the SDDbgValues and invalidate them. They are not deleted until the corresponding SelectionDAG is destroyed. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@99469 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
7c3d45a03e
commit
bfcb305189
@ -285,6 +285,11 @@ public:
|
|||||||
IsEarlyClobber = Val;
|
IsEarlyClobber = Val;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setIsDebug(bool Val = true) {
|
||||||
|
assert(isReg() && IsDef && "Wrong MachineOperand accessor");
|
||||||
|
IsDebug = Val;
|
||||||
|
}
|
||||||
|
|
||||||
//===--------------------------------------------------------------------===//
|
//===--------------------------------------------------------------------===//
|
||||||
// Accessors for various operand types.
|
// Accessors for various operand types.
|
||||||
//===--------------------------------------------------------------------===//
|
//===--------------------------------------------------------------------===//
|
||||||
|
@ -60,42 +60,40 @@ private:
|
|||||||
|
|
||||||
/// SDDbgInfo - Keeps track of dbg_value information through SDISel. We do
|
/// SDDbgInfo - Keeps track of dbg_value information through SDISel. We do
|
||||||
/// not build SDNodes for these so as not to perturb the generated code;
|
/// not build SDNodes for these so as not to perturb the generated code;
|
||||||
/// instead the info is kept off to the side in this structure. SDNodes may
|
/// instead the info is kept off to the side in this structure. Each SDNode may
|
||||||
/// have an associated dbg_value entry in DbgValMap. Debug info that is not
|
/// have one or more associated dbg_value entries. This information is kept in
|
||||||
/// associated with any SDNode is held in DbgConstMap. It is possible for
|
/// DbgValMap.
|
||||||
/// optimizations to change a variable to a constant, in which case the
|
|
||||||
/// corresponding debug info is moved from the variable to the constant table
|
|
||||||
/// (NYI).
|
|
||||||
class SDDbgInfo {
|
class SDDbgInfo {
|
||||||
DenseMap<const SDNode*, SDDbgValue*> DbgVblMap;
|
SmallVector<SDDbgValue*, 32> DbgValues;
|
||||||
SmallVector<SDDbgValue*, 4> DbgConstMap;
|
DenseMap<const SDNode*, SmallVector<SDDbgValue*, 2> > DbgVblMap;
|
||||||
|
|
||||||
void operator=(const SDDbgInfo&); // Do not implement.
|
void operator=(const SDDbgInfo&); // Do not implement.
|
||||||
SDDbgInfo(const SDDbgInfo&); // Do not implement.
|
SDDbgInfo(const SDDbgInfo&); // Do not implement.
|
||||||
public:
|
public:
|
||||||
SDDbgInfo() {}
|
SDDbgInfo() {}
|
||||||
|
|
||||||
void add(const SDNode *Node, SDDbgValue *V) {
|
void add(SDDbgValue *V, const SDNode *Node = 0) {
|
||||||
DbgVblMap[Node] = V;
|
if (Node)
|
||||||
|
DbgVblMap[Node].push_back(V);
|
||||||
|
DbgValues.push_back(V);
|
||||||
}
|
}
|
||||||
void add(SDDbgValue *V) { DbgConstMap.push_back(V); }
|
|
||||||
void remove(const SDNode *Node) {
|
|
||||||
DenseMap<const SDNode*, SDDbgValue*>::iterator Itr =
|
|
||||||
DbgVblMap.find(Node);
|
|
||||||
if (Itr != DbgVblMap.end())
|
|
||||||
DbgVblMap.erase(Itr);
|
|
||||||
}
|
|
||||||
// No need to remove a constant.
|
|
||||||
void clear() {
|
void clear() {
|
||||||
DbgVblMap.clear();
|
DbgVblMap.clear();
|
||||||
DbgConstMap.clear();
|
DbgValues.clear();
|
||||||
}
|
}
|
||||||
SDDbgValue *getSDDbgValue(const SDNode *Node) {
|
|
||||||
|
bool empty() const {
|
||||||
|
return DbgValues.empty();
|
||||||
|
}
|
||||||
|
|
||||||
|
SmallVector<SDDbgValue*,2> &getSDDbgValues(const SDNode *Node) {
|
||||||
return DbgVblMap[Node];
|
return DbgVblMap[Node];
|
||||||
}
|
}
|
||||||
typedef SmallVector<SDDbgValue*, 4>::iterator ConstDbgIterator;
|
|
||||||
ConstDbgIterator DbgConstBegin() { return DbgConstMap.begin(); }
|
typedef SmallVector<SDDbgValue*,32>::iterator DbgIterator;
|
||||||
ConstDbgIterator DbgConstEnd() { return DbgConstMap.end(); }
|
DbgIterator DbgBegin() { return DbgValues.begin(); }
|
||||||
|
DbgIterator DbgEnd() { return DbgValues.end(); }
|
||||||
};
|
};
|
||||||
|
|
||||||
enum CombineLevel {
|
enum CombineLevel {
|
||||||
@ -871,19 +869,21 @@ public:
|
|||||||
/// GetOrdering - Get the order for the SDNode.
|
/// GetOrdering - Get the order for the SDNode.
|
||||||
unsigned GetOrdering(const SDNode *SD) const;
|
unsigned GetOrdering(const SDNode *SD) const;
|
||||||
|
|
||||||
/// AssignDbgInfo - Assign debug info to the SDNode.
|
/// AddDbgValue - Add a dbg_value SDNode. If SD is non-null that means the
|
||||||
void AssignDbgInfo(SDNode *SD, SDDbgValue *db);
|
/// value is produced by SD.
|
||||||
|
void AddDbgValue(SDDbgValue *DB, SDNode *SD = 0);
|
||||||
|
|
||||||
/// RememberDbgInfo - Remember debug info with no associated SDNode.
|
/// GetDbgValues - Get the debug values which reference the given SDNode.
|
||||||
void RememberDbgInfo(SDDbgValue *db);
|
SmallVector<SDDbgValue*,2> &GetDbgValues(const SDNode* SD) {
|
||||||
|
return DbgInfo->getSDDbgValues(SD);
|
||||||
/// GetDbgInfo - Get the debug info for the SDNode.
|
|
||||||
SDDbgValue *GetDbgInfo(const SDNode* SD);
|
|
||||||
|
|
||||||
SDDbgInfo::ConstDbgIterator DbgConstBegin() {
|
|
||||||
return DbgInfo->DbgConstBegin();
|
|
||||||
}
|
}
|
||||||
SDDbgInfo::ConstDbgIterator DbgConstEnd() { return DbgInfo->DbgConstEnd(); }
|
|
||||||
|
/// hasDebugValues - Return true if there are any SDDbgValue nodes associated
|
||||||
|
/// with this SelectionDAG.
|
||||||
|
bool hasDebugValues() const { return !DbgInfo->empty(); }
|
||||||
|
|
||||||
|
SDDbgInfo::DbgIterator DbgBegin() { return DbgInfo->DbgBegin(); }
|
||||||
|
SDDbgInfo::DbgIterator DbgEnd() { return DbgInfo->DbgEnd(); }
|
||||||
|
|
||||||
void dump() const;
|
void dump() const;
|
||||||
|
|
||||||
|
@ -264,7 +264,8 @@ void
|
|||||||
InstrEmitter::AddRegisterOperand(MachineInstr *MI, SDValue Op,
|
InstrEmitter::AddRegisterOperand(MachineInstr *MI, SDValue Op,
|
||||||
unsigned IIOpNum,
|
unsigned IIOpNum,
|
||||||
const TargetInstrDesc *II,
|
const TargetInstrDesc *II,
|
||||||
DenseMap<SDValue, unsigned> &VRBaseMap) {
|
DenseMap<SDValue, unsigned> &VRBaseMap,
|
||||||
|
bool IsDebug) {
|
||||||
assert(Op.getValueType() != MVT::Other &&
|
assert(Op.getValueType() != MVT::Other &&
|
||||||
Op.getValueType() != MVT::Flag &&
|
Op.getValueType() != MVT::Flag &&
|
||||||
"Chain and flag operands should occur at end of operand list!");
|
"Chain and flag operands should occur at end of operand list!");
|
||||||
@ -295,7 +296,11 @@ InstrEmitter::AddRegisterOperand(MachineInstr *MI, SDValue Op,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
MI->addOperand(MachineOperand::CreateReg(VReg, isOptDef));
|
MI->addOperand(MachineOperand::CreateReg(VReg, isOptDef,
|
||||||
|
false/*isImp*/, false/*isKill*/,
|
||||||
|
false/*isDead*/, false/*isUndef*/,
|
||||||
|
false/*isEarlyClobber*/,
|
||||||
|
0/*SubReg*/, IsDebug));
|
||||||
}
|
}
|
||||||
|
|
||||||
/// AddOperand - Add the specified operand to the specified machine instr. II
|
/// AddOperand - Add the specified operand to the specified machine instr. II
|
||||||
@ -305,9 +310,10 @@ InstrEmitter::AddRegisterOperand(MachineInstr *MI, SDValue Op,
|
|||||||
void InstrEmitter::AddOperand(MachineInstr *MI, SDValue Op,
|
void InstrEmitter::AddOperand(MachineInstr *MI, SDValue Op,
|
||||||
unsigned IIOpNum,
|
unsigned IIOpNum,
|
||||||
const TargetInstrDesc *II,
|
const TargetInstrDesc *II,
|
||||||
DenseMap<SDValue, unsigned> &VRBaseMap) {
|
DenseMap<SDValue, unsigned> &VRBaseMap,
|
||||||
|
bool IsDebug) {
|
||||||
if (Op.isMachineOpcode()) {
|
if (Op.isMachineOpcode()) {
|
||||||
AddRegisterOperand(MI, Op, IIOpNum, II, VRBaseMap);
|
AddRegisterOperand(MI, Op, IIOpNum, II, VRBaseMap, IsDebug);
|
||||||
} else if (ConstantSDNode *C = dyn_cast<ConstantSDNode>(Op)) {
|
} else if (ConstantSDNode *C = dyn_cast<ConstantSDNode>(Op)) {
|
||||||
MI->addOperand(MachineOperand::CreateImm(C->getSExtValue()));
|
MI->addOperand(MachineOperand::CreateImm(C->getSExtValue()));
|
||||||
} else if (ConstantFPSDNode *F = dyn_cast<ConstantFPSDNode>(Op)) {
|
} else if (ConstantFPSDNode *F = dyn_cast<ConstantFPSDNode>(Op)) {
|
||||||
@ -356,7 +362,7 @@ void InstrEmitter::AddOperand(MachineInstr *MI, SDValue Op,
|
|||||||
assert(Op.getValueType() != MVT::Other &&
|
assert(Op.getValueType() != MVT::Other &&
|
||||||
Op.getValueType() != MVT::Flag &&
|
Op.getValueType() != MVT::Flag &&
|
||||||
"Chain and flag operands should occur at end of operand list!");
|
"Chain and flag operands should occur at end of operand list!");
|
||||||
AddRegisterOperand(MI, Op, IIOpNum, II, VRBaseMap);
|
AddRegisterOperand(MI, Op, IIOpNum, II, VRBaseMap, IsDebug);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -498,75 +504,48 @@ InstrEmitter::EmitCopyToRegClassNode(SDNode *Node,
|
|||||||
assert(isNew && "Node emitted out of order - early");
|
assert(isNew && "Node emitted out of order - early");
|
||||||
}
|
}
|
||||||
|
|
||||||
/// EmitDbgValue - Generate any debug info that refers to this Node. Constant
|
/// EmitDbgValue - Generate machine instruction for a dbg_value node.
|
||||||
/// dbg_value is not handled here.
|
///
|
||||||
void
|
MachineInstr *InstrEmitter::EmitDbgValue(SDDbgValue *SD,
|
||||||
InstrEmitter::EmitDbgValue(SDNode *Node,
|
MachineBasicBlock *InsertBB,
|
||||||
DenseMap<SDValue, unsigned> &VRBaseMap,
|
DenseMap<SDValue, unsigned> &VRBaseMap,
|
||||||
SDDbgValue *sd) {
|
|
||||||
if (!Node->getHasDebugValue())
|
|
||||||
return;
|
|
||||||
if (!sd)
|
|
||||||
return;
|
|
||||||
assert(sd->getKind() == SDDbgValue::SDNODE);
|
|
||||||
unsigned VReg = getVR(SDValue(sd->getSDNode(), sd->getResNo()), VRBaseMap);
|
|
||||||
const TargetInstrDesc &II = TII->get(TargetOpcode::DBG_VALUE);
|
|
||||||
DebugLoc DL = sd->getDebugLoc();
|
|
||||||
MachineInstr *MI;
|
|
||||||
if (VReg) {
|
|
||||||
MI = BuildMI(*MF, DL, II).addReg(VReg, RegState::Debug).
|
|
||||||
addImm(sd->getOffset()).
|
|
||||||
addMetadata(sd->getMDPtr());
|
|
||||||
} else {
|
|
||||||
// Insert an Undef so we can see what we dropped.
|
|
||||||
MI = BuildMI(*MF, DL, II).addReg(0U).addImm(sd->getOffset()).
|
|
||||||
addMetadata(sd->getMDPtr());
|
|
||||||
}
|
|
||||||
MBB->insert(InsertPos, MI);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// EmitDbgValue - Generate debug info that does not refer to a SDNode.
|
|
||||||
void
|
|
||||||
InstrEmitter::EmitDbgValue(SDDbgValue *sd,
|
|
||||||
DenseMap<MachineBasicBlock*, MachineBasicBlock*> *EM) {
|
DenseMap<MachineBasicBlock*, MachineBasicBlock*> *EM) {
|
||||||
if (!sd)
|
uint64_t Offset = SD->getOffset();
|
||||||
return;
|
MDNode* MDPtr = SD->getMDPtr();
|
||||||
|
DebugLoc DL = SD->getDebugLoc();
|
||||||
|
|
||||||
const TargetInstrDesc &II = TII->get(TargetOpcode::DBG_VALUE);
|
const TargetInstrDesc &II = TII->get(TargetOpcode::DBG_VALUE);
|
||||||
uint64_t Offset = sd->getOffset();
|
MachineInstrBuilder MIB = BuildMI(*MF, DL, II);
|
||||||
MDNode* mdPtr = sd->getMDPtr();
|
if (SD->getKind() == SDDbgValue::SDNODE) {
|
||||||
SDDbgValue::DbgValueKind kind = sd->getKind();
|
AddOperand(&*MIB, SDValue(SD->getSDNode(), SD->getResNo()),
|
||||||
DebugLoc DL = sd->getDebugLoc();
|
(*MIB).getNumOperands(), &II, VRBaseMap, true /*IsDebug*/);
|
||||||
MachineInstr* MI;
|
} else if (SD->getKind() == SDDbgValue::CONST) {
|
||||||
if (kind == SDDbgValue::CONST) {
|
Value *V = SD->getConst();
|
||||||
Value *V = sd->getConst();
|
|
||||||
if (ConstantInt *CI = dyn_cast<ConstantInt>(V)) {
|
if (ConstantInt *CI = dyn_cast<ConstantInt>(V)) {
|
||||||
MI = BuildMI(*MF, DL, II).addImm(CI->getZExtValue()).
|
MIB.addImm(CI->getSExtValue());
|
||||||
addImm(Offset).addMetadata(mdPtr);
|
|
||||||
} else if (ConstantFP *CF = dyn_cast<ConstantFP>(V)) {
|
} else if (ConstantFP *CF = dyn_cast<ConstantFP>(V)) {
|
||||||
MI = BuildMI(*MF, DL, II).addFPImm(CF).
|
MIB.addFPImm(CF);
|
||||||
addImm(Offset).addMetadata(mdPtr);
|
|
||||||
} else {
|
} else {
|
||||||
// Could be an Undef. In any case insert an Undef so we can see what we
|
// Could be an Undef. In any case insert an Undef so we can see what we
|
||||||
// dropped.
|
// dropped.
|
||||||
MI = BuildMI(*MF, DL, II).addReg(0U).
|
MIB.addReg(0U);
|
||||||
addImm(Offset).addMetadata(mdPtr);
|
|
||||||
}
|
}
|
||||||
} else if (kind == SDDbgValue::FRAMEIX) {
|
} else if (SD->getKind() == SDDbgValue::FRAMEIX) {
|
||||||
unsigned FrameIx = sd->getFrameIx();
|
unsigned FrameIx = SD->getFrameIx();
|
||||||
// Stack address; this needs to be lowered in target-dependent fashion.
|
// Stack address; this needs to be lowered in target-dependent fashion.
|
||||||
// FIXME test that the target supports this somehow; if not emit Undef.
|
// FIXME test that the target supports this somehow; if not emit Undef.
|
||||||
// Create a pseudo for EmitInstrWithCustomInserter's consumption.
|
// Create a pseudo for EmitInstrWithCustomInserter's consumption.
|
||||||
MI = BuildMI(*MF, DL, II).addImm(FrameIx).
|
MIB.addImm(FrameIx).addImm(Offset).addMetadata(MDPtr);
|
||||||
addImm(Offset).addMetadata(mdPtr);
|
abort();
|
||||||
MBB = TLI->EmitInstrWithCustomInserter(MI, MBB, EM);
|
TLI->EmitInstrWithCustomInserter(&*MIB, InsertBB, EM);
|
||||||
InsertPos = MBB->end();
|
return 0;
|
||||||
return;
|
|
||||||
} else {
|
} else {
|
||||||
// Insert an Undef so we can see what we dropped.
|
// Insert an Undef so we can see what we dropped.
|
||||||
MI = BuildMI(*MF, DL, II).addReg(0U).
|
MIB.addReg(0U);
|
||||||
addImm(Offset).addMetadata(mdPtr);
|
|
||||||
}
|
}
|
||||||
MBB->insert(InsertPos, MI);
|
|
||||||
|
MIB.addImm(Offset).addMetadata(MDPtr);
|
||||||
|
return &*MIB;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// EmitNode - Generate machine code for a node and needed dependencies.
|
/// EmitNode - Generate machine code for a node and needed dependencies.
|
||||||
|
@ -64,7 +64,8 @@ class InstrEmitter {
|
|||||||
void AddRegisterOperand(MachineInstr *MI, SDValue Op,
|
void AddRegisterOperand(MachineInstr *MI, SDValue Op,
|
||||||
unsigned IIOpNum,
|
unsigned IIOpNum,
|
||||||
const TargetInstrDesc *II,
|
const TargetInstrDesc *II,
|
||||||
DenseMap<SDValue, unsigned> &VRBaseMap);
|
DenseMap<SDValue, unsigned> &VRBaseMap,
|
||||||
|
bool IsDebug = false);
|
||||||
|
|
||||||
/// AddOperand - Add the specified operand to the specified machine instr. II
|
/// AddOperand - Add the specified operand to the specified machine instr. II
|
||||||
/// specifies the instruction information for the node, and IIOpNum is the
|
/// specifies the instruction information for the node, and IIOpNum is the
|
||||||
@ -73,7 +74,8 @@ class InstrEmitter {
|
|||||||
void AddOperand(MachineInstr *MI, SDValue Op,
|
void AddOperand(MachineInstr *MI, SDValue Op,
|
||||||
unsigned IIOpNum,
|
unsigned IIOpNum,
|
||||||
const TargetInstrDesc *II,
|
const TargetInstrDesc *II,
|
||||||
DenseMap<SDValue, unsigned> &VRBaseMap);
|
DenseMap<SDValue, unsigned> &VRBaseMap,
|
||||||
|
bool IsDebug = false);
|
||||||
|
|
||||||
/// EmitSubregNode - Generate machine code for subreg nodes.
|
/// EmitSubregNode - Generate machine code for subreg nodes.
|
||||||
///
|
///
|
||||||
@ -98,16 +100,12 @@ public:
|
|||||||
/// MachineInstr.
|
/// MachineInstr.
|
||||||
static unsigned CountOperands(SDNode *Node);
|
static unsigned CountOperands(SDNode *Node);
|
||||||
|
|
||||||
/// EmitDbgValue - Generate any debug info that refers to this Node. Constant
|
/// EmitDbgValue - Generate machine instruction for a dbg_value node.
|
||||||
/// dbg_value is not handled here.
|
///
|
||||||
void EmitDbgValue(SDNode *Node,
|
MachineInstr *EmitDbgValue(SDDbgValue *SD,
|
||||||
DenseMap<SDValue, unsigned> &VRBaseMap,
|
MachineBasicBlock *InsertBB,
|
||||||
SDDbgValue* sd);
|
DenseMap<SDValue, unsigned> &VRBaseMap,
|
||||||
|
DenseMap<MachineBasicBlock*, MachineBasicBlock*> *EM);
|
||||||
|
|
||||||
/// EmitDbgValue - Generate a constant DBG_VALUE. No node is involved.
|
|
||||||
void EmitDbgValue(SDDbgValue* sd,
|
|
||||||
DenseMap<MachineBasicBlock*, MachineBasicBlock*> *EM);
|
|
||||||
|
|
||||||
/// EmitNode - Generate machine code for a node and needed dependencies.
|
/// EmitNode - Generate machine code for a node and needed dependencies.
|
||||||
///
|
///
|
||||||
|
@ -47,10 +47,12 @@ private:
|
|||||||
uint64_t Offset;
|
uint64_t Offset;
|
||||||
DebugLoc DL;
|
DebugLoc DL;
|
||||||
unsigned Order;
|
unsigned Order;
|
||||||
|
bool Invalid;
|
||||||
public:
|
public:
|
||||||
// Constructor for non-constants.
|
// Constructor for non-constants.
|
||||||
SDDbgValue(MDNode *mdP, SDNode *N, unsigned R, uint64_t off, DebugLoc dl,
|
SDDbgValue(MDNode *mdP, SDNode *N, unsigned R, uint64_t off, DebugLoc dl,
|
||||||
unsigned O) : mdPtr(mdP), Offset(off), DL(dl), Order(O) {
|
unsigned O) : mdPtr(mdP), Offset(off), DL(dl), Order(O),
|
||||||
|
Invalid(false) {
|
||||||
kind = SDNODE;
|
kind = SDNODE;
|
||||||
u.s.Node = N;
|
u.s.Node = N;
|
||||||
u.s.ResNo = R;
|
u.s.ResNo = R;
|
||||||
@ -97,6 +99,12 @@ public:
|
|||||||
// Returns the SDNodeOrder. This is the order of the preceding node in the
|
// Returns the SDNodeOrder. This is the order of the preceding node in the
|
||||||
// input.
|
// input.
|
||||||
unsigned getOrder() { return Order; }
|
unsigned getOrder() { return Order; }
|
||||||
|
|
||||||
|
// setIsInvalidated / isInvalidated - Setter / getter of the "Invalidated"
|
||||||
|
// property. A SDDbgValue is invalid if the SDNode that produces the value is
|
||||||
|
// deleted.
|
||||||
|
void setIsInvalidated() { Invalid = true; }
|
||||||
|
bool isInvalidated() { return Invalid; }
|
||||||
};
|
};
|
||||||
|
|
||||||
} // end llvm namespace
|
} // end llvm namespace
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
#include "llvm/Target/TargetSubtarget.h"
|
#include "llvm/Target/TargetSubtarget.h"
|
||||||
#include "llvm/ADT/DenseMap.h"
|
#include "llvm/ADT/DenseMap.h"
|
||||||
#include "llvm/ADT/SmallPtrSet.h"
|
#include "llvm/ADT/SmallPtrSet.h"
|
||||||
|
#include "llvm/ADT/SmallSet.h"
|
||||||
#include "llvm/ADT/SmallVector.h"
|
#include "llvm/ADT/SmallVector.h"
|
||||||
#include "llvm/ADT/Statistic.h"
|
#include "llvm/ADT/Statistic.h"
|
||||||
#include "llvm/Support/Debug.h"
|
#include "llvm/Support/Debug.h"
|
||||||
@ -407,19 +408,67 @@ void ScheduleDAGSDNodes::dumpNode(const SUnit *SU) const {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
struct OrderSorter {
|
||||||
|
bool operator()(const std::pair<unsigned, MachineInstr*> &A,
|
||||||
|
const std::pair<unsigned, MachineInstr*> &B) {
|
||||||
|
return A.first < B.first;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// ProcessSourceNode - Process nodes with source order numbers. These are added
|
||||||
|
// to a vector which EmitSchedule use to determine how to insert dbg_value
|
||||||
|
// instructions in the right order.
|
||||||
|
static void ProcessSourceNode(SDNode *N, SelectionDAG *DAG,
|
||||||
|
InstrEmitter &Emitter,
|
||||||
|
DenseMap<MachineBasicBlock*, MachineBasicBlock*> *EM,
|
||||||
|
DenseMap<SDValue, unsigned> &VRBaseMap,
|
||||||
|
SmallVector<std::pair<unsigned, MachineInstr*>, 32> &Orders,
|
||||||
|
SmallSet<unsigned, 8> &Seen) {
|
||||||
|
unsigned Order = DAG->GetOrdering(N);
|
||||||
|
if (!Order || !Seen.insert(Order))
|
||||||
|
return;
|
||||||
|
|
||||||
|
MachineBasicBlock *BB = Emitter.getBlock();
|
||||||
|
if (BB->empty() || BB->back().isPHI()) {
|
||||||
|
// Did not insert any instruction.
|
||||||
|
Orders.push_back(std::make_pair(Order, (MachineInstr*)0));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Orders.push_back(std::make_pair(Order, &BB->back()));
|
||||||
|
if (!N->getHasDebugValue())
|
||||||
|
return;
|
||||||
|
// Opportunistically insert immediate dbg_value uses, i.e. those with source
|
||||||
|
// order number right after the N.
|
||||||
|
MachineBasicBlock::iterator InsertPos = Emitter.getInsertPos();
|
||||||
|
SmallVector<SDDbgValue*,2> &DVs = DAG->GetDbgValues(N);
|
||||||
|
for (unsigned i = 0, e = DVs.size(); i != e; ++i) {
|
||||||
|
if (DVs[i]->isInvalidated())
|
||||||
|
continue;
|
||||||
|
unsigned DVOrder = DVs[i]->getOrder();
|
||||||
|
if (DVOrder == ++Order) {
|
||||||
|
// FIXME: If the source node with next higher order is scheduled before
|
||||||
|
// this could end up generating funky debug info.
|
||||||
|
MachineInstr *DbgMI = Emitter.EmitDbgValue(DVs[i], BB, VRBaseMap, EM);
|
||||||
|
Orders.push_back(std::make_pair(DVOrder, DbgMI));
|
||||||
|
BB->insert(InsertPos, DbgMI);
|
||||||
|
DVs[i]->setIsInvalidated();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/// EmitSchedule - Emit the machine code in scheduled order.
|
/// EmitSchedule - Emit the machine code in scheduled order.
|
||||||
MachineBasicBlock *ScheduleDAGSDNodes::
|
MachineBasicBlock *ScheduleDAGSDNodes::
|
||||||
EmitSchedule(DenseMap<MachineBasicBlock*, MachineBasicBlock*> *EM) {
|
EmitSchedule(DenseMap<MachineBasicBlock*, MachineBasicBlock*> *EM) {
|
||||||
InstrEmitter Emitter(BB, InsertPos);
|
InstrEmitter Emitter(BB, InsertPos);
|
||||||
DenseMap<SDValue, unsigned> VRBaseMap;
|
DenseMap<SDValue, unsigned> VRBaseMap;
|
||||||
DenseMap<SUnit*, unsigned> CopyVRBaseMap;
|
DenseMap<SUnit*, unsigned> CopyVRBaseMap;
|
||||||
|
SmallVector<std::pair<unsigned, MachineInstr*>, 32> Orders;
|
||||||
// For now, any constant debug info nodes go at the beginning.
|
SmallSet<unsigned, 8> Seen;
|
||||||
for (SDDbgInfo::ConstDbgIterator I = DAG->DbgConstBegin(),
|
bool HasDbg = DAG->hasDebugValues();
|
||||||
E = DAG->DbgConstEnd(); I!=E; I++) {
|
|
||||||
Emitter.EmitDbgValue(*I, EM);
|
|
||||||
delete *I;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (unsigned i = 0, e = Sequence.size(); i != e; i++) {
|
for (unsigned i = 0, e = Sequence.size(); i != e; i++) {
|
||||||
SUnit *SU = Sequence[i];
|
SUnit *SU = Sequence[i];
|
||||||
@ -442,22 +491,72 @@ EmitSchedule(DenseMap<MachineBasicBlock*, MachineBasicBlock*> *EM) {
|
|||||||
N = N->getFlaggedNode())
|
N = N->getFlaggedNode())
|
||||||
FlaggedNodes.push_back(N);
|
FlaggedNodes.push_back(N);
|
||||||
while (!FlaggedNodes.empty()) {
|
while (!FlaggedNodes.empty()) {
|
||||||
|
SDNode *N = FlaggedNodes.back();
|
||||||
Emitter.EmitNode(FlaggedNodes.back(), SU->OrigNode != SU, SU->isCloned,
|
Emitter.EmitNode(FlaggedNodes.back(), SU->OrigNode != SU, SU->isCloned,
|
||||||
VRBaseMap, EM);
|
VRBaseMap, EM);
|
||||||
if (FlaggedNodes.back()->getHasDebugValue())
|
// Remember the the source order of the inserted instruction.
|
||||||
if (SDDbgValue *sd = DAG->GetDbgInfo(FlaggedNodes.back())) {
|
if (HasDbg)
|
||||||
Emitter.EmitDbgValue(FlaggedNodes.back(), VRBaseMap, sd);
|
ProcessSourceNode(N, DAG, Emitter, EM, VRBaseMap, Orders, Seen);
|
||||||
delete sd;
|
|
||||||
}
|
|
||||||
FlaggedNodes.pop_back();
|
FlaggedNodes.pop_back();
|
||||||
}
|
}
|
||||||
Emitter.EmitNode(SU->getNode(), SU->OrigNode != SU, SU->isCloned,
|
Emitter.EmitNode(SU->getNode(), SU->OrigNode != SU, SU->isCloned,
|
||||||
VRBaseMap, EM);
|
VRBaseMap, EM);
|
||||||
if (SU->getNode()->getHasDebugValue())
|
// Remember the the source order of the inserted instruction.
|
||||||
if (SDDbgValue *sd = DAG->GetDbgInfo(SU->getNode())) {
|
if (HasDbg)
|
||||||
Emitter.EmitDbgValue(SU->getNode(), VRBaseMap, sd);
|
ProcessSourceNode(SU->getNode(), DAG, Emitter, EM, VRBaseMap, Orders,
|
||||||
delete sd;
|
Seen);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Insert all the dbg_value which have not already been inserted in source
|
||||||
|
// order sequence.
|
||||||
|
if (HasDbg) {
|
||||||
|
MachineBasicBlock::iterator BBBegin = BB->empty() ? BB->end() : BB->begin();
|
||||||
|
while (BBBegin != BB->end() && BBBegin->isPHI())
|
||||||
|
++BBBegin;
|
||||||
|
|
||||||
|
// Sort the source order instructions and use the order to insert debug
|
||||||
|
// values.
|
||||||
|
std::sort(Orders.begin(), Orders.end(), OrderSorter());
|
||||||
|
|
||||||
|
SDDbgInfo::DbgIterator DI = DAG->DbgBegin();
|
||||||
|
SDDbgInfo::DbgIterator DE = DAG->DbgEnd();
|
||||||
|
// Now emit the rest according to source order.
|
||||||
|
unsigned LastOrder = 0;
|
||||||
|
MachineInstr *LastMI = 0;
|
||||||
|
for (unsigned i = 0, e = Orders.size(); i != e && DI != DE; ++i) {
|
||||||
|
unsigned Order = Orders[i].first;
|
||||||
|
MachineInstr *MI = Orders[i].second;
|
||||||
|
// Insert all SDDbgValue's whose order(s) are before "Order".
|
||||||
|
if (!MI)
|
||||||
|
continue;
|
||||||
|
MachineBasicBlock *MIBB = MI->getParent();
|
||||||
|
for (; DI != DE &&
|
||||||
|
(*DI)->getOrder() >= LastOrder && (*DI)->getOrder() < Order; ++DI) {
|
||||||
|
if ((*DI)->isInvalidated())
|
||||||
|
continue;
|
||||||
|
MachineInstr *DbgMI = Emitter.EmitDbgValue(*DI, MIBB, VRBaseMap, EM);
|
||||||
|
if (!LastOrder)
|
||||||
|
// Insert to start of the BB (after PHIs).
|
||||||
|
BB->insert(BBBegin, DbgMI);
|
||||||
|
else {
|
||||||
|
MachineBasicBlock::iterator Pos = MI;
|
||||||
|
MIBB->insert(llvm::next(Pos), DbgMI);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
LastOrder = Order;
|
||||||
|
LastMI = MI;
|
||||||
|
}
|
||||||
|
// Add trailing DbgValue's before the terminator. FIXME: May want to add
|
||||||
|
// some of them before one or more conditional branches?
|
||||||
|
while (DI != DE) {
|
||||||
|
MachineBasicBlock *InsertBB = Emitter.getBlock();
|
||||||
|
MachineBasicBlock::iterator Pos= Emitter.getBlock()->getFirstTerminator();
|
||||||
|
if (!(*DI)->isInvalidated()) {
|
||||||
|
MachineInstr *DbgMI= Emitter.EmitDbgValue(*DI, InsertBB, VRBaseMap, EM);
|
||||||
|
InsertBB->insert(Pos, DbgMI);
|
||||||
|
}
|
||||||
|
++DI;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
BB = Emitter.getBlock();
|
BB = Emitter.getBlock();
|
||||||
|
@ -598,8 +598,10 @@ void SelectionDAG::DeallocateNode(SDNode *N) {
|
|||||||
// Remove the ordering of this node.
|
// Remove the ordering of this node.
|
||||||
Ordering->remove(N);
|
Ordering->remove(N);
|
||||||
|
|
||||||
// And its entry in the debug info table, if any.
|
// If any of the SDDbgValue nodes refer to this SDNode, invalidate them.
|
||||||
DbgInfo->remove(N);
|
SmallVector<SDDbgValue*, 2> &DbgVals = DbgInfo->getSDDbgValues(N);
|
||||||
|
for (unsigned i = 0, e = DbgVals.size(); i != e; ++i)
|
||||||
|
DbgVals[i]->setIsInvalidated();
|
||||||
}
|
}
|
||||||
|
|
||||||
/// RemoveNodeFromCSEMaps - Take the specified node out of the CSE map that
|
/// RemoveNodeFromCSEMaps - Take the specified node out of the CSE map that
|
||||||
@ -811,6 +813,7 @@ void SelectionDAG::init(MachineFunction &mf, MachineModuleInfo *mmi,
|
|||||||
SelectionDAG::~SelectionDAG() {
|
SelectionDAG::~SelectionDAG() {
|
||||||
allnodes_clear();
|
allnodes_clear();
|
||||||
delete Ordering;
|
delete Ordering;
|
||||||
|
DbgInfo->clear();
|
||||||
delete DbgInfo;
|
delete DbgInfo;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -5241,24 +5244,12 @@ unsigned SelectionDAG::GetOrdering(const SDNode *SD) const {
|
|||||||
return Ordering->getOrder(SD);
|
return Ordering->getOrder(SD);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// AssignDbgInfo - Assign debug info to the SDNode.
|
/// AddDbgValue - Add a dbg_value SDNode. If SD is non-null that means the
|
||||||
void SelectionDAG::AssignDbgInfo(SDNode* SD, SDDbgValue* db) {
|
/// value is produced by SD.
|
||||||
assert(SD && "Trying to assign dbg info to a null node!");
|
void SelectionDAG::AddDbgValue(SDDbgValue *DB, SDNode *SD) {
|
||||||
DbgInfo->add(SD, db);
|
DbgInfo->add(DB, SD);
|
||||||
SD->setHasDebugValue(true);
|
if (SD)
|
||||||
}
|
SD->setHasDebugValue(true);
|
||||||
|
|
||||||
/// RememberDbgInfo - Remember debug info which is not assigned to an SDNode.
|
|
||||||
void SelectionDAG::RememberDbgInfo(SDDbgValue* db) {
|
|
||||||
DbgInfo->add(db);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// GetDbgInfo - Get the debug info, if any, for the SDNode.
|
|
||||||
SDDbgValue* SelectionDAG::GetDbgInfo(const SDNode *SD) {
|
|
||||||
assert(SD && "Trying to get the order of a null node!");
|
|
||||||
if (SD->getHasDebugValue())
|
|
||||||
return DbgInfo->getSDDbgValue(SD);
|
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
@ -3825,20 +3825,20 @@ SelectionDAGBuilder::visitIntrinsicCall(CallInst &I, unsigned Intrinsic) {
|
|||||||
++SDNodeOrder;
|
++SDNodeOrder;
|
||||||
if (isa<ConstantInt>(V) || isa<ConstantFP>(V)) {
|
if (isa<ConstantInt>(V) || isa<ConstantFP>(V)) {
|
||||||
SDDbgValue* dv = new SDDbgValue(Variable, V, Offset, dl, SDNodeOrder);
|
SDDbgValue* dv = new SDDbgValue(Variable, V, Offset, dl, SDNodeOrder);
|
||||||
DAG.RememberDbgInfo(dv);
|
DAG.AddDbgValue(dv);
|
||||||
} else {
|
} else {
|
||||||
SDValue &N = NodeMap[V];
|
SDValue &N = NodeMap[V];
|
||||||
if (N.getNode()) {
|
if (N.getNode()) {
|
||||||
SDDbgValue *dv = new SDDbgValue(Variable, N.getNode(),
|
SDDbgValue *dv = new SDDbgValue(Variable, N.getNode(),
|
||||||
N.getResNo(), Offset, dl, SDNodeOrder);
|
N.getResNo(), Offset, dl, SDNodeOrder);
|
||||||
DAG.AssignDbgInfo(N.getNode(), dv);
|
DAG.AddDbgValue(dv, N.getNode());
|
||||||
} else {
|
} else {
|
||||||
// We may expand this to cover more cases. One case where we have no
|
// We may expand this to cover more cases. One case where we have no
|
||||||
// data available is an unreferenced parameter; we need this fallback.
|
// data available is an unreferenced parameter; we need this fallback.
|
||||||
SDDbgValue* dv = new SDDbgValue(Variable,
|
SDDbgValue* dv = new SDDbgValue(Variable,
|
||||||
UndefValue::get(V->getType()),
|
UndefValue::get(V->getType()),
|
||||||
Offset, dl, SDNodeOrder);
|
Offset, dl, SDNodeOrder);
|
||||||
DAG.RememberDbgInfo(dv);
|
DAG.AddDbgValue(dv);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user