/* * Copyright (C) 2019 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "KernelInfo.h" #include "parse_string.h" namespace android { namespace vintf { const KernelVersion& KernelInfo::version() const { return mVersion; } const std::map& KernelInfo::configs() const { return mConfigs; } bool KernelInfo::matchKernelConfigs(const std::vector& matrixConfigs, std::string* error) const { for (const KernelConfig& matrixConfig : matrixConfigs) { const std::string& key = matrixConfig.first; auto it = this->mConfigs.find(key); if (it == this->mConfigs.end()) { // special case: n matches if the config doesn't exist. if (matrixConfig.second == KernelConfigTypedValue::gMissingConfig) { continue; } if (error != nullptr) { *error = "Missing config " + key; } return false; } const std::string& kernelValue = it->second; if (!matrixConfig.second.matchValue(kernelValue)) { if (error != nullptr) { *error = "For config " + key + ", value = " + kernelValue + " but required " + to_string(matrixConfig.second); } return false; } } return true; } bool KernelInfo::matchKernelVersion(const KernelVersion& minLts) const { return minLts.version == mVersion.version && minLts.majorRev == mVersion.majorRev && minLts.minorRev <= mVersion.minorRev; } bool KernelInfo::matchKernelRequirements(const std::vector& kernels, std::string* error) const { bool foundMatchedKernelVersion = false; bool foundMatchedConditions = false; for (const MatrixKernel& matrixKernel : kernels) { if (!matchKernelVersion(matrixKernel.minLts())) { continue; } foundMatchedKernelVersion = true; // ignore this fragment if not all conditions are met. if (!matchKernelConfigs(matrixKernel.conditions(), error)) { continue; } foundMatchedConditions = true; if (!matchKernelConfigs(matrixKernel.configs(), error)) { return false; } } if (!foundMatchedKernelVersion) { if (error != nullptr) { std::stringstream ss; ss << "Framework is incompatible with kernel version " << version() << ", compatible kernel versions are"; for (const MatrixKernel& matrixKernel : kernels) ss << " " << matrixKernel.minLts(); *error = ss.str(); } return false; } if (!foundMatchedConditions) { // This should not happen because first for each must be // empty. Reject here for inconsistency. if (error != nullptr) { error->insert(0, "Framework match kernel version with unmet conditions:"); } return false; } if (error != nullptr) { error->clear(); } return true; } bool KernelInfo::operator==(const KernelInfo& other) const { return mVersion == other.mVersion && mConfigs == other.mConfigs; } } // namespace vintf } // namespace android