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