/* * Copyright (C) 2018 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 "host/libs/config/cuttlefish_config.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include "common/libs/utils/environment.h" #include "common/libs/utils/files.h" #include "host/libs/vm_manager/qemu_manager.h" namespace { int InstanceFromEnvironment() { static constexpr char kInstanceEnvironmentVariable[] = "CUTTLEFISH_INSTANCE"; static constexpr int kDefaultInstance = 1; // CUTTLEFISH_INSTANCE environment variable const char* instance_str = std::getenv(kInstanceEnvironmentVariable); if (!instance_str) { // Try to get it from the user instead instance_str = std::getenv("USER"); if (!instance_str || std::strncmp(instance_str, vsoc::kVsocUserPrefix, sizeof(vsoc::kVsocUserPrefix) - 1)) { // No user or we don't recognize this user LOG(WARNING) << "No user or non-vsoc user, returning default config"; return kDefaultInstance; } instance_str += sizeof(vsoc::kVsocUserPrefix) - 1; // Set the environment variable so that child processes see it setenv(kInstanceEnvironmentVariable, instance_str, 0); } int instance = std::atoi(instance_str); if (instance <= 0) { instance = kDefaultInstance; } return instance; } const char* kInstances = "instances"; const char* kAssemblyDir = "assembly_dir"; const char* kSerialNumber = "serial_number"; const char* kInstanceDir = "instance_dir"; const char* kVmManager = "vm_manager"; const char* const kGpuMode = "gpu_mode"; const char* const kWaylandSocket = "wayland_socket"; const char* const kXDisplay = "x_display"; const char* kDeviceTitle = "device_title"; const char* kCpus = "cpus"; const char* kMemoryMb = "memory_mb"; const char* kDpi = "dpi"; const char* kXRes = "x_res"; const char* kYRes = "y_res"; const char* kRefreshRateHz = "refresh_rate_hz"; const char* kKernelImagePath = "kernel_image_path"; const char* kUseUnpackedKernel = "use_unpacked_kernel"; const char* kDecompressedKernelImagePath = "decompressed_kernel_image_path"; const char* kDecompressKernel = "decompress_kernel"; const char* kGdbFlag = "gdb_flag"; const char* kRamdiskImagePath = "ramdisk_image_path"; const char* kInitramfsPath = "initramfs_path"; const char* kFinalRamdiskPath = "final_ramdisk_path"; const char* kVendorRamdiskImagePath = "vendor_ramdisk_image_path"; const char* kVirtualDiskPaths = "virtual_disk_paths"; const char* kDeprecatedBootCompleted = "deprecated_boot_completed"; const char* kMobileBridgeName = "mobile_bridge_name"; const char* kMobileTapName = "mobile_tap_name"; const char* kWifiTapName = "wifi_tap_name"; const char* kVsockGuestCid = "vsock_guest_cid"; const char* kUuid = "uuid"; const char* kCuttlefishEnvPath = "cuttlefish_env_path"; const char* kAdbMode = "adb_mode"; const char* kHostPort = "host_port"; const char* kAdbIPAndPort = "adb_ip_and_port"; const char* kSetupWizardMode = "setupwizard_mode"; const char* kQemuBinary = "qemu_binary"; const char* kCrosvmBinary = "crosvm_binary"; const char* kConsoleForwarderBinary = "console_forwarder_binary"; const char* kKernelLogMonitorBinary = "kernel_log_monitor_binary"; const char* kEnableVncServer = "enable_vnc_server"; const char* kVncServerBinary = "vnc_server_binary"; const char* kVncServerPort = "vnc_server_port"; const char* kEnableWebRTC = "enable_webrtc"; const char* kWebRTCBinary = "webrtc_binary"; const char* kWebRTCAssetsDir = "webrtc_assets_dir"; const char* kWebRTCPublicIP = "webrtc_public_ip"; const char* kWebRTCEnableADBWebSocket = "webrtc_enable_adb_websocket"; const char* kEnableVehicleHalServer = "enable_vehicle_hal_server"; const char* kVehicleHalServerBinary = "vehicle_hal_server_binary"; const char* kVehicleHalServerPort = "vehicle_hal_server_port"; const char* kRestartSubprocesses = "restart_subprocesses"; const char* kRunAdbConnector = "run_adb_connector"; const char* kAdbConnectorBinary = "adb_connector_binary"; const char* kSocketVsockProxyBinary = "socket_vsock_proxy_binary"; const char* kRunAsDaemon = "run_as_daemon"; const char* kDataPolicy = "data_policy"; const char* kBlankDataImageMb = "blank_data_image_mb"; const char* kBlankDataImageFmt = "blank_data_image_fmt"; const char* kLogcatMode = "logcat_mode"; const char* kLogcatReceiverBinary = "logcat_receiver_binary"; const char* kConfigServerBinary = "config_server_binary"; const char* kRunTombstoneReceiver = "enable_tombstone_logger"; const char* kTombstoneReceiverBinary = "tombstone_receiver_binary"; const char* kWebRTCCertsDir = "webrtc_certs_dir"; const char* kBootloader = "bootloader"; const char* kUseBootloader = "use_bootloader"; const char* kBootSlot = "boot_slot"; const char* kLoopMaxPart = "loop_max_part"; const char* kGuestEnforceSecurity = "guest_enforce_security"; const char* kGuestAuditSecurity = "guest_audit_security"; const char* kGuestForceNormalBoot = "guest_force_normal_boot"; const char* kBootImageKernelCmdline = "boot_image_kernel_cmdline"; const char* kExtraKernelCmdline = "extra_kernel_cmdline"; const char* kWifiMacAddress = "wifi_mac_address"; } // namespace namespace vsoc { const char* const kGpuModeGuestSwiftshader = "guest_swiftshader"; const char* const kGpuModeDrmVirgl = "drm_virgl"; const char* const kGpuModeGfxStream = "gfxstream"; std::string DefaultEnvironmentPath(const char* environment_key, const char* default_value, const char* subpath) { return cvd::StringFromEnv(environment_key, default_value) + "/" + subpath; } Json::Value* CuttlefishConfig::MutableInstanceSpecific::Dictionary() { return &(*config_->dictionary_)[kInstances][id_]; } const Json::Value* CuttlefishConfig::InstanceSpecific::Dictionary() const { return &(*config_->dictionary_)[kInstances][id_]; } std::string CuttlefishConfig::assembly_dir() const { return (*dictionary_)[kAssemblyDir].asString(); } void CuttlefishConfig::set_assembly_dir(const std::string& assembly_dir) { (*dictionary_)[kAssemblyDir] = assembly_dir; } std::string CuttlefishConfig::InstanceSpecific::instance_dir() const { return (*Dictionary())[kInstanceDir].asString(); } void CuttlefishConfig::MutableInstanceSpecific::set_instance_dir( const std::string& instance_dir) { (*Dictionary())[kInstanceDir] = instance_dir; } std::string CuttlefishConfig::InstanceSpecific::instance_internal_dir() const { return PerInstancePath(kInternalDirName); } std::string CuttlefishConfig::vm_manager() const { return (*dictionary_)[kVmManager].asString(); } void CuttlefishConfig::set_vm_manager(const std::string& name) { (*dictionary_)[kVmManager] = name; } std::string CuttlefishConfig::gpu_mode() const { return (*dictionary_)[kGpuMode].asString(); } void CuttlefishConfig::set_gpu_mode(const std::string& name) { (*dictionary_)[kGpuMode] = name; } std::string CuttlefishConfig::wayland_socket() const { // Don't use SetPath here: the path is already fully formed. return (*dictionary_)[kWaylandSocket].asString(); } void CuttlefishConfig::set_wayland_socket(const std::string& path) { (*dictionary_)[kWaylandSocket] = path; } std::string CuttlefishConfig::x_display() const { return (*dictionary_)[kXDisplay].asString(); } void CuttlefishConfig::set_x_display(const std::string& address) { (*dictionary_)[kXDisplay] = address; } std::string CuttlefishConfig::InstanceSpecific::serial_number() const { return (*Dictionary())[kSerialNumber].asString(); } void CuttlefishConfig::MutableInstanceSpecific::set_serial_number( const std::string& serial_number) { (*Dictionary())[kSerialNumber] = serial_number; } int CuttlefishConfig::cpus() const { return (*dictionary_)[kCpus].asInt(); } void CuttlefishConfig::set_cpus(int cpus) { (*dictionary_)[kCpus] = cpus; } int CuttlefishConfig::memory_mb() const { return (*dictionary_)[kMemoryMb].asInt(); } void CuttlefishConfig::set_memory_mb(int memory_mb) { (*dictionary_)[kMemoryMb] = memory_mb; } int CuttlefishConfig::dpi() const { return (*dictionary_)[kDpi].asInt(); } void CuttlefishConfig::set_dpi(int dpi) { (*dictionary_)[kDpi] = dpi; } int CuttlefishConfig::x_res() const { return (*dictionary_)[kXRes].asInt(); } void CuttlefishConfig::set_x_res(int x_res) { (*dictionary_)[kXRes] = x_res; } int CuttlefishConfig::y_res() const { return (*dictionary_)[kYRes].asInt(); } void CuttlefishConfig::set_y_res(int y_res) { (*dictionary_)[kYRes] = y_res; } int CuttlefishConfig::refresh_rate_hz() const { return (*dictionary_)[kRefreshRateHz].asInt(); } void CuttlefishConfig::set_refresh_rate_hz(int refresh_rate_hz) { (*dictionary_)[kRefreshRateHz] = refresh_rate_hz; } std::string CuttlefishConfig::kernel_image_path() const { return (*dictionary_)[kKernelImagePath].asString(); } void CuttlefishConfig::SetPath(const std::string& key, const std::string& path) { if (!path.empty()) { (*dictionary_)[key] = cvd::AbsolutePath(path); } } void CuttlefishConfig::set_kernel_image_path( const std::string& kernel_image_path) { SetPath(kKernelImagePath, kernel_image_path); } bool CuttlefishConfig::use_unpacked_kernel() const { return (*dictionary_)[kUseUnpackedKernel].asBool(); } void CuttlefishConfig::set_use_unpacked_kernel(bool use_unpacked_kernel) { (*dictionary_)[kUseUnpackedKernel] = use_unpacked_kernel; } bool CuttlefishConfig::decompress_kernel() const { return (*dictionary_)[kDecompressKernel].asBool(); } void CuttlefishConfig::set_decompress_kernel(bool decompress_kernel) { (*dictionary_)[kDecompressKernel] = decompress_kernel; } std::string CuttlefishConfig::decompressed_kernel_image_path() const { return (*dictionary_)[kDecompressedKernelImagePath].asString(); } void CuttlefishConfig::set_decompressed_kernel_image_path( const std::string& path) { SetPath(kDecompressedKernelImagePath, path); } std::string CuttlefishConfig::gdb_flag() const { return (*dictionary_)[kGdbFlag].asString(); } void CuttlefishConfig::set_gdb_flag(const std::string& device) { (*dictionary_)[kGdbFlag] = device; } std::string CuttlefishConfig::ramdisk_image_path() const { return (*dictionary_)[kRamdiskImagePath].asString(); } void CuttlefishConfig::set_ramdisk_image_path( const std::string& ramdisk_image_path) { SetPath(kRamdiskImagePath, ramdisk_image_path); } std::string CuttlefishConfig::initramfs_path() const { return (*dictionary_)[kInitramfsPath].asString(); } void CuttlefishConfig::set_initramfs_path(const std::string& initramfs_path) { SetPath(kInitramfsPath, initramfs_path); } std::string CuttlefishConfig::final_ramdisk_path() const { return (*dictionary_)[kFinalRamdiskPath].asString(); } void CuttlefishConfig::set_final_ramdisk_path( const std::string& final_ramdisk_path) { SetPath(kFinalRamdiskPath, final_ramdisk_path); } std::string CuttlefishConfig::vendor_ramdisk_image_path() const { return (*dictionary_)[kVendorRamdiskImagePath].asString(); } void CuttlefishConfig::set_vendor_ramdisk_image_path( const std::string& vendor_ramdisk_image_path) { SetPath(kVendorRamdiskImagePath, vendor_ramdisk_image_path); } std::vector CuttlefishConfig::InstanceSpecific::virtual_disk_paths() const { std::vector virtual_disks; auto virtual_disks_json_obj = (*Dictionary())[kVirtualDiskPaths]; for (const auto& disk : virtual_disks_json_obj) { virtual_disks.push_back(disk.asString()); } return virtual_disks; } void CuttlefishConfig::MutableInstanceSpecific::set_virtual_disk_paths( const std::vector& virtual_disk_paths) { Json::Value virtual_disks_json_obj(Json::arrayValue); for (const auto& arg : virtual_disk_paths) { virtual_disks_json_obj.append(arg); } (*Dictionary())[kVirtualDiskPaths] = virtual_disks_json_obj; } std::string CuttlefishConfig::InstanceSpecific::kernel_log_pipe_name() const { return cvd::AbsolutePath(PerInstanceInternalPath("kernel-log-pipe")); } std::string CuttlefishConfig::InstanceSpecific::console_pipe_name() const { return cvd::AbsolutePath(PerInstanceInternalPath("console-pipe")); } bool CuttlefishConfig::deprecated_boot_completed() const { return (*dictionary_)[kDeprecatedBootCompleted].asBool(); } void CuttlefishConfig::set_deprecated_boot_completed( bool deprecated_boot_completed) { (*dictionary_)[kDeprecatedBootCompleted] = deprecated_boot_completed; } std::string CuttlefishConfig::InstanceSpecific::access_kregistry_path() const { return cvd::AbsolutePath(PerInstancePath("access-kregistry")); } std::string CuttlefishConfig::InstanceSpecific::console_path() const { return cvd::AbsolutePath(PerInstancePath("console")); } std::string CuttlefishConfig::InstanceSpecific::logcat_path() const { return cvd::AbsolutePath(PerInstancePath("logcat")); } std::string CuttlefishConfig::InstanceSpecific::launcher_monitor_socket_path() const { return cvd::AbsolutePath(PerInstancePath("launcher_monitor.sock")); } std::string CuttlefishConfig::InstanceSpecific::launcher_log_path() const { return cvd::AbsolutePath(PerInstancePath("launcher.log")); } std::string CuttlefishConfig::InstanceSpecific::sdcard_path() const { return cvd::AbsolutePath(PerInstancePath("sdcard.img")); } std::string CuttlefishConfig::InstanceSpecific::mobile_bridge_name() const { return (*Dictionary())[kMobileBridgeName].asString(); } void CuttlefishConfig::MutableInstanceSpecific::set_mobile_bridge_name( const std::string& mobile_bridge_name) { (*Dictionary())[kMobileBridgeName] = mobile_bridge_name; } std::string CuttlefishConfig::InstanceSpecific::mobile_tap_name() const { return (*Dictionary())[kMobileTapName].asString(); } void CuttlefishConfig::MutableInstanceSpecific::set_mobile_tap_name( const std::string& mobile_tap_name) { (*Dictionary())[kMobileTapName] = mobile_tap_name; } std::string CuttlefishConfig::InstanceSpecific::wifi_tap_name() const { return (*Dictionary())[kWifiTapName].asString(); } void CuttlefishConfig::MutableInstanceSpecific::set_wifi_tap_name( const std::string& wifi_tap_name) { (*Dictionary())[kWifiTapName] = wifi_tap_name; } int CuttlefishConfig::InstanceSpecific::vsock_guest_cid() const { return (*Dictionary())[kVsockGuestCid].asInt(); } void CuttlefishConfig::MutableInstanceSpecific::set_vsock_guest_cid( int vsock_guest_cid) { (*Dictionary())[kVsockGuestCid] = vsock_guest_cid; } std::string CuttlefishConfig::InstanceSpecific::uuid() const { return (*Dictionary())[kUuid].asString(); } void CuttlefishConfig::MutableInstanceSpecific::set_uuid(const std::string& uuid) { (*Dictionary())[kUuid] = uuid; } void CuttlefishConfig::set_cuttlefish_env_path(const std::string& path) { SetPath(kCuttlefishEnvPath, path); } std::string CuttlefishConfig::cuttlefish_env_path() const { return (*dictionary_)[kCuttlefishEnvPath].asString(); } static AdbMode stringToAdbMode(std::string mode) { std::transform(mode.begin(), mode.end(), mode.begin(), ::tolower); if (mode == "vsock_tunnel") { return AdbMode::VsockTunnel; } else if (mode == "vsock_half_tunnel") { return AdbMode::VsockHalfTunnel; } else if (mode == "native_vsock") { return AdbMode::NativeVsock; } else { return AdbMode::Unknown; } } std::set CuttlefishConfig::adb_mode() const { std::set args_set; for (auto& mode : (*dictionary_)[kAdbMode]) { args_set.insert(stringToAdbMode(mode.asString())); } return args_set; } void CuttlefishConfig::set_adb_mode(const std::set& mode) { Json::Value mode_json_obj(Json::arrayValue); for (const auto& arg : mode) { mode_json_obj.append(arg); } (*dictionary_)[kAdbMode] = mode_json_obj; } int CuttlefishConfig::InstanceSpecific::host_port() const { return (*Dictionary())[kHostPort].asInt(); } void CuttlefishConfig::MutableInstanceSpecific::set_host_port(int host_port) { (*Dictionary())[kHostPort] = host_port; } std::string CuttlefishConfig::InstanceSpecific::adb_ip_and_port() const { return (*Dictionary())[kAdbIPAndPort].asString(); } void CuttlefishConfig::MutableInstanceSpecific::set_adb_ip_and_port( const std::string& ip_port) { (*Dictionary())[kAdbIPAndPort] = ip_port; } std::string CuttlefishConfig::InstanceSpecific::adb_device_name() const { if (adb_ip_and_port() != "") { return adb_ip_and_port(); } LOG(ERROR) << "no adb_mode found, returning bad device name"; return "NO_ADB_MODE_SET_NO_VALID_DEVICE_NAME"; } std::string CuttlefishConfig::InstanceSpecific::device_title() const { return (*Dictionary())[kDeviceTitle].asString(); } void CuttlefishConfig::MutableInstanceSpecific::set_device_title( const std::string& title) { (*Dictionary())[kDeviceTitle] = title; } std::string CuttlefishConfig::setupwizard_mode() const { return (*dictionary_)[kSetupWizardMode].asString(); } void CuttlefishConfig::set_setupwizard_mode(const std::string& mode) { (*dictionary_)[kSetupWizardMode] = mode; } std::string CuttlefishConfig::qemu_binary() const { return (*dictionary_)[kQemuBinary].asString(); } void CuttlefishConfig::set_qemu_binary(const std::string& qemu_binary) { (*dictionary_)[kQemuBinary] = qemu_binary; } std::string CuttlefishConfig::crosvm_binary() const { return (*dictionary_)[kCrosvmBinary].asString(); } void CuttlefishConfig::set_crosvm_binary(const std::string& crosvm_binary) { (*dictionary_)[kCrosvmBinary] = crosvm_binary; } std::string CuttlefishConfig::console_forwarder_binary() const { return (*dictionary_)[kConsoleForwarderBinary].asString(); } void CuttlefishConfig::set_console_forwarder_binary( const std::string& binary) { (*dictionary_)[kConsoleForwarderBinary] = binary; } std::string CuttlefishConfig::kernel_log_monitor_binary() const { return (*dictionary_)[kKernelLogMonitorBinary].asString(); } void CuttlefishConfig::set_kernel_log_monitor_binary( const std::string& kernel_log_monitor_binary) { (*dictionary_)[kKernelLogMonitorBinary] = kernel_log_monitor_binary; } bool CuttlefishConfig::enable_vnc_server() const { return (*dictionary_)[kEnableVncServer].asBool(); } void CuttlefishConfig::set_enable_vnc_server(bool enable_vnc_server) { (*dictionary_)[kEnableVncServer] = enable_vnc_server; } std::string CuttlefishConfig::vnc_server_binary() const { return (*dictionary_)[kVncServerBinary].asString(); } void CuttlefishConfig::set_vnc_server_binary( const std::string& vnc_server_binary) { (*dictionary_)[kVncServerBinary] = vnc_server_binary; } int CuttlefishConfig::InstanceSpecific::vnc_server_port() const { return (*Dictionary())[kVncServerPort].asInt(); } void CuttlefishConfig::MutableInstanceSpecific::set_vnc_server_port(int vnc_server_port) { (*Dictionary())[kVncServerPort] = vnc_server_port; } int CuttlefishConfig::InstanceSpecific::vehicle_hal_server_port() const { return (*Dictionary())[kVehicleHalServerPort].asInt(); } void CuttlefishConfig::MutableInstanceSpecific::set_vehicle_hal_server_port(int vehicle_hal_server_port) { (*Dictionary())[kVehicleHalServerPort] = vehicle_hal_server_port; } void CuttlefishConfig::set_enable_webrtc(bool enable_webrtc) { (*dictionary_)[kEnableWebRTC] = enable_webrtc; } bool CuttlefishConfig::enable_webrtc() const { return (*dictionary_)[kEnableWebRTC].asBool(); } void CuttlefishConfig::set_webrtc_binary(const std::string& webrtc_binary) { (*dictionary_)[kWebRTCBinary] = webrtc_binary; } std::string CuttlefishConfig::webrtc_binary() const { return (*dictionary_)[kWebRTCBinary].asString(); } void CuttlefishConfig::set_enable_vehicle_hal_grpc_server(bool enable_vehicle_hal_grpc_server) { (*dictionary_)[kEnableVehicleHalServer] = enable_vehicle_hal_grpc_server; } bool CuttlefishConfig::enable_vehicle_hal_grpc_server() const { return (*dictionary_)[kEnableVehicleHalServer].asBool(); } void CuttlefishConfig::set_vehicle_hal_grpc_server_binary(const std::string& vehicle_hal_server_binary) { (*dictionary_)[kVehicleHalServerBinary] = vehicle_hal_server_binary; } std::string CuttlefishConfig::vehicle_hal_grpc_server_binary() const { return (*dictionary_)[kVehicleHalServerBinary].asString(); } void CuttlefishConfig::set_webrtc_assets_dir(const std::string& webrtc_assets_dir) { (*dictionary_)[kWebRTCAssetsDir] = webrtc_assets_dir; } std::string CuttlefishConfig::webrtc_assets_dir() const { return (*dictionary_)[kWebRTCAssetsDir].asString(); } void CuttlefishConfig::set_webrtc_public_ip( const std::string& webrtc_public_ip) { (*dictionary_)[kWebRTCPublicIP] = webrtc_public_ip; } std::string CuttlefishConfig::webrtc_public_ip() const { return (*dictionary_)[kWebRTCPublicIP].asString(); } void CuttlefishConfig::set_webrtc_enable_adb_websocket(bool enable) { (*dictionary_)[kWebRTCEnableADBWebSocket] = enable; } bool CuttlefishConfig::webrtc_enable_adb_websocket() const { return (*dictionary_)[kWebRTCEnableADBWebSocket].asBool(); } bool CuttlefishConfig::restart_subprocesses() const { return (*dictionary_)[kRestartSubprocesses].asBool(); } void CuttlefishConfig::set_restart_subprocesses(bool restart_subprocesses) { (*dictionary_)[kRestartSubprocesses] = restart_subprocesses; } bool CuttlefishConfig::run_adb_connector() const { return (*dictionary_)[kRunAdbConnector].asBool(); } void CuttlefishConfig::set_run_adb_connector(bool run_adb_connector) { (*dictionary_)[kRunAdbConnector] = run_adb_connector; } std::string CuttlefishConfig::adb_connector_binary() const { return (*dictionary_)[kAdbConnectorBinary].asString(); } void CuttlefishConfig::set_adb_connector_binary( const std::string& adb_connector_binary) { (*dictionary_)[kAdbConnectorBinary] = adb_connector_binary; } std::string CuttlefishConfig::socket_vsock_proxy_binary() const { return (*dictionary_)[kSocketVsockProxyBinary].asString(); } void CuttlefishConfig::set_socket_vsock_proxy_binary( const std::string& socket_vsock_proxy_binary) { (*dictionary_)[kSocketVsockProxyBinary] = socket_vsock_proxy_binary; } bool CuttlefishConfig::run_as_daemon() const { return (*dictionary_)[kRunAsDaemon].asBool(); } void CuttlefishConfig::set_run_as_daemon(bool run_as_daemon) { (*dictionary_)[kRunAsDaemon] = run_as_daemon; } std::string CuttlefishConfig::data_policy() const { return (*dictionary_)[kDataPolicy].asString(); } void CuttlefishConfig::set_data_policy(const std::string& data_policy) { (*dictionary_)[kDataPolicy] = data_policy; } int CuttlefishConfig::blank_data_image_mb() const { return (*dictionary_)[kBlankDataImageMb].asInt(); } void CuttlefishConfig::set_blank_data_image_mb(int blank_data_image_mb) { (*dictionary_)[kBlankDataImageMb] = blank_data_image_mb; } std::string CuttlefishConfig::blank_data_image_fmt() const { return (*dictionary_)[kBlankDataImageFmt].asString(); } void CuttlefishConfig::set_blank_data_image_fmt(const std::string& blank_data_image_fmt) { (*dictionary_)[kBlankDataImageFmt] = blank_data_image_fmt; } void CuttlefishConfig::set_logcat_mode(const std::string& mode) { (*dictionary_)[kLogcatMode] = mode; } std::string CuttlefishConfig::logcat_mode() const { return (*dictionary_)[kLogcatMode].asString(); } void CuttlefishConfig::set_logcat_receiver_binary(const std::string& binary) { SetPath(kLogcatReceiverBinary, binary); } std::string CuttlefishConfig::logcat_receiver_binary() const { return (*dictionary_)[kLogcatReceiverBinary].asString(); } void CuttlefishConfig::set_config_server_binary(const std::string& binary) { SetPath(kConfigServerBinary, binary); } std::string CuttlefishConfig::config_server_binary() const { return (*dictionary_)[kConfigServerBinary].asString(); } bool CuttlefishConfig::enable_tombstone_receiver() const { return (*dictionary_)[kRunTombstoneReceiver].asBool(); } void CuttlefishConfig::set_enable_tombstone_receiver(bool enable_tombstone_receiver) { (*dictionary_)[kRunTombstoneReceiver] = enable_tombstone_receiver; } std::string CuttlefishConfig::tombstone_receiver_binary() const { return (*dictionary_)[kTombstoneReceiverBinary].asString(); } void CuttlefishConfig::set_tombstone_receiver_binary(const std::string& e2e_test_binary) { (*dictionary_)[kTombstoneReceiverBinary] = e2e_test_binary; } bool CuttlefishConfig::use_bootloader() const { return (*dictionary_)[kUseBootloader].asBool(); } void CuttlefishConfig::set_use_bootloader(bool use_bootloader) { (*dictionary_)[kUseBootloader] = use_bootloader; } std::string CuttlefishConfig::bootloader() const { return (*dictionary_)[kBootloader].asString(); } void CuttlefishConfig::set_bootloader(const std::string& bootloader) { SetPath(kBootloader, bootloader); } void CuttlefishConfig::set_boot_slot(const std::string& boot_slot) { (*dictionary_)[kBootSlot] = boot_slot; } std::string CuttlefishConfig::boot_slot() const { return (*dictionary_)[kBootSlot].asString(); } void CuttlefishConfig::set_webrtc_certs_dir(const std::string& certs_dir) { (*dictionary_)[kWebRTCCertsDir] = certs_dir; } std::string CuttlefishConfig::webrtc_certs_dir() const { return (*dictionary_)[kWebRTCCertsDir].asString(); } std::string CuttlefishConfig::InstanceSpecific::touch_socket_path() const { return PerInstanceInternalPath("touch.sock"); } std::string CuttlefishConfig::InstanceSpecific::keyboard_socket_path() const { return PerInstanceInternalPath("keyboard.sock"); } std::string CuttlefishConfig::InstanceSpecific::frames_socket_path() const { return PerInstanceInternalPath("frames.sock"); } void CuttlefishConfig::set_loop_max_part(int loop_max_part) { (*dictionary_)[kLoopMaxPart] = loop_max_part; } int CuttlefishConfig::loop_max_part() const { return (*dictionary_)[kLoopMaxPart].asInt(); } void CuttlefishConfig::set_guest_enforce_security(bool guest_enforce_security) { (*dictionary_)[kGuestEnforceSecurity] = guest_enforce_security; } bool CuttlefishConfig::guest_enforce_security() const { return (*dictionary_)[kGuestEnforceSecurity].asBool(); } void CuttlefishConfig::set_guest_audit_security(bool guest_audit_security) { (*dictionary_)[kGuestAuditSecurity] = guest_audit_security; } bool CuttlefishConfig::guest_audit_security() const { return (*dictionary_)[kGuestAuditSecurity].asBool(); } void CuttlefishConfig::set_guest_force_normal_boot(bool guest_force_normal_boot) { (*dictionary_)[kGuestForceNormalBoot] = guest_force_normal_boot; } bool CuttlefishConfig::guest_force_normal_boot() const { return (*dictionary_)[kGuestForceNormalBoot].asBool(); } void CuttlefishConfig::MutableInstanceSpecific::set_wifi_mac_address( const std::array& mac_address) { Json::Value mac_address_obj(Json::arrayValue); for (const auto& num : mac_address) { mac_address_obj.append(num); } (*Dictionary())[kWifiMacAddress] = mac_address_obj; } std::array CuttlefishConfig::InstanceSpecific::wifi_mac_address() const { std::array mac_address{0, 0, 0, 0, 0, 0}; auto mac_address_obj = (*Dictionary())[kWifiMacAddress]; if (mac_address_obj.size() != 6) { LOG(ERROR) << kWifiMacAddress << " entry had wrong size"; return {}; } for (int i = 0; i < 6; i++) { mac_address[i] = mac_address_obj[i].asInt(); } return mac_address; } void CuttlefishConfig::set_boot_image_kernel_cmdline(std::string boot_image_kernel_cmdline) { Json::Value args_json_obj(Json::arrayValue); for (const auto& arg : android::base::Split(boot_image_kernel_cmdline, " ")) { args_json_obj.append(arg); } (*dictionary_)[kBootImageKernelCmdline] = args_json_obj; } std::vector CuttlefishConfig::boot_image_kernel_cmdline() const { std::vector cmdline; for (const Json::Value& arg : (*dictionary_)[kBootImageKernelCmdline]) { cmdline.push_back(arg.asString()); } return cmdline; } void CuttlefishConfig::set_extra_kernel_cmdline(std::string extra_cmdline) { Json::Value args_json_obj(Json::arrayValue); for (const auto& arg : android::base::Split(extra_cmdline, " ")) { args_json_obj.append(arg); } (*dictionary_)[kExtraKernelCmdline] = args_json_obj; } std::vector CuttlefishConfig::extra_kernel_cmdline() const { std::vector cmdline; for (const Json::Value& arg : (*dictionary_)[kExtraKernelCmdline]) { cmdline.push_back(arg.asString()); } return cmdline; } // Creates the (initially empty) config object and populates it with values from // the config file if the CUTTLEFISH_CONFIG_FILE env variable is present. // Returns nullptr if there was an error loading from file /*static*/ CuttlefishConfig* CuttlefishConfig::BuildConfigImpl() { auto config_file_path = cvd::StringFromEnv(kCuttlefishConfigEnvVarName, vsoc::GetGlobalConfigFileLink()); auto ret = new CuttlefishConfig(); if (ret) { auto loaded = ret->LoadFromFile(config_file_path.c_str()); if (!loaded) { delete ret; return nullptr; } } return ret; } /*static*/ const CuttlefishConfig* CuttlefishConfig::Get() { static std::shared_ptr config(BuildConfigImpl()); return config.get(); } CuttlefishConfig::CuttlefishConfig() : dictionary_(new Json::Value()) {} // Can't use '= default' on the header because the compiler complains of // Json::Value being an incomplete type CuttlefishConfig::~CuttlefishConfig() = default; CuttlefishConfig::CuttlefishConfig(CuttlefishConfig&&) = default; CuttlefishConfig& CuttlefishConfig::operator=(CuttlefishConfig&&) = default; bool CuttlefishConfig::LoadFromFile(const char* file) { auto real_file_path = cvd::AbsolutePath(file); if (real_file_path.empty()) { LOG(ERROR) << "Could not get real path for file " << file; return false; } Json::Reader reader; std::ifstream ifs(real_file_path); if (!reader.parse(ifs, *dictionary_)) { LOG(ERROR) << "Could not read config file " << file << ": " << reader.getFormattedErrorMessages(); return false; } return true; } bool CuttlefishConfig::SaveToFile(const std::string& file) const { std::ofstream ofs(file); if (!ofs.is_open()) { LOG(ERROR) << "Unable to write to file " << file; return false; } ofs << *dictionary_; return !ofs.fail(); } std::string CuttlefishConfig::AssemblyPath( const std::string& file_name) const { return cvd::AbsolutePath(assembly_dir() + "/" + file_name); } std::string CuttlefishConfig::composite_disk_path() const { return AssemblyPath("composite.img"); } std::string CuttlefishConfig::InstanceSpecific::PerInstancePath( const char* file_name) const { return (instance_dir() + "/") + file_name; } std::string CuttlefishConfig::InstanceSpecific::PerInstanceInternalPath( const char* file_name) const { if (file_name[0] == '\0') { // Don't append a / if file_name is empty. return PerInstancePath(kInternalDirName); } auto relative_path = (std::string(kInternalDirName) + "/") + file_name; return PerInstancePath(relative_path.c_str()); } std::string CuttlefishConfig::InstanceSpecific::instance_name() const { return ForCurrentInstance("cvd-"); } CuttlefishConfig::MutableInstanceSpecific CuttlefishConfig::ForInstance(int num) { return MutableInstanceSpecific(this, std::to_string(num)); } CuttlefishConfig::InstanceSpecific CuttlefishConfig::ForInstance(int num) const { return InstanceSpecific(this, std::to_string(num)); } CuttlefishConfig::InstanceSpecific CuttlefishConfig::ForDefaultInstance() const { return InstanceSpecific(this, std::to_string(GetInstance())); } std::vector CuttlefishConfig::Instances() const { const auto& json = (*dictionary_)[kInstances]; std::vector instances; for (const auto& name : json.getMemberNames()) { instances.push_back(CuttlefishConfig::InstanceSpecific(this, name)); } return instances; } int GetInstance() { static int instance_id = InstanceFromEnvironment(); return instance_id; } std::string GetGlobalConfigFileLink() { return cvd::StringFromEnv("HOME", ".") + "/.cuttlefish_config.json"; } std::string ForCurrentInstance(const char* prefix) { std::ostringstream stream; stream << prefix << std::setfill('0') << std::setw(2) << GetInstance(); return stream.str(); } int ForCurrentInstance(int base) { return base + GetInstance() - 1; } std::string RandomSerialNumber(const std::string& prefix) { const char hex_characters[] = "0123456789ABCDEF"; std::srand(time(0)); char str[10]; for(int i=0; i<10; i++){ str[i] = hex_characters[rand() % strlen(hex_characters)]; } return prefix + str; } std::string GetDefaultPerInstanceDir() { std::ostringstream stream; stream << std::getenv("HOME") << "/cuttlefish_runtime"; return stream.str(); } int GetDefaultPerInstanceVsockCid() { constexpr int kFirstGuestCid = 3; return vsoc::HostSupportsVsock() ? ForCurrentInstance(kFirstGuestCid) : 0; } std::string DefaultHostArtifactsPath(const std::string& file_name) { return (cvd::StringFromEnv("ANDROID_HOST_OUT", cvd::StringFromEnv("HOME", ".")) + "/") + file_name; } std::string DefaultGuestImagePath(const std::string& file_name) { return (cvd::StringFromEnv("ANDROID_PRODUCT_OUT", cvd::StringFromEnv("HOME", ".")) + "/") + file_name; } bool HostSupportsQemuCli() { static bool supported = std::system( "/usr/lib/cuttlefish-common/bin/capability_query.py qemu_cli") == 0; return supported; } bool HostSupportsVsock() { static bool supported = std::system( "/usr/lib/cuttlefish-common/bin/capability_query.py vsock") == 0; return supported; } } // namespace vsoc