// Copyright 2017 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/arc_util.h" #include #include #include "ash/public/cpp/app_types.h" #include "base/bind.h" #include "base/command_line.h" #include "base/feature_list.h" #include "chromeos/constants/chromeos_switches.h" #include "chromeos/dbus/session_manager/session_manager_client.h" #include "components/arc/arc_features.h" #include "components/exo/shell_surface_util.h" #include "components/user_manager/user_manager.h" #include "ui/aura/client/aura_constants.h" #include "ui/aura/window.h" namespace arc { namespace { // This is for finch. See also crbug.com/633704 for details. // TODO(hidehiko): More comments of the intention how this works, when // we unify the commandline flags. const base::Feature kEnableArcFeature{"EnableARC", base::FEATURE_DISABLED_BY_DEFAULT}; // Possible values for --arc-availability flag. constexpr char kAvailabilityNone[] = "none"; constexpr char kAvailabilityInstalled[] = "installed"; constexpr char kAvailabilityOfficiallySupported[] = "officially-supported"; constexpr char kAlwaysStartWithNoPlayStore[] = "always-start-with-no-play-store"; void SetArcCpuRestrictionCallback( login_manager::ContainerCpuRestrictionState state, bool success) { if (success) return; const char* message = (state == login_manager::CONTAINER_CPU_RESTRICTION_BACKGROUND) ? "unprioritize" : "prioritize"; LOG(ERROR) << "Failed to " << message << " ARC"; } } // namespace bool IsArcAvailable() { const auto* command_line = base::CommandLine::ForCurrentProcess(); if (command_line->HasSwitch(chromeos::switches::kArcAvailability)) { const std::string value = command_line->GetSwitchValueASCII(chromeos::switches::kArcAvailability); DCHECK(value == kAvailabilityNone || value == kAvailabilityInstalled || value == kAvailabilityOfficiallySupported) << "Unknown flag value: " << value; return value == kAvailabilityOfficiallySupported || (value == kAvailabilityInstalled && base::FeatureList::IsEnabled(kEnableArcFeature)); } // For transition, fallback to old flags. // TODO(hidehiko): Remove this and clean up whole this function, when // session_manager supports a new flag. return command_line->HasSwitch(chromeos::switches::kEnableArc) || (command_line->HasSwitch(chromeos::switches::kArcAvailable) && base::FeatureList::IsEnabled(kEnableArcFeature)); } bool IsArcVmEnabled() { return base::CommandLine::ForCurrentProcess()->HasSwitch( chromeos::switches::kEnableArcVm); } bool ShouldArcAlwaysStart() { const auto* command_line = base::CommandLine::ForCurrentProcess(); if (!command_line->HasSwitch(chromeos::switches::kArcStartMode)) return false; return command_line->GetSwitchValueASCII(chromeos::switches::kArcStartMode) == kAlwaysStartWithNoPlayStore; } bool ShouldArcAlwaysStartWithNoPlayStore() { return base::CommandLine::ForCurrentProcess()->GetSwitchValueASCII( chromeos::switches::kArcStartMode) == kAlwaysStartWithNoPlayStore; } bool ShouldShowOptInForTesting() { return base::CommandLine::ForCurrentProcess()->HasSwitch( chromeos::switches::kArcForceShowOptInUi); } void SetArcAlwaysStartWithoutPlayStoreForTesting() { base::CommandLine::ForCurrentProcess()->AppendSwitchASCII( chromeos::switches::kArcStartMode, kAlwaysStartWithNoPlayStore); } bool IsArcKioskAvailable() { const auto* command_line = base::CommandLine::ForCurrentProcess(); if (command_line->HasSwitch(chromeos::switches::kArcAvailability)) { std::string value = command_line->GetSwitchValueASCII(chromeos::switches::kArcAvailability); if (value == kAvailabilityInstalled) return true; return IsArcAvailable(); } // TODO(hidehiko): Remove this when session_manager supports the new flag. if (command_line->HasSwitch(chromeos::switches::kArcAvailable)) return true; // If not special kiosk device case, use general ARC check. return IsArcAvailable(); } void SetArcAvailableCommandLineForTesting(base::CommandLine* command_line) { command_line->AppendSwitchASCII(chromeos::switches::kArcAvailability, kAvailabilityOfficiallySupported); } bool IsArcKioskMode() { return user_manager::UserManager::IsInitialized() && user_manager::UserManager::Get()->IsLoggedInAsArcKioskApp(); } bool IsRobotOrOfflineDemoAccountMode() { return user_manager::UserManager::IsInitialized() && (user_manager::UserManager::Get()->IsLoggedInAsArcKioskApp() || user_manager::UserManager::Get()->IsLoggedInAsPublicAccount()); } bool IsArcAllowedForUser(const user_manager::User* user) { if (!user) { VLOG(1) << "No ARC for nullptr user."; return false; } // ARC is only supported for the following cases: // - Users have Gaia accounts; // - Active directory users; // - ARC kiosk session; // - Public Session users; // USER_TYPE_ARC_KIOSK_APP check is compatible with IsArcKioskMode() // above because ARC kiosk user is always the primary/active user of a // user session. The same for USER_TYPE_PUBLIC_ACCOUNT. if (!user->HasGaiaAccount() && !user->IsActiveDirectoryUser() && user->GetType() != user_manager::USER_TYPE_ARC_KIOSK_APP && user->GetType() != user_manager::USER_TYPE_PUBLIC_ACCOUNT) { VLOG(1) << "Users without GAIA or AD accounts, or not ARC kiosk apps are " "not supported in ARC."; return false; } if (user->GetType() == user_manager::USER_TYPE_CHILD && !base::FeatureList::IsEnabled(arc::kAvailableForChildAccountFeature)) { VLOG(1) << "ARC usage by Child users is prohibited"; return false; } return true; } bool IsArcOptInVerificationDisabled() { return base::CommandLine::ForCurrentProcess()->HasSwitch( chromeos::switches::kDisableArcOptInVerification); } bool IsArcAppWindow(const aura::Window* window) { if (!window) return false; return window->GetProperty(aura::client::kAppType) == static_cast(ash::AppType::ARC_APP); } int GetWindowTaskId(const aura::Window* window) { const std::string* arc_app_id = exo::GetShellApplicationId(window); if (!arc_app_id) return kNoTaskId; return GetTaskIdFromWindowAppId(*arc_app_id); } int GetTaskIdFromWindowAppId(const std::string& app_id) { int task_id; if (std::sscanf(app_id.c_str(), "org.chromium.arc.%d", &task_id) != 1) return kNoTaskId; return task_id; } void SetArcCpuRestriction(bool do_restrict) { if (!chromeos::SessionManagerClient::Get()) { LOG(WARNING) << "SessionManagerClient is not available"; return; } // Ignore any calls to restrict the ARC container if the specified command // line flag is set. if (chromeos::switches::IsArcCpuRestrictionDisabled() && do_restrict) return; const login_manager::ContainerCpuRestrictionState state = do_restrict ? login_manager::CONTAINER_CPU_RESTRICTION_BACKGROUND : login_manager::CONTAINER_CPU_RESTRICTION_FOREGROUND; chromeos::SessionManagerClient::Get()->SetArcCpuRestriction( state, base::BindOnce(SetArcCpuRestrictionCallback, state)); } bool IsArcForceCacheAppIcon() { return base::CommandLine::ForCurrentProcess()->HasSwitch( chromeos::switches::kArcForceCacheAppIcons); } bool IsArcDataCleanupOnStartRequested() { return base::CommandLine::ForCurrentProcess()->HasSwitch( chromeos::switches::kArcDataCleanupOnStart); } bool IsArcAppSyncFlowDisabled() { return base::CommandLine::ForCurrentProcess()->HasSwitch( chromeos::switches::kArcDisableAppSync); } bool IsArcLocaleSyncDisabled() { return base::CommandLine::ForCurrentProcess()->HasSwitch( chromeos::switches::kArcDisableLocaleSync); } bool IsArcPlayAutoInstallDisabled() { return base::CommandLine::ForCurrentProcess()->HasSwitch( chromeos::switches::kArcDisablePlayAutoInstall); } // static int32_t GetLcdDensityForDeviceScaleFactor(float device_scale_factor) { // Keep this consistent with wayland_client.cpp on Android side. // TODO(oshima): Consider sending this through wayland. constexpr float kEpsilon = 0.001; if (std::abs(device_scale_factor - 2.25f) < kEpsilon) return 280; if (std::abs(device_scale_factor - 1.6f) < kEpsilon) return 213; // TVDPI constexpr float kChromeScaleToAndroidScaleRatio = 0.75f; constexpr int32_t kDefaultDensityDpi = 160; return static_cast( std::max(1.0f, device_scale_factor * kChromeScaleToAndroidScaleRatio) * kDefaultDensityDpi); } } // namespace arc