axmap: fix continued sequential bit setting
[fio.git] / t / axmap.c
CommitLineData
ad1f90aa
JA
1#include <stdio.h>
2#include <stdlib.h>
ad1f90aa
JA
3#include <inttypes.h>
4
5#include "../lib/lfsr.h"
32bbd3ab 6#include "../lib/axmap.h"
ad1f90aa 7
a7448122 8static int test_regular(size_t size, int seed)
ad1f90aa
JA
9{
10 struct fio_lfsr lfsr;
ad1f90aa 11 struct axmap *map;
a7448122 12 size_t osize;
32bbd3ab 13 uint64_t ff;
a7448122 14 int err;
ad1f90aa 15
a7448122
JA
16 printf("Using %llu entries...", (unsigned long long) size);
17 fflush(stdout);
ad1f90aa 18
d474cbc9 19 lfsr_init(&lfsr, size, seed, seed & 0xF);
ad1f90aa 20 map = axmap_new(size);
32bbd3ab 21 osize = size;
a7448122 22 err = 0;
ad1f90aa
JA
23
24 while (size--) {
25 uint64_t val;
26
dd9bd2b2 27 if (lfsr_next(&lfsr, &val)) {
32bbd3ab 28 printf("lfsr: short loop\n");
a7448122 29 err = 1;
32bbd3ab
JA
30 break;
31 }
ab3379bc
JA
32 if (axmap_isset(map, val)) {
33 printf("bit already set\n");
a7448122 34 err = 1;
ab3379bc
JA
35 break;
36 }
ad1f90aa 37 axmap_set(map, val);
ab3379bc
JA
38 if (!axmap_isset(map, val)) {
39 printf("bit not set\n");
a7448122 40 err = 1;
ab3379bc
JA
41 break;
42 }
ad1f90aa
JA
43 }
44
a7448122
JA
45 if (err)
46 return err;
47
32bbd3ab
JA
48 ff = axmap_next_free(map, osize);
49 if (ff != (uint64_t) -1ULL) {
50 printf("axmap_next_free broken: got %llu\n", (unsigned long long) ff);
51 return 1;
52 }
53
a7448122
JA
54 printf("pass!\n");
55 axmap_free(map);
56 return 0;
57}
58
59static int test_multi(size_t size, unsigned int bit_off)
60{
61 unsigned int map_size = size;
62 struct axmap *map;
63 uint64_t val = bit_off;
64 int i, err;
65
66 printf("Test multi %llu entries %u offset...", (unsigned long long) size, bit_off);
67 fflush(stdout);
68
69 map = axmap_new(map_size);
70 while (val + 128 <= map_size) {
71 err = 0;
72 for (i = val; i < val + 128; i++) {
73 if (axmap_isset(map, val + i)) {
74 printf("bit already set\n");
75 err = 1;
76 break;
77 }
78 }
79
80 if (err)
81 break;
82
83 err = axmap_set_nr(map, val, 128);
84 if (err != 128) {
85 printf("only set %u bits\n", err);
86 break;
87 }
88
89 err = 0;
90 for (i = 0; i < 128; i++) {
91 if (!axmap_isset(map, val + i)) {
92 printf("bit not set: %llu\n", (unsigned long long) val + i);
93 err = 1;
94 break;
95 }
96 }
97
98 val += 128;
99 if (err)
100 break;
101 }
102
103 if (!err)
104 printf("pass!\n");
105
106 axmap_free(map);
107 return err;
108}
109
dc54a01f
JA
110struct overlap_test {
111 unsigned int start;
112 unsigned int nr;
113 unsigned int ret;
114};
115
9a5950bf
JA
116static int test_overlap(void)
117{
dc54a01f 118 struct overlap_test tests[] = {
5031b5cf
JA
119 {
120 .start = 0,
121 .nr = 0,
122 .ret = 0,
123 },
dc54a01f
JA
124 {
125 .start = 16,
126 .nr = 16,
127 .ret = 16,
128 },
5031b5cf
JA
129 {
130 .start = 16,
131 .nr = 0,
132 .ret = 0,
133 },
dc54a01f
JA
134 {
135 .start = 0,
136 .nr = 32,
137 .ret = 16,
138 },
139 {
140 .start = 48,
141 .nr = 32,
142 .ret = 32,
143 },
144 {
145 .start = 32,
146 .nr = 32,
147 .ret = 16,
148 },
64e6d4e6
JA
149 {
150 .start = 79,
151 .nr = 1,
152 .ret = 0,
153 },
154 {
155 .start = 80,
156 .nr = 21,
157 .ret = 21,
158 },
dc54a01f
JA
159 {
160 .start = 102,
161 .nr = 1,
162 .ret = 1,
163 },
164 {
165 .start = 101,
166 .nr = 3,
167 .ret = 1,
168 },
169 {
170 .start = 106,
171 .nr = 4,
172 .ret = 4,
173 },
174 {
175 .start = 105,
176 .nr = 3,
177 .ret = 1,
178 },
179 {
180 .start = 120,
181 .nr = 4,
182 .ret = 4,
183 },
184 {
185 .start = 118,
186 .nr = 2,
187 .ret = 2,
188 },
189 {
190 .start = 118,
191 .nr = 2,
192 .ret = 0,
193 },
194 {
195 .start = -1U,
196 },
197 };
9a5950bf 198 struct axmap *map;
dc54a01f 199 int entries, i, ret, err = 0;
9a5950bf 200
dc54a01f
JA
201 entries = 0;
202 for (i = 0; tests[i].start != 1U; i++) {
203 unsigned int this = tests[i].start + tests[i].nr;
53bf886a 204
dc54a01f
JA
205 if (this > entries)
206 entries = this;
53bf886a
JA
207 }
208
dc54a01f
JA
209 printf("Test overlaps...");
210 fflush(stdout);
9a5950bf 211
dc54a01f 212 map = axmap_new(entries);
9a5950bf 213
dc54a01f
JA
214 for (i = 0; tests[i].start != -1U; i++) {
215 struct overlap_test *t = &tests[i];
9a5950bf 216
dc54a01f
JA
217 ret = axmap_set_nr(map, t->start, t->nr);
218 if (ret != t->ret) {
219 printf("fail\n");
220 printf("start=%u, nr=%d, ret=%d: %d\n", t->start, t->nr,
221 t->ret, ret);
222 err = 1;
223 break;
224 }
9a5950bf
JA
225 }
226
2baa609d
JA
227 if (!err)
228 printf("pass!\n");
9a5950bf 229 axmap_free(map);
dc54a01f 230 return err;
9a5950bf
JA
231}
232
a7448122
JA
233int main(int argc, char *argv[])
234{
235 size_t size = (1UL << 23) - 200;
236 int seed = 1;
237
238 if (argc > 1) {
239 size = strtoul(argv[1], NULL, 10);
240 if (argc > 2)
241 seed = strtoul(argv[2], NULL, 10);
242 }
243
244 if (test_regular(size, seed))
245 return 1;
246 if (test_multi(size, 0))
247 return 2;
248 if (test_multi(size, 17))
249 return 3;
9a5950bf
JA
250 if (test_overlap())
251 return 4;
a7448122 252
ad1f90aa
JA
253 return 0;
254}