2020-07-08 07:42:40 +08:00
|
|
|
//===- mlir-reduce.cpp - The MLIR reducer ---------------------------------===//
|
|
|
|
//
|
|
|
|
// 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 implements the general framework of the MLIR reducer tool. It
|
|
|
|
// parses the command line arguments, parses the initial MLIR test case and sets
|
|
|
|
// up the testing environment. It outputs the most reduced test case variant
|
|
|
|
// after executing the reduction passes.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include "mlir/InitAllDialects.h"
|
|
|
|
#include "mlir/Parser.h"
|
|
|
|
#include "mlir/Pass/Pass.h"
|
|
|
|
#include "mlir/Pass/PassManager.h"
|
2020-08-19 00:47:06 +08:00
|
|
|
#include "mlir/Reducer/OptReductionPass.h"
|
2020-08-21 12:59:23 +08:00
|
|
|
#include "mlir/Reducer/Passes/OpReducer.h"
|
2020-08-08 07:17:27 +08:00
|
|
|
#include "mlir/Reducer/ReductionNode.h"
|
|
|
|
#include "mlir/Reducer/ReductionTreePass.h"
|
2020-07-08 07:42:40 +08:00
|
|
|
#include "mlir/Reducer/Tester.h"
|
|
|
|
#include "mlir/Support/FileUtilities.h"
|
|
|
|
#include "mlir/Support/LogicalResult.h"
|
|
|
|
#include "mlir/Transforms/Passes.h"
|
|
|
|
#include "llvm/Support/InitLLVM.h"
|
|
|
|
#include "llvm/Support/ToolOutputFile.h"
|
|
|
|
|
|
|
|
using namespace mlir;
|
|
|
|
|
2020-09-01 06:39:39 +08:00
|
|
|
namespace mlir {
|
2020-11-06 00:55:00 +08:00
|
|
|
namespace test {
|
2020-09-01 06:39:39 +08:00
|
|
|
void registerTestDialect(DialectRegistry &);
|
2020-11-06 00:55:00 +08:00
|
|
|
} // namespace test
|
|
|
|
} // namespace mlir
|
2020-09-01 06:39:39 +08:00
|
|
|
|
2020-07-08 07:42:40 +08:00
|
|
|
static llvm::cl::opt<std::string> inputFilename(llvm::cl::Positional,
|
|
|
|
llvm::cl::Required,
|
|
|
|
llvm::cl::desc("<input file>"));
|
|
|
|
|
|
|
|
static llvm::cl::opt<std::string>
|
|
|
|
testFilename("test", llvm::cl::Required, llvm::cl::desc("Testing script"));
|
|
|
|
|
|
|
|
static llvm::cl::list<std::string>
|
|
|
|
testArguments("test-args", llvm::cl::ZeroOrMore,
|
|
|
|
llvm::cl::desc("Testing script arguments"));
|
|
|
|
|
|
|
|
static llvm::cl::opt<std::string>
|
|
|
|
outputFilename("o",
|
|
|
|
llvm::cl::desc("Output filename for the reduced test case"),
|
|
|
|
llvm::cl::init("-"));
|
|
|
|
|
2020-08-19 00:47:06 +08:00
|
|
|
// TODO: Use PassPipelineCLParser to define pass pieplines in the command line.
|
|
|
|
static llvm::cl::opt<std::string>
|
|
|
|
passTestSpecifier("pass-test",
|
|
|
|
llvm::cl::desc("Indicate a specific pass to be tested"));
|
|
|
|
|
2020-07-08 07:42:40 +08:00
|
|
|
// Parse and verify the input MLIR file.
|
|
|
|
static LogicalResult loadModule(MLIRContext &context, OwningModuleRef &module,
|
|
|
|
StringRef inputFilename) {
|
|
|
|
module = parseSourceFile(inputFilename, &context);
|
|
|
|
if (!module)
|
|
|
|
return failure();
|
|
|
|
|
|
|
|
return success();
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char **argv) {
|
|
|
|
|
|
|
|
llvm::InitLLVM y(argc, argv);
|
|
|
|
|
|
|
|
registerMLIRContextCLOptions();
|
|
|
|
registerPassManagerCLOptions();
|
|
|
|
|
|
|
|
llvm::cl::ParseCommandLineOptions(argc, argv,
|
|
|
|
"MLIR test case reduction tool.\n");
|
|
|
|
|
|
|
|
std::string errorMessage;
|
|
|
|
|
|
|
|
auto testscript = openInputFile(testFilename, &errorMessage);
|
|
|
|
if (!testscript)
|
|
|
|
llvm::report_fatal_error(errorMessage);
|
|
|
|
|
|
|
|
auto output = openOutputFile(outputFilename, &errorMessage);
|
|
|
|
if (!output)
|
|
|
|
llvm::report_fatal_error(errorMessage);
|
|
|
|
|
2021-02-10 17:11:50 +08:00
|
|
|
mlir::DialectRegistry registry;
|
|
|
|
registerAllDialects(registry);
|
2020-09-01 06:39:39 +08:00
|
|
|
#ifdef MLIR_INCLUDE_TESTS
|
2021-02-10 17:11:50 +08:00
|
|
|
mlir::test::registerTestDialect(registry);
|
2020-09-01 06:39:39 +08:00
|
|
|
#endif
|
2021-02-10 17:11:50 +08:00
|
|
|
mlir::MLIRContext context(registry);
|
2020-07-08 07:42:40 +08:00
|
|
|
|
2020-09-01 06:39:39 +08:00
|
|
|
mlir::OwningModuleRef moduleRef;
|
2020-07-08 07:42:40 +08:00
|
|
|
if (failed(loadModule(context, moduleRef, inputFilename)))
|
|
|
|
llvm::report_fatal_error("Input test case can't be parsed");
|
|
|
|
|
|
|
|
// Initialize test environment.
|
2020-08-08 07:17:27 +08:00
|
|
|
const Tester test(testFilename, testArguments);
|
2020-07-08 07:42:40 +08:00
|
|
|
|
|
|
|
if (!test.isInteresting(inputFilename))
|
|
|
|
llvm::report_fatal_error(
|
|
|
|
"Input test case does not exhibit interesting behavior");
|
|
|
|
|
2020-08-08 07:17:27 +08:00
|
|
|
// Reduction pass pipeline.
|
|
|
|
PassManager pm(&context);
|
|
|
|
|
2020-08-19 00:47:06 +08:00
|
|
|
if (passTestSpecifier == "DCE") {
|
|
|
|
|
|
|
|
// Opt Reduction Pass with SymbolDCEPass as opt pass.
|
2020-08-21 12:59:23 +08:00
|
|
|
pm.addPass(std::make_unique<OptReductionPass>(test, &context,
|
2020-08-19 00:47:06 +08:00
|
|
|
createSymbolDCEPass()));
|
|
|
|
|
|
|
|
} else if (passTestSpecifier == "function-reducer") {
|
|
|
|
|
|
|
|
// Reduction tree pass with OpReducer variant generation and single path
|
|
|
|
// traversal.
|
2020-08-21 12:59:23 +08:00
|
|
|
pm.addPass(
|
|
|
|
std::make_unique<ReductionTreePass<OpReducer<FuncOp>, SinglePath>>(
|
|
|
|
test));
|
2020-08-19 00:47:06 +08:00
|
|
|
}
|
2020-08-08 07:17:27 +08:00
|
|
|
|
|
|
|
ModuleOp m = moduleRef.get().clone();
|
|
|
|
|
|
|
|
if (failed(pm.run(m)))
|
|
|
|
llvm::report_fatal_error("Error running the reduction pass pipeline");
|
|
|
|
|
|
|
|
m.print(output->os());
|
2020-07-08 07:42:40 +08:00
|
|
|
output->keep();
|
|
|
|
|
|
|
|
return 0;
|
2020-08-08 07:17:27 +08:00
|
|
|
}
|