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