chromium/sandbox/win/src/threadpool_unittest.cc

// Copyright 2006-2008 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "sandbox/win/src/threadpool.h"

#include <windows.h>

#include <stdint.h>

#include "testing/gtest/include/gtest/gtest.h"

void __stdcall EmptyCallBack(void*, unsigned char) {}

void __stdcall TestCallBack(void* context, unsigned char) {
  HANDLE event = reinterpret_cast<HANDLE>(context);
  ::SetEvent(event);
}

namespace sandbox {

// Test that register and unregister work, part 1.
TEST(IPCTest, ThreadPoolRegisterTest1) {
  ThreadPool thread_pool;

  EXPECT_EQ(0u, thread_pool.OutstandingWaits());

  HANDLE event1 = ::CreateEventW(nullptr, false, false, nullptr);
  HANDLE event2 = ::CreateEventW(nullptr, false, false, nullptr);

  uint32_t context = 0;
  EXPECT_FALSE(thread_pool.RegisterWait(0, event1, EmptyCallBack, &context));
  EXPECT_EQ(0u, thread_pool.OutstandingWaits());

  EXPECT_TRUE(thread_pool.RegisterWait(this, event1, EmptyCallBack, &context));
  EXPECT_EQ(1u, thread_pool.OutstandingWaits());
  EXPECT_TRUE(thread_pool.RegisterWait(this, event2, EmptyCallBack, &context));
  EXPECT_EQ(2u, thread_pool.OutstandingWaits());

  EXPECT_TRUE(thread_pool.UnRegisterWaits(this));
  EXPECT_EQ(0u, thread_pool.OutstandingWaits());

  EXPECT_TRUE(::CloseHandle(event1));
  EXPECT_TRUE(::CloseHandle(event2));
}

// Test that register and unregister work, part 2.
TEST(IPCTest, ThreadPoolRegisterTest2) {
  ThreadPool thread_pool;

  HANDLE event1 = ::CreateEventW(nullptr, false, false, nullptr);
  HANDLE event2 = ::CreateEventW(nullptr, false, false, nullptr);

  uint32_t context = 0;
  uint32_t c1 = 0;
  uint32_t c2 = 0;

  EXPECT_TRUE(thread_pool.RegisterWait(&c1, event1, EmptyCallBack, &context));
  EXPECT_EQ(1u, thread_pool.OutstandingWaits());
  EXPECT_TRUE(thread_pool.RegisterWait(&c2, event2, EmptyCallBack, &context));
  EXPECT_EQ(2u, thread_pool.OutstandingWaits());

  EXPECT_TRUE(thread_pool.UnRegisterWaits(&c2));
  EXPECT_EQ(1u, thread_pool.OutstandingWaits());
  EXPECT_TRUE(thread_pool.UnRegisterWaits(&c2));
  EXPECT_EQ(1u, thread_pool.OutstandingWaits());

  EXPECT_TRUE(thread_pool.UnRegisterWaits(&c1));
  EXPECT_EQ(0u, thread_pool.OutstandingWaits());

  EXPECT_TRUE(::CloseHandle(event1));
  EXPECT_TRUE(::CloseHandle(event2));
}

// Test that the thread pool has at least a thread that services an event.
// Test that when the event is un-registered is no longer serviced.
TEST(IPCTest, ThreadPoolSignalAndWaitTest) {
  ThreadPool thread_pool;

  // The events are auto reset and start not signaled.
  HANDLE event1 = ::CreateEventW(nullptr, false, false, nullptr);
  HANDLE event2 = ::CreateEventW(nullptr, false, false, nullptr);

  EXPECT_TRUE(thread_pool.RegisterWait(this, event1, TestCallBack, event2));

  EXPECT_EQ(WAIT_OBJECT_0, ::SignalObjectAndWait(event1, event2, 5000, false));
  EXPECT_EQ(WAIT_OBJECT_0, ::SignalObjectAndWait(event1, event2, 5000, false));

  EXPECT_TRUE(thread_pool.UnRegisterWaits(this));
  EXPECT_EQ(0u, thread_pool.OutstandingWaits());

  EXPECT_EQ(static_cast<DWORD>(WAIT_TIMEOUT),
            ::SignalObjectAndWait(event1, event2, 1000, false));

  EXPECT_TRUE(::CloseHandle(event1));
  EXPECT_TRUE(::CloseHandle(event2));
}

}  // namespace sandbox