/* * Copyright (c) 2010-2012 Niels Provos and Nick Mathewson * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * 3. The name of the author may not be used to endorse or promote products * derived from this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "event2/event-config.h" #ifdef _WIN32 #include #include #else #include #endif #include #include #ifdef EVENT__HAVE_SYS_TIME_H #include #endif #ifdef EVENT__HAVE_SYS_SOCKET_H #include #endif #include #include #include #include #include #include "event2/event.h" #include "event2/util.h" #include struct cpu_usage_timer { #ifdef _WIN32 HANDLE thread; FILETIME usertimeBegin; FILETIME kerneltimeBegin; #else clock_t ticksBegin; #endif struct timeval timeBegin; }; static void start_cpu_usage_timer(struct cpu_usage_timer *timer) { #ifdef _WIN32 int r; FILETIME createtime, exittime; timer->thread = GetCurrentThread(); r = GetThreadTimes(timer->thread, &createtime, &exittime, &timer->usertimeBegin, &timer->kerneltimeBegin); if (r==0) printf("GetThreadTimes failed."); #else timer->ticksBegin = clock(); #endif evutil_gettimeofday(&timer->timeBegin, NULL); } #ifdef _WIN32 static ev_int64_t filetime_to_100nsec(const FILETIME *ft) { /* Number of 100-nanosecond units */ ev_int64_t n = ft->dwHighDateTime; n <<= 32; n += ft->dwLowDateTime; return n; } static double filetime_diff(const FILETIME *ftStart, const FILETIME *ftEnd) { ev_int64_t s, e, diff; double r; s = filetime_to_100nsec(ftStart); e = filetime_to_100nsec(ftEnd); diff = e - s; r = (double) diff; return r / 1.0e7; } #endif static void get_cpu_usage(struct cpu_usage_timer *timer, double *secElapsedOut, double *secUsedOut, double *usageOut) { #ifdef _WIN32 double usertime_seconds, kerneltime_seconds; FILETIME createtime, exittime, usertimeEnd, kerneltimeEnd; int r; #else clock_t ticksEnd; #endif struct timeval timeEnd, timeDiff; double secondsPassed, secondsUsed; #ifdef _WIN32 r = GetThreadTimes(timer->thread, &createtime, &exittime, &usertimeEnd, &kerneltimeEnd); if (r==0) printf("GetThreadTimes failed."); usertime_seconds = filetime_diff(&timer->usertimeBegin, &usertimeEnd); kerneltime_seconds = filetime_diff(&timer->kerneltimeBegin, &kerneltimeEnd); secondsUsed = kerneltime_seconds + usertime_seconds; #else ticksEnd = clock(); secondsUsed = (ticksEnd - timer->ticksBegin) / (double)CLOCKS_PER_SEC; #endif evutil_gettimeofday(&timeEnd, NULL); evutil_timersub(&timeEnd, &timer->timeBegin, &timeDiff); secondsPassed = timeDiff.tv_sec + (timeDiff.tv_usec / 1.0e6); *secElapsedOut = secondsPassed; *secUsedOut = secondsUsed; *usageOut = secondsUsed / secondsPassed; } static void write_cb(evutil_socket_t fd, short event, void *arg) { printf("write callback. should only see this once\n"); /* got what we want remove the event */ event_del(*(struct event**)arg); /* opps changed my mind add it back again */ event_add(*(struct event**)arg,NULL); /* not a good day for decisiveness, I really didn't want it after all */ event_del(*(struct event**)arg); } static void timeout_cb(evutil_socket_t fd, short event, void *arg) { printf("timeout fired, time to end test\n"); event_del(*(struct event**)arg); return; } int main(int argc, char **argv) { struct event* ev = NULL; struct event* timeout = NULL; struct event_base* base = NULL; evutil_socket_t pair[2] = {EVUTIL_INVALID_SOCKET, EVUTIL_INVALID_SOCKET}; struct timeval tv; struct cpu_usage_timer timer; double usage, secPassed, secUsed; #ifdef _WIN32 WORD wVersionRequested; WSADATA wsaData; wVersionRequested = MAKEWORD(2, 2); (void) WSAStartup(wVersionRequested, &wsaData); #endif if (evutil_socketpair(AF_UNIX, SOCK_STREAM, 0, pair) == -1) return (1); /* Initialize the event library */ if (!(base = event_base_new())) return (1); /* Initialize a timeout to terminate the test */ timeout = evtimer_new(base,timeout_cb,&timeout); if (timeout == NULL) { perror("evtimer_new"); goto err; } /* and watch for writability on one end of the pipe */ ev = event_new(base,pair[1],EV_WRITE | EV_PERSIST, write_cb, &ev); if (ev == NULL) { perror("event_new"); goto err; } tv.tv_sec = 1; tv.tv_usec = 500*1000; evtimer_add(timeout, &tv); event_add(ev, NULL); start_cpu_usage_timer(&timer); event_base_dispatch(base); event_free(ev); event_free(timeout); event_base_free(base); get_cpu_usage(&timer, &secPassed, &secUsed, &usage); /* attempt to calculate our cpu usage over the test should be virtually nil */ printf("usec used=%d, usec passed=%d, cpu usage=%.2f%%\n", (int)(secUsed*1e6), (int)(secPassed*1e6), usage*100); if (usage > 50.0) /* way too high */ return 1; return 0; err: if (ev) event_free(ev); if (timeout) event_free(timeout); if (pair[0] != EVUTIL_INVALID_SOCKET) evutil_closesocket(pair[0]); if (pair[1] != EVUTIL_INVALID_SOCKET) evutil_closesocket(pair[1]); if (base) event_base_free(base); return 1; }