Merge tag 'xtensa-20180225' of git://github.com/jcmvbkbc/linux-xtensa
[cris-mirror.git] / tools / perf / util / time-utils.c
blob6193b46050a566a3a3752b3b955197d2b8cd6ac1
1 // SPDX-License-Identifier: GPL-2.0
2 #include <stdlib.h>
3 #include <string.h>
4 #include <sys/time.h>
5 #include <linux/time64.h>
6 #include <time.h>
7 #include <errno.h>
8 #include <inttypes.h>
9 #include <math.h>
11 #include "perf.h"
12 #include "debug.h"
13 #include "time-utils.h"
15 int parse_nsec_time(const char *str, u64 *ptime)
17 u64 time_sec, time_nsec;
18 char *end;
20 time_sec = strtoul(str, &end, 10);
21 if (*end != '.' && *end != '\0')
22 return -1;
24 if (*end == '.') {
25 int i;
26 char nsec_buf[10];
28 if (strlen(++end) > 9)
29 return -1;
31 strncpy(nsec_buf, end, 9);
32 nsec_buf[9] = '\0';
34 /* make it nsec precision */
35 for (i = strlen(nsec_buf); i < 9; i++)
36 nsec_buf[i] = '0';
38 time_nsec = strtoul(nsec_buf, &end, 10);
39 if (*end != '\0')
40 return -1;
41 } else
42 time_nsec = 0;
44 *ptime = time_sec * NSEC_PER_SEC + time_nsec;
45 return 0;
48 static int parse_timestr_sec_nsec(struct perf_time_interval *ptime,
49 char *start_str, char *end_str)
51 if (start_str && (*start_str != '\0') &&
52 (parse_nsec_time(start_str, &ptime->start) != 0)) {
53 return -1;
56 if (end_str && (*end_str != '\0') &&
57 (parse_nsec_time(end_str, &ptime->end) != 0)) {
58 return -1;
61 return 0;
64 static int split_start_end(char **start, char **end, const char *ostr, char ch)
66 char *start_str, *end_str;
67 char *d, *str;
69 if (ostr == NULL || *ostr == '\0')
70 return 0;
72 /* copy original string because we need to modify it */
73 str = strdup(ostr);
74 if (str == NULL)
75 return -ENOMEM;
77 start_str = str;
78 d = strchr(start_str, ch);
79 if (d) {
80 *d = '\0';
81 ++d;
83 end_str = d;
85 *start = start_str;
86 *end = end_str;
88 return 0;
91 int perf_time__parse_str(struct perf_time_interval *ptime, const char *ostr)
93 char *start_str = NULL, *end_str;
94 int rc;
96 rc = split_start_end(&start_str, &end_str, ostr, ',');
97 if (rc || !start_str)
98 return rc;
100 ptime->start = 0;
101 ptime->end = 0;
103 rc = parse_timestr_sec_nsec(ptime, start_str, end_str);
105 free(start_str);
107 /* make sure end time is after start time if it was given */
108 if (rc == 0 && ptime->end && ptime->end < ptime->start)
109 return -EINVAL;
111 pr_debug("start time %" PRIu64 ", ", ptime->start);
112 pr_debug("end time %" PRIu64 "\n", ptime->end);
114 return rc;
117 static int parse_percent(double *pcnt, char *str)
119 char *c, *endptr;
120 double d;
122 c = strchr(str, '%');
123 if (c)
124 *c = '\0';
125 else
126 return -1;
128 d = strtod(str, &endptr);
129 if (endptr != str + strlen(str))
130 return -1;
132 *pcnt = d / 100.0;
133 return 0;
136 static int percent_slash_split(char *str, struct perf_time_interval *ptime,
137 u64 start, u64 end)
139 char *p, *end_str;
140 double pcnt, start_pcnt, end_pcnt;
141 u64 total = end - start;
142 int i;
145 * Example:
146 * 10%/2: select the second 10% slice and the third 10% slice
149 /* We can modify this string since the original one is copied */
150 p = strchr(str, '/');
151 if (!p)
152 return -1;
154 *p = '\0';
155 if (parse_percent(&pcnt, str) < 0)
156 return -1;
158 p++;
159 i = (int)strtol(p, &end_str, 10);
160 if (*end_str)
161 return -1;
163 if (pcnt <= 0.0)
164 return -1;
166 start_pcnt = pcnt * (i - 1);
167 end_pcnt = pcnt * i;
169 if (start_pcnt < 0.0 || start_pcnt > 1.0 ||
170 end_pcnt < 0.0 || end_pcnt > 1.0) {
171 return -1;
174 ptime->start = start + round(start_pcnt * total);
175 ptime->end = start + round(end_pcnt * total);
177 return 0;
180 static int percent_dash_split(char *str, struct perf_time_interval *ptime,
181 u64 start, u64 end)
183 char *start_str = NULL, *end_str;
184 double start_pcnt, end_pcnt;
185 u64 total = end - start;
186 int ret;
189 * Example: 0%-10%
192 ret = split_start_end(&start_str, &end_str, str, '-');
193 if (ret || !start_str)
194 return ret;
196 if ((parse_percent(&start_pcnt, start_str) != 0) ||
197 (parse_percent(&end_pcnt, end_str) != 0)) {
198 free(start_str);
199 return -1;
202 free(start_str);
204 if (start_pcnt < 0.0 || start_pcnt > 1.0 ||
205 end_pcnt < 0.0 || end_pcnt > 1.0 ||
206 start_pcnt > end_pcnt) {
207 return -1;
210 ptime->start = start + round(start_pcnt * total);
211 ptime->end = start + round(end_pcnt * total);
213 return 0;
216 typedef int (*time_pecent_split)(char *, struct perf_time_interval *,
217 u64 start, u64 end);
219 static int percent_comma_split(struct perf_time_interval *ptime_buf, int num,
220 const char *ostr, u64 start, u64 end,
221 time_pecent_split func)
223 char *str, *p1, *p2;
224 int len, ret, i = 0;
226 str = strdup(ostr);
227 if (str == NULL)
228 return -ENOMEM;
230 len = strlen(str);
231 p1 = str;
233 while (p1 < str + len) {
234 if (i >= num) {
235 free(str);
236 return -1;
239 p2 = strchr(p1, ',');
240 if (p2)
241 *p2 = '\0';
243 ret = (func)(p1, &ptime_buf[i], start, end);
244 if (ret < 0) {
245 free(str);
246 return -1;
249 pr_debug("start time %d: %" PRIu64 ", ", i, ptime_buf[i].start);
250 pr_debug("end time %d: %" PRIu64 "\n", i, ptime_buf[i].end);
252 i++;
254 if (p2)
255 p1 = p2 + 1;
256 else
257 break;
260 free(str);
261 return i;
264 static int one_percent_convert(struct perf_time_interval *ptime_buf,
265 const char *ostr, u64 start, u64 end, char *c)
267 char *str;
268 int len = strlen(ostr), ret;
271 * c points to '%'.
272 * '%' should be the last character
274 if (ostr + len - 1 != c)
275 return -1;
278 * Construct a string like "xx%/1"
280 str = malloc(len + 3);
281 if (str == NULL)
282 return -ENOMEM;
284 memcpy(str, ostr, len);
285 strcpy(str + len, "/1");
287 ret = percent_slash_split(str, ptime_buf, start, end);
288 if (ret == 0)
289 ret = 1;
291 free(str);
292 return ret;
295 int perf_time__percent_parse_str(struct perf_time_interval *ptime_buf, int num,
296 const char *ostr, u64 start, u64 end)
298 char *c;
301 * ostr example:
302 * 10%/2,10%/3: select the second 10% slice and the third 10% slice
303 * 0%-10%,30%-40%: multiple time range
304 * 50%: just one percent
307 memset(ptime_buf, 0, sizeof(*ptime_buf) * num);
309 c = strchr(ostr, '/');
310 if (c) {
311 return percent_comma_split(ptime_buf, num, ostr, start,
312 end, percent_slash_split);
315 c = strchr(ostr, '-');
316 if (c) {
317 return percent_comma_split(ptime_buf, num, ostr, start,
318 end, percent_dash_split);
321 c = strchr(ostr, '%');
322 if (c)
323 return one_percent_convert(ptime_buf, ostr, start, end, c);
325 return -1;
328 struct perf_time_interval *perf_time__range_alloc(const char *ostr, int *size)
330 const char *p1, *p2;
331 int i = 1;
332 struct perf_time_interval *ptime;
335 * At least allocate one time range.
337 if (!ostr)
338 goto alloc;
340 p1 = ostr;
341 while (p1 < ostr + strlen(ostr)) {
342 p2 = strchr(p1, ',');
343 if (!p2)
344 break;
346 p1 = p2 + 1;
347 i++;
350 alloc:
351 *size = i;
352 ptime = calloc(i, sizeof(*ptime));
353 return ptime;
356 bool perf_time__skip_sample(struct perf_time_interval *ptime, u64 timestamp)
358 /* if time is not set don't drop sample */
359 if (timestamp == 0)
360 return false;
362 /* otherwise compare sample time to time window */
363 if ((ptime->start && timestamp < ptime->start) ||
364 (ptime->end && timestamp > ptime->end)) {
365 return true;
368 return false;
371 bool perf_time__ranges_skip_sample(struct perf_time_interval *ptime_buf,
372 int num, u64 timestamp)
374 struct perf_time_interval *ptime;
375 int i;
377 if ((timestamp == 0) || (num == 0))
378 return false;
380 if (num == 1)
381 return perf_time__skip_sample(&ptime_buf[0], timestamp);
384 * start/end of multiple time ranges must be valid.
386 for (i = 0; i < num; i++) {
387 ptime = &ptime_buf[i];
389 if (timestamp >= ptime->start &&
390 ((timestamp < ptime->end && i < num - 1) ||
391 (timestamp <= ptime->end && i == num - 1))) {
392 break;
396 return (i == num) ? true : false;
399 int timestamp__scnprintf_usec(u64 timestamp, char *buf, size_t sz)
401 u64 sec = timestamp / NSEC_PER_SEC;
402 u64 usec = (timestamp % NSEC_PER_SEC) / NSEC_PER_USEC;
404 return scnprintf(buf, sz, "%"PRIu64".%06"PRIu64, sec, usec);
407 int fetch_current_timestamp(char *buf, size_t sz)
409 struct timeval tv;
410 struct tm tm;
411 char dt[32];
413 if (gettimeofday(&tv, NULL) || !localtime_r(&tv.tv_sec, &tm))
414 return -1;
416 if (!strftime(dt, sizeof(dt), "%Y%m%d%H%M%S", &tm))
417 return -1;
419 scnprintf(buf, sz, "%s%02u", dt, (unsigned)tv.tv_usec / 10000);
421 return 0;