298 lines
8.5 KiB
C++
298 lines
8.5 KiB
C++
// Copyright 2013 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.
|
|
|
|
// NOTE(vtl): Some of these tests are inherently flaky (e.g., if run on a
|
|
// heavily-loaded system). Sorry. |test::EpsilonDeadline()| may be increased to
|
|
// increase tolerance and reduce observed flakiness (though doing so reduces the
|
|
// meaningfulness of the test).
|
|
|
|
#include "mojo/edk/system/waiter.h"
|
|
|
|
#include <stdint.h>
|
|
|
|
#include "base/macros.h"
|
|
#include "base/synchronization/lock.h"
|
|
#include "base/threading/simple_thread.h"
|
|
#include "mojo/edk/system/test_utils.h"
|
|
#include "testing/gtest/include/gtest/gtest.h"
|
|
|
|
namespace mojo {
|
|
namespace edk {
|
|
namespace {
|
|
|
|
const unsigned kPollTimeMs = 10;
|
|
|
|
class WaitingThread : public base::SimpleThread {
|
|
public:
|
|
explicit WaitingThread(MojoDeadline deadline)
|
|
: base::SimpleThread("waiting_thread"),
|
|
deadline_(deadline),
|
|
done_(false),
|
|
result_(MOJO_RESULT_UNKNOWN),
|
|
context_(static_cast<uintptr_t>(-1)) {
|
|
waiter_.Init();
|
|
}
|
|
|
|
~WaitingThread() override { Join(); }
|
|
|
|
void WaitUntilDone(MojoResult* result,
|
|
uintptr_t* context,
|
|
MojoDeadline* elapsed) {
|
|
for (;;) {
|
|
{
|
|
base::AutoLock locker(lock_);
|
|
if (done_) {
|
|
*result = result_;
|
|
*context = context_;
|
|
*elapsed = elapsed_;
|
|
break;
|
|
}
|
|
}
|
|
|
|
test::Sleep(test::DeadlineFromMilliseconds(kPollTimeMs));
|
|
}
|
|
}
|
|
|
|
Waiter* waiter() { return &waiter_; }
|
|
|
|
private:
|
|
void Run() override {
|
|
test::Stopwatch stopwatch;
|
|
MojoResult result;
|
|
uintptr_t context = static_cast<uintptr_t>(-1);
|
|
MojoDeadline elapsed;
|
|
|
|
stopwatch.Start();
|
|
result = waiter_.Wait(deadline_, &context);
|
|
elapsed = stopwatch.Elapsed();
|
|
|
|
{
|
|
base::AutoLock locker(lock_);
|
|
done_ = true;
|
|
result_ = result;
|
|
context_ = context;
|
|
elapsed_ = elapsed;
|
|
}
|
|
}
|
|
|
|
const MojoDeadline deadline_;
|
|
Waiter waiter_; // Thread-safe.
|
|
|
|
base::Lock lock_; // Protects the following members.
|
|
bool done_;
|
|
MojoResult result_;
|
|
uintptr_t context_;
|
|
MojoDeadline elapsed_;
|
|
|
|
DISALLOW_COPY_AND_ASSIGN(WaitingThread);
|
|
};
|
|
|
|
TEST(WaiterTest, Basic) {
|
|
MojoResult result;
|
|
uintptr_t context;
|
|
MojoDeadline elapsed;
|
|
|
|
// Finite deadline.
|
|
|
|
// Awake immediately after thread start.
|
|
{
|
|
WaitingThread thread(10 * test::EpsilonDeadline());
|
|
thread.Start();
|
|
thread.waiter()->Awake(MOJO_RESULT_OK, 1);
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(MOJO_RESULT_OK, result);
|
|
EXPECT_EQ(1u, context);
|
|
EXPECT_LT(elapsed, test::EpsilonDeadline());
|
|
}
|
|
|
|
// Awake before after thread start.
|
|
{
|
|
WaitingThread thread(10 * test::EpsilonDeadline());
|
|
thread.waiter()->Awake(MOJO_RESULT_CANCELLED, 2);
|
|
thread.Start();
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(MOJO_RESULT_CANCELLED, result);
|
|
EXPECT_EQ(2u, context);
|
|
EXPECT_LT(elapsed, test::EpsilonDeadline());
|
|
}
|
|
|
|
// Awake some time after thread start.
|
|
{
|
|
WaitingThread thread(10 * test::EpsilonDeadline());
|
|
thread.Start();
|
|
test::Sleep(2 * test::EpsilonDeadline());
|
|
thread.waiter()->Awake(1, 3);
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(1u, result);
|
|
EXPECT_EQ(3u, context);
|
|
EXPECT_GT(elapsed, (2 - 1) * test::EpsilonDeadline());
|
|
EXPECT_LT(elapsed, (2 + 1) * test::EpsilonDeadline());
|
|
}
|
|
|
|
// Awake some longer time after thread start.
|
|
{
|
|
WaitingThread thread(10 * test::EpsilonDeadline());
|
|
thread.Start();
|
|
test::Sleep(5 * test::EpsilonDeadline());
|
|
thread.waiter()->Awake(2, 4);
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(2u, result);
|
|
EXPECT_EQ(4u, context);
|
|
EXPECT_GT(elapsed, (5 - 1) * test::EpsilonDeadline());
|
|
EXPECT_LT(elapsed, (5 + 1) * test::EpsilonDeadline());
|
|
}
|
|
|
|
// Don't awake -- time out (on another thread).
|
|
{
|
|
WaitingThread thread(2 * test::EpsilonDeadline());
|
|
thread.Start();
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(MOJO_RESULT_DEADLINE_EXCEEDED, result);
|
|
EXPECT_EQ(static_cast<uintptr_t>(-1), context);
|
|
EXPECT_GT(elapsed, (2 - 1) * test::EpsilonDeadline());
|
|
EXPECT_LT(elapsed, (2 + 1) * test::EpsilonDeadline());
|
|
}
|
|
|
|
// No (indefinite) deadline.
|
|
|
|
// Awake immediately after thread start.
|
|
{
|
|
WaitingThread thread(MOJO_DEADLINE_INDEFINITE);
|
|
thread.Start();
|
|
thread.waiter()->Awake(MOJO_RESULT_OK, 5);
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(MOJO_RESULT_OK, result);
|
|
EXPECT_EQ(5u, context);
|
|
EXPECT_LT(elapsed, test::EpsilonDeadline());
|
|
}
|
|
|
|
// Awake before after thread start.
|
|
{
|
|
WaitingThread thread(MOJO_DEADLINE_INDEFINITE);
|
|
thread.waiter()->Awake(MOJO_RESULT_CANCELLED, 6);
|
|
thread.Start();
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(MOJO_RESULT_CANCELLED, result);
|
|
EXPECT_EQ(6u, context);
|
|
EXPECT_LT(elapsed, test::EpsilonDeadline());
|
|
}
|
|
|
|
// Awake some time after thread start.
|
|
{
|
|
WaitingThread thread(MOJO_DEADLINE_INDEFINITE);
|
|
thread.Start();
|
|
test::Sleep(2 * test::EpsilonDeadline());
|
|
thread.waiter()->Awake(1, 7);
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(1u, result);
|
|
EXPECT_EQ(7u, context);
|
|
EXPECT_GT(elapsed, (2 - 1) * test::EpsilonDeadline());
|
|
EXPECT_LT(elapsed, (2 + 1) * test::EpsilonDeadline());
|
|
}
|
|
|
|
// Awake some longer time after thread start.
|
|
{
|
|
WaitingThread thread(MOJO_DEADLINE_INDEFINITE);
|
|
thread.Start();
|
|
test::Sleep(5 * test::EpsilonDeadline());
|
|
thread.waiter()->Awake(2, 8);
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(2u, result);
|
|
EXPECT_EQ(8u, context);
|
|
EXPECT_GT(elapsed, (5 - 1) * test::EpsilonDeadline());
|
|
EXPECT_LT(elapsed, (5 + 1) * test::EpsilonDeadline());
|
|
}
|
|
}
|
|
|
|
TEST(WaiterTest, TimeOut) {
|
|
test::Stopwatch stopwatch;
|
|
MojoDeadline elapsed;
|
|
|
|
Waiter waiter;
|
|
uintptr_t context = 123;
|
|
|
|
waiter.Init();
|
|
stopwatch.Start();
|
|
EXPECT_EQ(MOJO_RESULT_DEADLINE_EXCEEDED, waiter.Wait(0, &context));
|
|
elapsed = stopwatch.Elapsed();
|
|
EXPECT_LT(elapsed, test::EpsilonDeadline());
|
|
EXPECT_EQ(123u, context);
|
|
|
|
waiter.Init();
|
|
stopwatch.Start();
|
|
EXPECT_EQ(MOJO_RESULT_DEADLINE_EXCEEDED,
|
|
waiter.Wait(2 * test::EpsilonDeadline(), &context));
|
|
elapsed = stopwatch.Elapsed();
|
|
EXPECT_GT(elapsed, (2 - 1) * test::EpsilonDeadline());
|
|
EXPECT_LT(elapsed, (2 + 1) * test::EpsilonDeadline());
|
|
EXPECT_EQ(123u, context);
|
|
|
|
waiter.Init();
|
|
stopwatch.Start();
|
|
EXPECT_EQ(MOJO_RESULT_DEADLINE_EXCEEDED,
|
|
waiter.Wait(5 * test::EpsilonDeadline(), &context));
|
|
elapsed = stopwatch.Elapsed();
|
|
EXPECT_GT(elapsed, (5 - 1) * test::EpsilonDeadline());
|
|
EXPECT_LT(elapsed, (5 + 1) * test::EpsilonDeadline());
|
|
EXPECT_EQ(123u, context);
|
|
}
|
|
|
|
// The first |Awake()| should always win.
|
|
TEST(WaiterTest, MultipleAwakes) {
|
|
MojoResult result;
|
|
uintptr_t context;
|
|
MojoDeadline elapsed;
|
|
|
|
{
|
|
WaitingThread thread(MOJO_DEADLINE_INDEFINITE);
|
|
thread.Start();
|
|
thread.waiter()->Awake(MOJO_RESULT_OK, 1);
|
|
thread.waiter()->Awake(1, 2);
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(MOJO_RESULT_OK, result);
|
|
EXPECT_EQ(1u, context);
|
|
EXPECT_LT(elapsed, test::EpsilonDeadline());
|
|
}
|
|
|
|
{
|
|
WaitingThread thread(MOJO_DEADLINE_INDEFINITE);
|
|
thread.waiter()->Awake(1, 3);
|
|
thread.Start();
|
|
thread.waiter()->Awake(MOJO_RESULT_OK, 4);
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(1u, result);
|
|
EXPECT_EQ(3u, context);
|
|
EXPECT_LT(elapsed, test::EpsilonDeadline());
|
|
}
|
|
|
|
{
|
|
WaitingThread thread(MOJO_DEADLINE_INDEFINITE);
|
|
thread.Start();
|
|
thread.waiter()->Awake(10, 5);
|
|
test::Sleep(2 * test::EpsilonDeadline());
|
|
thread.waiter()->Awake(20, 6);
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(10u, result);
|
|
EXPECT_EQ(5u, context);
|
|
EXPECT_LT(elapsed, test::EpsilonDeadline());
|
|
}
|
|
|
|
{
|
|
WaitingThread thread(10 * test::EpsilonDeadline());
|
|
thread.Start();
|
|
test::Sleep(1 * test::EpsilonDeadline());
|
|
thread.waiter()->Awake(MOJO_RESULT_FAILED_PRECONDITION, 7);
|
|
test::Sleep(2 * test::EpsilonDeadline());
|
|
thread.waiter()->Awake(MOJO_RESULT_OK, 8);
|
|
thread.WaitUntilDone(&result, &context, &elapsed);
|
|
EXPECT_EQ(MOJO_RESULT_FAILED_PRECONDITION, result);
|
|
EXPECT_EQ(7u, context);
|
|
EXPECT_GT(elapsed, (1 - 1) * test::EpsilonDeadline());
|
|
EXPECT_LT(elapsed, (1 + 1) * test::EpsilonDeadline());
|
|
}
|
|
}
|
|
|
|
} // namespace
|
|
} // namespace edk
|
|
} // namespace mojo
|