aboutsummaryrefslogtreecommitdiff
path: root/src/utils/threadpool_test.cc
diff options
context:
space:
mode:
Diffstat (limited to 'src/utils/threadpool_test.cc')
-rw-r--r--src/utils/threadpool_test.cc133
1 files changed, 133 insertions, 0 deletions
diff --git a/src/utils/threadpool_test.cc b/src/utils/threadpool_test.cc
new file mode 100644
index 0000000..17854dc
--- /dev/null
+++ b/src/utils/threadpool_test.cc
@@ -0,0 +1,133 @@
+// Copyright 2021 The libgav1 Authors
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+
+#include "src/utils/threadpool.h"
+
+#include <cassert>
+#include <cstdint>
+#include <memory>
+
+#include "absl/synchronization/mutex.h"
+#include "absl/time/clock.h"
+#include "absl/time/time.h"
+#include "gtest/gtest.h"
+#include "src/utils/compiler_attributes.h"
+#include "src/utils/executor.h"
+
+namespace libgav1 {
+namespace {
+
+class SimpleGuardedInteger {
+ public:
+ explicit SimpleGuardedInteger(int initial_value) : value_(initial_value) {}
+ SimpleGuardedInteger(const SimpleGuardedInteger&) = delete;
+ SimpleGuardedInteger& operator=(const SimpleGuardedInteger&) = delete;
+
+ void Decrement() {
+ absl::MutexLock l(&mutex_);
+ assert(value_ >= 1);
+ --value_;
+ changed_.SignalAll();
+ }
+
+ void Increment() {
+ absl::MutexLock l(&mutex_);
+ ++value_;
+ changed_.SignalAll();
+ }
+
+ int Value() {
+ absl::MutexLock l(&mutex_);
+ return value_;
+ }
+
+ void WaitForZero() {
+ absl::MutexLock l(&mutex_);
+ while (value_ != 0) {
+ changed_.Wait(&mutex_);
+ }
+ }
+
+ private:
+ absl::Mutex mutex_;
+ absl::CondVar changed_;
+ int value_ LIBGAV1_GUARDED_BY(mutex_);
+};
+
+// Loops for |milliseconds| of wall-clock time.
+void LoopForMs(int64_t milliseconds) {
+ const absl::Time deadline = absl::Now() + absl::Milliseconds(milliseconds);
+ while (absl::Now() < deadline) {
+ }
+}
+
+// A function that increments the given integer.
+void IncrementIntegerJob(SimpleGuardedInteger* value) {
+ LoopForMs(100);
+ value->Increment();
+}
+
+TEST(ThreadPoolTest, ThreadedIntegerIncrement) {
+ std::unique_ptr<ThreadPool> thread_pool = ThreadPool::Create(100);
+ ASSERT_NE(thread_pool, nullptr);
+ EXPECT_EQ(thread_pool->num_threads(), 100);
+ SimpleGuardedInteger count(0);
+ for (int i = 0; i < 1000; ++i) {
+ thread_pool->Schedule([&count]() { IncrementIntegerJob(&count); });
+ }
+ thread_pool.reset(nullptr);
+ EXPECT_EQ(count.Value(), 1000);
+}
+
+// Test a ThreadPool via the Executor interface.
+TEST(ThreadPoolTest, ExecutorInterface) {
+ std::unique_ptr<ThreadPool> thread_pool = ThreadPool::Create(100);
+ ASSERT_NE(thread_pool, nullptr);
+ std::unique_ptr<Executor> executor(thread_pool.release());
+ SimpleGuardedInteger count(0);
+ for (int i = 0; i < 1000; ++i) {
+ executor->Schedule([&count]() { IncrementIntegerJob(&count); });
+ }
+ executor.reset(nullptr);
+ EXPECT_EQ(count.Value(), 1000);
+}
+
+TEST(ThreadPoolTest, DestroyWithoutUse) {
+ std::unique_ptr<ThreadPool> thread_pool = ThreadPool::Create(100);
+ EXPECT_NE(thread_pool, nullptr);
+ thread_pool.reset(nullptr);
+}
+
+// If num_threads is 0, ThreadPool::Create() should return a null pointer.
+TEST(ThreadPoolTest, NumThreadsZero) {
+ std::unique_ptr<ThreadPool> thread_pool = ThreadPool::Create(0);
+ EXPECT_EQ(thread_pool, nullptr);
+}
+
+// If num_threads is 1, the closures are run in FIFO order.
+TEST(ThreadPoolTest, OneThreadRunsClosuresFIFO) {
+ int count = 0; // Declare first so that it outlives the thread pool.
+ std::unique_ptr<ThreadPool> pool = ThreadPool::Create(1);
+ ASSERT_NE(pool, nullptr);
+ EXPECT_EQ(pool->num_threads(), 1);
+ for (int i = 0; i < 1000; ++i) {
+ pool->Schedule([&count, i]() {
+ EXPECT_EQ(count, i);
+ count++;
+ });
+ }
+}
+
+} // namespace
+} // namespace libgav1