[mlir][PDL] Add support for PDL bytecode and expose PDL support to OwningRewritePatternList
PDL patterns are now supported via a new `PDLPatternModule` class. This class contains a ModuleOp with the pdl::PatternOp operations representing the patterns, as well as a collection of registered C++ functions for native constraints/creations/rewrites/etc. that may be invoked via the pdl patterns. Instances of this class are added to an OwningRewritePatternList in the same fashion as C++ RewritePatterns, i.e. via the `insert` method.
The PDL bytecode is an in-memory representation of the PDL interpreter dialect that can be efficiently interpreted/executed. The representation of the bytecode boils down to a code array(for opcodes/memory locations/etc) and a memory buffer(for storing attributes/operations/values/any other data necessary). The bytecode operations are effectively a 1-1 mapping to the PDLInterp dialect operations, with a few exceptions in cases where the in-memory representation of the bytecode can be more efficient than the MLIR representation. For example, a generic `AreEqual` bytecode op can be used to represent AreEqualOp, CheckAttributeOp, and CheckTypeOp.
The execution of the bytecode is split into two phases: matching and rewriting. When matching, all of the matched patterns are collected to avoid the overhead of re-running parts of the matcher. These matched patterns are then considered alongside the native C++ patterns, which rewrite immediately in-place via `RewritePattern::matchAndRewrite`, for the given root operation. When a PDL pattern is matched and has the highest benefit, it is passed back to the bytecode to execute its rewriter.
Differential Revision: https://reviews.llvm.org/D89107
2020-12-02 06:30:18 +08:00
|
|
|
//===- ByteCode.h - Pattern byte-code interpreter ---------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file declares a byte-code and interpreter for pattern rewrites in MLIR.
|
|
|
|
// The byte-code is constructed from the PDL Interpreter dialect.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef MLIR_REWRITE_BYTECODE_H_
|
|
|
|
#define MLIR_REWRITE_BYTECODE_H_
|
|
|
|
|
|
|
|
#include "mlir/IR/PatternMatch.h"
|
|
|
|
|
|
|
|
namespace mlir {
|
|
|
|
namespace pdl_interp {
|
|
|
|
class RecordMatchOp;
|
|
|
|
} // end namespace pdl_interp
|
|
|
|
|
|
|
|
namespace detail {
|
|
|
|
class PDLByteCode;
|
|
|
|
|
|
|
|
/// Use generic bytecode types. ByteCodeField refers to the actual bytecode
|
[mlir][PDL] Add support for variadic operands and results in the PDL byte code
Supporting ranges in the byte code requires additional complexity, given that a range can't be easily representable as an opaque void *, as is possible with the existing bytecode value types (Attribute, Type, Value, etc.). To enable representing a range with void *, an auxillary storage is used for the actual range itself, with the pointer being passed around in the normal byte code memory. For type ranges, a TypeRange is stored. For value ranges, a ValueRange is stored. The above problem represents a majority of the complexity involved in this revision, the rest is adapting/adding byte code operations to support the changes made to the PDL interpreter in the parent revision.
After this revision, PDL will have initial end-to-end support for variadic operands/results.
Differential Revision: https://reviews.llvm.org/D95723
2021-03-17 04:12:01 +08:00
|
|
|
/// entries. ByteCodeAddr refers to size of indices into the bytecode.
|
[mlir][PDL] Add support for PDL bytecode and expose PDL support to OwningRewritePatternList
PDL patterns are now supported via a new `PDLPatternModule` class. This class contains a ModuleOp with the pdl::PatternOp operations representing the patterns, as well as a collection of registered C++ functions for native constraints/creations/rewrites/etc. that may be invoked via the pdl patterns. Instances of this class are added to an OwningRewritePatternList in the same fashion as C++ RewritePatterns, i.e. via the `insert` method.
The PDL bytecode is an in-memory representation of the PDL interpreter dialect that can be efficiently interpreted/executed. The representation of the bytecode boils down to a code array(for opcodes/memory locations/etc) and a memory buffer(for storing attributes/operations/values/any other data necessary). The bytecode operations are effectively a 1-1 mapping to the PDLInterp dialect operations, with a few exceptions in cases where the in-memory representation of the bytecode can be more efficient than the MLIR representation. For example, a generic `AreEqual` bytecode op can be used to represent AreEqualOp, CheckAttributeOp, and CheckTypeOp.
The execution of the bytecode is split into two phases: matching and rewriting. When matching, all of the matched patterns are collected to avoid the overhead of re-running parts of the matcher. These matched patterns are then considered alongside the native C++ patterns, which rewrite immediately in-place via `RewritePattern::matchAndRewrite`, for the given root operation. When a PDL pattern is matched and has the highest benefit, it is passed back to the bytecode to execute its rewriter.
Differential Revision: https://reviews.llvm.org/D89107
2020-12-02 06:30:18 +08:00
|
|
|
using ByteCodeField = uint16_t;
|
|
|
|
using ByteCodeAddr = uint32_t;
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// PDLByteCodePattern
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
/// All of the data pertaining to a specific pattern within the bytecode.
|
|
|
|
class PDLByteCodePattern : public Pattern {
|
|
|
|
public:
|
|
|
|
static PDLByteCodePattern create(pdl_interp::RecordMatchOp matchOp,
|
|
|
|
ByteCodeAddr rewriterAddr);
|
|
|
|
|
|
|
|
/// Return the bytecode address of the rewriter for this pattern.
|
|
|
|
ByteCodeAddr getRewriterAddr() const { return rewriterAddr; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
template <typename... Args>
|
|
|
|
PDLByteCodePattern(ByteCodeAddr rewriterAddr, Args &&...patternArgs)
|
|
|
|
: Pattern(std::forward<Args>(patternArgs)...),
|
|
|
|
rewriterAddr(rewriterAddr) {}
|
|
|
|
|
|
|
|
/// The address of the rewriter for this pattern.
|
|
|
|
ByteCodeAddr rewriterAddr;
|
|
|
|
};
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// PDLByteCodeMutableState
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
/// This class contains the mutable state of a bytecode instance. This allows
|
|
|
|
/// for a bytecode instance to be cached and reused across various different
|
|
|
|
/// threads/drivers.
|
|
|
|
class PDLByteCodeMutableState {
|
|
|
|
public:
|
|
|
|
/// Set the new benefit for a bytecode pattern. The `patternIndex` corresponds
|
|
|
|
/// to the position of the pattern within the range returned by
|
|
|
|
/// `PDLByteCode::getPatterns`.
|
|
|
|
void updatePatternBenefit(unsigned patternIndex, PatternBenefit benefit);
|
|
|
|
|
[mlir][PDL] Add support for variadic operands and results in the PDL byte code
Supporting ranges in the byte code requires additional complexity, given that a range can't be easily representable as an opaque void *, as is possible with the existing bytecode value types (Attribute, Type, Value, etc.). To enable representing a range with void *, an auxillary storage is used for the actual range itself, with the pointer being passed around in the normal byte code memory. For type ranges, a TypeRange is stored. For value ranges, a ValueRange is stored. The above problem represents a majority of the complexity involved in this revision, the rest is adapting/adding byte code operations to support the changes made to the PDL interpreter in the parent revision.
After this revision, PDL will have initial end-to-end support for variadic operands/results.
Differential Revision: https://reviews.llvm.org/D95723
2021-03-17 04:12:01 +08:00
|
|
|
/// Cleanup any allocated state after a match/rewrite has been completed. This
|
|
|
|
/// method should be called irregardless of whether the match+rewrite was a
|
|
|
|
/// success or not.
|
|
|
|
void cleanupAfterMatchAndRewrite();
|
|
|
|
|
[mlir][PDL] Add support for PDL bytecode and expose PDL support to OwningRewritePatternList
PDL patterns are now supported via a new `PDLPatternModule` class. This class contains a ModuleOp with the pdl::PatternOp operations representing the patterns, as well as a collection of registered C++ functions for native constraints/creations/rewrites/etc. that may be invoked via the pdl patterns. Instances of this class are added to an OwningRewritePatternList in the same fashion as C++ RewritePatterns, i.e. via the `insert` method.
The PDL bytecode is an in-memory representation of the PDL interpreter dialect that can be efficiently interpreted/executed. The representation of the bytecode boils down to a code array(for opcodes/memory locations/etc) and a memory buffer(for storing attributes/operations/values/any other data necessary). The bytecode operations are effectively a 1-1 mapping to the PDLInterp dialect operations, with a few exceptions in cases where the in-memory representation of the bytecode can be more efficient than the MLIR representation. For example, a generic `AreEqual` bytecode op can be used to represent AreEqualOp, CheckAttributeOp, and CheckTypeOp.
The execution of the bytecode is split into two phases: matching and rewriting. When matching, all of the matched patterns are collected to avoid the overhead of re-running parts of the matcher. These matched patterns are then considered alongside the native C++ patterns, which rewrite immediately in-place via `RewritePattern::matchAndRewrite`, for the given root operation. When a PDL pattern is matched and has the highest benefit, it is passed back to the bytecode to execute its rewriter.
Differential Revision: https://reviews.llvm.org/D89107
2020-12-02 06:30:18 +08:00
|
|
|
private:
|
|
|
|
/// Allow access to data fields.
|
|
|
|
friend class PDLByteCode;
|
|
|
|
|
|
|
|
/// The mutable block of memory used during the matching and rewriting phases
|
|
|
|
/// of the bytecode.
|
|
|
|
std::vector<const void *> memory;
|
|
|
|
|
[mlir][PDL] Add support for variadic operands and results in the PDL byte code
Supporting ranges in the byte code requires additional complexity, given that a range can't be easily representable as an opaque void *, as is possible with the existing bytecode value types (Attribute, Type, Value, etc.). To enable representing a range with void *, an auxillary storage is used for the actual range itself, with the pointer being passed around in the normal byte code memory. For type ranges, a TypeRange is stored. For value ranges, a ValueRange is stored. The above problem represents a majority of the complexity involved in this revision, the rest is adapting/adding byte code operations to support the changes made to the PDL interpreter in the parent revision.
After this revision, PDL will have initial end-to-end support for variadic operands/results.
Differential Revision: https://reviews.llvm.org/D95723
2021-03-17 04:12:01 +08:00
|
|
|
/// A mutable block of memory used during the matching and rewriting phase of
|
|
|
|
/// the bytecode to store ranges of types.
|
|
|
|
std::vector<TypeRange> typeRangeMemory;
|
|
|
|
/// A set of type ranges that have been allocated by the byte code interpreter
|
|
|
|
/// to provide a guaranteed lifetime.
|
|
|
|
std::vector<llvm::OwningArrayRef<Type>> allocatedTypeRangeMemory;
|
|
|
|
|
|
|
|
/// A mutable block of memory used during the matching and rewriting phase of
|
|
|
|
/// the bytecode to store ranges of values.
|
|
|
|
std::vector<ValueRange> valueRangeMemory;
|
|
|
|
/// A set of value ranges that have been allocated by the byte code
|
|
|
|
/// interpreter to provide a guaranteed lifetime.
|
|
|
|
std::vector<llvm::OwningArrayRef<Value>> allocatedValueRangeMemory;
|
|
|
|
|
[mlir][PDL] Add support for PDL bytecode and expose PDL support to OwningRewritePatternList
PDL patterns are now supported via a new `PDLPatternModule` class. This class contains a ModuleOp with the pdl::PatternOp operations representing the patterns, as well as a collection of registered C++ functions for native constraints/creations/rewrites/etc. that may be invoked via the pdl patterns. Instances of this class are added to an OwningRewritePatternList in the same fashion as C++ RewritePatterns, i.e. via the `insert` method.
The PDL bytecode is an in-memory representation of the PDL interpreter dialect that can be efficiently interpreted/executed. The representation of the bytecode boils down to a code array(for opcodes/memory locations/etc) and a memory buffer(for storing attributes/operations/values/any other data necessary). The bytecode operations are effectively a 1-1 mapping to the PDLInterp dialect operations, with a few exceptions in cases where the in-memory representation of the bytecode can be more efficient than the MLIR representation. For example, a generic `AreEqual` bytecode op can be used to represent AreEqualOp, CheckAttributeOp, and CheckTypeOp.
The execution of the bytecode is split into two phases: matching and rewriting. When matching, all of the matched patterns are collected to avoid the overhead of re-running parts of the matcher. These matched patterns are then considered alongside the native C++ patterns, which rewrite immediately in-place via `RewritePattern::matchAndRewrite`, for the given root operation. When a PDL pattern is matched and has the highest benefit, it is passed back to the bytecode to execute its rewriter.
Differential Revision: https://reviews.llvm.org/D89107
2020-12-02 06:30:18 +08:00
|
|
|
/// The up-to-date benefits of the patterns held by the bytecode. The order
|
|
|
|
/// of this array corresponds 1-1 with the array of patterns in `PDLByteCode`.
|
|
|
|
std::vector<PatternBenefit> currentPatternBenefits;
|
|
|
|
};
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// PDLByteCode
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
/// The bytecode class is also the interpreter. Contains the bytecode itself,
|
|
|
|
/// the static info, addresses of the rewriter functions, the interpreter
|
|
|
|
/// memory buffer, and the execution context.
|
|
|
|
class PDLByteCode {
|
|
|
|
public:
|
|
|
|
/// Each successful match returns a MatchResult, which contains information
|
|
|
|
/// necessary to execute the rewriter and indicates the originating pattern.
|
|
|
|
struct MatchResult {
|
|
|
|
MatchResult(Location loc, const PDLByteCodePattern &pattern,
|
|
|
|
PatternBenefit benefit)
|
|
|
|
: location(loc), pattern(&pattern), benefit(benefit) {}
|
[mlir][PDL] Add support for variadic operands and results in the PDL byte code
Supporting ranges in the byte code requires additional complexity, given that a range can't be easily representable as an opaque void *, as is possible with the existing bytecode value types (Attribute, Type, Value, etc.). To enable representing a range with void *, an auxillary storage is used for the actual range itself, with the pointer being passed around in the normal byte code memory. For type ranges, a TypeRange is stored. For value ranges, a ValueRange is stored. The above problem represents a majority of the complexity involved in this revision, the rest is adapting/adding byte code operations to support the changes made to the PDL interpreter in the parent revision.
After this revision, PDL will have initial end-to-end support for variadic operands/results.
Differential Revision: https://reviews.llvm.org/D95723
2021-03-17 04:12:01 +08:00
|
|
|
MatchResult(const MatchResult &) = delete;
|
|
|
|
MatchResult &operator=(const MatchResult &) = delete;
|
|
|
|
MatchResult(MatchResult &&other) = default;
|
|
|
|
MatchResult &operator=(MatchResult &&) = default;
|
[mlir][PDL] Add support for PDL bytecode and expose PDL support to OwningRewritePatternList
PDL patterns are now supported via a new `PDLPatternModule` class. This class contains a ModuleOp with the pdl::PatternOp operations representing the patterns, as well as a collection of registered C++ functions for native constraints/creations/rewrites/etc. that may be invoked via the pdl patterns. Instances of this class are added to an OwningRewritePatternList in the same fashion as C++ RewritePatterns, i.e. via the `insert` method.
The PDL bytecode is an in-memory representation of the PDL interpreter dialect that can be efficiently interpreted/executed. The representation of the bytecode boils down to a code array(for opcodes/memory locations/etc) and a memory buffer(for storing attributes/operations/values/any other data necessary). The bytecode operations are effectively a 1-1 mapping to the PDLInterp dialect operations, with a few exceptions in cases where the in-memory representation of the bytecode can be more efficient than the MLIR representation. For example, a generic `AreEqual` bytecode op can be used to represent AreEqualOp, CheckAttributeOp, and CheckTypeOp.
The execution of the bytecode is split into two phases: matching and rewriting. When matching, all of the matched patterns are collected to avoid the overhead of re-running parts of the matcher. These matched patterns are then considered alongside the native C++ patterns, which rewrite immediately in-place via `RewritePattern::matchAndRewrite`, for the given root operation. When a PDL pattern is matched and has the highest benefit, it is passed back to the bytecode to execute its rewriter.
Differential Revision: https://reviews.llvm.org/D89107
2020-12-02 06:30:18 +08:00
|
|
|
|
|
|
|
/// The location of operations to be replaced.
|
|
|
|
Location location;
|
|
|
|
/// Memory values defined in the matcher that are passed to the rewriter.
|
[mlir][PDL] Add support for variadic operands and results in the PDL byte code
Supporting ranges in the byte code requires additional complexity, given that a range can't be easily representable as an opaque void *, as is possible with the existing bytecode value types (Attribute, Type, Value, etc.). To enable representing a range with void *, an auxillary storage is used for the actual range itself, with the pointer being passed around in the normal byte code memory. For type ranges, a TypeRange is stored. For value ranges, a ValueRange is stored. The above problem represents a majority of the complexity involved in this revision, the rest is adapting/adding byte code operations to support the changes made to the PDL interpreter in the parent revision.
After this revision, PDL will have initial end-to-end support for variadic operands/results.
Differential Revision: https://reviews.llvm.org/D95723
2021-03-17 04:12:01 +08:00
|
|
|
SmallVector<const void *> values;
|
|
|
|
/// Memory used for the range input values.
|
|
|
|
SmallVector<TypeRange, 0> typeRangeValues;
|
|
|
|
SmallVector<ValueRange, 0> valueRangeValues;
|
|
|
|
|
[mlir][PDL] Add support for PDL bytecode and expose PDL support to OwningRewritePatternList
PDL patterns are now supported via a new `PDLPatternModule` class. This class contains a ModuleOp with the pdl::PatternOp operations representing the patterns, as well as a collection of registered C++ functions for native constraints/creations/rewrites/etc. that may be invoked via the pdl patterns. Instances of this class are added to an OwningRewritePatternList in the same fashion as C++ RewritePatterns, i.e. via the `insert` method.
The PDL bytecode is an in-memory representation of the PDL interpreter dialect that can be efficiently interpreted/executed. The representation of the bytecode boils down to a code array(for opcodes/memory locations/etc) and a memory buffer(for storing attributes/operations/values/any other data necessary). The bytecode operations are effectively a 1-1 mapping to the PDLInterp dialect operations, with a few exceptions in cases where the in-memory representation of the bytecode can be more efficient than the MLIR representation. For example, a generic `AreEqual` bytecode op can be used to represent AreEqualOp, CheckAttributeOp, and CheckTypeOp.
The execution of the bytecode is split into two phases: matching and rewriting. When matching, all of the matched patterns are collected to avoid the overhead of re-running parts of the matcher. These matched patterns are then considered alongside the native C++ patterns, which rewrite immediately in-place via `RewritePattern::matchAndRewrite`, for the given root operation. When a PDL pattern is matched and has the highest benefit, it is passed back to the bytecode to execute its rewriter.
Differential Revision: https://reviews.llvm.org/D89107
2020-12-02 06:30:18 +08:00
|
|
|
/// The originating pattern that was matched. This is always non-null, but
|
|
|
|
/// represented with a pointer to allow for assignment.
|
|
|
|
const PDLByteCodePattern *pattern;
|
|
|
|
/// The current benefit of the pattern that was matched.
|
|
|
|
PatternBenefit benefit;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Create a ByteCode instance from the given module containing operations in
|
|
|
|
/// the PDL interpreter dialect.
|
|
|
|
PDLByteCode(ModuleOp module,
|
|
|
|
llvm::StringMap<PDLConstraintFunction> constraintFns,
|
|
|
|
llvm::StringMap<PDLRewriteFunction> rewriteFns);
|
|
|
|
|
|
|
|
/// Return the patterns held by the bytecode.
|
|
|
|
ArrayRef<PDLByteCodePattern> getPatterns() const { return patterns; }
|
|
|
|
|
|
|
|
/// Initialize the given state such that it can be used to execute the current
|
|
|
|
/// bytecode.
|
|
|
|
void initializeMutableState(PDLByteCodeMutableState &state) const;
|
|
|
|
|
|
|
|
/// Run the pattern matcher on the given root operation, collecting the
|
|
|
|
/// matched patterns in `matches`.
|
|
|
|
void match(Operation *op, PatternRewriter &rewriter,
|
|
|
|
SmallVectorImpl<MatchResult> &matches,
|
|
|
|
PDLByteCodeMutableState &state) const;
|
|
|
|
|
|
|
|
/// Run the rewriter of the given pattern that was previously matched in
|
|
|
|
/// `match`.
|
|
|
|
void rewrite(PatternRewriter &rewriter, const MatchResult &match,
|
|
|
|
PDLByteCodeMutableState &state) const;
|
|
|
|
|
|
|
|
private:
|
|
|
|
/// Execute the given byte code starting at the provided instruction `inst`.
|
|
|
|
/// `matches` is an optional field provided when this function is executed in
|
|
|
|
/// a matching context.
|
|
|
|
void executeByteCode(const ByteCodeField *inst, PatternRewriter &rewriter,
|
|
|
|
PDLByteCodeMutableState &state,
|
|
|
|
SmallVectorImpl<MatchResult> *matches) const;
|
|
|
|
|
2021-01-08 01:09:48 +08:00
|
|
|
/// A vector containing pointers to uniqued data. The storage is intentionally
|
[mlir][PDL] Add support for PDL bytecode and expose PDL support to OwningRewritePatternList
PDL patterns are now supported via a new `PDLPatternModule` class. This class contains a ModuleOp with the pdl::PatternOp operations representing the patterns, as well as a collection of registered C++ functions for native constraints/creations/rewrites/etc. that may be invoked via the pdl patterns. Instances of this class are added to an OwningRewritePatternList in the same fashion as C++ RewritePatterns, i.e. via the `insert` method.
The PDL bytecode is an in-memory representation of the PDL interpreter dialect that can be efficiently interpreted/executed. The representation of the bytecode boils down to a code array(for opcodes/memory locations/etc) and a memory buffer(for storing attributes/operations/values/any other data necessary). The bytecode operations are effectively a 1-1 mapping to the PDLInterp dialect operations, with a few exceptions in cases where the in-memory representation of the bytecode can be more efficient than the MLIR representation. For example, a generic `AreEqual` bytecode op can be used to represent AreEqualOp, CheckAttributeOp, and CheckTypeOp.
The execution of the bytecode is split into two phases: matching and rewriting. When matching, all of the matched patterns are collected to avoid the overhead of re-running parts of the matcher. These matched patterns are then considered alongside the native C++ patterns, which rewrite immediately in-place via `RewritePattern::matchAndRewrite`, for the given root operation. When a PDL pattern is matched and has the highest benefit, it is passed back to the bytecode to execute its rewriter.
Differential Revision: https://reviews.llvm.org/D89107
2020-12-02 06:30:18 +08:00
|
|
|
/// opaque such that we can store a wide range of data types. The types of
|
|
|
|
/// data stored here include:
|
|
|
|
/// * Attribute, Identifier, OperationName, Type
|
|
|
|
std::vector<const void *> uniquedData;
|
|
|
|
|
|
|
|
/// A vector containing the generated bytecode for the matcher.
|
|
|
|
SmallVector<ByteCodeField, 64> matcherByteCode;
|
|
|
|
|
|
|
|
/// A vector containing the generated bytecode for all of the rewriters.
|
|
|
|
SmallVector<ByteCodeField, 64> rewriterByteCode;
|
|
|
|
|
|
|
|
/// The set of patterns contained within the bytecode.
|
|
|
|
SmallVector<PDLByteCodePattern, 32> patterns;
|
|
|
|
|
|
|
|
/// A set of user defined functions invoked via PDL.
|
|
|
|
std::vector<PDLConstraintFunction> constraintFunctions;
|
|
|
|
std::vector<PDLRewriteFunction> rewriteFunctions;
|
|
|
|
|
|
|
|
/// The maximum memory index used by a value.
|
|
|
|
ByteCodeField maxValueMemoryIndex = 0;
|
[mlir][PDL] Add support for variadic operands and results in the PDL byte code
Supporting ranges in the byte code requires additional complexity, given that a range can't be easily representable as an opaque void *, as is possible with the existing bytecode value types (Attribute, Type, Value, etc.). To enable representing a range with void *, an auxillary storage is used for the actual range itself, with the pointer being passed around in the normal byte code memory. For type ranges, a TypeRange is stored. For value ranges, a ValueRange is stored. The above problem represents a majority of the complexity involved in this revision, the rest is adapting/adding byte code operations to support the changes made to the PDL interpreter in the parent revision.
After this revision, PDL will have initial end-to-end support for variadic operands/results.
Differential Revision: https://reviews.llvm.org/D95723
2021-03-17 04:12:01 +08:00
|
|
|
|
|
|
|
/// The maximum number of different types of ranges.
|
|
|
|
ByteCodeField maxTypeRangeCount = 0;
|
|
|
|
ByteCodeField maxValueRangeCount = 0;
|
[mlir][PDL] Add support for PDL bytecode and expose PDL support to OwningRewritePatternList
PDL patterns are now supported via a new `PDLPatternModule` class. This class contains a ModuleOp with the pdl::PatternOp operations representing the patterns, as well as a collection of registered C++ functions for native constraints/creations/rewrites/etc. that may be invoked via the pdl patterns. Instances of this class are added to an OwningRewritePatternList in the same fashion as C++ RewritePatterns, i.e. via the `insert` method.
The PDL bytecode is an in-memory representation of the PDL interpreter dialect that can be efficiently interpreted/executed. The representation of the bytecode boils down to a code array(for opcodes/memory locations/etc) and a memory buffer(for storing attributes/operations/values/any other data necessary). The bytecode operations are effectively a 1-1 mapping to the PDLInterp dialect operations, with a few exceptions in cases where the in-memory representation of the bytecode can be more efficient than the MLIR representation. For example, a generic `AreEqual` bytecode op can be used to represent AreEqualOp, CheckAttributeOp, and CheckTypeOp.
The execution of the bytecode is split into two phases: matching and rewriting. When matching, all of the matched patterns are collected to avoid the overhead of re-running parts of the matcher. These matched patterns are then considered alongside the native C++ patterns, which rewrite immediately in-place via `RewritePattern::matchAndRewrite`, for the given root operation. When a PDL pattern is matched and has the highest benefit, it is passed back to the bytecode to execute its rewriter.
Differential Revision: https://reviews.llvm.org/D89107
2020-12-02 06:30:18 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
} // end namespace detail
|
|
|
|
} // end namespace mlir
|
|
|
|
|
|
|
|
#endif // MLIR_REWRITE_BYTECODE_H_
|