2008-01-06 09:10:31 +08:00
|
|
|
//===- CodeGenDAGPatterns.cpp - Read DAG patterns from .td file -----------===//
|
2008-01-06 06:25:12 +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
|
2008-01-06 06:25:12 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2008-01-06 09:10:31 +08:00
|
|
|
// This file implements the CodeGenDAGPatterns class, which is used to read and
|
2008-01-06 06:25:12 +08:00
|
|
|
// represent the patterns present in a .td file for instructions.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2008-01-06 07:37:52 +08:00
|
|
|
#include "CodeGenDAGPatterns.h"
|
2018-08-28 23:42:08 +08:00
|
|
|
#include "llvm/ADT/BitVector.h"
|
2017-09-21 02:01:40 +08:00
|
|
|
#include "llvm/ADT/DenseSet.h"
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
#include "llvm/ADT/MapVector.h"
|
2010-03-15 14:00:16 +08:00
|
|
|
#include "llvm/ADT/STLExtras.h"
|
2017-09-15 00:56:21 +08:00
|
|
|
#include "llvm/ADT/SmallSet.h"
|
2015-11-28 16:23:02 +08:00
|
|
|
#include "llvm/ADT/SmallString.h"
|
2012-12-04 18:37:14 +08:00
|
|
|
#include "llvm/ADT/StringExtras.h"
|
2017-09-21 12:55:03 +08:00
|
|
|
#include "llvm/ADT/StringMap.h"
|
2012-04-19 01:46:41 +08:00
|
|
|
#include "llvm/ADT/Twine.h"
|
2008-01-06 06:25:12 +08:00
|
|
|
#include "llvm/Support/Debug.h"
|
2012-01-17 12:43:56 +08:00
|
|
|
#include "llvm/Support/ErrorHandling.h"
|
2019-08-14 18:48:39 +08:00
|
|
|
#include "llvm/Support/TypeSize.h"
|
2012-12-04 18:37:14 +08:00
|
|
|
#include "llvm/TableGen/Error.h"
|
|
|
|
#include "llvm/TableGen/Record.h"
|
2008-01-16 05:43:17 +08:00
|
|
|
#include <algorithm>
|
2012-03-23 19:35:30 +08:00
|
|
|
#include <cstdio>
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
#include <iterator>
|
2012-03-23 19:35:30 +08:00
|
|
|
#include <set>
|
2008-01-06 06:25:12 +08:00
|
|
|
using namespace llvm;
|
|
|
|
|
2014-04-22 06:55:11 +08:00
|
|
|
#define DEBUG_TYPE "dag-patterns"
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
static inline bool isIntegerOrPtr(MVT VT) {
|
|
|
|
return VT.isInteger() || VT == MVT::iPTR;
|
2008-06-06 20:08:01 +08:00
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
static inline bool isFloatingPoint(MVT VT) {
|
|
|
|
return VT.isFloatingPoint();
|
2008-06-06 20:08:01 +08:00
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
static inline bool isVector(MVT VT) {
|
|
|
|
return VT.isVector();
|
2008-06-06 20:08:01 +08:00
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
static inline bool isScalar(MVT VT) {
|
|
|
|
return !VT.isVector();
|
2010-03-20 01:41:26 +08:00
|
|
|
}
|
2008-06-06 20:08:01 +08:00
|
|
|
|
2017-09-20 02:42:34 +08:00
|
|
|
template <typename Predicate>
|
|
|
|
static bool berase_if(MachineValueTypeSet &S, Predicate P) {
|
2017-09-15 00:56:21 +08:00
|
|
|
bool Erased = false;
|
2017-09-20 02:42:34 +08:00
|
|
|
// It is ok to iterate over MachineValueTypeSet and remove elements from it
|
|
|
|
// at the same time.
|
|
|
|
for (MVT T : S) {
|
|
|
|
if (!P(T))
|
|
|
|
continue;
|
|
|
|
Erased = true;
|
|
|
|
S.erase(T);
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
return Erased;
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// --- TypeSetByHwMode
|
2010-03-15 14:00:16 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// This is a parameterized type-set class. For each mode there is a list
|
|
|
|
// of types that are currently possible for a given tree node. Type
|
|
|
|
// inference will apply to each mode separately.
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
TypeSetByHwMode::TypeSetByHwMode(ArrayRef<ValueTypeByHwMode> VTList) {
|
Add support for pointer types in patterns
Summary:
This adds support for defining patterns for global isel using pointer
types, for example:
def : Pat<(load GPR32:$src),
(p1 (LOAD GPR32:$src))>;
DAGISelEmitter will ignore the pointer information and treat these
types as integers with the same bit-width as the pointer type.
Reviewers: dsanders, rtereshin, arsenm
Reviewed By: arsenm
Subscribers: Petar.Avramovic, wdng, rovka, kristof.beyls, jfb, volkan, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D57065
llvm-svn: 354510
2019-02-21 03:43:47 +08:00
|
|
|
for (const ValueTypeByHwMode &VVT : VTList) {
|
2017-09-15 00:56:21 +08:00
|
|
|
insert(VVT);
|
Add support for pointer types in patterns
Summary:
This adds support for defining patterns for global isel using pointer
types, for example:
def : Pat<(load GPR32:$src),
(p1 (LOAD GPR32:$src))>;
DAGISelEmitter will ignore the pointer information and treat these
types as integers with the same bit-width as the pointer type.
Reviewers: dsanders, rtereshin, arsenm
Reviewed By: arsenm
Subscribers: Petar.Avramovic, wdng, rovka, kristof.beyls, jfb, volkan, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D57065
llvm-svn: 354510
2019-02-21 03:43:47 +08:00
|
|
|
AddrSpaces.push_back(VVT.PtrAddrSpace);
|
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeSetByHwMode::isValueTypeByHwMode(bool AllowEmpty) const {
|
|
|
|
for (const auto &I : *this) {
|
|
|
|
if (I.second.size() > 1)
|
|
|
|
return false;
|
|
|
|
if (!AllowEmpty && I.second.empty())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
ValueTypeByHwMode TypeSetByHwMode::getValueTypeByHwMode() const {
|
|
|
|
assert(isValueTypeByHwMode(true) &&
|
|
|
|
"The type set has multiple types for at least one HW mode");
|
|
|
|
ValueTypeByHwMode VVT;
|
Add support for pointer types in patterns
Summary:
This adds support for defining patterns for global isel using pointer
types, for example:
def : Pat<(load GPR32:$src),
(p1 (LOAD GPR32:$src))>;
DAGISelEmitter will ignore the pointer information and treat these
types as integers with the same bit-width as the pointer type.
Reviewers: dsanders, rtereshin, arsenm
Reviewed By: arsenm
Subscribers: Petar.Avramovic, wdng, rovka, kristof.beyls, jfb, volkan, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D57065
llvm-svn: 354510
2019-02-21 03:43:47 +08:00
|
|
|
auto ASI = AddrSpaces.begin();
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
for (const auto &I : *this) {
|
|
|
|
MVT T = I.second.empty() ? MVT::Other : *I.second.begin();
|
|
|
|
VVT.getOrCreateTypeForMode(I.first, T);
|
Add support for pointer types in patterns
Summary:
This adds support for defining patterns for global isel using pointer
types, for example:
def : Pat<(load GPR32:$src),
(p1 (LOAD GPR32:$src))>;
DAGISelEmitter will ignore the pointer information and treat these
types as integers with the same bit-width as the pointer type.
Reviewers: dsanders, rtereshin, arsenm
Reviewed By: arsenm
Subscribers: Petar.Avramovic, wdng, rovka, kristof.beyls, jfb, volkan, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D57065
llvm-svn: 354510
2019-02-21 03:43:47 +08:00
|
|
|
if (ASI != AddrSpaces.end())
|
|
|
|
VVT.PtrAddrSpace = *ASI++;
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
|
|
|
return VVT;
|
|
|
|
}
|
2012-10-26 04:33:17 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeSetByHwMode::isPossible() const {
|
|
|
|
for (const auto &I : *this)
|
|
|
|
if (!I.second.empty())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
2010-03-20 01:41:26 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeSetByHwMode::insert(const ValueTypeByHwMode &VVT) {
|
|
|
|
bool Changed = false;
|
2018-08-17 21:03:17 +08:00
|
|
|
bool ContainsDefault = false;
|
|
|
|
MVT DT = MVT::Other;
|
|
|
|
|
2017-09-21 02:01:40 +08:00
|
|
|
SmallDenseSet<unsigned, 4> Modes;
|
2017-09-15 00:56:21 +08:00
|
|
|
for (const auto &P : VVT) {
|
|
|
|
unsigned M = P.first;
|
|
|
|
Modes.insert(M);
|
|
|
|
// Make sure there exists a set for each specific mode from VVT.
|
|
|
|
Changed |= getOrCreate(M).insert(P.second).second;
|
2018-08-17 21:03:17 +08:00
|
|
|
// Cache VVT's default mode.
|
|
|
|
if (DefaultMode == M) {
|
|
|
|
ContainsDefault = true;
|
|
|
|
DT = P.second;
|
|
|
|
}
|
2012-10-26 04:33:17 +08:00
|
|
|
}
|
2010-03-20 01:41:26 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// If VVT has a default mode, add the corresponding type to all
|
|
|
|
// modes in "this" that do not exist in VVT.
|
2018-08-17 21:03:17 +08:00
|
|
|
if (ContainsDefault)
|
2017-09-15 00:56:21 +08:00
|
|
|
for (auto &I : *this)
|
|
|
|
if (!Modes.count(I.first))
|
|
|
|
Changed |= I.second.insert(DT).second;
|
2018-08-17 21:03:17 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
return Changed;
|
2010-03-19 12:54:36 +08:00
|
|
|
}
|
2010-03-15 14:00:16 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// Constrain the type set to be the intersection with VTS.
|
|
|
|
bool TypeSetByHwMode::constrain(const TypeSetByHwMode &VTS) {
|
|
|
|
bool Changed = false;
|
|
|
|
if (hasDefault()) {
|
|
|
|
for (const auto &I : VTS) {
|
|
|
|
unsigned M = I.first;
|
|
|
|
if (M == DefaultMode || hasMode(M))
|
|
|
|
continue;
|
2017-09-20 02:42:34 +08:00
|
|
|
Map.insert({M, Map.at(DefaultMode)});
|
2017-09-15 00:56:21 +08:00
|
|
|
Changed = true;
|
|
|
|
}
|
|
|
|
}
|
2010-03-15 14:00:16 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
for (auto &I : *this) {
|
|
|
|
unsigned M = I.first;
|
|
|
|
SetType &S = I.second;
|
|
|
|
if (VTS.hasMode(M) || VTS.hasDefault()) {
|
|
|
|
Changed |= intersect(I.second, VTS.get(M));
|
|
|
|
} else if (!S.empty()) {
|
|
|
|
S.clear();
|
|
|
|
Changed = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Changed;
|
2010-12-22 00:16:00 +08:00
|
|
|
}
|
2010-03-15 14:00:16 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
template <typename Predicate>
|
|
|
|
bool TypeSetByHwMode::constrain(Predicate P) {
|
|
|
|
bool Changed = false;
|
|
|
|
for (auto &I : *this)
|
2017-09-17 19:19:53 +08:00
|
|
|
Changed |= berase_if(I.second, [&P](MVT VT) { return !P(VT); });
|
2017-09-15 00:56:21 +08:00
|
|
|
return Changed;
|
2014-01-28 12:49:01 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
template <typename Predicate>
|
|
|
|
bool TypeSetByHwMode::assign_if(const TypeSetByHwMode &VTS, Predicate P) {
|
|
|
|
assert(empty());
|
|
|
|
for (const auto &I : VTS) {
|
|
|
|
SetType &S = getOrCreate(I.first);
|
|
|
|
for (auto J : I.second)
|
|
|
|
if (P(J))
|
|
|
|
S.insert(J);
|
|
|
|
}
|
|
|
|
return !empty();
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2009-08-11 09:14:02 +08:00
|
|
|
|
2017-09-21 02:01:40 +08:00
|
|
|
void TypeSetByHwMode::writeToStream(raw_ostream &OS) const {
|
|
|
|
SmallVector<unsigned, 4> Modes;
|
|
|
|
Modes.reserve(Map.size());
|
2010-03-15 14:00:16 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
for (const auto &I : *this)
|
|
|
|
Modes.push_back(I.first);
|
2017-09-21 02:01:40 +08:00
|
|
|
if (Modes.empty()) {
|
|
|
|
OS << "{}";
|
|
|
|
return;
|
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
array_pod_sort(Modes.begin(), Modes.end());
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-21 02:01:40 +08:00
|
|
|
OS << '{';
|
2017-09-15 00:56:21 +08:00
|
|
|
for (unsigned M : Modes) {
|
2017-09-21 02:01:40 +08:00
|
|
|
OS << ' ' << getModeName(M) << ':';
|
|
|
|
writeToStream(get(M), OS);
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
2017-09-21 02:01:40 +08:00
|
|
|
OS << " }";
|
2009-08-11 09:14:02 +08:00
|
|
|
}
|
2010-03-15 14:00:16 +08:00
|
|
|
|
2017-09-21 02:01:40 +08:00
|
|
|
void TypeSetByHwMode::writeToStream(const SetType &S, raw_ostream &OS) {
|
|
|
|
SmallVector<MVT, 4> Types(S.begin(), S.end());
|
2017-09-15 00:56:21 +08:00
|
|
|
array_pod_sort(Types.begin(), Types.end());
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-21 02:01:40 +08:00
|
|
|
OS << '[';
|
2017-09-15 00:56:21 +08:00
|
|
|
for (unsigned i = 0, e = Types.size(); i != e; ++i) {
|
2017-09-21 02:01:40 +08:00
|
|
|
OS << ValueTypeByHwMode::getMVTName(Types[i]);
|
2017-09-15 00:56:21 +08:00
|
|
|
if (i != e-1)
|
2017-09-21 02:01:40 +08:00
|
|
|
OS << ' ';
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
2017-09-21 02:01:40 +08:00
|
|
|
OS << ']';
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeSetByHwMode::operator==(const TypeSetByHwMode &VTS) const {
|
2018-08-17 00:16:28 +08:00
|
|
|
// The isSimple call is much quicker than hasDefault - check this first.
|
|
|
|
bool IsSimple = isSimple();
|
|
|
|
bool VTSIsSimple = VTS.isSimple();
|
|
|
|
if (IsSimple && VTSIsSimple)
|
|
|
|
return *begin() == *VTS.begin();
|
|
|
|
|
|
|
|
// Speedup: We have a default if the set is simple.
|
|
|
|
bool HaveDefault = IsSimple || hasDefault();
|
|
|
|
bool VTSHaveDefault = VTSIsSimple || VTS.hasDefault();
|
|
|
|
if (HaveDefault != VTSHaveDefault)
|
2017-09-15 00:56:21 +08:00
|
|
|
return false;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-21 02:01:40 +08:00
|
|
|
SmallDenseSet<unsigned, 4> Modes;
|
2017-09-15 00:56:21 +08:00
|
|
|
for (auto &I : *this)
|
|
|
|
Modes.insert(I.first);
|
|
|
|
for (const auto &I : VTS)
|
|
|
|
Modes.insert(I.first);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
if (HaveDefault) {
|
|
|
|
// Both sets have default mode.
|
|
|
|
for (unsigned M : Modes) {
|
|
|
|
if (get(M) != VTS.get(M))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Neither set has default mode.
|
|
|
|
for (unsigned M : Modes) {
|
|
|
|
// If there is no default mode, an empty set is equivalent to not having
|
|
|
|
// the corresponding mode.
|
|
|
|
bool NoModeThis = !hasMode(M) || get(M).empty();
|
|
|
|
bool NoModeVTS = !VTS.hasMode(M) || VTS.get(M).empty();
|
|
|
|
if (NoModeThis != NoModeVTS)
|
2010-03-15 14:00:16 +08:00
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
if (!NoModeThis)
|
|
|
|
if (get(M) != VTS.get(M))
|
|
|
|
return false;
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-09-23 02:29:37 +08:00
|
|
|
namespace llvm {
|
|
|
|
raw_ostream &operator<<(raw_ostream &OS, const TypeSetByHwMode &T) {
|
|
|
|
T.writeToStream(OS);
|
|
|
|
return OS;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
LLVM_DUMP_METHOD
|
|
|
|
void TypeSetByHwMode::dump() const {
|
2017-09-23 02:29:37 +08:00
|
|
|
dbgs() << *this << '\n';
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool TypeSetByHwMode::intersect(SetType &Out, const SetType &In) {
|
|
|
|
bool OutP = Out.count(MVT::iPTR), InP = In.count(MVT::iPTR);
|
|
|
|
auto Int = [&In](MVT T) -> bool { return !In.count(T); };
|
|
|
|
|
|
|
|
if (OutP == InP)
|
|
|
|
return berase_if(Out, Int);
|
|
|
|
|
|
|
|
// Compute the intersection of scalars separately to account for only
|
|
|
|
// one set containing iPTR.
|
|
|
|
// The itersection of iPTR with a set of integer scalar types that does not
|
|
|
|
// include iPTR will result in the most specific scalar type:
|
|
|
|
// - iPTR is more specific than any set with two elements or more
|
|
|
|
// - iPTR is less specific than any single integer scalar type.
|
|
|
|
// For example
|
|
|
|
// { iPTR } * { i32 } -> { i32 }
|
|
|
|
// { iPTR } * { i32 i64 } -> { iPTR }
|
2017-09-20 02:42:34 +08:00
|
|
|
// and
|
|
|
|
// { iPTR i32 } * { i32 } -> { i32 }
|
|
|
|
// { iPTR i32 } * { i32 i64 } -> { i32 i64 }
|
|
|
|
// { iPTR i32 } * { i32 i64 i128 } -> { iPTR i32 }
|
|
|
|
|
|
|
|
// Compute the difference between the two sets in such a way that the
|
|
|
|
// iPTR is in the set that is being subtracted. This is to see if there
|
|
|
|
// are any extra scalars in the set without iPTR that are not in the
|
|
|
|
// set containing iPTR. Then the iPTR could be considered a "wildcard"
|
|
|
|
// matching these scalars. If there is only one such scalar, it would
|
|
|
|
// replace the iPTR, if there are more, the iPTR would be retained.
|
2017-09-15 00:56:21 +08:00
|
|
|
SetType Diff;
|
|
|
|
if (InP) {
|
2017-09-20 02:42:34 +08:00
|
|
|
Diff = Out;
|
|
|
|
berase_if(Diff, [&In](MVT T) { return In.count(T); });
|
|
|
|
// Pre-remove these elements and rely only on InP/OutP to determine
|
|
|
|
// whether a change has been made.
|
2017-09-15 00:56:21 +08:00
|
|
|
berase_if(Out, [&Diff](MVT T) { return Diff.count(T); });
|
|
|
|
} else {
|
2017-09-20 02:42:34 +08:00
|
|
|
Diff = In;
|
|
|
|
berase_if(Diff, [&Out](MVT T) { return Out.count(T); });
|
2017-09-15 00:56:21 +08:00
|
|
|
Out.erase(MVT::iPTR);
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-20 02:42:34 +08:00
|
|
|
// The actual intersection.
|
2017-09-15 00:56:21 +08:00
|
|
|
bool Changed = berase_if(Out, Int);
|
|
|
|
unsigned NumD = Diff.size();
|
|
|
|
if (NumD == 0)
|
|
|
|
return Changed;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
if (NumD == 1) {
|
|
|
|
Out.insert(*Diff.begin());
|
|
|
|
// This is a change only if Out was the one with iPTR (which is now
|
|
|
|
// being replaced).
|
|
|
|
Changed |= OutP;
|
|
|
|
} else {
|
2017-09-20 02:42:34 +08:00
|
|
|
// Multiple elements from Out are now replaced with iPTR.
|
2017-09-15 00:56:21 +08:00
|
|
|
Out.insert(MVT::iPTR);
|
2017-09-20 02:42:34 +08:00
|
|
|
Changed |= !OutP;
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
return Changed;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-12-22 01:12:43 +08:00
|
|
|
bool TypeSetByHwMode::validate() const {
|
2017-09-15 00:56:21 +08:00
|
|
|
#ifndef NDEBUG
|
|
|
|
if (empty())
|
2017-12-22 01:12:43 +08:00
|
|
|
return true;
|
2017-09-15 00:56:21 +08:00
|
|
|
bool AllEmpty = true;
|
|
|
|
for (const auto &I : *this)
|
|
|
|
AllEmpty &= I.second.empty();
|
2017-12-22 01:12:43 +08:00
|
|
|
return !AllEmpty;
|
2017-09-15 00:56:21 +08:00
|
|
|
#endif
|
2017-12-22 01:12:43 +08:00
|
|
|
return true;
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
2010-03-15 14:00:16 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// --- TypeInfer
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeInfer::MergeInTypeInfo(TypeSetByHwMode &Out,
|
|
|
|
const TypeSetByHwMode &In) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Out, *this);
|
2017-09-15 00:56:21 +08:00
|
|
|
In.validate();
|
|
|
|
if (In.empty() || Out == In || TP.hasError())
|
2015-11-24 16:20:42 +08:00
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
if (Out.empty()) {
|
|
|
|
Out = In;
|
2015-11-24 16:20:42 +08:00
|
|
|
return true;
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool Changed = Out.constrain(In);
|
|
|
|
if (Changed && Out.empty())
|
|
|
|
TP.error("Type contradiction");
|
|
|
|
|
|
|
|
return Changed;
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeInfer::forceArbitrary(TypeSetByHwMode &Out) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Out, *this);
|
2012-10-26 04:33:17 +08:00
|
|
|
if (TP.hasError())
|
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
assert(!Out.empty() && "cannot pick from an empty set");
|
2015-11-23 15:19:08 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool Changed = false;
|
|
|
|
for (auto &I : Out) {
|
|
|
|
TypeSetByHwMode::SetType &S = I.second;
|
|
|
|
if (S.size() <= 1)
|
|
|
|
continue;
|
|
|
|
MVT T = *S.begin(); // Pick the first element.
|
|
|
|
S.clear();
|
|
|
|
S.insert(T);
|
|
|
|
Changed = true;
|
2012-10-26 04:33:17 +08:00
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
return Changed;
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeInfer::EnforceInteger(TypeSetByHwMode &Out) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Out, *this);
|
2012-10-26 04:33:17 +08:00
|
|
|
if (TP.hasError())
|
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
if (!Out.empty())
|
|
|
|
return Out.constrain(isIntegerOrPtr);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
return Out.assign_if(getLegalTypes(), isIntegerOrPtr);
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeInfer::EnforceFloatingPoint(TypeSetByHwMode &Out) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Out, *this);
|
2012-10-26 04:33:17 +08:00
|
|
|
if (TP.hasError())
|
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
if (!Out.empty())
|
|
|
|
return Out.constrain(isFloatingPoint);
|
2012-10-26 04:33:17 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
return Out.assign_if(getLegalTypes(), isFloatingPoint);
|
|
|
|
}
|
2010-03-20 01:41:26 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeInfer::EnforceScalar(TypeSetByHwMode &Out) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Out, *this);
|
2017-09-15 00:56:21 +08:00
|
|
|
if (TP.hasError())
|
2010-03-20 01:41:26 +08:00
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
if (!Out.empty())
|
|
|
|
return Out.constrain(isScalar);
|
2010-03-20 01:41:26 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
return Out.assign_if(getLegalTypes(), isScalar);
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeInfer::EnforceVector(TypeSetByHwMode &Out) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Out, *this);
|
2012-10-26 04:33:17 +08:00
|
|
|
if (TP.hasError())
|
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
if (!Out.empty())
|
|
|
|
return Out.constrain(isVector);
|
2012-10-26 04:33:17 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
return Out.assign_if(getLegalTypes(), isVector);
|
|
|
|
}
|
2010-03-20 01:41:26 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeInfer::EnforceAny(TypeSetByHwMode &Out) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Out, *this);
|
2017-09-15 00:56:21 +08:00
|
|
|
if (TP.hasError() || !Out.empty())
|
|
|
|
return false;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
Out = getLegalTypes();
|
|
|
|
return true;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
template <typename Iter, typename Pred, typename Less>
|
|
|
|
static Iter min_if(Iter B, Iter E, Pred P, Less L) {
|
|
|
|
if (B == E)
|
|
|
|
return E;
|
|
|
|
Iter Min = E;
|
|
|
|
for (Iter I = B; I != E; ++I) {
|
|
|
|
if (!P(*I))
|
|
|
|
continue;
|
|
|
|
if (Min == E || L(*I, *Min))
|
|
|
|
Min = I;
|
2012-10-26 04:33:17 +08:00
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
return Min;
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
template <typename Iter, typename Pred, typename Less>
|
|
|
|
static Iter max_if(Iter B, Iter E, Pred P, Less L) {
|
|
|
|
if (B == E)
|
|
|
|
return E;
|
|
|
|
Iter Max = E;
|
|
|
|
for (Iter I = B; I != E; ++I) {
|
|
|
|
if (!P(*I))
|
|
|
|
continue;
|
|
|
|
if (Max == E || L(*Max, *I))
|
|
|
|
Max = I;
|
|
|
|
}
|
|
|
|
return Max;
|
|
|
|
}
|
2010-03-15 14:00:16 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
/// Make sure that for each type in Small, there exists a larger type in Big.
|
|
|
|
bool TypeInfer::EnforceSmallerThan(TypeSetByHwMode &Small,
|
|
|
|
TypeSetByHwMode &Big) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Small, *this), _2(Big, *this);
|
2014-01-28 12:49:01 +08:00
|
|
|
if (TP.hasError())
|
2012-10-26 04:33:17 +08:00
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
bool Changed = false;
|
|
|
|
|
|
|
|
if (Small.empty())
|
|
|
|
Changed |= EnforceAny(Small);
|
|
|
|
if (Big.empty())
|
|
|
|
Changed |= EnforceAny(Big);
|
|
|
|
|
|
|
|
assert(Small.hasDefault() && Big.hasDefault());
|
|
|
|
|
|
|
|
std::vector<unsigned> Modes = union_modes(Small, Big);
|
|
|
|
|
|
|
|
// 1. Only allow integer or floating point types and make sure that
|
|
|
|
// both sides are both integer or both floating point.
|
|
|
|
// 2. Make sure that either both sides have vector types, or neither
|
|
|
|
// of them does.
|
|
|
|
for (unsigned M : Modes) {
|
|
|
|
TypeSetByHwMode::SetType &S = Small.get(M);
|
|
|
|
TypeSetByHwMode::SetType &B = Big.get(M);
|
|
|
|
|
2019-11-03 02:02:33 +08:00
|
|
|
if (any_of(S, isIntegerOrPtr) && any_of(S, isIntegerOrPtr)) {
|
2017-09-17 19:19:53 +08:00
|
|
|
auto NotInt = [](MVT VT) { return !isIntegerOrPtr(VT); };
|
2019-11-03 06:38:07 +08:00
|
|
|
Changed |= berase_if(S, NotInt);
|
|
|
|
Changed |= berase_if(B, NotInt);
|
2017-09-15 00:56:21 +08:00
|
|
|
} else if (any_of(S, isFloatingPoint) && any_of(B, isFloatingPoint)) {
|
2017-09-17 19:19:53 +08:00
|
|
|
auto NotFP = [](MVT VT) { return !isFloatingPoint(VT); };
|
2019-11-03 06:38:07 +08:00
|
|
|
Changed |= berase_if(S, NotFP);
|
|
|
|
Changed |= berase_if(B, NotFP);
|
2017-09-15 00:56:21 +08:00
|
|
|
} else if (S.empty() || B.empty()) {
|
|
|
|
Changed = !S.empty() || !B.empty();
|
|
|
|
S.clear();
|
|
|
|
B.clear();
|
|
|
|
} else {
|
|
|
|
TP.error("Incompatible types");
|
|
|
|
return Changed;
|
2011-02-02 03:12:32 +08:00
|
|
|
}
|
2014-01-28 12:49:01 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
if (none_of(S, isVector) || none_of(B, isVector)) {
|
2019-11-03 06:38:07 +08:00
|
|
|
Changed |= berase_if(S, isVector);
|
|
|
|
Changed |= berase_if(B, isVector);
|
2011-02-02 03:12:32 +08:00
|
|
|
}
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
auto LT = [](MVT A, MVT B) -> bool {
|
2019-08-14 18:48:39 +08:00
|
|
|
// Always treat non-scalable MVTs as smaller than scalable MVTs for the
|
|
|
|
// purposes of ordering.
|
|
|
|
auto ASize = std::make_tuple(A.isScalableVector(), A.getScalarSizeInBits(),
|
|
|
|
A.getSizeInBits());
|
|
|
|
auto BSize = std::make_tuple(B.isScalableVector(), B.getScalarSizeInBits(),
|
|
|
|
B.getSizeInBits());
|
|
|
|
return ASize < BSize;
|
2017-09-15 00:56:21 +08:00
|
|
|
};
|
2019-08-14 18:48:39 +08:00
|
|
|
auto SameKindLE = [](MVT A, MVT B) -> bool {
|
2017-09-15 00:56:21 +08:00
|
|
|
// This function is used when removing elements: when a vector is compared
|
2019-08-14 18:48:39 +08:00
|
|
|
// to a non-vector or a scalable vector to any non-scalable MVT, it should
|
|
|
|
// return false (to avoid removal).
|
|
|
|
if (std::make_tuple(A.isVector(), A.isScalableVector()) !=
|
|
|
|
std::make_tuple(B.isVector(), B.isScalableVector()))
|
2017-09-15 00:56:21 +08:00
|
|
|
return false;
|
2015-03-05 15:11:34 +08:00
|
|
|
|
2019-08-14 18:48:39 +08:00
|
|
|
return std::make_tuple(A.getScalarSizeInBits(), A.getSizeInBits()) <=
|
|
|
|
std::make_tuple(B.getScalarSizeInBits(), B.getSizeInBits());
|
2017-09-15 00:56:21 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
for (unsigned M : Modes) {
|
|
|
|
TypeSetByHwMode::SetType &S = Small.get(M);
|
|
|
|
TypeSetByHwMode::SetType &B = Big.get(M);
|
|
|
|
// MinS = min scalar in Small, remove all scalars from Big that are
|
|
|
|
// smaller-or-equal than MinS.
|
|
|
|
auto MinS = min_if(S.begin(), S.end(), isScalar, LT);
|
2017-10-15 23:39:56 +08:00
|
|
|
if (MinS != S.end())
|
2019-08-14 18:48:39 +08:00
|
|
|
Changed |= berase_if(B, std::bind(SameKindLE,
|
|
|
|
std::placeholders::_1, *MinS));
|
2017-10-15 23:39:56 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// MaxS = max scalar in Big, remove all scalars from Small that are
|
|
|
|
// larger than MaxS.
|
|
|
|
auto MaxS = max_if(B.begin(), B.end(), isScalar, LT);
|
2017-10-15 23:39:56 +08:00
|
|
|
if (MaxS != B.end())
|
2019-08-14 18:48:39 +08:00
|
|
|
Changed |= berase_if(S, std::bind(SameKindLE,
|
|
|
|
*MaxS, std::placeholders::_1));
|
2015-03-05 15:11:34 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// MinV = min vector in Small, remove all vectors from Big that are
|
|
|
|
// smaller-or-equal than MinV.
|
|
|
|
auto MinV = min_if(S.begin(), S.end(), isVector, LT);
|
2017-10-15 23:39:56 +08:00
|
|
|
if (MinV != S.end())
|
2019-08-14 18:48:39 +08:00
|
|
|
Changed |= berase_if(B, std::bind(SameKindLE,
|
|
|
|
std::placeholders::_1, *MinV));
|
2017-10-15 23:39:56 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// MaxV = max vector in Big, remove all vectors from Small that are
|
|
|
|
// larger than MaxV.
|
|
|
|
auto MaxV = max_if(B.begin(), B.end(), isVector, LT);
|
2017-10-15 23:39:56 +08:00
|
|
|
if (MaxV != B.end())
|
2019-08-14 18:48:39 +08:00
|
|
|
Changed |= berase_if(S, std::bind(SameKindLE,
|
|
|
|
*MaxV, std::placeholders::_1));
|
2015-03-05 15:11:34 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
return Changed;
|
2015-03-05 15:11:34 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
/// 1. Ensure that for each type T in Vec, T is a vector type, and that
|
|
|
|
/// for each type U in Elem, U is a scalar type.
|
|
|
|
/// 2. Ensure that for each (scalar) type U in Elem, there exists a (vector)
|
|
|
|
/// type T in Vec, such that U is the element type of T.
|
|
|
|
bool TypeInfer::EnforceVectorEltTypeIs(TypeSetByHwMode &Vec,
|
|
|
|
TypeSetByHwMode &Elem) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Vec, *this), _2(Elem, *this);
|
2012-10-26 04:33:17 +08:00
|
|
|
if (TP.hasError())
|
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
bool Changed = false;
|
|
|
|
|
|
|
|
if (Vec.empty())
|
|
|
|
Changed |= EnforceVector(Vec);
|
|
|
|
if (Elem.empty())
|
|
|
|
Changed |= EnforceScalar(Elem);
|
|
|
|
|
|
|
|
for (unsigned M : union_modes(Vec, Elem)) {
|
|
|
|
TypeSetByHwMode::SetType &V = Vec.get(M);
|
|
|
|
TypeSetByHwMode::SetType &E = Elem.get(M);
|
|
|
|
|
|
|
|
Changed |= berase_if(V, isScalar); // Scalar = !vector
|
|
|
|
Changed |= berase_if(E, isVector); // Vector = !scalar
|
|
|
|
assert(!V.empty() && !E.empty());
|
|
|
|
|
|
|
|
SmallSet<MVT,4> VT, ST;
|
|
|
|
// Collect element types from the "vector" set.
|
|
|
|
for (MVT T : V)
|
|
|
|
VT.insert(T.getVectorElementType());
|
|
|
|
// Collect scalar types from the "element" set.
|
|
|
|
for (MVT T : E)
|
|
|
|
ST.insert(T);
|
|
|
|
|
|
|
|
// Remove from V all (vector) types whose element type is not in S.
|
|
|
|
Changed |= berase_if(V, [&ST](MVT T) -> bool {
|
|
|
|
return !ST.count(T.getVectorElementType());
|
|
|
|
});
|
|
|
|
// Remove from E all (scalar) types, for which there is no corresponding
|
|
|
|
// type in V.
|
|
|
|
Changed |= berase_if(E, [&VT](MVT T) -> bool { return !VT.count(T); });
|
2010-03-24 08:01:16 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
return Changed;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TypeInfer::EnforceVectorEltTypeIs(TypeSetByHwMode &Vec,
|
|
|
|
const ValueTypeByHwMode &VVT) {
|
|
|
|
TypeSetByHwMode Tmp(VVT);
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Vec, *this), _2(Tmp, *this);
|
2017-09-15 00:56:21 +08:00
|
|
|
return EnforceVectorEltTypeIs(Vec, Tmp);
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
/// Ensure that for each type T in Sub, T is a vector type, and there
|
|
|
|
/// exists a type U in Vec such that U is a vector type with the same
|
|
|
|
/// element type as T and at least as many elements as T.
|
|
|
|
bool TypeInfer::EnforceVectorSubVectorTypeIs(TypeSetByHwMode &Vec,
|
|
|
|
TypeSetByHwMode &Sub) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(Vec, *this), _2(Sub, *this);
|
2014-01-26 01:40:33 +08:00
|
|
|
if (TP.hasError())
|
|
|
|
return false;
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
/// Return true if B is a suB-vector of P, i.e. P is a suPer-vector of B.
|
|
|
|
auto IsSubVec = [](MVT B, MVT P) -> bool {
|
|
|
|
if (!B.isVector() || !P.isVector())
|
2014-01-26 01:40:33 +08:00
|
|
|
return false;
|
2017-11-07 18:43:56 +08:00
|
|
|
// Logically a <4 x i32> is a valid subvector of <n x 4 x i32>
|
|
|
|
// but until there are obvious use-cases for this, keep the
|
|
|
|
// types separate.
|
|
|
|
if (B.isScalableVector() != P.isScalableVector())
|
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
if (B.getVectorElementType() != P.getVectorElementType())
|
2014-01-26 01:40:33 +08:00
|
|
|
return false;
|
2017-09-15 00:56:21 +08:00
|
|
|
return B.getVectorNumElements() < P.getVectorNumElements();
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Return true if S has no element (vector type) that T is a sub-vector of,
|
|
|
|
/// i.e. has the same element type as T and more elements.
|
|
|
|
auto NoSubV = [&IsSubVec](const TypeSetByHwMode::SetType &S, MVT T) -> bool {
|
2019-12-23 01:58:32 +08:00
|
|
|
for (auto I : S)
|
2017-09-15 00:56:21 +08:00
|
|
|
if (IsSubVec(T, I))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
};
|
2011-01-25 04:53:18 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
/// Return true if S has no element (vector type) that T is a super-vector
|
|
|
|
/// of, i.e. has the same element type as T and fewer elements.
|
|
|
|
auto NoSupV = [&IsSubVec](const TypeSetByHwMode::SetType &S, MVT T) -> bool {
|
2019-12-23 01:58:32 +08:00
|
|
|
for (auto I : S)
|
2017-09-15 00:56:21 +08:00
|
|
|
if (IsSubVec(I, T))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
};
|
2011-01-25 04:53:18 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool Changed = false;
|
2015-03-05 15:11:34 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
if (Vec.empty())
|
|
|
|
Changed |= EnforceVector(Vec);
|
|
|
|
if (Sub.empty())
|
|
|
|
Changed |= EnforceVector(Sub);
|
2015-03-05 15:11:34 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
for (unsigned M : union_modes(Vec, Sub)) {
|
|
|
|
TypeSetByHwMode::SetType &S = Sub.get(M);
|
|
|
|
TypeSetByHwMode::SetType &V = Vec.get(M);
|
2015-03-05 15:11:34 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
Changed |= berase_if(S, isScalar);
|
2015-03-05 15:11:34 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// Erase all types from S that are not sub-vectors of a type in V.
|
|
|
|
Changed |= berase_if(S, std::bind(NoSubV, V, std::placeholders::_1));
|
2015-11-24 16:20:47 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// Erase all types from V that are not super-vectors of a type in S.
|
|
|
|
Changed |= berase_if(V, std::bind(NoSupV, S, std::placeholders::_1));
|
2015-03-05 15:11:34 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
return Changed;
|
2015-03-05 15:11:34 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
/// 1. Ensure that V has a scalar type iff W has a scalar type.
|
|
|
|
/// 2. Ensure that for each vector type T in V, there exists a vector
|
|
|
|
/// type U in W, such that T and U have the same number of elements.
|
|
|
|
/// 3. Ensure that for each vector type U in W, there exists a vector
|
|
|
|
/// type T in V, such that T and U have the same number of elements
|
|
|
|
/// (reverse of 2).
|
|
|
|
bool TypeInfer::EnforceSameNumElts(TypeSetByHwMode &V, TypeSetByHwMode &W) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(V, *this), _2(W, *this);
|
2015-11-26 15:02:18 +08:00
|
|
|
if (TP.hasError())
|
|
|
|
return false;
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool Changed = false;
|
|
|
|
if (V.empty())
|
|
|
|
Changed |= EnforceAny(V);
|
|
|
|
if (W.empty())
|
|
|
|
Changed |= EnforceAny(W);
|
|
|
|
|
|
|
|
// An actual vector type cannot have 0 elements, so we can treat scalars
|
|
|
|
// as zero-length vectors. This way both vectors and scalars can be
|
|
|
|
// processed identically.
|
|
|
|
auto NoLength = [](const SmallSet<unsigned,2> &Lengths, MVT T) -> bool {
|
|
|
|
return !Lengths.count(T.isVector() ? T.getVectorNumElements() : 0);
|
|
|
|
};
|
|
|
|
|
|
|
|
for (unsigned M : union_modes(V, W)) {
|
|
|
|
TypeSetByHwMode::SetType &VS = V.get(M);
|
|
|
|
TypeSetByHwMode::SetType &WS = W.get(M);
|
|
|
|
|
|
|
|
SmallSet<unsigned,2> VN, WN;
|
|
|
|
for (MVT T : VS)
|
|
|
|
VN.insert(T.isVector() ? T.getVectorNumElements() : 0);
|
|
|
|
for (MVT T : WS)
|
|
|
|
WN.insert(T.isVector() ? T.getVectorNumElements() : 0);
|
|
|
|
|
|
|
|
Changed |= berase_if(VS, std::bind(NoLength, WN, std::placeholders::_1));
|
|
|
|
Changed |= berase_if(WS, std::bind(NoLength, VN, std::placeholders::_1));
|
|
|
|
}
|
|
|
|
return Changed;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 1. Ensure that for each type T in A, there exists a type U in B,
|
|
|
|
/// such that T and U have equal size in bits.
|
|
|
|
/// 2. Ensure that for each type U in B, there exists a type T in A
|
|
|
|
/// such that T and U have equal size in bits (reverse of 1).
|
|
|
|
bool TypeInfer::EnforceSameSize(TypeSetByHwMode &A, TypeSetByHwMode &B) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(A, *this), _2(B, *this);
|
2017-09-15 00:56:21 +08:00
|
|
|
if (TP.hasError())
|
|
|
|
return false;
|
|
|
|
bool Changed = false;
|
|
|
|
if (A.empty())
|
|
|
|
Changed |= EnforceAny(A);
|
|
|
|
if (B.empty())
|
|
|
|
Changed |= EnforceAny(B);
|
2015-11-26 15:02:18 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
auto NoSize = [](const SmallSet<unsigned,2> &Sizes, MVT T) -> bool {
|
|
|
|
return !Sizes.count(T.getSizeInBits());
|
|
|
|
};
|
2015-11-26 15:02:18 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
for (unsigned M : union_modes(A, B)) {
|
|
|
|
TypeSetByHwMode::SetType &AS = A.get(M);
|
|
|
|
TypeSetByHwMode::SetType &BS = B.get(M);
|
|
|
|
SmallSet<unsigned,2> AN, BN;
|
2015-11-26 15:02:18 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
for (MVT T : AS)
|
|
|
|
AN.insert(T.getSizeInBits());
|
|
|
|
for (MVT T : BS)
|
|
|
|
BN.insert(T.getSizeInBits());
|
2015-11-26 15:02:18 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
Changed |= berase_if(AS, std::bind(NoSize, BN, std::placeholders::_1));
|
|
|
|
Changed |= berase_if(BS, std::bind(NoSize, AN, std::placeholders::_1));
|
2015-11-26 15:02:18 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
return Changed;
|
2015-11-26 15:02:18 +08:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
void TypeInfer::expandOverloads(TypeSetByHwMode &VTS) {
|
2017-12-22 01:12:43 +08:00
|
|
|
ValidateOnExit _1(VTS, *this);
|
2018-08-17 23:54:07 +08:00
|
|
|
const TypeSetByHwMode &Legal = getLegalTypes();
|
|
|
|
assert(Legal.isDefaultOnly() && "Default-mode only expected");
|
|
|
|
const TypeSetByHwMode::SetType &LegalTypes = Legal.get(DefaultMode);
|
2008-03-06 01:49:05 +08:00
|
|
|
|
2018-08-17 23:54:07 +08:00
|
|
|
for (auto &I : VTS)
|
|
|
|
expandOverloads(I.second, LegalTypes);
|
2008-03-06 01:49:05 +08:00
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
|
|
|
|
void TypeInfer::expandOverloads(TypeSetByHwMode::SetType &Out,
|
|
|
|
const TypeSetByHwMode::SetType &Legal) {
|
|
|
|
std::set<MVT> Ovs;
|
2017-09-20 02:42:34 +08:00
|
|
|
for (MVT T : Out) {
|
|
|
|
if (!T.isOverloaded())
|
2017-09-15 00:56:21 +08:00
|
|
|
continue;
|
2017-09-21 02:01:40 +08:00
|
|
|
|
2017-09-20 02:42:34 +08:00
|
|
|
Ovs.insert(T);
|
|
|
|
// MachineValueTypeSet allows iteration and erasing.
|
|
|
|
Out.erase(T);
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
for (MVT Ov : Ovs) {
|
|
|
|
switch (Ov.SimpleTy) {
|
|
|
|
case MVT::iPTRAny:
|
|
|
|
Out.insert(MVT::iPTR);
|
|
|
|
return;
|
|
|
|
case MVT::iAny:
|
|
|
|
for (MVT T : MVT::integer_valuetypes())
|
|
|
|
if (Legal.count(T))
|
|
|
|
Out.insert(T);
|
2019-09-17 18:19:23 +08:00
|
|
|
for (MVT T : MVT::integer_fixedlen_vector_valuetypes())
|
|
|
|
if (Legal.count(T))
|
|
|
|
Out.insert(T);
|
|
|
|
for (MVT T : MVT::integer_scalable_vector_valuetypes())
|
2017-09-15 00:56:21 +08:00
|
|
|
if (Legal.count(T))
|
|
|
|
Out.insert(T);
|
|
|
|
return;
|
|
|
|
case MVT::fAny:
|
|
|
|
for (MVT T : MVT::fp_valuetypes())
|
|
|
|
if (Legal.count(T))
|
|
|
|
Out.insert(T);
|
2019-09-17 18:19:23 +08:00
|
|
|
for (MVT T : MVT::fp_fixedlen_vector_valuetypes())
|
|
|
|
if (Legal.count(T))
|
|
|
|
Out.insert(T);
|
|
|
|
for (MVT T : MVT::fp_scalable_vector_valuetypes())
|
2017-09-15 00:56:21 +08:00
|
|
|
if (Legal.count(T))
|
|
|
|
Out.insert(T);
|
|
|
|
return;
|
|
|
|
case MVT::vAny:
|
|
|
|
for (MVT T : MVT::vector_valuetypes())
|
|
|
|
if (Legal.count(T))
|
|
|
|
Out.insert(T);
|
|
|
|
return;
|
|
|
|
case MVT::Any:
|
|
|
|
for (MVT T : MVT::all_valuetypes())
|
|
|
|
if (Legal.count(T))
|
|
|
|
Out.insert(T);
|
|
|
|
return;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2008-03-06 01:49:05 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-17 23:54:07 +08:00
|
|
|
const TypeSetByHwMode &TypeInfer::getLegalTypes() {
|
2017-09-20 02:42:34 +08:00
|
|
|
if (!LegalTypesCached) {
|
2018-08-17 23:54:07 +08:00
|
|
|
TypeSetByHwMode::SetType &LegalTypes = LegalCache.getOrCreate(DefaultMode);
|
2017-09-20 02:42:34 +08:00
|
|
|
// Stuff all types from all modes into the default mode.
|
|
|
|
const TypeSetByHwMode <S = TP.getDAGPatterns().getLegalTypes();
|
|
|
|
for (const auto &I : LTS)
|
2018-08-17 23:54:07 +08:00
|
|
|
LegalTypes.insert(I.second);
|
2017-09-20 02:42:34 +08:00
|
|
|
LegalTypesCached = true;
|
|
|
|
}
|
2018-08-17 23:54:07 +08:00
|
|
|
assert(LegalCache.isDefaultOnly() && "Default-mode only expected");
|
|
|
|
return LegalCache;
|
2008-03-06 01:49:05 +08:00
|
|
|
}
|
2011-04-18 05:38:24 +08:00
|
|
|
|
2017-12-22 01:12:43 +08:00
|
|
|
#ifndef NDEBUG
|
|
|
|
TypeInfer::ValidateOnExit::~ValidateOnExit() {
|
2018-07-14 00:42:15 +08:00
|
|
|
if (Infer.Validate && !VTS.validate()) {
|
2017-12-22 01:12:43 +08:00
|
|
|
dbgs() << "Type set is empty for each HW mode:\n"
|
|
|
|
"possible type contradiction in the pattern below "
|
|
|
|
"(use -print-records with llvm-tblgen to see all "
|
|
|
|
"expanded records).\n";
|
|
|
|
Infer.TP.dump();
|
|
|
|
llvm_unreachable(nullptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// ScopedName Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
bool ScopedName::operator==(const ScopedName &o) const {
|
|
|
|
return Scope == o.Scope && Identifier == o.Identifier;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ScopedName::operator!=(const ScopedName &o) const {
|
|
|
|
return !(*this == o);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-04-18 05:38:24 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// TreePredicateFn Implementation
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2011-04-18 06:05:17 +08:00
|
|
|
/// TreePredicateFn constructor. Here 'N' is a subclass of PatFrag.
|
|
|
|
TreePredicateFn::TreePredicateFn(TreePattern *N) : PatFragRec(N) {
|
2017-10-16 03:01:32 +08:00
|
|
|
assert(
|
|
|
|
(!hasPredCode() || !hasImmCode()) &&
|
|
|
|
".td file corrupt: can't have a node predicate *and* an imm predicate");
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TreePredicateFn::hasPredCode() const {
|
2017-11-14 06:26:13 +08:00
|
|
|
return isLoad() || isStore() || isAtomic() ||
|
2017-10-16 03:01:32 +08:00
|
|
|
!PatFragRec->getRecord()->getValueAsString("PredicateCode").empty();
|
2011-04-18 06:05:17 +08:00
|
|
|
}
|
|
|
|
|
2017-10-15 10:06:44 +08:00
|
|
|
std::string TreePredicateFn::getPredCode() const {
|
|
|
|
std::string Code = "";
|
|
|
|
|
2017-11-14 06:26:13 +08:00
|
|
|
if (!isLoad() && !isStore() && !isAtomic()) {
|
|
|
|
Record *MemoryVT = getMemoryVT();
|
|
|
|
|
|
|
|
if (MemoryVT)
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"MemoryVT requires IsLoad or IsStore");
|
|
|
|
}
|
|
|
|
|
2017-10-15 10:06:44 +08:00
|
|
|
if (!isLoad() && !isStore()) {
|
|
|
|
if (isUnindexed())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsUnindexed requires IsLoad or IsStore");
|
|
|
|
|
|
|
|
Record *ScalarMemoryVT = getScalarMemoryVT();
|
|
|
|
|
|
|
|
if (ScalarMemoryVT)
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"ScalarMemoryVT requires IsLoad or IsStore");
|
|
|
|
}
|
|
|
|
|
2017-11-14 06:26:13 +08:00
|
|
|
if (isLoad() + isStore() + isAtomic() > 1)
|
2017-10-15 10:06:44 +08:00
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
2017-11-14 06:26:13 +08:00
|
|
|
"IsLoad, IsStore, and IsAtomic are mutually exclusive");
|
2017-10-15 10:06:44 +08:00
|
|
|
|
|
|
|
if (isLoad()) {
|
|
|
|
if (!isUnindexed() && !isNonExtLoad() && !isAnyExtLoad() &&
|
|
|
|
!isSignExtLoad() && !isZeroExtLoad() && getMemoryVT() == nullptr &&
|
2019-07-31 08:14:43 +08:00
|
|
|
getScalarMemoryVT() == nullptr && getAddressSpaces() == nullptr &&
|
|
|
|
getMinAlignment() < 1)
|
2017-10-15 10:06:44 +08:00
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsLoad cannot be used by itself");
|
|
|
|
} else {
|
|
|
|
if (isNonExtLoad())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsNonExtLoad requires IsLoad");
|
|
|
|
if (isAnyExtLoad())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAnyExtLoad requires IsLoad");
|
|
|
|
if (isSignExtLoad())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsSignExtLoad requires IsLoad");
|
|
|
|
if (isZeroExtLoad())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsZeroExtLoad requires IsLoad");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isStore()) {
|
|
|
|
if (!isUnindexed() && !isTruncStore() && !isNonTruncStore() &&
|
2019-07-31 08:14:43 +08:00
|
|
|
getMemoryVT() == nullptr && getScalarMemoryVT() == nullptr &&
|
|
|
|
getAddressSpaces() == nullptr && getMinAlignment() < 1)
|
2017-10-15 10:06:44 +08:00
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsStore cannot be used by itself");
|
|
|
|
} else {
|
|
|
|
if (isNonTruncStore())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsNonTruncStore requires IsStore");
|
|
|
|
if (isTruncStore())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsTruncStore requires IsStore");
|
|
|
|
}
|
|
|
|
|
2017-11-14 06:26:13 +08:00
|
|
|
if (isAtomic()) {
|
2017-11-14 07:03:47 +08:00
|
|
|
if (getMemoryVT() == nullptr && !isAtomicOrderingMonotonic() &&
|
2019-09-10 00:02:07 +08:00
|
|
|
getAddressSpaces() == nullptr &&
|
2017-11-14 07:03:47 +08:00
|
|
|
!isAtomicOrderingAcquire() && !isAtomicOrderingRelease() &&
|
|
|
|
!isAtomicOrderingAcquireRelease() &&
|
2017-12-01 05:05:59 +08:00
|
|
|
!isAtomicOrderingSequentiallyConsistent() &&
|
|
|
|
!isAtomicOrderingAcquireOrStronger() &&
|
|
|
|
!isAtomicOrderingReleaseOrStronger() &&
|
|
|
|
!isAtomicOrderingWeakerThanAcquire() &&
|
|
|
|
!isAtomicOrderingWeakerThanRelease())
|
2017-11-14 06:26:13 +08:00
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAtomic cannot be used by itself");
|
2017-11-14 07:03:47 +08:00
|
|
|
} else {
|
|
|
|
if (isAtomicOrderingMonotonic())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAtomicOrderingMonotonic requires IsAtomic");
|
|
|
|
if (isAtomicOrderingAcquire())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAtomicOrderingAcquire requires IsAtomic");
|
|
|
|
if (isAtomicOrderingRelease())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAtomicOrderingRelease requires IsAtomic");
|
|
|
|
if (isAtomicOrderingAcquireRelease())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAtomicOrderingAcquireRelease requires IsAtomic");
|
|
|
|
if (isAtomicOrderingSequentiallyConsistent())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAtomicOrderingSequentiallyConsistent requires IsAtomic");
|
2017-12-01 05:05:59 +08:00
|
|
|
if (isAtomicOrderingAcquireOrStronger())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAtomicOrderingAcquireOrStronger requires IsAtomic");
|
|
|
|
if (isAtomicOrderingReleaseOrStronger())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAtomicOrderingReleaseOrStronger requires IsAtomic");
|
|
|
|
if (isAtomicOrderingWeakerThanAcquire())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAtomicOrderingWeakerThanAcquire requires IsAtomic");
|
2017-11-14 06:26:13 +08:00
|
|
|
}
|
2017-11-14 07:03:47 +08:00
|
|
|
|
2017-11-14 06:26:13 +08:00
|
|
|
if (isLoad() || isStore() || isAtomic()) {
|
2019-07-16 04:59:42 +08:00
|
|
|
if (ListInit *AddressSpaces = getAddressSpaces()) {
|
|
|
|
Code += "unsigned AddrSpace = cast<MemSDNode>(N)->getAddressSpace();\n"
|
|
|
|
" if (";
|
|
|
|
|
|
|
|
bool First = true;
|
|
|
|
for (Init *Val : AddressSpaces->getValues()) {
|
|
|
|
if (First)
|
|
|
|
First = false;
|
|
|
|
else
|
|
|
|
Code += " && ";
|
|
|
|
|
|
|
|
IntInit *IntVal = dyn_cast<IntInit>(Val);
|
|
|
|
if (!IntVal) {
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"AddressSpaces element must be integer");
|
|
|
|
}
|
|
|
|
|
|
|
|
Code += "AddrSpace != " + utostr(IntVal->getValue());
|
|
|
|
}
|
|
|
|
|
|
|
|
Code += ")\nreturn false;\n";
|
|
|
|
}
|
2017-11-14 06:26:13 +08:00
|
|
|
|
2019-07-31 08:14:43 +08:00
|
|
|
int64_t MinAlign = getMinAlignment();
|
|
|
|
if (MinAlign > 0) {
|
|
|
|
Code += "if (cast<MemSDNode>(N)->getAlignment() < ";
|
|
|
|
Code += utostr(MinAlign);
|
|
|
|
Code += ")\nreturn false;\n";
|
|
|
|
}
|
|
|
|
|
2017-11-14 06:26:13 +08:00
|
|
|
Record *MemoryVT = getMemoryVT();
|
|
|
|
|
|
|
|
if (MemoryVT)
|
2019-07-16 04:59:42 +08:00
|
|
|
Code += ("if (cast<MemSDNode>(N)->getMemoryVT() != MVT::" +
|
2017-11-14 06:26:13 +08:00
|
|
|
MemoryVT->getName() + ") return false;\n")
|
|
|
|
.str();
|
|
|
|
}
|
|
|
|
|
2017-11-14 07:03:47 +08:00
|
|
|
if (isAtomic() && isAtomicOrderingMonotonic())
|
|
|
|
Code += "if (cast<AtomicSDNode>(N)->getOrdering() != "
|
|
|
|
"AtomicOrdering::Monotonic) return false;\n";
|
|
|
|
if (isAtomic() && isAtomicOrderingAcquire())
|
|
|
|
Code += "if (cast<AtomicSDNode>(N)->getOrdering() != "
|
|
|
|
"AtomicOrdering::Acquire) return false;\n";
|
|
|
|
if (isAtomic() && isAtomicOrderingRelease())
|
|
|
|
Code += "if (cast<AtomicSDNode>(N)->getOrdering() != "
|
|
|
|
"AtomicOrdering::Release) return false;\n";
|
|
|
|
if (isAtomic() && isAtomicOrderingAcquireRelease())
|
|
|
|
Code += "if (cast<AtomicSDNode>(N)->getOrdering() != "
|
|
|
|
"AtomicOrdering::AcquireRelease) return false;\n";
|
|
|
|
if (isAtomic() && isAtomicOrderingSequentiallyConsistent())
|
|
|
|
Code += "if (cast<AtomicSDNode>(N)->getOrdering() != "
|
|
|
|
"AtomicOrdering::SequentiallyConsistent) return false;\n";
|
|
|
|
|
2017-12-01 05:05:59 +08:00
|
|
|
if (isAtomic() && isAtomicOrderingAcquireOrStronger())
|
|
|
|
Code += "if (!isAcquireOrStronger(cast<AtomicSDNode>(N)->getOrdering())) "
|
|
|
|
"return false;\n";
|
|
|
|
if (isAtomic() && isAtomicOrderingWeakerThanAcquire())
|
|
|
|
Code += "if (isAcquireOrStronger(cast<AtomicSDNode>(N)->getOrdering())) "
|
|
|
|
"return false;\n";
|
|
|
|
|
|
|
|
if (isAtomic() && isAtomicOrderingReleaseOrStronger())
|
|
|
|
Code += "if (!isReleaseOrStronger(cast<AtomicSDNode>(N)->getOrdering())) "
|
|
|
|
"return false;\n";
|
|
|
|
if (isAtomic() && isAtomicOrderingWeakerThanRelease())
|
|
|
|
Code += "if (isReleaseOrStronger(cast<AtomicSDNode>(N)->getOrdering())) "
|
|
|
|
"return false;\n";
|
|
|
|
|
2017-10-15 10:06:44 +08:00
|
|
|
if (isLoad() || isStore()) {
|
|
|
|
StringRef SDNodeName = isLoad() ? "LoadSDNode" : "StoreSDNode";
|
|
|
|
|
|
|
|
if (isUnindexed())
|
|
|
|
Code += ("if (cast<" + SDNodeName +
|
|
|
|
">(N)->getAddressingMode() != ISD::UNINDEXED) "
|
|
|
|
"return false;\n")
|
|
|
|
.str();
|
|
|
|
|
|
|
|
if (isLoad()) {
|
|
|
|
if ((isNonExtLoad() + isAnyExtLoad() + isSignExtLoad() +
|
|
|
|
isZeroExtLoad()) > 1)
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsNonExtLoad, IsAnyExtLoad, IsSignExtLoad, and "
|
|
|
|
"IsZeroExtLoad are mutually exclusive");
|
|
|
|
if (isNonExtLoad())
|
|
|
|
Code += "if (cast<LoadSDNode>(N)->getExtensionType() != "
|
|
|
|
"ISD::NON_EXTLOAD) return false;\n";
|
|
|
|
if (isAnyExtLoad())
|
|
|
|
Code += "if (cast<LoadSDNode>(N)->getExtensionType() != ISD::EXTLOAD) "
|
|
|
|
"return false;\n";
|
|
|
|
if (isSignExtLoad())
|
|
|
|
Code += "if (cast<LoadSDNode>(N)->getExtensionType() != ISD::SEXTLOAD) "
|
|
|
|
"return false;\n";
|
|
|
|
if (isZeroExtLoad())
|
|
|
|
Code += "if (cast<LoadSDNode>(N)->getExtensionType() != ISD::ZEXTLOAD) "
|
|
|
|
"return false;\n";
|
|
|
|
} else {
|
|
|
|
if ((isNonTruncStore() + isTruncStore()) > 1)
|
|
|
|
PrintFatalError(
|
|
|
|
getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsNonTruncStore, and IsTruncStore are mutually exclusive");
|
|
|
|
if (isNonTruncStore())
|
|
|
|
Code +=
|
|
|
|
" if (cast<StoreSDNode>(N)->isTruncatingStore()) return false;\n";
|
|
|
|
if (isTruncStore())
|
|
|
|
Code +=
|
|
|
|
" if (!cast<StoreSDNode>(N)->isTruncatingStore()) return false;\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
Record *ScalarMemoryVT = getScalarMemoryVT();
|
|
|
|
|
|
|
|
if (ScalarMemoryVT)
|
|
|
|
Code += ("if (cast<" + SDNodeName +
|
|
|
|
">(N)->getMemoryVT().getScalarType() != MVT::" +
|
|
|
|
ScalarMemoryVT->getName() + ") return false;\n")
|
|
|
|
.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string PredicateCode = PatFragRec->getRecord()->getValueAsString("PredicateCode");
|
|
|
|
|
|
|
|
Code += PredicateCode;
|
|
|
|
|
|
|
|
if (PredicateCode.empty() && !Code.empty())
|
|
|
|
Code += "return true;\n";
|
|
|
|
|
|
|
|
return Code;
|
2011-04-18 05:38:24 +08:00
|
|
|
}
|
|
|
|
|
2017-10-16 03:01:32 +08:00
|
|
|
bool TreePredicateFn::hasImmCode() const {
|
|
|
|
return !PatFragRec->getRecord()->getValueAsString("ImmediateCode").empty();
|
|
|
|
}
|
|
|
|
|
2017-10-15 10:06:44 +08:00
|
|
|
std::string TreePredicateFn::getImmCode() const {
|
2012-01-13 11:38:34 +08:00
|
|
|
return PatFragRec->getRecord()->getValueAsString("ImmediateCode");
|
2011-04-18 06:05:17 +08:00
|
|
|
}
|
|
|
|
|
[aarch64] Support APInt and APFloat in ImmLeaf subclasses and make AArch64 use them.
Summary:
The purpose of this patch is to expose more information about ImmLeaf-like
PatLeaf's so that GlobalISel can learn to import them. Previously, ImmLeaf
could only be used to test int64_t's produced by sign-extending an APInt.
Other tests on immediates had to use the generic PatLeaf and extract the
constant using C++.
With this patch, tablegen will know how to generate predicates for APInt,
and APFloat. This will allow it to 'do the right thing' for both SelectionDAG
and GlobalISel which require different methods of extracting the immediate
from the IR.
This is NFC for SelectionDAG since the new code is equivalent to the
previous code. It's also NFC for FastISel because FastIselShouldIgnore is 1
for the ImmLeaf subclasses. Enabling FastIselShouldIgnore == 0 for these new
subclasses will require a significant re-factor of FastISel.
For GlobalISel, it's currently NFC because the relevant code to import the
affected rules is not yet present. This will be added in a later patch.
Depends on D36086
Reviewers: ab, t.p.northover, qcolombet, rovka, aditya_nandakumar
Reviewed By: qcolombet
Subscribers: bjope, aemerson, rengolin, javed.absar, igorb, llvm-commits, kristof.beyls
Differential Revision: https://reviews.llvm.org/D36534
llvm-svn: 315747
2017-10-14 04:42:18 +08:00
|
|
|
bool TreePredicateFn::immCodeUsesAPInt() const {
|
|
|
|
return getOrigPatFragRecord()->getRecord()->getValueAsBit("IsAPInt");
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TreePredicateFn::immCodeUsesAPFloat() const {
|
|
|
|
bool Unset;
|
|
|
|
// The return value will be false when IsAPFloat is unset.
|
|
|
|
return getOrigPatFragRecord()->getRecord()->getValueAsBitOrUnset("IsAPFloat",
|
|
|
|
Unset);
|
|
|
|
}
|
|
|
|
|
2017-10-15 10:06:44 +08:00
|
|
|
bool TreePredicateFn::isPredefinedPredicateEqualTo(StringRef Field,
|
|
|
|
bool Value) const {
|
|
|
|
bool Unset;
|
|
|
|
bool Result =
|
|
|
|
getOrigPatFragRecord()->getRecord()->getValueAsBitOrUnset(Field, Unset);
|
|
|
|
if (Unset)
|
|
|
|
return false;
|
|
|
|
return Result == Value;
|
|
|
|
}
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
bool TreePredicateFn::usesOperands() const {
|
|
|
|
return isPredefinedPredicateEqualTo("PredicateCodeUsesOperands", true);
|
|
|
|
}
|
2017-10-15 10:06:44 +08:00
|
|
|
bool TreePredicateFn::isLoad() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsLoad", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isStore() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsStore", true);
|
|
|
|
}
|
2017-11-14 06:26:13 +08:00
|
|
|
bool TreePredicateFn::isAtomic() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAtomic", true);
|
|
|
|
}
|
2017-10-15 10:06:44 +08:00
|
|
|
bool TreePredicateFn::isUnindexed() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsUnindexed", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isNonExtLoad() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsNonExtLoad", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isAnyExtLoad() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAnyExtLoad", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isSignExtLoad() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsSignExtLoad", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isZeroExtLoad() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsZeroExtLoad", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isNonTruncStore() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsTruncStore", false);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isTruncStore() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsTruncStore", true);
|
|
|
|
}
|
2017-11-14 07:03:47 +08:00
|
|
|
bool TreePredicateFn::isAtomicOrderingMonotonic() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAtomicOrderingMonotonic", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isAtomicOrderingAcquire() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAtomicOrderingAcquire", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isAtomicOrderingRelease() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAtomicOrderingRelease", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isAtomicOrderingAcquireRelease() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAtomicOrderingAcquireRelease", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isAtomicOrderingSequentiallyConsistent() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAtomicOrderingSequentiallyConsistent",
|
|
|
|
true);
|
|
|
|
}
|
2017-12-01 05:05:59 +08:00
|
|
|
bool TreePredicateFn::isAtomicOrderingAcquireOrStronger() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAtomicOrderingAcquireOrStronger", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isAtomicOrderingWeakerThanAcquire() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAtomicOrderingAcquireOrStronger", false);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isAtomicOrderingReleaseOrStronger() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAtomicOrderingReleaseOrStronger", true);
|
|
|
|
}
|
|
|
|
bool TreePredicateFn::isAtomicOrderingWeakerThanRelease() const {
|
|
|
|
return isPredefinedPredicateEqualTo("IsAtomicOrderingReleaseOrStronger", false);
|
|
|
|
}
|
2017-10-15 10:06:44 +08:00
|
|
|
Record *TreePredicateFn::getMemoryVT() const {
|
|
|
|
Record *R = getOrigPatFragRecord()->getRecord();
|
|
|
|
if (R->isValueUnset("MemoryVT"))
|
|
|
|
return nullptr;
|
|
|
|
return R->getValueAsDef("MemoryVT");
|
|
|
|
}
|
2019-07-16 04:59:42 +08:00
|
|
|
|
|
|
|
ListInit *TreePredicateFn::getAddressSpaces() const {
|
|
|
|
Record *R = getOrigPatFragRecord()->getRecord();
|
|
|
|
if (R->isValueUnset("AddressSpaces"))
|
|
|
|
return nullptr;
|
|
|
|
return R->getValueAsListInit("AddressSpaces");
|
|
|
|
}
|
|
|
|
|
2019-07-31 08:14:43 +08:00
|
|
|
int64_t TreePredicateFn::getMinAlignment() const {
|
|
|
|
Record *R = getOrigPatFragRecord()->getRecord();
|
|
|
|
if (R->isValueUnset("MinAlignment"))
|
|
|
|
return 0;
|
|
|
|
return R->getValueAsInt("MinAlignment");
|
|
|
|
}
|
|
|
|
|
2017-10-15 10:06:44 +08:00
|
|
|
Record *TreePredicateFn::getScalarMemoryVT() const {
|
|
|
|
Record *R = getOrigPatFragRecord()->getRecord();
|
|
|
|
if (R->isValueUnset("ScalarMemoryVT"))
|
|
|
|
return nullptr;
|
|
|
|
return R->getValueAsDef("ScalarMemoryVT");
|
|
|
|
}
|
2018-06-16 07:13:43 +08:00
|
|
|
bool TreePredicateFn::hasGISelPredicateCode() const {
|
|
|
|
return !PatFragRec->getRecord()
|
|
|
|
->getValueAsString("GISelPredicateCode")
|
|
|
|
.empty();
|
|
|
|
}
|
|
|
|
std::string TreePredicateFn::getGISelPredicateCode() const {
|
|
|
|
return PatFragRec->getRecord()->getValueAsString("GISelPredicateCode");
|
|
|
|
}
|
2017-10-15 10:06:44 +08:00
|
|
|
|
2017-10-15 05:27:53 +08:00
|
|
|
StringRef TreePredicateFn::getImmType() const {
|
[aarch64] Support APInt and APFloat in ImmLeaf subclasses and make AArch64 use them.
Summary:
The purpose of this patch is to expose more information about ImmLeaf-like
PatLeaf's so that GlobalISel can learn to import them. Previously, ImmLeaf
could only be used to test int64_t's produced by sign-extending an APInt.
Other tests on immediates had to use the generic PatLeaf and extract the
constant using C++.
With this patch, tablegen will know how to generate predicates for APInt,
and APFloat. This will allow it to 'do the right thing' for both SelectionDAG
and GlobalISel which require different methods of extracting the immediate
from the IR.
This is NFC for SelectionDAG since the new code is equivalent to the
previous code. It's also NFC for FastISel because FastIselShouldIgnore is 1
for the ImmLeaf subclasses. Enabling FastIselShouldIgnore == 0 for these new
subclasses will require a significant re-factor of FastISel.
For GlobalISel, it's currently NFC because the relevant code to import the
affected rules is not yet present. This will be added in a later patch.
Depends on D36086
Reviewers: ab, t.p.northover, qcolombet, rovka, aditya_nandakumar
Reviewed By: qcolombet
Subscribers: bjope, aemerson, rengolin, javed.absar, igorb, llvm-commits, kristof.beyls
Differential Revision: https://reviews.llvm.org/D36534
llvm-svn: 315747
2017-10-14 04:42:18 +08:00
|
|
|
if (immCodeUsesAPInt())
|
|
|
|
return "const APInt &";
|
|
|
|
if (immCodeUsesAPFloat())
|
|
|
|
return "const APFloat &";
|
|
|
|
return "int64_t";
|
|
|
|
}
|
2011-04-18 05:38:24 +08:00
|
|
|
|
2017-10-15 05:27:53 +08:00
|
|
|
StringRef TreePredicateFn::getImmTypeIdentifier() const {
|
[globalisel][tablegen] Add support for fpimm and import of APInt/APFloat based ImmLeaf.
Summary:
There's only a tablegen testcase for IntImmLeaf and not a CodeGen one
because the relevant rules are rejected for other reasons at the moment.
On AArch64, it's because there's an SDNodeXForm attached to the operand.
On X86, it's because the rule either emits multiple instructions or has
another predicate using PatFrag which cannot easily be supported at the
same time.
Reviewers: ab, t.p.northover, qcolombet, rovka, aditya_nandakumar
Reviewed By: qcolombet
Subscribers: aemerson, javed.absar, igorb, llvm-commits, kristof.beyls
Differential Revision: https://reviews.llvm.org/D36569
llvm-svn: 315761
2017-10-14 05:28:03 +08:00
|
|
|
if (immCodeUsesAPInt())
|
|
|
|
return "APInt";
|
|
|
|
else if (immCodeUsesAPFloat())
|
|
|
|
return "APFloat";
|
|
|
|
return "I64";
|
|
|
|
}
|
|
|
|
|
2011-04-18 05:38:24 +08:00
|
|
|
/// isAlwaysTrue - Return true if this is a noop predicate.
|
|
|
|
bool TreePredicateFn::isAlwaysTrue() const {
|
2017-10-16 03:01:32 +08:00
|
|
|
return !hasPredCode() && !hasImmCode();
|
2011-04-18 05:38:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the name to use in the generated code to reference this, this is
|
|
|
|
/// "Predicate_foo" if from a pattern fragment "foo".
|
|
|
|
std::string TreePredicateFn::getFnName() const {
|
2016-12-04 13:48:16 +08:00
|
|
|
return "Predicate_" + PatFragRec->getRecord()->getName().str();
|
2011-04-18 05:38:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// getCodeToRunOnSDNode - Return the code for the function body that
|
|
|
|
/// evaluates this predicate. The argument is expected to be in "Node",
|
|
|
|
/// not N. This handles casting and conversion to a concrete node type as
|
|
|
|
/// appropriate.
|
|
|
|
std::string TreePredicateFn::getCodeToRunOnSDNode() const {
|
2011-04-18 06:05:17 +08:00
|
|
|
// Handle immediate predicates first.
|
2017-10-15 10:06:44 +08:00
|
|
|
std::string ImmCode = getImmCode();
|
2011-04-18 06:05:17 +08:00
|
|
|
if (!ImmCode.empty()) {
|
2017-10-15 10:06:44 +08:00
|
|
|
if (isLoad())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsLoad cannot be used with ImmLeaf or its subclasses");
|
|
|
|
if (isStore())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsStore cannot be used with ImmLeaf or its subclasses");
|
|
|
|
if (isUnindexed())
|
|
|
|
PrintFatalError(
|
|
|
|
getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsUnindexed cannot be used with ImmLeaf or its subclasses");
|
|
|
|
if (isNonExtLoad())
|
|
|
|
PrintFatalError(
|
|
|
|
getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsNonExtLoad cannot be used with ImmLeaf or its subclasses");
|
|
|
|
if (isAnyExtLoad())
|
|
|
|
PrintFatalError(
|
|
|
|
getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsAnyExtLoad cannot be used with ImmLeaf or its subclasses");
|
|
|
|
if (isSignExtLoad())
|
|
|
|
PrintFatalError(
|
|
|
|
getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsSignExtLoad cannot be used with ImmLeaf or its subclasses");
|
|
|
|
if (isZeroExtLoad())
|
|
|
|
PrintFatalError(
|
|
|
|
getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsZeroExtLoad cannot be used with ImmLeaf or its subclasses");
|
|
|
|
if (isNonTruncStore())
|
|
|
|
PrintFatalError(
|
|
|
|
getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsNonTruncStore cannot be used with ImmLeaf or its subclasses");
|
|
|
|
if (isTruncStore())
|
|
|
|
PrintFatalError(
|
|
|
|
getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"IsTruncStore cannot be used with ImmLeaf or its subclasses");
|
|
|
|
if (getMemoryVT())
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"MemoryVT cannot be used with ImmLeaf or its subclasses");
|
|
|
|
if (getScalarMemoryVT())
|
|
|
|
PrintFatalError(
|
|
|
|
getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"ScalarMemoryVT cannot be used with ImmLeaf or its subclasses");
|
|
|
|
|
|
|
|
std::string Result = (" " + getImmType() + " Imm = ").str();
|
[aarch64] Support APInt and APFloat in ImmLeaf subclasses and make AArch64 use them.
Summary:
The purpose of this patch is to expose more information about ImmLeaf-like
PatLeaf's so that GlobalISel can learn to import them. Previously, ImmLeaf
could only be used to test int64_t's produced by sign-extending an APInt.
Other tests on immediates had to use the generic PatLeaf and extract the
constant using C++.
With this patch, tablegen will know how to generate predicates for APInt,
and APFloat. This will allow it to 'do the right thing' for both SelectionDAG
and GlobalISel which require different methods of extracting the immediate
from the IR.
This is NFC for SelectionDAG since the new code is equivalent to the
previous code. It's also NFC for FastISel because FastIselShouldIgnore is 1
for the ImmLeaf subclasses. Enabling FastIselShouldIgnore == 0 for these new
subclasses will require a significant re-factor of FastISel.
For GlobalISel, it's currently NFC because the relevant code to import the
affected rules is not yet present. This will be added in a later patch.
Depends on D36086
Reviewers: ab, t.p.northover, qcolombet, rovka, aditya_nandakumar
Reviewed By: qcolombet
Subscribers: bjope, aemerson, rengolin, javed.absar, igorb, llvm-commits, kristof.beyls
Differential Revision: https://reviews.llvm.org/D36534
llvm-svn: 315747
2017-10-14 04:42:18 +08:00
|
|
|
if (immCodeUsesAPFloat())
|
|
|
|
Result += "cast<ConstantFPSDNode>(Node)->getValueAPF();\n";
|
|
|
|
else if (immCodeUsesAPInt())
|
|
|
|
Result += "cast<ConstantSDNode>(Node)->getAPIntValue();\n";
|
|
|
|
else
|
|
|
|
Result += "cast<ConstantSDNode>(Node)->getSExtValue();\n";
|
2017-10-15 10:06:44 +08:00
|
|
|
return Result + ImmCode;
|
2011-04-18 06:05:17 +08:00
|
|
|
}
|
2017-09-23 00:57:28 +08:00
|
|
|
|
2011-04-18 06:05:17 +08:00
|
|
|
// Handle arbitrary node predicates.
|
2017-10-16 03:01:32 +08:00
|
|
|
assert(hasPredCode() && "Don't have any predicate code!");
|
2019-12-31 01:05:25 +08:00
|
|
|
|
|
|
|
// If this is using PatFrags, there are multiple trees to search. They should
|
|
|
|
// all have the same class. FIXME: Is there a way to find a common
|
|
|
|
// superclass?
|
2017-10-15 05:27:53 +08:00
|
|
|
StringRef ClassName;
|
2019-12-31 01:05:25 +08:00
|
|
|
for (const auto &Tree : PatFragRec->getTrees()) {
|
|
|
|
StringRef TreeClassName;
|
|
|
|
if (Tree->isLeaf())
|
|
|
|
TreeClassName = "SDNode";
|
|
|
|
else {
|
|
|
|
Record *Op = Tree->getOperator();
|
|
|
|
const SDNodeInfo &Info = PatFragRec->getDAGPatterns().getSDNodeInfo(Op);
|
|
|
|
TreeClassName = Info.getSDClassName();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ClassName.empty())
|
|
|
|
ClassName = TreeClassName;
|
|
|
|
else if (ClassName != TreeClassName) {
|
|
|
|
PrintFatalError(getOrigPatFragRecord()->getRecord()->getLoc(),
|
|
|
|
"PatFrags trees do not have consistent class");
|
|
|
|
}
|
2011-04-18 05:38:24 +08:00
|
|
|
}
|
2019-12-31 01:05:25 +08:00
|
|
|
|
2011-04-18 05:38:24 +08:00
|
|
|
std::string Result;
|
|
|
|
if (ClassName == "SDNode")
|
|
|
|
Result = " SDNode *N = Node;\n";
|
|
|
|
else
|
2017-10-15 05:27:53 +08:00
|
|
|
Result = " auto *N = cast<" + ClassName.str() + ">(Node);\n";
|
2017-09-23 00:57:28 +08:00
|
|
|
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
return (Twine(Result) + " (void)N;\n" + getPredCode()).str();
|
2008-03-06 01:49:05 +08:00
|
|
|
}
|
|
|
|
|
2008-08-22 08:20:26 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// PatternToMatch implementation
|
|
|
|
//
|
|
|
|
|
2019-03-10 13:21:52 +08:00
|
|
|
static bool isImmAllOnesAllZerosMatch(const TreePatternNode *P) {
|
|
|
|
if (!P->isLeaf())
|
|
|
|
return false;
|
|
|
|
DefInit *DI = dyn_cast<DefInit>(P->getLeafValue());
|
|
|
|
if (!DI)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
Record *R = DI->getDef();
|
|
|
|
return R->getName() == "immAllOnesV" || R->getName() == "immAllZerosV";
|
|
|
|
}
|
|
|
|
|
2010-03-29 09:40:38 +08:00
|
|
|
/// getPatternSize - Return the 'size' of this pattern. We want to match large
|
|
|
|
/// patterns before small ones. This is used to determine the size of a
|
|
|
|
/// pattern.
|
2018-06-15 04:32:58 +08:00
|
|
|
static unsigned getPatternSize(const TreePatternNode *P,
|
2010-03-29 09:40:38 +08:00
|
|
|
const CodeGenDAGPatterns &CGP) {
|
|
|
|
unsigned Size = 3; // The node itself.
|
|
|
|
// If the root node is a ConstantSDNode, increases its size.
|
|
|
|
// e.g. (set R32:$dst, 0).
|
2018-06-15 04:32:58 +08:00
|
|
|
if (P->isLeaf() && isa<IntInit>(P->getLeafValue()))
|
2010-03-29 09:40:38 +08:00
|
|
|
Size += 2;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
if (const ComplexPattern *AM = P->getComplexPatternInfo(CGP)) {
|
2016-11-10 07:53:43 +08:00
|
|
|
Size += AM->getComplexity();
|
2014-05-20 19:52:46 +08:00
|
|
|
// We don't want to count any children twice, so return early.
|
|
|
|
return Size;
|
|
|
|
}
|
|
|
|
|
2010-03-29 09:40:38 +08:00
|
|
|
// If this node has some predicate function that must match, it adds to the
|
|
|
|
// complexity of this node.
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
if (!P->getPredicateCalls().empty())
|
2010-03-29 09:40:38 +08:00
|
|
|
++Size;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-29 09:40:38 +08:00
|
|
|
// Count children in the count if they are also nodes.
|
2018-06-15 04:32:58 +08:00
|
|
|
for (unsigned i = 0, e = P->getNumChildren(); i != e; ++i) {
|
|
|
|
const TreePatternNode *Child = P->getChild(i);
|
|
|
|
if (!Child->isLeaf() && Child->getNumTypes()) {
|
2018-08-16 04:41:19 +08:00
|
|
|
const TypeSetByHwMode &T0 = Child->getExtType(0);
|
2017-09-15 00:56:21 +08:00
|
|
|
// At this point, all variable type sets should be simple, i.e. only
|
|
|
|
// have a default mode.
|
|
|
|
if (T0.getMachineValueType() != MVT::Other) {
|
|
|
|
Size += getPatternSize(Child, CGP);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2018-06-15 04:32:58 +08:00
|
|
|
if (Child->isLeaf()) {
|
|
|
|
if (isa<IntInit>(Child->getLeafValue()))
|
2010-03-29 09:40:38 +08:00
|
|
|
Size += 5; // Matches a ConstantSDNode (+3) and a specific value (+2).
|
2018-06-15 04:32:58 +08:00
|
|
|
else if (Child->getComplexPatternInfo(CGP))
|
2010-03-29 09:40:38 +08:00
|
|
|
Size += getPatternSize(Child, CGP);
|
2019-03-10 13:21:52 +08:00
|
|
|
else if (isImmAllOnesAllZerosMatch(Child))
|
|
|
|
Size += 4; // Matches a build_vector(+3) and a predicate (+1).
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
else if (!Child->getPredicateCalls().empty())
|
2010-03-29 09:40:38 +08:00
|
|
|
++Size;
|
|
|
|
}
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-29 09:40:38 +08:00
|
|
|
return Size;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Compute the complexity metric for the input pattern. This roughly
|
|
|
|
/// corresponds to the number of nodes that are covered.
|
2014-08-01 08:32:36 +08:00
|
|
|
int PatternToMatch::
|
2010-03-29 09:40:38 +08:00
|
|
|
getPatternComplexity(const CodeGenDAGPatterns &CGP) const {
|
2018-06-15 04:32:58 +08:00
|
|
|
return getPatternSize(getSrcPattern(), CGP) + getAddedComplexity();
|
2010-03-29 09:40:38 +08:00
|
|
|
}
|
|
|
|
|
2008-08-22 08:20:26 +08:00
|
|
|
/// getPredicateCheck - Return a single string containing all of this
|
|
|
|
/// pattern's predicates concatenated with "&&" operators.
|
|
|
|
///
|
|
|
|
std::string PatternToMatch::getPredicateCheck() const {
|
2017-09-15 00:56:21 +08:00
|
|
|
SmallVector<const Predicate*,4> PredList;
|
2019-07-30 23:56:43 +08:00
|
|
|
for (const Predicate &P : Predicates) {
|
|
|
|
if (!P.getCondString().empty())
|
|
|
|
PredList.push_back(&P);
|
|
|
|
}
|
2019-08-23 01:31:59 +08:00
|
|
|
llvm::sort(PredList, deref<std::less<>>());
|
2015-11-27 13:44:04 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
std::string Check;
|
|
|
|
for (unsigned i = 0, e = PredList.size(); i != e; ++i) {
|
|
|
|
if (i != 0)
|
|
|
|
Check += " && ";
|
|
|
|
Check += '(' + PredList[i]->getCondString() + ')';
|
2015-11-27 13:44:04 +08:00
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
return Check;
|
2008-08-22 08:20:26 +08:00
|
|
|
}
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// SDTypeConstraint implementation
|
|
|
|
//
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
SDTypeConstraint::SDTypeConstraint(Record *R, const CodeGenHwModes &CGH) {
|
2008-01-06 06:25:12 +08:00
|
|
|
OperandNo = R->getValueAsInt("OperandNum");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
if (R->isSubClassOf("SDTCisVT")) {
|
|
|
|
ConstraintType = SDTCisVT;
|
2017-09-15 00:56:21 +08:00
|
|
|
VVT = getValueTypeByHwMode(R->getValueAsDef("VT"), CGH);
|
|
|
|
for (const auto &P : VVT)
|
|
|
|
if (P.second == MVT::isVoid)
|
|
|
|
PrintFatalError(R->getLoc(), "Cannot use 'Void' as type to SDTCisVT");
|
2008-01-06 06:25:12 +08:00
|
|
|
} else if (R->isSubClassOf("SDTCisPtrTy")) {
|
|
|
|
ConstraintType = SDTCisPtrTy;
|
|
|
|
} else if (R->isSubClassOf("SDTCisInt")) {
|
|
|
|
ConstraintType = SDTCisInt;
|
|
|
|
} else if (R->isSubClassOf("SDTCisFP")) {
|
|
|
|
ConstraintType = SDTCisFP;
|
2009-08-13 06:30:59 +08:00
|
|
|
} else if (R->isSubClassOf("SDTCisVec")) {
|
|
|
|
ConstraintType = SDTCisVec;
|
2008-01-06 06:25:12 +08:00
|
|
|
} else if (R->isSubClassOf("SDTCisSameAs")) {
|
|
|
|
ConstraintType = SDTCisSameAs;
|
|
|
|
x.SDTCisSameAs_Info.OtherOperandNum = R->getValueAsInt("OtherOperandNum");
|
|
|
|
} else if (R->isSubClassOf("SDTCisVTSmallerThanOp")) {
|
|
|
|
ConstraintType = SDTCisVTSmallerThanOp;
|
2010-12-22 00:16:00 +08:00
|
|
|
x.SDTCisVTSmallerThanOp_Info.OtherOperandNum =
|
2008-01-06 06:25:12 +08:00
|
|
|
R->getValueAsInt("OtherOperandNum");
|
|
|
|
} else if (R->isSubClassOf("SDTCisOpSmallerThanOp")) {
|
|
|
|
ConstraintType = SDTCisOpSmallerThanOp;
|
2010-12-22 00:16:00 +08:00
|
|
|
x.SDTCisOpSmallerThanOp_Info.BigOperandNum =
|
2008-01-06 06:25:12 +08:00
|
|
|
R->getValueAsInt("BigOperandNum");
|
2008-02-09 09:37:05 +08:00
|
|
|
} else if (R->isSubClassOf("SDTCisEltOfVec")) {
|
|
|
|
ConstraintType = SDTCisEltOfVec;
|
2010-03-15 14:00:16 +08:00
|
|
|
x.SDTCisEltOfVec_Info.OtherOperandNum = R->getValueAsInt("OtherOpNum");
|
2011-01-25 04:53:18 +08:00
|
|
|
} else if (R->isSubClassOf("SDTCisSubVecOfVec")) {
|
|
|
|
ConstraintType = SDTCisSubVecOfVec;
|
|
|
|
x.SDTCisSubVecOfVec_Info.OtherOperandNum =
|
|
|
|
R->getValueAsInt("OtherOpNum");
|
2015-03-05 15:11:34 +08:00
|
|
|
} else if (R->isSubClassOf("SDTCVecEltisVT")) {
|
|
|
|
ConstraintType = SDTCVecEltisVT;
|
2017-09-15 00:56:21 +08:00
|
|
|
VVT = getValueTypeByHwMode(R->getValueAsDef("VT"), CGH);
|
|
|
|
for (const auto &P : VVT) {
|
|
|
|
MVT T = P.second;
|
|
|
|
if (T.isVector())
|
|
|
|
PrintFatalError(R->getLoc(),
|
|
|
|
"Cannot use vector type as SDTCVecEltisVT");
|
|
|
|
if (!T.isInteger() && !T.isFloatingPoint())
|
|
|
|
PrintFatalError(R->getLoc(), "Must use integer or floating point type "
|
|
|
|
"as SDTCVecEltisVT");
|
|
|
|
}
|
2015-03-05 15:11:34 +08:00
|
|
|
} else if (R->isSubClassOf("SDTCisSameNumEltsAs")) {
|
|
|
|
ConstraintType = SDTCisSameNumEltsAs;
|
|
|
|
x.SDTCisSameNumEltsAs_Info.OtherOperandNum =
|
|
|
|
R->getValueAsInt("OtherOperandNum");
|
2015-11-26 15:02:18 +08:00
|
|
|
} else if (R->isSubClassOf("SDTCisSameSizeAs")) {
|
|
|
|
ConstraintType = SDTCisSameSizeAs;
|
|
|
|
x.SDTCisSameSizeAs_Info.OtherOperandNum =
|
|
|
|
R->getValueAsInt("OtherOperandNum");
|
2008-01-06 06:25:12 +08:00
|
|
|
} else {
|
[tablegen] Add locations to many PrintFatalError() calls
Summary:
While working on the GISel Combiner, I noticed I was producing location-less
error messages fairly often and set about fixing this. In the process, I
noticed quite a few places elsewhere in TableGen that also neglected to include
a relevant location.
This patch adds locations to errors that relate to a specific record (or a
field within it) and also have easy access to the relevant location. This is
particularly useful when multiclasses are involved as many of these errors
refer to the full name of a record and it's difficult to guess which substring
is grep-able.
Unfortunately, tablegen currently only supports Record granularity so it's not
currently possible to point at a specific Init so these sometimes point at the
record that caused the error rather than the precise origin of the error.
Reviewers: bogner, aditya_nandakumar, volkan, aemerson, paquette, nhaehnle
Reviewed By: nhaehnle
Subscribers: jdoerfert, nhaehnle, asb, rbar, johnrusso, simoncook, apazos, sabuasal, niosHD, jrtc27, zzheng, edward-jones, rogfer01, MartinMosbeck, brucehoult, the_o, PkmX, jocewei, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D58077
llvm-svn: 353862
2019-02-13 01:36:57 +08:00
|
|
|
PrintFatalError(R->getLoc(),
|
|
|
|
"Unrecognized SDTypeConstraint '" + R->getName() + "'!\n");
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getOperandNum - Return the node corresponding to operand #OpNo in tree
|
2010-03-20 05:56:21 +08:00
|
|
|
/// N, and the result number in ResNo.
|
2018-06-15 04:32:58 +08:00
|
|
|
static TreePatternNode *getOperandNum(unsigned OpNo, TreePatternNode *N,
|
2010-03-20 05:56:21 +08:00
|
|
|
const SDNodeInfo &NodeInfo,
|
|
|
|
unsigned &ResNo) {
|
|
|
|
unsigned NumResults = NodeInfo.getNumResults();
|
|
|
|
if (OpNo < NumResults) {
|
|
|
|
ResNo = OpNo;
|
|
|
|
return N;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-20 05:56:21 +08:00
|
|
|
OpNo -= NumResults;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
if (OpNo >= N->getNumChildren()) {
|
2015-05-12 06:17:13 +08:00
|
|
|
std::string S;
|
|
|
|
raw_string_ostream OS(S);
|
|
|
|
OS << "Invalid operand number in type constraint "
|
2010-03-20 05:56:21 +08:00
|
|
|
<< (OpNo+NumResults) << " ";
|
2018-06-15 04:32:58 +08:00
|
|
|
N->print(OS);
|
2015-05-12 06:17:13 +08:00
|
|
|
PrintFatalError(OS.str());
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
return N->getChild(OpNo);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// ApplyTypeConstraint - Given a node in a pattern, apply this type
|
|
|
|
/// constraint to the nodes operands. This returns true if it makes a
|
2012-10-26 04:33:17 +08:00
|
|
|
/// change, false otherwise. If a type contradiction is found, flag an error.
|
2018-06-15 04:32:58 +08:00
|
|
|
bool SDTypeConstraint::ApplyTypeConstraint(TreePatternNode *N,
|
2008-01-06 06:25:12 +08:00
|
|
|
const SDNodeInfo &NodeInfo,
|
|
|
|
TreePattern &TP) const {
|
2012-10-26 04:33:17 +08:00
|
|
|
if (TP.hasError())
|
|
|
|
return false;
|
|
|
|
|
2010-03-20 05:56:21 +08:00
|
|
|
unsigned ResNo = 0; // The result number being referenced.
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *NodeToApply = getOperandNum(OperandNo, N, NodeInfo, ResNo);
|
2017-09-15 00:56:21 +08:00
|
|
|
TypeInfer &TI = TP.getInfer();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
switch (ConstraintType) {
|
|
|
|
case SDTCisVT:
|
|
|
|
// Operand must be a particular type.
|
2018-06-15 04:32:58 +08:00
|
|
|
return NodeToApply->UpdateNodeType(ResNo, VVT, TP);
|
2010-03-15 14:00:16 +08:00
|
|
|
case SDTCisPtrTy:
|
2008-01-06 06:25:12 +08:00
|
|
|
// Operand must be same as target pointer type.
|
2018-06-15 04:32:58 +08:00
|
|
|
return NodeToApply->UpdateNodeType(ResNo, MVT::iPTR, TP);
|
2010-03-15 14:00:16 +08:00
|
|
|
case SDTCisInt:
|
|
|
|
// Require it to be one of the legal integer VTs.
|
2018-06-15 04:32:58 +08:00
|
|
|
return TI.EnforceInteger(NodeToApply->getExtType(ResNo));
|
2010-03-15 14:00:16 +08:00
|
|
|
case SDTCisFP:
|
|
|
|
// Require it to be one of the legal fp VTs.
|
2018-06-15 04:32:58 +08:00
|
|
|
return TI.EnforceFloatingPoint(NodeToApply->getExtType(ResNo));
|
2010-03-15 14:00:16 +08:00
|
|
|
case SDTCisVec:
|
|
|
|
// Require it to be one of the legal vector VTs.
|
2018-06-15 04:32:58 +08:00
|
|
|
return TI.EnforceVector(NodeToApply->getExtType(ResNo));
|
2008-01-06 06:25:12 +08:00
|
|
|
case SDTCisSameAs: {
|
2010-03-20 05:56:21 +08:00
|
|
|
unsigned OResNo = 0;
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *OtherNode =
|
2010-03-20 05:56:21 +08:00
|
|
|
getOperandNum(x.SDTCisSameAs_Info.OtherOperandNum, N, NodeInfo, OResNo);
|
2018-06-15 04:32:58 +08:00
|
|
|
return NodeToApply->UpdateNodeType(ResNo, OtherNode->getExtType(OResNo),TP)|
|
|
|
|
OtherNode->UpdateNodeType(OResNo,NodeToApply->getExtType(ResNo),TP);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
case SDTCisVTSmallerThanOp: {
|
|
|
|
// The NodeToApply must be a leaf node that is a VT. OtherOperandNum must
|
|
|
|
// have an integer type that is smaller than the VT.
|
2018-06-15 04:32:58 +08:00
|
|
|
if (!NodeToApply->isLeaf() ||
|
|
|
|
!isa<DefInit>(NodeToApply->getLeafValue()) ||
|
|
|
|
!static_cast<DefInit*>(NodeToApply->getLeafValue())->getDef()
|
2012-10-26 04:33:17 +08:00
|
|
|
->isSubClassOf("ValueType")) {
|
2018-06-15 04:32:58 +08:00
|
|
|
TP.error(N->getOperator()->getName() + " expects a VT operand!");
|
2012-10-26 04:33:17 +08:00
|
|
|
return false;
|
|
|
|
}
|
2018-06-15 04:32:58 +08:00
|
|
|
DefInit *DI = static_cast<DefInit*>(NodeToApply->getLeafValue());
|
2017-09-15 00:56:21 +08:00
|
|
|
const CodeGenTarget &T = TP.getDAGPatterns().getTargetInfo();
|
|
|
|
auto VVT = getValueTypeByHwMode(DI->getDef(), T.getHwModes());
|
|
|
|
TypeSetByHwMode TypeListTmp(VVT);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-20 05:56:21 +08:00
|
|
|
unsigned OResNo = 0;
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *OtherNode =
|
2010-03-20 05:56:21 +08:00
|
|
|
getOperandNum(x.SDTCisVTSmallerThanOp_Info.OtherOperandNum, N, NodeInfo,
|
|
|
|
OResNo);
|
2010-03-15 14:00:16 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
return TI.EnforceSmallerThan(TypeListTmp, OtherNode->getExtType(OResNo));
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
case SDTCisOpSmallerThanOp: {
|
2010-03-20 05:56:21 +08:00
|
|
|
unsigned BResNo = 0;
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *BigOperand =
|
2010-03-20 05:56:21 +08:00
|
|
|
getOperandNum(x.SDTCisOpSmallerThanOp_Info.BigOperandNum, N, NodeInfo,
|
|
|
|
BResNo);
|
2018-06-15 04:32:58 +08:00
|
|
|
return TI.EnforceSmallerThan(NodeToApply->getExtType(ResNo),
|
|
|
|
BigOperand->getExtType(BResNo));
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2008-02-09 09:37:05 +08:00
|
|
|
case SDTCisEltOfVec: {
|
2010-03-20 05:56:21 +08:00
|
|
|
unsigned VResNo = 0;
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *VecOperand =
|
2010-03-20 05:56:21 +08:00
|
|
|
getOperandNum(x.SDTCisEltOfVec_Info.OtherOperandNum, N, NodeInfo,
|
|
|
|
VResNo);
|
2010-03-24 08:01:16 +08:00
|
|
|
// Filter vector types out of VecOperand that don't have the right element
|
|
|
|
// type.
|
2018-06-15 04:32:58 +08:00
|
|
|
return TI.EnforceVectorEltTypeIs(VecOperand->getExtType(VResNo),
|
|
|
|
NodeToApply->getExtType(ResNo));
|
2008-02-09 09:37:05 +08:00
|
|
|
}
|
2011-01-25 04:53:18 +08:00
|
|
|
case SDTCisSubVecOfVec: {
|
|
|
|
unsigned VResNo = 0;
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *BigVecOperand =
|
2011-01-25 04:53:18 +08:00
|
|
|
getOperandNum(x.SDTCisSubVecOfVec_Info.OtherOperandNum, N, NodeInfo,
|
|
|
|
VResNo);
|
|
|
|
|
|
|
|
// Filter vector types out of BigVecOperand that don't have the
|
|
|
|
// right subvector type.
|
2018-06-15 04:32:58 +08:00
|
|
|
return TI.EnforceVectorSubVectorTypeIs(BigVecOperand->getExtType(VResNo),
|
|
|
|
NodeToApply->getExtType(ResNo));
|
2011-01-25 04:53:18 +08:00
|
|
|
}
|
2015-03-05 15:11:34 +08:00
|
|
|
case SDTCVecEltisVT: {
|
2018-06-15 04:32:58 +08:00
|
|
|
return TI.EnforceVectorEltTypeIs(NodeToApply->getExtType(ResNo), VVT);
|
2015-03-05 15:11:34 +08:00
|
|
|
}
|
|
|
|
case SDTCisSameNumEltsAs: {
|
|
|
|
unsigned OResNo = 0;
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *OtherNode =
|
2015-03-05 15:11:34 +08:00
|
|
|
getOperandNum(x.SDTCisSameNumEltsAs_Info.OtherOperandNum,
|
|
|
|
N, NodeInfo, OResNo);
|
2018-06-15 04:32:58 +08:00
|
|
|
return TI.EnforceSameNumElts(OtherNode->getExtType(OResNo),
|
|
|
|
NodeToApply->getExtType(ResNo));
|
2015-03-05 15:11:34 +08:00
|
|
|
}
|
2015-11-26 15:02:18 +08:00
|
|
|
case SDTCisSameSizeAs: {
|
|
|
|
unsigned OResNo = 0;
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *OtherNode =
|
2015-11-26 15:02:18 +08:00
|
|
|
getOperandNum(x.SDTCisSameSizeAs_Info.OtherOperandNum,
|
|
|
|
N, NodeInfo, OResNo);
|
2018-06-15 04:32:58 +08:00
|
|
|
return TI.EnforceSameSize(OtherNode->getExtType(OResNo),
|
|
|
|
NodeToApply->getExtType(ResNo));
|
2015-11-26 15:02:18 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
}
|
2012-01-17 15:00:13 +08:00
|
|
|
llvm_unreachable("Invalid ConstraintType!");
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2013-03-18 12:08:07 +08:00
|
|
|
// Update the node type to match an instruction operand or result as specified
|
|
|
|
// in the ins or outs lists on the instruction definition. Return true if the
|
|
|
|
// type was actually changed.
|
|
|
|
bool TreePatternNode::UpdateNodeTypeFromInst(unsigned ResNo,
|
|
|
|
Record *Operand,
|
|
|
|
TreePattern &TP) {
|
|
|
|
// The 'unknown' operand indicates that types should be inferred from the
|
|
|
|
// context.
|
|
|
|
if (Operand->isSubClassOf("unknown_class"))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// The Operand class specifies a type directly.
|
2017-09-15 00:56:21 +08:00
|
|
|
if (Operand->isSubClassOf("Operand")) {
|
|
|
|
Record *R = Operand->getValueAsDef("Type");
|
|
|
|
const CodeGenTarget &T = TP.getDAGPatterns().getTargetInfo();
|
|
|
|
return UpdateNodeType(ResNo, getValueTypeByHwMode(R, T.getHwModes()), TP);
|
|
|
|
}
|
2013-03-18 12:08:07 +08:00
|
|
|
|
|
|
|
// PointerLikeRegClass has a type that is determined at runtime.
|
|
|
|
if (Operand->isSubClassOf("PointerLikeRegClass"))
|
|
|
|
return UpdateNodeType(ResNo, MVT::iPTR, TP);
|
|
|
|
|
|
|
|
// Both RegisterClass and RegisterOperand operands derive their types from a
|
|
|
|
// register class def.
|
2014-04-15 15:20:03 +08:00
|
|
|
Record *RC = nullptr;
|
2013-03-18 12:08:07 +08:00
|
|
|
if (Operand->isSubClassOf("RegisterClass"))
|
|
|
|
RC = Operand;
|
|
|
|
else if (Operand->isSubClassOf("RegisterOperand"))
|
|
|
|
RC = Operand->getValueAsDef("RegClass");
|
|
|
|
|
|
|
|
assert(RC && "Unknown operand type");
|
|
|
|
CodeGenTarget &Tgt = TP.getDAGPatterns().getTargetInfo();
|
|
|
|
return UpdateNodeType(ResNo, Tgt.getRegisterClass(RC).getValueTypes(), TP);
|
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool TreePatternNode::ContainsUnresolvedType(TreePattern &TP) const {
|
|
|
|
for (unsigned i = 0, e = Types.size(); i != e; ++i)
|
|
|
|
if (!TP.getInfer().isConcrete(Types[i], true))
|
|
|
|
return true;
|
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
if (getChild(i)->ContainsUnresolvedType(TP))
|
2017-09-15 00:56:21 +08:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TreePatternNode::hasProperTypeByHwMode() const {
|
|
|
|
for (const TypeSetByHwMode &S : Types)
|
|
|
|
if (!S.isDefaultOnly())
|
|
|
|
return true;
|
2018-05-31 05:00:18 +08:00
|
|
|
for (const TreePatternNodePtr &C : Children)
|
2017-09-15 00:56:21 +08:00
|
|
|
if (C->hasProperTypeByHwMode())
|
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TreePatternNode::hasPossibleType() const {
|
|
|
|
for (const TypeSetByHwMode &S : Types)
|
|
|
|
if (!S.isPossible())
|
|
|
|
return false;
|
2018-05-31 05:00:18 +08:00
|
|
|
for (const TreePatternNodePtr &C : Children)
|
2017-09-15 00:56:21 +08:00
|
|
|
if (!C->hasPossibleType())
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool TreePatternNode::setDefaultMode(unsigned Mode) {
|
|
|
|
for (TypeSetByHwMode &S : Types) {
|
|
|
|
S.makeSimple(Mode);
|
|
|
|
// Check if the selected mode had a type conflict.
|
|
|
|
if (S.get(DefaultMode).empty())
|
|
|
|
return false;
|
|
|
|
}
|
2018-05-31 05:00:18 +08:00
|
|
|
for (const TreePatternNodePtr &C : Children)
|
2017-09-15 00:56:21 +08:00
|
|
|
if (!C->setDefaultMode(Mode))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
2013-03-18 12:08:07 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// SDNodeInfo implementation
|
|
|
|
//
|
2017-09-15 00:56:21 +08:00
|
|
|
SDNodeInfo::SDNodeInfo(Record *R, const CodeGenHwModes &CGH) : Def(R) {
|
2008-01-06 06:25:12 +08:00
|
|
|
EnumName = R->getValueAsString("Opcode");
|
|
|
|
SDClassName = R->getValueAsString("SDClass");
|
|
|
|
Record *TypeProfile = R->getValueAsDef("TypeProfile");
|
|
|
|
NumResults = TypeProfile->getValueAsInt("NumResults");
|
|
|
|
NumOperands = TypeProfile->getValueAsInt("NumOperands");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Parse the properties.
|
2017-12-21 03:36:28 +08:00
|
|
|
Properties = parseSDPatternOperatorProperties(R);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Parse the type constraints.
|
|
|
|
std::vector<Record*> ConstraintList =
|
|
|
|
TypeProfile->getValueAsListOfDefs("Constraints");
|
2017-09-15 00:56:21 +08:00
|
|
|
for (Record *R : ConstraintList)
|
|
|
|
TypeConstraints.emplace_back(R, CGH);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2010-02-28 08:22:30 +08:00
|
|
|
/// getKnownType - If the type constraints on this node imply a fixed type
|
|
|
|
/// (e.g. all stores return void, etc), then return it as an
|
2010-03-19 09:14:27 +08:00
|
|
|
/// MVT::SimpleValueType. Otherwise, return EEVT::Other.
|
2010-03-24 08:41:19 +08:00
|
|
|
MVT::SimpleValueType SDNodeInfo::getKnownType(unsigned ResNo) const {
|
2010-02-28 08:22:30 +08:00
|
|
|
unsigned NumResults = getNumResults();
|
|
|
|
assert(NumResults <= 1 &&
|
|
|
|
"We only work with nodes with zero or one result so far!");
|
2010-03-24 08:41:19 +08:00
|
|
|
assert(ResNo == 0 && "Only handles single result nodes so far");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2015-11-23 04:46:24 +08:00
|
|
|
for (const SDTypeConstraint &Constraint : TypeConstraints) {
|
2010-02-28 08:22:30 +08:00
|
|
|
// Make sure that this applies to the correct node result.
|
2015-11-23 04:46:24 +08:00
|
|
|
if (Constraint.OperandNo >= NumResults) // FIXME: need value #
|
2010-02-28 08:22:30 +08:00
|
|
|
continue;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2015-11-23 04:46:24 +08:00
|
|
|
switch (Constraint.ConstraintType) {
|
2010-02-28 08:22:30 +08:00
|
|
|
default: break;
|
|
|
|
case SDTypeConstraint::SDTCisVT:
|
2017-09-15 00:56:21 +08:00
|
|
|
if (Constraint.VVT.isSimple())
|
|
|
|
return Constraint.VVT.getSimple().SimpleTy;
|
|
|
|
break;
|
2010-02-28 08:22:30 +08:00
|
|
|
case SDTypeConstraint::SDTCisPtrTy:
|
|
|
|
return MVT::iPTR;
|
|
|
|
}
|
|
|
|
}
|
2010-03-19 09:14:27 +08:00
|
|
|
return MVT::Other;
|
2010-02-28 08:22:30 +08:00
|
|
|
}
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// TreePatternNode implementation
|
|
|
|
//
|
|
|
|
|
2010-03-20 05:37:09 +08:00
|
|
|
static unsigned GetNumNodeResults(Record *Operator, CodeGenDAGPatterns &CDP) {
|
|
|
|
if (Operator->getName() == "set" ||
|
2010-03-27 10:53:27 +08:00
|
|
|
Operator->getName() == "implicit")
|
2010-03-20 05:37:09 +08:00
|
|
|
return 0; // All return nothing.
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-23 04:56:36 +08:00
|
|
|
if (Operator->isSubClassOf("Intrinsic"))
|
|
|
|
return CDP.getIntrinsic(Operator).IS.RetVTs.size();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-20 05:37:09 +08:00
|
|
|
if (Operator->isSubClassOf("SDNode"))
|
|
|
|
return CDP.getSDNodeInfo(Operator).getNumResults();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (Operator->isSubClassOf("PatFrags")) {
|
2010-03-20 05:37:09 +08:00
|
|
|
// If we've already parsed this pattern fragment, get it. Otherwise, handle
|
|
|
|
// the forward reference case where one pattern fragment references another
|
|
|
|
// before it is processed.
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (TreePattern *PFRec = CDP.getPatternFragmentIfRead(Operator)) {
|
|
|
|
// The number of results of a fragment with alternative records is the
|
|
|
|
// maximum number of results across all alternatives.
|
|
|
|
unsigned NumResults = 0;
|
|
|
|
for (auto T : PFRec->getTrees())
|
|
|
|
NumResults = std::max(NumResults, T->getNumTypes());
|
|
|
|
return NumResults;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
ListInit *LI = Operator->getValueAsListInit("Fragments");
|
|
|
|
assert(LI && "Invalid Fragment");
|
|
|
|
unsigned NumResults = 0;
|
|
|
|
for (Init *I : LI->getValues()) {
|
|
|
|
Record *Op = nullptr;
|
|
|
|
if (DagInit *Dag = dyn_cast<DagInit>(I))
|
|
|
|
if (DefInit *DI = dyn_cast<DefInit>(Dag->getOperator()))
|
|
|
|
Op = DI->getDef();
|
|
|
|
assert(Op && "Invalid Fragment");
|
|
|
|
NumResults = std::max(NumResults, GetNumNodeResults(Op, CDP));
|
|
|
|
}
|
|
|
|
return NumResults;
|
2010-03-20 05:37:09 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-20 05:37:09 +08:00
|
|
|
if (Operator->isSubClassOf("Instruction")) {
|
|
|
|
CodeGenInstruction &InstInfo = CDP.getTargetInfo().getInstruction(Operator);
|
2010-03-28 03:15:02 +08:00
|
|
|
|
2015-03-20 13:09:06 +08:00
|
|
|
unsigned NumDefsToAdd = InstInfo.Operands.NumDefs;
|
|
|
|
|
|
|
|
// Subtract any defaulted outputs.
|
|
|
|
for (unsigned i = 0; i != InstInfo.Operands.NumDefs; ++i) {
|
|
|
|
Record *OperandNode = InstInfo.Operands[i].Rec;
|
|
|
|
|
|
|
|
if (OperandNode->isSubClassOf("OperandWithDefaultOps") &&
|
|
|
|
!CDP.getDefaultOperand(OperandNode).DefaultOps.empty())
|
|
|
|
--NumDefsToAdd;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-28 04:09:24 +08:00
|
|
|
// Add on one implicit def if it has a resolvable type.
|
|
|
|
if (InstInfo.HasOneImplicitDefWithKnownVT(CDP.getTargetInfo()) !=MVT::Other)
|
|
|
|
++NumDefsToAdd;
|
2010-03-28 03:15:02 +08:00
|
|
|
return NumDefsToAdd;
|
2010-03-20 05:37:09 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-20 05:37:09 +08:00
|
|
|
if (Operator->isSubClassOf("SDNodeXForm"))
|
|
|
|
return 1; // FIXME: Generalize SDNodeXForm
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2014-01-03 04:47:05 +08:00
|
|
|
if (Operator->isSubClassOf("ValueType"))
|
|
|
|
return 1; // A type-cast of one result.
|
|
|
|
|
2014-05-20 19:52:46 +08:00
|
|
|
if (Operator->isSubClassOf("ComplexPattern"))
|
|
|
|
return 1;
|
|
|
|
|
2017-01-28 10:02:38 +08:00
|
|
|
errs() << *Operator;
|
2015-05-12 06:17:13 +08:00
|
|
|
PrintFatalError("Unhandled node in GetNumNodeResults");
|
2010-03-20 05:37:09 +08:00
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
|
2009-07-03 08:10:29 +08:00
|
|
|
void TreePatternNode::print(raw_ostream &OS) const {
|
2010-03-20 05:37:09 +08:00
|
|
|
if (isLeaf())
|
2008-01-06 06:25:12 +08:00
|
|
|
OS << *getLeafValue();
|
2010-03-20 05:37:09 +08:00
|
|
|
else
|
2010-02-23 15:50:58 +08:00
|
|
|
OS << '(' << getOperator()->getName();
|
2010-03-20 05:37:09 +08:00
|
|
|
|
2017-09-21 02:01:40 +08:00
|
|
|
for (unsigned i = 0, e = Types.size(); i != e; ++i) {
|
|
|
|
OS << ':';
|
|
|
|
getExtType(i).writeToStream(OS);
|
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
if (!isLeaf()) {
|
|
|
|
if (getNumChildren() != 0) {
|
|
|
|
OS << " ";
|
2018-06-15 04:32:58 +08:00
|
|
|
getChild(0)->print(OS);
|
2008-01-06 06:25:12 +08:00
|
|
|
for (unsigned i = 1, e = getNumChildren(); i != e; ++i) {
|
|
|
|
OS << ", ";
|
2018-06-15 04:32:58 +08:00
|
|
|
getChild(i)->print(OS);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
OS << ")";
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
for (const TreePredicateCall &Pred : PredicateCalls) {
|
|
|
|
OS << "<<P:";
|
|
|
|
if (Pred.Scope)
|
|
|
|
OS << Pred.Scope << ":";
|
|
|
|
OS << Pred.Fn.getFnName() << ">>";
|
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
if (TransformFn)
|
|
|
|
OS << "<<X:" << TransformFn->getName() << ">>";
|
|
|
|
if (!getName().empty())
|
|
|
|
OS << ":$" << getName();
|
|
|
|
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
for (const ScopedName &Name : NamesAsPredicateArg)
|
|
|
|
OS << ":$pred:" << Name.getScope() << ":" << Name.getIdentifier();
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
void TreePatternNode::dump() const {
|
2009-07-03 08:10:29 +08:00
|
|
|
print(errs());
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2008-03-06 01:49:05 +08:00
|
|
|
/// isIsomorphicTo - Return true if this node is recursively
|
|
|
|
/// isomorphic to the specified node. For this comparison, the node's
|
|
|
|
/// entire state is considered. The assigned name is ignored, since
|
|
|
|
/// nodes with differing names are considered isomorphic. However, if
|
|
|
|
/// the assigned name is present in the dependent variable set, then
|
|
|
|
/// the assigned name is considered significant and the node is
|
|
|
|
/// isomorphic if the names match.
|
2018-06-15 04:32:58 +08:00
|
|
|
bool TreePatternNode::isIsomorphicTo(const TreePatternNode *N,
|
2008-03-06 01:49:05 +08:00
|
|
|
const MultipleUseVarSet &DepVars) const {
|
2018-06-15 04:32:58 +08:00
|
|
|
if (N == this) return true;
|
|
|
|
if (N->isLeaf() != isLeaf() || getExtTypes() != N->getExtTypes() ||
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
getPredicateCalls() != N->getPredicateCalls() ||
|
2018-06-15 04:32:58 +08:00
|
|
|
getTransformFn() != N->getTransformFn())
|
2008-01-06 06:25:12 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
if (isLeaf()) {
|
2012-10-11 04:24:43 +08:00
|
|
|
if (DefInit *DI = dyn_cast<DefInit>(getLeafValue())) {
|
2018-06-15 04:32:58 +08:00
|
|
|
if (DefInit *NDI = dyn_cast<DefInit>(N->getLeafValue())) {
|
2008-03-20 09:22:40 +08:00
|
|
|
return ((DI->getDef() == NDI->getDef())
|
|
|
|
&& (DepVars.find(getName()) == DepVars.end()
|
2018-06-15 04:32:58 +08:00
|
|
|
|| getName() == N->getName()));
|
2008-03-06 01:49:05 +08:00
|
|
|
}
|
|
|
|
}
|
2018-06-15 04:32:58 +08:00
|
|
|
return getLeafValue() == N->getLeafValue();
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
if (N->getOperator() != getOperator() ||
|
|
|
|
N->getNumChildren() != getNumChildren()) return false;
|
2008-01-06 06:25:12 +08:00
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
if (!getChild(i)->isIsomorphicTo(N->getChild(i), DepVars))
|
2008-01-06 06:25:12 +08:00
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// clone - Make a copy of this tree and all of its children.
|
|
|
|
///
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr TreePatternNode::clone() const {
|
|
|
|
TreePatternNodePtr New;
|
2008-01-06 06:25:12 +08:00
|
|
|
if (isLeaf()) {
|
2018-05-31 05:00:18 +08:00
|
|
|
New = std::make_shared<TreePatternNode>(getLeafValue(), getNumTypes());
|
2008-01-06 06:25:12 +08:00
|
|
|
} else {
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> CChildren;
|
2008-01-06 06:25:12 +08:00
|
|
|
CChildren.reserve(Children.size());
|
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
CChildren.push_back(getChild(i)->clone());
|
2018-07-15 14:52:49 +08:00
|
|
|
New = std::make_shared<TreePatternNode>(getOperator(), std::move(CChildren),
|
2018-05-31 05:00:18 +08:00
|
|
|
getNumTypes());
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
New->setName(getName());
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
New->setNamesAsPredicateArg(getNamesAsPredicateArg());
|
2010-03-20 05:37:09 +08:00
|
|
|
New->Types = Types;
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
New->setPredicateCalls(getPredicateCalls());
|
2008-01-06 06:25:12 +08:00
|
|
|
New->setTransformFn(getTransformFn());
|
|
|
|
return New;
|
|
|
|
}
|
|
|
|
|
2010-02-15 06:22:58 +08:00
|
|
|
/// RemoveAllTypes - Recursively strip all the types of this tree.
|
|
|
|
void TreePatternNode::RemoveAllTypes() {
|
2015-11-23 04:46:22 +08:00
|
|
|
// Reset to unknown type.
|
2017-09-15 00:56:21 +08:00
|
|
|
std::fill(Types.begin(), Types.end(), TypeSetByHwMode());
|
2010-02-15 06:22:58 +08:00
|
|
|
if (isLeaf()) return;
|
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
getChild(i)->RemoveAllTypes();
|
2010-02-15 06:22:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
/// SubstituteFormalArguments - Replace the formal arguments in this tree
|
|
|
|
/// with actual values specified by ArgMap.
|
2018-05-31 05:00:18 +08:00
|
|
|
void TreePatternNode::SubstituteFormalArguments(
|
|
|
|
std::map<std::string, TreePatternNodePtr> &ArgMap) {
|
2008-01-06 06:25:12 +08:00
|
|
|
if (isLeaf()) return;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i) {
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *Child = getChild(i);
|
|
|
|
if (Child->isLeaf()) {
|
|
|
|
Init *Val = Child->getLeafValue();
|
2014-02-28 08:26:56 +08:00
|
|
|
// Note that, when substituting into an output pattern, Val might be an
|
|
|
|
// UnsetInit.
|
|
|
|
if (isa<UnsetInit>(Val) || (isa<DefInit>(Val) &&
|
|
|
|
cast<DefInit>(Val)->getDef()->getName() == "node")) {
|
2008-01-06 06:25:12 +08:00
|
|
|
// We found a use of a formal argument, replace it with its value.
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNodePtr NewChild = ArgMap[Child->getName()];
|
2008-10-15 14:17:21 +08:00
|
|
|
assert(NewChild && "Couldn't find formal argument!");
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
assert((Child->getPredicateCalls().empty() ||
|
|
|
|
NewChild->getPredicateCalls() == Child->getPredicateCalls()) &&
|
2008-10-15 14:17:21 +08:00
|
|
|
"Non-empty child predicate clobbered!");
|
2018-06-14 19:56:19 +08:00
|
|
|
setChild(i, std::move(NewChild));
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
} else {
|
2018-06-15 04:32:58 +08:00
|
|
|
getChild(i)->SubstituteFormalArguments(ArgMap);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// InlinePatternFragments - If this pattern refers to any pattern
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
/// fragments, return the set of inlined versions (this can be more than
|
|
|
|
/// one if a PatFrags record has multiple alternatives).
|
|
|
|
void TreePatternNode::InlinePatternFragments(
|
|
|
|
TreePatternNodePtr T, TreePattern &TP,
|
|
|
|
std::vector<TreePatternNodePtr> &OutAlternatives) {
|
|
|
|
|
2012-10-26 04:33:17 +08:00
|
|
|
if (TP.hasError())
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (isLeaf()) {
|
|
|
|
OutAlternatives.push_back(T); // nothing to do.
|
|
|
|
return;
|
|
|
|
}
|
2012-10-26 04:33:17 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
Record *Op = getOperator();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (!Op->isSubClassOf("PatFrags")) {
|
|
|
|
if (getNumChildren() == 0) {
|
|
|
|
OutAlternatives.push_back(T);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recursively inline children nodes.
|
|
|
|
std::vector<std::vector<TreePatternNodePtr> > ChildAlternatives;
|
|
|
|
ChildAlternatives.resize(getNumChildren());
|
2008-10-15 14:17:21 +08:00
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i) {
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr Child = getChildShared(i);
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
Child->InlinePatternFragments(Child, TP, ChildAlternatives[i]);
|
|
|
|
// If there are no alternatives for any child, there are no
|
|
|
|
// alternatives for this expression as whole.
|
|
|
|
if (ChildAlternatives[i].empty())
|
|
|
|
return;
|
2008-10-15 14:17:21 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
for (auto NewChild : ChildAlternatives[i])
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
assert((Child->getPredicateCalls().empty() ||
|
|
|
|
NewChild->getPredicateCalls() == Child->getPredicateCalls()) &&
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
"Non-empty child predicate clobbered!");
|
2008-10-15 14:17:21 +08:00
|
|
|
}
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
|
|
|
|
// The end result is an all-pairs construction of the resultant pattern.
|
|
|
|
std::vector<unsigned> Idxs;
|
|
|
|
Idxs.resize(ChildAlternatives.size());
|
|
|
|
bool NotDone;
|
|
|
|
do {
|
|
|
|
// Create the variant and add it to the output list.
|
|
|
|
std::vector<TreePatternNodePtr> NewChildren;
|
|
|
|
for (unsigned i = 0, e = ChildAlternatives.size(); i != e; ++i)
|
|
|
|
NewChildren.push_back(ChildAlternatives[i][Idxs[i]]);
|
|
|
|
TreePatternNodePtr R = std::make_shared<TreePatternNode>(
|
2018-07-15 14:52:49 +08:00
|
|
|
getOperator(), std::move(NewChildren), getNumTypes());
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
|
|
|
|
// Copy over properties.
|
|
|
|
R->setName(getName());
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
R->setNamesAsPredicateArg(getNamesAsPredicateArg());
|
|
|
|
R->setPredicateCalls(getPredicateCalls());
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
R->setTransformFn(getTransformFn());
|
|
|
|
for (unsigned i = 0, e = getNumTypes(); i != e; ++i)
|
|
|
|
R->setType(i, getExtType(i));
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
for (unsigned i = 0, e = getNumResults(); i != e; ++i)
|
|
|
|
R->setResultIndex(i, getResultIndex(i));
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
|
|
|
|
// Register alternative.
|
|
|
|
OutAlternatives.push_back(R);
|
|
|
|
|
|
|
|
// Increment indices to the next permutation by incrementing the
|
|
|
|
// indices from last index backward, e.g., generate the sequence
|
|
|
|
// [0, 0], [0, 1], [1, 0], [1, 1].
|
|
|
|
int IdxsIdx;
|
|
|
|
for (IdxsIdx = Idxs.size() - 1; IdxsIdx >= 0; --IdxsIdx) {
|
|
|
|
if (++Idxs[IdxsIdx] == ChildAlternatives[IdxsIdx].size())
|
|
|
|
Idxs[IdxsIdx] = 0;
|
|
|
|
else
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
NotDone = (IdxsIdx >= 0);
|
|
|
|
} while (NotDone);
|
|
|
|
|
|
|
|
return;
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, we found a reference to a fragment. First, look up its
|
|
|
|
// TreePattern record.
|
|
|
|
TreePattern *Frag = TP.getDAGPatterns().getPatternFragment(Op);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Verify that we are passing the right number of operands.
|
2012-10-26 04:33:17 +08:00
|
|
|
if (Frag->getNumArgs() != Children.size()) {
|
2008-01-06 06:25:12 +08:00
|
|
|
TP.error("'" + Op->getName() + "' fragment requires " +
|
2017-12-29 00:58:54 +08:00
|
|
|
Twine(Frag->getNumArgs()) + " operands!");
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
return;
|
2012-10-26 04:33:17 +08:00
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
TreePredicateFn PredFn(Frag);
|
|
|
|
unsigned Scope = 0;
|
|
|
|
if (TreePredicateFn(Frag).usesOperands())
|
|
|
|
Scope = TP.getDAGPatterns().allocateScope();
|
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// Compute the map of formal to actual arguments.
|
|
|
|
std::map<std::string, TreePatternNodePtr> ArgMap;
|
|
|
|
for (unsigned i = 0, e = Frag->getNumArgs(); i != e; ++i) {
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
TreePatternNodePtr Child = getChildShared(i);
|
|
|
|
if (Scope != 0) {
|
|
|
|
Child = Child->clone();
|
|
|
|
Child->addNameAsPredicateArg(ScopedName(Scope, Frag->getArgName(i)));
|
|
|
|
}
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
ArgMap[Frag->getArgName(i)] = Child;
|
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// Loop over all fragment alternatives.
|
|
|
|
for (auto Alternative : Frag->getTrees()) {
|
|
|
|
TreePatternNodePtr FragTree = Alternative->clone();
|
2008-10-15 14:17:21 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (!PredFn.isAlwaysTrue())
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
FragTree->addPredicateCall(PredFn, Scope);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// Resolve formal arguments to their actual value.
|
|
|
|
if (Frag->getNumArgs())
|
|
|
|
FragTree->SubstituteFormalArguments(ArgMap);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// Transfer types. Note that the resolved alternative may have fewer
|
|
|
|
// (but not more) results than the PatFrags node.
|
|
|
|
FragTree->setName(getName());
|
|
|
|
for (unsigned i = 0, e = FragTree->getNumTypes(); i != e; ++i)
|
|
|
|
FragTree->UpdateNodeType(i, getExtType(i), TP);
|
2008-10-15 14:17:21 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// Transfer in the old predicates.
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
for (const TreePredicateCall &Pred : getPredicateCalls())
|
|
|
|
FragTree->addPredicateCall(Pred);
|
2008-10-15 14:17:21 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// The fragment we inlined could have recursive inlining that is needed. See
|
|
|
|
// if there are any pattern fragments in it and inline them as needed.
|
|
|
|
FragTree->InlinePatternFragments(FragTree, TP, OutAlternatives);
|
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// getImplicitType - Check to see if the specified record has an implicit
|
2009-06-17 12:23:52 +08:00
|
|
|
/// type which should be applied to it. This will infer the type of register
|
2008-01-06 06:25:12 +08:00
|
|
|
/// references from the register file information, for example.
|
|
|
|
///
|
2013-03-24 02:08:44 +08:00
|
|
|
/// When Unnamed is set, return the type of a DAG operand with no name, such as
|
|
|
|
/// the F8RC register class argument in:
|
|
|
|
///
|
|
|
|
/// (COPY_TO_REGCLASS GPR:$src, F8RC)
|
|
|
|
///
|
|
|
|
/// When Unnamed is false, return the type of a named DAG operand such as the
|
|
|
|
/// GPR:$src operand above.
|
|
|
|
///
|
2017-09-15 00:56:21 +08:00
|
|
|
static TypeSetByHwMode getImplicitType(Record *R, unsigned ResNo,
|
|
|
|
bool NotRegisters,
|
|
|
|
bool Unnamed,
|
|
|
|
TreePattern &TP) {
|
|
|
|
CodeGenDAGPatterns &CDP = TP.getDAGPatterns();
|
|
|
|
|
2011-06-28 05:06:21 +08:00
|
|
|
// Check to see if this is a register operand.
|
|
|
|
if (R->isSubClassOf("RegisterOperand")) {
|
|
|
|
assert(ResNo == 0 && "Regoperand ref only has one result!");
|
|
|
|
if (NotRegisters)
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(); // Unknown.
|
2011-06-28 05:06:21 +08:00
|
|
|
Record *RegClass = R->getValueAsDef("RegClass");
|
|
|
|
const CodeGenTarget &T = TP.getDAGPatterns().getTargetInfo();
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(T.getRegisterClass(RegClass).getValueTypes());
|
2011-06-28 05:06:21 +08:00
|
|
|
}
|
|
|
|
|
2010-03-15 14:00:16 +08:00
|
|
|
// Check to see if this is a register or a register class.
|
2008-01-06 06:25:12 +08:00
|
|
|
if (R->isSubClassOf("RegisterClass")) {
|
2010-03-24 07:50:31 +08:00
|
|
|
assert(ResNo == 0 && "Regclass ref only has one result!");
|
2013-03-24 02:08:44 +08:00
|
|
|
// An unnamed register class represents itself as an i32 immediate, for
|
|
|
|
// example on a COPY_TO_REGCLASS instruction.
|
|
|
|
if (Unnamed)
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(MVT::i32);
|
2013-03-24 02:08:44 +08:00
|
|
|
|
|
|
|
// In a named operand, the register class provides the possible set of
|
|
|
|
// types.
|
2010-12-22 00:16:00 +08:00
|
|
|
if (NotRegisters)
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(); // Unknown.
|
2010-03-15 14:00:16 +08:00
|
|
|
const CodeGenTarget &T = TP.getDAGPatterns().getTargetInfo();
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(T.getRegisterClass(R).getValueTypes());
|
2010-03-24 07:50:31 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (R->isSubClassOf("PatFrags")) {
|
2010-03-24 07:50:31 +08:00
|
|
|
assert(ResNo == 0 && "FIXME: PatFrag with multiple results?");
|
2008-01-06 06:25:12 +08:00
|
|
|
// Pattern fragment types will be resolved when they are inlined.
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(); // Unknown.
|
2010-03-24 07:50:31 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-24 07:50:31 +08:00
|
|
|
if (R->isSubClassOf("Register")) {
|
|
|
|
assert(ResNo == 0 && "Registers only produce one result!");
|
2010-12-22 00:16:00 +08:00
|
|
|
if (NotRegisters)
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(); // Unknown.
|
2008-01-06 06:25:12 +08:00
|
|
|
const CodeGenTarget &T = TP.getDAGPatterns().getTargetInfo();
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(T.getRegisterVTs(R));
|
2010-03-24 07:50:31 +08:00
|
|
|
}
|
2010-05-24 22:48:12 +08:00
|
|
|
|
|
|
|
if (R->isSubClassOf("SubRegIndex")) {
|
|
|
|
assert(ResNo == 0 && "SubRegisterIndices only produce one result!");
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(MVT::i32);
|
2010-05-24 22:48:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2013-03-24 04:35:01 +08:00
|
|
|
if (R->isSubClassOf("ValueType")) {
|
2010-03-24 07:50:31 +08:00
|
|
|
assert(ResNo == 0 && "This node only has one result!");
|
2013-03-24 04:35:01 +08:00
|
|
|
// An unnamed VTSDNode represents itself as an MVT::Other immediate.
|
|
|
|
//
|
|
|
|
// (sext_inreg GPR:$src, i16)
|
|
|
|
// ~~~
|
|
|
|
if (Unnamed)
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(MVT::Other);
|
2013-03-24 04:35:01 +08:00
|
|
|
// With a name, the ValueType simply provides the type of the named
|
|
|
|
// variable.
|
|
|
|
//
|
|
|
|
// (sext_inreg i32:$src, i16)
|
|
|
|
// ~~~~~~~~
|
2013-03-24 08:56:16 +08:00
|
|
|
if (NotRegisters)
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(); // Unknown.
|
|
|
|
const CodeGenHwModes &CGH = CDP.getTargetInfo().getHwModes();
|
|
|
|
return TypeSetByHwMode(getValueTypeByHwMode(R, CGH));
|
2013-03-24 04:35:01 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (R->isSubClassOf("CondCode")) {
|
|
|
|
assert(ResNo == 0 && "This node only has one result!");
|
|
|
|
// Using a CondCodeSDNode.
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(MVT::Other);
|
2010-03-24 07:50:31 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-24 07:50:31 +08:00
|
|
|
if (R->isSubClassOf("ComplexPattern")) {
|
|
|
|
assert(ResNo == 0 && "FIXME: ComplexPattern with multiple results?");
|
2010-12-22 00:16:00 +08:00
|
|
|
if (NotRegisters)
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(); // Unknown.
|
|
|
|
return TypeSetByHwMode(CDP.getComplexPattern(R).getValueType());
|
2010-03-24 07:50:31 +08:00
|
|
|
}
|
|
|
|
if (R->isSubClassOf("PointerLikeRegClass")) {
|
|
|
|
assert(ResNo == 0 && "Regclass can only have one result!");
|
2017-09-15 00:56:21 +08:00
|
|
|
TypeSetByHwMode VTS(MVT::iPTR);
|
|
|
|
TP.getInfer().expandOverloads(VTS);
|
|
|
|
return VTS;
|
2010-03-24 07:50:31 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-24 07:50:31 +08:00
|
|
|
if (R->getName() == "node" || R->getName() == "srcvalue" ||
|
2019-03-10 13:21:52 +08:00
|
|
|
R->getName() == "zero_reg" || R->getName() == "immAllOnesV" ||
|
2019-05-30 15:30:37 +08:00
|
|
|
R->getName() == "immAllZerosV" || R->getName() == "undef_tied_input") {
|
2008-01-06 06:25:12 +08:00
|
|
|
// Placeholder.
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(); // Unknown.
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
if (R->isSubClassOf("Operand")) {
|
|
|
|
const CodeGenHwModes &CGH = CDP.getTargetInfo().getHwModes();
|
|
|
|
Record *T = R->getValueAsDef("Type");
|
|
|
|
return TypeSetByHwMode(getValueTypeByHwMode(T, CGH));
|
|
|
|
}
|
2014-05-20 19:52:46 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
TP.error("Unknown node flavor used in pattern: " + R->getName());
|
2017-09-15 00:56:21 +08:00
|
|
|
return TypeSetByHwMode(MVT::Other);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2008-01-06 13:36:50 +08:00
|
|
|
|
|
|
|
/// getIntrinsicInfo - If this node corresponds to an intrinsic, return the
|
|
|
|
/// CodeGenIntrinsic information for it, otherwise return a null pointer.
|
|
|
|
const CodeGenIntrinsic *TreePatternNode::
|
|
|
|
getIntrinsicInfo(const CodeGenDAGPatterns &CDP) const {
|
|
|
|
if (getOperator() != CDP.get_intrinsic_void_sdnode() &&
|
|
|
|
getOperator() != CDP.get_intrinsic_w_chain_sdnode() &&
|
|
|
|
getOperator() != CDP.get_intrinsic_wo_chain_sdnode())
|
2014-04-15 15:20:03 +08:00
|
|
|
return nullptr;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
unsigned IID = cast<IntInit>(getChild(0)->getLeafValue())->getValue();
|
2008-01-06 13:36:50 +08:00
|
|
|
return &CDP.getIntrinsicInfo(IID);
|
|
|
|
}
|
|
|
|
|
2010-02-15 06:22:58 +08:00
|
|
|
/// getComplexPatternInfo - If this node corresponds to a ComplexPattern,
|
|
|
|
/// return the ComplexPattern information, otherwise return null.
|
|
|
|
const ComplexPattern *
|
|
|
|
TreePatternNode::getComplexPatternInfo(const CodeGenDAGPatterns &CGP) const {
|
2014-05-20 19:52:46 +08:00
|
|
|
Record *Rec;
|
|
|
|
if (isLeaf()) {
|
|
|
|
DefInit *DI = dyn_cast<DefInit>(getLeafValue());
|
|
|
|
if (!DI)
|
|
|
|
return nullptr;
|
|
|
|
Rec = DI->getDef();
|
|
|
|
} else
|
|
|
|
Rec = getOperator();
|
|
|
|
|
|
|
|
if (!Rec->isSubClassOf("ComplexPattern"))
|
|
|
|
return nullptr;
|
|
|
|
return &CGP.getComplexPattern(Rec);
|
|
|
|
}
|
|
|
|
|
|
|
|
unsigned TreePatternNode::getNumMIResults(const CodeGenDAGPatterns &CGP) const {
|
|
|
|
// A ComplexPattern specifically declares how many results it fills in.
|
|
|
|
if (const ComplexPattern *CP = getComplexPatternInfo(CGP))
|
|
|
|
return CP->getNumOperands();
|
|
|
|
|
|
|
|
// If MIOperandInfo is specified, that gives the count.
|
|
|
|
if (isLeaf()) {
|
|
|
|
DefInit *DI = dyn_cast<DefInit>(getLeafValue());
|
|
|
|
if (DI && DI->getDef()->isSubClassOf("Operand")) {
|
|
|
|
DagInit *MIOps = DI->getDef()->getValueAsDag("MIOperandInfo");
|
|
|
|
if (MIOps->getNumArgs())
|
|
|
|
return MIOps->getNumArgs();
|
|
|
|
}
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2014-05-20 19:52:46 +08:00
|
|
|
// Otherwise there is just one result.
|
|
|
|
return 1;
|
2010-02-15 06:22:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// NodeHasProperty - Return true if this node has the specified property.
|
|
|
|
bool TreePatternNode::NodeHasProperty(SDNP Property,
|
2010-02-15 06:33:49 +08:00
|
|
|
const CodeGenDAGPatterns &CGP) const {
|
2010-02-15 06:22:58 +08:00
|
|
|
if (isLeaf()) {
|
|
|
|
if (const ComplexPattern *CP = getComplexPatternInfo(CGP))
|
|
|
|
return CP->hasProperty(Property);
|
2017-12-21 03:36:28 +08:00
|
|
|
|
2010-02-15 06:22:58 +08:00
|
|
|
return false;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-12-21 03:36:28 +08:00
|
|
|
if (Property != SDNPHasChain) {
|
|
|
|
// The chain proprety is already present on the different intrinsic node
|
|
|
|
// types (intrinsic_w_chain, intrinsic_void), and is not explicitly listed
|
|
|
|
// on the intrinsic. Anything else is specific to the individual intrinsic.
|
|
|
|
if (const CodeGenIntrinsic *Int = getIntrinsicInfo(CGP))
|
|
|
|
return Int->hasProperty(Property);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!Operator->isSubClassOf("SDPatternOperator"))
|
|
|
|
return false;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-02-15 06:22:58 +08:00
|
|
|
return CGP.getSDNodeInfo(Operator).hasProperty(Property);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/// TreeHasProperty - Return true if any node in this tree has the specified
|
|
|
|
/// property.
|
|
|
|
bool TreePatternNode::TreeHasProperty(SDNP Property,
|
2010-02-15 06:33:49 +08:00
|
|
|
const CodeGenDAGPatterns &CGP) const {
|
2010-02-15 06:22:58 +08:00
|
|
|
if (NodeHasProperty(Property, CGP))
|
|
|
|
return true;
|
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
if (getChild(i)->TreeHasProperty(Property, CGP))
|
2010-02-15 06:22:58 +08:00
|
|
|
return true;
|
|
|
|
return false;
|
2010-12-22 00:16:00 +08:00
|
|
|
}
|
2010-02-15 06:22:58 +08:00
|
|
|
|
2008-06-17 04:29:38 +08:00
|
|
|
/// isCommutativeIntrinsic - Return true if the node corresponds to a
|
|
|
|
/// commutative intrinsic.
|
|
|
|
bool
|
|
|
|
TreePatternNode::isCommutativeIntrinsic(const CodeGenDAGPatterns &CDP) const {
|
|
|
|
if (const CodeGenIntrinsic *Int = getIntrinsicInfo(CDP))
|
|
|
|
return Int->isCommutative;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
static bool isOperandClass(const TreePatternNode *N, StringRef Class) {
|
|
|
|
if (!N->isLeaf())
|
|
|
|
return N->getOperator()->isSubClassOf(Class);
|
2008-01-06 13:36:50 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
DefInit *DI = dyn_cast<DefInit>(N->getLeafValue());
|
2014-11-03 07:46:51 +08:00
|
|
|
if (DI && DI->getDef()->isSubClassOf(Class))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2014-12-12 06:27:14 +08:00
|
|
|
|
|
|
|
static void emitTooManyOperandsError(TreePattern &TP,
|
|
|
|
StringRef InstName,
|
|
|
|
unsigned Expected,
|
|
|
|
unsigned Actual) {
|
|
|
|
TP.error("Instruction '" + InstName + "' was provided " + Twine(Actual) +
|
|
|
|
" operands but expected only " + Twine(Expected) + "!");
|
|
|
|
}
|
|
|
|
|
|
|
|
static void emitTooFewOperandsError(TreePattern &TP,
|
|
|
|
StringRef InstName,
|
|
|
|
unsigned Actual) {
|
|
|
|
TP.error("Instruction '" + InstName +
|
|
|
|
"' expects more than the provided " + Twine(Actual) + " operands!");
|
|
|
|
}
|
|
|
|
|
2009-01-06 01:23:09 +08:00
|
|
|
/// ApplyTypeConstraints - Apply all of the type constraints relevant to
|
2008-01-06 06:25:12 +08:00
|
|
|
/// this node and its children in the tree. This returns true if it makes a
|
2012-10-26 04:33:17 +08:00
|
|
|
/// change, false otherwise. If a type contradiction is found, flag an error.
|
2008-01-06 06:25:12 +08:00
|
|
|
bool TreePatternNode::ApplyTypeConstraints(TreePattern &TP, bool NotRegisters) {
|
2012-10-26 04:33:17 +08:00
|
|
|
if (TP.hasError())
|
|
|
|
return false;
|
|
|
|
|
2008-01-06 09:10:31 +08:00
|
|
|
CodeGenDAGPatterns &CDP = TP.getDAGPatterns();
|
2008-01-06 06:25:12 +08:00
|
|
|
if (isLeaf()) {
|
2012-10-11 04:24:43 +08:00
|
|
|
if (DefInit *DI = dyn_cast<DefInit>(getLeafValue())) {
|
2008-01-06 06:25:12 +08:00
|
|
|
// If it's a regclass or something else known, include the type.
|
2010-03-20 05:37:09 +08:00
|
|
|
bool MadeChange = false;
|
|
|
|
for (unsigned i = 0, e = Types.size(); i != e; ++i)
|
|
|
|
MadeChange |= UpdateNodeType(i, getImplicitType(DI->getDef(), i,
|
2013-03-24 02:08:44 +08:00
|
|
|
NotRegisters,
|
|
|
|
!hasName(), TP), TP);
|
2010-03-20 05:37:09 +08:00
|
|
|
return MadeChange;
|
2010-02-15 05:10:15 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2012-10-11 04:24:43 +08:00
|
|
|
if (IntInit *II = dyn_cast<IntInit>(getLeafValue())) {
|
2010-03-20 05:37:09 +08:00
|
|
|
assert(Types.size() == 1 && "Invalid IntInit");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Int inits are always integers. :)
|
2017-09-15 00:56:21 +08:00
|
|
|
bool MadeChange = TP.getInfer().EnforceInteger(Types[0]);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
if (!TP.getInfer().isConcrete(Types[0], false))
|
2010-03-15 14:00:16 +08:00
|
|
|
return MadeChange;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
ValueTypeByHwMode VVT = TP.getInfer().getConcrete(Types[0], false);
|
|
|
|
for (auto &P : VVT) {
|
|
|
|
MVT::SimpleValueType VT = P.second.SimpleTy;
|
|
|
|
if (VT == MVT::iPTR || VT == MVT::iPTRAny)
|
|
|
|
continue;
|
|
|
|
unsigned Size = MVT(VT).getSizeInBits();
|
|
|
|
// Make sure that the value is representable for this type.
|
|
|
|
if (Size >= 32)
|
|
|
|
continue;
|
|
|
|
// Check that the value doesn't use more bits than we have. It must
|
|
|
|
// either be a sign- or zero-extended equivalent of the original.
|
|
|
|
int64_t SignBitAndAbove = II->getValue() >> (Size - 1);
|
|
|
|
if (SignBitAndAbove == -1 || SignBitAndAbove == 0 ||
|
|
|
|
SignBitAndAbove == 1)
|
|
|
|
continue;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-12-29 00:58:54 +08:00
|
|
|
TP.error("Integer value '" + Twine(II->getValue()) +
|
2017-09-15 00:56:21 +08:00
|
|
|
"' is out of range for type '" + getEnumName(VT) + "'!");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return MadeChange;
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2017-09-15 00:56:21 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
return false;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-02-23 13:51:07 +08:00
|
|
|
if (const CodeGenIntrinsic *Int = getIntrinsicInfo(CDP)) {
|
2008-01-06 06:25:12 +08:00
|
|
|
bool MadeChange = false;
|
2008-06-06 20:08:01 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Apply the result type to the node.
|
2008-11-13 17:08:33 +08:00
|
|
|
unsigned NumRetVTs = Int->IS.RetVTs.size();
|
|
|
|
unsigned NumParamVTs = Int->IS.ParamVTs.size();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-11-13 17:08:33 +08:00
|
|
|
for (unsigned i = 0, e = NumRetVTs; i != e; ++i)
|
2010-03-20 05:37:09 +08:00
|
|
|
MadeChange |= UpdateNodeType(i, Int->IS.RetVTs[i], TP);
|
2008-11-13 17:08:33 +08:00
|
|
|
|
2012-10-26 04:33:17 +08:00
|
|
|
if (getNumChildren() != NumParamVTs + 1) {
|
2017-12-29 00:58:54 +08:00
|
|
|
TP.error("Intrinsic '" + Int->Name + "' expects " + Twine(NumParamVTs) +
|
|
|
|
" operands, not " + Twine(getNumChildren() - 1) + " operands!");
|
2012-10-26 04:33:17 +08:00
|
|
|
return false;
|
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
// Apply type info to the intrinsic ID.
|
2018-06-15 04:32:58 +08:00
|
|
|
MadeChange |= getChild(0)->UpdateNodeType(0, MVT::iPTR, TP);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-20 05:37:09 +08:00
|
|
|
for (unsigned i = 0, e = getNumChildren()-1; i != e; ++i) {
|
2018-06-15 04:32:58 +08:00
|
|
|
MadeChange |= getChild(i+1)->ApplyTypeConstraints(TP, NotRegisters);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-20 05:37:09 +08:00
|
|
|
MVT::SimpleValueType OpVT = Int->IS.ParamVTs[i];
|
2018-06-15 04:32:58 +08:00
|
|
|
assert(getChild(i+1)->getNumTypes() == 1 && "Unhandled case");
|
|
|
|
MadeChange |= getChild(i+1)->UpdateNodeType(0, OpVT, TP);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
return MadeChange;
|
2010-02-23 13:51:07 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-02-23 13:51:07 +08:00
|
|
|
if (getOperator()->isSubClassOf("SDNode")) {
|
2008-01-06 06:25:12 +08:00
|
|
|
const SDNodeInfo &NI = CDP.getSDNodeInfo(getOperator());
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-28 16:48:47 +08:00
|
|
|
// Check that the number of operands is sane. Negative operands -> varargs.
|
|
|
|
if (NI.getNumOperands() >= 0 &&
|
2012-10-26 04:33:17 +08:00
|
|
|
getNumChildren() != (unsigned)NI.getNumOperands()) {
|
2010-03-28 16:48:47 +08:00
|
|
|
TP.error(getOperator()->getName() + " node requires exactly " +
|
2017-12-29 00:58:54 +08:00
|
|
|
Twine(NI.getNumOperands()) + " operands!");
|
2012-10-26 04:33:17 +08:00
|
|
|
return false;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
bool MadeChange = false;
|
2008-01-06 06:25:12 +08:00
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
MadeChange |= getChild(i)->ApplyTypeConstraints(TP, NotRegisters);
|
|
|
|
MadeChange |= NI.ApplyTypeConstraints(this, TP);
|
2010-03-20 05:37:09 +08:00
|
|
|
return MadeChange;
|
2010-02-23 13:51:07 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-02-23 13:51:07 +08:00
|
|
|
if (getOperator()->isSubClassOf("Instruction")) {
|
2008-01-06 06:25:12 +08:00
|
|
|
const DAGInstruction &Inst = CDP.getInstruction(getOperator());
|
|
|
|
CodeGenInstruction &InstInfo =
|
2010-03-19 08:07:20 +08:00
|
|
|
CDP.getTargetInfo().getInstruction(getOperator());
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-28 03:15:02 +08:00
|
|
|
bool MadeChange = false;
|
|
|
|
|
|
|
|
// Apply the result types to the node, these come from the things in the
|
|
|
|
// (outs) list of the instruction.
|
2015-03-20 13:09:06 +08:00
|
|
|
unsigned NumResultsToAdd = std::min(InstInfo.Operands.NumDefs,
|
|
|
|
Inst.getNumResults());
|
2013-03-18 12:08:07 +08:00
|
|
|
for (unsigned ResNo = 0; ResNo != NumResultsToAdd; ++ResNo)
|
|
|
|
MadeChange |= UpdateNodeTypeFromInst(ResNo, Inst.getResult(ResNo), TP);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-28 03:15:02 +08:00
|
|
|
// If the instruction has implicit defs, we apply the first one as a result.
|
|
|
|
// FIXME: This sucks, it should apply all implicit defs.
|
|
|
|
if (!InstInfo.ImplicitDefs.empty()) {
|
|
|
|
unsigned ResNo = NumResultsToAdd;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-28 04:09:24 +08:00
|
|
|
// FIXME: Generalize to multiple possible types and multiple possible
|
|
|
|
// ImplicitDefs.
|
|
|
|
MVT::SimpleValueType VT =
|
|
|
|
InstInfo.HasOneImplicitDefWithKnownVT(CDP.getTargetInfo());
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-28 04:09:24 +08:00
|
|
|
if (VT != MVT::Other)
|
|
|
|
MadeChange |= UpdateNodeType(ResNo, VT, TP);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-15 14:00:16 +08:00
|
|
|
// If this is an INSERT_SUBREG, constrain the source and destination VTs to
|
|
|
|
// be the same.
|
|
|
|
if (getOperator()->getName() == "INSERT_SUBREG") {
|
2018-06-15 04:32:58 +08:00
|
|
|
assert(getChild(0)->getNumTypes() == 1 && "FIXME: Unhandled");
|
|
|
|
MadeChange |= UpdateNodeType(0, getChild(0)->getExtType(0), TP);
|
|
|
|
MadeChange |= getChild(0)->UpdateNodeType(0, getExtType(0), TP);
|
2014-11-03 07:46:51 +08:00
|
|
|
} else if (getOperator()->getName() == "REG_SEQUENCE") {
|
|
|
|
// We need to do extra, custom typechecking for REG_SEQUENCE since it is
|
|
|
|
// variadic.
|
|
|
|
|
|
|
|
unsigned NChild = getNumChildren();
|
|
|
|
if (NChild < 3) {
|
|
|
|
TP.error("REG_SEQUENCE requires at least 3 operands!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (NChild % 2 == 0) {
|
|
|
|
TP.error("REG_SEQUENCE requires an odd number of operands!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isOperandClass(getChild(0), "RegisterClass")) {
|
|
|
|
TP.error("REG_SEQUENCE requires a RegisterClass for first operand!");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned I = 1; I < NChild; I += 2) {
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *SubIdxChild = getChild(I + 1);
|
2014-11-03 07:46:51 +08:00
|
|
|
if (!isOperandClass(SubIdxChild, "SubRegIndex")) {
|
|
|
|
TP.error("REG_SEQUENCE requires a SubRegIndex for operand " +
|
2017-12-29 00:58:54 +08:00
|
|
|
Twine(I + 1) + "!");
|
2014-11-03 07:46:51 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
|
[TableGen] Allow DAG isel patterns to override default operands.
When a Tablegen instruction description uses `OperandWithDefaultOps`,
isel patterns for that instruction don't have to fill in the default
value for the operand in question. But the flip side is that they
actually //can't// override the defaults even if they want to.
This will be very inconvenient for the Arm backend, when we start
wanting to write isel patterns that generate the many MVE predicated
vector instructions, in the form with predication actually enabled. So
this small Tablegen fix makes it possible to write an isel pattern
either with or without values for a defaulted operand, and have the
default values filled in only if they are not overridden.
If all the defaulted operands come at the end of the instruction's
operand list, there's a natural way to match them up to the arguments
supplied in the pattern: consume pattern arguments until you run out,
then fill in any missing instruction operands with their default
values. But if defaulted and non-defaulted operands are interleaved,
it's less clear what to do. This does happen in existing targets (the
first example I came across was KILLGT, in the AMDGPU/R600 backend),
and of course they expect the previous behaviour (that the default for
those operands is used and a pattern argument is not consumed), so for
backwards compatibility I've stuck with that.
Reviewers: nhaehnle, hfinkel, dmgreen
Subscribers: mehdi_amini, javed.absar, tpr, kristof.beyls, steven_wu, dexonsmith, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63814
llvm-svn: 365114
2019-07-04 16:43:20 +08:00
|
|
|
// If one or more operands with a default value appear at the end of the
|
|
|
|
// formal operand list for an instruction, we allow them to be overridden
|
|
|
|
// by optional operands provided in the pattern.
|
|
|
|
//
|
|
|
|
// But if an operand B without a default appears at any point after an
|
|
|
|
// operand A with a default, then we don't allow A to be overridden,
|
|
|
|
// because there would be no way to specify whether the next operand in
|
|
|
|
// the pattern was intended to override A or skip it.
|
|
|
|
unsigned NonOverridableOperands = Inst.getNumOperands();
|
|
|
|
while (NonOverridableOperands > 0 &&
|
|
|
|
CDP.operandHasDefault(Inst.getOperand(NonOverridableOperands-1)))
|
|
|
|
--NonOverridableOperands;
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
unsigned ChildNo = 0;
|
|
|
|
for (unsigned i = 0, e = Inst.getNumOperands(); i != e; ++i) {
|
|
|
|
Record *OperandNode = Inst.getOperand(i);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
[TableGen] Allow DAG isel patterns to override default operands.
When a Tablegen instruction description uses `OperandWithDefaultOps`,
isel patterns for that instruction don't have to fill in the default
value for the operand in question. But the flip side is that they
actually //can't// override the defaults even if they want to.
This will be very inconvenient for the Arm backend, when we start
wanting to write isel patterns that generate the many MVE predicated
vector instructions, in the form with predication actually enabled. So
this small Tablegen fix makes it possible to write an isel pattern
either with or without values for a defaulted operand, and have the
default values filled in only if they are not overridden.
If all the defaulted operands come at the end of the instruction's
operand list, there's a natural way to match them up to the arguments
supplied in the pattern: consume pattern arguments until you run out,
then fill in any missing instruction operands with their default
values. But if defaulted and non-defaulted operands are interleaved,
it's less clear what to do. This does happen in existing targets (the
first example I came across was KILLGT, in the AMDGPU/R600 backend),
and of course they expect the previous behaviour (that the default for
those operands is used and a pattern argument is not consumed), so for
backwards compatibility I've stuck with that.
Reviewers: nhaehnle, hfinkel, dmgreen
Subscribers: mehdi_amini, javed.absar, tpr, kristof.beyls, steven_wu, dexonsmith, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63814
llvm-svn: 365114
2019-07-04 16:43:20 +08:00
|
|
|
// If the operand has a default value, do we use it? We must use the
|
|
|
|
// default if we've run out of children of the pattern DAG to consume,
|
|
|
|
// or if the operand is followed by a non-defaulted one.
|
|
|
|
if (CDP.operandHasDefault(OperandNode) &&
|
|
|
|
(i < NonOverridableOperands || ChildNo >= getNumChildren()))
|
2008-01-06 06:25:12 +08:00
|
|
|
continue;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
[TableGen] Allow DAG isel patterns to override default operands.
When a Tablegen instruction description uses `OperandWithDefaultOps`,
isel patterns for that instruction don't have to fill in the default
value for the operand in question. But the flip side is that they
actually //can't// override the defaults even if they want to.
This will be very inconvenient for the Arm backend, when we start
wanting to write isel patterns that generate the many MVE predicated
vector instructions, in the form with predication actually enabled. So
this small Tablegen fix makes it possible to write an isel pattern
either with or without values for a defaulted operand, and have the
default values filled in only if they are not overridden.
If all the defaulted operands come at the end of the instruction's
operand list, there's a natural way to match them up to the arguments
supplied in the pattern: consume pattern arguments until you run out,
then fill in any missing instruction operands with their default
values. But if defaulted and non-defaulted operands are interleaved,
it's less clear what to do. This does happen in existing targets (the
first example I came across was KILLGT, in the AMDGPU/R600 backend),
and of course they expect the previous behaviour (that the default for
those operands is used and a pattern argument is not consumed), so for
backwards compatibility I've stuck with that.
Reviewers: nhaehnle, hfinkel, dmgreen
Subscribers: mehdi_amini, javed.absar, tpr, kristof.beyls, steven_wu, dexonsmith, llvm-commits
Tags: #llvm
Differential Revision: https://reviews.llvm.org/D63814
llvm-svn: 365114
2019-07-04 16:43:20 +08:00
|
|
|
// If we have run out of child nodes and there _isn't_ a default
|
|
|
|
// value we can use for the next operand, give an error.
|
2012-10-26 04:33:17 +08:00
|
|
|
if (ChildNo >= getNumChildren()) {
|
2014-12-12 06:27:14 +08:00
|
|
|
emitTooFewOperandsError(TP, getOperator()->getName(), getNumChildren());
|
2012-10-26 04:33:17 +08:00
|
|
|
return false;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *Child = getChild(ChildNo++);
|
2010-03-28 03:15:02 +08:00
|
|
|
unsigned ChildResNo = 0; // Instructions always use res #0 of their op.
|
Extend TableGen instruction selection matcher to improve handling
of complex instruction operands (e.g. address modes).
Currently, if a Pat pattern creates an instruction that has a complex
operand (i.e. one that consists of multiple sub-operands at the MI
level), this operand must match a ComplexPattern DAG pattern with the
correct number of output operands.
This commit extends TableGen to alternatively allow match a complex
operands against multiple separate operands at the DAG level.
This allows using Pat patterns to match pre-increment nodes like
pre_store (which must have separate operands at the DAG level) onto
an instruction pattern that uses a multi-operand memory operand,
like the following example on PowerPC (will be committed as a
follow-on patch):
def STWU : DForm_1<37, (outs ptr_rc:$ea_res), (ins GPRC:$rS, memri:$dst),
"stwu $rS, $dst", LdStStoreUpd, []>,
RegConstraint<"$dst.reg = $ea_res">, NoEncode<"$ea_res">;
def : Pat<(pre_store GPRC:$rS, ptr_rc:$ptrreg, iaddroff:$ptroff),
(STWU GPRC:$rS, iaddroff:$ptroff, ptr_rc:$ptrreg)>;
Here, the pair of "ptroff" and "ptrreg" operands is matched onto the
complex operand "dst" of class "memri" in the "STWU" instruction.
Approved by Jakob Stoklund Olesen.
llvm-svn: 177428
2013-03-20 03:51:09 +08:00
|
|
|
|
|
|
|
// If the operand has sub-operands, they may be provided by distinct
|
|
|
|
// child patterns, so attempt to match each sub-operand separately.
|
|
|
|
if (OperandNode->isSubClassOf("Operand")) {
|
|
|
|
DagInit *MIOpInfo = OperandNode->getValueAsDag("MIOperandInfo");
|
|
|
|
if (unsigned NumArgs = MIOpInfo->getNumArgs()) {
|
|
|
|
// But don't do that if the whole operand is being provided by
|
2014-05-22 19:56:09 +08:00
|
|
|
// a single ComplexPattern-related Operand.
|
|
|
|
|
|
|
|
if (Child->getNumMIResults(CDP) < NumArgs) {
|
Extend TableGen instruction selection matcher to improve handling
of complex instruction operands (e.g. address modes).
Currently, if a Pat pattern creates an instruction that has a complex
operand (i.e. one that consists of multiple sub-operands at the MI
level), this operand must match a ComplexPattern DAG pattern with the
correct number of output operands.
This commit extends TableGen to alternatively allow match a complex
operands against multiple separate operands at the DAG level.
This allows using Pat patterns to match pre-increment nodes like
pre_store (which must have separate operands at the DAG level) onto
an instruction pattern that uses a multi-operand memory operand,
like the following example on PowerPC (will be committed as a
follow-on patch):
def STWU : DForm_1<37, (outs ptr_rc:$ea_res), (ins GPRC:$rS, memri:$dst),
"stwu $rS, $dst", LdStStoreUpd, []>,
RegConstraint<"$dst.reg = $ea_res">, NoEncode<"$ea_res">;
def : Pat<(pre_store GPRC:$rS, ptr_rc:$ptrreg, iaddroff:$ptroff),
(STWU GPRC:$rS, iaddroff:$ptroff, ptr_rc:$ptrreg)>;
Here, the pair of "ptroff" and "ptrreg" operands is matched onto the
complex operand "dst" of class "memri" in the "STWU" instruction.
Approved by Jakob Stoklund Olesen.
llvm-svn: 177428
2013-03-20 03:51:09 +08:00
|
|
|
// Match first sub-operand against the child we already have.
|
|
|
|
Record *SubRec = cast<DefInit>(MIOpInfo->getArg(0))->getDef();
|
|
|
|
MadeChange |=
|
|
|
|
Child->UpdateNodeTypeFromInst(ChildResNo, SubRec, TP);
|
|
|
|
|
|
|
|
// And the remaining sub-operands against subsequent children.
|
|
|
|
for (unsigned Arg = 1; Arg < NumArgs; ++Arg) {
|
|
|
|
if (ChildNo >= getNumChildren()) {
|
2014-12-12 06:27:14 +08:00
|
|
|
emitTooFewOperandsError(TP, getOperator()->getName(),
|
|
|
|
getNumChildren());
|
Extend TableGen instruction selection matcher to improve handling
of complex instruction operands (e.g. address modes).
Currently, if a Pat pattern creates an instruction that has a complex
operand (i.e. one that consists of multiple sub-operands at the MI
level), this operand must match a ComplexPattern DAG pattern with the
correct number of output operands.
This commit extends TableGen to alternatively allow match a complex
operands against multiple separate operands at the DAG level.
This allows using Pat patterns to match pre-increment nodes like
pre_store (which must have separate operands at the DAG level) onto
an instruction pattern that uses a multi-operand memory operand,
like the following example on PowerPC (will be committed as a
follow-on patch):
def STWU : DForm_1<37, (outs ptr_rc:$ea_res), (ins GPRC:$rS, memri:$dst),
"stwu $rS, $dst", LdStStoreUpd, []>,
RegConstraint<"$dst.reg = $ea_res">, NoEncode<"$ea_res">;
def : Pat<(pre_store GPRC:$rS, ptr_rc:$ptrreg, iaddroff:$ptroff),
(STWU GPRC:$rS, iaddroff:$ptroff, ptr_rc:$ptrreg)>;
Here, the pair of "ptroff" and "ptrreg" operands is matched onto the
complex operand "dst" of class "memri" in the "STWU" instruction.
Approved by Jakob Stoklund Olesen.
llvm-svn: 177428
2013-03-20 03:51:09 +08:00
|
|
|
return false;
|
|
|
|
}
|
2018-06-15 04:32:58 +08:00
|
|
|
Child = getChild(ChildNo++);
|
Extend TableGen instruction selection matcher to improve handling
of complex instruction operands (e.g. address modes).
Currently, if a Pat pattern creates an instruction that has a complex
operand (i.e. one that consists of multiple sub-operands at the MI
level), this operand must match a ComplexPattern DAG pattern with the
correct number of output operands.
This commit extends TableGen to alternatively allow match a complex
operands against multiple separate operands at the DAG level.
This allows using Pat patterns to match pre-increment nodes like
pre_store (which must have separate operands at the DAG level) onto
an instruction pattern that uses a multi-operand memory operand,
like the following example on PowerPC (will be committed as a
follow-on patch):
def STWU : DForm_1<37, (outs ptr_rc:$ea_res), (ins GPRC:$rS, memri:$dst),
"stwu $rS, $dst", LdStStoreUpd, []>,
RegConstraint<"$dst.reg = $ea_res">, NoEncode<"$ea_res">;
def : Pat<(pre_store GPRC:$rS, ptr_rc:$ptrreg, iaddroff:$ptroff),
(STWU GPRC:$rS, iaddroff:$ptroff, ptr_rc:$ptrreg)>;
Here, the pair of "ptroff" and "ptrreg" operands is matched onto the
complex operand "dst" of class "memri" in the "STWU" instruction.
Approved by Jakob Stoklund Olesen.
llvm-svn: 177428
2013-03-20 03:51:09 +08:00
|
|
|
|
|
|
|
SubRec = cast<DefInit>(MIOpInfo->getArg(Arg))->getDef();
|
|
|
|
MadeChange |=
|
|
|
|
Child->UpdateNodeTypeFromInst(ChildResNo, SubRec, TP);
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we didn't match by pieces above, attempt to match the whole
|
|
|
|
// operand now.
|
2013-03-18 12:08:07 +08:00
|
|
|
MadeChange |= Child->UpdateNodeTypeFromInst(ChildResNo, OperandNode, TP);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2008-03-11 17:33:47 +08:00
|
|
|
|
2014-11-03 07:46:51 +08:00
|
|
|
if (!InstInfo.Operands.isVariadic && ChildNo != getNumChildren()) {
|
2014-12-12 06:27:14 +08:00
|
|
|
emitTooManyOperandsError(TP, getOperator()->getName(),
|
|
|
|
ChildNo, getNumChildren());
|
2012-10-26 04:33:17 +08:00
|
|
|
return false;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
Extend TableGen instruction selection matcher to improve handling
of complex instruction operands (e.g. address modes).
Currently, if a Pat pattern creates an instruction that has a complex
operand (i.e. one that consists of multiple sub-operands at the MI
level), this operand must match a ComplexPattern DAG pattern with the
correct number of output operands.
This commit extends TableGen to alternatively allow match a complex
operands against multiple separate operands at the DAG level.
This allows using Pat patterns to match pre-increment nodes like
pre_store (which must have separate operands at the DAG level) onto
an instruction pattern that uses a multi-operand memory operand,
like the following example on PowerPC (will be committed as a
follow-on patch):
def STWU : DForm_1<37, (outs ptr_rc:$ea_res), (ins GPRC:$rS, memri:$dst),
"stwu $rS, $dst", LdStStoreUpd, []>,
RegConstraint<"$dst.reg = $ea_res">, NoEncode<"$ea_res">;
def : Pat<(pre_store GPRC:$rS, ptr_rc:$ptrreg, iaddroff:$ptroff),
(STWU GPRC:$rS, iaddroff:$ptroff, ptr_rc:$ptrreg)>;
Here, the pair of "ptroff" and "ptrreg" operands is matched onto the
complex operand "dst" of class "memri" in the "STWU" instruction.
Approved by Jakob Stoklund Olesen.
llvm-svn: 177428
2013-03-20 03:51:09 +08:00
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
MadeChange |= getChild(i)->ApplyTypeConstraints(TP, NotRegisters);
|
2008-01-06 06:25:12 +08:00
|
|
|
return MadeChange;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2014-05-20 19:52:46 +08:00
|
|
|
if (getOperator()->isSubClassOf("ComplexPattern")) {
|
|
|
|
bool MadeChange = false;
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < getNumChildren(); ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
MadeChange |= getChild(i)->ApplyTypeConstraints(TP, NotRegisters);
|
2014-05-20 19:52:46 +08:00
|
|
|
|
|
|
|
return MadeChange;
|
|
|
|
}
|
|
|
|
|
2010-02-23 13:51:07 +08:00
|
|
|
assert(getOperator()->isSubClassOf("SDNodeXForm") && "Unknown node type!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-02-23 13:51:07 +08:00
|
|
|
// Node transforms always take one operand.
|
2012-10-26 04:33:17 +08:00
|
|
|
if (getNumChildren() != 1) {
|
2010-02-23 13:51:07 +08:00
|
|
|
TP.error("Node transform '" + getOperator()->getName() +
|
|
|
|
"' requires one operand!");
|
2012-10-26 04:33:17 +08:00
|
|
|
return false;
|
|
|
|
}
|
2010-02-23 13:51:07 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
bool MadeChange = getChild(0)->ApplyTypeConstraints(TP, NotRegisters);
|
2010-03-15 14:00:16 +08:00
|
|
|
return MadeChange;
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// OnlyOnRHSOfCommutative - Return true if this value is only allowed on the
|
|
|
|
/// RHS of a commutative operation, not the on LHS.
|
2018-06-15 04:32:58 +08:00
|
|
|
static bool OnlyOnRHSOfCommutative(TreePatternNode *N) {
|
|
|
|
if (!N->isLeaf() && N->getOperator()->getName() == "imm")
|
2008-01-06 06:25:12 +08:00
|
|
|
return true;
|
2018-06-15 04:32:58 +08:00
|
|
|
if (N->isLeaf() && isa<IntInit>(N->getLeafValue()))
|
2008-01-06 06:25:12 +08:00
|
|
|
return true;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// canPatternMatch - If it is impossible for this pattern to match on this
|
|
|
|
/// target, fill in Reason and return false. Otherwise, return true. This is
|
2009-03-27 00:17:51 +08:00
|
|
|
/// used as a sanity check for .td files (to prevent people from writing stuff
|
2008-01-06 06:25:12 +08:00
|
|
|
/// that can never possibly work), and to prevent the pattern permuter from
|
|
|
|
/// generating stuff that is useless.
|
2010-12-22 00:16:00 +08:00
|
|
|
bool TreePatternNode::canPatternMatch(std::string &Reason,
|
2008-04-03 08:02:49 +08:00
|
|
|
const CodeGenDAGPatterns &CDP) {
|
2008-01-06 06:25:12 +08:00
|
|
|
if (isLeaf()) return true;
|
|
|
|
|
|
|
|
for (unsigned i = 0, e = getNumChildren(); i != e; ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
if (!getChild(i)->canPatternMatch(Reason, CDP))
|
2008-01-06 06:25:12 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// If this is an intrinsic, handle cases that would make it not match. For
|
|
|
|
// example, if an operand is required to be an immediate.
|
|
|
|
if (getOperator()->isSubClassOf("Intrinsic")) {
|
|
|
|
// TODO:
|
|
|
|
return true;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2014-05-20 19:52:46 +08:00
|
|
|
if (getOperator()->isSubClassOf("ComplexPattern"))
|
|
|
|
return true;
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// If this node is a commutative operator, check that the LHS isn't an
|
|
|
|
// immediate.
|
|
|
|
const SDNodeInfo &NodeInfo = CDP.getSDNodeInfo(getOperator());
|
2008-06-17 04:29:38 +08:00
|
|
|
bool isCommIntrinsic = isCommutativeIntrinsic(CDP);
|
|
|
|
if (NodeInfo.hasProperty(SDNPCommutative) || isCommIntrinsic) {
|
2008-01-06 06:25:12 +08:00
|
|
|
// Scan all of the operands of the node and make sure that only the last one
|
|
|
|
// is a constant node, unless the RHS also is.
|
|
|
|
if (!OnlyOnRHSOfCommutative(getChild(getNumChildren()-1))) {
|
2016-12-19 16:35:08 +08:00
|
|
|
unsigned Skip = isCommIntrinsic ? 1 : 0; // First operand is intrinsic id.
|
2008-06-17 04:29:38 +08:00
|
|
|
for (unsigned i = Skip, e = getNumChildren()-1; i != e; ++i)
|
2008-01-06 06:25:12 +08:00
|
|
|
if (OnlyOnRHSOfCommutative(getChild(i))) {
|
|
|
|
Reason="Immediate value must be on the RHS of commutative operators!";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// TreePattern implementation
|
|
|
|
//
|
|
|
|
|
2011-07-30 06:43:06 +08:00
|
|
|
TreePattern::TreePattern(Record *TheRec, ListInit *RawPat, bool isInput,
|
2012-10-26 04:33:17 +08:00
|
|
|
CodeGenDAGPatterns &cdp) : TheRecord(TheRec), CDP(cdp),
|
2017-09-15 00:56:21 +08:00
|
|
|
isInputPattern(isInput), HasError(false),
|
|
|
|
Infer(*this) {
|
2015-06-02 12:15:51 +08:00
|
|
|
for (Init *I : RawPat->getValues())
|
|
|
|
Trees.push_back(ParseTreePattern(I, ""));
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2011-07-30 06:43:06 +08:00
|
|
|
TreePattern::TreePattern(Record *TheRec, DagInit *Pat, bool isInput,
|
2012-10-26 04:33:17 +08:00
|
|
|
CodeGenDAGPatterns &cdp) : TheRecord(TheRec), CDP(cdp),
|
2017-09-15 00:56:21 +08:00
|
|
|
isInputPattern(isInput), HasError(false),
|
|
|
|
Infer(*this) {
|
2010-03-28 14:50:34 +08:00
|
|
|
Trees.push_back(ParseTreePattern(Pat, ""));
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePattern::TreePattern(Record *TheRec, TreePatternNodePtr Pat, bool isInput,
|
|
|
|
CodeGenDAGPatterns &cdp)
|
|
|
|
: TheRecord(TheRec), CDP(cdp), isInputPattern(isInput), HasError(false),
|
|
|
|
Infer(*this) {
|
2014-11-18 06:55:41 +08:00
|
|
|
Trees.push_back(Pat);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2014-11-12 07:48:11 +08:00
|
|
|
void TreePattern::error(const Twine &Msg) {
|
2012-10-26 04:33:17 +08:00
|
|
|
if (HasError)
|
|
|
|
return;
|
2008-01-06 06:25:12 +08:00
|
|
|
dump();
|
2012-10-26 04:33:17 +08:00
|
|
|
PrintError(TheRecord->getLoc(), "In " + TheRecord->getName() + ": " + Msg);
|
|
|
|
HasError = true;
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2010-03-15 14:00:16 +08:00
|
|
|
void TreePattern::ComputeNamedNodes() {
|
2018-06-15 04:32:58 +08:00
|
|
|
for (TreePatternNodePtr &Tree : Trees)
|
|
|
|
ComputeNamedNodes(Tree.get());
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
void TreePattern::ComputeNamedNodes(TreePatternNode *N) {
|
2010-03-15 14:00:16 +08:00
|
|
|
if (!N->getName().empty())
|
2018-06-15 04:32:58 +08:00
|
|
|
NamedNodes[N->getName()].push_back(N);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-15 14:00:16 +08:00
|
|
|
for (unsigned i = 0, e = N->getNumChildren(); i != e; ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
ComputeNamedNodes(N->getChild(i));
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr TreePattern::ParseTreePattern(Init *TheInit,
|
|
|
|
StringRef OpName) {
|
2012-10-11 04:24:43 +08:00
|
|
|
if (DefInit *DI = dyn_cast<DefInit>(TheInit)) {
|
2010-03-28 14:50:34 +08:00
|
|
|
Record *R = DI->getDef();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-28 14:50:34 +08:00
|
|
|
// Direct reference to a leaf DagNode or PatFrag? Turn it into a
|
2011-07-07 07:38:13 +08:00
|
|
|
// TreePatternNode of its own. For example:
|
2010-03-28 14:50:34 +08:00
|
|
|
/// (foo GPR, imm) -> (foo GPR, (imm))
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (R->isSubClassOf("SDNode") || R->isSubClassOf("PatFrags"))
|
2011-07-30 03:07:07 +08:00
|
|
|
return ParseTreePattern(
|
2016-12-05 14:00:41 +08:00
|
|
|
DagInit::get(DI, nullptr,
|
2016-12-05 14:00:46 +08:00
|
|
|
std::vector<std::pair<Init*, StringInit*> >()),
|
2011-07-30 03:07:07 +08:00
|
|
|
OpName);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-28 14:50:34 +08:00
|
|
|
// Input argument?
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr Res = std::make_shared<TreePatternNode>(DI, 1);
|
2010-03-28 16:48:47 +08:00
|
|
|
if (R->getName() == "node" && !OpName.empty()) {
|
2010-03-28 14:50:34 +08:00
|
|
|
if (OpName.empty())
|
|
|
|
error("'node' argument requires a name to match with operand list");
|
|
|
|
Args.push_back(OpName);
|
|
|
|
}
|
|
|
|
|
|
|
|
Res->setName(OpName);
|
|
|
|
return Res;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2013-03-25 03:37:00 +08:00
|
|
|
// ?:$name or just $name.
|
2015-04-22 10:09:45 +08:00
|
|
|
if (isa<UnsetInit>(TheInit)) {
|
2013-03-25 03:37:00 +08:00
|
|
|
if (OpName.empty())
|
|
|
|
error("'?' argument requires a name to match with operand list");
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr Res = std::make_shared<TreePatternNode>(TheInit, 1);
|
2013-03-25 03:37:00 +08:00
|
|
|
Args.push_back(OpName);
|
|
|
|
Res->setName(OpName);
|
|
|
|
return Res;
|
|
|
|
}
|
|
|
|
|
2018-06-04 22:45:12 +08:00
|
|
|
if (isa<IntInit>(TheInit) || isa<BitInit>(TheInit)) {
|
2010-03-28 14:50:34 +08:00
|
|
|
if (!OpName.empty())
|
2018-06-04 22:45:12 +08:00
|
|
|
error("Constant int or bit argument should not have a name!");
|
|
|
|
if (isa<BitInit>(TheInit))
|
|
|
|
TheInit = TheInit->convertInitializerTo(IntRecTy::get());
|
|
|
|
return std::make_shared<TreePatternNode>(TheInit, 1);
|
2010-03-28 14:50:34 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2012-10-11 04:24:43 +08:00
|
|
|
if (BitsInit *BI = dyn_cast<BitsInit>(TheInit)) {
|
2010-03-28 14:50:34 +08:00
|
|
|
// Turn this into an IntInit.
|
2011-07-30 06:43:06 +08:00
|
|
|
Init *II = BI->convertInitializerTo(IntRecTy::get());
|
2014-04-15 15:20:03 +08:00
|
|
|
if (!II || !isa<IntInit>(II))
|
2010-03-28 14:50:34 +08:00
|
|
|
error("Bits value must be constants!");
|
2010-03-28 14:57:56 +08:00
|
|
|
return ParseTreePattern(II, OpName);
|
2010-03-28 14:50:34 +08:00
|
|
|
}
|
|
|
|
|
2012-10-11 04:24:43 +08:00
|
|
|
DagInit *Dag = dyn_cast<DagInit>(TheInit);
|
2010-03-28 14:50:34 +08:00
|
|
|
if (!Dag) {
|
2017-01-28 10:02:38 +08:00
|
|
|
TheInit->print(errs());
|
2010-03-28 14:50:34 +08:00
|
|
|
error("Pattern has unexpected init kind!");
|
|
|
|
}
|
2012-10-11 04:24:43 +08:00
|
|
|
DefInit *OpDef = dyn_cast<DefInit>(Dag->getOperator());
|
2008-01-06 06:25:12 +08:00
|
|
|
if (!OpDef) error("Pattern has unexpected operator type!");
|
|
|
|
Record *Operator = OpDef->getDef();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
if (Operator->isSubClassOf("ValueType")) {
|
|
|
|
// If the operator is a ValueType, then this must be "type cast" of a leaf
|
|
|
|
// node.
|
|
|
|
if (Dag->getNumArgs() != 1)
|
|
|
|
error("Type cast only takes one operand!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr New =
|
|
|
|
ParseTreePattern(Dag->getArg(0), Dag->getArgNameStr(0));
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Apply the type cast.
|
2010-03-20 05:37:09 +08:00
|
|
|
assert(New->getNumTypes() == 1 && "FIXME: Unhandled");
|
2017-09-15 00:56:21 +08:00
|
|
|
const CodeGenHwModes &CGH = getDAGPatterns().getTargetInfo().getHwModes();
|
|
|
|
New->UpdateNodeType(0, getValueTypeByHwMode(Operator, CGH), *this);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-28 14:50:34 +08:00
|
|
|
if (!OpName.empty())
|
|
|
|
error("ValueType cast should not have a name!");
|
2008-01-06 06:25:12 +08:00
|
|
|
return New;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Verify that this is something that makes sense for an operator.
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (!Operator->isSubClassOf("PatFrags") &&
|
2009-03-19 13:21:56 +08:00
|
|
|
!Operator->isSubClassOf("SDNode") &&
|
2010-12-22 00:16:00 +08:00
|
|
|
!Operator->isSubClassOf("Instruction") &&
|
2008-01-06 06:25:12 +08:00
|
|
|
!Operator->isSubClassOf("SDNodeXForm") &&
|
|
|
|
!Operator->isSubClassOf("Intrinsic") &&
|
2014-05-20 19:52:46 +08:00
|
|
|
!Operator->isSubClassOf("ComplexPattern") &&
|
2008-01-06 06:25:12 +08:00
|
|
|
Operator->getName() != "set" &&
|
2010-03-27 10:53:27 +08:00
|
|
|
Operator->getName() != "implicit")
|
2008-01-06 06:25:12 +08:00
|
|
|
error("Unrecognized node '" + Operator->getName() + "'!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Check to see if this is something that is illegal in an input pattern.
|
2010-03-28 14:57:56 +08:00
|
|
|
if (isInputPattern) {
|
|
|
|
if (Operator->isSubClassOf("Instruction") ||
|
|
|
|
Operator->isSubClassOf("SDNodeXForm"))
|
|
|
|
error("Cannot use '" + Operator->getName() + "' in an input pattern!");
|
|
|
|
} else {
|
|
|
|
if (Operator->isSubClassOf("Intrinsic"))
|
|
|
|
error("Cannot use '" + Operator->getName() + "' in an output pattern!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-28 14:57:56 +08:00
|
|
|
if (Operator->isSubClassOf("SDNode") &&
|
|
|
|
Operator->getName() != "imm" &&
|
2019-09-23 03:49:39 +08:00
|
|
|
Operator->getName() != "timm" &&
|
2010-03-28 14:57:56 +08:00
|
|
|
Operator->getName() != "fpimm" &&
|
|
|
|
Operator->getName() != "tglobaltlsaddr" &&
|
|
|
|
Operator->getName() != "tconstpool" &&
|
|
|
|
Operator->getName() != "tjumptable" &&
|
|
|
|
Operator->getName() != "tframeindex" &&
|
|
|
|
Operator->getName() != "texternalsym" &&
|
|
|
|
Operator->getName() != "tblockaddress" &&
|
|
|
|
Operator->getName() != "tglobaladdr" &&
|
|
|
|
Operator->getName() != "bb" &&
|
2015-06-23 01:46:53 +08:00
|
|
|
Operator->getName() != "vt" &&
|
|
|
|
Operator->getName() != "mcsym")
|
2010-03-28 14:57:56 +08:00
|
|
|
error("Cannot use '" + Operator->getName() + "' in an output pattern!");
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> Children;
|
2010-03-28 14:50:34 +08:00
|
|
|
|
|
|
|
// Parse all the operands.
|
|
|
|
for (unsigned i = 0, e = Dag->getNumArgs(); i != e; ++i)
|
2016-12-05 14:00:46 +08:00
|
|
|
Children.push_back(ParseTreePattern(Dag->getArg(i), Dag->getArgNameStr(i)));
|
2010-12-22 00:16:00 +08:00
|
|
|
|
[TableGen] Add support of Intrinsics with multiple returns
This change deals with intrinsics with multiple outputs, for example load
instrinsic with address updated.
DAG selection for Instrinsics could be done either through source code or
tablegen. Handling all intrinsics in source code would introduce a huge chunk
of repetitive code if we have a large number of intrinsic that return multiple
values (see NVPTX as an example). While intrinsic class in tablegen supports
multiple outputs, tablegen only supports Intrinsics with zero or one output on
TreePattern. This appears to be a simple bug in tablegen that is fixed by this
change.
For Intrinsics defined as:
def int_xxx_load_addr_updated: Intrinsic<[llvm_i32_ty, llvm_ptr_ty], [llvm_ptr_ty, llvm_i32_ty], []>;
Instruction will be defined as:
def L32_X: Inst<(outs reg:$d1, reg:$d2), (ins reg:$s1, reg:$s2), "ld32_x $d1, $d2, $s2", [(set i32:$d1, i32:$d2, (int_xxx_load_addr_updated i32:$s1, i32:$s2))]>;
Patch by Wenbo Sun, thanks!
Differential Revision: https://reviews.llvm.org/D32888
llvm-svn: 321704
2018-01-03 19:35:09 +08:00
|
|
|
// Get the actual number of results before Operator is converted to an intrinsic
|
|
|
|
// node (which is hard-coded to have either zero or one result).
|
|
|
|
unsigned NumResults = GetNumNodeResults(Operator, CDP);
|
|
|
|
|
2018-03-31 06:22:31 +08:00
|
|
|
// If the operator is an intrinsic, then this is just syntactic sugar for
|
2010-12-22 00:16:00 +08:00
|
|
|
// (intrinsic_* <number>, ..children..). Pick the right intrinsic node, and
|
2008-01-06 06:25:12 +08:00
|
|
|
// convert the intrinsic name to a number.
|
|
|
|
if (Operator->isSubClassOf("Intrinsic")) {
|
|
|
|
const CodeGenIntrinsic &Int = getDAGPatterns().getIntrinsic(Operator);
|
|
|
|
unsigned IID = getDAGPatterns().getIntrinsicID(Operator)+1;
|
|
|
|
|
|
|
|
// If this intrinsic returns void, it must have side-effects and thus a
|
|
|
|
// chain.
|
2010-03-28 14:50:34 +08:00
|
|
|
if (Int.IS.RetVTs.empty())
|
2008-01-06 06:25:12 +08:00
|
|
|
Operator = getDAGPatterns().get_intrinsic_void_sdnode();
|
2019-07-17 18:53:13 +08:00
|
|
|
else if (Int.ModRef != CodeGenIntrinsic::NoMem || Int.hasSideEffects)
|
2008-01-06 06:25:12 +08:00
|
|
|
// Has side-effects, requires chain.
|
|
|
|
Operator = getDAGPatterns().get_intrinsic_w_chain_sdnode();
|
2010-03-28 14:50:34 +08:00
|
|
|
else // Otherwise, no chain.
|
2008-01-06 06:25:12 +08:00
|
|
|
Operator = getDAGPatterns().get_intrinsic_wo_chain_sdnode();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-06-14 19:56:19 +08:00
|
|
|
Children.insert(Children.begin(),
|
|
|
|
std::make_shared<TreePatternNode>(IntInit::get(IID), 1));
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2014-05-20 19:52:46 +08:00
|
|
|
if (Operator->isSubClassOf("ComplexPattern")) {
|
|
|
|
for (unsigned i = 0; i < Children.size(); ++i) {
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNodePtr Child = Children[i];
|
2014-05-20 19:52:46 +08:00
|
|
|
|
|
|
|
if (Child->getName().empty())
|
|
|
|
error("All arguments to a ComplexPattern must be named");
|
|
|
|
|
|
|
|
// Check that the ComplexPattern uses are consistent: "(MY_PAT $a, $b)"
|
|
|
|
// and "(MY_PAT $b, $a)" should not be allowed in the same pattern;
|
|
|
|
// neither should "(MY_PAT_1 $a, $b)" and "(MY_PAT_2 $a, $b)".
|
|
|
|
auto OperandId = std::make_pair(Operator, i);
|
|
|
|
auto PrevOp = ComplexPatternOperands.find(Child->getName());
|
|
|
|
if (PrevOp != ComplexPatternOperands.end()) {
|
|
|
|
if (PrevOp->getValue() != OperandId)
|
|
|
|
error("All ComplexPattern operands must appear consistently: "
|
|
|
|
"in the same order in just one ComplexPattern instance.");
|
|
|
|
} else
|
|
|
|
ComplexPatternOperands[Child->getName()] = OperandId;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNodePtr Result =
|
2018-07-15 14:52:49 +08:00
|
|
|
std::make_shared<TreePatternNode>(Operator, std::move(Children),
|
|
|
|
NumResults);
|
2010-03-28 14:50:34 +08:00
|
|
|
Result->setName(OpName);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2016-12-05 14:00:41 +08:00
|
|
|
if (Dag->getName()) {
|
2010-03-28 14:50:34 +08:00
|
|
|
assert(Result->getName().empty());
|
2016-12-05 14:00:41 +08:00
|
|
|
Result->setName(Dag->getNameStr());
|
2010-03-28 14:50:34 +08:00
|
|
|
}
|
2009-03-19 13:21:56 +08:00
|
|
|
return Result;
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2010-03-28 16:38:32 +08:00
|
|
|
/// SimplifyTree - See if we can simplify this tree to eliminate something that
|
|
|
|
/// will never match in favor of something obvious that will. This is here
|
|
|
|
/// strictly as a convenience to target authors because it allows them to write
|
|
|
|
/// more type generic things and have useless type casts fold away.
|
|
|
|
///
|
|
|
|
/// This returns true if any change is made.
|
2018-05-31 05:00:18 +08:00
|
|
|
static bool SimplifyTree(TreePatternNodePtr &N) {
|
2010-03-28 16:38:32 +08:00
|
|
|
if (N->isLeaf())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// If we have a bitconvert with a resolved type and if the source and
|
|
|
|
// destination types are the same, then the bitconvert is useless, remove it.
|
|
|
|
if (N->getOperator()->getName() == "bitconvert" &&
|
2017-09-15 00:56:21 +08:00
|
|
|
N->getExtType(0).isValueTypeByHwMode(false) &&
|
2018-06-15 04:32:58 +08:00
|
|
|
N->getExtType(0) == N->getChild(0)->getExtType(0) &&
|
2010-03-28 16:38:32 +08:00
|
|
|
N->getName().empty()) {
|
2018-05-31 05:00:18 +08:00
|
|
|
N = N->getChildShared(0);
|
2010-03-28 16:38:32 +08:00
|
|
|
SimplifyTree(N);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Walk all children.
|
|
|
|
bool MadeChange = false;
|
|
|
|
for (unsigned i = 0, e = N->getNumChildren(); i != e; ++i) {
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr Child = N->getChildShared(i);
|
2010-03-28 16:38:32 +08:00
|
|
|
MadeChange |= SimplifyTree(Child);
|
2018-06-14 19:56:19 +08:00
|
|
|
N->setChild(i, std::move(Child));
|
2010-03-28 16:38:32 +08:00
|
|
|
}
|
|
|
|
return MadeChange;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
/// InferAllTypes - Infer/propagate as many types throughout the expression
|
2009-03-27 00:17:51 +08:00
|
|
|
/// patterns as possible. Return true if all types are inferred, false
|
2012-10-26 04:33:17 +08:00
|
|
|
/// otherwise. Flags an error if a type contradiction is found.
|
2010-03-15 14:00:16 +08:00
|
|
|
bool TreePattern::
|
|
|
|
InferAllTypes(const StringMap<SmallVector<TreePatternNode*,1> > *InNamedTypes) {
|
|
|
|
if (NamedNodes.empty())
|
|
|
|
ComputeNamedNodes();
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
bool MadeChange = true;
|
|
|
|
while (MadeChange) {
|
|
|
|
MadeChange = false;
|
2018-05-31 05:00:18 +08:00
|
|
|
for (TreePatternNodePtr &Tree : Trees) {
|
2015-11-23 04:46:24 +08:00
|
|
|
MadeChange |= Tree->ApplyTypeConstraints(*this, false);
|
|
|
|
MadeChange |= SimplifyTree(Tree);
|
2010-03-28 16:38:32 +08:00
|
|
|
}
|
2010-03-15 14:00:16 +08:00
|
|
|
|
|
|
|
// If there are constraints on our named nodes, apply them.
|
2015-11-23 04:46:24 +08:00
|
|
|
for (auto &Entry : NamedNodes) {
|
|
|
|
SmallVectorImpl<TreePatternNode*> &Nodes = Entry.second;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-15 14:00:16 +08:00
|
|
|
// If we have input named node types, propagate their types to the named
|
|
|
|
// values here.
|
|
|
|
if (InNamedTypes) {
|
2015-11-23 04:46:24 +08:00
|
|
|
if (!InNamedTypes->count(Entry.getKey())) {
|
|
|
|
error("Node '" + std::string(Entry.getKey()) +
|
2014-07-10 02:55:49 +08:00
|
|
|
"' in output pattern but not input pattern");
|
|
|
|
return true;
|
|
|
|
}
|
2010-03-15 14:00:16 +08:00
|
|
|
|
|
|
|
const SmallVectorImpl<TreePatternNode*> &InNodes =
|
2015-11-23 04:46:24 +08:00
|
|
|
InNamedTypes->find(Entry.getKey())->second;
|
2010-03-15 14:00:16 +08:00
|
|
|
|
|
|
|
// The input types should be fully resolved by now.
|
2015-11-23 04:46:24 +08:00
|
|
|
for (TreePatternNode *Node : Nodes) {
|
2010-03-15 14:00:16 +08:00
|
|
|
// If this node is a register class, and it is the root of the pattern
|
|
|
|
// then we're mapping something onto an input register. We allow
|
|
|
|
// changing the type of the input register in this case. This allows
|
|
|
|
// us to match things like:
|
|
|
|
// def : Pat<(v1i64 (bitconvert(v2i32 DPR:$src))), (v1i64 DPR:$src)>;
|
2018-05-31 05:00:18 +08:00
|
|
|
if (Node == Trees[0].get() && Node->isLeaf()) {
|
2015-11-23 04:46:24 +08:00
|
|
|
DefInit *DI = dyn_cast<DefInit>(Node->getLeafValue());
|
2011-06-28 05:06:21 +08:00
|
|
|
if (DI && (DI->getDef()->isSubClassOf("RegisterClass") ||
|
|
|
|
DI->getDef()->isSubClassOf("RegisterOperand")))
|
2010-03-15 14:00:16 +08:00
|
|
|
continue;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2015-11-23 04:46:24 +08:00
|
|
|
assert(Node->getNumTypes() == 1 &&
|
2010-03-20 05:37:09 +08:00
|
|
|
InNodes[0]->getNumTypes() == 1 &&
|
|
|
|
"FIXME: cannot name multiple result nodes yet");
|
2015-11-23 04:46:24 +08:00
|
|
|
MadeChange |= Node->UpdateNodeType(0, InNodes[0]->getExtType(0),
|
|
|
|
*this);
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-15 14:00:16 +08:00
|
|
|
// If there are multiple nodes with the same name, they must all have the
|
|
|
|
// same type.
|
2015-11-23 04:46:24 +08:00
|
|
|
if (Entry.second.size() > 1) {
|
2010-03-15 14:00:16 +08:00
|
|
|
for (unsigned i = 0, e = Nodes.size()-1; i != e; ++i) {
|
2010-03-20 05:37:09 +08:00
|
|
|
TreePatternNode *N1 = Nodes[i], *N2 = Nodes[i+1];
|
2010-03-21 09:38:21 +08:00
|
|
|
assert(N1->getNumTypes() == 1 && N2->getNumTypes() == 1 &&
|
2010-03-20 05:37:09 +08:00
|
|
|
"FIXME: cannot name multiple result nodes yet");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-20 05:37:09 +08:00
|
|
|
MadeChange |= N1->UpdateNodeType(0, N2->getExtType(0), *this);
|
|
|
|
MadeChange |= N2->UpdateNodeType(0, N1->getExtType(0), *this);
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
bool HasUnresolvedTypes = false;
|
2018-05-31 05:00:18 +08:00
|
|
|
for (const TreePatternNodePtr &Tree : Trees)
|
2017-09-15 00:56:21 +08:00
|
|
|
HasUnresolvedTypes |= Tree->ContainsUnresolvedType(*this);
|
2008-01-06 06:25:12 +08:00
|
|
|
return !HasUnresolvedTypes;
|
|
|
|
}
|
|
|
|
|
2009-07-03 08:10:29 +08:00
|
|
|
void TreePattern::print(raw_ostream &OS) const {
|
2008-01-06 06:25:12 +08:00
|
|
|
OS << getRecord()->getName();
|
|
|
|
if (!Args.empty()) {
|
|
|
|
OS << "(" << Args[0];
|
|
|
|
for (unsigned i = 1, e = Args.size(); i != e; ++i)
|
|
|
|
OS << ", " << Args[i];
|
|
|
|
OS << ")";
|
|
|
|
}
|
|
|
|
OS << ": ";
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
if (Trees.size() > 1)
|
|
|
|
OS << "[\n";
|
2018-05-31 05:00:18 +08:00
|
|
|
for (const TreePatternNodePtr &Tree : Trees) {
|
2008-01-06 06:25:12 +08:00
|
|
|
OS << "\t";
|
2015-11-23 04:46:24 +08:00
|
|
|
Tree->print(OS);
|
2008-01-06 06:25:12 +08:00
|
|
|
OS << "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Trees.size() > 1)
|
|
|
|
OS << "]\n";
|
|
|
|
}
|
|
|
|
|
2009-07-03 08:10:29 +08:00
|
|
|
void TreePattern::dump() const { print(errs()); }
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
2008-01-06 09:10:31 +08:00
|
|
|
// CodeGenDAGPatterns implementation
|
2008-01-06 06:25:12 +08:00
|
|
|
//
|
|
|
|
|
2017-11-11 11:23:44 +08:00
|
|
|
CodeGenDAGPatterns::CodeGenDAGPatterns(RecordKeeper &R,
|
|
|
|
PatternRewriterFn PatternRewriter)
|
|
|
|
: Records(R), Target(R), LegalVTS(Target.getLegalValueTypes()),
|
|
|
|
PatternRewriter(PatternRewriter) {
|
2010-12-13 08:23:57 +08:00
|
|
|
|
2019-12-11 23:37:16 +08:00
|
|
|
Intrinsics = CodeGenIntrinsicTable(Records);
|
2008-01-06 06:25:12 +08:00
|
|
|
ParseNodeInfo();
|
2008-01-06 06:54:53 +08:00
|
|
|
ParseNodeTransforms();
|
2008-01-06 06:25:12 +08:00
|
|
|
ParseComplexPatterns();
|
2008-01-06 06:43:57 +08:00
|
|
|
ParsePatternFragments();
|
2008-01-06 06:25:12 +08:00
|
|
|
ParseDefaultOperands();
|
|
|
|
ParseInstructions();
|
2014-02-28 08:26:56 +08:00
|
|
|
ParsePatternFragments(/*OutFrags*/true);
|
2008-01-06 06:25:12 +08:00
|
|
|
ParsePatterns();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// Break patterns with parameterized types into a series of patterns,
|
|
|
|
// where each one has a fixed type and is predicated on the conditions
|
|
|
|
// of the associated HW mode.
|
|
|
|
ExpandHwModeBasedTypes();
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Generate variants. For example, commutative patterns can match
|
|
|
|
// multiple ways. Add them to PatternsToMatch as well.
|
|
|
|
GenerateVariants();
|
2008-04-03 08:02:49 +08:00
|
|
|
|
|
|
|
// Infer instruction flags. For example, we can detect loads,
|
|
|
|
// stores, and side effects in many cases by examining an
|
|
|
|
// instruction's pattern.
|
|
|
|
InferInstructionFlags();
|
2012-08-28 11:26:49 +08:00
|
|
|
|
|
|
|
// Verify that instruction flags match the patterns.
|
|
|
|
VerifyInstructionFlags();
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2017-10-14 03:00:01 +08:00
|
|
|
Record *CodeGenDAGPatterns::getSDNodeNamed(const std::string &Name) const {
|
2008-01-06 06:25:12 +08:00
|
|
|
Record *N = Records.getDef(Name);
|
2015-05-12 06:17:13 +08:00
|
|
|
if (!N || !N->isSubClassOf("SDNode"))
|
|
|
|
PrintFatalError("Error getting SDNode '" + Name + "'!");
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
return N;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parse all of the SDNode definitions for the target, populating SDNodes.
|
2008-01-06 09:10:31 +08:00
|
|
|
void CodeGenDAGPatterns::ParseNodeInfo() {
|
2008-01-06 06:25:12 +08:00
|
|
|
std::vector<Record*> Nodes = Records.getAllDerivedDefinitions("SDNode");
|
2017-09-15 00:56:21 +08:00
|
|
|
const CodeGenHwModes &CGH = getTargetInfo().getHwModes();
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
while (!Nodes.empty()) {
|
2017-09-15 00:56:21 +08:00
|
|
|
Record *R = Nodes.back();
|
|
|
|
SDNodes.insert(std::make_pair(R, SDNodeInfo(R, CGH)));
|
2008-01-06 06:25:12 +08:00
|
|
|
Nodes.pop_back();
|
|
|
|
}
|
|
|
|
|
2009-03-27 00:17:51 +08:00
|
|
|
// Get the builtin intrinsic nodes.
|
2008-01-06 06:25:12 +08:00
|
|
|
intrinsic_void_sdnode = getSDNodeNamed("intrinsic_void");
|
|
|
|
intrinsic_w_chain_sdnode = getSDNodeNamed("intrinsic_w_chain");
|
|
|
|
intrinsic_wo_chain_sdnode = getSDNodeNamed("intrinsic_wo_chain");
|
|
|
|
}
|
|
|
|
|
|
|
|
/// ParseNodeTransforms - Parse all SDNodeXForm instances into the SDNodeXForms
|
|
|
|
/// map, and emit them to the file as functions.
|
2008-01-06 09:10:31 +08:00
|
|
|
void CodeGenDAGPatterns::ParseNodeTransforms() {
|
2008-01-06 06:25:12 +08:00
|
|
|
std::vector<Record*> Xforms = Records.getAllDerivedDefinitions("SDNodeXForm");
|
|
|
|
while (!Xforms.empty()) {
|
|
|
|
Record *XFormNode = Xforms.back();
|
|
|
|
Record *SDNode = XFormNode->getValueAsDef("Opcode");
|
2017-06-01 05:12:46 +08:00
|
|
|
StringRef Code = XFormNode->getValueAsString("XFormFunction");
|
2008-01-06 06:54:53 +08:00
|
|
|
SDNodeXForms.insert(std::make_pair(XFormNode, NodeXForm(SDNode, Code)));
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
Xforms.pop_back();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-01-06 09:10:31 +08:00
|
|
|
void CodeGenDAGPatterns::ParseComplexPatterns() {
|
2008-01-06 06:25:12 +08:00
|
|
|
std::vector<Record*> AMs = Records.getAllDerivedDefinitions("ComplexPattern");
|
|
|
|
while (!AMs.empty()) {
|
|
|
|
ComplexPatterns.insert(std::make_pair(AMs.back(), AMs.back()));
|
|
|
|
AMs.pop_back();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// ParsePatternFragments - Parse all of the PatFrag definitions in the .td
|
|
|
|
/// file, building up the PatternFragments map. After we've collected them all,
|
|
|
|
/// inline fragments together as necessary, so that there are no references left
|
|
|
|
/// inside a pattern fragment to a pattern fragment.
|
|
|
|
///
|
2014-02-28 08:26:56 +08:00
|
|
|
void CodeGenDAGPatterns::ParsePatternFragments(bool OutFrags) {
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
std::vector<Record*> Fragments = Records.getAllDerivedDefinitions("PatFrags");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:43:57 +08:00
|
|
|
// First step, parse all of the fragments.
|
2015-11-23 04:46:24 +08:00
|
|
|
for (Record *Frag : Fragments) {
|
|
|
|
if (OutFrags != Frag->isSubClassOf("OutPatFrag"))
|
2014-02-28 08:26:56 +08:00
|
|
|
continue;
|
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
ListInit *LI = Frag->getValueAsListInit("Fragments");
|
2014-02-28 08:26:56 +08:00
|
|
|
TreePattern *P =
|
2019-08-15 23:54:37 +08:00
|
|
|
(PatternFragments[Frag] = std::make_unique<TreePattern>(
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
Frag, LI, !Frag->isSubClassOf("OutPatFrag"),
|
2014-11-14 05:40:02 +08:00
|
|
|
*this)).get();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:43:57 +08:00
|
|
|
// Validate the argument list, converting it to set, to discard duplicates.
|
2008-01-06 06:25:12 +08:00
|
|
|
std::vector<std::string> &Args = P->getArgList();
|
2017-09-21 02:01:40 +08:00
|
|
|
// Copy the args so we can take StringRefs to them.
|
|
|
|
auto ArgsCopy = Args;
|
|
|
|
SmallDenseSet<StringRef, 4> OperandsSet;
|
|
|
|
OperandsSet.insert(ArgsCopy.begin(), ArgsCopy.end());
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:43:57 +08:00
|
|
|
if (OperandsSet.count(""))
|
2008-01-06 06:25:12 +08:00
|
|
|
P->error("Cannot have unnamed 'node' values in pattern fragment!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Parse the operands list.
|
2015-11-23 04:46:24 +08:00
|
|
|
DagInit *OpsList = Frag->getValueAsDag("Operands");
|
2012-10-11 04:24:43 +08:00
|
|
|
DefInit *OpsOp = dyn_cast<DefInit>(OpsList->getOperator());
|
2008-01-06 06:25:12 +08:00
|
|
|
// Special cases: ops == outs == ins. Different names are used to
|
2009-03-27 00:17:51 +08:00
|
|
|
// improve readability.
|
2008-01-06 06:25:12 +08:00
|
|
|
if (!OpsOp ||
|
|
|
|
(OpsOp->getDef()->getName() != "ops" &&
|
|
|
|
OpsOp->getDef()->getName() != "outs" &&
|
|
|
|
OpsOp->getDef()->getName() != "ins"))
|
|
|
|
P->error("Operands list should start with '(ops ... '!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
|
|
|
// Copy over the arguments.
|
2008-01-06 06:25:12 +08:00
|
|
|
Args.clear();
|
|
|
|
for (unsigned j = 0, e = OpsList->getNumArgs(); j != e; ++j) {
|
2012-10-11 04:24:47 +08:00
|
|
|
if (!isa<DefInit>(OpsList->getArg(j)) ||
|
|
|
|
cast<DefInit>(OpsList->getArg(j))->getDef()->getName() != "node")
|
2008-01-06 06:25:12 +08:00
|
|
|
P->error("Operands list should all be 'node' values.");
|
2016-12-05 14:00:46 +08:00
|
|
|
if (!OpsList->getArgName(j))
|
2008-01-06 06:25:12 +08:00
|
|
|
P->error("Operands list should have names for each operand!");
|
2016-12-05 14:00:46 +08:00
|
|
|
StringRef ArgNameStr = OpsList->getArgNameStr(j);
|
|
|
|
if (!OperandsSet.count(ArgNameStr))
|
|
|
|
P->error("'" + ArgNameStr +
|
2008-01-06 06:25:12 +08:00
|
|
|
"' does not occur in pattern or was multiply specified!");
|
2016-12-05 14:00:46 +08:00
|
|
|
OperandsSet.erase(ArgNameStr);
|
|
|
|
Args.push_back(ArgNameStr);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:43:57 +08:00
|
|
|
if (!OperandsSet.empty())
|
2008-01-06 06:25:12 +08:00
|
|
|
P->error("Operands list does not contain an entry for operand '" +
|
2008-01-06 06:43:57 +08:00
|
|
|
*OperandsSet.begin() + "'!");
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
// If there is a node transformation corresponding to this, keep track of
|
|
|
|
// it.
|
2015-11-23 04:46:24 +08:00
|
|
|
Record *Transform = Frag->getValueAsDef("OperandTransform");
|
2008-01-06 06:25:12 +08:00
|
|
|
if (!getSDNodeTransform(Transform).second.empty()) // not noop xform?
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
for (auto T : P->getTrees())
|
|
|
|
T->setTransformFn(Transform);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Now that we've parsed all of the tree fragments, do a closure on them so
|
|
|
|
// that there are not references to PatFrags left inside of them.
|
2015-11-23 04:46:24 +08:00
|
|
|
for (Record *Frag : Fragments) {
|
|
|
|
if (OutFrags != Frag->isSubClassOf("OutPatFrag"))
|
2014-02-28 08:26:56 +08:00
|
|
|
continue;
|
|
|
|
|
2015-11-23 04:46:24 +08:00
|
|
|
TreePattern &ThePat = *PatternFragments[Frag];
|
2014-11-14 05:40:02 +08:00
|
|
|
ThePat.InlinePatternFragments();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Infer as many types as possible. Don't worry about it if we don't infer
|
2018-07-14 00:42:15 +08:00
|
|
|
// all of them, some may depend on the inputs of the pattern. Also, don't
|
|
|
|
// validate type sets; validation may cause spurious failures e.g. if a
|
|
|
|
// fragment needs floating-point types but the current target does not have
|
|
|
|
// any (this is only an error if that fragment is ever used!).
|
|
|
|
{
|
|
|
|
TypeInfer::SuppressValidation SV(ThePat.getInfer());
|
|
|
|
ThePat.InferAllTypes();
|
|
|
|
ThePat.resetError();
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// If debugging, print out the pattern fragment result.
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(ThePat.dump());
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-01-06 09:10:31 +08:00
|
|
|
void CodeGenDAGPatterns::ParseDefaultOperands() {
|
2012-09-06 22:15:52 +08:00
|
|
|
std::vector<Record*> DefaultOps;
|
|
|
|
DefaultOps = Records.getAllDerivedDefinitions("OperandWithDefaultOps");
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
// Find some SDNode.
|
|
|
|
assert(!SDNodes.empty() && "No SDNodes parsed?");
|
2011-07-30 06:43:06 +08:00
|
|
|
Init *SomeSDNode = DefInit::get(SDNodes.begin()->first);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2012-09-06 22:15:52 +08:00
|
|
|
for (unsigned i = 0, e = DefaultOps.size(); i != e; ++i) {
|
|
|
|
DagInit *DefaultInfo = DefaultOps[i]->getValueAsDag("DefaultOps");
|
|
|
|
|
|
|
|
// Clone the DefaultInfo dag node, changing the operator from 'ops' to
|
|
|
|
// SomeSDnode so that we can parse this.
|
2016-12-05 14:00:46 +08:00
|
|
|
std::vector<std::pair<Init*, StringInit*> > Ops;
|
2012-09-06 22:15:52 +08:00
|
|
|
for (unsigned op = 0, e = DefaultInfo->getNumArgs(); op != e; ++op)
|
|
|
|
Ops.push_back(std::make_pair(DefaultInfo->getArg(op),
|
|
|
|
DefaultInfo->getArgName(op)));
|
2016-12-05 14:00:41 +08:00
|
|
|
DagInit *DI = DagInit::get(SomeSDNode, nullptr, Ops);
|
2012-09-06 22:15:52 +08:00
|
|
|
|
|
|
|
// Create a TreePattern to parse this.
|
|
|
|
TreePattern P(DefaultOps[i], DI, false, *this);
|
|
|
|
assert(P.getNumTrees() == 1 && "This ctor can only produce one tree!");
|
|
|
|
|
|
|
|
// Copy the operands over into a DAGDefaultOperand.
|
|
|
|
DAGDefaultOperand DefaultOpInfo;
|
|
|
|
|
2018-05-31 05:00:18 +08:00
|
|
|
const TreePatternNodePtr &T = P.getTree(0);
|
2012-09-06 22:15:52 +08:00
|
|
|
for (unsigned op = 0, e = T->getNumChildren(); op != e; ++op) {
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr TPN = T->getChildShared(op);
|
2012-09-06 22:15:52 +08:00
|
|
|
while (TPN->ApplyTypeConstraints(P, false))
|
|
|
|
/* Resolve all types */;
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
if (TPN->ContainsUnresolvedType(P)) {
|
2014-03-30 01:17:15 +08:00
|
|
|
PrintFatalError("Value #" + Twine(i) + " of OperandWithDefaultOps '" +
|
|
|
|
DefaultOps[i]->getName() +
|
|
|
|
"' doesn't have a concrete type!");
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2018-06-14 19:56:19 +08:00
|
|
|
DefaultOpInfo.DefaultOps.push_back(std::move(TPN));
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2012-09-06 22:15:52 +08:00
|
|
|
|
|
|
|
// Insert it into the DefaultOperands map so we can find it later.
|
|
|
|
DefaultOperands[DefaultOps[i]] = DefaultOpInfo;
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// HandleUse - Given "Pat" a leaf in the pattern, check to see if it is an
|
|
|
|
/// instruction input. Return true if this is a real use.
|
2018-06-12 06:14:43 +08:00
|
|
|
static bool HandleUse(TreePattern &I, TreePatternNodePtr Pat,
|
2018-05-31 05:00:18 +08:00
|
|
|
std::map<std::string, TreePatternNodePtr> &InstInputs) {
|
2008-01-06 06:25:12 +08:00
|
|
|
// No name -> not interesting.
|
|
|
|
if (Pat->getName().empty()) {
|
|
|
|
if (Pat->isLeaf()) {
|
2012-10-11 04:24:43 +08:00
|
|
|
DefInit *DI = dyn_cast<DefInit>(Pat->getLeafValue());
|
2011-06-28 05:06:21 +08:00
|
|
|
if (DI && (DI->getDef()->isSubClassOf("RegisterClass") ||
|
|
|
|
DI->getDef()->isSubClassOf("RegisterOperand")))
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("Input " + DI->getDef()->getName() + " must be named!");
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Record *Rec;
|
|
|
|
if (Pat->isLeaf()) {
|
2012-10-11 04:24:43 +08:00
|
|
|
DefInit *DI = dyn_cast<DefInit>(Pat->getLeafValue());
|
2018-06-12 06:14:43 +08:00
|
|
|
if (!DI)
|
|
|
|
I.error("Input $" + Pat->getName() + " must be an identifier!");
|
2008-01-06 06:25:12 +08:00
|
|
|
Rec = DI->getDef();
|
|
|
|
} else {
|
|
|
|
Rec = Pat->getOperator();
|
|
|
|
}
|
|
|
|
|
|
|
|
// SRCVALUE nodes are ignored.
|
|
|
|
if (Rec->getName() == "srcvalue")
|
|
|
|
return false;
|
|
|
|
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr &Slot = InstInputs[Pat->getName()];
|
2008-01-06 06:25:12 +08:00
|
|
|
if (!Slot) {
|
|
|
|
Slot = Pat;
|
2010-02-23 13:59:10 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
Record *SlotRec;
|
|
|
|
if (Slot->isLeaf()) {
|
2012-10-11 04:24:47 +08:00
|
|
|
SlotRec = cast<DefInit>(Slot->getLeafValue())->getDef();
|
2008-01-06 06:25:12 +08:00
|
|
|
} else {
|
2010-02-23 13:59:10 +08:00
|
|
|
assert(Slot->getNumChildren() == 0 && "can't be a use with children!");
|
|
|
|
SlotRec = Slot->getOperator();
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-02-23 13:59:10 +08:00
|
|
|
// Ensure that the inputs agree if we've already seen this input.
|
|
|
|
if (Rec != SlotRec)
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("All $" + Pat->getName() + " inputs must agree with each other");
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// Ensure that the types can agree as well.
|
|
|
|
Slot->UpdateNodeType(0, Pat->getExtType(0), I);
|
|
|
|
Pat->UpdateNodeType(0, Slot->getExtType(0), I);
|
2010-03-20 05:37:09 +08:00
|
|
|
if (Slot->getExtTypes() != Pat->getExtTypes())
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("All $" + Pat->getName() + " inputs must agree with each other");
|
2008-01-06 06:25:12 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// FindPatternInputsAndOutputs - Scan the specified TreePatternNode (which is
|
|
|
|
/// part of "I", the instruction), computing the set of inputs and outputs of
|
|
|
|
/// the pattern. Report errors if we see anything naughty.
|
2018-05-31 05:00:18 +08:00
|
|
|
void CodeGenDAGPatterns::FindPatternInputsAndOutputs(
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePattern &I, TreePatternNodePtr Pat,
|
2018-05-31 05:00:18 +08:00
|
|
|
std::map<std::string, TreePatternNodePtr> &InstInputs,
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
MapVector<std::string, TreePatternNodePtr, std::map<std::string, unsigned>>
|
|
|
|
&InstResults,
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<Record *> &InstImpResults) {
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
|
|
|
|
// The instruction pattern still has unresolved fragments. For *named*
|
|
|
|
// nodes we must resolve those here. This may not result in multiple
|
|
|
|
// alternatives.
|
|
|
|
if (!Pat->getName().empty()) {
|
|
|
|
TreePattern SrcPattern(I.getRecord(), Pat, true, *this);
|
|
|
|
SrcPattern.InlinePatternFragments();
|
|
|
|
SrcPattern.InferAllTypes();
|
|
|
|
Pat = SrcPattern.getOnlyTree();
|
|
|
|
}
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
if (Pat->isLeaf()) {
|
2010-04-20 14:30:25 +08:00
|
|
|
bool isUse = HandleUse(I, Pat, InstInputs);
|
2008-01-06 06:25:12 +08:00
|
|
|
if (!isUse && Pat->getTransformFn())
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("Cannot specify a transform function for a non-input value!");
|
2008-01-06 06:25:12 +08:00
|
|
|
return;
|
2010-02-17 14:53:36 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-02-17 14:53:36 +08:00
|
|
|
if (Pat->getOperator()->getName() == "implicit") {
|
2008-01-06 06:25:12 +08:00
|
|
|
for (unsigned i = 0, e = Pat->getNumChildren(); i != e; ++i) {
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *Dest = Pat->getChild(i);
|
|
|
|
if (!Dest->isLeaf())
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("implicitly defined value should be a register!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
DefInit *Val = dyn_cast<DefInit>(Dest->getLeafValue());
|
2008-01-06 06:25:12 +08:00
|
|
|
if (!Val || !Val->getDef()->isSubClassOf("Register"))
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("implicitly defined value should be a register!");
|
2008-01-06 06:25:12 +08:00
|
|
|
InstImpResults.push_back(Val->getDef());
|
|
|
|
}
|
|
|
|
return;
|
2010-02-17 14:53:36 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-02-17 14:53:36 +08:00
|
|
|
if (Pat->getOperator()->getName() != "set") {
|
2008-01-06 06:25:12 +08:00
|
|
|
// If this is not a set, verify that the children nodes are not void typed,
|
|
|
|
// and recurse.
|
|
|
|
for (unsigned i = 0, e = Pat->getNumChildren(); i != e; ++i) {
|
2018-06-15 04:32:58 +08:00
|
|
|
if (Pat->getChild(i)->getNumTypes() == 0)
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("Cannot have void nodes inside of patterns!");
|
2018-05-31 05:00:18 +08:00
|
|
|
FindPatternInputsAndOutputs(I, Pat->getChildShared(i), InstInputs,
|
|
|
|
InstResults, InstImpResults);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// If this is a non-leaf node with no children, treat it basically as if
|
|
|
|
// it were a leaf. This handles nodes like (imm).
|
2010-04-20 14:30:25 +08:00
|
|
|
bool isUse = HandleUse(I, Pat, InstInputs);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
if (!isUse && Pat->getTransformFn())
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("Cannot specify a transform function for a non-input value!");
|
2008-01-06 06:25:12 +08:00
|
|
|
return;
|
2010-02-17 14:53:36 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Otherwise, this is a set, validate and collect instruction results.
|
|
|
|
if (Pat->getNumChildren() == 0)
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("set requires operands!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
if (Pat->getTransformFn())
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("Cannot specify a transform function on a set node!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Check the set destinations.
|
|
|
|
unsigned NumDests = Pat->getNumChildren()-1;
|
|
|
|
for (unsigned i = 0; i != NumDests; ++i) {
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNodePtr Dest = Pat->getChildShared(i);
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// For set destinations we also must resolve fragments here.
|
|
|
|
TreePattern DestPattern(I.getRecord(), Dest, false, *this);
|
|
|
|
DestPattern.InlinePatternFragments();
|
|
|
|
DestPattern.InferAllTypes();
|
|
|
|
Dest = DestPattern.getOnlyTree();
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
if (!Dest->isLeaf())
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("set destination should be a register!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2012-10-11 04:24:43 +08:00
|
|
|
DefInit *Val = dyn_cast<DefInit>(Dest->getLeafValue());
|
2014-12-13 05:48:03 +08:00
|
|
|
if (!Val) {
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("set destination should be a register!");
|
2014-12-13 05:48:03 +08:00
|
|
|
continue;
|
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
if (Val->getDef()->isSubClassOf("RegisterClass") ||
|
2013-03-24 08:56:16 +08:00
|
|
|
Val->getDef()->isSubClassOf("ValueType") ||
|
2011-06-28 05:06:21 +08:00
|
|
|
Val->getDef()->isSubClassOf("RegisterOperand") ||
|
2009-07-30 04:43:05 +08:00
|
|
|
Val->getDef()->isSubClassOf("PointerLikeRegClass")) {
|
2008-01-06 06:25:12 +08:00
|
|
|
if (Dest->getName().empty())
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("set destination must have a name!");
|
2008-01-06 06:25:12 +08:00
|
|
|
if (InstResults.count(Dest->getName()))
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("cannot set '" + Dest->getName() + "' multiple times");
|
2008-01-06 06:25:12 +08:00
|
|
|
InstResults[Dest->getName()] = Dest;
|
|
|
|
} else if (Val->getDef()->isSubClassOf("Register")) {
|
|
|
|
InstImpResults.push_back(Val->getDef());
|
|
|
|
} else {
|
2018-06-12 06:14:43 +08:00
|
|
|
I.error("set destination should be a register!");
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Verify and collect info from the computation.
|
2018-05-31 05:00:18 +08:00
|
|
|
FindPatternInputsAndOutputs(I, Pat->getChildShared(NumDests), InstInputs,
|
|
|
|
InstResults, InstImpResults);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2008-04-03 08:02:49 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Instruction Analysis
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
class InstAnalyzer {
|
|
|
|
const CodeGenDAGPatterns &CDP;
|
|
|
|
public:
|
2012-08-24 08:31:16 +08:00
|
|
|
bool hasSideEffects;
|
|
|
|
bool mayStore;
|
|
|
|
bool mayLoad;
|
|
|
|
bool isBitcast;
|
|
|
|
bool isVariadic;
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
bool hasChain;
|
2008-04-03 08:02:49 +08:00
|
|
|
|
2012-08-24 08:31:16 +08:00
|
|
|
InstAnalyzer(const CodeGenDAGPatterns &cdp)
|
|
|
|
: CDP(cdp), hasSideEffects(false), mayStore(false), mayLoad(false),
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
isBitcast(false), isVariadic(false), hasChain(false) {}
|
2008-04-03 08:02:49 +08:00
|
|
|
|
2017-06-21 00:34:37 +08:00
|
|
|
void Analyze(const PatternToMatch &Pat) {
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
const TreePatternNode *N = Pat.getSrcPattern();
|
|
|
|
AnalyzeNode(N);
|
|
|
|
// These properties are detected only on the root node.
|
|
|
|
isBitcast = IsNodeBitcast(N);
|
2012-08-25 06:46:53 +08:00
|
|
|
}
|
|
|
|
|
2008-04-03 08:02:49 +08:00
|
|
|
private:
|
2018-06-15 04:32:58 +08:00
|
|
|
bool IsNodeBitcast(const TreePatternNode *N) const {
|
2012-08-24 08:31:16 +08:00
|
|
|
if (hasSideEffects || mayLoad || mayStore || isVariadic)
|
2011-03-15 13:09:26 +08:00
|
|
|
return false;
|
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (N->isLeaf())
|
2011-03-15 13:09:26 +08:00
|
|
|
return false;
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (N->getNumChildren() != 1 || !N->getChild(0)->isLeaf())
|
2011-03-15 13:09:26 +08:00
|
|
|
return false;
|
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
const SDNodeInfo &OpInfo = CDP.getSDNodeInfo(N->getOperator());
|
2011-03-15 13:09:26 +08:00
|
|
|
if (OpInfo.getNumResults() != 1 || OpInfo.getNumOperands() != 1)
|
|
|
|
return false;
|
|
|
|
return OpInfo.getEnumName() == "ISD::BITCAST";
|
|
|
|
}
|
|
|
|
|
2012-08-28 11:26:49 +08:00
|
|
|
public:
|
2018-06-15 04:32:58 +08:00
|
|
|
void AnalyzeNode(const TreePatternNode *N) {
|
|
|
|
if (N->isLeaf()) {
|
|
|
|
if (DefInit *DI = dyn_cast<DefInit>(N->getLeafValue())) {
|
2008-04-03 08:02:49 +08:00
|
|
|
Record *LeafRec = DI->getDef();
|
|
|
|
// Handle ComplexPattern leaves.
|
|
|
|
if (LeafRec->isSubClassOf("ComplexPattern")) {
|
|
|
|
const ComplexPattern &CP = CDP.getComplexPattern(LeafRec);
|
|
|
|
if (CP.hasProperty(SDNPMayStore)) mayStore = true;
|
|
|
|
if (CP.hasProperty(SDNPMayLoad)) mayLoad = true;
|
2012-08-24 08:31:16 +08:00
|
|
|
if (CP.hasProperty(SDNPSideEffect)) hasSideEffects = true;
|
2008-04-03 08:02:49 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Analyze children.
|
2018-06-15 04:32:58 +08:00
|
|
|
for (unsigned i = 0, e = N->getNumChildren(); i != e; ++i)
|
|
|
|
AnalyzeNode(N->getChild(i));
|
2008-04-03 08:02:49 +08:00
|
|
|
|
|
|
|
// Notice properties of the node.
|
2018-06-15 04:32:58 +08:00
|
|
|
if (N->NodeHasProperty(SDNPMayStore, CDP)) mayStore = true;
|
|
|
|
if (N->NodeHasProperty(SDNPMayLoad, CDP)) mayLoad = true;
|
|
|
|
if (N->NodeHasProperty(SDNPSideEffect, CDP)) hasSideEffects = true;
|
|
|
|
if (N->NodeHasProperty(SDNPVariadic, CDP)) isVariadic = true;
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (N->NodeHasProperty(SDNPHasChain, CDP)) hasChain = true;
|
2008-04-03 08:02:49 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
if (const CodeGenIntrinsic *IntInfo = N->getIntrinsicInfo(CDP)) {
|
2008-04-03 08:02:49 +08:00
|
|
|
// If this is an intrinsic, analyze it.
|
2016-04-20 05:58:33 +08:00
|
|
|
if (IntInfo->ModRef & CodeGenIntrinsic::MR_Ref)
|
2008-04-03 08:02:49 +08:00
|
|
|
mayLoad = true;// These may load memory.
|
|
|
|
|
2016-04-20 05:58:33 +08:00
|
|
|
if (IntInfo->ModRef & CodeGenIntrinsic::MR_Mod)
|
2008-04-03 08:02:49 +08:00
|
|
|
mayStore = true;// Intrinsics that can write to memory are 'mayStore'.
|
|
|
|
|
2017-04-29 05:01:46 +08:00
|
|
|
if (IntInfo->ModRef >= CodeGenIntrinsic::ReadWriteMem ||
|
|
|
|
IntInfo->hasSideEffects)
|
2016-04-20 05:58:33 +08:00
|
|
|
// ReadWriteMem intrinsics can have other strange effects.
|
2012-08-24 08:31:16 +08:00
|
|
|
hasSideEffects = true;
|
2008-04-03 08:02:49 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
|
2012-08-25 01:08:41 +08:00
|
|
|
static bool InferFromPattern(CodeGenInstruction &InstInfo,
|
2012-08-24 08:31:16 +08:00
|
|
|
const InstAnalyzer &PatInfo,
|
|
|
|
Record *PatDef) {
|
2012-08-25 01:08:41 +08:00
|
|
|
bool Error = false;
|
|
|
|
|
2012-08-24 08:31:16 +08:00
|
|
|
// Remember where InstInfo got its flags.
|
|
|
|
if (InstInfo.hasUndefFlags())
|
|
|
|
InstInfo.InferredFrom = PatDef;
|
2008-04-03 08:02:49 +08:00
|
|
|
|
2012-08-25 01:08:41 +08:00
|
|
|
// Check explicitly set flags for consistency.
|
|
|
|
if (InstInfo.hasSideEffects != PatInfo.hasSideEffects &&
|
|
|
|
!InstInfo.hasSideEffects_Unset) {
|
|
|
|
// Allow explicitly setting hasSideEffects = 1 on instructions, even when
|
|
|
|
// the pattern has no side effects. That could be useful for div/rem
|
|
|
|
// instructions that may trap.
|
|
|
|
if (!InstInfo.hasSideEffects) {
|
|
|
|
Error = true;
|
|
|
|
PrintError(PatDef->getLoc(), "Pattern doesn't match hasSideEffects = " +
|
|
|
|
Twine(InstInfo.hasSideEffects));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (InstInfo.mayStore != PatInfo.mayStore && !InstInfo.mayStore_Unset) {
|
|
|
|
Error = true;
|
|
|
|
PrintError(PatDef->getLoc(), "Pattern doesn't match mayStore = " +
|
|
|
|
Twine(InstInfo.mayStore));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (InstInfo.mayLoad != PatInfo.mayLoad && !InstInfo.mayLoad_Unset) {
|
|
|
|
// Allow explicitly setting mayLoad = 1, even when the pattern has no loads.
|
2015-09-12 09:17:08 +08:00
|
|
|
// Some targets translate immediates to loads.
|
2012-08-25 01:08:41 +08:00
|
|
|
if (!InstInfo.mayLoad) {
|
|
|
|
Error = true;
|
|
|
|
PrintError(PatDef->getLoc(), "Pattern doesn't match mayLoad = " +
|
|
|
|
Twine(InstInfo.mayLoad));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-24 08:31:16 +08:00
|
|
|
// Transfer inferred flags.
|
|
|
|
InstInfo.hasSideEffects |= PatInfo.hasSideEffects;
|
|
|
|
InstInfo.mayStore |= PatInfo.mayStore;
|
|
|
|
InstInfo.mayLoad |= PatInfo.mayLoad;
|
2008-04-03 08:02:49 +08:00
|
|
|
|
2012-08-24 08:31:16 +08:00
|
|
|
// These flags are silently added without any verification.
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// FIXME: To match historical behavior of TableGen, for now add those flags
|
|
|
|
// only when we're inferring from the primary instruction pattern.
|
|
|
|
if (PatDef->isSubClassOf("Instruction")) {
|
|
|
|
InstInfo.isBitcast |= PatInfo.isBitcast;
|
|
|
|
InstInfo.hasChain |= PatInfo.hasChain;
|
|
|
|
InstInfo.hasChain_Inferred = true;
|
|
|
|
}
|
2012-08-25 05:08:09 +08:00
|
|
|
|
|
|
|
// Don't infer isVariadic. This flag means something different on SDNodes and
|
|
|
|
// instructions. For example, a CALL SDNode is variadic because it has the
|
|
|
|
// call arguments as operands, but a CALL instruction is not variadic - it
|
|
|
|
// has argument registers as implicit, not explicit uses.
|
2012-08-25 01:08:41 +08:00
|
|
|
|
|
|
|
return Error;
|
2008-04-03 08:02:49 +08:00
|
|
|
}
|
|
|
|
|
TableGen: Allow conditional instruction pattern in multiclass.
Define a 'null_frag' SDPatternOperator node, which if referenced in an
instruction Pattern, results in the pattern being collapsed to be as-if
'[]' had been specified instead. This allows supporting a multiclass
definition where some instaniations have ISel patterns associated and
others do not.
For example,
multiclass myMulti<RegisterClass rc, SDPatternOperator OpNode = null_frag> {
def _x : myI<(outs rc:), (ins rc:), []>;
def _r : myI<(outs rc:), (ins rc:), [(set rc:, (OpNode rc:))]>;
}
defm foo : myMulti<GRa, not>;
defm bar : myMulti<GRb>;
llvm-svn: 160333
2012-07-17 08:47:06 +08:00
|
|
|
/// hasNullFragReference - Return true if the DAG has any reference to the
|
|
|
|
/// null_frag operator.
|
|
|
|
static bool hasNullFragReference(DagInit *DI) {
|
2012-10-11 04:24:43 +08:00
|
|
|
DefInit *OpDef = dyn_cast<DefInit>(DI->getOperator());
|
TableGen: Allow conditional instruction pattern in multiclass.
Define a 'null_frag' SDPatternOperator node, which if referenced in an
instruction Pattern, results in the pattern being collapsed to be as-if
'[]' had been specified instead. This allows supporting a multiclass
definition where some instaniations have ISel patterns associated and
others do not.
For example,
multiclass myMulti<RegisterClass rc, SDPatternOperator OpNode = null_frag> {
def _x : myI<(outs rc:), (ins rc:), []>;
def _r : myI<(outs rc:), (ins rc:), [(set rc:, (OpNode rc:))]>;
}
defm foo : myMulti<GRa, not>;
defm bar : myMulti<GRb>;
llvm-svn: 160333
2012-07-17 08:47:06 +08:00
|
|
|
if (!OpDef) return false;
|
|
|
|
Record *Operator = OpDef->getDef();
|
|
|
|
|
|
|
|
// If this is the null fragment, return true.
|
|
|
|
if (Operator->getName() == "null_frag") return true;
|
|
|
|
// If any of the arguments reference the null fragment, return true.
|
|
|
|
for (unsigned i = 0, e = DI->getNumArgs(); i != e; ++i) {
|
2012-10-11 04:24:43 +08:00
|
|
|
DagInit *Arg = dyn_cast<DagInit>(DI->getArg(i));
|
TableGen: Allow conditional instruction pattern in multiclass.
Define a 'null_frag' SDPatternOperator node, which if referenced in an
instruction Pattern, results in the pattern being collapsed to be as-if
'[]' had been specified instead. This allows supporting a multiclass
definition where some instaniations have ISel patterns associated and
others do not.
For example,
multiclass myMulti<RegisterClass rc, SDPatternOperator OpNode = null_frag> {
def _x : myI<(outs rc:), (ins rc:), []>;
def _r : myI<(outs rc:), (ins rc:), [(set rc:, (OpNode rc:))]>;
}
defm foo : myMulti<GRa, not>;
defm bar : myMulti<GRb>;
llvm-svn: 160333
2012-07-17 08:47:06 +08:00
|
|
|
if (Arg && hasNullFragReference(Arg))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// hasNullFragReference - Return true if any DAG in the list references
|
|
|
|
/// the null_frag operator.
|
|
|
|
static bool hasNullFragReference(ListInit *LI) {
|
2015-06-02 12:15:51 +08:00
|
|
|
for (Init *I : LI->getValues()) {
|
|
|
|
DagInit *DI = dyn_cast<DagInit>(I);
|
TableGen: Allow conditional instruction pattern in multiclass.
Define a 'null_frag' SDPatternOperator node, which if referenced in an
instruction Pattern, results in the pattern being collapsed to be as-if
'[]' had been specified instead. This allows supporting a multiclass
definition where some instaniations have ISel patterns associated and
others do not.
For example,
multiclass myMulti<RegisterClass rc, SDPatternOperator OpNode = null_frag> {
def _x : myI<(outs rc:), (ins rc:), []>;
def _r : myI<(outs rc:), (ins rc:), [(set rc:, (OpNode rc:))]>;
}
defm foo : myMulti<GRa, not>;
defm bar : myMulti<GRb>;
llvm-svn: 160333
2012-07-17 08:47:06 +08:00
|
|
|
assert(DI && "non-dag in an instruction Pattern list?!");
|
|
|
|
if (hasNullFragReference(DI))
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-08-25 06:46:53 +08:00
|
|
|
/// Get all the instructions in a tree.
|
|
|
|
static void
|
2018-06-15 04:32:58 +08:00
|
|
|
getInstructionsInTree(TreePatternNode *Tree, SmallVectorImpl<Record*> &Instrs) {
|
|
|
|
if (Tree->isLeaf())
|
2012-08-25 06:46:53 +08:00
|
|
|
return;
|
2018-06-15 04:32:58 +08:00
|
|
|
if (Tree->getOperator()->isSubClassOf("Instruction"))
|
|
|
|
Instrs.push_back(Tree->getOperator());
|
|
|
|
for (unsigned i = 0, e = Tree->getNumChildren(); i != e; ++i)
|
|
|
|
getInstructionsInTree(Tree->getChild(i), Instrs);
|
2012-08-25 06:46:53 +08:00
|
|
|
}
|
|
|
|
|
2013-03-24 08:56:16 +08:00
|
|
|
/// Check the class of a pattern leaf node against the instruction operand it
|
|
|
|
/// represents.
|
|
|
|
static bool checkOperandClass(CGIOperandList::OperandInfo &OI,
|
|
|
|
Record *Leaf) {
|
|
|
|
if (OI.Rec == Leaf)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Allow direct value types to be used in instruction set patterns.
|
|
|
|
// The type will be checked later.
|
|
|
|
if (Leaf->isSubClassOf("ValueType"))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
// Patterns can also be ComplexPattern instances.
|
|
|
|
if (Leaf->isSubClassOf("ComplexPattern"))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
void CodeGenDAGPatterns::parseInstructionPattern(
|
2013-10-29 02:07:21 +08:00
|
|
|
CodeGenInstruction &CGI, ListInit *Pat, DAGInstMap &DAGInsts) {
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2015-03-10 11:25:04 +08:00
|
|
|
assert(!DAGInsts.count(CGI.TheDef) && "Instruction already parsed!");
|
|
|
|
|
|
|
|
// Parse the instruction.
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
TreePattern I(CGI.TheDef, Pat, true, *this);
|
2015-03-10 11:25:04 +08:00
|
|
|
|
|
|
|
// InstInputs - Keep track of all of the inputs of the instruction, along
|
|
|
|
// with the record they are declared as.
|
2018-05-31 05:00:18 +08:00
|
|
|
std::map<std::string, TreePatternNodePtr> InstInputs;
|
2015-03-10 11:25:04 +08:00
|
|
|
|
|
|
|
// InstResults - Keep track of all the virtual registers that are 'set'
|
|
|
|
// in the instruction, including what reg class they are.
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
MapVector<std::string, TreePatternNodePtr, std::map<std::string, unsigned>>
|
|
|
|
InstResults;
|
2015-03-10 11:25:04 +08:00
|
|
|
|
|
|
|
std::vector<Record*> InstImpResults;
|
|
|
|
|
|
|
|
// Verify that the top-level forms in the instruction are of void type, and
|
|
|
|
// fill in the InstResults map.
|
2017-09-21 02:01:40 +08:00
|
|
|
SmallString<32> TypesString;
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
for (unsigned j = 0, e = I.getNumTrees(); j != e; ++j) {
|
2017-09-21 02:01:40 +08:00
|
|
|
TypesString.clear();
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
TreePatternNodePtr Pat = I.getTree(j);
|
2016-04-20 05:58:10 +08:00
|
|
|
if (Pat->getNumTypes() != 0) {
|
2017-09-21 02:01:40 +08:00
|
|
|
raw_svector_ostream OS(TypesString);
|
2016-04-20 05:58:10 +08:00
|
|
|
for (unsigned k = 0, ke = Pat->getNumTypes(); k != ke; ++k) {
|
|
|
|
if (k > 0)
|
2017-09-21 02:01:40 +08:00
|
|
|
OS << ", ";
|
|
|
|
Pat->getExtType(k).writeToStream(OS);
|
2016-04-20 05:58:10 +08:00
|
|
|
}
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
I.error("Top-level forms in instruction pattern should have"
|
2017-09-21 02:01:40 +08:00
|
|
|
" void types, has types " +
|
|
|
|
OS.str());
|
2016-04-20 05:58:10 +08:00
|
|
|
}
|
2015-03-10 11:25:04 +08:00
|
|
|
|
|
|
|
// Find inputs and outputs, and verify the structure of the uses/defs.
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
FindPatternInputsAndOutputs(I, Pat, InstInputs, InstResults,
|
2015-03-10 11:25:04 +08:00
|
|
|
InstImpResults);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now that we have inputs and outputs of the pattern, inspect the operands
|
|
|
|
// list for the instruction. This determines the order that operands are
|
|
|
|
// added to the machine instruction the node corresponds to.
|
|
|
|
unsigned NumResults = InstResults.size();
|
|
|
|
|
|
|
|
// Parse the operands list from the (ops) list, validating it.
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
assert(I.getArgList().empty() && "Args list should still be empty here!");
|
2015-03-10 11:25:04 +08:00
|
|
|
|
|
|
|
// Check that all of the results occur first in the list.
|
|
|
|
std::vector<Record*> Results;
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
std::vector<unsigned> ResultIndices;
|
2018-05-31 05:00:18 +08:00
|
|
|
SmallVector<TreePatternNodePtr, 2> ResNodes;
|
2015-03-10 11:25:04 +08:00
|
|
|
for (unsigned i = 0; i != NumResults; ++i) {
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
if (i == CGI.Operands.size()) {
|
|
|
|
const std::string &OpName =
|
|
|
|
std::find_if(InstResults.begin(), InstResults.end(),
|
|
|
|
[](const std::pair<std::string, TreePatternNodePtr> &P) {
|
|
|
|
return P.second;
|
|
|
|
})
|
|
|
|
->first;
|
|
|
|
|
|
|
|
I.error("'" + OpName + "' set but does not appear in operand list!");
|
|
|
|
}
|
|
|
|
|
2015-03-10 11:25:04 +08:00
|
|
|
const std::string &OpName = CGI.Operands[i].Name;
|
|
|
|
|
|
|
|
// Check that it exists in InstResults.
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
auto InstResultIter = InstResults.find(OpName);
|
|
|
|
if (InstResultIter == InstResults.end() || !InstResultIter->second)
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
I.error("Operand $" + OpName + " does not exist in operand list!");
|
2015-03-10 11:25:04 +08:00
|
|
|
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
TreePatternNodePtr RNode = InstResultIter->second;
|
2015-03-10 11:25:04 +08:00
|
|
|
Record *R = cast<DefInit>(RNode->getLeafValue())->getDef();
|
2018-06-14 19:56:19 +08:00
|
|
|
ResNodes.push_back(std::move(RNode));
|
2015-03-10 11:25:04 +08:00
|
|
|
if (!R)
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
I.error("Operand $" + OpName + " should be a set destination: all "
|
2015-03-10 11:25:04 +08:00
|
|
|
"outputs must occur before inputs in operand list!");
|
|
|
|
|
|
|
|
if (!checkOperandClass(CGI.Operands[i], R))
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
I.error("Operand $" + OpName + " class mismatch!");
|
2015-03-10 11:25:04 +08:00
|
|
|
|
|
|
|
// Remember the return type.
|
|
|
|
Results.push_back(CGI.Operands[i].Rec);
|
|
|
|
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
// Remember the result index.
|
|
|
|
ResultIndices.push_back(std::distance(InstResults.begin(), InstResultIter));
|
|
|
|
|
2015-03-10 11:25:04 +08:00
|
|
|
// Okay, this one checks out.
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
InstResultIter->second = nullptr;
|
2015-03-10 11:25:04 +08:00
|
|
|
}
|
|
|
|
|
2018-07-15 14:52:48 +08:00
|
|
|
// Loop over the inputs next.
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> ResultNodeOperands;
|
2015-03-10 11:25:04 +08:00
|
|
|
std::vector<Record*> Operands;
|
|
|
|
for (unsigned i = NumResults, e = CGI.Operands.size(); i != e; ++i) {
|
|
|
|
CGIOperandList::OperandInfo &Op = CGI.Operands[i];
|
|
|
|
const std::string &OpName = Op.Name;
|
|
|
|
if (OpName.empty())
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
I.error("Operand #" + Twine(i) + " in operands list has no name!");
|
2015-03-10 11:25:04 +08:00
|
|
|
|
2018-07-15 14:52:48 +08:00
|
|
|
if (!InstInputs.count(OpName)) {
|
2015-03-10 11:25:04 +08:00
|
|
|
// If this is an operand with a DefaultOps set filled in, we can ignore
|
|
|
|
// this. When we codegen it, we will do so as always executed.
|
|
|
|
if (Op.Rec->isSubClassOf("OperandWithDefaultOps")) {
|
|
|
|
// Does it have a non-empty DefaultOps field? If so, ignore this
|
|
|
|
// operand.
|
|
|
|
if (!getDefaultOperand(Op.Rec).DefaultOps.empty())
|
|
|
|
continue;
|
|
|
|
}
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
I.error("Operand $" + OpName +
|
2015-03-10 11:25:04 +08:00
|
|
|
" does not appear in the instruction pattern");
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2018-07-15 14:52:48 +08:00
|
|
|
TreePatternNodePtr InVal = InstInputs[OpName];
|
|
|
|
InstInputs.erase(OpName); // It occurred, remove from map.
|
2015-03-10 11:25:04 +08:00
|
|
|
|
|
|
|
if (InVal->isLeaf() && isa<DefInit>(InVal->getLeafValue())) {
|
|
|
|
Record *InRec = static_cast<DefInit*>(InVal->getLeafValue())->getDef();
|
|
|
|
if (!checkOperandClass(Op, InRec))
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
I.error("Operand $" + OpName + "'s register class disagrees"
|
2015-03-10 11:25:04 +08:00
|
|
|
" between the operand and pattern");
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2015-03-10 11:25:04 +08:00
|
|
|
Operands.push_back(Op.Rec);
|
2008-01-06 06:25:12 +08:00
|
|
|
|
2015-03-10 11:25:04 +08:00
|
|
|
// Construct the result for the dest-pattern operand list.
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr OpNode = InVal->clone();
|
2008-01-06 06:25:12 +08:00
|
|
|
|
2015-03-10 11:25:04 +08:00
|
|
|
// No predicate is useful on the result.
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
OpNode->clearPredicateCalls();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2015-03-10 11:25:04 +08:00
|
|
|
// Promote the xform function to be an explicit node if set.
|
|
|
|
if (Record *Xform = OpNode->getTransformFn()) {
|
|
|
|
OpNode->setTransformFn(nullptr);
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> Children;
|
2015-03-10 11:25:04 +08:00
|
|
|
Children.push_back(OpNode);
|
2018-07-15 14:52:49 +08:00
|
|
|
OpNode = std::make_shared<TreePatternNode>(Xform, std::move(Children),
|
2018-06-15 04:32:58 +08:00
|
|
|
OpNode->getNumTypes());
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-06-14 19:56:19 +08:00
|
|
|
ResultNodeOperands.push_back(std::move(OpNode));
|
2015-03-10 11:25:04 +08:00
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
|
2018-07-15 14:52:48 +08:00
|
|
|
if (!InstInputs.empty())
|
|
|
|
I.error("Input operand $" + InstInputs.begin()->first +
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
" occurs in pattern but not in operands list!");
|
2008-01-06 06:25:12 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNodePtr ResultPattern = std::make_shared<TreePatternNode>(
|
2018-07-15 14:52:49 +08:00
|
|
|
I.getRecord(), std::move(ResultNodeOperands),
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
GetNumNodeResults(I.getRecord(), *this));
|
2015-03-20 13:09:06 +08:00
|
|
|
// Copy fully inferred output node types to instruction result pattern.
|
|
|
|
for (unsigned i = 0; i != NumResults; ++i) {
|
|
|
|
assert(ResNodes[i]->getNumTypes() == 1 && "FIXME: Unhandled");
|
|
|
|
ResultPattern->setType(i, ResNodes[i]->getExtType(0));
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
ResultPattern->setResultIndex(i, ResultIndices[i]);
|
2015-03-20 13:09:06 +08:00
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// FIXME: Assume only the first tree is the pattern. The others are clobber
|
|
|
|
// nodes.
|
|
|
|
TreePatternNodePtr Pattern = I.getTree(0);
|
|
|
|
TreePatternNodePtr SrcPattern;
|
|
|
|
if (Pattern->getOperator()->getName() == "set") {
|
|
|
|
SrcPattern = Pattern->getChild(Pattern->getNumChildren()-1)->clone();
|
|
|
|
} else{
|
|
|
|
// Not a set (store or something?)
|
|
|
|
SrcPattern = Pattern;
|
|
|
|
}
|
|
|
|
|
2015-03-10 11:25:04 +08:00
|
|
|
// Create and insert the instruction.
|
|
|
|
// FIXME: InstImpResults should not be part of DAGInstruction.
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
Record *R = I.getRecord();
|
|
|
|
DAGInsts.emplace(std::piecewise_construct, std::forward_as_tuple(R),
|
|
|
|
std::forward_as_tuple(Results, Operands, InstImpResults,
|
|
|
|
SrcPattern, ResultPattern));
|
2008-01-06 06:25:12 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
LLVM_DEBUG(I.dump());
|
2015-03-10 11:25:04 +08:00
|
|
|
}
|
2013-10-29 02:07:21 +08:00
|
|
|
|
|
|
|
/// ParseInstructions - Parse all of the instructions, inlining and resolving
|
|
|
|
/// any fragments involved. This populates the Instructions list with fully
|
|
|
|
/// resolved instructions.
|
|
|
|
void CodeGenDAGPatterns::ParseInstructions() {
|
|
|
|
std::vector<Record*> Instrs = Records.getAllDerivedDefinitions("Instruction");
|
|
|
|
|
2015-11-23 04:46:24 +08:00
|
|
|
for (Record *Instr : Instrs) {
|
2014-04-15 15:20:03 +08:00
|
|
|
ListInit *LI = nullptr;
|
2013-10-29 02:07:21 +08:00
|
|
|
|
2015-11-23 04:46:24 +08:00
|
|
|
if (isa<ListInit>(Instr->getValueInit("Pattern")))
|
|
|
|
LI = Instr->getValueAsListInit("Pattern");
|
2013-10-29 02:07:21 +08:00
|
|
|
|
|
|
|
// If there is no pattern, only collect minimal information about the
|
|
|
|
// instruction for its operand list. We have to assume that there is one
|
|
|
|
// result, as we have no detailed info. A pattern which references the
|
|
|
|
// null_frag operator is as-if no pattern were specified. Normally this
|
|
|
|
// is from a multiclass expansion w/ a SDPatternOperator passed in as
|
|
|
|
// null_frag.
|
2015-05-14 13:53:53 +08:00
|
|
|
if (!LI || LI->empty() || hasNullFragReference(LI)) {
|
2013-10-29 02:07:21 +08:00
|
|
|
std::vector<Record*> Results;
|
|
|
|
std::vector<Record*> Operands;
|
|
|
|
|
2015-11-23 04:46:24 +08:00
|
|
|
CodeGenInstruction &InstInfo = Target.getInstruction(Instr);
|
2013-10-29 02:07:21 +08:00
|
|
|
|
|
|
|
if (InstInfo.Operands.size() != 0) {
|
2015-03-20 13:09:06 +08:00
|
|
|
for (unsigned j = 0, e = InstInfo.Operands.NumDefs; j < e; ++j)
|
|
|
|
Results.push_back(InstInfo.Operands[j].Rec);
|
|
|
|
|
|
|
|
// The rest are inputs.
|
|
|
|
for (unsigned j = InstInfo.Operands.NumDefs,
|
|
|
|
e = InstInfo.Operands.size(); j < e; ++j)
|
|
|
|
Operands.push_back(InstInfo.Operands[j].Rec);
|
2013-10-29 02:07:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Create and insert the instruction.
|
|
|
|
std::vector<Record*> ImpResults;
|
2015-11-23 04:46:24 +08:00
|
|
|
Instructions.insert(std::make_pair(Instr,
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
DAGInstruction(Results, Operands, ImpResults)));
|
2013-10-29 02:07:21 +08:00
|
|
|
continue; // no pattern.
|
|
|
|
}
|
|
|
|
|
2015-11-23 04:46:24 +08:00
|
|
|
CodeGenInstruction &CGI = Target.getInstruction(Instr);
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
parseInstructionPattern(CGI, LI, Instructions);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// If we can, convert the instructions to be patterns that are matched!
|
2015-11-23 04:46:24 +08:00
|
|
|
for (auto &Entry : Instructions) {
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
Record *Instr = Entry.first;
|
2015-11-23 04:46:24 +08:00
|
|
|
DAGInstruction &TheInst = Entry.second;
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
TreePatternNodePtr SrcPattern = TheInst.getSrcPattern();
|
|
|
|
TreePatternNodePtr ResultPattern = TheInst.getResultPattern();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
if (SrcPattern && ResultPattern) {
|
|
|
|
TreePattern Pattern(Instr, SrcPattern, true, *this);
|
|
|
|
TreePattern Result(Instr, ResultPattern, false, *this);
|
|
|
|
ParseOnePattern(Instr, Pattern, Result, TheInst.getImpResults());
|
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
typedef std::pair<TreePatternNode *, unsigned> NameRecord;
|
2010-02-23 14:55:24 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
static void FindNames(TreePatternNode *P,
|
2010-02-23 15:22:28 +08:00
|
|
|
std::map<std::string, NameRecord> &Names,
|
2012-10-26 04:33:17 +08:00
|
|
|
TreePattern *PatternTop) {
|
2018-06-15 04:32:58 +08:00
|
|
|
if (!P->getName().empty()) {
|
|
|
|
NameRecord &Rec = Names[P->getName()];
|
2010-02-23 14:55:24 +08:00
|
|
|
// If this is the first instance of the name, remember the node.
|
|
|
|
if (Rec.second++ == 0)
|
2018-06-15 04:32:58 +08:00
|
|
|
Rec.first = P;
|
|
|
|
else if (Rec.first->getExtTypes() != P->getExtTypes())
|
|
|
|
PatternTop->error("repetition of value: $" + P->getName() +
|
2010-02-23 15:22:28 +08:00
|
|
|
" where different uses have different types!");
|
2010-02-23 14:55:24 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
if (!P->isLeaf()) {
|
|
|
|
for (unsigned i = 0, e = P->getNumChildren(); i != e; ++i)
|
|
|
|
FindNames(P->getChild(i), Names, PatternTop);
|
2010-02-23 14:35:45 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
std::vector<Predicate> CodeGenDAGPatterns::makePredList(ListInit *L) {
|
|
|
|
std::vector<Predicate> Preds;
|
|
|
|
for (Init *I : L->getValues()) {
|
|
|
|
if (DefInit *Pred = dyn_cast<DefInit>(I))
|
|
|
|
Preds.push_back(Pred->getDef());
|
|
|
|
else
|
|
|
|
llvm_unreachable("Non-def on the list");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort so that different orders get canonicalized to the same string.
|
llvm::sort(C.begin(), C.end(), ...) -> llvm::sort(C, ...)
Summary: The convenience wrapper in STLExtras is available since rL342102.
Reviewers: dblaikie, javed.absar, JDevlieghere, andreadb
Subscribers: MatzeB, sanjoy, arsenm, dschuff, mehdi_amini, sdardis, nemanjai, jvesely, nhaehnle, sbc100, jgravelle-google, eraman, aheejin, kbarton, JDevlieghere, javed.absar, gbedwell, jrtc27, mgrang, atanasyan, steven_wu, george.burgess.iv, dexonsmith, kristina, jsji, llvm-commits
Differential Revision: https://reviews.llvm.org/D52573
llvm-svn: 343163
2018-09-27 10:13:45 +08:00
|
|
|
llvm::sort(Preds);
|
2017-09-15 00:56:21 +08:00
|
|
|
return Preds;
|
|
|
|
}
|
|
|
|
|
2012-10-26 04:33:17 +08:00
|
|
|
void CodeGenDAGPatterns::AddPatternToMatch(TreePattern *Pattern,
|
2017-06-26 01:33:49 +08:00
|
|
|
PatternToMatch &&PTM) {
|
2010-02-23 14:35:45 +08:00
|
|
|
// Do some sanity checking on the pattern we're about to match.
|
2010-02-23 14:16:51 +08:00
|
|
|
std::string Reason;
|
2012-09-20 06:15:06 +08:00
|
|
|
if (!PTM.getSrcPattern()->canPatternMatch(Reason, *this)) {
|
|
|
|
PrintWarning(Pattern->getRecord()->getLoc(),
|
|
|
|
Twine("Pattern can never match: ") + Reason);
|
|
|
|
return;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-02 06:29:19 +08:00
|
|
|
// If the source pattern's root is a complex pattern, that complex pattern
|
|
|
|
// must specify the nodes it can potentially match.
|
|
|
|
if (const ComplexPattern *CP =
|
|
|
|
PTM.getSrcPattern()->getComplexPatternInfo(*this))
|
|
|
|
if (CP->getRootNodes().empty())
|
|
|
|
Pattern->error("ComplexPattern at root must specify list of opcodes it"
|
|
|
|
" could match");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
|
|
|
|
2010-02-23 14:35:45 +08:00
|
|
|
// Find all of the named values in the input and output, ensure they have the
|
|
|
|
// same type.
|
2010-02-23 14:55:24 +08:00
|
|
|
std::map<std::string, NameRecord> SrcNames, DstNames;
|
2018-06-15 04:32:58 +08:00
|
|
|
FindNames(PTM.getSrcPattern(), SrcNames, Pattern);
|
|
|
|
FindNames(PTM.getDstPattern(), DstNames, Pattern);
|
2010-02-23 14:35:45 +08:00
|
|
|
|
|
|
|
// Scan all of the named values in the destination pattern, rejecting them if
|
|
|
|
// they don't exist in the input pattern.
|
2015-11-23 04:46:24 +08:00
|
|
|
for (const auto &Entry : DstNames) {
|
|
|
|
if (SrcNames[Entry.first].first == nullptr)
|
2010-02-23 14:35:45 +08:00
|
|
|
Pattern->error("Pattern has input without matching name in output: $" +
|
2015-11-23 04:46:24 +08:00
|
|
|
Entry.first);
|
2010-02-23 15:50:58 +08:00
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-02-23 14:55:24 +08:00
|
|
|
// Scan all of the named values in the source pattern, rejecting them if the
|
|
|
|
// name isn't used in the dest, and isn't used to tie two values together.
|
2015-11-23 04:46:24 +08:00
|
|
|
for (const auto &Entry : SrcNames)
|
|
|
|
if (DstNames[Entry.first].first == nullptr &&
|
|
|
|
SrcNames[Entry.first].second == 1)
|
|
|
|
Pattern->error("Pattern has dead named input: $" + Entry.first);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2018-06-14 19:56:19 +08:00
|
|
|
PatternsToMatch.push_back(PTM);
|
2010-02-23 14:16:51 +08:00
|
|
|
}
|
|
|
|
|
2008-04-03 08:02:49 +08:00
|
|
|
void CodeGenDAGPatterns::InferInstructionFlags() {
|
2016-02-01 09:33:42 +08:00
|
|
|
ArrayRef<const CodeGenInstruction*> Instructions =
|
2010-03-19 08:34:35 +08:00
|
|
|
Target.getInstructionsByEnumValue();
|
2012-08-24 08:31:16 +08:00
|
|
|
|
2012-08-25 01:08:41 +08:00
|
|
|
unsigned Errors = 0;
|
2012-08-24 08:31:16 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// Try to infer flags from all patterns in PatternToMatch. These include
|
|
|
|
// both the primary instruction patterns (which always come first) and
|
|
|
|
// patterns defined outside the instruction.
|
2017-06-21 00:34:35 +08:00
|
|
|
for (const PatternToMatch &PTM : ptms()) {
|
2012-08-25 06:46:53 +08:00
|
|
|
// We can only infer from single-instruction patterns, otherwise we won't
|
|
|
|
// know which instruction should get the flags.
|
|
|
|
SmallVector<Record*, 8> PatInstrs;
|
2018-06-15 04:32:58 +08:00
|
|
|
getInstructionsInTree(PTM.getDstPattern(), PatInstrs);
|
2012-08-25 06:46:53 +08:00
|
|
|
if (PatInstrs.size() != 1)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Get the single instruction.
|
|
|
|
CodeGenInstruction &InstInfo = Target.getInstruction(PatInstrs.front());
|
|
|
|
|
|
|
|
// Only infer properties from the first pattern. We'll verify the others.
|
|
|
|
if (InstInfo.InferredFrom)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
InstAnalyzer PatInfo(*this);
|
2017-06-21 00:34:37 +08:00
|
|
|
PatInfo.Analyze(PTM);
|
2012-08-25 06:46:53 +08:00
|
|
|
Errors += InferFromPattern(InstInfo, PatInfo, PTM.getSrcRecord());
|
|
|
|
}
|
|
|
|
|
2012-08-25 01:08:41 +08:00
|
|
|
if (Errors)
|
2012-10-26 04:33:17 +08:00
|
|
|
PrintFatalError("pattern conflicts");
|
2012-08-25 01:08:41 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
// If requested by the target, guess any undefined properties.
|
2012-08-24 08:31:16 +08:00
|
|
|
if (Target.guessInstructionProperties()) {
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
for (unsigned i = 0, e = Instructions.size(); i != e; ++i) {
|
|
|
|
CodeGenInstruction *InstInfo =
|
|
|
|
const_cast<CodeGenInstruction *>(Instructions[i]);
|
2015-11-23 04:46:24 +08:00
|
|
|
if (InstInfo->InferredFrom)
|
2012-08-24 08:31:16 +08:00
|
|
|
continue;
|
|
|
|
// The mayLoad and mayStore flags default to false.
|
|
|
|
// Conservatively assume hasSideEffects if it wasn't explicit.
|
2015-11-23 04:46:24 +08:00
|
|
|
if (InstInfo->hasSideEffects_Unset)
|
|
|
|
InstInfo->hasSideEffects = true;
|
2012-08-24 08:31:16 +08:00
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Complain about any flags that are still undefined.
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
for (unsigned i = 0, e = Instructions.size(); i != e; ++i) {
|
|
|
|
CodeGenInstruction *InstInfo =
|
|
|
|
const_cast<CodeGenInstruction *>(Instructions[i]);
|
2015-11-23 04:46:24 +08:00
|
|
|
if (InstInfo->InferredFrom)
|
2012-08-24 08:31:16 +08:00
|
|
|
continue;
|
2015-11-23 04:46:24 +08:00
|
|
|
if (InstInfo->hasSideEffects_Unset)
|
|
|
|
PrintError(InstInfo->TheDef->getLoc(),
|
2012-08-24 08:31:16 +08:00
|
|
|
"Can't infer hasSideEffects from patterns");
|
2015-11-23 04:46:24 +08:00
|
|
|
if (InstInfo->mayStore_Unset)
|
|
|
|
PrintError(InstInfo->TheDef->getLoc(),
|
2012-08-24 08:31:16 +08:00
|
|
|
"Can't infer mayStore from patterns");
|
2015-11-23 04:46:24 +08:00
|
|
|
if (InstInfo->mayLoad_Unset)
|
|
|
|
PrintError(InstInfo->TheDef->getLoc(),
|
2012-08-24 08:31:16 +08:00
|
|
|
"Can't infer mayLoad from patterns");
|
2008-04-03 08:02:49 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-28 11:26:49 +08:00
|
|
|
|
|
|
|
/// Verify instruction flags against pattern node properties.
|
|
|
|
void CodeGenDAGPatterns::VerifyInstructionFlags() {
|
|
|
|
unsigned Errors = 0;
|
|
|
|
for (ptm_iterator I = ptm_begin(), E = ptm_end(); I != E; ++I) {
|
|
|
|
const PatternToMatch &PTM = *I;
|
|
|
|
SmallVector<Record*, 8> Instrs;
|
2018-06-15 04:32:58 +08:00
|
|
|
getInstructionsInTree(PTM.getDstPattern(), Instrs);
|
2012-08-28 11:26:49 +08:00
|
|
|
if (Instrs.empty())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Count the number of instructions with each flag set.
|
|
|
|
unsigned NumSideEffects = 0;
|
|
|
|
unsigned NumStores = 0;
|
|
|
|
unsigned NumLoads = 0;
|
2015-11-23 04:46:24 +08:00
|
|
|
for (const Record *Instr : Instrs) {
|
|
|
|
const CodeGenInstruction &InstInfo = Target.getInstruction(Instr);
|
2012-08-28 11:26:49 +08:00
|
|
|
NumSideEffects += InstInfo.hasSideEffects;
|
|
|
|
NumStores += InstInfo.mayStore;
|
|
|
|
NumLoads += InstInfo.mayLoad;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Analyze the source pattern.
|
|
|
|
InstAnalyzer PatInfo(*this);
|
2017-06-21 00:34:37 +08:00
|
|
|
PatInfo.Analyze(PTM);
|
2012-08-28 11:26:49 +08:00
|
|
|
|
|
|
|
// Collect error messages.
|
|
|
|
SmallVector<std::string, 4> Msgs;
|
|
|
|
|
|
|
|
// Check for missing flags in the output.
|
|
|
|
// Permit extra flags for now at least.
|
|
|
|
if (PatInfo.hasSideEffects && !NumSideEffects)
|
|
|
|
Msgs.push_back("pattern has side effects, but hasSideEffects isn't set");
|
|
|
|
|
|
|
|
// Don't verify store flags on instructions with side effects. At least for
|
|
|
|
// intrinsics, side effects implies mayStore.
|
|
|
|
if (!PatInfo.hasSideEffects && PatInfo.mayStore && !NumStores)
|
|
|
|
Msgs.push_back("pattern may store, but mayStore isn't set");
|
|
|
|
|
|
|
|
// Similarly, mayStore implies mayLoad on intrinsics.
|
|
|
|
if (!PatInfo.mayStore && PatInfo.mayLoad && !NumLoads)
|
|
|
|
Msgs.push_back("pattern may load, but mayLoad isn't set");
|
|
|
|
|
|
|
|
// Print error messages.
|
|
|
|
if (Msgs.empty())
|
|
|
|
continue;
|
|
|
|
++Errors;
|
|
|
|
|
2015-11-23 04:46:24 +08:00
|
|
|
for (const std::string &Msg : Msgs)
|
|
|
|
PrintError(PTM.getSrcRecord()->getLoc(), Twine(Msg) + " on the " +
|
2012-08-28 11:26:49 +08:00
|
|
|
(Instrs.size() == 1 ?
|
|
|
|
"instruction" : "output instructions"));
|
|
|
|
// Provide the location of the relevant instruction definitions.
|
2015-11-23 04:46:24 +08:00
|
|
|
for (const Record *Instr : Instrs) {
|
|
|
|
if (Instr != PTM.getSrcRecord())
|
|
|
|
PrintError(Instr->getLoc(), "defined here");
|
|
|
|
const CodeGenInstruction &InstInfo = Target.getInstruction(Instr);
|
2012-08-28 11:26:49 +08:00
|
|
|
if (InstInfo.InferredFrom &&
|
|
|
|
InstInfo.InferredFrom != InstInfo.TheDef &&
|
|
|
|
InstInfo.InferredFrom != PTM.getSrcRecord())
|
2015-09-12 09:17:08 +08:00
|
|
|
PrintError(InstInfo.InferredFrom->getLoc(), "inferred from pattern");
|
2012-08-28 11:26:49 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (Errors)
|
2012-10-26 04:33:17 +08:00
|
|
|
PrintFatalError("Errors in DAG patterns");
|
2012-08-28 11:26:49 +08:00
|
|
|
}
|
|
|
|
|
2010-03-15 14:00:16 +08:00
|
|
|
/// Given a pattern result with an unresolved type, see if we can find one
|
|
|
|
/// instruction with an unresolved result type. Force this result type to an
|
|
|
|
/// arbitrary element if it's possible types to converge results.
|
2018-06-15 04:32:58 +08:00
|
|
|
static bool ForceArbitraryInstResultType(TreePatternNode *N, TreePattern &TP) {
|
|
|
|
if (N->isLeaf())
|
2010-03-15 14:00:16 +08:00
|
|
|
return false;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-15 14:00:16 +08:00
|
|
|
// Analyze children.
|
2018-06-15 04:32:58 +08:00
|
|
|
for (unsigned i = 0, e = N->getNumChildren(); i != e; ++i)
|
|
|
|
if (ForceArbitraryInstResultType(N->getChild(i), TP))
|
2010-03-15 14:00:16 +08:00
|
|
|
return true;
|
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
if (!N->getOperator()->isSubClassOf("Instruction"))
|
2010-03-15 14:00:16 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// If this type is already concrete or completely unknown we can't do
|
|
|
|
// anything.
|
2017-09-15 00:56:21 +08:00
|
|
|
TypeInfer &TI = TP.getInfer();
|
2018-06-15 04:32:58 +08:00
|
|
|
for (unsigned i = 0, e = N->getNumTypes(); i != e; ++i) {
|
|
|
|
if (N->getExtType(i).empty() || TI.isConcrete(N->getExtType(i), false))
|
2010-03-20 05:37:09 +08:00
|
|
|
continue;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2017-09-15 00:56:21 +08:00
|
|
|
// Otherwise, force its type to an arbitrary choice.
|
2018-06-15 04:32:58 +08:00
|
|
|
if (TI.forceArbitrary(N->getExtType(i)))
|
2010-03-20 05:37:09 +08:00
|
|
|
return true;
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2010-03-20 05:37:09 +08:00
|
|
|
return false;
|
2010-03-15 14:00:16 +08:00
|
|
|
}
|
|
|
|
|
2018-08-01 19:57:58 +08:00
|
|
|
// Promote xform function to be an explicit node wherever set.
|
|
|
|
static TreePatternNodePtr PromoteXForms(TreePatternNodePtr N) {
|
|
|
|
if (Record *Xform = N->getTransformFn()) {
|
|
|
|
N->setTransformFn(nullptr);
|
|
|
|
std::vector<TreePatternNodePtr> Children;
|
|
|
|
Children.push_back(PromoteXForms(N));
|
|
|
|
return std::make_shared<TreePatternNode>(Xform, std::move(Children),
|
|
|
|
N->getNumTypes());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!N->isLeaf())
|
|
|
|
for (unsigned i = 0, e = N->getNumChildren(); i != e; ++i) {
|
|
|
|
TreePatternNodePtr Child = N->getChildShared(i);
|
2018-08-01 20:07:32 +08:00
|
|
|
N->setChild(i, PromoteXForms(Child));
|
2018-08-01 19:57:58 +08:00
|
|
|
}
|
|
|
|
return N;
|
|
|
|
}
|
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
void CodeGenDAGPatterns::ParseOnePattern(Record *TheDef,
|
|
|
|
TreePattern &Pattern, TreePattern &Result,
|
|
|
|
const std::vector<Record *> &InstImpResults) {
|
|
|
|
|
|
|
|
// Inline pattern fragments and expand multiple alternatives.
|
|
|
|
Pattern.InlinePatternFragments();
|
|
|
|
Result.InlinePatternFragments();
|
|
|
|
|
|
|
|
if (Result.getNumTrees() != 1)
|
|
|
|
Result.error("Cannot use multi-alternative fragments in result pattern!");
|
|
|
|
|
|
|
|
// Infer types.
|
|
|
|
bool IterateInference;
|
|
|
|
bool InferredAllPatternTypes, InferredAllResultTypes;
|
|
|
|
do {
|
|
|
|
// Infer as many types as possible. If we cannot infer all of them, we
|
|
|
|
// can never do anything with this pattern: report it to the user.
|
|
|
|
InferredAllPatternTypes =
|
|
|
|
Pattern.InferAllTypes(&Pattern.getNamedNodesMap());
|
|
|
|
|
|
|
|
// Infer as many types as possible. If we cannot infer all of them, we
|
|
|
|
// can never do anything with this pattern: report it to the user.
|
|
|
|
InferredAllResultTypes =
|
|
|
|
Result.InferAllTypes(&Pattern.getNamedNodesMap());
|
|
|
|
|
|
|
|
IterateInference = false;
|
|
|
|
|
|
|
|
// Apply the type of the result to the source pattern. This helps us
|
|
|
|
// resolve cases where the input type is known to be a pointer type (which
|
|
|
|
// is considered resolved), but the result knows it needs to be 32- or
|
|
|
|
// 64-bits. Infer the other way for good measure.
|
|
|
|
for (auto T : Pattern.getTrees())
|
|
|
|
for (unsigned i = 0, e = std::min(Result.getOnlyTree()->getNumTypes(),
|
|
|
|
T->getNumTypes());
|
|
|
|
i != e; ++i) {
|
|
|
|
IterateInference |= T->UpdateNodeType(
|
|
|
|
i, Result.getOnlyTree()->getExtType(i), Result);
|
|
|
|
IterateInference |= Result.getOnlyTree()->UpdateNodeType(
|
|
|
|
i, T->getExtType(i), Result);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If our iteration has converged and the input pattern's types are fully
|
|
|
|
// resolved but the result pattern is not fully resolved, we may have a
|
|
|
|
// situation where we have two instructions in the result pattern and
|
|
|
|
// the instructions require a common register class, but don't care about
|
|
|
|
// what actual MVT is used. This is actually a bug in our modelling:
|
|
|
|
// output patterns should have register classes, not MVTs.
|
|
|
|
//
|
|
|
|
// In any case, to handle this, we just go through and disambiguate some
|
|
|
|
// arbitrary types to the result pattern's nodes.
|
|
|
|
if (!IterateInference && InferredAllPatternTypes &&
|
|
|
|
!InferredAllResultTypes)
|
|
|
|
IterateInference =
|
|
|
|
ForceArbitraryInstResultType(Result.getTree(0).get(), Result);
|
|
|
|
} while (IterateInference);
|
|
|
|
|
|
|
|
// Verify that we inferred enough types that we can do something with the
|
|
|
|
// pattern and result. If these fire the user has to add type casts.
|
|
|
|
if (!InferredAllPatternTypes)
|
|
|
|
Pattern.error("Could not infer all types in pattern!");
|
|
|
|
if (!InferredAllResultTypes) {
|
|
|
|
Pattern.dump();
|
|
|
|
Result.error("Could not infer all types in pattern result!");
|
|
|
|
}
|
|
|
|
|
2018-08-01 19:57:58 +08:00
|
|
|
// Promote xform function to be an explicit node wherever set.
|
|
|
|
TreePatternNodePtr DstShared = PromoteXForms(Result.getOnlyTree());
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
|
|
|
|
TreePattern Temp(Result.getRecord(), DstShared, false, *this);
|
|
|
|
Temp.InferAllTypes();
|
|
|
|
|
|
|
|
ListInit *Preds = TheDef->getValueAsListInit("Predicates");
|
|
|
|
int Complexity = TheDef->getValueAsInt("AddedComplexity");
|
|
|
|
|
|
|
|
if (PatternRewriter)
|
|
|
|
PatternRewriter(&Pattern);
|
|
|
|
|
|
|
|
// A pattern may end up with an "impossible" type, i.e. a situation
|
|
|
|
// where all types have been eliminated for some node in this pattern.
|
|
|
|
// This could occur for intrinsics that only make sense for a specific
|
|
|
|
// value type, and use a specific register class. If, for some mode,
|
|
|
|
// that register class does not accept that type, the type inference
|
|
|
|
// will lead to a contradiction, which is not an error however, but
|
|
|
|
// a sign that this pattern will simply never match.
|
|
|
|
if (Temp.getOnlyTree()->hasPossibleType())
|
|
|
|
for (auto T : Pattern.getTrees())
|
|
|
|
if (T->hasPossibleType())
|
|
|
|
AddPatternToMatch(&Pattern,
|
|
|
|
PatternToMatch(TheDef, makePredList(Preds),
|
|
|
|
T, Temp.getOnlyTree(),
|
|
|
|
InstImpResults, Complexity,
|
|
|
|
TheDef->getID()));
|
|
|
|
}
|
|
|
|
|
2008-01-06 09:10:31 +08:00
|
|
|
void CodeGenDAGPatterns::ParsePatterns() {
|
2008-01-06 06:25:12 +08:00
|
|
|
std::vector<Record*> Patterns = Records.getAllDerivedDefinitions("Pattern");
|
|
|
|
|
2015-11-23 04:46:24 +08:00
|
|
|
for (Record *CurPattern : Patterns) {
|
2011-07-30 06:43:06 +08:00
|
|
|
DagInit *Tree = CurPattern->getValueAsDag("PatternToMatch");
|
2012-07-18 02:39:36 +08:00
|
|
|
|
|
|
|
// If the pattern references the null_frag, there's nothing to do.
|
|
|
|
if (hasNullFragReference(Tree))
|
|
|
|
continue;
|
|
|
|
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePattern Pattern(CurPattern, Tree, true, *this);
|
2008-01-06 06:25:12 +08:00
|
|
|
|
2011-07-30 06:43:06 +08:00
|
|
|
ListInit *LI = CurPattern->getValueAsListInit("ResultInstrs");
|
2015-05-14 13:53:53 +08:00
|
|
|
if (LI->empty()) continue; // no pattern.
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Parse the instruction.
|
2014-11-15 05:53:50 +08:00
|
|
|
TreePattern Result(CurPattern, LI, false, *this);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2014-11-15 05:53:50 +08:00
|
|
|
if (Result.getNumTrees() != 1)
|
|
|
|
Result.error("Cannot handle instructions producing instructions "
|
|
|
|
"with temporaries yet!");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Validate that the input pattern is correct.
|
2018-05-31 05:00:18 +08:00
|
|
|
std::map<std::string, TreePatternNodePtr> InstInputs;
|
[TableGen] Preserve order of output operands in DAGISelMatcherGen
Summary:
This fixes support in DAGISelMatcher backend for DAG nodes with multiple
result values. Previously the order of results in selected DAG nodes always
matched the order of results in ISel patterns. After the change the order of
results matches the order of operands in OutOperandList instead.
For example, given this definition from the attached test case:
def INSTR : Instruction {
let OutOperandList = (outs GPR:$r1, GPR:$r0);
let InOperandList = (ins GPR:$t0, GPR:$t1);
let Pattern = [(set i32:$r0, i32:$r1, (udivrem i32:$t0, i32:$t1))];
}
the DAGISelMatcher backend currently produces a matcher that creates INSTR
nodes with the first result `$r0` and the second result `$r1`, contrary to the
order in the OutOperandList. The order of operands in OutOperandList does not
matter at all, which is unexpected (and unfortunate) because the order of
results of a DAG node does matters, perhaps a lot.
With this change, if the order in OutOperandList does not match the order in
Pattern, DAGISelMatcherGen emits CompleteMatch opcodes with the order of
results taken from OutOperandList. Backend writers can use it to express
result reorderings in TableGen.
If the order in OutOperandList matches the order in Pattern, the result of
DAGISelMatcherGen is unaffected.
Patch by Eugene Sharygin
Reviewers: andreadb, bjope, hfinkel, RKSimon, craig.topper
Reviewed By: craig.topper
Subscribers: nhaehnle, craig.topper, llvm-commits
Differential Revision: https://reviews.llvm.org/D55055
llvm-svn: 348326
2018-12-05 08:47:59 +08:00
|
|
|
MapVector<std::string, TreePatternNodePtr, std::map<std::string, unsigned>>
|
|
|
|
InstResults;
|
2008-01-06 06:25:12 +08:00
|
|
|
std::vector<Record*> InstImpResults;
|
2018-05-31 05:00:18 +08:00
|
|
|
for (unsigned j = 0, ee = Pattern.getNumTrees(); j != ee; ++j)
|
2018-06-12 06:14:43 +08:00
|
|
|
FindPatternInputsAndOutputs(Pattern, Pattern.getTree(j), InstInputs,
|
2018-05-31 05:00:18 +08:00
|
|
|
InstResults, InstImpResults);
|
2008-01-06 06:25:12 +08:00
|
|
|
|
[TableGen] Support multi-alternative pattern fragments
A TableGen instruction record usually contains a DAG pattern that will
describe the SelectionDAG operation that can be implemented by this
instruction. However, there will be cases where several different DAG
patterns can all be implemented by the same instruction. The way to
represent this today is to write additional patterns in the Pattern
(or usually Pat) class that map those extra DAG patterns to the
instruction. This usually also works fine.
However, I've noticed cases where the current setup seems to require
quite a bit of extra (and duplicated) text in the target .td files.
For example, in the SystemZ back-end, there are quite a number of
instructions that can implement an "add-with-overflow" operation.
The same instructions also need to be used to implement just plain
addition (simply ignoring the extra overflow output). The current
solution requires creating extra Pat pattern for every instruction,
duplicating the information about which particular add operands
map best to which particular instruction.
This patch enhances TableGen to support a new PatFrags class, which
can be used to encapsulate multiple alternative patterns that may
all match to the same instruction. It operates the same way as the
existing PatFrag class, except that it accepts a list of DAG patterns
to match instead of just a single one. As an example, we can now define
a PatFrags to match either an "add-with-overflow" or a regular add
operation:
def z_sadd : PatFrags<(ops node:$src1, node:$src2),
[(z_saddo node:$src1, node:$src2),
(add node:$src1, node:$src2)]>;
and then use this in the add instruction pattern:
defm AR : BinaryRRAndK<"ar", 0x1A, 0xB9F8, z_sadd, GR32, GR32>;
These SystemZ target changes are implemented here as well.
Note that PatFrag is now defined as a subclass of PatFrags, which
means that some users of internals of PatFrag need to be updated.
(E.g. instead of using PatFrag.Fragment you now need to use
!head(PatFrag.Fragments).)
The implementation is based on the following main ideas:
- InlinePatternFragments may now replace each original pattern
with several result patterns, not just one.
- parseInstructionPattern delays calling InlinePatternFragments
and InferAllTypes. Instead, it extracts a single DAG match
pattern from the main instruction pattern.
- Processing of the DAG match pattern part of the main instruction
pattern now shares most code with processing match patterns from
the Pattern class.
- Direct use of main instruction patterns in InferFromPattern and
EmitResultInstructionAsOperand is removed; everything now operates
solely on DAG match patterns.
Reviewed by: hfinkel
Differential Revision: https://reviews.llvm.org/D48545
llvm-svn: 336999
2018-07-13 21:18:00 +08:00
|
|
|
ParseOnePattern(CurPattern, Pattern, Result, InstImpResults);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
static void collectModes(std::set<unsigned> &Modes, const TreePatternNode *N) {
|
2017-09-15 00:56:21 +08:00
|
|
|
for (const TypeSetByHwMode &VTS : N->getExtTypes())
|
|
|
|
for (const auto &I : VTS)
|
|
|
|
Modes.insert(I.first);
|
|
|
|
|
|
|
|
for (unsigned i = 0, e = N->getNumChildren(); i != e; ++i)
|
2018-06-15 04:32:58 +08:00
|
|
|
collectModes(Modes, N->getChild(i));
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void CodeGenDAGPatterns::ExpandHwModeBasedTypes() {
|
|
|
|
const CodeGenHwModes &CGH = getTargetInfo().getHwModes();
|
|
|
|
std::map<unsigned,std::vector<Predicate>> ModeChecks;
|
|
|
|
std::vector<PatternToMatch> Copy = PatternsToMatch;
|
|
|
|
PatternsToMatch.clear();
|
|
|
|
|
2018-05-31 05:00:18 +08:00
|
|
|
auto AppendPattern = [this, &ModeChecks](PatternToMatch &P, unsigned Mode) {
|
|
|
|
TreePatternNodePtr NewSrc = P.SrcPattern->clone();
|
|
|
|
TreePatternNodePtr NewDst = P.DstPattern->clone();
|
2017-09-15 00:56:21 +08:00
|
|
|
if (!NewSrc->setDefaultMode(Mode) || !NewDst->setDefaultMode(Mode)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::vector<Predicate> Preds = P.Predicates;
|
|
|
|
const std::vector<Predicate> &MC = ModeChecks[Mode];
|
|
|
|
Preds.insert(Preds.end(), MC.begin(), MC.end());
|
2018-06-14 19:56:19 +08:00
|
|
|
PatternsToMatch.emplace_back(P.getSrcRecord(), Preds, std::move(NewSrc),
|
|
|
|
std::move(NewDst), P.getDstRegs(),
|
|
|
|
P.getAddedComplexity(), Record::getNewUID(),
|
|
|
|
Mode);
|
2017-09-15 00:56:21 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
for (PatternToMatch &P : Copy) {
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr SrcP = nullptr, DstP = nullptr;
|
2018-06-15 04:32:58 +08:00
|
|
|
if (P.SrcPattern->hasProperTypeByHwMode())
|
|
|
|
SrcP = P.SrcPattern;
|
|
|
|
if (P.DstPattern->hasProperTypeByHwMode())
|
|
|
|
DstP = P.DstPattern;
|
|
|
|
if (!SrcP && !DstP) {
|
2017-09-15 00:56:21 +08:00
|
|
|
PatternsToMatch.push_back(P);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::set<unsigned> Modes;
|
2018-06-15 04:32:58 +08:00
|
|
|
if (SrcP)
|
|
|
|
collectModes(Modes, SrcP.get());
|
|
|
|
if (DstP)
|
|
|
|
collectModes(Modes, DstP.get());
|
2017-09-15 00:56:21 +08:00
|
|
|
|
|
|
|
// The predicate for the default mode needs to be constructed for each
|
|
|
|
// pattern separately.
|
|
|
|
// Since not all modes must be present in each pattern, if a mode m is
|
|
|
|
// absent, then there is no point in constructing a check for m. If such
|
|
|
|
// a check was created, it would be equivalent to checking the default
|
|
|
|
// mode, except not all modes' predicates would be a part of the checking
|
|
|
|
// code. The subsequently generated check for the default mode would then
|
|
|
|
// have the exact same patterns, but a different predicate code. To avoid
|
|
|
|
// duplicated patterns with different predicate checks, construct the
|
|
|
|
// default check as a negation of all predicates that are actually present
|
|
|
|
// in the source/destination patterns.
|
|
|
|
std::vector<Predicate> DefaultPred;
|
|
|
|
|
|
|
|
for (unsigned M : Modes) {
|
|
|
|
if (M == DefaultMode)
|
|
|
|
continue;
|
|
|
|
if (ModeChecks.find(M) != ModeChecks.end())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Fill the map entry for this mode.
|
|
|
|
const HwMode &HM = CGH.getMode(M);
|
|
|
|
ModeChecks[M].emplace_back(Predicate(HM.Features, true));
|
|
|
|
|
|
|
|
// Add negations of the HM's predicates to the default predicate.
|
|
|
|
DefaultPred.emplace_back(Predicate(HM.Features, false));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned M : Modes) {
|
|
|
|
if (M == DefaultMode)
|
|
|
|
continue;
|
|
|
|
AppendPattern(P, M);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool HasDefault = Modes.count(DefaultMode);
|
|
|
|
if (HasDefault)
|
|
|
|
AppendPattern(P, DefaultMode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Dependent variable map for CodeGenDAGPattern variant generation
|
2017-09-21 02:01:40 +08:00
|
|
|
typedef StringMap<int> DepVarMap;
|
2017-09-15 00:56:21 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
static void FindDepVarsOf(TreePatternNode *N, DepVarMap &DepMap) {
|
|
|
|
if (N->isLeaf()) {
|
|
|
|
if (N->hasName() && isa<DefInit>(N->getLeafValue()))
|
|
|
|
DepMap[N->getName()]++;
|
2017-09-15 00:56:21 +08:00
|
|
|
} else {
|
2018-06-15 04:32:58 +08:00
|
|
|
for (size_t i = 0, e = N->getNumChildren(); i != e; ++i)
|
|
|
|
FindDepVarsOf(N->getChild(i), DepMap);
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Find dependent variables within child patterns
|
2018-06-15 04:32:58 +08:00
|
|
|
static void FindDepVars(TreePatternNode *N, MultipleUseVarSet &DepVars) {
|
2017-09-15 00:56:21 +08:00
|
|
|
DepVarMap depcounts;
|
|
|
|
FindDepVarsOf(N, depcounts);
|
2017-09-21 02:01:40 +08:00
|
|
|
for (const auto &Pair : depcounts) {
|
|
|
|
if (Pair.getValue() > 1)
|
|
|
|
DepVars.insert(Pair.getKey());
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef NDEBUG
|
|
|
|
/// Dump the dependent variable set:
|
|
|
|
static void DumpDepVars(MultipleUseVarSet &DepVars) {
|
|
|
|
if (DepVars.empty()) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(errs() << "<empty set>");
|
2017-09-15 00:56:21 +08:00
|
|
|
} else {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(errs() << "[ ");
|
2017-09-21 02:01:40 +08:00
|
|
|
for (const auto &DepVar : DepVars) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(errs() << DepVar.getKey() << " ");
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(errs() << "]");
|
2017-09-15 00:56:21 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
/// CombineChildVariants - Given a bunch of permutations of each child of the
|
|
|
|
/// 'operator' node, put them together in all possible ways.
|
2018-05-31 05:00:18 +08:00
|
|
|
static void CombineChildVariants(
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNodePtr Orig,
|
2018-05-31 05:00:18 +08:00
|
|
|
const std::vector<std::vector<TreePatternNodePtr>> &ChildVariants,
|
|
|
|
std::vector<TreePatternNodePtr> &OutVariants, CodeGenDAGPatterns &CDP,
|
|
|
|
const MultipleUseVarSet &DepVars) {
|
2008-01-06 06:25:12 +08:00
|
|
|
// Make sure that each operand has at least one variant to choose from.
|
2015-11-23 04:46:24 +08:00
|
|
|
for (const auto &Variants : ChildVariants)
|
|
|
|
if (Variants.empty())
|
2008-01-06 06:25:12 +08:00
|
|
|
return;
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// The end result is an all-pairs construction of the resultant pattern.
|
|
|
|
std::vector<unsigned> Idxs;
|
|
|
|
Idxs.resize(ChildVariants.size());
|
2008-03-06 01:49:05 +08:00
|
|
|
bool NotDone;
|
|
|
|
do {
|
|
|
|
#ifndef NDEBUG
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(if (!Idxs.empty()) {
|
2018-06-15 04:32:58 +08:00
|
|
|
errs() << Orig->getOperator()->getName() << ": Idxs = [ ";
|
2018-05-14 20:53:11 +08:00
|
|
|
for (unsigned Idx : Idxs) {
|
|
|
|
errs() << Idx << " ";
|
|
|
|
}
|
|
|
|
errs() << "]\n";
|
|
|
|
});
|
2008-03-06 01:49:05 +08:00
|
|
|
#endif
|
2008-01-06 06:25:12 +08:00
|
|
|
// Create the variant and add it to the output list.
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> NewChildren;
|
2008-01-06 06:25:12 +08:00
|
|
|
for (unsigned i = 0, e = ChildVariants.size(); i != e; ++i)
|
|
|
|
NewChildren.push_back(ChildVariants[i][Idxs[i]]);
|
2018-05-31 05:00:18 +08:00
|
|
|
TreePatternNodePtr R = std::make_shared<TreePatternNode>(
|
2018-07-15 14:52:49 +08:00
|
|
|
Orig->getOperator(), std::move(NewChildren), Orig->getNumTypes());
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Copy over properties.
|
2018-06-15 04:32:58 +08:00
|
|
|
R->setName(Orig->getName());
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
R->setNamesAsPredicateArg(Orig->getNamesAsPredicateArg());
|
|
|
|
R->setPredicateCalls(Orig->getPredicateCalls());
|
2018-06-15 04:32:58 +08:00
|
|
|
R->setTransformFn(Orig->getTransformFn());
|
|
|
|
for (unsigned i = 0, e = Orig->getNumTypes(); i != e; ++i)
|
|
|
|
R->setType(i, Orig->getExtType(i));
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-03-06 01:49:05 +08:00
|
|
|
// If this pattern cannot match, do not include it as a variant.
|
2008-01-06 06:25:12 +08:00
|
|
|
std::string ErrString;
|
2015-11-23 04:11:21 +08:00
|
|
|
// Scan to see if this pattern has already been emitted. We can get
|
|
|
|
// duplication due to things like commuting:
|
|
|
|
// (and GPRC:$a, GPRC:$b) -> (and GPRC:$b, GPRC:$a)
|
|
|
|
// which are the same pattern. Ignore the dups.
|
|
|
|
if (R->canPatternMatch(ErrString, CDP) &&
|
2018-05-31 05:00:18 +08:00
|
|
|
none_of(OutVariants, [&](TreePatternNodePtr Variant) {
|
2018-06-15 04:32:58 +08:00
|
|
|
return R->isIsomorphicTo(Variant.get(), DepVars);
|
2016-08-12 05:15:00 +08:00
|
|
|
}))
|
2018-05-31 05:00:18 +08:00
|
|
|
OutVariants.push_back(R);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-03-06 01:49:05 +08:00
|
|
|
// Increment indices to the next permutation by incrementing the
|
2015-09-12 09:17:08 +08:00
|
|
|
// indices from last index backward, e.g., generate the sequence
|
2008-03-06 01:49:05 +08:00
|
|
|
// [0, 0], [0, 1], [1, 0], [1, 1].
|
|
|
|
int IdxsIdx;
|
|
|
|
for (IdxsIdx = Idxs.size() - 1; IdxsIdx >= 0; --IdxsIdx) {
|
|
|
|
if (++Idxs[IdxsIdx] == ChildVariants[IdxsIdx].size())
|
|
|
|
Idxs[IdxsIdx] = 0;
|
|
|
|
else
|
2008-01-06 06:25:12 +08:00
|
|
|
break;
|
|
|
|
}
|
2008-03-06 01:49:05 +08:00
|
|
|
NotDone = (IdxsIdx >= 0);
|
|
|
|
} while (NotDone);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// CombineChildVariants - A helper function for binary operators.
|
|
|
|
///
|
2018-06-15 04:32:58 +08:00
|
|
|
static void CombineChildVariants(TreePatternNodePtr Orig,
|
2018-05-31 05:00:18 +08:00
|
|
|
const std::vector<TreePatternNodePtr> &LHS,
|
|
|
|
const std::vector<TreePatternNodePtr> &RHS,
|
|
|
|
std::vector<TreePatternNodePtr> &OutVariants,
|
2008-03-06 01:49:05 +08:00
|
|
|
CodeGenDAGPatterns &CDP,
|
|
|
|
const MultipleUseVarSet &DepVars) {
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<std::vector<TreePatternNodePtr>> ChildVariants;
|
2008-01-06 06:25:12 +08:00
|
|
|
ChildVariants.push_back(LHS);
|
|
|
|
ChildVariants.push_back(RHS);
|
2008-03-06 01:49:05 +08:00
|
|
|
CombineChildVariants(Orig, ChildVariants, OutVariants, CDP, DepVars);
|
2010-12-22 00:16:00 +08:00
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
|
2018-05-31 05:00:18 +08:00
|
|
|
static void
|
2018-06-15 04:32:58 +08:00
|
|
|
GatherChildrenOfAssociativeOpcode(TreePatternNodePtr N,
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> &Children) {
|
2008-01-06 06:25:12 +08:00
|
|
|
assert(N->getNumChildren()==2 &&"Associative but doesn't have 2 children!");
|
|
|
|
Record *Operator = N->getOperator();
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Only permit raw nodes.
|
TableGen/ISel: Allow PatFrag predicate code to access captured operands
Summary:
This simplifies writing predicates for pattern fragments that are
automatically re-associated or commuted.
For example, a followup patch adds patterns for fragments of the form
(add (shl $x, $y), $z) to the AMDGPU backend. Such patterns are
automatically commuted to (add $z, (shl $x, $y)), which makes it basically
impossible to refer to $x, $y, and $z generically in the PredicateCode.
With this change, the PredicateCode can refer to $x, $y, and $z simply
as `Operands[i]`.
Test confirmed that there are no changes to any of the generated files
when building all (non-experimental) targets.
Change-Id: I61c00ace7eed42c1d4edc4c5351174b56b77a79c
Reviewers: arsenm, rampitec, RKSimon, craig.topper, hfinkel, uweigand
Subscribers: wdng, tpr, llvm-commits
Differential Revision: https://reviews.llvm.org/D51994
llvm-svn: 347992
2018-11-30 22:15:13 +08:00
|
|
|
if (!N->getName().empty() || !N->getPredicateCalls().empty() ||
|
2008-01-06 06:25:12 +08:00
|
|
|
N->getTransformFn()) {
|
|
|
|
Children.push_back(N);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
if (N->getChild(0)->isLeaf() || N->getChild(0)->getOperator() != Operator)
|
2018-05-31 05:00:18 +08:00
|
|
|
Children.push_back(N->getChildShared(0));
|
2008-01-06 06:25:12 +08:00
|
|
|
else
|
2018-05-31 05:00:18 +08:00
|
|
|
GatherChildrenOfAssociativeOpcode(N->getChildShared(0), Children);
|
2008-01-06 06:25:12 +08:00
|
|
|
|
2018-06-15 04:32:58 +08:00
|
|
|
if (N->getChild(1)->isLeaf() || N->getChild(1)->getOperator() != Operator)
|
2018-05-31 05:00:18 +08:00
|
|
|
Children.push_back(N->getChildShared(1));
|
2008-01-06 06:25:12 +08:00
|
|
|
else
|
2018-05-31 05:00:18 +08:00
|
|
|
GatherChildrenOfAssociativeOpcode(N->getChildShared(1), Children);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// GenerateVariantsOf - Given a pattern N, generate all permutations we can of
|
|
|
|
/// the (potentially recursive) pattern by using algebraic laws.
|
|
|
|
///
|
2018-06-15 04:32:58 +08:00
|
|
|
static void GenerateVariantsOf(TreePatternNodePtr N,
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> &OutVariants,
|
2008-03-06 01:49:05 +08:00
|
|
|
CodeGenDAGPatterns &CDP,
|
|
|
|
const MultipleUseVarSet &DepVars) {
|
2014-05-20 19:52:46 +08:00
|
|
|
// We cannot permute leaves or ComplexPattern uses.
|
|
|
|
if (N->isLeaf() || N->getOperator()->isSubClassOf("ComplexPattern")) {
|
2008-01-06 06:25:12 +08:00
|
|
|
OutVariants.push_back(N);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Look up interesting info about the node.
|
|
|
|
const SDNodeInfo &NodeInfo = CDP.getSDNodeInfo(N->getOperator());
|
|
|
|
|
2009-03-27 00:17:51 +08:00
|
|
|
// If this node is associative, re-associate.
|
2008-01-06 06:25:12 +08:00
|
|
|
if (NodeInfo.hasProperty(SDNPAssociative)) {
|
2010-12-22 00:16:00 +08:00
|
|
|
// Re-associate by pulling together all of the linked operators
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> MaximalChildren;
|
2008-01-06 06:25:12 +08:00
|
|
|
GatherChildrenOfAssociativeOpcode(N, MaximalChildren);
|
|
|
|
|
|
|
|
// Only handle child sizes of 3. Otherwise we'll end up trying too many
|
|
|
|
// permutations.
|
|
|
|
if (MaximalChildren.size() == 3) {
|
|
|
|
// Find the variants of all of our maximal children.
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> AVariants, BVariants, CVariants;
|
2008-03-06 01:49:05 +08:00
|
|
|
GenerateVariantsOf(MaximalChildren[0], AVariants, CDP, DepVars);
|
|
|
|
GenerateVariantsOf(MaximalChildren[1], BVariants, CDP, DepVars);
|
|
|
|
GenerateVariantsOf(MaximalChildren[2], CVariants, CDP, DepVars);
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// There are only two ways we can permute the tree:
|
|
|
|
// (A op B) op C and A op (B op C)
|
|
|
|
// Within these forms, we can also permute A/B/C.
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Generate legal pair permutations of A/B/C.
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> ABVariants;
|
|
|
|
std::vector<TreePatternNodePtr> BAVariants;
|
|
|
|
std::vector<TreePatternNodePtr> ACVariants;
|
|
|
|
std::vector<TreePatternNodePtr> CAVariants;
|
|
|
|
std::vector<TreePatternNodePtr> BCVariants;
|
|
|
|
std::vector<TreePatternNodePtr> CBVariants;
|
2018-06-15 04:32:58 +08:00
|
|
|
CombineChildVariants(N, AVariants, BVariants, ABVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, BVariants, AVariants, BAVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, AVariants, CVariants, ACVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, CVariants, AVariants, CAVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, BVariants, CVariants, BCVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, CVariants, BVariants, CBVariants, CDP, DepVars);
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
// Combine those into the result: (x op x) op x
|
2018-06-15 04:32:58 +08:00
|
|
|
CombineChildVariants(N, ABVariants, CVariants, OutVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, BAVariants, CVariants, OutVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, ACVariants, BVariants, OutVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, CAVariants, BVariants, OutVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, BCVariants, AVariants, OutVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, CBVariants, AVariants, OutVariants, CDP, DepVars);
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
// Combine those into the result: x op (x op x)
|
2018-06-15 04:32:58 +08:00
|
|
|
CombineChildVariants(N, CVariants, ABVariants, OutVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, CVariants, BAVariants, OutVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, BVariants, ACVariants, OutVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, BVariants, CAVariants, OutVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, AVariants, BCVariants, OutVariants, CDP, DepVars);
|
|
|
|
CombineChildVariants(N, AVariants, CBVariants, OutVariants, CDP, DepVars);
|
2008-01-06 06:25:12 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Compute permutations of all children.
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<std::vector<TreePatternNodePtr>> ChildVariants;
|
2008-01-06 06:25:12 +08:00
|
|
|
ChildVariants.resize(N->getNumChildren());
|
|
|
|
for (unsigned i = 0, e = N->getNumChildren(); i != e; ++i)
|
2018-05-31 05:00:18 +08:00
|
|
|
GenerateVariantsOf(N->getChildShared(i), ChildVariants[i], CDP, DepVars);
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
// Build all permutations based on how the children were formed.
|
2018-06-15 04:32:58 +08:00
|
|
|
CombineChildVariants(N, ChildVariants, OutVariants, CDP, DepVars);
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
// If this node is commutative, consider the commuted order.
|
2008-06-17 04:29:38 +08:00
|
|
|
bool isCommIntrinsic = N->isCommutativeIntrinsic(CDP);
|
|
|
|
if (NodeInfo.hasProperty(SDNPCommutative) || isCommIntrinsic) {
|
2017-09-04 11:44:33 +08:00
|
|
|
assert((N->getNumChildren()>=2 || isCommIntrinsic) &&
|
2008-06-17 04:29:38 +08:00
|
|
|
"Commutative but doesn't have 2 children!");
|
2008-01-06 06:25:12 +08:00
|
|
|
// Don't count children which are actually register references.
|
|
|
|
unsigned NC = 0;
|
|
|
|
for (unsigned i = 0, e = N->getNumChildren(); i != e; ++i) {
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNode *Child = N->getChild(i);
|
|
|
|
if (Child->isLeaf())
|
|
|
|
if (DefInit *DI = dyn_cast<DefInit>(Child->getLeafValue())) {
|
2008-01-06 06:25:12 +08:00
|
|
|
Record *RR = DI->getDef();
|
|
|
|
if (RR->isSubClassOf("Register"))
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
NC++;
|
|
|
|
}
|
|
|
|
// Consider the commuted order.
|
2008-06-17 04:29:38 +08:00
|
|
|
if (isCommIntrinsic) {
|
|
|
|
// Commutative intrinsic. First operand is the intrinsic id, 2nd and 3rd
|
|
|
|
// operands are the commutative operands, and there might be more operands
|
|
|
|
// after those.
|
|
|
|
assert(NC >= 3 &&
|
2015-09-12 09:17:08 +08:00
|
|
|
"Commutative intrinsic should have at least 3 children!");
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<std::vector<TreePatternNodePtr>> Variants;
|
2018-06-14 19:56:19 +08:00
|
|
|
Variants.push_back(std::move(ChildVariants[0])); // Intrinsic id.
|
|
|
|
Variants.push_back(std::move(ChildVariants[2]));
|
|
|
|
Variants.push_back(std::move(ChildVariants[1]));
|
2008-06-17 04:29:38 +08:00
|
|
|
for (unsigned i = 3; i != NC; ++i)
|
2018-06-14 19:56:19 +08:00
|
|
|
Variants.push_back(std::move(ChildVariants[i]));
|
2018-06-15 04:32:58 +08:00
|
|
|
CombineChildVariants(N, Variants, OutVariants, CDP, DepVars);
|
2017-09-04 11:44:33 +08:00
|
|
|
} else if (NC == N->getNumChildren()) {
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<std::vector<TreePatternNodePtr>> Variants;
|
2018-06-14 19:56:19 +08:00
|
|
|
Variants.push_back(std::move(ChildVariants[1]));
|
|
|
|
Variants.push_back(std::move(ChildVariants[0]));
|
2017-09-04 11:44:33 +08:00
|
|
|
for (unsigned i = 2; i != NC; ++i)
|
2018-06-14 19:56:19 +08:00
|
|
|
Variants.push_back(std::move(ChildVariants[i]));
|
2018-06-15 04:32:58 +08:00
|
|
|
CombineChildVariants(N, Variants, OutVariants, CDP, DepVars);
|
2017-09-04 11:44:33 +08:00
|
|
|
}
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// GenerateVariants - Generate variants. For example, commutative patterns can
|
|
|
|
// match multiple ways. Add them to PatternsToMatch as well.
|
2008-01-06 09:10:31 +08:00
|
|
|
void CodeGenDAGPatterns::GenerateVariants() {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(errs() << "Generating instruction variants.\n");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Loop over all of the patterns we've collected, checking to see if we can
|
|
|
|
// generate variants of the instruction, through the exploitation of
|
2009-03-27 00:17:51 +08:00
|
|
|
// identities. This permits the target to provide aggressive matching without
|
2008-01-06 06:25:12 +08:00
|
|
|
// the .td file having to contain tons of variants of instructions.
|
|
|
|
//
|
|
|
|
// Note that this loop adds new patterns to the PatternsToMatch list, but we
|
|
|
|
// intentionally do not reconsider these. Any variants of added patterns have
|
|
|
|
// already been added.
|
|
|
|
//
|
2018-09-18 19:30:30 +08:00
|
|
|
const unsigned NumOriginalPatterns = PatternsToMatch.size();
|
|
|
|
BitVector MatchedPatterns(NumOriginalPatterns);
|
|
|
|
std::vector<BitVector> MatchedPredicates(NumOriginalPatterns,
|
|
|
|
BitVector(NumOriginalPatterns));
|
|
|
|
|
|
|
|
typedef std::pair<MultipleUseVarSet, std::vector<TreePatternNodePtr>>
|
|
|
|
DepsAndVariants;
|
|
|
|
std::map<unsigned, DepsAndVariants> PatternsWithVariants;
|
|
|
|
|
|
|
|
// Collect patterns with more than one variant.
|
|
|
|
for (unsigned i = 0; i != NumOriginalPatterns; ++i) {
|
|
|
|
MultipleUseVarSet DepVars;
|
2018-05-31 05:00:18 +08:00
|
|
|
std::vector<TreePatternNodePtr> Variants;
|
2018-06-15 04:32:58 +08:00
|
|
|
FindDepVars(PatternsToMatch[i].getSrcPattern(), DepVars);
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(errs() << "Dependent/multiply used variables: ");
|
|
|
|
LLVM_DEBUG(DumpDepVars(DepVars));
|
|
|
|
LLVM_DEBUG(errs() << "\n");
|
2018-05-31 05:00:18 +08:00
|
|
|
GenerateVariantsOf(PatternsToMatch[i].getSrcPatternShared(), Variants,
|
|
|
|
*this, DepVars);
|
2008-01-06 06:25:12 +08:00
|
|
|
|
|
|
|
assert(!Variants.empty() && "Must create at least original variant!");
|
2018-09-18 19:30:30 +08:00
|
|
|
if (Variants.size() == 1) // No additional variants for this pattern.
|
2008-01-06 06:25:12 +08:00
|
|
|
continue;
|
|
|
|
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(errs() << "FOUND VARIANTS OF: ";
|
|
|
|
PatternsToMatch[i].getSrcPattern()->dump(); errs() << "\n");
|
2008-01-06 06:25:12 +08:00
|
|
|
|
2018-09-18 19:30:30 +08:00
|
|
|
PatternsWithVariants[i] = std::make_pair(DepVars, Variants);
|
|
|
|
|
2018-08-28 23:42:08 +08:00
|
|
|
// Cache matching predicates.
|
2018-09-18 19:30:30 +08:00
|
|
|
if (MatchedPatterns[i])
|
|
|
|
continue;
|
|
|
|
|
|
|
|
const std::vector<Predicate> &Predicates =
|
|
|
|
PatternsToMatch[i].getPredicates();
|
|
|
|
|
|
|
|
BitVector &Matches = MatchedPredicates[i];
|
2018-09-19 20:23:50 +08:00
|
|
|
MatchedPatterns.set(i);
|
|
|
|
Matches.set(i);
|
2018-09-18 19:30:30 +08:00
|
|
|
|
|
|
|
// Don't test patterns that have already been cached - it won't match.
|
|
|
|
for (unsigned p = 0; p != NumOriginalPatterns; ++p)
|
|
|
|
if (!MatchedPatterns[p])
|
|
|
|
Matches[p] = (Predicates == PatternsToMatch[p].getPredicates());
|
|
|
|
|
|
|
|
// Copy this to all the matching patterns.
|
|
|
|
for (int p = Matches.find_first(); p != -1; p = Matches.find_next(p))
|
2018-09-18 20:01:25 +08:00
|
|
|
if (p != (int)i) {
|
2018-09-19 20:23:50 +08:00
|
|
|
MatchedPatterns.set(p);
|
2018-09-18 19:30:30 +08:00
|
|
|
MatchedPredicates[p] = Matches;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto it : PatternsWithVariants) {
|
|
|
|
unsigned i = it.first;
|
|
|
|
const MultipleUseVarSet &DepVars = it.second.first;
|
|
|
|
const std::vector<TreePatternNodePtr> &Variants = it.second.second;
|
2018-08-28 23:42:08 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
for (unsigned v = 0, e = Variants.size(); v != e; ++v) {
|
2018-06-15 04:32:58 +08:00
|
|
|
TreePatternNodePtr Variant = Variants[v];
|
2018-09-18 19:30:30 +08:00
|
|
|
BitVector &Matches = MatchedPredicates[i];
|
2008-01-06 06:25:12 +08:00
|
|
|
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(errs() << " VAR#" << v << ": "; Variant->dump();
|
|
|
|
errs() << "\n");
|
2010-12-22 00:16:00 +08:00
|
|
|
|
2008-01-06 06:25:12 +08:00
|
|
|
// Scan to see if an instruction or explicit pattern already matches this.
|
|
|
|
bool AlreadyExists = false;
|
2015-11-23 06:43:40 +08:00
|
|
|
for (unsigned p = 0, e = PatternsToMatch.size(); p != e; ++p) {
|
2009-06-26 13:59:16 +08:00
|
|
|
// Skip if the top level predicates do not match.
|
2018-09-18 19:30:30 +08:00
|
|
|
if (!Matches[p])
|
2009-06-26 13:59:16 +08:00
|
|
|
continue;
|
2008-01-06 06:25:12 +08:00
|
|
|
// Check to see if this variant already exists.
|
2018-06-15 04:32:58 +08:00
|
|
|
if (Variant->isIsomorphicTo(PatternsToMatch[p].getSrcPattern(),
|
2015-11-23 06:43:40 +08:00
|
|
|
DepVars)) {
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(errs() << " *** ALREADY EXISTS, ignoring variant.\n");
|
2008-01-06 06:25:12 +08:00
|
|
|
AlreadyExists = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// If we already have it, ignore the variant.
|
|
|
|
if (AlreadyExists) continue;
|
|
|
|
|
|
|
|
// Otherwise, add it to the list of patterns we have.
|
2017-06-27 15:10:20 +08:00
|
|
|
PatternsToMatch.push_back(PatternToMatch(
|
2015-11-23 06:43:40 +08:00
|
|
|
PatternsToMatch[i].getSrcRecord(), PatternsToMatch[i].getPredicates(),
|
2018-05-31 05:00:18 +08:00
|
|
|
Variant, PatternsToMatch[i].getDstPatternShared(),
|
2015-11-23 06:43:40 +08:00
|
|
|
PatternsToMatch[i].getDstRegs(),
|
2017-06-27 15:10:20 +08:00
|
|
|
PatternsToMatch[i].getAddedComplexity(), Record::getNewUID()));
|
2018-09-18 19:30:30 +08:00
|
|
|
MatchedPredicates.push_back(Matches);
|
|
|
|
|
2018-09-18 22:05:07 +08:00
|
|
|
// Add a new match the same as this pattern.
|
|
|
|
for (auto &P : MatchedPredicates)
|
2018-09-19 19:18:49 +08:00
|
|
|
P.push_back(P[i]);
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
|
2018-05-14 20:53:11 +08:00
|
|
|
LLVM_DEBUG(errs() << "\n");
|
2008-01-06 06:25:12 +08:00
|
|
|
}
|
|
|
|
}
|