mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-12 21:30:45 +01:00

Bug: webrtc:12338 Change-Id: I72fcb505a92f03b2ace7160ee33d555a977eddfd Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/226955 Reviewed-by: Harald Alvestrand <hta@webrtc.org> Commit-Queue: Artem Titov <titovartem@webrtc.org> Cr-Commit-Position: refs/heads/master@{#34587}
182 lines
4.9 KiB
C++
182 lines
4.9 KiB
C++
/*
|
|
* Copyright 2019 The WebRTC project authors. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
#include "rtc_base/task_utils/pending_task_safety_flag.h"
|
|
|
|
#include <memory>
|
|
|
|
#include "rtc_base/event.h"
|
|
#include "rtc_base/logging.h"
|
|
#include "rtc_base/task_queue_for_test.h"
|
|
#include "rtc_base/task_utils/to_queued_task.h"
|
|
#include "test/gmock.h"
|
|
#include "test/gtest.h"
|
|
|
|
namespace webrtc {
|
|
namespace {
|
|
using ::testing::AtLeast;
|
|
using ::testing::Invoke;
|
|
using ::testing::MockFunction;
|
|
using ::testing::NiceMock;
|
|
using ::testing::Return;
|
|
} // namespace
|
|
|
|
TEST(PendingTaskSafetyFlagTest, Basic) {
|
|
rtc::scoped_refptr<PendingTaskSafetyFlag> safety_flag;
|
|
{
|
|
// Scope for the `owner` instance.
|
|
class Owner {
|
|
public:
|
|
Owner() = default;
|
|
~Owner() { flag_->SetNotAlive(); }
|
|
|
|
rtc::scoped_refptr<PendingTaskSafetyFlag> flag_ =
|
|
PendingTaskSafetyFlag::Create();
|
|
} owner;
|
|
EXPECT_TRUE(owner.flag_->alive());
|
|
safety_flag = owner.flag_;
|
|
EXPECT_TRUE(safety_flag->alive());
|
|
}
|
|
// `owner` now out of scope.
|
|
EXPECT_FALSE(safety_flag->alive());
|
|
}
|
|
|
|
TEST(PendingTaskSafetyFlagTest, BasicScoped) {
|
|
rtc::scoped_refptr<PendingTaskSafetyFlag> safety_flag;
|
|
{
|
|
struct Owner {
|
|
ScopedTaskSafety safety;
|
|
} owner;
|
|
safety_flag = owner.safety.flag();
|
|
EXPECT_TRUE(safety_flag->alive());
|
|
}
|
|
// `owner` now out of scope.
|
|
EXPECT_FALSE(safety_flag->alive());
|
|
}
|
|
|
|
TEST(PendingTaskSafetyFlagTest, PendingTaskSuccess) {
|
|
TaskQueueForTest tq1("OwnerHere");
|
|
TaskQueueForTest tq2("OwnerNotHere");
|
|
|
|
class Owner {
|
|
public:
|
|
Owner() : tq_main_(TaskQueueBase::Current()) { RTC_DCHECK(tq_main_); }
|
|
~Owner() {
|
|
RTC_DCHECK(tq_main_->IsCurrent());
|
|
flag_->SetNotAlive();
|
|
}
|
|
|
|
void DoStuff() {
|
|
RTC_DCHECK(!tq_main_->IsCurrent());
|
|
tq_main_->PostTask(ToQueuedTask([safe = flag_, this]() {
|
|
if (!safe->alive())
|
|
return;
|
|
stuff_done_ = true;
|
|
}));
|
|
}
|
|
|
|
bool stuff_done() const { return stuff_done_; }
|
|
|
|
private:
|
|
TaskQueueBase* const tq_main_;
|
|
bool stuff_done_ = false;
|
|
rtc::scoped_refptr<PendingTaskSafetyFlag> flag_{
|
|
PendingTaskSafetyFlag::Create()};
|
|
};
|
|
|
|
std::unique_ptr<Owner> owner;
|
|
tq1.SendTask(
|
|
[&owner]() {
|
|
owner.reset(new Owner());
|
|
EXPECT_FALSE(owner->stuff_done());
|
|
},
|
|
RTC_FROM_HERE);
|
|
ASSERT_TRUE(owner);
|
|
tq2.SendTask([&owner]() { owner->DoStuff(); }, RTC_FROM_HERE);
|
|
tq1.SendTask(
|
|
[&owner]() {
|
|
EXPECT_TRUE(owner->stuff_done());
|
|
owner.reset();
|
|
},
|
|
RTC_FROM_HERE);
|
|
ASSERT_FALSE(owner);
|
|
}
|
|
|
|
TEST(PendingTaskSafetyFlagTest, PendingTaskDropped) {
|
|
TaskQueueForTest tq1("OwnerHere");
|
|
TaskQueueForTest tq2("OwnerNotHere");
|
|
|
|
class Owner {
|
|
public:
|
|
explicit Owner(bool* stuff_done)
|
|
: tq_main_(TaskQueueBase::Current()), stuff_done_(stuff_done) {
|
|
RTC_DCHECK(tq_main_);
|
|
*stuff_done_ = false;
|
|
}
|
|
~Owner() {
|
|
RTC_DCHECK(tq_main_->IsCurrent());
|
|
}
|
|
|
|
void DoStuff() {
|
|
RTC_DCHECK(!tq_main_->IsCurrent());
|
|
tq_main_->PostTask(
|
|
ToQueuedTask(safety_, [this]() { *stuff_done_ = true; }));
|
|
}
|
|
|
|
private:
|
|
TaskQueueBase* const tq_main_;
|
|
bool* const stuff_done_;
|
|
ScopedTaskSafety safety_;
|
|
};
|
|
|
|
std::unique_ptr<Owner> owner;
|
|
bool stuff_done = false;
|
|
tq1.SendTask([&owner, &stuff_done]() { owner.reset(new Owner(&stuff_done)); },
|
|
RTC_FROM_HERE);
|
|
ASSERT_TRUE(owner);
|
|
// Queue up a task on tq1 that will execute before the 'DoStuff' task
|
|
// can, and delete the `owner` before the 'stuff' task can execute.
|
|
rtc::Event blocker;
|
|
tq1.PostTask([&blocker, &owner]() {
|
|
blocker.Wait(rtc::Event::kForever);
|
|
owner.reset();
|
|
});
|
|
|
|
// Queue up a DoStuff...
|
|
tq2.SendTask([&owner]() { owner->DoStuff(); }, RTC_FROM_HERE);
|
|
|
|
ASSERT_TRUE(owner);
|
|
blocker.Set();
|
|
|
|
// Run an empty task on tq1 to flush all the queued tasks.
|
|
tq1.WaitForPreviouslyPostedTasks();
|
|
ASSERT_FALSE(owner);
|
|
EXPECT_FALSE(stuff_done);
|
|
}
|
|
|
|
TEST(PendingTaskSafetyFlagTest, PendingTaskNotAliveInitialized) {
|
|
TaskQueueForTest tq("PendingTaskNotAliveInitialized");
|
|
|
|
// Create a new flag that initially not `alive`.
|
|
auto flag = PendingTaskSafetyFlag::CreateDetachedInactive();
|
|
tq.SendTask([&flag]() { EXPECT_FALSE(flag->alive()); }, RTC_FROM_HERE);
|
|
|
|
bool task_1_ran = false;
|
|
bool task_2_ran = false;
|
|
tq.PostTask(ToQueuedTask(flag, [&task_1_ran]() { task_1_ran = true; }));
|
|
tq.PostTask([&flag]() { flag->SetAlive(); });
|
|
tq.PostTask(ToQueuedTask(flag, [&task_2_ran]() { task_2_ran = true; }));
|
|
|
|
tq.WaitForPreviouslyPostedTasks();
|
|
EXPECT_FALSE(task_1_ran);
|
|
EXPECT_TRUE(task_2_ran);
|
|
}
|
|
|
|
} // namespace webrtc
|