blob: 7604fd9a85129bfce5db05bb8a7dadc653d2193f [file] [log] [blame]
/*-------------------------------------------------------------------------
* drawElements Quality Program Helper Library
* -------------------------------------------
*
* Copyright 2014 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
*//*!
* \file
* \brief Watch dog for detecting timeouts
*//*--------------------------------------------------------------------*/
#include "qpWatchDog.h"
#include "deThread.h"
#include "deClock.h"
#include "deMemory.h"
#include <stdio.h>
#if 0
# define DBGPRINT(X) qpPrintf X
#else
# define DBGPRINT(X)
#endif
typedef enum Status_e
{
STATUS_THREAD_RUNNING = 0,
STATUS_STOP_THREAD,
STATUS_LAST
} Status;
struct qpWatchDog_s
{
qpWatchDogFunc timeOutFunc;
void* timeOutUserPtr;
int totalTimeLimit; /* Total test case time limit in seconds */
int intervalTimeLimit; /* Iteration length limit in seconds */
/*
Iteration time limit in seconds specified to the constructor. This is stored so that
intervalTimeLimit can be restored after qpWatchDog_touchAndDisableIntervalTimeLimit
is called.
*/
int defaultIntervalTimeLimit;
volatile deUint64 resetTime;
volatile deUint64 lastTouchTime;
deThread watchDogThread;
volatile Status status;
};
static void watchDogThreadFunc (void* arg)
{
qpWatchDog* dog = (qpWatchDog*)arg;
DE_ASSERT(dog);
DBGPRINT(("watchDogThreadFunc(): start\n"));
while (dog->status == STATUS_THREAD_RUNNING)
{
deUint64 curTime = deGetMicroseconds();
int totalSecondsPassed = (int)((curTime - dog->resetTime) / 1000000ull);
int secondsSinceLastTouch = (int)((curTime - dog->lastTouchTime) / 1000000ull);
deBool overIntervalLimit = secondsSinceLastTouch > dog->intervalTimeLimit;
deBool overTotalLimit = totalSecondsPassed > dog->totalTimeLimit;
if (overIntervalLimit || overTotalLimit)
{
qpTimeoutReason reason = overTotalLimit ? QP_TIMEOUT_REASON_TOTAL_LIMIT : QP_TIMEOUT_REASON_INTERVAL_LIMIT;
DBGPRINT(("watchDogThreadFunc(): call timeout func\n"));
dog->timeOutFunc(dog, dog->timeOutUserPtr, reason);
break;
}
deSleep(100);
}
DBGPRINT(("watchDogThreadFunc(): stop\n"));
}
qpWatchDog* qpWatchDog_create (qpWatchDogFunc timeOutFunc, void* userPtr, int totalTimeLimitSecs, int intervalTimeLimitSecs)
{
/* Allocate & initialize. */
qpWatchDog* dog = (qpWatchDog*)deCalloc(sizeof(qpWatchDog));
if (!dog)
return dog;
DE_ASSERT(timeOutFunc);
DE_ASSERT((totalTimeLimitSecs > 0) && (intervalTimeLimitSecs > 0));
DBGPRINT(("qpWatchDog::create(%ds, %ds)\n", totalTimeLimitSecs, intervalTimeLimitSecs));
dog->timeOutFunc = timeOutFunc;
dog->timeOutUserPtr = userPtr;
dog->totalTimeLimit = totalTimeLimitSecs;
dog->intervalTimeLimit = intervalTimeLimitSecs;
dog->defaultIntervalTimeLimit = intervalTimeLimitSecs;
/* Reset (sets time values). */
qpWatchDog_reset(dog);
/* Initialize watchdog thread. */
dog->status = STATUS_THREAD_RUNNING;
dog->watchDogThread = deThread_create(watchDogThreadFunc, dog, DE_NULL);
if (!dog->watchDogThread)
{
deFree(dog);
return DE_NULL;
}
return dog;
}
void qpWatchDog_reset (qpWatchDog* dog)
{
deUint64 curTime = deGetMicroseconds();
DE_ASSERT(dog);
DBGPRINT(("qpWatchDog::reset()\n"));
dog->resetTime = curTime;
dog->lastTouchTime = curTime;
}
void qpWatchDog_destroy (qpWatchDog* dog)
{
DE_ASSERT(dog);
DBGPRINT(("qpWatchDog::destroy()\n"));
/* Finish the watchdog thread. */
dog->status = STATUS_STOP_THREAD;
deThread_join(dog->watchDogThread);
deThread_destroy(dog->watchDogThread);
DBGPRINT(("qpWatchDog::destroy() finished\n"));
deFree(dog);
}
void qpWatchDog_touch (qpWatchDog* dog)
{
DE_ASSERT(dog);
DBGPRINT(("qpWatchDog::touch()\n"));
dog->lastTouchTime = deGetMicroseconds();
}
/*
These function exists to allow the interval timer to be disabled for special cases
like very long shader compilations. Heavy code can be put between calls
to qpWatchDog_touchAndDisableIntervalTimeLimit and qpWatchDog_touchAndEnableIntervalTimeLimit
and during that period the interval time limit will become the same as the total
time limit. Afterwards, the interval timer is set back to its default.
*/
void qpWatchDog_touchAndDisableIntervalTimeLimit(qpWatchDog *dog)
{
dog->intervalTimeLimit = dog->totalTimeLimit;
qpWatchDog_touch(dog);
}
void qpWatchDog_touchAndEnableIntervalTimeLimit(qpWatchDog *dog)
{
dog->intervalTimeLimit = dog->defaultIntervalTimeLimit;
qpWatchDog_touch(dog);
}