• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2019 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 #include "KernelInfo.h"
17 
18 #include "parse_string.h"
19 #include "parse_xml.h"
20 #include "parse_xml_internal.h"
21 #include "utils.h"
22 
23 namespace android {
24 namespace vintf {
25 
26 using details::mergeField;
27 
version() const28 const KernelVersion& KernelInfo::version() const {
29     return mVersion;
30 }
31 
configs() const32 const std::map<std::string, std::string>& KernelInfo::configs() const {
33     return mConfigs;
34 }
35 
level() const36 Level KernelInfo::level() const {
37     return mLevel;
38 }
39 
matchKernelConfigs(const std::vector<KernelConfig> & matrixConfigs,std::string * error) const40 bool KernelInfo::matchKernelConfigs(const std::vector<KernelConfig>& matrixConfigs,
41                                     std::string* error) const {
42     for (const KernelConfig& matrixConfig : matrixConfigs) {
43         const std::string& key = matrixConfig.first;
44         auto it = this->mConfigs.find(key);
45         if (it == this->mConfigs.end()) {
46             // special case: <value type="tristate">n</value> matches if the config doesn't exist.
47             if (matrixConfig.second == KernelConfigTypedValue::gMissingConfig) {
48                 continue;
49             }
50             if (error != nullptr) {
51                 *error = "Missing config " + key;
52             }
53             return false;
54         }
55         const std::string& kernelValue = it->second;
56         if (!matrixConfig.second.matchValue(kernelValue)) {
57             if (error != nullptr) {
58                 *error = "For config " + key + ", value = " + kernelValue + " but required " +
59                          to_string(matrixConfig.second);
60             }
61             return false;
62         }
63     }
64     return true;
65 }
66 
matchKernelVersion(const KernelVersion & minLts) const67 bool KernelInfo::matchKernelVersion(const KernelVersion& minLts) const {
68     return mVersion.dropMinor() == minLts.dropMinor() && minLts.minorRev <= mVersion.minorRev;
69 }
70 
getMatchedKernelRequirements(const std::vector<MatrixKernel> & kernels,Level kernelLevel,std::string * error) const71 std::vector<const MatrixKernel*> KernelInfo::getMatchedKernelRequirements(
72     const std::vector<MatrixKernel>& kernels, Level kernelLevel, std::string* error) const {
73     std::map<Level, std::vector<const MatrixKernel*>> kernelsForLevel;
74     for (const MatrixKernel& matrixKernel : kernels) {
75         const auto& minLts = matrixKernel.minLts();
76         auto matrixKernelLevel = matrixKernel.getSourceMatrixLevel();
77 
78         // Filter out kernels with different x.y.
79         if (mVersion.dropMinor() != minLts.dropMinor()) {
80             continue;
81         }
82 
83         // Check matrix kernel level
84 
85         // Use legacy behavior when kernel FCM version is not specified. Blindly add all of them
86         // here. The correct one (with smallest matrixKernelLevel) will be picked later.
87         if (kernelLevel == Level::UNSPECIFIED) {
88             kernelsForLevel[matrixKernelLevel].push_back(&matrixKernel);
89             continue;
90         }
91 
92         if (matrixKernelLevel == Level::UNSPECIFIED) {
93             if (error) {
94                 *error = "Seen unspecified source matrix level; this should not happen.";
95             }
96             return {};
97         }
98 
99         if (matrixKernelLevel < kernelLevel) {
100             continue;
101         }
102 
103         // matrix level >= kernel level
104 
105         // for kernel level >= S, do not allow matrix level > kernel level; i.e. only check
106         // matching KMI.
107         if (kernelLevel >= Level::S && matrixKernelLevel > kernelLevel) {
108             continue;
109         }
110 
111         kernelsForLevel[matrixKernelLevel].push_back(&matrixKernel);
112     }
113 
114     if (kernelsForLevel.empty()) {
115         if (error) {
116             std::stringstream ss;
117             ss << "No kernel entry found for kernel version " << mVersion.dropMinor()
118                << " at kernel FCM version "
119                << (kernelLevel == Level::UNSPECIFIED ? "unspecified" : to_string(kernelLevel))
120                << ". The following kernel requirements are checked:";
121             for (const MatrixKernel& matrixKernel : kernels) {
122                 ss << "\n  Minimum LTS: " << matrixKernel.minLts()
123                    << ", kernel FCM version: " << matrixKernel.getSourceMatrixLevel()
124                    << (matrixKernel.conditions().empty() ? "" : ", with conditionals");
125             };
126             *error = ss.str();
127         }
128         return {};
129     }
130 
131     // At this point, kernelsForLevel contains kernel requirements for each level.
132     // For example, if the running kernel version is 4.14.y then kernelsForLevel contains
133     // 4.14-p, 4.14-q, 4.14-r.
134     // (This excludes kernels < kernel FCM version, or device FCM version if kernel FCM version is
135     // empty. For example, if device level = Q and kernel level is unspecified, this list only
136     // contains 4.14-q and 4.14-r).
137 
138     // Use legacy behavior when kernel FCM version is not specified. e.g. target FCM version 3 (P)
139     // matches kernel 4.4-p, 4.9-p, 4.14-p, 4.19-q, etc., but not 4.9-q or 4.14-q.
140     // Since we already filtered |kernels| based on kernel version, we only need to check the first
141     // item in kernelsForLevel.
142     // Note that this excludes *-r and above kernels. Devices with target FCM version >= 5 (R) must
143     // state kernel FCM version explicitly in the device manifest. The value is automatically
144     // inserted for devices with target FCM version >= 5 when manifest is built with assemble_vintf.
145     if (kernelLevel == Level::UNSPECIFIED) {
146         auto [matrixKernelLevel, matrixKernels] = *kernelsForLevel.begin();
147 
148         // Do not allow *-r and above kernels.
149         if (matrixKernelLevel != Level::UNSPECIFIED && matrixKernelLevel >= Level::R) {
150             if (error) {
151                 KernelInfo msg;
152                 msg.mLevel = Level::R;
153                 *error = "Kernel FCM version is not specified, but kernel version " +
154                          to_string(mVersion) +
155                          " is found. Fix by specifying kernel FCM version in device manifest. "
156                          "For example, for a *-r kernel:\n" +
157                          toXml(msg);
158             }
159             return {};
160         }
161 
162         auto matchedMatrixKernels = getMatchedKernelVersionAndConfigs(matrixKernels, error);
163         if (matchedMatrixKernels.empty()) {
164             return {};
165         }
166         return matchedMatrixKernels;
167     }
168 
169     // Use new behavior when kernel FCM version is specified. e.g. kernel FCM version 3 (P)
170     // matches kernel 4.4-p, 4.9-p, 4.14-p, 4.9-q, 4.14-q, 4.14-r etc., but not 5.4-r.
171     // For kernel FCM version >= S, only matching KMI is accepted. e.g. kernel FCM version 6 (S)
172     // matches 4.19-stable, 5.10-android12, 5.4-android12, not x.y-android13.
173     // Note we already filtered |kernels| based on kernel version.
174     auto [firstMatrixKernelLevel, firstMatrixKernels] = *kernelsForLevel.begin();
175     if (firstMatrixKernelLevel == Level::UNSPECIFIED || firstMatrixKernelLevel > kernelLevel) {
176         if (error) {
177             *error = "Kernel FCM Version is " + to_string(kernelLevel) + " and kernel version is " +
178                      to_string(mVersion) +
179                      ", but the first kernel FCM version allowed for kernel version " +
180                      to_string(mVersion.dropMinor()) + ".y is " + to_string(firstMatrixKernelLevel);
181         }
182         return {};
183     }
184     for (auto [matrixKernelLevel, matrixKernels] : kernelsForLevel) {
185         if (matrixKernelLevel == Level::UNSPECIFIED || matrixKernelLevel < kernelLevel) {
186             continue;
187         }
188         std::string errorForLevel;
189         auto matchedMatrixKernels =
190             getMatchedKernelVersionAndConfigs(matrixKernels, &errorForLevel);
191         if (matchedMatrixKernels.empty()) {
192             if (error) {
193                 *error += "For kernel requirements at matrix level " +
194                           to_string(matrixKernelLevel) + ", " + errorForLevel + "\n";
195             }
196             continue;
197         }
198         return matchedMatrixKernels;
199     }
200 
201     if (error) {
202         error->insert(0, "No compatible kernel requirement found (kernel FCM version = " +
203                              to_string(kernelLevel) + ").\n");
204     }
205     return {};
206 }
207 
getMatchedKernelVersionAndConfigs(const std::vector<const MatrixKernel * > & kernels,std::string * error) const208 std::vector<const MatrixKernel*> KernelInfo::getMatchedKernelVersionAndConfigs(
209     const std::vector<const MatrixKernel*>& kernels, std::string* error) const {
210     std::vector<const MatrixKernel*> result;
211     bool foundMatchedKernelVersion = false;
212     for (const MatrixKernel* matrixKernel : kernels) {
213         if (!matchKernelVersion(matrixKernel->minLts())) {
214             continue;
215         }
216         foundMatchedKernelVersion = true;
217         // ignore this fragment if not all conditions are met.
218         if (!matchKernelConfigs(matrixKernel->conditions(), error)) {
219             continue;
220         }
221         if (!matchKernelConfigs(matrixKernel->configs(), error)) {
222             return {};
223         }
224         result.push_back(matrixKernel);
225     }
226     if (!foundMatchedKernelVersion) {
227         if (error != nullptr) {
228             std::stringstream ss;
229             ss << "Framework is incompatible with kernel version " << version()
230                << ", compatible kernel versions are:";
231             for (const MatrixKernel* matrixKernel : kernels) {
232                 ss << "\n  Minimum LTS: " << matrixKernel->minLts()
233                    << ", kernel FCM version: " << matrixKernel->getSourceMatrixLevel()
234                    << (matrixKernel->conditions().empty() ? "" : ", with conditionals");
235             };
236             *error = ss.str();
237         }
238         return {};
239     }
240     if (result.empty()) {
241         // This means matchKernelVersion passes but all matchKernelConfigs(conditions) fails.
242         // This should not happen because first <conditions> for each <kernel> must be
243         // empty. Reject here for inconsistency.
244         if (error != nullptr) {
245             error->insert(0, "Framework matches kernel version with unmet conditions.");
246         }
247         return {};
248     }
249     if (error != nullptr) {
250         error->clear();
251     }
252     return result;
253 }
254 
operator ==(const KernelInfo & other) const255 bool KernelInfo::operator==(const KernelInfo& other) const {
256     return mVersion == other.mVersion && mConfigs == other.mConfigs;
257 }
258 
merge(KernelInfo * other,std::string * error)259 bool KernelInfo::merge(KernelInfo* other, std::string* error) {
260     if (!mergeField(&mVersion, &other->mVersion)) {
261         if (error) {
262             *error = "Conflicting kernel version: " + to_string(version()) + " vs. " +
263                      to_string(other->version());
264         }
265         return false;
266     }
267 
268     // Do not allow merging configs. One of them must be empty.
269     if (!mergeField(&mConfigs, &other->mConfigs)) {
270         if (error) {
271             *error = "Found <kernel><config> items in two manifests.";
272         }
273         return false;
274     }
275 
276     if (!mergeField(&mLevel, &other->mLevel, Level::UNSPECIFIED)) {
277         if (error) {
278             *error = "Conflicting kernel level: " + to_string(level()) + " vs. " +
279                      to_string(other->level());
280         }
281         return false;
282     }
283     return true;
284 }
285 
286 }  // namespace vintf
287 }  // namespace android
288