| /* |
| * Copyright (C) 2020 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 android.hdmicec.cts.common; |
| |
| import static com.google.common.truth.Truth.assertThat; |
| import static com.google.common.truth.Truth.assertWithMessage; |
| |
| import android.hdmicec.cts.BaseHdmiCecCtsTest; |
| import android.hdmicec.cts.CecMessage; |
| import android.hdmicec.cts.CecOperand; |
| import android.hdmicec.cts.HdmiCecConstants; |
| import android.hdmicec.cts.LogicalAddress; |
| |
| import com.android.tradefed.device.ITestDevice; |
| import com.android.tradefed.testtype.DeviceJUnit4ClassRunner; |
| |
| import org.junit.Ignore; |
| import org.junit.Rule; |
| import org.junit.Test; |
| import org.junit.rules.RuleChain; |
| import org.junit.runner.RunWith; |
| |
| import java.util.Arrays; |
| import java.util.List; |
| import java.util.concurrent.TimeUnit; |
| |
| /** |
| * HDMI CEC tests verifying power status related messages of the device (CEC 2.0 CTS Section 7.6) |
| */ |
| @RunWith(DeviceJUnit4ClassRunner.class) |
| public final class HdmiCecPowerStatusTest extends BaseHdmiCecCtsTest { |
| |
| private static final int ON = 0x0; |
| private static final int OFF = 0x1; |
| private static final int IN_TRANSITION_TO_STANDBY = 0x3; |
| |
| private static final int SLEEP_TIMESTEP_SECONDS = 1; |
| private static final int WAIT_TIME = 5; |
| private static final int MAX_SLEEP_TIME = 8; |
| |
| @Rule |
| public RuleChain mRuleChain = |
| RuleChain |
| .outerRule(CecRules.requiresCec(this)) |
| .around(CecRules.requiresLeanback(this)) |
| .around(hdmiCecClient); |
| |
| /** |
| * Test HF4-6-20 |
| * |
| * Verifies that {@code <Report Power Status>} message is broadcast when the device transitions |
| * from standby to on. |
| */ |
| @Test |
| public void cect_hf4_6_20_broadcastsWhenTurningOn() throws Exception { |
| ITestDevice device = getDevice(); |
| setCec20(); |
| |
| // Move device to standby |
| device.executeShellCommand("input keyevent KEYCODE_SLEEP"); |
| TimeUnit.SECONDS.sleep(WAIT_TIME); |
| |
| // Turn device on |
| device.executeShellCommand("input keyevent KEYCODE_WAKEUP"); |
| TimeUnit.SECONDS.sleep(WAIT_TIME); |
| |
| String reportPowerStatus = hdmiCecClient.checkExpectedOutput(LogicalAddress.BROADCAST, |
| CecOperand.REPORT_POWER_STATUS); |
| |
| if (CecMessage.getParams(reportPowerStatus) == HdmiCecConstants.CEC_POWER_STATUS_STANDBY) { |
| // Received the "turning off" broadcast, check for the next broadcast message |
| reportPowerStatus = hdmiCecClient.checkExpectedOutput(LogicalAddress.BROADCAST, |
| CecOperand.REPORT_POWER_STATUS); |
| } |
| |
| assertThat(CecMessage.getParams(reportPowerStatus)).isEqualTo( |
| HdmiCecConstants.CEC_POWER_STATUS_ON); |
| } |
| |
| /** |
| * Test HF4-6-21 |
| * |
| * Verifies that {@code <Report Power Status>} message is broadcast when the device transitions |
| * from on to standby. |
| */ |
| @Test |
| public void cect_hf4_6_21_broadcastsWhenGoingToStandby() throws Exception { |
| ITestDevice device = getDevice(); |
| setCec20(); |
| |
| // Turn device on |
| device.executeShellCommand("input keyevent KEYCODE_WAKEUP"); |
| TimeUnit.SECONDS.sleep(WAIT_TIME); |
| |
| // Move device to standby |
| device.executeShellCommand("input keyevent KEYCODE_SLEEP"); |
| TimeUnit.SECONDS.sleep(WAIT_TIME); |
| |
| String reportPowerStatus = hdmiCecClient.checkExpectedOutput(LogicalAddress.BROADCAST, |
| CecOperand.REPORT_POWER_STATUS); |
| |
| if (CecMessage.getParams(reportPowerStatus) == HdmiCecConstants.CEC_POWER_STATUS_ON) { |
| // Received the "wake up" broadcast, check for the next broadcast message |
| reportPowerStatus = hdmiCecClient.checkExpectedOutput(LogicalAddress.BROADCAST, |
| CecOperand.REPORT_POWER_STATUS); |
| } |
| |
| assertThat(CecMessage.getParams(reportPowerStatus)).isEqualTo( |
| HdmiCecConstants.CEC_POWER_STATUS_STANDBY); |
| } |
| |
| /** |
| * Test 11.1.14-1, 11.2.14-1 |
| * |
| * <p>Tests that the device sends a {@code <REPORT_POWER_STATUS>} with params 0x0 when the |
| * device is powered on. |
| */ |
| @Test |
| public void cect_PowerStatusWhenOn() throws Exception { |
| ITestDevice device = getDevice(); |
| /* Make sure the device is not booting up/in standby */ |
| device.waitForBootComplete(HdmiCecConstants.REBOOT_TIMEOUT); |
| LogicalAddress cecClientDevice = hdmiCecClient.getSelfDevice(); |
| hdmiCecClient.sendCecMessage(cecClientDevice, CecOperand.GIVE_POWER_STATUS); |
| String message = |
| hdmiCecClient.checkExpectedOutput(cecClientDevice, CecOperand.REPORT_POWER_STATUS); |
| assertThat(CecMessage.getParams(message)).isEqualTo(ON); |
| } |
| |
| /** |
| * Test 11.2.14-1, 11.2.14-2 |
| * |
| * <p>Tests that the device sends a {@code <REPORT_POWER_STATUS>} with params 0x1 when the |
| * device is powered off. |
| */ |
| @Test |
| public void cect_PowerStatusWhenOff() throws Exception { |
| ITestDevice device = getDevice(); |
| try { |
| /* Make sure the device is not booting up/in standby */ |
| device.waitForBootComplete(HdmiCecConstants.REBOOT_TIMEOUT); |
| /* The sleep below could send some devices into a deep suspend state. */ |
| device.executeShellCommand("input keyevent KEYCODE_SLEEP"); |
| TimeUnit.SECONDS.sleep(WAIT_TIME); |
| int waitTimeSeconds = WAIT_TIME; |
| int powerStatus; |
| LogicalAddress cecClientDevice = hdmiCecClient.getSelfDevice(); |
| do { |
| TimeUnit.SECONDS.sleep(SLEEP_TIMESTEP_SECONDS); |
| waitTimeSeconds += SLEEP_TIMESTEP_SECONDS; |
| hdmiCecClient.sendCecMessage(cecClientDevice, CecOperand.GIVE_POWER_STATUS); |
| powerStatus = |
| CecMessage.getParams( |
| hdmiCecClient.checkExpectedOutput( |
| cecClientDevice, CecOperand.REPORT_POWER_STATUS)); |
| } while (powerStatus == IN_TRANSITION_TO_STANDBY && waitTimeSeconds <= MAX_SLEEP_TIME); |
| assertThat(powerStatus).isEqualTo(OFF); |
| } finally { |
| /* Wake up the device */ |
| device.executeShellCommand("input keyevent KEYCODE_WAKEUP"); |
| } |
| } |
| |
| /** |
| * Test HF4-6-8 |
| * |
| * Tests that a device comes out of the standby state when it receives a {@code <User Control |
| * Pressed>} message with power related operands. |
| */ |
| @Ignore("b/178083922") |
| @Test |
| public void cect_hf4_6_8_userControlPressed_powerOn() throws Exception { |
| ITestDevice device = getDevice(); |
| List<Integer> powerControlOperands = Arrays.asList(HdmiCecConstants.CEC_CONTROL_POWER, |
| HdmiCecConstants.CEC_CONTROL_POWER_ON_FUNCTION, |
| HdmiCecConstants.CEC_CONTROL_POWER_TOGGLE_FUNCTION); |
| |
| LogicalAddress source = mDutLogicalAddress == LogicalAddress.TV ? LogicalAddress.PLAYBACK_1 |
| : LogicalAddress.TV; |
| |
| for (Integer operand : powerControlOperands) { |
| try { |
| device.executeShellCommand("input keyevent KEYCODE_SLEEP"); |
| TimeUnit.SECONDS.sleep(MAX_SLEEP_TIME); |
| String wakeStateBefore = device.executeShellCommand( |
| "dumpsys power | grep mWakefulness="); |
| assertThat(wakeStateBefore.trim()).isEqualTo("mWakefulness=Asleep"); |
| |
| hdmiCecClient.sendUserControlPressAndRelease(source, mDutLogicalAddress, operand, |
| false); |
| |
| TimeUnit.SECONDS.sleep(WAIT_TIME); |
| String wakeStateAfter = device.executeShellCommand( |
| "dumpsys power | grep mWakefulness="); |
| assertWithMessage("Device should wake up on <User Control Pressed> %s", operand) |
| .that(wakeStateAfter.trim()).isEqualTo("mWakefulness=Awake"); |
| } finally { |
| device.executeShellCommand("input keyevent KEYCODE_WAKEUP"); |
| } |
| } |
| } |
| |
| /** |
| * Test HF4-6-10 |
| * |
| * Tests that a device comes enters the standby state when it receives a {@code <User Control |
| * Pressed>} message with power related operands. |
| */ |
| @Test |
| public void cect_hf4_6_10_userControlPressed_powerOff() throws Exception { |
| ITestDevice device = getDevice(); |
| List<Integer> powerControlOperands = Arrays.asList( |
| HdmiCecConstants.CEC_CONTROL_POWER_OFF_FUNCTION, |
| HdmiCecConstants.CEC_CONTROL_POWER_TOGGLE_FUNCTION); |
| |
| LogicalAddress source = mDutLogicalAddress == LogicalAddress.TV ? LogicalAddress.PLAYBACK_1 |
| : LogicalAddress.TV; |
| |
| for (Integer operand : powerControlOperands) { |
| try { |
| device.executeShellCommand("input keyevent KEYCODE_WAKEUP"); |
| TimeUnit.SECONDS.sleep(WAIT_TIME); |
| String wakeStateBefore = device.executeShellCommand( |
| "dumpsys power | grep mWakefulness="); |
| assertThat(wakeStateBefore.trim()).isEqualTo("mWakefulness=Awake"); |
| |
| hdmiCecClient.sendUserControlPressAndRelease(source, mDutLogicalAddress, operand, |
| false); |
| |
| TimeUnit.SECONDS.sleep(WAIT_TIME); |
| String wakeStateAfter = device.executeShellCommand( |
| "dumpsys power | grep mWakefulness="); |
| assertWithMessage("Device should go to standby on <User Control Pressed> %s", |
| operand) |
| .that(wakeStateAfter.trim()).isEqualTo("mWakefulness=Asleep"); |
| } finally { |
| device.executeShellCommand("input keyevent KEYCODE_WAKEUP"); |
| } |
| } |
| } |
| } |