forked from OSchip/llvm-project
[DebugInfo][NFCI] Split salvageDebugInfo into helper functions
Some use cases are appearing where salvaging is needed that does not correspond to an instruction being deleted -- for example an instruction being sunk, or a Value not being available in a block being isel'd. Enable more fine grained control over how salavging occurs by splitting the logic into helper functions, separating things that are specific to working on DbgVariableIntrinsics from those specific to interpreting IR and building DIExpressions. Differential Revision: https://reviews.llvm.org/D57696 llvm-svn: 353156
This commit is contained in:
parent
7279895454
commit
84ca706be1
|
@ -341,6 +341,18 @@ void replaceDbgValueForAlloca(AllocaInst *AI, Value *NewAllocaAddress,
|
|||
/// Returns true if any debug users were updated.
|
||||
bool salvageDebugInfo(Instruction &I);
|
||||
|
||||
/// Implementation of salvageDebugInfo, applying only to instructions in
|
||||
/// \p Insns, rather than all debug users of \p I.
|
||||
bool salvageDebugInfoForDbgValues(Instruction &I,
|
||||
ArrayRef<DbgVariableIntrinsic *> Insns);
|
||||
|
||||
/// Given an instruction \p I and DIExpression \p DIExpr operating on it, write
|
||||
/// the effects of \p I into the returned DIExpression, or return nullptr if
|
||||
/// it cannot be salvaged. \p StackVal: whether DW_OP_stack_value should be
|
||||
/// appended to the expression.
|
||||
DIExpression *salvageDebugInfoImpl(Instruction &I, DIExpression *DIExpr,
|
||||
bool StackVal);
|
||||
|
||||
/// Point debug users of \p From to \p To or salvage them. Use this function
|
||||
/// only when replacing all uses of \p From with \p To, with a guarantee that
|
||||
/// \p From is going to be deleted.
|
||||
|
|
|
@ -1593,120 +1593,120 @@ bool llvm::salvageDebugInfo(Instruction &I) {
|
|||
if (DbgUsers.empty())
|
||||
return false;
|
||||
|
||||
auto &M = *I.getModule();
|
||||
auto &DL = M.getDataLayout();
|
||||
return salvageDebugInfoForDbgValues(I, DbgUsers);
|
||||
}
|
||||
|
||||
bool llvm::salvageDebugInfoForDbgValues(
|
||||
Instruction &I, ArrayRef<DbgVariableIntrinsic *> DbgUsers) {
|
||||
auto &Ctx = I.getContext();
|
||||
auto wrapMD = [&](Value *V) { return wrapValueInMetadata(Ctx, V); };
|
||||
|
||||
auto doSalvage = [&](DbgVariableIntrinsic *DII, SmallVectorImpl<uint64_t> &Ops) {
|
||||
auto *DIExpr = DII->getExpression();
|
||||
if (!Ops.empty()) {
|
||||
// Do not add DW_OP_stack_value for DbgDeclare and DbgAddr, because they
|
||||
// are implicitly pointing out the value as a DWARF memory location
|
||||
// description.
|
||||
bool WithStackValue = isa<DbgValueInst>(DII);
|
||||
DIExpr = DIExpression::prependOpcodes(DIExpr, Ops, WithStackValue);
|
||||
}
|
||||
for (auto *DII : DbgUsers) {
|
||||
// Do not add DW_OP_stack_value for DbgDeclare and DbgAddr, because they
|
||||
// are implicitly pointing out the value as a DWARF memory location
|
||||
// description.
|
||||
bool StackValue = isa<DbgValueInst>(DII);
|
||||
|
||||
DIExpression *DIExpr =
|
||||
salvageDebugInfoImpl(I, DII->getExpression(), StackValue);
|
||||
|
||||
// salvageDebugInfoImpl should fail on examining the first element of
|
||||
// DbgUsers, or none of them.
|
||||
if (!DIExpr)
|
||||
return false;
|
||||
|
||||
DII->setOperand(0, wrapMD(I.getOperand(0)));
|
||||
DII->setOperand(2, MetadataAsValue::get(Ctx, DIExpr));
|
||||
LLVM_DEBUG(dbgs() << "SALVAGE: " << *DII << '\n');
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
DIExpression *llvm::salvageDebugInfoImpl(Instruction &I,
|
||||
DIExpression *SrcDIExpr,
|
||||
bool WithStackValue) {
|
||||
auto &M = *I.getModule();
|
||||
auto &DL = M.getDataLayout();
|
||||
|
||||
// Apply a vector of opcodes to the source DIExpression.
|
||||
auto doSalvage = [&](SmallVectorImpl<uint64_t> &Ops) -> DIExpression * {
|
||||
DIExpression *DIExpr = SrcDIExpr;
|
||||
if (!Ops.empty()) {
|
||||
DIExpr = DIExpression::prependOpcodes(DIExpr, Ops, WithStackValue);
|
||||
}
|
||||
return DIExpr;
|
||||
};
|
||||
|
||||
auto applyOffset = [&](DbgVariableIntrinsic *DII, uint64_t Offset) {
|
||||
// Apply the given offset to the source DIExpression.
|
||||
auto applyOffset = [&](uint64_t Offset) -> DIExpression * {
|
||||
SmallVector<uint64_t, 8> Ops;
|
||||
DIExpression::appendOffset(Ops, Offset);
|
||||
doSalvage(DII, Ops);
|
||||
return doSalvage(Ops);
|
||||
};
|
||||
|
||||
auto applyOps = [&](DbgVariableIntrinsic *DII,
|
||||
std::initializer_list<uint64_t> Opcodes) {
|
||||
// initializer-list helper for applying operators to the source DIExpression.
|
||||
auto applyOps =
|
||||
[&](std::initializer_list<uint64_t> Opcodes) -> DIExpression * {
|
||||
SmallVector<uint64_t, 8> Ops(Opcodes);
|
||||
doSalvage(DII, Ops);
|
||||
return doSalvage(Ops);
|
||||
};
|
||||
|
||||
if (auto *CI = dyn_cast<CastInst>(&I)) {
|
||||
if (!CI->isNoopCast(DL))
|
||||
return false;
|
||||
return nullptr;
|
||||
|
||||
// No-op casts are irrelevant for debug info.
|
||||
MetadataAsValue *CastSrc = wrapMD(I.getOperand(0));
|
||||
for (auto *DII : DbgUsers) {
|
||||
DII->setOperand(0, CastSrc);
|
||||
LLVM_DEBUG(dbgs() << "SALVAGE: " << *DII << '\n');
|
||||
}
|
||||
return true;
|
||||
return SrcDIExpr;
|
||||
} else if (auto *GEP = dyn_cast<GetElementPtrInst>(&I)) {
|
||||
unsigned BitWidth =
|
||||
M.getDataLayout().getIndexSizeInBits(GEP->getPointerAddressSpace());
|
||||
// Rewrite a constant GEP into a DIExpression. Since we are performing
|
||||
// arithmetic to compute the variable's *value* in the DIExpression, we
|
||||
// need to mark the expression with a DW_OP_stack_value.
|
||||
// Rewrite a constant GEP into a DIExpression.
|
||||
APInt Offset(BitWidth, 0);
|
||||
if (GEP->accumulateConstantOffset(M.getDataLayout(), Offset))
|
||||
for (auto *DII : DbgUsers)
|
||||
applyOffset(DII, Offset.getSExtValue());
|
||||
return true;
|
||||
if (GEP->accumulateConstantOffset(M.getDataLayout(), Offset)) {
|
||||
return applyOffset(Offset.getSExtValue());
|
||||
} else {
|
||||
return nullptr;
|
||||
}
|
||||
} else if (auto *BI = dyn_cast<BinaryOperator>(&I)) {
|
||||
// Rewrite binary operations with constant integer operands.
|
||||
auto *ConstInt = dyn_cast<ConstantInt>(I.getOperand(1));
|
||||
if (!ConstInt || ConstInt->getBitWidth() > 64)
|
||||
return false;
|
||||
return nullptr;
|
||||
|
||||
uint64_t Val = ConstInt->getSExtValue();
|
||||
for (auto *DII : DbgUsers) {
|
||||
switch (BI->getOpcode()) {
|
||||
case Instruction::Add:
|
||||
applyOffset(DII, Val);
|
||||
break;
|
||||
case Instruction::Sub:
|
||||
applyOffset(DII, -int64_t(Val));
|
||||
break;
|
||||
case Instruction::Mul:
|
||||
applyOps(DII, {dwarf::DW_OP_constu, Val, dwarf::DW_OP_mul});
|
||||
break;
|
||||
case Instruction::SDiv:
|
||||
applyOps(DII, {dwarf::DW_OP_constu, Val, dwarf::DW_OP_div});
|
||||
break;
|
||||
case Instruction::SRem:
|
||||
applyOps(DII, {dwarf::DW_OP_constu, Val, dwarf::DW_OP_mod});
|
||||
break;
|
||||
case Instruction::Or:
|
||||
applyOps(DII, {dwarf::DW_OP_constu, Val, dwarf::DW_OP_or});
|
||||
break;
|
||||
case Instruction::And:
|
||||
applyOps(DII, {dwarf::DW_OP_constu, Val, dwarf::DW_OP_and});
|
||||
break;
|
||||
case Instruction::Xor:
|
||||
applyOps(DII, {dwarf::DW_OP_constu, Val, dwarf::DW_OP_xor});
|
||||
break;
|
||||
case Instruction::Shl:
|
||||
applyOps(DII, {dwarf::DW_OP_constu, Val, dwarf::DW_OP_shl});
|
||||
break;
|
||||
case Instruction::LShr:
|
||||
applyOps(DII, {dwarf::DW_OP_constu, Val, dwarf::DW_OP_shr});
|
||||
break;
|
||||
case Instruction::AShr:
|
||||
applyOps(DII, {dwarf::DW_OP_constu, Val, dwarf::DW_OP_shra});
|
||||
break;
|
||||
default:
|
||||
// TODO: Salvage constants from each kind of binop we know about.
|
||||
return false;
|
||||
}
|
||||
switch (BI->getOpcode()) {
|
||||
case Instruction::Add:
|
||||
return applyOffset(Val);
|
||||
case Instruction::Sub:
|
||||
return applyOffset(-int64_t(Val));
|
||||
case Instruction::Mul:
|
||||
return applyOps({dwarf::DW_OP_constu, Val, dwarf::DW_OP_mul});
|
||||
case Instruction::SDiv:
|
||||
return applyOps({dwarf::DW_OP_constu, Val, dwarf::DW_OP_div});
|
||||
case Instruction::SRem:
|
||||
return applyOps({dwarf::DW_OP_constu, Val, dwarf::DW_OP_mod});
|
||||
case Instruction::Or:
|
||||
return applyOps({dwarf::DW_OP_constu, Val, dwarf::DW_OP_or});
|
||||
case Instruction::And:
|
||||
return applyOps({dwarf::DW_OP_constu, Val, dwarf::DW_OP_and});
|
||||
case Instruction::Xor:
|
||||
return applyOps({dwarf::DW_OP_constu, Val, dwarf::DW_OP_xor});
|
||||
case Instruction::Shl:
|
||||
return applyOps({dwarf::DW_OP_constu, Val, dwarf::DW_OP_shl});
|
||||
case Instruction::LShr:
|
||||
return applyOps({dwarf::DW_OP_constu, Val, dwarf::DW_OP_shr});
|
||||
case Instruction::AShr:
|
||||
return applyOps({dwarf::DW_OP_constu, Val, dwarf::DW_OP_shra});
|
||||
default:
|
||||
// TODO: Salvage constants from each kind of binop we know about.
|
||||
return nullptr;
|
||||
}
|
||||
return true;
|
||||
} else if (isa<LoadInst>(&I)) {
|
||||
MetadataAsValue *AddrMD = wrapMD(I.getOperand(0));
|
||||
for (auto *DII : DbgUsers) {
|
||||
// Rewrite the load into DW_OP_deref.
|
||||
auto *DIExpr = DII->getExpression();
|
||||
DIExpr = DIExpression::prepend(DIExpr, DIExpression::WithDeref);
|
||||
DII->setOperand(0, AddrMD);
|
||||
DII->setOperand(2, MetadataAsValue::get(Ctx, DIExpr));
|
||||
LLVM_DEBUG(dbgs() << "SALVAGE: " << *DII << '\n');
|
||||
}
|
||||
return true;
|
||||
// Rewrite the load into DW_OP_deref.
|
||||
return DIExpression::prepend(SrcDIExpr, DIExpression::WithDeref);
|
||||
}
|
||||
return false;
|
||||
return nullptr;
|
||||
}
|
||||
|
||||
/// A replacement for a dbg.value expression.
|
||||
|
|
Loading…
Reference in New Issue