| /*------------------------------------------------------------------------- |
| * 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); |
| } |