2020-08-14 11:43:47 +08:00
|
|
|
//===- PassManagerTest.cpp - PassManager unit tests -----------------------===//
|
|
|
|
//
|
|
|
|
// 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
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "mlir/Pass/PassManager.h"
|
|
|
|
#include "mlir/IR/Builders.h"
|
2020-11-20 02:43:12 +08:00
|
|
|
#include "mlir/IR/BuiltinOps.h"
|
2020-08-14 11:43:47 +08:00
|
|
|
#include "mlir/Pass/Pass.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
|
|
|
using namespace mlir;
|
|
|
|
using namespace mlir::detail;
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
/// Analysis that operates on any operation.
|
|
|
|
struct GenericAnalysis {
|
|
|
|
GenericAnalysis(Operation *op) : isFunc(isa<FuncOp>(op)) {}
|
|
|
|
const bool isFunc;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Analysis that operates on a specific operation.
|
|
|
|
struct OpSpecificAnalysis {
|
|
|
|
OpSpecificAnalysis(FuncOp op) : isSecret(op.getName() == "secret") {}
|
|
|
|
const bool isSecret;
|
|
|
|
};
|
|
|
|
|
|
|
|
/// Simple pass to annotate a FuncOp with the results of analysis.
|
|
|
|
/// Note: not using FunctionPass as it skip external functions.
|
|
|
|
struct AnnotateFunctionPass
|
|
|
|
: public PassWrapper<AnnotateFunctionPass, OperationPass<FuncOp>> {
|
|
|
|
void runOnOperation() override {
|
|
|
|
FuncOp op = getOperation();
|
2020-12-14 15:32:31 +08:00
|
|
|
Builder builder(op->getParentOfType<ModuleOp>());
|
2020-08-14 11:43:47 +08:00
|
|
|
|
|
|
|
auto &ga = getAnalysis<GenericAnalysis>();
|
|
|
|
auto &sa = getAnalysis<OpSpecificAnalysis>();
|
|
|
|
|
2020-12-14 15:32:31 +08:00
|
|
|
op->setAttr("isFunc", builder.getBoolAttr(ga.isFunc));
|
|
|
|
op->setAttr("isSecret", builder.getBoolAttr(sa.isSecret));
|
2020-08-14 11:43:47 +08:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST(PassManagerTest, OpSpecificAnalysis) {
|
|
|
|
MLIRContext context;
|
|
|
|
Builder builder(&context);
|
|
|
|
|
|
|
|
// Create a module with 2 functions.
|
|
|
|
OwningModuleRef module(ModuleOp::create(UnknownLoc::get(&context)));
|
|
|
|
for (StringRef name : {"secret", "not_secret"}) {
|
|
|
|
FuncOp func =
|
|
|
|
FuncOp::create(builder.getUnknownLoc(), name,
|
|
|
|
builder.getFunctionType(llvm::None, llvm::None));
|
2020-11-14 05:04:53 +08:00
|
|
|
func.setPrivate();
|
2020-08-14 11:43:47 +08:00
|
|
|
module->push_back(func);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Instantiate and run our pass.
|
|
|
|
PassManager pm(&context);
|
|
|
|
pm.addNestedPass<FuncOp>(std::make_unique<AnnotateFunctionPass>());
|
|
|
|
LogicalResult result = pm.run(module.get());
|
|
|
|
EXPECT_TRUE(succeeded(result));
|
|
|
|
|
|
|
|
// Verify that each function got annotated with expected attributes.
|
|
|
|
for (FuncOp func : module->getOps<FuncOp>()) {
|
2020-12-14 15:32:31 +08:00
|
|
|
ASSERT_TRUE(func->getAttr("isFunc").isa<BoolAttr>());
|
|
|
|
EXPECT_TRUE(func->getAttr("isFunc").cast<BoolAttr>().getValue());
|
2020-08-14 11:43:47 +08:00
|
|
|
|
|
|
|
bool isSecret = func.getName() == "secret";
|
2020-12-14 15:32:31 +08:00
|
|
|
ASSERT_TRUE(func->getAttr("isSecret").isa<BoolAttr>());
|
|
|
|
EXPECT_EQ(func->getAttr("isSecret").cast<BoolAttr>().getValue(), isSecret);
|
2020-08-14 11:43:47 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-03 04:09:07 +08:00
|
|
|
namespace {
|
|
|
|
struct InvalidPass : Pass {
|
|
|
|
InvalidPass() : Pass(TypeID::get<InvalidPass>(), StringRef("invalid_op")) {}
|
|
|
|
StringRef getName() const override { return "Invalid Pass"; }
|
|
|
|
void runOnOperation() override {}
|
|
|
|
|
|
|
|
/// A clone method to create a copy of this pass.
|
|
|
|
std::unique_ptr<Pass> clonePass() const override {
|
|
|
|
return std::make_unique<InvalidPass>(
|
|
|
|
*static_cast<const InvalidPass *>(this));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} // anonymous namespace
|
|
|
|
|
|
|
|
TEST(PassManagerTest, InvalidPass) {
|
|
|
|
MLIRContext context;
|
|
|
|
|
|
|
|
// Create a module
|
|
|
|
OwningModuleRef module(ModuleOp::create(UnknownLoc::get(&context)));
|
|
|
|
|
|
|
|
// Add a single "invalid_op" operation
|
|
|
|
OpBuilder builder(&module->getBodyRegion());
|
|
|
|
OperationState state(UnknownLoc::get(&context), "invalid_op");
|
|
|
|
builder.insert(Operation::create(state));
|
|
|
|
|
|
|
|
// Register a diagnostic handler to capture the diagnostic so that we can
|
|
|
|
// check it later.
|
|
|
|
std::unique_ptr<Diagnostic> diagnostic;
|
|
|
|
context.getDiagEngine().registerHandler([&](Diagnostic &diag) {
|
|
|
|
diagnostic.reset(new Diagnostic(std::move(diag)));
|
|
|
|
});
|
|
|
|
|
|
|
|
// Instantiate and run our pass.
|
|
|
|
PassManager pm(&context);
|
2020-11-03 19:17:39 +08:00
|
|
|
pm.nest("invalid_op").addPass(std::make_unique<InvalidPass>());
|
2020-09-03 04:09:07 +08:00
|
|
|
LogicalResult result = pm.run(module.get());
|
|
|
|
EXPECT_TRUE(failed(result));
|
|
|
|
ASSERT_TRUE(diagnostic.get() != nullptr);
|
|
|
|
EXPECT_EQ(
|
|
|
|
diagnostic->str(),
|
|
|
|
"'invalid_op' op trying to schedule a pass on an unregistered operation");
|
2020-11-03 19:17:39 +08:00
|
|
|
|
|
|
|
// Check that adding the pass at the top-level triggers a fatal error.
|
|
|
|
ASSERT_DEATH(pm.addPass(std::make_unique<InvalidPass>()), "");
|
2020-09-03 04:09:07 +08:00
|
|
|
}
|
|
|
|
|
2020-08-14 11:43:47 +08:00
|
|
|
} // end namespace
|