1 /*
2  * Copyright (c) 2016 Facebook
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of version 2 of the GNU General Public
6  * License as published by the Free Software Foundation.
7  */
8 #define _GNU_SOURCE
9 #include <stdio.h>
10 #include <unistd.h>
11 #include <errno.h>
12 #include <string.h>
13 #include <assert.h>
14 #include <sched.h>
15 #include <stdlib.h>
16 #include <time.h>
17 
18 #include <sys/wait.h>
19 
20 #include <bpf/bpf.h>
21 
22 #include "bpf_util.h"
23 #include "bpf_rlimit.h"
24 
25 #define LOCAL_FREE_TARGET	(128)
26 #define PERCPU_FREE_TARGET	(4)
27 
28 static int nr_cpus;
29 
30 static int create_map(int map_type, int map_flags, unsigned int size)
31 {
32 	int map_fd;
33 
34 	map_fd = bpf_create_map(map_type, sizeof(unsigned long long),
35 				sizeof(unsigned long long), size, map_flags);
36 
37 	if (map_fd == -1)
38 		perror("bpf_create_map");
39 
40 	return map_fd;
41 }
42 
43 static int map_subset(int map0, int map1)
44 {
45 	unsigned long long next_key = 0;
46 	unsigned long long value0[nr_cpus], value1[nr_cpus];
47 	int ret;
48 
49 	while (!bpf_map_get_next_key(map1, &next_key, &next_key)) {
50 		assert(!bpf_map_lookup_elem(map1, &next_key, value1));
51 		ret = bpf_map_lookup_elem(map0, &next_key, value0);
52 		if (ret) {
53 			printf("key:%llu not found from map. %s(%d)\n",
54 			       next_key, strerror(errno), errno);
55 			return 0;
56 		}
57 		if (value0[0] != value1[0]) {
58 			printf("key:%llu value0:%llu != value1:%llu\n",
59 			       next_key, value0[0], value1[0]);
60 			return 0;
61 		}
62 	}
63 	return 1;
64 }
65 
66 static int map_equal(int lru_map, int expected)
67 {
68 	return map_subset(lru_map, expected) && map_subset(expected, lru_map);
69 }
70 
71 static int sched_next_online(int pid, int *next_to_try)
72 {
73 	cpu_set_t cpuset;
74 	int next = *next_to_try;
75 	int ret = -1;
76 
77 	while (next < nr_cpus) {
78 		CPU_ZERO(&cpuset);
79 		CPU_SET(next++, &cpuset);
80 		if (!sched_setaffinity(pid, sizeof(cpuset), &cpuset)) {
81 			ret = 0;
82 			break;
83 		}
84 	}
85 
86 	*next_to_try = next;
87 	return ret;
88 }
89 
90 /* Size of the LRU amp is 2
91  * Add key=1 (+1 key)
92  * Add key=2 (+1 key)
93  * Lookup Key=1
94  * Add Key=3
95  *   => Key=2 will be removed by LRU
96  * Iterate map.  Only found key=1 and key=3
97  */
98 static void test_lru_sanity0(int map_type, int map_flags)
99 {
100 	unsigned long long key, value[nr_cpus];
101 	int lru_map_fd, expected_map_fd;
102 	int next_cpu = 0;
103 
104 	printf("%s (map_type:%d map_flags:0x%X): ", __func__, map_type,
105 	       map_flags);
106 
107 	assert(sched_next_online(0, &next_cpu) != -1);
108 
109 	if (map_flags & BPF_F_NO_COMMON_LRU)
110 		lru_map_fd = create_map(map_type, map_flags, 2 * nr_cpus);
111 	else
112 		lru_map_fd = create_map(map_type, map_flags, 2);
113 	assert(lru_map_fd != -1);
114 
115 	expected_map_fd = create_map(BPF_MAP_TYPE_HASH, 0, 2);
116 	assert(expected_map_fd != -1);
117 
118 	value[0] = 1234;
119 
120 	/* insert key=1 element */
121 
122 	key = 1;
123 	assert(!bpf_map_update_elem(lru_map_fd, &key, value, BPF_NOEXIST));
124 	assert(!bpf_map_update_elem(expected_map_fd, &key, value,
125 				    BPF_NOEXIST));
126 
127 	/* BPF_NOEXIST means: add new element if it doesn't exist */
128 	assert(bpf_map_update_elem(lru_map_fd, &key, value, BPF_NOEXIST) == -1
129 	       /* key=1 already exists */
130 	       && errno == EEXIST);
131 
132 	assert(bpf_map_update_elem(lru_map_fd, &key, value, -1) == -1 &&
133 	       errno == EINVAL);
134 
135 	/* insert key=2 element */
136 
137 	/* check that key=2 is not found */
138 	key = 2;
139 	assert(bpf_map_lookup_elem(lru_map_fd, &key, value) == -1 &&
140 	       errno == ENOENT);
141 
142 	/* BPF_EXIST means: update existing element */
143 	assert(bpf_map_update_elem(lru_map_fd, &key, value, BPF_EXIST) == -1 &&
144 	       /* key=2 is not there */
145 	       errno == ENOENT);
146 
147 	assert(!bpf_map_update_elem(lru_map_fd, &key, value, BPF_NOEXIST));
148 
149 	/* insert key=3 element */
150 
151 	/* check that key=3 is not found */
152 	key = 3;
153 	assert(bpf_map_lookup_elem(lru_map_fd, &key, value) == -1 &&
154 	       errno == ENOENT);
155 
156 	/* check that key=1 can be found and mark the ref bit to
157 	 * stop LRU from removing key=1
158 	 */
159 	key = 1;
160 	assert(!bpf_map_lookup_elem(lru_map_fd, &key, value));
161 	assert(value[0] == 1234);
162 
163 	key = 3;
164 	assert(!bpf_map_update_elem(lru_map_fd, &key, value, BPF_NOEXIST));
165 	assert(!bpf_map_update_elem(expected_map_fd, &key, value,
166 				    BPF_NOEXIST));
167 
168 	/* key=2 has been removed from the LRU */
169 	key = 2;
170 	assert(bpf_map_lookup_elem(lru_map_fd, &key, value) == -1);
171 
172 	assert(map_equal(lru_map_fd, expected_map_fd));
173 
174 	close(expected_map_fd);
175 	close(lru_map_fd);
176 
177 	printf("Pass\n");
178 }
179 
180 /* Size of the LRU map is 1.5*tgt_free
181  * Insert 1 to tgt_free (+tgt_free keys)
182  * Lookup 1 to tgt_free/2
183  * Insert 1+tgt_free to 2*tgt_free (+tgt_free keys)
184  * => 1+tgt_free/2 to LOCALFREE_TARGET will be removed by LRU
185  */
186 static void test_lru_sanity1(int map_type, int map_flags, unsigned int tgt_free)
187 {
188 	unsigned long long key, end_key, value[nr_cpus];
189 	int lru_map_fd, expected_map_fd;
190 	unsigned int batch_size;
191 	unsigned int map_size;
192 	int next_cpu = 0;
193 
194 	if (map_flags & BPF_F_NO_COMMON_LRU)
195 		/* This test is only applicable to common LRU list */
196 		return;
197 
198 	printf("%s (map_type:%d map_flags:0x%X): ", __func__, map_type,
199 	       map_flags);
200 
201 	assert(sched_next_online(0, &next_cpu) != -1);
202 
203 	batch_size = tgt_free / 2;
204 	assert(batch_size * 2 == tgt_free);
205 
206 	map_size = tgt_free + batch_size;
207 	lru_map_fd = create_map(map_type, map_flags, map_size);
208 	assert(lru_map_fd != -1);
209 
210 	expected_map_fd = create_map(BPF_MAP_TYPE_HASH, 0, map_size);
211 	assert(expected_map_fd != -1);
212 
213 	value[0] = 1234;
214 
215 	/* Insert 1 to tgt_free (+tgt_free keys) */
216 	end_key = 1 + tgt_free;
217 	for (key = 1; key < end_key; key++)
218 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
219 					    BPF_NOEXIST));
220 
221 	/* Lookup 1 to tgt_free/2 */
222 	end_key = 1 + batch_size;
223 	for (key = 1; key < end_key; key++) {
224 		assert(!bpf_map_lookup_elem(lru_map_fd, &key, value));
225 		assert(!bpf_map_update_elem(expected_map_fd, &key, value,
226 					    BPF_NOEXIST));
227 	}
228 
229 	/* Insert 1+tgt_free to 2*tgt_free
230 	 * => 1+tgt_free/2 to LOCALFREE_TARGET will be
231 	 * removed by LRU
232 	 */
233 	key = 1 + tgt_free;
234 	end_key = key + tgt_free;
235 	for (; key < end_key; key++) {
236 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
237 					    BPF_NOEXIST));
238 		assert(!bpf_map_update_elem(expected_map_fd, &key, value,
239 					    BPF_NOEXIST));
240 	}
241 
242 	assert(map_equal(lru_map_fd, expected_map_fd));
243 
244 	close(expected_map_fd);
245 	close(lru_map_fd);
246 
247 	printf("Pass\n");
248 }
249 
250 /* Size of the LRU map 1.5 * tgt_free
251  * Insert 1 to tgt_free (+tgt_free keys)
252  * Update 1 to tgt_free/2
253  *   => The original 1 to tgt_free/2 will be removed due to
254  *      the LRU shrink process
255  * Re-insert 1 to tgt_free/2 again and do a lookup immeidately
256  * Insert 1+tgt_free to tgt_free*3/2
257  * Insert 1+tgt_free*3/2 to tgt_free*5/2
258  *   => Key 1+tgt_free to tgt_free*3/2
259  *      will be removed from LRU because it has never
260  *      been lookup and ref bit is not set
261  */
262 static void test_lru_sanity2(int map_type, int map_flags, unsigned int tgt_free)
263 {
264 	unsigned long long key, value[nr_cpus];
265 	unsigned long long end_key;
266 	int lru_map_fd, expected_map_fd;
267 	unsigned int batch_size;
268 	unsigned int map_size;
269 	int next_cpu = 0;
270 
271 	if (map_flags & BPF_F_NO_COMMON_LRU)
272 		/* This test is only applicable to common LRU list */
273 		return;
274 
275 	printf("%s (map_type:%d map_flags:0x%X): ", __func__, map_type,
276 	       map_flags);
277 
278 	assert(sched_next_online(0, &next_cpu) != -1);
279 
280 	batch_size = tgt_free / 2;
281 	assert(batch_size * 2 == tgt_free);
282 
283 	map_size = tgt_free + batch_size;
284 	lru_map_fd = create_map(map_type, map_flags, map_size);
285 	assert(lru_map_fd != -1);
286 
287 	expected_map_fd = create_map(BPF_MAP_TYPE_HASH, 0, map_size);
288 	assert(expected_map_fd != -1);
289 
290 	value[0] = 1234;
291 
292 	/* Insert 1 to tgt_free (+tgt_free keys) */
293 	end_key = 1 + tgt_free;
294 	for (key = 1; key < end_key; key++)
295 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
296 					    BPF_NOEXIST));
297 
298 	/* Any bpf_map_update_elem will require to acquire a new node
299 	 * from LRU first.
300 	 *
301 	 * The local list is running out of free nodes.
302 	 * It gets from the global LRU list which tries to
303 	 * shrink the inactive list to get tgt_free
304 	 * number of free nodes.
305 	 *
306 	 * Hence, the oldest key 1 to tgt_free/2
307 	 * are removed from the LRU list.
308 	 */
309 	key = 1;
310 	if (map_type == BPF_MAP_TYPE_LRU_PERCPU_HASH) {
311 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
312 					    BPF_NOEXIST));
313 		assert(!bpf_map_delete_elem(lru_map_fd, &key));
314 	} else {
315 		assert(bpf_map_update_elem(lru_map_fd, &key, value,
316 					   BPF_EXIST));
317 	}
318 
319 	/* Re-insert 1 to tgt_free/2 again and do a lookup
320 	 * immeidately.
321 	 */
322 	end_key = 1 + batch_size;
323 	value[0] = 4321;
324 	for (key = 1; key < end_key; key++) {
325 		assert(bpf_map_lookup_elem(lru_map_fd, &key, value));
326 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
327 					    BPF_NOEXIST));
328 		assert(!bpf_map_lookup_elem(lru_map_fd, &key, value));
329 		assert(value[0] == 4321);
330 		assert(!bpf_map_update_elem(expected_map_fd, &key, value,
331 					    BPF_NOEXIST));
332 	}
333 
334 	value[0] = 1234;
335 
336 	/* Insert 1+tgt_free to tgt_free*3/2 */
337 	end_key = 1 + tgt_free + batch_size;
338 	for (key = 1 + tgt_free; key < end_key; key++)
339 		/* These newly added but not referenced keys will be
340 		 * gone during the next LRU shrink.
341 		 */
342 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
343 					    BPF_NOEXIST));
344 
345 	/* Insert 1+tgt_free*3/2 to  tgt_free*5/2 */
346 	end_key = key + tgt_free;
347 	for (; key < end_key; key++) {
348 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
349 					    BPF_NOEXIST));
350 		assert(!bpf_map_update_elem(expected_map_fd, &key, value,
351 					    BPF_NOEXIST));
352 	}
353 
354 	assert(map_equal(lru_map_fd, expected_map_fd));
355 
356 	close(expected_map_fd);
357 	close(lru_map_fd);
358 
359 	printf("Pass\n");
360 }
361 
362 /* Size of the LRU map is 2*tgt_free
363  * It is to test the active/inactive list rotation
364  * Insert 1 to 2*tgt_free (+2*tgt_free keys)
365  * Lookup key 1 to tgt_free*3/2
366  * Add 1+2*tgt_free to tgt_free*5/2 (+tgt_free/2 keys)
367  *  => key 1+tgt_free*3/2 to 2*tgt_free are removed from LRU
368  */
369 static void test_lru_sanity3(int map_type, int map_flags, unsigned int tgt_free)
370 {
371 	unsigned long long key, end_key, value[nr_cpus];
372 	int lru_map_fd, expected_map_fd;
373 	unsigned int batch_size;
374 	unsigned int map_size;
375 	int next_cpu = 0;
376 
377 	if (map_flags & BPF_F_NO_COMMON_LRU)
378 		/* This test is only applicable to common LRU list */
379 		return;
380 
381 	printf("%s (map_type:%d map_flags:0x%X): ", __func__, map_type,
382 	       map_flags);
383 
384 	assert(sched_next_online(0, &next_cpu) != -1);
385 
386 	batch_size = tgt_free / 2;
387 	assert(batch_size * 2 == tgt_free);
388 
389 	map_size = tgt_free * 2;
390 	lru_map_fd = create_map(map_type, map_flags, map_size);
391 	assert(lru_map_fd != -1);
392 
393 	expected_map_fd = create_map(BPF_MAP_TYPE_HASH, 0, map_size);
394 	assert(expected_map_fd != -1);
395 
396 	value[0] = 1234;
397 
398 	/* Insert 1 to 2*tgt_free (+2*tgt_free keys) */
399 	end_key = 1 + (2 * tgt_free);
400 	for (key = 1; key < end_key; key++)
401 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
402 					    BPF_NOEXIST));
403 
404 	/* Lookup key 1 to tgt_free*3/2 */
405 	end_key = tgt_free + batch_size;
406 	for (key = 1; key < end_key; key++) {
407 		assert(!bpf_map_lookup_elem(lru_map_fd, &key, value));
408 		assert(!bpf_map_update_elem(expected_map_fd, &key, value,
409 					    BPF_NOEXIST));
410 	}
411 
412 	/* Add 1+2*tgt_free to tgt_free*5/2
413 	 * (+tgt_free/2 keys)
414 	 */
415 	key = 2 * tgt_free + 1;
416 	end_key = key + batch_size;
417 	for (; key < end_key; key++) {
418 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
419 					    BPF_NOEXIST));
420 		assert(!bpf_map_update_elem(expected_map_fd, &key, value,
421 					    BPF_NOEXIST));
422 	}
423 
424 	assert(map_equal(lru_map_fd, expected_map_fd));
425 
426 	close(expected_map_fd);
427 	close(lru_map_fd);
428 
429 	printf("Pass\n");
430 }
431 
432 /* Test deletion */
433 static void test_lru_sanity4(int map_type, int map_flags, unsigned int tgt_free)
434 {
435 	int lru_map_fd, expected_map_fd;
436 	unsigned long long key, value[nr_cpus];
437 	unsigned long long end_key;
438 	int next_cpu = 0;
439 
440 	printf("%s (map_type:%d map_flags:0x%X): ", __func__, map_type,
441 	       map_flags);
442 
443 	assert(sched_next_online(0, &next_cpu) != -1);
444 
445 	if (map_flags & BPF_F_NO_COMMON_LRU)
446 		lru_map_fd = create_map(map_type, map_flags,
447 					3 * tgt_free * nr_cpus);
448 	else
449 		lru_map_fd = create_map(map_type, map_flags, 3 * tgt_free);
450 	assert(lru_map_fd != -1);
451 
452 	expected_map_fd = create_map(BPF_MAP_TYPE_HASH, 0,
453 				     3 * tgt_free);
454 	assert(expected_map_fd != -1);
455 
456 	value[0] = 1234;
457 
458 	for (key = 1; key <= 2 * tgt_free; key++)
459 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
460 					    BPF_NOEXIST));
461 
462 	key = 1;
463 	assert(bpf_map_update_elem(lru_map_fd, &key, value, BPF_NOEXIST));
464 
465 	for (key = 1; key <= tgt_free; key++) {
466 		assert(!bpf_map_lookup_elem(lru_map_fd, &key, value));
467 		assert(!bpf_map_update_elem(expected_map_fd, &key, value,
468 					    BPF_NOEXIST));
469 	}
470 
471 	for (; key <= 2 * tgt_free; key++) {
472 		assert(!bpf_map_delete_elem(lru_map_fd, &key));
473 		assert(bpf_map_delete_elem(lru_map_fd, &key));
474 	}
475 
476 	end_key = key + 2 * tgt_free;
477 	for (; key < end_key; key++) {
478 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
479 					    BPF_NOEXIST));
480 		assert(!bpf_map_update_elem(expected_map_fd, &key, value,
481 					    BPF_NOEXIST));
482 	}
483 
484 	assert(map_equal(lru_map_fd, expected_map_fd));
485 
486 	close(expected_map_fd);
487 	close(lru_map_fd);
488 
489 	printf("Pass\n");
490 }
491 
492 static void do_test_lru_sanity5(unsigned long long last_key, int map_fd)
493 {
494 	unsigned long long key, value[nr_cpus];
495 
496 	/* Ensure the last key inserted by previous CPU can be found */
497 	assert(!bpf_map_lookup_elem(map_fd, &last_key, value));
498 
499 	value[0] = 1234;
500 
501 	key = last_key + 1;
502 	assert(!bpf_map_update_elem(map_fd, &key, value, BPF_NOEXIST));
503 	assert(!bpf_map_lookup_elem(map_fd, &key, value));
504 
505 	/* Cannot find the last key because it was removed by LRU */
506 	assert(bpf_map_lookup_elem(map_fd, &last_key, value));
507 }
508 
509 /* Test map with only one element */
510 static void test_lru_sanity5(int map_type, int map_flags)
511 {
512 	unsigned long long key, value[nr_cpus];
513 	int next_cpu = 0;
514 	int map_fd;
515 
516 	if (map_flags & BPF_F_NO_COMMON_LRU)
517 		return;
518 
519 	printf("%s (map_type:%d map_flags:0x%X): ", __func__, map_type,
520 	       map_flags);
521 
522 	map_fd = create_map(map_type, map_flags, 1);
523 	assert(map_fd != -1);
524 
525 	value[0] = 1234;
526 	key = 0;
527 	assert(!bpf_map_update_elem(map_fd, &key, value, BPF_NOEXIST));
528 
529 	while (sched_next_online(0, &next_cpu) != -1) {
530 		pid_t pid;
531 
532 		pid = fork();
533 		if (pid == 0) {
534 			do_test_lru_sanity5(key, map_fd);
535 			exit(0);
536 		} else if (pid == -1) {
537 			printf("couldn't spawn process to test key:%llu\n",
538 			       key);
539 			exit(1);
540 		} else {
541 			int status;
542 
543 			assert(waitpid(pid, &status, 0) == pid);
544 			assert(status == 0);
545 			key++;
546 		}
547 	}
548 
549 	close(map_fd);
550 	/* At least one key should be tested */
551 	assert(key > 0);
552 
553 	printf("Pass\n");
554 }
555 
556 /* Test list rotation for BPF_F_NO_COMMON_LRU map */
557 static void test_lru_sanity6(int map_type, int map_flags, int tgt_free)
558 {
559 	int lru_map_fd, expected_map_fd;
560 	unsigned long long key, value[nr_cpus];
561 	unsigned int map_size = tgt_free * 2;
562 	int next_cpu = 0;
563 
564 	if (!(map_flags & BPF_F_NO_COMMON_LRU))
565 		return;
566 
567 	printf("%s (map_type:%d map_flags:0x%X): ", __func__, map_type,
568 	       map_flags);
569 
570 	assert(sched_next_online(0, &next_cpu) != -1);
571 
572 	expected_map_fd = create_map(BPF_MAP_TYPE_HASH, 0, map_size);
573 	assert(expected_map_fd != -1);
574 
575 	lru_map_fd = create_map(map_type, map_flags, map_size * nr_cpus);
576 	assert(lru_map_fd != -1);
577 
578 	value[0] = 1234;
579 
580 	for (key = 1; key <= tgt_free; key++) {
581 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
582 					    BPF_NOEXIST));
583 		assert(!bpf_map_update_elem(expected_map_fd, &key, value,
584 					    BPF_NOEXIST));
585 	}
586 
587 	for (; key <= tgt_free * 2; key++) {
588 		unsigned long long stable_key;
589 
590 		/* Make ref bit sticky for key: [1, tgt_free] */
591 		for (stable_key = 1; stable_key <= tgt_free; stable_key++) {
592 			/* Mark the ref bit */
593 			assert(!bpf_map_lookup_elem(lru_map_fd, &stable_key,
594 						    value));
595 		}
596 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
597 					    BPF_NOEXIST));
598 	}
599 
600 	for (; key <= tgt_free * 3; key++) {
601 		assert(!bpf_map_update_elem(lru_map_fd, &key, value,
602 					    BPF_NOEXIST));
603 		assert(!bpf_map_update_elem(expected_map_fd, &key, value,
604 					    BPF_NOEXIST));
605 	}
606 
607 	assert(map_equal(lru_map_fd, expected_map_fd));
608 
609 	close(expected_map_fd);
610 	close(lru_map_fd);
611 
612 	printf("Pass\n");
613 }
614 
615 int main(int argc, char **argv)
616 {
617 	int map_types[] = {BPF_MAP_TYPE_LRU_HASH,
618 			     BPF_MAP_TYPE_LRU_PERCPU_HASH};
619 	int map_flags[] = {0, BPF_F_NO_COMMON_LRU};
620 	int t, f;
621 
622 	setbuf(stdout, NULL);
623 
624 	nr_cpus = bpf_num_possible_cpus();
625 	assert(nr_cpus != -1);
626 	printf("nr_cpus:%d\n\n", nr_cpus);
627 
628 	for (f = 0; f < sizeof(map_flags) / sizeof(*map_flags); f++) {
629 		unsigned int tgt_free = (map_flags[f] & BPF_F_NO_COMMON_LRU) ?
630 			PERCPU_FREE_TARGET : LOCAL_FREE_TARGET;
631 
632 		for (t = 0; t < sizeof(map_types) / sizeof(*map_types); t++) {
633 			test_lru_sanity0(map_types[t], map_flags[f]);
634 			test_lru_sanity1(map_types[t], map_flags[f], tgt_free);
635 			test_lru_sanity2(map_types[t], map_flags[f], tgt_free);
636 			test_lru_sanity3(map_types[t], map_flags[f], tgt_free);
637 			test_lru_sanity4(map_types[t], map_flags[f], tgt_free);
638 			test_lru_sanity5(map_types[t], map_flags[f]);
639 			test_lru_sanity6(map_types[t], map_flags[f], tgt_free);
640 
641 			printf("\n");
642 		}
643 	}
644 
645 	return 0;
646 }
647