/* * 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. */ package com.android.systemui; import android.content.Context; import android.content.res.Configuration; import android.hardware.SensorManager; import android.os.Handler; import android.os.HandlerThread; import android.os.Looper; import android.os.Process; import android.os.ServiceManager; import android.util.ArrayMap; import android.view.IWindowManager; import android.view.WindowManagerGlobal; import com.android.internal.annotations.VisibleForTesting; import com.android.internal.app.ColorDisplayController; import com.android.internal.logging.MetricsLogger; import com.android.internal.statusbar.IStatusBarService; import com.android.internal.util.Preconditions; import com.android.settingslib.bluetooth.LocalBluetoothManager; import com.android.systemui.assist.AssistManager; import com.android.systemui.colorextraction.SysuiColorExtractor; import com.android.systemui.fragments.FragmentService; import com.android.systemui.keyguard.ScreenLifecycle; import com.android.systemui.keyguard.WakefulnessLifecycle; import com.android.systemui.plugins.ActivityStarter; import com.android.systemui.plugins.PluginDependencyProvider; import com.android.systemui.plugins.PluginManager; import com.android.systemui.plugins.PluginManagerImpl; import com.android.systemui.plugins.VolumeDialogController; import com.android.systemui.power.EnhancedEstimates; import com.android.systemui.power.EnhancedEstimatesImpl; import com.android.systemui.power.PowerNotificationWarnings; import com.android.systemui.power.PowerUI; import com.android.systemui.statusbar.AppOpsListener; import com.android.systemui.statusbar.VibratorHelper; import com.android.systemui.statusbar.phone.ConfigurationControllerImpl; import com.android.systemui.statusbar.phone.DarkIconDispatcherImpl; import com.android.systemui.statusbar.phone.LightBarController; import com.android.systemui.statusbar.phone.ManagedProfileController; import com.android.systemui.statusbar.phone.ManagedProfileControllerImpl; import com.android.systemui.statusbar.phone.StatusBarIconController; import com.android.systemui.statusbar.phone.StatusBarIconControllerImpl; import com.android.systemui.statusbar.phone.StatusBarWindowManager; import com.android.systemui.statusbar.policy.AccessibilityController; import com.android.systemui.statusbar.policy.AccessibilityManagerWrapper; import com.android.systemui.statusbar.policy.BatteryController; import com.android.systemui.statusbar.policy.BatteryControllerImpl; import com.android.systemui.statusbar.policy.BluetoothController; import com.android.systemui.statusbar.policy.BluetoothControllerImpl; import com.android.systemui.statusbar.policy.CastController; import com.android.systemui.statusbar.policy.CastControllerImpl; import com.android.systemui.statusbar.policy.ConfigurationController; import com.android.systemui.statusbar.policy.DarkIconDispatcher; import com.android.systemui.statusbar.policy.DataSaverController; import com.android.systemui.statusbar.policy.DeviceProvisionedController; import com.android.systemui.statusbar.policy.DeviceProvisionedControllerImpl; import com.android.systemui.statusbar.policy.ExtensionController; import com.android.systemui.statusbar.policy.ExtensionControllerImpl; import com.android.systemui.statusbar.policy.FlashlightController; import com.android.systemui.statusbar.policy.FlashlightControllerImpl; import com.android.systemui.statusbar.policy.HotspotController; import com.android.systemui.statusbar.policy.HotspotControllerImpl; import com.android.systemui.statusbar.policy.IconLogger; import com.android.systemui.statusbar.policy.IconLoggerImpl; import com.android.systemui.statusbar.policy.KeyguardMonitor; import com.android.systemui.statusbar.policy.KeyguardMonitorImpl; import com.android.systemui.statusbar.policy.LocationController; import com.android.systemui.statusbar.policy.LocationControllerImpl; import com.android.systemui.statusbar.policy.NetworkController; import com.android.systemui.statusbar.policy.NetworkControllerImpl; import com.android.systemui.statusbar.policy.NextAlarmController; import com.android.systemui.statusbar.policy.NextAlarmControllerImpl; import com.android.systemui.statusbar.policy.RotationLockController; import com.android.systemui.statusbar.policy.RotationLockControllerImpl; import com.android.systemui.statusbar.policy.SecurityController; import com.android.systemui.statusbar.policy.SecurityControllerImpl; import com.android.systemui.statusbar.policy.UserInfoController; import com.android.systemui.statusbar.policy.UserInfoControllerImpl; import com.android.systemui.statusbar.policy.UserSwitcherController; import com.android.systemui.statusbar.policy.ZenModeController; import com.android.systemui.statusbar.policy.ZenModeControllerImpl; import com.android.systemui.tuner.TunablePadding.TunablePaddingService; import com.android.systemui.tuner.TunerService; import com.android.systemui.tuner.TunerServiceImpl; import com.android.systemui.util.AsyncSensorManager; import com.android.systemui.util.leak.GarbageMonitor; import com.android.systemui.util.leak.LeakDetector; import com.android.systemui.util.leak.LeakReporter; import com.android.systemui.volume.VolumeDialogControllerImpl; import java.io.FileDescriptor; import java.io.PrintWriter; import java.util.HashMap; import java.util.function.Consumer; /** * Class to handle ugly dependencies throughout sysui until we determine the * long-term dependency injection solution. * * Classes added here should be things that are expected to live the lifetime of sysui, * and are generally applicable to many parts of sysui. They will be lazily * initialized to ensure they aren't created on form factors that don't need them * (e.g. HotspotController on TV). Despite being lazily initialized, it is expected * that all dependencies will be gotten during sysui startup, and not during runtime * to avoid jank. * * All classes used here are expected to manage their own lifecycle, meaning if * they have no clients they should not have any registered resources like bound * services, registered receivers, etc. */ public class Dependency extends SystemUI { private static final String TAG = "Dependency"; /** * Key for getting a background Looper for background work. */ public static final DependencyKey BG_LOOPER = new DependencyKey<>("background_looper"); /** * Key for getting a Handler for receiving time tick broadcasts on. */ public static final DependencyKey TIME_TICK_HANDLER = new DependencyKey<>("time_tick_handler"); /** * Generic handler on the main thread. */ public static final DependencyKey MAIN_HANDLER = new DependencyKey<>("main_handler"); /** * An email address to send memory leak reports to by default. */ public static final DependencyKey LEAK_REPORT_EMAIL = new DependencyKey<>("leak_report_email"); private final ArrayMap mDependencies = new ArrayMap<>(); private final ArrayMap mProviders = new ArrayMap<>(); @Override public void start() { // TODO: Think about ways to push these creation rules out of Dependency to cut down // on imports. mProviders.put(TIME_TICK_HANDLER, () -> { HandlerThread thread = new HandlerThread("TimeTick"); thread.start(); return new Handler(thread.getLooper()); }); mProviders.put(BG_LOOPER, () -> { HandlerThread thread = new HandlerThread("SysUiBg", Process.THREAD_PRIORITY_BACKGROUND); thread.start(); return thread.getLooper(); }); mProviders.put(MAIN_HANDLER, () -> new Handler(Looper.getMainLooper())); mProviders.put(ActivityStarter.class, () -> new ActivityStarterDelegate()); mProviders.put(ActivityStarterDelegate.class, () -> getDependency(ActivityStarter.class)); mProviders.put(AsyncSensorManager.class, () -> new AsyncSensorManager(mContext.getSystemService(SensorManager.class))); mProviders.put(BluetoothController.class, () -> new BluetoothControllerImpl(mContext, getDependency(BG_LOOPER))); mProviders.put(LocationController.class, () -> new LocationControllerImpl(mContext, getDependency(BG_LOOPER))); mProviders.put(RotationLockController.class, () -> new RotationLockControllerImpl(mContext)); mProviders.put(NetworkController.class, () -> new NetworkControllerImpl(mContext, getDependency(BG_LOOPER), getDependency(DeviceProvisionedController.class))); mProviders.put(ZenModeController.class, () -> new ZenModeControllerImpl(mContext, getDependency(MAIN_HANDLER))); mProviders.put(HotspotController.class, () -> new HotspotControllerImpl(mContext)); mProviders.put(CastController.class, () -> new CastControllerImpl(mContext)); mProviders.put(FlashlightController.class, () -> new FlashlightControllerImpl(mContext)); mProviders.put(KeyguardMonitor.class, () -> new KeyguardMonitorImpl(mContext)); mProviders.put(UserSwitcherController.class, () -> new UserSwitcherController(mContext, getDependency(KeyguardMonitor.class), getDependency(MAIN_HANDLER), getDependency(ActivityStarter.class))); mProviders.put(UserInfoController.class, () -> new UserInfoControllerImpl(mContext)); mProviders.put(BatteryController.class, () -> new BatteryControllerImpl(mContext)); mProviders.put(ColorDisplayController.class, () -> new ColorDisplayController(mContext)); mProviders.put(ManagedProfileController.class, () -> new ManagedProfileControllerImpl(mContext)); mProviders.put(NextAlarmController.class, () -> new NextAlarmControllerImpl(mContext)); mProviders.put(DataSaverController.class, () -> get(NetworkController.class).getDataSaverController()); mProviders.put(AccessibilityController.class, () -> new AccessibilityController(mContext)); mProviders.put(DeviceProvisionedController.class, () -> new DeviceProvisionedControllerImpl(mContext)); mProviders.put(PluginManager.class, () -> new PluginManagerImpl(mContext)); mProviders.put(AssistManager.class, () -> new AssistManager(getDependency(DeviceProvisionedController.class), mContext)); mProviders.put(SecurityController.class, () -> new SecurityControllerImpl(mContext)); mProviders.put(LeakDetector.class, LeakDetector::create); mProviders.put(LEAK_REPORT_EMAIL, () -> null); mProviders.put(LeakReporter.class, () -> new LeakReporter( mContext, getDependency(LeakDetector.class), getDependency(LEAK_REPORT_EMAIL))); mProviders.put( GarbageMonitor.class, () -> new GarbageMonitor( mContext, getDependency(BG_LOOPER), getDependency(LeakDetector.class), getDependency(LeakReporter.class))); mProviders.put(TunerService.class, () -> new TunerServiceImpl(mContext)); mProviders.put(StatusBarWindowManager.class, () -> new StatusBarWindowManager(mContext)); mProviders.put(DarkIconDispatcher.class, () -> new DarkIconDispatcherImpl(mContext)); mProviders.put(ConfigurationController.class, () -> new ConfigurationControllerImpl(mContext)); mProviders.put(StatusBarIconController.class, () -> new StatusBarIconControllerImpl(mContext)); mProviders.put(ScreenLifecycle.class, () -> new ScreenLifecycle()); mProviders.put(WakefulnessLifecycle.class, () -> new WakefulnessLifecycle()); mProviders.put(FragmentService.class, () -> new FragmentService(mContext)); mProviders.put(ExtensionController.class, () -> new ExtensionControllerImpl(mContext)); mProviders.put(PluginDependencyProvider.class, () -> new PluginDependencyProvider(get(PluginManager.class))); mProviders.put(LocalBluetoothManager.class, () -> LocalBluetoothManager.getInstance(mContext, null)); mProviders.put(VolumeDialogController.class, () -> new VolumeDialogControllerImpl(mContext)); mProviders.put(MetricsLogger.class, () -> new MetricsLogger()); mProviders.put(AccessibilityManagerWrapper.class, () -> new AccessibilityManagerWrapper(mContext)); // Creating a new instance will trigger color extraction. // Thankfully this only happens once - during boot - and WallpaperManagerService // loads colors from cache. mProviders.put(SysuiColorExtractor.class, () -> new SysuiColorExtractor(mContext)); mProviders.put(TunablePaddingService.class, () -> new TunablePaddingService()); mProviders.put(ForegroundServiceController.class, () -> new ForegroundServiceControllerImpl(mContext)); mProviders.put(UiOffloadThread.class, UiOffloadThread::new); mProviders.put(PowerUI.WarningsUI.class, () -> new PowerNotificationWarnings(mContext)); mProviders.put(IconLogger.class, () -> new IconLoggerImpl(mContext, getDependency(BG_LOOPER), getDependency(MetricsLogger.class))); mProviders.put(LightBarController.class, () -> new LightBarController(mContext)); mProviders.put(IWindowManager.class, () -> WindowManagerGlobal.getWindowManagerService()); mProviders.put(OverviewProxyService.class, () -> new OverviewProxyService(mContext)); mProviders.put(EnhancedEstimates.class, () -> new EnhancedEstimatesImpl()); mProviders.put(AppOpsListener.class, () -> new AppOpsListener(mContext)); mProviders.put(VibratorHelper.class, () -> new VibratorHelper(mContext)); mProviders.put(IStatusBarService.class, () -> IStatusBarService.Stub.asInterface( ServiceManager.getService(Context.STATUS_BAR_SERVICE))); // Put all dependencies above here so the factory can override them if it wants. SystemUIFactory.getInstance().injectDependencies(mProviders, mContext); sDependency = this; } @Override public synchronized void dump(FileDescriptor fd, PrintWriter pw, String[] args) { super.dump(fd, pw, args); pw.println("Dumping existing controllers:"); mDependencies.values().stream().filter(obj -> obj instanceof Dumpable) .forEach(o -> ((Dumpable) o).dump(fd, pw, args)); } @Override protected synchronized void onConfigurationChanged(Configuration newConfig) { super.onConfigurationChanged(newConfig); mDependencies.values().stream().filter(obj -> obj instanceof ConfigurationChangedReceiver) .forEach(o -> ((ConfigurationChangedReceiver) o).onConfigurationChanged(newConfig)); } protected final T getDependency(Class cls) { return getDependencyInner(cls); } protected final T getDependency(DependencyKey key) { return getDependencyInner(key); } private synchronized T getDependencyInner(Object key) { @SuppressWarnings("unchecked") T obj = (T) mDependencies.get(key); if (obj == null) { obj = createDependency(key); mDependencies.put(key, obj); } return obj; } @VisibleForTesting protected T createDependency(Object cls) { Preconditions.checkArgument(cls instanceof DependencyKey || cls instanceof Class); @SuppressWarnings("unchecked") DependencyProvider provider = mProviders.get(cls); if (provider == null) { throw new IllegalArgumentException("Unsupported dependency " + cls + ". " + mProviders.size() + " providers known."); } return provider.createDependency(); } private static Dependency sDependency; public interface DependencyProvider { T createDependency(); } private void destroyDependency(Class cls, Consumer destroy) { T dep = (T) mDependencies.remove(cls); if (dep != null && destroy != null) { destroy.accept(dep); } } /** * Used in separate processes (like tuner settings) to init the dependencies. */ public static void initDependencies(Context context) { if (sDependency != null) return; Dependency d = new Dependency(); d.mContext = context; d.mComponents = new HashMap<>(); d.start(); } /** * Used in separate process teardown to ensure the context isn't leaked. * * TODO: Remove once PreferenceFragment doesn't reference getActivity() * anymore and these context hacks are no longer needed. */ public static void clearDependencies() { sDependency = null; } /** * Checks to see if a dependency is instantiated, if it is it removes it from * the cache and calls the destroy callback. */ public static void destroy(Class cls, Consumer destroy) { sDependency.destroyDependency(cls, destroy); } public static T get(Class cls) { return sDependency.getDependency(cls); } public static T get(DependencyKey cls) { return sDependency.getDependency(cls); } public static final class DependencyKey { private final String mDisplayName; public DependencyKey(String displayName) { mDisplayName = displayName; } @Override public String toString() { return mDisplayName; } } }