261 lines
10 KiB
C++
261 lines
10 KiB
C++
/*
|
|
* Copyright (C) 2017 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 <inttypes.h>
|
|
|
|
#include <string>
|
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <android-base/file.h>
|
|
#include <android-base/stringprintf.h>
|
|
#include <android-base/strings.h>
|
|
|
|
#include <private/android_filesystem_config.h>
|
|
|
|
#include "fs_config.h"
|
|
|
|
extern const fs_path_config* __for_testing_only__android_dirs;
|
|
extern const fs_path_config* __for_testing_only__android_files;
|
|
extern bool (*__for_testing_only__fs_config_cmp)(bool, const char*, size_t, const char*, size_t);
|
|
|
|
// Maximum entries in system/core/libcutils/fs_config.cpp:android_* before we
|
|
// hit a nullptr termination, before we declare the list is just too big or
|
|
// could be missing the nullptr.
|
|
static constexpr size_t max_idx = 4096;
|
|
|
|
static const struct fs_config_cmp_test {
|
|
bool dir;
|
|
const char* prefix;
|
|
const char* path;
|
|
bool match;
|
|
} fs_config_cmp_tests[] = {
|
|
// clang-format off
|
|
{ true, "system/lib", "system/lib/hw", true },
|
|
{ true, "vendor/lib", "system/vendor/lib/hw", true },
|
|
{ true, "system/vendor/lib", "vendor/lib/hw", false },
|
|
{ true, "system/vendor/lib", "system/vendor/lib/hw", true },
|
|
{ true, "foo/*/bar/*", "foo/1/bar/2", true },
|
|
{ true, "foo/*/bar/*", "foo/1/bar", true },
|
|
{ true, "foo/*/bar/*", "foo/1/bar/2/3", true },
|
|
{ true, "foo/*/bar/*", "foo/1/bar/2/3/", true },
|
|
{ false, "vendor/bin/wifi", "system/vendor/bin/w", false },
|
|
{ false, "vendor/bin/wifi", "system/vendor/bin/wifi", true },
|
|
{ false, "vendor/bin/wifi", "system/vendor/bin/wifi2", false },
|
|
{ false, "system/vendor/bin/wifi", "system/vendor/bin/wifi", true, },
|
|
{ false, "odm/bin/wifi", "system/odm/bin/wifi", false },
|
|
{ false, "odm/bin/wifi", "vendor/odm/bin/wifi", true },
|
|
{ false, "oem/bin/wifi", "system/oem/bin/wifi", false },
|
|
{ false, "data/bin/wifi", "system/data/bin/wifi", false },
|
|
{ false, "system/bin/*", "system/bin/wifi", true },
|
|
{ false, "vendor/bin/*", "system/vendor/bin/wifi", true },
|
|
{ false, "system/bin/*", "system/bin", false },
|
|
{ false, "system/vendor/bin/*", "vendor/bin/wifi", false },
|
|
{ false, "foo/*/bar/*", "foo/1/bar/2", true },
|
|
{ false, "foo/*/bar/*", "foo/1/bar", false },
|
|
{ false, "foo/*/bar/*", "foo/1/bar/2/3", true },
|
|
{ false, "foo/*/bar/*.so", "foo/1/bar/2/3", false },
|
|
{ false, "foo/*/bar/*.so", "foo/1/bar/2.so", true },
|
|
{ false, "foo/*/bar/*.so", "foo/1/bar/2/3.so", true },
|
|
{ false, NULL, NULL, false },
|
|
// clang-format on
|
|
};
|
|
|
|
static bool check_unique(std::vector<const char*>& paths, const std::string& config_name,
|
|
const std::string& prefix) {
|
|
bool retval = false;
|
|
|
|
std::string alternate = "system/" + prefix;
|
|
|
|
for (size_t idx = 0; idx < paths.size(); ++idx) {
|
|
size_t second;
|
|
std::string path(paths[idx]);
|
|
// check if there are multiple identical paths
|
|
for (second = idx + 1; second < paths.size(); ++second) {
|
|
if (path == paths[second]) {
|
|
GTEST_LOG_(ERROR) << "duplicate paths in " << config_name << ": " << paths[idx];
|
|
retval = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// check if path is <partition>/
|
|
if (android::base::StartsWith(path, prefix)) {
|
|
// rebuild path to be system/<partition>/... to check for alias
|
|
path = alternate + path.substr(prefix.size());
|
|
for (second = 0; second < paths.size(); ++second) {
|
|
if (path == paths[second]) {
|
|
GTEST_LOG_(ERROR) << "duplicate alias paths in " << config_name << ": "
|
|
<< paths[idx] << " and " << paths[second]
|
|
<< " (remove latter)";
|
|
retval = true;
|
|
break;
|
|
}
|
|
}
|
|
continue;
|
|
}
|
|
|
|
// check if path is system/<partition>/
|
|
if (android::base::StartsWith(path, alternate)) {
|
|
// rebuild path to be <partition>/... to check for alias
|
|
path = prefix + path.substr(alternate.size());
|
|
for (second = 0; second < paths.size(); ++second) {
|
|
if (path == paths[second]) break;
|
|
}
|
|
if (second >= paths.size()) {
|
|
GTEST_LOG_(ERROR) << "replace path in " << config_name << ": " << paths[idx]
|
|
<< " with " << path;
|
|
retval = true;
|
|
}
|
|
}
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
static bool check_unique(const fs_path_config* paths, const char* type_name,
|
|
const std::string& prefix) {
|
|
std::string config("system/core/libcutils/fs_config.cpp:android_");
|
|
config += type_name;
|
|
config += "[]";
|
|
|
|
bool retval = false;
|
|
std::vector<const char*> paths_tmp;
|
|
for (size_t idx = 0; paths[idx].prefix; ++idx) {
|
|
if (idx > max_idx) {
|
|
GTEST_LOG_(WARNING) << config << ": has no end (missing null prefix)";
|
|
retval = true;
|
|
break;
|
|
}
|
|
paths_tmp.push_back(paths[idx].prefix);
|
|
}
|
|
|
|
return check_unique(paths_tmp, config, prefix) || retval;
|
|
}
|
|
|
|
static bool check_fs_config_cmp(const fs_config_cmp_test* tests) {
|
|
bool match, retval = false;
|
|
for (size_t idx = 0; tests[idx].prefix; ++idx) {
|
|
match = __for_testing_only__fs_config_cmp(tests[idx].dir, tests[idx].prefix,
|
|
strlen(tests[idx].prefix), tests[idx].path,
|
|
strlen(tests[idx].path));
|
|
if (match != tests[idx].match) {
|
|
GTEST_LOG_(ERROR) << tests[idx].path << (match ? " matched " : " didn't match ")
|
|
<< tests[idx].prefix;
|
|
retval = true;
|
|
break;
|
|
}
|
|
}
|
|
return retval;
|
|
}
|
|
|
|
#define endof(pointer, field) (offsetof(typeof(*(pointer)), field) + sizeof((pointer)->field))
|
|
|
|
static bool check_unique(const std::string& config, const std::string& prefix) {
|
|
int retval = false;
|
|
|
|
std::string data;
|
|
if (!android::base::ReadFileToString(config, &data)) return retval;
|
|
|
|
const fs_path_config_from_file* pc =
|
|
reinterpret_cast<const fs_path_config_from_file*>(data.c_str());
|
|
size_t len = data.size();
|
|
|
|
std::vector<const char*> paths_tmp;
|
|
size_t entry_number = 0;
|
|
while (len > 0) {
|
|
uint16_t host_len = (len >= endof(pc, len)) ? pc->len : INT16_MAX;
|
|
if (host_len > len) {
|
|
GTEST_LOG_(WARNING) << config << ": truncated at entry " << entry_number << " ("
|
|
<< host_len << " > " << len << ")";
|
|
const std::string unknown("?");
|
|
GTEST_LOG_(WARNING)
|
|
<< config << ": entry[" << entry_number << "]={ "
|
|
<< "len=" << ((len >= endof(pc, len))
|
|
? android::base::StringPrintf("%" PRIu16, pc->len)
|
|
: unknown)
|
|
<< ", mode=" << ((len >= endof(pc, mode))
|
|
? android::base::StringPrintf("0%" PRIo16, pc->mode)
|
|
: unknown)
|
|
<< ", uid=" << ((len >= endof(pc, uid))
|
|
? android::base::StringPrintf("%" PRIu16, pc->uid)
|
|
: unknown)
|
|
<< ", gid=" << ((len >= endof(pc, gid))
|
|
? android::base::StringPrintf("%" PRIu16, pc->gid)
|
|
: unknown)
|
|
<< ", capabilities="
|
|
<< ((len >= endof(pc, capabilities))
|
|
? android::base::StringPrintf("0x%" PRIx64, pc->capabilities)
|
|
: unknown)
|
|
<< ", prefix="
|
|
<< ((len >= offsetof(fs_path_config_from_file, prefix))
|
|
? android::base::StringPrintf(
|
|
"\"%.*s...", (int)(len - offsetof(fs_path_config_from_file, prefix)),
|
|
pc->prefix)
|
|
: unknown)
|
|
<< " }";
|
|
retval = true;
|
|
break;
|
|
}
|
|
paths_tmp.push_back(pc->prefix);
|
|
|
|
pc = reinterpret_cast<const fs_path_config_from_file*>(reinterpret_cast<const char*>(pc) +
|
|
host_len);
|
|
len -= host_len;
|
|
++entry_number;
|
|
}
|
|
|
|
return check_unique(paths_tmp, config, prefix) || retval;
|
|
}
|
|
|
|
void check_two(const fs_path_config* paths, const char* type_name, const char* prefix) {
|
|
ASSERT_FALSE(paths == nullptr);
|
|
ASSERT_FALSE(type_name == nullptr);
|
|
ASSERT_FALSE(prefix == nullptr);
|
|
bool check_internal = check_unique(paths, type_name, prefix);
|
|
EXPECT_FALSE(check_internal);
|
|
bool check_overrides =
|
|
check_unique(std::string("/") + prefix + "etc/fs_config_" + type_name, prefix);
|
|
EXPECT_FALSE(check_overrides);
|
|
}
|
|
|
|
TEST(fs_config, vendor_dirs_alias) {
|
|
check_two(__for_testing_only__android_dirs, "dirs", "vendor/");
|
|
}
|
|
|
|
TEST(fs_config, vendor_files_alias) {
|
|
check_two(__for_testing_only__android_files, "files", "vendor/");
|
|
}
|
|
|
|
TEST(fs_config, oem_dirs_alias) {
|
|
check_two(__for_testing_only__android_dirs, "dirs", "oem/");
|
|
}
|
|
|
|
TEST(fs_config, oem_files_alias) {
|
|
check_two(__for_testing_only__android_files, "files", "oem/");
|
|
}
|
|
|
|
TEST(fs_config, odm_dirs_alias) {
|
|
check_two(__for_testing_only__android_dirs, "dirs", "odm/");
|
|
}
|
|
|
|
TEST(fs_config, odm_files_alias) {
|
|
check_two(__for_testing_only__android_files, "files", "odm/");
|
|
}
|
|
|
|
TEST(fs_config, system_alias) {
|
|
EXPECT_FALSE(check_fs_config_cmp(fs_config_cmp_tests));
|
|
}
|