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

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