blob: a4b1ca911b75858b6726c464e968665eea572221 [file] [log] [blame]
// Copyright 2019 The IREE Authors
//
// Licensed 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 "iree_tf_compiler/TF/Passes.h"
#include "iree/compiler/Dialect/Flow/Transforms/Passes.h"
#include "mlir-hlo/Dialect/mhlo/transforms/passes.h"
#include "mlir/Dialect/MemRef/Transforms/Passes.h"
#include "mlir/Dialect/Shape/Transforms/Passes.h"
#include "mlir/Pass/PassManager.h"
#include "mlir/Pass/PassRegistry.h"
#include "mlir/Transforms/Passes.h"
#include "tensorflow/compiler/mlir/tensorflow/transforms/passes.h"
#include "tensorflow/compiler/mlir/tensorflow/transforms/tf_saved_model_passes.h"
#include "tensorflow/compiler/mlir/tosa/tf_passes.h"
namespace mlir {
namespace iree_integrations {
namespace TF {
// All IREE-specific passes that lower TF representations before reaching the
// IREE core should go here.
void buildTFImportPassPipeline(OpPassManager &pm, bool useTosa) {
//----------------------------------------------------------------------------
// Clean up tf_executor and extraneous unused functions.
//----------------------------------------------------------------------------
pm.addPass(createSymbolDCEPass());
pm.addPass(tf_executor::CreateTFExecutorGraphPruningPass());
pm.addPass(::mlir::TF::CreateGuaranteeAllFuncsOneUsePass());
::mlir::TF::CreateTFStandardPipeline(pm,
::mlir::TF::StandardPipelineOptions());
pm.addPass(::mlir::TF::CreateDeviceIndexSelectorPass());
//----------------------------------------------------------------------------
// Try to get the IR in good condition.
//----------------------------------------------------------------------------
pm.addPass(createStripAssertsPass());
pm.addPass(createInlinerPass());
pm.addPass(createCanonicalizerPass());
pm.addPass(TFDevice::CreateDecomposeResourceOpsPass());
pm.addPass(createPropagateResourceCastsPass());
pm.addPass(::mlir::TF::CreateTFShapeInferencePass());
//----------------------------------------------------------------------------
// Lower to CFG.
// After this point, most TF optimizations won't work properly besides
// simple canonicalizations.
//----------------------------------------------------------------------------
pm.addPass(::mlir::TF::CreateTFFunctionalControlFlowToCFG());
// Inline, as tf-functional-control-flow-to-cfg leaves in calls.
pm.addPass(createInlinerPass());
//----------------------------------------------------------------------------
// Some further cleanups now that control flow is in better shape.
//----------------------------------------------------------------------------
pm.addPass(createSymbolDCEPass());
pm.addPass(createCanonicalizerPass());
//----------------------------------------------------------------------------
// Legalize to TOSA/XLA
//----------------------------------------------------------------------------
if (useTosa) {
tosa::TOSATFLegalizationPipelineOptions tosaOptions;
tosa::createTFtoTOSALegalizationPipeline(pm, tosaOptions);
} else {
pm.addPass(createConvertToMHLOPass());
pm.addPass(createCanonicalizerPass());
}
//----------------------------------------------------------------------------
// Now that the IR is starting to look nice, optimize global tensors.
//----------------------------------------------------------------------------
pm.addPass(tf_saved_model::CreateOptimizeGlobalTensorsPass());
//----------------------------------------------------------------------------
// Lowering shape-related constructs.
//----------------------------------------------------------------------------
// pm.addPass(iree_compiler::Shape::createConvertHLOToShapePass());
// TODO(#2277): Lower HLO shape constraints instead of eliding them here.
pm.addPass(createRemoveShapeConstraintsPass());
pm.addPass(createCanonicalizerPass());
// pm.addPass(iree_compiler::Shape::createConvertShapeToShapexPass());
// pm.addPass(createCanonicalizerPass());
//----------------------------------------------------------------------------
// Lowering tf_saved_model dialect to IREE dialects
//----------------------------------------------------------------------------
// First, eliminate tf_saved_model.global_tensor's and corresponding
// tf_saved_model.bound_input's.
pm.addPass(createLowerGlobalTensorsPass());
// Lower exported functions.
//
// This pass must run second because:
// - It assumes that tf_saved_model.bound_inputs have been eliminated
// - It removes tf_saved_model.semantics from the module, which we can only
// do at the very end.
pm.addPass(createSavedModelToIREEABIPass());
// Inline the wrapper functions.
pm.addPass(createInlinerPass());
//----------------------------------------------------------------------------
// Ensure that all Tensorflow has been legalized away
//----------------------------------------------------------------------------
pm.addPass(createStripModuleMetadataPass());
pm.nest<ModuleOp>().addPass(createStripFunctionMetadataPass());
pm.addPass(createVerifyFullyConvertedPass());
}
void registerTFImportPassPipeline() {
mlir::PassPipelineRegistration<> pipeline(
"iree-import-tf-pipeline",
"Run IREE-specific passes for importing TF code into IREE",
[](OpPassManager &passManager) {
buildTFImportPassPipeline(passManager, false);
});
}
void registerTFTosaImportPassPipeline() {
mlir::PassPipelineRegistration<> pipeline(
"iree-import-tf-tosa-pipeline",
"Run IREE-specific passes for importing TF code into IREE",
[](OpPassManager &passManager) {
buildTFImportPassPipeline(passManager, true);
});
}
} // namespace TF
} // namespace iree_integrations
} // namespace mlir