You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
graphengine/ge/graph/passes/cast_remove_pass.cc

151 lines
4.8 KiB

/**
* Copyright 2020 Huawei Technologies Co., Ltd
*
* 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 "graph/passes/cast_remove_pass.h"
#include <string>
#include <vector>
#include "framework/common/debug/ge_log.h"
#include "graph/common/transop_util.h"
#include "graph/debug/ge_attr_define.h"
#include "graph/utils/type_utils.h"
namespace ge {
Status CastRemovePass::Run(NodePtr &node) {
if (node == nullptr) {
GELOGE(PARAM_INVALID, "Param [node] must not be null.");
return PARAM_INVALID;
}
OpDescPtr op_desc = node->GetOpDesc();
if (op_desc == nullptr) {
GELOGE(PARAM_INVALID, "OpDesc of param [node] must not be null.");
return PARAM_INVALID;
}
// begin with not trans op, and only has one out data anchor
if (TransOpUtil::IsTransOp(node) || node->GetAllOutDataAnchorsSize() != 1) {
return SUCCESS;
}
std::vector<NodePtr> nodes_to_fuse;
NodePtr end_node = GetTheEndNode(node, nodes_to_fuse);
if (nodes_to_fuse.empty()) {
return SUCCESS;
}
OpDescPtr end_op_desc = end_node->GetOpDesc();
if (end_op_desc == nullptr) {
GELOGE(PARAM_INVALID, "OpDesc of end node must not be null.");
return PARAM_INVALID;
}
if (!CheckPrecisionLoss(nodes_to_fuse)) {
return SUCCESS;
}
DataType type = DT_UNDEFINED;
if (!HasSameDataType(op_desc, end_op_desc, type)) {
return SUCCESS;
}
if (RemoveCast(type, nodes_to_fuse) != SUCCESS) {
return FAILED;
}
return SUCCESS;
}
bool CastRemovePass::CheckPrecisionLoss(const std::vector<NodePtr> &nodes_to_fuse) {
for (const NodePtr &node : nodes_to_fuse) {
if (!TransOpUtil::CheckPrecisionLoss(node)) {
return false;
}
}
return true;
}
bool CastRemovePass::HasSameDataType(OpDescPtr &begin_op_desc, OpDescPtr &end_op_desc, DataType &type) const {
if (begin_op_desc->GetName() == end_op_desc->GetName()) {
return false;
}
auto end_out_desc = end_op_desc->MutableOutputDesc(0);
DataType end_out_datatype = end_out_desc->GetDataType();
auto begin_out_desc = begin_op_desc->MutableOutputDesc(0);
DataType begin_out_datatype = begin_out_desc->GetDataType();
if (begin_out_datatype == end_out_datatype && (begin_out_datatype == DT_FLOAT16 || begin_out_datatype == DT_FLOAT)) {
type = begin_out_datatype;
return true;
}
return false;
}
// op1->TransData->Cast->TransposeD->Cast->TransData->op2
// change to be
// op1->TransData->TransposeD->TransData->op2
Status CastRemovePass::RemoveCast(DataType &type, std::vector<NodePtr> &nodes_to_fuse) {
string cast_name;
for (NodePtr &node : nodes_to_fuse) {
if (node->GetType() == CAST) {
GELOGI("CastRemovePass, remove Cast %s.", node->GetName().c_str());
cast_name = node->GetName();
if (IsolateAndDeleteNode(node, {0}) != SUCCESS) {
GELOGE(FAILED, "IsolateAndDeleteNode %s failed.", node->GetName().c_str());
return FAILED;
}
}
}
if (cast_name.empty()) {
return SUCCESS;
}
for (auto &node : nodes_to_fuse) {
if (node->GetType() == CAST) {
continue;
}
OpDescPtr op_desc = node->GetOpDesc();
if (op_desc == nullptr) {
GELOGE(FAILED, "OpDesc must not be null.");
return FAILED;
}
// change node name for recompile cache, will be abandoned in April
string new_node_name = cast_name + op_desc->GetName();
op_desc->SetName(new_node_name);
// add attr to changed TransData, then will be rebuild
if (!AttrUtils::SetBool(op_desc, ATTR_NEED_COMPILE, true)) {
GELOGE(FAILED, "Set ATTR_NEED_COMPILE Attr fail.");
return FAILED;
}
auto in_desc = op_desc->MutableInputDesc(0);
auto out_desc = op_desc->MutableOutputDesc(0);
in_desc->SetDataType(type);
out_desc->SetDataType(type);
GELOGI("CastRemovePass, change %s %s datatype to be %s.", node->GetType().c_str(), node->GetName().c_str(),
TypeUtils::DataTypeToSerialString(type).c_str());
}
return SUCCESS;
}
NodePtr CastRemovePass::GetTheEndNode(NodePtr begin_node, std::vector<NodePtr> &nodes_to_fuse) {
while (begin_node->GetOutDataNodes().size() == 1) {
auto out_node = begin_node->GetOutDataNodes().at(0);
if (!TransOpUtil::IsTransOp(out_node)) {
return begin_node; // when seen not trans op
}
begin_node = out_node;
nodes_to_fuse.emplace_back(begin_node);
}
return begin_node; // when seen branch
}
} // namespace ge