|  | // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "base/synchronization/waitable_event.h" | 
|  |  | 
|  | #include "base/compiler_specific.h" | 
|  | #include "base/threading/platform_thread.h" | 
|  | #include "base/time/time.h" | 
|  | #include "testing/gtest/include/gtest/gtest.h" | 
|  |  | 
|  | namespace base { | 
|  |  | 
|  | TEST(WaitableEventTest, ManualBasics) { | 
|  | WaitableEvent event(true, false); | 
|  |  | 
|  | EXPECT_FALSE(event.IsSignaled()); | 
|  |  | 
|  | event.Signal(); | 
|  | EXPECT_TRUE(event.IsSignaled()); | 
|  | EXPECT_TRUE(event.IsSignaled()); | 
|  |  | 
|  | event.Reset(); | 
|  | EXPECT_FALSE(event.IsSignaled()); | 
|  | EXPECT_FALSE(event.TimedWait(TimeDelta::FromMilliseconds(10))); | 
|  |  | 
|  | event.Signal(); | 
|  | event.Wait(); | 
|  | EXPECT_TRUE(event.TimedWait(TimeDelta::FromMilliseconds(10))); | 
|  | } | 
|  |  | 
|  | TEST(WaitableEventTest, AutoBasics) { | 
|  | WaitableEvent event(false, false); | 
|  |  | 
|  | EXPECT_FALSE(event.IsSignaled()); | 
|  |  | 
|  | event.Signal(); | 
|  | EXPECT_TRUE(event.IsSignaled()); | 
|  | EXPECT_FALSE(event.IsSignaled()); | 
|  |  | 
|  | event.Reset(); | 
|  | EXPECT_FALSE(event.IsSignaled()); | 
|  | EXPECT_FALSE(event.TimedWait(TimeDelta::FromMilliseconds(10))); | 
|  |  | 
|  | event.Signal(); | 
|  | event.Wait(); | 
|  | EXPECT_FALSE(event.TimedWait(TimeDelta::FromMilliseconds(10))); | 
|  |  | 
|  | event.Signal(); | 
|  | EXPECT_TRUE(event.TimedWait(TimeDelta::FromMilliseconds(10))); | 
|  | } | 
|  |  | 
|  | TEST(WaitableEventTest, WaitManyShortcut) { | 
|  | WaitableEvent* ev[5]; | 
|  | for (unsigned i = 0; i < 5; ++i) | 
|  | ev[i] = new WaitableEvent(false, false); | 
|  |  | 
|  | ev[3]->Signal(); | 
|  | EXPECT_EQ(WaitableEvent::WaitMany(ev, 5), 3u); | 
|  |  | 
|  | ev[3]->Signal(); | 
|  | EXPECT_EQ(WaitableEvent::WaitMany(ev, 5), 3u); | 
|  |  | 
|  | ev[4]->Signal(); | 
|  | EXPECT_EQ(WaitableEvent::WaitMany(ev, 5), 4u); | 
|  |  | 
|  | ev[0]->Signal(); | 
|  | EXPECT_EQ(WaitableEvent::WaitMany(ev, 5), 0u); | 
|  |  | 
|  | for (unsigned i = 0; i < 5; ++i) | 
|  | delete ev[i]; | 
|  | } | 
|  |  | 
|  | class WaitableEventSignaler : public PlatformThread::Delegate { | 
|  | public: | 
|  | WaitableEventSignaler(TimeDelta delay, WaitableEvent* event) | 
|  | : delay_(delay), | 
|  | event_(event) { | 
|  | } | 
|  |  | 
|  | void ThreadMain() override { | 
|  | PlatformThread::Sleep(delay_); | 
|  | event_->Signal(); | 
|  | } | 
|  |  | 
|  | private: | 
|  | const TimeDelta delay_; | 
|  | WaitableEvent* event_; | 
|  | }; | 
|  |  | 
|  | // Tests that a WaitableEvent can be safely deleted when |Wait| is done without | 
|  | // additional synchronization. | 
|  | TEST(WaitableEventTest, WaitAndDelete) { | 
|  | WaitableEvent* ev = new WaitableEvent(false, false); | 
|  |  | 
|  | WaitableEventSignaler signaler(TimeDelta::FromMilliseconds(10), ev); | 
|  | PlatformThreadHandle thread; | 
|  | PlatformThread::Create(0, &signaler, &thread); | 
|  |  | 
|  | ev->Wait(); | 
|  | delete ev; | 
|  |  | 
|  | PlatformThread::Join(thread); | 
|  | } | 
|  |  | 
|  | // Tests that a WaitableEvent can be safely deleted when |WaitMany| is done | 
|  | // without additional synchronization. | 
|  | TEST(WaitableEventTest, WaitMany) { | 
|  | WaitableEvent* ev[5]; | 
|  | for (unsigned i = 0; i < 5; ++i) | 
|  | ev[i] = new WaitableEvent(false, false); | 
|  |  | 
|  | WaitableEventSignaler signaler(TimeDelta::FromMilliseconds(10), ev[2]); | 
|  | PlatformThreadHandle thread; | 
|  | PlatformThread::Create(0, &signaler, &thread); | 
|  |  | 
|  | size_t index = WaitableEvent::WaitMany(ev, 5); | 
|  |  | 
|  | for (unsigned i = 0; i < 5; ++i) | 
|  | delete ev[i]; | 
|  |  | 
|  | PlatformThread::Join(thread); | 
|  | EXPECT_EQ(2u, index); | 
|  | } | 
|  |  | 
|  | // Tests that using TimeDelta::Max() on TimedWait() is not the same as passing | 
|  | // a timeout of 0. (crbug.com/465948) | 
|  | #if defined(OS_POSIX) | 
|  | // crbug.com/465948 not fixed yet. | 
|  | #define MAYBE_TimedWait DISABLED_TimedWait | 
|  | #else | 
|  | #define MAYBE_TimedWait TimedWait | 
|  | #endif | 
|  | TEST(WaitableEventTest, MAYBE_TimedWait) { | 
|  | WaitableEvent* ev = new WaitableEvent(false, false); | 
|  |  | 
|  | TimeDelta thread_delay = TimeDelta::FromMilliseconds(10); | 
|  | WaitableEventSignaler signaler(thread_delay, ev); | 
|  | PlatformThreadHandle thread; | 
|  | TimeTicks start = TimeTicks::Now(); | 
|  | PlatformThread::Create(0, &signaler, &thread); | 
|  |  | 
|  | ev->TimedWait(TimeDelta::Max()); | 
|  | EXPECT_GE(TimeTicks::Now() - start, thread_delay); | 
|  | delete ev; | 
|  |  | 
|  | PlatformThread::Join(thread); | 
|  | } | 
|  |  | 
|  | }  // namespace base |