| /* |
| * Copyright 2004 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/event.h" |
| |
| #include "api/units/time_delta.h" |
| #include "rtc_base/platform_thread.h" |
| #include "system_wrappers/include/clock.h" |
| #include "test/gtest.h" |
| |
| namespace rtc { |
| |
| TEST(EventTest, InitiallySignaled) { |
| Event event(false, true); |
| ASSERT_TRUE(event.Wait(webrtc::TimeDelta::Zero())); |
| } |
| |
| TEST(EventTest, ManualReset) { |
| Event event(true, false); |
| ASSERT_FALSE(event.Wait(webrtc::TimeDelta::Zero())); |
| |
| event.Set(); |
| ASSERT_TRUE(event.Wait(webrtc::TimeDelta::Zero())); |
| ASSERT_TRUE(event.Wait(webrtc::TimeDelta::Zero())); |
| |
| event.Reset(); |
| ASSERT_FALSE(event.Wait(webrtc::TimeDelta::Zero())); |
| } |
| |
| TEST(EventTest, AutoReset) { |
| Event event; |
| ASSERT_FALSE(event.Wait(webrtc::TimeDelta::Zero())); |
| |
| event.Set(); |
| ASSERT_TRUE(event.Wait(webrtc::TimeDelta::Zero())); |
| ASSERT_FALSE(event.Wait(webrtc::TimeDelta::Zero())); |
| } |
| |
| class SignalerThread { |
| public: |
| void Start(Event* writer, Event* reader) { |
| writer_ = writer; |
| reader_ = reader; |
| thread_ = PlatformThread::SpawnJoinable( |
| [this] { |
| while (!stop_event_.Wait(webrtc::TimeDelta::Zero())) { |
| writer_->Set(); |
| reader_->Wait(Event::kForever); |
| } |
| }, |
| "EventPerf"); |
| } |
| void Stop() { |
| stop_event_.Set(); |
| thread_.Finalize(); |
| } |
| Event stop_event_; |
| Event* writer_; |
| Event* reader_; |
| PlatformThread thread_; |
| }; |
| |
| TEST(EventTest, UnsignaledWaitDoesNotReturnBeforeTimeout) { |
| constexpr webrtc::TimeDelta kDuration = webrtc::TimeDelta::Micros(10'499); |
| Event event; |
| auto begin = webrtc::Clock::GetRealTimeClock()->CurrentTime(); |
| EXPECT_FALSE(event.Wait(kDuration)); |
| EXPECT_GE(webrtc::Clock::GetRealTimeClock()->CurrentTime(), |
| begin + kDuration); |
| } |
| |
| // These tests are disabled by default and only intended to be run manually. |
| TEST(EventTest, DISABLED_PerformanceSingleThread) { |
| static const int kNumIterations = 10000000; |
| Event event; |
| for (int i = 0; i < kNumIterations; ++i) { |
| event.Set(); |
| event.Wait(webrtc::TimeDelta::Zero()); |
| } |
| } |
| |
| TEST(EventTest, DISABLED_PerformanceMultiThread) { |
| static const int kNumIterations = 10000; |
| Event read; |
| Event write; |
| SignalerThread thread; |
| thread.Start(&read, &write); |
| |
| for (int i = 0; i < kNumIterations; ++i) { |
| write.Set(); |
| read.Wait(Event::kForever); |
| } |
| write.Set(); |
| |
| thread.Stop(); |
| } |
| |
| #if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID) |
| // Tests that we crash if we attempt to call rtc::Event::Wait while we're |
| // not allowed to (as per `RTC_DISALLOW_WAIT()`). |
| TEST(EventTestDeathTest, DisallowEventWait) { |
| Event event; |
| RTC_DISALLOW_WAIT(); |
| EXPECT_DEATH(event.Wait(Event::kForever), ""); |
| } |
| #endif // RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID) |
| |
| } // namespace rtc |