blob: a383e3623dd0c18c8c75be6155bcb9d64d3a175d [file] [log] [blame]
Yifan Hong676447a2016-11-15 12:57:23 -08001/*
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
17#include "CompatibilityMatrix.h"
18
Yifan Hongb6c7f492018-02-27 14:07:57 -080019#include <iostream>
Yifan Hongdbe9db32017-12-11 19:06:11 -080020#include <utility>
21
Yifan Hong1e8febd2019-08-07 16:17:19 -070022#include <android-base/logging.h>
Yifan Honge7837b12018-10-11 10:38:57 -070023#include <android-base/strings.h>
24
Yifan Hongbbfff302017-06-06 17:10:13 -070025#include "parse_string.h"
Yifan Hongddae77e2017-12-18 16:57:07 -080026#include "parse_xml.h"
Yifan Hongb6c7f492018-02-27 14:07:57 -080027#include "utils.h"
Yifan Hongddae77e2017-12-18 16:57:07 -080028
Yifan Hong676447a2016-11-15 12:57:23 -080029namespace android {
30namespace vintf {
31
Yifan Hong4c6962d2019-01-30 15:50:46 -080032using details::mergeField;
33
Yifan Honge7837b12018-10-11 10:38:57 -070034bool CompatibilityMatrix::addKernel(MatrixKernel&& kernel, std::string* error) {
Yifan Hong7c7d7062017-04-04 16:26:51 -070035 if (mType != SchemaType::FRAMEWORK) {
Yifan Honge7837b12018-10-11 10:38:57 -070036 if (error) {
37 *error = "Cannot add <kernel> to a " + to_string(mType) + " compatibility matrix.";
38 }
Yifan Hong7c7d7062017-04-04 16:26:51 -070039 return false;
40 }
Yifan Honge7837b12018-10-11 10:38:57 -070041
Yifan Hongb343c5b2019-12-11 18:13:59 -080042 if (kernel.getSourceMatrixLevel() == Level::UNSPECIFIED) {
43 kernel.setSourceMatrixLevel(level());
44 }
45
Yifan Honge7837b12018-10-11 10:38:57 -070046 auto it = framework.mKernels.begin();
47 for (; it != framework.mKernels.end(); ++it) {
48 if (it->minLts() == kernel.minLts()) {
49 break;
50 }
51 if (it->minLts().version == kernel.minLts().version &&
52 it->minLts().majorRev == kernel.minLts().majorRev) {
53 if (error) {
54 *error = "Kernel version mismatch; cannot add " + to_string(kernel.minLts()) +
55 " because " + to_string(it->minLts()) + " was added.";
56 }
57 return false;
58 }
59 }
60
61 bool seenVersion = it != framework.mKernels.end();
62
63 if (seenVersion) {
64 // If no conditions, must be the first among the same minLts
65 // because O libvintf only checks the first <kernel> tag that version matches.
66 if (kernel.conditions().empty()) {
67 // Found first <kernel> with the same minLts.
68 // Append config if it does not have <condition>s, else error.
69 if (it->conditions().empty()) {
70 const auto& configs = kernel.configs();
71 it->mConfigs.insert(it->mConfigs.end(), configs.begin(), configs.end());
72 } else {
73 if (error) {
74 *error =
75 "Base compatibility matrix has <condition> for the first <kernel> "
76 "with minlts " +
77 to_string(kernel.minLts()) + " for unknown reason.";
78 }
79 return false;
80 }
81 return true;
82 }
83 } else {
84 // First <kernel> of a minLts must not have <condition>'s for backwards compatibility
85 // with O libvintf.
86 if (!kernel.conditions().empty()) {
87 framework.mKernels.push_back(MatrixKernel(KernelVersion{kernel.minLts()}, {}));
88 }
89 }
90
Yifan Hong7c7d7062017-04-04 16:26:51 -070091 framework.mKernels.push_back(std::move(kernel));
Yifan Hong676447a2016-11-15 12:57:23 -080092 return true;
93}
94
Yifan Hong398f4c72017-04-13 20:18:01 -070095SchemaType CompatibilityMatrix::type() const {
96 return mType;
97}
98
Yifan Hong2027a492017-12-11 15:21:19 -080099Level CompatibilityMatrix::level() const {
100 return mLevel;
101}
102
Yifan Hong9f78c182018-07-12 14:45:52 -0700103status_t CompatibilityMatrix::fetchAllInformation(const FileSystem* fileSystem,
104 const std::string& path, std::string* error) {
105 return details::fetchAllInformation(fileSystem, path, gCompatibilityMatrixConverter, this,
106 error);
Yifan Hong1e5a0542017-04-28 14:37:56 -0700107}
108
Yifan Hongbbfff302017-06-06 17:10:13 -0700109std::string CompatibilityMatrix::getXmlSchemaPath(const std::string& xmlFileName,
110 const Version& version) const {
111 using std::literals::string_literals::operator""s;
112 auto range = getXmlFiles(xmlFileName);
113 for (auto it = range.first; it != range.second; ++it) {
114 const MatrixXmlFile& matrixXmlFile = it->second;
115 if (matrixXmlFile.versionRange().contains(version)) {
116 if (!matrixXmlFile.overriddenPath().empty()) {
117 return matrixXmlFile.overriddenPath();
118 }
119 return "/"s + (type() == SchemaType::DEVICE ? "vendor" : "system") + "/etc/" +
120 xmlFileName + "_V" + std::to_string(matrixXmlFile.versionRange().majorVer) +
121 "_" + std::to_string(matrixXmlFile.versionRange().maxMinor) + "." +
122 to_string(matrixXmlFile.format());
123 }
124 }
125 return "";
126}
127
Yifan Honge7e45532018-03-16 18:11:49 -0700128// Split existingHal into a HAL that contains only interface/instance and a HAL
129// that does not contain it. Return the HAL that contains only interface/instance.
130// - Return nullptr if existingHal does not contain interface/instance
131// - Return existingHal if existingHal contains only interface/instance
132// - Remove interface/instance from existingHal, and return a new MatrixHal (that is added
133// to "this") that contains only interface/instance.
134MatrixHal* CompatibilityMatrix::splitInstance(MatrixHal* existingHal, const std::string& interface,
Yifan Hong643a9ef2018-03-21 14:13:55 -0700135 const std::string& instanceOrPattern, bool isRegex) {
136 bool found = false;
137 bool foundOthers = false;
138 existingHal->forEachInstance([&](const auto& matrixInstance) {
139 bool interfaceMatch = matrixInstance.interface() == interface;
140 bool instanceMatch = false;
141 if (matrixInstance.isRegex() && isRegex) {
142 instanceMatch = (matrixInstance.regexPattern() == instanceOrPattern);
143 } else if (!matrixInstance.isRegex() && !isRegex) {
144 instanceMatch = (matrixInstance.exactInstance() == instanceOrPattern);
145 }
146
147 bool match = interfaceMatch && instanceMatch;
148
149 found |= match;
150 foundOthers |= (!match);
151
152 return !found || !foundOthers;
153 });
154
155 if (!found) {
Yifan Honge7e45532018-03-16 18:11:49 -0700156 return nullptr;
157 }
158
Yifan Hong643a9ef2018-03-21 14:13:55 -0700159 if (!foundOthers) {
Yifan Honge7e45532018-03-16 18:11:49 -0700160 return existingHal;
161 }
162
Yifan Hong643a9ef2018-03-21 14:13:55 -0700163 existingHal->removeInstance(interface, instanceOrPattern, isRegex);
Yifan Honge7e45532018-03-16 18:11:49 -0700164 MatrixHal copy = *existingHal;
165 copy.clearInstances();
Yifan Hong643a9ef2018-03-21 14:13:55 -0700166 copy.insertInstance(interface, instanceOrPattern, isRegex);
Yifan Honge7e45532018-03-16 18:11:49 -0700167
168 return addInternal(std::move(copy));
169}
170
Yifan Hong7967d7b2018-03-15 17:08:58 -0700171// Add all package@other_version::interface/instance as an optional instance.
172// If package@this_version::interface/instance is in this (that is, some instance
173// with the same package and interface and instance exists), then other_version is
174// considered a possible replacement to this_version.
175// See LibVintfTest.AddOptionalHal* tests for details.
Yifan Hongdbe9db32017-12-11 19:06:11 -0800176bool CompatibilityMatrix::addAllHalsAsOptional(CompatibilityMatrix* other, std::string* error) {
177 if (other == nullptr || other->level() <= level()) {
178 return true;
179 }
180
181 for (auto& pair : other->mHals) {
182 const std::string& name = pair.first;
183 MatrixHal& halToAdd = pair.second;
Yifan Hongdbe9db32017-12-11 19:06:11 -0800184
Yifan Honge7e45532018-03-16 18:11:49 -0700185 std::set<std::pair<std::string, std::string>> insertedInstances;
Yifan Hong643a9ef2018-03-21 14:13:55 -0700186 std::set<std::pair<std::string, std::string>> insertedRegex;
Yifan Honge7e45532018-03-16 18:11:49 -0700187 auto existingHals = getHals(name);
188
189 halToAdd.forEachInstance([&](const std::vector<VersionRange>& versionRanges,
Yifan Hong643a9ef2018-03-21 14:13:55 -0700190 const std::string& interface,
191 const std::string& instanceOrPattern, bool isRegex) {
Yifan Honge7e45532018-03-16 18:11:49 -0700192 for (auto* existingHal : existingHals) {
Yifan Hong5bb4fbf2019-09-10 14:47:28 -0700193 // Ignore HALs with different format.
194 if (halToAdd.format != existingHal->format) {
195 continue;
196 }
197
Yifan Hong643a9ef2018-03-21 14:13:55 -0700198 MatrixHal* splitInstance =
199 this->splitInstance(existingHal, interface, instanceOrPattern, isRegex);
Yifan Honge7e45532018-03-16 18:11:49 -0700200 if (splitInstance != nullptr) {
201 splitInstance->insertVersionRanges(versionRanges);
Yifan Hong643a9ef2018-03-21 14:13:55 -0700202 if (isRegex) {
203 insertedRegex.insert(std::make_pair(interface, instanceOrPattern));
204 } else {
205 insertedInstances.insert(std::make_pair(interface, instanceOrPattern));
206 }
Yifan Honge7e45532018-03-16 18:11:49 -0700207 }
Yifan Hongdbe9db32017-12-11 19:06:11 -0800208 }
Yifan Honge7e45532018-03-16 18:11:49 -0700209 return true;
210 });
211
212 // Add the remaining instances.
213 for (const auto& pair : insertedInstances) {
Yifan Hong643a9ef2018-03-21 14:13:55 -0700214 halToAdd.removeInstance(pair.first, pair.second, false /* isRegex */);
215 }
216 for (const auto& pair : insertedRegex) {
217 halToAdd.removeInstance(pair.first, pair.second, true /* isRegex */);
Yifan Hong7967d7b2018-03-15 17:08:58 -0700218 }
Yifan Hongdbe9db32017-12-11 19:06:11 -0800219
Yifan Hong7e9e04d2018-03-20 13:06:00 -0700220 if (halToAdd.instancesCount() > 0) {
Yifan Hong7967d7b2018-03-15 17:08:58 -0700221 halToAdd.setOptional(true);
222 if (!add(std::move(halToAdd))) {
223 if (error) {
224 *error = "Cannot add HAL " + name + " for unknown reason.";
Yifan Hongdbe9db32017-12-11 19:06:11 -0800225 }
226 return false;
227 }
Yifan Hongdbe9db32017-12-11 19:06:11 -0800228 }
229 }
230 return true;
231}
232
Yifan Hongd4b92fe2017-12-20 15:29:03 -0800233bool CompatibilityMatrix::addAllXmlFilesAsOptional(CompatibilityMatrix* other, std::string* error) {
234 if (other == nullptr || other->level() <= level()) {
235 return true;
236 }
237 for (auto& pair : other->mXmlFiles) {
238 const std::string& name = pair.first;
239 MatrixXmlFile& xmlFileToAdd = pair.second;
240
241 xmlFileToAdd.mOptional = true;
242 if (!addXmlFile(std::move(xmlFileToAdd))) {
243 if (error) {
244 *error = "Cannot add XML File " + name + " for unknown reason.";
245 }
246 return false;
247 }
248 }
249 return true;
250}
251
Yifan Honge7837b12018-10-11 10:38:57 -0700252// Merge Kernel.
253// Add <kernel> from exact "level", then optionally add <kernel> from high levels to low levels.
254// For example, (each letter is a kernel version x.y.z)
255// 1.xml: A1, B1
256// 2.xml: B2, C2, D2
257// 3.xml: D3, E3
258// Then the combined 1.xml should have
259// A1, B1 (from 1.xml, required), C2, D2, E3 (optional, use earliest possible).
260bool CompatibilityMatrix::addAllKernels(CompatibilityMatrix* other, std::string* error) {
261 for (MatrixKernel& kernel : other->framework.mKernels) {
Yifan Honga39ecfe2019-12-11 17:55:30 -0800262 if (kernel.getSourceMatrixLevel() == Level::UNSPECIFIED) {
263 kernel.setSourceMatrixLevel(other->level());
264 }
Yifan Honge7837b12018-10-11 10:38:57 -0700265 KernelVersion ver = kernel.minLts();
266 if (!addKernel(std::move(kernel), error)) {
267 if (error) {
268 *error = "Cannot add kernel version " + to_string(ver) + ": " + *error;
269 }
270 return false;
271 }
272 }
273 return true;
274}
275
Yifan Hongd6de7f62018-04-26 18:40:02 -0700276bool CompatibilityMatrix::addAllKernelsAsOptional(CompatibilityMatrix* other, std::string* error) {
277 if (other == nullptr || other->level() <= level()) {
278 return true;
279 }
280
281 for (MatrixKernel& kernelToAdd : other->framework.mKernels) {
282 bool exists =
283 std::any_of(this->framework.mKernels.begin(), this->framework.mKernels.end(),
284 [&kernelToAdd](const MatrixKernel& existing) {
285 return kernelToAdd.minLts().version == existing.minLts().version &&
286 kernelToAdd.minLts().majorRev == existing.minLts().majorRev;
287 });
288
289 if (exists) {
290 // Shouldn't retroactively add requirements to minLts(), so ignore this.
291 // This happens even when kernelToAdd.conditions() != existing.conditions().
292 continue;
293 }
294
Yifan Honga39ecfe2019-12-11 17:55:30 -0800295 if (kernelToAdd.getSourceMatrixLevel() == Level::UNSPECIFIED) {
296 kernelToAdd.setSourceMatrixLevel(other->level());
297 }
Yifan Hong1e8febd2019-08-07 16:17:19 -0700298
Yifan Hongd6de7f62018-04-26 18:40:02 -0700299 KernelVersion minLts = kernelToAdd.minLts();
Yifan Honge7837b12018-10-11 10:38:57 -0700300 if (!addKernel(std::move(kernelToAdd), error)) {
Yifan Hongd6de7f62018-04-26 18:40:02 -0700301 if (error) {
Yifan Honge7837b12018-10-11 10:38:57 -0700302 *error = "Cannot add " + to_string(minLts) + ": " + *error;
Yifan Hongd6de7f62018-04-26 18:40:02 -0700303 }
304 return false;
305 }
306 }
307 return true;
308}
309
Yifan Honge7837b12018-10-11 10:38:57 -0700310bool CompatibilityMatrix::addSepolicy(CompatibilityMatrix* other, std::string* error) {
311 bool success = mergeField(&this->framework.mSepolicy, &other->framework.mSepolicy);
312 if (!success && error) *error = "<sepolicy> is already defined";
313 return success;
314}
315
316bool CompatibilityMatrix::addAvbMetaVersion(CompatibilityMatrix* other, std::string* error) {
317 bool success = mergeField(&this->framework.mAvbMetaVersion, &other->framework.mAvbMetaVersion);
318 if (!success && error) *error = "<avb><vbmeta-version> is already defined";
319 return success;
320}
321
Yifan Hong5a51ea52019-04-22 14:54:57 -0700322bool CompatibilityMatrix::addVndk(CompatibilityMatrix* other, std::string* error) {
323#pragma clang diagnostic push
324#pragma clang diagnostic ignored "-Wdeprecated-declarations"
325 bool success = mergeField(&this->device.mVndk, &other->device.mVndk);
326#pragma clang diagnostic pop
327 if (!success && error) *error = "<vndk> is already defined";
328 return success;
329}
330
331bool CompatibilityMatrix::addVendorNdk(CompatibilityMatrix* other, std::string* error) {
332 bool success = mergeField(&this->device.mVendorNdk, &other->device.mVendorNdk);
333 if (!success && error) *error = "<vendor-ndk> is already defined";
334 return success;
335}
336
337bool CompatibilityMatrix::addSystemSdk(CompatibilityMatrix* other, std::string* /* error */) {
338 this->device.mSystemSdk.addAll(&other->device.mSystemSdk);
339 return true;
340}
341
Yifan Hongfb7469c2017-04-05 19:15:21 -0700342bool operator==(const CompatibilityMatrix &lft, const CompatibilityMatrix &rgt) {
Yifan Hong2027a492017-12-11 15:21:19 -0800343 return lft.mType == rgt.mType && lft.mLevel == rgt.mLevel && lft.mHals == rgt.mHals &&
344 lft.mXmlFiles == rgt.mXmlFiles &&
Yifan Hongfeb454e2018-01-09 16:16:40 -0800345 (lft.mType != SchemaType::DEVICE ||
346 (
Yifan Hong0f529fa2018-01-10 14:51:59 -0800347#pragma clang diagnostic push
348#pragma clang diagnostic ignored "-Wdeprecated-declarations"
Yifan Hongfeb454e2018-01-09 16:16:40 -0800349 lft.device.mVndk == rgt.device.mVndk &&
Yifan Hong0f529fa2018-01-10 14:51:59 -0800350#pragma clang diagnostic pop
Yifan Honga28729e2018-01-17 13:40:35 -0800351 lft.device.mVendorNdk == rgt.device.mVendorNdk &&
352 lft.device.mSystemSdk == rgt.device.mSystemSdk)) &&
Yifan Hongd4857902017-06-13 14:13:56 -0700353 (lft.mType != SchemaType::FRAMEWORK ||
354 (lft.framework.mKernels == rgt.framework.mKernels &&
355 lft.framework.mSepolicy == rgt.framework.mSepolicy &&
356 lft.framework.mAvbMetaVersion == rgt.framework.mAvbMetaVersion));
Yifan Hongfb7469c2017-04-05 19:15:21 -0700357}
358
Yifan Honge7837b12018-10-11 10:38:57 -0700359std::unique_ptr<CompatibilityMatrix> CompatibilityMatrix::combine(
360 Level deviceLevel, std::vector<Named<CompatibilityMatrix>>* matrices, std::string* error) {
361 // Check type.
362 for (const auto& e : *matrices) {
363 if (e.object.type() != SchemaType::FRAMEWORK) {
364 if (error) {
365 *error = "File \"" + e.name + "\" is not a framework compatibility matrix.";
366 return nullptr;
367 }
368 }
369 }
370
371 // Matrices with unspecified (empty) level are auto-filled with deviceLevel.
Yifan Hongddae77e2017-12-18 16:57:07 -0800372 for (auto& e : *matrices) {
Yifan Hongffcaf992018-01-09 17:08:51 -0800373 if (e.object.level() == Level::UNSPECIFIED) {
Yifan Honge7837b12018-10-11 10:38:57 -0700374 e.object.mLevel = deviceLevel;
Yifan Hongddae77e2017-12-18 16:57:07 -0800375 }
376 }
377
Yifan Honge7837b12018-10-11 10:38:57 -0700378 // Add from low to high FCM version so that optional <kernel> requirements are added correctly.
379 // See comment in addAllAsOptional.
380 std::sort(matrices->begin(), matrices->end(),
381 [](const auto& x, const auto& y) { return x.object.level() < y.object.level(); });
382
383 auto baseMatrix = std::make_unique<CompatibilityMatrix>();
384 baseMatrix->mLevel = deviceLevel;
385 baseMatrix->mType = SchemaType::FRAMEWORK;
386
387 std::vector<std::string> parsedFiles;
388 for (auto& e : *matrices) {
389 if (e.object.level() < deviceLevel) {
390 continue;
Yifan Hongddae77e2017-12-18 16:57:07 -0800391 }
Yifan Honge7837b12018-10-11 10:38:57 -0700392
393 bool success = false;
394 if (e.object.level() == deviceLevel) {
395 success = baseMatrix->addAll(&e, error);
396 } else {
397 success = baseMatrix->addAllAsOptional(&e, error);
398 }
399 if (!success) {
400 if (error) {
401 *error = "Conflict when merging \"" + e.name + "\": " + *error + "\n" +
402 "Previous files:\n" + base::Join(parsedFiles, "\n");
403 }
404 return nullptr;
405 }
406 parsedFiles.push_back(e.name);
Yifan Hongddae77e2017-12-18 16:57:07 -0800407 }
Yifan Honge7837b12018-10-11 10:38:57 -0700408
409 return baseMatrix;
Yifan Hongddae77e2017-12-18 16:57:07 -0800410}
411
Yifan Hong5a51ea52019-04-22 14:54:57 -0700412std::unique_ptr<CompatibilityMatrix> CompatibilityMatrix::combineDeviceMatrices(
413 std::vector<Named<CompatibilityMatrix>>* matrices, std::string* error) {
414 auto baseMatrix = std::make_unique<CompatibilityMatrix>();
415 baseMatrix->mType = SchemaType::DEVICE;
416
417 std::vector<std::string> parsedFiles;
418 for (auto& e : *matrices) {
419 bool success = baseMatrix->addAll(&e, error);
420 if (!success) {
421 if (error) {
422 *error = "Conflict when merging \"" + e.name + "\": " + *error + "\n" +
423 "Previous files:\n" + base::Join(parsedFiles, "\n");
424 }
425 return nullptr;
426 }
427 parsedFiles.push_back(e.name);
428 }
429 return baseMatrix;
430}
431
Yifan Honge7837b12018-10-11 10:38:57 -0700432bool CompatibilityMatrix::addAll(Named<CompatibilityMatrix>* inputMatrix, std::string* error) {
433 if (!addAllHals(&inputMatrix->object, error) || !addAllXmlFiles(&inputMatrix->object, error) ||
434 !addAllKernels(&inputMatrix->object, error) || !addSepolicy(&inputMatrix->object, error) ||
Yifan Hong5a51ea52019-04-22 14:54:57 -0700435 !addAvbMetaVersion(&inputMatrix->object, error) || !addVndk(&inputMatrix->object, error) ||
436 !addVendorNdk(&inputMatrix->object, error) || !addSystemSdk(&inputMatrix->object, error)) {
Yifan Honge7837b12018-10-11 10:38:57 -0700437 if (error) {
438 *error = "File \"" + inputMatrix->name + "\" cannot be added: " + *error + ".";
Yifan Hongd6de7f62018-04-26 18:40:02 -0700439 }
Yifan Hong5390e3b2019-01-31 14:13:00 -0800440 return false;
Yifan Hongd6de7f62018-04-26 18:40:02 -0700441 }
442 return true;
443}
444
Yifan Honge7837b12018-10-11 10:38:57 -0700445bool CompatibilityMatrix::addAllAsOptional(Named<CompatibilityMatrix>* inputMatrix,
446 std::string* error) {
447 if (!addAllHalsAsOptional(&inputMatrix->object, error) ||
448 !addAllXmlFilesAsOptional(&inputMatrix->object, error) ||
449 !addAllKernelsAsOptional(&inputMatrix->object, error)) {
450 if (error) {
451 *error = "File \"" + inputMatrix->name + "\" cannot be added: " + *error;
Yifan Hongddae77e2017-12-18 16:57:07 -0800452 }
Yifan Honge7837b12018-10-11 10:38:57 -0700453 return false;
Yifan Hongddae77e2017-12-18 16:57:07 -0800454 }
Yifan Honge7837b12018-10-11 10:38:57 -0700455 // ignore <sepolicy> requirement from higher level
456 // ignore <avb> requirement from higher level
457 return true;
Yifan Hongddae77e2017-12-18 16:57:07 -0800458}
459
Yifan Hong2a90ffe2018-03-05 17:45:34 -0800460bool CompatibilityMatrix::forEachInstanceOfVersion(
Yifan Hongfc7ad272019-09-10 19:49:15 -0700461 HalFormat format, const std::string& package, const Version& expectVersion,
Yifan Hong2a90ffe2018-03-05 17:45:34 -0800462 const std::function<bool(const MatrixInstance&)>& func) const {
463 for (const MatrixHal* hal : getHals(package)) {
464 bool cont = hal->forEachInstance([&](const MatrixInstance& matrixInstance) {
Yifan Hongfc7ad272019-09-10 19:49:15 -0700465 if (matrixInstance.format() == format &&
466 matrixInstance.versionRange().contains(expectVersion)) {
Yifan Hong2a90ffe2018-03-05 17:45:34 -0800467 return func(matrixInstance);
Yifan Honge3a92342018-01-25 17:00:16 -0800468 }
Yifan Hong2a90ffe2018-03-05 17:45:34 -0800469 return true;
470 });
471 if (!cont) return false;
Yifan Honge3a92342018-01-25 17:00:16 -0800472 }
Yifan Hong2a90ffe2018-03-05 17:45:34 -0800473 return true;
Yifan Honge3a92342018-01-25 17:00:16 -0800474}
475
Yifan Hong1dc5a472019-09-10 19:40:06 -0700476bool CompatibilityMatrix::matchInstance(HalFormat format, const std::string& halName,
477 const Version& version, const std::string& interfaceName,
Yifan Hongdef7e7f2018-03-20 13:27:36 -0700478 const std::string& instance) const {
479 bool found = false;
Yifan Hong1dc5a472019-09-10 19:40:06 -0700480 (void)forEachInstanceOfInterface(format, halName, version, interfaceName,
Yifan Hongdef7e7f2018-03-20 13:27:36 -0700481 [&found, &instance](const auto& e) {
482 found |= (e.matchInstance(instance));
483 return !found; // if not found, continue
484 });
485 return found;
486}
487
Yifan Hong52b7fae2018-05-22 16:21:31 -0700488std::string CompatibilityMatrix::getVendorNdkVersion() const {
489 return type() == SchemaType::DEVICE ? device.mVendorNdk.version() : "";
490}
491
Yifan Hong1e8febd2019-08-07 16:17:19 -0700492Level CompatibilityMatrix::getSourceMatrixLevel(const MatrixKernel* matrixKernel) const {
493 CHECK(std::find_if(framework.mKernels.begin(), framework.mKernels.end(),
494 [matrixKernel](const auto& e) { return &e == matrixKernel; }) !=
495 framework.mKernels.end());
496 Level ret = matrixKernel->getSourceMatrixLevel();
497 if (ret != Level::UNSPECIFIED) return ret;
498 return level();
499}
500
Yifan Hong676447a2016-11-15 12:57:23 -0800501} // namespace vintf
502} // namespace android