2017-09-12 07:00:48 +08:00
|
|
|
//===- MachineVerifier.cpp - Machine Code Verifier ------------------------===//
|
2009-05-16 08:33:53 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// 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
|
2009-05-16 08:33:53 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Pass to verify generated machine code. The following is checked:
|
|
|
|
//
|
|
|
|
// Operand counts: All explicit operands must be present.
|
|
|
|
//
|
|
|
|
// Register classes: All physical and virtual register operands must be
|
|
|
|
// compatible with the register class required by the instruction descriptor.
|
|
|
|
//
|
|
|
|
// Register live intervals: Registers must be defined only once, and must be
|
|
|
|
// defined before use.
|
|
|
|
//
|
2020-01-27 07:47:32 +08:00
|
|
|
// The machine code verifier is enabled with the command-line option
|
|
|
|
// -verify-machineinstrs.
|
2009-05-16 08:33:53 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/ADT/BitVector.h"
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/ADT/DenseSet.h"
|
2013-07-16 05:26:31 +08:00
|
|
|
#include "llvm/ADT/DepthFirstIterator.h"
|
[MachineVerifier] Doing ::calcRegsPassed in RPO: ~35% faster MV, NFC
Depending on the target, test suite, pipeline config and perhaps other
factors machine verifier when forced on with -verify-machineinstrs can
increase compile time 2-2.5 times over (Release, Asserts On), taking up
~60% of the time. An invaluable tool, it significantly slows down
machine verifier-enabled testing.
Nearly 75% of its time MachineVerifier spends in the calcRegsPassed
method. It's a classic forward dataflow analysis executed over sets, but
visiting MBBs in arbitrary order. We switch that to RPO here.
This speeds up MachineVerifier by about 35%, decreasing the overall
compile time with -verify-machineinstrs by 20-25% or so.
calcRegsPassed itself gets 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: bogner
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75032
2020-02-24 13:53:19 +08:00
|
|
|
#include "llvm/ADT/PostOrderIterator.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/ADT/STLExtras.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/ADT/SetOperations.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/ADT/SmallPtrSet.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/ADT/SmallVector.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/ADT/StringRef.h"
|
|
|
|
#include "llvm/ADT/Twine.h"
|
2015-12-03 07:06:39 +08:00
|
|
|
#include "llvm/Analysis/EHPersonalities.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/CodeGen/GlobalISel/RegisterBank.h"
|
|
|
|
#include "llvm/CodeGen/LiveInterval.h"
|
2020-03-23 10:08:29 +08:00
|
|
|
#include "llvm/CodeGen/LiveIntervalCalc.h"
|
2017-12-13 10:51:04 +08:00
|
|
|
#include "llvm/CodeGen/LiveIntervals.h"
|
2017-12-19 07:19:44 +08:00
|
|
|
#include "llvm/CodeGen/LiveStacks.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/CodeGen/LiveVariables.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/CodeGen/MachineBasicBlock.h"
|
2009-08-14 00:19:51 +08:00
|
|
|
#include "llvm/CodeGen/MachineFrameInfo.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/CodeGen/MachineFunction.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/CodeGen/MachineFunctionPass.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/CodeGen/MachineInstr.h"
|
|
|
|
#include "llvm/CodeGen/MachineInstrBundle.h"
|
2009-10-08 01:36:00 +08:00
|
|
|
#include "llvm/CodeGen/MachineMemOperand.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/CodeGen/MachineOperand.h"
|
2009-05-16 08:33:53 +08:00
|
|
|
#include "llvm/CodeGen/MachineRegisterInfo.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/CodeGen/PseudoSourceValue.h"
|
|
|
|
#include "llvm/CodeGen/SlotIndexes.h"
|
2017-06-03 00:36:37 +08:00
|
|
|
#include "llvm/CodeGen/StackMaps.h"
|
2017-11-08 09:01:31 +08:00
|
|
|
#include "llvm/CodeGen/TargetInstrInfo.h"
|
2017-11-17 09:07:10 +08:00
|
|
|
#include "llvm/CodeGen/TargetOpcodes.h"
|
|
|
|
#include "llvm/CodeGen/TargetRegisterInfo.h"
|
|
|
|
#include "llvm/CodeGen/TargetSubtargetInfo.h"
|
2013-01-02 19:36:10 +08:00
|
|
|
#include "llvm/IR/BasicBlock.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/IR/Function.h"
|
2013-01-02 19:36:10 +08:00
|
|
|
#include "llvm/IR/InlineAsm.h"
|
|
|
|
#include "llvm/IR/Instructions.h"
|
Sink all InitializePasses.h includes
This file lists every pass in LLVM, and is included by Pass.h, which is
very popular. Every time we add, remove, or rename a pass in LLVM, it
caused lots of recompilation.
I found this fact by looking at this table, which is sorted by the
number of times a file was changed over the last 100,000 git commits
multiplied by the number of object files that depend on it in the
current checkout:
recompiles touches affected_files header
342380 95 3604 llvm/include/llvm/ADT/STLExtras.h
314730 234 1345 llvm/include/llvm/InitializePasses.h
307036 118 2602 llvm/include/llvm/ADT/APInt.h
213049 59 3611 llvm/include/llvm/Support/MathExtras.h
170422 47 3626 llvm/include/llvm/Support/Compiler.h
162225 45 3605 llvm/include/llvm/ADT/Optional.h
158319 63 2513 llvm/include/llvm/ADT/Triple.h
140322 39 3598 llvm/include/llvm/ADT/StringRef.h
137647 59 2333 llvm/include/llvm/Support/Error.h
131619 73 1803 llvm/include/llvm/Support/FileSystem.h
Before this change, touching InitializePasses.h would cause 1345 files
to recompile. After this change, touching it only causes 550 compiles in
an incremental rebuild.
Reviewers: bkramer, asbirlea, bollu, jdoerfert
Differential Revision: https://reviews.llvm.org/D70211
2019-11-14 05:15:01 +08:00
|
|
|
#include "llvm/InitializePasses.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/MC/LaneBitmask.h"
|
2011-05-05 06:54:05 +08:00
|
|
|
#include "llvm/MC/MCAsmInfo.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/MC/MCInstrDesc.h"
|
|
|
|
#include "llvm/MC/MCRegisterInfo.h"
|
|
|
|
#include "llvm/MC/MCTargetOptions.h"
|
|
|
|
#include "llvm/Pass.h"
|
|
|
|
#include "llvm/Support/Casting.h"
|
2009-07-11 21:10:19 +08:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include "llvm/Support/LowLevelTypeImpl.h"
|
|
|
|
#include "llvm/Support/MathExtras.h"
|
2009-07-11 21:10:19 +08:00
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2012-12-04 00:50:05 +08:00
|
|
|
#include "llvm/Target/TargetMachine.h"
|
2017-09-12 07:00:48 +08:00
|
|
|
#include <algorithm>
|
|
|
|
#include <cassert>
|
|
|
|
#include <cstddef>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <iterator>
|
|
|
|
#include <string>
|
|
|
|
#include <utility>
|
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2017-09-12 07:00:48 +08:00
|
|
|
struct MachineVerifier {
|
|
|
|
MachineVerifier(Pass *pass, const char *b) : PASS(pass), Banner(b) {}
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2016-02-16 03:25:31 +08:00
|
|
|
unsigned verify(MachineFunction &MF);
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2009-11-19 04:36:57 +08:00
|
|
|
Pass *const PASS;
|
2010-12-18 08:06:56 +08:00
|
|
|
const char *Banner;
|
2009-05-16 08:33:53 +08:00
|
|
|
const MachineFunction *MF;
|
|
|
|
const TargetMachine *TM;
|
2011-06-28 05:26:13 +08:00
|
|
|
const TargetInstrInfo *TII;
|
2009-05-16 08:33:53 +08:00
|
|
|
const TargetRegisterInfo *TRI;
|
|
|
|
const MachineRegisterInfo *MRI;
|
|
|
|
|
|
|
|
unsigned foundErrors;
|
|
|
|
|
2016-08-03 00:17:15 +08:00
|
|
|
// Avoid querying the MachineFunctionProperties for each operand.
|
|
|
|
bool isFunctionRegBankSelected;
|
2016-08-03 00:49:22 +08:00
|
|
|
bool isFunctionSelected;
|
2016-08-03 00:17:15 +08:00
|
|
|
|
2017-09-12 07:00:48 +08:00
|
|
|
using RegVector = SmallVector<unsigned, 16>;
|
|
|
|
using RegMaskVector = SmallVector<const uint32_t *, 4>;
|
|
|
|
using RegSet = DenseSet<unsigned>;
|
|
|
|
using RegMap = DenseMap<unsigned, const MachineInstr *>;
|
|
|
|
using BlockSet = SmallPtrSet<const MachineBasicBlock *, 8>;
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2018-10-04 06:05:31 +08:00
|
|
|
const MachineInstr *FirstNonPHI;
|
2011-09-24 06:45:39 +08:00
|
|
|
const MachineInstr *FirstTerminator;
|
2012-08-21 04:52:06 +08:00
|
|
|
BlockSet FunctionBlocks;
|
2011-09-24 06:45:39 +08:00
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
BitVector regsReserved;
|
|
|
|
RegSet regsLive;
|
2009-08-08 21:19:25 +08:00
|
|
|
RegVector regsDefined, regsDead, regsKilled;
|
2012-02-28 09:42:41 +08:00
|
|
|
RegMaskVector regMasks;
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2011-01-13 05:27:48 +08:00
|
|
|
SlotIndex lastIndex;
|
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
// Add Reg and any sub-registers to RV
|
|
|
|
void addRegWithSubRegs(RegVector &RV, unsigned Reg) {
|
|
|
|
RV.push_back(Reg);
|
2019-08-02 07:27:28 +08:00
|
|
|
if (Register::isPhysicalRegister(Reg))
|
2019-12-05 17:16:08 +08:00
|
|
|
for (const MCPhysReg &SubReg : TRI->subregs(Reg))
|
|
|
|
RV.push_back(SubReg);
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
struct BBInfo {
|
|
|
|
// Is this MBB reachable from the MF entry point?
|
2017-09-12 07:00:48 +08:00
|
|
|
bool reachable = false;
|
2009-05-16 08:33:53 +08:00
|
|
|
|
|
|
|
// Vregs that must be live in because they are used without being
|
|
|
|
// defined. Map value is the user.
|
|
|
|
RegMap vregsLiveIn;
|
|
|
|
|
|
|
|
// Regs killed in MBB. They may be defined again, and will then be in both
|
|
|
|
// regsKilled and regsLiveOut.
|
|
|
|
RegSet regsKilled;
|
|
|
|
|
|
|
|
// Regs defined in MBB and live out. Note that vregs passing through may
|
|
|
|
// be live out without being mentioned here.
|
|
|
|
RegSet regsLiveOut;
|
|
|
|
|
|
|
|
// Vregs that pass through MBB untouched. This set is disjoint from
|
|
|
|
// regsKilled and regsLiveOut.
|
|
|
|
RegSet vregsPassed;
|
|
|
|
|
2009-11-19 04:36:57 +08:00
|
|
|
// Vregs that must pass through MBB because they are needed by a successor
|
|
|
|
// block. This set is disjoint from regsLiveOut.
|
|
|
|
RegSet vregsRequired;
|
|
|
|
|
2012-08-21 04:52:06 +08:00
|
|
|
// Set versions of block's predecessor and successor lists.
|
|
|
|
BlockSet Preds, Succs;
|
|
|
|
|
2017-09-12 07:00:48 +08:00
|
|
|
BBInfo() = default;
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2009-11-19 04:36:57 +08:00
|
|
|
// Add register to vregsRequired if it belongs there. Return true if
|
|
|
|
// anything changed.
|
|
|
|
bool addRequired(unsigned Reg) {
|
2019-08-02 07:27:28 +08:00
|
|
|
if (!Register::isVirtualRegister(Reg))
|
2009-11-19 04:36:57 +08:00
|
|
|
return false;
|
|
|
|
if (regsLiveOut.count(Reg))
|
|
|
|
return false;
|
|
|
|
return vregsRequired.insert(Reg).second;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Same for a full set.
|
|
|
|
bool addRequired(const RegSet &RS) {
|
|
|
|
bool changed = false;
|
|
|
|
for (RegSet::const_iterator I = RS.begin(), E = RS.end(); I != E; ++I)
|
|
|
|
if (addRequired(*I))
|
|
|
|
changed = true;
|
|
|
|
return changed;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Same for a full map.
|
|
|
|
bool addRequired(const RegMap &RM) {
|
|
|
|
bool changed = false;
|
|
|
|
for (RegMap::const_iterator I = RM.begin(), E = RM.end(); I != E; ++I)
|
|
|
|
if (addRequired(I->first))
|
|
|
|
changed = true;
|
|
|
|
return changed;
|
|
|
|
}
|
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
// Live-out registers are either in regsLiveOut or vregsPassed.
|
|
|
|
bool isLiveOut(unsigned Reg) const {
|
|
|
|
return regsLiveOut.count(Reg) || vregsPassed.count(Reg);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Extra register info per MBB.
|
|
|
|
DenseMap<const MachineBasicBlock*, BBInfo> MBBInfoMap;
|
|
|
|
|
|
|
|
bool isReserved(unsigned Reg) {
|
2009-08-05 03:18:01 +08:00
|
|
|
return Reg < regsReserved.size() && regsReserved.test(Reg);
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
2017-05-06 06:04:05 +08:00
|
|
|
bool isAllocatable(unsigned Reg) const {
|
|
|
|
return Reg < TRI->getNumRegs() && TRI->isInAllocatableClass(Reg) &&
|
2019-06-19 08:25:39 +08:00
|
|
|
!regsReserved.test(Reg);
|
2012-02-15 03:17:48 +08:00
|
|
|
}
|
|
|
|
|
2009-11-19 04:36:57 +08:00
|
|
|
// Analysis information if available
|
|
|
|
LiveVariables *LiveVars;
|
2010-10-27 06:36:07 +08:00
|
|
|
LiveIntervals *LiveInts;
|
2010-11-02 03:49:52 +08:00
|
|
|
LiveStacks *LiveStks;
|
2010-10-27 04:21:46 +08:00
|
|
|
SlotIndexes *Indexes;
|
2009-11-19 04:36:57 +08:00
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
void visitMachineFunctionBefore();
|
|
|
|
void visitMachineBasicBlockBefore(const MachineBasicBlock *MBB);
|
2012-06-07 06:34:30 +08:00
|
|
|
void visitMachineBundleBefore(const MachineInstr *MI);
|
2019-02-05 07:29:11 +08:00
|
|
|
|
2019-02-05 07:29:16 +08:00
|
|
|
bool verifyVectorElementMatch(LLT Ty0, LLT Ty1, const MachineInstr *MI);
|
2019-02-05 07:29:11 +08:00
|
|
|
void verifyPreISelGenericInstruction(const MachineInstr *MI);
|
2009-05-16 08:33:53 +08:00
|
|
|
void visitMachineInstrBefore(const MachineInstr *MI);
|
|
|
|
void visitMachineOperand(const MachineOperand *MO, unsigned MONum);
|
|
|
|
void visitMachineInstrAfter(const MachineInstr *MI);
|
2012-06-07 06:34:30 +08:00
|
|
|
void visitMachineBundleAfter(const MachineInstr *MI);
|
2009-05-16 08:33:53 +08:00
|
|
|
void visitMachineBasicBlockAfter(const MachineBasicBlock *MBB);
|
|
|
|
void visitMachineFunctionAfter();
|
|
|
|
|
|
|
|
void report(const char *msg, const MachineFunction *MF);
|
|
|
|
void report(const char *msg, const MachineBasicBlock *MBB);
|
|
|
|
void report(const char *msg, const MachineInstr *MI);
|
2018-05-08 06:31:12 +08:00
|
|
|
void report(const char *msg, const MachineOperand *MO, unsigned MONum,
|
|
|
|
LLT MOVRegType = LLT{});
|
2015-11-10 07:59:33 +08:00
|
|
|
|
|
|
|
void report_context(const LiveInterval &LI) const;
|
2016-07-26 03:39:01 +08:00
|
|
|
void report_context(const LiveRange &LR, unsigned VRegUnit,
|
2015-11-10 07:59:33 +08:00
|
|
|
LaneBitmask LaneMask) const;
|
|
|
|
void report_context(const LiveRange::Segment &S) const;
|
|
|
|
void report_context(const VNInfo &VNI) const;
|
2016-02-02 10:44:25 +08:00
|
|
|
void report_context(SlotIndex Pos) const;
|
2019-01-08 23:16:23 +08:00
|
|
|
void report_context(MCPhysReg PhysReg) const;
|
2016-02-02 10:44:25 +08:00
|
|
|
void report_context_liverange(const LiveRange &LR) const;
|
2016-02-03 04:04:51 +08:00
|
|
|
void report_context_lanemask(LaneBitmask LaneMask) const;
|
2016-05-12 05:31:39 +08:00
|
|
|
void report_context_vreg(unsigned VReg) const;
|
2018-07-17 02:51:40 +08:00
|
|
|
void report_context_vreg_regunit(unsigned VRegOrUnit) const;
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2012-08-30 02:11:05 +08:00
|
|
|
void verifyInlineAsm(const MachineInstr *MI);
|
|
|
|
|
2012-03-29 04:47:35 +08:00
|
|
|
void checkLiveness(const MachineOperand *MO, unsigned MONum);
|
2016-02-03 04:04:51 +08:00
|
|
|
void checkLivenessAtUse(const MachineOperand *MO, unsigned MONum,
|
2018-07-17 02:51:40 +08:00
|
|
|
SlotIndex UseIdx, const LiveRange &LR, unsigned VRegOrUnit,
|
2016-12-15 22:36:06 +08:00
|
|
|
LaneBitmask LaneMask = LaneBitmask::getNone());
|
2016-02-03 04:04:51 +08:00
|
|
|
void checkLivenessAtDef(const MachineOperand *MO, unsigned MONum,
|
2018-07-17 02:51:40 +08:00
|
|
|
SlotIndex DefIdx, const LiveRange &LR, unsigned VRegOrUnit,
|
2018-09-20 14:59:18 +08:00
|
|
|
bool SubRangeCheck = false,
|
2016-12-15 22:36:06 +08:00
|
|
|
LaneBitmask LaneMask = LaneBitmask::getNone());
|
2016-02-03 04:04:51 +08:00
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
void markReachable(const MachineBasicBlock *MBB);
|
2010-01-06 04:59:36 +08:00
|
|
|
void calcRegsPassed();
|
2017-11-28 11:54:19 +08:00
|
|
|
void checkPHIOps(const MachineBasicBlock &MBB);
|
2009-11-19 04:36:57 +08:00
|
|
|
|
|
|
|
void calcRegsRequired();
|
|
|
|
void verifyLiveVariables();
|
2010-08-07 02:04:19 +08:00
|
|
|
void verifyLiveIntervals();
|
2012-08-02 08:20:20 +08:00
|
|
|
void verifyLiveInterval(const LiveInterval&);
|
2014-12-10 09:12:10 +08:00
|
|
|
void verifyLiveRangeValue(const LiveRange&, const VNInfo*, unsigned,
|
2016-12-15 22:36:06 +08:00
|
|
|
LaneBitmask);
|
2013-10-11 05:28:54 +08:00
|
|
|
void verifyLiveRangeSegment(const LiveRange&,
|
2014-12-10 09:12:10 +08:00
|
|
|
const LiveRange::const_iterator I, unsigned,
|
2016-12-15 22:36:06 +08:00
|
|
|
LaneBitmask);
|
|
|
|
void verifyLiveRange(const LiveRange&, unsigned,
|
|
|
|
LaneBitmask LaneMask = LaneBitmask::getNone());
|
2013-07-16 05:26:31 +08:00
|
|
|
|
|
|
|
void verifyStackFrame();
|
2015-09-10 01:49:46 +08:00
|
|
|
|
|
|
|
void verifySlotIndexes() const;
|
2016-03-30 01:40:22 +08:00
|
|
|
void verifyProperties(const MachineFunction &MF);
|
2009-11-19 04:36:57 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
struct MachineVerifierPass : public MachineFunctionPass {
|
|
|
|
static char ID; // Pass ID, replacement for typeid
|
2017-09-12 07:00:48 +08:00
|
|
|
|
2014-12-12 03:41:51 +08:00
|
|
|
const std::string Banner;
|
2009-11-19 04:36:57 +08:00
|
|
|
|
2017-03-29 23:25:06 +08:00
|
|
|
MachineVerifierPass(std::string banner = std::string())
|
2017-03-29 17:08:25 +08:00
|
|
|
: MachineFunctionPass(ID), Banner(std::move(banner)) {
|
2010-10-20 01:21:58 +08:00
|
|
|
initializeMachineVerifierPassPass(*PassRegistry::getPassRegistry());
|
|
|
|
}
|
2009-11-19 04:36:57 +08:00
|
|
|
|
2014-03-07 17:26:03 +08:00
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override {
|
2009-11-19 04:36:57 +08:00
|
|
|
AU.setPreservesAll();
|
|
|
|
MachineFunctionPass::getAnalysisUsage(AU);
|
|
|
|
}
|
|
|
|
|
2014-03-07 17:26:03 +08:00
|
|
|
bool runOnMachineFunction(MachineFunction &MF) override {
|
2016-02-16 03:25:31 +08:00
|
|
|
unsigned FoundErrors = MachineVerifier(this, Banner.c_str()).verify(MF);
|
|
|
|
if (FoundErrors)
|
|
|
|
report_fatal_error("Found "+Twine(FoundErrors)+" machine code errors.");
|
2009-11-19 04:36:57 +08:00
|
|
|
return false;
|
|
|
|
}
|
2009-05-16 08:33:53 +08:00
|
|
|
};
|
2009-11-19 04:36:57 +08:00
|
|
|
|
2017-09-12 07:00:48 +08:00
|
|
|
} // end anonymous namespace
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2009-11-19 04:36:57 +08:00
|
|
|
char MachineVerifierPass::ID = 0;
|
2017-09-12 07:00:48 +08:00
|
|
|
|
2010-08-24 01:52:01 +08:00
|
|
|
INITIALIZE_PASS(MachineVerifierPass, "machineverifier",
|
2010-10-08 06:25:06 +08:00
|
|
|
"Verify generated machine code", false, false)
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2014-12-12 03:41:51 +08:00
|
|
|
FunctionPass *llvm::createMachineVerifierPass(const std::string &Banner) {
|
2010-12-18 08:06:56 +08:00
|
|
|
return new MachineVerifierPass(Banner);
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
2016-02-16 03:25:31 +08:00
|
|
|
bool MachineFunction::verify(Pass *p, const char *Banner, bool AbortOnErrors)
|
|
|
|
const {
|
|
|
|
MachineFunction &MF = const_cast<MachineFunction&>(*this);
|
|
|
|
unsigned FoundErrors = MachineVerifier(p, Banner).verify(MF);
|
|
|
|
if (AbortOnErrors && FoundErrors)
|
|
|
|
report_fatal_error("Found "+Twine(FoundErrors)+" machine code errors.");
|
|
|
|
return FoundErrors == 0;
|
2009-11-14 05:56:09 +08:00
|
|
|
}
|
|
|
|
|
2015-09-10 01:49:46 +08:00
|
|
|
void MachineVerifier::verifySlotIndexes() const {
|
|
|
|
if (Indexes == nullptr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Ensure the IdxMBB list is sorted by slot indexes.
|
|
|
|
SlotIndex Last;
|
|
|
|
for (SlotIndexes::MBBIndexIterator I = Indexes->MBBIndexBegin(),
|
|
|
|
E = Indexes->MBBIndexEnd(); I != E; ++I) {
|
|
|
|
assert(!Last.isValid() || I->first > Last);
|
|
|
|
Last = I->first;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-30 01:40:22 +08:00
|
|
|
void MachineVerifier::verifyProperties(const MachineFunction &MF) {
|
|
|
|
// If a pass has introduced virtual registers without clearing the
|
2016-08-25 09:27:13 +08:00
|
|
|
// NoVRegs property (or set it without allocating the vregs)
|
2016-03-30 01:40:22 +08:00
|
|
|
// then report an error.
|
|
|
|
if (MF.getProperties().hasProperty(
|
2016-08-25 09:27:13 +08:00
|
|
|
MachineFunctionProperties::Property::NoVRegs) &&
|
|
|
|
MRI->getNumVirtRegs())
|
|
|
|
report("Function has NoVRegs property but there are VReg operands", &MF);
|
2016-03-30 01:40:22 +08:00
|
|
|
}
|
|
|
|
|
2016-02-16 03:25:31 +08:00
|
|
|
unsigned MachineVerifier::verify(MachineFunction &MF) {
|
2009-05-16 08:33:53 +08:00
|
|
|
foundErrors = 0;
|
|
|
|
|
|
|
|
this->MF = &MF;
|
|
|
|
TM = &MF.getTarget();
|
2014-10-14 15:00:33 +08:00
|
|
|
TII = MF.getSubtarget().getInstrInfo();
|
|
|
|
TRI = MF.getSubtarget().getRegisterInfo();
|
2009-05-16 08:33:53 +08:00
|
|
|
MRI = &MF.getRegInfo();
|
|
|
|
|
[GlobalISel] Print/Parse FailedISel MachineFunction property
FailedISel MachineFunction property is part of the CodeGen pipeline
state as much as every other property, notably, Legalized,
RegBankSelected, and Selected. Let's make that part of the state also
serializable / de-serializable, so if GlobalISel aborts on some of the
functions of a large module, but not the others, it could be easily seen
and the state of the pipeline could be maintained through llc's
invocations with -stop-after / -start-after.
To make MIR printable and generally to not to break it too much too
soon, this patch also defers cleaning up the vreg -> LLT map until
ResetMachineFunctionPass.
To make MIR with FailedISel: true also machine verifiable, machine
verifier is changed so it treats a MIR-module as non-regbankselected and
non-selected if there is FailedISel property set.
Reviewers: qcolombet, ab
Reviewed By: dsanders
Subscribers: javed.absar, rovka, kristof.beyls, llvm-commits
Differential Revision: https://reviews.llvm.org/D42877
llvm-svn: 326343
2018-03-01 01:55:45 +08:00
|
|
|
const bool isFunctionFailedISel = MF.getProperties().hasProperty(
|
|
|
|
MachineFunctionProperties::Property::FailedISel);
|
2018-10-03 01:56:58 +08:00
|
|
|
|
|
|
|
// If we're mid-GlobalISel and we already triggered the fallback path then
|
|
|
|
// it's expected that the MIR is somewhat broken but that's ok since we'll
|
|
|
|
// reset it and clear the FailedISel attribute in ResetMachineFunctions.
|
|
|
|
if (isFunctionFailedISel)
|
|
|
|
return foundErrors;
|
|
|
|
|
2020-01-27 07:47:32 +08:00
|
|
|
isFunctionRegBankSelected = MF.getProperties().hasProperty(
|
|
|
|
MachineFunctionProperties::Property::RegBankSelected);
|
|
|
|
isFunctionSelected = MF.getProperties().hasProperty(
|
|
|
|
MachineFunctionProperties::Property::Selected);
|
|
|
|
|
2014-04-14 08:51:57 +08:00
|
|
|
LiveVars = nullptr;
|
|
|
|
LiveInts = nullptr;
|
|
|
|
LiveStks = nullptr;
|
|
|
|
Indexes = nullptr;
|
2009-11-19 04:36:57 +08:00
|
|
|
if (PASS) {
|
2010-08-06 06:32:21 +08:00
|
|
|
LiveInts = PASS->getAnalysisIfAvailable<LiveIntervals>();
|
2010-08-06 07:51:26 +08:00
|
|
|
// We don't want to verify LiveVariables if LiveIntervals is available.
|
|
|
|
if (!LiveInts)
|
|
|
|
LiveVars = PASS->getAnalysisIfAvailable<LiveVariables>();
|
2010-11-02 03:49:52 +08:00
|
|
|
LiveStks = PASS->getAnalysisIfAvailable<LiveStacks>();
|
2010-10-27 04:21:46 +08:00
|
|
|
Indexes = PASS->getAnalysisIfAvailable<SlotIndexes>();
|
2009-11-19 04:36:57 +08:00
|
|
|
}
|
|
|
|
|
2015-09-10 01:49:46 +08:00
|
|
|
verifySlotIndexes();
|
|
|
|
|
2016-03-30 01:40:22 +08:00
|
|
|
verifyProperties(MF);
|
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
visitMachineFunctionBefore();
|
|
|
|
for (MachineFunction::const_iterator MFI = MF.begin(), MFE = MF.end();
|
|
|
|
MFI!=MFE; ++MFI) {
|
2015-10-10 03:40:45 +08:00
|
|
|
visitMachineBasicBlockBefore(&*MFI);
|
2012-06-07 06:34:30 +08:00
|
|
|
// Keep track of the current bundle header.
|
2014-04-14 08:51:57 +08:00
|
|
|
const MachineInstr *CurBundle = nullptr;
|
2012-12-19 06:55:07 +08:00
|
|
|
// Do we expect the next instruction to be part of the same bundle?
|
|
|
|
bool InBundle = false;
|
|
|
|
|
2011-12-14 10:11:42 +08:00
|
|
|
for (MachineBasicBlock::const_instr_iterator MBBI = MFI->instr_begin(),
|
|
|
|
MBBE = MFI->instr_end(); MBBI != MBBE; ++MBBI) {
|
2015-10-10 03:40:45 +08:00
|
|
|
if (MBBI->getParent() != &*MFI) {
|
2016-09-03 09:22:56 +08:00
|
|
|
report("Bad instruction parent pointer", &*MFI);
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "Instruction: " << *MBBI;
|
2011-01-13 05:27:41 +08:00
|
|
|
continue;
|
|
|
|
}
|
2012-12-19 06:55:07 +08:00
|
|
|
|
|
|
|
// Check for consistent bundle flags.
|
|
|
|
if (InBundle && !MBBI->isBundledWithPred())
|
|
|
|
report("Missing BundledPred flag, "
|
2015-10-10 03:40:45 +08:00
|
|
|
"BundledSucc was set on predecessor",
|
|
|
|
&*MBBI);
|
2012-12-19 06:55:07 +08:00
|
|
|
if (!InBundle && MBBI->isBundledWithPred())
|
|
|
|
report("BundledPred flag is set, "
|
2015-10-10 03:40:45 +08:00
|
|
|
"but BundledSucc not set on predecessor",
|
|
|
|
&*MBBI);
|
2012-12-19 06:55:07 +08:00
|
|
|
|
2012-06-07 06:34:30 +08:00
|
|
|
// Is this a bundle header?
|
|
|
|
if (!MBBI->isInsideBundle()) {
|
|
|
|
if (CurBundle)
|
|
|
|
visitMachineBundleAfter(CurBundle);
|
2015-10-10 03:40:45 +08:00
|
|
|
CurBundle = &*MBBI;
|
2012-06-07 06:34:30 +08:00
|
|
|
visitMachineBundleBefore(CurBundle);
|
|
|
|
} else if (!CurBundle)
|
2016-09-03 09:22:56 +08:00
|
|
|
report("No bundle header", &*MBBI);
|
2015-10-10 03:40:45 +08:00
|
|
|
visitMachineInstrBefore(&*MBBI);
|
2015-05-01 03:35:41 +08:00
|
|
|
for (unsigned I = 0, E = MBBI->getNumOperands(); I != E; ++I) {
|
|
|
|
const MachineInstr &MI = *MBBI;
|
|
|
|
const MachineOperand &Op = MI.getOperand(I);
|
|
|
|
if (Op.getParent() != &MI) {
|
2015-05-01 07:20:56 +08:00
|
|
|
// Make sure to use correct addOperand / RemoveOperand / ChangeTo
|
2015-05-01 03:35:41 +08:00
|
|
|
// functions when replacing operands of a MachineInstr.
|
|
|
|
report("Instruction has operand with wrong parent set", &MI);
|
|
|
|
}
|
|
|
|
|
|
|
|
visitMachineOperand(&Op, I);
|
|
|
|
}
|
|
|
|
|
2015-10-10 03:40:45 +08:00
|
|
|
visitMachineInstrAfter(&*MBBI);
|
2012-12-19 06:55:07 +08:00
|
|
|
|
|
|
|
// Was this the last bundled instruction?
|
|
|
|
InBundle = MBBI->isBundledWithSucc();
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
2012-06-07 06:34:30 +08:00
|
|
|
if (CurBundle)
|
|
|
|
visitMachineBundleAfter(CurBundle);
|
2012-12-19 06:55:07 +08:00
|
|
|
if (InBundle)
|
|
|
|
report("BundledSucc flag set on last instruction in block", &MFI->back());
|
2015-10-10 03:40:45 +08:00
|
|
|
visitMachineBasicBlockAfter(&*MFI);
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
visitMachineFunctionAfter();
|
|
|
|
|
2009-08-08 23:34:50 +08:00
|
|
|
// Clean up.
|
|
|
|
regsLive.clear();
|
|
|
|
regsDefined.clear();
|
|
|
|
regsDead.clear();
|
|
|
|
regsKilled.clear();
|
2012-02-28 09:42:41 +08:00
|
|
|
regMasks.clear();
|
2009-08-08 23:34:50 +08:00
|
|
|
MBBInfoMap.clear();
|
|
|
|
|
2016-02-16 03:25:31 +08:00
|
|
|
return foundErrors;
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
2009-08-23 09:03:30 +08:00
|
|
|
void MachineVerifier::report(const char *msg, const MachineFunction *MF) {
|
2009-05-16 08:33:53 +08:00
|
|
|
assert(MF);
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << '\n';
|
2010-12-18 08:06:56 +08:00
|
|
|
if (!foundErrors++) {
|
|
|
|
if (Banner)
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "# " << Banner << '\n';
|
2015-11-10 07:59:23 +08:00
|
|
|
if (LiveInts != nullptr)
|
|
|
|
LiveInts->print(errs());
|
|
|
|
else
|
|
|
|
MF->print(errs(), Indexes);
|
2010-12-18 08:06:56 +08:00
|
|
|
}
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "*** Bad machine code: " << msg << " ***\n"
|
2012-08-22 14:07:19 +08:00
|
|
|
<< "- function: " << MF->getName() << "\n";
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
2009-10-05 02:18:39 +08:00
|
|
|
void MachineVerifier::report(const char *msg, const MachineBasicBlock *MBB) {
|
2009-05-16 08:33:53 +08:00
|
|
|
assert(MBB);
|
|
|
|
report(msg, MBB->getParent());
|
2017-12-05 01:18:51 +08:00
|
|
|
errs() << "- basic block: " << printMBBReference(*MBB) << ' '
|
|
|
|
<< MBB->getName() << " (" << (const void *)MBB << ')';
|
2010-10-27 04:21:46 +08:00
|
|
|
if (Indexes)
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << " [" << Indexes->getMBBStartIdx(MBB)
|
2010-10-27 04:21:46 +08:00
|
|
|
<< ';' << Indexes->getMBBEndIdx(MBB) << ')';
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << '\n';
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
2009-10-05 02:18:39 +08:00
|
|
|
void MachineVerifier::report(const char *msg, const MachineInstr *MI) {
|
2009-05-16 08:33:53 +08:00
|
|
|
assert(MI);
|
|
|
|
report(msg, MI->getParent());
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "- instruction: ";
|
2016-02-27 14:40:41 +08:00
|
|
|
if (Indexes && Indexes->hasIndex(*MI))
|
|
|
|
errs() << Indexes->getInstructionIndex(*MI) << '\t';
|
2015-11-10 07:59:25 +08:00
|
|
|
MI->print(errs(), /*SkipOpers=*/true);
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
2018-05-08 06:31:12 +08:00
|
|
|
void MachineVerifier::report(const char *msg, const MachineOperand *MO,
|
|
|
|
unsigned MONum, LLT MOVRegType) {
|
2009-05-16 08:33:53 +08:00
|
|
|
assert(MO);
|
|
|
|
report(msg, MO->getParent());
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "- operand " << MONum << ": ";
|
2018-05-08 06:31:12 +08:00
|
|
|
MO->print(errs(), MOVRegType, TRI);
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "\n";
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
2016-02-02 10:44:25 +08:00
|
|
|
void MachineVerifier::report_context(SlotIndex Pos) const {
|
|
|
|
errs() << "- at: " << Pos << '\n';
|
|
|
|
}
|
|
|
|
|
2015-11-10 07:59:33 +08:00
|
|
|
void MachineVerifier::report_context(const LiveInterval &LI) const {
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "- interval: " << LI << '\n';
|
2012-08-02 22:31:49 +08:00
|
|
|
}
|
|
|
|
|
2016-07-26 03:39:01 +08:00
|
|
|
void MachineVerifier::report_context(const LiveRange &LR, unsigned VRegUnit,
|
2015-11-10 07:59:33 +08:00
|
|
|
LaneBitmask LaneMask) const {
|
2016-02-02 10:44:25 +08:00
|
|
|
report_context_liverange(LR);
|
2016-07-26 03:39:01 +08:00
|
|
|
report_context_vreg_regunit(VRegUnit);
|
2016-12-17 03:11:56 +08:00
|
|
|
if (LaneMask.any())
|
2016-02-03 04:04:51 +08:00
|
|
|
report_context_lanemask(LaneMask);
|
2013-10-11 05:28:54 +08:00
|
|
|
}
|
|
|
|
|
2015-11-10 07:59:33 +08:00
|
|
|
void MachineVerifier::report_context(const LiveRange::Segment &S) const {
|
|
|
|
errs() << "- segment: " << S << '\n';
|
|
|
|
}
|
|
|
|
|
|
|
|
void MachineVerifier::report_context(const VNInfo &VNI) const {
|
|
|
|
errs() << "- ValNo: " << VNI.id << " (def " << VNI.def << ")\n";
|
2013-10-11 05:28:54 +08:00
|
|
|
}
|
|
|
|
|
2016-02-02 10:44:25 +08:00
|
|
|
void MachineVerifier::report_context_liverange(const LiveRange &LR) const {
|
|
|
|
errs() << "- liverange: " << LR << '\n';
|
|
|
|
}
|
|
|
|
|
2019-01-08 23:16:23 +08:00
|
|
|
void MachineVerifier::report_context(MCPhysReg PReg) const {
|
|
|
|
errs() << "- p. register: " << printReg(PReg, TRI) << '\n';
|
|
|
|
}
|
|
|
|
|
2016-05-12 05:31:39 +08:00
|
|
|
void MachineVerifier::report_context_vreg(unsigned VReg) const {
|
2017-11-28 20:42:37 +08:00
|
|
|
errs() << "- v. register: " << printReg(VReg, TRI) << '\n';
|
2016-05-12 05:31:39 +08:00
|
|
|
}
|
|
|
|
|
2016-02-03 04:04:51 +08:00
|
|
|
void MachineVerifier::report_context_vreg_regunit(unsigned VRegOrUnit) const {
|
2019-08-02 07:27:28 +08:00
|
|
|
if (Register::isVirtualRegister(VRegOrUnit)) {
|
2016-05-12 05:31:39 +08:00
|
|
|
report_context_vreg(VRegOrUnit);
|
2016-02-03 04:04:51 +08:00
|
|
|
} else {
|
2017-11-28 20:42:37 +08:00
|
|
|
errs() << "- regunit: " << printRegUnit(VRegOrUnit, TRI) << '\n';
|
2016-02-03 04:04:51 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MachineVerifier::report_context_lanemask(LaneBitmask LaneMask) const {
|
|
|
|
errs() << "- lanemask: " << PrintLaneMask(LaneMask) << '\n';
|
|
|
|
}
|
|
|
|
|
2009-10-05 02:18:39 +08:00
|
|
|
void MachineVerifier::markReachable(const MachineBasicBlock *MBB) {
|
2009-05-16 08:33:53 +08:00
|
|
|
BBInfo &MInfo = MBBInfoMap[MBB];
|
|
|
|
if (!MInfo.reachable) {
|
|
|
|
MInfo.reachable = true;
|
|
|
|
for (MachineBasicBlock::const_succ_iterator SuI = MBB->succ_begin(),
|
|
|
|
SuE = MBB->succ_end(); SuI != SuE; ++SuI)
|
|
|
|
markReachable(*SuI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-05 02:18:39 +08:00
|
|
|
void MachineVerifier::visitMachineFunctionBefore() {
|
2011-01-13 05:27:48 +08:00
|
|
|
lastIndex = SlotIndex();
|
2017-05-06 06:04:05 +08:00
|
|
|
regsReserved = MRI->reservedRegsFrozen() ? MRI->getReservedRegs()
|
|
|
|
: TRI->getReservedRegs(*MF);
|
2009-08-05 03:18:01 +08:00
|
|
|
|
2017-04-12 03:32:41 +08:00
|
|
|
if (!MF->empty())
|
|
|
|
markReachable(&MF->front());
|
2012-08-21 04:52:06 +08:00
|
|
|
|
|
|
|
// Build a set of the basic blocks in the function.
|
|
|
|
FunctionBlocks.clear();
|
2014-05-01 02:29:51 +08:00
|
|
|
for (const auto &MBB : *MF) {
|
|
|
|
FunctionBlocks.insert(&MBB);
|
|
|
|
BBInfo &MInfo = MBBInfoMap[&MBB];
|
|
|
|
|
|
|
|
MInfo.Preds.insert(MBB.pred_begin(), MBB.pred_end());
|
|
|
|
if (MInfo.Preds.size() != MBB.pred_size())
|
|
|
|
report("MBB has duplicate entries in its predecessor list.", &MBB);
|
|
|
|
|
|
|
|
MInfo.Succs.insert(MBB.succ_begin(), MBB.succ_end());
|
|
|
|
if (MInfo.Succs.size() != MBB.succ_size())
|
|
|
|
report("MBB has duplicate entries in its successor list.", &MBB);
|
2012-08-21 04:52:06 +08:00
|
|
|
}
|
2013-04-20 05:40:57 +08:00
|
|
|
|
|
|
|
// Check that the register use lists are sane.
|
|
|
|
MRI->verifyUseLists();
|
2013-07-16 05:26:31 +08:00
|
|
|
|
2017-04-12 03:32:41 +08:00
|
|
|
if (!MF->empty())
|
|
|
|
verifyStackFrame();
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
2009-11-14 05:55:54 +08:00
|
|
|
// Does iterator point to a and b as the first two elements?
|
2010-04-16 01:08:50 +08:00
|
|
|
static bool matchPair(MachineBasicBlock::const_succ_iterator i,
|
|
|
|
const MachineBasicBlock *a, const MachineBasicBlock *b) {
|
2009-11-14 05:55:54 +08:00
|
|
|
if (*i == a)
|
|
|
|
return *++i == b;
|
|
|
|
if (*i == b)
|
|
|
|
return *++i == a;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
MachineVerifier::visitMachineBasicBlockBefore(const MachineBasicBlock *MBB) {
|
2014-04-14 08:51:57 +08:00
|
|
|
FirstTerminator = nullptr;
|
2018-10-04 06:05:31 +08:00
|
|
|
FirstNonPHI = nullptr;
|
2011-09-24 06:45:39 +08:00
|
|
|
|
2016-08-24 09:32:41 +08:00
|
|
|
if (!MF->getProperties().hasProperty(
|
2017-01-06 04:01:19 +08:00
|
|
|
MachineFunctionProperties::Property::NoPHIs) && MRI->tracksLiveness()) {
|
2012-02-15 03:17:48 +08:00
|
|
|
// If this block has allocatable physical registers live-in, check that
|
|
|
|
// it is an entry block or landing pad.
|
2015-09-10 02:08:03 +08:00
|
|
|
for (const auto &LI : MBB->liveins()) {
|
|
|
|
if (isAllocatable(LI.PhysReg) && !MBB->isEHPad() &&
|
Allow "callbr" to return non-void values
Summary:
Terminators in LLVM aren't prohibited from returning values. This means that
the "callbr" instruction, which is used for "asm goto", can support "asm goto
with outputs."
This patch removes all restrictions against "callbr" returning values. The
heavy lifting is done by the code generator. The "INLINEASM_BR" instruction's
a terminator, and the code generator doesn't allow non-terminator instructions
after a terminator. In order to correctly model the feature, we need to copy
outputs from "INLINEASM_BR" into virtual registers. Of course, those copies
aren't terminators.
To get around this issue, we split the block containing the "INLINEASM_BR"
right before the "COPY" instructions. This results in two cheats:
- Any physical registers defined by "INLINEASM_BR" need to be marked as
live-in into the block with the "COPY" instructions. This violates an
assumption that physical registers aren't marked as "live-in" until after
register allocation. But it seems as if the live-in information only
needs to be correct after register allocation. So we're able to get away
with this.
- The indirect branches from the "INLINEASM_BR" are moved to the "COPY"
block. This is to satisfy PHI nodes.
I've been told that MLIR can support this handily, but until we're able to
use it, we'll have to stick with the above.
Reviewers: jyknight, nickdesaulniers, hfinkel, MaskRay, lattner
Reviewed By: nickdesaulniers, MaskRay, lattner
Subscribers: rriddle, qcolombet, jdoerfert, MatzeB, echristo, MaskRay, xbolva00, aaron.ballman, cfe-commits, JonChesterfield, hiraditya, llvm-commits, rnk, craig.topper
Tags: #llvm, #clang
Differential Revision: https://reviews.llvm.org/D69868
2020-02-25 10:28:32 +08:00
|
|
|
!MBB->isInlineAsmBrDefaultTarget() &&
|
2016-02-22 04:39:50 +08:00
|
|
|
MBB->getIterator() != MBB->getParent()->begin()) {
|
2017-02-16 06:19:06 +08:00
|
|
|
report("MBB has allocatable live-in, but isn't entry or landing-pad.", MBB);
|
2019-01-08 23:16:23 +08:00
|
|
|
report_context(LI.PhysReg);
|
2012-02-15 03:17:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-10-22 02:47:06 +08:00
|
|
|
// Count the number of landing pad successors.
|
Allow "callbr" to return non-void values
Summary:
Terminators in LLVM aren't prohibited from returning values. This means that
the "callbr" instruction, which is used for "asm goto", can support "asm goto
with outputs."
This patch removes all restrictions against "callbr" returning values. The
heavy lifting is done by the code generator. The "INLINEASM_BR" instruction's
a terminator, and the code generator doesn't allow non-terminator instructions
after a terminator. In order to correctly model the feature, we need to copy
outputs from "INLINEASM_BR" into virtual registers. Of course, those copies
aren't terminators.
To get around this issue, we split the block containing the "INLINEASM_BR"
right before the "COPY" instructions. This results in two cheats:
- Any physical registers defined by "INLINEASM_BR" need to be marked as
live-in into the block with the "COPY" instructions. This violates an
assumption that physical registers aren't marked as "live-in" until after
register allocation. But it seems as if the live-in information only
needs to be correct after register allocation. So we're able to get away
with this.
- The indirect branches from the "INLINEASM_BR" are moved to the "COPY"
block. This is to satisfy PHI nodes.
I've been told that MLIR can support this handily, but until we're able to
use it, we'll have to stick with the above.
Reviewers: jyknight, nickdesaulniers, hfinkel, MaskRay, lattner
Reviewed By: nickdesaulniers, MaskRay, lattner
Subscribers: rriddle, qcolombet, jdoerfert, MatzeB, echristo, MaskRay, xbolva00, aaron.ballman, cfe-commits, JonChesterfield, hiraditya, llvm-commits, rnk, craig.topper
Tags: #llvm, #clang
Differential Revision: https://reviews.llvm.org/D69868
2020-02-25 10:28:32 +08:00
|
|
|
SmallPtrSet<const MachineBasicBlock*, 4> LandingPadSuccs;
|
|
|
|
for (const auto *succ : MBB->successors()) {
|
|
|
|
if (succ->isEHPad())
|
|
|
|
LandingPadSuccs.insert(succ);
|
|
|
|
if (!FunctionBlocks.count(succ))
|
2020-01-08 04:53:44 +08:00
|
|
|
report("MBB has successor that isn't part of the function.", MBB);
|
Allow "callbr" to return non-void values
Summary:
Terminators in LLVM aren't prohibited from returning values. This means that
the "callbr" instruction, which is used for "asm goto", can support "asm goto
with outputs."
This patch removes all restrictions against "callbr" returning values. The
heavy lifting is done by the code generator. The "INLINEASM_BR" instruction's
a terminator, and the code generator doesn't allow non-terminator instructions
after a terminator. In order to correctly model the feature, we need to copy
outputs from "INLINEASM_BR" into virtual registers. Of course, those copies
aren't terminators.
To get around this issue, we split the block containing the "INLINEASM_BR"
right before the "COPY" instructions. This results in two cheats:
- Any physical registers defined by "INLINEASM_BR" need to be marked as
live-in into the block with the "COPY" instructions. This violates an
assumption that physical registers aren't marked as "live-in" until after
register allocation. But it seems as if the live-in information only
needs to be correct after register allocation. So we're able to get away
with this.
- The indirect branches from the "INLINEASM_BR" are moved to the "COPY"
block. This is to satisfy PHI nodes.
I've been told that MLIR can support this handily, but until we're able to
use it, we'll have to stick with the above.
Reviewers: jyknight, nickdesaulniers, hfinkel, MaskRay, lattner
Reviewed By: nickdesaulniers, MaskRay, lattner
Subscribers: rriddle, qcolombet, jdoerfert, MatzeB, echristo, MaskRay, xbolva00, aaron.ballman, cfe-commits, JonChesterfield, hiraditya, llvm-commits, rnk, craig.topper
Tags: #llvm, #clang
Differential Revision: https://reviews.llvm.org/D69868
2020-02-25 10:28:32 +08:00
|
|
|
if (!MBBInfoMap[succ].Preds.count(MBB)) {
|
2020-01-08 04:53:44 +08:00
|
|
|
report("Inconsistent CFG", MBB);
|
|
|
|
errs() << "MBB is not in the predecessor list of the successor "
|
Allow "callbr" to return non-void values
Summary:
Terminators in LLVM aren't prohibited from returning values. This means that
the "callbr" instruction, which is used for "asm goto", can support "asm goto
with outputs."
This patch removes all restrictions against "callbr" returning values. The
heavy lifting is done by the code generator. The "INLINEASM_BR" instruction's
a terminator, and the code generator doesn't allow non-terminator instructions
after a terminator. In order to correctly model the feature, we need to copy
outputs from "INLINEASM_BR" into virtual registers. Of course, those copies
aren't terminators.
To get around this issue, we split the block containing the "INLINEASM_BR"
right before the "COPY" instructions. This results in two cheats:
- Any physical registers defined by "INLINEASM_BR" need to be marked as
live-in into the block with the "COPY" instructions. This violates an
assumption that physical registers aren't marked as "live-in" until after
register allocation. But it seems as if the live-in information only
needs to be correct after register allocation. So we're able to get away
with this.
- The indirect branches from the "INLINEASM_BR" are moved to the "COPY"
block. This is to satisfy PHI nodes.
I've been told that MLIR can support this handily, but until we're able to
use it, we'll have to stick with the above.
Reviewers: jyknight, nickdesaulniers, hfinkel, MaskRay, lattner
Reviewed By: nickdesaulniers, MaskRay, lattner
Subscribers: rriddle, qcolombet, jdoerfert, MatzeB, echristo, MaskRay, xbolva00, aaron.ballman, cfe-commits, JonChesterfield, hiraditya, llvm-commits, rnk, craig.topper
Tags: #llvm, #clang
Differential Revision: https://reviews.llvm.org/D69868
2020-02-25 10:28:32 +08:00
|
|
|
<< printMBBReference(*succ) << ".\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Count the number of INLINEASM_BR indirect target successors.
|
|
|
|
SmallPtrSet<const MachineBasicBlock*, 4> IndirectTargetSuccs;
|
|
|
|
for (const auto *succ : MBB->successors()) {
|
|
|
|
if (MBB->isInlineAsmBrIndirectTarget(succ))
|
|
|
|
IndirectTargetSuccs.insert(succ);
|
|
|
|
if (!FunctionBlocks.count(succ))
|
|
|
|
report("MBB has successor that isn't part of the function.", MBB);
|
|
|
|
if (!MBBInfoMap[succ].Preds.count(MBB)) {
|
|
|
|
report("Inconsistent CFG", MBB);
|
|
|
|
errs() << "MBB is not in the predecessor list of the successor "
|
|
|
|
<< printMBBReference(*succ) << ".\n";
|
2012-08-21 04:52:06 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check the predecessor list.
|
|
|
|
for (MachineBasicBlock::const_pred_iterator I = MBB->pred_begin(),
|
|
|
|
E = MBB->pred_end(); I != E; ++I) {
|
|
|
|
if (!FunctionBlocks.count(*I))
|
|
|
|
report("MBB has predecessor that isn't part of the function.", MBB);
|
|
|
|
if (!MBBInfoMap[*I].Succs.count(MBB)) {
|
|
|
|
report("Inconsistent CFG", MBB);
|
2017-12-05 01:18:51 +08:00
|
|
|
errs() << "MBB is not in the successor list of the predecessor "
|
|
|
|
<< printMBBReference(*(*I)) << ".\n";
|
2012-08-21 04:52:06 +08:00
|
|
|
}
|
2010-12-20 12:19:48 +08:00
|
|
|
}
|
2011-05-05 06:54:05 +08:00
|
|
|
|
|
|
|
const MCAsmInfo *AsmInfo = TM->getMCAsmInfo();
|
|
|
|
const BasicBlock *BB = MBB->getBasicBlock();
|
2017-12-16 06:22:58 +08:00
|
|
|
const Function &F = MF->getFunction();
|
2011-05-05 06:54:05 +08:00
|
|
|
if (LandingPadSuccs.size() > 1 &&
|
|
|
|
!(AsmInfo &&
|
|
|
|
AsmInfo->getExceptionHandlingType() == ExceptionHandling::SjLj &&
|
2015-11-10 05:04:00 +08:00
|
|
|
BB && isa<SwitchInst>(BB->getTerminator())) &&
|
[WebAssembly] Add Wasm personality and isScopedEHPersonality()
Summary:
- Add wasm personality function
- Re-categorize the existing `isFuncletEHPersonality()` function into
two different functions: `isFuncletEHPersonality()` and
`isScopedEHPersonality(). This becomes necessary as wasm EH uses scoped
EH instructions (catchswitch, catchpad/ret, and cleanuppad/ret) but not
outlined funclets.
- Changed some callsites of `isFuncletEHPersonality()` to
`isScopedEHPersonality()` if they are related to scoped EH IR-level
stuff.
Reviewers: majnemer, dschuff, rnk
Subscribers: jfb, sbc100, jgravelle-google, eraman, JDevlieghere, sunfish, llvm-commits
Differential Revision: https://reviews.llvm.org/D45559
llvm-svn: 332667
2018-05-18 04:52:03 +08:00
|
|
|
!isScopedEHPersonality(classifyEHPersonality(F.getPersonalityFn())))
|
2010-10-22 02:47:06 +08:00
|
|
|
report("MBB has more than one landing pad successor", MBB);
|
|
|
|
|
2020-01-21 23:47:35 +08:00
|
|
|
// Call analyzeBranch. If it succeeds, there several more conditions to check.
|
2014-04-14 08:51:57 +08:00
|
|
|
MachineBasicBlock *TBB = nullptr, *FBB = nullptr;
|
2009-08-27 10:43:49 +08:00
|
|
|
SmallVector<MachineOperand, 4> Cond;
|
2016-07-15 22:41:04 +08:00
|
|
|
if (!TII->analyzeBranch(*const_cast<MachineBasicBlock *>(MBB), TBB, FBB,
|
|
|
|
Cond)) {
|
2020-01-21 23:47:35 +08:00
|
|
|
// Ok, analyzeBranch thinks it knows what's going on with this block. Let's
|
2009-08-27 10:43:49 +08:00
|
|
|
// check whether its answers match up with reality.
|
|
|
|
if (!TBB && !FBB) {
|
|
|
|
// Block falls through to its successor.
|
2015-10-10 03:40:45 +08:00
|
|
|
MachineFunction::const_iterator MBBI = MBB->getIterator();
|
2009-08-27 10:43:49 +08:00
|
|
|
++MBBI;
|
|
|
|
if (MBBI == MF->end()) {
|
2009-08-28 02:14:26 +08:00
|
|
|
// It's possible that the block legitimately ends with a noreturn
|
|
|
|
// call or an unreachable, in which case it won't actually fall
|
|
|
|
// out the bottom of the function.
|
Allow "callbr" to return non-void values
Summary:
Terminators in LLVM aren't prohibited from returning values. This means that
the "callbr" instruction, which is used for "asm goto", can support "asm goto
with outputs."
This patch removes all restrictions against "callbr" returning values. The
heavy lifting is done by the code generator. The "INLINEASM_BR" instruction's
a terminator, and the code generator doesn't allow non-terminator instructions
after a terminator. In order to correctly model the feature, we need to copy
outputs from "INLINEASM_BR" into virtual registers. Of course, those copies
aren't terminators.
To get around this issue, we split the block containing the "INLINEASM_BR"
right before the "COPY" instructions. This results in two cheats:
- Any physical registers defined by "INLINEASM_BR" need to be marked as
live-in into the block with the "COPY" instructions. This violates an
assumption that physical registers aren't marked as "live-in" until after
register allocation. But it seems as if the live-in information only
needs to be correct after register allocation. So we're able to get away
with this.
- The indirect branches from the "INLINEASM_BR" are moved to the "COPY"
block. This is to satisfy PHI nodes.
I've been told that MLIR can support this handily, but until we're able to
use it, we'll have to stick with the above.
Reviewers: jyknight, nickdesaulniers, hfinkel, MaskRay, lattner
Reviewed By: nickdesaulniers, MaskRay, lattner
Subscribers: rriddle, qcolombet, jdoerfert, MatzeB, echristo, MaskRay, xbolva00, aaron.ballman, cfe-commits, JonChesterfield, hiraditya, llvm-commits, rnk, craig.topper
Tags: #llvm, #clang
Differential Revision: https://reviews.llvm.org/D69868
2020-02-25 10:28:32 +08:00
|
|
|
} else if (MBB->succ_size() == LandingPadSuccs.size() ||
|
|
|
|
MBB->succ_size() == IndirectTargetSuccs.size()) {
|
2009-08-28 02:14:26 +08:00
|
|
|
// It's possible that the block legitimately ends with a noreturn
|
2019-01-09 13:11:10 +08:00
|
|
|
// call or an unreachable, in which case it won't actually fall
|
2009-08-28 02:14:26 +08:00
|
|
|
// out of the block.
|
Allow "callbr" to return non-void values
Summary:
Terminators in LLVM aren't prohibited from returning values. This means that
the "callbr" instruction, which is used for "asm goto", can support "asm goto
with outputs."
This patch removes all restrictions against "callbr" returning values. The
heavy lifting is done by the code generator. The "INLINEASM_BR" instruction's
a terminator, and the code generator doesn't allow non-terminator instructions
after a terminator. In order to correctly model the feature, we need to copy
outputs from "INLINEASM_BR" into virtual registers. Of course, those copies
aren't terminators.
To get around this issue, we split the block containing the "INLINEASM_BR"
right before the "COPY" instructions. This results in two cheats:
- Any physical registers defined by "INLINEASM_BR" need to be marked as
live-in into the block with the "COPY" instructions. This violates an
assumption that physical registers aren't marked as "live-in" until after
register allocation. But it seems as if the live-in information only
needs to be correct after register allocation. So we're able to get away
with this.
- The indirect branches from the "INLINEASM_BR" are moved to the "COPY"
block. This is to satisfy PHI nodes.
I've been told that MLIR can support this handily, but until we're able to
use it, we'll have to stick with the above.
Reviewers: jyknight, nickdesaulniers, hfinkel, MaskRay, lattner
Reviewed By: nickdesaulniers, MaskRay, lattner
Subscribers: rriddle, qcolombet, jdoerfert, MatzeB, echristo, MaskRay, xbolva00, aaron.ballman, cfe-commits, JonChesterfield, hiraditya, llvm-commits, rnk, craig.topper
Tags: #llvm, #clang
Differential Revision: https://reviews.llvm.org/D69868
2020-02-25 10:28:32 +08:00
|
|
|
} else if ((LandingPadSuccs.size() &&
|
|
|
|
MBB->succ_size() != 1 + LandingPadSuccs.size()) ||
|
|
|
|
(IndirectTargetSuccs.size() &&
|
|
|
|
MBB->succ_size() != 1 + IndirectTargetSuccs.size())) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via unconditional fall-through but doesn't have "
|
|
|
|
"exactly one CFG successor!", MBB);
|
2015-10-10 03:40:45 +08:00
|
|
|
} else if (!MBB->isSuccessor(&*MBBI)) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via unconditional fall-through but its successor "
|
|
|
|
"differs from its CFG successor!", MBB);
|
|
|
|
}
|
2014-05-24 21:31:10 +08:00
|
|
|
if (!MBB->empty() && MBB->back().isBarrier() &&
|
2016-02-23 10:46:52 +08:00
|
|
|
!TII->isPredicated(MBB->back())) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via unconditional fall-through but ends with a "
|
|
|
|
"barrier instruction!", MBB);
|
|
|
|
}
|
|
|
|
if (!Cond.empty()) {
|
|
|
|
report("MBB exits via unconditional fall-through but has a condition!",
|
|
|
|
MBB);
|
|
|
|
}
|
|
|
|
} else if (TBB && !FBB && Cond.empty()) {
|
|
|
|
// Block unconditionally branches somewhere.
|
2014-12-02 02:43:53 +08:00
|
|
|
// If the block has exactly one successor, that happens to be a
|
|
|
|
// landingpad, accept it as valid control flow.
|
|
|
|
if (MBB->succ_size() != 1+LandingPadSuccs.size() &&
|
|
|
|
(MBB->succ_size() != 1 || LandingPadSuccs.size() != 1 ||
|
Allow "callbr" to return non-void values
Summary:
Terminators in LLVM aren't prohibited from returning values. This means that
the "callbr" instruction, which is used for "asm goto", can support "asm goto
with outputs."
This patch removes all restrictions against "callbr" returning values. The
heavy lifting is done by the code generator. The "INLINEASM_BR" instruction's
a terminator, and the code generator doesn't allow non-terminator instructions
after a terminator. In order to correctly model the feature, we need to copy
outputs from "INLINEASM_BR" into virtual registers. Of course, those copies
aren't terminators.
To get around this issue, we split the block containing the "INLINEASM_BR"
right before the "COPY" instructions. This results in two cheats:
- Any physical registers defined by "INLINEASM_BR" need to be marked as
live-in into the block with the "COPY" instructions. This violates an
assumption that physical registers aren't marked as "live-in" until after
register allocation. But it seems as if the live-in information only
needs to be correct after register allocation. So we're able to get away
with this.
- The indirect branches from the "INLINEASM_BR" are moved to the "COPY"
block. This is to satisfy PHI nodes.
I've been told that MLIR can support this handily, but until we're able to
use it, we'll have to stick with the above.
Reviewers: jyknight, nickdesaulniers, hfinkel, MaskRay, lattner
Reviewed By: nickdesaulniers, MaskRay, lattner
Subscribers: rriddle, qcolombet, jdoerfert, MatzeB, echristo, MaskRay, xbolva00, aaron.ballman, cfe-commits, JonChesterfield, hiraditya, llvm-commits, rnk, craig.topper
Tags: #llvm, #clang
Differential Revision: https://reviews.llvm.org/D69868
2020-02-25 10:28:32 +08:00
|
|
|
*MBB->succ_begin() != *LandingPadSuccs.begin()) &&
|
|
|
|
MBB->succ_size() != 1 + IndirectTargetSuccs.size() &&
|
|
|
|
(MBB->succ_size() != 1 || IndirectTargetSuccs.size() != 1 ||
|
|
|
|
*MBB->succ_begin() != *IndirectTargetSuccs.begin())) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via unconditional branch but doesn't have "
|
|
|
|
"exactly one CFG successor!", MBB);
|
2010-10-22 02:47:06 +08:00
|
|
|
} else if (!MBB->isSuccessor(TBB)) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via unconditional branch but the CFG "
|
|
|
|
"successor doesn't match the actual successor!", MBB);
|
|
|
|
}
|
|
|
|
if (MBB->empty()) {
|
|
|
|
report("MBB exits via unconditional branch but doesn't contain "
|
|
|
|
"any instructions!", MBB);
|
2014-05-24 21:31:10 +08:00
|
|
|
} else if (!MBB->back().isBarrier()) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via unconditional branch but doesn't end with a "
|
|
|
|
"barrier instruction!", MBB);
|
2014-05-24 21:31:10 +08:00
|
|
|
} else if (!MBB->back().isTerminator()) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via unconditional branch but the branch isn't a "
|
|
|
|
"terminator instruction!", MBB);
|
|
|
|
}
|
|
|
|
} else if (TBB && !FBB && !Cond.empty()) {
|
|
|
|
// Block conditionally branches somewhere, otherwise falls through.
|
2015-10-10 03:40:45 +08:00
|
|
|
MachineFunction::const_iterator MBBI = MBB->getIterator();
|
2009-08-27 10:43:49 +08:00
|
|
|
++MBBI;
|
|
|
|
if (MBBI == MF->end()) {
|
|
|
|
report("MBB conditionally falls through out of function!", MBB);
|
2012-12-20 06:13:01 +08:00
|
|
|
} else if (MBB->succ_size() == 1) {
|
2012-08-21 05:39:52 +08:00
|
|
|
// A conditional branch with only one successor is weird, but allowed.
|
|
|
|
if (&*MBBI != TBB)
|
|
|
|
report("MBB exits via conditional branch/fall-through but only has "
|
|
|
|
"one CFG successor!", MBB);
|
|
|
|
else if (TBB != *MBB->succ_begin())
|
|
|
|
report("MBB exits via conditional branch/fall-through but the CFG "
|
|
|
|
"successor don't match the actual successor!", MBB);
|
|
|
|
} else if (MBB->succ_size() != 2) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via conditional branch/fall-through but doesn't have "
|
|
|
|
"exactly two CFG successors!", MBB);
|
2015-10-10 03:40:45 +08:00
|
|
|
} else if (!matchPair(MBB->succ_begin(), TBB, &*MBBI)) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via conditional branch/fall-through but the CFG "
|
|
|
|
"successors don't match the actual successors!", MBB);
|
|
|
|
}
|
|
|
|
if (MBB->empty()) {
|
|
|
|
report("MBB exits via conditional branch/fall-through but doesn't "
|
|
|
|
"contain any instructions!", MBB);
|
2014-05-24 21:31:10 +08:00
|
|
|
} else if (MBB->back().isBarrier()) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via conditional branch/fall-through but ends with a "
|
|
|
|
"barrier instruction!", MBB);
|
2014-05-24 21:31:10 +08:00
|
|
|
} else if (!MBB->back().isTerminator()) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via conditional branch/fall-through but the branch "
|
|
|
|
"isn't a terminator instruction!", MBB);
|
|
|
|
}
|
|
|
|
} else if (TBB && FBB) {
|
|
|
|
// Block conditionally branches somewhere, otherwise branches
|
|
|
|
// somewhere else.
|
2012-08-21 05:39:52 +08:00
|
|
|
if (MBB->succ_size() == 1) {
|
|
|
|
// A conditional branch with only one successor is weird, but allowed.
|
|
|
|
if (FBB != TBB)
|
|
|
|
report("MBB exits via conditional branch/branch through but only has "
|
|
|
|
"one CFG successor!", MBB);
|
|
|
|
else if (TBB != *MBB->succ_begin())
|
|
|
|
report("MBB exits via conditional branch/branch through but the CFG "
|
|
|
|
"successor don't match the actual successor!", MBB);
|
|
|
|
} else if (MBB->succ_size() != 2) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via conditional branch/branch but doesn't have "
|
|
|
|
"exactly two CFG successors!", MBB);
|
2009-11-14 05:55:54 +08:00
|
|
|
} else if (!matchPair(MBB->succ_begin(), TBB, FBB)) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via conditional branch/branch but the CFG "
|
|
|
|
"successors don't match the actual successors!", MBB);
|
|
|
|
}
|
|
|
|
if (MBB->empty()) {
|
|
|
|
report("MBB exits via conditional branch/branch but doesn't "
|
|
|
|
"contain any instructions!", MBB);
|
2014-05-24 21:13:17 +08:00
|
|
|
} else if (!MBB->back().isBarrier()) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via conditional branch/branch but doesn't end with a "
|
|
|
|
"barrier instruction!", MBB);
|
2014-05-24 21:13:17 +08:00
|
|
|
} else if (!MBB->back().isTerminator()) {
|
2009-08-27 10:43:49 +08:00
|
|
|
report("MBB exits via conditional branch/branch but the branch "
|
|
|
|
"isn't a terminator instruction!", MBB);
|
|
|
|
}
|
|
|
|
if (Cond.empty()) {
|
2018-10-24 05:23:52 +08:00
|
|
|
report("MBB exits via conditional branch/branch but there's no "
|
2009-08-27 10:43:49 +08:00
|
|
|
"condition!", MBB);
|
|
|
|
}
|
|
|
|
} else {
|
2020-01-21 23:47:35 +08:00
|
|
|
report("analyzeBranch returned invalid data!", MBB);
|
2009-08-27 10:43:49 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
regsLive.clear();
|
2017-01-06 04:01:19 +08:00
|
|
|
if (MRI->tracksLiveness()) {
|
|
|
|
for (const auto &LI : MBB->liveins()) {
|
2019-08-02 07:27:28 +08:00
|
|
|
if (!Register::isPhysicalRegister(LI.PhysReg)) {
|
2017-01-06 04:01:19 +08:00
|
|
|
report("MBB live-in list contains non-physical register", MBB);
|
|
|
|
continue;
|
|
|
|
}
|
2019-12-05 17:16:08 +08:00
|
|
|
for (const MCPhysReg &SubReg : TRI->subregs_inclusive(LI.PhysReg))
|
|
|
|
regsLive.insert(SubReg);
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
}
|
2009-08-14 00:19:51 +08:00
|
|
|
|
2016-07-29 02:40:00 +08:00
|
|
|
const MachineFrameInfo &MFI = MF->getFrameInfo();
|
|
|
|
BitVector PR = MFI.getPristineRegs(*MF);
|
2017-05-17 09:07:53 +08:00
|
|
|
for (unsigned I : PR.set_bits()) {
|
2019-12-05 17:16:08 +08:00
|
|
|
for (const MCPhysReg &SubReg : TRI->subregs_inclusive(I))
|
|
|
|
regsLive.insert(SubReg);
|
2009-08-14 00:19:51 +08:00
|
|
|
}
|
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
regsKilled.clear();
|
|
|
|
regsDefined.clear();
|
2011-01-13 05:27:48 +08:00
|
|
|
|
|
|
|
if (Indexes)
|
|
|
|
lastIndex = Indexes->getMBBStartIdx(MBB);
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
2012-06-07 06:34:30 +08:00
|
|
|
// This function gets called for all bundle headers, including normal
|
|
|
|
// stand-alone unbundled instructions.
|
|
|
|
void MachineVerifier::visitMachineBundleBefore(const MachineInstr *MI) {
|
2016-02-27 14:40:41 +08:00
|
|
|
if (Indexes && Indexes->hasIndex(*MI)) {
|
|
|
|
SlotIndex idx = Indexes->getInstructionIndex(*MI);
|
2012-06-07 06:34:30 +08:00
|
|
|
if (!(idx > lastIndex)) {
|
|
|
|
report("Instruction index out of order", MI);
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "Last instruction was at " << lastIndex << '\n';
|
2012-06-07 06:34:30 +08:00
|
|
|
}
|
|
|
|
lastIndex = idx;
|
|
|
|
}
|
2012-06-08 01:41:39 +08:00
|
|
|
|
|
|
|
// Ensure non-terminators don't follow terminators.
|
|
|
|
// Ignore predicated terminators formed by if conversion.
|
|
|
|
// FIXME: If conversion shouldn't need to violate this rule.
|
2016-02-23 10:46:52 +08:00
|
|
|
if (MI->isTerminator() && !TII->isPredicated(*MI)) {
|
2012-06-08 01:41:39 +08:00
|
|
|
if (!FirstTerminator)
|
|
|
|
FirstTerminator = MI;
|
2020-03-04 03:27:09 +08:00
|
|
|
} else if (FirstTerminator) {
|
2020-01-08 05:43:04 +08:00
|
|
|
report("Non-terminator instruction after the first terminator", MI);
|
|
|
|
errs() << "First terminator was:\t" << *FirstTerminator;
|
2012-06-08 01:41:39 +08:00
|
|
|
}
|
2012-06-07 06:34:30 +08:00
|
|
|
}
|
|
|
|
|
2012-08-30 02:11:05 +08:00
|
|
|
// The operands on an INLINEASM instruction must follow a template.
|
|
|
|
// Verify that the flag operands make sense.
|
|
|
|
void MachineVerifier::verifyInlineAsm(const MachineInstr *MI) {
|
|
|
|
// The first two operands on INLINEASM are the asm string and global flags.
|
|
|
|
if (MI->getNumOperands() < 2) {
|
|
|
|
report("Too few operands on inline asm", MI);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!MI->getOperand(0).isSymbol())
|
|
|
|
report("Asm string must be an external symbol", MI);
|
|
|
|
if (!MI->getOperand(1).isImm())
|
|
|
|
report("Asm flags must be an immediate", MI);
|
2012-10-31 03:11:54 +08:00
|
|
|
// Allowed flags are Extra_HasSideEffects = 1, Extra_IsAlignStack = 2,
|
2016-06-23 02:51:08 +08:00
|
|
|
// Extra_AsmDialect = 4, Extra_MayLoad = 8, and Extra_MayStore = 16,
|
|
|
|
// and Extra_IsConvergent = 32.
|
|
|
|
if (!isUInt<6>(MI->getOperand(1).getImm()))
|
2012-08-30 02:11:05 +08:00
|
|
|
report("Unknown asm flags", &MI->getOperand(1), 1);
|
|
|
|
|
2015-03-16 17:53:42 +08:00
|
|
|
static_assert(InlineAsm::MIOp_FirstOperand == 2, "Asm format changed");
|
2012-08-30 02:11:05 +08:00
|
|
|
|
|
|
|
unsigned OpNo = InlineAsm::MIOp_FirstOperand;
|
|
|
|
unsigned NumOps;
|
|
|
|
for (unsigned e = MI->getNumOperands(); OpNo < e; OpNo += NumOps) {
|
|
|
|
const MachineOperand &MO = MI->getOperand(OpNo);
|
|
|
|
// There may be implicit ops after the fixed operands.
|
|
|
|
if (!MO.isImm())
|
|
|
|
break;
|
|
|
|
NumOps = 1 + InlineAsm::getNumOperandRegisters(MO.getImm());
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
2009-10-08 01:36:00 +08:00
|
|
|
|
2012-08-30 02:11:05 +08:00
|
|
|
if (OpNo > MI->getNumOperands())
|
|
|
|
report("Missing operands in last group", MI);
|
|
|
|
|
|
|
|
// An optional MDNode follows the groups.
|
|
|
|
if (OpNo < MI->getNumOperands() && MI->getOperand(OpNo).isMetadata())
|
|
|
|
++OpNo;
|
|
|
|
|
|
|
|
// All trailing operands must be implicit registers.
|
|
|
|
for (unsigned e = MI->getNumOperands(); OpNo < e; ++OpNo) {
|
|
|
|
const MachineOperand &MO = MI->getOperand(OpNo);
|
|
|
|
if (!MO.isReg() || !MO.isImplicit())
|
|
|
|
report("Expected implicit register after groups", &MO, OpNo);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-05 07:29:16 +08:00
|
|
|
/// Check that types are consistent when two operands need to have the same
|
|
|
|
/// number of vector elements.
|
|
|
|
/// \return true if the types are valid.
|
|
|
|
bool MachineVerifier::verifyVectorElementMatch(LLT Ty0, LLT Ty1,
|
|
|
|
const MachineInstr *MI) {
|
|
|
|
if (Ty0.isVector() != Ty1.isVector()) {
|
|
|
|
report("operand types must be all-vector or all-scalar", MI);
|
|
|
|
// Generally we try to report as many issues as possible at once, but in
|
|
|
|
// this case it's not clear what should we be comparing the size of the
|
|
|
|
// scalar with: the size of the whole vector or its lane. Instead of
|
|
|
|
// making an arbitrary choice and emitting not so helpful message, let's
|
|
|
|
// avoid the extra noise and stop here.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Ty0.isVector() && Ty0.getNumElements() != Ty1.getNumElements()) {
|
|
|
|
report("operand types must preserve number of vector elements", MI);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-02-05 07:29:11 +08:00
|
|
|
void MachineVerifier::verifyPreISelGenericInstruction(const MachineInstr *MI) {
|
|
|
|
if (isFunctionSelected)
|
|
|
|
report("Unexpected generic instruction in a Selected function", MI);
|
2016-08-24 05:19:49 +08:00
|
|
|
|
2019-02-05 07:29:11 +08:00
|
|
|
const MCInstrDesc &MCID = MI->getDesc();
|
|
|
|
unsigned NumOps = MI->getNumOperands();
|
2012-08-29 08:38:03 +08:00
|
|
|
|
2019-02-05 07:29:11 +08:00
|
|
|
// Check types.
|
|
|
|
SmallVector<LLT, 4> Types;
|
|
|
|
for (unsigned I = 0, E = std::min(MCID.getNumOperands(), NumOps);
|
2018-05-08 06:31:12 +08:00
|
|
|
I != E; ++I) {
|
2019-02-05 07:29:11 +08:00
|
|
|
if (!MCID.OpInfo[I].isGenericType())
|
|
|
|
continue;
|
|
|
|
// Generic instructions specify type equality constraints between some of
|
|
|
|
// their operands. Make sure these are consistent.
|
|
|
|
size_t TypeIdx = MCID.OpInfo[I].getGenericTypeIndex();
|
|
|
|
Types.resize(std::max(TypeIdx + 1, Types.size()));
|
|
|
|
|
|
|
|
const MachineOperand *MO = &MI->getOperand(I);
|
2019-02-05 08:53:22 +08:00
|
|
|
if (!MO->isReg()) {
|
|
|
|
report("generic instruction must use register operands", MI);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-02-05 07:29:11 +08:00
|
|
|
LLT OpTy = MRI->getType(MO->getReg());
|
|
|
|
// Don't report a type mismatch if there is no actual mismatch, only a
|
|
|
|
// type missing, to reduce noise:
|
|
|
|
if (OpTy.isValid()) {
|
|
|
|
// Only the first valid type for a type index will be printed: don't
|
|
|
|
// overwrite it later so it's always clear which type was expected:
|
|
|
|
if (!Types[TypeIdx].isValid())
|
|
|
|
Types[TypeIdx] = OpTy;
|
|
|
|
else if (Types[TypeIdx] != OpTy)
|
|
|
|
report("Type mismatch in generic instruction", MO, I, OpTy);
|
2010-08-06 06:32:21 +08:00
|
|
|
} else {
|
2019-02-05 07:29:11 +08:00
|
|
|
// Generic instructions must have types attached to their operands.
|
|
|
|
report("Generic instruction is missing a virtual register type", MO, I);
|
2010-08-06 06:32:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-05 07:29:11 +08:00
|
|
|
// Generic opcodes must not have physical register operands.
|
|
|
|
for (unsigned I = 0; I < MI->getNumOperands(); ++I) {
|
|
|
|
const MachineOperand *MO = &MI->getOperand(I);
|
2019-08-02 07:27:28 +08:00
|
|
|
if (MO->isReg() && Register::isPhysicalRegister(MO->getReg()))
|
2019-02-05 07:29:11 +08:00
|
|
|
report("Generic instruction cannot have physical register", MO, I);
|
2016-08-31 02:52:46 +08:00
|
|
|
}
|
|
|
|
|
2019-02-05 07:29:11 +08:00
|
|
|
// Avoid out of bounds in checks below. This was already reported earlier.
|
|
|
|
if (MI->getNumOperands() < MCID.getNumOperands())
|
|
|
|
return;
|
|
|
|
|
2011-09-21 10:20:46 +08:00
|
|
|
StringRef ErrorInfo;
|
2016-06-30 08:01:54 +08:00
|
|
|
if (!TII->verifyInstruction(*MI, ErrorInfo))
|
2011-09-21 10:20:46 +08:00
|
|
|
report(ErrorInfo.data(), MI);
|
2017-06-03 00:36:37 +08:00
|
|
|
|
|
|
|
// Verify properties of various specific instruction types
|
2019-02-05 07:29:11 +08:00
|
|
|
switch (MI->getOpcode()) {
|
2019-01-23 02:53:41 +08:00
|
|
|
case TargetOpcode::G_CONSTANT:
|
|
|
|
case TargetOpcode::G_FCONSTANT: {
|
|
|
|
if (MI->getNumOperands() < MCID.getNumOperands())
|
|
|
|
break;
|
|
|
|
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
if (DstTy.isVector())
|
|
|
|
report("Instruction cannot use a vector result type", MI);
|
2019-02-05 07:29:31 +08:00
|
|
|
|
|
|
|
if (MI->getOpcode() == TargetOpcode::G_CONSTANT) {
|
|
|
|
if (!MI->getOperand(1).isCImm()) {
|
|
|
|
report("G_CONSTANT operand must be cimm", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ConstantInt *CI = MI->getOperand(1).getCImm();
|
|
|
|
if (CI->getBitWidth() != DstTy.getSizeInBits())
|
|
|
|
report("inconsistent constant size", MI);
|
|
|
|
} else {
|
|
|
|
if (!MI->getOperand(1).isFPImm()) {
|
|
|
|
report("G_FCONSTANT operand must be fpimm", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
const ConstantFP *CF = MI->getOperand(1).getFPImm();
|
|
|
|
|
|
|
|
if (APFloat::getSizeInBits(CF->getValueAPF().getSemantics()) !=
|
|
|
|
DstTy.getSizeInBits()) {
|
|
|
|
report("inconsistent constant size", MI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-23 02:53:41 +08:00
|
|
|
break;
|
|
|
|
}
|
2017-06-03 00:36:37 +08:00
|
|
|
case TargetOpcode::G_LOAD:
|
|
|
|
case TargetOpcode::G_STORE:
|
2019-01-27 19:34:41 +08:00
|
|
|
case TargetOpcode::G_ZEXTLOAD:
|
2019-01-27 23:57:23 +08:00
|
|
|
case TargetOpcode::G_SEXTLOAD: {
|
2019-01-30 09:10:42 +08:00
|
|
|
LLT ValTy = MRI->getType(MI->getOperand(0).getReg());
|
2019-01-27 23:57:23 +08:00
|
|
|
LLT PtrTy = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
if (!PtrTy.isPointer())
|
|
|
|
report("Generic memory instruction must access a pointer", MI);
|
|
|
|
|
2017-06-03 00:36:37 +08:00
|
|
|
// Generic loads and stores must have a single MachineMemOperand
|
|
|
|
// describing that access.
|
2019-01-27 19:34:41 +08:00
|
|
|
if (!MI->hasOneMemOperand()) {
|
2017-06-03 00:36:37 +08:00
|
|
|
report("Generic instruction accessing memory must have one mem operand",
|
|
|
|
MI);
|
2019-01-27 19:34:41 +08:00
|
|
|
} else {
|
2019-01-30 09:10:42 +08:00
|
|
|
const MachineMemOperand &MMO = **MI->memoperands_begin();
|
2019-01-27 19:34:41 +08:00
|
|
|
if (MI->getOpcode() == TargetOpcode::G_ZEXTLOAD ||
|
|
|
|
MI->getOpcode() == TargetOpcode::G_SEXTLOAD) {
|
2019-04-18 06:21:05 +08:00
|
|
|
if (MMO.getSizeInBits() >= ValTy.getSizeInBits())
|
2019-01-27 19:34:41 +08:00
|
|
|
report("Generic extload must have a narrower memory type", MI);
|
2019-01-30 09:10:42 +08:00
|
|
|
} else if (MI->getOpcode() == TargetOpcode::G_LOAD) {
|
2019-04-18 06:21:05 +08:00
|
|
|
if (MMO.getSize() > ValTy.getSizeInBytes())
|
2019-01-30 09:10:42 +08:00
|
|
|
report("load memory size cannot exceed result size", MI);
|
|
|
|
} else if (MI->getOpcode() == TargetOpcode::G_STORE) {
|
2019-04-18 06:21:05 +08:00
|
|
|
if (ValTy.getSizeInBytes() < MMO.getSize())
|
2019-01-30 09:10:42 +08:00
|
|
|
report("store memory size cannot exceed value size", MI);
|
2019-01-27 19:34:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-03 00:36:37 +08:00
|
|
|
break;
|
2019-01-27 23:57:23 +08:00
|
|
|
}
|
2017-08-24 04:45:48 +08:00
|
|
|
case TargetOpcode::G_PHI: {
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
if (!DstTy.isValid() ||
|
|
|
|
!std::all_of(MI->operands_begin() + 1, MI->operands_end(),
|
|
|
|
[this, &DstTy](const MachineOperand &MO) {
|
|
|
|
if (!MO.isReg())
|
|
|
|
return true;
|
|
|
|
LLT Ty = MRI->getType(MO.getReg());
|
|
|
|
if (!Ty.isValid() || (Ty != DstTy))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}))
|
|
|
|
report("Generic Instruction G_PHI has operands with incompatible/missing "
|
|
|
|
"types",
|
|
|
|
MI);
|
|
|
|
break;
|
|
|
|
}
|
2019-01-19 05:04:59 +08:00
|
|
|
case TargetOpcode::G_BITCAST: {
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT SrcTy = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
if (!DstTy.isValid() || !SrcTy.isValid())
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (SrcTy.isPointer() != DstTy.isPointer())
|
|
|
|
report("bitcast cannot convert between pointers and other types", MI);
|
|
|
|
|
|
|
|
if (SrcTy.getSizeInBits() != DstTy.getSizeInBits())
|
|
|
|
report("bitcast sizes must match", MI);
|
|
|
|
break;
|
|
|
|
}
|
2019-01-30 07:29:00 +08:00
|
|
|
case TargetOpcode::G_INTTOPTR:
|
|
|
|
case TargetOpcode::G_PTRTOINT:
|
|
|
|
case TargetOpcode::G_ADDRSPACE_CAST: {
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT SrcTy = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
if (!DstTy.isValid() || !SrcTy.isValid())
|
|
|
|
break;
|
|
|
|
|
2019-02-05 07:29:16 +08:00
|
|
|
verifyVectorElementMatch(DstTy, SrcTy, MI);
|
2019-01-30 07:29:00 +08:00
|
|
|
|
|
|
|
DstTy = DstTy.getScalarType();
|
|
|
|
SrcTy = SrcTy.getScalarType();
|
|
|
|
|
|
|
|
if (MI->getOpcode() == TargetOpcode::G_INTTOPTR) {
|
|
|
|
if (!DstTy.isPointer())
|
|
|
|
report("inttoptr result type must be a pointer", MI);
|
|
|
|
if (SrcTy.isPointer())
|
|
|
|
report("inttoptr source type must not be a pointer", MI);
|
|
|
|
} else if (MI->getOpcode() == TargetOpcode::G_PTRTOINT) {
|
|
|
|
if (!SrcTy.isPointer())
|
|
|
|
report("ptrtoint source type must be a pointer", MI);
|
|
|
|
if (DstTy.isPointer())
|
|
|
|
report("ptrtoint result type must not be a pointer", MI);
|
|
|
|
} else {
|
|
|
|
assert(MI->getOpcode() == TargetOpcode::G_ADDRSPACE_CAST);
|
|
|
|
if (!SrcTy.isPointer() || !DstTy.isPointer())
|
|
|
|
report("addrspacecast types must be pointers", MI);
|
|
|
|
else {
|
|
|
|
if (SrcTy.getAddressSpace() == DstTy.getAddressSpace())
|
|
|
|
report("addrspacecast must convert different address spaces", MI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
[globalisel] Rename G_GEP to G_PTR_ADD
Summary:
G_GEP is rather poorly named. It's a simple pointer+scalar addition and
doesn't support any of the complexities of getelementptr. I therefore
propose that we rename it. There's a G_PTR_MASK so let's follow that
convention and go with G_PTR_ADD
Reviewers: volkan, aditya_nandakumar, bogner, rovka, arsenm
Subscribers: sdardis, jvesely, wdng, nhaehnle, hiraditya, jrtc27, atanasyan, arphaman, Petar.Avramovic, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D69734
2019-11-02 04:18:00 +08:00
|
|
|
case TargetOpcode::G_PTR_ADD: {
|
2019-02-06 04:04:12 +08:00
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT PtrTy = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
LLT OffsetTy = MRI->getType(MI->getOperand(2).getReg());
|
|
|
|
if (!DstTy.isValid() || !PtrTy.isValid() || !OffsetTy.isValid())
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (!PtrTy.getScalarType().isPointer())
|
|
|
|
report("gep first operand must be a pointer", MI);
|
|
|
|
|
|
|
|
if (OffsetTy.getScalarType().isPointer())
|
|
|
|
report("gep offset operand must not be a pointer", MI);
|
|
|
|
|
|
|
|
// TODO: Is the offset allowed to be a scalar with a vector?
|
|
|
|
break;
|
|
|
|
}
|
2018-05-08 10:48:15 +08:00
|
|
|
case TargetOpcode::G_SEXT:
|
|
|
|
case TargetOpcode::G_ZEXT:
|
|
|
|
case TargetOpcode::G_ANYEXT:
|
|
|
|
case TargetOpcode::G_TRUNC:
|
|
|
|
case TargetOpcode::G_FPEXT:
|
|
|
|
case TargetOpcode::G_FPTRUNC: {
|
|
|
|
// Number of operands and presense of types is already checked (and
|
|
|
|
// reported in case of any issues), so no need to report them again. As
|
|
|
|
// we're trying to report as many issues as possible at once, however, the
|
|
|
|
// instructions aren't guaranteed to have the right number of operands or
|
|
|
|
// types attached to them at this point
|
|
|
|
assert(MCID.getNumOperands() == 2 && "Expected 2 operands G_*{EXT,TRUNC}");
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT SrcTy = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
if (!DstTy.isValid() || !SrcTy.isValid())
|
|
|
|
break;
|
|
|
|
|
2019-02-05 07:29:16 +08:00
|
|
|
LLT DstElTy = DstTy.getScalarType();
|
|
|
|
LLT SrcElTy = SrcTy.getScalarType();
|
2018-05-08 10:48:15 +08:00
|
|
|
if (DstElTy.isPointer() || SrcElTy.isPointer())
|
|
|
|
report("Generic extend/truncate can not operate on pointers", MI);
|
|
|
|
|
2019-02-05 07:29:16 +08:00
|
|
|
verifyVectorElementMatch(DstTy, SrcTy, MI);
|
|
|
|
|
2018-05-08 10:48:15 +08:00
|
|
|
unsigned DstSize = DstElTy.getSizeInBits();
|
|
|
|
unsigned SrcSize = SrcElTy.getSizeInBits();
|
|
|
|
switch (MI->getOpcode()) {
|
|
|
|
default:
|
|
|
|
if (DstSize <= SrcSize)
|
|
|
|
report("Generic extend has destination type no larger than source", MI);
|
|
|
|
break;
|
|
|
|
case TargetOpcode::G_TRUNC:
|
|
|
|
case TargetOpcode::G_FPTRUNC:
|
|
|
|
if (DstSize >= SrcSize)
|
|
|
|
report("Generic truncate has destination type no smaller than source",
|
|
|
|
MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2019-02-05 07:29:16 +08:00
|
|
|
case TargetOpcode::G_SELECT: {
|
|
|
|
LLT SelTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT CondTy = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
if (!SelTy.isValid() || !CondTy.isValid())
|
|
|
|
break;
|
|
|
|
|
|
|
|
// Scalar condition select on a vector is valid.
|
|
|
|
if (CondTy.isVector())
|
|
|
|
verifyVectorElementMatch(SelTy, CondTy, MI);
|
|
|
|
break;
|
|
|
|
}
|
2018-12-11 02:44:58 +08:00
|
|
|
case TargetOpcode::G_MERGE_VALUES: {
|
|
|
|
// G_MERGE_VALUES should only be used to merge scalars into a larger scalar,
|
|
|
|
// e.g. s2N = MERGE sN, sN
|
|
|
|
// Merging multiple scalars into a vector is not allowed, should use
|
|
|
|
// G_BUILD_VECTOR for that.
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT SrcTy = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
if (DstTy.isVector() || SrcTy.isVector())
|
|
|
|
report("G_MERGE_VALUES cannot operate on vectors", MI);
|
2019-07-02 02:01:35 +08:00
|
|
|
|
|
|
|
const unsigned NumOps = MI->getNumOperands();
|
|
|
|
if (DstTy.getSizeInBits() != SrcTy.getSizeInBits() * (NumOps - 1))
|
|
|
|
report("G_MERGE_VALUES result size is inconsistent", MI);
|
|
|
|
|
|
|
|
for (unsigned I = 2; I != NumOps; ++I) {
|
|
|
|
if (MRI->getType(MI->getOperand(I).getReg()) != SrcTy)
|
|
|
|
report("G_MERGE_VALUES source types do not match", MI);
|
|
|
|
}
|
|
|
|
|
2018-12-11 02:44:58 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TargetOpcode::G_UNMERGE_VALUES: {
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT SrcTy = MRI->getType(MI->getOperand(MI->getNumOperands()-1).getReg());
|
|
|
|
// For now G_UNMERGE can split vectors.
|
|
|
|
for (unsigned i = 0; i < MI->getNumOperands()-1; ++i) {
|
|
|
|
if (MRI->getType(MI->getOperand(i).getReg()) != DstTy)
|
|
|
|
report("G_UNMERGE_VALUES destination types do not match", MI);
|
|
|
|
}
|
|
|
|
if (SrcTy.getSizeInBits() !=
|
|
|
|
(DstTy.getSizeInBits() * (MI->getNumOperands() - 1))) {
|
|
|
|
report("G_UNMERGE_VALUES source operand does not cover dest operands",
|
|
|
|
MI);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2018-12-06 07:53:30 +08:00
|
|
|
case TargetOpcode::G_BUILD_VECTOR: {
|
|
|
|
// Source types must be scalars, dest type a vector. Total size of scalars
|
|
|
|
// must match the dest vector size.
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT SrcEltTy = MRI->getType(MI->getOperand(1).getReg());
|
2019-02-15 23:24:34 +08:00
|
|
|
if (!DstTy.isVector() || SrcEltTy.isVector()) {
|
2018-12-06 07:53:30 +08:00
|
|
|
report("G_BUILD_VECTOR must produce a vector from scalar operands", MI);
|
2019-02-15 23:24:34 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (DstTy.getElementType() != SrcEltTy)
|
|
|
|
report("G_BUILD_VECTOR result element type must match source type", MI);
|
|
|
|
|
|
|
|
if (DstTy.getNumElements() != MI->getNumOperands() - 1)
|
|
|
|
report("G_BUILD_VECTOR must have an operand for each elemement", MI);
|
|
|
|
|
2018-12-06 07:53:30 +08:00
|
|
|
for (unsigned i = 2; i < MI->getNumOperands(); ++i) {
|
|
|
|
if (MRI->getType(MI->getOperand(1).getReg()) !=
|
|
|
|
MRI->getType(MI->getOperand(i).getReg()))
|
|
|
|
report("G_BUILD_VECTOR source operand types are not homogeneous", MI);
|
|
|
|
}
|
2019-02-15 23:24:34 +08:00
|
|
|
|
2018-12-06 07:53:30 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TargetOpcode::G_BUILD_VECTOR_TRUNC: {
|
|
|
|
// Source types must be scalars, dest type a vector. Scalar types must be
|
|
|
|
// larger than the dest vector elt type, as this is a truncating operation.
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT SrcEltTy = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
if (!DstTy.isVector() || SrcEltTy.isVector())
|
|
|
|
report("G_BUILD_VECTOR_TRUNC must produce a vector from scalar operands",
|
|
|
|
MI);
|
|
|
|
for (unsigned i = 2; i < MI->getNumOperands(); ++i) {
|
|
|
|
if (MRI->getType(MI->getOperand(1).getReg()) !=
|
|
|
|
MRI->getType(MI->getOperand(i).getReg()))
|
|
|
|
report("G_BUILD_VECTOR_TRUNC source operand types are not homogeneous",
|
|
|
|
MI);
|
|
|
|
}
|
|
|
|
if (SrcEltTy.getSizeInBits() <= DstTy.getElementType().getSizeInBits())
|
|
|
|
report("G_BUILD_VECTOR_TRUNC source operand types are not larger than "
|
|
|
|
"dest elt type",
|
|
|
|
MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TargetOpcode::G_CONCAT_VECTORS: {
|
|
|
|
// Source types should be vectors, and total size should match the dest
|
|
|
|
// vector size.
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT SrcTy = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
if (!DstTy.isVector() || !SrcTy.isVector())
|
|
|
|
report("G_CONCAT_VECTOR requires vector source and destination operands",
|
|
|
|
MI);
|
|
|
|
for (unsigned i = 2; i < MI->getNumOperands(); ++i) {
|
|
|
|
if (MRI->getType(MI->getOperand(1).getReg()) !=
|
|
|
|
MRI->getType(MI->getOperand(i).getReg()))
|
|
|
|
report("G_CONCAT_VECTOR source operand types are not homogeneous", MI);
|
|
|
|
}
|
|
|
|
if (DstTy.getNumElements() !=
|
|
|
|
SrcTy.getNumElements() * (MI->getNumOperands() - 1))
|
|
|
|
report("G_CONCAT_VECTOR num dest and source elements should match", MI);
|
|
|
|
break;
|
|
|
|
}
|
2019-02-05 07:29:11 +08:00
|
|
|
case TargetOpcode::G_ICMP:
|
|
|
|
case TargetOpcode::G_FCMP: {
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT SrcTy = MRI->getType(MI->getOperand(2).getReg());
|
|
|
|
|
|
|
|
if ((DstTy.isVector() != SrcTy.isVector()) ||
|
|
|
|
(DstTy.isVector() && DstTy.getNumElements() != SrcTy.getNumElements()))
|
|
|
|
report("Generic vector icmp/fcmp must preserve number of lanes", MI);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2019-02-12 06:12:43 +08:00
|
|
|
case TargetOpcode::G_EXTRACT: {
|
|
|
|
const MachineOperand &SrcOp = MI->getOperand(1);
|
|
|
|
if (!SrcOp.isReg()) {
|
|
|
|
report("extract source must be a register", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const MachineOperand &OffsetOp = MI->getOperand(2);
|
|
|
|
if (!OffsetOp.isImm()) {
|
|
|
|
report("extract offset must be a constant", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned DstSize = MRI->getType(MI->getOperand(0).getReg()).getSizeInBits();
|
|
|
|
unsigned SrcSize = MRI->getType(SrcOp.getReg()).getSizeInBits();
|
|
|
|
if (SrcSize == DstSize)
|
|
|
|
report("extract source must be larger than result", MI);
|
|
|
|
|
|
|
|
if (DstSize + OffsetOp.getImm() > SrcSize)
|
|
|
|
report("extract reads past end of register", MI);
|
|
|
|
break;
|
|
|
|
}
|
2019-02-20 00:10:16 +08:00
|
|
|
case TargetOpcode::G_INSERT: {
|
|
|
|
const MachineOperand &SrcOp = MI->getOperand(2);
|
|
|
|
if (!SrcOp.isReg()) {
|
|
|
|
report("insert source must be a register", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
const MachineOperand &OffsetOp = MI->getOperand(3);
|
|
|
|
if (!OffsetOp.isImm()) {
|
|
|
|
report("insert offset must be a constant", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned DstSize = MRI->getType(MI->getOperand(0).getReg()).getSizeInBits();
|
|
|
|
unsigned SrcSize = MRI->getType(SrcOp.getReg()).getSizeInBits();
|
|
|
|
|
|
|
|
if (DstSize <= SrcSize)
|
|
|
|
report("inserted size must be smaller than total register", MI);
|
|
|
|
|
|
|
|
if (SrcSize + OffsetOp.getImm() > DstSize)
|
|
|
|
report("insert writes past end of register", MI);
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
2019-06-12 03:58:06 +08:00
|
|
|
case TargetOpcode::G_JUMP_TABLE: {
|
|
|
|
if (!MI->getOperand(1).isJTI())
|
|
|
|
report("G_JUMP_TABLE source operand must be a jump table index", MI);
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
if (!DstTy.isPointer())
|
|
|
|
report("G_JUMP_TABLE dest operand must have a pointer type", MI);
|
|
|
|
break;
|
|
|
|
}
|
2019-06-15 01:55:48 +08:00
|
|
|
case TargetOpcode::G_BRJT: {
|
|
|
|
if (!MRI->getType(MI->getOperand(0).getReg()).isPointer())
|
|
|
|
report("G_BRJT src operand 0 must be a pointer type", MI);
|
|
|
|
|
|
|
|
if (!MI->getOperand(1).isJTI())
|
|
|
|
report("G_BRJT src operand 1 must be a jump table index", MI);
|
|
|
|
|
|
|
|
const auto &IdxOp = MI->getOperand(2);
|
|
|
|
if (!IdxOp.isReg() || MRI->getType(IdxOp.getReg()).isPointer())
|
|
|
|
report("G_BRJT src operand 2 must be a scalar reg type", MI);
|
|
|
|
break;
|
|
|
|
}
|
2019-06-18 01:01:32 +08:00
|
|
|
case TargetOpcode::G_INTRINSIC:
|
|
|
|
case TargetOpcode::G_INTRINSIC_W_SIDE_EFFECTS: {
|
|
|
|
// TODO: Should verify number of def and use operands, but the current
|
|
|
|
// interface requires passing in IR types for mangling.
|
|
|
|
const MachineOperand &IntrIDOp = MI->getOperand(MI->getNumExplicitDefs());
|
|
|
|
if (!IntrIDOp.isIntrinsicID()) {
|
|
|
|
report("G_INTRINSIC first src operand must be an intrinsic ID", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool NoSideEffects = MI->getOpcode() == TargetOpcode::G_INTRINSIC;
|
|
|
|
unsigned IntrID = IntrIDOp.getIntrinsicID();
|
|
|
|
if (IntrID != 0 && IntrID < Intrinsic::num_intrinsics) {
|
|
|
|
AttributeList Attrs
|
|
|
|
= Intrinsic::getAttributes(MF->getFunction().getContext(),
|
|
|
|
static_cast<Intrinsic::ID>(IntrID));
|
|
|
|
bool DeclHasSideEffects = !Attrs.hasFnAttribute(Attribute::ReadNone);
|
|
|
|
if (NoSideEffects && DeclHasSideEffects) {
|
|
|
|
report("G_INTRINSIC used with intrinsic that accesses memory", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!NoSideEffects && !DeclHasSideEffects) {
|
|
|
|
report("G_INTRINSIC_W_SIDE_EFFECTS used with readnone intrinsic", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2019-09-28 13:33:21 +08:00
|
|
|
switch (IntrID) {
|
|
|
|
case Intrinsic::memcpy:
|
|
|
|
if (MI->getNumOperands() != 5)
|
|
|
|
report("Expected memcpy intrinsic to have 5 operands", MI);
|
|
|
|
break;
|
|
|
|
case Intrinsic::memmove:
|
|
|
|
if (MI->getNumOperands() != 5)
|
|
|
|
report("Expected memmove intrinsic to have 5 operands", MI);
|
|
|
|
break;
|
|
|
|
case Intrinsic::memset:
|
|
|
|
if (MI->getNumOperands() != 5)
|
|
|
|
report("Expected memset intrinsic to have 5 operands", MI);
|
|
|
|
break;
|
|
|
|
}
|
[globalisel] Add G_SEXT_INREG
Summary:
Targets often have instructions that can sign-extend certain cases faster
than the equivalent shift-left/arithmetic-shift-right. Such cases can be
identified by matching a shift-left/shift-right pair but there are some
issues with this in the context of combines. For example, suppose you can
sign-extend 8-bit up to 32-bit with a target extend instruction.
%1:_(s32) = G_SHL %0:_(s32), i32 24 # (I've inlined the G_CONSTANT for brevity)
%2:_(s32) = G_ASHR %1:_(s32), i32 24
%3:_(s32) = G_ASHR %2:_(s32), i32 1
would reasonably combine to:
%1:_(s32) = G_SHL %0:_(s32), i32 24
%2:_(s32) = G_ASHR %1:_(s32), i32 25
which no longer matches the special case. If your shifts and extend are
equal cost, this would break even as a pair of shifts but if your shift is
more expensive than the extend then it's cheaper as:
%2:_(s32) = G_SEXT_INREG %0:_(s32), i32 8
%3:_(s32) = G_ASHR %2:_(s32), i32 1
It's possible to match the shift-pair in ISel and emit an extend and ashr.
However, this is far from the only way to break this shift pair and make
it hard to match the extends. Another example is that with the right
known-zeros, this:
%1:_(s32) = G_SHL %0:_(s32), i32 24
%2:_(s32) = G_ASHR %1:_(s32), i32 24
%3:_(s32) = G_MUL %2:_(s32), i32 2
can become:
%1:_(s32) = G_SHL %0:_(s32), i32 24
%2:_(s32) = G_ASHR %1:_(s32), i32 23
All upstream targets have been configured to lower it to the current
G_SHL,G_ASHR pair but will likely want to make it legal in some cases to
handle their faster cases.
To follow-up: Provide a way to legalize based on the constant. At the
moment, I'm thinking that the best way to achieve this is to provide the
MI in LegalityQuery but that opens the door to breaking core principles
of the legalizer (legality is not context sensitive). That said, it's
worth noting that looking at other instructions and acting on that
information doesn't violate this principle in itself. It's only a
violation if, at the end of legalization, a pass that checks legality
without being able to see the context would say an instruction might not be
legal. That's a fairly subtle distinction so to give a concrete example,
saying %2 in:
%1 = G_CONSTANT 16
%2 = G_SEXT_INREG %0, %1
is legal is in violation of that principle if the legality of %2 depends
on %1 being constant and/or being 16. However, legalizing to either:
%2 = G_SEXT_INREG %0, 16
or:
%1 = G_CONSTANT 16
%2:_(s32) = G_SHL %0, %1
%3:_(s32) = G_ASHR %2, %1
depending on whether %1 is constant and 16 does not violate that principle
since both outputs are genuinely legal.
Reviewers: bogner, aditya_nandakumar, volkan, aemerson, paquette, arsenm
Subscribers: sdardis, jvesely, wdng, nhaehnle, rovka, kristof.beyls, javed.absar, hiraditya, jrtc27, atanasyan, Petar.Avramovic, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D61289
llvm-svn: 368487
2019-08-10 05:11:20 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TargetOpcode::G_SEXT_INREG: {
|
|
|
|
if (!MI->getOperand(2).isImm()) {
|
|
|
|
report("G_SEXT_INREG expects an immediate operand #2", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT SrcTy = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
verifyVectorElementMatch(DstTy, SrcTy, MI);
|
2019-06-18 01:01:32 +08:00
|
|
|
|
[globalisel] Add G_SEXT_INREG
Summary:
Targets often have instructions that can sign-extend certain cases faster
than the equivalent shift-left/arithmetic-shift-right. Such cases can be
identified by matching a shift-left/shift-right pair but there are some
issues with this in the context of combines. For example, suppose you can
sign-extend 8-bit up to 32-bit with a target extend instruction.
%1:_(s32) = G_SHL %0:_(s32), i32 24 # (I've inlined the G_CONSTANT for brevity)
%2:_(s32) = G_ASHR %1:_(s32), i32 24
%3:_(s32) = G_ASHR %2:_(s32), i32 1
would reasonably combine to:
%1:_(s32) = G_SHL %0:_(s32), i32 24
%2:_(s32) = G_ASHR %1:_(s32), i32 25
which no longer matches the special case. If your shifts and extend are
equal cost, this would break even as a pair of shifts but if your shift is
more expensive than the extend then it's cheaper as:
%2:_(s32) = G_SEXT_INREG %0:_(s32), i32 8
%3:_(s32) = G_ASHR %2:_(s32), i32 1
It's possible to match the shift-pair in ISel and emit an extend and ashr.
However, this is far from the only way to break this shift pair and make
it hard to match the extends. Another example is that with the right
known-zeros, this:
%1:_(s32) = G_SHL %0:_(s32), i32 24
%2:_(s32) = G_ASHR %1:_(s32), i32 24
%3:_(s32) = G_MUL %2:_(s32), i32 2
can become:
%1:_(s32) = G_SHL %0:_(s32), i32 24
%2:_(s32) = G_ASHR %1:_(s32), i32 23
All upstream targets have been configured to lower it to the current
G_SHL,G_ASHR pair but will likely want to make it legal in some cases to
handle their faster cases.
To follow-up: Provide a way to legalize based on the constant. At the
moment, I'm thinking that the best way to achieve this is to provide the
MI in LegalityQuery but that opens the door to breaking core principles
of the legalizer (legality is not context sensitive). That said, it's
worth noting that looking at other instructions and acting on that
information doesn't violate this principle in itself. It's only a
violation if, at the end of legalization, a pass that checks legality
without being able to see the context would say an instruction might not be
legal. That's a fairly subtle distinction so to give a concrete example,
saying %2 in:
%1 = G_CONSTANT 16
%2 = G_SEXT_INREG %0, %1
is legal is in violation of that principle if the legality of %2 depends
on %1 being constant and/or being 16. However, legalizing to either:
%2 = G_SEXT_INREG %0, 16
or:
%1 = G_CONSTANT 16
%2:_(s32) = G_SHL %0, %1
%3:_(s32) = G_ASHR %2, %1
depending on whether %1 is constant and 16 does not violate that principle
since both outputs are genuinely legal.
Reviewers: bogner, aditya_nandakumar, volkan, aemerson, paquette, arsenm
Subscribers: sdardis, jvesely, wdng, nhaehnle, rovka, kristof.beyls, javed.absar, hiraditya, jrtc27, atanasyan, Petar.Avramovic, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D61289
llvm-svn: 368487
2019-08-10 05:11:20 +08:00
|
|
|
int64_t Imm = MI->getOperand(2).getImm();
|
|
|
|
if (Imm <= 0)
|
|
|
|
report("G_SEXT_INREG size must be >= 1", MI);
|
|
|
|
if (Imm >= SrcTy.getScalarSizeInBits())
|
|
|
|
report("G_SEXT_INREG size must be less than source bit width", MI);
|
2019-06-18 01:01:32 +08:00
|
|
|
break;
|
|
|
|
}
|
2019-08-13 23:34:38 +08:00
|
|
|
case TargetOpcode::G_SHUFFLE_VECTOR: {
|
|
|
|
const MachineOperand &MaskOp = MI->getOperand(3);
|
|
|
|
if (!MaskOp.isShuffleMask()) {
|
|
|
|
report("Incorrect mask operand type for G_SHUFFLE_VECTOR", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-08-13 23:52:21 +08:00
|
|
|
LLT DstTy = MRI->getType(MI->getOperand(0).getReg());
|
|
|
|
LLT Src0Ty = MRI->getType(MI->getOperand(1).getReg());
|
|
|
|
LLT Src1Ty = MRI->getType(MI->getOperand(2).getReg());
|
|
|
|
|
|
|
|
if (Src0Ty != Src1Ty)
|
|
|
|
report("Source operands must be the same type", MI);
|
|
|
|
|
|
|
|
if (Src0Ty.getScalarType() != DstTy.getScalarType())
|
|
|
|
report("G_SHUFFLE_VECTOR cannot change element type", MI);
|
|
|
|
|
|
|
|
// Don't check that all operands are vector because scalars are used in
|
|
|
|
// place of 1 element vectors.
|
|
|
|
int SrcNumElts = Src0Ty.isVector() ? Src0Ty.getNumElements() : 1;
|
|
|
|
int DstNumElts = DstTy.isVector() ? DstTy.getNumElements() : 1;
|
|
|
|
|
2020-01-14 07:32:45 +08:00
|
|
|
ArrayRef<int> MaskIdxes = MaskOp.getShuffleMask();
|
2019-08-13 23:52:21 +08:00
|
|
|
|
|
|
|
if (static_cast<int>(MaskIdxes.size()) != DstNumElts)
|
|
|
|
report("Wrong result type for shufflemask", MI);
|
|
|
|
|
|
|
|
for (int Idx : MaskIdxes) {
|
|
|
|
if (Idx < 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (Idx >= 2 * SrcNumElts)
|
|
|
|
report("Out of bounds shuffle index", MI);
|
|
|
|
}
|
|
|
|
|
2019-08-13 23:34:38 +08:00
|
|
|
break;
|
|
|
|
}
|
2019-08-24 10:25:56 +08:00
|
|
|
case TargetOpcode::G_DYN_STACKALLOC: {
|
|
|
|
const MachineOperand &DstOp = MI->getOperand(0);
|
|
|
|
const MachineOperand &AllocOp = MI->getOperand(1);
|
|
|
|
const MachineOperand &AlignOp = MI->getOperand(2);
|
|
|
|
|
|
|
|
if (!DstOp.isReg() || !MRI->getType(DstOp.getReg()).isPointer()) {
|
|
|
|
report("dst operand 0 must be a pointer type", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!AllocOp.isReg() || !MRI->getType(AllocOp.getReg()).isScalar()) {
|
|
|
|
report("src operand 1 must be a scalar reg type", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!AlignOp.isImm()) {
|
|
|
|
report("src operand 2 must be an immediate type", MI);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2019-02-05 07:29:11 +08:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MachineVerifier::visitMachineInstrBefore(const MachineInstr *MI) {
|
|
|
|
const MCInstrDesc &MCID = MI->getDesc();
|
|
|
|
if (MI->getNumOperands() < MCID.getNumOperands()) {
|
|
|
|
report("Too few operands", MI);
|
|
|
|
errs() << MCID.getNumOperands() << " operands expected, but "
|
|
|
|
<< MI->getNumOperands() << " given.\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MI->isPHI()) {
|
|
|
|
if (MF->getProperties().hasProperty(
|
|
|
|
MachineFunctionProperties::Property::NoPHIs))
|
|
|
|
report("Found PHI instruction with NoPHIs property set", MI);
|
|
|
|
|
|
|
|
if (FirstNonPHI)
|
|
|
|
report("Found PHI instruction after non-PHI", MI);
|
|
|
|
} else if (FirstNonPHI == nullptr)
|
|
|
|
FirstNonPHI = MI;
|
|
|
|
|
|
|
|
// Check the tied operands.
|
|
|
|
if (MI->isInlineAsm())
|
|
|
|
verifyInlineAsm(MI);
|
|
|
|
|
|
|
|
// Check the MachineMemOperands for basic consistency.
|
|
|
|
for (MachineInstr::mmo_iterator I = MI->memoperands_begin(),
|
|
|
|
E = MI->memoperands_end();
|
|
|
|
I != E; ++I) {
|
|
|
|
if ((*I)->isLoad() && !MI->mayLoad())
|
|
|
|
report("Missing mayLoad flag", MI);
|
|
|
|
if ((*I)->isStore() && !MI->mayStore())
|
|
|
|
report("Missing mayStore flag", MI);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Debug values must not have a slot index.
|
|
|
|
// Other instructions must have one, unless they are inside a bundle.
|
|
|
|
if (LiveInts) {
|
|
|
|
bool mapped = !LiveInts->isNotInMIMap(*MI);
|
|
|
|
if (MI->isDebugInstr()) {
|
|
|
|
if (mapped)
|
|
|
|
report("Debug instruction has a slot index", MI);
|
|
|
|
} else if (MI->isInsideBundle()) {
|
|
|
|
if (mapped)
|
|
|
|
report("Instruction inside bundle has a slot index", MI);
|
|
|
|
} else {
|
|
|
|
if (!mapped)
|
|
|
|
report("Missing slot index", MI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isPreISelGenericOpcode(MCID.getOpcode())) {
|
|
|
|
verifyPreISelGenericInstruction(MI);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
StringRef ErrorInfo;
|
|
|
|
if (!TII->verifyInstruction(*MI, ErrorInfo))
|
|
|
|
report(ErrorInfo.data(), MI);
|
|
|
|
|
|
|
|
// Verify properties of various specific instruction types
|
|
|
|
switch (MI->getOpcode()) {
|
2018-02-09 09:27:23 +08:00
|
|
|
case TargetOpcode::COPY: {
|
|
|
|
if (foundErrors)
|
|
|
|
break;
|
|
|
|
const MachineOperand &DstOp = MI->getOperand(0);
|
|
|
|
const MachineOperand &SrcOp = MI->getOperand(1);
|
|
|
|
LLT DstTy = MRI->getType(DstOp.getReg());
|
|
|
|
LLT SrcTy = MRI->getType(SrcOp.getReg());
|
|
|
|
if (SrcTy.isValid() && DstTy.isValid()) {
|
|
|
|
// If both types are valid, check that the types are the same.
|
|
|
|
if (SrcTy != DstTy) {
|
|
|
|
report("Copy Instruction is illegal with mismatching types", MI);
|
|
|
|
errs() << "Def = " << DstTy << ", Src = " << SrcTy << "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (SrcTy.isValid() || DstTy.isValid()) {
|
|
|
|
// If one of them have valid types, let's just check they have the same
|
|
|
|
// size.
|
|
|
|
unsigned SrcSize = TRI->getRegSizeInBits(SrcOp.getReg(), *MRI);
|
|
|
|
unsigned DstSize = TRI->getRegSizeInBits(DstOp.getReg(), *MRI);
|
|
|
|
assert(SrcSize && "Expecting size here");
|
|
|
|
assert(DstSize && "Expecting size here");
|
|
|
|
if (SrcSize != DstSize)
|
|
|
|
if (!DstOp.getSubReg() && !SrcOp.getSubReg()) {
|
|
|
|
report("Copy Instruction is illegal with mismatching sizes", MI);
|
|
|
|
errs() << "Def Size = " << DstSize << ", Src Size = " << SrcSize
|
|
|
|
<< "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2020-04-14 23:30:51 +08:00
|
|
|
case TargetOpcode::STATEPOINT: {
|
|
|
|
StatepointOpers SO(MI);
|
|
|
|
if (!MI->getOperand(SO.getIDPos()).isImm() ||
|
|
|
|
!MI->getOperand(SO.getNBytesPos()).isImm() ||
|
|
|
|
!MI->getOperand(SO.getNCallArgsPos()).isImm()) {
|
2017-06-03 00:36:37 +08:00
|
|
|
report("meta operands to STATEPOINT not constant!", MI);
|
2020-04-14 23:30:51 +08:00
|
|
|
break;
|
|
|
|
}
|
2017-06-03 01:02:33 +08:00
|
|
|
|
|
|
|
auto VerifyStackMapConstant = [&](unsigned Offset) {
|
2020-04-14 23:30:51 +08:00
|
|
|
if (!MI->getOperand(Offset - 1).isImm() ||
|
|
|
|
MI->getOperand(Offset - 1).getImm() != StackMaps::ConstantOp ||
|
|
|
|
!MI->getOperand(Offset).isImm())
|
2017-06-03 01:02:33 +08:00
|
|
|
report("stack map constant to STATEPOINT not well formed!", MI);
|
|
|
|
};
|
2020-04-14 23:30:51 +08:00
|
|
|
VerifyStackMapConstant(SO.getCCIdx());
|
|
|
|
VerifyStackMapConstant(SO.getFlagsIdx());
|
|
|
|
VerifyStackMapConstant(SO.getNumDeoptArgsIdx());
|
2017-06-03 01:02:33 +08:00
|
|
|
|
|
|
|
// TODO: verify we have properly encoded deopt arguments
|
2020-04-14 23:30:51 +08:00
|
|
|
} break;
|
2019-02-05 07:29:11 +08:00
|
|
|
}
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2009-10-05 02:18:39 +08:00
|
|
|
MachineVerifier::visitMachineOperand(const MachineOperand *MO, unsigned MONum) {
|
2009-05-16 08:33:53 +08:00
|
|
|
const MachineInstr *MI = MO->getParent();
|
2011-06-29 03:10:37 +08:00
|
|
|
const MCInstrDesc &MCID = MI->getDesc();
|
2015-08-11 05:47:36 +08:00
|
|
|
unsigned NumDefs = MCID.getNumDefs();
|
|
|
|
if (MCID.getOpcode() == TargetOpcode::PATCHPOINT)
|
|
|
|
NumDefs = (MONum == 0 && MO->isReg()) ? NumDefs : 0;
|
2009-05-16 15:25:20 +08:00
|
|
|
|
2011-06-29 03:10:37 +08:00
|
|
|
// The first MCID.NumDefs operands must be explicit register defines
|
2015-08-11 05:47:36 +08:00
|
|
|
if (MONum < NumDefs) {
|
2012-08-15 09:39:31 +08:00
|
|
|
const MCOperandInfo &MCOI = MCID.OpInfo[MONum];
|
2009-05-16 15:25:20 +08:00
|
|
|
if (!MO->isReg())
|
|
|
|
report("Explicit definition must be a register", MO, MONum);
|
2012-05-30 03:40:44 +08:00
|
|
|
else if (!MO->isDef() && !MCOI.isOptionalDef())
|
2009-05-16 15:25:20 +08:00
|
|
|
report("Explicit definition marked as use", MO, MONum);
|
|
|
|
else if (MO->isImplicit())
|
|
|
|
report("Explicit definition marked as implicit", MO, MONum);
|
2011-06-29 03:10:37 +08:00
|
|
|
} else if (MONum < MCID.getNumOperands()) {
|
2012-08-15 09:39:31 +08:00
|
|
|
const MCOperandInfo &MCOI = MCID.OpInfo[MONum];
|
2010-11-17 08:55:36 +08:00
|
|
|
// Don't check if it's the last operand in a variadic instruction. See,
|
2019-11-01 17:14:22 +08:00
|
|
|
// e.g., LDM_RET in the arm back end. Check non-variadic operands only.
|
|
|
|
bool IsOptional = MI->isVariadic() && MONum == MCID.getNumOperands() - 1;
|
|
|
|
if (!IsOptional) {
|
|
|
|
if (MO->isReg()) {
|
2020-02-06 14:35:01 +08:00
|
|
|
if (MO->isDef() && !MCOI.isOptionalDef() && !MCID.variadicOpsAreDefs())
|
2019-11-01 17:14:22 +08:00
|
|
|
report("Explicit operand marked as def", MO, MONum);
|
|
|
|
if (MO->isImplicit())
|
|
|
|
report("Explicit operand marked as implicit", MO, MONum);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that an instruction has register operands only as expected.
|
|
|
|
if (MCOI.OperandType == MCOI::OPERAND_REGISTER &&
|
|
|
|
!MO->isReg() && !MO->isFI())
|
|
|
|
report("Expected a register operand.", MO, MONum);
|
|
|
|
if ((MCOI.OperandType == MCOI::OPERAND_IMMEDIATE ||
|
|
|
|
MCOI.OperandType == MCOI::OPERAND_PCREL) && MO->isReg())
|
|
|
|
report("Expected a non-register operand.", MO, MONum);
|
2009-09-24 04:57:55 +08:00
|
|
|
}
|
2012-08-29 08:38:03 +08:00
|
|
|
|
2012-09-05 02:38:28 +08:00
|
|
|
int TiedTo = MCID.getOperandConstraint(MONum, MCOI::TIED_TO);
|
|
|
|
if (TiedTo != -1) {
|
2012-08-29 08:38:03 +08:00
|
|
|
if (!MO->isReg())
|
|
|
|
report("Tied use must be a register", MO, MONum);
|
|
|
|
else if (!MO->isTied())
|
|
|
|
report("Operand should be tied", MO, MONum);
|
2012-09-05 02:38:28 +08:00
|
|
|
else if (unsigned(TiedTo) != MI->findTiedOperandIdx(MONum))
|
|
|
|
report("Tied def doesn't match MCInstrDesc", MO, MONum);
|
2019-08-02 07:27:28 +08:00
|
|
|
else if (Register::isPhysicalRegister(MO->getReg())) {
|
2017-07-06 21:18:21 +08:00
|
|
|
const MachineOperand &MOTied = MI->getOperand(TiedTo);
|
|
|
|
if (!MOTied.isReg())
|
|
|
|
report("Tied counterpart must be a register", &MOTied, TiedTo);
|
2019-08-02 07:27:28 +08:00
|
|
|
else if (Register::isPhysicalRegister(MOTied.getReg()) &&
|
2017-07-06 21:18:21 +08:00
|
|
|
MO->getReg() != MOTied.getReg())
|
|
|
|
report("Tied physical registers must match.", &MOTied, TiedTo);
|
|
|
|
}
|
2012-08-29 08:38:03 +08:00
|
|
|
} else if (MO->isReg() && MO->isTied())
|
|
|
|
report("Explicit operand should not be tied", MO, MONum);
|
2009-09-24 04:57:55 +08:00
|
|
|
} else {
|
2009-12-23 05:48:20 +08:00
|
|
|
// ARM adds %reg0 operands to indicate predicates. We'll allow that.
|
2011-12-07 15:15:52 +08:00
|
|
|
if (MO->isReg() && !MO->isImplicit() && !MI->isVariadic() && MO->getReg())
|
2009-09-24 04:57:55 +08:00
|
|
|
report("Extra explicit operand on non-variadic instruction", MO, MONum);
|
2009-05-16 15:25:20 +08:00
|
|
|
}
|
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
switch (MO->getType()) {
|
|
|
|
case MachineOperand::MO_Register: {
|
Apply llvm-prefer-register-over-unsigned from clang-tidy to LLVM
Summary:
This clang-tidy check is looking for unsigned integer variables whose initializer
starts with an implicit cast from llvm::Register and changes the type of the
variable to llvm::Register (dropping the llvm:: where possible).
Partial reverts in:
X86FrameLowering.cpp - Some functions return unsigned and arguably should be MCRegister
X86FixupLEAs.cpp - Some functions return unsigned and arguably should be MCRegister
X86FrameLowering.cpp - Some functions return unsigned and arguably should be MCRegister
HexagonBitSimplify.cpp - Function takes BitTracker::RegisterRef which appears to be unsigned&
MachineVerifier.cpp - Ambiguous operator==() given MCRegister and const Register
PPCFastISel.cpp - No Register::operator-=()
PeepholeOptimizer.cpp - TargetInstrInfo::optimizeLoadInstr() takes an unsigned&
MachineTraceMetrics.cpp - MachineTraceMetrics lacks a suitable constructor
Manual fixups in:
ARMFastISel.cpp - ARMEmitLoad() now takes a Register& instead of unsigned&
HexagonSplitDouble.cpp - Ternary operator was ambiguous between unsigned/Register
HexagonConstExtenders.cpp - Has a local class named Register, used llvm::Register instead of Register.
PPCFastISel.cpp - PPCEmitLoad() now takes a Register& instead of unsigned&
Depends on D65919
Reviewers: arsenm, bogner, craig.topper, RKSimon
Reviewed By: arsenm
Subscribers: RKSimon, craig.topper, lenary, aemerson, wuzish, jholewinski, MatzeB, qcolombet, dschuff, jyknight, dylanmckay, sdardis, nemanjai, jvesely, wdng, nhaehnle, sbc100, jgravelle-google, kristof.beyls, hiraditya, aheejin, kbarton, fedor.sergeev, javed.absar, asb, rbar, johnrusso, simoncook, apazos, sabuasal, niosHD, jrtc27, MaskRay, zzheng, edward-jones, atanasyan, rogfer01, MartinMosbeck, brucehoult, the_o, tpr, PkmX, jocewei, jsji, Petar.Avramovic, asbirlea, Jim, s.egerton, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D65962
llvm-svn: 369041
2019-08-16 03:22:08 +08:00
|
|
|
const Register Reg = MO->getReg();
|
2009-05-16 08:33:53 +08:00
|
|
|
if (!Reg)
|
|
|
|
return;
|
2012-03-29 04:47:35 +08:00
|
|
|
if (MRI->tracksLiveness() && !MI->isDebugValue())
|
|
|
|
checkLiveness(MO, MONum);
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2012-09-05 02:38:28 +08:00
|
|
|
// Verify the consistency of tied operands.
|
|
|
|
if (MO->isTied()) {
|
|
|
|
unsigned OtherIdx = MI->findTiedOperandIdx(MONum);
|
|
|
|
const MachineOperand &OtherMO = MI->getOperand(OtherIdx);
|
|
|
|
if (!OtherMO.isReg())
|
|
|
|
report("Must be tied to a register", MO, MONum);
|
|
|
|
if (!OtherMO.isTied())
|
|
|
|
report("Missing tie flags on tied operand", MO, MONum);
|
|
|
|
if (MI->findTiedOperandIdx(OtherIdx) != MONum)
|
|
|
|
report("Inconsistent tie links", MO, MONum);
|
|
|
|
if (MONum < MCID.getNumDefs()) {
|
|
|
|
if (OtherIdx < MCID.getNumOperands()) {
|
|
|
|
if (-1 == MCID.getOperandConstraint(OtherIdx, MCOI::TIED_TO))
|
|
|
|
report("Explicit def tied to explicit use without tie constraint",
|
|
|
|
MO, MONum);
|
|
|
|
} else {
|
|
|
|
if (!OtherMO.isImplicit())
|
|
|
|
report("Explicit def should be tied to implicit use", MO, MONum);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-26 00:49:11 +08:00
|
|
|
// Verify two-address constraints after leaving SSA form.
|
|
|
|
unsigned DefIdx;
|
|
|
|
if (!MRI->isSSA() && MO->isUse() &&
|
|
|
|
MI->isRegTiedToDefOperand(MONum, &DefIdx) &&
|
|
|
|
Reg != MI->getOperand(DefIdx).getReg())
|
|
|
|
report("Two-address instruction operands must be identical", MO, MONum);
|
2009-05-16 08:33:53 +08:00
|
|
|
|
|
|
|
// Check register classes.
|
2017-11-28 11:54:20 +08:00
|
|
|
unsigned SubIdx = MO->getSubReg();
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2019-08-02 07:27:28 +08:00
|
|
|
if (Register::isPhysicalRegister(Reg)) {
|
2017-11-28 11:54:20 +08:00
|
|
|
if (SubIdx) {
|
|
|
|
report("Illegal subregister index for physical register", MO, MONum);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (MONum < MCID.getNumOperands()) {
|
2012-05-08 06:10:26 +08:00
|
|
|
if (const TargetRegisterClass *DRC =
|
|
|
|
TII->getRegClass(MCID, MONum, TRI, *MF)) {
|
2011-10-06 06:12:57 +08:00
|
|
|
if (!DRC->contains(Reg)) {
|
2009-05-16 08:33:53 +08:00
|
|
|
report("Illegal physical register for instruction", MO, MONum);
|
2017-12-01 00:12:24 +08:00
|
|
|
errs() << printReg(Reg, TRI) << " is not a "
|
|
|
|
<< TRI->getRegClassName(DRC) << " register.\n";
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
}
|
2017-11-28 11:54:20 +08:00
|
|
|
}
|
2018-01-30 02:57:07 +08:00
|
|
|
if (MO->isRenamable()) {
|
[MachineOperand][Target] MachineOperand::isRenamable semantics changes
Summary:
Add a target option AllowRegisterRenaming that is used to opt in to
post-register-allocation renaming of registers. This is set to 0 by
default, which causes the hasExtraSrcRegAllocReq/hasExtraDstRegAllocReq
fields of all opcodes to be set to 1, causing
MachineOperand::isRenamable to always return false.
Set the AllowRegisterRenaming flag to 1 for all in-tree targets that
have lit tests that were effected by enabling COPY forwarding in
MachineCopyPropagation (AArch64, AMDGPU, ARM, Hexagon, Mips, PowerPC,
RISCV, Sparc, SystemZ and X86).
Add some more comments describing the semantics of the
MachineOperand::isRenamable function and how it is set and maintained.
Change isRenamable to check the operand's opcode
hasExtraSrcRegAllocReq/hasExtraDstRegAllocReq bit directly instead of
relying on it being consistently reflected in the IsRenamable bit
setting.
Clear the IsRenamable bit when changing an operand's register value.
Remove target code that was clearing the IsRenamable bit when changing
registers/opcodes now that this is done conservatively by default.
Change setting of hasExtraSrcRegAllocReq in AMDGPU target to be done in
one place covering all opcodes that have constant pipe read limit
restrictions.
Reviewers: qcolombet, MatzeB
Subscribers: aemerson, arsenm, jyknight, mcrosier, sdardis, nhaehnle, javed.absar, tpr, arichardson, kristof.beyls, kbarton, fedor.sergeev, asb, rbar, johnrusso, simoncook, jordy.potman.lists, apazos, sabuasal, niosHD, escha, nemanjai, llvm-commits
Differential Revision: https://reviews.llvm.org/D43042
llvm-svn: 325931
2018-02-24 02:25:08 +08:00
|
|
|
if (MRI->isReserved(Reg)) {
|
2018-01-30 02:57:07 +08:00
|
|
|
report("isRenamable set on reserved register", MO, MONum);
|
[MachineOperand][Target] MachineOperand::isRenamable semantics changes
Summary:
Add a target option AllowRegisterRenaming that is used to opt in to
post-register-allocation renaming of registers. This is set to 0 by
default, which causes the hasExtraSrcRegAllocReq/hasExtraDstRegAllocReq
fields of all opcodes to be set to 1, causing
MachineOperand::isRenamable to always return false.
Set the AllowRegisterRenaming flag to 1 for all in-tree targets that
have lit tests that were effected by enabling COPY forwarding in
MachineCopyPropagation (AArch64, AMDGPU, ARM, Hexagon, Mips, PowerPC,
RISCV, Sparc, SystemZ and X86).
Add some more comments describing the semantics of the
MachineOperand::isRenamable function and how it is set and maintained.
Change isRenamable to check the operand's opcode
hasExtraSrcRegAllocReq/hasExtraDstRegAllocReq bit directly instead of
relying on it being consistently reflected in the IsRenamable bit
setting.
Clear the IsRenamable bit when changing an operand's register value.
Remove target code that was clearing the IsRenamable bit when changing
registers/opcodes now that this is done conservatively by default.
Change setting of hasExtraSrcRegAllocReq in AMDGPU target to be done in
one place covering all opcodes that have constant pipe read limit
restrictions.
Reviewers: qcolombet, MatzeB
Subscribers: aemerson, arsenm, jyknight, mcrosier, sdardis, nhaehnle, javed.absar, tpr, arichardson, kristof.beyls, kbarton, fedor.sergeev, asb, rbar, johnrusso, simoncook, jordy.potman.lists, apazos, sabuasal, niosHD, escha, nemanjai, llvm-commits
Differential Revision: https://reviews.llvm.org/D43042
llvm-svn: 325931
2018-02-24 02:25:08 +08:00
|
|
|
return;
|
|
|
|
}
|
2017-12-13 01:53:59 +08:00
|
|
|
}
|
[DebugInfo] Make sure all DBG_VALUEs' reguse operands have IsDebug property
Summary:
In some cases, these operands lacked the IsDebug property, which is meant to signal that
they should not affect codegen. This patch adds a check for this property in the
MachineVerifier and adds it where it was missing.
This includes refactorings to use MachineInstrBuilder construction functions instead of
manually setting up the intrinsic everywhere.
Patch by: JesperAntonsson
Reviewers: aprantl, rnk, echristo, javed.absar
Reviewed By: aprantl
Subscribers: qcolombet, sdardis, nemanjai, JDevlieghere, atanasyan, llvm-commits
Differential Revision: https://reviews.llvm.org/D48319
llvm-svn: 335214
2018-06-21 18:03:34 +08:00
|
|
|
if (MI->isDebugValue() && MO->isUse() && !MO->isDebug()) {
|
|
|
|
report("Use-reg is not IsDebug in a DBG_VALUE", MO, MONum);
|
|
|
|
return;
|
|
|
|
}
|
2017-11-28 11:54:20 +08:00
|
|
|
} else {
|
|
|
|
// Virtual register.
|
|
|
|
const TargetRegisterClass *RC = MRI->getRegClassOrNull(Reg);
|
|
|
|
if (!RC) {
|
|
|
|
// This is a generic virtual register.
|
|
|
|
|
|
|
|
// If we're post-Select, we can't have gvregs anymore.
|
|
|
|
if (isFunctionSelected) {
|
|
|
|
report("Generic virtual register invalid in a Selected function",
|
|
|
|
MO, MONum);
|
|
|
|
return;
|
|
|
|
}
|
2016-08-03 00:49:22 +08:00
|
|
|
|
2017-11-28 11:54:20 +08:00
|
|
|
// The gvreg must have a type and it must not have a SubIdx.
|
|
|
|
LLT Ty = MRI->getType(Reg);
|
|
|
|
if (!Ty.isValid()) {
|
|
|
|
report("Generic virtual register must have a valid type", MO,
|
|
|
|
MONum);
|
|
|
|
return;
|
|
|
|
}
|
2016-08-03 00:17:15 +08:00
|
|
|
|
2017-11-28 11:54:20 +08:00
|
|
|
const RegisterBank *RegBank = MRI->getRegBankOrNull(Reg);
|
2016-08-03 00:17:15 +08:00
|
|
|
|
2017-11-28 11:54:20 +08:00
|
|
|
// If we're post-RegBankSelect, the gvreg must have a bank.
|
|
|
|
if (!RegBank && isFunctionRegBankSelected) {
|
|
|
|
report("Generic virtual register must have a bank in a "
|
|
|
|
"RegBankSelected function",
|
|
|
|
MO, MONum);
|
|
|
|
return;
|
|
|
|
}
|
2016-08-03 00:17:15 +08:00
|
|
|
|
2017-11-28 11:54:20 +08:00
|
|
|
// Make sure the register fits into its register bank if any.
|
|
|
|
if (RegBank && Ty.isValid() &&
|
|
|
|
RegBank->getSize() < Ty.getSizeInBits()) {
|
|
|
|
report("Register bank is too small for virtual register", MO,
|
|
|
|
MONum);
|
|
|
|
errs() << "Register bank " << RegBank->getName() << " too small("
|
|
|
|
<< RegBank->getSize() << ") to fit " << Ty.getSizeInBits()
|
|
|
|
<< "-bits\n";
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (SubIdx) {
|
2019-02-05 08:53:22 +08:00
|
|
|
report("Generic virtual register does not allow subregister index", MO,
|
2017-11-28 11:54:20 +08:00
|
|
|
MONum);
|
|
|
|
return;
|
|
|
|
}
|
2016-12-23 05:56:39 +08:00
|
|
|
|
2017-11-28 11:54:20 +08:00
|
|
|
// If this is a target specific instruction and this operand
|
|
|
|
// has register class constraint, the virtual register must
|
|
|
|
// comply to it.
|
|
|
|
if (!isPreISelGenericOpcode(MCID.getOpcode()) &&
|
|
|
|
MONum < MCID.getNumOperands() &&
|
|
|
|
TII->getRegClass(MCID, MONum, TRI, *MF)) {
|
|
|
|
report("Virtual register does not match instruction constraint", MO,
|
|
|
|
MONum);
|
|
|
|
errs() << "Expect register class "
|
|
|
|
<< TRI->getRegClassName(
|
|
|
|
TII->getRegClass(MCID, MONum, TRI, *MF))
|
|
|
|
<< " but got nothing\n";
|
|
|
|
return;
|
|
|
|
}
|
2016-12-23 05:56:39 +08:00
|
|
|
|
2017-11-28 11:54:20 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (SubIdx) {
|
|
|
|
const TargetRegisterClass *SRC =
|
|
|
|
TRI->getSubClassWithSubReg(RC, SubIdx);
|
|
|
|
if (!SRC) {
|
|
|
|
report("Invalid subregister index for virtual register", MO, MONum);
|
|
|
|
errs() << "Register class " << TRI->getRegClassName(RC)
|
|
|
|
<< " does not support subreg index " << SubIdx << "\n";
|
|
|
|
return;
|
2016-04-09 00:35:22 +08:00
|
|
|
}
|
2017-11-28 11:54:20 +08:00
|
|
|
if (RC != SRC) {
|
|
|
|
report("Invalid register class for subregister index", MO, MONum);
|
|
|
|
errs() << "Register class " << TRI->getRegClassName(RC)
|
|
|
|
<< " does not fully support subreg index " << SubIdx << "\n";
|
|
|
|
return;
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
2017-11-28 11:54:20 +08:00
|
|
|
}
|
|
|
|
if (MONum < MCID.getNumOperands()) {
|
2012-05-08 06:10:26 +08:00
|
|
|
if (const TargetRegisterClass *DRC =
|
|
|
|
TII->getRegClass(MCID, MONum, TRI, *MF)) {
|
2011-10-06 06:12:57 +08:00
|
|
|
if (SubIdx) {
|
|
|
|
const TargetRegisterClass *SuperRC =
|
2015-03-11 07:46:01 +08:00
|
|
|
TRI->getLargestLegalSuperClass(RC, *MF);
|
2011-10-06 06:12:57 +08:00
|
|
|
if (!SuperRC) {
|
|
|
|
report("No largest legal super class exists.", MO, MONum);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
DRC = TRI->getMatchingSuperRegClass(SuperRC, DRC, SubIdx);
|
|
|
|
if (!DRC) {
|
|
|
|
report("No matching super-reg register class.", MO, MONum);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2011-06-02 13:43:46 +08:00
|
|
|
if (!RC->hasSuperClassEq(DRC)) {
|
2009-05-16 08:33:53 +08:00
|
|
|
report("Illegal virtual register for instruction", MO, MONum);
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "Expected a " << TRI->getRegClassName(DRC)
|
2014-11-17 13:50:14 +08:00
|
|
|
<< " register, but got a " << TRI->getRegClassName(RC)
|
|
|
|
<< " register\n";
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2009-09-21 15:19:08 +08:00
|
|
|
|
2012-02-28 09:42:41 +08:00
|
|
|
case MachineOperand::MO_RegisterMask:
|
|
|
|
regMasks.push_back(MO->getRegMask());
|
|
|
|
break;
|
|
|
|
|
2009-09-21 15:19:08 +08:00
|
|
|
case MachineOperand::MO_MachineBasicBlock:
|
2010-02-10 03:54:29 +08:00
|
|
|
if (MI->isPHI() && !MO->getMBB()->isSuccessor(MI->getParent()))
|
|
|
|
report("PHI operand is not in the CFG", MO, MONum);
|
2009-09-21 15:19:08 +08:00
|
|
|
break;
|
|
|
|
|
2010-11-02 03:49:52 +08:00
|
|
|
case MachineOperand::MO_FrameIndex:
|
|
|
|
if (LiveStks && LiveStks->hasInterval(MO->getIndex()) &&
|
2016-02-27 14:40:41 +08:00
|
|
|
LiveInts && !LiveInts->isNotInMIMap(*MI)) {
|
2015-10-29 16:28:35 +08:00
|
|
|
int FI = MO->getIndex();
|
|
|
|
LiveInterval &LI = LiveStks->getInterval(FI);
|
2016-02-27 14:40:41 +08:00
|
|
|
SlotIndex Idx = LiveInts->getInstructionIndex(*MI);
|
2015-10-21 15:39:47 +08:00
|
|
|
|
|
|
|
bool stores = MI->mayStore();
|
2015-10-29 16:28:35 +08:00
|
|
|
bool loads = MI->mayLoad();
|
|
|
|
// For a memory-to-memory move, we need to check if the frame
|
|
|
|
// index is used for storing or loading, by inspecting the
|
|
|
|
// memory operands.
|
|
|
|
if (stores && loads) {
|
|
|
|
for (auto *MMO : MI->memoperands()) {
|
|
|
|
const PseudoSourceValue *PSV = MMO->getPseudoValue();
|
|
|
|
if (PSV == nullptr) continue;
|
|
|
|
const FixedStackPseudoSourceValue *Value =
|
|
|
|
dyn_cast<FixedStackPseudoSourceValue>(PSV);
|
|
|
|
if (Value == nullptr) continue;
|
|
|
|
if (Value->getFrameIndex() != FI) continue;
|
|
|
|
|
|
|
|
if (MMO->isStore())
|
|
|
|
loads = false;
|
|
|
|
else
|
|
|
|
stores = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (loads == stores)
|
|
|
|
report("Missing fixed stack memoperand.", MI);
|
|
|
|
}
|
|
|
|
if (loads && !LI.liveAt(Idx.getRegSlot(true))) {
|
2010-11-02 03:49:52 +08:00
|
|
|
report("Instruction loads from dead spill slot", MO, MONum);
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "Live stack: " << LI << '\n';
|
2010-11-02 03:49:52 +08:00
|
|
|
}
|
2015-10-21 15:39:47 +08:00
|
|
|
if (stores && !LI.liveAt(Idx.getRegSlot())) {
|
2010-11-02 03:49:52 +08:00
|
|
|
report("Instruction stores to dead spill slot", MO, MONum);
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "Live stack: " << LI << '\n';
|
2010-11-02 03:49:52 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-03 04:04:51 +08:00
|
|
|
void MachineVerifier::checkLivenessAtUse(const MachineOperand *MO,
|
|
|
|
unsigned MONum, SlotIndex UseIdx, const LiveRange &LR, unsigned VRegOrUnit,
|
|
|
|
LaneBitmask LaneMask) {
|
|
|
|
LiveQueryResult LRQ = LR.Query(UseIdx);
|
|
|
|
// Check if we have a segment at the use, note however that we only need one
|
|
|
|
// live subregister range, the others may be dead.
|
2016-12-15 22:36:06 +08:00
|
|
|
if (!LRQ.valueIn() && LaneMask.none()) {
|
2016-02-03 04:04:51 +08:00
|
|
|
report("No live segment at use", MO, MONum);
|
|
|
|
report_context_liverange(LR);
|
|
|
|
report_context_vreg_regunit(VRegOrUnit);
|
|
|
|
report_context(UseIdx);
|
|
|
|
}
|
|
|
|
if (MO->isKill() && !LRQ.isKill()) {
|
|
|
|
report("Live range continues after kill flag", MO, MONum);
|
|
|
|
report_context_liverange(LR);
|
|
|
|
report_context_vreg_regunit(VRegOrUnit);
|
2016-12-17 03:11:56 +08:00
|
|
|
if (LaneMask.any())
|
2016-02-03 04:04:51 +08:00
|
|
|
report_context_lanemask(LaneMask);
|
|
|
|
report_context(UseIdx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void MachineVerifier::checkLivenessAtDef(const MachineOperand *MO,
|
|
|
|
unsigned MONum, SlotIndex DefIdx, const LiveRange &LR, unsigned VRegOrUnit,
|
2018-09-20 14:59:18 +08:00
|
|
|
bool SubRangeCheck, LaneBitmask LaneMask) {
|
2016-02-03 04:04:51 +08:00
|
|
|
if (const VNInfo *VNI = LR.getVNInfoAt(DefIdx)) {
|
|
|
|
assert(VNI && "NULL valno is not allowed");
|
|
|
|
if (VNI->def != DefIdx) {
|
|
|
|
report("Inconsistent valno->def", MO, MONum);
|
|
|
|
report_context_liverange(LR);
|
|
|
|
report_context_vreg_regunit(VRegOrUnit);
|
2016-12-17 03:11:56 +08:00
|
|
|
if (LaneMask.any())
|
2016-02-03 04:04:51 +08:00
|
|
|
report_context_lanemask(LaneMask);
|
|
|
|
report_context(*VNI);
|
|
|
|
report_context(DefIdx);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
report("No live segment at def", MO, MONum);
|
|
|
|
report_context_liverange(LR);
|
|
|
|
report_context_vreg_regunit(VRegOrUnit);
|
2016-12-17 03:11:56 +08:00
|
|
|
if (LaneMask.any())
|
2016-02-03 04:04:51 +08:00
|
|
|
report_context_lanemask(LaneMask);
|
|
|
|
report_context(DefIdx);
|
|
|
|
}
|
|
|
|
// Check that, if the dead def flag is present, LiveInts agree.
|
|
|
|
if (MO->isDead()) {
|
|
|
|
LiveQueryResult LRQ = LR.Query(DefIdx);
|
|
|
|
if (!LRQ.isDeadDef()) {
|
2019-08-02 07:27:28 +08:00
|
|
|
assert(Register::isVirtualRegister(VRegOrUnit) &&
|
2018-09-20 14:59:18 +08:00
|
|
|
"Expecting a virtual register.");
|
|
|
|
// A dead subreg def only tells us that the specific subreg is dead. There
|
|
|
|
// could be other non-dead defs of other subregs, or we could have other
|
|
|
|
// parts of the register being live through the instruction. So unless we
|
|
|
|
// are checking liveness for a subrange it is ok for the live range to
|
|
|
|
// continue, given that we have a dead def of a subregister.
|
|
|
|
if (SubRangeCheck || MO->getSubReg() == 0) {
|
2016-02-03 04:04:51 +08:00
|
|
|
report("Live range continues after dead def flag", MO, MONum);
|
|
|
|
report_context_liverange(LR);
|
|
|
|
report_context_vreg_regunit(VRegOrUnit);
|
2016-12-17 03:11:56 +08:00
|
|
|
if (LaneMask.any())
|
2016-02-03 04:04:51 +08:00
|
|
|
report_context_lanemask(LaneMask);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-29 04:47:35 +08:00
|
|
|
void MachineVerifier::checkLiveness(const MachineOperand *MO, unsigned MONum) {
|
|
|
|
const MachineInstr *MI = MO->getParent();
|
|
|
|
const unsigned Reg = MO->getReg();
|
|
|
|
|
|
|
|
// Both use and def operands can read a register.
|
|
|
|
if (MO->readsReg()) {
|
2012-07-26 00:49:11 +08:00
|
|
|
if (MO->isKill())
|
2012-03-29 04:47:35 +08:00
|
|
|
addRegWithSubRegs(regsKilled, Reg);
|
|
|
|
|
|
|
|
// Check that LiveVars knows this kill.
|
2019-08-02 07:27:28 +08:00
|
|
|
if (LiveVars && Register::isVirtualRegister(Reg) && MO->isKill()) {
|
2012-03-29 04:47:35 +08:00
|
|
|
LiveVariables::VarInfo &VI = LiveVars->getVarInfo(Reg);
|
2016-08-12 06:21:41 +08:00
|
|
|
if (!is_contained(VI.Kills, MI))
|
2012-03-29 04:47:35 +08:00
|
|
|
report("Kill missing from LiveVariables", MO, MONum);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check LiveInts liveness and kill.
|
2016-02-27 14:40:41 +08:00
|
|
|
if (LiveInts && !LiveInts->isNotInMIMap(*MI)) {
|
|
|
|
SlotIndex UseIdx = LiveInts->getInstructionIndex(*MI);
|
2012-08-02 07:52:40 +08:00
|
|
|
// Check the cached regunit intervals.
|
2019-08-02 07:27:28 +08:00
|
|
|
if (Register::isPhysicalRegister(Reg) && !isReserved(Reg)) {
|
2012-08-02 07:52:40 +08:00
|
|
|
for (MCRegUnitIterator Units(Reg, TRI); Units.isValid(); ++Units) {
|
2017-09-02 02:36:26 +08:00
|
|
|
if (MRI->isReservedRegUnit(*Units))
|
|
|
|
continue;
|
2016-02-03 04:04:51 +08:00
|
|
|
if (const LiveRange *LR = LiveInts->getCachedRegUnit(*Units))
|
|
|
|
checkLivenessAtUse(MO, MONum, UseIdx, *LR, *Units);
|
2012-03-29 04:47:35 +08:00
|
|
|
}
|
2012-08-02 07:52:40 +08:00
|
|
|
}
|
|
|
|
|
2019-08-02 07:27:28 +08:00
|
|
|
if (Register::isVirtualRegister(Reg)) {
|
2012-08-02 07:52:40 +08:00
|
|
|
if (LiveInts->hasInterval(Reg)) {
|
|
|
|
// This is a virtual register interval.
|
|
|
|
const LiveInterval &LI = LiveInts->getInterval(Reg);
|
2016-02-03 04:04:51 +08:00
|
|
|
checkLivenessAtUse(MO, MONum, UseIdx, LI, Reg);
|
|
|
|
|
|
|
|
if (LI.hasSubRanges() && !MO->isDef()) {
|
|
|
|
unsigned SubRegIdx = MO->getSubReg();
|
|
|
|
LaneBitmask MOMask = SubRegIdx != 0
|
|
|
|
? TRI->getSubRegIndexLaneMask(SubRegIdx)
|
|
|
|
: MRI->getMaxLaneMaskForVReg(Reg);
|
2016-12-15 22:36:06 +08:00
|
|
|
LaneBitmask LiveInMask;
|
2016-02-03 04:04:51 +08:00
|
|
|
for (const LiveInterval::SubRange &SR : LI.subranges()) {
|
2016-12-15 22:36:06 +08:00
|
|
|
if ((MOMask & SR.LaneMask).none())
|
2016-02-03 04:04:51 +08:00
|
|
|
continue;
|
|
|
|
checkLivenessAtUse(MO, MONum, UseIdx, SR, Reg, SR.LaneMask);
|
|
|
|
LiveQueryResult LRQ = SR.Query(UseIdx);
|
|
|
|
if (LRQ.valueIn())
|
|
|
|
LiveInMask |= SR.LaneMask;
|
|
|
|
}
|
|
|
|
// At least parts of the register has to be live at the use.
|
2016-12-15 22:36:06 +08:00
|
|
|
if ((LiveInMask & MOMask).none()) {
|
2016-02-03 04:04:51 +08:00
|
|
|
report("No live subrange at use", MO, MONum);
|
|
|
|
report_context(LI);
|
|
|
|
report_context(UseIdx);
|
|
|
|
}
|
2012-08-02 07:52:40 +08:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
report("Virtual register has no live interval", MO, MONum);
|
2012-03-29 04:47:35 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Use of a dead register.
|
|
|
|
if (!regsLive.count(Reg)) {
|
2019-08-02 07:27:28 +08:00
|
|
|
if (Register::isPhysicalRegister(Reg)) {
|
2012-03-29 04:47:35 +08:00
|
|
|
// Reserved registers may be used even when 'dead'.
|
2014-12-10 09:13:13 +08:00
|
|
|
bool Bad = !isReserved(Reg);
|
|
|
|
// We are fine if just any subregister has a defined value.
|
|
|
|
if (Bad) {
|
2019-12-05 17:16:08 +08:00
|
|
|
|
|
|
|
for (const MCPhysReg &SubReg : TRI->subregs(Reg)) {
|
|
|
|
if (regsLive.count(SubReg)) {
|
2014-12-10 09:13:13 +08:00
|
|
|
Bad = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-01-15 06:25:14 +08:00
|
|
|
// If there is an additional implicit-use of a super register we stop
|
|
|
|
// here. By definition we are fine if the super register is not
|
|
|
|
// (completely) dead, if the complete super register is dead we will
|
|
|
|
// get a report for its operand.
|
|
|
|
if (Bad) {
|
|
|
|
for (const MachineOperand &MOP : MI->uses()) {
|
2018-08-28 01:40:09 +08:00
|
|
|
if (!MOP.isReg() || !MOP.isImplicit())
|
2015-01-15 06:25:14 +08:00
|
|
|
continue;
|
2018-08-28 01:40:09 +08:00
|
|
|
|
2019-08-02 07:27:28 +08:00
|
|
|
if (!Register::isPhysicalRegister(MOP.getReg()))
|
2015-01-15 06:25:14 +08:00
|
|
|
continue;
|
2018-08-28 01:40:09 +08:00
|
|
|
|
2019-12-05 17:16:08 +08:00
|
|
|
for (const MCPhysReg &SubReg : TRI->subregs(MOP.getReg())) {
|
|
|
|
if (SubReg == Reg) {
|
2015-01-15 06:25:14 +08:00
|
|
|
Bad = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-12-10 09:13:13 +08:00
|
|
|
if (Bad)
|
2012-03-29 04:47:35 +08:00
|
|
|
report("Using an undefined physical register", MO, MONum);
|
2012-07-20 07:40:38 +08:00
|
|
|
} else if (MRI->def_empty(Reg)) {
|
|
|
|
report("Reading virtual register without a def", MO, MONum);
|
2012-03-29 04:47:35 +08:00
|
|
|
} else {
|
|
|
|
BBInfo &MInfo = MBBInfoMap[MI->getParent()];
|
|
|
|
// We don't know which virtual registers are live in, so only complain
|
|
|
|
// if vreg was killed in this MBB. Otherwise keep track of vregs that
|
|
|
|
// must be live in. PHI instructions are handled separately.
|
|
|
|
if (MInfo.regsKilled.count(Reg))
|
|
|
|
report("Using a killed virtual register", MO, MONum);
|
|
|
|
else if (!MI->isPHI())
|
|
|
|
MInfo.vregsLiveIn.insert(std::make_pair(Reg, MI));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MO->isDef()) {
|
|
|
|
// Register defined.
|
|
|
|
// TODO: verify that earlyclobber ops are not used.
|
|
|
|
if (MO->isDead())
|
|
|
|
addRegWithSubRegs(regsDead, Reg);
|
|
|
|
else
|
|
|
|
addRegWithSubRegs(regsDefined, Reg);
|
|
|
|
|
|
|
|
// Verify SSA form.
|
2019-08-02 07:27:28 +08:00
|
|
|
if (MRI->isSSA() && Register::isVirtualRegister(Reg) &&
|
2014-03-02 20:27:27 +08:00
|
|
|
std::next(MRI->def_begin(Reg)) != MRI->def_end())
|
2012-03-29 04:47:35 +08:00
|
|
|
report("Multiple virtual register defs in SSA form", MO, MONum);
|
|
|
|
|
2013-10-11 05:28:43 +08:00
|
|
|
// Check LiveInts for a live segment, but only for virtual registers.
|
2016-02-27 14:40:41 +08:00
|
|
|
if (LiveInts && !LiveInts->isNotInMIMap(*MI)) {
|
|
|
|
SlotIndex DefIdx = LiveInts->getInstructionIndex(*MI);
|
2012-06-23 06:23:58 +08:00
|
|
|
DefIdx = DefIdx.getRegSlot(MO->isEarlyClobber());
|
2016-02-03 04:04:51 +08:00
|
|
|
|
2019-08-02 07:27:28 +08:00
|
|
|
if (Register::isVirtualRegister(Reg)) {
|
2016-02-03 04:04:51 +08:00
|
|
|
if (LiveInts->hasInterval(Reg)) {
|
|
|
|
const LiveInterval &LI = LiveInts->getInterval(Reg);
|
|
|
|
checkLivenessAtDef(MO, MONum, DefIdx, LI, Reg);
|
|
|
|
|
|
|
|
if (LI.hasSubRanges()) {
|
|
|
|
unsigned SubRegIdx = MO->getSubReg();
|
|
|
|
LaneBitmask MOMask = SubRegIdx != 0
|
|
|
|
? TRI->getSubRegIndexLaneMask(SubRegIdx)
|
|
|
|
: MRI->getMaxLaneMaskForVReg(Reg);
|
|
|
|
for (const LiveInterval::SubRange &SR : LI.subranges()) {
|
2016-12-15 22:36:06 +08:00
|
|
|
if ((SR.LaneMask & MOMask).none())
|
2016-02-03 04:04:51 +08:00
|
|
|
continue;
|
2018-09-20 14:59:18 +08:00
|
|
|
checkLivenessAtDef(MO, MONum, DefIdx, SR, Reg, true, SR.LaneMask);
|
2016-02-03 04:04:51 +08:00
|
|
|
}
|
2012-03-29 04:47:35 +08:00
|
|
|
}
|
|
|
|
} else {
|
2016-02-03 04:04:51 +08:00
|
|
|
report("Virtual register has no Live interval", MO, MONum);
|
2012-03-29 04:47:35 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-12 07:00:48 +08:00
|
|
|
void MachineVerifier::visitMachineInstrAfter(const MachineInstr *MI) {}
|
2012-06-07 06:34:30 +08:00
|
|
|
|
|
|
|
// This function gets called after visiting all instructions in a bundle. The
|
|
|
|
// argument points to the bundle header.
|
|
|
|
// Normal stand-alone instructions are also considered 'bundles', and this
|
|
|
|
// function is called for all of them.
|
|
|
|
void MachineVerifier::visitMachineBundleAfter(const MachineInstr *MI) {
|
2009-05-16 08:33:53 +08:00
|
|
|
BBInfo &MInfo = MBBInfoMap[MI->getParent()];
|
|
|
|
set_union(MInfo.regsKilled, regsKilled);
|
2010-08-06 02:59:59 +08:00
|
|
|
set_subtract(regsLive, regsKilled); regsKilled.clear();
|
2012-02-28 09:42:41 +08:00
|
|
|
// Kill any masked registers.
|
|
|
|
while (!regMasks.empty()) {
|
|
|
|
const uint32_t *Mask = regMasks.pop_back_val();
|
|
|
|
for (RegSet::iterator I = regsLive.begin(), E = regsLive.end(); I != E; ++I)
|
2019-08-02 07:27:28 +08:00
|
|
|
if (Register::isPhysicalRegister(*I) &&
|
2012-02-28 09:42:41 +08:00
|
|
|
MachineOperand::clobbersPhysReg(Mask, *I))
|
|
|
|
regsDead.push_back(*I);
|
|
|
|
}
|
2010-08-06 02:59:59 +08:00
|
|
|
set_subtract(regsLive, regsDead); regsDead.clear();
|
|
|
|
set_union(regsLive, regsDefined); regsDefined.clear();
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2009-10-05 02:18:39 +08:00
|
|
|
MachineVerifier::visitMachineBasicBlockAfter(const MachineBasicBlock *MBB) {
|
2009-05-16 08:33:53 +08:00
|
|
|
MBBInfoMap[MBB].regsLiveOut = regsLive;
|
|
|
|
regsLive.clear();
|
2011-01-13 05:27:48 +08:00
|
|
|
|
|
|
|
if (Indexes) {
|
|
|
|
SlotIndex stop = Indexes->getMBBEndIdx(MBB);
|
|
|
|
if (!(stop > lastIndex)) {
|
|
|
|
report("Block ends before last instruction index", MBB);
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "Block ends at " << stop
|
2011-01-13 05:27:48 +08:00
|
|
|
<< " last instruction was at " << lastIndex << '\n';
|
|
|
|
}
|
|
|
|
lastIndex = stop;
|
|
|
|
}
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
[MachineVerifier] Doing ::calcRegsPassed over faster sets: ~15-20% faster MV, NFC
MachineVerifier still takes 45-50% of total compile time with
-verify-machineinstrs, with calcRegsPassed dataflow taking ~50-60% of
MachineVerifier.
The majority of that time is spent in BBInfo::addPassed, mostly within
DenseSet implementing the sets the dataflow is operating over.
In particular, 1/4 of that DenseSet time is spent just iterating over it
(operator++), 40-50% on insertions, and most of the rest in ::count.
Given that, we're implementing custom sets just for this analysis here,
focusing on cheap insertions and O(n) iteration time (as opposed to
O(U), where U is the universe).
As it's based _mostly_ on BitVector for sparse and SmallVector for
dense, it may remotely resemble SparseSet. The difference is, our
solution is a lot less clever, doesn't have constant time `clear` that
we won't use anyway as reusing these sets across analyses is cumbersome,
and thus more space efficient and safer (got a resizable Universe and a
fallback to DenseSet for sparse if it gets too big).
With this patch MachineVerifier gets ~15-20% faster, its contribution to
total compile time drops from 45-50% to ~35%, while contribution of
calcRegsPassed to MachineVerifier drops from 50-60% to ~35% as well.
calcRegsPassed itself gets another 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: rudkx
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75033
2020-02-24 13:53:53 +08:00
|
|
|
namespace {
|
|
|
|
// This implements a set of registers that serves as a filter: can filter other
|
|
|
|
// sets by passing through elements not in the filter and blocking those that
|
|
|
|
// are. Any filter implicitly includes the full set of physical registers upon
|
|
|
|
// creation, thus filtering them all out. The filter itself as a set only grows,
|
|
|
|
// and needs to be as efficient as possible.
|
|
|
|
struct VRegFilter {
|
|
|
|
// Add elements to the filter itself. \pre Input set \p FromRegSet must have
|
|
|
|
// no duplicates. Both virtual and physical registers are fine.
|
|
|
|
template <typename RegSetT> void add(const RegSetT &FromRegSet) {
|
|
|
|
SmallVector<unsigned, 0> VRegsBuffer;
|
|
|
|
filterAndAdd(FromRegSet, VRegsBuffer);
|
|
|
|
}
|
|
|
|
// Filter \p FromRegSet through the filter and append passed elements into \p
|
|
|
|
// ToVRegs. All elements appended are then added to the filter itself.
|
|
|
|
// \returns true if anything changed.
|
|
|
|
template <typename RegSetT>
|
|
|
|
bool filterAndAdd(const RegSetT &FromRegSet,
|
|
|
|
SmallVectorImpl<unsigned> &ToVRegs) {
|
|
|
|
unsigned SparseUniverse = Sparse.size();
|
|
|
|
unsigned NewSparseUniverse = SparseUniverse;
|
|
|
|
unsigned NewDenseSize = Dense.size();
|
|
|
|
size_t Begin = ToVRegs.size();
|
|
|
|
for (unsigned Reg : FromRegSet) {
|
|
|
|
if (!Register::isVirtualRegister(Reg))
|
|
|
|
continue;
|
|
|
|
unsigned Index = Register::virtReg2Index(Reg);
|
|
|
|
if (Index < SparseUniverseMax) {
|
|
|
|
if (Index < SparseUniverse && Sparse.test(Index))
|
|
|
|
continue;
|
|
|
|
NewSparseUniverse = std::max(NewSparseUniverse, Index + 1);
|
|
|
|
} else {
|
|
|
|
if (Dense.count(Reg))
|
|
|
|
continue;
|
|
|
|
++NewDenseSize;
|
|
|
|
}
|
|
|
|
ToVRegs.push_back(Reg);
|
|
|
|
}
|
|
|
|
size_t End = ToVRegs.size();
|
|
|
|
if (Begin == End)
|
|
|
|
return false;
|
|
|
|
// Reserving space in sets once performs better than doing so continuously
|
|
|
|
// and pays easily for double look-ups (even in Dense with SparseUniverseMax
|
|
|
|
// tuned all the way down) and double iteration (the second one is over a
|
|
|
|
// SmallVector, which is a lot cheaper compared to DenseSet or BitVector).
|
|
|
|
Sparse.resize(NewSparseUniverse);
|
|
|
|
Dense.reserve(NewDenseSize);
|
|
|
|
for (unsigned I = Begin; I < End; ++I) {
|
|
|
|
unsigned Reg = ToVRegs[I];
|
|
|
|
unsigned Index = Register::virtReg2Index(Reg);
|
|
|
|
if (Index < SparseUniverseMax)
|
|
|
|
Sparse.set(Index);
|
|
|
|
else
|
|
|
|
Dense.insert(Reg);
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static constexpr unsigned SparseUniverseMax = 10 * 1024 * 8;
|
|
|
|
// VRegs indexed within SparseUniverseMax are tracked by Sparse, those beyound
|
|
|
|
// are tracked by Dense. The only purpose of the threashold and the Dense set
|
|
|
|
// is to have a reasonably growing memory usage in pathological cases (large
|
|
|
|
// number of very sparse VRegFilter instances live at the same time). In
|
|
|
|
// practice even in the worst-by-execution time cases having all elements
|
|
|
|
// tracked by Sparse (very large SparseUniverseMax scenario) tends to be more
|
|
|
|
// space efficient than if tracked by Dense. The threashold is set to keep the
|
|
|
|
// worst-case memory usage within 2x of figures determined empirically for
|
|
|
|
// "all Dense" scenario in such worst-by-execution-time cases.
|
|
|
|
BitVector Sparse;
|
|
|
|
DenseSet<unsigned> Dense;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Implements both a transfer function and a (binary, in-place) join operator
|
|
|
|
// for a dataflow over register sets with set union join and filtering transfer
|
|
|
|
// (out_b = in_b \ filter_b). filter_b is expected to be set-up ahead of time.
|
|
|
|
// Maintains out_b as its state, allowing for O(n) iteration over it at any
|
|
|
|
// time, where n is the size of the set (as opposed to O(U) where U is the
|
|
|
|
// universe). filter_b implicitly contains all physical registers at all times.
|
|
|
|
class FilteringVRegSet {
|
|
|
|
VRegFilter Filter;
|
|
|
|
SmallVector<unsigned, 0> VRegs;
|
|
|
|
|
|
|
|
public:
|
|
|
|
// Set-up the filter_b. \pre Input register set \p RS must have no duplicates.
|
|
|
|
// Both virtual and physical registers are fine.
|
|
|
|
template <typename RegSetT> void addToFilter(const RegSetT &RS) {
|
|
|
|
Filter.add(RS);
|
|
|
|
}
|
|
|
|
// Passes \p RS through the filter_b (transfer function) and adds what's left
|
|
|
|
// to itself (out_b).
|
|
|
|
template <typename RegSetT> bool add(const RegSetT &RS) {
|
|
|
|
// Double-duty the Filter: to maintain VRegs a set (and the join operation
|
|
|
|
// a set union) just add everything being added here to the Filter as well.
|
|
|
|
return Filter.filterAndAdd(RS, VRegs);
|
|
|
|
}
|
|
|
|
using const_iterator = decltype(VRegs)::const_iterator;
|
|
|
|
const_iterator begin() const { return VRegs.begin(); }
|
|
|
|
const_iterator end() const { return VRegs.end(); }
|
|
|
|
size_t size() const { return VRegs.size(); }
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
// Calculate the largest possible vregsPassed sets. These are the registers that
|
|
|
|
// can pass through an MBB live, but may not be live every time. It is assumed
|
|
|
|
// that all vregsPassed sets are empty before the call.
|
2010-01-06 04:59:36 +08:00
|
|
|
void MachineVerifier::calcRegsPassed() {
|
[MachineVerifier] Doing ::calcRegsPassed in RPO: ~35% faster MV, NFC
Depending on the target, test suite, pipeline config and perhaps other
factors machine verifier when forced on with -verify-machineinstrs can
increase compile time 2-2.5 times over (Release, Asserts On), taking up
~60% of the time. An invaluable tool, it significantly slows down
machine verifier-enabled testing.
Nearly 75% of its time MachineVerifier spends in the calcRegsPassed
method. It's a classic forward dataflow analysis executed over sets, but
visiting MBBs in arbitrary order. We switch that to RPO here.
This speeds up MachineVerifier by about 35%, decreasing the overall
compile time with -verify-machineinstrs by 20-25% or so.
calcRegsPassed itself gets 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: bogner
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75032
2020-02-24 13:53:19 +08:00
|
|
|
// This is a forward dataflow, doing it in RPO. A standard map serves as a
|
|
|
|
// priority (sorting by RPO number) queue, deduplicating worklist, and an RPO
|
|
|
|
// number to MBB mapping all at once.
|
|
|
|
std::map<unsigned, const MachineBasicBlock *> RPOWorklist;
|
|
|
|
DenseMap<const MachineBasicBlock *, unsigned> RPONumbers;
|
|
|
|
if (MF->empty()) {
|
|
|
|
// ReversePostOrderTraversal doesn't handle empty functions.
|
|
|
|
return;
|
|
|
|
}
|
[MachineVerifier] Doing ::calcRegsPassed over faster sets: ~15-20% faster MV, NFC
MachineVerifier still takes 45-50% of total compile time with
-verify-machineinstrs, with calcRegsPassed dataflow taking ~50-60% of
MachineVerifier.
The majority of that time is spent in BBInfo::addPassed, mostly within
DenseSet implementing the sets the dataflow is operating over.
In particular, 1/4 of that DenseSet time is spent just iterating over it
(operator++), 40-50% on insertions, and most of the rest in ::count.
Given that, we're implementing custom sets just for this analysis here,
focusing on cheap insertions and O(n) iteration time (as opposed to
O(U), where U is the universe).
As it's based _mostly_ on BitVector for sparse and SmallVector for
dense, it may remotely resemble SparseSet. The difference is, our
solution is a lot less clever, doesn't have constant time `clear` that
we won't use anyway as reusing these sets across analyses is cumbersome,
and thus more space efficient and safer (got a resizable Universe and a
fallback to DenseSet for sparse if it gets too big).
With this patch MachineVerifier gets ~15-20% faster, its contribution to
total compile time drops from 45-50% to ~35%, while contribution of
calcRegsPassed to MachineVerifier drops from 50-60% to ~35% as well.
calcRegsPassed itself gets another 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: rudkx
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75033
2020-02-24 13:53:53 +08:00
|
|
|
std::vector<FilteringVRegSet> VRegsPassedSets(MF->size());
|
[MachineVerifier] Doing ::calcRegsPassed in RPO: ~35% faster MV, NFC
Depending on the target, test suite, pipeline config and perhaps other
factors machine verifier when forced on with -verify-machineinstrs can
increase compile time 2-2.5 times over (Release, Asserts On), taking up
~60% of the time. An invaluable tool, it significantly slows down
machine verifier-enabled testing.
Nearly 75% of its time MachineVerifier spends in the calcRegsPassed
method. It's a classic forward dataflow analysis executed over sets, but
visiting MBBs in arbitrary order. We switch that to RPO here.
This speeds up MachineVerifier by about 35%, decreasing the overall
compile time with -verify-machineinstrs by 20-25% or so.
calcRegsPassed itself gets 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: bogner
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75032
2020-02-24 13:53:19 +08:00
|
|
|
for (const MachineBasicBlock *MBB :
|
|
|
|
ReversePostOrderTraversal<const MachineFunction *>(MF)) {
|
|
|
|
// Careful with the evaluation order, fetch next number before allocating.
|
|
|
|
unsigned Number = RPONumbers.size();
|
|
|
|
RPONumbers[MBB] = Number;
|
[MachineVerifier] Doing ::calcRegsPassed over faster sets: ~15-20% faster MV, NFC
MachineVerifier still takes 45-50% of total compile time with
-verify-machineinstrs, with calcRegsPassed dataflow taking ~50-60% of
MachineVerifier.
The majority of that time is spent in BBInfo::addPassed, mostly within
DenseSet implementing the sets the dataflow is operating over.
In particular, 1/4 of that DenseSet time is spent just iterating over it
(operator++), 40-50% on insertions, and most of the rest in ::count.
Given that, we're implementing custom sets just for this analysis here,
focusing on cheap insertions and O(n) iteration time (as opposed to
O(U), where U is the universe).
As it's based _mostly_ on BitVector for sparse and SmallVector for
dense, it may remotely resemble SparseSet. The difference is, our
solution is a lot less clever, doesn't have constant time `clear` that
we won't use anyway as reusing these sets across analyses is cumbersome,
and thus more space efficient and safer (got a resizable Universe and a
fallback to DenseSet for sparse if it gets too big).
With this patch MachineVerifier gets ~15-20% faster, its contribution to
total compile time drops from 45-50% to ~35%, while contribution of
calcRegsPassed to MachineVerifier drops from 50-60% to ~35% as well.
calcRegsPassed itself gets another 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: rudkx
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75033
2020-02-24 13:53:53 +08:00
|
|
|
// Set-up the transfer functions for all blocks.
|
|
|
|
const BBInfo &MInfo = MBBInfoMap[MBB];
|
|
|
|
VRegsPassedSets[Number].addToFilter(MInfo.regsKilled);
|
|
|
|
VRegsPassedSets[Number].addToFilter(MInfo.regsLiveOut);
|
[MachineVerifier] Doing ::calcRegsPassed in RPO: ~35% faster MV, NFC
Depending on the target, test suite, pipeline config and perhaps other
factors machine verifier when forced on with -verify-machineinstrs can
increase compile time 2-2.5 times over (Release, Asserts On), taking up
~60% of the time. An invaluable tool, it significantly slows down
machine verifier-enabled testing.
Nearly 75% of its time MachineVerifier spends in the calcRegsPassed
method. It's a classic forward dataflow analysis executed over sets, but
visiting MBBs in arbitrary order. We switch that to RPO here.
This speeds up MachineVerifier by about 35%, decreasing the overall
compile time with -verify-machineinstrs by 20-25% or so.
calcRegsPassed itself gets 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: bogner
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75032
2020-02-24 13:53:19 +08:00
|
|
|
}
|
2009-05-16 08:33:53 +08:00
|
|
|
// First push live-out regs to successors' vregsPassed. Remember the MBBs that
|
|
|
|
// have any vregsPassed.
|
[MachineVerifier] Doing ::calcRegsPassed in RPO: ~35% faster MV, NFC
Depending on the target, test suite, pipeline config and perhaps other
factors machine verifier when forced on with -verify-machineinstrs can
increase compile time 2-2.5 times over (Release, Asserts On), taking up
~60% of the time. An invaluable tool, it significantly slows down
machine verifier-enabled testing.
Nearly 75% of its time MachineVerifier spends in the calcRegsPassed
method. It's a classic forward dataflow analysis executed over sets, but
visiting MBBs in arbitrary order. We switch that to RPO here.
This speeds up MachineVerifier by about 35%, decreasing the overall
compile time with -verify-machineinstrs by 20-25% or so.
calcRegsPassed itself gets 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: bogner
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75032
2020-02-24 13:53:19 +08:00
|
|
|
for (const MachineBasicBlock &MBB : *MF) {
|
[MachineVerifier] Doing ::calcRegsPassed over faster sets: ~15-20% faster MV, NFC
MachineVerifier still takes 45-50% of total compile time with
-verify-machineinstrs, with calcRegsPassed dataflow taking ~50-60% of
MachineVerifier.
The majority of that time is spent in BBInfo::addPassed, mostly within
DenseSet implementing the sets the dataflow is operating over.
In particular, 1/4 of that DenseSet time is spent just iterating over it
(operator++), 40-50% on insertions, and most of the rest in ::count.
Given that, we're implementing custom sets just for this analysis here,
focusing on cheap insertions and O(n) iteration time (as opposed to
O(U), where U is the universe).
As it's based _mostly_ on BitVector for sparse and SmallVector for
dense, it may remotely resemble SparseSet. The difference is, our
solution is a lot less clever, doesn't have constant time `clear` that
we won't use anyway as reusing these sets across analyses is cumbersome,
and thus more space efficient and safer (got a resizable Universe and a
fallback to DenseSet for sparse if it gets too big).
With this patch MachineVerifier gets ~15-20% faster, its contribution to
total compile time drops from 45-50% to ~35%, while contribution of
calcRegsPassed to MachineVerifier drops from 50-60% to ~35% as well.
calcRegsPassed itself gets another 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: rudkx
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75033
2020-02-24 13:53:53 +08:00
|
|
|
const BBInfo &MInfo = MBBInfoMap[&MBB];
|
2009-05-16 08:33:53 +08:00
|
|
|
if (!MInfo.reachable)
|
|
|
|
continue;
|
[MachineVerifier] Doing ::calcRegsPassed in RPO: ~35% faster MV, NFC
Depending on the target, test suite, pipeline config and perhaps other
factors machine verifier when forced on with -verify-machineinstrs can
increase compile time 2-2.5 times over (Release, Asserts On), taking up
~60% of the time. An invaluable tool, it significantly slows down
machine verifier-enabled testing.
Nearly 75% of its time MachineVerifier spends in the calcRegsPassed
method. It's a classic forward dataflow analysis executed over sets, but
visiting MBBs in arbitrary order. We switch that to RPO here.
This speeds up MachineVerifier by about 35%, decreasing the overall
compile time with -verify-machineinstrs by 20-25% or so.
calcRegsPassed itself gets 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: bogner
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75032
2020-02-24 13:53:19 +08:00
|
|
|
for (const MachineBasicBlock *Succ : MBB.successors()) {
|
[MachineVerifier] Doing ::calcRegsPassed over faster sets: ~15-20% faster MV, NFC
MachineVerifier still takes 45-50% of total compile time with
-verify-machineinstrs, with calcRegsPassed dataflow taking ~50-60% of
MachineVerifier.
The majority of that time is spent in BBInfo::addPassed, mostly within
DenseSet implementing the sets the dataflow is operating over.
In particular, 1/4 of that DenseSet time is spent just iterating over it
(operator++), 40-50% on insertions, and most of the rest in ::count.
Given that, we're implementing custom sets just for this analysis here,
focusing on cheap insertions and O(n) iteration time (as opposed to
O(U), where U is the universe).
As it's based _mostly_ on BitVector for sparse and SmallVector for
dense, it may remotely resemble SparseSet. The difference is, our
solution is a lot less clever, doesn't have constant time `clear` that
we won't use anyway as reusing these sets across analyses is cumbersome,
and thus more space efficient and safer (got a resizable Universe and a
fallback to DenseSet for sparse if it gets too big).
With this patch MachineVerifier gets ~15-20% faster, its contribution to
total compile time drops from 45-50% to ~35%, while contribution of
calcRegsPassed to MachineVerifier drops from 50-60% to ~35% as well.
calcRegsPassed itself gets another 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: rudkx
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75033
2020-02-24 13:53:53 +08:00
|
|
|
unsigned SuccNumber = RPONumbers[Succ];
|
|
|
|
FilteringVRegSet &SuccSet = VRegsPassedSets[SuccNumber];
|
|
|
|
if (SuccSet.add(MInfo.regsLiveOut))
|
|
|
|
RPOWorklist.emplace(SuccNumber, Succ);
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
[MachineVerifier] Doing ::calcRegsPassed in RPO: ~35% faster MV, NFC
Depending on the target, test suite, pipeline config and perhaps other
factors machine verifier when forced on with -verify-machineinstrs can
increase compile time 2-2.5 times over (Release, Asserts On), taking up
~60% of the time. An invaluable tool, it significantly slows down
machine verifier-enabled testing.
Nearly 75% of its time MachineVerifier spends in the calcRegsPassed
method. It's a classic forward dataflow analysis executed over sets, but
visiting MBBs in arbitrary order. We switch that to RPO here.
This speeds up MachineVerifier by about 35%, decreasing the overall
compile time with -verify-machineinstrs by 20-25% or so.
calcRegsPassed itself gets 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: bogner
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75032
2020-02-24 13:53:19 +08:00
|
|
|
// Iteratively push vregsPassed to successors.
|
|
|
|
while (!RPOWorklist.empty()) {
|
|
|
|
auto Next = RPOWorklist.begin();
|
|
|
|
const MachineBasicBlock *MBB = Next->second;
|
|
|
|
RPOWorklist.erase(Next);
|
[MachineVerifier] Doing ::calcRegsPassed over faster sets: ~15-20% faster MV, NFC
MachineVerifier still takes 45-50% of total compile time with
-verify-machineinstrs, with calcRegsPassed dataflow taking ~50-60% of
MachineVerifier.
The majority of that time is spent in BBInfo::addPassed, mostly within
DenseSet implementing the sets the dataflow is operating over.
In particular, 1/4 of that DenseSet time is spent just iterating over it
(operator++), 40-50% on insertions, and most of the rest in ::count.
Given that, we're implementing custom sets just for this analysis here,
focusing on cheap insertions and O(n) iteration time (as opposed to
O(U), where U is the universe).
As it's based _mostly_ on BitVector for sparse and SmallVector for
dense, it may remotely resemble SparseSet. The difference is, our
solution is a lot less clever, doesn't have constant time `clear` that
we won't use anyway as reusing these sets across analyses is cumbersome,
and thus more space efficient and safer (got a resizable Universe and a
fallback to DenseSet for sparse if it gets too big).
With this patch MachineVerifier gets ~15-20% faster, its contribution to
total compile time drops from 45-50% to ~35%, while contribution of
calcRegsPassed to MachineVerifier drops from 50-60% to ~35% as well.
calcRegsPassed itself gets another 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: rudkx
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75033
2020-02-24 13:53:53 +08:00
|
|
|
FilteringVRegSet &MSet = VRegsPassedSets[RPONumbers[MBB]];
|
[MachineVerifier] Doing ::calcRegsPassed in RPO: ~35% faster MV, NFC
Depending on the target, test suite, pipeline config and perhaps other
factors machine verifier when forced on with -verify-machineinstrs can
increase compile time 2-2.5 times over (Release, Asserts On), taking up
~60% of the time. An invaluable tool, it significantly slows down
machine verifier-enabled testing.
Nearly 75% of its time MachineVerifier spends in the calcRegsPassed
method. It's a classic forward dataflow analysis executed over sets, but
visiting MBBs in arbitrary order. We switch that to RPO here.
This speeds up MachineVerifier by about 35%, decreasing the overall
compile time with -verify-machineinstrs by 20-25% or so.
calcRegsPassed itself gets 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: bogner
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75032
2020-02-24 13:53:19 +08:00
|
|
|
for (const MachineBasicBlock *Succ : MBB->successors()) {
|
|
|
|
if (Succ == MBB)
|
2009-05-16 08:33:53 +08:00
|
|
|
continue;
|
[MachineVerifier] Doing ::calcRegsPassed over faster sets: ~15-20% faster MV, NFC
MachineVerifier still takes 45-50% of total compile time with
-verify-machineinstrs, with calcRegsPassed dataflow taking ~50-60% of
MachineVerifier.
The majority of that time is spent in BBInfo::addPassed, mostly within
DenseSet implementing the sets the dataflow is operating over.
In particular, 1/4 of that DenseSet time is spent just iterating over it
(operator++), 40-50% on insertions, and most of the rest in ::count.
Given that, we're implementing custom sets just for this analysis here,
focusing on cheap insertions and O(n) iteration time (as opposed to
O(U), where U is the universe).
As it's based _mostly_ on BitVector for sparse and SmallVector for
dense, it may remotely resemble SparseSet. The difference is, our
solution is a lot less clever, doesn't have constant time `clear` that
we won't use anyway as reusing these sets across analyses is cumbersome,
and thus more space efficient and safer (got a resizable Universe and a
fallback to DenseSet for sparse if it gets too big).
With this patch MachineVerifier gets ~15-20% faster, its contribution to
total compile time drops from 45-50% to ~35%, while contribution of
calcRegsPassed to MachineVerifier drops from 50-60% to ~35% as well.
calcRegsPassed itself gets another 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: rudkx
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75033
2020-02-24 13:53:53 +08:00
|
|
|
unsigned SuccNumber = RPONumbers[Succ];
|
|
|
|
FilteringVRegSet &SuccSet = VRegsPassedSets[SuccNumber];
|
|
|
|
if (SuccSet.add(MSet))
|
|
|
|
RPOWorklist.emplace(SuccNumber, Succ);
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
}
|
[MachineVerifier] Doing ::calcRegsPassed over faster sets: ~15-20% faster MV, NFC
MachineVerifier still takes 45-50% of total compile time with
-verify-machineinstrs, with calcRegsPassed dataflow taking ~50-60% of
MachineVerifier.
The majority of that time is spent in BBInfo::addPassed, mostly within
DenseSet implementing the sets the dataflow is operating over.
In particular, 1/4 of that DenseSet time is spent just iterating over it
(operator++), 40-50% on insertions, and most of the rest in ::count.
Given that, we're implementing custom sets just for this analysis here,
focusing on cheap insertions and O(n) iteration time (as opposed to
O(U), where U is the universe).
As it's based _mostly_ on BitVector for sparse and SmallVector for
dense, it may remotely resemble SparseSet. The difference is, our
solution is a lot less clever, doesn't have constant time `clear` that
we won't use anyway as reusing these sets across analyses is cumbersome,
and thus more space efficient and safer (got a resizable Universe and a
fallback to DenseSet for sparse if it gets too big).
With this patch MachineVerifier gets ~15-20% faster, its contribution to
total compile time drops from 45-50% to ~35%, while contribution of
calcRegsPassed to MachineVerifier drops from 50-60% to ~35% as well.
calcRegsPassed itself gets another 2x faster here.
All measured on a large suite of shaders targeting a number of GPUs.
Reviewers: bogner, stoklund, rudkx, qcolombet
Reviewed By: rudkx
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D75033
2020-02-24 13:53:53 +08:00
|
|
|
// Copy the results back to BBInfos.
|
|
|
|
for (const MachineBasicBlock &MBB : *MF) {
|
|
|
|
BBInfo &MInfo = MBBInfoMap[&MBB];
|
|
|
|
if (!MInfo.reachable)
|
|
|
|
continue;
|
|
|
|
const FilteringVRegSet &MSet = VRegsPassedSets[RPONumbers[&MBB]];
|
|
|
|
MInfo.vregsPassed.reserve(MSet.size());
|
|
|
|
MInfo.vregsPassed.insert(MSet.begin(), MSet.end());
|
|
|
|
}
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
2009-11-19 04:36:57 +08:00
|
|
|
// Calculate the set of virtual registers that must be passed through each basic
|
|
|
|
// block in order to satisfy the requirements of successor blocks. This is very
|
2010-01-06 04:59:36 +08:00
|
|
|
// similar to calcRegsPassed, only backwards.
|
2009-11-19 04:36:57 +08:00
|
|
|
void MachineVerifier::calcRegsRequired() {
|
|
|
|
// First push live-in regs to predecessors' vregsRequired.
|
2012-03-10 08:36:04 +08:00
|
|
|
SmallPtrSet<const MachineBasicBlock*, 8> todo;
|
2014-05-01 02:29:51 +08:00
|
|
|
for (const auto &MBB : *MF) {
|
2009-11-19 04:36:57 +08:00
|
|
|
BBInfo &MInfo = MBBInfoMap[&MBB];
|
|
|
|
for (MachineBasicBlock::const_pred_iterator PrI = MBB.pred_begin(),
|
|
|
|
PrE = MBB.pred_end(); PrI != PrE; ++PrI) {
|
|
|
|
BBInfo &PInfo = MBBInfoMap[*PrI];
|
|
|
|
if (PInfo.addRequired(MInfo.vregsLiveIn))
|
|
|
|
todo.insert(*PrI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Iteratively push vregsRequired to predecessors. This will converge to the
|
|
|
|
// same final state regardless of DenseSet iteration order.
|
|
|
|
while (!todo.empty()) {
|
|
|
|
const MachineBasicBlock *MBB = *todo.begin();
|
|
|
|
todo.erase(MBB);
|
|
|
|
BBInfo &MInfo = MBBInfoMap[MBB];
|
|
|
|
for (MachineBasicBlock::const_pred_iterator PrI = MBB->pred_begin(),
|
|
|
|
PrE = MBB->pred_end(); PrI != PrE; ++PrI) {
|
|
|
|
if (*PrI == MBB)
|
|
|
|
continue;
|
|
|
|
BBInfo &SInfo = MBBInfoMap[*PrI];
|
|
|
|
if (SInfo.addRequired(MInfo.vregsRequired))
|
|
|
|
todo.insert(*PrI);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-16 08:33:53 +08:00
|
|
|
// Check PHI instructions at the beginning of MBB. It is assumed that
|
2010-01-06 04:59:36 +08:00
|
|
|
// calcRegsPassed has been run so BBInfo::isLiveOut is valid.
|
2017-11-28 11:54:19 +08:00
|
|
|
void MachineVerifier::checkPHIOps(const MachineBasicBlock &MBB) {
|
|
|
|
BBInfo &MInfo = MBBInfoMap[&MBB];
|
|
|
|
|
2012-03-10 08:36:04 +08:00
|
|
|
SmallPtrSet<const MachineBasicBlock*, 8> seen;
|
2017-11-28 11:54:19 +08:00
|
|
|
for (const MachineInstr &Phi : MBB) {
|
|
|
|
if (!Phi.isPHI())
|
2014-05-01 06:17:38 +08:00
|
|
|
break;
|
2012-03-10 08:36:04 +08:00
|
|
|
seen.clear();
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2017-11-28 11:54:19 +08:00
|
|
|
const MachineOperand &MODef = Phi.getOperand(0);
|
|
|
|
if (!MODef.isReg() || !MODef.isDef()) {
|
|
|
|
report("Expected first PHI operand to be a register def", &MODef, 0);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (MODef.isTied() || MODef.isImplicit() || MODef.isInternalRead() ||
|
|
|
|
MODef.isEarlyClobber() || MODef.isDebug())
|
|
|
|
report("Unexpected flag on PHI operand", &MODef, 0);
|
Apply llvm-prefer-register-over-unsigned from clang-tidy to LLVM
Summary:
This clang-tidy check is looking for unsigned integer variables whose initializer
starts with an implicit cast from llvm::Register and changes the type of the
variable to llvm::Register (dropping the llvm:: where possible).
Partial reverts in:
X86FrameLowering.cpp - Some functions return unsigned and arguably should be MCRegister
X86FixupLEAs.cpp - Some functions return unsigned and arguably should be MCRegister
X86FrameLowering.cpp - Some functions return unsigned and arguably should be MCRegister
HexagonBitSimplify.cpp - Function takes BitTracker::RegisterRef which appears to be unsigned&
MachineVerifier.cpp - Ambiguous operator==() given MCRegister and const Register
PPCFastISel.cpp - No Register::operator-=()
PeepholeOptimizer.cpp - TargetInstrInfo::optimizeLoadInstr() takes an unsigned&
MachineTraceMetrics.cpp - MachineTraceMetrics lacks a suitable constructor
Manual fixups in:
ARMFastISel.cpp - ARMEmitLoad() now takes a Register& instead of unsigned&
HexagonSplitDouble.cpp - Ternary operator was ambiguous between unsigned/Register
HexagonConstExtenders.cpp - Has a local class named Register, used llvm::Register instead of Register.
PPCFastISel.cpp - PPCEmitLoad() now takes a Register& instead of unsigned&
Depends on D65919
Reviewers: arsenm, bogner, craig.topper, RKSimon
Reviewed By: arsenm
Subscribers: RKSimon, craig.topper, lenary, aemerson, wuzish, jholewinski, MatzeB, qcolombet, dschuff, jyknight, dylanmckay, sdardis, nemanjai, jvesely, wdng, nhaehnle, sbc100, jgravelle-google, kristof.beyls, hiraditya, aheejin, kbarton, fedor.sergeev, javed.absar, asb, rbar, johnrusso, simoncook, apazos, sabuasal, niosHD, jrtc27, MaskRay, zzheng, edward-jones, atanasyan, rogfer01, MartinMosbeck, brucehoult, the_o, tpr, PkmX, jocewei, jsji, Petar.Avramovic, asbirlea, Jim, s.egerton, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D65962
llvm-svn: 369041
2019-08-16 03:22:08 +08:00
|
|
|
Register DefReg = MODef.getReg();
|
2019-08-02 07:27:28 +08:00
|
|
|
if (!Register::isVirtualRegister(DefReg))
|
2017-11-28 11:54:19 +08:00
|
|
|
report("Expected first PHI operand to be a virtual register", &MODef, 0);
|
|
|
|
|
|
|
|
for (unsigned I = 1, E = Phi.getNumOperands(); I != E; I += 2) {
|
|
|
|
const MachineOperand &MO0 = Phi.getOperand(I);
|
|
|
|
if (!MO0.isReg()) {
|
|
|
|
report("Expected PHI operand to be a register", &MO0, I);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (MO0.isImplicit() || MO0.isInternalRead() || MO0.isEarlyClobber() ||
|
|
|
|
MO0.isDebug() || MO0.isTied())
|
|
|
|
report("Unexpected flag on PHI operand", &MO0, I);
|
|
|
|
|
|
|
|
const MachineOperand &MO1 = Phi.getOperand(I + 1);
|
|
|
|
if (!MO1.isMBB()) {
|
|
|
|
report("Expected PHI operand to be a basic block", &MO1, I + 1);
|
2009-05-16 08:33:53 +08:00
|
|
|
continue;
|
2017-11-28 11:54:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
const MachineBasicBlock &Pre = *MO1.getMBB();
|
|
|
|
if (!Pre.isSuccessor(&MBB)) {
|
|
|
|
report("PHI input is not a predecessor block", &MO1, I + 1);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MInfo.reachable) {
|
|
|
|
seen.insert(&Pre);
|
|
|
|
BBInfo &PrInfo = MBBInfoMap[&Pre];
|
2017-12-05 02:57:48 +08:00
|
|
|
if (!MO0.isUndef() && PrInfo.reachable &&
|
|
|
|
!PrInfo.isLiveOut(MO0.getReg()))
|
2017-11-28 11:54:19 +08:00
|
|
|
report("PHI operand is not live-out from predecessor", &MO0, I);
|
|
|
|
}
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Did we see all predecessors?
|
2017-11-28 11:54:19 +08:00
|
|
|
if (MInfo.reachable) {
|
|
|
|
for (MachineBasicBlock *Pred : MBB.predecessors()) {
|
|
|
|
if (!seen.count(Pred)) {
|
|
|
|
report("Missing PHI operand", &Phi);
|
2017-12-05 01:18:51 +08:00
|
|
|
errs() << printMBBReference(*Pred)
|
|
|
|
<< " is a predecessor according to the CFG.\n";
|
2017-11-28 11:54:19 +08:00
|
|
|
}
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-10-05 02:18:39 +08:00
|
|
|
void MachineVerifier::visitMachineFunctionAfter() {
|
2010-01-06 04:59:36 +08:00
|
|
|
calcRegsPassed();
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2020-01-16 22:01:27 +08:00
|
|
|
for (const MachineBasicBlock &MBB : *MF)
|
2017-11-28 11:54:19 +08:00
|
|
|
checkPHIOps(MBB);
|
2009-05-16 08:33:53 +08:00
|
|
|
|
2010-08-07 02:04:19 +08:00
|
|
|
// Now check liveness info if available
|
2012-03-10 08:36:06 +08:00
|
|
|
calcRegsRequired();
|
|
|
|
|
2012-06-30 05:00:00 +08:00
|
|
|
// Check for killed virtual registers that should be live out.
|
2014-05-01 02:29:51 +08:00
|
|
|
for (const auto &MBB : *MF) {
|
|
|
|
BBInfo &MInfo = MBBInfoMap[&MBB];
|
2012-06-30 05:00:00 +08:00
|
|
|
for (RegSet::iterator
|
|
|
|
I = MInfo.vregsRequired.begin(), E = MInfo.vregsRequired.end(); I != E;
|
|
|
|
++I)
|
|
|
|
if (MInfo.regsKilled.count(*I)) {
|
2014-05-01 02:29:51 +08:00
|
|
|
report("Virtual register killed in block, but needed live out.", &MBB);
|
2017-11-28 20:42:37 +08:00
|
|
|
errs() << "Virtual register " << printReg(*I)
|
2017-12-01 00:12:24 +08:00
|
|
|
<< " is used after the block.\n";
|
2012-06-30 05:00:00 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-06-26 02:18:27 +08:00
|
|
|
if (!MF->empty()) {
|
2012-03-10 08:36:06 +08:00
|
|
|
BBInfo &MInfo = MBBInfoMap[&MF->front()];
|
|
|
|
for (RegSet::iterator
|
|
|
|
I = MInfo.vregsRequired.begin(), E = MInfo.vregsRequired.end(); I != E;
|
2016-05-12 05:31:39 +08:00
|
|
|
++I) {
|
|
|
|
report("Virtual register defs don't dominate all uses.", MF);
|
|
|
|
report_context_vreg(*I);
|
|
|
|
}
|
2012-03-10 08:36:06 +08:00
|
|
|
}
|
|
|
|
|
2010-08-07 02:04:19 +08:00
|
|
|
if (LiveVars)
|
2009-11-19 04:36:57 +08:00
|
|
|
verifyLiveVariables();
|
2010-08-07 02:04:19 +08:00
|
|
|
if (LiveInts)
|
|
|
|
verifyLiveIntervals();
|
2019-06-27 15:48:06 +08:00
|
|
|
|
2019-11-19 19:34:49 +08:00
|
|
|
// Check live-in list of each MBB. If a register is live into MBB, check
|
|
|
|
// that the register is in regsLiveOut of each predecessor block. Since
|
|
|
|
// this must come from a definition in the predecesssor or its live-in
|
|
|
|
// list, this will catch a live-through case where the predecessor does not
|
|
|
|
// have the register in its live-in list. This currently only checks
|
|
|
|
// registers that have no aliases, are not allocatable and are not
|
|
|
|
// reserved, which could mean a condition code register for instance.
|
|
|
|
if (MRI->tracksLiveness())
|
|
|
|
for (const auto &MBB : *MF)
|
|
|
|
for (MachineBasicBlock::RegisterMaskPair P : MBB.liveins()) {
|
|
|
|
MCPhysReg LiveInReg = P.PhysReg;
|
|
|
|
bool hasAliases = MCRegAliasIterator(LiveInReg, TRI, false).isValid();
|
|
|
|
if (hasAliases || isAllocatable(LiveInReg) || isReserved(LiveInReg))
|
|
|
|
continue;
|
|
|
|
for (const MachineBasicBlock *Pred : MBB.predecessors()) {
|
|
|
|
BBInfo &PInfo = MBBInfoMap[Pred];
|
|
|
|
if (!PInfo.regsLiveOut.count(LiveInReg)) {
|
|
|
|
report("Live in register not found to be live out from predecessor.",
|
|
|
|
&MBB);
|
|
|
|
errs() << TRI->getName(LiveInReg)
|
|
|
|
<< " not found to be live out from "
|
|
|
|
<< printMBBReference(*Pred) << "\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-27 15:48:06 +08:00
|
|
|
for (auto CSInfo : MF->getCallSitesInfo())
|
|
|
|
if (!CSInfo.first->isCall())
|
|
|
|
report("Call site info referencing instruction that is not call", MF);
|
2009-05-16 08:33:53 +08:00
|
|
|
}
|
2009-11-19 04:36:57 +08:00
|
|
|
|
|
|
|
void MachineVerifier::verifyLiveVariables() {
|
|
|
|
assert(LiveVars && "Don't call verifyLiveVariables without LiveVars");
|
2011-01-09 07:11:02 +08:00
|
|
|
for (unsigned i = 0, e = MRI->getNumVirtRegs(); i != e; ++i) {
|
2019-08-02 07:27:28 +08:00
|
|
|
unsigned Reg = Register::index2VirtReg(i);
|
2009-11-19 04:36:57 +08:00
|
|
|
LiveVariables::VarInfo &VI = LiveVars->getVarInfo(Reg);
|
2014-05-01 02:29:51 +08:00
|
|
|
for (const auto &MBB : *MF) {
|
|
|
|
BBInfo &MInfo = MBBInfoMap[&MBB];
|
2009-11-19 04:36:57 +08:00
|
|
|
|
|
|
|
// Our vregsRequired should be identical to LiveVariables' AliveBlocks
|
|
|
|
if (MInfo.vregsRequired.count(Reg)) {
|
2014-05-01 02:29:51 +08:00
|
|
|
if (!VI.AliveBlocks.test(MBB.getNumber())) {
|
|
|
|
report("LiveVariables: Block missing from AliveBlocks", &MBB);
|
2017-11-28 20:42:37 +08:00
|
|
|
errs() << "Virtual register " << printReg(Reg)
|
2017-12-01 00:12:24 +08:00
|
|
|
<< " must be live through the block.\n";
|
2009-11-19 04:36:57 +08:00
|
|
|
}
|
|
|
|
} else {
|
2014-05-01 02:29:51 +08:00
|
|
|
if (VI.AliveBlocks.test(MBB.getNumber())) {
|
|
|
|
report("LiveVariables: Block should not be in AliveBlocks", &MBB);
|
2017-11-28 20:42:37 +08:00
|
|
|
errs() << "Virtual register " << printReg(Reg)
|
2017-12-01 00:12:24 +08:00
|
|
|
<< " is not needed live through the block.\n";
|
2009-11-19 04:36:57 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-07 02:04:19 +08:00
|
|
|
void MachineVerifier::verifyLiveIntervals() {
|
|
|
|
assert(LiveInts && "Don't call verifyLiveIntervals without LiveInts");
|
2012-06-21 07:23:59 +08:00
|
|
|
for (unsigned i = 0, e = MRI->getNumVirtRegs(); i != e; ++i) {
|
2019-08-02 07:27:28 +08:00
|
|
|
unsigned Reg = Register::index2VirtReg(i);
|
2010-10-07 07:54:35 +08:00
|
|
|
|
|
|
|
// Spilling and splitting may leave unused registers around. Skip them.
|
2012-06-21 07:23:59 +08:00
|
|
|
if (MRI->reg_nodbg_empty(Reg))
|
2010-10-07 07:54:35 +08:00
|
|
|
continue;
|
|
|
|
|
2012-06-21 07:23:59 +08:00
|
|
|
if (!LiveInts->hasInterval(Reg)) {
|
|
|
|
report("Missing live interval for virtual register", MF);
|
2017-11-28 20:42:37 +08:00
|
|
|
errs() << printReg(Reg, TRI) << " still has defs or uses\n";
|
2010-10-29 04:44:22 +08:00
|
|
|
continue;
|
2012-06-21 07:23:59 +08:00
|
|
|
}
|
2010-10-29 04:44:22 +08:00
|
|
|
|
2012-06-21 07:23:59 +08:00
|
|
|
const LiveInterval &LI = LiveInts->getInterval(Reg);
|
|
|
|
assert(Reg == LI.reg && "Invalid reg to interval mapping");
|
2012-08-02 08:20:20 +08:00
|
|
|
verifyLiveInterval(LI);
|
|
|
|
}
|
2012-08-03 00:36:50 +08:00
|
|
|
|
|
|
|
// Verify all the cached regunit intervals.
|
|
|
|
for (unsigned i = 0, e = TRI->getNumRegUnits(); i != e; ++i)
|
2013-10-11 05:29:02 +08:00
|
|
|
if (const LiveRange *LR = LiveInts->getCachedRegUnit(i))
|
|
|
|
verifyLiveRange(*LR, i);
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
2010-08-07 02:04:19 +08:00
|
|
|
|
2013-10-11 05:28:54 +08:00
|
|
|
void MachineVerifier::verifyLiveRangeValue(const LiveRange &LR,
|
2014-12-10 09:12:10 +08:00
|
|
|
const VNInfo *VNI, unsigned Reg,
|
2015-09-26 05:51:14 +08:00
|
|
|
LaneBitmask LaneMask) {
|
2012-08-02 08:20:20 +08:00
|
|
|
if (VNI->isUnused())
|
|
|
|
return;
|
2010-08-07 02:04:19 +08:00
|
|
|
|
2013-10-11 05:28:54 +08:00
|
|
|
const VNInfo *DefVNI = LR.getVNInfoAt(VNI->def);
|
2010-08-07 02:04:19 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
if (!DefVNI) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Value not live at VNInfo def and not marked unused", MF);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(*VNI);
|
2012-08-02 08:20:20 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-08-07 02:04:19 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
if (DefVNI != VNI) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Live segment at def has different VNInfo", MF);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(*VNI);
|
2012-08-02 08:20:20 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-08-07 02:04:19 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
const MachineBasicBlock *MBB = LiveInts->getMBBFromIndex(VNI->def);
|
|
|
|
if (!MBB) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Invalid VNInfo definition index", MF);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(*VNI);
|
2012-08-02 08:20:20 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-10-23 06:48:58 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
if (VNI->isPHIDef()) {
|
|
|
|
if (VNI->def != LiveInts->getMBBStartIdx(MBB)) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("PHIDef VNInfo is not defined at MBB start", MBB);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(*VNI);
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
2010-12-20 11:15:20 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
// Non-PHI def.
|
|
|
|
const MachineInstr *MI = LiveInts->getInstructionFromIndex(VNI->def);
|
|
|
|
if (!MI) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("No instruction at VNInfo def index", MBB);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(*VNI);
|
2012-08-02 08:20:20 +08:00
|
|
|
return;
|
|
|
|
}
|
2012-02-28 02:24:30 +08:00
|
|
|
|
2013-10-11 05:28:54 +08:00
|
|
|
if (Reg != 0) {
|
|
|
|
bool hasDef = false;
|
|
|
|
bool isEarlyClobber = false;
|
2016-02-28 01:05:33 +08:00
|
|
|
for (ConstMIBundleOperands MOI(*MI); MOI.isValid(); ++MOI) {
|
2013-10-11 05:28:54 +08:00
|
|
|
if (!MOI->isReg() || !MOI->isDef())
|
2012-08-02 08:20:20 +08:00
|
|
|
continue;
|
2019-08-02 07:27:28 +08:00
|
|
|
if (Register::isVirtualRegister(Reg)) {
|
2013-10-11 05:28:54 +08:00
|
|
|
if (MOI->getReg() != Reg)
|
|
|
|
continue;
|
|
|
|
} else {
|
2019-08-02 07:27:28 +08:00
|
|
|
if (!Register::isPhysicalRegister(MOI->getReg()) ||
|
2013-10-11 05:28:54 +08:00
|
|
|
!TRI->hasRegUnit(MOI->getReg(), Reg))
|
|
|
|
continue;
|
|
|
|
}
|
2016-12-17 03:11:56 +08:00
|
|
|
if (LaneMask.any() &&
|
2016-12-15 22:36:06 +08:00
|
|
|
(TRI->getSubRegIndexLaneMask(MOI->getSubReg()) & LaneMask).none())
|
2014-12-10 09:12:10 +08:00
|
|
|
continue;
|
2013-10-11 05:28:54 +08:00
|
|
|
hasDef = true;
|
|
|
|
if (MOI->isEarlyClobber())
|
|
|
|
isEarlyClobber = true;
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
2010-12-20 11:15:20 +08:00
|
|
|
|
2013-10-11 05:28:54 +08:00
|
|
|
if (!hasDef) {
|
|
|
|
report("Defining instruction does not modify register", MI);
|
2015-11-10 07:59:33 +08:00
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(*VNI);
|
2013-10-11 05:28:54 +08:00
|
|
|
}
|
2012-08-02 08:20:20 +08:00
|
|
|
|
2013-10-11 05:28:54 +08:00
|
|
|
// Early clobber defs begin at USE slots, but other defs must begin at
|
|
|
|
// DEF slots.
|
|
|
|
if (isEarlyClobber) {
|
|
|
|
if (!VNI->def.isEarlyClobber()) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Early clobber def must be at an early-clobber slot", MBB);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(*VNI);
|
2013-10-11 05:28:54 +08:00
|
|
|
}
|
|
|
|
} else if (!VNI->def.isRegister()) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Non-PHI, non-early clobber def must be at a register slot", MBB);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(*VNI);
|
2010-08-07 02:04:19 +08:00
|
|
|
}
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
|
|
|
}
|
2010-08-07 02:04:19 +08:00
|
|
|
|
2013-10-11 05:28:54 +08:00
|
|
|
void MachineVerifier::verifyLiveRangeSegment(const LiveRange &LR,
|
|
|
|
const LiveRange::const_iterator I,
|
2015-09-26 05:51:14 +08:00
|
|
|
unsigned Reg, LaneBitmask LaneMask)
|
|
|
|
{
|
2013-10-11 05:28:54 +08:00
|
|
|
const LiveRange::Segment &S = *I;
|
|
|
|
const VNInfo *VNI = S.valno;
|
2013-10-11 05:28:43 +08:00
|
|
|
assert(VNI && "Live segment has no valno");
|
2012-08-02 08:20:20 +08:00
|
|
|
|
2013-10-11 05:28:54 +08:00
|
|
|
if (VNI->id >= LR.getNumValNums() || VNI != LR.getValNumInfo(VNI->id)) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Foreign valno in live segment", MF);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
|
|
|
report_context(*VNI);
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
2010-08-07 02:04:19 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
if (VNI->isUnused()) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Live segment valno is marked unused", MF);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
2010-08-07 02:04:19 +08:00
|
|
|
|
2013-10-11 05:28:54 +08:00
|
|
|
const MachineBasicBlock *MBB = LiveInts->getMBBFromIndex(S.start);
|
2012-08-02 08:20:20 +08:00
|
|
|
if (!MBB) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Bad start of live segment, no basic block", MF);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
2012-08-02 08:20:20 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
SlotIndex MBBStartIdx = LiveInts->getMBBStartIdx(MBB);
|
2013-10-11 05:28:54 +08:00
|
|
|
if (S.start != MBBStartIdx && S.start != VNI->def) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Live segment must begin at MBB entry or valno def", MBB);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
2010-08-07 02:04:19 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
const MachineBasicBlock *EndMBB =
|
2013-10-11 05:28:54 +08:00
|
|
|
LiveInts->getMBBFromIndex(S.end.getPrevSlot());
|
2012-08-02 08:20:20 +08:00
|
|
|
if (!EndMBB) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Bad end of live segment, no basic block", MF);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
2012-08-02 08:20:20 +08:00
|
|
|
return;
|
|
|
|
}
|
2010-10-23 08:49:09 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
// No more checks for live-out segments.
|
2013-10-11 05:28:54 +08:00
|
|
|
if (S.end == LiveInts->getMBBEndIdx(EndMBB))
|
2012-08-02 08:20:20 +08:00
|
|
|
return;
|
|
|
|
|
2012-08-03 00:36:50 +08:00
|
|
|
// RegUnit intervals are allowed dead phis.
|
2019-08-02 07:27:28 +08:00
|
|
|
if (!Register::isVirtualRegister(Reg) && VNI->isPHIDef() &&
|
2013-10-11 05:28:54 +08:00
|
|
|
S.start == VNI->def && S.end == VNI->def.getDeadSlot())
|
2012-08-03 00:36:50 +08:00
|
|
|
return;
|
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
// The live segment is ending inside EndMBB
|
|
|
|
const MachineInstr *MI =
|
2013-10-11 05:28:54 +08:00
|
|
|
LiveInts->getInstructionFromIndex(S.end.getPrevSlot());
|
2012-08-02 08:20:20 +08:00
|
|
|
if (!MI) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Live segment doesn't end at a valid instruction", EndMBB);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
2012-08-02 08:20:20 +08:00
|
|
|
return;
|
|
|
|
}
|
2012-02-28 02:24:30 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
// The block slot must refer to a basic block boundary.
|
2013-10-11 05:28:54 +08:00
|
|
|
if (S.end.isBlock()) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Live segment ends at B slot of an instruction", EndMBB);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
2012-02-28 02:24:30 +08:00
|
|
|
|
2013-10-11 05:28:54 +08:00
|
|
|
if (S.end.isDead()) {
|
2012-08-02 08:20:20 +08:00
|
|
|
// Segment ends on the dead slot.
|
|
|
|
// That means there must be a dead def.
|
2013-10-11 05:28:54 +08:00
|
|
|
if (!SlotIndex::isSameInstr(S.start, S.end)) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Live segment ending at dead slot spans instructions", EndMBB);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// A live segment can only end at an early-clobber slot if it is being
|
|
|
|
// redefined by an early-clobber def.
|
2013-10-11 05:28:54 +08:00
|
|
|
if (S.end.isEarlyClobber()) {
|
|
|
|
if (I+1 == LR.end() || (I+1)->start != S.end) {
|
2012-08-02 08:20:20 +08:00
|
|
|
report("Live segment ending at early clobber slot must be "
|
2015-11-10 07:59:33 +08:00
|
|
|
"redefined by an EC def in the same instruction", EndMBB);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// The following checks only apply to virtual registers. Physreg liveness
|
|
|
|
// is too weird to check.
|
2019-08-02 07:27:28 +08:00
|
|
|
if (Register::isVirtualRegister(Reg)) {
|
2013-10-11 05:28:43 +08:00
|
|
|
// A live segment can end with either a redefinition, a kill flag on a
|
2012-08-02 08:20:20 +08:00
|
|
|
// use, or a dead flag on a def.
|
|
|
|
bool hasRead = false;
|
2014-12-10 09:13:11 +08:00
|
|
|
bool hasSubRegDef = false;
|
2016-03-30 03:07:43 +08:00
|
|
|
bool hasDeadDef = false;
|
2016-02-28 01:05:33 +08:00
|
|
|
for (ConstMIBundleOperands MOI(*MI); MOI.isValid(); ++MOI) {
|
2013-10-11 05:28:54 +08:00
|
|
|
if (!MOI->isReg() || MOI->getReg() != Reg)
|
2012-02-28 02:24:30 +08:00
|
|
|
continue;
|
2016-08-24 21:37:55 +08:00
|
|
|
unsigned Sub = MOI->getSubReg();
|
2016-12-15 22:36:06 +08:00
|
|
|
LaneBitmask SLM = Sub != 0 ? TRI->getSubRegIndexLaneMask(Sub)
|
|
|
|
: LaneBitmask::getAll();
|
2016-03-30 03:07:43 +08:00
|
|
|
if (MOI->isDef()) {
|
2016-08-24 21:37:55 +08:00
|
|
|
if (Sub != 0) {
|
2016-03-30 03:07:43 +08:00
|
|
|
hasSubRegDef = true;
|
2017-12-07 18:40:31 +08:00
|
|
|
// An operand %0:sub0 reads %0:sub1..n. Invert the lane
|
2016-08-24 21:37:55 +08:00
|
|
|
// mask for subregister defs. Read-undef defs will be handled by
|
|
|
|
// readsReg below.
|
2016-08-29 21:15:35 +08:00
|
|
|
SLM = ~SLM;
|
2016-08-24 21:37:55 +08:00
|
|
|
}
|
2016-03-30 03:07:43 +08:00
|
|
|
if (MOI->isDead())
|
|
|
|
hasDeadDef = true;
|
|
|
|
}
|
2016-12-17 03:11:56 +08:00
|
|
|
if (LaneMask.any() && (LaneMask & SLM).none())
|
2016-08-24 21:37:55 +08:00
|
|
|
continue;
|
2012-08-02 08:20:20 +08:00
|
|
|
if (MOI->readsReg())
|
|
|
|
hasRead = true;
|
|
|
|
}
|
2016-03-30 03:07:43 +08:00
|
|
|
if (S.end.isDead()) {
|
|
|
|
// Make sure that the corresponding machine operand for a "dead" live
|
|
|
|
// range has the dead flag. We cannot perform this check for subregister
|
|
|
|
// liveranges as partially dead values are allowed.
|
2016-12-15 22:36:06 +08:00
|
|
|
if (LaneMask.none() && !hasDeadDef) {
|
2016-03-30 03:07:43 +08:00
|
|
|
report("Instruction ending live segment on dead slot has no dead flag",
|
|
|
|
MI);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
|
|
|
}
|
|
|
|
} else {
|
2012-08-02 08:20:20 +08:00
|
|
|
if (!hasRead) {
|
2014-12-10 09:13:11 +08:00
|
|
|
// When tracking subregister liveness, the main range must start new
|
|
|
|
// values on partial register writes, even if there is no read.
|
2016-12-17 03:11:56 +08:00
|
|
|
if (!MRI->shouldTrackSubRegLiveness(Reg) || LaneMask.any() ||
|
2015-03-19 08:21:58 +08:00
|
|
|
!hasSubRegDef) {
|
2014-12-10 09:13:11 +08:00
|
|
|
report("Instruction ending live segment doesn't read the register",
|
|
|
|
MI);
|
2015-11-10 07:59:33 +08:00
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(S);
|
2014-12-10 09:13:11 +08:00
|
|
|
}
|
2012-02-28 02:24:30 +08:00
|
|
|
}
|
2012-08-02 08:20:20 +08:00
|
|
|
}
|
|
|
|
}
|
2012-02-28 02:24:30 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
// Now check all the basic blocks in this live segment.
|
2015-10-10 03:40:45 +08:00
|
|
|
MachineFunction::const_iterator MFI = MBB->getIterator();
|
2013-10-11 05:28:43 +08:00
|
|
|
// Is this live segment the beginning of a non-PHIDef VN?
|
2013-10-11 05:28:54 +08:00
|
|
|
if (S.start == VNI->def && !VNI->isPHIDef()) {
|
2012-08-02 08:20:20 +08:00
|
|
|
// Not live-in to any blocks.
|
|
|
|
if (MBB == EndMBB)
|
|
|
|
return;
|
|
|
|
// Skip this block.
|
|
|
|
++MFI;
|
|
|
|
}
|
2018-08-17 03:13:28 +08:00
|
|
|
|
|
|
|
SmallVector<SlotIndex, 4> Undefs;
|
|
|
|
if (LaneMask.any()) {
|
|
|
|
LiveInterval &OwnerLI = LiveInts->getInterval(Reg);
|
|
|
|
OwnerLI.computeSubRangeUndefs(Undefs, LaneMask, *MRI, *Indexes);
|
|
|
|
}
|
|
|
|
|
2017-09-12 07:00:48 +08:00
|
|
|
while (true) {
|
2015-10-10 03:40:45 +08:00
|
|
|
assert(LiveInts->isLiveInToMBB(LR, &*MFI));
|
2012-08-02 08:20:20 +08:00
|
|
|
// We don't know how to track physregs into a landing pad.
|
2019-08-02 07:27:28 +08:00
|
|
|
if (!Register::isVirtualRegister(Reg) && MFI->isEHPad()) {
|
2012-08-02 08:20:20 +08:00
|
|
|
if (&*MFI == EndMBB)
|
|
|
|
break;
|
|
|
|
++MFI;
|
|
|
|
continue;
|
|
|
|
}
|
2010-10-23 08:49:09 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
// Is VNI a PHI-def in the current block?
|
|
|
|
bool IsPHI = VNI->isPHIDef() &&
|
2015-10-10 03:40:45 +08:00
|
|
|
VNI->def == LiveInts->getMBBStartIdx(&*MFI);
|
2012-08-02 08:20:20 +08:00
|
|
|
|
|
|
|
// Check that VNI is live-out of all predecessors.
|
|
|
|
for (MachineBasicBlock::const_pred_iterator PI = MFI->pred_begin(),
|
|
|
|
PE = MFI->pred_end(); PI != PE; ++PI) {
|
|
|
|
SlotIndex PEnd = LiveInts->getMBBEndIdx(*PI);
|
2013-10-11 05:28:54 +08:00
|
|
|
const VNInfo *PVNI = LR.getVNInfoBefore(PEnd);
|
2012-08-02 08:20:20 +08:00
|
|
|
|
2017-06-09 05:30:54 +08:00
|
|
|
// All predecessors must have a live-out value. However for a phi
|
|
|
|
// instruction with subregister intervals
|
|
|
|
// only one of the subregisters (not necessarily the current one) needs to
|
|
|
|
// be defined.
|
2018-08-17 03:13:28 +08:00
|
|
|
if (!PVNI && (LaneMask.none() || !IsPHI)) {
|
|
|
|
if (LiveRangeCalc::isJointlyDominated(*PI, Undefs, *Indexes))
|
|
|
|
continue;
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Register not marked live out of predecessor", *PI);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
|
|
|
report_context(*VNI);
|
2017-12-05 01:18:51 +08:00
|
|
|
errs() << " live into " << printMBBReference(*MFI) << '@'
|
|
|
|
<< LiveInts->getMBBStartIdx(&*MFI) << ", not live before "
|
2015-10-10 03:40:45 +08:00
|
|
|
<< PEnd << '\n';
|
2012-08-02 08:20:20 +08:00
|
|
|
continue;
|
2010-10-23 08:49:09 +08:00
|
|
|
}
|
2010-12-29 07:45:38 +08:00
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
// Only PHI-defs can take different predecessor values.
|
|
|
|
if (!IsPHI && PVNI != VNI) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Different value live out of predecessor", *PI);
|
|
|
|
report_context(LR, Reg, LaneMask);
|
2017-12-05 01:18:51 +08:00
|
|
|
errs() << "Valno #" << PVNI->id << " live out of "
|
|
|
|
<< printMBBReference(*(*PI)) << '@' << PEnd << "\nValno #"
|
|
|
|
<< VNI->id << " live into " << printMBBReference(*MFI) << '@'
|
2015-10-10 03:40:45 +08:00
|
|
|
<< LiveInts->getMBBStartIdx(&*MFI) << '\n';
|
2010-10-23 08:49:09 +08:00
|
|
|
}
|
2010-08-07 02:04:19 +08:00
|
|
|
}
|
2012-08-02 08:20:20 +08:00
|
|
|
if (&*MFI == EndMBB)
|
|
|
|
break;
|
|
|
|
++MFI;
|
|
|
|
}
|
|
|
|
}
|
2010-10-27 06:36:07 +08:00
|
|
|
|
2014-12-10 09:12:10 +08:00
|
|
|
void MachineVerifier::verifyLiveRange(const LiveRange &LR, unsigned Reg,
|
2015-09-26 05:51:14 +08:00
|
|
|
LaneBitmask LaneMask) {
|
2014-12-11 07:07:54 +08:00
|
|
|
for (const VNInfo *VNI : LR.valnos)
|
|
|
|
verifyLiveRangeValue(LR, VNI, Reg, LaneMask);
|
2012-08-02 08:20:20 +08:00
|
|
|
|
2013-10-11 05:28:54 +08:00
|
|
|
for (LiveRange::const_iterator I = LR.begin(), E = LR.end(); I != E; ++I)
|
2014-12-10 09:12:10 +08:00
|
|
|
verifyLiveRangeSegment(LR, I, Reg, LaneMask);
|
2013-10-11 05:28:54 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void MachineVerifier::verifyLiveInterval(const LiveInterval &LI) {
|
2014-12-10 09:12:10 +08:00
|
|
|
unsigned Reg = LI.reg;
|
2019-08-02 07:27:28 +08:00
|
|
|
assert(Register::isVirtualRegister(Reg));
|
2015-03-26 05:18:22 +08:00
|
|
|
verifyLiveRange(LI, Reg);
|
|
|
|
|
2016-12-15 22:36:06 +08:00
|
|
|
LaneBitmask Mask;
|
2015-09-26 05:51:14 +08:00
|
|
|
LaneBitmask MaxMask = MRI->getMaxLaneMaskForVReg(Reg);
|
2015-03-26 05:18:22 +08:00
|
|
|
for (const LiveInterval::SubRange &SR : LI.subranges()) {
|
2016-12-17 03:11:56 +08:00
|
|
|
if ((Mask & SR.LaneMask).any()) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Lane masks of sub ranges overlap in live interval", MF);
|
|
|
|
report_context(LI);
|
|
|
|
}
|
2016-12-17 03:11:56 +08:00
|
|
|
if ((SR.LaneMask & ~MaxMask).any()) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Subrange lanemask is invalid", MF);
|
|
|
|
report_context(LI);
|
|
|
|
}
|
|
|
|
if (SR.empty()) {
|
|
|
|
report("Subrange must not be empty", MF);
|
|
|
|
report_context(SR, LI.reg, SR.LaneMask);
|
|
|
|
}
|
2015-03-26 05:18:22 +08:00
|
|
|
Mask |= SR.LaneMask;
|
|
|
|
verifyLiveRange(SR, LI.reg, SR.LaneMask);
|
2015-11-10 07:59:33 +08:00
|
|
|
if (!LI.covers(SR)) {
|
|
|
|
report("A Subrange is not covered by the main range", MF);
|
|
|
|
report_context(LI);
|
|
|
|
}
|
2014-12-10 09:12:10 +08:00
|
|
|
}
|
|
|
|
|
2012-08-02 08:20:20 +08:00
|
|
|
// Check the LI only has one connected component.
|
2015-03-26 05:18:22 +08:00
|
|
|
ConnectedVNInfoEqClasses ConEQ(*LiveInts);
|
2016-01-08 09:16:35 +08:00
|
|
|
unsigned NumComp = ConEQ.Classify(LI);
|
2015-03-26 05:18:22 +08:00
|
|
|
if (NumComp > 1) {
|
2015-11-10 07:59:33 +08:00
|
|
|
report("Multiple connected components in live interval", MF);
|
|
|
|
report_context(LI);
|
2015-03-26 05:18:22 +08:00
|
|
|
for (unsigned comp = 0; comp != NumComp; ++comp) {
|
|
|
|
errs() << comp << ": valnos";
|
|
|
|
for (LiveInterval::const_vni_iterator I = LI.vni_begin(),
|
|
|
|
E = LI.vni_end(); I!=E; ++I)
|
|
|
|
if (comp == ConEQ.getEqClass(*I))
|
|
|
|
errs() << ' ' << (*I)->id;
|
|
|
|
errs() << '\n';
|
2010-10-27 06:36:07 +08:00
|
|
|
}
|
2010-08-07 02:04:19 +08:00
|
|
|
}
|
|
|
|
}
|
2013-07-16 05:26:31 +08:00
|
|
|
|
|
|
|
namespace {
|
2017-09-12 07:00:48 +08:00
|
|
|
|
2013-07-16 05:26:31 +08:00
|
|
|
// FrameSetup and FrameDestroy can have zero adjustment, so using a single
|
|
|
|
// integer, we can't tell whether it is a FrameSetup or FrameDestroy if the
|
|
|
|
// value is zero.
|
|
|
|
// We use a bool plus an integer to capture the stack state.
|
|
|
|
struct StackStateOfBB {
|
2017-09-12 07:00:48 +08:00
|
|
|
StackStateOfBB() = default;
|
2013-07-16 05:26:31 +08:00
|
|
|
StackStateOfBB(int EntryVal, int ExitVal, bool EntrySetup, bool ExitSetup) :
|
|
|
|
EntryValue(EntryVal), ExitValue(ExitVal), EntryIsSetup(EntrySetup),
|
2017-09-12 07:00:48 +08:00
|
|
|
ExitIsSetup(ExitSetup) {}
|
|
|
|
|
2013-07-16 05:26:31 +08:00
|
|
|
// Can be negative, which means we are setting up a frame.
|
2017-09-12 07:00:48 +08:00
|
|
|
int EntryValue = 0;
|
|
|
|
int ExitValue = 0;
|
|
|
|
bool EntryIsSetup = false;
|
|
|
|
bool ExitIsSetup = false;
|
2013-07-16 05:26:31 +08:00
|
|
|
};
|
2017-09-12 07:00:48 +08:00
|
|
|
|
|
|
|
} // end anonymous namespace
|
2013-07-16 05:26:31 +08:00
|
|
|
|
|
|
|
/// Make sure on every path through the CFG, a FrameSetup <n> is always followed
|
|
|
|
/// by a FrameDestroy <n>, stack adjustments are identical on all
|
|
|
|
/// CFG edges to a merge point, and frame is destroyed at end of a return block.
|
|
|
|
void MachineVerifier::verifyStackFrame() {
|
2015-05-19 04:27:55 +08:00
|
|
|
unsigned FrameSetupOpcode = TII->getCallFrameSetupOpcode();
|
|
|
|
unsigned FrameDestroyOpcode = TII->getCallFrameDestroyOpcode();
|
2017-04-20 09:34:04 +08:00
|
|
|
if (FrameSetupOpcode == ~0u && FrameDestroyOpcode == ~0u)
|
|
|
|
return;
|
2013-07-16 05:26:31 +08:00
|
|
|
|
|
|
|
SmallVector<StackStateOfBB, 8> SPState;
|
|
|
|
SPState.resize(MF->getNumBlockIDs());
|
2016-10-06 05:36:16 +08:00
|
|
|
df_iterator_default_set<const MachineBasicBlock*> Reachable;
|
2013-07-16 05:26:31 +08:00
|
|
|
|
|
|
|
// Visit the MBBs in DFS order.
|
2017-09-12 07:00:48 +08:00
|
|
|
for (df_ext_iterator<const MachineFunction *,
|
|
|
|
df_iterator_default_set<const MachineBasicBlock *>>
|
2013-07-16 05:26:31 +08:00
|
|
|
DFI = df_ext_begin(MF, Reachable), DFE = df_ext_end(MF, Reachable);
|
|
|
|
DFI != DFE; ++DFI) {
|
|
|
|
const MachineBasicBlock *MBB = *DFI;
|
|
|
|
|
|
|
|
StackStateOfBB BBState;
|
|
|
|
// Check the exit state of the DFS stack predecessor.
|
|
|
|
if (DFI.getPathLength() >= 2) {
|
|
|
|
const MachineBasicBlock *StackPred = DFI.getPath(DFI.getPathLength() - 2);
|
|
|
|
assert(Reachable.count(StackPred) &&
|
|
|
|
"DFS stack predecessor is already visited.\n");
|
|
|
|
BBState.EntryValue = SPState[StackPred->getNumber()].ExitValue;
|
|
|
|
BBState.EntryIsSetup = SPState[StackPred->getNumber()].ExitIsSetup;
|
|
|
|
BBState.ExitValue = BBState.EntryValue;
|
|
|
|
BBState.ExitIsSetup = BBState.EntryIsSetup;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update stack state by checking contents of MBB.
|
2014-05-01 06:17:38 +08:00
|
|
|
for (const auto &I : *MBB) {
|
|
|
|
if (I.getOpcode() == FrameSetupOpcode) {
|
2013-07-16 05:26:31 +08:00
|
|
|
if (BBState.ExitIsSetup)
|
2014-05-01 06:17:38 +08:00
|
|
|
report("FrameSetup is after another FrameSetup", &I);
|
2017-05-09 21:35:13 +08:00
|
|
|
BBState.ExitValue -= TII->getFrameTotalSize(I);
|
2013-07-16 05:26:31 +08:00
|
|
|
BBState.ExitIsSetup = true;
|
|
|
|
}
|
|
|
|
|
2014-05-01 06:17:38 +08:00
|
|
|
if (I.getOpcode() == FrameDestroyOpcode) {
|
2017-05-09 21:35:13 +08:00
|
|
|
int Size = TII->getFrameTotalSize(I);
|
2013-07-16 05:26:31 +08:00
|
|
|
if (!BBState.ExitIsSetup)
|
2014-05-01 06:17:38 +08:00
|
|
|
report("FrameDestroy is not after a FrameSetup", &I);
|
2013-07-16 05:26:31 +08:00
|
|
|
int AbsSPAdj = BBState.ExitValue < 0 ? -BBState.ExitValue :
|
|
|
|
BBState.ExitValue;
|
|
|
|
if (BBState.ExitIsSetup && AbsSPAdj != Size) {
|
2014-05-01 06:17:38 +08:00
|
|
|
report("FrameDestroy <n> is after FrameSetup <m>", &I);
|
2015-02-04 08:02:59 +08:00
|
|
|
errs() << "FrameDestroy <" << Size << "> is after FrameSetup <"
|
2013-07-16 05:26:31 +08:00
|
|
|
<< AbsSPAdj << ">.\n";
|
|
|
|
}
|
|
|
|
BBState.ExitValue += Size;
|
|
|
|
BBState.ExitIsSetup = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
SPState[MBB->getNumber()] = BBState;
|
|
|
|
|
|
|
|
// Make sure the exit state of any predecessor is consistent with the entry
|
|
|
|
// state.
|
|
|
|
for (MachineBasicBlock::const_pred_iterator I = MBB->pred_begin(),
|
|
|
|
E = MBB->pred_end(); I != E; ++I) {
|
|
|
|
if (Reachable.count(*I) &&
|
|
|
|
(SPState[(*I)->getNumber()].ExitValue != BBState.EntryValue ||
|
|
|
|
SPState[(*I)->getNumber()].ExitIsSetup != BBState.EntryIsSetup)) {
|
|
|
|
report("The exit stack state of a predecessor is inconsistent.", MBB);
|
2017-12-05 01:18:51 +08:00
|
|
|
errs() << "Predecessor " << printMBBReference(*(*I))
|
|
|
|
<< " has exit state (" << SPState[(*I)->getNumber()].ExitValue
|
|
|
|
<< ", " << SPState[(*I)->getNumber()].ExitIsSetup << "), while "
|
|
|
|
<< printMBBReference(*MBB) << " has entry state ("
|
|
|
|
<< BBState.EntryValue << ", " << BBState.EntryIsSetup << ").\n";
|
2013-07-16 05:26:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure the entry state of any successor is consistent with the exit
|
|
|
|
// state.
|
|
|
|
for (MachineBasicBlock::const_succ_iterator I = MBB->succ_begin(),
|
|
|
|
E = MBB->succ_end(); I != E; ++I) {
|
|
|
|
if (Reachable.count(*I) &&
|
|
|
|
(SPState[(*I)->getNumber()].EntryValue != BBState.ExitValue ||
|
|
|
|
SPState[(*I)->getNumber()].EntryIsSetup != BBState.ExitIsSetup)) {
|
|
|
|
report("The entry stack state of a successor is inconsistent.", MBB);
|
2017-12-05 01:18:51 +08:00
|
|
|
errs() << "Successor " << printMBBReference(*(*I))
|
|
|
|
<< " has entry state (" << SPState[(*I)->getNumber()].EntryValue
|
|
|
|
<< ", " << SPState[(*I)->getNumber()].EntryIsSetup << "), while "
|
|
|
|
<< printMBBReference(*MBB) << " has exit state ("
|
|
|
|
<< BBState.ExitValue << ", " << BBState.ExitIsSetup << ").\n";
|
2013-07-16 05:26:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure a basic block with return ends with zero stack adjustment.
|
|
|
|
if (!MBB->empty() && MBB->back().isReturn()) {
|
|
|
|
if (BBState.ExitIsSetup)
|
|
|
|
report("A return block ends with a FrameSetup.", MBB);
|
|
|
|
if (BBState.ExitValue)
|
|
|
|
report("A return block ends with a nonzero stack adjustment.", MBB);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|