Merge "Cache OTP callback if the device is still under address allocation"

This commit is contained in:
Amy Zhang
2019-10-04 20:36:53 +00:00
committed by Android (Google) Code Review
3 changed files with 236 additions and 7 deletions

View File

@@ -310,6 +310,11 @@ public class HdmiControlService extends SystemService {
// Invoke once new local device is ready.
private IHdmiControlCallback mDisplayStatusCallback = null;
@Nullable
// Save callback when the device is still under logcial address allocation
// Invoke once new local device is ready.
private IHdmiControlCallback mOtpCallbackPendingAddressAllocation = null;
@Nullable
private HdmiCecController mCecController;
@@ -785,17 +790,21 @@ public class HdmiControlService extends SystemService {
// Address allocation completed for all devices. Notify each device.
if (allocatingDevices.size() == ++finished[0]) {
mAddressAllocated = true;
// Reinvoke the saved display status callback once the local device is ready.
if (mDisplayStatusCallback != null) {
queryDisplayStatus(mDisplayStatusCallback);
mDisplayStatusCallback = null;
}
if (initiatedBy != INITIATED_BY_HOTPLUG) {
// In case of the hotplug we don't call onInitializeCecComplete()
// since we reallocate the logical address only.
onInitializeCecComplete(initiatedBy);
}
notifyAddressAllocated(allocatedDevices, initiatedBy);
// Reinvoke the saved display status callback once the local device is ready.
if (mDisplayStatusCallback != null) {
queryDisplayStatus(mDisplayStatusCallback);
mDisplayStatusCallback = null;
}
if (mOtpCallbackPendingAddressAllocation != null) {
oneTouchPlay(mOtpCallbackPendingAddressAllocation);
mOtpCallbackPendingAddressAllocation = null;
}
mCecMessageBuffer.processMessages();
}
}
@@ -2246,8 +2255,16 @@ public class HdmiControlService extends SystemService {
}
@ServiceThreadOnly
private void oneTouchPlay(final IHdmiControlCallback callback) {
@VisibleForTesting
protected void oneTouchPlay(final IHdmiControlCallback callback) {
assertRunOnServiceThread();
if (!mAddressAllocated) {
mOtpCallbackPendingAddressAllocation = callback;
Slog.d(TAG, "Local device is under address allocation. "
+ "Save OTP callback for later process.");
return;
}
HdmiCecLocalDeviceSource source = playback();
if (source == null) {
source = audioSystem();

View File

@@ -77,7 +77,6 @@ final class OneTouchPlayAction extends HdmiCecFeatureAction {
sendCommand(HdmiCecMessageBuilder.buildTextViewOn(getSourceAddress(), mTargetAddress));
broadcastActiveSource();
queryDevicePowerStatus();
mState = STATE_WAITING_FOR_REPORT_POWER_STATUS;
addTimer(mState, HdmiConfig.TIMEOUT_MS);
return true;
}
@@ -99,6 +98,7 @@ final class OneTouchPlayAction extends HdmiCecFeatureAction {
}
private void queryDevicePowerStatus() {
mState = STATE_WAITING_FOR_REPORT_POWER_STATUS;
sendCommand(HdmiCecMessageBuilder.buildGiveDevicePowerStatus(getSourceAddress(),
mTargetAddress));
}

View File

@@ -0,0 +1,212 @@
/*
* Copyright (C) 2019 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.server.hdmi;
import static android.os.SystemClock.sleep;
import static com.android.server.hdmi.HdmiControlService.INITIATED_BY_ENABLE_CEC;
import static com.google.common.truth.Truth.assertThat;
import static junit.framework.Assert.assertEquals;
import android.hardware.hdmi.HdmiControlManager;
import android.hardware.hdmi.HdmiPortInfo;
import android.hardware.hdmi.IHdmiControlCallback;
import android.os.Looper;
import android.os.SystemProperties;
import android.os.test.TestLooper;
import android.util.Slog;
import androidx.test.InstrumentationRegistry;
import androidx.test.filters.SmallTest;
import java.util.ArrayList;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/**
* Tests for {@link HdmiControlServiceBinderAPITest} class.
*/
@SmallTest
@RunWith(JUnit4.class)
public class HdmiControlServiceBinderAPITest {
private class HdmiCecLocalDeviceMyDevice extends HdmiCecLocalDevice {
private boolean mCanGoToStandby;
private boolean mIsStandby;
private boolean mIsDisabled;
protected HdmiCecLocalDeviceMyDevice(HdmiControlService service, int deviceType) {
super(service, deviceType);
}
@Override
protected void onAddressAllocated(int logicalAddress, int reason) {
}
@Override
protected int getPreferredAddress() {
return 0;
}
@Override
protected void setPreferredAddress(int addr) {
}
@Override
protected boolean canGoToStandby() {
return mCanGoToStandby;
}
@Override
protected void disableDevice(
boolean initiatedByCec, final PendingActionClearedCallback originalCallback) {
mIsDisabled = true;
originalCallback.onCleared(this);
}
@Override
protected void onStandby(boolean initiatedByCec, int standbyAction) {
mIsStandby = true;
}
protected boolean isStandby() {
return mIsStandby;
}
protected boolean isDisabled() {
return mIsDisabled;
}
protected void setCanGoToStandby(boolean canGoToStandby) {
mCanGoToStandby = canGoToStandby;
}
}
private static final String TAG = "HdmiControlServiceBinderAPITest";
private HdmiControlService mHdmiControlService;
private HdmiCecController mHdmiCecController;
private HdmiCecLocalDevicePlayback mPlaybackDevice;
private FakeNativeWrapper mNativeWrapper;
private Looper mMyLooper;
private TestLooper mTestLooper = new TestLooper();
private ArrayList<HdmiCecLocalDevice> mLocalDevices = new ArrayList<>();
private HdmiPortInfo[] mHdmiPortInfo;
private int mResult;
private int mPowerStatus;
@Before
public void SetUp() {
mHdmiControlService =
new HdmiControlService(InstrumentationRegistry.getTargetContext()) {
@Override
void sendCecCommand(HdmiCecMessage command) {
switch (command.getOpcode()) {
case Constants.MESSAGE_GIVE_DEVICE_POWER_STATUS:
HdmiCecMessage message =
HdmiCecMessageBuilder.buildReportPowerStatus(
Constants.ADDR_TV,
Constants.ADDR_PLAYBACK_1,
HdmiControlManager.POWER_STATUS_ON);
handleCecCommand(message);
break;
default:
return;
}
}
@Override
boolean isPowerStandby() {
return mPowerStatus == HdmiControlManager.POWER_STATUS_STANDBY;
}
};
mMyLooper = mTestLooper.getLooper();
mPlaybackDevice = new HdmiCecLocalDevicePlayback(mHdmiControlService) {
@Override
void setIsActiveSource(boolean on) {
mIsActiveSource = on;
}
@Override
protected void wakeUpIfActiveSource() {}
@Override
protected void setPreferredAddress(int addr) {}
@Override
protected int getPreferredAddress() {
return Constants.ADDR_PLAYBACK_1;
}
};
mPlaybackDevice.init();
mHdmiControlService.setIoLooper(mMyLooper);
mNativeWrapper = new FakeNativeWrapper();
mHdmiCecController =
HdmiCecController.createWithNativeWrapper(mHdmiControlService, mNativeWrapper);
mHdmiControlService.setCecController(mHdmiCecController);
mHdmiControlService.setHdmiMhlController(HdmiMhlControllerStub.create(mHdmiControlService));
mHdmiControlService.setMessageValidator(new HdmiCecMessageValidator(mHdmiControlService));
mLocalDevices.add(mPlaybackDevice);
mHdmiPortInfo = new HdmiPortInfo[1];
mHdmiPortInfo[0] =
new HdmiPortInfo(1, HdmiPortInfo.PORT_INPUT, 0x2100, true, false, false);
mNativeWrapper.setPortInfo(mHdmiPortInfo);
mHdmiControlService.initPortInfo();
mResult = -1;
mPowerStatus = HdmiControlManager.POWER_STATUS_ON;
mTestLooper.dispatchAll();
}
@Test
public void oneTouchPlay_addressNotAllocated() {
assertThat(mHdmiControlService.isAddressAllocated()).isFalse();
mHdmiControlService.oneTouchPlay(new IHdmiControlCallback.Stub() {
@Override
public void onComplete(int result) {
mResult = result;
}
});
assertEquals(mResult, -1);
assertThat(mPlaybackDevice.mIsActiveSource).isFalse();
mHdmiControlService.allocateLogicalAddress(mLocalDevices, INITIATED_BY_ENABLE_CEC);
mTestLooper.dispatchAll();
assertThat(mHdmiControlService.isAddressAllocated()).isTrue();
assertEquals(mResult, HdmiControlManager.RESULT_SUCCESS);
assertThat(mPlaybackDevice.mIsActiveSource).isTrue();
}
@Test
public void oneTouchPlay_addressAllocated() {
mHdmiControlService.allocateLogicalAddress(mLocalDevices, INITIATED_BY_ENABLE_CEC);
mTestLooper.dispatchAll();
assertThat(mHdmiControlService.isAddressAllocated()).isTrue();
mHdmiControlService.oneTouchPlay(new IHdmiControlCallback.Stub() {
@Override
public void onComplete(int result) {
mResult = result;
}
});
assertEquals(mResult, HdmiControlManager.RESULT_SUCCESS);
assertThat(mPlaybackDevice.mIsActiveSource).isTrue();
}
}