1 /*
2  * Copyright (C) 2017 The Android Open Source Project
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 #define LOG_TAG "VtsTestabilityChecker"
17 
18 #include "VtsTestabilityChecker.h"
19 
20 #include <algorithm>
21 #include <iostream>
22 #include <set>
23 
24 #include <android-base/strings.h>
25 #include <vintf/parse_string.h>
26 
27 using android::base::Join;
28 using android::vintf::Arch;
29 using android::vintf::CompatibilityMatrix;
30 using android::vintf::gArchStrings;
31 using android::vintf::HalManifest;
32 using android::vintf::ManifestHal;
33 using android::vintf::ManifestInstance;
34 using android::vintf::MatrixHal;
35 using android::vintf::MatrixInstance;
36 using android::vintf::toFQNameString;
37 using android::vintf::Transport;
38 using android::vintf::Version;
39 using android::vintf::operator<<;
40 using std::set;
41 using std::string;
42 using std::vector;
43 
44 namespace android {
45 namespace vts {
46 
CheckHalForComplianceTest(const string & hal_package_name,const Version & hal_version,const string & hal_interface_name,const Arch & arch,set<string> * instances)47 bool VtsTestabilityChecker::CheckHalForComplianceTest(
48     const string& hal_package_name, const Version& hal_version,
49     const string& hal_interface_name, const Arch& arch,
50     set<string>* instances) {
51   CHECK(instances) << "instances set should not be NULL.";
52   set<string> famework_hal_instances;
53   set<string> vendor_hal_instances;
54   bool check_framework_hal = CheckFrameworkManifestHal(
55       hal_package_name, hal_version, hal_interface_name, arch,
56       &famework_hal_instances);
57   bool check_vendor_hal =
58       CheckVendorManifestHal(hal_package_name, hal_version, hal_interface_name,
59                              arch, &vendor_hal_instances);
60   set_union(famework_hal_instances.begin(), famework_hal_instances.end(),
61             vendor_hal_instances.begin(), vendor_hal_instances.end(),
62             std::inserter(*instances, instances->begin()));
63   return check_framework_hal || check_vendor_hal;
64 }
65 
CheckHalForNonComplianceTest(const string & hal_package_name,const Version & hal_version,const string & hal_interface_name,const Arch & arch,set<string> * instances)66 bool VtsTestabilityChecker::CheckHalForNonComplianceTest(
67     const string& hal_package_name, const Version& hal_version,
68     const string& hal_interface_name, const Arch& arch,
69     set<string>* instances) {
70   CHECK(instances) << "instances set should not be NULL.";
71   set<string> vendor_hal_instances;
72   set<string> test_hal_instances;
73   bool check_vendor_hal =
74       CheckVendorManifestHal(hal_package_name, hal_version, hal_interface_name,
75                              arch, &vendor_hal_instances);
76 
77   bool check_test_hal = CheckTestHalWithHwManager(
78       hal_package_name, hal_version, hal_interface_name, &test_hal_instances);
79 
80   set_union(vendor_hal_instances.begin(), vendor_hal_instances.end(),
81             test_hal_instances.begin(), test_hal_instances.end(),
82             std::inserter(*instances, instances->begin()));
83   return check_vendor_hal || check_test_hal;
84 }
85 
FindInstance(const vector<ManifestInstance> & manifest_instances,const MatrixInstance & matrix_instance)86 vector<const ManifestInstance*> VtsTestabilityChecker::FindInstance(
87     const vector<ManifestInstance>& manifest_instances,
88     const MatrixInstance& matrix_instance) {
89   vector<const ManifestInstance*> ret;
90   for (const auto& e : manifest_instances) {
91     if (matrix_instance.matchInstance(e.instance())) {
92       ret.push_back(&e);
93     }
94   }
95   return ret;
96 }
97 
FindInterface(const vector<ManifestInstance> & manifest_instances,const MatrixInstance & matrix_instance)98 vector<const ManifestInstance*> VtsTestabilityChecker::FindInterface(
99     const vector<ManifestInstance>& manifest_instances,
100     const MatrixInstance& matrix_instance) {
101   vector<const ManifestInstance*> ret;
102   for (const auto& e : manifest_instances) {
103     if (e.interface() == matrix_instance.interface()) {
104       ret.push_back(&e);
105     }
106   }
107   return ret;
108 }
109 
CheckFrameworkCompatibleHal(const string & hal_package_name,const Version & hal_version,const string & hal_interface_name,const Arch & arch,set<string> * instances)110 bool VtsTestabilityChecker::CheckFrameworkCompatibleHal(
111     const string& hal_package_name, const Version& hal_version,
112     const string& hal_interface_name, const Arch& arch,
113     set<string>* instances) {
114   CHECK(instances) << "instances set should not be NULL.";
115 
116   auto matrix_instances = framework_comp_matrix_->getHidlFqInstances(
117       hal_package_name, hal_version, hal_interface_name);
118   auto manifest_instances = device_hal_manifest_->getHidlFqInstances(
119       hal_package_name, hal_version, hal_interface_name);
120 
121   bool testable = false;
122 
123   for (const auto& matrix_instance : matrix_instances) {
124     const auto& matched_instances =
125         FindInstance(manifest_instances, matrix_instance);
126     if (!matrix_instance.optional() && matched_instances.empty()) {
127       // In matrix but not in manifest.
128       // The test should still run, but expect the test
129       // to fail (due to incompatible vendor and framework HAL).
130       LOG(ERROR) << "Compatibility error. Hal " << hal_package_name
131                  << " is required by framework but not supported by vendor";
132       if (!hal_interface_name.empty()) {
133         if (!matrix_instance.isRegex()) {
134           instances->insert(matrix_instance.exactInstance());
135         } else {
136           LOG(ERROR) << "Ignore regex-instance '"
137                      << matrix_instance.regexPattern();
138         }
139       }
140       testable |= true;
141       continue;
142     }
143 
144     if (hal_interface_name.empty()) {
145       testable |= !matched_instances.empty();
146       continue;
147     }
148 
149     auto get_testable_instances =
150         [&](const vector<const vintf::ManifestInstance*>& manifest_instances) {
151           vector<string> ret;
152           for (const auto& manifest_instance : manifest_instances) {
153             if ((manifest_instance->transport() == Transport::PASSTHROUGH &&
154                  CheckPassthroughManifestArch(manifest_instance->arch(),
155                                               arch)) ||
156                 manifest_instance->transport() == Transport::HWBINDER) {
157               ret.push_back(manifest_instance->instance());
158             }
159           }
160           return ret;
161         };
162 
163     auto testable_instances = get_testable_instances(matched_instances);
164     if (!testable_instances.empty()) {
165       instances->insert(testable_instances.begin(), testable_instances.end());
166       testable |= true;
167       continue;
168     }
169 
170     // Special case: if a.h.foo@1.0::IFoo/default is in matrix but /custom
171     // is in manifest, the interface is still testable, but /default should
172     // not be added to instances.
173     const auto& matched_interface_instances =
174         FindInterface(manifest_instances, matrix_instance);
175     auto testable_interfaces =
176         get_testable_instances(matched_interface_instances);
177     if (!testable_interfaces.empty()) {
178       testable |= true;
179       continue;
180     }
181   }
182   if (instances->empty()) {
183     LOG(ERROR) << "Hal "
184                << toFQNameString(hal_package_name, hal_version,
185                                  hal_interface_name)
186                << " has no testable instance";
187   }
188   return testable;
189 }
190 
CheckPassthroughManifestArch(const Arch & manifest_arch,const Arch & arch)191 bool VtsTestabilityChecker::CheckPassthroughManifestArch(
192     const Arch& manifest_arch, const Arch& arch) {
193   switch (arch) {
194     case Arch::ARCH_32: {
195       if (android::vintf::has32(manifest_arch)) {
196         return true;
197       }
198       break;
199     }
200     case Arch::ARCH_64: {
201       if (android::vintf::has64(manifest_arch)) {
202         return true;
203       }
204       break;
205     }
206     default: {
207       LOG(ERROR) << "Unexpected arch to check: " << arch;
208       break;
209     }
210   }
211   return false;
212 }
213 
CheckFrameworkManifestHal(const string & hal_package_name,const Version & hal_version,const string & hal_interface_name,const Arch & arch,set<string> * instances)214 bool VtsTestabilityChecker::CheckFrameworkManifestHal(
215     const string& hal_package_name, const Version& hal_version,
216     const string& hal_interface_name, const Arch& arch,
217     set<string>* instances) {
218   return CheckManifestHal(framework_hal_manifest_, hal_package_name,
219                           hal_version, hal_interface_name, arch, instances);
220 }
221 
CheckVendorManifestHal(const string & hal_package_name,const Version & hal_version,const string & hal_interface_name,const Arch & arch,set<string> * instances)222 bool VtsTestabilityChecker::CheckVendorManifestHal(
223     const string& hal_package_name, const Version& hal_version,
224     const string& hal_interface_name, const Arch& arch,
225     set<string>* instances) {
226   return CheckManifestHal(device_hal_manifest_, hal_package_name, hal_version,
227                           hal_interface_name, arch, instances);
228 }
229 
CheckManifestHal(const HalManifest * hal_manifest,const string & hal_package_name,const Version & hal_version,const string & hal_interface_name,const Arch & arch,set<string> * instances)230 bool VtsTestabilityChecker::CheckManifestHal(const HalManifest* hal_manifest,
231                                              const string& hal_package_name,
232                                              const Version& hal_version,
233                                              const string& hal_interface_name,
234                                              const Arch& arch,
235                                              set<string>* instances) {
236   CHECK(instances) << "instances set should not be NULL.";
237 
238   const auto& manifest_instances = hal_manifest->getHidlFqInstances(
239       hal_package_name, hal_version, hal_interface_name);
240 
241   const auto& fq_instance_name =
242       toFQNameString(hal_package_name, hal_version, hal_interface_name);
243 
244   if (manifest_instances.empty()) {
245     LOG(DEBUG) << "Does not find instances for " << fq_instance_name
246                << " in manifest file";
247     return false;
248   }
249 
250   bool testable = false;
251   for (const auto& manifest_instance : manifest_instances) {
252     if (manifest_instance.transport() == Transport::PASSTHROUGH &&
253         !CheckPassthroughManifestArch(manifest_instance.arch(), arch)) {
254       LOG(DEBUG) << "Manifest HAL " << fq_instance_name
255                  << " is passthrough and does not support arch " << arch;
256       continue;  // skip this instance
257     }
258     if (!hal_interface_name.empty()) {
259       instances->insert(manifest_instance.instance());
260     }
261     testable = true;
262   }
263   return testable;
264 }
265 
CheckTestHalWithHwManager(const string & hal_package_name,const Version & hal_version,const string & hal_interface_name,set<string> * instances)266 bool VtsTestabilityChecker::CheckTestHalWithHwManager(
267     const string& hal_package_name, const Version& hal_version,
268     const string& hal_interface_name, set<string>* instances) {
269   CHECK(instances) << "instances set should not be NULL.";
270 
271   string fqName =
272       toFQNameString(hal_package_name, hal_version, hal_interface_name);
273   bool registered = false;
274   hardware::Return<void> res;
275   if (!hal_interface_name.empty()) {
276     res = sm_->listByInterface(fqName, [&](const auto& registered_instances) {
277       for (const string& instance : registered_instances) {
278         registered = true;
279         instances->insert(instance);
280       }
281     });
282   } else {  // handle legacy data without interface info.
283     res = sm_->list([&](const auto& services) {
284       for (const string& service : services) {
285         if (service.find(fqName) == 0) {
286           registered = true;
287           break;
288         }
289       }
290     });
291   }
292   if (!res.isOk()) {
293     LOG(ERROR) << "failed to check services: " << res.description();
294     return false;
295   }
296   return registered;
297 }
298 
299 }  // namespace vts
300 }  // namespace android
301