blob: 38f58fd79002ba2dac0a14b370e666e4efa1d72f [file] [log] [blame]
/*
* Copyright (C) 2016 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.telecom.tests;
import android.media.AudioManager;
import android.os.HandlerThread;
import android.test.suitebuilder.annotation.SmallTest;
import com.android.server.telecom.CallAudioManager;
import com.android.server.telecom.CallAudioModeStateMachine;
import com.android.server.telecom.CallAudioRouteStateMachine;
import com.android.server.telecom.CallAudioModeStateMachine.MessageArgs.Builder;
import com.android.server.telecom.SystemStateHelper;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
import org.mockito.Mock;
import static org.junit.Assert.assertEquals;
import static org.mockito.ArgumentMatchers.nullable;
import static org.mockito.Matchers.anyInt;
import static org.mockito.Mockito.clearInvocations;
import static org.mockito.Mockito.never;
import static org.mockito.Mockito.times;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
@RunWith(JUnit4.class)
public class CallAudioModeStateMachineTest extends TelecomTestCase {
private static final int TEST_TIMEOUT = 1000;
@Mock private SystemStateHelper mSystemStateHelper;
@Mock private AudioManager mAudioManager;
@Mock private CallAudioManager mCallAudioManager;
private HandlerThread mTestThread;
@Override
@Before
public void setUp() throws Exception {
mTestThread = new HandlerThread("CallAudioModeStateMachineTest");
mTestThread.start();
super.setUp();
}
@Override
@After
public void tearDown() throws Exception {
mTestThread.quit();
mTestThread.join();
super.tearDown();
}
@SmallTest
@Test
public void testNoFocusWhenRingerSilenced() throws Throwable {
CallAudioModeStateMachine sm = new CallAudioModeStateMachine(mSystemStateHelper,
mAudioManager, mTestThread.getLooper());
sm.setCallAudioManager(mCallAudioManager);
sm.sendMessage(CallAudioModeStateMachine.ABANDON_FOCUS_FOR_TESTING);
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
resetMocks();
when(mCallAudioManager.startRinging()).thenReturn(false);
sm.sendMessage(CallAudioModeStateMachine.NEW_RINGING_CALL, new Builder()
.setHasActiveOrDialingCalls(false)
.setHasRingingCalls(true)
.setHasHoldingCalls(false)
.setIsTonePlaying(false)
.setForegroundCallIsVoip(false)
.setSession(null)
.build());
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
assertEquals(CallAudioModeStateMachine.RING_STATE_NAME, sm.getCurrentStateName());
verify(mAudioManager, never()).requestAudioFocusForCall(anyInt(), anyInt());
verify(mAudioManager, never()).setMode(anyInt());
verify(mCallAudioManager, never()).stopRinging();
verify(mCallAudioManager).stopCallWaiting();
}
@SmallTest
@Test
public void testNoRingWhenDeviceIsAtEar() {
CallAudioModeStateMachine sm = new CallAudioModeStateMachine(mSystemStateHelper,
mAudioManager, mTestThread.getLooper());
sm.setCallAudioManager(mCallAudioManager);
sm.sendMessage(CallAudioModeStateMachine.ABANDON_FOCUS_FOR_TESTING);
sm.sendMessage(CallAudioModeStateMachine.NEW_HOLDING_CALL, new Builder()
.setHasActiveOrDialingCalls(false)
.setHasRingingCalls(false)
.setHasHoldingCalls(true)
.setIsTonePlaying(false)
.setForegroundCallIsVoip(false)
.setSession(null)
.build());
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
assertEquals(CallAudioModeStateMachine.TONE_HOLD_STATE_NAME, sm.getCurrentStateName());
when(mSystemStateHelper.isDeviceAtEar()).thenReturn(true);
resetMocks();
sm.sendMessage(CallAudioModeStateMachine.NEW_RINGING_CALL, new Builder()
.setHasActiveOrDialingCalls(false)
.setHasRingingCalls(true)
.setHasHoldingCalls(true)
.setIsTonePlaying(false)
.setForegroundCallIsVoip(false)
.setSession(null)
.build());
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
verify(mAudioManager, never()).requestAudioFocusForCall(anyInt(), anyInt());
verify(mAudioManager, never()).setMode(anyInt());
verify(mCallAudioManager, never()).startRinging();
verify(mCallAudioManager).startCallWaiting(nullable(String.class));
}
@SmallTest
@Test
public void testRegainFocusWhenHfpIsConnectedSilenced() throws Throwable {
CallAudioModeStateMachine sm = new CallAudioModeStateMachine(mSystemStateHelper,
mAudioManager, mTestThread.getLooper());
sm.setCallAudioManager(mCallAudioManager);
sm.sendMessage(CallAudioModeStateMachine.ABANDON_FOCUS_FOR_TESTING);
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
resetMocks();
when(mCallAudioManager.startRinging()).thenReturn(false);
sm.sendMessage(CallAudioModeStateMachine.NEW_RINGING_CALL, new Builder()
.setHasActiveOrDialingCalls(false)
.setHasRingingCalls(true)
.setHasHoldingCalls(false)
.setIsTonePlaying(false)
.setForegroundCallIsVoip(false)
.setSession(null)
.build());
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
assertEquals(CallAudioModeStateMachine.RING_STATE_NAME, sm.getCurrentStateName());
verify(mAudioManager, never()).requestAudioFocusForCall(anyInt(), anyInt());
verify(mAudioManager, never()).setMode(anyInt());
verify(mCallAudioManager, never()).stopRinging();
verify(mCallAudioManager).stopCallWaiting();
when(mCallAudioManager.startRinging()).thenReturn(true);
sm.sendMessage(CallAudioModeStateMachine.RINGER_MODE_CHANGE);
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
verify(mCallAudioManager, times(2)).startRinging();
verify(mAudioManager).requestAudioFocusForCall(AudioManager.STREAM_RING,
AudioManager.AUDIOFOCUS_GAIN_TRANSIENT);
verify(mAudioManager).setMode(AudioManager.MODE_RINGTONE);
verify(mCallAudioManager).setCallAudioRouteFocusState(
CallAudioRouteStateMachine.RINGING_FOCUS);
}
@SmallTest
@Test
public void testDoNotRingTwiceWhenHfpConnected() {
CallAudioModeStateMachine sm = new CallAudioModeStateMachine(mSystemStateHelper,
mAudioManager, mTestThread.getLooper());
sm.setCallAudioManager(mCallAudioManager);
sm.sendMessage(CallAudioModeStateMachine.ABANDON_FOCUS_FOR_TESTING);
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
resetMocks();
when(mCallAudioManager.startRinging()).thenReturn(true);
sm.sendMessage(CallAudioModeStateMachine.NEW_RINGING_CALL, new Builder()
.setHasActiveOrDialingCalls(false)
.setHasRingingCalls(true)
.setHasHoldingCalls(false)
.setIsTonePlaying(false)
.setForegroundCallIsVoip(false)
.setSession(null)
.build());
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
assertEquals(CallAudioModeStateMachine.RING_STATE_NAME, sm.getCurrentStateName());
verify(mAudioManager).requestAudioFocusForCall(AudioManager.STREAM_RING,
AudioManager.AUDIOFOCUS_GAIN_TRANSIENT);
verify(mAudioManager).setMode(AudioManager.MODE_RINGTONE);
verify(mCallAudioManager).setCallAudioRouteFocusState(
CallAudioRouteStateMachine.RINGING_FOCUS);
when(mCallAudioManager.isRingtonePlaying()).thenReturn(true);
sm.sendMessage(CallAudioModeStateMachine.RINGER_MODE_CHANGE);
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
// Make sure we don't try and start ringing again.
verify(mCallAudioManager, times(1)).startRinging();
}
@SmallTest
@Test
public void testStartRingingAfterHfpConnectedIfNotAlreadyPlaying() {
CallAudioModeStateMachine sm = new CallAudioModeStateMachine(mSystemStateHelper,
mAudioManager, mTestThread.getLooper());
sm.setCallAudioManager(mCallAudioManager);
sm.sendMessage(CallAudioModeStateMachine.ABANDON_FOCUS_FOR_TESTING);
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
resetMocks();
when(mCallAudioManager.startRinging()).thenReturn(true);
sm.sendMessage(CallAudioModeStateMachine.NEW_RINGING_CALL, new Builder()
.setHasActiveOrDialingCalls(false)
.setHasRingingCalls(true)
.setHasHoldingCalls(false)
.setIsTonePlaying(false)
.setForegroundCallIsVoip(false)
.setSession(null)
.build());
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
assertEquals(CallAudioModeStateMachine.RING_STATE_NAME, sm.getCurrentStateName());
verify(mAudioManager).requestAudioFocusForCall(AudioManager.STREAM_RING,
AudioManager.AUDIOFOCUS_GAIN_TRANSIENT);
verify(mAudioManager).setMode(AudioManager.MODE_RINGTONE);
verify(mCallAudioManager).setCallAudioRouteFocusState(
CallAudioRouteStateMachine.RINGING_FOCUS);
when(mCallAudioManager.isRingtonePlaying()).thenReturn(false);
sm.sendMessage(CallAudioModeStateMachine.RINGER_MODE_CHANGE);
waitForHandlerAction(sm.getHandler(), TEST_TIMEOUT);
// Make sure we do try and start ringing again, since the ringtone wasn't already playing.
verify(mCallAudioManager, times(2)).startRinging();
}
private void resetMocks() {
clearInvocations(mCallAudioManager, mAudioManager);
}
}