added PThreadVideoTimer and tests for it. still it doesn't work as expected.

This commit is contained in:
rachytski 2011-11-24 16:28:32 +04:00 committed by Alex Zolotarev
parent 3f2e3fc1e4
commit e5ceafbe9d
5 changed files with 185 additions and 3 deletions

View file

@ -41,7 +41,8 @@ include($$ROOT_DIR/common.pri)
ios_concurrent_runner.mm \
platform_ios.mm
} else:android* {
SOURCES += platform_android.cpp
SOURCES += platform_android.cpp \
pthread_video_timer.cpp
}
macx*|iphone* {
@ -68,3 +69,4 @@ SOURCES += \
video_timer.cpp \
http_request.cpp \
chunks_download_strategy.cpp \
pthread_video_timer.cpp \

View file

@ -20,7 +20,7 @@ win32* {
}
macx* {
QT *= gui # needed for QApplication with event loop, to test async events (downloader, etc.)
LIBS *= "-framework Foundation" "-framework IOKit"
LIBS *= "-framework Foundation" "-framework IOKit" "-framework QuartzCore"
}
win32*|linux* {
QT *= network
@ -33,4 +33,4 @@ SOURCES += \
concurrent_runner_test.cpp \
language_test.cpp \
downloader_test.cpp \
video_timer_test.cpp \

View file

@ -0,0 +1,46 @@
#include "../../testing/testing.hpp"
#include "../video_timer.hpp"
#include "../../base/thread.hpp"
#include "../../base/logging.hpp"
#include "../../std/bind.hpp"
void incrementValue(int & i)
{
++i;
}
UNIT_TEST(TimerTest)
{
int i = 0;
#ifdef OMIM_OS_MAC
VideoTimer * videoTimer = CreatePThreadVideoTimer(bind(&incrementValue, ref(i)));
#endif
LOG(LINFO, ("checking for approximately 60 cycles in second"));
videoTimer->start();
threads::Sleep(1000);
videoTimer->pause();
TEST((i >= 57) && (i <= 61), ("timer doesn't work, i=", i));
videoTimer->resume();
threads::Sleep(200);
videoTimer->stop();
videoTimer->start();
threads::Sleep(200);
videoTimer->stop();
}

View file

@ -0,0 +1,133 @@
#include <pthread.h>
#include "video_timer.hpp"
#include "../base/logging.hpp"
#include <sys/time.h>
#include <sys/errno.h>
class PThreadVideoTimer : public VideoTimer
{
private:
pthread_t m_handle;
pthread_mutex_t m_mutex;
pthread_cond_t m_cond;
int m_frameRate;
public:
PThreadVideoTimer(VideoTimer::TFrameFn frameFn)
: VideoTimer(frameFn), m_frameRate(60)
{
::pthread_mutex_init(&m_mutex, 0);
::pthread_cond_init(&m_cond, 0);
}
~PThreadVideoTimer()
{
stop();
::pthread_mutex_destroy(&m_mutex);
::pthread_cond_destroy(&m_cond);
}
static void * TimerProc(void * p)
{
PThreadVideoTimer * t = reinterpret_cast<PThreadVideoTimer*>(p);
::timeval prevtv;
::gettimeofday(&prevtv, 0);
::timeval curtv;
int64_t interval = 1000000000 / t->m_frameRate;
int64_t halfInterval = interval / 2;
while (true)
{
::pthread_mutex_lock(&t->m_mutex);
t->perform();
::gettimeofday(&curtv, 0);
int64_t sec = (int64_t)curtv.tv_sec - (int64_t)prevtv.tv_sec;
int64_t nsec = ((int64_t)curtv.tv_usec - (int64_t)prevtv.tv_usec) * 1000;
int64_t nsecDiff = sec * (int64_t)1000000000 + nsec;
int64_t ceiledDiff = ((nsecDiff + interval - 1) / interval) * interval;
/// how much we should wait
if ((ceiledDiff - nsecDiff) < halfInterval)
/// less than a half-frame left, should wait till next frame
ceiledDiff += interval;
::timespec ts;
ts.tv_sec = prevtv.tv_sec + (prevtv.tv_usec * 1000 + ceiledDiff) / 1000000000;
ts.tv_nsec = (prevtv.tv_usec * 1000 + ceiledDiff) % 1000000000;
::pthread_cond_timedwait(&t->m_cond, &t->m_mutex, &ts);
::gettimeofday(&prevtv, 0);
if (t->m_state == EStopped)
{
::pthread_mutex_unlock(&t->m_mutex);
break;
}
::pthread_mutex_unlock(&t->m_mutex);
}
::pthread_exit(0);
return 0;
}
void start()
{
if (m_state == EStopped)
{
::pthread_create(&m_handle, 0, &TimerProc, reinterpret_cast<void*>(this));
m_state = ERunning;
}
}
void resume()
{
if (m_state == EPaused)
{
m_state = EStopped;
start();
}
}
void pause()
{
stop();
m_state = EPaused;
}
void stop()
{
if (m_state == ERunning)
{
::pthread_mutex_lock(&m_mutex);
m_state = EStopped;
::pthread_cond_signal(&m_cond);
::pthread_mutex_unlock(&m_mutex);
::pthread_join(m_handle, 0);
}
}
void perform()
{
m_frameFn();
}
};
VideoTimer * CreatePThreadVideoTimer(VideoTimer::TFrameFn frameFn)
{
return new PThreadVideoTimer(frameFn);
}

View file

@ -40,3 +40,4 @@ public:
extern "C" VideoTimer * CreateIOSVideoTimer(VideoTimer::TFrameFn frameFn);
extern "C" VideoTimer * CreateAppleVideoTimer(VideoTimer::TFrameFn frameFn);
extern "C" VideoTimer * CreateWin32VideoTimer(VideoTimer::TFrameFn frameFn);
extern "C" VideoTimer * CreatePThreadVideoTimer(VideoTimer::TFrameFn frameFn);