251 lines
6.9 KiB
C++
251 lines
6.9 KiB
C++
|
/*
|
||
|
* Copyright 2012 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 "webrtc/base/sigslot.h"
|
||
|
|
||
|
#include "webrtc/base/gunit.h"
|
||
|
|
||
|
// This function, when passed a has_slots or signalx, will break the build if
|
||
|
// its threading requirement is not single threaded
|
||
|
static bool TemplateIsST(const sigslot::single_threaded* p) {
|
||
|
return true;
|
||
|
}
|
||
|
// This function, when passed a has_slots or signalx, will break the build if
|
||
|
// its threading requirement is not multi threaded
|
||
|
static bool TemplateIsMT(const sigslot::multi_threaded_local* p) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
class SigslotDefault : public testing::Test, public sigslot::has_slots<> {
|
||
|
protected:
|
||
|
sigslot::signal0<> signal_;
|
||
|
};
|
||
|
|
||
|
template<class slot_policy = sigslot::single_threaded,
|
||
|
class signal_policy = sigslot::single_threaded>
|
||
|
class SigslotReceiver : public sigslot::has_slots<slot_policy> {
|
||
|
public:
|
||
|
SigslotReceiver() : signal_(NULL), signal_count_(0) {
|
||
|
}
|
||
|
~SigslotReceiver() {
|
||
|
}
|
||
|
|
||
|
void Connect(sigslot::signal0<signal_policy>* signal) {
|
||
|
if (!signal) return;
|
||
|
Disconnect();
|
||
|
signal_ = signal;
|
||
|
signal->connect(this,
|
||
|
&SigslotReceiver<slot_policy, signal_policy>::OnSignal);
|
||
|
}
|
||
|
void Disconnect() {
|
||
|
if (!signal_) return;
|
||
|
signal_->disconnect(this);
|
||
|
signal_ = NULL;
|
||
|
}
|
||
|
void OnSignal() {
|
||
|
++signal_count_;
|
||
|
}
|
||
|
int signal_count() { return signal_count_; }
|
||
|
|
||
|
private:
|
||
|
sigslot::signal0<signal_policy>* signal_;
|
||
|
int signal_count_;
|
||
|
};
|
||
|
|
||
|
template<class slot_policy = sigslot::single_threaded,
|
||
|
class mt_signal_policy = sigslot::multi_threaded_local>
|
||
|
class SigslotSlotTest : public testing::Test {
|
||
|
protected:
|
||
|
SigslotSlotTest() {
|
||
|
mt_signal_policy mt_policy;
|
||
|
TemplateIsMT(&mt_policy);
|
||
|
}
|
||
|
|
||
|
virtual void SetUp() {
|
||
|
Connect();
|
||
|
}
|
||
|
virtual void TearDown() {
|
||
|
Disconnect();
|
||
|
}
|
||
|
|
||
|
void Disconnect() {
|
||
|
st_receiver_.Disconnect();
|
||
|
mt_receiver_.Disconnect();
|
||
|
}
|
||
|
|
||
|
void Connect() {
|
||
|
st_receiver_.Connect(&SignalSTLoopback);
|
||
|
mt_receiver_.Connect(&SignalMTLoopback);
|
||
|
}
|
||
|
|
||
|
int st_loop_back_count() { return st_receiver_.signal_count(); }
|
||
|
int mt_loop_back_count() { return mt_receiver_.signal_count(); }
|
||
|
|
||
|
sigslot::signal0<> SignalSTLoopback;
|
||
|
SigslotReceiver<slot_policy, sigslot::single_threaded> st_receiver_;
|
||
|
sigslot::signal0<mt_signal_policy> SignalMTLoopback;
|
||
|
SigslotReceiver<slot_policy, mt_signal_policy> mt_receiver_;
|
||
|
};
|
||
|
|
||
|
typedef SigslotSlotTest<> SigslotSTSlotTest;
|
||
|
typedef SigslotSlotTest<sigslot::multi_threaded_local,
|
||
|
sigslot::multi_threaded_local> SigslotMTSlotTest;
|
||
|
|
||
|
class multi_threaded_local_fake : public sigslot::multi_threaded_local {
|
||
|
public:
|
||
|
multi_threaded_local_fake() : lock_count_(0), unlock_count_(0) {
|
||
|
}
|
||
|
|
||
|
virtual void lock() {
|
||
|
++lock_count_;
|
||
|
}
|
||
|
virtual void unlock() {
|
||
|
++unlock_count_;
|
||
|
}
|
||
|
|
||
|
int lock_count() { return lock_count_; }
|
||
|
|
||
|
bool InCriticalSection() { return lock_count_ != unlock_count_; }
|
||
|
|
||
|
protected:
|
||
|
int lock_count_;
|
||
|
int unlock_count_;
|
||
|
};
|
||
|
|
||
|
typedef SigslotSlotTest<multi_threaded_local_fake,
|
||
|
multi_threaded_local_fake> SigslotMTLockBase;
|
||
|
|
||
|
class SigslotMTLockTest : public SigslotMTLockBase {
|
||
|
protected:
|
||
|
SigslotMTLockTest() {}
|
||
|
|
||
|
virtual void SetUp() {
|
||
|
EXPECT_EQ(0, SlotLockCount());
|
||
|
SigslotMTLockBase::SetUp();
|
||
|
// Connects to two signals (ST and MT). However,
|
||
|
// SlotLockCount() only gets the count for the
|
||
|
// MT signal (there are two separate SigslotReceiver which
|
||
|
// keep track of their own count).
|
||
|
EXPECT_EQ(1, SlotLockCount());
|
||
|
}
|
||
|
virtual void TearDown() {
|
||
|
const int previous_lock_count = SlotLockCount();
|
||
|
SigslotMTLockBase::TearDown();
|
||
|
// Disconnects from two signals. Note analogous to SetUp().
|
||
|
EXPECT_EQ(previous_lock_count + 1, SlotLockCount());
|
||
|
}
|
||
|
|
||
|
int SlotLockCount() { return mt_receiver_.lock_count(); }
|
||
|
void Signal() { SignalMTLoopback(); }
|
||
|
int SignalLockCount() { return SignalMTLoopback.lock_count(); }
|
||
|
int signal_count() { return mt_loop_back_count(); }
|
||
|
bool InCriticalSection() { return SignalMTLoopback.InCriticalSection(); }
|
||
|
};
|
||
|
|
||
|
// This test will always succeed. However, if the default template instantiation
|
||
|
// changes from single threaded to multi threaded it will break the build here.
|
||
|
TEST_F(SigslotDefault, DefaultIsST) {
|
||
|
EXPECT_TRUE(TemplateIsST(this));
|
||
|
EXPECT_TRUE(TemplateIsST(&signal_));
|
||
|
}
|
||
|
|
||
|
// ST slot, ST signal
|
||
|
TEST_F(SigslotSTSlotTest, STLoopbackTest) {
|
||
|
SignalSTLoopback();
|
||
|
EXPECT_EQ(1, st_loop_back_count());
|
||
|
EXPECT_EQ(0, mt_loop_back_count());
|
||
|
}
|
||
|
|
||
|
// ST slot, MT signal
|
||
|
TEST_F(SigslotSTSlotTest, MTLoopbackTest) {
|
||
|
SignalMTLoopback();
|
||
|
EXPECT_EQ(1, mt_loop_back_count());
|
||
|
EXPECT_EQ(0, st_loop_back_count());
|
||
|
}
|
||
|
|
||
|
// ST slot, both ST and MT (separate) signal
|
||
|
TEST_F(SigslotSTSlotTest, AllLoopbackTest) {
|
||
|
SignalSTLoopback();
|
||
|
SignalMTLoopback();
|
||
|
EXPECT_EQ(1, mt_loop_back_count());
|
||
|
EXPECT_EQ(1, st_loop_back_count());
|
||
|
}
|
||
|
|
||
|
TEST_F(SigslotSTSlotTest, Reconnect) {
|
||
|
SignalSTLoopback();
|
||
|
SignalMTLoopback();
|
||
|
EXPECT_EQ(1, mt_loop_back_count());
|
||
|
EXPECT_EQ(1, st_loop_back_count());
|
||
|
Disconnect();
|
||
|
SignalSTLoopback();
|
||
|
SignalMTLoopback();
|
||
|
EXPECT_EQ(1, mt_loop_back_count());
|
||
|
EXPECT_EQ(1, st_loop_back_count());
|
||
|
Connect();
|
||
|
SignalSTLoopback();
|
||
|
SignalMTLoopback();
|
||
|
EXPECT_EQ(2, mt_loop_back_count());
|
||
|
EXPECT_EQ(2, st_loop_back_count());
|
||
|
}
|
||
|
|
||
|
// MT slot, ST signal
|
||
|
TEST_F(SigslotMTSlotTest, STLoopbackTest) {
|
||
|
SignalSTLoopback();
|
||
|
EXPECT_EQ(1, st_loop_back_count());
|
||
|
EXPECT_EQ(0, mt_loop_back_count());
|
||
|
}
|
||
|
|
||
|
// MT slot, MT signal
|
||
|
TEST_F(SigslotMTSlotTest, MTLoopbackTest) {
|
||
|
SignalMTLoopback();
|
||
|
EXPECT_EQ(1, mt_loop_back_count());
|
||
|
EXPECT_EQ(0, st_loop_back_count());
|
||
|
}
|
||
|
|
||
|
// MT slot, both ST and MT (separate) signal
|
||
|
TEST_F(SigslotMTSlotTest, AllLoopbackTest) {
|
||
|
SignalMTLoopback();
|
||
|
SignalSTLoopback();
|
||
|
EXPECT_EQ(1, st_loop_back_count());
|
||
|
EXPECT_EQ(1, mt_loop_back_count());
|
||
|
}
|
||
|
|
||
|
// Test that locks are acquired and released correctly.
|
||
|
TEST_F(SigslotMTLockTest, LockSanity) {
|
||
|
const int lock_count = SignalLockCount();
|
||
|
Signal();
|
||
|
EXPECT_FALSE(InCriticalSection());
|
||
|
EXPECT_EQ(lock_count + 1, SignalLockCount());
|
||
|
EXPECT_EQ(1, signal_count());
|
||
|
}
|
||
|
|
||
|
// Destroy signal and slot in different orders.
|
||
|
TEST(DestructionOrder, SignalFirst) {
|
||
|
sigslot::signal0<>* signal = new sigslot::signal0<>;
|
||
|
SigslotReceiver<>* receiver = new SigslotReceiver<>();
|
||
|
receiver->Connect(signal);
|
||
|
(*signal)();
|
||
|
EXPECT_EQ(1, receiver->signal_count());
|
||
|
delete signal;
|
||
|
delete receiver;
|
||
|
}
|
||
|
|
||
|
TEST(DestructionOrder, SlotFirst) {
|
||
|
sigslot::signal0<>* signal = new sigslot::signal0<>;
|
||
|
SigslotReceiver<>* receiver = new SigslotReceiver<>();
|
||
|
receiver->Connect(signal);
|
||
|
(*signal)();
|
||
|
EXPECT_EQ(1, receiver->signal_count());
|
||
|
|
||
|
delete receiver;
|
||
|
(*signal)();
|
||
|
delete signal;
|
||
|
}
|