Jinsuk Kim | 2918e9e | 2014-05-20 16:45:45 +0900 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2014 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
| 16 | |
| 17 | package com.android.server.hdmi; |
| 18 | |
Jinsuk Kim | c0c20d0 | 2014-07-04 14:34:31 +0900 | [diff] [blame] | 19 | import android.hardware.hdmi.HdmiControlManager; |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 20 | import android.hardware.hdmi.HdmiDeviceInfo; |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 21 | import android.hardware.hdmi.IHdmiControlCallback; |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 22 | import android.os.PowerManager; |
| 23 | import android.os.PowerManager.WakeLock; |
Jinsuk Kim | af2acf0 | 2014-07-11 18:43:04 +0900 | [diff] [blame] | 24 | import android.os.SystemProperties; |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 25 | import android.provider.Settings.Global; |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 26 | import android.util.Slog; |
Jinsuk Kim | 2918e9e | 2014-05-20 16:45:45 +0900 | [diff] [blame] | 27 | |
Amy | aefab64 | 2018-08-22 19:10:14 -0700 | [diff] [blame] | 28 | import com.android.internal.annotations.VisibleForTesting; |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 29 | import com.android.internal.app.LocalePicker; |
| 30 | import com.android.internal.app.LocalePicker.LocaleInfo; |
Terry Heo | 959d2db | 2014-08-28 16:45:41 +0900 | [diff] [blame] | 31 | import com.android.internal.util.IndentingPrintWriter; |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 32 | import com.android.server.hdmi.HdmiAnnotations.ServiceThreadOnly; |
| 33 | |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 34 | import java.io.UnsupportedEncodingException; |
| 35 | import java.util.List; |
Jinsuk Kim | 8ea452e | 2015-07-10 13:01:06 +0900 | [diff] [blame] | 36 | import java.util.Locale; |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 37 | |
Jinsuk Kim | 2918e9e | 2014-05-20 16:45:45 +0900 | [diff] [blame] | 38 | /** |
| 39 | * Represent a logical device of type Playback residing in Android system. |
| 40 | */ |
Amy | 645c361 | 2018-09-25 10:48:19 -0700 | [diff] [blame] | 41 | public class HdmiCecLocalDevicePlayback extends HdmiCecLocalDeviceSource { |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 42 | private static final String TAG = "HdmiCecLocalDevicePlayback"; |
Jinsuk Kim | 2918e9e | 2014-05-20 16:45:45 +0900 | [diff] [blame] | 43 | |
Jinsuk Kim | 659c486 | 2015-05-07 12:12:55 +0900 | [diff] [blame] | 44 | private static final boolean WAKE_ON_HOTPLUG = |
| 45 | SystemProperties.getBoolean(Constants.PROPERTY_WAKE_ON_HOTPLUG, true); |
| 46 | |
Donghyun Cho | a09256c | 2016-03-11 17:35:37 +0900 | [diff] [blame] | 47 | private static final boolean SET_MENU_LANGUAGE = |
| 48 | SystemProperties.getBoolean(Constants.PROPERTY_SET_MENU_LANGUAGE, false); |
| 49 | |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 50 | // Used to keep the device awake while it is the active source. For devices that |
| 51 | // cannot wake up via CEC commands, this address the inconvenience of having to |
Jinsuk Kim | bad8393 | 2015-02-10 07:10:40 +0900 | [diff] [blame] | 52 | // turn them on. True by default, and can be disabled (i.e. device can go to sleep |
| 53 | // in active device status) by explicitly setting the system property |
| 54 | // persist.sys.hdmi.keep_awake to false. |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 55 | // Lazily initialized - should call getWakeLock() to get the instance. |
Jinsuk Kim | bad8393 | 2015-02-10 07:10:40 +0900 | [diff] [blame] | 56 | private ActiveWakeLock mWakeLock; |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 57 | |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 58 | // If true, turn off TV upon standby. False by default. |
| 59 | private boolean mAutoTvOff; |
| 60 | |
Amy | aefab64 | 2018-08-22 19:10:14 -0700 | [diff] [blame] | 61 | // Local active port number used for Routing Control. |
| 62 | // Default 0 means HOME is the current active path. Temp solution only. |
| 63 | // TODO(amyjojo): adding system constants for input ports to TIF mapping. |
| 64 | private int mLocalActivePath = 0; |
| 65 | |
Jungshik Jang | 3ee6572 | 2014-06-03 16:22:30 +0900 | [diff] [blame] | 66 | HdmiCecLocalDevicePlayback(HdmiControlService service) { |
Jungshik Jang | 61f4fbd | 2014-08-06 19:21:12 +0900 | [diff] [blame] | 67 | super(service, HdmiDeviceInfo.DEVICE_PLAYBACK); |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 68 | |
| 69 | mAutoTvOff = mService.readBooleanSetting(Global.HDMI_CONTROL_AUTO_DEVICE_OFF_ENABLED, false); |
| 70 | |
| 71 | // The option is false by default. Update settings db as well to have the right |
| 72 | // initial setting on UI. |
| 73 | mService.writeBooleanSetting(Global.HDMI_CONTROL_AUTO_DEVICE_OFF_ENABLED, mAutoTvOff); |
Jungshik Jang | 8b308d9 | 2014-05-29 21:52:28 +0900 | [diff] [blame] | 74 | } |
| 75 | |
| 76 | @Override |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 77 | @ServiceThreadOnly |
Yuncheol Heo | fc44e4e | 2014-08-04 19:41:09 +0900 | [diff] [blame] | 78 | protected void onAddressAllocated(int logicalAddress, int reason) { |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 79 | assertRunOnServiceThread(); |
Amy | b9d7f43 | 2018-11-30 15:08:30 -0800 | [diff] [blame] | 80 | if (reason == mService.INITIATED_BY_ENABLE_CEC) { |
| 81 | mService.setAndBroadcastActiveSource(mService.getPhysicalAddress(), |
| 82 | getDeviceInfo().getDeviceType(), Constants.ADDR_BROADCAST); |
| 83 | } |
Jungshik Jang | 3ee6572 | 2014-06-03 16:22:30 +0900 | [diff] [blame] | 84 | mService.sendCecCommand(HdmiCecMessageBuilder.buildReportPhysicalAddressCommand( |
| 85 | mAddress, mService.getPhysicalAddress(), mDeviceType)); |
Jinsuk Kim | ad515e8 | 2014-10-07 07:43:44 +0900 | [diff] [blame] | 86 | mService.sendCecCommand(HdmiCecMessageBuilder.buildDeviceVendorIdCommand( |
| 87 | mAddress, mService.getVendorId())); |
Jinsuk Kim | 6f87b4e | 2014-10-10 14:40:29 +0900 | [diff] [blame] | 88 | startQueuedActions(); |
Jinsuk Kim | 2918e9e | 2014-05-20 16:45:45 +0900 | [diff] [blame] | 89 | } |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 90 | |
Jinsuk Kim | af2acf0 | 2014-07-11 18:43:04 +0900 | [diff] [blame] | 91 | @Override |
| 92 | @ServiceThreadOnly |
| 93 | protected int getPreferredAddress() { |
| 94 | assertRunOnServiceThread(); |
| 95 | return SystemProperties.getInt(Constants.PROPERTY_PREFERRED_ADDRESS_PLAYBACK, |
| 96 | Constants.ADDR_UNREGISTERED); |
| 97 | } |
| 98 | |
| 99 | @Override |
| 100 | @ServiceThreadOnly |
| 101 | protected void setPreferredAddress(int addr) { |
| 102 | assertRunOnServiceThread(); |
Amy | 59c06c1 | 2019-01-18 15:35:15 -0800 | [diff] [blame] | 103 | mService.writeStringSystemProperty(Constants.PROPERTY_PREFERRED_ADDRESS_PLAYBACK, |
Jinsuk Kim | af2acf0 | 2014-07-11 18:43:04 +0900 | [diff] [blame] | 104 | String.valueOf(addr)); |
| 105 | } |
| 106 | |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 107 | @ServiceThreadOnly |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 108 | void queryDisplayStatus(IHdmiControlCallback callback) { |
| 109 | assertRunOnServiceThread(); |
Jinsuk Kim | cb80287 | 2015-10-13 08:22:09 +0900 | [diff] [blame] | 110 | List<DevicePowerStatusAction> actions = getActions(DevicePowerStatusAction.class); |
| 111 | if (!actions.isEmpty()) { |
| 112 | Slog.i(TAG, "queryDisplayStatus already in progress"); |
| 113 | actions.get(0).addCallback(callback); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 114 | return; |
| 115 | } |
Jinsuk Kim | cb80287 | 2015-10-13 08:22:09 +0900 | [diff] [blame] | 116 | DevicePowerStatusAction action = DevicePowerStatusAction.create(this, Constants.ADDR_TV, |
| 117 | callback); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 118 | if (action == null) { |
| 119 | Slog.w(TAG, "Cannot initiate queryDisplayStatus"); |
Jinsuk Kim | c0c20d0 | 2014-07-04 14:34:31 +0900 | [diff] [blame] | 120 | invokeCallback(callback, HdmiControlManager.RESULT_EXCEPTION); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 121 | return; |
| 122 | } |
| 123 | addAndStartAction(action); |
| 124 | } |
| 125 | |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 126 | @Override |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 127 | @ServiceThreadOnly |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 128 | void onHotplug(int portId, boolean connected) { |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 129 | assertRunOnServiceThread(); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 130 | mCecMessageCache.flushAll(); |
Yuncheol Heo | 89ec14e | 2014-09-16 15:53:59 +0900 | [diff] [blame] | 131 | // We'll not clear mIsActiveSource on the hotplug event to pass CETC 11.2.2-2 ~ 3. |
Jinsuk Kim | 659c486 | 2015-05-07 12:12:55 +0900 | [diff] [blame] | 132 | if (WAKE_ON_HOTPLUG && connected && mService.isPowerStandbyOrTransient()) { |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 133 | mService.wakeUp(); |
| 134 | } |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 135 | if (!connected) { |
| 136 | getWakeLock().release(); |
| 137 | } |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 138 | } |
| 139 | |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 140 | @Override |
| 141 | @ServiceThreadOnly |
| 142 | protected void onStandby(boolean initiatedByCec, int standbyAction) { |
| 143 | assertRunOnServiceThread(); |
Donghyun Cho | 9ccff51 | 2016-04-01 14:31:11 +0900 | [diff] [blame] | 144 | if (!mService.isControlEnabled() || initiatedByCec || !mAutoTvOff) { |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 145 | return; |
| 146 | } |
| 147 | switch (standbyAction) { |
| 148 | case HdmiControlService.STANDBY_SCREEN_OFF: |
Donghyun Cho | 9ccff51 | 2016-04-01 14:31:11 +0900 | [diff] [blame] | 149 | mService.sendCecCommand( |
| 150 | HdmiCecMessageBuilder.buildStandby(mAddress, Constants.ADDR_TV)); |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 151 | break; |
| 152 | case HdmiControlService.STANDBY_SHUTDOWN: |
| 153 | // ACTION_SHUTDOWN is taken as a signal to power off all the devices. |
| 154 | mService.sendCecCommand( |
| 155 | HdmiCecMessageBuilder.buildStandby(mAddress, Constants.ADDR_BROADCAST)); |
| 156 | break; |
| 157 | } |
| 158 | } |
| 159 | |
| 160 | @Override |
| 161 | @ServiceThreadOnly |
| 162 | void setAutoDeviceOff(boolean enabled) { |
| 163 | assertRunOnServiceThread(); |
| 164 | mAutoTvOff = enabled; |
| 165 | } |
| 166 | |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 167 | @ServiceThreadOnly |
Amy | 645c361 | 2018-09-25 10:48:19 -0700 | [diff] [blame] | 168 | @VisibleForTesting |
Amy | 9a59d9c | 2018-08-31 13:47:24 -0700 | [diff] [blame] | 169 | void setIsActiveSource(boolean on) { |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 170 | assertRunOnServiceThread(); |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 171 | mIsActiveSource = on; |
| 172 | if (on) { |
| 173 | getWakeLock().acquire(); |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 174 | } else { |
| 175 | getWakeLock().release(); |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 176 | } |
| 177 | } |
| 178 | |
| 179 | @ServiceThreadOnly |
Jinsuk Kim | bad8393 | 2015-02-10 07:10:40 +0900 | [diff] [blame] | 180 | private ActiveWakeLock getWakeLock() { |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 181 | assertRunOnServiceThread(); |
| 182 | if (mWakeLock == null) { |
Jinsuk Kim | bad8393 | 2015-02-10 07:10:40 +0900 | [diff] [blame] | 183 | if (SystemProperties.getBoolean(Constants.PROPERTY_KEEP_AWAKE, true)) { |
| 184 | mWakeLock = new SystemWakeLock(); |
| 185 | } else { |
| 186 | // Create a dummy lock object that doesn't do anything about wake lock, |
| 187 | // hence allows the device to go to sleep even if it's the active source. |
| 188 | mWakeLock = new ActiveWakeLock() { |
| 189 | @Override |
| 190 | public void acquire() { } |
| 191 | @Override |
| 192 | public void release() { } |
| 193 | @Override |
| 194 | public boolean isHeld() { return false; } |
| 195 | }; |
| 196 | HdmiLogger.debug("No wakelock is used to keep the display on."); |
| 197 | } |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 198 | } |
| 199 | return mWakeLock; |
| 200 | } |
| 201 | |
| 202 | @Override |
| 203 | protected boolean canGoToStandby() { |
| 204 | return !getWakeLock().isHeld(); |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 205 | } |
| 206 | |
Jinsuk Kim | 9b8507c | 2015-03-24 16:55:01 +0900 | [diff] [blame] | 207 | @ServiceThreadOnly |
| 208 | protected boolean handleUserControlPressed(HdmiCecMessage message) { |
| 209 | assertRunOnServiceThread(); |
| 210 | wakeUpIfActiveSource(); |
| 211 | return super.handleUserControlPressed(message); |
| 212 | } |
| 213 | |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 214 | @Override |
Amy | 3403742 | 2018-09-06 13:21:08 -0700 | [diff] [blame] | 215 | protected void wakeUpIfActiveSource() { |
Jinsuk Kim | 9b8507c | 2015-03-24 16:55:01 +0900 | [diff] [blame] | 216 | if (!mIsActiveSource) { |
| 217 | return; |
| 218 | } |
| 219 | // Wake up the device if the power is in standby mode, or its screen is off - |
| 220 | // which can happen if the device is holding a partial lock. |
| 221 | if (mService.isPowerStandbyOrTransient() || !mService.getPowerManager().isScreenOn()) { |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 222 | mService.wakeUp(); |
| 223 | } |
| 224 | } |
| 225 | |
Amy | 225d55a | 2018-09-06 11:03:51 -0700 | [diff] [blame] | 226 | @Override |
| 227 | protected void maySendActiveSource(int dest) { |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 228 | if (mIsActiveSource) { |
| 229 | mService.sendCecCommand(HdmiCecMessageBuilder.buildActiveSource( |
| 230 | mAddress, mService.getPhysicalAddress())); |
Yuncheol Heo | 8ecb721 | 2014-10-27 17:29:42 +0900 | [diff] [blame] | 231 | // Always reports menu-status active to receive RCP. |
| 232 | mService.sendCecCommand(HdmiCecMessageBuilder.buildReportMenuStatus( |
| 233 | mAddress, dest, Constants.MENU_STATE_ACTIVATED)); |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 234 | } |
| 235 | } |
| 236 | |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 237 | @ServiceThreadOnly |
| 238 | protected boolean handleSetMenuLanguage(HdmiCecMessage message) { |
| 239 | assertRunOnServiceThread(); |
Donghyun Cho | a09256c | 2016-03-11 17:35:37 +0900 | [diff] [blame] | 240 | if (!SET_MENU_LANGUAGE) { |
| 241 | return false; |
| 242 | } |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 243 | |
| 244 | try { |
| 245 | String iso3Language = new String(message.getParams(), 0, 3, "US-ASCII"); |
Jinsuk Kim | 8ea452e | 2015-07-10 13:01:06 +0900 | [diff] [blame] | 246 | Locale currentLocale = mService.getContext().getResources().getConfiguration().locale; |
| 247 | if (currentLocale.getISO3Language().equals(iso3Language)) { |
| 248 | // Do not switch language if the new language is the same as the current one. |
| 249 | // This helps avoid accidental country variant switching from en_US to en_AU |
| 250 | // due to the limitation of CEC. See the warning below. |
| 251 | return true; |
| 252 | } |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 253 | |
| 254 | // Don't use Locale.getAvailableLocales() since it returns a locale |
| 255 | // which is not available on Settings. |
| 256 | final List<LocaleInfo> localeInfos = LocalePicker.getAllAssetLocales( |
| 257 | mService.getContext(), false); |
| 258 | for (LocaleInfo localeInfo : localeInfos) { |
| 259 | if (localeInfo.getLocale().getISO3Language().equals(iso3Language)) { |
| 260 | // WARNING: CEC adopts ISO/FDIS-2 for language code, while Android requires |
| 261 | // additional country variant to pinpoint the locale. This keeps the right |
| 262 | // locale from being chosen. 'eng' in the CEC command, for instance, |
| 263 | // will always be mapped to en-AU among other variants like en-US, en-GB, |
| 264 | // an en-IN, which may not be the expected one. |
| 265 | LocalePicker.updateLocale(localeInfo.getLocale()); |
| 266 | return true; |
| 267 | } |
| 268 | } |
| 269 | Slog.w(TAG, "Can't handle <Set Menu Language> of " + iso3Language); |
| 270 | return false; |
| 271 | } catch (UnsupportedEncodingException e) { |
Donghyun Cho | a09256c | 2016-03-11 17:35:37 +0900 | [diff] [blame] | 272 | Slog.w(TAG, "Can't handle <Set Menu Language>", e); |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 273 | return false; |
| 274 | } |
| 275 | } |
| 276 | |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 277 | @Override |
Donghyun Cho | 7609bc3 | 2016-12-02 15:31:52 +0900 | [diff] [blame] | 278 | protected int findKeyReceiverAddress() { |
| 279 | return Constants.ADDR_TV; |
| 280 | } |
| 281 | |
| 282 | @Override |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 283 | @ServiceThreadOnly |
Jungshik Jang | 4fc1d10 | 2014-07-09 19:24:50 +0900 | [diff] [blame] | 284 | protected void disableDevice(boolean initiatedByCec, PendingActionClearedCallback callback) { |
| 285 | super.disableDevice(initiatedByCec, callback); |
| 286 | |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 287 | assertRunOnServiceThread(); |
| 288 | if (!initiatedByCec && mIsActiveSource) { |
| 289 | mService.sendCecCommand(HdmiCecMessageBuilder.buildInactiveSource( |
| 290 | mAddress, mService.getPhysicalAddress())); |
| 291 | } |
Amy | 9a59d9c | 2018-08-31 13:47:24 -0700 | [diff] [blame] | 292 | setIsActiveSource(false); |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 293 | checkIfPendingActionsCleared(); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 294 | } |
Terry Heo | 959d2db | 2014-08-28 16:45:41 +0900 | [diff] [blame] | 295 | |
Amy | aefab64 | 2018-08-22 19:10:14 -0700 | [diff] [blame] | 296 | private void routeToPort(int portId) { |
| 297 | // TODO(AMYJOJO): route to specific input of the port |
| 298 | mLocalActivePath = portId; |
| 299 | } |
| 300 | |
| 301 | @VisibleForTesting |
| 302 | protected int getLocalActivePath() { |
| 303 | return mLocalActivePath; |
| 304 | } |
| 305 | |
Terry Heo | 959d2db | 2014-08-28 16:45:41 +0900 | [diff] [blame] | 306 | @Override |
| 307 | protected void dump(final IndentingPrintWriter pw) { |
| 308 | super.dump(pw); |
| 309 | pw.println("mIsActiveSource: " + mIsActiveSource); |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 310 | pw.println("mAutoTvOff:" + mAutoTvOff); |
Terry Heo | 959d2db | 2014-08-28 16:45:41 +0900 | [diff] [blame] | 311 | } |
Jinsuk Kim | bad8393 | 2015-02-10 07:10:40 +0900 | [diff] [blame] | 312 | |
| 313 | // Wrapper interface over PowerManager.WakeLock |
| 314 | private interface ActiveWakeLock { |
| 315 | void acquire(); |
| 316 | void release(); |
| 317 | boolean isHeld(); |
| 318 | } |
| 319 | |
| 320 | private class SystemWakeLock implements ActiveWakeLock { |
| 321 | private final WakeLock mWakeLock; |
| 322 | public SystemWakeLock() { |
| 323 | mWakeLock = mService.getPowerManager().newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, TAG); |
| 324 | mWakeLock.setReferenceCounted(false); |
| 325 | } |
| 326 | |
| 327 | @Override |
| 328 | public void acquire() { |
| 329 | mWakeLock.acquire(); |
| 330 | HdmiLogger.debug("active source: %b. Wake lock acquired", mIsActiveSource); |
| 331 | } |
| 332 | |
| 333 | @Override |
| 334 | public void release() { |
| 335 | mWakeLock.release(); |
| 336 | HdmiLogger.debug("Wake lock released"); |
| 337 | } |
| 338 | |
| 339 | @Override |
| 340 | public boolean isHeld() { |
| 341 | return mWakeLock.isHeld(); |
| 342 | } |
| 343 | } |
Jinsuk Kim | 84659ed | 2014-10-06 23:48:19 +0000 | [diff] [blame] | 344 | } |