You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

ge_generator.cc 37 kB

5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
4 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
4 years ago
5 years ago
5 years ago
4 years ago
4 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
4 years ago
4 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
5 years ago
4 years ago
5 years ago
5 years ago
5 years ago
5 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957
  1. /**
  2. * Copyright 2020 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. #include "generator/ge_generator.h"
  17. #include <atomic>
  18. #include "common/ge/ge_util.h"
  19. #include "common/ge/plugin_manager.h"
  20. #include "common/helper/model_helper.h"
  21. #include "common/helper/om_file_helper.h"
  22. #include "common/util.h"
  23. #include "common/util/error_manager/error_manager.h"
  24. #include "framework/common/debug/ge_log.h"
  25. #include "framework/common/debug/log.h"
  26. #include "ge/ge_api.h"
  27. #include "graph/debug/ge_attr_define.h"
  28. #include "graph/ge_context.h"
  29. #include "graph/manager/graph_manager.h"
  30. #include "graph/manager/util/rt_context_util.h"
  31. #include "graph/opsproto_manager.h"
  32. #include "graph/utils/graph_utils.h"
  33. #include "graph/utils/type_utils.h"
  34. #include "init/gelib.h"
  35. #include "model/ge_model.h"
  36. using std::map;
  37. using std::string;
  38. using std::vector;
  39. namespace {
  40. const char *const kAttrOpType = "op_type";
  41. const char *const kEngineNameDefault = "default";
  42. const char *const kVectorEngine = "VectorEngine";
  43. const char *const kAIcoreEngine = "AIcoreEngine";
  44. const char *const kFileNameSuffix = "online";
  45. const char *const kAicpuAllshape = "_AllShape";
  46. constexpr char const *kAttrSupportDynamicShape = "support_dynamicshape";
  47. const int64_t kDynamicDimValue = -2;
  48. std::map<ge::OpEngineType, std::string> engine_type_map{
  49. {ge::ENGINE_SYS, kEngineNameDefault},
  50. {ge::ENGINE_AICORE, kAIcoreEngine},
  51. {ge::ENGINE_VECTOR, kVectorEngine}};
  52. bool ContainsDynamicInpus(const ge::OpDesc &op_desc) {
  53. for (auto &tensor_desc : op_desc.GetAllInputsDescPtr()) {
  54. if (tensor_desc->MutableShape().IsUnknownShape()) {
  55. GELOGI("Contains unknown shape input. set is_dynamic_input to true.");
  56. return true;
  57. }
  58. }
  59. return false;
  60. }
  61. } // namespace
  62. namespace ge {
  63. static Status CheckEngineTypeSupport(const NodePtr &node, OpEngineType engine_type) {
  64. const OpDescPtr &op_desc = node->GetOpDesc();
  65. GE_CHECK_NOTNULL_EXEC(op_desc, return PARAM_INVALID);
  66. if (engine_type == ENGINE_SYS) {
  67. GELOGI("CheckEngineType: use default engine.");
  68. return SUCCESS;
  69. }
  70. // get op engine name
  71. string op_engine_name;
  72. auto iter = engine_type_map.find(engine_type);
  73. if (iter != engine_type_map.end()) {
  74. op_engine_name = iter->second;
  75. GELOGI("CheckEngineType: engine type: %d", static_cast<int>(engine_type));
  76. } else {
  77. ErrorManager::GetInstance().ATCReportErrMessage("E14001", {"opname", "optype", "value", "reason"},
  78. {op_desc->GetName(), op_desc->GetType(), "engine type",
  79. "it only support kEngineNameDefault/kAIcoreEngine/kVectorEngine"});
  80. GELOGE(FAILED, "CheckEngineType: engine type: %d not support.", static_cast<int>(engine_type));
  81. return FAILED;
  82. }
  83. if (op_desc->HasAttr(ATTR_NAME_UNREGST_OPPATH)) {
  84. op_desc->SetOpEngineName(op_engine_name);
  85. op_desc->SetOpKernelLibName(op_engine_name);
  86. return SUCCESS;
  87. }
  88. // set op engine name and opkernelLib. when engine support
  89. std::shared_ptr<GELib> instance_ptr = ge::GELib::GetInstance();
  90. if ((instance_ptr == nullptr) || (!instance_ptr->InitFlag())) {
  91. GELOGE(GE_CLI_GE_NOT_INITIALIZED, "CheckEngineType failed.");
  92. return FAILED;
  93. }
  94. OpsKernelManager &ops_kernel_manager = instance_ptr->OpsKernelManagerObj();
  95. std::vector<OpInfo> op_infos = ops_kernel_manager.GetOpsKernelInfo(op_desc->GetType());
  96. if (op_infos.empty()) {
  97. ErrorManager::GetInstance().ATCReportErrMessage("E14001", {"opname", "optype", "value", "reason"},
  98. {op_desc->GetName(), op_desc->GetType(), "optype", "it can not find"});
  99. GELOGE(FAILED, "CheckEngineType: Can not get op info by op type %s", op_desc->GetType().c_str());
  100. return FAILED;
  101. }
  102. string kernel_name;
  103. for (const auto &it : op_infos) {
  104. if (it.engine == op_engine_name) {
  105. kernel_name = it.opKernelLib;
  106. break;
  107. }
  108. }
  109. if (kernel_name.empty()) {
  110. ErrorManager::GetInstance().ATCReportErrMessage("E14001", {"opname", "optype", "value", "reason"},
  111. {op_desc->GetName(), op_desc->GetType(), "engine name" + FmtToStr(op_engine_name), "it can not find"});
  112. GELOGE(FAILED, "CheckEngineType:Can not find ops kernel, engine name: %s.", op_engine_name.c_str());
  113. return FAILED;
  114. }
  115. auto &kernel_map = ops_kernel_manager.GetAllOpsKernelInfoStores();
  116. auto kernel_info_store = kernel_map.find(kernel_name);
  117. if (kernel_info_store != kernel_map.end()) {
  118. std::string unsupported_reason;
  119. if (kernel_info_store->second->CheckSupported(node, unsupported_reason)) {
  120. op_desc->SetOpEngineName(op_engine_name);
  121. op_desc->SetOpKernelLibName(kernel_name);
  122. GELOGI("CheckEngineType:Set OpKernelLibName %s and engine name %s into op_desc %s", kernel_name.c_str(),
  123. op_engine_name.c_str(), op_desc->GetName().c_str());
  124. return SUCCESS;
  125. } else {
  126. ErrorManager::GetInstance().ATCReportErrMessage(
  127. "E13002", {"optype", "opskernel", "reason"}, {op_desc->GetType(), kernel_name, unsupported_reason});
  128. GELOGE(FAILED, "CheckEngineType: check support failed, Op type %s of ops kernel %s is unsupported, reason:%s",
  129. op_desc->GetType().c_str(), kernel_name.c_str(), unsupported_reason.c_str());
  130. return FAILED;
  131. }
  132. } else {
  133. ErrorManager::GetInstance().ATCReportErrMessage(
  134. "E13003", {"opname", "optype"}, {op_desc->GetName(), op_desc->GetType()});
  135. GELOGE(FAILED,
  136. "CheckEngineType:Can not find any supported ops kernel info store by kernel_name %s,"
  137. "op type is %s, op name is %s",
  138. kernel_name.c_str(), op_desc->GetType().c_str(), op_desc->GetName().c_str());
  139. }
  140. return FAILED;
  141. }
  142. static Status AddInputs(const ComputeGraphPtr &graph, const NodePtr &node, const GeTensorDesc &tensor, int32_t index,
  143. bool attr) {
  144. GE_CHECK_NOTNULL_EXEC(graph, return PARAM_INVALID);
  145. GE_CHECK_NOTNULL_EXEC(node, return PARAM_INVALID);
  146. auto format = tensor.GetFormat();
  147. auto data_type = tensor.GetDataType();
  148. if (format == FORMAT_RESERVED && data_type == DT_UNDEFINED) {
  149. return SUCCESS;
  150. }
  151. string op_type;
  152. bool is_const = false;
  153. (void)AttrUtils::GetBool(tensor, CONST_ATTR_NAME_INPUT, is_const);
  154. if (is_const) {
  155. GELOGD("Get input[%d] is const", index);
  156. op_type = CONSTANTOP;
  157. } else if (!AttrUtils::GetStr(tensor, kAttrOpType, op_type) || op_type.empty()) {
  158. op_type = DATA;
  159. }
  160. string op_name = node->GetName() + "_in_" + std::to_string(index);
  161. OpDescPtr data_op = MakeShared<ge::OpDesc>(op_name, op_type);
  162. if (data_op == nullptr) {
  163. return FAILED;
  164. }
  165. if (is_const) {
  166. ConstGeTensorPtr tensor_value;
  167. if (!AttrUtils::GetTensor(tensor, ge::ATTR_NAME_WEIGHTS, tensor_value)) {
  168. GELOGE(FAILED, "Get value failed, node name:%s.", tensor.GetName().c_str());
  169. return FAILED;
  170. }
  171. if (!AttrUtils::SetTensor(data_op, ge::ATTR_NAME_WEIGHTS, tensor_value)) {
  172. GELOGE(FAILED, "Set attr ATTR_NAME_WEIGHTS fail.");
  173. return FAILED;
  174. }
  175. }
  176. (void)AttrUtils::SetBool(data_op, "_is_single_op", true);
  177. GE_CHK_BOOL_EXEC(data_op->AddInputDesc(tensor) == GRAPH_SUCCESS, return FAILED, "Add input desc fail");
  178. GE_CHK_BOOL_EXEC(data_op->AddOutputDesc(tensor) == GRAPH_SUCCESS, return FAILED, "Add output desc fail");
  179. if (attr) {
  180. GE_CHK_BOOL_EXEC(AttrUtils::SetInt(data_op, ATTR_NAME_INDEX, index), return FAILED, "Set index fail");
  181. }
  182. ge::NodePtr arg_node = graph->AddNode(data_op);
  183. GE_CHK_BOOL_EXEC(arg_node != nullptr, return FAILED, "Insert Data node fail");
  184. GE_CHK_STATUS(GraphUtils::AddEdge(arg_node->GetOutDataAnchor(0), node->GetInDataAnchor(index)),
  185. "Add edge[%s->%s] fail", data_op->GetName().c_str(), node->GetName().c_str());
  186. return SUCCESS;
  187. }
  188. static Status AddOutputs(const ComputeGraphPtr &graph, const NodePtr &node, const vector<GeTensor> &outputs) {
  189. OpDescPtr op_desc = MakeShared<ge::OpDesc>(graph->GetName() + "_" + NODE_NAME_NET_OUTPUT, NETOUTPUT);
  190. if (op_desc == nullptr) {
  191. return FAILED;
  192. }
  193. (void)AttrUtils::SetBool(op_desc, "_is_single_op", true);
  194. int32_t count = 0;
  195. for (const auto &out_desc : outputs) {
  196. GeTensorDesc tensor = out_desc.GetTensorDesc();
  197. TensorUtils::SetInputTensor(tensor, true);
  198. GE_CHK_BOOL_EXEC(op_desc->AddInputDesc(tensor) == GRAPH_SUCCESS, return FAILED, "Add input desc fail.");
  199. TensorUtils::SetInputTensor(tensor, false);
  200. TensorUtils::SetOutputTensor(tensor, true);
  201. GE_CHK_BOOL_EXEC(op_desc->AddOutputDesc(tensor) == GRAPH_SUCCESS, return FAILED, "Add output desc fail.");
  202. count++;
  203. }
  204. GE_CHECK_NOTNULL_EXEC(graph, return PARAM_INVALID);
  205. ge::NodePtr out_node = graph->AddNode(op_desc);
  206. GE_CHK_BOOL_EXEC(out_node != nullptr, return FAILED, "Insert Output node fail");
  207. GE_CHECK_NOTNULL_EXEC(node, return PARAM_INVALID);
  208. for (int32_t i = 0; i < count; ++i) {
  209. GE_CHK_STATUS(GraphUtils::AddEdge(node->GetOutDataAnchor(i), out_node->GetInDataAnchor(i)),
  210. "Add edge[%s->%s] fail", node->GetName().c_str(), out_node->GetName().c_str());
  211. }
  212. return SUCCESS;
  213. }
  214. static void GetOpsProtoPath(string &opsproto_path) {
  215. const char *path_env = std::getenv("ASCEND_OPP_PATH");
  216. if (path_env != nullptr) {
  217. string path = path_env;
  218. string file_path = RealPath(path.c_str());
  219. if (file_path.empty()) {
  220. GELOGE(FAILED, "File path %s is invalid.", path.c_str());
  221. return;
  222. }
  223. opsproto_path = (path + "/op_proto/custom/" + ":") + (path + "/op_proto/built-in/");
  224. GELOGI("Get opsproto so path from env : %s", path.c_str());
  225. return;
  226. }
  227. string path_base = PluginManager::GetPath();
  228. GELOGI("path_base is %s.", path_base.c_str());
  229. path_base = path_base.substr(0, path_base.rfind('/'));
  230. path_base = path_base.substr(0, path_base.rfind('/') + 1);
  231. opsproto_path = (path_base + "ops/op_proto/custom/" + ":") + (path_base + "ops/op_proto/built-in/");
  232. }
  233. static Status ResetTensorVecShape(const vector<GeTensor> &inputs, vector<GeTensor> &inputs_dynamic) {
  234. for (auto input : inputs) {
  235. auto input_desc = input.GetTensorDesc();
  236. GeShape shape_ori = input_desc.GetShape();
  237. std::vector<int64_t> dynamic_shape_dims = {kDynamicDimValue};
  238. GeShape dynamic_shape(dynamic_shape_dims);
  239. std::vector<std::pair<int64_t, int64_t>> dynamic_shape_range;
  240. ge::GeTensor inputTensor;
  241. ge::GeTensorDesc desc(input_desc);
  242. bool is_const = false;
  243. (void)AttrUtils::GetBool(input_desc, CONST_ATTR_NAME_INPUT, is_const);
  244. if (!is_const) {
  245. int64_t storage_format = FORMAT_NCHW;
  246. if (ge::AttrUtils::GetInt(desc, ge::ATTR_NAME_STORAGE_FORMAT, storage_format) &&
  247. !ge::AttrUtils::SetListInt(desc, ge::ATTR_NAME_STORAGE_SHAPE, dynamic_shape_dims)) {
  248. GELOGE(FAILED, "Set attr ATTR_NAME_STORAGE_SHAPE fail.");
  249. return FAILED;
  250. }
  251. desc.SetShape(dynamic_shape);
  252. desc.SetShapeRange(dynamic_shape_range);
  253. }
  254. inputTensor.SetTensorDesc(desc);
  255. inputs_dynamic.push_back(inputTensor);
  256. }
  257. return SUCCESS;
  258. }
  259. class GeGenerator::Impl {
  260. public:
  261. Impl(OmgContext &omg_context) : omg_context_(omg_context) {}
  262. ~Impl() = default;
  263. Status BuildModel(const Graph &graph, const vector<GeTensor> &inputs, GeRootModelPtr &ge_models);
  264. Status SaveModel(const string &file_name_prefix, GeModelPtr &models, ModelBufferData &model);
  265. Status SaveRootModel(const string &file_name_prefix, GeRootModelPtr &model, ModelBufferData &model_buff);
  266. Status SaveParams(GeModelPtr &ge_model, const string &type, const map<string, GeAttrValue> &attrs,
  267. const vector<GeTensor> &inputs, const vector<GeTensor> &outputs);
  268. Status GenerateInfershapeGraph(const Graph &graph);
  269. OmgContext &omg_context_;
  270. GraphManager graph_manager_;
  271. SaveParam save_param_;
  272. bool is_offline_ = true;
  273. bool is_singleop_unregistered_ = false;
  274. std::string build_mode_;
  275. std::string build_step_;
  276. static std::mutex mutex_;
  277. private:
  278. static std::string Trim(const std::string &str);
  279. bool ParseVersion(const std::string &line, std::string &version);
  280. bool GetVersionFromPath(const std::string &file_path, std::string &version);
  281. bool SetAtcVersionInfo(AttrHolder &obj);
  282. bool SetOppVersionInfo(AttrHolder &obj);
  283. bool SetOmSystemInfo(AttrHolder &obj);
  284. };
  285. Status GeGenerator::Initialize(const map<string, string> &options) {
  286. return Initialize(options, domi::GetContext());
  287. }
  288. Status GeGenerator::Initialize(const map<string, string> &options, OmgContext &omg_context) {
  289. impl_ = ge::MakeShared<Impl>(omg_context);
  290. if (impl_ == nullptr) {
  291. GELOGE(MEMALLOC_FAILED, "Make shared failed");
  292. return MEMALLOC_FAILED;
  293. }
  294. ErrorManager::GetInstance().SetStage(ErrorMessage::kInitialize, ErrorMessage::kOpsProtoInit);
  295. string opsproto_path;
  296. GetOpsProtoPath(opsproto_path);
  297. GELOGI("Get opsproto path is %s.", opsproto_path.c_str());
  298. OpsProtoManager *manager = OpsProtoManager::Instance();
  299. map<string, string> option_tmp;
  300. option_tmp.emplace(std::pair<string, string>(string("ge.opsProtoLibPath"), opsproto_path));
  301. (void)manager->Initialize(option_tmp);
  302. Status ret = impl_->graph_manager_.Initialize(options);
  303. if (ret != SUCCESS) {
  304. GELOGE(GE_GENERATOR_GRAPH_MANAGER_INIT_FAILED, "Graph manager initialize failed.");
  305. return GE_GENERATOR_GRAPH_MANAGER_INIT_FAILED;
  306. }
  307. // get ek file
  308. auto iter = options.find(EK_FILE);
  309. if (iter != options.end()) {
  310. impl_->save_param_.ek_file = iter->second;
  311. }
  312. // get cert file
  313. iter = options.find(CERT_FILE);
  314. if (iter != options.end()) {
  315. impl_->save_param_.cert_file = iter->second;
  316. }
  317. // get hw key file
  318. iter = options.find(HW_KEY_FILE);
  319. if (iter != options.end()) {
  320. impl_->save_param_.hw_key_file = iter->second;
  321. }
  322. // get private file
  323. iter = options.find(PRIVATE_KEY_FILE);
  324. if (iter != options.end()) {
  325. impl_->save_param_.pri_key_file = iter->second;
  326. }
  327. // get build mode
  328. iter = options.find(BUILD_MODE);
  329. if (iter != options.end()) {
  330. impl_->build_mode_ = iter->second;
  331. }
  332. // get build step
  333. iter = options.find(BUILD_STEP);
  334. if (iter != options.end()) {
  335. impl_->build_step_ = iter->second;
  336. }
  337. return SUCCESS;
  338. }
  339. Status GeGenerator::Finalize() {
  340. ErrorManager::GetInstance().SetStage(ErrorMessage::kFinalize, ErrorMessage::kFinalize);
  341. GE_CHECK_NOTNULL_EXEC(impl_, return PARAM_INVALID);
  342. Status ret = impl_->graph_manager_.Finalize();
  343. if (ret != SUCCESS) {
  344. GELOGE(GE_GENERATOR_GRAPH_MANAGER_FINALIZE_FAILED, "Graph manager finalize failed.");
  345. return GE_GENERATOR_GRAPH_MANAGER_FINALIZE_FAILED;
  346. }
  347. return SUCCESS;
  348. }
  349. Status GeGenerator::GenerateOfflineModel(const Graph &graph, const string &file_name_prefix,
  350. const vector<GeTensor> &inputs) {
  351. ErrorManager::GetInstance().SetStage(ErrorMessage::kModelCompile, ErrorMessage::kOther);
  352. GELOGI("Start to generate offline model.");
  353. ModelBufferData model;
  354. return GenerateModel(graph, file_name_prefix, inputs, model, true);
  355. }
  356. Status GeGenerator::GenerateOnlineModel(const Graph &graph, const vector<GeTensor> &inputs, ModelBufferData &model) {
  357. ErrorManager::GetInstance().SetStage(ErrorMessage::kModelCompile, ErrorMessage::kOther);
  358. return GenerateModel(graph, "online", inputs, model, false);
  359. }
  360. Status GeGenerator::GenerateInfershapeGraph(const Graph &graph) {
  361. GE_CHECK_NOTNULL_EXEC(impl_, return PARAM_INVALID);
  362. Status ret = impl_->GenerateInfershapeGraph(graph);
  363. if (ret != SUCCESS) {
  364. GELOGE(ret, "Dump infershape json failed");
  365. if (impl_->graph_manager_.Finalize() != SUCCESS) {
  366. GELOGE(FAILED, "graph_manager finalize fail.");
  367. }
  368. return ret;
  369. }
  370. GELOGI("Generate infer shape graph success");
  371. return SUCCESS;
  372. }
  373. std::mutex GeGenerator::Impl::mutex_;
  374. // Remove the space and tab before and after the string
  375. std::string GeGenerator::Impl::Trim(const std::string &str) {
  376. if (str.empty()) {
  377. return str;
  378. }
  379. std::string::size_type start = str.find_first_not_of(" \t\r\n");
  380. if (start == std::string::npos) {
  381. return str;
  382. }
  383. std::string::size_type end = str.find_last_not_of(" \t\r\n") + 1;
  384. return str.substr(start, end);
  385. }
  386. // Parsing the command line
  387. bool GeGenerator::Impl::ParseVersion(const std::string &line, std::string &version) {
  388. std::string flag = "Version=";
  389. std::string temp = Trim(line);
  390. if (temp.empty()) {
  391. GELOGW("line is empty.");
  392. return false;
  393. }
  394. std::string::size_type pos = temp.find(flag);
  395. if (pos == std::string::npos) {
  396. GELOGW("Incorrect line [%s], it must include [%s].", line.c_str(), flag.c_str());
  397. return false;
  398. }
  399. if (temp.size() == flag.size()) {
  400. GELOGW("version information is empty. %s", line.c_str());
  401. return false;
  402. }
  403. version = temp.substr(pos + flag.size());
  404. return true;
  405. }
  406. bool GeGenerator::Impl::GetVersionFromPath(const std::string &file_path, std::string &version) {
  407. // Normalize the path
  408. string resolved_file_path = RealPath(file_path.c_str());
  409. if (resolved_file_path.empty()) {
  410. GELOGW("Invalid input file path [%s], make sure that the file path is correct.", file_path.c_str());
  411. return false;
  412. }
  413. std::ifstream fs(resolved_file_path, std::ifstream::in);
  414. if (!fs.is_open()) {
  415. GELOGW("Open %s failed.", file_path.c_str());
  416. return false;
  417. }
  418. std::string line;
  419. if (getline(fs, line)) {
  420. if (!ParseVersion(line, version)) {
  421. GELOGW("Parse version failed. content is [%s].", line.c_str());
  422. fs.close();
  423. return false;
  424. }
  425. } else {
  426. GELOGW("No version information found in the file path:%s", file_path.c_str());
  427. fs.close();
  428. return false;
  429. }
  430. fs.close(); // close the file
  431. return true;
  432. }
  433. // Set package version information in the model
  434. bool GeGenerator::Impl::SetAtcVersionInfo(AttrHolder &obj) {
  435. std::string path_base = ge::GELib::GetPath();
  436. path_base = path_base.substr(0, path_base.rfind('/'));
  437. path_base = path_base.substr(0, path_base.rfind('/') + 1);
  438. std::string version_path = path_base + "version.info";
  439. std::string version;
  440. if (!GetVersionFromPath(version_path, version)) {
  441. GELOGW("Get atc version information failed!");
  442. return false;
  443. }
  444. // set version info
  445. if (!ge::AttrUtils::SetStr(obj, ATTR_MODEL_ATC_VERSION, version)) {
  446. GELOGW("Ge model set atc version failed!");
  447. return false;
  448. }
  449. return true;
  450. }
  451. // Set package version information in the model
  452. bool GeGenerator::Impl::SetOppVersionInfo(AttrHolder &obj) {
  453. const char *path_env = std::getenv("ASCEND_OPP_PATH");
  454. if (path_env == nullptr) {
  455. GELOGW("Get environment variable ASCEND_OPP_PATH failed!");
  456. return false;
  457. }
  458. std::string version_path = path_env;
  459. version_path += "/version.info";
  460. std::string version;
  461. if (!GetVersionFromPath(version_path, version)) {
  462. GELOGW("Get opp version information failed!");
  463. return false;
  464. }
  465. // set version info
  466. if (!ge::AttrUtils::SetStr(obj, ATTR_MODEL_OPP_VERSION, version)) {
  467. GELOGW("Ge model set opp version failed!");
  468. return false;
  469. }
  470. return true;
  471. }
  472. bool GeGenerator::Impl::SetOmSystemInfo(AttrHolder &obj) {
  473. std::string soc_version;
  474. (void)ge::GetContext().GetOption(ge::SOC_VERSION, soc_version);
  475. GELOGI("SetOmSystemInfo soc_version: %s", soc_version.c_str());
  476. if (!ge::AttrUtils::SetStr(obj, "soc_version", soc_version)) {
  477. GELOGW("SetStr of soc_version failed.");
  478. return false;
  479. }
  480. std::string framework_type;
  481. (void)ge::GetContext().GetOption(ge::FRAMEWORK_TYPE, framework_type);
  482. GELOGI("SetOmSystemInfo framework_type: %s", framework_type.c_str());
  483. auto iter = ge::kFwkTypeToStr.find(framework_type);
  484. if (iter == ge::kFwkTypeToStr.end()) {
  485. GELOGW("Can not find framework_type in the map.");
  486. return false;
  487. }
  488. if (!ge::AttrUtils::SetStr(obj, "framework_type", iter->second)) {
  489. GELOGW("SetStr of framework_type failed.");
  490. return false;
  491. }
  492. return true;
  493. }
  494. Status GeGenerator::GenerateModel(const Graph &graph, const string &file_name_prefix, const vector<GeTensor> &inputs,
  495. ModelBufferData &model, bool is_offline) {
  496. rtContext_t ctx = nullptr;
  497. auto rt = rtCtxGetCurrent(&ctx);
  498. if (rt != RT_ERROR_NONE) {
  499. GELOGD("Current ctx is null.");
  500. ctx = nullptr;
  501. }
  502. GeRootModelPtr ge_root_model = nullptr;
  503. GE_CHECK_NOTNULL_EXEC(impl_, return PARAM_INVALID);
  504. impl_->is_offline_ = is_offline;
  505. Status ret = impl_->BuildModel(graph, inputs, ge_root_model);
  506. if (ret != SUCCESS) {
  507. GELOGE(ret, "Build model failed.");
  508. if (impl_->graph_manager_.Finalize() != SUCCESS) {
  509. GELOGE(FAILED, "graph_manager finalize fail.");
  510. }
  511. return ret;
  512. }
  513. /// BUILD_MODE_TUNING with BUILD_STEP_BEFORE_UB_MATCH no need save model;
  514. /// BUILD_MODE_TUNING with BUILD_STEP_AFTER_BUILDER no need save model;
  515. /// BUILD_MODE_TUNING with BUILD_STEP_AFTER_BUILDER_SUB no need save model.
  516. if ((impl_->build_mode_ == BUILD_MODE_TUNING) &&
  517. (impl_->build_step_ == BUILD_STEP_BEFORE_UB_MATCH || impl_->build_step_ == BUILD_STEP_AFTER_BUILDER ||
  518. impl_->build_step_ == BUILD_STEP_AFTER_BUILDER_SUB)) {
  519. GELOGI("Build mode:%s with step:%s no need SaveModel.",
  520. impl_->build_mode_.c_str(),
  521. impl_->build_step_.c_str());
  522. return SUCCESS;
  523. }
  524. GE_CHECK_NOTNULL(ge_root_model);
  525. GE_CHECK_NOTNULL(ge_root_model->GetRootGraph());
  526. ModelHelper model_helper;
  527. string model_name = "";
  528. Status name_ret = model_helper.GetModelNameFromMergedGraphName(ge_root_model->GetRootGraph()->GetName(),
  529. model_name);
  530. if (name_ret != SUCCESS) {
  531. ErrorManager::GetInstance().ATCReportErrMessage("E10000", {"parameter"}, {"output"});
  532. GELOGE(FAILED, "Get model_name failed. Param --output is invalid.");
  533. return PARAM_INVALID;
  534. }
  535. map<string, GeModelPtr> name_to_ge_model = ge_root_model->GetSubgraphInstanceNameToModel();
  536. GeModelPtr &ge_model = name_to_ge_model[ge_root_model->GetRootGraph()->GetName()];
  537. GE_RETURN_WITH_LOG_IF_FALSE(ge_model != nullptr, "ge_model cannot be null");
  538. ge_model->SetName(model_name);
  539. ret = impl_->SaveRootModel(file_name_prefix, ge_root_model, model);
  540. if (ret != SUCCESS) {
  541. GELOGE(ret, "Save model failed");
  542. if (impl_->graph_manager_.Finalize() != SUCCESS) {
  543. GELOGE(FAILED, "graph_manager finalize fail.");
  544. }
  545. return ret;
  546. }
  547. if (ctx != nullptr) {
  548. (void)rtCtxSetCurrent(ctx);
  549. }
  550. return SUCCESS;
  551. }
  552. namespace {
  553. bool IsNeedConnectInputOpForSingleOp(GeTensorDesc &tensor_desc) {
  554. bool is_need = true;
  555. // format and dtype is all reserved, stand for Optional input. When singleop scene
  556. if (tensor_desc.GetFormat() == FORMAT_RESERVED && tensor_desc.GetDataType() == DT_UNDEFINED) {
  557. is_need = false;
  558. }
  559. return is_need;
  560. }
  561. Status CheckDynamicSupport(GeModelPtr &ge_model, const ComputeGraphPtr &graph) {
  562. bool support_dynamic = true;
  563. bool is_dynamic = false;
  564. for (const auto &node : graph->GetDirectNode()) {
  565. GE_CHECK_NOTNULL(node);
  566. auto op_desc = node->GetOpDesc();
  567. GE_CHECK_NOTNULL(op_desc);
  568. if (op_desc->GetOpEngineName() != kAIcoreEngine) {
  569. continue;
  570. }
  571. if (AttrUtils::HasAttr(op_desc, kAttrSupportDynamicShape)) {
  572. is_dynamic = true;
  573. (void) AttrUtils::GetBool(op_desc, kAttrSupportDynamicShape, support_dynamic);
  574. if (!support_dynamic) {
  575. GELOGW("Node[%s] doesn't support dynamic shape.", node->GetName().c_str());
  576. break;
  577. }
  578. }
  579. }
  580. if (is_dynamic) {
  581. (void) AttrUtils::SetBool(ge_model, kAttrSupportDynamicShape, support_dynamic);
  582. }
  583. return SUCCESS;
  584. }
  585. }
  586. Status GeGenerator::CheckForSingleOp(OpDescPtr &op_desc, const vector<GeTensor> &inputs,
  587. const vector<GeTensor> &outputs) {
  588. GE_CHECK_NOTNULL_EXEC(op_desc, return PARAM_INVALID);
  589. if (!inputs.empty() && (inputs.size() != op_desc->GetAllInputsSize())) {
  590. ErrorManager::GetInstance().ATCReportErrMessage("E14001", {"opname", "optype", "value", "reason"},
  591. {op_desc->GetName(), op_desc->GetType(), "inputs size" + FmtToStr(op_desc->GetAllInputsSize()),
  592. "tensor size is " + FmtToStr(inputs.size())});
  593. GELOGE(PARAM_INVALID, "Tensor size: %zu, Inputs size: %zu", inputs.size(), op_desc->GetAllInputsSize());
  594. return PARAM_INVALID;
  595. }
  596. if (!outputs.empty() && (outputs.size() != op_desc->GetOutputsSize())) {
  597. ErrorManager::GetInstance().ATCReportErrMessage("E14001", {"opname", "optype", "value", "reason"},
  598. {op_desc->GetName(), op_desc->GetType(), "outputs size" + FmtToStr(op_desc->GetOutputsSize()),
  599. "tensor size is " + FmtToStr(outputs.size())});
  600. GELOGE(PARAM_INVALID, "Tensor size: %zu, Outputs size: %zu", outputs.size(), op_desc->GetOutputsSize());
  601. return PARAM_INVALID;
  602. }
  603. return SUCCESS;
  604. }
  605. Status GeGenerator::BuildSingleOp(OpDescPtr &op_desc, const vector<GeTensor> &inputs, const vector<GeTensor> &outputs,
  606. const string &model_file_name, OpEngineType engine_type, ModelBufferData &model_buff,
  607. bool is_offline) {
  608. GE_CHECK_NOTNULL_EXEC(impl_, return PARAM_INVALID);
  609. impl_->is_offline_ = is_offline;
  610. if (!is_offline) {
  611. (void)AttrUtils::SetBool(op_desc, ATTR_SINGLE_OP_SCENE, true);
  612. }
  613. if (CheckForSingleOp(op_desc, inputs, outputs) != SUCCESS) {
  614. GELOGE(PARAM_INVALID, "input param is invalid when build single op!");
  615. return PARAM_INVALID;
  616. }
  617. OmgContext &omg_context = (impl_ == nullptr) ? domi::GetContext() : impl_->omg_context_;
  618. omg_context.is_dynamic_input = ContainsDynamicInpus(*op_desc);
  619. if (op_desc->HasAttr(ATTR_NAME_UNREGST_OPPATH)) {
  620. impl_->is_singleop_unregistered_ = true;
  621. }
  622. // 0. Save original attributes.
  623. OpDescPtr op_desc_tmp = AttrUtils::CloneOpDesc(op_desc);
  624. GE_CHECK_NOTNULL(op_desc_tmp);
  625. // 1. Create ComputeGraph.
  626. string name = ge::CurrentTimeInStr() + "_" + model_file_name;
  627. Graph graph;
  628. GE_CHK_STATUS(BuildSingleOpGraph(op_desc, inputs, outputs, name, graph), "make graph fail.");
  629. // 2. check engine type when compile online
  630. if (model_file_name == kFileNameSuffix) {
  631. auto comp_graph = GraphUtils::GetComputeGraph(graph);
  632. GE_CHECK_NOTNULL(comp_graph);
  633. auto node = comp_graph->FindNode(op_desc->GetName());
  634. Status ret = CheckEngineTypeSupport(node, engine_type);
  635. if (ret != SUCCESS) {
  636. GELOGE(ret, "check engine type failed");
  637. return ret;
  638. }
  639. }
  640. GELOGI("ATC parser success in single op build.");
  641. GeRootModelPtr ge_root_model = nullptr;
  642. GE_CHK_STATUS_RET_NOLOG(impl_->BuildModel(graph, inputs, ge_root_model));
  643. map<string, GeAttrValue> op_attrs = op_desc_tmp->GetAllAttrs();
  644. GE_CHECK_NOTNULL(ge_root_model);
  645. GE_CHECK_NOTNULL(ge_root_model->GetRootGraph());
  646. map<string, GeModelPtr> name_to_ge_model = ge_root_model->GetSubgraphInstanceNameToModel();
  647. if (name_to_ge_model.empty()) {
  648. GELOGE(PARAM_INVALID, "GetSubgraphInstanceNameToModel is empty.");
  649. return PARAM_INVALID;
  650. }
  651. const ComputeGraphPtr root_graph = ge_root_model->GetRootGraph();
  652. GeModelPtr &ge_model = name_to_ge_model.begin()->second;
  653. GE_CHK_STATUS_RET_NOLOG(CheckDynamicSupport(ge_model, root_graph));
  654. GELOGI("After build model, The opType in op_desc_tmp is [%s]", op_desc_tmp->GetType().c_str());
  655. bool all_shape = false;
  656. (void)AttrUtils::GetBool(op_desc, kAicpuAllshape, all_shape);
  657. if (all_shape) {
  658. GELOGD("Get aicpu all_shape kernel!");
  659. vector<GeTensor> inputs_dynamic;
  660. vector<GeTensor> outputs_dynamic;
  661. GE_CHK_STATUS_RET_NOLOG(ResetTensorVecShape(inputs, inputs_dynamic));
  662. GE_CHK_STATUS_RET_NOLOG(ResetTensorVecShape(outputs, outputs_dynamic));
  663. GE_CHK_STATUS_RET_NOLOG(
  664. impl_->SaveParams(ge_model, op_desc_tmp->GetType(), op_attrs, inputs_dynamic, outputs_dynamic));
  665. } else {
  666. GE_CHK_STATUS_RET_NOLOG(impl_->SaveParams(ge_model, op_desc_tmp->GetType(), op_attrs, inputs, outputs));
  667. }
  668. GELOGI("Start save GeModel to Model buffer");
  669. GE_CHK_STATUS_RET_NOLOG(impl_->SaveModel(model_file_name, ge_model, model_buff));
  670. return SUCCESS;
  671. }
  672. /**
  673. * @ingroup ge
  674. * @brief Compiling a single operator into an offline model
  675. * @param [in] OpDescPtr &op_desc: Operator description info that needs to be compiled into an offline model file
  676. * @param [in] vector<GeTensor> &inputs: Operator input data description information.
  677. * @param [in] vector<GeTensor> &outputs: Operator output data description information.
  678. * @param [in] const string &model_file_name: Offline model filename.
  679. * @return SUCCESS handle successfully / others handle failed
  680. */
  681. Status GeGenerator::BuildSingleOpModel(OpDescPtr &op_desc, const vector<GeTensor> &inputs,
  682. const vector<GeTensor> &outputs, const string &model_file_name) {
  683. ErrorManager::GetInstance().SetStage(ErrorMessage::kModelCompile, ErrorMessage::kOther);
  684. GELOGI("Start to build single op offline model, input size: %zu, output size: %zu", inputs.size(), outputs.size());
  685. ModelBufferData model_buff;
  686. OpEngineType engine_type = ENGINE_SYS;
  687. Status status = BuildSingleOp(op_desc, inputs, outputs, model_file_name, engine_type, model_buff, true);
  688. GELOGI("Finish build single offline model, status: %u", status);
  689. return status;
  690. }
  691. /**
  692. * @ingroup ge
  693. * @brief Compiling a single operator into online buffer
  694. * @param [in] OpDescPtr &op_desc: Operator description info that needs to be compiled into an offline model file
  695. * @param [in] vector<GeTensor> &inputs: Operator input data description information.
  696. * @param [in] vector<GeTensor> &outputs: Operator output data description information.
  697. * @param [in] engine_type: specific engine.
  698. * @param [out] ModelBufferData &Model_buff: Model_buff: model buffer of the op.
  699. * @return SUCCESS handle successfully / others handle failed
  700. */
  701. Status GeGenerator::BuildSingleOpModel(OpDescPtr &op_desc, const vector<GeTensor> &inputs,
  702. const vector<GeTensor> &outputs, OpEngineType engine_type,
  703. ModelBufferData &model_buff) {
  704. ErrorManager::GetInstance().SetStage(ErrorMessage::kModelCompile, ErrorMessage::kOther);
  705. GELOGI("Start to build single op online, input size: %zu, output size: %zu.", inputs.size(), outputs.size());
  706. Status status = BuildSingleOp(op_desc, inputs, outputs, kFileNameSuffix, engine_type, model_buff, false);
  707. GELOGI("Finish build single online model, status: %u.", status);
  708. return status;
  709. }
  710. Status GeGenerator::BuildSingleOpGraph(OpDescPtr &op_desc, const vector<GeTensor> &inputs,
  711. const vector<GeTensor> &outputs, std::string graph_name, Graph &graph) {
  712. ge::ComputeGraphPtr compute_graph = MakeShared<ComputeGraph>(graph_name);
  713. GE_CHECK_NOTNULL_EXEC(compute_graph, return INTERNAL_ERROR);
  714. // 1. Add Node to ComputeGraph.
  715. NodePtr op_node = compute_graph->AddNode(op_desc);
  716. GE_CHECK_NOTNULL_EXEC(op_node, return INTERNAL_ERROR);
  717. // 2. Create InputData node.
  718. int32_t arg_index = 0;
  719. if (inputs.empty()) {
  720. for (const auto &input_desc : op_desc->GetAllInputsDescPtr()) {
  721. GE_CHECK_NOTNULL_EXEC(input_desc, return INTERNAL_ERROR);
  722. if (!IsNeedConnectInputOpForSingleOp(*input_desc)) {
  723. continue;
  724. }
  725. GE_CHK_STATUS_RET_NOLOG(AddInputs(compute_graph, op_node, *input_desc, arg_index, false));
  726. arg_index++;
  727. }
  728. } else {
  729. for (const auto &in_desc : inputs) {
  730. GE_CHK_STATUS_RET_NOLOG(AddInputs(compute_graph, op_node, in_desc.GetTensorDesc(), arg_index, true));
  731. arg_index++;
  732. }
  733. }
  734. // 3. Create Output node.
  735. if (!outputs.empty()) {
  736. GE_CHK_STATUS_RET_NOLOG(AddOutputs(compute_graph, op_node, outputs));
  737. }
  738. // dump ComputeGraph node.
  739. compute_graph->Dump();
  740. graph = ge::GraphUtils::CreateGraphFromComputeGraph(compute_graph);
  741. return SUCCESS;
  742. }
  743. Status GeGenerator::Impl::SaveParams(GeModelPtr &ge_model, const string &type, const map<string, GeAttrValue> &attrs,
  744. const vector<GeTensor> &inputs, const vector<GeTensor> &outputs) {
  745. GE_CHECK_NOTNULL_EXEC(ge_model, return PARAM_INVALID);
  746. GE_CHK_BOOL_EXEC_NOLOG(graph_manager_.SaveParams(*ge_model, type, attrs, inputs, outputs) == SUCCESS,
  747. (void)graph_manager_.Finalize();
  748. return FAILED);
  749. return SUCCESS;
  750. }
  751. Status GeGenerator::Impl::SaveModel(const string &file_name_prefix, GeModelPtr &model, ModelBufferData &model_buff) {
  752. // set atc version
  753. if (!SetAtcVersionInfo(*(model.get()))) {
  754. GELOGW("SetPackageVersionInfo of atc failed!");
  755. }
  756. // set opp version
  757. if (!SetOppVersionInfo(*(model.get()))) {
  758. GELOGW("SetPackageVersionInfo of ops failed!");
  759. }
  760. ModelHelper model_helper;
  761. model_helper.SetSaveMode(is_offline_);
  762. Status ret = model_helper.SaveToOmModel(model, save_param_, file_name_prefix, model_buff);
  763. if (ret != SUCCESS) {
  764. GELOGE(ret, "Save to om model failed");
  765. return ret;
  766. }
  767. return SUCCESS;
  768. }
  769. Status GeGenerator::Impl::SaveRootModel(const string &file_name_prefix, GeRootModelPtr &ge_root_model,
  770. ModelBufferData &model_buff) {
  771. bool is_unknown_shape = false;
  772. auto ret = ge_root_model->CheckIsUnknownShape(is_unknown_shape);
  773. if (ret != SUCCESS) {
  774. GELOGE(FAILED, "Check root model is unkonwn shape failed");
  775. return FAILED;
  776. }
  777. GELOGD("begin save root model, cur model is unkonwn shape model ? : %d", is_unknown_shape);
  778. GE_CHK_BOOL_EXEC(!ge_root_model->GetSubgraphInstanceNameToModel().empty(), return FAILED,
  779. "ge root model has no sub model")
  780. GeModelPtr model_root = nullptr;
  781. if (is_unknown_shape) {
  782. model_root = make_shared<GeModel>();
  783. model_root->SetGraph(GraphUtils::CreateGraphFromComputeGraph(ge_root_model->GetRootGraph()));
  784. ge_root_model->SetSubgraphInstanceNameToModel(ge_root_model->GetRootGraph()->GetName(), model_root);
  785. model_root->SetName(ge_root_model->GetRootGraph()->GetName());
  786. } else {
  787. model_root = ge_root_model->GetSubgraphInstanceNameToModel().begin()->second;
  788. }
  789. // set atc version
  790. if (!SetAtcVersionInfo(*(model_root.get()))) {
  791. GELOGW("SetPackageVersionInfo of atc failed!");
  792. }
  793. // set opp version
  794. if (!SetOppVersionInfo(*(model_root.get()))) {
  795. GELOGW("SetPackageVersionInfo of ops failed!");
  796. }
  797. if (!SetOmSystemInfo(*(model_root.get()))) {
  798. GELOGW("SetOmsystemInfo failed!");
  799. }
  800. ModelHelper model_helper;
  801. model_helper.SetSaveMode(is_offline_);
  802. ret = model_helper.SaveToOmRootModel(ge_root_model, save_param_, file_name_prefix, model_buff, is_unknown_shape);
  803. if (ret != SUCCESS) {
  804. GELOGE(ret, "Save to om model failed");
  805. return ret;
  806. }
  807. return SUCCESS;
  808. }
  809. Status GeGenerator::Impl::BuildModel(const Graph &graph, const vector<GeTensor> &inputs,
  810. GeRootModelPtr &ge_root_model) {
  811. static std::atomic<GraphId> atomic_graph_id(0);
  812. auto graph_id = atomic_graph_id.fetch_add(1);
  813. const std::map<std::string, std::string> options;
  814. Status ret = graph_manager_.AddGraph(graph_id, graph, options, omg_context_);
  815. if (ret != SUCCESS) {
  816. GELOGE(GE_GENERATOR_GRAPH_MANAGER_ADD_GRAPH_FAILED, "GraphManager add graph fail, graph id: %u", graph_id);
  817. (void)graph_manager_.Finalize();
  818. return GE_GENERATOR_GRAPH_MANAGER_ADD_GRAPH_FAILED;
  819. }
  820. graph_manager_.SetOptionsRunGraphFlag(false);
  821. static std::atomic<uint64_t> atomic_session_id(0);
  822. auto session_id = atomic_session_id.fetch_add(1);
  823. if (is_singleop_unregistered_) {
  824. ret = graph_manager_.BuildGraphForUnregisteredOp(graph_id, inputs, ge_root_model, session_id);
  825. } else {
  826. ret = graph_manager_.BuildGraph(graph_id, inputs, ge_root_model, session_id);
  827. }
  828. ErrorManager::GetInstance().SetStage(ErrorMessage::kModelCompile, ErrorMessage::kOther);
  829. if (ret != SUCCESS) {
  830. GELOGE(GE_GENERATOR_GRAPH_MANAGER_BUILD_GRAPH_FAILED, "GraphManager build graph fail, graph id: %u", graph_id);
  831. VarManagerPool::Instance().RemoveVarManager(session_id);
  832. return GE_GENERATOR_GRAPH_MANAGER_BUILD_GRAPH_FAILED;
  833. }
  834. VarManagerPool::Instance().RemoveVarManager(session_id);
  835. return SUCCESS;
  836. }
  837. Status GeGenerator::Impl::GenerateInfershapeGraph(const Graph &graph) {
  838. static std::atomic<GraphId> atomic_graph_id(0);
  839. auto graph_id = atomic_graph_id.fetch_add(1);
  840. const std::map<std::string, std::string> options;
  841. Status ret = graph_manager_.AddGraph(graph_id, graph, options, omg_context_);
  842. if (ret != SUCCESS) {
  843. GELOGE(GE_GENERATOR_GRAPH_MANAGER_ADD_GRAPH_FAILED, "GraphManager add graph failed, graph id: %u", graph_id);
  844. (void)graph_manager_.Finalize();
  845. return GE_GENERATOR_GRAPH_MANAGER_ADD_GRAPH_FAILED;
  846. }
  847. ret = graph_manager_.GenerateInfershapeGraph(graph_id);
  848. if (ret != SUCCESS) {
  849. GELOGE(GE_GENERATOR_GRAPH_MANAGER_BUILD_GRAPH_FAILED, "GraphManager generate graph failed");
  850. return GE_GENERATOR_GRAPH_MANAGER_BUILD_GRAPH_FAILED;
  851. }
  852. return SUCCESS;
  853. }
  854. } // namespace ge

图引擎模块(GE)是MindSpore的一个子模块,其代码由C++实现,位于前端模块ME和底层硬件之间,起到承接作用。图引擎模块以ME下发的图作为输入,然后进行一系列的深度图优化操作,最后输出一张可以在底层硬件上高效运行的图。GE针对昇腾AI处理器的硬件结构特点,做了特定的优化工作,以此来充分发挥出昇腾AI处理器的强大算力。在进行模型训练/推理时,GE会被自动调用而用户并不感知。GE主要由GE API和GE Core两部分组成,详细的架构图如下所示