| // Copyright 2019 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "components/arc/session/file_system_status.h" |
| |
| #include <linux/magic.h> |
| #include <sys/statvfs.h> |
| |
| #include <string> |
| |
| #include "base/containers/span.h" |
| #include "base/files/file_util.h" |
| #include "base/json/json_reader.h" |
| #include "base/logging.h" |
| #include "base/threading/scoped_blocking_call.h" |
| #include "base/values.h" |
| #include "build/build_config.h" |
| |
| namespace arc { |
| namespace { |
| |
| constexpr const char kAdbdJson[] = "/etc/arc/adbd.json"; |
| constexpr const char kArcVmConfigJsonPath[] = "/usr/share/arcvm/config.json"; |
| constexpr const char kBuiltinPath[] = "/opt/google/vms/android"; |
| constexpr const char kFstabPath[] = "/run/arcvm/host_generated/fstab"; |
| constexpr const char kKernel[] = "vmlinux"; |
| constexpr const char kRootFs[] = "system.raw.img"; |
| constexpr const char kVendorImage[] = "vendor.raw.img"; |
| |
| } // namespace |
| |
| FileSystemStatus::FileSystemStatus(FileSystemStatus&& other) = default; |
| FileSystemStatus::~FileSystemStatus() = default; |
| FileSystemStatus& FileSystemStatus::operator=(FileSystemStatus&& other) = |
| default; |
| |
| FileSystemStatus::FileSystemStatus() |
| : is_android_debuggable_( |
| IsAndroidDebuggable(base::FilePath(kArcVmConfigJsonPath))), |
| is_host_rootfs_writable_(IsHostRootfsWritable()), |
| system_image_path_(base::FilePath(kBuiltinPath).Append(kRootFs)), |
| vendor_image_path_(base::FilePath(kBuiltinPath).Append(kVendorImage)), |
| guest_kernel_path_(base::FilePath(kBuiltinPath).Append(kKernel)), |
| fstab_path_(kFstabPath), |
| is_system_image_ext_format_(IsSystemImageExtFormat(system_image_path_)), |
| has_adbd_json_(base::PathExists(base::FilePath(kAdbdJson))) {} |
| |
| // static |
| bool FileSystemStatus::IsAndroidDebuggable(const base::FilePath& json_path) { |
| if (!base::PathExists(json_path)) |
| return false; |
| |
| std::string content; |
| if (!base::ReadFileToString(json_path, &content)) |
| return false; |
| |
| base::JSONReader::ValueWithError result( |
| base::JSONReader::ReadAndReturnValueWithError(content, |
| base::JSON_PARSE_RFC)); |
| if (!result.value) { |
| LOG(ERROR) << "Error parsing " << json_path |
| << ", message=" << result.error_message << ": " << content; |
| return false; |
| } |
| if (!result.value->is_dict()) { |
| LOG(ERROR) << "Error parsing " << json_path << ": " << *(result.value); |
| return false; |
| } |
| |
| const base::Value* debuggable = result.value->FindKeyOfType( |
| "ANDROID_DEBUGGABLE", base::Value::Type::BOOLEAN); |
| if (!debuggable) { |
| LOG(ERROR) << "ANDROID_DEBUGGABLE is not found in " << json_path; |
| return false; |
| } |
| |
| return debuggable->GetBool(); |
| } |
| |
| // static |
| bool FileSystemStatus::IsHostRootfsWritable() { |
| base::ScopedBlockingCall scoped_blocking_call(FROM_HERE, |
| base::BlockingType::MAY_BLOCK); |
| struct statvfs buf; |
| if (statvfs("/", &buf) < 0) { |
| PLOG(ERROR) << "statvfs() failed"; |
| return false; |
| } |
| const bool rw = !(buf.f_flag & ST_RDONLY); |
| VLOG(1) << "Host's rootfs is " << (rw ? "rw" : "ro"); |
| return rw; |
| } |
| |
| // static |
| bool FileSystemStatus::IsSystemImageExtFormat(const base::FilePath& path) { |
| base::File file(path, base::File::FLAG_OPEN | base::File::FLAG_READ); |
| if (!file.IsValid()) { |
| PLOG(ERROR) << "Cannot open system image file: " << path.value(); |
| return false; |
| } |
| |
| uint8_t buf[2]; |
| if (!file.ReadAndCheck(0x400 + 0x38, base::make_span(buf, sizeof(buf)))) { |
| PLOG(ERROR) << "File read error on system image file: " << path.value(); |
| return false; |
| } |
| |
| uint16_t magic_le = *reinterpret_cast<uint16_t*>(buf); |
| #if defined(ARCH_CPU_LITTLE_ENDIAN) |
| return magic_le == EXT4_SUPER_MAGIC; |
| #else |
| #error Unsupported platform |
| #endif |
| } |
| |
| } // namespace arc |