InstrRefBasedLDV is marginally slower than VarlocBasedLDV when analysing optimised code -- however, it's much slower when analysing code compiled -O0. To avoid this: don't use instruction referencing for -O0 functions. In the "pure" case of unoptimised code, this won't really harm the debugging experience because most variables won't have been promoted off the stack, so can't go missing. It becomes more complicated when optimised code is inlined into functions marked optnone; however these are rare, and as -O0 doesn't run many optimisations there should be little damage to the debug experience as a result. I've taken the opportunity to refactor testing for instruction-referencing into a MachineFunction method, which seems the most appropriate place to put it. Differential Revision: https://reviews.llvm.org/D108585
115 lines
3.9 KiB
C++
115 lines
3.9 KiB
C++
//===- LiveDebugValues.cpp - Tracking Debug Value MIs ---------------------===//
|
|
//
|
|
// 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
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "LiveDebugValues.h"
|
|
|
|
#include "llvm/CodeGen/MachineBasicBlock.h"
|
|
#include "llvm/CodeGen/MachineFrameInfo.h"
|
|
#include "llvm/CodeGen/MachineFunctionPass.h"
|
|
#include "llvm/CodeGen/Passes.h"
|
|
#include "llvm/InitializePasses.h"
|
|
#include "llvm/Pass.h"
|
|
#include "llvm/Support/CommandLine.h"
|
|
#include "llvm/Target/TargetMachine.h"
|
|
|
|
/// \file LiveDebugValues.cpp
|
|
///
|
|
/// The LiveDebugValues pass extends the range of variable locations
|
|
/// (specified by DBG_VALUE instructions) from single blocks to successors
|
|
/// and any other code locations where the variable location is valid.
|
|
/// There are currently two implementations: the "VarLoc" implementation
|
|
/// explicitly tracks the location of a variable, while the "InstrRef"
|
|
/// implementation tracks the values defined by instructions through locations.
|
|
///
|
|
/// This file implements neither; it merely registers the pass, allows the
|
|
/// user to pick which implementation will be used to propagate variable
|
|
/// locations.
|
|
|
|
#define DEBUG_TYPE "livedebugvalues"
|
|
|
|
using namespace llvm;
|
|
|
|
static cl::opt<bool>
|
|
ForceInstrRefLDV("force-instr-ref-livedebugvalues", cl::Hidden,
|
|
cl::desc("Use instruction-ref based LiveDebugValues with "
|
|
"normal DBG_VALUE inputs"),
|
|
cl::init(false));
|
|
|
|
// Options to prevent pathological compile-time behavior. If InputBBLimit and
|
|
// InputDbgValueLimit are both exceeded, range extension is disabled.
|
|
static cl::opt<unsigned> InputBBLimit(
|
|
"livedebugvalues-input-bb-limit",
|
|
cl::desc("Maximum input basic blocks before DBG_VALUE limit applies"),
|
|
cl::init(10000), cl::Hidden);
|
|
static cl::opt<unsigned> InputDbgValueLimit(
|
|
"livedebugvalues-input-dbg-value-limit",
|
|
cl::desc(
|
|
"Maximum input DBG_VALUE insts supported by debug range extension"),
|
|
cl::init(50000), cl::Hidden);
|
|
|
|
/// Generic LiveDebugValues pass. Calls through to VarLocBasedLDV or
|
|
/// InstrRefBasedLDV to perform location propagation, via the LDVImpl
|
|
/// base class.
|
|
class LiveDebugValues : public MachineFunctionPass {
|
|
public:
|
|
static char ID;
|
|
|
|
LiveDebugValues();
|
|
~LiveDebugValues() {
|
|
if (TheImpl)
|
|
delete TheImpl;
|
|
}
|
|
|
|
/// Calculate the liveness information for the given machine function.
|
|
bool runOnMachineFunction(MachineFunction &MF) override;
|
|
|
|
MachineFunctionProperties getRequiredProperties() const override {
|
|
return MachineFunctionProperties().set(
|
|
MachineFunctionProperties::Property::NoVRegs);
|
|
}
|
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override {
|
|
AU.setPreservesCFG();
|
|
MachineFunctionPass::getAnalysisUsage(AU);
|
|
}
|
|
|
|
private:
|
|
LDVImpl *TheImpl;
|
|
TargetPassConfig *TPC;
|
|
};
|
|
|
|
char LiveDebugValues::ID = 0;
|
|
|
|
char &llvm::LiveDebugValuesID = LiveDebugValues::ID;
|
|
|
|
INITIALIZE_PASS(LiveDebugValues, DEBUG_TYPE, "Live DEBUG_VALUE analysis", false,
|
|
false)
|
|
|
|
/// Default construct and initialize the pass.
|
|
LiveDebugValues::LiveDebugValues() : MachineFunctionPass(ID) {
|
|
initializeLiveDebugValuesPass(*PassRegistry::getPassRegistry());
|
|
TheImpl = nullptr;
|
|
}
|
|
|
|
bool LiveDebugValues::runOnMachineFunction(MachineFunction &MF) {
|
|
if (!TheImpl) {
|
|
TPC = getAnalysisIfAvailable<TargetPassConfig>();
|
|
bool InstrRefBased = MF.useDebugInstrRef();
|
|
|
|
// Allow the user to force selection of InstrRef LDV.
|
|
InstrRefBased |= ForceInstrRefLDV;
|
|
|
|
if (InstrRefBased)
|
|
TheImpl = llvm::makeInstrRefBasedLiveDebugValues();
|
|
else
|
|
TheImpl = llvm::makeVarLocBasedLiveDebugValues();
|
|
}
|
|
|
|
return TheImpl->ExtendRanges(MF, TPC, InputBBLimit, InputDbgValueLimit);
|
|
}
|