| /* |
| * Copyright (C) 2018 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.tv.cec.V1_0.SendMessageResult; |
| |
| import com.android.internal.annotations.VisibleForTesting; |
| |
| /** |
| * Feature action that handles System Audio Mode initiated by AVR devices. |
| */ |
| public class SystemAudioInitiationActionFromAvr extends HdmiCecFeatureAction { |
| |
| // State that waits for <Active Source> once send <Request Active Source>. |
| private static final int STATE_WAITING_FOR_ACTIVE_SOURCE = 1; |
| // State that waits for TV supporting Audio System Mode or not |
| // once received <Active Source> |
| private static final int STATE_WAITING_FOR_TV_SUPPORT = 2; |
| @VisibleForTesting |
| static final int MAX_RETRY_COUNT = 5; |
| |
| private int mSendRequestActiveSourceRetryCount = 0; |
| private int mSendSetSystemAudioModeRetryCount = 0; |
| |
| SystemAudioInitiationActionFromAvr(HdmiCecLocalDevice source) { |
| super(source); |
| } |
| |
| @Override |
| boolean start() { |
| if (audioSystem().getActiveSource().physicalAddress == Constants.INVALID_PHYSICAL_ADDRESS) { |
| mState = STATE_WAITING_FOR_ACTIVE_SOURCE; |
| addTimer(mState, HdmiConfig.TIMEOUT_MS); |
| sendRequestActiveSource(); |
| } else { |
| queryTvSystemAudioModeSupport(); |
| } |
| return true; |
| } |
| |
| @Override |
| boolean processCommand(HdmiCecMessage cmd) { |
| switch (cmd.getOpcode()) { |
| case Constants.MESSAGE_ACTIVE_SOURCE: |
| // received <Active Source> |
| if (mState != STATE_WAITING_FOR_ACTIVE_SOURCE) { |
| return false; |
| } |
| mActionTimer.clearTimerMessage(); |
| // Broadcast message is also handled by other device types |
| audioSystem().handleActiveSource(cmd); |
| mState = STATE_WAITING_FOR_TV_SUPPORT; |
| queryTvSystemAudioModeSupport(); |
| return true; |
| } |
| return false; |
| } |
| |
| @Override |
| void handleTimerEvent(int state) { |
| if (mState != state) { |
| return; |
| } |
| |
| switch (mState) { |
| case STATE_WAITING_FOR_ACTIVE_SOURCE: |
| handleActiveSourceTimeout(); |
| break; |
| } |
| } |
| |
| protected void sendRequestActiveSource() { |
| sendCommand(HdmiCecMessageBuilder.buildRequestActiveSource(getSourceAddress()), |
| result -> { |
| if (result != SendMessageResult.SUCCESS) { |
| if (mSendRequestActiveSourceRetryCount < MAX_RETRY_COUNT) { |
| mSendRequestActiveSourceRetryCount++; |
| sendRequestActiveSource(); |
| } else { |
| audioSystem().checkSupportAndSetSystemAudioMode(false); |
| finish(); |
| } |
| } |
| }); |
| } |
| |
| protected void sendSetSystemAudioMode(boolean on, int dest) { |
| sendCommand(HdmiCecMessageBuilder.buildSetSystemAudioMode(getSourceAddress(), |
| dest, on), result -> { |
| if (result != SendMessageResult.SUCCESS) { |
| if (mSendSetSystemAudioModeRetryCount < MAX_RETRY_COUNT) { |
| mSendSetSystemAudioModeRetryCount++; |
| sendSetSystemAudioMode(on, dest); |
| } else { |
| audioSystem().checkSupportAndSetSystemAudioMode(false); |
| finish(); |
| } |
| } |
| }); |
| } |
| |
| private void handleActiveSourceTimeout() { |
| HdmiLogger.debug("Cannot get active source."); |
| // If not able to find Active Source and the current device has playbcak functionality, |
| // claim Active Source and start to query TV system audio mode support. |
| if (audioSystem().mService.isPlaybackDevice()) { |
| audioSystem().mService.setAndBroadcastActiveSourceFromOneDeviceType( |
| Constants.ADDR_BROADCAST, getSourcePath()); |
| mState = STATE_WAITING_FOR_TV_SUPPORT; |
| queryTvSystemAudioModeSupport(); |
| } else { |
| audioSystem().checkSupportAndSetSystemAudioMode(false); |
| } |
| finish(); |
| } |
| |
| private void queryTvSystemAudioModeSupport() { |
| audioSystem().queryTvSystemAudioModeSupport( |
| supported -> { |
| if (supported) { |
| if (audioSystem().checkSupportAndSetSystemAudioMode(true)) { |
| sendSetSystemAudioMode(true, Constants.ADDR_BROADCAST); |
| } |
| finish(); |
| } else { |
| audioSystem().checkSupportAndSetSystemAudioMode(false); |
| finish(); |
| } |
| }); |
| } |
| |
| private void switchToRelevantInputForDeviceAt(int physicalAddress) { |
| // TODO(shubang): implement this method |
| } |
| } |