Donate to e Foundation | Murena handsets with /e/OS | Own a part of Murena! Learn more

Commit af034343 authored by Grace Jia's avatar Grace Jia Committed by Automerger Merge Worker
Browse files

Add synchronized lock for ringback playing start/stop request to avoid am: 7b6c9ba0

parents 6162f2a1 7b6c9ba0
Loading
Loading
Loading
Loading
+43 −28
Original line number Original line Diff line number Diff line
@@ -19,6 +19,7 @@ package com.android.server.telecom;
import static com.android.server.telecom.LogUtils.Events.START_RINBACK;
import static com.android.server.telecom.LogUtils.Events.START_RINBACK;
import static com.android.server.telecom.LogUtils.Events.STOP_RINGBACK;
import static com.android.server.telecom.LogUtils.Events.STOP_RINGBACK;


import com.android.internal.annotations.VisibleForTesting;
import com.android.internal.util.Preconditions;
import com.android.internal.util.Preconditions;
import android.telecom.Log;
import android.telecom.Log;


@@ -42,8 +43,12 @@ public class RingbackPlayer {
     */
     */
    private InCallTonePlayer mTonePlayer;
    private InCallTonePlayer mTonePlayer;


    RingbackPlayer(InCallTonePlayer.Factory playerFactory) {
    private final Object mLock;

    @VisibleForTesting
    public RingbackPlayer(InCallTonePlayer.Factory playerFactory) {
        mPlayerFactory = playerFactory;
        mPlayerFactory = playerFactory;
        mLock = new Object();
    }
    }


    /**
    /**
@@ -52,6 +57,7 @@ public class RingbackPlayer {
     * @param call The call for which to ringback.
     * @param call The call for which to ringback.
     */
     */
    public void startRingbackForCall(Call call) {
    public void startRingbackForCall(Call call) {
        synchronized (mLock) {
            Preconditions.checkState(call.getState() == CallState.DIALING);
            Preconditions.checkState(call.getState() == CallState.DIALING);


            if (mCall == call) {
            if (mCall == call) {
@@ -73,6 +79,7 @@ public class RingbackPlayer {
                mTonePlayer.startTone();
                mTonePlayer.startTone();
            }
            }
        }
        }
    }


    /**
    /**
     * Stops the ringback for the specified dialing call as needed.
     * Stops the ringback for the specified dialing call as needed.
@@ -80,6 +87,7 @@ public class RingbackPlayer {
     * @param call The call for which to stop ringback.
     * @param call The call for which to stop ringback.
     */
     */
    public void stopRingbackForCall(Call call) {
    public void stopRingbackForCall(Call call) {
        synchronized (mLock) {
            if (mCall == call) {
            if (mCall == call) {
                // The foreground call is no longer dialing or is no longer the foreground call. In
                // The foreground call is no longer dialing or is no longer the foreground call. In
                // either case, stop the ringback tone.
                // either case, stop the ringback tone.
@@ -96,3 +104,10 @@ public class RingbackPlayer {
            }
            }
        }
        }
    }
    }

    public boolean isRingbackPlaying() {
        synchronized (mLock) {
            return mTonePlayer != null;
        }
    }
}
 No newline at end of file
+79 −0
Original line number Original line Diff line number Diff line
/*
 * Copyright (C) 2022 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 static org.junit.Assert.assertFalse;
import static org.mockito.ArgumentMatchers.anyInt;
import static org.mockito.Mockito.doAnswer;
import static org.mockito.Mockito.doReturn;
import static org.mockito.Mockito.when;

import android.test.suitebuilder.annotation.SmallTest;

import com.android.server.telecom.Call;
import com.android.server.telecom.CallState;
import com.android.server.telecom.InCallTonePlayer;
import com.android.server.telecom.RingbackPlayer;

import org.junit.Before;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
import org.mockito.Mock;

import java.util.concurrent.CountDownLatch;

@RunWith(JUnit4.class)
public class RingbackPlayerTest extends TelecomTestCase {
    @Mock InCallTonePlayer.Factory mFactory;
    @Mock Call mCall;
    @Mock InCallTonePlayer mTonePlayer;

    private RingbackPlayer mRingbackPlayer;

    @Before
    @Override
    public void setUp() throws Exception {
        super.setUp();
        when(mFactory.createPlayer(anyInt())).thenReturn(mTonePlayer);
        mRingbackPlayer = new RingbackPlayer(mFactory);
    }

    @SmallTest
    @Test
    public void testPlayerSync() {
        // make sure InCallTonePlayer try to start playing the tone after RingbackPlayer receives
        // stop tone request.
        CountDownLatch latch = new CountDownLatch(1);
        doReturn(CallState.DIALING).when(mCall).getState();
        doAnswer(x -> {
            new Thread(() -> {
                try {
                    latch.wait();
                } catch (InterruptedException e) {
                    // Ignore
                }
            }).start();
            return true;
        }).when(mTonePlayer).startTone();

        mRingbackPlayer.startRingbackForCall(mCall);
        mRingbackPlayer.stopRingbackForCall(mCall);
        assertFalse(mRingbackPlayer.isRingbackPlaying());
        latch.countDown();
    }
}