/* * Copyright (C) 2014 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 android.hardware.hdmi.HdmiControlManager; import android.hardware.hdmi.IHdmiControlCallback; import android.hardware.hdmi.HdmiPlaybackClient.OneTouchPlayCallback; import android.os.RemoteException; import android.util.Slog; import java.util.ArrayList; import java.util.List; /** * Feature action that performs one touch play against TV/Display device. This action is initiated * via {@link android.hardware.hdmi.HdmiPlaybackClient#oneTouchPlay(OneTouchPlayCallback)} from the * Android system working as playback device to turn on the TV, and switch the input. *

* Package-private, accessed by {@link HdmiControlService} only. */ final class OneTouchPlayAction extends HdmiCecFeatureAction { private static final String TAG = "OneTouchPlayAction"; // State in which the action is waiting for . In normal situation // source device can simply send and in succession // since the standard requires that the TV/Display should buffer the // if the TV is brought of out standby state. // // But there are TV's that fail to buffer the while getting out of // standby mode, and do not accept the command until their power status becomes 'ON'. // For a workaround, we send commands periodically to make sure // the device switches its status to 'ON'. Then we send additional . private static final int STATE_WAITING_FOR_REPORT_POWER_STATUS = 1; // The maximum number of times we send before we give up. // We wait up to RESPONSE_TIMEOUT_MS * LOOP_COUNTER_MAX = 20 seconds. private static final int LOOP_COUNTER_MAX = 10; private final int mTargetAddress; private final List mCallbacks = new ArrayList<>(); private int mPowerStatusCounter = 0; // Factory method. Ensures arguments are valid. static OneTouchPlayAction create(HdmiCecLocalDevicePlayback source, int targetAddress, IHdmiControlCallback callback) { if (source == null || callback == null) { Slog.e(TAG, "Wrong arguments"); return null; } return new OneTouchPlayAction(source, targetAddress, callback); } private OneTouchPlayAction(HdmiCecLocalDevice localDevice, int targetAddress, IHdmiControlCallback callback) { super(localDevice); mTargetAddress = targetAddress; addCallback(callback); } @Override boolean start() { sendCommand(HdmiCecMessageBuilder.buildTextViewOn(getSourceAddress(), mTargetAddress)); broadcastActiveSource(); queryDevicePowerStatus(); mState = STATE_WAITING_FOR_REPORT_POWER_STATUS; addTimer(mState, HdmiConfig.TIMEOUT_MS); return true; } private void broadcastActiveSource() { sendCommand(HdmiCecMessageBuilder.buildActiveSource(getSourceAddress(), getSourcePath())); // Because only playback device can create this action, it's safe to cast. playback().setActiveSource(true); } private void queryDevicePowerStatus() { sendCommand(HdmiCecMessageBuilder.buildGiveDevicePowerStatus(getSourceAddress(), mTargetAddress)); } @Override boolean processCommand(HdmiCecMessage cmd) { if (mState != STATE_WAITING_FOR_REPORT_POWER_STATUS || mTargetAddress != cmd.getSource()) { return false; } if (cmd.getOpcode() == Constants.MESSAGE_REPORT_POWER_STATUS) { int status = cmd.getParams()[0]; if (status == HdmiControlManager.POWER_STATUS_ON) { broadcastActiveSource(); invokeCallback(HdmiControlManager.RESULT_SUCCESS); finish(); } return true; } return false; } @Override void handleTimerEvent(int state) { if (mState != state) { return; } if (state == STATE_WAITING_FOR_REPORT_POWER_STATUS) { if (mPowerStatusCounter++ < LOOP_COUNTER_MAX) { queryDevicePowerStatus(); addTimer(mState, HdmiConfig.TIMEOUT_MS); } else { // Couldn't wake up the TV for whatever reason. Report failure. invokeCallback(HdmiControlManager.RESULT_TIMEOUT); finish(); } } } public void addCallback(IHdmiControlCallback callback) { mCallbacks.add(callback); } private void invokeCallback(int result) { try { for (IHdmiControlCallback callback : mCallbacks) { callback.onComplete(result); } } catch (RemoteException e) { Slog.e(TAG, "Callback failed:" + e); } } }