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.

symbol_resolver.h 4.5 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  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. #ifndef MINDSPORE_CCSRC_FRONTEND_OPTIMIZER_IRPASS_SYMBOL_RESOLVER_H_
  17. #define MINDSPORE_CCSRC_FRONTEND_OPTIMIZER_IRPASS_SYMBOL_RESOLVER_H_
  18. #include <string>
  19. #include <memory>
  20. #include "frontend/optimizer/optimizer.h"
  21. #include "frontend/optimizer/optimizer_caller.h"
  22. #include "frontend/optimizer/irpass.h"
  23. #include "frontend/optimizer/anf_visitor.h"
  24. #include "frontend/operator/ops.h"
  25. #include "ir/pattern_matcher.h"
  26. #include "pipeline/jit/parse/data_converter.h"
  27. #include "pipeline/jit/parse/python_adapter.h"
  28. #include "pipeline/jit/parse/parse_base.h"
  29. namespace mindspore {
  30. namespace opt {
  31. namespace irpass {
  32. const char PARSE_SUPER_NAME[] = "namespace";
  33. // {prim::kPrimResolve, Ns, Sym}
  34. class ResolverResolve : public AnfVisitor {
  35. public:
  36. AnfNodePtr operator()(const OptimizerPtr &optimizer, const AnfNodePtr &node) override {
  37. Reset();
  38. AnfVisitor::Match(prim::kPrimResolve, {IsVNode, IsVNode})(node);
  39. if (sym_ != nullptr) {
  40. return parse::ResolveSymbol(optimizer->manager(), ns_, sym_, node);
  41. }
  42. return nullptr;
  43. }
  44. void Visit(const ValueNodePtr &vnode) override {
  45. if (IsValueNode<parse::NameSpace>(vnode)) {
  46. ns_ = GetValueNode<parse::NameSpacePtr>(vnode);
  47. } else if (ns_ != nullptr && IsValueNode<parse::Symbol>(vnode)) {
  48. sym_ = GetValueNode<parse::SymbolPtr>(vnode);
  49. }
  50. }
  51. void Reset() {
  52. ns_ = nullptr;
  53. sym_ = nullptr;
  54. }
  55. private:
  56. parse::NameSpacePtr ns_{nullptr};
  57. parse::SymbolPtr sym_{nullptr};
  58. };
  59. // {prim::kPrimGetAttr, Ns, Str}
  60. class ResolverGetattr : public AnfVisitor {
  61. public:
  62. AnfNodePtr operator()(const OptimizerPtr &optimizer, const AnfNodePtr &node) override {
  63. Reset();
  64. AnfVisitor::Match(prim::kPrimGetAttr, {IsVNode, IsVNode})(node);
  65. if (sym_ != nullptr) {
  66. return parse::ResolveSymbol(optimizer->manager(), ns_, sym_, node);
  67. }
  68. return nullptr;
  69. }
  70. void Visit(const AnfNodePtr &node) override {
  71. if (IsValueNode<parse::NameSpace>(node)) {
  72. ns_ = GetValueNode<parse::NameSpacePtr>(node);
  73. } else if (ns_ != nullptr && IsValueNode<StringImm>(node)) {
  74. auto str = GetValue<std::string>(GetValueNode(node));
  75. sym_ = std::make_shared<parse::Symbol>(str);
  76. }
  77. }
  78. void Reset() {
  79. ns_ = nullptr;
  80. sym_ = nullptr;
  81. }
  82. private:
  83. parse::NameSpacePtr ns_{nullptr};
  84. parse::SymbolPtr sym_{nullptr};
  85. };
  86. // {prim::kPrimGetAttr, {prim::kPrimResolve, ns_node, sym_node}, attr_node}
  87. class ResolveAttr : public OptimizerCaller {
  88. public:
  89. AnfNodePtr operator()(const OptimizerPtr &optimizer, const AnfNodePtr &node) override {
  90. PatternNode<AnfNodePtr> ns_node, sym_node, attr_node;
  91. auto ResolveAttrLambda = [&node, &ns_node, &sym_node, &attr_node, &optimizer]() -> AnfNodePtr {
  92. auto node_to_getattr = node->cast<CNodePtr>()->input(1);
  93. std::string attr_as_string = GetValueNode<StringImmPtr>(attr_node.GetNode(node))->value();
  94. auto ns_ = GetValueNode<parse::NameSpacePtr>(ns_node.GetNode(node));
  95. auto sym_ = GetValueNode<parse::SymbolPtr>(sym_node.GetNode(node));
  96. if (ns_->module() == parse::RESOLVE_NAMESPACE_NAME_CLASS_MEMBER && sym_->symbol() != PARSE_SUPER_NAME) {
  97. // deal with the case of getting attr from a class member
  98. // and avoid the case of getting attr from self (the result of ParseSuper)
  99. auto result = parse::ResolveCellwithAttr(optimizer->manager(), ns_, sym_, node_to_getattr, attr_as_string);
  100. return result;
  101. }
  102. return nullptr;
  103. };
  104. MATCH_REPLACE_LAMBDA_IF(
  105. node, PPrimitive(prim::kPrimGetAttr, PPrimitive(prim::kPrimResolve, ns_node, sym_node), attr_node),
  106. ResolveAttrLambda, attr_node.CheckFunc(IsValueNode<StringImm>, node));
  107. return nullptr;
  108. }
  109. };
  110. } // namespace irpass
  111. } // namespace opt
  112. } // namespace mindspore
  113. #endif // MINDSPORE_CCSRC_FRONTEND_OPTIMIZER_IRPASS_SYMBOL_RESOLVER_H_