forked from OSchip/llvm-project
Introduce [post]dominator tree and related infrastructure, use it in CFG func
verifier. We get most of this infrastructure directly from LLVM, we just need to adapt it to our CFG abstraction. This has a few unrelated changes engangled in it: - getFunction() in various classes was const incorrect, fix it. - This moves Verifier.cpp to the analysis library, since Verifier depends on dominance and these are both really analyses. - IndexedAccessorIterator::reference was defined wrong, leading to really exciting template errors that were fun to diagnose. - This flips the boolean sense of the foldOperation() function in constant folding pass in response to previous patch feedback. PiperOrigin-RevId: 214046593
This commit is contained in:
parent
948dea045b
commit
d6f8ec7bac
|
@ -0,0 +1,191 @@
|
||||||
|
//===- Dominance.h - Dominator analysis for CFG Functions -------*- C++ -*-===//
|
||||||
|
//
|
||||||
|
// Copyright 2019 The MLIR Authors.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
// =============================================================================
|
||||||
|
|
||||||
|
#ifndef MLIR_ANALYSIS_DOMINANCE_H
|
||||||
|
#define MLIR_ANALYSIS_DOMINANCE_H
|
||||||
|
|
||||||
|
#include "mlir/IR/CFGFunction.h"
|
||||||
|
#include "llvm/Support/GenericDomTree.h"
|
||||||
|
|
||||||
|
namespace llvm {
|
||||||
|
template <> struct GraphTraits<mlir::BasicBlock *> {
|
||||||
|
using ChildIteratorType = mlir::BasicBlock::succ_iterator;
|
||||||
|
using Node = mlir::BasicBlock;
|
||||||
|
using NodeRef = Node *;
|
||||||
|
|
||||||
|
static NodeRef getEntryNode(NodeRef bb) { return bb; }
|
||||||
|
|
||||||
|
static ChildIteratorType child_begin(NodeRef node) {
|
||||||
|
return node->succ_begin();
|
||||||
|
}
|
||||||
|
static ChildIteratorType child_end(NodeRef node) { return node->succ_end(); }
|
||||||
|
};
|
||||||
|
|
||||||
|
template <> struct GraphTraits<const mlir::BasicBlock *> {
|
||||||
|
using ChildIteratorType = mlir::BasicBlock::const_succ_iterator;
|
||||||
|
using Node = const mlir::BasicBlock;
|
||||||
|
using NodeRef = Node *;
|
||||||
|
|
||||||
|
static NodeRef getEntryNode(NodeRef bb) { return bb; }
|
||||||
|
|
||||||
|
static ChildIteratorType child_begin(NodeRef node) {
|
||||||
|
return node->succ_begin();
|
||||||
|
}
|
||||||
|
static ChildIteratorType child_end(NodeRef node) { return node->succ_end(); }
|
||||||
|
};
|
||||||
|
|
||||||
|
template <> struct GraphTraits<Inverse<mlir::BasicBlock *>> {
|
||||||
|
using ChildIteratorType = mlir::BasicBlock::pred_iterator;
|
||||||
|
using Node = mlir::BasicBlock;
|
||||||
|
using NodeRef = Node *;
|
||||||
|
static NodeRef getEntryNode(Inverse<mlir::BasicBlock *> inverseGraph) {
|
||||||
|
return inverseGraph.Graph;
|
||||||
|
}
|
||||||
|
static inline ChildIteratorType child_begin(NodeRef node) {
|
||||||
|
return node->pred_begin();
|
||||||
|
}
|
||||||
|
static inline ChildIteratorType child_end(NodeRef node) {
|
||||||
|
return node->pred_end();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <> struct GraphTraits<Inverse<const mlir::BasicBlock *>> {
|
||||||
|
using ChildIteratorType = mlir::BasicBlock::const_pred_iterator;
|
||||||
|
using Node = const mlir::BasicBlock;
|
||||||
|
using NodeRef = Node *;
|
||||||
|
|
||||||
|
static NodeRef getEntryNode(Inverse<const mlir::BasicBlock *> inverseGraph) {
|
||||||
|
return inverseGraph.Graph;
|
||||||
|
}
|
||||||
|
static inline ChildIteratorType child_begin(NodeRef node) {
|
||||||
|
return node->pred_begin();
|
||||||
|
}
|
||||||
|
static inline ChildIteratorType child_end(NodeRef node) {
|
||||||
|
return node->pred_end();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <>
|
||||||
|
struct GraphTraits<mlir::CFGFunction *>
|
||||||
|
: public GraphTraits<mlir::BasicBlock *> {
|
||||||
|
using GraphType = mlir::CFGFunction *;
|
||||||
|
using NodeRef = mlir::BasicBlock *;
|
||||||
|
|
||||||
|
static NodeRef getEntryNode(GraphType fn) { return &fn->front(); }
|
||||||
|
|
||||||
|
using nodes_iterator = pointer_iterator<mlir::CFGFunction::iterator>;
|
||||||
|
static nodes_iterator nodes_begin(GraphType fn) {
|
||||||
|
return nodes_iterator(fn->begin());
|
||||||
|
}
|
||||||
|
static nodes_iterator nodes_end(GraphType fn) {
|
||||||
|
return nodes_iterator(fn->end());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
template <>
|
||||||
|
struct GraphTraits<Inverse<mlir::CFGFunction *>>
|
||||||
|
: public GraphTraits<Inverse<mlir::BasicBlock *>> {
|
||||||
|
using GraphType = Inverse<mlir::CFGFunction *>;
|
||||||
|
using NodeRef = NodeRef;
|
||||||
|
|
||||||
|
static NodeRef getEntryNode(GraphType fn) { return &fn.Graph->front(); }
|
||||||
|
|
||||||
|
using nodes_iterator = pointer_iterator<mlir::CFGFunction::iterator>;
|
||||||
|
static nodes_iterator nodes_begin(GraphType fn) {
|
||||||
|
return nodes_iterator(fn.Graph->begin());
|
||||||
|
}
|
||||||
|
static nodes_iterator nodes_end(GraphType fn) {
|
||||||
|
return nodes_iterator(fn.Graph->end());
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} // namespace llvm
|
||||||
|
|
||||||
|
extern template class llvm::DominatorTreeBase<mlir::BasicBlock, false>;
|
||||||
|
extern template class llvm::DominatorTreeBase<mlir::BasicBlock, true>;
|
||||||
|
extern template class llvm::DomTreeNodeBase<mlir::BasicBlock>;
|
||||||
|
|
||||||
|
namespace llvm {
|
||||||
|
namespace DomTreeBuilder {
|
||||||
|
|
||||||
|
using MLIRDomTree = llvm::DomTreeBase<mlir::BasicBlock>;
|
||||||
|
using MLIRPostDomTree = llvm::PostDomTreeBase<mlir::BasicBlock>;
|
||||||
|
|
||||||
|
// extern template void Calculate<MLIRDomTree>(MLIRDomTree &DT);
|
||||||
|
// extern template void Calculate<MLIRPostDomTree>(MLIRPostDomTree &DT);
|
||||||
|
|
||||||
|
} // namespace DomTreeBuilder
|
||||||
|
} // namespace llvm
|
||||||
|
|
||||||
|
namespace mlir {
|
||||||
|
using DominatorTreeBase = llvm::DominatorTreeBase<BasicBlock, false>;
|
||||||
|
using PostDominatorTreeBase = llvm::DominatorTreeBase<BasicBlock, true>;
|
||||||
|
using DominanceInfoNode = llvm::DomTreeNodeBase<BasicBlock>;
|
||||||
|
|
||||||
|
/// A class for computing basic dominance information.
|
||||||
|
class DominanceInfo : public DominatorTreeBase {
|
||||||
|
using super = DominatorTreeBase;
|
||||||
|
|
||||||
|
public:
|
||||||
|
DominanceInfo(CFGFunction *F);
|
||||||
|
|
||||||
|
/// Return true if instruction A properly dominates instruction B.
|
||||||
|
bool properlyDominates(const Instruction *a, const Instruction *b);
|
||||||
|
|
||||||
|
/// Return true if instruction A dominates instruction B.
|
||||||
|
bool dominates(const Instruction *a, const Instruction *b) {
|
||||||
|
return a == b || properlyDominates(a, b);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return true if value A properly dominates instruction B.
|
||||||
|
bool properlyDominates(const SSAValue *a, const Instruction *b);
|
||||||
|
|
||||||
|
/// Return true if instruction A dominates instruction B.
|
||||||
|
bool dominates(const SSAValue *a, const Instruction *b) {
|
||||||
|
return a->getDefiningInst() == b || properlyDominates(a, b);
|
||||||
|
}
|
||||||
|
|
||||||
|
// dominates/properlyDominates for basic blocks.
|
||||||
|
using DominatorTreeBase::dominates;
|
||||||
|
using DominatorTreeBase::properlyDominates;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // end namespace mlir
|
||||||
|
|
||||||
|
namespace llvm {
|
||||||
|
|
||||||
|
/// DominatorTree GraphTraits specialization so the DominatorTree can be
|
||||||
|
/// iterated by generic graph iterators.
|
||||||
|
template <> struct GraphTraits<mlir::DominanceInfoNode *> {
|
||||||
|
using ChildIteratorType = mlir::DominanceInfoNode::iterator;
|
||||||
|
using NodeRef = mlir::DominanceInfoNode *;
|
||||||
|
|
||||||
|
static NodeRef getEntryNode(NodeRef N) { return N; }
|
||||||
|
static inline ChildIteratorType child_begin(NodeRef N) { return N->begin(); }
|
||||||
|
static inline ChildIteratorType child_end(NodeRef N) { return N->end(); }
|
||||||
|
};
|
||||||
|
|
||||||
|
template <> struct GraphTraits<const mlir::DominanceInfoNode *> {
|
||||||
|
using ChildIteratorType = mlir::DominanceInfoNode::const_iterator;
|
||||||
|
using NodeRef = const mlir::DominanceInfoNode *;
|
||||||
|
|
||||||
|
static NodeRef getEntryNode(NodeRef N) { return N; }
|
||||||
|
static inline ChildIteratorType child_begin(NodeRef N) { return N->begin(); }
|
||||||
|
static inline ChildIteratorType child_end(NodeRef N) { return N->end(); }
|
||||||
|
};
|
||||||
|
|
||||||
|
} // end namespace llvm
|
||||||
|
#endif
|
|
@ -39,9 +39,12 @@ public:
|
||||||
~BasicBlock();
|
~BasicBlock();
|
||||||
|
|
||||||
/// Return the function that a BasicBlock is part of.
|
/// Return the function that a BasicBlock is part of.
|
||||||
CFGFunction *getFunction() const {
|
CFGFunction *getFunction() { return function; }
|
||||||
return function;
|
const CFGFunction *getFunction() const { return function; }
|
||||||
}
|
|
||||||
|
/// Return the function that a BasicBlock is part of.
|
||||||
|
const CFGFunction *getParent() const { return function; }
|
||||||
|
CFGFunction *getParent() { return function; }
|
||||||
|
|
||||||
//===--------------------------------------------------------------------===//
|
//===--------------------------------------------------------------------===//
|
||||||
// Block arguments management
|
// Block arguments management
|
||||||
|
@ -194,6 +197,11 @@ public:
|
||||||
void print(raw_ostream &os) const;
|
void print(raw_ostream &os) const;
|
||||||
void dump() const;
|
void dump() const;
|
||||||
|
|
||||||
|
/// Print out the name of the basic block without printing its body.
|
||||||
|
/// NOTE: The printType argument is ignored. We keep it for compatibility
|
||||||
|
/// with LLVM dominator machinery that expects it to exist.
|
||||||
|
void printAsOperand(raw_ostream &os, bool printType = true);
|
||||||
|
|
||||||
/// getSublistAccess() - Returns pointer to member of operation list
|
/// getSublistAccess() - Returns pointer to member of operation list
|
||||||
static OperationListType BasicBlock::*getSublistAccess(OperationInst*) {
|
static OperationListType BasicBlock::*getSublistAccess(OperationInst*) {
|
||||||
return &BasicBlock::operations;
|
return &BasicBlock::operations;
|
||||||
|
@ -300,6 +308,9 @@ public:
|
||||||
: IndexedAccessorIterator<SuccessorIterator<BlockType>, BlockType,
|
: IndexedAccessorIterator<SuccessorIterator<BlockType>, BlockType,
|
||||||
BlockType>(object, index) {}
|
BlockType>(object, index) {}
|
||||||
|
|
||||||
|
SuccessorIterator(const SuccessorIterator &other)
|
||||||
|
: SuccessorIterator(other.object, other.index) {}
|
||||||
|
|
||||||
/// Support converting to the const variant. This will be a no-op for const
|
/// Support converting to the const variant. This will be a no-op for const
|
||||||
/// variant.
|
/// variant.
|
||||||
operator SuccessorIterator<const BlockType>() const {
|
operator SuccessorIterator<const BlockType>() const {
|
||||||
|
|
|
@ -77,7 +77,10 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Return the function that this argument is defined in.
|
/// Return the function that this argument is defined in.
|
||||||
CFGFunction *getFunction() const;
|
CFGFunction *getFunction();
|
||||||
|
const CFGFunction *getFunction() const {
|
||||||
|
return const_cast<BBArgument *>(this)->getFunction();
|
||||||
|
}
|
||||||
|
|
||||||
BasicBlock *getOwner() { return owner; }
|
BasicBlock *getOwner() { return owner; }
|
||||||
const BasicBlock *getOwner() const { return owner; }
|
const BasicBlock *getOwner() const { return owner; }
|
||||||
|
|
|
@ -287,7 +287,8 @@ private:
|
||||||
template <typename ConcreteType, typename ObjectType, typename ElementType>
|
template <typename ConcreteType, typename ObjectType, typename ElementType>
|
||||||
class IndexedAccessorIterator
|
class IndexedAccessorIterator
|
||||||
: public llvm::iterator_facade_base<
|
: public llvm::iterator_facade_base<
|
||||||
ConcreteType, std::random_access_iterator_tag, ElementType *> {
|
ConcreteType, std::random_access_iterator_tag, ElementType *,
|
||||||
|
std::ptrdiff_t, ElementType *, ElementType *> {
|
||||||
public:
|
public:
|
||||||
ptrdiff_t operator-(const IndexedAccessorIterator &rhs) const {
|
ptrdiff_t operator-(const IndexedAccessorIterator &rhs) const {
|
||||||
assert(object == rhs.object && "incompatible iterators");
|
assert(object == rhs.object && "incompatible iterators");
|
||||||
|
|
|
@ -0,0 +1,81 @@
|
||||||
|
//===- Dominance.cpp - Dominator analysis for CFG Functions ---------------===//
|
||||||
|
//
|
||||||
|
// Copyright 2019 The MLIR Authors.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
// =============================================================================
|
||||||
|
//
|
||||||
|
// Implementation of dominance related classes and instantiations of extern
|
||||||
|
// templates.
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
|
#include "mlir/Analysis/Dominance.h"
|
||||||
|
#include "llvm/Support/GenericDomTreeConstruction.h"
|
||||||
|
using namespace mlir;
|
||||||
|
|
||||||
|
template class llvm::DominatorTreeBase<BasicBlock, false>;
|
||||||
|
template class llvm::DominatorTreeBase<BasicBlock, true>;
|
||||||
|
template class llvm::DomTreeNodeBase<BasicBlock>;
|
||||||
|
|
||||||
|
/// Compute the immediate-dominators map.
|
||||||
|
DominanceInfo::DominanceInfo(CFGFunction *function) : DominatorTreeBase() {
|
||||||
|
// Build the dominator tree for the function.
|
||||||
|
recalculate(*function);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return true if instruction A properly dominates instruction B.
|
||||||
|
bool DominanceInfo::properlyDominates(const Instruction *a,
|
||||||
|
const Instruction *b) {
|
||||||
|
auto *aBlock = a->getBlock(), *bBlock = b->getBlock();
|
||||||
|
|
||||||
|
// If the blocks are different, it's as easy as whether A's block
|
||||||
|
// dominates B's block.
|
||||||
|
if (aBlock != bBlock)
|
||||||
|
return properlyDominates(a->getBlock(), b->getBlock());
|
||||||
|
|
||||||
|
// If a/b are the same, then they don't properly dominate each other.
|
||||||
|
if (a == b)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// If one is a terminator, then the other dominates it.
|
||||||
|
auto *aOp = dyn_cast<OperationInst>(a);
|
||||||
|
if (!aOp)
|
||||||
|
return false;
|
||||||
|
|
||||||
|
auto *bOp = dyn_cast<OperationInst>(b);
|
||||||
|
if (!bOp)
|
||||||
|
return true;
|
||||||
|
|
||||||
|
// Otherwise, do a linear scan to determine whether B comes after A.
|
||||||
|
auto aIter = BasicBlock::const_iterator(aOp);
|
||||||
|
auto bIter = BasicBlock::const_iterator(bOp);
|
||||||
|
auto fIter = aBlock->begin();
|
||||||
|
while (bIter != fIter) {
|
||||||
|
--bIter;
|
||||||
|
if (aIter == bIter)
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return true if value A properly dominates instruction B.
|
||||||
|
bool DominanceInfo::properlyDominates(const SSAValue *a, const Instruction *b) {
|
||||||
|
if (auto *aInst = a->getDefiningInst())
|
||||||
|
return properlyDominates(aInst, b);
|
||||||
|
|
||||||
|
// bbarguments properly dominate all instructions in their own block, so we
|
||||||
|
// use a dominates check here, not a properlyDominates check.
|
||||||
|
return dominates(cast<BBArgument>(a)->getOwner(), b->getBlock());
|
||||||
|
}
|
|
@ -1,4 +1,4 @@
|
||||||
//===- HyperRectangularSet.cpp - MLIR HyperRectangularSet Class--*- C++ -*-===//
|
//===- HyperRectangularSet.cpp - MLIR HyperRectangularSet Class -----------===//
|
||||||
//
|
//
|
||||||
// Copyright 2019 The MLIR Authors.
|
// Copyright 2019 The MLIR Authors.
|
||||||
//
|
//
|
||||||
|
@ -20,13 +20,10 @@
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
#include "mlir/Analysis/HyperRectangularSet.h"
|
#include "mlir/Analysis/HyperRectangularSet.h"
|
||||||
|
|
||||||
#include <algorithm>
|
|
||||||
|
|
||||||
#include "mlir/IR/AffineExpr.h"
|
#include "mlir/IR/AffineExpr.h"
|
||||||
#include "mlir/IR/IntegerSet.h"
|
#include "mlir/IR/IntegerSet.h"
|
||||||
#include "llvm/Support/raw_ostream.h"
|
#include "llvm/Support/raw_ostream.h"
|
||||||
|
#include <algorithm>
|
||||||
using namespace mlir;
|
using namespace mlir;
|
||||||
|
|
||||||
// TODO(bondhugula): clean this code up.
|
// TODO(bondhugula): clean this code up.
|
||||||
|
|
|
@ -33,6 +33,7 @@
|
||||||
//
|
//
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
|
#include "mlir/Analysis/Dominance.h"
|
||||||
#include "mlir/IR/Attributes.h"
|
#include "mlir/IR/Attributes.h"
|
||||||
#include "mlir/IR/CFGFunction.h"
|
#include "mlir/IR/CFGFunction.h"
|
||||||
#include "mlir/IR/MLFunction.h"
|
#include "mlir/IR/MLFunction.h"
|
||||||
|
@ -163,12 +164,15 @@ bool Verifier::verifyOperation(const Operation &op) {
|
||||||
namespace {
|
namespace {
|
||||||
struct CFGFuncVerifier : public Verifier {
|
struct CFGFuncVerifier : public Verifier {
|
||||||
const CFGFunction &fn;
|
const CFGFunction &fn;
|
||||||
|
DominanceInfo domInfo;
|
||||||
|
|
||||||
CFGFuncVerifier(const CFGFunction &fn) : Verifier(fn), fn(fn) {}
|
CFGFuncVerifier(const CFGFunction &fn)
|
||||||
|
: Verifier(fn), fn(fn), domInfo(const_cast<CFGFunction *>(&fn)) {}
|
||||||
|
|
||||||
bool verify();
|
bool verify();
|
||||||
bool verifyBlock(const BasicBlock &block);
|
bool verifyBlock(const BasicBlock &block);
|
||||||
bool verifyTerminator(const TerminatorInst &term);
|
bool verifyTerminator(const TerminatorInst &term);
|
||||||
|
bool verifyInstOperands(const Instruction &inst);
|
||||||
|
|
||||||
bool verifyBBArguments(ArrayRef<InstOperand> operands,
|
bool verifyBBArguments(ArrayRef<InstOperand> operands,
|
||||||
const BasicBlock *destBB, const TerminatorInst &term);
|
const BasicBlock *destBB, const TerminatorInst &term);
|
||||||
|
@ -212,6 +216,24 @@ bool CFGFuncVerifier::verify() {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool CFGFuncVerifier::verifyInstOperands(const Instruction &inst) {
|
||||||
|
// Check that operands properly dominate this use.
|
||||||
|
for (unsigned operandNo = 0, e = inst.getNumOperands(); operandNo != e;
|
||||||
|
++operandNo) {
|
||||||
|
auto *op = inst.getOperand(operandNo);
|
||||||
|
if (domInfo.properlyDominates(op, &inst))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
inst.emitError("operand #" + Twine(operandNo) +
|
||||||
|
" does not dominate this use");
|
||||||
|
if (auto *useInst = op->getDefiningInst())
|
||||||
|
useInst->emitNote("operand defined here");
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
bool CFGFuncVerifier::verifyBlock(const BasicBlock &block) {
|
bool CFGFuncVerifier::verifyBlock(const BasicBlock &block) {
|
||||||
if (!block.getTerminator())
|
if (!block.getTerminator())
|
||||||
return failure("basic block with no terminator", block);
|
return failure("basic block with no terminator", block);
|
||||||
|
@ -225,7 +247,7 @@ bool CFGFuncVerifier::verifyBlock(const BasicBlock &block) {
|
||||||
}
|
}
|
||||||
|
|
||||||
for (auto &inst : block) {
|
for (auto &inst : block) {
|
||||||
if (verifyOperation(inst))
|
if (verifyOperation(inst) || verifyInstOperands(inst))
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
|
@ -244,6 +266,9 @@ bool CFGFuncVerifier::verifyTerminator(const TerminatorInst &term) {
|
||||||
return failure("reference to operand defined in another function", term);
|
return failure("reference to operand defined in another function", term);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Verify dominance of values.
|
||||||
|
verifyInstOperands(term);
|
||||||
|
|
||||||
// Check that successors are in the right function.
|
// Check that successors are in the right function.
|
||||||
for (auto *succ : term.getBlock()->getSuccessors()) {
|
for (auto *succ : term.getBlock()->getSuccessors()) {
|
||||||
if (succ->getFunction() != &fn)
|
if (succ->getFunction() != &fn)
|
|
@ -1077,6 +1077,8 @@ public:
|
||||||
void print(const BranchInst *inst);
|
void print(const BranchInst *inst);
|
||||||
void print(const CondBranchInst *inst);
|
void print(const CondBranchInst *inst);
|
||||||
|
|
||||||
|
void printBBName(const BasicBlock *block) { os << "bb" << getBBID(block); }
|
||||||
|
|
||||||
unsigned getBBID(const BasicBlock *block) {
|
unsigned getBBID(const BasicBlock *block) {
|
||||||
auto it = basicBlockIDs.find(block);
|
auto it = basicBlockIDs.find(block);
|
||||||
assert(it != basicBlockIDs.end() && "Block not in this function?");
|
assert(it != basicBlockIDs.end() && "Block not in this function?");
|
||||||
|
@ -1129,7 +1131,7 @@ void CFGFunctionPrinter::print() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void CFGFunctionPrinter::print(const BasicBlock *block) {
|
void CFGFunctionPrinter::print(const BasicBlock *block) {
|
||||||
os << "bb" << getBBID(block);
|
printBBName(block);
|
||||||
|
|
||||||
if (!block->args_empty()) {
|
if (!block->args_empty()) {
|
||||||
os << '(';
|
os << '(';
|
||||||
|
@ -1150,7 +1152,8 @@ void CFGFunctionPrinter::print(const BasicBlock *block) {
|
||||||
if (block != &block->getFunction()->front())
|
if (block != &block->getFunction()->front())
|
||||||
os << "\t// no predecessors";
|
os << "\t// no predecessors";
|
||||||
} else if (auto *pred = block->getSinglePredecessor()) {
|
} else if (auto *pred = block->getSinglePredecessor()) {
|
||||||
os << "\t// pred: bb" << getBBID(pred);
|
os << "\t// pred: ";
|
||||||
|
printBBName(pred);
|
||||||
} else {
|
} else {
|
||||||
// We want to print the predecessors in increasing numeric order, not in
|
// We want to print the predecessors in increasing numeric order, not in
|
||||||
// whatever order the use-list is in, so gather and sort them.
|
// whatever order the use-list is in, so gather and sort them.
|
||||||
|
@ -1198,7 +1201,8 @@ void CFGFunctionPrinter::print(const OperationInst *inst) {
|
||||||
}
|
}
|
||||||
|
|
||||||
void CFGFunctionPrinter::print(const BranchInst *inst) {
|
void CFGFunctionPrinter::print(const BranchInst *inst) {
|
||||||
os << "br bb" << getBBID(inst->getDest());
|
os << "br ";
|
||||||
|
printBBName(inst->getDest());
|
||||||
|
|
||||||
if (inst->getNumOperands() != 0) {
|
if (inst->getNumOperands() != 0) {
|
||||||
os << '(';
|
os << '(';
|
||||||
|
@ -1215,7 +1219,8 @@ void CFGFunctionPrinter::print(const CondBranchInst *inst) {
|
||||||
os << "cond_br ";
|
os << "cond_br ";
|
||||||
printValueID(inst->getCondition());
|
printValueID(inst->getCondition());
|
||||||
|
|
||||||
os << ", bb" << getBBID(inst->getTrueDest());
|
os << ", ";
|
||||||
|
printBBName(inst->getTrueDest());
|
||||||
if (inst->getNumTrueOperands() != 0) {
|
if (inst->getNumTrueOperands() != 0) {
|
||||||
os << '(';
|
os << '(';
|
||||||
interleaveComma(inst->getTrueOperands(),
|
interleaveComma(inst->getTrueOperands(),
|
||||||
|
@ -1227,7 +1232,8 @@ void CFGFunctionPrinter::print(const CondBranchInst *inst) {
|
||||||
os << ")";
|
os << ")";
|
||||||
}
|
}
|
||||||
|
|
||||||
os << ", bb" << getBBID(inst->getFalseDest());
|
os << ", ";
|
||||||
|
printBBName(inst->getFalseDest());
|
||||||
if (inst->getNumFalseOperands() != 0) {
|
if (inst->getNumFalseOperands() != 0) {
|
||||||
os << '(';
|
os << '(';
|
||||||
interleaveComma(inst->getFalseOperands(),
|
interleaveComma(inst->getFalseOperands(),
|
||||||
|
@ -1555,6 +1561,17 @@ void BasicBlock::print(raw_ostream &os) const {
|
||||||
|
|
||||||
void BasicBlock::dump() const { print(llvm::errs()); }
|
void BasicBlock::dump() const { print(llvm::errs()); }
|
||||||
|
|
||||||
|
/// Print out the name of the basic block without printing its body.
|
||||||
|
void BasicBlock::printAsOperand(raw_ostream &os, bool printType) {
|
||||||
|
if (!getFunction()) {
|
||||||
|
os << "<<UNLINKED BLOCK>>\n";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
ModuleState state(getFunction()->getContext());
|
||||||
|
ModulePrinter modulePrinter(os, state);
|
||||||
|
CFGFunctionPrinter(getFunction(), modulePrinter).printBBName(this);
|
||||||
|
}
|
||||||
|
|
||||||
void Statement::print(raw_ostream &os) const {
|
void Statement::print(raw_ostream &os) const {
|
||||||
MLFunction *function = findFunction();
|
MLFunction *function = findFunction();
|
||||||
if (!function) {
|
if (!function) {
|
||||||
|
|
|
@ -77,7 +77,7 @@ CFGFunction *CFGValue::getFunction() {
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
/// Return the function that this argument is defined in.
|
/// Return the function that this argument is defined in.
|
||||||
CFGFunction *BBArgument::getFunction() const {
|
CFGFunction *BBArgument::getFunction() {
|
||||||
if (auto *owner = getOwner())
|
if (auto *owner = getOwner())
|
||||||
return owner->getFunction();
|
return owner->getFunction();
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
|
@ -40,7 +40,7 @@ struct ConstantFold : public FunctionPass {
|
||||||
/// Attempt to fold the specified operation, updating the IR to match. If
|
/// Attempt to fold the specified operation, updating the IR to match. If
|
||||||
/// constants are found, we keep track of them in the existingConstants list.
|
/// constants are found, we keep track of them in the existingConstants list.
|
||||||
///
|
///
|
||||||
/// This returns true if the operation was folded.
|
/// This returns false if the operation was successfully folded.
|
||||||
bool ConstantFold::foldOperation(Operation *op,
|
bool ConstantFold::foldOperation(Operation *op,
|
||||||
SmallVectorImpl<SSAValue *> &existingConstants,
|
SmallVectorImpl<SSAValue *> &existingConstants,
|
||||||
ConstantFactoryType constantFactory) {
|
ConstantFactoryType constantFactory) {
|
||||||
|
@ -49,7 +49,7 @@ bool ConstantFold::foldOperation(Operation *op,
|
||||||
// later, and don't try to fold it.
|
// later, and don't try to fold it.
|
||||||
if (op->is<ConstantOp>()) {
|
if (op->is<ConstantOp>()) {
|
||||||
existingConstants.push_back(op->getResult(0));
|
existingConstants.push_back(op->getResult(0));
|
||||||
return false;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check to see if each of the operands is a trivial constant. If so, get
|
// Check to see if each of the operands is a trivial constant. If so, get
|
||||||
|
@ -63,13 +63,13 @@ bool ConstantFold::foldOperation(Operation *op,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// If one of the operands was non-constant, then we can't fold it.
|
// If one of the operands was non-constant, then we can't fold it.
|
||||||
return false;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Attempt to constant fold the operation.
|
// Attempt to constant fold the operation.
|
||||||
SmallVector<Attribute *, 8> resultConstants;
|
SmallVector<Attribute *, 8> resultConstants;
|
||||||
if (op->constantFold(operandConstants, resultConstants))
|
if (op->constantFold(operandConstants, resultConstants))
|
||||||
return false;
|
return true;
|
||||||
|
|
||||||
// Ok, if everything succeeded, then we can create constants corresponding
|
// Ok, if everything succeeded, then we can create constants corresponding
|
||||||
// to the result of the call.
|
// to the result of the call.
|
||||||
|
@ -88,7 +88,7 @@ bool ConstantFold::foldOperation(Operation *op,
|
||||||
res->replaceAllUsesWith(cst);
|
res->replaceAllUsesWith(cst);
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// For now, we do a simple top-down pass over a function folding constants. We
|
// For now, we do a simple top-down pass over a function folding constants. We
|
||||||
|
@ -108,7 +108,7 @@ PassResult ConstantFold::runOnCFGFunction(CFGFunction *f) {
|
||||||
->getResult();
|
->getResult();
|
||||||
};
|
};
|
||||||
|
|
||||||
if (foldOperation(&inst, existingConstants, constantFactory)) {
|
if (!foldOperation(&inst, existingConstants, constantFactory)) {
|
||||||
// At this point the operation is dead, remove it.
|
// At this point the operation is dead, remove it.
|
||||||
// TODO: This is assuming that all constant foldable operations have no
|
// TODO: This is assuming that all constant foldable operations have no
|
||||||
// side effects. When we have side effect modeling, we should verify
|
// side effects. When we have side effect modeling, we should verify
|
||||||
|
@ -160,7 +160,7 @@ void ConstantFold::foldStmtBlock(
|
||||||
->getResult();
|
->getResult();
|
||||||
};
|
};
|
||||||
|
|
||||||
if (foldOperation(opStmt, existingConstants, constantFactory)) {
|
if (!foldOperation(opStmt, existingConstants, constantFactory)) {
|
||||||
// At this point the operation is dead, remove it.
|
// At this point the operation is dead, remove it.
|
||||||
// TODO: This is assuming that all constant foldable operations have no
|
// TODO: This is assuming that all constant foldable operations have no
|
||||||
// side effects. When we have side effect modeling, we should verify that
|
// side effects. When we have side effect modeling, we should verify that
|
||||||
|
|
|
@ -396,6 +396,18 @@ mlfunc @dominance_failure() {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// -----
|
||||||
|
|
||||||
|
cfgfunc @dominance_failure() {
|
||||||
|
bb0:
|
||||||
|
"foo"(%x) : (i32) -> () // expected-error {{operand #0 does not dominate this use}}
|
||||||
|
br bb1
|
||||||
|
bb1:
|
||||||
|
%x = "bar"() : () -> i32 // expected-error {{operand defined here}}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// -----
|
// -----
|
||||||
|
|
||||||
mlfunc @return_type_mismatch() -> i32 {
|
mlfunc @return_type_mismatch() -> i32 {
|
||||||
|
|
Loading…
Reference in New Issue