blob: 94874aac310b46a5a58c8161b7ca2d5c30b61936 [file] [log] [blame]
/*
* Copyright (C) 2015 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "Utilities.h"
#include <wtf/RunLoop.h>
#include <wtf/Threading.h>
namespace TestWebKitAPI {
static bool testFinished;
static int count = 100;
TEST(WTF_RunLoop, Deadlock)
{
WTF::initializeMainThread();
struct DispatchFromDestructorTester {
~DispatchFromDestructorTester() {
RunLoop::main().dispatch([] {
if (!(--count))
testFinished = true;
});
}
};
for (int i = 0; i < count; ++i) {
auto capture = std::make_shared<DispatchFromDestructorTester>();
RunLoop::main().dispatch([capture] { });
}
Util::run(&testFinished);
}
TEST(WTF_RunLoop, NestedInOrder)
{
WTF::initializeMainThread();
bool done = false;
bool didExecuteOuter = false;
RunLoop::main().dispatch([&done, &didExecuteOuter] {
RunLoop::main().dispatch([&done, &didExecuteOuter] {
EXPECT_TRUE(didExecuteOuter);
done = true;
});
Util::run(&done);
});
RunLoop::main().dispatch([&didExecuteOuter] {
didExecuteOuter = true;
});
Util::run(&done);
}
TEST(WTF_RunLoop, DispatchCrossThreadWhileNested)
{
WTF::initializeMainThread();
bool done = false;
RunLoop::main().dispatch([&done] {
Thread::create("DispatchCrossThread", [&done] {
RunLoop::main().dispatch([&done] {
done = true;
});
});
Util::run(&done);
});
RunLoop::main().dispatch([] { });
Util::run(&done);
}
TEST(WTF_RunLoop, CallOnMainCrossThreadWhileNested)
{
WTF::initializeMainThread();
bool done = false;
callOnMainThread([&done] {
Thread::create("CallOnMainCrossThread", [&done] {
callOnMainThread([&done] {
done = true;
});
});
Util::run(&done);
});
callOnMainThread([] { });
Util::run(&done);
}
class DerivedOneShotTimer : public RunLoop::Timer<DerivedOneShotTimer> {
public:
DerivedOneShotTimer(bool& testFinished)
: RunLoop::Timer<DerivedOneShotTimer>(RunLoop::current(), this, &DerivedOneShotTimer::fired)
, m_testFinished(testFinished)
{
}
void fired()
{
EXPECT_FALSE(isActive());
EXPECT_EQ(secondsUntilFire(), Seconds(0));
m_testFinished = true;
stop();
}
private:
bool& m_testFinished;
};
TEST(WTF_RunLoop, OneShotTimer)
{
WTF::initializeMainThread();
bool testFinished = false;
DerivedOneShotTimer timer(testFinished);
timer.startOneShot(100_ms);
Util::run(&testFinished);
}
class DerivedRepeatingTimer : public RunLoop::Timer<DerivedRepeatingTimer> {
public:
DerivedRepeatingTimer(bool& testFinished)
: RunLoop::Timer<DerivedRepeatingTimer>(RunLoop::current(), this, &DerivedRepeatingTimer::fired)
, m_testFinished(testFinished)
{
}
void fired()
{
EXPECT_TRUE(isActive());
if (++m_count == 10) {
m_testFinished = true;
stop();
EXPECT_FALSE(isActive());
EXPECT_EQ(secondsUntilFire(), Seconds(0));
}
}
private:
unsigned m_count { 0 };
bool& m_testFinished;
};
TEST(WTF_RunLoop, RepeatingTimer)
{
WTF::initializeMainThread();
bool testFinished = false;
DerivedRepeatingTimer timer(testFinished);
timer.startRepeating(10_ms);
Util::run(&testFinished);
ASSERT_FALSE(timer.isActive());
}
TEST(WTF_RunLoop, ManyTimes)
{
class Counter {
public:
void run()
{
if (++m_count == 100000) {
RunLoop::current().stop();
return;
}
RunLoop::current().dispatch([this] {
run();
});
}
private:
unsigned m_count { 0 };
};
Thread::create("RunLoopManyTimes", [] {
Counter counter;
RunLoop::current().dispatch([&counter] {
counter.run();
});
RunLoop::run();
})->waitForCompletion();
}
TEST(WTF_RunLoop, ThreadTerminationSelfReferenceCleanup)
{
RefPtr<RunLoop> runLoop;
Thread::create("RunLoopThreadTerminationSelfReferenceCleanup", [&] {
runLoop = &RunLoop::current();
// This stores a RunLoop reference in the dispatch queue that will not be released
// via the usual dispatch, but should still be released upon thread termination.
// After that, the observing RefPtr should be the only one holding a reference
// to the RunLoop object.
runLoop->dispatch([ref = runLoop.copyRef()] { });
})->waitForCompletion();
EXPECT_TRUE(runLoop->hasOneRef());
}
} // namespace TestWebKitAPI