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 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263
|
//===-- llvm/CodeGen/SDNodeDbgValue.h - SelectionDAG dbg_value --*- C++ -*-===//
//
// 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
//
//===----------------------------------------------------------------------===//
//
// This file declares the SDDbgValue class.
//
//===----------------------------------------------------------------------===//
#ifndef LLVM_LIB_CODEGEN_SELECTIONDAG_SDNODEDBGVALUE_H
#define LLVM_LIB_CODEGEN_SELECTIONDAG_SDNODEDBGVALUE_H
#include "llvm/IR/DebugLoc.h"
#include "llvm/Support/DataTypes.h"
#include <utility>
namespace llvm {
class DIVariable;
class DIExpression;
class SDNode;
class Value;
class raw_ostream;
/// Holds the information for a single machine location through SDISel; either
/// an SDNode, a constant, a stack location, or a virtual register.
class SDDbgOperand {
public:
enum Kind {
SDNODE = 0, ///< Value is the result of an expression.
CONST = 1, ///< Value is a constant.
FRAMEIX = 2, ///< Value is contents of a stack location.
VREG = 3 ///< Value is a virtual register.
};
Kind getKind() const { return kind; }
/// Returns the SDNode* for a register ref
SDNode *getSDNode() const {
assert(kind == SDNODE);
return u.s.Node;
}
/// Returns the ResNo for a register ref
unsigned getResNo() const {
assert(kind == SDNODE);
return u.s.ResNo;
}
/// Returns the Value* for a constant
const Value *getConst() const {
assert(kind == CONST);
return u.Const;
}
/// Returns the FrameIx for a stack object
unsigned getFrameIx() const {
assert(kind == FRAMEIX);
return u.FrameIx;
}
/// Returns the Virtual Register for a VReg
unsigned getVReg() const {
assert(kind == VREG);
return u.VReg;
}
static SDDbgOperand fromNode(SDNode *Node, unsigned ResNo) {
return SDDbgOperand(Node, ResNo);
}
static SDDbgOperand fromFrameIdx(unsigned FrameIdx) {
return SDDbgOperand(FrameIdx, FRAMEIX);
}
static SDDbgOperand fromVReg(unsigned VReg) {
return SDDbgOperand(VReg, VREG);
}
static SDDbgOperand fromConst(const Value *Const) {
return SDDbgOperand(Const);
}
bool operator!=(const SDDbgOperand &Other) const { return !(*this == Other); }
bool operator==(const SDDbgOperand &Other) const {
if (kind != Other.kind)
return false;
switch (kind) {
case SDNODE:
return getSDNode() == Other.getSDNode() && getResNo() == Other.getResNo();
case CONST:
return getConst() == Other.getConst();
case VREG:
return getVReg() == Other.getVReg();
case FRAMEIX:
return getFrameIx() == Other.getFrameIx();
}
return false;
}
private:
Kind kind;
union {
struct {
SDNode *Node; ///< Valid for expressions.
unsigned ResNo; ///< Valid for expressions.
} s;
const Value *Const; ///< Valid for constants.
unsigned FrameIx; ///< Valid for stack objects.
unsigned VReg; ///< Valid for registers.
} u;
/// Constructor for non-constants.
SDDbgOperand(SDNode *N, unsigned R) : kind(SDNODE) {
u.s.Node = N;
u.s.ResNo = R;
}
/// Constructor for constants.
SDDbgOperand(const Value *C) : kind(CONST) { u.Const = C; }
/// Constructor for virtual registers and frame indices.
SDDbgOperand(unsigned VRegOrFrameIdx, Kind Kind) : kind(Kind) {
assert((Kind == VREG || Kind == FRAMEIX) &&
"Invalid SDDbgValue constructor");
if (kind == VREG)
u.VReg = VRegOrFrameIdx;
else
u.FrameIx = VRegOrFrameIdx;
}
};
/// Holds the information from a dbg_value node through SDISel.
/// We do not use SDValue here to avoid including its header.
class SDDbgValue {
public:
private:
// SDDbgValues are allocated by a BumpPtrAllocator, which means the destructor
// may not be called; therefore all member arrays must also be allocated by
// that BumpPtrAllocator, to ensure that they are correctly freed.
size_t NumLocationOps;
SDDbgOperand *LocationOps;
// SDNode dependencies will be calculated as SDNodes that appear in
// LocationOps plus these AdditionalDependencies.
size_t NumAdditionalDependencies;
SDNode **AdditionalDependencies;
DIVariable *Var;
DIExpression *Expr;
DebugLoc DL;
unsigned Order;
bool IsIndirect;
bool IsVariadic;
bool Invalid = false;
bool Emitted = false;
public:
SDDbgValue(BumpPtrAllocator &Alloc, DIVariable *Var, DIExpression *Expr,
ArrayRef<SDDbgOperand> L, ArrayRef<SDNode *> Dependencies,
bool IsIndirect, DebugLoc DL, unsigned O, bool IsVariadic)
: NumLocationOps(L.size()),
LocationOps(Alloc.Allocate<SDDbgOperand>(L.size())),
NumAdditionalDependencies(Dependencies.size()),
AdditionalDependencies(Alloc.Allocate<SDNode *>(Dependencies.size())),
Var(Var), Expr(Expr), DL(DL), Order(O), IsIndirect(IsIndirect),
IsVariadic(IsVariadic) {
assert(IsVariadic || L.size() == 1);
assert(!(IsVariadic && IsIndirect));
std::copy(L.begin(), L.end(), LocationOps);
std::copy(Dependencies.begin(), Dependencies.end(), AdditionalDependencies);
}
// We allocate arrays with the BumpPtrAllocator and never free or copy them,
// for LocationOps and AdditionalDependencies, as we never expect to copy or
// destroy an SDDbgValue. If we ever start copying or destroying instances, we
// should manage the allocated memory appropriately.
SDDbgValue(const SDDbgValue &Other) = delete;
SDDbgValue &operator=(const SDDbgValue &Other) = delete;
~SDDbgValue() = delete;
/// Returns the DIVariable pointer for the variable.
DIVariable *getVariable() const { return Var; }
/// Returns the DIExpression pointer for the expression.
DIExpression *getExpression() const { return Expr; }
ArrayRef<SDDbgOperand> getLocationOps() const {
return ArrayRef<SDDbgOperand>(LocationOps, NumLocationOps);
}
SmallVector<SDDbgOperand> copyLocationOps() const {
return SmallVector<SDDbgOperand>(LocationOps, LocationOps + NumLocationOps);
}
// Returns the SDNodes which this SDDbgValue depends on.
SmallVector<SDNode *> getSDNodes() const {
SmallVector<SDNode *> Dependencies;
for (const SDDbgOperand &DbgOp : getLocationOps())
if (DbgOp.getKind() == SDDbgOperand::SDNODE)
Dependencies.push_back(DbgOp.getSDNode());
for (SDNode *Node : getAdditionalDependencies())
Dependencies.push_back(Node);
return Dependencies;
}
ArrayRef<SDNode *> getAdditionalDependencies() const {
return ArrayRef<SDNode *>(AdditionalDependencies,
NumAdditionalDependencies);
}
/// Returns whether this is an indirect value.
bool isIndirect() const { return IsIndirect; }
bool isVariadic() const { return IsVariadic; }
/// Returns the DebugLoc.
const DebugLoc &getDebugLoc() const { return DL; }
/// Returns the SDNodeOrder. This is the order of the preceding node in the
/// input.
unsigned getOrder() const { 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() const { return Invalid; }
/// setIsEmitted / isEmitted - Getter/Setter for flag indicating that this
/// SDDbgValue has been emitted to an MBB.
void setIsEmitted() { Emitted = true; }
bool isEmitted() const { return Emitted; }
/// clearIsEmitted - Reset Emitted flag, for certain special cases where
/// dbg.addr is emitted twice.
void clearIsEmitted() { Emitted = false; }
LLVM_DUMP_METHOD void dump() const;
LLVM_DUMP_METHOD void print(raw_ostream &OS) const;
};
/// Holds the information from a dbg_label node through SDISel.
/// We do not use SDValue here to avoid including its header.
class SDDbgLabel {
MDNode *Label;
DebugLoc DL;
unsigned Order;
public:
SDDbgLabel(MDNode *Label, DebugLoc dl, unsigned O)
: Label(Label), DL(std::move(dl)), Order(O) {}
/// Returns the MDNode pointer for the label.
MDNode *getLabel() const { return Label; }
/// Returns the DebugLoc.
const DebugLoc &getDebugLoc() const { return DL; }
/// Returns the SDNodeOrder. This is the order of the preceding node in the
/// input.
unsigned getOrder() const { return Order; }
};
} // end llvm namespace
#endif
|