1*70a7ec85SAndroid Build Coastguard Worker /*
2*70a7ec85SAndroid Build Coastguard Worker * Copyright (C) 2017 The Android Open Source Project
3*70a7ec85SAndroid Build Coastguard Worker *
4*70a7ec85SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*70a7ec85SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*70a7ec85SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*70a7ec85SAndroid Build Coastguard Worker *
8*70a7ec85SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*70a7ec85SAndroid Build Coastguard Worker *
10*70a7ec85SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*70a7ec85SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*70a7ec85SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*70a7ec85SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*70a7ec85SAndroid Build Coastguard Worker * limitations under the License.
15*70a7ec85SAndroid Build Coastguard Worker */
16*70a7ec85SAndroid Build Coastguard Worker
17*70a7ec85SAndroid Build Coastguard Worker
18*70a7ec85SAndroid Build Coastguard Worker #define LOG_TAG "libvintf"
19*70a7ec85SAndroid Build Coastguard Worker #include <android-base/logging.h>
20*70a7ec85SAndroid Build Coastguard Worker
21*70a7ec85SAndroid Build Coastguard Worker #include "RuntimeInfo.h"
22*70a7ec85SAndroid Build Coastguard Worker
23*70a7ec85SAndroid Build Coastguard Worker #include "CompatibilityMatrix.h"
24*70a7ec85SAndroid Build Coastguard Worker #include "parse_string.h"
25*70a7ec85SAndroid Build Coastguard Worker
26*70a7ec85SAndroid Build Coastguard Worker #include <dirent.h>
27*70a7ec85SAndroid Build Coastguard Worker #include <errno.h>
28*70a7ec85SAndroid Build Coastguard Worker #include <sys/utsname.h>
29*70a7ec85SAndroid Build Coastguard Worker #include <unistd.h>
30*70a7ec85SAndroid Build Coastguard Worker
31*70a7ec85SAndroid Build Coastguard Worker #include <fstream>
32*70a7ec85SAndroid Build Coastguard Worker #include <iostream>
33*70a7ec85SAndroid Build Coastguard Worker #include <sstream>
34*70a7ec85SAndroid Build Coastguard Worker
35*70a7ec85SAndroid Build Coastguard Worker #include <android-base/properties.h>
36*70a7ec85SAndroid Build Coastguard Worker #include <selinux/selinux.h>
37*70a7ec85SAndroid Build Coastguard Worker
38*70a7ec85SAndroid Build Coastguard Worker #include "KernelConfigs.h"
39*70a7ec85SAndroid Build Coastguard Worker
40*70a7ec85SAndroid Build Coastguard Worker namespace android {
41*70a7ec85SAndroid Build Coastguard Worker namespace vintf {
42*70a7ec85SAndroid Build Coastguard Worker
43*70a7ec85SAndroid Build Coastguard Worker struct RuntimeInfoFetcher {
RuntimeInfoFetcherandroid::vintf::RuntimeInfoFetcher44*70a7ec85SAndroid Build Coastguard Worker RuntimeInfoFetcher(RuntimeInfo *ki) : mRuntimeInfo(ki) { }
45*70a7ec85SAndroid Build Coastguard Worker status_t fetchAllInformation(RuntimeInfo::FetchFlags flags);
46*70a7ec85SAndroid Build Coastguard Worker
47*70a7ec85SAndroid Build Coastguard Worker private:
48*70a7ec85SAndroid Build Coastguard Worker status_t fetchVersion(RuntimeInfo::FetchFlags flags);
49*70a7ec85SAndroid Build Coastguard Worker status_t fetchKernelConfigs(RuntimeInfo::FetchFlags flags);
50*70a7ec85SAndroid Build Coastguard Worker status_t fetchCpuInfo(RuntimeInfo::FetchFlags flags);
51*70a7ec85SAndroid Build Coastguard Worker status_t fetchKernelSepolicyVers(RuntimeInfo::FetchFlags flags);
52*70a7ec85SAndroid Build Coastguard Worker status_t fetchAvb(RuntimeInfo::FetchFlags flags);
53*70a7ec85SAndroid Build Coastguard Worker status_t parseKernelVersion();
54*70a7ec85SAndroid Build Coastguard Worker RuntimeInfo* mRuntimeInfo;
55*70a7ec85SAndroid Build Coastguard Worker };
56*70a7ec85SAndroid Build Coastguard Worker
57*70a7ec85SAndroid Build Coastguard Worker // decompress /proc/config.gz and read its contents.
fetchKernelConfigs(RuntimeInfo::FetchFlags)58*70a7ec85SAndroid Build Coastguard Worker status_t RuntimeInfoFetcher::fetchKernelConfigs(RuntimeInfo::FetchFlags) {
59*70a7ec85SAndroid Build Coastguard Worker return kernelconfigs::LoadKernelConfigs(&mRuntimeInfo->mKernel.mConfigs);
60*70a7ec85SAndroid Build Coastguard Worker }
61*70a7ec85SAndroid Build Coastguard Worker
fetchCpuInfo(RuntimeInfo::FetchFlags)62*70a7ec85SAndroid Build Coastguard Worker status_t RuntimeInfoFetcher::fetchCpuInfo(RuntimeInfo::FetchFlags) {
63*70a7ec85SAndroid Build Coastguard Worker // TODO implement this; 32-bit and 64-bit has different format.
64*70a7ec85SAndroid Build Coastguard Worker std::ifstream in{"/proc/cpuinfo"};
65*70a7ec85SAndroid Build Coastguard Worker if (!in.is_open()) {
66*70a7ec85SAndroid Build Coastguard Worker LOG(WARNING) << "Cannot read /proc/cpuinfo";
67*70a7ec85SAndroid Build Coastguard Worker return UNKNOWN_ERROR;
68*70a7ec85SAndroid Build Coastguard Worker }
69*70a7ec85SAndroid Build Coastguard Worker std::stringstream sstream;
70*70a7ec85SAndroid Build Coastguard Worker sstream << in.rdbuf();
71*70a7ec85SAndroid Build Coastguard Worker mRuntimeInfo->mCpuInfo = sstream.str();
72*70a7ec85SAndroid Build Coastguard Worker return OK;
73*70a7ec85SAndroid Build Coastguard Worker }
74*70a7ec85SAndroid Build Coastguard Worker
fetchKernelSepolicyVers(RuntimeInfo::FetchFlags)75*70a7ec85SAndroid Build Coastguard Worker status_t RuntimeInfoFetcher::fetchKernelSepolicyVers(RuntimeInfo::FetchFlags) {
76*70a7ec85SAndroid Build Coastguard Worker int pv;
77*70a7ec85SAndroid Build Coastguard Worker #ifdef LIBVINTF_TARGET
78*70a7ec85SAndroid Build Coastguard Worker pv = security_policyvers();
79*70a7ec85SAndroid Build Coastguard Worker #else
80*70a7ec85SAndroid Build Coastguard Worker pv = 0;
81*70a7ec85SAndroid Build Coastguard Worker #endif
82*70a7ec85SAndroid Build Coastguard Worker if (pv < 0) {
83*70a7ec85SAndroid Build Coastguard Worker return pv;
84*70a7ec85SAndroid Build Coastguard Worker }
85*70a7ec85SAndroid Build Coastguard Worker mRuntimeInfo->mKernelSepolicyVersion = pv;
86*70a7ec85SAndroid Build Coastguard Worker return OK;
87*70a7ec85SAndroid Build Coastguard Worker }
88*70a7ec85SAndroid Build Coastguard Worker
fetchVersion(RuntimeInfo::FetchFlags flags)89*70a7ec85SAndroid Build Coastguard Worker status_t RuntimeInfoFetcher::fetchVersion(RuntimeInfo::FetchFlags flags) {
90*70a7ec85SAndroid Build Coastguard Worker struct utsname buf;
91*70a7ec85SAndroid Build Coastguard Worker if (uname(&buf)) {
92*70a7ec85SAndroid Build Coastguard Worker return -errno;
93*70a7ec85SAndroid Build Coastguard Worker }
94*70a7ec85SAndroid Build Coastguard Worker mRuntimeInfo->mOsName = buf.sysname;
95*70a7ec85SAndroid Build Coastguard Worker mRuntimeInfo->mNodeName = buf.nodename;
96*70a7ec85SAndroid Build Coastguard Worker mRuntimeInfo->mOsRelease = buf.release;
97*70a7ec85SAndroid Build Coastguard Worker mRuntimeInfo->mOsVersion = buf.version;
98*70a7ec85SAndroid Build Coastguard Worker mRuntimeInfo->mHardwareId = buf.machine;
99*70a7ec85SAndroid Build Coastguard Worker
100*70a7ec85SAndroid Build Coastguard Worker mRuntimeInfo->mIsMainline = RuntimeInfo::kernelReleaseIsMainline(mRuntimeInfo->mOsRelease);
101*70a7ec85SAndroid Build Coastguard Worker
102*70a7ec85SAndroid Build Coastguard Worker status_t err = RuntimeInfo::parseGkiKernelRelease(flags, mRuntimeInfo->mOsRelease,
103*70a7ec85SAndroid Build Coastguard Worker &mRuntimeInfo->mKernel.mVersion,
104*70a7ec85SAndroid Build Coastguard Worker &mRuntimeInfo->mKernel.mLevel);
105*70a7ec85SAndroid Build Coastguard Worker if (err == OK) return OK;
106*70a7ec85SAndroid Build Coastguard Worker
107*70a7ec85SAndroid Build Coastguard Worker err = parseKernelVersion();
108*70a7ec85SAndroid Build Coastguard Worker if (err != OK) {
109*70a7ec85SAndroid Build Coastguard Worker LOG(ERROR) << "Could not parse kernel version from \""
110*70a7ec85SAndroid Build Coastguard Worker << mRuntimeInfo->mOsRelease << "\"";
111*70a7ec85SAndroid Build Coastguard Worker }
112*70a7ec85SAndroid Build Coastguard Worker return err;
113*70a7ec85SAndroid Build Coastguard Worker }
114*70a7ec85SAndroid Build Coastguard Worker
parseKernelVersion()115*70a7ec85SAndroid Build Coastguard Worker status_t RuntimeInfoFetcher::parseKernelVersion() {
116*70a7ec85SAndroid Build Coastguard Worker auto pos = mRuntimeInfo->mOsRelease.find('.');
117*70a7ec85SAndroid Build Coastguard Worker if (pos == std::string::npos) {
118*70a7ec85SAndroid Build Coastguard Worker return UNKNOWN_ERROR;
119*70a7ec85SAndroid Build Coastguard Worker }
120*70a7ec85SAndroid Build Coastguard Worker pos = mRuntimeInfo->mOsRelease.find('.', pos + 1);
121*70a7ec85SAndroid Build Coastguard Worker if (pos == std::string::npos) {
122*70a7ec85SAndroid Build Coastguard Worker return UNKNOWN_ERROR;
123*70a7ec85SAndroid Build Coastguard Worker }
124*70a7ec85SAndroid Build Coastguard Worker pos = mRuntimeInfo->mOsRelease.find_first_not_of("0123456789", pos + 1);
125*70a7ec85SAndroid Build Coastguard Worker // no need to check pos == std::string::npos, because substr will handle this
126*70a7ec85SAndroid Build Coastguard Worker if (!parse(mRuntimeInfo->mOsRelease.substr(0, pos), &mRuntimeInfo->mKernel.mVersion)) {
127*70a7ec85SAndroid Build Coastguard Worker return UNKNOWN_ERROR;
128*70a7ec85SAndroid Build Coastguard Worker }
129*70a7ec85SAndroid Build Coastguard Worker return OK;
130*70a7ec85SAndroid Build Coastguard Worker }
131*70a7ec85SAndroid Build Coastguard Worker
fetchAvb(RuntimeInfo::FetchFlags)132*70a7ec85SAndroid Build Coastguard Worker status_t RuntimeInfoFetcher::fetchAvb(RuntimeInfo::FetchFlags) {
133*70a7ec85SAndroid Build Coastguard Worker std::string prop = android::base::GetProperty("ro.boot.vbmeta.avb_version", "0.0");
134*70a7ec85SAndroid Build Coastguard Worker if (!parse(prop, &mRuntimeInfo->mBootVbmetaAvbVersion)) {
135*70a7ec85SAndroid Build Coastguard Worker return UNKNOWN_ERROR;
136*70a7ec85SAndroid Build Coastguard Worker }
137*70a7ec85SAndroid Build Coastguard Worker prop = android::base::GetProperty("ro.boot.avb_version", "0.0");
138*70a7ec85SAndroid Build Coastguard Worker if (!parse(prop, &mRuntimeInfo->mBootAvbVersion)) {
139*70a7ec85SAndroid Build Coastguard Worker return UNKNOWN_ERROR;
140*70a7ec85SAndroid Build Coastguard Worker }
141*70a7ec85SAndroid Build Coastguard Worker return OK;
142*70a7ec85SAndroid Build Coastguard Worker }
143*70a7ec85SAndroid Build Coastguard Worker
144*70a7ec85SAndroid Build Coastguard Worker struct FetchFunction {
145*70a7ec85SAndroid Build Coastguard Worker RuntimeInfo::FetchFlags flags;
146*70a7ec85SAndroid Build Coastguard Worker std::function<status_t(RuntimeInfoFetcher*, RuntimeInfo::FetchFlags)> fetch;
147*70a7ec85SAndroid Build Coastguard Worker std::string description;
148*70a7ec85SAndroid Build Coastguard Worker };
149*70a7ec85SAndroid Build Coastguard Worker
fetchAllInformation(RuntimeInfo::FetchFlags flags)150*70a7ec85SAndroid Build Coastguard Worker status_t RuntimeInfoFetcher::fetchAllInformation(RuntimeInfo::FetchFlags flags) {
151*70a7ec85SAndroid Build Coastguard Worker using F = RuntimeInfo::FetchFlag;
152*70a7ec85SAndroid Build Coastguard Worker using RF = RuntimeInfoFetcher;
153*70a7ec85SAndroid Build Coastguard Worker // clang-format off
154*70a7ec85SAndroid Build Coastguard Worker const static std::vector<FetchFunction> gFetchFunctions({
155*70a7ec85SAndroid Build Coastguard Worker {F::CPU_VERSION | F::KERNEL_FCM, &RF::fetchVersion, "/proc/version"},
156*70a7ec85SAndroid Build Coastguard Worker {F::CONFIG_GZ, &RF::fetchKernelConfigs, "/proc/config.gz"},
157*70a7ec85SAndroid Build Coastguard Worker {F::CPU_INFO, &RF::fetchCpuInfo, "/proc/cpuinfo"},
158*70a7ec85SAndroid Build Coastguard Worker {F::POLICYVERS, &RF::fetchKernelSepolicyVers, "kernel sepolicy version"},
159*70a7ec85SAndroid Build Coastguard Worker {F::AVB, &RF::fetchAvb, "avb version"},
160*70a7ec85SAndroid Build Coastguard Worker });
161*70a7ec85SAndroid Build Coastguard Worker // clang-format on
162*70a7ec85SAndroid Build Coastguard Worker
163*70a7ec85SAndroid Build Coastguard Worker status_t err;
164*70a7ec85SAndroid Build Coastguard Worker for (const auto& fetchFunction : gFetchFunctions)
165*70a7ec85SAndroid Build Coastguard Worker if ((flags & fetchFunction.flags) && ((err = fetchFunction.fetch(this, flags)) != OK))
166*70a7ec85SAndroid Build Coastguard Worker LOG(WARNING) << "Cannot fetch or parse " << fetchFunction.description << ": "
167*70a7ec85SAndroid Build Coastguard Worker << strerror(-err);
168*70a7ec85SAndroid Build Coastguard Worker
169*70a7ec85SAndroid Build Coastguard Worker return OK;
170*70a7ec85SAndroid Build Coastguard Worker }
171*70a7ec85SAndroid Build Coastguard Worker
fetchAllInformation(RuntimeInfo::FetchFlags flags)172*70a7ec85SAndroid Build Coastguard Worker status_t RuntimeInfo::fetchAllInformation(RuntimeInfo::FetchFlags flags) {
173*70a7ec85SAndroid Build Coastguard Worker return RuntimeInfoFetcher(this).fetchAllInformation(flags);
174*70a7ec85SAndroid Build Coastguard Worker }
175*70a7ec85SAndroid Build Coastguard Worker
176*70a7ec85SAndroid Build Coastguard Worker } // namespace vintf
177*70a7ec85SAndroid Build Coastguard Worker } // namespace android
178