MLIR  20.0.0git
IRAffine.cpp
Go to the documentation of this file.
1 //===- IRAffine.cpp - Exports 'ir' module affine related bindings ---------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include <cstddef>
10 #include <cstdint>
11 #include <pybind11/cast.h>
12 #include <pybind11/detail/common.h>
13 #include <pybind11/pybind11.h>
14 #include <pybind11/pytypes.h>
15 #include <string>
16 #include <utility>
17 #include <vector>
18 
19 #include "IRModule.h"
20 
21 #include "PybindUtils.h"
22 
23 #include "mlir-c/AffineExpr.h"
24 #include "mlir-c/AffineMap.h"
26 #include "mlir-c/IntegerSet.h"
27 #include "mlir/Support/LLVM.h"
28 #include "llvm/ADT/Hashing.h"
29 #include "llvm/ADT/SmallVector.h"
30 #include "llvm/ADT/StringRef.h"
31 #include "llvm/ADT/Twine.h"
32 
33 namespace py = pybind11;
34 using namespace mlir;
35 using namespace mlir::python;
36 
37 using llvm::SmallVector;
38 using llvm::StringRef;
39 using llvm::Twine;
40 
41 static const char kDumpDocstring[] =
42  R"(Dumps a debug representation of the object to stderr.)";
43 
44 /// Attempts to populate `result` with the content of `list` casted to the
45 /// appropriate type (Python and C types are provided as template arguments).
46 /// Throws errors in case of failure, using "action" to describe what the caller
47 /// was attempting to do.
48 template <typename PyType, typename CType>
49 static void pyListToVector(const py::list &list,
51  StringRef action) {
52  result.reserve(py::len(list));
53  for (py::handle item : list) {
54  try {
55  result.push_back(item.cast<PyType>());
56  } catch (py::cast_error &err) {
57  std::string msg = (llvm::Twine("Invalid expression when ") + action +
58  " (" + err.what() + ")")
59  .str();
60  throw py::cast_error(msg);
61  } catch (py::reference_cast_error &err) {
62  std::string msg = (llvm::Twine("Invalid expression (None?) when ") +
63  action + " (" + err.what() + ")")
64  .str();
65  throw py::cast_error(msg);
66  }
67  }
68 }
69 
70 template <typename PermutationTy>
71 static bool isPermutation(std::vector<PermutationTy> permutation) {
72  llvm::SmallVector<bool, 8> seen(permutation.size(), false);
73  for (auto val : permutation) {
74  if (val < permutation.size()) {
75  if (seen[val])
76  return false;
77  seen[val] = true;
78  continue;
79  }
80  return false;
81  }
82  return true;
83 }
84 
85 namespace {
86 
87 /// CRTP base class for Python MLIR affine expressions that subclass AffineExpr
88 /// and should be castable from it. Intermediate hierarchy classes can be
89 /// modeled by specifying BaseTy.
90 template <typename DerivedTy, typename BaseTy = PyAffineExpr>
91 class PyConcreteAffineExpr : public BaseTy {
92 public:
93  // Derived classes must define statics for:
94  // IsAFunctionTy isaFunction
95  // const char *pyClassName
96  // and redefine bindDerived.
97  using ClassTy = py::class_<DerivedTy, BaseTy>;
98  using IsAFunctionTy = bool (*)(MlirAffineExpr);
99 
100  PyConcreteAffineExpr() = default;
101  PyConcreteAffineExpr(PyMlirContextRef contextRef, MlirAffineExpr affineExpr)
102  : BaseTy(std::move(contextRef), affineExpr) {}
103  PyConcreteAffineExpr(PyAffineExpr &orig)
104  : PyConcreteAffineExpr(orig.getContext(), castFrom(orig)) {}
105 
106  static MlirAffineExpr castFrom(PyAffineExpr &orig) {
107  if (!DerivedTy::isaFunction(orig)) {
108  auto origRepr = py::repr(py::cast(orig)).cast<std::string>();
109  throw py::value_error((Twine("Cannot cast affine expression to ") +
110  DerivedTy::pyClassName + " (from " + origRepr +
111  ")")
112  .str());
113  }
114  return orig;
115  }
116 
117  static void bind(py::module &m) {
118  auto cls = ClassTy(m, DerivedTy::pyClassName, py::module_local());
119  cls.def(py::init<PyAffineExpr &>(), py::arg("expr"));
120  cls.def_static(
121  "isinstance",
122  [](PyAffineExpr &otherAffineExpr) -> bool {
123  return DerivedTy::isaFunction(otherAffineExpr);
124  },
125  py::arg("other"));
126  DerivedTy::bindDerived(cls);
127  }
128 
129  /// Implemented by derived classes to add methods to the Python subclass.
130  static void bindDerived(ClassTy &m) {}
131 };
132 
133 class PyAffineConstantExpr : public PyConcreteAffineExpr<PyAffineConstantExpr> {
134 public:
135  static constexpr IsAFunctionTy isaFunction = mlirAffineExprIsAConstant;
136  static constexpr const char *pyClassName = "AffineConstantExpr";
137  using PyConcreteAffineExpr::PyConcreteAffineExpr;
138 
139  static PyAffineConstantExpr get(intptr_t value,
140  DefaultingPyMlirContext context) {
141  MlirAffineExpr affineExpr =
142  mlirAffineConstantExprGet(context->get(), static_cast<int64_t>(value));
143  return PyAffineConstantExpr(context->getRef(), affineExpr);
144  }
145 
146  static void bindDerived(ClassTy &c) {
147  c.def_static("get", &PyAffineConstantExpr::get, py::arg("value"),
148  py::arg("context") = py::none());
149  c.def_property_readonly("value", [](PyAffineConstantExpr &self) {
150  return mlirAffineConstantExprGetValue(self);
151  });
152  }
153 };
154 
155 class PyAffineDimExpr : public PyConcreteAffineExpr<PyAffineDimExpr> {
156 public:
157  static constexpr IsAFunctionTy isaFunction = mlirAffineExprIsADim;
158  static constexpr const char *pyClassName = "AffineDimExpr";
159  using PyConcreteAffineExpr::PyConcreteAffineExpr;
160 
161  static PyAffineDimExpr get(intptr_t pos, DefaultingPyMlirContext context) {
162  MlirAffineExpr affineExpr = mlirAffineDimExprGet(context->get(), pos);
163  return PyAffineDimExpr(context->getRef(), affineExpr);
164  }
165 
166  static void bindDerived(ClassTy &c) {
167  c.def_static("get", &PyAffineDimExpr::get, py::arg("position"),
168  py::arg("context") = py::none());
169  c.def_property_readonly("position", [](PyAffineDimExpr &self) {
170  return mlirAffineDimExprGetPosition(self);
171  });
172  }
173 };
174 
175 class PyAffineSymbolExpr : public PyConcreteAffineExpr<PyAffineSymbolExpr> {
176 public:
177  static constexpr IsAFunctionTy isaFunction = mlirAffineExprIsASymbol;
178  static constexpr const char *pyClassName = "AffineSymbolExpr";
179  using PyConcreteAffineExpr::PyConcreteAffineExpr;
180 
181  static PyAffineSymbolExpr get(intptr_t pos, DefaultingPyMlirContext context) {
182  MlirAffineExpr affineExpr = mlirAffineSymbolExprGet(context->get(), pos);
183  return PyAffineSymbolExpr(context->getRef(), affineExpr);
184  }
185 
186  static void bindDerived(ClassTy &c) {
187  c.def_static("get", &PyAffineSymbolExpr::get, py::arg("position"),
188  py::arg("context") = py::none());
189  c.def_property_readonly("position", [](PyAffineSymbolExpr &self) {
190  return mlirAffineSymbolExprGetPosition(self);
191  });
192  }
193 };
194 
195 class PyAffineBinaryExpr : public PyConcreteAffineExpr<PyAffineBinaryExpr> {
196 public:
197  static constexpr IsAFunctionTy isaFunction = mlirAffineExprIsABinary;
198  static constexpr const char *pyClassName = "AffineBinaryExpr";
199  using PyConcreteAffineExpr::PyConcreteAffineExpr;
200 
201  PyAffineExpr lhs() {
202  MlirAffineExpr lhsExpr = mlirAffineBinaryOpExprGetLHS(get());
203  return PyAffineExpr(getContext(), lhsExpr);
204  }
205 
206  PyAffineExpr rhs() {
207  MlirAffineExpr rhsExpr = mlirAffineBinaryOpExprGetRHS(get());
208  return PyAffineExpr(getContext(), rhsExpr);
209  }
210 
211  static void bindDerived(ClassTy &c) {
212  c.def_property_readonly("lhs", &PyAffineBinaryExpr::lhs);
213  c.def_property_readonly("rhs", &PyAffineBinaryExpr::rhs);
214  }
215 };
216 
217 class PyAffineAddExpr
218  : public PyConcreteAffineExpr<PyAffineAddExpr, PyAffineBinaryExpr> {
219 public:
220  static constexpr IsAFunctionTy isaFunction = mlirAffineExprIsAAdd;
221  static constexpr const char *pyClassName = "AffineAddExpr";
222  using PyConcreteAffineExpr::PyConcreteAffineExpr;
223 
224  static PyAffineAddExpr get(PyAffineExpr lhs, const PyAffineExpr &rhs) {
225  MlirAffineExpr expr = mlirAffineAddExprGet(lhs, rhs);
226  return PyAffineAddExpr(lhs.getContext(), expr);
227  }
228 
229  static PyAffineAddExpr getRHSConstant(PyAffineExpr lhs, intptr_t rhs) {
230  MlirAffineExpr expr = mlirAffineAddExprGet(
232  return PyAffineAddExpr(lhs.getContext(), expr);
233  }
234 
235  static PyAffineAddExpr getLHSConstant(intptr_t lhs, PyAffineExpr rhs) {
236  MlirAffineExpr expr = mlirAffineAddExprGet(
238  return PyAffineAddExpr(rhs.getContext(), expr);
239  }
240 
241  static void bindDerived(ClassTy &c) {
242  c.def_static("get", &PyAffineAddExpr::get);
243  }
244 };
245 
246 class PyAffineMulExpr
247  : public PyConcreteAffineExpr<PyAffineMulExpr, PyAffineBinaryExpr> {
248 public:
249  static constexpr IsAFunctionTy isaFunction = mlirAffineExprIsAMul;
250  static constexpr const char *pyClassName = "AffineMulExpr";
251  using PyConcreteAffineExpr::PyConcreteAffineExpr;
252 
253  static PyAffineMulExpr get(PyAffineExpr lhs, const PyAffineExpr &rhs) {
254  MlirAffineExpr expr = mlirAffineMulExprGet(lhs, rhs);
255  return PyAffineMulExpr(lhs.getContext(), expr);
256  }
257 
258  static PyAffineMulExpr getRHSConstant(PyAffineExpr lhs, intptr_t rhs) {
259  MlirAffineExpr expr = mlirAffineMulExprGet(
261  return PyAffineMulExpr(lhs.getContext(), expr);
262  }
263 
264  static PyAffineMulExpr getLHSConstant(intptr_t lhs, PyAffineExpr rhs) {
265  MlirAffineExpr expr = mlirAffineMulExprGet(
267  return PyAffineMulExpr(rhs.getContext(), expr);
268  }
269 
270  static void bindDerived(ClassTy &c) {
271  c.def_static("get", &PyAffineMulExpr::get);
272  }
273 };
274 
275 class PyAffineModExpr
276  : public PyConcreteAffineExpr<PyAffineModExpr, PyAffineBinaryExpr> {
277 public:
278  static constexpr IsAFunctionTy isaFunction = mlirAffineExprIsAMod;
279  static constexpr const char *pyClassName = "AffineModExpr";
280  using PyConcreteAffineExpr::PyConcreteAffineExpr;
281 
282  static PyAffineModExpr get(PyAffineExpr lhs, const PyAffineExpr &rhs) {
283  MlirAffineExpr expr = mlirAffineModExprGet(lhs, rhs);
284  return PyAffineModExpr(lhs.getContext(), expr);
285  }
286 
287  static PyAffineModExpr getRHSConstant(PyAffineExpr lhs, intptr_t rhs) {
288  MlirAffineExpr expr = mlirAffineModExprGet(
290  return PyAffineModExpr(lhs.getContext(), expr);
291  }
292 
293  static PyAffineModExpr getLHSConstant(intptr_t lhs, PyAffineExpr rhs) {
294  MlirAffineExpr expr = mlirAffineModExprGet(
296  return PyAffineModExpr(rhs.getContext(), expr);
297  }
298 
299  static void bindDerived(ClassTy &c) {
300  c.def_static("get", &PyAffineModExpr::get);
301  }
302 };
303 
304 class PyAffineFloorDivExpr
305  : public PyConcreteAffineExpr<PyAffineFloorDivExpr, PyAffineBinaryExpr> {
306 public:
307  static constexpr IsAFunctionTy isaFunction = mlirAffineExprIsAFloorDiv;
308  static constexpr const char *pyClassName = "AffineFloorDivExpr";
309  using PyConcreteAffineExpr::PyConcreteAffineExpr;
310 
311  static PyAffineFloorDivExpr get(PyAffineExpr lhs, const PyAffineExpr &rhs) {
312  MlirAffineExpr expr = mlirAffineFloorDivExprGet(lhs, rhs);
313  return PyAffineFloorDivExpr(lhs.getContext(), expr);
314  }
315 
316  static PyAffineFloorDivExpr getRHSConstant(PyAffineExpr lhs, intptr_t rhs) {
317  MlirAffineExpr expr = mlirAffineFloorDivExprGet(
319  return PyAffineFloorDivExpr(lhs.getContext(), expr);
320  }
321 
322  static PyAffineFloorDivExpr getLHSConstant(intptr_t lhs, PyAffineExpr rhs) {
323  MlirAffineExpr expr = mlirAffineFloorDivExprGet(
325  return PyAffineFloorDivExpr(rhs.getContext(), expr);
326  }
327 
328  static void bindDerived(ClassTy &c) {
329  c.def_static("get", &PyAffineFloorDivExpr::get);
330  }
331 };
332 
333 class PyAffineCeilDivExpr
334  : public PyConcreteAffineExpr<PyAffineCeilDivExpr, PyAffineBinaryExpr> {
335 public:
336  static constexpr IsAFunctionTy isaFunction = mlirAffineExprIsACeilDiv;
337  static constexpr const char *pyClassName = "AffineCeilDivExpr";
338  using PyConcreteAffineExpr::PyConcreteAffineExpr;
339 
340  static PyAffineCeilDivExpr get(PyAffineExpr lhs, const PyAffineExpr &rhs) {
341  MlirAffineExpr expr = mlirAffineCeilDivExprGet(lhs, rhs);
342  return PyAffineCeilDivExpr(lhs.getContext(), expr);
343  }
344 
345  static PyAffineCeilDivExpr getRHSConstant(PyAffineExpr lhs, intptr_t rhs) {
346  MlirAffineExpr expr = mlirAffineCeilDivExprGet(
348  return PyAffineCeilDivExpr(lhs.getContext(), expr);
349  }
350 
351  static PyAffineCeilDivExpr getLHSConstant(intptr_t lhs, PyAffineExpr rhs) {
352  MlirAffineExpr expr = mlirAffineCeilDivExprGet(
354  return PyAffineCeilDivExpr(rhs.getContext(), expr);
355  }
356 
357  static void bindDerived(ClassTy &c) {
358  c.def_static("get", &PyAffineCeilDivExpr::get);
359  }
360 };
361 
362 } // namespace
363 
364 bool PyAffineExpr::operator==(const PyAffineExpr &other) const {
365  return mlirAffineExprEqual(affineExpr, other.affineExpr);
366 }
367 
369  return py::reinterpret_steal<py::object>(
371 }
372 
374  MlirAffineExpr rawAffineExpr = mlirPythonCapsuleToAffineExpr(capsule.ptr());
375  if (mlirAffineExprIsNull(rawAffineExpr))
376  throw py::error_already_set();
377  return PyAffineExpr(
379  rawAffineExpr);
380 }
381 
382 //------------------------------------------------------------------------------
383 // PyAffineMap and utilities.
384 //------------------------------------------------------------------------------
385 namespace {
386 
387 /// A list of expressions contained in an affine map. Internally these are
388 /// stored as a consecutive array leading to inexpensive random access. Both
389 /// the map and the expression are owned by the context so we need not bother
390 /// with lifetime extension.
391 class PyAffineMapExprList
392  : public Sliceable<PyAffineMapExprList, PyAffineExpr> {
393 public:
394  static constexpr const char *pyClassName = "AffineExprList";
395 
396  PyAffineMapExprList(const PyAffineMap &map, intptr_t startIndex = 0,
397  intptr_t length = -1, intptr_t step = 1)
398  : Sliceable(startIndex,
399  length == -1 ? mlirAffineMapGetNumResults(map) : length,
400  step),
401  affineMap(map) {}
402 
403 private:
404  /// Give the parent CRTP class access to hook implementations below.
405  friend class Sliceable<PyAffineMapExprList, PyAffineExpr>;
406 
407  intptr_t getRawNumElements() { return mlirAffineMapGetNumResults(affineMap); }
408 
409  PyAffineExpr getRawElement(intptr_t pos) {
410  return PyAffineExpr(affineMap.getContext(),
411  mlirAffineMapGetResult(affineMap, pos));
412  }
413 
414  PyAffineMapExprList slice(intptr_t startIndex, intptr_t length,
415  intptr_t step) {
416  return PyAffineMapExprList(affineMap, startIndex, length, step);
417  }
418 
419  PyAffineMap affineMap;
420 };
421 } // namespace
422 
423 bool PyAffineMap::operator==(const PyAffineMap &other) const {
424  return mlirAffineMapEqual(affineMap, other.affineMap);
425 }
426 
428  return py::reinterpret_steal<py::object>(mlirPythonAffineMapToCapsule(*this));
429 }
430 
432  MlirAffineMap rawAffineMap = mlirPythonCapsuleToAffineMap(capsule.ptr());
433  if (mlirAffineMapIsNull(rawAffineMap))
434  throw py::error_already_set();
435  return PyAffineMap(
437  rawAffineMap);
438 }
439 
440 //------------------------------------------------------------------------------
441 // PyIntegerSet and utilities.
442 //------------------------------------------------------------------------------
443 namespace {
444 
445 class PyIntegerSetConstraint {
446 public:
447  PyIntegerSetConstraint(PyIntegerSet set, intptr_t pos)
448  : set(std::move(set)), pos(pos) {}
449 
450  PyAffineExpr getExpr() {
451  return PyAffineExpr(set.getContext(),
452  mlirIntegerSetGetConstraint(set, pos));
453  }
454 
455  bool isEq() { return mlirIntegerSetIsConstraintEq(set, pos); }
456 
457  static void bind(py::module &m) {
458  py::class_<PyIntegerSetConstraint>(m, "IntegerSetConstraint",
459  py::module_local())
460  .def_property_readonly("expr", &PyIntegerSetConstraint::getExpr)
461  .def_property_readonly("is_eq", &PyIntegerSetConstraint::isEq);
462  }
463 
464 private:
465  PyIntegerSet set;
466  intptr_t pos;
467 };
468 
469 class PyIntegerSetConstraintList
470  : public Sliceable<PyIntegerSetConstraintList, PyIntegerSetConstraint> {
471 public:
472  static constexpr const char *pyClassName = "IntegerSetConstraintList";
473 
474  PyIntegerSetConstraintList(const PyIntegerSet &set, intptr_t startIndex = 0,
475  intptr_t length = -1, intptr_t step = 1)
476  : Sliceable(startIndex,
477  length == -1 ? mlirIntegerSetGetNumConstraints(set) : length,
478  step),
479  set(set) {}
480 
481 private:
482  /// Give the parent CRTP class access to hook implementations below.
483  friend class Sliceable<PyIntegerSetConstraintList, PyIntegerSetConstraint>;
484 
485  intptr_t getRawNumElements() { return mlirIntegerSetGetNumConstraints(set); }
486 
487  PyIntegerSetConstraint getRawElement(intptr_t pos) {
488  return PyIntegerSetConstraint(set, pos);
489  }
490 
491  PyIntegerSetConstraintList slice(intptr_t startIndex, intptr_t length,
492  intptr_t step) {
493  return PyIntegerSetConstraintList(set, startIndex, length, step);
494  }
495 
496  PyIntegerSet set;
497 };
498 } // namespace
499 
500 bool PyIntegerSet::operator==(const PyIntegerSet &other) const {
501  return mlirIntegerSetEqual(integerSet, other.integerSet);
502 }
503 
505  return py::reinterpret_steal<py::object>(
507 }
508 
510  MlirIntegerSet rawIntegerSet = mlirPythonCapsuleToIntegerSet(capsule.ptr());
511  if (mlirIntegerSetIsNull(rawIntegerSet))
512  throw py::error_already_set();
513  return PyIntegerSet(
515  rawIntegerSet);
516 }
517 
518 void mlir::python::populateIRAffine(py::module &m) {
519  //----------------------------------------------------------------------------
520  // Mapping of PyAffineExpr and derived classes.
521  //----------------------------------------------------------------------------
522  py::class_<PyAffineExpr>(m, "AffineExpr", py::module_local())
523  .def_property_readonly(MLIR_PYTHON_CAPI_PTR_ATTR,
526  .def("__add__", &PyAffineAddExpr::get)
527  .def("__add__", &PyAffineAddExpr::getRHSConstant)
528  .def("__radd__", &PyAffineAddExpr::getRHSConstant)
529  .def("__mul__", &PyAffineMulExpr::get)
530  .def("__mul__", &PyAffineMulExpr::getRHSConstant)
531  .def("__rmul__", &PyAffineMulExpr::getRHSConstant)
532  .def("__mod__", &PyAffineModExpr::get)
533  .def("__mod__", &PyAffineModExpr::getRHSConstant)
534  .def("__rmod__",
535  [](PyAffineExpr &self, intptr_t other) {
536  return PyAffineModExpr::get(
537  PyAffineConstantExpr::get(other, *self.getContext().get()),
538  self);
539  })
540  .def("__sub__",
541  [](PyAffineExpr &self, PyAffineExpr &other) {
542  auto negOne =
544  return PyAffineAddExpr::get(self,
545  PyAffineMulExpr::get(negOne, other));
546  })
547  .def("__sub__",
548  [](PyAffineExpr &self, intptr_t other) {
549  return PyAffineAddExpr::get(
550  self,
551  PyAffineConstantExpr::get(-other, *self.getContext().get()));
552  })
553  .def("__rsub__",
554  [](PyAffineExpr &self, intptr_t other) {
555  return PyAffineAddExpr::getLHSConstant(
556  other, PyAffineMulExpr::getLHSConstant(-1, self));
557  })
558  .def("__eq__", [](PyAffineExpr &self,
559  PyAffineExpr &other) { return self == other; })
560  .def("__eq__",
561  [](PyAffineExpr &self, py::object &other) { return false; })
562  .def("__str__",
563  [](PyAffineExpr &self) {
564  PyPrintAccumulator printAccum;
565  mlirAffineExprPrint(self, printAccum.getCallback(),
566  printAccum.getUserData());
567  return printAccum.join();
568  })
569  .def("__repr__",
570  [](PyAffineExpr &self) {
571  PyPrintAccumulator printAccum;
572  printAccum.parts.append("AffineExpr(");
573  mlirAffineExprPrint(self, printAccum.getCallback(),
574  printAccum.getUserData());
575  printAccum.parts.append(")");
576  return printAccum.join();
577  })
578  .def("__hash__",
579  [](PyAffineExpr &self) {
580  return static_cast<size_t>(llvm::hash_value(self.get().ptr));
581  })
582  .def_property_readonly(
583  "context",
584  [](PyAffineExpr &self) { return self.getContext().getObject(); })
585  .def("compose",
586  [](PyAffineExpr &self, PyAffineMap &other) {
587  return PyAffineExpr(self.getContext(),
588  mlirAffineExprCompose(self, other));
589  })
590  .def_static(
591  "get_add", &PyAffineAddExpr::get,
592  "Gets an affine expression containing a sum of two expressions.")
593  .def_static("get_add", &PyAffineAddExpr::getLHSConstant,
594  "Gets an affine expression containing a sum of a constant "
595  "and another expression.")
596  .def_static("get_add", &PyAffineAddExpr::getRHSConstant,
597  "Gets an affine expression containing a sum of an expression "
598  "and a constant.")
599  .def_static(
600  "get_mul", &PyAffineMulExpr::get,
601  "Gets an affine expression containing a product of two expressions.")
602  .def_static("get_mul", &PyAffineMulExpr::getLHSConstant,
603  "Gets an affine expression containing a product of a "
604  "constant and another expression.")
605  .def_static("get_mul", &PyAffineMulExpr::getRHSConstant,
606  "Gets an affine expression containing a product of an "
607  "expression and a constant.")
608  .def_static("get_mod", &PyAffineModExpr::get,
609  "Gets an affine expression containing the modulo of dividing "
610  "one expression by another.")
611  .def_static("get_mod", &PyAffineModExpr::getLHSConstant,
612  "Gets a semi-affine expression containing the modulo of "
613  "dividing a constant by an expression.")
614  .def_static("get_mod", &PyAffineModExpr::getRHSConstant,
615  "Gets an affine expression containing the module of dividing"
616  "an expression by a constant.")
617  .def_static("get_floor_div", &PyAffineFloorDivExpr::get,
618  "Gets an affine expression containing the rounded-down "
619  "result of dividing one expression by another.")
620  .def_static("get_floor_div", &PyAffineFloorDivExpr::getLHSConstant,
621  "Gets a semi-affine expression containing the rounded-down "
622  "result of dividing a constant by an expression.")
623  .def_static("get_floor_div", &PyAffineFloorDivExpr::getRHSConstant,
624  "Gets an affine expression containing the rounded-down "
625  "result of dividing an expression by a constant.")
626  .def_static("get_ceil_div", &PyAffineCeilDivExpr::get,
627  "Gets an affine expression containing the rounded-up result "
628  "of dividing one expression by another.")
629  .def_static("get_ceil_div", &PyAffineCeilDivExpr::getLHSConstant,
630  "Gets a semi-affine expression containing the rounded-up "
631  "result of dividing a constant by an expression.")
632  .def_static("get_ceil_div", &PyAffineCeilDivExpr::getRHSConstant,
633  "Gets an affine expression containing the rounded-up result "
634  "of dividing an expression by a constant.")
635  .def_static("get_constant", &PyAffineConstantExpr::get, py::arg("value"),
636  py::arg("context") = py::none(),
637  "Gets a constant affine expression with the given value.")
638  .def_static(
639  "get_dim", &PyAffineDimExpr::get, py::arg("position"),
640  py::arg("context") = py::none(),
641  "Gets an affine expression of a dimension at the given position.")
642  .def_static(
643  "get_symbol", &PyAffineSymbolExpr::get, py::arg("position"),
644  py::arg("context") = py::none(),
645  "Gets an affine expression of a symbol at the given position.")
646  .def(
647  "dump", [](PyAffineExpr &self) { mlirAffineExprDump(self); },
649  PyAffineConstantExpr::bind(m);
650  PyAffineDimExpr::bind(m);
651  PyAffineSymbolExpr::bind(m);
652  PyAffineBinaryExpr::bind(m);
653  PyAffineAddExpr::bind(m);
654  PyAffineMulExpr::bind(m);
655  PyAffineModExpr::bind(m);
656  PyAffineFloorDivExpr::bind(m);
657  PyAffineCeilDivExpr::bind(m);
658 
659  //----------------------------------------------------------------------------
660  // Mapping of PyAffineMap.
661  //----------------------------------------------------------------------------
662  py::class_<PyAffineMap>(m, "AffineMap", py::module_local())
663  .def_property_readonly(MLIR_PYTHON_CAPI_PTR_ATTR,
666  .def("__eq__",
667  [](PyAffineMap &self, PyAffineMap &other) { return self == other; })
668  .def("__eq__", [](PyAffineMap &self, py::object &other) { return false; })
669  .def("__str__",
670  [](PyAffineMap &self) {
671  PyPrintAccumulator printAccum;
672  mlirAffineMapPrint(self, printAccum.getCallback(),
673  printAccum.getUserData());
674  return printAccum.join();
675  })
676  .def("__repr__",
677  [](PyAffineMap &self) {
678  PyPrintAccumulator printAccum;
679  printAccum.parts.append("AffineMap(");
680  mlirAffineMapPrint(self, printAccum.getCallback(),
681  printAccum.getUserData());
682  printAccum.parts.append(")");
683  return printAccum.join();
684  })
685  .def("__hash__",
686  [](PyAffineMap &self) {
687  return static_cast<size_t>(llvm::hash_value(self.get().ptr));
688  })
689  .def_static("compress_unused_symbols",
690  [](py::list affineMaps, DefaultingPyMlirContext context) {
692  pyListToVector<PyAffineMap, MlirAffineMap>(
693  affineMaps, maps, "attempting to create an AffineMap");
694  std::vector<MlirAffineMap> compressed(affineMaps.size());
695  auto populate = [](void *result, intptr_t idx,
696  MlirAffineMap m) {
697  static_cast<MlirAffineMap *>(result)[idx] = (m);
698  };
700  maps.data(), maps.size(), compressed.data(), populate);
701  std::vector<PyAffineMap> res;
702  res.reserve(compressed.size());
703  for (auto m : compressed)
704  res.emplace_back(context->getRef(), m);
705  return res;
706  })
707  .def_property_readonly(
708  "context",
709  [](PyAffineMap &self) { return self.getContext().getObject(); },
710  "Context that owns the Affine Map")
711  .def(
712  "dump", [](PyAffineMap &self) { mlirAffineMapDump(self); },
714  .def_static(
715  "get",
716  [](intptr_t dimCount, intptr_t symbolCount, py::list exprs,
717  DefaultingPyMlirContext context) {
718  SmallVector<MlirAffineExpr> affineExprs;
719  pyListToVector<PyAffineExpr, MlirAffineExpr>(
720  exprs, affineExprs, "attempting to create an AffineMap");
721  MlirAffineMap map =
722  mlirAffineMapGet(context->get(), dimCount, symbolCount,
723  affineExprs.size(), affineExprs.data());
724  return PyAffineMap(context->getRef(), map);
725  },
726  py::arg("dim_count"), py::arg("symbol_count"), py::arg("exprs"),
727  py::arg("context") = py::none(),
728  "Gets a map with the given expressions as results.")
729  .def_static(
730  "get_constant",
731  [](intptr_t value, DefaultingPyMlirContext context) {
732  MlirAffineMap affineMap =
733  mlirAffineMapConstantGet(context->get(), value);
734  return PyAffineMap(context->getRef(), affineMap);
735  },
736  py::arg("value"), py::arg("context") = py::none(),
737  "Gets an affine map with a single constant result")
738  .def_static(
739  "get_empty",
740  [](DefaultingPyMlirContext context) {
741  MlirAffineMap affineMap = mlirAffineMapEmptyGet(context->get());
742  return PyAffineMap(context->getRef(), affineMap);
743  },
744  py::arg("context") = py::none(), "Gets an empty affine map.")
745  .def_static(
746  "get_identity",
747  [](intptr_t nDims, DefaultingPyMlirContext context) {
748  MlirAffineMap affineMap =
749  mlirAffineMapMultiDimIdentityGet(context->get(), nDims);
750  return PyAffineMap(context->getRef(), affineMap);
751  },
752  py::arg("n_dims"), py::arg("context") = py::none(),
753  "Gets an identity map with the given number of dimensions.")
754  .def_static(
755  "get_minor_identity",
756  [](intptr_t nDims, intptr_t nResults,
757  DefaultingPyMlirContext context) {
758  MlirAffineMap affineMap =
759  mlirAffineMapMinorIdentityGet(context->get(), nDims, nResults);
760  return PyAffineMap(context->getRef(), affineMap);
761  },
762  py::arg("n_dims"), py::arg("n_results"),
763  py::arg("context") = py::none(),
764  "Gets a minor identity map with the given number of dimensions and "
765  "results.")
766  .def_static(
767  "get_permutation",
768  [](std::vector<unsigned> permutation,
769  DefaultingPyMlirContext context) {
770  if (!isPermutation(permutation))
771  throw py::cast_error("Invalid permutation when attempting to "
772  "create an AffineMap");
773  MlirAffineMap affineMap = mlirAffineMapPermutationGet(
774  context->get(), permutation.size(), permutation.data());
775  return PyAffineMap(context->getRef(), affineMap);
776  },
777  py::arg("permutation"), py::arg("context") = py::none(),
778  "Gets an affine map that permutes its inputs.")
779  .def(
780  "get_submap",
781  [](PyAffineMap &self, std::vector<intptr_t> &resultPos) {
782  intptr_t numResults = mlirAffineMapGetNumResults(self);
783  for (intptr_t pos : resultPos) {
784  if (pos < 0 || pos >= numResults)
785  throw py::value_error("result position out of bounds");
786  }
787  MlirAffineMap affineMap = mlirAffineMapGetSubMap(
788  self, resultPos.size(), resultPos.data());
789  return PyAffineMap(self.getContext(), affineMap);
790  },
791  py::arg("result_positions"))
792  .def(
793  "get_major_submap",
794  [](PyAffineMap &self, intptr_t nResults) {
795  if (nResults >= mlirAffineMapGetNumResults(self))
796  throw py::value_error("number of results out of bounds");
797  MlirAffineMap affineMap =
798  mlirAffineMapGetMajorSubMap(self, nResults);
799  return PyAffineMap(self.getContext(), affineMap);
800  },
801  py::arg("n_results"))
802  .def(
803  "get_minor_submap",
804  [](PyAffineMap &self, intptr_t nResults) {
805  if (nResults >= mlirAffineMapGetNumResults(self))
806  throw py::value_error("number of results out of bounds");
807  MlirAffineMap affineMap =
808  mlirAffineMapGetMinorSubMap(self, nResults);
809  return PyAffineMap(self.getContext(), affineMap);
810  },
811  py::arg("n_results"))
812  .def(
813  "replace",
814  [](PyAffineMap &self, PyAffineExpr &expression,
815  PyAffineExpr &replacement, intptr_t numResultDims,
816  intptr_t numResultSyms) {
817  MlirAffineMap affineMap = mlirAffineMapReplace(
818  self, expression, replacement, numResultDims, numResultSyms);
819  return PyAffineMap(self.getContext(), affineMap);
820  },
821  py::arg("expr"), py::arg("replacement"), py::arg("n_result_dims"),
822  py::arg("n_result_syms"))
823  .def_property_readonly(
824  "is_permutation",
825  [](PyAffineMap &self) { return mlirAffineMapIsPermutation(self); })
826  .def_property_readonly("is_projected_permutation",
827  [](PyAffineMap &self) {
829  })
830  .def_property_readonly(
831  "n_dims",
832  [](PyAffineMap &self) { return mlirAffineMapGetNumDims(self); })
833  .def_property_readonly(
834  "n_inputs",
835  [](PyAffineMap &self) { return mlirAffineMapGetNumInputs(self); })
836  .def_property_readonly(
837  "n_symbols",
838  [](PyAffineMap &self) { return mlirAffineMapGetNumSymbols(self); })
839  .def_property_readonly("results", [](PyAffineMap &self) {
840  return PyAffineMapExprList(self);
841  });
842  PyAffineMapExprList::bind(m);
843 
844  //----------------------------------------------------------------------------
845  // Mapping of PyIntegerSet.
846  //----------------------------------------------------------------------------
847  py::class_<PyIntegerSet>(m, "IntegerSet", py::module_local())
848  .def_property_readonly(MLIR_PYTHON_CAPI_PTR_ATTR,
851  .def("__eq__", [](PyIntegerSet &self,
852  PyIntegerSet &other) { return self == other; })
853  .def("__eq__", [](PyIntegerSet &self, py::object other) { return false; })
854  .def("__str__",
855  [](PyIntegerSet &self) {
856  PyPrintAccumulator printAccum;
857  mlirIntegerSetPrint(self, printAccum.getCallback(),
858  printAccum.getUserData());
859  return printAccum.join();
860  })
861  .def("__repr__",
862  [](PyIntegerSet &self) {
863  PyPrintAccumulator printAccum;
864  printAccum.parts.append("IntegerSet(");
865  mlirIntegerSetPrint(self, printAccum.getCallback(),
866  printAccum.getUserData());
867  printAccum.parts.append(")");
868  return printAccum.join();
869  })
870  .def("__hash__",
871  [](PyIntegerSet &self) {
872  return static_cast<size_t>(llvm::hash_value(self.get().ptr));
873  })
874  .def_property_readonly(
875  "context",
876  [](PyIntegerSet &self) { return self.getContext().getObject(); })
877  .def(
878  "dump", [](PyIntegerSet &self) { mlirIntegerSetDump(self); },
880  .def_static(
881  "get",
882  [](intptr_t numDims, intptr_t numSymbols, py::list exprs,
883  std::vector<bool> eqFlags, DefaultingPyMlirContext context) {
884  if (exprs.size() != eqFlags.size())
885  throw py::value_error(
886  "Expected the number of constraints to match "
887  "that of equality flags");
888  if (exprs.empty())
889  throw py::value_error("Expected non-empty list of constraints");
890 
891  // Copy over to a SmallVector because std::vector has a
892  // specialization for booleans that packs data and does not
893  // expose a `bool *`.
894  SmallVector<bool, 8> flags(eqFlags.begin(), eqFlags.end());
895 
896  SmallVector<MlirAffineExpr> affineExprs;
897  pyListToVector<PyAffineExpr>(exprs, affineExprs,
898  "attempting to create an IntegerSet");
899  MlirIntegerSet set = mlirIntegerSetGet(
900  context->get(), numDims, numSymbols, exprs.size(),
901  affineExprs.data(), flags.data());
902  return PyIntegerSet(context->getRef(), set);
903  },
904  py::arg("num_dims"), py::arg("num_symbols"), py::arg("exprs"),
905  py::arg("eq_flags"), py::arg("context") = py::none())
906  .def_static(
907  "get_empty",
908  [](intptr_t numDims, intptr_t numSymbols,
909  DefaultingPyMlirContext context) {
910  MlirIntegerSet set =
911  mlirIntegerSetEmptyGet(context->get(), numDims, numSymbols);
912  return PyIntegerSet(context->getRef(), set);
913  },
914  py::arg("num_dims"), py::arg("num_symbols"),
915  py::arg("context") = py::none())
916  .def(
917  "get_replaced",
918  [](PyIntegerSet &self, py::list dimExprs, py::list symbolExprs,
919  intptr_t numResultDims, intptr_t numResultSymbols) {
920  if (static_cast<intptr_t>(dimExprs.size()) !=
922  throw py::value_error(
923  "Expected the number of dimension replacement expressions "
924  "to match that of dimensions");
925  if (static_cast<intptr_t>(symbolExprs.size()) !=
927  throw py::value_error(
928  "Expected the number of symbol replacement expressions "
929  "to match that of symbols");
930 
931  SmallVector<MlirAffineExpr> dimAffineExprs, symbolAffineExprs;
932  pyListToVector<PyAffineExpr>(
933  dimExprs, dimAffineExprs,
934  "attempting to create an IntegerSet by replacing dimensions");
935  pyListToVector<PyAffineExpr>(
936  symbolExprs, symbolAffineExprs,
937  "attempting to create an IntegerSet by replacing symbols");
938  MlirIntegerSet set = mlirIntegerSetReplaceGet(
939  self, dimAffineExprs.data(), symbolAffineExprs.data(),
940  numResultDims, numResultSymbols);
941  return PyIntegerSet(self.getContext(), set);
942  },
943  py::arg("dim_exprs"), py::arg("symbol_exprs"),
944  py::arg("num_result_dims"), py::arg("num_result_symbols"))
945  .def_property_readonly("is_canonical_empty",
946  [](PyIntegerSet &self) {
947  return mlirIntegerSetIsCanonicalEmpty(self);
948  })
949  .def_property_readonly(
950  "n_dims",
951  [](PyIntegerSet &self) { return mlirIntegerSetGetNumDims(self); })
952  .def_property_readonly(
953  "n_symbols",
954  [](PyIntegerSet &self) { return mlirIntegerSetGetNumSymbols(self); })
955  .def_property_readonly(
956  "n_inputs",
957  [](PyIntegerSet &self) { return mlirIntegerSetGetNumInputs(self); })
958  .def_property_readonly("n_equalities",
959  [](PyIntegerSet &self) {
960  return mlirIntegerSetGetNumEqualities(self);
961  })
962  .def_property_readonly("n_inequalities",
963  [](PyIntegerSet &self) {
965  })
966  .def_property_readonly("constraints", [](PyIntegerSet &self) {
967  return PyIntegerSetConstraintList(self);
968  });
969  PyIntegerSetConstraint::bind(m);
970  PyIntegerSetConstraintList::bind(m);
971 }
static void pyListToVector(const py::list &list, llvm::SmallVectorImpl< CType > &result, StringRef action)
Attempts to populate result with the content of list casted to the appropriate type (Python and C typ...
Definition: IRAffine.cpp:49
static const char kDumpDocstring[]
Definition: IRAffine.cpp:41
static bool isPermutation(std::vector< PermutationTy > permutation)
Definition: IRAffine.cpp:71
static MLIRContext * getContext(OpFoldResult val)
static MlirIntegerSet mlirPythonCapsuleToIntegerSet(PyObject *capsule)
Extracts an MlirIntegerSet from a capsule as produced from mlirPythonIntegerSetToCapsule.
Definition: Interop.h:414
#define MLIR_PYTHON_CAPI_PTR_ATTR
Attribute on MLIR Python objects that expose their C-API pointer.
Definition: Interop.h:97
static PyObject * mlirPythonIntegerSetToCapsule(MlirIntegerSet integerSet)
Creates a capsule object encapsulating the raw C-API MlirIntegerSet.
Definition: Interop.h:405
static MlirAffineMap mlirPythonCapsuleToAffineMap(PyObject *capsule)
Extracts an MlirAffineMap from a capsule as produced from mlirPythonAffineMapToCapsule.
Definition: Interop.h:395
#define MLIR_PYTHON_CAPI_FACTORY_ATTR
Attribute on MLIR Python objects that exposes a factory function for constructing the corresponding P...
Definition: Interop.h:110
static PyObject * mlirPythonAffineExprToCapsule(MlirAffineExpr expr)
Creates a capsule object encapsulating the raw C-API MlirAffineExpr.
Definition: Interop.h:161
static PyObject * mlirPythonAffineMapToCapsule(MlirAffineMap affineMap)
Creates a capsule object encapsulating the raw C-API MlirAffineMap.
Definition: Interop.h:386
static MlirAffineExpr mlirPythonCapsuleToAffineExpr(PyObject *capsule)
Extracts an MlirAffineExpr from a capsule as produced from mlirPythonAffineExprToCapsule.
Definition: Interop.h:170
A CRTP base class for pseudo-containers willing to support Python-type slicing access on top of index...
Definition: PybindUtils.h:209
PyMlirContextRef & getContext()
Accesses the context reference.
Definition: IRModule.h:311
Used in function arguments when None should resolve to the current context manager set instance.
Definition: IRModule.h:292
ReferrentTy * get() const
Definition: PybindUtils.h:47
Wrapper around MlirAffineExpr. Affine expressions are owned by the context.
Definition: IRModule.h:1163
static PyAffineExpr createFromCapsule(pybind11::object capsule)
Creates a PyAffineExpr from the MlirAffineExpr wrapped by a capsule.
Definition: IRAffine.cpp:373
pybind11::object getCapsule()
Gets a capsule wrapping the void* within the MlirAffineExpr.
Definition: IRAffine.cpp:368
PyAffineExpr(PyMlirContextRef contextRef, MlirAffineExpr affineExpr)
Definition: IRModule.h:1165
bool operator==(const PyAffineExpr &other) const
Definition: IRAffine.cpp:364
pybind11::object getCapsule()
Gets a capsule wrapping the void* within the MlirAffineMap.
Definition: IRAffine.cpp:427
PyAffineMap(PyMlirContextRef contextRef, MlirAffineMap affineMap)
Definition: IRModule.h:1192
bool operator==(const PyAffineMap &other) const
Definition: IRAffine.cpp:423
static PyAffineMap createFromCapsule(pybind11::object capsule)
Creates a PyAffineMap from the MlirAffineMap wrapped by a capsule.
Definition: IRAffine.cpp:431
static PyIntegerSet createFromCapsule(pybind11::object capsule)
Creates a PyIntegerSet from the MlirAffineMap wrapped by a capsule.
Definition: IRAffine.cpp:509
bool operator==(const PyIntegerSet &other) const
Definition: IRAffine.cpp:500
pybind11::object getCapsule()
Gets a capsule wrapping the void* within the MlirIntegerSet.
Definition: IRAffine.cpp:504
PyIntegerSet(PyMlirContextRef contextRef, MlirIntegerSet integerSet)
Definition: IRModule.h:1213
static PyMlirContextRef forContext(MlirContext context)
Returns a context reference for the singleton PyMlirContext wrapper for the given context.
Definition: IRCore.cpp:633
Wrapper around the generic MlirType.
Definition: IRModule.h:880
MLIR_CAPI_EXPORTED bool mlirAffineExprIsAConstant(MlirAffineExpr affineExpr)
Checks whether the given affine expression is a constant expression.
Definition: AffineExpr.cpp:100
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineBinaryOpExprGetLHS(MlirAffineExpr affineExpr)
Returns the left hand side affine expression of the given affine binary operation expression.
Definition: AffineExpr.cpp:187
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineMulExprGet(MlirAffineExpr lhs, MlirAffineExpr rhs)
Creates an affine mul expression with 'lhs' and 'rhs'.
Definition: AffineExpr.cpp:133
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineConstantExprGet(MlirContext ctx, int64_t constant)
Creates an affine constant expression with 'constant' in the context.
Definition: AffineExpr.cpp:104
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineBinaryOpExprGetRHS(MlirAffineExpr affineExpr)
Returns the right hand side affine expression of the given affine binary operation expression.
Definition: AffineExpr.cpp:191
MLIR_CAPI_EXPORTED void mlirAffineExprPrint(MlirAffineExpr affineExpr, MlirStringCallback callback, void *userData)
Prints an affine expression by sending chunks of the string representation and forwarding userData to...
Definition: AffineExpr.cpp:28
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineAddExprGet(MlirAffineExpr lhs, MlirAffineExpr rhs)
Creates an affine add expression with 'lhs' and 'rhs'.
Definition: AffineExpr.cpp:120
MLIR_CAPI_EXPORTED intptr_t mlirAffineDimExprGetPosition(MlirAffineExpr affineExpr)
Returns the position of the given affine dimension expression.
Definition: AffineExpr.cpp:76
MLIR_CAPI_EXPORTED bool mlirAffineExprIsAAdd(MlirAffineExpr affineExpr)
Checks whether the given affine expression is an add expression.
Definition: AffineExpr.cpp:116
MLIR_CAPI_EXPORTED bool mlirAffineExprIsABinary(MlirAffineExpr affineExpr)
Checks whether the given affine expression is binary.
Definition: AffineExpr.cpp:183
MLIR_CAPI_EXPORTED bool mlirAffineExprIsAMul(MlirAffineExpr affineExpr)
Checks whether the given affine expression is an mul expression.
Definition: AffineExpr.cpp:129
MLIR_CAPI_EXPORTED int64_t mlirAffineConstantExprGetValue(MlirAffineExpr affineExpr)
Returns the value of the given affine constant expression.
Definition: AffineExpr.cpp:108
MLIR_CAPI_EXPORTED MlirContext mlirAffineExprGetContext(MlirAffineExpr affineExpr)
Gets the context that owns the affine expression.
Definition: AffineExpr.cpp:20
MLIR_CAPI_EXPORTED bool mlirAffineExprIsASymbol(MlirAffineExpr affineExpr)
Checks whether the given affine expression is a symbol expression.
Definition: AffineExpr.cpp:84
MLIR_CAPI_EXPORTED bool mlirAffineExprIsACeilDiv(MlirAffineExpr affineExpr)
Checks whether the given affine expression is an ceildiv expression.
Definition: AffineExpr.cpp:169
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineFloorDivExprGet(MlirAffineExpr lhs, MlirAffineExpr rhs)
Creates an affine floordiv expression with 'lhs' and 'rhs'.
Definition: AffineExpr.cpp:159
MLIR_CAPI_EXPORTED bool mlirAffineExprIsAFloorDiv(MlirAffineExpr affineExpr)
Checks whether the given affine expression is an floordiv expression.
Definition: AffineExpr.cpp:155
MLIR_CAPI_EXPORTED bool mlirAffineExprEqual(MlirAffineExpr lhs, MlirAffineExpr rhs)
Returns true if the two affine expressions are equal.
Definition: AffineExpr.cpp:24
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineDimExprGet(MlirContext ctx, intptr_t position)
Creates an affine dimension expression with 'position' in the context.
Definition: AffineExpr.cpp:72
MLIR_CAPI_EXPORTED bool mlirAffineExprIsAMod(MlirAffineExpr affineExpr)
Checks whether the given affine expression is an mod expression.
Definition: AffineExpr.cpp:142
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineModExprGet(MlirAffineExpr lhs, MlirAffineExpr rhs)
Creates an affine mod expression with 'lhs' and 'rhs'.
Definition: AffineExpr.cpp:146
MLIR_CAPI_EXPORTED void mlirAffineExprDump(MlirAffineExpr affineExpr)
Prints the affine expression to the standard error stream.
Definition: AffineExpr.cpp:34
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineExprCompose(MlirAffineExpr affineExpr, struct MlirAffineMap affineMap)
Composes the given map with the given expression.
Definition: AffineExpr.cpp:59
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineSymbolExprGet(MlirContext ctx, intptr_t position)
Creates an affine symbol expression with 'position' in the context.
Definition: AffineExpr.cpp:88
static bool mlirAffineExprIsNull(MlirAffineExpr affineExpr)
Returns true if the given affine expression is a null expression.
Definition: AffineExpr.h:54
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineCeilDivExprGet(MlirAffineExpr lhs, MlirAffineExpr rhs)
Creates an affine ceildiv expression with 'lhs' and 'rhs'.
Definition: AffineExpr.cpp:173
MLIR_CAPI_EXPORTED intptr_t mlirAffineSymbolExprGetPosition(MlirAffineExpr affineExpr)
Returns the position of the given affine symbol expression.
Definition: AffineExpr.cpp:92
MLIR_CAPI_EXPORTED bool mlirAffineExprIsADim(MlirAffineExpr affineExpr)
Checks whether the given affine expression is a dimension expression.
Definition: AffineExpr.cpp:68
MLIR_CAPI_EXPORTED bool mlirAffineMapEqual(MlirAffineMap a1, MlirAffineMap a2)
Checks if two affine maps are equal.
Definition: AffineMap.cpp:25
MLIR_CAPI_EXPORTED MlirAffineMap mlirAffineMapEmptyGet(MlirContext ctx)
Creates a zero result affine map with no dimensions or symbols in the context.
Definition: AffineMap.cpp:37
MLIR_CAPI_EXPORTED MlirAffineMap mlirAffineMapGet(MlirContext ctx, intptr_t dimCount, intptr_t symbolCount, intptr_t nAffineExprs, MlirAffineExpr *affineExprs)
Creates an affine map with results defined by the given list of affine expressions.
Definition: AffineMap.cpp:46
MLIR_CAPI_EXPORTED intptr_t mlirAffineMapGetNumDims(MlirAffineMap affineMap)
Returns the number of dimensions of the given affine map.
Definition: AffineMap.cpp:94
MLIR_CAPI_EXPORTED MlirContext mlirAffineMapGetContext(MlirAffineMap affineMap)
Gets the context that the given affine map was created with.
Definition: AffineMap.cpp:21
MLIR_CAPI_EXPORTED MlirAffineExpr mlirAffineMapGetResult(MlirAffineMap affineMap, intptr_t pos)
Returns the result at the given position.
Definition: AffineMap.cpp:106
MLIR_CAPI_EXPORTED void mlirAffineMapDump(MlirAffineMap affineMap)
Prints the affine map to the standard error stream.
Definition: AffineMap.cpp:35
MLIR_CAPI_EXPORTED MlirAffineMap mlirAffineMapConstantGet(MlirContext ctx, int64_t val)
Creates a single constant result affine map in the context.
Definition: AffineMap.cpp:54
MLIR_CAPI_EXPORTED MlirAffineMap mlirAffineMapMultiDimIdentityGet(MlirContext ctx, intptr_t numDims)
Creates an affine map with 'numDims' identity in the context.
Definition: AffineMap.cpp:58
MLIR_CAPI_EXPORTED void mlirAffineMapPrint(MlirAffineMap affineMap, MlirStringCallback callback, void *userData)
Prints an affine map by sending chunks of the string representation and forwarding userData tocallbac...
Definition: AffineMap.cpp:29
MLIR_CAPI_EXPORTED MlirAffineMap mlirAffineMapGetMinorSubMap(MlirAffineMap affineMap, intptr_t numResults)
Returns the affine map consisting of the most minor numResults results.
Definition: AffineMap.cpp:136
MLIR_CAPI_EXPORTED MlirAffineMap mlirAffineMapGetMajorSubMap(MlirAffineMap affineMap, intptr_t numResults)
Returns the affine map consisting of the most major numResults results.
Definition: AffineMap.cpp:131
MLIR_CAPI_EXPORTED MlirAffineMap mlirAffineMapGetSubMap(MlirAffineMap affineMap, intptr_t size, intptr_t *resultPos)
Returns the affine map consisting of the resultPos subset.
Definition: AffineMap.cpp:122
MLIR_CAPI_EXPORTED intptr_t mlirAffineMapGetNumResults(MlirAffineMap affineMap)
Returns the number of results of the given affine map.
Definition: AffineMap.cpp:102
MLIR_CAPI_EXPORTED intptr_t mlirAffineMapGetNumSymbols(MlirAffineMap affineMap)
Returns the number of symbols of the given affine map.
Definition: AffineMap.cpp:98
MLIR_CAPI_EXPORTED MlirAffineMap mlirAffineMapPermutationGet(MlirContext ctx, intptr_t size, unsigned *permutation)
Creates an affine map with a permutation expression and its size in the context.
Definition: AffineMap.cpp:68
MLIR_CAPI_EXPORTED void mlirAffineMapCompressUnusedSymbols(MlirAffineMap *affineMaps, intptr_t size, void *result, void(*populateResult)(void *res, intptr_t idx, MlirAffineMap m))
Returns the simplified affine map resulting from dropping the symbols that do not appear in any of th...
Definition: AffineMap.cpp:150
MLIR_CAPI_EXPORTED bool mlirAffineMapIsProjectedPermutation(MlirAffineMap affineMap)
Checks whether the given affine map represents a subset of a symbol-less permutation map.
Definition: AffineMap.cpp:114
MLIR_CAPI_EXPORTED intptr_t mlirAffineMapGetNumInputs(MlirAffineMap affineMap)
Returns the number of inputs (dimensions + symbols) of the given affine map.
Definition: AffineMap.cpp:110
MLIR_CAPI_EXPORTED bool mlirAffineMapIsPermutation(MlirAffineMap affineMap)
Checks whether the given affine map represents a symbol-less permutation map.
Definition: AffineMap.cpp:118
MLIR_CAPI_EXPORTED MlirAffineMap mlirAffineMapReplace(MlirAffineMap affineMap, MlirAffineExpr expression, MlirAffineExpr replacement, intptr_t numResultDims, intptr_t numResultSyms)
Apply AffineExpr::replace(map) to each of the results and return a new new AffineMap with the new res...
Definition: AffineMap.cpp:141
MLIR_CAPI_EXPORTED MlirAffineMap mlirAffineMapMinorIdentityGet(MlirContext ctx, intptr_t dims, intptr_t results)
Creates an identity affine map on the most minor dimensions in the context.
Definition: AffineMap.cpp:63
static bool mlirAffineMapIsNull(MlirAffineMap affineMap)
Checks whether an affine map is null.
Definition: AffineMap.h:47
MLIR_CAPI_EXPORTED intptr_t mlirIntegerSetGetNumConstraints(MlirIntegerSet set)
Returns the number of constraints (equalities + inequalities) in the given set.
Definition: IntegerSet.cpp:85
MLIR_CAPI_EXPORTED MlirAffineExpr mlirIntegerSetGetConstraint(MlirIntegerSet set, intptr_t pos)
Returns pos-th constraint of the set.
Definition: IntegerSet.cpp:97
MLIR_CAPI_EXPORTED bool mlirIntegerSetIsCanonicalEmpty(MlirIntegerSet set)
Checks whether the given set is a canonical empty set, e.g., the set returned by mlirIntegerSetEmptyG...
Definition: IntegerSet.cpp:69
MLIR_CAPI_EXPORTED intptr_t mlirIntegerSetGetNumInputs(MlirIntegerSet set)
Returns the number of inputs (dimensions + symbols) in the given set.
Definition: IntegerSet.cpp:81
MLIR_CAPI_EXPORTED MlirIntegerSet mlirIntegerSetEmptyGet(MlirContext context, intptr_t numDims, intptr_t numSymbols)
Gets or creates a new canonically empty integer set with the give number of dimensions and symbols in...
Definition: IntegerSet.cpp:35
MLIR_CAPI_EXPORTED MlirIntegerSet mlirIntegerSetGet(MlirContext context, intptr_t numDims, intptr_t numSymbols, intptr_t numConstraints, const MlirAffineExpr *constraints, const bool *eqFlags)
Gets or creates a new integer set in the given context.
Definition: IntegerSet.cpp:42
MLIR_CAPI_EXPORTED intptr_t mlirIntegerSetGetNumEqualities(MlirIntegerSet set)
Returns the number of equalities in the given set.
Definition: IntegerSet.cpp:89
MLIR_CAPI_EXPORTED intptr_t mlirIntegerSetGetNumDims(MlirIntegerSet set)
Returns the number of dimensions in the given set.
Definition: IntegerSet.cpp:73
MLIR_CAPI_EXPORTED MlirIntegerSet mlirIntegerSetReplaceGet(MlirIntegerSet set, const MlirAffineExpr *dimReplacements, const MlirAffineExpr *symbolReplacements, intptr_t numResultDims, intptr_t numResultSymbols)
Gets or creates a new integer set in which the values and dimensions of the given set are replaced wi...
Definition: IntegerSet.cpp:56
MLIR_CAPI_EXPORTED intptr_t mlirIntegerSetGetNumSymbols(MlirIntegerSet set)
Returns the number of symbols in the given set.
Definition: IntegerSet.cpp:77
static bool mlirIntegerSetIsNull(MlirIntegerSet set)
Checks whether an integer set is a null object.
Definition: IntegerSet.h:46
MLIR_CAPI_EXPORTED MlirContext mlirIntegerSetGetContext(MlirIntegerSet set)
Gets the context in which the given integer set lives.
Definition: IntegerSet.cpp:19
MLIR_CAPI_EXPORTED void mlirIntegerSetPrint(MlirIntegerSet set, MlirStringCallback callback, void *userData)
Prints an integer set by sending chunks of the string representation and forwarding userData tocallba...
Definition: IntegerSet.cpp:27
MLIR_CAPI_EXPORTED bool mlirIntegerSetEqual(MlirIntegerSet s1, MlirIntegerSet s2)
Checks if two integer set objects are equal.
Definition: IntegerSet.cpp:23
MLIR_CAPI_EXPORTED bool mlirIntegerSetIsConstraintEq(MlirIntegerSet set, intptr_t pos)
Returns true of the pos-th constraint of the set is an equality constraint, false otherwise.
Definition: IntegerSet.cpp:101
MLIR_CAPI_EXPORTED void mlirIntegerSetDump(MlirIntegerSet set)
Prints an integer set to the standard error stream.
Definition: IntegerSet.cpp:33
MLIR_CAPI_EXPORTED intptr_t mlirIntegerSetGetNumInequalities(MlirIntegerSet set)
Returns the number of inequalities in the given set.
Definition: IntegerSet.cpp:93
inline ::llvm::hash_code hash_value(const PolynomialBase< D, T > &arg)
Definition: Polynomial.h:262
void populateIRAffine(pybind11::module &m)
Include the generated interface declarations.
auto get(MLIRContext *context, Ts &&...params)
Helper method that injects context only if needed, this helps unify some of the attribute constructio...
Accumulates into a python string from a method that accepts an MlirStringCallback.
Definition: PybindUtils.h:102
pybind11::list parts
Definition: PybindUtils.h:103
pybind11::str join()
Definition: PybindUtils.h:117
MlirStringCallback getCallback()
Definition: PybindUtils.h:107