1 // Copyright 2014 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
7 #include "src/base/platform/platform.h"
8 #include "src/base/platform/semaphore.h"
9 #include "src/base/platform/time.h"
10 #include "testing/gtest/include/gtest/gtest.h"
17 static const char kAlphabet[] = "XKOAD";
18 static const size_t kAlphabetSize = sizeof(kAlphabet) - 1;
19 static const size_t kBufferSize = 987; // GCD(buffer size, alphabet size) = 1
20 static const size_t kDataSize = kBufferSize * kAlphabetSize * 10;
23 class ProducerThread FINAL : public Thread {
25 ProducerThread(char* buffer, Semaphore* free_space, Semaphore* used_space)
26 : Thread(Options("ProducerThread")),
28 free_space_(free_space),
29 used_space_(used_space) {}
30 virtual ~ProducerThread() {}
32 virtual void Run() OVERRIDE {
33 for (size_t n = 0; n < kDataSize; ++n) {
35 buffer_[n % kBufferSize] = kAlphabet[n % kAlphabetSize];
36 used_space_->Signal();
42 Semaphore* const free_space_;
43 Semaphore* const used_space_;
47 class ConsumerThread FINAL : public Thread {
49 ConsumerThread(const char* buffer, Semaphore* free_space,
50 Semaphore* used_space)
51 : Thread(Options("ConsumerThread")),
53 free_space_(free_space),
54 used_space_(used_space) {}
55 virtual ~ConsumerThread() {}
57 virtual void Run() OVERRIDE {
58 for (size_t n = 0; n < kDataSize; ++n) {
60 EXPECT_EQ(kAlphabet[n % kAlphabetSize], buffer_[n % kBufferSize]);
61 free_space_->Signal();
67 Semaphore* const free_space_;
68 Semaphore* const used_space_;
72 class WaitAndSignalThread FINAL : public Thread {
74 explicit WaitAndSignalThread(Semaphore* semaphore)
75 : Thread(Options("WaitAndSignalThread")), semaphore_(semaphore) {}
76 virtual ~WaitAndSignalThread() {}
78 virtual void Run() OVERRIDE {
79 for (int n = 0; n < 100; ++n) {
81 ASSERT_FALSE(semaphore_->WaitFor(TimeDelta::FromMicroseconds(1)));
87 Semaphore* const semaphore_;
93 TEST(Semaphore, ProducerConsumer) {
94 char buffer[kBufferSize];
95 std::memset(buffer, 0, sizeof(buffer));
96 Semaphore free_space(kBufferSize);
97 Semaphore used_space(0);
98 ProducerThread producer_thread(buffer, &free_space, &used_space);
99 ConsumerThread consumer_thread(buffer, &free_space, &used_space);
100 producer_thread.Start();
101 consumer_thread.Start();
102 producer_thread.Join();
103 consumer_thread.Join();
107 TEST(Semaphore, WaitAndSignal) {
108 Semaphore semaphore(0);
109 WaitAndSignalThread t1(&semaphore);
110 WaitAndSignalThread t2(&semaphore);
115 // Make something available.
123 EXPECT_FALSE(semaphore.WaitFor(TimeDelta::FromMicroseconds(1)));
127 TEST(Semaphore, WaitFor) {
128 Semaphore semaphore(0);
130 // Semaphore not signalled - timeout.
131 ASSERT_FALSE(semaphore.WaitFor(TimeDelta::FromMicroseconds(0)));
132 ASSERT_FALSE(semaphore.WaitFor(TimeDelta::FromMicroseconds(100)));
133 ASSERT_FALSE(semaphore.WaitFor(TimeDelta::FromMicroseconds(1000)));
135 // Semaphore signalled - no timeout.
137 ASSERT_TRUE(semaphore.WaitFor(TimeDelta::FromMicroseconds(0)));
139 ASSERT_TRUE(semaphore.WaitFor(TimeDelta::FromMicroseconds(100)));
141 ASSERT_TRUE(semaphore.WaitFor(TimeDelta::FromMicroseconds(1000)));