blob: 4683d3513b78abf6b3bbe7cfbb3143cbbde8d0a4 [file] [log] [blame]
Pierre Ossman236c03c2014-07-04 14:12:49 +02001/* Copyright 2013-2014 Pierre Ossman <ossman@cendio.se> for Cendio AB
2 *
3 * This is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License as published by
5 * the Free Software Foundation; either version 2 of the License, or
6 * (at your option) any later version.
7 *
8 * This software is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 *
13 * You should have received a copy of the GNU General Public License
14 * along with this software; if not, write to the Free Software
15 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
16 * USA.
17 */
18
19#include <stdint.h>
Pierre Ossman8ac31112015-02-06 13:50:36 +010020#include <stdlib.h>
21#include <string.h>
Pierre Ossman236c03c2014-07-04 14:12:49 +020022
23#ifdef WIN32
24#include <windows.h>
25#else
26#include <sys/resource.h>
27#endif
28
Pierre Ossman8ac31112015-02-06 13:50:36 +010029#include "util.h"
30
Pierre Ossman236c03c2014-07-04 14:12:49 +020031#ifdef WIN32
Pierre Ossmanb591c9d2015-11-09 15:29:30 +010032typedef struct {
33 FILETIME kernelTime;
34 FILETIME userTime;
35} syscounter_t;
Pierre Ossman236c03c2014-07-04 14:12:49 +020036#else
Pierre Ossman8ac31112015-02-06 13:50:36 +010037typedef struct rusage syscounter_t;
Pierre Ossman236c03c2014-07-04 14:12:49 +020038#endif
39
Pierre Ossman8ac31112015-02-06 13:50:36 +010040static syscounter_t _globalCounter[2];
41static cpucounter_t globalCounter = _globalCounter;
42
43void startCpuCounter(void)
44{
45 startCpuCounter(globalCounter);
46}
47
48void endCpuCounter(void)
49{
50 endCpuCounter(globalCounter);
51}
52
53double getCpuCounter(void)
54{
55 return getCpuCounter(globalCounter);
56}
57
58cpucounter_t newCpuCounter(void)
59{
60 syscounter_t *c;
61
62 c = (syscounter_t*)malloc(sizeof(syscounter_t) * 2);
63 if (c == NULL)
64 return NULL;
65
66 memset(c, 0, sizeof(syscounter_t) * 2);
67
68 return c;
69}
70
71void freeCpuCounter(cpucounter_t c)
72{
73 free(c);
74}
75
76static void measureCpu(syscounter_t *counter)
Pierre Ossman236c03c2014-07-04 14:12:49 +020077{
78#ifdef WIN32
Pierre Ossmanb591c9d2015-11-09 15:29:30 +010079 FILETIME dummy1, dummy2;
Pierre Ossman236c03c2014-07-04 14:12:49 +020080
81 GetProcessTimes(GetCurrentProcess(), &dummy1, &dummy2,
Pierre Ossmanb591c9d2015-11-09 15:29:30 +010082 &counter->kernelTime, &counter->userTime);
Pierre Ossman236c03c2014-07-04 14:12:49 +020083#else
Pierre Ossman8ac31112015-02-06 13:50:36 +010084 getrusage(RUSAGE_SELF, counter);
Pierre Ossman236c03c2014-07-04 14:12:49 +020085#endif
86}
87
Pierre Ossman8ac31112015-02-06 13:50:36 +010088void startCpuCounter(cpucounter_t c)
Pierre Ossman236c03c2014-07-04 14:12:49 +020089{
Pierre Ossman8ac31112015-02-06 13:50:36 +010090 syscounter_t *s = (syscounter_t*)c;
91 measureCpu(&s[0]);
Pierre Ossman236c03c2014-07-04 14:12:49 +020092}
93
Pierre Ossman8ac31112015-02-06 13:50:36 +010094void endCpuCounter(cpucounter_t c)
Pierre Ossman236c03c2014-07-04 14:12:49 +020095{
Pierre Ossman8ac31112015-02-06 13:50:36 +010096 syscounter_t *s = (syscounter_t*)c;
97 measureCpu(&s[1]);
Pierre Ossman236c03c2014-07-04 14:12:49 +020098}
99
Pierre Ossman8ac31112015-02-06 13:50:36 +0100100double getCpuCounter(cpucounter_t c)
Pierre Ossman236c03c2014-07-04 14:12:49 +0200101{
Pierre Ossman8ac31112015-02-06 13:50:36 +0100102 syscounter_t *s = (syscounter_t*)c;
Pierre Ossmanb591c9d2015-11-09 15:29:30 +0100103 double sysSeconds, userSeconds;
Pierre Ossman236c03c2014-07-04 14:12:49 +0200104
105#ifdef WIN32
106 uint64_t counters[2];
107
Pierre Ossmanb591c9d2015-11-09 15:29:30 +0100108 counters[0] = (uint64_t)s[0].kernelTime.dwHighDateTime << 32 |
109 s[0].kernelTime.dwLowDateTime;
110 counters[1] = (uint64_t)s[1].kernelTime.dwHighDateTime << 32 |
111 s[1].kernelTime.dwLowDateTime;
Pierre Ossman236c03c2014-07-04 14:12:49 +0200112
Pierre Ossmanb591c9d2015-11-09 15:29:30 +0100113 sysSeconds = (double)(counters[1] - counters[0]) / 10000000.0;
114
115 counters[0] = (uint64_t)s[0].userTime.dwHighDateTime << 32 |
116 s[0].userTime.dwLowDateTime;
117 counters[1] = (uint64_t)s[1].userTime.dwHighDateTime << 32 |
118 s[1].userTime.dwLowDateTime;
119
120 userSeconds = (double)(counters[1] - counters[0]) / 10000000.0;
Pierre Ossman236c03c2014-07-04 14:12:49 +0200121#else
Pierre Ossmanb591c9d2015-11-09 15:29:30 +0100122 sysSeconds = (double)(s[1].ru_stime.tv_sec -
123 s[0].ru_stime.tv_sec);
124 sysSeconds += (double)(s[1].ru_stime.tv_usec -
125 s[0].ru_stime.tv_usec) / 1000000.0;
126
127 userSeconds = (double)(s[1].ru_utime.tv_sec -
128 s[0].ru_utime.tv_sec);
129 userSeconds += (double)(s[1].ru_utime.tv_usec -
130 s[0].ru_utime.tv_usec) / 1000000.0;
Pierre Ossman236c03c2014-07-04 14:12:49 +0200131#endif
132
Pierre Ossmanb591c9d2015-11-09 15:29:30 +0100133 return sysSeconds + userSeconds;
Pierre Ossman236c03c2014-07-04 14:12:49 +0200134}