#include <memory>
#include <utility>
#include "base/functional/bind.h"
#include "base/memory/raw_ptr.h"
#include "base/synchronization/waitable_event.h"
#include "base/task/sequenced_task_runner.h"
#include "base/test/task_environment.h"
#include "base/threading/thread.h"
#include "base/time/time.h"
#include "media/audio/alive_checker.h"
#include "testing/gtest/include/gtest/gtest.h"
namespace media {
namespace {
constexpr int kCheckIntervalMs = 10;
constexpr int kTimeoutMs = 50;
}
class MockPowerObserverHelper : public PowerObserverHelper {
public:
MockPowerObserverHelper(scoped_refptr<base::SequencedTaskRunner> task_runner,
base::RepeatingClosure suspend_callback,
base::RepeatingClosure resume_callback)
: PowerObserverHelper(std::move(task_runner),
std::move(suspend_callback),
std::move(resume_callback)) {}
bool IsSuspending() const override {
DCHECK(TaskRunnerForTesting()->RunsTasksInCurrentSequence());
return is_suspending_;
}
void Suspend() {
DCHECK(TaskRunnerForTesting()->RunsTasksInCurrentSequence());
is_suspending_ = true;
SuspendCallbackForTesting()->Run();
}
void Resume() {
DCHECK(TaskRunnerForTesting()->RunsTasksInCurrentSequence());
is_suspending_ = false;
ResumeCallbackForTesting()->Run();
}
private:
bool is_suspending_ = false;
};
class AliveCheckerTest : public testing::Test {
public:
AliveCheckerTest()
: alive_checker_thread_("AliveCheckerThread"),
detected_dead_event_(base::WaitableEvent::ResetPolicy::MANUAL,
base::WaitableEvent::InitialState::NOT_SIGNALED) {
alive_checker_thread_.StartAndWaitForTesting();
}
AliveCheckerTest(const AliveCheckerTest&) = delete;
AliveCheckerTest& operator=(const AliveCheckerTest&) = delete;
void OnDetectedDead() {
EXPECT_TRUE(alive_checker_thread_.task_runner()->BelongsToCurrentThread());
detected_dead_event_.Signal();
}
std::unique_ptr<PowerObserverHelper> CreatePowerObserverHelper(
scoped_refptr<base::SequencedTaskRunner> task_runner,
base::RepeatingClosure suspend_callback,
base::RepeatingClosure resume_callback) {
std::unique_ptr<MockPowerObserverHelper> mock_power_observer_helper =
std::make_unique<MockPowerObserverHelper>(std::move(task_runner),
std::move(suspend_callback),
std::move(resume_callback));
mock_power_observer_helper_ = mock_power_observer_helper.get();
return mock_power_observer_helper;
}
protected:
~AliveCheckerTest() override {
base::WaitableEvent done(base::WaitableEvent::ResetPolicy::MANUAL,
base::WaitableEvent::InitialState::NOT_SIGNALED);
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE,
base::BindOnce(&AliveCheckerTest::ResetAliveCheckerOnAliveCheckerThread,
base::Unretained(this), &done));
done.Wait();
}
void CreateAliveChecker(bool stop_at_first_alive_notification,
bool pause_check_during_suspend) {
base::WaitableEvent done(base::WaitableEvent::ResetPolicy::MANUAL,
base::WaitableEvent::InitialState::NOT_SIGNALED);
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE,
base::BindOnce(
&AliveCheckerTest::CreateAliveCheckerOnAliveCheckerThread,
base::Unretained(this), stop_at_first_alive_notification,
pause_check_during_suspend, &done));
done.Wait();
}
void StartAliveChecker() {
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&AliveChecker::Start,
base::Unretained(alive_checker_.get())));
}
void StopAliveChecker() {
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&AliveChecker::Stop,
base::Unretained(alive_checker_.get())));
}
void NotifyAliveMultipleTimes(int remaining_notifications,
base::TimeDelta delay) {
alive_checker_->NotifyAlive();
if (remaining_notifications > 1) {
alive_checker_thread_.task_runner()->PostDelayedTask(
FROM_HERE,
base::BindOnce(&AliveCheckerTest::NotifyAliveMultipleTimes,
base::Unretained(this), remaining_notifications - 1,
delay),
delay);
}
}
void WaitUntilDetectedDead() {
detected_dead_event_.Wait();
detected_dead_event_.Reset();
}
bool WaitUntilDetectedDeadWithTimeout(base::TimeDelta timeout) {
bool signaled = detected_dead_event_.TimedWait(timeout);
detected_dead_event_.Reset();
return signaled;
}
bool GetDetectedDead() {
bool detected_dead = false;
base::WaitableEvent done(base::WaitableEvent::ResetPolicy::MANUAL,
base::WaitableEvent::InitialState::NOT_SIGNALED);
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE,
base::BindOnce(&AliveCheckerTest::GetDetectedDeadOnAliveCheckerThread,
base::Unretained(this), &detected_dead, &done));
done.Wait();
return detected_dead;
}
base::test::TaskEnvironment task_environment_;
base::Thread alive_checker_thread_;
std::unique_ptr<AliveChecker> alive_checker_;
raw_ptr<MockPowerObserverHelper> mock_power_observer_helper_;
private:
void CreateAliveCheckerOnAliveCheckerThread(
bool stop_at_first_alive_notification,
bool pause_check_during_suspend,
base::WaitableEvent* done) {
EXPECT_TRUE(alive_checker_thread_.task_runner()->BelongsToCurrentThread());
if (pause_check_during_suspend) {
alive_checker_ = std::make_unique<AliveChecker>(
base::BindRepeating(&AliveCheckerTest::OnDetectedDead,
base::Unretained(this)),
base::Milliseconds(kCheckIntervalMs), base::Milliseconds(kTimeoutMs),
stop_at_first_alive_notification,
base::BindOnce(&AliveCheckerTest::CreatePowerObserverHelper,
base::Unretained(this)));
} else {
alive_checker_ = std::make_unique<AliveChecker>(
base::BindRepeating(&AliveCheckerTest::OnDetectedDead,
base::Unretained(this)),
base::Milliseconds(kCheckIntervalMs), base::Milliseconds(kTimeoutMs),
stop_at_first_alive_notification, false);
}
done->Signal();
}
void GetDetectedDeadOnAliveCheckerThread(bool* detected_dead,
base::WaitableEvent* done) {
EXPECT_TRUE(alive_checker_thread_.task_runner()->BelongsToCurrentThread());
*detected_dead = alive_checker_->DetectedDead();
done->Signal();
}
void ResetAliveCheckerOnAliveCheckerThread(base::WaitableEvent* done) {
EXPECT_TRUE(alive_checker_thread_.task_runner()->BelongsToCurrentThread());
mock_power_observer_helper_ = nullptr;
alive_checker_.reset();
done->Signal();
}
base::WaitableEvent detected_dead_event_;
};
TEST_F(AliveCheckerTest, NoAliveNotificationsDetectTwice) {
CreateAliveChecker(false, false);
StartAliveChecker();
EXPECT_FALSE(GetDetectedDead());
WaitUntilDetectedDead();
EXPECT_TRUE(GetDetectedDead());
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_TRUE(GetDetectedDead());
StartAliveChecker();
EXPECT_FALSE(GetDetectedDead());
WaitUntilDetectedDead();
EXPECT_TRUE(GetDetectedDead());
}
TEST_F(AliveCheckerTest, StopAtFirstAliveNotification_DoNotify) {
CreateAliveChecker(true, false);
StartAliveChecker();
alive_checker_->NotifyAlive();
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
}
TEST_F(AliveCheckerTest, StopAtFirstAliveNotification_DontNotify) {
CreateAliveChecker(true, false);
StartAliveChecker();
WaitUntilDetectedDead();
EXPECT_TRUE(GetDetectedDead());
}
TEST_F(AliveCheckerTest, SuspendResume_StartBetweenSuspendAndResume) {
CreateAliveChecker(false, true);
ASSERT_TRUE(mock_power_observer_helper_);
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Suspend,
base::Unretained(mock_power_observer_helper_)));
StartAliveChecker();
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Resume,
base::Unretained(mock_power_observer_helper_)));
WaitUntilDetectedDead();
EXPECT_TRUE(GetDetectedDead());
}
TEST_F(AliveCheckerTest, SuspendResumeWithAutoStop_NotifyBeforeSuspend) {
CreateAliveChecker(true, true);
ASSERT_TRUE(mock_power_observer_helper_);
StartAliveChecker();
alive_checker_->NotifyAlive();
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Suspend,
base::Unretained(mock_power_observer_helper_)));
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Resume,
base::Unretained(mock_power_observer_helper_)));
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
}
TEST_F(AliveCheckerTest,
SuspendResumeWithAutoStop_NotifyBeforeSuspendAndRestart) {
CreateAliveChecker(true, true);
ASSERT_TRUE(mock_power_observer_helper_);
StartAliveChecker();
alive_checker_->NotifyAlive();
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
StartAliveChecker();
EXPECT_FALSE(GetDetectedDead());
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Suspend,
base::Unretained(mock_power_observer_helper_)));
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Resume,
base::Unretained(mock_power_observer_helper_)));
WaitUntilDetectedDead();
EXPECT_TRUE(GetDetectedDead());
}
TEST_F(AliveCheckerTest,
SuspendResumeWithAutoStop_NotifyBetweenSuspendAndResume) {
CreateAliveChecker(true, true);
ASSERT_TRUE(mock_power_observer_helper_);
StartAliveChecker();
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Suspend,
base::Unretained(mock_power_observer_helper_)));
alive_checker_->NotifyAlive();
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Resume,
base::Unretained(mock_power_observer_helper_)));
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
}
TEST_F(AliveCheckerTest, SuspendResumeWithAutoStop_NotifyAfterResume) {
CreateAliveChecker(true, true);
ASSERT_TRUE(mock_power_observer_helper_);
StartAliveChecker();
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Suspend,
base::Unretained(mock_power_observer_helper_)));
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Resume,
base::Unretained(mock_power_observer_helper_)));
alive_checker_->NotifyAlive();
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
}
TEST_F(AliveCheckerTest, SuspendResumeWithAutoStop_DontNotify) {
CreateAliveChecker(true, true);
ASSERT_TRUE(mock_power_observer_helper_);
StartAliveChecker();
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Suspend,
base::Unretained(mock_power_observer_helper_)));
EXPECT_FALSE(WaitUntilDetectedDeadWithTimeout(
base::Milliseconds(kTimeoutMs + kCheckIntervalMs + 10)));
EXPECT_FALSE(GetDetectedDead());
alive_checker_thread_.task_runner()->PostTask(
FROM_HERE, base::BindOnce(&MockPowerObserverHelper::Resume,
base::Unretained(mock_power_observer_helper_)));
WaitUntilDetectedDead();
EXPECT_TRUE(GetDetectedDead());
}
}