/* * 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 #include #include #include #include #include #include "utility/ValidateXml.h" using std::string_literals::operator""s; static void get_files_in_dirs(const char* dir_path, std::vector& files) { DIR* d; struct dirent* de; d = opendir(dir_path); if (d == nullptr) { return; } while ((de = readdir(d))) { if (de->d_type != DT_REG) { continue; } files.push_back(de->d_name); } closedir(d); } static std::vector glob(const std::string& pattern) { glob_t glob_result; auto ret = glob(pattern.c_str(), GLOB_MARK, nullptr, &glob_result); auto guard = android::base::make_scope_guard([&glob_result] { globfree(&glob_result); }); std::vector files; if (ret == 0) { for (size_t i = 0; i < glob_result.gl_pathc; i++) { files.emplace_back(glob_result.gl_pathv[i]); } } return files; } TEST(CheckConfig, halManifestValidation) { if (android::base::GetIntProperty("ro.product.first_api_level", INT64_MAX) <= 28) { GTEST_SKIP(); } RecordProperty("description", "Verify that the hal manifest file " "is valid according to the schema"); constexpr const char* xsd = "/data/local/tmp/hal_manifest.xsd"; // There may be compatibility matrices in .../etc/vintf. Manifests are only loaded from // manifest.xml and manifest_*.xml, so only check those. std::vector vintf_locations = {"/vendor/etc/vintf", "/odm/etc/vintf"}; for (const char* dir_path : vintf_locations) { std::vector files; get_files_in_dirs(dir_path, files); for (std::string file_name : files) { if (android::base::StartsWith(file_name, "manifest")) { EXPECT_VALID_XML((dir_path + "/"s + file_name).c_str(), xsd); } } } // .../etc/vintf/manifest should only contain manifest fragments, so all of them must match the // schema. std::vector fragment_locations = {"/vendor/etc/vintf/manifest", "/odm/etc/vintf/manifest"}; for (const char* dir_path : fragment_locations) { std::vector files; get_files_in_dirs(dir_path, files); for (std::string file_name : files) { EXPECT_VALID_XML((dir_path + "/"s + file_name).c_str(), xsd); } } // APEXes contain fragments as well. auto fragments = glob("/apex/*/etc/vintf/*.xml"); for (const auto& fragment : fragments) { // Skip /apex/name@version paths to avoid double processing auto parts = android::base::Split(fragment, "/"); if (parts.size() < 3 || parts[2].find('@') != std::string::npos) continue; EXPECT_VALID_XML(fragment.c_str(), xsd); } }