-
Notifications
You must be signed in to change notification settings - Fork 7
/
stress-bsearch.c
128 lines (116 loc) · 3.2 KB
/
stress-bsearch.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
/*
* Copyright (C) 2013-2018 Canonical, Ltd.
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*
* This code is a complete clean re-write of the stress tool by
* Colin Ian King <colin.king@canonical.com> and attempts to be
* backwardly compatible with the stress tool by Amos Waterland
* <apw@rossby.metr.ou.edu> but has more stress tests and more
* functionality.
*
*/
#include "stress-ng.h"
/*
* stress_set_bsearch_size()
* set bsearch size from given option string
*/
void stress_set_bsearch_size(const char *opt)
{
uint64_t bsearch_size;
bsearch_size = get_uint64(opt);
check_range("bsearch-size", bsearch_size,
MIN_BSEARCH_SIZE, MAX_BSEARCH_SIZE);
set_setting("bsearch-size", TYPE_ID_UINT64, &bsearch_size);
}
/*
* cmp()
* compare int32 values for bsearch
*/
static int cmp(const void *p1, const void *p2)
{
const int32_t *i1 = (const int32_t *)p1;
const int32_t *i2 = (const int32_t *)p2;
if (*i1 > *i2)
return 1;
else if (*i1 < *i2)
return -1;
else
return 0;
}
/*
* Monotonically increasing values
*/
#define SETDATA(d, i, v, prev) \
d[i] = 1 + prev + (v & 0x7); \
v >>= 2; \
prev = d[i]; \
i++; \
/*
* stress_bsearch()
* stress bsearch
*/
int stress_bsearch(const args_t *args)
{
int32_t *data, *ptr, prev = 0;
size_t n, n8, i;
uint64_t bsearch_size = DEFAULT_BSEARCH_SIZE;
if (!get_setting("bsearch-size", &bsearch_size)) {
if (g_opt_flags & OPT_FLAGS_MAXIMIZE)
bsearch_size = MAX_BSEARCH_SIZE;
if (g_opt_flags & OPT_FLAGS_MINIMIZE)
bsearch_size = MIN_BSEARCH_SIZE;
}
n = (size_t)bsearch_size;
n8 = (n + 7) & ~7;
/* allocate in multiples of 8 */
if ((data = calloc(n8, sizeof(*data))) == NULL) {
pr_fail_dbg("malloc");
return EXIT_NO_RESOURCE;
}
/* Populate with ascending data */
prev = 0;
for (i = 0; i < n;) {
uint64_t v = mwc64();
SETDATA(data, i, v, prev);
SETDATA(data, i, v, prev);
SETDATA(data, i, v, prev);
SETDATA(data, i, v, prev);
SETDATA(data, i, v, prev);
SETDATA(data, i, v, prev);
SETDATA(data, i, v, prev);
SETDATA(data, i, v, prev);
}
do {
for (ptr = data, i = 0; i < n; i++, ptr++) {
int32_t *result;
result = bsearch(ptr, data, n, sizeof(*ptr), cmp);
if (g_opt_flags & OPT_FLAGS_VERIFY) {
if (result == NULL)
pr_fail("%s: element %zu could "
"not be found\n",
args->name, i);
else if (*result != *ptr)
pr_fail("%s: element %zu "
"found %" PRIu32
", expecting %" PRIu32 "\n",
args->name, i, *result, *ptr);
}
}
inc_counter(args);
} while (keep_stressing());
free(data);
return EXIT_SUCCESS;
}