2005-09-03 01:14:03 +00:00
|
|
|
//===- DAGISelEmitter.h - Generate an instruction selector ------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file was developed by Chris Lattner and is distributed under
|
|
|
|
// the University of Illinois Open Source License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This tablegen backend emits a DAG instruction selector.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef DAGISEL_EMITTER_H
|
|
|
|
#define DAGISEL_EMITTER_H
|
|
|
|
|
|
|
|
#include "TableGenBackend.h"
|
|
|
|
#include "CodeGenTarget.h"
|
|
|
|
|
|
|
|
namespace llvm {
|
2005-09-07 23:44:43 +00:00
|
|
|
class Record;
|
2005-09-10 02:00:02 +00:00
|
|
|
struct Init;
|
2005-09-15 22:23:50 +00:00
|
|
|
class ListInit;
|
2005-09-07 23:44:43 +00:00
|
|
|
class DagInit;
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
class SDNodeInfo;
|
2005-09-07 23:44:43 +00:00
|
|
|
class TreePattern;
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
class TreePatternNode;
|
2005-09-07 23:44:43 +00:00
|
|
|
class DAGISelEmitter;
|
2005-09-08 21:03:01 +00:00
|
|
|
|
2005-10-14 06:12:03 +00:00
|
|
|
/// MVT::DAGISelGenValueType - These are some extended forms of MVT::ValueType
|
|
|
|
/// that we use as lattice values during type inferrence.
|
|
|
|
namespace MVT {
|
|
|
|
enum DAGISelGenValueType {
|
|
|
|
isFP = MVT::LAST_VALUETYPE,
|
|
|
|
isInt,
|
|
|
|
isUnknown
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2005-09-08 21:27:15 +00:00
|
|
|
/// SDTypeConstraint - This is a discriminated union of constraints,
|
|
|
|
/// corresponding to the SDTypeConstraint tablegen class in Target.td.
|
|
|
|
struct SDTypeConstraint {
|
|
|
|
SDTypeConstraint(Record *R);
|
|
|
|
|
|
|
|
unsigned OperandNo; // The operand # this constraint applies to.
|
|
|
|
enum {
|
2005-10-14 04:53:53 +00:00
|
|
|
SDTCisVT, SDTCisInt, SDTCisFP, SDTCisSameAs, SDTCisVTSmallerThanOp,
|
|
|
|
SDTCisOpSmallerThanOp
|
2005-09-08 21:27:15 +00:00
|
|
|
} ConstraintType;
|
|
|
|
|
|
|
|
union { // The discriminated union.
|
|
|
|
struct {
|
|
|
|
MVT::ValueType VT;
|
|
|
|
} SDTCisVT_Info;
|
|
|
|
struct {
|
|
|
|
unsigned OtherOperandNum;
|
|
|
|
} SDTCisSameAs_Info;
|
|
|
|
struct {
|
|
|
|
unsigned OtherOperandNum;
|
|
|
|
} SDTCisVTSmallerThanOp_Info;
|
2005-10-14 04:53:53 +00:00
|
|
|
struct {
|
|
|
|
unsigned BigOperandNum;
|
|
|
|
} SDTCisOpSmallerThanOp_Info;
|
2005-09-08 21:27:15 +00:00
|
|
|
} x;
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
|
|
|
|
/// ApplyTypeConstraint - Given a node in a pattern, apply this type
|
|
|
|
/// constraint to the nodes operands. This returns true if it makes a
|
|
|
|
/// change, false otherwise. If a type contradiction is found, throw an
|
|
|
|
/// exception.
|
|
|
|
bool ApplyTypeConstraint(TreePatternNode *N, const SDNodeInfo &NodeInfo,
|
|
|
|
TreePattern &TP) const;
|
|
|
|
|
|
|
|
/// getOperandNum - Return the node corresponding to operand #OpNo in tree
|
|
|
|
/// N, which has NumResults results.
|
|
|
|
TreePatternNode *getOperandNum(unsigned OpNo, TreePatternNode *N,
|
|
|
|
unsigned NumResults) const;
|
2005-09-08 21:27:15 +00:00
|
|
|
};
|
|
|
|
|
2005-09-08 21:03:01 +00:00
|
|
|
/// SDNodeInfo - One of these records is created for each SDNode instance in
|
|
|
|
/// the target .td file. This represents the various dag nodes we will be
|
|
|
|
/// processing.
|
|
|
|
class SDNodeInfo {
|
|
|
|
Record *Def;
|
|
|
|
std::string EnumName;
|
|
|
|
std::string SDClassName;
|
2005-09-28 18:28:29 +00:00
|
|
|
unsigned Properties;
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
unsigned NumResults;
|
|
|
|
int NumOperands;
|
2005-09-08 21:27:15 +00:00
|
|
|
std::vector<SDTypeConstraint> TypeConstraints;
|
2005-09-08 21:03:01 +00:00
|
|
|
public:
|
|
|
|
SDNodeInfo(Record *R); // Parse the specified record.
|
|
|
|
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
unsigned getNumResults() const { return NumResults; }
|
2005-09-08 21:27:15 +00:00
|
|
|
int getNumOperands() const { return NumOperands; }
|
2005-09-08 21:03:01 +00:00
|
|
|
Record *getRecord() const { return Def; }
|
|
|
|
const std::string &getEnumName() const { return EnumName; }
|
|
|
|
const std::string &getSDClassName() const { return SDClassName; }
|
2005-09-08 21:27:15 +00:00
|
|
|
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
const std::vector<SDTypeConstraint> &getTypeConstraints() const {
|
2005-09-08 21:27:15 +00:00
|
|
|
return TypeConstraints;
|
|
|
|
}
|
2005-09-28 18:28:29 +00:00
|
|
|
|
|
|
|
// SelectionDAG node properties.
|
2005-09-28 20:58:06 +00:00
|
|
|
enum SDNP { SDNPCommutative, SDNPAssociative };
|
2005-09-28 18:28:29 +00:00
|
|
|
|
|
|
|
/// hasProperty - Return true if this node has the specified property.
|
|
|
|
///
|
|
|
|
bool hasProperty(enum SDNP Prop) const { return Properties & (1 << Prop); }
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
|
|
|
|
/// ApplyTypeConstraints - Given a node in a pattern, apply the type
|
|
|
|
/// constraints for this node to the operands of the node. This returns
|
|
|
|
/// true if it makes a change, false otherwise. If a type contradiction is
|
|
|
|
/// found, throw an exception.
|
|
|
|
bool ApplyTypeConstraints(TreePatternNode *N, TreePattern &TP) const {
|
|
|
|
bool MadeChange = false;
|
|
|
|
for (unsigned i = 0, e = TypeConstraints.size(); i != e; ++i)
|
|
|
|
MadeChange |= TypeConstraints[i].ApplyTypeConstraint(N, *this, TP);
|
|
|
|
return MadeChange;
|
|
|
|
}
|
2005-09-08 21:03:01 +00:00
|
|
|
};
|
2005-09-03 01:14:03 +00:00
|
|
|
|
2005-09-07 23:44:43 +00:00
|
|
|
/// FIXME: TreePatternNode's can be shared in some cases (due to dag-shaped
|
|
|
|
/// patterns), and as such should be ref counted. We currently just leak all
|
|
|
|
/// TreePatternNode objects!
|
|
|
|
class TreePatternNode {
|
|
|
|
/// The inferred type for this node, or MVT::LAST_VALUETYPE if it hasn't
|
|
|
|
/// been determined yet.
|
2005-10-14 06:12:03 +00:00
|
|
|
unsigned char Ty;
|
|
|
|
|
2005-09-07 23:44:43 +00:00
|
|
|
/// Operator - The Record for the operator if this is an interior node (not
|
|
|
|
/// a leaf).
|
|
|
|
Record *Operator;
|
|
|
|
|
|
|
|
/// Val - The init value (e.g. the "GPRC" record, or "7") for a leaf.
|
|
|
|
///
|
|
|
|
Init *Val;
|
|
|
|
|
|
|
|
/// Name - The name given to this node with the :$foo notation.
|
|
|
|
///
|
|
|
|
std::string Name;
|
|
|
|
|
|
|
|
/// PredicateFn - The predicate function to execute on this node to check
|
|
|
|
/// for a match. If this string is empty, no predicate is involved.
|
|
|
|
std::string PredicateFn;
|
|
|
|
|
2005-09-13 21:51:00 +00:00
|
|
|
/// TransformFn - The transformation function to execute on this node before
|
|
|
|
/// it can be substituted into the resulting instruction on a pattern match.
|
2005-09-14 22:55:26 +00:00
|
|
|
Record *TransformFn;
|
2005-09-13 21:51:00 +00:00
|
|
|
|
2005-09-07 23:44:43 +00:00
|
|
|
std::vector<TreePatternNode*> Children;
|
|
|
|
public:
|
|
|
|
TreePatternNode(Record *Op, const std::vector<TreePatternNode*> &Ch)
|
2005-10-14 06:12:03 +00:00
|
|
|
: Ty(MVT::isUnknown), Operator(Op), Val(0), TransformFn(0),
|
2005-09-14 22:55:26 +00:00
|
|
|
Children(Ch) {}
|
2005-09-07 23:44:43 +00:00
|
|
|
TreePatternNode(Init *val) // leaf ctor
|
2005-10-14 06:12:03 +00:00
|
|
|
: Ty(MVT::isUnknown), Operator(0), Val(val), TransformFn(0) {}
|
2005-09-07 23:44:43 +00:00
|
|
|
~TreePatternNode();
|
|
|
|
|
|
|
|
const std::string &getName() const { return Name; }
|
|
|
|
void setName(const std::string &N) { Name = N; }
|
|
|
|
|
|
|
|
bool isLeaf() const { return Val != 0; }
|
2005-10-14 06:12:03 +00:00
|
|
|
bool hasTypeSet() const { return Ty < MVT::LAST_VALUETYPE; }
|
|
|
|
bool isTypeCompletelyUnknown() const {
|
|
|
|
return Ty == MVT::isUnknown;
|
|
|
|
}
|
|
|
|
MVT::ValueType getType() const {
|
|
|
|
assert(hasTypeSet() && "Doesn't have a type yet!");
|
|
|
|
return (MVT::ValueType)Ty;
|
|
|
|
}
|
|
|
|
unsigned char getExtType() const { return Ty; }
|
|
|
|
void setType(unsigned char VT) { Ty = VT; }
|
2005-09-07 23:44:43 +00:00
|
|
|
|
|
|
|
Init *getLeafValue() const { assert(isLeaf()); return Val; }
|
|
|
|
Record *getOperator() const { assert(!isLeaf()); return Operator; }
|
|
|
|
|
|
|
|
unsigned getNumChildren() const { return Children.size(); }
|
|
|
|
TreePatternNode *getChild(unsigned N) const { return Children[N]; }
|
|
|
|
void setChild(unsigned i, TreePatternNode *N) {
|
|
|
|
Children[i] = N;
|
|
|
|
}
|
|
|
|
|
|
|
|
const std::string &getPredicateFn() const { return PredicateFn; }
|
|
|
|
void setPredicateFn(const std::string &Fn) { PredicateFn = Fn; }
|
2005-09-13 21:51:00 +00:00
|
|
|
|
2005-09-14 22:55:26 +00:00
|
|
|
Record *getTransformFn() const { return TransformFn; }
|
|
|
|
void setTransformFn(Record *Fn) { TransformFn = Fn; }
|
2005-09-07 23:44:43 +00:00
|
|
|
|
|
|
|
void print(std::ostream &OS) const;
|
|
|
|
void dump() const;
|
|
|
|
|
|
|
|
public: // Higher level manipulation routines.
|
|
|
|
|
|
|
|
/// clone - Return a new copy of this tree.
|
|
|
|
///
|
|
|
|
TreePatternNode *clone() const;
|
|
|
|
|
2005-09-29 19:28:10 +00:00
|
|
|
/// isIsomorphicTo - Return true if this node is recursively isomorphic to
|
|
|
|
/// the specified node. For this comparison, all of the state of the node
|
|
|
|
/// is considered, except for the assigned name. Nodes with differing names
|
|
|
|
/// that are otherwise identical are considered isomorphic.
|
|
|
|
bool isIsomorphicTo(const TreePatternNode *N) const;
|
|
|
|
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
/// SubstituteFormalArguments - Replace the formal arguments in this tree
|
|
|
|
/// with actual values specified by ArgMap.
|
2005-09-07 23:44:43 +00:00
|
|
|
void SubstituteFormalArguments(std::map<std::string,
|
|
|
|
TreePatternNode*> &ArgMap);
|
|
|
|
|
|
|
|
/// InlinePatternFragments - If this pattern refers to any pattern
|
|
|
|
/// fragments, inline them into place, giving us a pattern without any
|
|
|
|
/// PatFrag references.
|
|
|
|
TreePatternNode *InlinePatternFragments(TreePattern &TP);
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
|
|
|
|
/// ApplyTypeConstraints - Apply all of the type constraints relevent to
|
|
|
|
/// this node and its children in the tree. This returns true if it makes a
|
|
|
|
/// change, false otherwise. If a type contradiction is found, throw an
|
|
|
|
/// exception.
|
2005-10-14 04:11:13 +00:00
|
|
|
bool ApplyTypeConstraints(TreePattern &TP, bool NotRegisters);
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
|
|
|
|
/// UpdateNodeType - Set the node type of N to VT if VT contains
|
|
|
|
/// information. If N already contains a conflicting type, then throw an
|
|
|
|
/// exception. This returns true if any information was updated.
|
|
|
|
///
|
2005-10-14 06:12:03 +00:00
|
|
|
bool UpdateNodeType(unsigned char EVT, TreePattern &TP);
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
|
|
|
|
/// ContainsUnresolvedType - Return true if this tree contains any
|
|
|
|
/// unresolved types.
|
|
|
|
bool ContainsUnresolvedType() const {
|
|
|
|
if (Ty == MVT::LAST_VALUETYPE) return true;
|
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i)
|
|
|
|
if (getChild(i)->ContainsUnresolvedType()) return true;
|
|
|
|
return false;
|
|
|
|
}
|
Emit an error if instructions or patterns are defined but can never match.
Currently we check that immediate values live on the RHS of commutative
operators. Defining ORI like this, for example:
def ORI : DForm_4<24, (ops GPRC:$dst, GPRC:$src1, u16imm:$src2),
"ori $dst, $src1, $src2",
[(set GPRC:$dst, (or immZExt16:$src2, GPRC:$src1))]>;
results in:
tblgen: In ORI: Instruction can never match: Immediate values must be on the RHS of commutative operators!
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23501 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-28 19:27:25 +00:00
|
|
|
|
2005-09-28 20:58:06 +00:00
|
|
|
/// canPatternMatch - If it is impossible for this pattern to match on this
|
|
|
|
/// target, fill in Reason and return false. Otherwise, return true.
|
Emit an error if instructions or patterns are defined but can never match.
Currently we check that immediate values live on the RHS of commutative
operators. Defining ORI like this, for example:
def ORI : DForm_4<24, (ops GPRC:$dst, GPRC:$src1, u16imm:$src2),
"ori $dst, $src1, $src2",
[(set GPRC:$dst, (or immZExt16:$src2, GPRC:$src1))]>;
results in:
tblgen: In ORI: Instruction can never match: Immediate values must be on the RHS of commutative operators!
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23501 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-28 19:27:25 +00:00
|
|
|
bool canPatternMatch(std::string &Reason, DAGISelEmitter &ISE);
|
2005-09-07 23:44:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2005-09-13 21:20:49 +00:00
|
|
|
/// TreePattern - Represent a pattern, used for instructions, pattern
|
|
|
|
/// fragments, etc.
|
2005-09-07 23:44:43 +00:00
|
|
|
///
|
|
|
|
class TreePattern {
|
|
|
|
/// Trees - The list of pattern trees which corresponds to this pattern.
|
|
|
|
/// Note that PatFrag's only have a single tree.
|
|
|
|
///
|
|
|
|
std::vector<TreePatternNode*> Trees;
|
|
|
|
|
|
|
|
/// TheRecord - The actual TableGen record corresponding to this pattern.
|
|
|
|
///
|
|
|
|
Record *TheRecord;
|
|
|
|
|
|
|
|
/// Args - This is a list of all of the arguments to this pattern (for
|
|
|
|
/// PatFrag patterns), which are the 'node' markers in this pattern.
|
|
|
|
std::vector<std::string> Args;
|
|
|
|
|
|
|
|
/// ISE - the DAG isel emitter coordinating this madness.
|
|
|
|
///
|
|
|
|
DAGISelEmitter &ISE;
|
|
|
|
public:
|
|
|
|
|
|
|
|
/// TreePattern constructor - Parse the specified DagInits into the
|
|
|
|
/// current record.
|
2005-09-15 22:23:50 +00:00
|
|
|
TreePattern(Record *TheRec, ListInit *RawPat, DAGISelEmitter &ise);
|
|
|
|
TreePattern(Record *TheRec, DagInit *Pat, DAGISelEmitter &ise);
|
|
|
|
TreePattern(Record *TheRec, TreePatternNode *Pat, DAGISelEmitter &ise);
|
2005-09-07 23:44:43 +00:00
|
|
|
|
|
|
|
/// getTrees - Return the tree patterns which corresponds to this pattern.
|
|
|
|
///
|
|
|
|
const std::vector<TreePatternNode*> &getTrees() const { return Trees; }
|
2005-09-09 01:11:17 +00:00
|
|
|
unsigned getNumTrees() const { return Trees.size(); }
|
|
|
|
TreePatternNode *getTree(unsigned i) const { return Trees[i]; }
|
2005-09-09 01:15:01 +00:00
|
|
|
TreePatternNode *getOnlyTree() const {
|
|
|
|
assert(Trees.size() == 1 && "Doesn't have exactly one pattern!");
|
|
|
|
return Trees[0];
|
|
|
|
}
|
2005-09-07 23:44:43 +00:00
|
|
|
|
|
|
|
/// getRecord - Return the actual TableGen record corresponding to this
|
|
|
|
/// pattern.
|
|
|
|
///
|
|
|
|
Record *getRecord() const { return TheRecord; }
|
|
|
|
|
|
|
|
unsigned getNumArgs() const { return Args.size(); }
|
|
|
|
const std::string &getArgName(unsigned i) const {
|
|
|
|
assert(i < Args.size() && "Argument reference out of range!");
|
|
|
|
return Args[i];
|
|
|
|
}
|
2005-09-13 21:20:49 +00:00
|
|
|
std::vector<std::string> &getArgList() { return Args; }
|
2005-09-07 23:44:43 +00:00
|
|
|
|
|
|
|
DAGISelEmitter &getDAGISelEmitter() const { return ISE; }
|
|
|
|
|
|
|
|
/// InlinePatternFragments - If this pattern refers to any pattern
|
|
|
|
/// fragments, inline them into place, giving us a pattern without any
|
|
|
|
/// PatFrag references.
|
|
|
|
void InlinePatternFragments() {
|
|
|
|
for (unsigned i = 0, e = Trees.size(); i != e; ++i)
|
|
|
|
Trees[i] = Trees[i]->InlinePatternFragments(*this);
|
|
|
|
}
|
|
|
|
|
Implement a complete type inference system for dag patterns, based on the
constraints defined in the DAG node definitions in the .td files. This
allows us to infer (and check!) the types for all nodes in the current
ppc .td file. For example, instead of:
Inst pattern EQV: (set GPRC:i32:$rT, (xor (xor GPRC:i32:$rA, GPRC:i32:$rB), (imm)<<Predicate_immAllOnes>>))
we now fully infer:
Inst pattern EQV: (set:void GPRC:i32:$rT, (xor:i32 (xor:i32 GPRC:i32:$rA, GPRC:i32:$rB), (imm:i32)<<Predicate_immAllOnes>>))
from: (set GPRC:$rT, (not (xor GPRC:$rA, GPRC:$rB)))
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23284 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-08 23:22:48 +00:00
|
|
|
/// InferAllTypes - Infer/propagate as many types throughout the expression
|
|
|
|
/// patterns as possible. Return true if all types are infered, false
|
|
|
|
/// otherwise. Throw an exception if a type contradiction is found.
|
|
|
|
bool InferAllTypes();
|
|
|
|
|
2005-09-07 23:44:43 +00:00
|
|
|
/// error - Throw an exception, prefixing it with information about this
|
|
|
|
/// pattern.
|
|
|
|
void error(const std::string &Msg) const;
|
|
|
|
|
|
|
|
void print(std::ostream &OS) const;
|
|
|
|
void dump() const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
TreePatternNode *ParseTreePattern(DagInit *DI);
|
|
|
|
};
|
2005-09-14 04:03:16 +00:00
|
|
|
|
|
|
|
|
|
|
|
class DAGInstruction {
|
|
|
|
TreePattern *Pattern;
|
|
|
|
unsigned NumResults;
|
|
|
|
unsigned NumOperands;
|
2005-09-15 21:51:12 +00:00
|
|
|
std::vector<MVT::ValueType> ResultTypes;
|
|
|
|
std::vector<MVT::ValueType> OperandTypes;
|
2005-09-14 22:55:26 +00:00
|
|
|
TreePatternNode *ResultPattern;
|
2005-09-14 04:03:16 +00:00
|
|
|
public:
|
2005-09-15 21:51:12 +00:00
|
|
|
DAGInstruction(TreePattern *TP,
|
|
|
|
const std::vector<MVT::ValueType> &resultTypes,
|
2005-09-15 22:23:50 +00:00
|
|
|
const std::vector<MVT::ValueType> &operandTypes)
|
2005-09-15 21:51:12 +00:00
|
|
|
: Pattern(TP), ResultTypes(resultTypes), OperandTypes(operandTypes),
|
2005-09-15 22:23:50 +00:00
|
|
|
ResultPattern(0) {}
|
2005-09-14 04:03:16 +00:00
|
|
|
|
|
|
|
TreePattern *getPattern() const { return Pattern; }
|
2005-09-15 21:51:12 +00:00
|
|
|
unsigned getNumResults() const { return ResultTypes.size(); }
|
|
|
|
unsigned getNumOperands() const { return OperandTypes.size(); }
|
|
|
|
|
2005-09-15 22:23:50 +00:00
|
|
|
void setResultPattern(TreePatternNode *R) { ResultPattern = R; }
|
|
|
|
|
2005-09-15 21:51:12 +00:00
|
|
|
MVT::ValueType getResultType(unsigned RN) const {
|
|
|
|
assert(RN < ResultTypes.size());
|
|
|
|
return ResultTypes[RN];
|
|
|
|
}
|
|
|
|
|
|
|
|
MVT::ValueType getOperandType(unsigned ON) const {
|
|
|
|
assert(ON < OperandTypes.size());
|
|
|
|
return OperandTypes[ON];
|
|
|
|
}
|
2005-09-14 22:55:26 +00:00
|
|
|
TreePatternNode *getResultPattern() const { return ResultPattern; }
|
2005-09-14 04:03:16 +00:00
|
|
|
};
|
2005-09-07 23:44:43 +00:00
|
|
|
|
|
|
|
|
2005-09-03 01:14:03 +00:00
|
|
|
/// InstrSelectorEmitter - The top-level class which coordinates construction
|
|
|
|
/// and emission of the instruction selector.
|
|
|
|
///
|
|
|
|
class DAGISelEmitter : public TableGenBackend {
|
2005-09-23 20:52:47 +00:00
|
|
|
public:
|
|
|
|
typedef std::pair<TreePatternNode*, TreePatternNode*> PatternToMatch;
|
|
|
|
private:
|
2005-09-03 01:14:03 +00:00
|
|
|
RecordKeeper &Records;
|
|
|
|
CodeGenTarget Target;
|
|
|
|
|
2005-09-08 21:03:01 +00:00
|
|
|
std::map<Record*, SDNodeInfo> SDNodes;
|
2005-09-13 21:51:00 +00:00
|
|
|
std::map<Record*, std::pair<Record*, std::string> > SDNodeXForms;
|
2005-09-07 23:44:43 +00:00
|
|
|
std::map<Record*, TreePattern*> PatternFragments;
|
2005-09-15 21:57:35 +00:00
|
|
|
std::map<Record*, DAGInstruction> Instructions;
|
2005-09-14 00:09:24 +00:00
|
|
|
|
|
|
|
/// PatternsToMatch - All of the things we are matching on the DAG. The first
|
|
|
|
/// value is the pattern to match, the second pattern is the result to
|
|
|
|
/// emit.
|
2005-09-23 19:36:15 +00:00
|
|
|
std::vector<PatternToMatch> PatternsToMatch;
|
2005-09-03 01:14:03 +00:00
|
|
|
public:
|
|
|
|
DAGISelEmitter(RecordKeeper &R) : Records(R) {}
|
|
|
|
|
|
|
|
// run - Output the isel, returning true on failure.
|
2005-09-07 23:44:43 +00:00
|
|
|
void run(std::ostream &OS);
|
2005-09-08 21:03:01 +00:00
|
|
|
|
2005-10-14 04:11:13 +00:00
|
|
|
const CodeGenTarget &getTargetInfo() const { return Target; }
|
|
|
|
|
2005-09-08 21:03:01 +00:00
|
|
|
const SDNodeInfo &getSDNodeInfo(Record *R) const {
|
|
|
|
assert(SDNodes.count(R) && "Unknown node!");
|
|
|
|
return SDNodes.find(R)->second;
|
|
|
|
}
|
2005-09-07 23:44:43 +00:00
|
|
|
|
|
|
|
TreePattern *getPatternFragment(Record *R) const {
|
|
|
|
assert(PatternFragments.count(R) && "Invalid pattern fragment request!");
|
|
|
|
return PatternFragments.find(R)->second;
|
|
|
|
}
|
|
|
|
|
2005-09-13 21:59:15 +00:00
|
|
|
const std::pair<Record*, std::string> &getSDNodeTransform(Record *R) const {
|
|
|
|
assert(SDNodeXForms.count(R) && "Invalid transform!");
|
|
|
|
return SDNodeXForms.find(R)->second;
|
|
|
|
}
|
|
|
|
|
2005-09-15 21:57:35 +00:00
|
|
|
const DAGInstruction &getInstruction(Record *R) const {
|
|
|
|
assert(Instructions.count(R) && "Unknown instruction!");
|
|
|
|
return Instructions.find(R)->second;
|
|
|
|
}
|
|
|
|
|
2005-09-07 23:44:43 +00:00
|
|
|
private:
|
2005-09-08 21:03:01 +00:00
|
|
|
void ParseNodeInfo();
|
2005-09-13 21:51:00 +00:00
|
|
|
void ParseNodeTransforms(std::ostream &OS);
|
2005-09-15 02:38:02 +00:00
|
|
|
void ParsePatternFragments(std::ostream &OS);
|
|
|
|
void ParseInstructions();
|
|
|
|
void ParsePatterns();
|
Emit an error if instructions or patterns are defined but can never match.
Currently we check that immediate values live on the RHS of commutative
operators. Defining ORI like this, for example:
def ORI : DForm_4<24, (ops GPRC:$dst, GPRC:$src1, u16imm:$src2),
"ori $dst, $src1, $src2",
[(set GPRC:$dst, (or immZExt16:$src2, GPRC:$src1))]>;
results in:
tblgen: In ORI: Instruction can never match: Immediate values must be on the RHS of commutative operators!
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@23501 91177308-0d34-0410-b5e6-96231b3b80d8
2005-09-28 19:27:25 +00:00
|
|
|
void GenerateVariants();
|
2005-09-14 20:53:42 +00:00
|
|
|
void FindPatternInputsAndOutputs(TreePattern *I, TreePatternNode *Pat,
|
|
|
|
std::map<std::string,
|
|
|
|
TreePatternNode*> &InstInputs,
|
|
|
|
std::map<std::string, Record*> &InstResults);
|
2005-09-23 21:33:23 +00:00
|
|
|
void EmitMatchForPattern(TreePatternNode *N, const std::string &RootName,
|
2005-09-23 23:16:51 +00:00
|
|
|
std::map<std::string,std::string> &VarMap,
|
2005-09-23 21:33:23 +00:00
|
|
|
unsigned PatternNo, std::ostream &OS);
|
2005-09-24 00:40:24 +00:00
|
|
|
unsigned CodeGenPatternResult(TreePatternNode *N, unsigned &Ctr,
|
|
|
|
std::map<std::string,std::string> &VariableMap,
|
|
|
|
std::ostream &OS);
|
2005-09-23 20:52:47 +00:00
|
|
|
void EmitCodeForPattern(PatternToMatch &Pattern, std::ostream &OS);
|
2005-09-07 23:44:43 +00:00
|
|
|
void EmitInstructionSelector(std::ostream &OS);
|
2005-09-03 01:14:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // End llvm namespace
|
|
|
|
|
|
|
|
#endif
|