* Copyright (c) 2025 Huawei Technologies Co., Ltd.
* This program is free software, you can redistribute it and/or modify it under the terms and conditions of
* CANN Open Software License Agreement Version 2.0 (the "License").
* Please refer to the License for details. You may not use this file except in compliance with the License.
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED,
* INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY, OR FITNESS FOR A PARTICULAR PURPOSE.
* See LICENSE in the root of the software repository for the full text of the License.
*/
* \file iso_paritioner_checker.cpp
* \brief
*/
#include "iso_partitioner_checker.h"
#include "passes/pass_log/pass_log.h"
#define MODULE_NAME "GraphPartition"
namespace npu {
namespace tile_fwk {
Status GraphPartitionChecker::DoPreCheck(Function& function)
{
APASS_LOG_INFO_F(Elements::Function, "PreCheck for GraphPartition.");
if (CheckValidOp(function) != SUCCESS) {
APASS_LOG_ERROR_F(Elements::Function, "CheckValidOp failed; Please check the CheckValidOp method.");
return FAILED;
}
return SUCCESS;
}
Status GraphPartitionChecker::DoPostCheck(Function& function)
{
APASS_LOG_INFO_F(Elements::Function, "PostCheck for GraphPartition.");
std::vector<std::vector<Operation*>> subgraphs(function.GetTotalSubGraphCount());
for (auto& op : function.Operations()) {
int32_t curSubgraphID = op.GetSubgraphID();
if (curSubgraphID == -1) {
APASS_LOG_ERROR_F(
Elements::Operation,
"Operation (opmagic: %d) is not in any subgraph; Please review the error messages generated during the "
"processing procedure.%s",
op.GetOpMagic(), GetFormatBacktrace(op).c_str());
return FAILED;
}
if (curSubgraphID < 0 || curSubgraphID >= static_cast<int32_t>(function.GetTotalSubGraphCount())) {
APASS_LOG_ERROR_F(
Elements::Operation,
"Operation (opmagic: %d) has illegal SubgraphID; Please review the error messages generated during the "
"processing procedure.%s",
op.GetOpMagic(), GetFormatBacktrace(op).c_str());
return FAILED;
}
subgraphs[curSubgraphID].push_back(&op);
}
for (int graphID = 0; graphID < static_cast<int>(subgraphs.size()); graphID++) {
if (subgraphs[graphID].size() == 0) {
APASS_LOG_ERROR_F(Elements::Operation, "Subgraph %d includes no Operation.", graphID);
return FAILED;
}
}
if (!function.LoopCheck().empty()) {
APASS_LOG_ERROR_F(Elements::Function, "Loopcheck failed after GraphPartition.");
return FAILED;
}
if (PostOperationCheck(function) != SUCCESS) {
APASS_LOG_ERROR_F(Elements::Function, "Operation post check failed.");
return FAILED;
}
return SUCCESS;
}
Status GraphPartitionChecker::PostOperationCheck(Function& function)
{
for (auto& op : function.Operations()) {
int32_t curSubgraphID = op.GetSubgraphID();
bool isStartNodeInSubgraph = true;
for (auto iTensor : op.GetIOperands()) {
for (auto& producer : iTensor->GetProducers()) {
if (curSubgraphID == producer->GetSubgraphID()) {
isStartNodeInSubgraph = false;
break;
}
}
if (!isStartNodeInSubgraph) {
break;
}
}
if (isStartNodeInSubgraph) {
APASS_LOG_DEBUG_F(
Elements::Operation, "Operation (opmagic: %d) is start node in subgraph.", op.GetOpMagic());
if (op.GetOpcode() == Opcode::OP_ASSEMBLE || op.GetOpcode() == Opcode::OP_COPY_OUT) {
APASS_LOG_ERROR_F(
Elements::Operation,
"Operation (opmagic: %d) is the start node of the subgraph, opcode should not be %s.%s",
op.GetOpMagic(), op.GetOpcodeStr().c_str(), GetFormatBacktrace(op).c_str());
return FAILED;
}
}
bool isEndNodeInSubgraph = true;
for (auto oTensor : op.GetOOperands()) {
for (auto& consumer : oTensor->GetConsumers()) {
if (curSubgraphID == consumer->GetSubgraphID()) {
isEndNodeInSubgraph = false;
break;
}
}
if (!isEndNodeInSubgraph) {
break;
}
}
if (isEndNodeInSubgraph) {
APASS_LOG_DEBUG_F(Elements::Operation, "Operation (opmagic: %d) is end node in subgraph.", op.GetOpMagic());
if (op.GetOpcode() == Opcode::OP_VIEW || op.GetOpcode() == Opcode::OP_COPY_IN) {
APASS_LOG_ERROR_F(
Elements::Operation,
"Operation (opmagic: %d) is the end node of the subgraph, opcode should not be %s.%s",
op.GetOpMagic(), op.GetOpcodeStr().c_str(), GetFormatBacktrace(op).c_str());
return FAILED;
}
}
}
return SUCCESS;
}
}
}