/* * Copyright (c) 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 "system_wrappers/interface/condition_variable_wrapper.h" #include "gtest/gtest.h" #include "system_wrappers/interface/critical_section_wrapper.h" #include "system_wrappers/interface/thread_wrapper.h" #include "system_wrappers/interface/trace.h" #include "system_wrappers/source/unittest_utilities.h" namespace webrtc { namespace { const int kLogTrace = false; // Set to true to enable debug logging to stdout. const int kLongWaitMs = 100*1000; // A long time in testing terms const int kShortWaitMs = 2*1000; // Long enough for process switches to happen #define LOG(...) WEBRTC_TRACE(kTraceStateInfo, kTraceUtility, -1, __VA_ARGS__); // A Baton is one possible control structure one can build using // conditional variables. // A Baton is always held by one and only one active thread - unlike // a lock, it can never be free. // One can pass it or grab it - both calls have timeouts. // Note - a production tool would guard against passing it without // grabbing it first. This one is for testing, so it doesn't. class Baton { public: Baton() : giver_sect_(CriticalSectionWrapper::CreateCriticalSection()), crit_sect_(CriticalSectionWrapper::CreateCriticalSection()), cond_var_(ConditionVariableWrapper::CreateConditionVariable()), being_passed_(false), pass_count_(0) { } ~Baton() { delete giver_sect_; delete crit_sect_; delete cond_var_; } // Pass the baton. Returns false if baton is not picked up in |max_msecs|. // Only one process can pass at the same time; this property is // ensured by the |giver_sect_| lock. bool Pass(WebRtc_UWord32 max_msecs) { LOG("Locking giver_sect"); CriticalSectionScoped cs_giver(giver_sect_); LOG("Locked giver_sect, locking crit_sect"); CriticalSectionScoped cs(crit_sect_); SignalBatonAvailable(); const bool result = TakeBatonIfStillFree(max_msecs); if (result) { ++pass_count_; LOG("Pass count is %d", pass_count_); } return result; } // Grab the baton. Returns false if baton is not passed. bool Grab(WebRtc_UWord32 max_msecs) { CriticalSectionScoped cs(crit_sect_); return WaitUntilBatonOffered(max_msecs); } int PassCount() { // We don't allow polling PassCount() during a Pass()-call since there is // no guarantee that |pass_count_| is incremented until the Pass()-call // finishes. I.e. the Grab()-call may finish before |pass_count_| has been // incremented. // Thus, this function waits on giver_sect_. CriticalSectionScoped cs(giver_sect_); return pass_count_; } private: // Wait/Signal forms a classical semaphore on |being_passed_|. // These functions must be called with crit_sect_ held. bool WaitUntilBatonOffered(int timeout_ms) { while (!being_passed_) { LOG("Wait waiting"); if (!cond_var_->SleepCS(*crit_sect_, timeout_ms)) { LOG("Wait timeout"); return false; } } being_passed_ = false; cond_var_->Wake(); return true; } void SignalBatonAvailable() { assert(!being_passed_); being_passed_ = true; LOG("Signal waking"); cond_var_->Wake(); } // Timeout extension: Wait for a limited time for someone else to // take it, and take it if it's not taken. // Returns true if resource is taken by someone else, false // if it is taken back by the caller. // This function must be called with both |giver_sect_| and // |crit_sect_| held. bool TakeBatonIfStillFree(int timeout_ms) { bool not_timeout = true; while (being_passed_ && not_timeout) { LOG("Takeback waiting"); not_timeout = cond_var_->SleepCS(*crit_sect_, timeout_ms); // If we're woken up while variable is still held, we may have // gotten a wakeup destined for a grabber thread. // This situation is not treated specially here. } if (!being_passed_) { return true; } else { LOG("Takeback grab"); assert(!not_timeout); being_passed_ = false; return false; } } // Lock that ensures that there is only one thread in the active // part of Pass() at a time. // |giver_sect_| must always be acquired before |cond_var_|. CriticalSectionWrapper* giver_sect_; // Lock that protects |being_passed_|. CriticalSectionWrapper* crit_sect_; ConditionVariableWrapper* cond_var_; bool being_passed_; // Statistics information: Number of successfull passes. int pass_count_; }; // Function that waits on a Baton, and passes it right back. // We expect these calls never to time out. bool WaitingRunFunction(void* obj) { Baton* the_baton = static_cast (obj); LOG("Thread waiting"); EXPECT_TRUE(the_baton->Grab(kLongWaitMs)); LOG("Thread waking parent"); EXPECT_TRUE(the_baton->Pass(kLongWaitMs)); return true; } class CondVarTest : public ::testing::Test { public: CondVarTest() : trace_(kLogTrace) { } virtual void SetUp() { thread_ = ThreadWrapper::CreateThread(&WaitingRunFunction, &baton_); unsigned int id = 42; ASSERT_TRUE(thread_->Start(id)); } virtual void TearDown() { // We have to wake the thread in order to make it obey the stop order. // But we don't know if the thread has completed the run function, so // we don't know if it will exit before or after the Pass. // Thus, we need to pin it down inside its Run function (between Grab // and Pass). ASSERT_TRUE(baton_.Pass(kShortWaitMs)); thread_->SetNotAlive(); ASSERT_TRUE(baton_.Grab(kShortWaitMs)); ASSERT_TRUE(thread_->Stop()); delete thread_; } protected: Baton baton_; private: ScopedTracing trace_; ThreadWrapper* thread_; }; // The SetUp and TearDown functions use condition variables. // This test verifies those pieces in isolation. TEST_F(CondVarTest, InitFunctionsWork) { // All relevant asserts are in the SetUp and TearDown functions. } // This test verifies that one can use the baton multiple times. TEST_F(CondVarTest, PassBatonMultipleTimes) { const int kNumberOfRounds = 2; for (int i = 0; i < kNumberOfRounds; ++i) { ASSERT_TRUE(baton_.Pass(kShortWaitMs)); ASSERT_TRUE(baton_.Grab(kShortWaitMs)); } EXPECT_EQ(2*kNumberOfRounds, baton_.PassCount()); } } // anonymous namespace } // namespace webrtc