tvm
stmt_functor.h
Go to the documentation of this file.
1 /*
2  * Licensed to the Apache Software Foundation (ASF) under one
3  * or more contributor license agreements. See the NOTICE file
4  * distributed with this work for additional information
5  * regarding copyright ownership. The ASF licenses this file
6  * to you under the Apache License, Version 2.0 (the
7  * "License"); you may not use this file except in compliance
8  * with the License. You may obtain a copy of the License at
9  *
10  * http://www.apache.org/licenses/LICENSE-2.0
11  *
12  * Unless required by applicable law or agreed to in writing,
13  * software distributed under the License is distributed on an
14  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
15  * KIND, either express or implied. See the License for the
16  * specific language governing permissions and limitations
17  * under the License.
18  */
19 
26 #ifndef TVM_TIR_STMT_FUNCTOR_H_
27 #define TVM_TIR_STMT_FUNCTOR_H_
28 
29 #include <tvm/node/functor.h>
30 #include <tvm/tir/expr.h>
31 #include <tvm/tir/expr_functor.h>
32 #include <tvm/tir/function.h>
33 #include <tvm/tir/stmt.h>
34 
35 #include <unordered_map>
36 #include <utility>
37 
38 namespace tvm {
39 namespace tir {
45 template <typename FType>
47 
48 #define STMT_FUNCTOR_DEFAULT \
49  { return VisitStmtDefault_(op, std::forward<Args>(args)...); }
50 
51 #define IR_STMT_FUNCTOR_DISPATCH(OP) \
52  vtable.template set_dispatch<OP>([](const ObjectRef& n, TSelf* self, Args... args) { \
53  return self->VisitStmt_(static_cast<const OP*>(n.get()), std::forward<Args>(args)...); \
54  });
55 
56 template <typename R, typename... Args>
57 class StmtFunctor<R(const Stmt& n, Args... args)> {
58  private:
59  using TSelf = StmtFunctor<R(const Stmt& n, Args... args)>;
60  using FType = NodeFunctor<R(const ObjectRef& n, TSelf* self, Args... args)>;
61 
62  public:
64  using result_type = R;
66  virtual ~StmtFunctor() {}
73  R operator()(const Stmt& n, Args... args) { return VisitStmt(n, std::forward<Args>(args)...); }
80  virtual R VisitStmt(const Stmt& n, Args... args) {
81  static FType vtable = InitVTable();
82  return vtable(n, this, std::forward<Args>(args)...);
83  }
84  // Functions that can be overriden by subclass
85  virtual R VisitStmt_(const LetStmtNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
86  virtual R VisitStmt_(const AttrStmtNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
87  virtual R VisitStmt_(const IfThenElseNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
88  virtual R VisitStmt_(const ForNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
89  virtual R VisitStmt_(const WhileNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
90  virtual R VisitStmt_(const AllocateNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
91  virtual R VisitStmt_(const AllocateConstNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
92  virtual R VisitStmt_(const DeclBufferNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
93  virtual R VisitStmt_(const StoreNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
94  virtual R VisitStmt_(const BufferStoreNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
95  virtual R VisitStmt_(const BufferRealizeNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
96  virtual R VisitStmt_(const AssertStmtNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
97  virtual R VisitStmt_(const ProducerStoreNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
98  virtual R VisitStmt_(const ProducerRealizeNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
99  virtual R VisitStmt_(const PrefetchNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
100  virtual R VisitStmt_(const SeqStmtNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
101  virtual R VisitStmt_(const EvaluateNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
102  virtual R VisitStmt_(const BlockNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
103  virtual R VisitStmt_(const BlockRealizeNode* op, Args... args) STMT_FUNCTOR_DEFAULT;
104  virtual R VisitStmtDefault_(const Object* op, Args...) {
105  LOG(FATAL) << "Do not have a default for " << op->GetTypeKey();
106  }
107 
108  private:
109  // initialize the vtable.
110  static FType InitVTable() {
111  FType vtable;
131  return vtable;
132  }
133 };
134 
135 #undef IR_STMT_FUNCTOR_DISPATCH
136 #undef STMT_FUNCTOR_DEFAULT
137 
141 class TVM_DLL StmtVisitor : protected StmtFunctor<void(const Stmt&)> {
142  public:
143  using StmtFunctor::operator();
144 
145  protected:
146  using StmtFunctor::VisitStmt;
154  virtual void VisitExpr(const PrimExpr& e) {}
155  // statement visitor
156  void VisitStmt_(const AttrStmtNode* op) override;
157  void VisitStmt_(const IfThenElseNode* op) override;
158  void VisitStmt_(const LetStmtNode* op) override;
159  void VisitStmt_(const ForNode* op) override;
160  void VisitStmt_(const WhileNode* op) override;
161  void VisitStmt_(const AllocateNode* op) override;
162  void VisitStmt_(const AllocateConstNode* op) override;
163  void VisitStmt_(const DeclBufferNode* op) override;
164  void VisitStmt_(const StoreNode* op) override;
165  void VisitStmt_(const BufferStoreNode* op) override;
166  void VisitStmt_(const BufferRealizeNode* op) override;
167  void VisitStmt_(const AssertStmtNode* op) override;
168  void VisitStmt_(const ProducerStoreNode* op) override;
169  void VisitStmt_(const ProducerRealizeNode* op) override;
170  void VisitStmt_(const PrefetchNode* op) override;
171  void VisitStmt_(const SeqStmtNode* op) override;
172  void VisitStmt_(const EvaluateNode* op) override;
173  void VisitStmt_(const BlockNode* op) override;
174  void VisitStmt_(const BlockRealizeNode* op) override;
175 };
176 
180 class TVM_DLL StmtMutator : protected StmtFunctor<Stmt(const Stmt&)> {
181  public:
191  allow_copy_on_write_ = true;
192  return VisitStmt(stmt);
193  }
194 
195  protected:
196  // We perform copy on write optimizations on the StmtMutator
197  // so that an unique copy of parent can be mutated inplace
198  // when some of its children changed.
199  // We only do such optimization for Stmt nests(instead of Exprs) for now
200  // as Stmt's parent state is more likely remain unchanged when one of
201  // its child block changes.
206  bool allow_copy_on_write_{false};
216  template <typename TNode>
217  ObjectPtr<TNode> CopyOnWrite(const TNode* node) {
218  static_assert(std::is_base_of<StmtNode, TNode>::value,
219  "StmtMutator:: CopyOnWrite requires us to track uniqueness of all parent "
220  "nodes during the recursion. Because the child classes do not necessarily "
221  "check the Array, Expr and other structures during the visit, it is only safe to "
222  "call this function with StmtNodes for now. "
223  "Please create a new node directly in other cases.");
224  if (allow_copy_on_write_) {
225  // return the old node.
226  return runtime::GetObjectPtr<TNode>(const_cast<TNode*>(node));
227  } else {
228  // Make a new copy of the node.
229  // need to rely on the default copy constructor
230  return runtime::make_object<TNode>(*node);
231  }
232  }
239  Stmt VisitStmt(const Stmt& stmt) override {
240  if (allow_copy_on_write_ && !stmt.unique()) {
241  allow_copy_on_write_ = false;
242  Stmt ret = StmtFunctor::VisitStmt(stmt);
243  allow_copy_on_write_ = true;
244  return ret;
245  } else {
246  return StmtFunctor::VisitStmt(stmt);
247  }
248  }
256  virtual PrimExpr VisitExpr(const PrimExpr& e) { return e; }
257  // statement visitor
258  Stmt VisitStmt_(const AttrStmtNode* op) override;
259  Stmt VisitStmt_(const IfThenElseNode* op) override;
260  Stmt VisitStmt_(const LetStmtNode* op) override;
261  Stmt VisitStmt_(const ForNode* op) override;
262  Stmt VisitStmt_(const WhileNode* op) override;
263  Stmt VisitStmt_(const AllocateNode* op) override;
264  Stmt VisitStmt_(const AllocateConstNode* op) override;
265  Stmt VisitStmt_(const DeclBufferNode* op) override;
266  Stmt VisitStmt_(const StoreNode* op) override;
267  Stmt VisitStmt_(const BufferStoreNode* op) override;
268  Stmt VisitStmt_(const BufferRealizeNode* op) override;
269  Stmt VisitStmt_(const AssertStmtNode* op) override;
270  Stmt VisitStmt_(const ProducerStoreNode* op) override;
271  Stmt VisitStmt_(const ProducerRealizeNode* op) override;
272  Stmt VisitStmt_(const PrefetchNode* op) override;
273  Stmt VisitStmt_(const SeqStmtNode* op) override;
274  Stmt VisitStmt_(const EvaluateNode* op) override;
275  Stmt VisitStmt_(const BlockNode* op) override;
276  Stmt VisitStmt_(const BlockRealizeNode* op) override;
289  Stmt VisitSeqStmt_(const SeqStmtNode* op, bool flatten_before_visit,
290  std::function<Stmt(const Stmt&)> fmutate = nullptr);
291 
292  // internal helper.
293  class Internal;
294 };
295 
299 class StmtExprVisitor : public StmtVisitor, public ExprVisitor {
300  public:
301  using StmtVisitor::operator();
302  using ExprVisitor::operator();
303 
304  protected:
305  using ExprVisitor::VisitExpr;
306  using StmtVisitor::VisitStmt;
307 
308  void VisitExpr(const PrimExpr& e) override { return ExprVisitor::VisitExpr(e); }
309 };
310 
314 class StmtExprMutator : public StmtMutator, public ExprMutator {
315  public:
316  using StmtMutator::operator();
317  using ExprMutator::operator();
318 
319  protected:
320  using ExprMutator::VisitExpr;
322 
323  PrimExpr VisitExpr(const PrimExpr& e) override { return ExprMutator::VisitExpr(e); }
324 };
325 
341 TVM_DLL Stmt IRTransform(Stmt stmt, const runtime::PackedFunc& preorder,
342  const runtime::PackedFunc& postorder,
343  Optional<Array<String>> only_enable = NullOpt);
344 
351 TVM_DLL void PostOrderVisit(const ObjectRef& node, std::function<void(const ObjectRef&)> fvisit);
352 
359 TVM_DLL Stmt Substitute(Stmt stmt, std::function<Optional<PrimExpr>(const Var& var)> vmap);
360 
367 TVM_DLL PrimExpr Substitute(PrimExpr expr, std::function<Optional<PrimExpr>(const Var& var)> vmap);
368 
375 TVM_DLL Array<Range> Substitute(const Array<Range>& region, const Map<Var, PrimExpr>& vmap);
376 
384 template <typename T>
385 inline auto Substitute(T input, const Map<Var, PrimExpr>& value_map) {
386  auto vmap = [&](const Var& var) -> Optional<PrimExpr> {
387  auto it = value_map.find(var);
388  if (it != value_map.end()) return (*it).second;
389  return Optional<PrimExpr>(nullptr);
390  };
391  return Substitute(std::move(input), vmap);
392 }
393 
401 template <typename T>
402 inline T Substitute(T input, const std::unordered_map<const VarNode*, PrimExpr>& value_map) {
403  auto vmap = [&](const Var& var) -> Optional<PrimExpr> {
404  auto it = value_map.find(var.get());
405  if (it != value_map.end()) return (*it).second;
406  return Optional<PrimExpr>(nullptr);
407  };
408  return Substitute(std::move(input), vmap);
409 }
410 
422  std::function<Optional<PrimExpr>(const Var&)> vmap);
423 
435  PrimExpr expr, std::function<Optional<PrimExpr>(const Var&)> vmap);
436 
444 TVM_DLL void PreOrderVisit(const ObjectRef& stmt_or_expr,
445  const std::function<bool(const ObjectRef&)>& fvisit);
446 
454 TVM_DLL PrimFunc RenewDefs(const PrimFunc& func);
455 
466 template <typename Node, typename = std::enable_if_t<std::is_base_of_v<StmtNode, Node>>>
467 bool ContainsNode(const Stmt& stmt) {
468  struct Visitor : StmtVisitor {
469  // Early bail-out, if we already found the node.
470  void VisitStmt(const Stmt& stmt) final {
471  if (contains_node) {
472  return;
473  }
474  StmtVisitor::VisitStmt(stmt);
475  }
476 
477  void VisitStmt_(const Node* block) override { contains_node = true; }
478 
479  bool contains_node{false};
480  };
481 
482  Visitor visitor;
483  visitor(stmt);
484  return visitor.contains_node;
485 }
486 
487 } // namespace tir
488 } // namespace tvm
489 
490 #endif // TVM_TIR_STMT_FUNCTOR_H_
virtual R VisitStmt_(const ProducerRealizeNode *op, Args... args)
Definition: stmt_functor.h:98
virtual R VisitStmt_(const LetStmtNode *op, Args... args)
Definition: stmt_functor.h:85
A dynamically dispatched functor on the type of the first argument.
Definition: functor.h:64
Define certain auxiliary attribute for the body to be a symbolic value. This provide auxiliary inform...
Definition: stmt.h:118
Store value into mult-dimensional array that will be read by the consumer of the producer.
Definition: stmt.h:408
A prefetch hint for a buffer.
Definition: stmt.h:1073
Declare a buffer that can be used in the body.
Definition: stmt.h:692
virtual R VisitStmt_(const SeqStmtNode *op, Args... args)
Definition: stmt_functor.h:100
void VisitExpr(const PrimExpr &e) override
Visitor to Exprs, can be overriden to do recursive changes to Exprs.
Definition: stmt_functor.h:308
A custom smart pointer for Object.
Definition: object.h:358
StmtMutator that mutates the statements.
Definition: stmt_functor.h:180
A block realization node represents execution of the block at the binding values. ...
Definition: stmt.h:1325
virtual PrimExpr VisitExpr(const PrimExpr &e)
Visitor to Exprs, can be overriden to do recursive changes to Exprs.
Definition: stmt_functor.h:256
Visitor that recursively visit stmts and exprs on them.
Definition: stmt_functor.h:299
virtual R VisitStmt_(const WhileNode *op, Args... args)
Definition: stmt_functor.h:89
virtual R VisitStmt_(const AssertStmtNode *op, Args... args)
Definition: stmt_functor.h:96
virtual R VisitStmt_(const AllocateConstNode *op, Args... args)
Definition: stmt_functor.h:91
runtime implementation for LibTorch/TorchScript.
Definition: analyzer.h:36
ExprVisitor.
Definition: expr_functor.h:208
virtual void VisitExpr(const PrimExpr &e)
Visitor to Exprs, can be overriden to do recursive changes to Exprs.
Definition: stmt_functor.h:154
The container of seq statement. Represent a sequence of statements.
Definition: stmt.h:730
a named variable in TIR
Definition: var.h:88
IfThenElse statment.
Definition: stmt.h:828
PrimExpr VisitExpr(const PrimExpr &e) override
Visitor to Exprs, can be overriden to do recursive changes to Exprs.
Definition: stmt_functor.h:323
Same as ExprFunctor except it is applied on statements.
Definition: stmt_functor.h:46
virtual ~StmtFunctor()
virtual destructor
Definition: stmt_functor.h:66
virtual R VisitStmt_(const StoreNode *op, Args... args)
Definition: stmt_functor.h:93
TIR Function.
base class of all object containers.
Definition: object.h:167
Stmt operator()(Stmt stmt)
Mutate stmt.
Definition: stmt_functor.h:190
void PostOrderVisit(const ObjectRef &node, std::function< void(const ObjectRef &)> fvisit)
Recursively visit the ir in post DFS order node, apply fvisit Each node is guaranteed to be visited o...
Stmt VisitStmt(const Stmt &stmt) override
Internal mutator that everyone calls.
Definition: stmt_functor.h:239
PrimFunc RenewDefs(const PrimFunc &func)
Renew the definition nodes for a TIR, including Var, Buffer and IterVar. This pass works as a simple ...
#define IR_STMT_FUNCTOR_DISPATCH(OP)
Definition: stmt_functor.h:51
Functors for tir expressions.
R operator()(const Stmt &n, Args... args)
Same as call.
Definition: stmt_functor.h:73
Annotate the bounds where the data produced by the producer need to be written and read in body...
Definition: stmt.h:463
virtual R VisitStmt_(const ProducerStoreNode *op, Args... args)
Definition: stmt_functor.h:97
TIR statements.
Stmt SubstituteWithDataTypeLegalization(Stmt stmt, std::function< Optional< PrimExpr >(const Var &)> vmap)
Substitute the var specified by vmap and legalize data types after substitution.
TIR expressions.
Array, container representing a contiguous sequence of ObjectRefs.
Definition: array.h:289
Managed reference to PrimFuncNode.
Definition: function.h:143
A While loop.
Definition: stmt.h:1032
virtual R VisitStmt(const Stmt &n, Args... args)
The functor call.
Definition: stmt_functor.h:80
iterator find(const K &key) const
Definition: map.h:1383
virtual R VisitStmtDefault_(const Object *op, Args...)
Definition: stmt_functor.h:104
Container of all statements.
Definition: stmt.h:57
R result_type
the result type of this functor
Definition: stmt_functor.h:64
virtual R VisitStmt_(const DeclBufferNode *op, Args... args)
Definition: stmt_functor.h:92
ObjectPtr< TNode > CopyOnWrite(const TNode *node)
Perform copy on write on node.
Definition: stmt_functor.h:217
bool ContainsNode(const Stmt &stmt)
Check if the statement contains the specified node type.
Definition: stmt_functor.h:467
Allocate a buffer that can be used in body.
Definition: stmt.h:519
Evaluates an expression. This is mostly used for putting a Call node into Stmt.
Definition: stmt.h:878
bool unique() const
Definition: object.h:550
virtual R VisitStmt_(const BufferRealizeNode *op, Args... args)
Definition: stmt_functor.h:95
virtual R VisitStmt_(const PrefetchNode *op, Args... args)
Definition: stmt_functor.h:99
const VarNode * get() const
Get pointer to the internal value.
Definition: var.h:128
A block is a basic schedule unit in TIR.
Definition: stmt.h:1241
virtual R VisitStmt_(const AttrStmtNode *op, Args... args)
Definition: stmt_functor.h:86
Var var(std::string name_hint, DataType t=DataType::Int(32))
Construct a new Var expression.
Allocate a buffer that can be used in body.
Definition: stmt.h:601
Defines the Functor data structures.
Base class of all object reference.
Definition: object.h:511
Store value to the buffer.
Definition: stmt.h:232
std::string GetTypeKey() const
Definition: object.h:180
virtual R VisitStmt_(const IfThenElseNode *op, Args... args)
Definition: stmt_functor.h:87
Stmt Substitute(Stmt stmt, std::function< Optional< PrimExpr >(const Var &var)> vmap)
Substitute the var specified by vmap.
Mutator that recursively mutates stmts and exprs on them.
Definition: stmt_functor.h:314
Store value to the high dimension buffer.
Definition: stmt.h:290
iterator end() const
Definition: map.h:1381
virtual R VisitStmt_(const BlockRealizeNode *op, Args... args)
Definition: stmt_functor.h:103
virtual R VisitStmt_(const ForNode *op, Args... args)
Definition: stmt_functor.h:88
Assert condition, if an error occurs, return the error message.
Definition: stmt.h:168
Map container of NodeRef->NodeRef in DSL graph. Map implements copy on write semantics, which means map is mutable but copy will happen when array is referenced in more than two places.
Definition: map.h:1271
PrimExpr ret(PrimExpr value, Span span=Span())
Return the value.
Stmt IRTransform(Stmt stmt, const runtime::PackedFunc &preorder, const runtime::PackedFunc &postorder, Optional< Array< String >> only_enable=NullOpt)
recursively visit the ir nodes in post DFS order, and transform it
Packed function is a type-erased function. The arguments are passed by packed format.
Definition: packed_func.h:138
Optional container that to represent to a Nullable variant of T.
Definition: optional.h:51
void PreOrderVisit(const ObjectRef &stmt_or_expr, const std::function< bool(const ObjectRef &)> &fvisit)
Recursively visit the IR in pre DFS order node, apply fvisit. If fvisit returns false, it won&#39;t visit the children of the node.
A for loop, with poissible type annotations.
Definition: stmt.h:950
#define STMT_FUNCTOR_DEFAULT
Definition: stmt_functor.h:48
virtual R VisitStmt_(const BufferStoreNode *op, Args... args)
Definition: stmt_functor.h:94
Let binding, bind var to value, then run body.
Definition: stmt.h:65
Reference to PrimExprNode.
Definition: expr.h:112
StmtVisitor.
Definition: stmt_functor.h:141
constexpr runtime::NullOptType NullOpt
Definition: optional.h:160
Annotate the region where the buffer need to be read and write in the body. We only need to allocate ...
Definition: stmt.h:345
virtual R VisitStmt_(const EvaluateNode *op, Args... args)
Definition: stmt_functor.h:101
virtual R VisitStmt_(const BlockNode *op, Args... args)
Definition: stmt_functor.h:102
virtual R VisitStmt_(const AllocateNode *op, Args... args)
Definition: stmt_functor.h:90
ExprMutator that mutates expressions.
Definition: expr_functor.h:255