| /*---------------------------------------------------------------------------* |
| * ptimer.c * |
| * * |
| * Copyright 2007, 2008 Nuance Communciations, Inc. * |
| * * |
| * 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. * |
| * * |
| *---------------------------------------------------------------------------*/ |
| |
| |
| |
| #include "pmemory.h" |
| #include "ptimer.h" |
| #include "pmutex.h" |
| |
| #ifdef _WIN32 |
| |
| /* |
| Note that this implementation assumes that QueryPerformanceCounter is |
| available (requires NT 3.1 and above) and that 64 bit arithmetic is |
| available (requires VC) |
| */ |
| |
| struct PTimer_t |
| { |
| LARGE_INTEGER PerformanceFreq; |
| LARGE_INTEGER RefTime; |
| LARGE_INTEGER elapsed; |
| }; |
| |
| |
| |
| /** |
| * Creates a new timer object. |
| **/ |
| ESR_ReturnCode PTimerCreate(PTimer **timer) |
| { |
| PTimer *tmp = NULL; |
| |
| if (timer == NULL) |
| return ESR_INVALID_ARGUMENT; |
| tmp = NEW(PTimer, "PTimer"); |
| if (tmp == NULL) |
| return ESR_OUT_OF_MEMORY; |
| |
| if (QueryPerformanceFrequency(&tmp->PerformanceFreq) == 0) |
| { |
| FREE(tmp); |
| return ESR_NOT_SUPPORTED; |
| } |
| tmp->PerformanceFreq.QuadPart /= 1000; |
| |
| tmp->RefTime.QuadPart = 0; |
| tmp->elapsed.QuadPart = 0; |
| *timer = tmp; |
| return ESR_SUCCESS; |
| } |
| |
| ESR_ReturnCode PTimerDestroy(PTimer *timer) |
| { |
| if (timer == NULL) return ESR_INVALID_ARGUMENT; |
| FREE(timer); |
| return ESR_SUCCESS; |
| } |
| |
| /** |
| * Starts the timer. This sets the reference time from which all new elapsed |
| * time are computed. This does not reset the elapsed time to 0. This is |
| * useful to pause the timer. |
| **/ |
| ESR_ReturnCode PTimerStart(PTimer *timer) |
| { |
| if (timer == NULL) return ESR_INVALID_ARGUMENT; |
| return (QueryPerformanceCounter(&timer->RefTime) ? |
| ESR_SUCCESS : |
| ESR_NOT_SUPPORTED); |
| } |
| |
| /** |
| * Stops the timer. |
| **/ |
| ESR_ReturnCode PTimerStop(PTimer *timer) |
| { |
| if (timer == NULL) return ESR_INVALID_ARGUMENT; |
| if (timer->RefTime.QuadPart != 0) |
| { |
| LARGE_INTEGER now; |
| if (!QueryPerformanceCounter(&now)) return ESR_NOT_SUPPORTED; |
| timer->elapsed.QuadPart += now.QuadPart - timer->RefTime.QuadPart; |
| timer->RefTime.QuadPart = 0; |
| } |
| return ESR_SUCCESS; |
| } |
| |
| /** |
| * Returns the timer elapsed time. If the Timer is in the stopped state, |
| * successive calls to getElapsed() will always return the same value. If |
| * the Timer is in the started state, successive calls will return the |
| * elapsed time since the last time PTimerStart() was called. |
| */ |
| ESR_ReturnCode PTimerGetElapsed(PTimer *timer, asr_uint32_t* elapsed) |
| { |
| if (timer == NULL || elapsed == NULL) |
| return ESR_INVALID_ARGUMENT; |
| |
| if (timer->RefTime.QuadPart != 0) |
| { |
| LARGE_INTEGER now; |
| if (!QueryPerformanceCounter(&now)) return ESR_NOT_SUPPORTED; |
| *elapsed = (asr_uint32_t) ((timer->elapsed.QuadPart + (now.QuadPart - timer->RefTime.QuadPart)) |
| / timer->PerformanceFreq.QuadPart); |
| } |
| else |
| *elapsed = (asr_uint32_t) (timer->elapsed.QuadPart / timer->PerformanceFreq.QuadPart); |
| |
| return ESR_SUCCESS; |
| } |
| |
| |
| /** |
| * Resets the elapsed time to 0 and resets the reference time of the Timer. |
| * This effectively reset the timer in the same state it was right after creation. |
| **/ |
| ESR_ReturnCode PTimerReset(PTimer *timer) |
| { |
| if (timer == NULL) return ESR_INVALID_ARGUMENT; |
| timer->RefTime.QuadPart = 0; |
| timer->elapsed.QuadPart = 0; |
| return ESR_SUCCESS; |
| } |
| |
| #elif defined(POSIX) |
| #include "ptrd.h" |
| /* |
| POSIX has a timer |
| */ |
| /* Clocks and timers: clock_settime, clock_gettime, clock_getres, timer_xxx and nanosleep */ |
| #ifndef _POSIX_TIMERS |
| #ifndef __vxworks /* __vxworks does not define it! */ |
| #error "Timer is not defined!" |
| #endif /* __vxworks */ |
| #endif /* _POSIX_TIMERS */ |
| |
| #define TIMER_MAX_VAL 10000 |
| |
| struct PTimer_t |
| { |
| timer_t timer; |
| asr_uint32_t elapsed; |
| }; |
| |
| /** |
| * Creates a new timer object. |
| **/ |
| ESR_ReturnCode PTimerCreate(PTimer **timer) |
| { |
| PTimer *tmp = NULL; |
| |
| if (timer == NULL) return ESR_INVALID_ARGUMENT; |
| tmp = NEW(PTimer, "PTimer"); |
| if (tmp == NULL) return ESR_OUT_OF_MEMORY; |
| |
| *timer = tmp; |
| if (timer_create(CLOCK_REALTIME, NULL, &(tmp->timer)) < 0) |
| return ESR_NOT_SUPPORTED; |
| |
| return ESR_SUCCESS; |
| } |
| |
| ESR_ReturnCode PTimerDestroy(PTimer *timer) |
| { |
| if (timer == NULL) return ESR_INVALID_ARGUMENT; |
| timer_delete(timer->timer); |
| FREE(timer); |
| |
| return ESR_SUCCESS; |
| } |
| |
| /** |
| * Starts the timer. This sets the reference time from which all new elapsed |
| * time are computed. This does not reset the elapsed time to 0. This is |
| * useful to pause the timer. |
| **/ |
| ESR_ReturnCode PTimerStart(PTimer *timer) |
| { |
| struct itimerspec expire_time; |
| |
| if (timer == NULL) return ESR_INVALID_ARGUMENT; |
| |
| expire_time.it_value.tv_sec = TIMER_MAX_VAL; /* set a large time for the timer */ |
| expire_time.it_value.tv_nsec = 0; |
| return (timer_settime(timer->timer, 0, &expire_time, NULL) == 0 ? |
| ESR_SUCCESS : |
| ESR_NOT_SUPPORTED); |
| } |
| |
| /** |
| * Stops the timer. |
| **/ |
| ESR_ReturnCode PTimerStop(PTimer *timer) |
| { |
| struct itimerspec remaining; |
| |
| if (timer == NULL) return ESR_INVALID_ARGUMENT; |
| |
| if (timer_gettime(timer->timer, &remaining) != 0) return ESR_NOT_SUPPORTED; |
| #if defined(__vxworks) |
| timer_cancel(timer->timer); |
| #endif |
| timer->elapsed = (asr_uint32_t) ((TIMER_MAX_VAL - remaining.it_value.tv_sec) * SECOND2MSECOND |
| - remaining.it_value.tv_nsec / MSECOND2NSECOND); |
| |
| return ESR_SUCCESS; |
| } |
| |
| /** |
| * Returns the timer elapsed time. If the Timer is in the stopped state, |
| * successive calls to getElapsed() will always return the same value. If |
| * the Timer is in the started state, successive calls will return the |
| * elapsed time since the last time PTimerStart() was called. |
| */ |
| ESR_ReturnCode PTimerGetElapsed(PTimer *timer, asr_uint32_t* elapsed) |
| { |
| if (timer == NULL || elapsed == NULL) |
| return ESR_INVALID_ARGUMENT; |
| |
| if (timer->elapsed == 0) /* stop is not called */ |
| { |
| struct itimerspec remaining; |
| if (timer_gettime(timer->timer, &remaining) != 0) return ESR_NOT_SUPPORTED; |
| *elapsed = (asr_uint32_t) ((TIMER_MAX_VAL - remaining.it_value.tv_sec) * SECOND2MSECOND |
| - remaining.it_value.tv_nsec / MSECOND2NSECOND); |
| } |
| else |
| *elapsed = timer->elapsed; |
| |
| return ESR_SUCCESS; |
| } |
| |
| |
| /** |
| * Resets the elapsed time to 0 and resets the reference time of the Timer. |
| * This effectively reset the timer in the same state it was right after creation. |
| **/ |
| ESR_ReturnCode PTimerReset(PTimer *timer) |
| { |
| if (timer == NULL) return ESR_INVALID_ARGUMENT; |
| timer->elapsed = 0; |
| return ESR_SUCCESS; |
| } |
| |
| #else |
| #error "Ptimer not implemented for this platform." |
| #endif |