2019-01-06 00:11:29 +08:00
|
|
|
//===- Predicate.cpp - Predicate class ------------------------------------===//
|
|
|
|
//
|
2020-01-26 11:58:30 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
2019-12-24 01:35:36 +08:00
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2019-01-06 00:11:29 +08:00
|
|
|
//
|
2019-12-24 01:35:36 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2019-01-06 00:11:29 +08:00
|
|
|
//
|
|
|
|
// Wrapper around predicates defined in TableGen.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "mlir/TableGen/Predicate.h"
|
2019-01-11 23:41:12 +08:00
|
|
|
#include "llvm/ADT/SetVector.h"
|
2019-01-17 04:36:10 +08:00
|
|
|
#include "llvm/ADT/SmallPtrSet.h"
|
2019-01-06 00:11:29 +08:00
|
|
|
#include "llvm/ADT/StringExtras.h"
|
2022-01-20 19:55:14 +08:00
|
|
|
#include "llvm/ADT/StringSwitch.h"
|
2019-01-06 00:11:29 +08:00
|
|
|
#include "llvm/Support/FormatVariadic.h"
|
|
|
|
#include "llvm/TableGen/Error.h"
|
|
|
|
#include "llvm/TableGen/Record.h"
|
|
|
|
|
|
|
|
using namespace mlir;
|
2020-08-12 08:47:07 +08:00
|
|
|
using namespace tblgen;
|
2019-01-06 00:11:29 +08:00
|
|
|
|
2019-01-16 02:42:21 +08:00
|
|
|
// Construct a Predicate from a record.
|
2020-08-12 08:47:07 +08:00
|
|
|
Pred::Pred(const llvm::Record *record) : def(record) {
|
2019-01-16 02:42:21 +08:00
|
|
|
assert(def->isSubClassOf("Pred") &&
|
|
|
|
"must be a subclass of TableGen 'Pred' class");
|
2019-01-09 09:19:22 +08:00
|
|
|
}
|
|
|
|
|
2019-01-16 02:42:21 +08:00
|
|
|
// Construct a Predicate from an initializer.
|
2022-01-14 09:35:30 +08:00
|
|
|
Pred::Pred(const llvm::Init *init) {
|
2019-01-16 02:42:21 +08:00
|
|
|
if (const auto *defInit = dyn_cast_or_null<llvm::DefInit>(init))
|
|
|
|
def = defInit->getDef();
|
2019-01-09 09:19:22 +08:00
|
|
|
}
|
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
std::string Pred::getCondition() const {
|
2019-01-17 04:36:10 +08:00
|
|
|
// Static dispatch to subclasses.
|
|
|
|
if (def->isSubClassOf("CombinedPred"))
|
|
|
|
return static_cast<const CombinedPred *>(this)->getConditionImpl();
|
|
|
|
if (def->isSubClassOf("CPred"))
|
|
|
|
return static_cast<const CPred *>(this)->getConditionImpl();
|
|
|
|
llvm_unreachable("Pred::getCondition must be overridden in subclasses");
|
|
|
|
}
|
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
bool Pred::isCombined() const {
|
2019-01-17 04:36:10 +08:00
|
|
|
return def && def->isSubClassOf("CombinedPred");
|
|
|
|
}
|
|
|
|
|
2022-01-27 07:49:53 +08:00
|
|
|
ArrayRef<SMLoc> Pred::getLoc() const { return def->getLoc(); }
|
2019-01-17 04:36:10 +08:00
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
CPred::CPred(const llvm::Record *record) : Pred(record) {
|
2019-01-17 04:36:10 +08:00
|
|
|
assert(def->isSubClassOf("CPred") &&
|
|
|
|
"must be a subclass of Tablegen 'CPred' class");
|
|
|
|
}
|
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
CPred::CPred(const llvm::Init *init) : Pred(init) {
|
2019-01-17 04:36:10 +08:00
|
|
|
assert((!def || def->isSubClassOf("CPred")) &&
|
|
|
|
"must be a subclass of Tablegen 'CPred' class");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get condition of the C Predicate.
|
2020-08-12 08:47:07 +08:00
|
|
|
std::string CPred::getConditionImpl() const {
|
2019-01-16 02:42:21 +08:00
|
|
|
assert(!isNull() && "null predicate does not have a condition");
|
2020-01-29 03:23:46 +08:00
|
|
|
return std::string(def->getValueAsString("predExpr"));
|
2019-01-06 00:11:29 +08:00
|
|
|
}
|
2019-01-17 04:36:10 +08:00
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
CombinedPred::CombinedPred(const llvm::Record *record) : Pred(record) {
|
2019-01-17 04:36:10 +08:00
|
|
|
assert(def->isSubClassOf("CombinedPred") &&
|
|
|
|
"must be a subclass of Tablegen 'CombinedPred' class");
|
|
|
|
}
|
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
CombinedPred::CombinedPred(const llvm::Init *init) : Pred(init) {
|
2019-01-17 04:36:10 +08:00
|
|
|
assert((!def || def->isSubClassOf("CombinedPred")) &&
|
|
|
|
"must be a subclass of Tablegen 'CombinedPred' class");
|
|
|
|
}
|
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
const llvm::Record *CombinedPred::getCombinerDef() const {
|
2019-01-17 04:36:10 +08:00
|
|
|
assert(def->getValue("kind") && "CombinedPred must have a value 'kind'");
|
|
|
|
return def->getValueAsDef("kind");
|
|
|
|
}
|
|
|
|
|
2022-01-02 09:50:43 +08:00
|
|
|
std::vector<llvm::Record *> CombinedPred::getChildren() const {
|
2019-01-17 04:36:10 +08:00
|
|
|
assert(def->getValue("children") &&
|
|
|
|
"CombinedPred must have a value 'children'");
|
|
|
|
return def->getValueAsListOfDefs("children");
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
// Kinds of nodes in a logical predicate tree.
|
|
|
|
enum class PredCombinerKind {
|
|
|
|
Leaf,
|
|
|
|
And,
|
|
|
|
Or,
|
|
|
|
Not,
|
|
|
|
SubstLeaves,
|
2019-04-05 22:22:28 +08:00
|
|
|
Concat,
|
2019-01-17 04:36:10 +08:00
|
|
|
// Special kinds that are used in simplification.
|
|
|
|
False,
|
|
|
|
True
|
|
|
|
};
|
|
|
|
|
|
|
|
// A node in a logical predicate tree.
|
|
|
|
struct PredNode {
|
|
|
|
PredCombinerKind kind;
|
2020-08-12 08:47:07 +08:00
|
|
|
const Pred *predicate;
|
2019-01-17 04:36:10 +08:00
|
|
|
SmallVector<PredNode *, 4> children;
|
|
|
|
std::string expr;
|
2019-04-05 22:22:28 +08:00
|
|
|
|
|
|
|
// Prefix and suffix are used by ConcatPred.
|
|
|
|
std::string prefix;
|
|
|
|
std::string suffix;
|
2019-01-17 04:36:10 +08:00
|
|
|
};
|
2021-12-08 02:27:58 +08:00
|
|
|
} // namespace
|
2019-01-17 04:36:10 +08:00
|
|
|
|
|
|
|
// Get a predicate tree node kind based on the kind used in the predicate
|
|
|
|
// TableGen record.
|
2020-08-12 08:47:07 +08:00
|
|
|
static PredCombinerKind getPredCombinerKind(const Pred &pred) {
|
2019-01-17 04:36:10 +08:00
|
|
|
if (!pred.isCombined())
|
|
|
|
return PredCombinerKind::Leaf;
|
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
const auto &combinedPred = static_cast<const CombinedPred &>(pred);
|
2020-10-07 22:17:35 +08:00
|
|
|
return StringSwitch<PredCombinerKind>(
|
2019-01-17 04:36:10 +08:00
|
|
|
combinedPred.getCombinerDef()->getName())
|
|
|
|
.Case("PredCombinerAnd", PredCombinerKind::And)
|
|
|
|
.Case("PredCombinerOr", PredCombinerKind::Or)
|
|
|
|
.Case("PredCombinerNot", PredCombinerKind::Not)
|
2019-04-05 22:22:28 +08:00
|
|
|
.Case("PredCombinerSubstLeaves", PredCombinerKind::SubstLeaves)
|
|
|
|
.Case("PredCombinerConcat", PredCombinerKind::Concat);
|
2019-01-17 04:36:10 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
// Substitution<pattern, replacement>.
|
|
|
|
using Subst = std::pair<StringRef, StringRef>;
|
2021-12-08 02:27:58 +08:00
|
|
|
} // namespace
|
2019-01-17 04:36:10 +08:00
|
|
|
|
2021-03-17 04:11:50 +08:00
|
|
|
/// Perform the given substitutions on 'str' in-place.
|
|
|
|
static void performSubstitutions(std::string &str,
|
|
|
|
ArrayRef<Subst> substitutions) {
|
|
|
|
// Apply all parent substitutions from innermost to outermost.
|
|
|
|
for (const auto &subst : llvm::reverse(substitutions)) {
|
|
|
|
auto pos = str.find(std::string(subst.first));
|
|
|
|
while (pos != std::string::npos) {
|
|
|
|
str.replace(pos, subst.first.size(), std::string(subst.second));
|
|
|
|
// Skip the newly inserted substring, which itself may consider the
|
|
|
|
// pattern to match.
|
|
|
|
pos += subst.second.size();
|
|
|
|
// Find the next possible match position.
|
|
|
|
pos = str.find(std::string(subst.first), pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-17 04:36:10 +08:00
|
|
|
// Build the predicate tree starting from the top-level predicate, which may
|
|
|
|
// have children, and perform leaf substitutions inplace. Note that after
|
|
|
|
// substitution, nodes are still pointing to the original TableGen record.
|
|
|
|
// All nodes are created within "allocator".
|
2020-02-07 01:03:15 +08:00
|
|
|
static PredNode *
|
2020-08-12 08:47:07 +08:00
|
|
|
buildPredicateTree(const Pred &root,
|
2020-02-07 01:03:15 +08:00
|
|
|
llvm::SpecificBumpPtrAllocator<PredNode> &allocator,
|
|
|
|
ArrayRef<Subst> substitutions) {
|
|
|
|
auto *rootNode = allocator.Allocate();
|
2019-01-17 04:36:10 +08:00
|
|
|
new (rootNode) PredNode;
|
|
|
|
rootNode->kind = getPredCombinerKind(root);
|
|
|
|
rootNode->predicate = &root;
|
|
|
|
if (!root.isCombined()) {
|
|
|
|
rootNode->expr = root.getCondition();
|
2021-03-17 04:11:50 +08:00
|
|
|
performSubstitutions(rootNode->expr, substitutions);
|
2019-01-17 04:36:10 +08:00
|
|
|
return rootNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the current combined predicate is a leaf substitution, append it to the
|
2019-10-20 15:11:03 +08:00
|
|
|
// list before continuing.
|
2019-01-17 04:36:10 +08:00
|
|
|
auto allSubstitutions = llvm::to_vector<4>(substitutions);
|
|
|
|
if (rootNode->kind == PredCombinerKind::SubstLeaves) {
|
2020-08-12 08:47:07 +08:00
|
|
|
const auto &substPred = static_cast<const SubstLeavesPred &>(root);
|
2019-01-17 04:36:10 +08:00
|
|
|
allSubstitutions.push_back(
|
|
|
|
{substPred.getPattern(), substPred.getReplacement()});
|
2021-03-17 04:11:50 +08:00
|
|
|
|
|
|
|
// If the current predicate is a ConcatPred, record the prefix and suffix.
|
|
|
|
} else if (rootNode->kind == PredCombinerKind::Concat) {
|
2020-08-12 08:47:07 +08:00
|
|
|
const auto &concatPred = static_cast<const ConcatPred &>(root);
|
2020-01-29 03:23:46 +08:00
|
|
|
rootNode->prefix = std::string(concatPred.getPrefix());
|
2021-03-17 04:11:50 +08:00
|
|
|
performSubstitutions(rootNode->prefix, substitutions);
|
2020-01-29 03:23:46 +08:00
|
|
|
rootNode->suffix = std::string(concatPred.getSuffix());
|
2021-03-17 04:11:50 +08:00
|
|
|
performSubstitutions(rootNode->suffix, substitutions);
|
2019-04-05 22:22:28 +08:00
|
|
|
}
|
2019-01-17 04:36:10 +08:00
|
|
|
|
|
|
|
// Build child subtrees.
|
2020-08-12 08:47:07 +08:00
|
|
|
auto combined = static_cast<const CombinedPred &>(root);
|
2019-01-17 04:36:10 +08:00
|
|
|
for (const auto *record : combined.getChildren()) {
|
2021-12-21 03:45:05 +08:00
|
|
|
auto *childTree =
|
2020-08-12 08:47:07 +08:00
|
|
|
buildPredicateTree(Pred(record), allocator, allSubstitutions);
|
2019-01-17 04:36:10 +08:00
|
|
|
rootNode->children.push_back(childTree);
|
|
|
|
}
|
|
|
|
return rootNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Simplify a predicate tree rooted at "node" using the predicates that are
|
|
|
|
// known to be true(false). For AND(OR) combined predicates, if any of the
|
|
|
|
// children is known to be false(true), the result is also false(true).
|
|
|
|
// Furthermore, for AND(OR) combined predicates, children that are known to be
|
|
|
|
// true(false) don't have to be checked dynamically.
|
2020-08-12 08:47:07 +08:00
|
|
|
static PredNode *
|
|
|
|
propagateGroundTruth(PredNode *node,
|
|
|
|
const llvm::SmallPtrSetImpl<Pred *> &knownTruePreds,
|
|
|
|
const llvm::SmallPtrSetImpl<Pred *> &knownFalsePreds) {
|
2019-01-17 04:36:10 +08:00
|
|
|
// If the current predicate is known to be true or false, change the kind of
|
|
|
|
// the node and return immediately.
|
|
|
|
if (knownTruePreds.count(node->predicate) != 0) {
|
|
|
|
node->kind = PredCombinerKind::True;
|
|
|
|
node->children.clear();
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
if (knownFalsePreds.count(node->predicate) != 0) {
|
|
|
|
node->kind = PredCombinerKind::False;
|
|
|
|
node->children.clear();
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the current node is a substitution, stop recursion now.
|
|
|
|
// The expressions in the leaves below this node were rewritten, but the nodes
|
|
|
|
// still point to the original predicate records. While the original
|
|
|
|
// predicate may be known to be true or false, it is not necessarily the case
|
|
|
|
// after rewriting.
|
2020-07-07 16:35:23 +08:00
|
|
|
// TODO: we can support ground truth for rewritten
|
2019-01-17 04:36:10 +08:00
|
|
|
// predicates by either (a) having our own unique'ing of the predicates
|
|
|
|
// instead of relying on TableGen record pointers or (b) taking ground truth
|
2019-10-20 15:11:03 +08:00
|
|
|
// values optionally prefixed with a list of substitutions to apply, e.g.
|
2019-01-17 04:36:10 +08:00
|
|
|
// "predX is true by itself as well as predSubY leaf substitution had been
|
|
|
|
// applied to it".
|
|
|
|
if (node->kind == PredCombinerKind::SubstLeaves) {
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, look at child nodes.
|
|
|
|
|
|
|
|
// Move child nodes into some local variable so that they can be optimized
|
|
|
|
// separately and re-added if necessary.
|
|
|
|
llvm::SmallVector<PredNode *, 4> children;
|
|
|
|
std::swap(node->children, children);
|
|
|
|
|
|
|
|
for (auto &child : children) {
|
|
|
|
// First, simplify the child. This maintains the predicate as it was.
|
2021-12-21 03:45:05 +08:00
|
|
|
auto *simplifiedChild =
|
2019-01-17 04:36:10 +08:00
|
|
|
propagateGroundTruth(child, knownTruePreds, knownFalsePreds);
|
|
|
|
|
|
|
|
// Just add the child if we don't know how to simplify the current node.
|
|
|
|
if (node->kind != PredCombinerKind::And &&
|
|
|
|
node->kind != PredCombinerKind::Or) {
|
|
|
|
node->children.push_back(simplifiedChild);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Second, based on the type define which known values of child predicates
|
|
|
|
// immediately collapse this predicate to a known value, and which others
|
|
|
|
// may be safely ignored.
|
|
|
|
// OR(..., True, ...) = True
|
|
|
|
// OR(..., False, ...) = OR(..., ...)
|
|
|
|
// AND(..., False, ...) = False
|
|
|
|
// AND(..., True, ...) = AND(..., ...)
|
|
|
|
auto collapseKind = node->kind == PredCombinerKind::And
|
|
|
|
? PredCombinerKind::False
|
|
|
|
: PredCombinerKind::True;
|
|
|
|
auto eraseKind = node->kind == PredCombinerKind::And
|
|
|
|
? PredCombinerKind::True
|
|
|
|
: PredCombinerKind::False;
|
|
|
|
const auto &collapseList =
|
|
|
|
node->kind == PredCombinerKind::And ? knownFalsePreds : knownTruePreds;
|
|
|
|
const auto &eraseList =
|
|
|
|
node->kind == PredCombinerKind::And ? knownTruePreds : knownFalsePreds;
|
|
|
|
if (simplifiedChild->kind == collapseKind ||
|
|
|
|
collapseList.count(simplifiedChild->predicate) != 0) {
|
|
|
|
node->kind = collapseKind;
|
|
|
|
node->children.clear();
|
|
|
|
return node;
|
2021-12-21 03:45:05 +08:00
|
|
|
}
|
|
|
|
if (simplifiedChild->kind == eraseKind ||
|
|
|
|
eraseList.count(simplifiedChild->predicate) != 0) {
|
2019-01-17 04:36:10 +08:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
node->children.push_back(simplifiedChild);
|
|
|
|
}
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Combine a list of predicate expressions using a binary combiner. If a list
|
|
|
|
// is empty, return "init".
|
|
|
|
static std::string combineBinary(ArrayRef<std::string> children,
|
2022-01-02 09:26:44 +08:00
|
|
|
const std::string &combiner,
|
|
|
|
std::string init) {
|
2019-01-17 04:36:10 +08:00
|
|
|
if (children.empty())
|
|
|
|
return init;
|
|
|
|
|
|
|
|
auto size = children.size();
|
|
|
|
if (size == 1)
|
|
|
|
return children.front();
|
|
|
|
|
|
|
|
std::string str;
|
|
|
|
llvm::raw_string_ostream os(str);
|
|
|
|
os << '(' << children.front() << ')';
|
|
|
|
for (unsigned i = 1; i < size; ++i) {
|
|
|
|
os << ' ' << combiner << " (" << children[i] << ')';
|
|
|
|
}
|
|
|
|
return os.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Prepend negation to the only condition in the predicate expression list.
|
|
|
|
static std::string combineNot(ArrayRef<std::string> children) {
|
|
|
|
assert(children.size() == 1 && "expected exactly one child predicate of Neg");
|
|
|
|
return (Twine("!(") + children.front() + Twine(')')).str();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Recursively traverse the predicate tree in depth-first post-order and build
|
|
|
|
// the final expression.
|
|
|
|
static std::string getCombinedCondition(const PredNode &root) {
|
|
|
|
// Immediately return for non-combiner predicates that don't have children.
|
|
|
|
if (root.kind == PredCombinerKind::Leaf)
|
|
|
|
return root.expr;
|
|
|
|
if (root.kind == PredCombinerKind::True)
|
|
|
|
return "true";
|
|
|
|
if (root.kind == PredCombinerKind::False)
|
|
|
|
return "false";
|
|
|
|
|
|
|
|
// Recurse into children.
|
|
|
|
llvm::SmallVector<std::string, 4> childExpressions;
|
|
|
|
childExpressions.reserve(root.children.size());
|
|
|
|
for (const auto &child : root.children)
|
|
|
|
childExpressions.push_back(getCombinedCondition(*child));
|
|
|
|
|
|
|
|
// Combine the expressions based on the predicate node kind.
|
|
|
|
if (root.kind == PredCombinerKind::And)
|
|
|
|
return combineBinary(childExpressions, "&&", "true");
|
|
|
|
if (root.kind == PredCombinerKind::Or)
|
|
|
|
return combineBinary(childExpressions, "||", "false");
|
|
|
|
if (root.kind == PredCombinerKind::Not)
|
|
|
|
return combineNot(childExpressions);
|
2019-04-05 22:22:28 +08:00
|
|
|
if (root.kind == PredCombinerKind::Concat) {
|
|
|
|
assert(childExpressions.size() == 1 &&
|
|
|
|
"ConcatPred should only have one child");
|
|
|
|
return root.prefix + childExpressions.front() + root.suffix;
|
|
|
|
}
|
2019-01-17 04:36:10 +08:00
|
|
|
|
|
|
|
// Substitutions were applied before so just ignore them.
|
|
|
|
if (root.kind == PredCombinerKind::SubstLeaves) {
|
|
|
|
assert(childExpressions.size() == 1 &&
|
|
|
|
"substitution predicate must have one child");
|
|
|
|
return childExpressions[0];
|
|
|
|
}
|
|
|
|
|
|
|
|
llvm::PrintFatalError(root.predicate->getLoc(), "unsupported predicate kind");
|
|
|
|
}
|
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
std::string CombinedPred::getConditionImpl() const {
|
2020-02-07 01:03:15 +08:00
|
|
|
llvm::SpecificBumpPtrAllocator<PredNode> allocator;
|
2021-12-21 03:45:05 +08:00
|
|
|
auto *predicateTree = buildPredicateTree(*this, allocator, {});
|
2020-08-12 08:47:07 +08:00
|
|
|
predicateTree =
|
|
|
|
propagateGroundTruth(predicateTree,
|
|
|
|
/*knownTruePreds=*/llvm::SmallPtrSet<Pred *, 2>(),
|
|
|
|
/*knownFalsePreds=*/llvm::SmallPtrSet<Pred *, 2>());
|
2019-01-17 04:36:10 +08:00
|
|
|
|
|
|
|
return getCombinedCondition(*predicateTree);
|
|
|
|
}
|
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
StringRef SubstLeavesPred::getPattern() const {
|
2019-01-17 04:36:10 +08:00
|
|
|
return def->getValueAsString("pattern");
|
|
|
|
}
|
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
StringRef SubstLeavesPred::getReplacement() const {
|
2019-01-17 04:36:10 +08:00
|
|
|
return def->getValueAsString("replacement");
|
|
|
|
}
|
2019-04-05 22:22:28 +08:00
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
StringRef ConcatPred::getPrefix() const {
|
2019-04-05 22:22:28 +08:00
|
|
|
return def->getValueAsString("prefix");
|
|
|
|
}
|
|
|
|
|
2020-08-12 08:47:07 +08:00
|
|
|
StringRef ConcatPred::getSuffix() const {
|
2019-04-05 22:22:28 +08:00
|
|
|
return def->getValueAsString("suffix");
|
|
|
|
}
|