• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /**
2  * Copyright 2021 Huawei Technologies Co., Ltd
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  * http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "backend/kernel_compiler/tbe/tbe_json/fusion_tbe_json_creator.h"
18 #include <algorithm>
19 #include <string>
20 #include <vector>
21 #include "base/core_ops.h"
22 #include "backend/session/anf_runtime_algorithm.h"
23 #include "backend/kernel_compiler/tbe/tbe_adapter.h"
24 #include "backend/kernel_compiler/tbe/tbe_convert_utils.h"
25 #include "backend/kernel_compiler/tbe/tbe_dynaminc_shape_util.h"
26 #include "backend/kernel_compiler/tbe/tbe_utils.h"
27 #include "runtime/dev.h"
28 #include "utils/json_operation_utils.h"
29 #include "backend/kernel_compiler/tbe/tbe_json/tbe_json_utils.h"
30 
31 namespace mindspore::kernel {
32 using mindspore::kernel::tbe::TbeAdapter;
GenJson(const FusionScopeInfo & fusion_scope_info,nlohmann::json * fusion_json)33 bool FusionBuildTbeJsonCreator::GenJson(const FusionScopeInfo &fusion_scope_info, nlohmann::json *fusion_json) {
34   MS_EXCEPTION_IF_NULL(fusion_json);
35 
36   MS_LOG(DEBUG) << "Start Generate Fusion Json, Fusion Node: " << fusion_scope_info.full_name;
37   nlohmann::json soc_info_json = kernel::tbe::TbeUtils::GenSocInfo();
38   (*fusion_json)[kJSocInfo] = soc_info_json;
39 
40   std::vector<nlohmann::json> op_list_json;
41   if (!GenOpListJson(fusion_scope_info, &op_list_json)) {
42     MS_LOG(WARNING) << "Fusion Error: generate fusion json failed.";
43     return false;
44   }
45   (*fusion_json)[kJOpList] = op_list_json;
46 
47   GenFusionOpName(fusion_json, kJFusionKernelNamePrefix);
48   AddOpNameForComputeNode(fusion_json);
49   (*fusion_json)[kJL1Size] = -1;
50   (*fusion_json)[kJGraphName] = "";
51   (*fusion_json)[kJScopeID] = fusion_scope_info.scope_id;
52   (*fusion_json)[kJFullName] = fusion_scope_info.full_name;
53   MS_LOG(DEBUG) << "Json name is : " << GetJsonName() << ", fusion json:" << fusion_json->dump();
54   return true;
55 }
56 
GenOpListJson(const FusionScopeInfo & fusion_scope_info,std::vector<nlohmann::json> * fusion_json)57 bool FusionBuildTbeJsonCreator::GenOpListJson(const FusionScopeInfo &fusion_scope_info,
58                                               std::vector<nlohmann::json> *fusion_json) {
59   MS_EXCEPTION_IF_NULL(fusion_json);
60   MS_LOG(DEBUG) << "Start";
61   if (!CheckInput(fusion_scope_info)) {
62     for (const auto &cnode : fusion_scope_info.compute_nodes) {
63       MS_LOG(WARNING) << "Fusion Error: check input failed, scope id: " << fusion_scope_info.scope_id
64                       << ", compute node: " << cnode->fullname_with_scope();
65     }
66     return false;
67   }
68 
69   optional_index_ = 0;
70   auto compute_nodes = fusion_scope_info.compute_nodes;
71   std::vector<nlohmann::json> compute_list;
72   for (const auto &compute_node : compute_nodes) {
73     nlohmann::json compute_json;
74     if (!GenComputeJson(compute_node, &compute_json)) {
75       MS_LOG(WARNING) << "Fusion Error: gen fusion compute json failed. node full name: "
76                       << compute_node->fullname_with_scope();
77       return false;
78     }
79     compute_json[kJOriName] = {fusion_scope_info.full_name};
80     compute_list.push_back(compute_json);
81   }
82 
83   // FusionDataType fusion_data_type: speceial process json desc output shape [kFusionAddN, kFusionReLUGradV2]
84   ANodeFusionDataTypeMap spec_data_input;
85   if (!TbeAdapter::GetSpecDataInput(fusion_scope_info, &spec_data_input)) {
86     return false;
87   }
88   GenDataJson(compute_nodes, compute_list, fusion_json, spec_data_input);
89   (*fusion_json).insert((*fusion_json).end(), compute_list.begin(), compute_list.end());
90   MS_LOG(DEBUG) << "End";
91   return true;
92 }
93 
CheckInput(const FusionScopeInfo & fusion_scope_info)94 bool FusionBuildTbeJsonCreator::CheckInput(const FusionScopeInfo &fusion_scope_info) {
95   MS_LOG(DEBUG) << "Start";
96   auto input_nodes = fusion_scope_info.input_nodes;
97   auto compute_nodes = fusion_scope_info.compute_nodes;
98   size_t input_size = 0;
99   for (const auto &node : compute_nodes) {
100     MS_EXCEPTION_IF_NULL(node);
101     auto cnode = node->cast<CNodePtr>();
102     if (cnode == nullptr) {
103       MS_LOG(WARNING) << "Fusion Error: fusion compute node must be cnode, but the node is " << cnode->DebugString();
104       return false;
105     }
106     for (size_t i = 1; i < cnode->inputs().size(); ++i) {
107       auto input = cnode->input(i);
108       auto find_iter = std::find(input_nodes.begin(), input_nodes.end(), input);
109       if (find_iter != input_nodes.end()) {
110         input_size++;
111       }
112     }
113   }
114   if (input_nodes.size() != input_size) {
115     MS_LOG(WARNING) << "Fusion Error: compute node input size: [ " << input_size
116                     << " ] is not equal to input nodes num: [ " << input_nodes.size() << " ].";
117     return false;
118   }
119   MS_LOG(DEBUG) << "End";
120   return true;
121 }
122 
GenDataJson(const std::vector<AnfNodePtr> & compute_nodes,const std::vector<nlohmann::json> & compute_json,std::vector<nlohmann::json> * op_list_json,const ANodeFusionDataTypeMap & spec_data_input)123 void FusionBuildTbeJsonCreator::GenDataJson(const std::vector<AnfNodePtr> &compute_nodes,
124                                             const std::vector<nlohmann::json> &compute_json,
125                                             std::vector<nlohmann::json> *op_list_json,
126                                             const ANodeFusionDataTypeMap &spec_data_input) {
127   MS_EXCEPTION_IF_NULL(op_list_json);
128   MS_LOG(DEBUG) << "Start.";
129   std::vector<std::string> compute_nodes_fullname;
130   std::transform(compute_nodes.begin(), compute_nodes.end(), back_inserter(compute_nodes_fullname),
131                  [](const AnfNodePtr &node) { return node->fullname_with_scope(); });
132   for (size_t i = 0; i < compute_nodes.size(); i++) {
133     auto inputs_desc = GetJsonValue<std::vector<nlohmann::json>>(compute_json.at(i), kJInputDesc);
134     for (const auto &input_desc : inputs_desc) {
135       if (std::find(compute_nodes_fullname.begin(), compute_nodes_fullname.end(),
136                     GetJsonValue<std::string>(input_desc, kJName)) != compute_nodes_fullname.end()) {
137         continue;
138       }
139       nlohmann::json data_json;
140       nlohmann::json output_desc = input_desc;
141       std::vector<nlohmann::json> output_desc_list;
142       if (input_desc.find(kJOriShape) != input_desc.end()) {
143         auto input_node = GetInputCNode(compute_nodes[i], output_desc);
144         TbeAdapter::FusionDescJsonPass(input_node, &output_desc, spec_data_input);
145       }
146       output_desc_list.push_back(output_desc);
147       data_json[kJName] = GetJsonValue<std::string>(input_desc, kJName);
148       data_json[kJType] = kJData;
149       data_json[kJOutputDesc] = output_desc_list;
150       (*op_list_json).push_back(data_json);
151     }
152   }
153   MS_LOG(DEBUG) << "End.";
154 }
GetInputCNode(const AnfNodePtr & node,const nlohmann::json & input_desc)155 AnfNodePtr FusionBuildTbeJsonCreator::GetInputCNode(const AnfNodePtr &node, const nlohmann::json &input_desc) {
156   auto input_name = GetJsonValue<std::string>(input_desc, kJName);
157   auto cnode = node->cast<CNodePtr>();
158   MS_EXCEPTION_IF_NULL(cnode);
159   for (size_t i = 1; i < cnode->inputs().size(); i++) {
160     auto kernel_idx = AnfAlgo::VisitKernel(cnode->input(i), 0);
161     auto full_name = kernel_idx.first->fullname_with_scope();
162     std::string desc_name = kernel_idx.second > 0 ? (full_name + "_" + std::to_string(kernel_idx.second)) : full_name;
163     if (input_name == desc_name) {
164       return cnode->input(i);
165     }
166   }
167   MS_LOG(EXCEPTION) << "Can not find node:[" << node->fullname_with_scope() << "]'s input [" << input_name << "]";
168 }
169 
GenInputsJson(const AnfNodePtr & anf_node,nlohmann::json * compute_json)170 bool FusionBuildTbeJsonCreator::GenInputsJson(const AnfNodePtr &anf_node, nlohmann::json *compute_json) {
171   MS_EXCEPTION_IF_NULL(anf_node);
172   MS_EXCEPTION_IF_NULL(compute_json);
173   std::vector<nlohmann::json> input_desc_list_tmp = {};
174   auto op_name = AnfAlgo::GetCNodeName(anf_node);
175   auto cnode = anf_node->cast<CNodePtr>();
176   MS_EXCEPTION_IF_NULL(cnode);
177   bool is_dynamic_input = AnfAlgo::HasNodeAttr(kAttrDynInputSizes, cnode);
178   if (is_dynamic_input) {
179     MS_LOG(INFO) << op_name << " has dynamic input.";
180     if (!CheckDynamicInput(cnode)) {
181       return false;
182     }
183   }
184   size_t input_index = 0;
185   for (size_t i = 1; i < cnode->inputs().size(); ++i) {
186     auto input = cnode->input(i);
187     if (HasAbstractMonad(input)) {
188       continue;
189     }
190     auto kernel_idx = AnfAlgo::VisitKernel(input, 0);
191     nlohmann::json input_desc;
192     GenDescJson(kernel_idx.first, kernel_idx.second, kernel_idx.second, &input_desc);
193     GenInputConstValue(anf_node, i - 1, &input_desc);
194     if (is_dynamic_input) {
195       input_desc[kJDynIndex] = (i - 1);
196     }
197     input_desc_list_tmp.emplace_back(input_desc);
198     input_index++;
199   }
200   std::vector<size_t> inputs_tensor_num;
201   auto op_info = tbe::TbeDynamicShapeUtil::FindOp(op_name, anf_node);
202   if (!TbeJsonUtils::GetInputsRealNum(anf_node, op_info->inputs_ptr(), &inputs_tensor_num)) {
203     return false;
204   }
205   size_t need_input_num = std::accumulate(inputs_tensor_num.begin(), inputs_tensor_num.end(), static_cast<size_t>(0));
206 
207   for (size_t i = input_index; i < need_input_num; ++i) {
208     nlohmann::json optional_input_desc;
209     optional_input_desc[kJName] = std::string(kJOptional) + std::to_string(optional_index_);
210     optional_input_desc[kJShape] = kJNull;
211     optional_input_desc[kJDataType] = 0;
212     optional_index_++;
213     input_desc_list_tmp.emplace_back(optional_input_desc);
214   }
215   std::vector<nlohmann::json> input_desc_list;
216   TbeAdapter::InputOrderPass<nlohmann::json>(anf_node, input_desc_list_tmp, &input_desc_list);
217   (*compute_json)[kJInputDesc] = input_desc_list;
218   return true;
219 }
220 
CheckDynamicInput(const CNodePtr & cnode)221 bool FusionBuildTbeJsonCreator::CheckDynamicInput(const CNodePtr &cnode) {
222   MS_EXCEPTION_IF_NULL(cnode);
223   if (!AnfAlgo::HasNodeAttr(kAttrDynInputSizes, cnode)) {
224     MS_LOG(WARNING) << "Fusion Error: cnode [ " << AnfAlgo::GetCNodeName(cnode) << "] has not attr dyn_input_sizes.";
225     return false;
226   }
227   // for dynamic input number, dyn_input_sizes has the info of dynamic input num for each input.
228   auto dyn_input_sizes = AnfAlgo::GetNodeAttr<std::vector<int64_t>>(cnode, kAttrDynInputSizes);
229   if (dyn_input_sizes.size() != 1) {
230     MS_LOG(WARNING) << "Fusion Error: fusion build not support dynamic input size > 1";
231     return false;
232   }
233   auto real_input_size = cnode->inputs().size() - 1;
234   if (LongToSize(dyn_input_sizes[0]) != real_input_size) {
235     MS_LOG(WARNING) << "Fusion Error: dyn_input_size" << dyn_input_sizes[0] << "not equal real_input_size"
236                     << real_input_size;
237     return false;
238   }
239   return true;
240 }
241 
GenOutputsJson(const AnfNodePtr & anf_node,nlohmann::json * compute_json)242 bool FusionBuildTbeJsonCreator::GenOutputsJson(const AnfNodePtr &anf_node, nlohmann::json *compute_json) {
243   MS_EXCEPTION_IF_NULL(anf_node);
244   MS_EXCEPTION_IF_NULL(compute_json);
245   auto output_size = AnfAlgo::GetOutputTensorNum(anf_node);
246   auto cnode = anf_node->cast<CNodePtr>();
247   MS_EXCEPTION_IF_NULL(cnode);
248   std::vector<nlohmann::json> output_desc_list;
249   if (AnfAlgo::HasNodeAttr(kAttrOutputUsedNum, cnode)) {
250     auto output_used_nums = AnfAlgo::GetNodeAttr<std::vector<int64_t>>(anf_node, kAttrOutputUsedNum);
251     if (output_used_nums.size() != output_size) {
252       MS_LOG(WARNING) << "Fusion Error: [" << AnfAlgo::GetCNodeName(anf_node) << " ]'s output tensor num("
253                       << output_size << ")"
254                       << " is not match output used num(" << output_used_nums.size() << ")";
255       return false;
256     }
257     auto desc_output_index = GetDescOutputIndex(output_used_nums);
258     for (size_t i = 0; i < output_size; ++i) {
259       MS_LOG(DEBUG) << "Fusion index: " << i << ", desc_output_index: " << desc_output_index[i];
260       nlohmann::json output_desc;
261       GenDescJson(anf_node, i, desc_output_index[i], &output_desc);
262       output_desc_list.emplace_back(output_desc);
263     }
264     for (size_t j = output_size; j < desc_output_index.size(); ++j) {
265       MS_LOG(DEBUG) << "Fusion index: " << j << ", desc_output_index: " << desc_output_index[j];
266       nlohmann::json output_desc;
267       GenReusedOutputDesc(anf_node, j, desc_output_index[j], &output_desc, output_size);
268       output_desc_list.emplace_back(output_desc);
269     }
270   } else {
271     for (size_t i = 0; i < output_size; ++i) {
272       nlohmann::json output_desc;
273       GenDescJson(anf_node, i, i, &output_desc);
274       output_desc_list.emplace_back(output_desc);
275     }
276   }
277   (*compute_json)[kJOutputDesc] = output_desc_list;
278   return true;
279 }
280 
GenReusedOutputDesc(const AnfNodePtr & anf_node,size_t index,size_t output_index,nlohmann::json * output_desc,size_t out_size)281 void FusionBuildTbeJsonCreator::GenReusedOutputDesc(const AnfNodePtr &anf_node, size_t index, size_t output_index,
282                                                     nlohmann::json *output_desc, size_t out_size) {
283   GenDesJsonCommon(output_desc);
284   std::string output_desc_name = anf_node->fullname_with_scope() + "_" + std::to_string(index);
285   (*output_desc)[kJName] = output_desc_name;
286   (*output_desc)[kJOutputIndex] = output_index;
287   std::vector<size_t> shape;
288   (*output_desc)[kJShape] = shape;
289   (*output_desc)[kJDataType] = tbe::TypeIdToString(AnfAlgo::GetOutputDeviceDataType(anf_node, out_size - 1));
290 }
291 
GetDescOutputIndex(const std::vector<int64_t> & output_used_nums)292 std::vector<size_t> FusionBuildTbeJsonCreator::GetDescOutputIndex(const std::vector<int64_t> &output_used_nums) {
293   std::vector<size_t> desc_output_index = {};
294   for (size_t idx = 0; idx < output_used_nums.size(); ++idx) {
295     desc_output_index.emplace_back(idx);
296     if (output_used_nums[idx] > 1) {
297       desc_output_index.emplace_back(idx);
298     }
299   }
300   return desc_output_index;
301 }
302 
AttrsJsonPostProcessing(const AnfNodePtr & anf_node,const OpInfoPtr & op_info_ptr,nlohmann::json * attrs_json)303 bool FusionBuildTbeJsonCreator::AttrsJsonPostProcessing(const AnfNodePtr &anf_node, const OpInfoPtr &op_info_ptr,
304                                                         nlohmann::json *attrs_json) {
305   // just keep it
306   // tbe::TbeAdapter::CastAttrJsonPost(anf_node, attrs_json);
307   return true;
308 }
309 
GenOtherJson(const AnfNodePtr & anf_node,nlohmann::json * compute_json)310 void FusionBuildTbeJsonCreator::GenOtherJson(const AnfNodePtr &anf_node, nlohmann::json *compute_json) {
311   MS_EXCEPTION_IF_NULL(anf_node);
312   MS_EXCEPTION_IF_NULL(compute_json);
313   (*compute_json)[kJUnknowShape] = tbe::TbeDynamicShapeUtil::GetDynamicShapeAttr(anf_node);
314 }
315 }  // namespace mindspore::kernel
316