Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[tf-frontend] remove shape.split_at, shape.cstr_broadcastable, mhlo.cstr_reshapable #119

Merged
merged 1 commit into from
Feb 8, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,7 @@

#include "mlir/Dialect/Func/IR/FuncOps.h"
#include "mlir/Dialect/Shape/IR/Shape.h"
#include "mlir/Dialect/Shape/Transforms/Passes.h"
#include "mlir/IR/BuiltinOps.h"
#include "mlir/Pass/PassManager.h"
#include "mlir/Transforms/Passes.h"
Expand All @@ -56,12 +57,14 @@ struct CustomizedTfToMhloPipelinePass
const std::vector<std::string> &customcall_ops, bool remove_control_flow,
bool staticalize_dynamic_shape, bool stop_after_rewrite_custom_call,
const std::unordered_map<std::string, Attribute>
&additional_main_func_attrs) {
&additional_main_func_attrs,
bool set_assuming_to_be_true) {
this->customCallOps = customcall_ops;
this->removeControlFlow = remove_control_flow;
this->staticalizeDynamicShape = staticalize_dynamic_shape;
this->stopAfterRewriteCustomCall = stop_after_rewrite_custom_call;
this->additional_main_func_attrs = additional_main_func_attrs;
this->setAssumingToBeTrue = set_assuming_to_be_true;
}

void runOnOperation() override {
Expand All @@ -72,15 +75,13 @@ struct CustomizedTfToMhloPipelinePass
pm.addPass(mlir::createSCCPPass());
pm.addPass(mlir::createCanonicalizerPass());

//// not safe remove-control-flow pass
// if (removeControlFlow)
// pm.addNestedPass<mlir::func::FuncOp>(
// mlir::tfext::createRemoveControlFlowPass());
// prun useless tf node
pm.addNestedPass<mlir::func::FuncOp>(
mlir::tf_executor::CreateTFExecutorGraphPruningPass());
if (removeControlFlow) {
pm.addNestedPass<mlir::func::FuncOp>(
mlir::tfext::createTFSwitchMergeToIfPass());
}

// prun useless tf node
pm.addNestedPass<mlir::func::FuncOp>(
mlir::tf_executor::CreateTFExecutorGraphPruningPass());
Expand Down Expand Up @@ -209,6 +210,12 @@ struct CustomizedTfToMhloPipelinePass
mlir::tfext::createRewriteFuncAttrToByteIRPass(
additional_main_func_attrs));

if (setAssumingToBeTrue) {
pm.addNestedPass<mlir::func::FuncOp>(
mlir::createRemoveShapeConstraintsPass());
pm.addNestedPass<mlir::func::FuncOp>(
mlir::tfext::createRemoveCstrReshapablePass());
}
pm.addPass(mlir::createCanonicalizerPass());

pm.addPass(mlir::mhlo::createHloLegalizeToStablehloPass());
Expand All @@ -230,8 +237,10 @@ mlir::tfext::createCustomizedTfToMhloPipelinePass(
bool staticalize_dynamic_shape /*= false*/,
bool stop_after_rewrite_custom_call /*= false*/,
const std::unordered_map<std::string, Attribute>
&additional_main_func_attrs /*= {}*/) {
&additional_main_func_attrs /*= {}*/,
bool set_assuming_to_be_true /*= true*/) {
return std::make_unique<CustomizedTfToMhloPipelinePass>(
customcall_ops, remove_control_flow, staticalize_dynamic_shape,
stop_after_rewrite_custom_call, additional_main_func_attrs);
stop_after_rewrite_custom_call, additional_main_func_attrs,
set_assuming_to_be_true);
}
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,8 @@ std::unique_ptr<OperationPass<ModuleOp>> createCustomizedTfToMhloPipelinePass(
bool remove_control_flow = false, bool staticalize_dynamic_shape = false,
bool stop_after_rewrite_custom_call = false,
const std::unordered_map<std::string, Attribute>
&additional_main_func_attrs = {});
&additional_main_func_attrs = {},
bool set_assuming_to_be_true = true);

} // namespace tfext
} // namespace mlir
Expand Down
4 changes: 3 additions & 1 deletion frontends/tf-frontend/tf_mlir_ext/pipelines/passes.td
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,9 @@ def CustomizedTfToMhloPipeline : Pass<"customized-tf-to-mhlo", "mlir::ModuleOp">
/*default*/"false", "Aggresively and experimentally try to rewrite "
"the dynamic graph to a equivalent static graph">,
Option<"stopAfterRewriteCustomCall", "stop-after-rewrite-customcall", "bool", /*default=*/"false",
"stop after rewrite customcall ops">
"stop after rewrite customcall ops">,
Option<"setAssumingToBeTrue", "set-assuming-to-be-true", "bool", /*default=*/"true",
"remove cstr_reshapable,cstr_broadcastable, and set assuming to be true">
];
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -19,3 +19,11 @@ func.func @static_strided_slice(%arg0: tensor<1x26xf16>) -> tensor<1x12xf16> {
}
// CHECK-LABEL: static_strided_slice
// CHECK: "tf.StridedSlice"

func.func @static_batch_matmul_v2(%arg0: tensor<32x246x16xf16>, %arg1: tensor<32x16x16xf16>) -> tensor<32x246x16xf16> {
%0 = "tf.BatchMatMulV2"(%arg0, %arg1) {adj_x = false, adj_y = false, device = ""} : (tensor<32x246x16xf16>, tensor<32x16x16xf16>) -> tensor<32x246x16xf16>
return %0 : tensor<32x246x16xf16>
}
// CHECK-LABEL: @static_batch_matmul_v2
// CHECK-NEXT: %0 = "mhlo.dot_general"(%arg0, %arg1) {dot_dimension_numbers = #mhlo.dot<lhs_batching_dimensions = [0], rhs_batching_dimensions = [0], lhs_contracting_dimensions = [2], rhs_contracting_dimensions = [1]>, precision_config = [#mhlo<precision DEFAULT>, #mhlo<precision DEFAULT>]} : (tensor<32x246x16xf16>, tensor<32x16x16xf16>) -> tensor<32x246x16xf16>
// CHECK-NEXT: return %0 : tensor<32x246x16xf16>
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
// RUN: tf-ext-opt -remove-cstr-reshapable -remove-shape-constraints -canonicalize %s | FileCheck %s

func.func @remove_cstr_reshapable(%arg0: index, %arg1: tensor<3xi32>, %arg2: tensor<32x?xf16>) -> tensor<?x?x64xf16> {
%0 = mhlo.cstr_reshapable %arg0, %arg1 : (index, tensor<3xi32>) -> !shape.witness
%1 = shape.assuming %0 -> (tensor<?x?x64xf16>) {
%2 = mhlo.compute_reshape_shape %arg0, %arg1 : (index, tensor<3xi32>) -> tensor<3xi32>
%3 = mhlo.dynamic_reshape %arg2, %2 : (tensor<32x?xf16>, tensor<3xi32>) -> tensor<?x?x64xf16>
shape.assuming_yield %3 : tensor<?x?x64xf16>
}
return %1 : tensor<?x?x64xf16>
}
// CHECK-LABEL: @remove_cstr_reshapable
// CHECK-NEXT: %0 = mhlo.compute_reshape_shape %arg0, %arg1 : (index, tensor<3xi32>) -> tensor<3xi32>
// CHECK-NEXT: %1 = mhlo.dynamic_reshape %arg2, %0 : (tensor<32x?xf16>, tensor<3xi32>) -> tensor<?x?x64xf16>
// CHECK-NEXT: return %1 : tensor<?x?x64xf16>

func.func @remove_cstr_broadcastable(%arg0: tensor<3xindex>, %arg1: tensor<3xindex>, %arg2: tensor<?x?x64xf16>, %arg3: tensor<32x?x64xf16>) -> tensor<32x?x64xf16> {
%0 = shape.cstr_broadcastable %arg0, %arg1 : tensor<3xindex>, tensor<3xindex>
%1 = shape.assuming %0 -> (tensor<32x?x64xf16>) {
%2 = shape.broadcast %arg0, %arg1 : tensor<3xindex>, tensor<3xindex> -> tensor<3xindex>
%3 = "mhlo.dynamic_broadcast_in_dim"(%arg2, %2) {broadcast_dimensions = dense<[0, 1, 2]> : tensor<3xi64>} : (tensor<?x?x64xf16>, tensor<3xindex>) -> tensor<32x?x64xf16>
%4 = "mhlo.dynamic_broadcast_in_dim"(%arg3, %2) {broadcast_dimensions = dense<[0, 1, 2]> : tensor<3xi64>} : (tensor<32x?x64xf16>, tensor<3xindex>) -> tensor<32x?x64xf16>
%5 = mhlo.multiply %3, %4 : tensor<32x?x64xf16>
shape.assuming_yield %5 : tensor<32x?x64xf16>
}
return %1 : tensor<32x?x64xf16>
}
// CHECK-LABEL: @remove_cstr_broadcastable
// CHECK-NEXT: %0 = shape.broadcast %arg0, %arg1 : tensor<3xindex>, tensor<3xindex> -> tensor<3xindex>
// CHECK-NEXT: %1 = "mhlo.dynamic_broadcast_in_dim"(%arg2, %0) {broadcast_dimensions = dense<[0, 1, 2]> : tensor<3xi64>} : (tensor<?x?x64xf16>, tensor<3xindex>) -> tensor<32x?x64xf16>
// CHECK-NEXT: %2 = "mhlo.dynamic_broadcast_in_dim"(%arg3, %0) {broadcast_dimensions = dense<[0, 1, 2]> : tensor<3xi64>} : (tensor<32x?x64xf16>, tensor<3xindex>) -> tensor<32x?x64xf16>
// CHECK-NEXT: %3 = mhlo.multiply %1, %2 : tensor<32x?x64xf16>
// CHECK-NEXT: return %3 : tensor<32x?x64xf16>
2 changes: 2 additions & 0 deletions frontends/tf-frontend/tf_mlir_ext/transforms/BUILD
Original file line number Diff line number Diff line change
Expand Up @@ -88,6 +88,7 @@ cc_library(
"rewrite_to_custom_call.cc",
"tf_fallback_to_custom_call.cc",
"tf_switch_merge_to_if.cc",
"remove_cstr_reshapable.cc",
],
hdrs = [
"constant_folding.h",
Expand All @@ -102,6 +103,7 @@ cc_library(
"rewrite_to_custom_call.h",
"tf_fallback_to_custom_call.h",
"tf_switch_merge_to_if.h",
"remove_cstr_reshapable.h",
],
textual_hdrs = [
"passes_detail.h",
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -467,9 +467,65 @@ class ConvertStridedSliceOp : public OpRewritePattern<TF::StridedSliceOp> {
}
};

// Returns a PrecisionConfig as an array attribute based on whether TF32
// execution is enabled
static ArrayAttr GetPrecisionConfig(Builder *builder) {
mlir::mhlo::Precision precision = mhlo::Precision::DEFAULT;
llvm::SmallVector<mlir::Attribute, 2> attr_vec;
const int num_inputs = 2;
for (int i = 0; i < num_inputs; i++) {
attr_vec.push_back(
mlir::mhlo::PrecisionAttr::get(builder->getContext(), precision));
}
return builder->getArrayAttr(attr_vec);
}

class ConvertBatchMatMulV2Op : public OpRewritePattern<TF::BatchMatMulV2Op> {
public:
using OpRewritePattern<TF::BatchMatMulV2Op>::OpRewritePattern;

LogicalResult matchAndRewrite(TF::BatchMatMulV2Op op,
PatternRewriter &rewriter) const override {
Value lhs = op.getX();
Value rhs = op.getY();
auto lhs_type = lhs.getType().dyn_cast<RankedTensorType>();
auto rhs_type = rhs.getType().dyn_cast<RankedTensorType>();
if (!lhs_type || !rhs_type)
return failure();
if (lhs_type.getRank() != rhs_type.getRank())
return failure();
auto lhs_batch = lhs_type.getShape().drop_back(2);
auto rhs_batch = rhs_type.getShape().drop_back(2);
bool batch_equal =
std::equal(lhs_batch.begin(), lhs_batch.end(), rhs_batch.begin());
bool is_static = std::all_of(lhs_batch.begin(), lhs_batch.end(),
[](int64_t dim) { return dim > 0; });
if (!batch_equal || !is_static) {
return failure();
}
int64_t rank = lhs_type.getRank();
auto batch_dimensions = llvm::to_vector<4>(llvm::seq<int64_t>(0, rank - 2));
auto lhs_contracting_dimensions = llvm::to_vector<4>(
llvm::ArrayRef({op.getAdjX() ? rank - 2 : rank - 1}));
auto rhs_contracting_dimensions = llvm::to_vector<4>(
llvm::ArrayRef({op.getAdjY() ? rank - 1 : rank - 2}));
auto dimension_numbers = mhlo::DotDimensionNumbersAttr::get(
rewriter.getContext(),
/*lhs_batching_dimensions=*/batch_dimensions,
/*rhs_batching_dimensions=*/batch_dimensions,
/*lhs_contracting_dimensions=*/lhs_contracting_dimensions,
/*rhs_contracting_dimensions=*/rhs_contracting_dimensions);
rewriter.replaceOpWithNewOp<mhlo::DotGeneralOp>(
op, op.getType(), lhs, rhs, dimension_numbers,
/*precision_config=*/GetPrecisionConfig(&rewriter));
return success();
}
};

void PopulateMhloLegalizeTfExtPatterns(MLIRContext *context,
RewritePatternSet *patterns) {
patterns->add(std::make_unique<ConvertStridedSliceOp>(context));
patterns->add(std::make_unique<ConvertBatchMatMulV2Op>(context));
}

struct MhloLegalizeTfExtPass
Expand Down
1 change: 1 addition & 0 deletions frontends/tf-frontend/tf_mlir_ext/transforms/passes.h
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@
#include "tf_mlir_ext/transforms/mhlo_legalize_tf_ext.h"
#include "tf_mlir_ext/transforms/process_dynamic_stitch_as_static.h"
#include "tf_mlir_ext/transforms/remove_control_flow.h"
#include "tf_mlir_ext/transforms/remove_cstr_reshapable.h"
#include "tf_mlir_ext/transforms/reshape_movedown_string.h"
#include "tf_mlir_ext/transforms/rewrite_func_attr_to_byteir.h"
#include "tf_mlir_ext/transforms/rewrite_to_custom_call.h"
Expand Down
5 changes: 5 additions & 0 deletions frontends/tf-frontend/tf_mlir_ext/transforms/passes.td
Original file line number Diff line number Diff line change
Expand Up @@ -108,4 +108,9 @@ def TFSwitchMergeToIf : Pass<"tf-switch-merge-to-if", "func::FuncOp"> {
let constructor = "mlir::tfext::createTFSwitchMergeToIfPass()";
}

def RemoveCstrReshapable : Pass<"remove-cstr-reshapable", "func::FuncOp"> {
let summary = "replace cstr_reshapable to true";
let constructor = "mlir::tfext::createRemoveCstrReshapablePass()";
}

#endif // TF_MLIR_EXT_PASSES
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
//===- remove_cstr_reshapable.cc ------------------------------*--- C++ -*-===//
//
// Copyright 2022 ByteDance Ltd. and/or its affiliates. All rights reserved.
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
//===----------------------------------------------------------------------===//

#include "mhlo/IR/hlo_ops.h"
#include "mlir/Dialect/Func/IR/FuncOps.h"
#include "mlir/Dialect/Shape/IR/Shape.h"
#include "mlir/Pass/PassManager.h"
#include "mlir/Transforms/GreedyPatternRewriteDriver.h"
#include "mlir/Transforms/Passes.h"
#include "mlir/include/mlir/IR/IRMapping.h"
#include "llvm/Support/Debug.h"
#include "llvm/Support/raw_ostream.h"

#include "tf_mlir_ext/transforms/passes_detail.h"
#include "tf_mlir_ext/transforms/remove_cstr_reshapable.h"

using namespace mlir;
using namespace llvm;

#define DEBUG_TYPE "remove_cstr_reshapable"

namespace {
/// Removal patterns.
class RemoveCstrReshapableOp : public OpRewritePattern<mhlo::CstrReshapableOp> {
public:
using OpRewritePattern::OpRewritePattern;

LogicalResult matchAndRewrite(mhlo::CstrReshapableOp op,
PatternRewriter &rewriter) const override {
rewriter.replaceOpWithNewOp<shape::ConstWitnessOp>(op.getOperation(), true);
return success();
}
};

/// Removal pass.
class RemoveCstrReshapablePass
: public RemoveCstrReshapableBase<RemoveCstrReshapablePass> {

void runOnOperation() override {
MLIRContext &ctx = getContext();

RewritePatternSet patterns(&ctx);
patterns.add<RemoveCstrReshapableOp>(patterns.getContext());

(void)applyPatternsAndFoldGreedily(getOperation(), std::move(patterns));
}
};

} // namespace

std::unique_ptr<OperationPass<func::FuncOp>>
mlir::tfext::createRemoveCstrReshapablePass() {
return std::make_unique<RemoveCstrReshapablePass>();
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
//===- remove_cstr_reshapable.h -------------------------------*--- C++ -*-===//
//
// Copyright 2022 ByteDance Ltd. and/or its affiliates. All rights reserved.
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
//===----------------------------------------------------------------------===//

#ifndef TFEXT_TRANSFORMS_REMOVE_CSTR_RESHAPABLE
#define TFEXT_TRANSFORMS_REMOVE_CSTR_RESHAPABLE

#include <memory>

#include "mlir/IR/MLIRContext.h" // from @llvm-project
#include "mlir/IR/PatternMatch.h" // from @llvm-project
#include "mlir/Pass/Pass.h" // from @llvm-project
#include "tensorflow/compiler/mlir/tensorflow/ir/tf_executor.h"

namespace mlir {
namespace tfext {

// -------------------------------------------
std::unique_ptr<OperationPass<func::FuncOp>> createRemoveCstrReshapablePass();

} // namespace tfext
} // namespace mlir

#endif // TFEXT_TRANSFORMS_REMOVE_CSTR_RESHAPABLE
9 changes: 8 additions & 1 deletion frontends/tf-frontend/tools/tf_frontend_main.cc
Original file line number Diff line number Diff line change
Expand Up @@ -119,6 +119,12 @@ static llvm::cl::opt<bool> keep_original_input_names(
llvm::cl::desc("put original input names in main func as an ArrayAttr"),
llvm::cl::init(false));

static llvm::cl::opt<bool> set_assuming_to_be_true(
"set-assuming-to-be-true",
llvm::cl::desc("remove cstr_reshapable and cstr_broadcastable,"
"and remove assuming"),
llvm::cl::init(true));

int main(int argc, char **argv) {
tensorflow::InitMlir y(&argc, &argv);

Expand Down Expand Up @@ -252,7 +258,8 @@ int main(int argc, char **argv) {
tf_frontend_manager.addPass(
::mlir::tfext::createCustomizedTfToMhloPipelinePass(
customcall_ops_array, remove_control_flow, staticalize_dynamic_shape,
stop_after_rewrite_customcall, additional_main_func_attrs));
stop_after_rewrite_customcall, additional_main_func_attrs,
set_assuming_to_be_true));
if (mlir::failed(tf_frontend_manager.run(*module))) {
llvm::outs() << "tf frontend customized-tf-to-mhlo pipeline failed\n";
return 1;
Expand Down
Loading