Reference: https://mlir.llvm.org/docs/Tutorials/Toy/Ch-4/
Note: Check Setup the Environment of MLIR for the environment setup.
1. Introduction#
Problem: Naively implementing each transformation for each dialect leads to large amounts of code duplication, as the internal algorithms are generally very similar.
Solution: To provide the ability for transformations to opaquely hook into dialects like Toy to get the information they need.
2. Add OpPrintInterface#
Define a new env var:
| 1
 | export TOY_CH4_HOME="$MLIR_HOME/examples/toy/Ch4"
 | 
2.1. Define OpPrintInterface#
First, create a new file $TOY_CH4_HOME/include/toy/OpPrintInterface.td, define OpPrintOpInterface with method opPrint which returns a std::string:
|  1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
 | #ifndef PRINT_INTERFACE
#define PRINT_INTERFACE
include "mlir/IR/OpBase.td"
def OpPrintOpInterface : OpInterface<"OpPrint">
{
    let description = [{
        Interface to print something in an operator.
    }];
    let methods = [
        InterfaceMethod< "Print some information in the current operation", "std::string", "opPrint" >
    ];
}
#endif // PRINT_INTERFACE
 | 
Second, create a file $TOY_CH4_HOME/include/toy/OpPrintInterface.hpp:
|  1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
 | #ifndef OPPRINTINTERFACE_HPP_
#define OPPRINTINTERFACE_HPP_
#include "mlir/IR/OpDefinition.h"
namespace mlir::toy {
/// Include the auto-generated declarations.
#include "toy/OpPrintOpInterface.h.inc"
} // namespace mlir::toy
#endif 
 | 
Third, in $TOY_CH4_HOME/include/toy/Dialect.h, include the new interface:
| 1
 | #include "toy/OpPrintInterface.hpp"
 | 
Fourth, make some modifications in $TOY_CH4_HOME/include/toy/Ops.td. Include the interface’s td at the beginning of the file:
| 1
 | include "toy/OpPrintInterface.td"
 | 
Then, for example, change the AddOp to declare that it implements the OpPrint interface:
| 1
2
3
4
5
6
7
 | def AddOp : Toy_Op<"add", [
    Pure, 
    DeclareOpInterfaceMethods<ShapeInferenceOpInterface>,
    DeclareOpInterfaceMethods<OpPrintOpInterface>
  ]> {
  // ...
}
 | 
You can also do the similar declaration for other operations.
Finally, some CMakeLists need to be modified.
Add following lines in $TOY_CH4_HOME/include/toy/CMakeLists.txt:
| 1
2
3
4
 | set(LLVM_TARGET_DEFINITIONS OpPrintInterface.td)
mlir_tablegen(OpPrintOpInterface.h.inc -gen-op-interface-decls)
mlir_tablegen(OpPrintOpInterface.cpp.inc -gen-op-interface-defs)
add_public_tablegen_target(ToyCh4OpPrintInterfaceIncGen)
 | 
Change the add_toy_chapter in $TOY_CH4_HOME/CMakelists.txt to:
|  1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
 | add_toy_chapter(toyc-ch4
  toyc.cpp
  parser/AST.cpp
  mlir/MLIRGen.cpp
  mlir/Dialect.cpp
  mlir/ShapeInferencePass.cpp
  mlir/OpPrintInterfacePass.cpp
  mlir/ToyCombine.cpp
  DEPENDS
  ToyCh4OpsIncGen
  ToyCh4ShapeInferenceInterfaceIncGen
  ToyCh4CombineIncGen
  ToyCh4OpPrintInterfaceIncGen
  )
 | 
To match the listed source files, create a blank file $TOY_CH4_HOME/mlir/OpPrintInterfacePass.cpp. We will implement the pass later.
Now build the MLIR:
| 1
 | bash $LLVM_PROJ_HOME/scripts/build-mlir.sh
 | 
Errors pop out because we haven’t implemented the OpPrintInterface which is declared in AddOp. Don’t worry, it will be implemented in the next section.
Now you can check the generated C++ class declarations in, for example, $LLVM_PROJ_HOME/build/tools/mlir/examples/toy/Ch4/include/toy/OpPrintOpInterface.h.inc.
2.3. Implement OpPrintInterface#
Since the interface is declared in AddOp (which is actually implemented by inheriting OpPrintOpInterface which provides a pure virtual function opPrint), we need to implement the function.
In $TOY_CH4_HOME/mlir/Dialect.cpp, add the following code:
| 1
 | std::string AddOp::opPrint() { return "I am AddOp"; }
 | 
2.4. Implement OpPrintPass#
In $TOY_CH4_HOME/include/toy/Passes.h, add the following line under mlir/examples/toy/Ch4/include/toy/Passes.h (inside namespace mlir::toy):
| 1
 | std::unique_ptr<Pass> createOpPrintPass();
 | 
In $TOY_CH4_HOME/mlir/OpPrintInterfacePass.cpp, add the following code:
|  1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
 | #include "mlir/IR/Operation.h"
#include "mlir/Pass/Pass.h"
#include "mlir/Support/LLVM.h"
#include "mlir/Support/TypeID.h"
#include "toy/Dialect.h"
#include "toy/OpPrintInterface.hpp"
#include "toy/Passes.h"
#include <iostream>
#include <memory>
using namespace mlir;
using namespace toy;
#include "toy/OpPrintOpInterface.cpp.inc"
namespace
{
struct OpPrintIntercfacePass
    : public mlir::PassWrapper<OpPrintIntercfacePass, OperationPass<toy::FuncOp>>
{
    MLIR_DEFINE_EXPLICIT_INTERNAL_INLINE_TYPE_ID(OpPrintIntercfacePass)
    void runOnOperation() override
    {
        auto f = getOperation();
        f.walk([&](mlir::Operation* op) {
            if (auto shapeOp = dyn_cast<OpPrint>(op)) {
                std::cout << shapeOp.opPrint() << std::endl;
            }
        });
    }
};
}  // namespace
std::unique_ptr<Pass> createOpPrintPass()
{
    return std::make_unique<OpPrintIntercfacePass>();
}
 | 
2.5. Add Pass to Pass Manager#
In $TOY_CH4_HOME/toyc.cpp, add the following line after mlir::OpPassManager &optPM = pm.nest<mlir::toy::FuncOp>();:
| 1
 | optPM.addPass(mlir::toy::createOpPrintPass());
 | 
Implement OpPrintPass in $TOY_CH4_HOME/mlir/OpPrintInterfacePass.cpp:
|  1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
 | #include "mlir/IR/Operation.h"
#include "mlir/Pass/Pass.h"
#include "mlir/Support/LLVM.h"
#include "mlir/Support/TypeID.h"
#include "toy/Dialect.h"
#include "toy/OpPrintInterface.hpp"
#include "toy/Passes.h"
#include <iostream>
#include <memory>
using namespace mlir;
using namespace toy;
#include "toy/OpPrintOpInterface.cpp.inc"
namespace
{
struct OpPrintIntercfacePass
    : public mlir::PassWrapper<OpPrintIntercfacePass, OperationPass<toy::FuncOp>>
{
    MLIR_DEFINE_EXPLICIT_INTERNAL_INLINE_TYPE_ID(OpPrintIntercfacePass)
    void runOnOperation() override
    {
        auto f = getOperation();
        f.walk([&](mlir::Operation* op) {
            if (auto shapeOp = dyn_cast<OpPrint>(op)) {
                std::cout << shapeOp.opPrint() << std::endl;
            }
        });
    }
};
}  // namespace
std::unique_ptr<Pass> mlir::toy::createOpPrintPass()
{
    return std::make_unique<OpPrintIntercfacePass>();
}
 | 
2.6. Test the Pass#
Now, rebuild the MLIR:
| 1
 | bash $LLVM_PROJ_HOME/scripts/build-mlir.sh
 | 
Run the Test:
| 1
 | toyc-ch4 $MLIR_HOME/test/Examples/Toy/Ch4/codegen.toy -emit=mlir -opt
 |