• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2024 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  * http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include "libabckit/include/cpp/abckit_cpp.h"
17 #include "helpers/helpers.h"
18 #include "helpers/helpers_runtime.h"
19 #include "libabckit/src/logger.h"
20 
21 #include <gtest/gtest.h>
22 #include <optional>
23 #include <string>
24 #include <set>
25 
26 namespace {
27 
28 class GTestAssertErrorHandler final : public abckit::IErrorHandler {
29 public:
HandleError(abckit::Exception && err)30     void HandleError(abckit::Exception &&err) override
31     {
32         EXPECT_TRUE(false) << "Abckit expection raised: " << err.what();
33     }
34 };
35 
36 using FunctionCallback = std::function<bool(abckit::core::Function)>;
37 using ClassCallback = std::function<bool(abckit::core::Class)>;
38 using NamespaceCallback = std::function<bool(abckit::core::Namespace)>;
39 
40 struct ClassMeta {
41     std::string modulePath;
42     std::string className;
43 
operator ==(const ClassMeta & lhs,const ClassMeta & rhs)44     friend bool operator==(const ClassMeta &lhs, const ClassMeta &rhs)
45     {
46         return std::tie(lhs.modulePath, lhs.className) == std::tie(rhs.modulePath, rhs.className);
47     }
48 
operator <(const ClassMeta & lhs,const ClassMeta & rhs)49     friend bool operator<(const ClassMeta &lhs, const ClassMeta &rhs)
50     {
51         return std::tie(lhs.modulePath, lhs.className) < std::tie(rhs.modulePath, rhs.className);
52     }
53 };
54 
EnumerateAllMethods(const abckit::core::Module & mod,const FunctionCallback & fnUserCallback)55 void EnumerateAllMethods(const abckit::core::Module &mod, const FunctionCallback &fnUserCallback)
56 {
57     ClassCallback clsCallback;
58 
59     FunctionCallback fnCallback = [&](const abckit::core::Function &fun) -> bool {
60         if (!fnUserCallback(fun)) {
61             return false;
62         }
63         fun.EnumerateNestedFunctions(fnCallback);
64         fun.EnumerateNestedClasses(clsCallback);
65         return true;
66     };
67 
68     clsCallback = [&](const abckit::core::Class &cls) -> bool {
69         cls.EnumerateMethods(fnCallback);
70         return true;
71     };
72 
73     NamespaceCallback nsCallback = [&](const abckit::core::Namespace &ns) -> bool {
74         ns.EnumerateNamespaces(nsCallback);
75         ns.EnumerateClasses(clsCallback);
76         ns.EnumerateTopLevelFunctions(fnCallback);
77         return true;
78     };
79 
80     mod.EnumerateNamespaces(nsCallback);
81     mod.EnumerateClasses(clsCallback);
82     mod.EnumerateTopLevelFunctions(fnCallback);
83 }
84 
GetImportDescriptors(const abckit::core::Module & mod,const std::set<ClassMeta> & baseClasses)85 std::vector<abckit::core::ImportDescriptor> GetImportDescriptors(const abckit::core::Module &mod,
86                                                                  const std::set<ClassMeta> &baseClasses)
87 {
88     std::vector<abckit::core::ImportDescriptor> idescs;
89     mod.EnumerateImports([&](abckit::core::ImportDescriptor idesc) -> bool {
90         const ClassMeta idescMeta {idesc.GetImportedModule().GetName(), idesc.GetName()};
91         for (const auto &baseClass : baseClasses) {
92             if (idescMeta == baseClass) {
93                 idescs.push_back(std::move(idesc));
94                 return true;
95             }
96         }
97         return true;
98     });
99     return idescs;
100 }
101 
FindClassMetaFromLoadApi(const abckit::core::ImportDescriptor & idesc,const abckit::Instruction & inst)102 std::optional<ClassMeta> FindClassMetaFromLoadApi(const abckit::core::ImportDescriptor &idesc,
103                                                   const abckit::Instruction &inst)
104 {
105     if (inst.GetGraph()->DynIsa().GetOpcode(inst) != ABCKIT_ISA_API_DYNAMIC_OPCODE_LDEXTERNALMODULEVAR ||
106         inst.GetGraph()->DynIsa().GetImportDescriptor(inst) != idesc) {
107         return std::nullopt;
108     }
109 
110     std::optional<ClassMeta> definedClass;
111     inst.VisitUsers([&](const abckit::Instruction &user) -> bool {
112         if (!definedClass &&
113             user.GetGraph()->DynIsa().GetOpcode(user) == ABCKIT_ISA_API_DYNAMIC_OPCODE_DEFINECLASSWITHBUFFER) {
114             auto klass = user.GetFunction().GetParentClass();
115             auto mod = klass.GetModule();
116             definedClass = {mod.GetName(), klass.GetName()};
117             return false;
118         }
119         return true;
120     });
121 
122     return definedClass;
123 }
124 
CollectSubClasses(const abckit::Instruction & inst,const std::vector<abckit::core::ImportDescriptor> & importDescriptors,std::set<ClassMeta> & subClasses)125 void CollectSubClasses(const abckit::Instruction &inst,
126                        const std::vector<abckit::core::ImportDescriptor> &importDescriptors,
127                        std::set<ClassMeta> &subClasses)
128 {
129     for (auto &idesc : importDescriptors) {
130         if (auto maybeMeta = FindClassMetaFromLoadApi(idesc, inst)) {
131             subClasses.insert(*maybeMeta);
132         }
133     }
134 }
135 
CollectSubClasses(const abckit::core::Function & method,const std::vector<abckit::core::ImportDescriptor> & importDescriptors,std::set<ClassMeta> & subClasses)136 void CollectSubClasses(const abckit::core::Function &method,
137                        const std::vector<abckit::core::ImportDescriptor> &importDescriptors,
138                        std::set<ClassMeta> &subClasses)
139 {
140     auto graph = method.CreateGraph();
141     graph.EnumerateBasicBlocksRpo([&](const abckit::BasicBlock &bb) -> bool {
142         for (auto inst = bb.GetFirstInst(); !!inst; inst = inst.GetNext()) {
143             CollectSubClasses(inst, importDescriptors, subClasses);
144         }
145         return true;
146     });
147 }
148 
149 }  // namespace
150 
151 namespace libabckit::test {
152 
153 class AbckitScenarioCppTestClean : public ::testing::Test {};
154 
155 // Test: test-kind=scenario, abc-kind=ArkTS1, category=positive, extension=cpp
TEST_F(AbckitScenarioCppTestClean,LibAbcKitTestScanSubclassesClean)156 TEST_F(AbckitScenarioCppTestClean, LibAbcKitTestScanSubclassesClean)
157 {
158     const std::string inputPath = ABCKIT_ABC_DIR "clean_scenarios/cpp_api/dynamic/scan_subclasses/scan_subclasses.abc";
159     abckit::File file(inputPath, std::make_unique<GTestAssertErrorHandler>());
160 
161     const std::set<ClassMeta> baseClasses = {{"modules/base", "Base"}};
162 
163     std::set<ClassMeta> subClasses;
164     file.EnumerateModules([&](const abckit::core::Module &mod) {
165         auto importDescriptors = GetImportDescriptors(mod, baseClasses);
166         if (importDescriptors.empty()) {
167             return true;
168         }
169 
170         EnumerateAllMethods(mod, [&](const abckit::core::Function &method) {
171             CollectSubClasses(method, importDescriptors, subClasses);
172             return true;
173         });
174         return true;
175     });
176 
177     const std::set<ClassMeta> expectedSubClasses = {{"scan_subclasses", "Child1"}, {"scan_subclasses", "Child2"}};
178 
179     EXPECT_FALSE(subClasses.empty());
180     EXPECT_TRUE(subClasses == expectedSubClasses);
181 }
182 
183 }  // namespace libabckit::test
184