1
0
mirror of https://github.com/VCMP-SqMod/SqMod.git synced 2025-02-25 06:07:13 +01:00
Sandu Liviu Catalin 4a6bfc086c Major plugin refactor and cleanup.
Switched to POCO library for unified platform/library interface.
Deprecated the external module API. It was creating more problems than solving.
Removed most built-in libraries in favor of system libraries for easier maintenance.
Cleaned and secured code with help from static analyzers.
2021-01-30 08:51:39 +02:00

481 lines
8.7 KiB
C++

//
// ThreadTest.cpp
//
// Copyright (c) 2004-2006, Applied Informatics Software Engineering GmbH.
// and Contributors.
//
// SPDX-License-Identifier: BSL-1.0
//
#include "ThreadTest.h"
#include "CppUnit/TestCaller.h"
#include "CppUnit/TestSuite.h"
#include "Poco/Thread.h"
#include "Poco/Runnable.h"
#include "Poco/ThreadTarget.h"
#include "Poco/Event.h"
#include "Poco/Timestamp.h"
#include "Poco/Timespan.h"
//#include <iostream>
#if defined(__sun) && defined(__SVR4) && !defined(__EXTENSIONS__)
#define __EXTENSIONS__
#endif
#include <climits>
using Poco::Thread;
using Poco::Runnable;
using Poco::ThreadTarget;
using Poco::Event;
class MyRunnable: public Runnable
{
public:
MyRunnable(): _ran(false)
{
}
void run()
{
Thread* pThread = Thread::current();
if (pThread)
_threadName = pThread->name();
_ran = true;
_event.wait();
}
bool ran() const
{
return _ran;
}
const std::string& threadName() const
{
return _threadName;
}
void notify()
{
_event.set();
}
static void staticFunc()
{
++_staticVar;
}
static int _staticVar;
private:
bool _ran;
std::string _threadName;
Event _event;
};
int MyRunnable::_staticVar = 0;
void freeFunc()
{
++MyRunnable::_staticVar;
}
void freeFunc(void* pData)
{
MyRunnable::_staticVar += *reinterpret_cast<int*>(pData);
}
class NonJoinRunnable : public Runnable
{
public:
NonJoinRunnable() : _finished(false)
{
}
void run()
{
_finished = true;
}
bool finished() const
{
return _finished;
}
private:
bool _finished;
};
class TrySleepRunnable : public Runnable
{
public:
TrySleepRunnable() : _counter(0), _sleepy(true)
{
}
void run()
{
_sleepy = !Thread::trySleep(300000);
++_counter;
_sleepy = !Thread::trySleep(300000);
++_counter;
_sleepy = !Thread::trySleep(100);
++_counter;
}
int counter() const
{
return _counter;
}
bool isSleepy() const
{
return _sleepy;
}
private:
int _counter;
bool _sleepy;
};
ThreadTest::ThreadTest(const std::string& name): CppUnit::TestCase(name)
{
}
ThreadTest::~ThreadTest()
{
}
void ThreadTest::testThread()
{
Thread thread;
MyRunnable r;
assertTrue (!thread.isRunning());
thread.start(r);
Thread::sleep(200);
assertTrue (thread.isRunning());
r.notify();
thread.join();
assertTrue (!thread.isRunning());
assertTrue (r.ran());
assertTrue (!r.threadName().empty());
}
void ThreadTest::testNamedThread()
{
Thread thread("MyThread");
MyRunnable r;
thread.start(r);
r.notify();
thread.join();
assertTrue (r.ran());
assertTrue (r.threadName() == "MyThread");
}
void ThreadTest::testCurrent()
{
assertNullPtr (Thread::current());
}
void ThreadTest::testThreads()
{
Thread thread1("Thread1");
Thread thread2("Thread2");
Thread thread3("Thread3");
Thread thread4("Thread4");
MyRunnable r1;
MyRunnable r2;
MyRunnable r3;
MyRunnable r4;
assertTrue (!thread1.isRunning());
assertTrue (!thread2.isRunning());
assertTrue (!thread3.isRunning());
assertTrue (!thread4.isRunning());
thread1.start(r1);
Thread::sleep(200);
assertTrue (thread1.isRunning());
assertTrue (!thread2.isRunning());
assertTrue (!thread3.isRunning());
assertTrue (!thread4.isRunning());
thread2.start(r2);
thread3.start(r3);
thread4.start(r4);
Thread::sleep(200);
assertTrue (thread1.isRunning());
assertTrue (thread2.isRunning());
assertTrue (thread3.isRunning());
assertTrue (thread4.isRunning());
r4.notify();
thread4.join();
assertTrue (!thread4.isRunning());
assertTrue (thread1.isRunning());
assertTrue (thread2.isRunning());
assertTrue (thread3.isRunning());
r3.notify();
thread3.join();
assertTrue (!thread3.isRunning());
r2.notify();
thread2.join();
assertTrue (!thread2.isRunning());
r1.notify();
thread1.join();
assertTrue (!thread1.isRunning());
assertTrue (r1.ran());
assertTrue (r1.threadName() == "Thread1");
assertTrue (r2.ran());
assertTrue (r2.threadName() == "Thread2");
assertTrue (r3.ran());
assertTrue (r3.threadName() == "Thread3");
assertTrue (r4.ran());
assertTrue (r4.threadName() == "Thread4");
}
void ThreadTest::testJoin()
{
Thread thread;
MyRunnable r;
assertTrue (!thread.isRunning());
thread.start(r);
Thread::sleep(200);
assertTrue (thread.isRunning());
assertTrue (!thread.tryJoin(100));
r.notify();
assertTrue (thread.tryJoin(500));
assertTrue (!thread.isRunning());
}
void ThreadTest::testNotJoin()
{
Thread thread;
NonJoinRunnable r;
thread.start(r);
while (!r.finished())
{
Thread::sleep(10);
}
Thread::sleep(100);
assertTrue (!thread.isRunning());
}
void ThreadTest::testTrySleep()
{
Thread thread;
TrySleepRunnable r;
assertTrue (r.isSleepy());
assertTrue (!thread.isRunning());
assertTrue (r.counter() == 0);
thread.start(r);
assertTrue (thread.isRunning());
assertTrue (r.counter() == 0);
assertTrue (r.isSleepy());
Thread::sleep(100);
assertTrue (r.counter() == 0);
assertTrue (r.isSleepy());
thread.wakeUp(); Thread::sleep(10);
assertTrue (r.counter() == 1);
assertTrue (r.isSleepy());
Thread::sleep(100);
assertTrue (r.counter() == 1);
thread.wakeUp(); Thread::sleep(10);
assertTrue (r.counter() == 2);
assertTrue (r.isSleepy());
Thread::sleep(200);
assertTrue (r.counter() == 3);
assertTrue (!r.isSleepy());
assertTrue (!thread.isRunning());
thread.wakeUp();
assertTrue (!thread.isRunning());
}
void ThreadTest::testNotRun()
{
Thread thread;
}
void ThreadTest::testNotRunJoin()
{
Thread thread;
thread.join();
}
void ThreadTest::testThreadTarget()
{
ThreadTarget te(&MyRunnable::staticFunc);
Thread thread;
assertTrue (!thread.isRunning());
int tmp = MyRunnable::_staticVar;
thread.start(te);
thread.join();
assertTrue (tmp + 1 == MyRunnable::_staticVar);
ThreadTarget te1(freeFunc);
assertTrue (!thread.isRunning());
tmp = MyRunnable::_staticVar;
thread.start(te1);
thread.join();
assertTrue (tmp + 1 == MyRunnable::_staticVar);
}
void ThreadTest::testThreadFunction()
{
Thread thread;
assertTrue (!thread.isRunning());
int tmp = MyRunnable::_staticVar;
thread.start(freeFunc, &tmp);
thread.join();
assertTrue (tmp * 2 == MyRunnable::_staticVar);
assertTrue (!thread.isRunning());
tmp = MyRunnable::_staticVar = 0;
thread.start(freeFunc, &tmp);
thread.join();
assertTrue (0 == MyRunnable::_staticVar);
}
struct Functor
{
void operator () ()
{
++MyRunnable::_staticVar;
}
};
void ThreadTest::testThreadFunctor()
{
Thread thread;
assertTrue (!thread.isRunning());
MyRunnable::_staticVar = 0;
thread.startFunc(Functor());
thread.join();
assertTrue (1 == MyRunnable::_staticVar);
assertTrue (!thread.isRunning());
Thread thread2;
assertTrue (!thread2.isRunning());
MyRunnable::_staticVar = 0;
thread.startFunc([] ()
{
MyRunnable::_staticVar++;
});
thread.join();
assertTrue (1 == MyRunnable::_staticVar);
assertTrue (!thread2.isRunning());
}
void ThreadTest::testThreadStackSize()
{
int stackSize = 50000000;
Thread thread;
assertTrue (0 == thread.getStackSize());
thread.setStackSize(stackSize);
assertTrue (stackSize <= thread.getStackSize());
int tmp = MyRunnable::_staticVar;
thread.start(freeFunc, &tmp);
thread.join();
assertTrue (tmp * 2 == MyRunnable::_staticVar);
stackSize = 1;
thread.setStackSize(stackSize);
#if !defined(POCO_OS_FAMILY_BSD) // on BSD family, stack size is rounded
#ifdef PTHREAD_STACK_MIN
assertTrue (PTHREAD_STACK_MIN == thread.getStackSize());
#else
assertTrue (stackSize >= thread.getStackSize());
#endif
#endif
tmp = MyRunnable::_staticVar;
thread.start(freeFunc, &tmp);
thread.join();
assertTrue (tmp * 2 == MyRunnable::_staticVar);
thread.setStackSize(0);
assertTrue (0 == thread.getStackSize());
tmp = MyRunnable::_staticVar;
thread.start(freeFunc, &tmp);
thread.join();
assertTrue (tmp * 2 == MyRunnable::_staticVar);
}
void ThreadTest::testSleep()
{
Poco::Timestamp start;
Thread::sleep(200);
Poco::Timespan elapsed = start.elapsed();
assertTrue (elapsed.totalMilliseconds() >= 190 && elapsed.totalMilliseconds() < 250);
}
void ThreadTest::setUp()
{
}
void ThreadTest::tearDown()
{
}
CppUnit::Test* ThreadTest::suite()
{
CppUnit::TestSuite* pSuite = new CppUnit::TestSuite("ThreadTest");
CppUnit_addTest(pSuite, ThreadTest, testThread);
CppUnit_addTest(pSuite, ThreadTest, testNamedThread);
CppUnit_addTest(pSuite, ThreadTest, testCurrent);
CppUnit_addTest(pSuite, ThreadTest, testThreads);
CppUnit_addTest(pSuite, ThreadTest, testJoin);
CppUnit_addTest(pSuite, ThreadTest, testNotJoin);
CppUnit_addTest(pSuite, ThreadTest, testNotRun);
CppUnit_addTest(pSuite, ThreadTest, testNotRunJoin);
CppUnit_addTest(pSuite, ThreadTest, testTrySleep);
CppUnit_addTest(pSuite, ThreadTest, testThreadTarget);
CppUnit_addTest(pSuite, ThreadTest, testThreadFunction);
CppUnit_addTest(pSuite, ThreadTest, testThreadFunctor);
CppUnit_addTest(pSuite, ThreadTest, testThreadStackSize);
CppUnit_addTest(pSuite, ThreadTest, testSleep);
return pSuite;
}