|
|
@@ -17,6 +17,7 @@ |
|
|
|
|
|
|
|
#include "graph/utils/node_utils.h" |
|
|
|
#include "graph/utils/tensor_utils.h" |
|
|
|
#include "graph/debug/ge_attr_define.h" |
|
|
|
|
|
|
|
namespace ge { |
|
|
|
const size_t kTwoInputNodesSize = 2; |
|
|
@@ -32,53 +33,110 @@ Status MarkAgnosticPass::Run(ComputeGraphPtr graph) { |
|
|
|
GELOGD("Op: %s, Index:0,has no input", node->GetName().c_str()); |
|
|
|
continue; |
|
|
|
} |
|
|
|
AttrUtils::SetInt(op_tensor, "_format_continuous", 1); |
|
|
|
AttrUtils::SetInt(node->GetOpDesc(), "_format_agnostic", 1); |
|
|
|
AttrUtils::SetListInt(node->GetOpDesc(), "_format_agnostic_except_input", std::vector<int64_t>({1})); |
|
|
|
AttrUtils::SetInt(op_tensor, ATTR_NAME_FORMAT_CONTINUOUS, 1); |
|
|
|
AttrUtils::SetInt(node->GetOpDesc(), ATTR_NAME_FORMAT_AGNOSTIC, 1); |
|
|
|
AttrUtils::SetListInt(node->GetOpDesc(), ATTR_NAME_FORMAT_AGNOSTIC_EXCEPT_INPUT, std::vector<int64_t>({1})); |
|
|
|
continue; |
|
|
|
} |
|
|
|
if (node_type == IDENTITY) { |
|
|
|
GELOGD("Mark format agnostic for identity node %s", node->GetName().c_str()); |
|
|
|
AttrUtils::SetInt(node->GetOpDesc(), "_format_agnostic", 1); |
|
|
|
AttrUtils::SetInt(node->GetOpDesc(), ATTR_NAME_FORMAT_AGNOSTIC, 1); |
|
|
|
continue; |
|
|
|
} |
|
|
|
if (node_type == REFMERGE || node_type == REFSWITCH) { |
|
|
|
GELOGD("Mark format agnostic for regmerge and refswitch node %s", node->GetName().c_str()); |
|
|
|
AttrUtils::SetInt(node->GetOpDesc(), "_format_agnostic", 1); |
|
|
|
AttrUtils::SetListInt(node->GetOpDesc(), "_format_agnostic_except_input", std::vector<int64_t>({1})); |
|
|
|
AttrUtils::SetInt(node->GetOpDesc(), ATTR_NAME_FORMAT_AGNOSTIC, 1); |
|
|
|
AttrUtils::SetListInt(node->GetOpDesc(), ATTR_NAME_FORMAT_AGNOSTIC_EXCEPT_INPUT, std::vector<int64_t>({1})); |
|
|
|
continue; |
|
|
|
} |
|
|
|
if (node_type == MERGE) { |
|
|
|
GELOGD("Mark format agnostic and continuous for merge node %s", node->GetName().c_str()); |
|
|
|
const auto &input_nodes = node->GetInAllNodes(); |
|
|
|
/// Enter-----------+ |
|
|
|
/// +-> Merge |
|
|
|
/// NextIteration---+ |
|
|
|
if (input_nodes.size() == kTwoInputNodesSize) { |
|
|
|
if (input_nodes.at(0)->GetType() == ENTER && input_nodes.at(1)->GetType() == NEXTITERATION) { |
|
|
|
continue; |
|
|
|
} |
|
|
|
} |
|
|
|
const OpDescPtr op_desc = node->GetOpDesc(); |
|
|
|
const GeTensorDescPtr op_tensor = op_desc->MutableOutputDesc(0); |
|
|
|
if (op_tensor == nullptr) { |
|
|
|
GELOGD("Op: %s, Index:0,has no output", node->GetName().c_str()); |
|
|
|
continue; |
|
|
|
} |
|
|
|
AttrUtils::SetInt(op_tensor, "_format_continuous", 1); |
|
|
|
|
|
|
|
// Merge----------->NetOutput only set format_cofntinuous attr |
|
|
|
const auto &output_nodes = node->GetOutAllNodes(); |
|
|
|
if (output_nodes.size() > 0) { |
|
|
|
// Always set continuous attr for merge output 0 |
|
|
|
GE_CHK_STATUS_RET(SetContinuousAttr(node, {0})); |
|
|
|
|
|
|
|
// Merge-->NetOutput only set merge output 0's continuous attr |
|
|
|
const auto &output_nodes = node->GetOutDataNodes(); |
|
|
|
if (!output_nodes.empty()) { |
|
|
|
if (output_nodes.at(0)->GetType() == NETOUTPUT) { |
|
|
|
continue; |
|
|
|
} |
|
|
|
} |
|
|
|
AttrUtils::SetInt(node->GetOpDesc(), "_format_agnostic", 1); |
|
|
|
AttrUtils::SetListInt(node->GetOpDesc(), "_format_agnostic_except_output", std::vector<int64_t>({1})); |
|
|
|
// Set format agnostic attr for merge in and out tensordesc |
|
|
|
AttrUtils::SetInt(node->GetOpDesc(), ATTR_NAME_FORMAT_AGNOSTIC, 1); |
|
|
|
AttrUtils::SetListInt(node->GetOpDesc(), ATTR_NAME_FORMAT_AGNOSTIC_EXCEPT_OUTPUT, std::vector<int64_t>({1})); |
|
|
|
|
|
|
|
// Set attr for enter and nextiteration |
|
|
|
if (HandWhileLoop(node) != SUCCESS) { |
|
|
|
GELOGE(FAILED, "Node: %s type merge handle while loop failed", node->GetName().c_str()); |
|
|
|
return FAILED; |
|
|
|
} |
|
|
|
continue; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
return SUCCESS; |
|
|
|
} |
|
|
|
|
|
|
|
bool MarkAgnosticPass::IsWhileLoop(const NodePtr &merge_node, NodePtr &enter, NodePtr &next) { |
|
|
|
auto node_type = NodeUtils::GetNodeType(*merge_node); |
|
|
|
if (node_type != MERGE) { |
|
|
|
GELOGW("Node %s type %s is not merge op.", merge_node->GetName().c_str(), node_type.c_str()); |
|
|
|
return false; |
|
|
|
} |
|
|
|
/// Enter-----------+ |
|
|
|
/// +-> Merge |
|
|
|
/// NextIteration---+ |
|
|
|
auto input_nodes = merge_node->GetInDataNodes(); |
|
|
|
if (input_nodes.size() != kTwoInputNodesSize) { |
|
|
|
GELOGD("Node %s type %s with [data input size[%zu]] is not enter-merge-nextiteration target.", |
|
|
|
merge_node->GetName().c_str(), node_type.c_str(), input_nodes.size()); |
|
|
|
return false; |
|
|
|
} |
|
|
|
auto in_node0 = input_nodes.at(0); |
|
|
|
auto in_node1 = input_nodes.at(1); |
|
|
|
auto in_type0 = NodeUtils::GetNodeType(in_node0); |
|
|
|
auto in_type1 = NodeUtils::GetNodeType(in_node1); |
|
|
|
if ((in_type0 != ENTER || in_type1 != NEXTITERATION) && (in_type0 != NEXTITERATION || in_type1 != ENTER)) { |
|
|
|
GELOGD("Node %s type %s with [data input0's type %s input1's type %s] is not enter-merge-nextiteration target.", |
|
|
|
merge_node->GetName().c_str(), node_type.c_str(), in_type0.c_str(), in_type1.c_str()); |
|
|
|
return false; |
|
|
|
} |
|
|
|
enter = in_node0; |
|
|
|
next = in_node1; |
|
|
|
return true; |
|
|
|
} |
|
|
|
|
|
|
|
Status MarkAgnosticPass::HandWhileLoop(const NodePtr &node) { |
|
|
|
NodePtr enter = nullptr; |
|
|
|
NodePtr next = nullptr; |
|
|
|
if (!IsWhileLoop(node, enter, next)) { |
|
|
|
return SUCCESS; |
|
|
|
} |
|
|
|
GE_CHECK_NOTNULL(enter); |
|
|
|
GE_CHECK_NOTNULL(next); |
|
|
|
// Set continuous attr |
|
|
|
GE_CHK_STATUS_RET(SetContinuousAttr(enter, {0})); |
|
|
|
GE_CHK_STATUS_RET(SetContinuousAttr(next, {0})); |
|
|
|
// Set format agnostic attr |
|
|
|
(void)AttrUtils::SetInt(enter->GetOpDesc(), ATTR_NAME_FORMAT_AGNOSTIC, 1); |
|
|
|
(void)AttrUtils::SetInt(next->GetOpDesc(), ATTR_NAME_FORMAT_AGNOSTIC, 1); |
|
|
|
|
|
|
|
return SUCCESS; |
|
|
|
} |
|
|
|
|
|
|
|
Status MarkAgnosticPass::SetContinuousAttr(const NodePtr &node, const std::vector<uint32_t> &indexes) { |
|
|
|
auto op_desc = node->GetOpDesc(); |
|
|
|
GE_CHECK_NOTNULL(op_desc); |
|
|
|
// This flag is for fe performance optimization |
|
|
|
(void)AttrUtils::SetBool(op_desc, ATTR_NAME_REFRESH_CONTINUOUS_FLAG, true); |
|
|
|
for (auto index : indexes) { |
|
|
|
auto out = op_desc->MutableOutputDesc(index); |
|
|
|
GE_CHECK_NOTNULL(out); |
|
|
|
// This attr is for out's dtype and format continuous with it's peer input |
|
|
|
(void)AttrUtils::SetInt(out, ATTR_NAME_FORMAT_CONTINUOUS, 1); |
|
|
|
} |
|
|
|
|
|
|
|
return SUCCESS; |
|
|
|
} |
|
|
|
} // namespace ge |