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; |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 24 | import android.os.RemoteException; |
Jinsuk Kim | af2acf0 | 2014-07-11 18:43:04 +0900 | [diff] [blame] | 25 | import android.os.SystemProperties; |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 26 | import android.provider.Settings.Global; |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 27 | import android.util.Slog; |
Jinsuk Kim | 2918e9e | 2014-05-20 16:45:45 +0900 | [diff] [blame] | 28 | |
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 | cb80287 | 2015-10-13 08:22:09 +0900 | [diff] [blame] | 38 | import java.util.List; |
| 39 | |
Jinsuk Kim | 2918e9e | 2014-05-20 16:45:45 +0900 | [diff] [blame] | 40 | /** |
| 41 | * Represent a logical device of type Playback residing in Android system. |
| 42 | */ |
| 43 | final class HdmiCecLocalDevicePlayback extends HdmiCecLocalDevice { |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 44 | private static final String TAG = "HdmiCecLocalDevicePlayback"; |
Jinsuk Kim | 2918e9e | 2014-05-20 16:45:45 +0900 | [diff] [blame] | 45 | |
Jinsuk Kim | 659c486 | 2015-05-07 12:12:55 +0900 | [diff] [blame] | 46 | private static final boolean WAKE_ON_HOTPLUG = |
| 47 | SystemProperties.getBoolean(Constants.PROPERTY_WAKE_ON_HOTPLUG, true); |
| 48 | |
Donghyun Cho | a09256c | 2016-03-11 17:35:37 +0900 | [diff] [blame] | 49 | private static final boolean SET_MENU_LANGUAGE = |
| 50 | SystemProperties.getBoolean(Constants.PROPERTY_SET_MENU_LANGUAGE, false); |
| 51 | |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 52 | private boolean mIsActiveSource = false; |
| 53 | |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 54 | // Used to keep the device awake while it is the active source. For devices that |
| 55 | // 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] | 56 | // turn them on. True by default, and can be disabled (i.e. device can go to sleep |
| 57 | // in active device status) by explicitly setting the system property |
| 58 | // persist.sys.hdmi.keep_awake to false. |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 59 | // Lazily initialized - should call getWakeLock() to get the instance. |
Jinsuk Kim | bad8393 | 2015-02-10 07:10:40 +0900 | [diff] [blame] | 60 | private ActiveWakeLock mWakeLock; |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 61 | |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 62 | // If true, turn off TV upon standby. False by default. |
| 63 | private boolean mAutoTvOff; |
| 64 | |
Jungshik Jang | 3ee6572 | 2014-06-03 16:22:30 +0900 | [diff] [blame] | 65 | HdmiCecLocalDevicePlayback(HdmiControlService service) { |
Jungshik Jang | 61f4fbd | 2014-08-06 19:21:12 +0900 | [diff] [blame] | 66 | super(service, HdmiDeviceInfo.DEVICE_PLAYBACK); |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 67 | |
| 68 | mAutoTvOff = mService.readBooleanSetting(Global.HDMI_CONTROL_AUTO_DEVICE_OFF_ENABLED, false); |
| 69 | |
| 70 | // The option is false by default. Update settings db as well to have the right |
| 71 | // initial setting on UI. |
| 72 | mService.writeBooleanSetting(Global.HDMI_CONTROL_AUTO_DEVICE_OFF_ENABLED, mAutoTvOff); |
Jungshik Jang | 8b308d9 | 2014-05-29 21:52:28 +0900 | [diff] [blame] | 73 | } |
| 74 | |
| 75 | @Override |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 76 | @ServiceThreadOnly |
Yuncheol Heo | fc44e4e | 2014-08-04 19:41:09 +0900 | [diff] [blame] | 77 | protected void onAddressAllocated(int logicalAddress, int reason) { |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 78 | assertRunOnServiceThread(); |
Jungshik Jang | 3ee6572 | 2014-06-03 16:22:30 +0900 | [diff] [blame] | 79 | mService.sendCecCommand(HdmiCecMessageBuilder.buildReportPhysicalAddressCommand( |
| 80 | mAddress, mService.getPhysicalAddress(), mDeviceType)); |
Jinsuk Kim | ad515e8 | 2014-10-07 07:43:44 +0900 | [diff] [blame] | 81 | mService.sendCecCommand(HdmiCecMessageBuilder.buildDeviceVendorIdCommand( |
| 82 | mAddress, mService.getVendorId())); |
Jinsuk Kim | 6f87b4e | 2014-10-10 14:40:29 +0900 | [diff] [blame] | 83 | startQueuedActions(); |
Jinsuk Kim | 2918e9e | 2014-05-20 16:45:45 +0900 | [diff] [blame] | 84 | } |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 85 | |
Jinsuk Kim | af2acf0 | 2014-07-11 18:43:04 +0900 | [diff] [blame] | 86 | @Override |
| 87 | @ServiceThreadOnly |
| 88 | protected int getPreferredAddress() { |
| 89 | assertRunOnServiceThread(); |
| 90 | return SystemProperties.getInt(Constants.PROPERTY_PREFERRED_ADDRESS_PLAYBACK, |
| 91 | Constants.ADDR_UNREGISTERED); |
| 92 | } |
| 93 | |
| 94 | @Override |
| 95 | @ServiceThreadOnly |
| 96 | protected void setPreferredAddress(int addr) { |
| 97 | assertRunOnServiceThread(); |
| 98 | SystemProperties.set(Constants.PROPERTY_PREFERRED_ADDRESS_PLAYBACK, |
| 99 | String.valueOf(addr)); |
| 100 | } |
| 101 | |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 102 | @ServiceThreadOnly |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 103 | void oneTouchPlay(IHdmiControlCallback callback) { |
| 104 | assertRunOnServiceThread(); |
Jinsuk Kim | cb80287 | 2015-10-13 08:22:09 +0900 | [diff] [blame] | 105 | List<OneTouchPlayAction> actions = getActions(OneTouchPlayAction.class); |
| 106 | if (!actions.isEmpty()) { |
| 107 | Slog.i(TAG, "oneTouchPlay already in progress"); |
| 108 | actions.get(0).addCallback(callback); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 109 | return; |
| 110 | } |
Jinsuk Kim | c0c20d0 | 2014-07-04 14:34:31 +0900 | [diff] [blame] | 111 | OneTouchPlayAction action = OneTouchPlayAction.create(this, Constants.ADDR_TV, |
| 112 | callback); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 113 | if (action == null) { |
| 114 | Slog.w(TAG, "Cannot initiate oneTouchPlay"); |
Jinsuk Kim | c0c20d0 | 2014-07-04 14:34:31 +0900 | [diff] [blame] | 115 | invokeCallback(callback, HdmiControlManager.RESULT_EXCEPTION); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 116 | return; |
| 117 | } |
| 118 | addAndStartAction(action); |
| 119 | } |
| 120 | |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 121 | @ServiceThreadOnly |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 122 | void queryDisplayStatus(IHdmiControlCallback callback) { |
| 123 | assertRunOnServiceThread(); |
Jinsuk Kim | cb80287 | 2015-10-13 08:22:09 +0900 | [diff] [blame] | 124 | List<DevicePowerStatusAction> actions = getActions(DevicePowerStatusAction.class); |
| 125 | if (!actions.isEmpty()) { |
| 126 | Slog.i(TAG, "queryDisplayStatus already in progress"); |
| 127 | actions.get(0).addCallback(callback); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 128 | return; |
| 129 | } |
Jinsuk Kim | cb80287 | 2015-10-13 08:22:09 +0900 | [diff] [blame] | 130 | DevicePowerStatusAction action = DevicePowerStatusAction.create(this, Constants.ADDR_TV, |
| 131 | callback); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 132 | if (action == null) { |
| 133 | Slog.w(TAG, "Cannot initiate queryDisplayStatus"); |
Jinsuk Kim | c0c20d0 | 2014-07-04 14:34:31 +0900 | [diff] [blame] | 134 | invokeCallback(callback, HdmiControlManager.RESULT_EXCEPTION); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 135 | return; |
| 136 | } |
| 137 | addAndStartAction(action); |
| 138 | } |
| 139 | |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 140 | @ServiceThreadOnly |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 141 | private void invokeCallback(IHdmiControlCallback callback, int result) { |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 142 | assertRunOnServiceThread(); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 143 | try { |
| 144 | callback.onComplete(result); |
| 145 | } catch (RemoteException e) { |
| 146 | Slog.e(TAG, "Invoking callback failed:" + e); |
| 147 | } |
| 148 | } |
| 149 | |
| 150 | @Override |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 151 | @ServiceThreadOnly |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 152 | void onHotplug(int portId, boolean connected) { |
Jungshik Jang | a5b7414 | 2014-06-23 18:03:10 +0900 | [diff] [blame] | 153 | assertRunOnServiceThread(); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 154 | mCecMessageCache.flushAll(); |
Yuncheol Heo | 89ec14e | 2014-09-16 15:53:59 +0900 | [diff] [blame] | 155 | // 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] | 156 | if (WAKE_ON_HOTPLUG && connected && mService.isPowerStandbyOrTransient()) { |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 157 | mService.wakeUp(); |
| 158 | } |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 159 | if (!connected) { |
| 160 | getWakeLock().release(); |
| 161 | } |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 162 | } |
| 163 | |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 164 | @Override |
| 165 | @ServiceThreadOnly |
| 166 | protected void onStandby(boolean initiatedByCec, int standbyAction) { |
| 167 | assertRunOnServiceThread(); |
| 168 | if (!mService.isControlEnabled() || initiatedByCec) { |
| 169 | return; |
| 170 | } |
| 171 | switch (standbyAction) { |
| 172 | case HdmiControlService.STANDBY_SCREEN_OFF: |
| 173 | if (mAutoTvOff) { |
| 174 | mService.sendCecCommand( |
| 175 | HdmiCecMessageBuilder.buildStandby(mAddress, Constants.ADDR_TV)); |
| 176 | } |
| 177 | break; |
| 178 | case HdmiControlService.STANDBY_SHUTDOWN: |
| 179 | // ACTION_SHUTDOWN is taken as a signal to power off all the devices. |
| 180 | mService.sendCecCommand( |
| 181 | HdmiCecMessageBuilder.buildStandby(mAddress, Constants.ADDR_BROADCAST)); |
| 182 | break; |
| 183 | } |
| 184 | } |
| 185 | |
| 186 | @Override |
| 187 | @ServiceThreadOnly |
| 188 | void setAutoDeviceOff(boolean enabled) { |
| 189 | assertRunOnServiceThread(); |
| 190 | mAutoTvOff = enabled; |
| 191 | } |
| 192 | |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 193 | @ServiceThreadOnly |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 194 | void setActiveSource(boolean on) { |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 195 | assertRunOnServiceThread(); |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 196 | mIsActiveSource = on; |
| 197 | if (on) { |
| 198 | getWakeLock().acquire(); |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 199 | } else { |
| 200 | getWakeLock().release(); |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 201 | } |
| 202 | } |
| 203 | |
| 204 | @ServiceThreadOnly |
Jinsuk Kim | bad8393 | 2015-02-10 07:10:40 +0900 | [diff] [blame] | 205 | private ActiveWakeLock getWakeLock() { |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 206 | assertRunOnServiceThread(); |
| 207 | if (mWakeLock == null) { |
Jinsuk Kim | bad8393 | 2015-02-10 07:10:40 +0900 | [diff] [blame] | 208 | if (SystemProperties.getBoolean(Constants.PROPERTY_KEEP_AWAKE, true)) { |
| 209 | mWakeLock = new SystemWakeLock(); |
| 210 | } else { |
| 211 | // Create a dummy lock object that doesn't do anything about wake lock, |
| 212 | // hence allows the device to go to sleep even if it's the active source. |
| 213 | mWakeLock = new ActiveWakeLock() { |
| 214 | @Override |
| 215 | public void acquire() { } |
| 216 | @Override |
| 217 | public void release() { } |
| 218 | @Override |
| 219 | public boolean isHeld() { return false; } |
| 220 | }; |
| 221 | HdmiLogger.debug("No wakelock is used to keep the display on."); |
| 222 | } |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 223 | } |
| 224 | return mWakeLock; |
| 225 | } |
| 226 | |
| 227 | @Override |
| 228 | protected boolean canGoToStandby() { |
| 229 | return !getWakeLock().isHeld(); |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 230 | } |
| 231 | |
| 232 | @Override |
| 233 | @ServiceThreadOnly |
| 234 | protected boolean handleActiveSource(HdmiCecMessage message) { |
| 235 | assertRunOnServiceThread(); |
| 236 | int physicalAddress = HdmiUtils.twoBytesToInt(message.getParams()); |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 237 | mayResetActiveSource(physicalAddress); |
| 238 | return true; // Broadcast message. |
| 239 | } |
| 240 | |
| 241 | private void mayResetActiveSource(int physicalAddress) { |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 242 | if (physicalAddress != mService.getPhysicalAddress()) { |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 243 | setActiveSource(false); |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 244 | } |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 245 | } |
| 246 | |
Jinsuk Kim | 9b8507c | 2015-03-24 16:55:01 +0900 | [diff] [blame] | 247 | @ServiceThreadOnly |
| 248 | protected boolean handleUserControlPressed(HdmiCecMessage message) { |
| 249 | assertRunOnServiceThread(); |
| 250 | wakeUpIfActiveSource(); |
| 251 | return super.handleUserControlPressed(message); |
| 252 | } |
| 253 | |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 254 | @Override |
| 255 | @ServiceThreadOnly |
| 256 | protected boolean handleSetStreamPath(HdmiCecMessage message) { |
| 257 | assertRunOnServiceThread(); |
| 258 | int physicalAddress = HdmiUtils.twoBytesToInt(message.getParams()); |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 259 | maySetActiveSource(physicalAddress); |
Yuncheol Heo | 8ecb721 | 2014-10-27 17:29:42 +0900 | [diff] [blame] | 260 | maySendActiveSource(message.getSource()); |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 261 | wakeUpIfActiveSource(); |
| 262 | return true; // Broadcast message. |
| 263 | } |
| 264 | |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 265 | // Samsung model we tested sends <Routing Change> and <Request Active Source> |
| 266 | // in a row, and then changes the input to the internal source if there is no |
| 267 | // <Active Source> in response. To handle this, we'll set ActiveSource aggressively. |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 268 | @Override |
| 269 | @ServiceThreadOnly |
| 270 | protected boolean handleRoutingChange(HdmiCecMessage message) { |
| 271 | assertRunOnServiceThread(); |
| 272 | int newPath = HdmiUtils.twoBytesToInt(message.getParams(), 2); |
| 273 | maySetActiveSource(newPath); |
| 274 | return true; // Broadcast message. |
| 275 | } |
| 276 | |
| 277 | @Override |
| 278 | @ServiceThreadOnly |
| 279 | protected boolean handleRoutingInformation(HdmiCecMessage message) { |
| 280 | assertRunOnServiceThread(); |
| 281 | int physicalAddress = HdmiUtils.twoBytesToInt(message.getParams()); |
| 282 | maySetActiveSource(physicalAddress); |
| 283 | return true; // Broadcast message. |
| 284 | } |
| 285 | |
| 286 | private void maySetActiveSource(int physicalAddress) { |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 287 | setActiveSource(physicalAddress == mService.getPhysicalAddress()); |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 288 | } |
| 289 | |
| 290 | private void wakeUpIfActiveSource() { |
Jinsuk Kim | 9b8507c | 2015-03-24 16:55:01 +0900 | [diff] [blame] | 291 | if (!mIsActiveSource) { |
| 292 | return; |
| 293 | } |
| 294 | // Wake up the device if the power is in standby mode, or its screen is off - |
| 295 | // which can happen if the device is holding a partial lock. |
| 296 | if (mService.isPowerStandbyOrTransient() || !mService.getPowerManager().isScreenOn()) { |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 297 | mService.wakeUp(); |
| 298 | } |
| 299 | } |
| 300 | |
Yuncheol Heo | 8ecb721 | 2014-10-27 17:29:42 +0900 | [diff] [blame] | 301 | private void maySendActiveSource(int dest) { |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 302 | if (mIsActiveSource) { |
| 303 | mService.sendCecCommand(HdmiCecMessageBuilder.buildActiveSource( |
| 304 | mAddress, mService.getPhysicalAddress())); |
Yuncheol Heo | 8ecb721 | 2014-10-27 17:29:42 +0900 | [diff] [blame] | 305 | // Always reports menu-status active to receive RCP. |
| 306 | mService.sendCecCommand(HdmiCecMessageBuilder.buildReportMenuStatus( |
| 307 | mAddress, dest, Constants.MENU_STATE_ACTIVATED)); |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 308 | } |
| 309 | } |
| 310 | |
| 311 | @Override |
| 312 | @ServiceThreadOnly |
| 313 | protected boolean handleRequestActiveSource(HdmiCecMessage message) { |
| 314 | assertRunOnServiceThread(); |
Yuncheol Heo | 8ecb721 | 2014-10-27 17:29:42 +0900 | [diff] [blame] | 315 | maySendActiveSource(message.getSource()); |
Yuncheol Heo | 64bafd9 | 2014-08-11 11:17:54 +0900 | [diff] [blame] | 316 | return true; // Broadcast message. |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 317 | } |
| 318 | |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 319 | @ServiceThreadOnly |
| 320 | protected boolean handleSetMenuLanguage(HdmiCecMessage message) { |
| 321 | assertRunOnServiceThread(); |
Donghyun Cho | a09256c | 2016-03-11 17:35:37 +0900 | [diff] [blame] | 322 | if (!SET_MENU_LANGUAGE) { |
| 323 | return false; |
| 324 | } |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 325 | |
| 326 | try { |
| 327 | String iso3Language = new String(message.getParams(), 0, 3, "US-ASCII"); |
Jinsuk Kim | 8ea452e | 2015-07-10 13:01:06 +0900 | [diff] [blame] | 328 | Locale currentLocale = mService.getContext().getResources().getConfiguration().locale; |
| 329 | if (currentLocale.getISO3Language().equals(iso3Language)) { |
| 330 | // Do not switch language if the new language is the same as the current one. |
| 331 | // This helps avoid accidental country variant switching from en_US to en_AU |
| 332 | // due to the limitation of CEC. See the warning below. |
| 333 | return true; |
| 334 | } |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 335 | |
| 336 | // Don't use Locale.getAvailableLocales() since it returns a locale |
| 337 | // which is not available on Settings. |
| 338 | final List<LocaleInfo> localeInfos = LocalePicker.getAllAssetLocales( |
| 339 | mService.getContext(), false); |
| 340 | for (LocaleInfo localeInfo : localeInfos) { |
| 341 | if (localeInfo.getLocale().getISO3Language().equals(iso3Language)) { |
| 342 | // WARNING: CEC adopts ISO/FDIS-2 for language code, while Android requires |
| 343 | // additional country variant to pinpoint the locale. This keeps the right |
| 344 | // locale from being chosen. 'eng' in the CEC command, for instance, |
| 345 | // will always be mapped to en-AU among other variants like en-US, en-GB, |
| 346 | // an en-IN, which may not be the expected one. |
| 347 | LocalePicker.updateLocale(localeInfo.getLocale()); |
| 348 | return true; |
| 349 | } |
| 350 | } |
| 351 | Slog.w(TAG, "Can't handle <Set Menu Language> of " + iso3Language); |
| 352 | return false; |
| 353 | } catch (UnsupportedEncodingException e) { |
Donghyun Cho | a09256c | 2016-03-11 17:35:37 +0900 | [diff] [blame] | 354 | Slog.w(TAG, "Can't handle <Set Menu Language>", e); |
Terry Heo | 795415b | 2014-10-01 15:03:53 +0900 | [diff] [blame] | 355 | return false; |
| 356 | } |
| 357 | } |
| 358 | |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 359 | @Override |
| 360 | @ServiceThreadOnly |
Jinsuk Kim | 8d115eb | 2015-03-18 10:52:13 +0900 | [diff] [blame] | 361 | protected void sendStandby(int deviceId) { |
| 362 | assertRunOnServiceThread(); |
| 363 | |
| 364 | // Playback device can send <Standby> to TV only. Ignore the parameter. |
| 365 | int targetAddress = Constants.ADDR_TV; |
| 366 | mService.sendCecCommand(HdmiCecMessageBuilder.buildStandby(mAddress, targetAddress)); |
| 367 | } |
| 368 | |
| 369 | @Override |
| 370 | @ServiceThreadOnly |
Jungshik Jang | 4fc1d10 | 2014-07-09 19:24:50 +0900 | [diff] [blame] | 371 | protected void disableDevice(boolean initiatedByCec, PendingActionClearedCallback callback) { |
| 372 | super.disableDevice(initiatedByCec, callback); |
| 373 | |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 374 | assertRunOnServiceThread(); |
| 375 | if (!initiatedByCec && mIsActiveSource) { |
| 376 | mService.sendCecCommand(HdmiCecMessageBuilder.buildInactiveSource( |
| 377 | mAddress, mService.getPhysicalAddress())); |
| 378 | } |
Jinsuk Kim | e26d833 | 2015-01-09 08:55:41 +0900 | [diff] [blame] | 379 | setActiveSource(false); |
Yuncheol Heo | 38db629 | 2014-07-01 14:15:14 +0900 | [diff] [blame] | 380 | checkIfPendingActionsCleared(); |
Jungshik Jang | 79c58a4 | 2014-06-16 16:45:36 +0900 | [diff] [blame] | 381 | } |
Terry Heo | 959d2db | 2014-08-28 16:45:41 +0900 | [diff] [blame] | 382 | |
| 383 | @Override |
| 384 | protected void dump(final IndentingPrintWriter pw) { |
| 385 | super.dump(pw); |
| 386 | pw.println("mIsActiveSource: " + mIsActiveSource); |
Jinsuk Kim | e6e8f3d | 2015-05-11 14:17:04 +0900 | [diff] [blame] | 387 | pw.println("mAutoTvOff:" + mAutoTvOff); |
Terry Heo | 959d2db | 2014-08-28 16:45:41 +0900 | [diff] [blame] | 388 | } |
Jinsuk Kim | bad8393 | 2015-02-10 07:10:40 +0900 | [diff] [blame] | 389 | |
| 390 | // Wrapper interface over PowerManager.WakeLock |
| 391 | private interface ActiveWakeLock { |
| 392 | void acquire(); |
| 393 | void release(); |
| 394 | boolean isHeld(); |
| 395 | } |
| 396 | |
| 397 | private class SystemWakeLock implements ActiveWakeLock { |
| 398 | private final WakeLock mWakeLock; |
| 399 | public SystemWakeLock() { |
| 400 | mWakeLock = mService.getPowerManager().newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, TAG); |
| 401 | mWakeLock.setReferenceCounted(false); |
| 402 | } |
| 403 | |
| 404 | @Override |
| 405 | public void acquire() { |
| 406 | mWakeLock.acquire(); |
| 407 | HdmiLogger.debug("active source: %b. Wake lock acquired", mIsActiveSource); |
| 408 | } |
| 409 | |
| 410 | @Override |
| 411 | public void release() { |
| 412 | mWakeLock.release(); |
| 413 | HdmiLogger.debug("Wake lock released"); |
| 414 | } |
| 415 | |
| 416 | @Override |
| 417 | public boolean isHeld() { |
| 418 | return mWakeLock.isHeld(); |
| 419 | } |
| 420 | } |
Jinsuk Kim | 84659ed | 2014-10-06 23:48:19 +0000 | [diff] [blame] | 421 | } |