2 * Copyright (c) 2016 Facebook
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.
19 #include <sys/resource.h>
24 #define LOCAL_FREE_TARGET (128)
25 #define PERCPU_FREE_TARGET (16)
29 static int create_map(int map_type
, int map_flags
, unsigned int size
)
33 map_fd
= bpf_create_map(map_type
, sizeof(unsigned long long),
34 sizeof(unsigned long long), size
, map_flags
);
37 perror("bpf_create_map");
42 static int map_subset(int map0
, int map1
)
44 unsigned long long next_key
= 0;
45 unsigned long long value0
[nr_cpus
], value1
[nr_cpus
];
48 while (!bpf_map_get_next_key(map1
, &next_key
, &next_key
)) {
49 assert(!bpf_map_lookup_elem(map1
, &next_key
, value1
));
50 ret
= bpf_map_lookup_elem(map0
, &next_key
, value0
);
52 printf("key:%llu not found from map. %s(%d)\n",
53 next_key
, strerror(errno
), errno
);
56 if (value0
[0] != value1
[0]) {
57 printf("key:%llu value0:%llu != value1:%llu\n",
58 next_key
, value0
[0], value1
[0]);
65 static int map_equal(int lru_map
, int expected
)
67 return map_subset(lru_map
, expected
) && map_subset(expected
, lru_map
);
70 static int sched_next_online(int pid
, int *next_to_try
)
73 int next
= *next_to_try
;
76 while (next
< nr_cpus
) {
78 CPU_SET(next
++, &cpuset
);
79 if (!sched_setaffinity(pid
, sizeof(cpuset
), &cpuset
)) {
89 /* Size of the LRU amp is 2
94 * => Key=2 will be removed by LRU
95 * Iterate map. Only found key=1 and key=3
97 static void test_lru_sanity0(int map_type
, int map_flags
)
99 unsigned long long key
, value
[nr_cpus
];
100 int lru_map_fd
, expected_map_fd
;
103 printf("%s (map_type:%d map_flags:0x%X): ", __func__
, map_type
,
106 assert(sched_next_online(0, &next_cpu
) != -1);
108 if (map_flags
& BPF_F_NO_COMMON_LRU
)
109 lru_map_fd
= create_map(map_type
, map_flags
, 2 * nr_cpus
);
111 lru_map_fd
= create_map(map_type
, map_flags
, 2);
112 assert(lru_map_fd
!= -1);
114 expected_map_fd
= create_map(BPF_MAP_TYPE_HASH
, 0, 2);
115 assert(expected_map_fd
!= -1);
119 /* insert key=1 element */
122 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
, BPF_NOEXIST
));
123 assert(!bpf_map_update_elem(expected_map_fd
, &key
, value
,
126 /* BPF_NOEXIST means: add new element if it doesn't exist */
127 assert(bpf_map_update_elem(lru_map_fd
, &key
, value
, BPF_NOEXIST
) == -1
128 /* key=1 already exists */
131 assert(bpf_map_update_elem(lru_map_fd
, &key
, value
, -1) == -1 &&
134 /* insert key=2 element */
136 /* check that key=2 is not found */
138 assert(bpf_map_lookup_elem(lru_map_fd
, &key
, value
) == -1 &&
141 /* BPF_EXIST means: update existing element */
142 assert(bpf_map_update_elem(lru_map_fd
, &key
, value
, BPF_EXIST
) == -1 &&
143 /* key=2 is not there */
146 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
, BPF_NOEXIST
));
148 /* insert key=3 element */
150 /* check that key=3 is not found */
152 assert(bpf_map_lookup_elem(lru_map_fd
, &key
, value
) == -1 &&
155 /* check that key=1 can be found and mark the ref bit to
156 * stop LRU from removing key=1
159 assert(!bpf_map_lookup_elem(lru_map_fd
, &key
, value
));
160 assert(value
[0] == 1234);
163 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
, BPF_NOEXIST
));
164 assert(!bpf_map_update_elem(expected_map_fd
, &key
, value
,
167 /* key=2 has been removed from the LRU */
169 assert(bpf_map_lookup_elem(lru_map_fd
, &key
, value
) == -1);
171 assert(map_equal(lru_map_fd
, expected_map_fd
));
173 close(expected_map_fd
);
179 /* Size of the LRU map is 1.5*tgt_free
180 * Insert 1 to tgt_free (+tgt_free keys)
181 * Lookup 1 to tgt_free/2
182 * Insert 1+tgt_free to 2*tgt_free (+tgt_free keys)
183 * => 1+tgt_free/2 to LOCALFREE_TARGET will be removed by LRU
185 static void test_lru_sanity1(int map_type
, int map_flags
, unsigned int tgt_free
)
187 unsigned long long key
, end_key
, value
[nr_cpus
];
188 int lru_map_fd
, expected_map_fd
;
189 unsigned int batch_size
;
190 unsigned int map_size
;
193 if (map_flags
& BPF_F_NO_COMMON_LRU
)
194 /* Ther percpu lru list (i.e each cpu has its own LRU
195 * list) does not have a local free list. Hence,
196 * it will only free old nodes till there is no free
197 * from the LRU list. Hence, this test does not apply
198 * to BPF_F_NO_COMMON_LRU
202 printf("%s (map_type:%d map_flags:0x%X): ", __func__
, map_type
,
205 assert(sched_next_online(0, &next_cpu
) != -1);
207 batch_size
= tgt_free
/ 2;
208 assert(batch_size
* 2 == tgt_free
);
210 map_size
= tgt_free
+ batch_size
;
211 lru_map_fd
= create_map(map_type
, map_flags
, map_size
);
212 assert(lru_map_fd
!= -1);
214 expected_map_fd
= create_map(BPF_MAP_TYPE_HASH
, 0, map_size
);
215 assert(expected_map_fd
!= -1);
219 /* Insert 1 to tgt_free (+tgt_free keys) */
220 end_key
= 1 + tgt_free
;
221 for (key
= 1; key
< end_key
; key
++)
222 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
225 /* Lookup 1 to tgt_free/2 */
226 end_key
= 1 + batch_size
;
227 for (key
= 1; key
< end_key
; key
++) {
228 assert(!bpf_map_lookup_elem(lru_map_fd
, &key
, value
));
229 assert(!bpf_map_update_elem(expected_map_fd
, &key
, value
,
233 /* Insert 1+tgt_free to 2*tgt_free
234 * => 1+tgt_free/2 to LOCALFREE_TARGET will be
238 end_key
= key
+ tgt_free
;
239 for (; key
< end_key
; key
++) {
240 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
242 assert(!bpf_map_update_elem(expected_map_fd
, &key
, value
,
246 assert(map_equal(lru_map_fd
, expected_map_fd
));
248 close(expected_map_fd
);
254 /* Size of the LRU map 1.5 * tgt_free
255 * Insert 1 to tgt_free (+tgt_free keys)
256 * Update 1 to tgt_free/2
257 * => The original 1 to tgt_free/2 will be removed due to
258 * the LRU shrink process
259 * Re-insert 1 to tgt_free/2 again and do a lookup immeidately
260 * Insert 1+tgt_free to tgt_free*3/2
261 * Insert 1+tgt_free*3/2 to tgt_free*5/2
262 * => Key 1+tgt_free to tgt_free*3/2
263 * will be removed from LRU because it has never
264 * been lookup and ref bit is not set
266 static void test_lru_sanity2(int map_type
, int map_flags
, unsigned int tgt_free
)
268 unsigned long long key
, value
[nr_cpus
];
269 unsigned long long end_key
;
270 int lru_map_fd
, expected_map_fd
;
271 unsigned int batch_size
;
272 unsigned int map_size
;
275 if (map_flags
& BPF_F_NO_COMMON_LRU
)
276 /* Ther percpu lru list (i.e each cpu has its own LRU
277 * list) does not have a local free list. Hence,
278 * it will only free old nodes till there is no free
279 * from the LRU list. Hence, this test does not apply
280 * to BPF_F_NO_COMMON_LRU
284 printf("%s (map_type:%d map_flags:0x%X): ", __func__
, map_type
,
287 assert(sched_next_online(0, &next_cpu
) != -1);
289 batch_size
= tgt_free
/ 2;
290 assert(batch_size
* 2 == tgt_free
);
292 map_size
= tgt_free
+ batch_size
;
293 if (map_flags
& BPF_F_NO_COMMON_LRU
)
294 lru_map_fd
= create_map(map_type
, map_flags
,
297 lru_map_fd
= create_map(map_type
, map_flags
, map_size
);
298 assert(lru_map_fd
!= -1);
300 expected_map_fd
= create_map(BPF_MAP_TYPE_HASH
, 0, map_size
);
301 assert(expected_map_fd
!= -1);
305 /* Insert 1 to tgt_free (+tgt_free keys) */
306 end_key
= 1 + tgt_free
;
307 for (key
= 1; key
< end_key
; key
++)
308 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
311 /* Any bpf_map_update_elem will require to acquire a new node
314 * The local list is running out of free nodes.
315 * It gets from the global LRU list which tries to
316 * shrink the inactive list to get tgt_free
317 * number of free nodes.
319 * Hence, the oldest key 1 to tgt_free/2
320 * are removed from the LRU list.
323 if (map_type
== BPF_MAP_TYPE_LRU_PERCPU_HASH
) {
324 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
326 assert(!bpf_map_delete_elem(lru_map_fd
, &key
));
328 assert(bpf_map_update_elem(lru_map_fd
, &key
, value
,
332 /* Re-insert 1 to tgt_free/2 again and do a lookup
335 end_key
= 1 + batch_size
;
337 for (key
= 1; key
< end_key
; key
++) {
338 assert(bpf_map_lookup_elem(lru_map_fd
, &key
, value
));
339 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
341 assert(!bpf_map_lookup_elem(lru_map_fd
, &key
, value
));
342 assert(value
[0] == 4321);
343 assert(!bpf_map_update_elem(expected_map_fd
, &key
, value
,
349 /* Insert 1+tgt_free to tgt_free*3/2 */
350 end_key
= 1 + tgt_free
+ batch_size
;
351 for (key
= 1 + tgt_free
; key
< end_key
; key
++)
352 /* These newly added but not referenced keys will be
353 * gone during the next LRU shrink.
355 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
358 /* Insert 1+tgt_free*3/2 to tgt_free*5/2 */
359 end_key
= key
+ tgt_free
;
360 for (; key
< end_key
; key
++) {
361 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
363 assert(!bpf_map_update_elem(expected_map_fd
, &key
, value
,
367 assert(map_equal(lru_map_fd
, expected_map_fd
));
369 close(expected_map_fd
);
375 /* Size of the LRU map is 2*tgt_free
376 * It is to test the active/inactive list rotation
377 * Insert 1 to 2*tgt_free (+2*tgt_free keys)
378 * Lookup key 1 to tgt_free*3/2
379 * Add 1+2*tgt_free to tgt_free*5/2 (+tgt_free/2 keys)
380 * => key 1+tgt_free*3/2 to 2*tgt_free are removed from LRU
382 static void test_lru_sanity3(int map_type
, int map_flags
, unsigned int tgt_free
)
384 unsigned long long key
, end_key
, value
[nr_cpus
];
385 int lru_map_fd
, expected_map_fd
;
386 unsigned int batch_size
;
387 unsigned int map_size
;
390 printf("%s (map_type:%d map_flags:0x%X): ", __func__
, map_type
,
393 assert(sched_next_online(0, &next_cpu
) != -1);
395 batch_size
= tgt_free
/ 2;
396 assert(batch_size
* 2 == tgt_free
);
398 map_size
= tgt_free
* 2;
399 if (map_flags
& BPF_F_NO_COMMON_LRU
)
400 lru_map_fd
= create_map(map_type
, map_flags
,
403 lru_map_fd
= create_map(map_type
, map_flags
, map_size
);
404 assert(lru_map_fd
!= -1);
406 expected_map_fd
= create_map(BPF_MAP_TYPE_HASH
, 0, map_size
);
407 assert(expected_map_fd
!= -1);
411 /* Insert 1 to 2*tgt_free (+2*tgt_free keys) */
412 end_key
= 1 + (2 * tgt_free
);
413 for (key
= 1; key
< end_key
; key
++)
414 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
417 /* Lookup key 1 to tgt_free*3/2 */
418 end_key
= tgt_free
+ batch_size
;
419 for (key
= 1; key
< end_key
; key
++) {
420 assert(!bpf_map_lookup_elem(lru_map_fd
, &key
, value
));
421 assert(!bpf_map_update_elem(expected_map_fd
, &key
, value
,
425 /* Add 1+2*tgt_free to tgt_free*5/2
428 key
= 2 * tgt_free
+ 1;
429 end_key
= key
+ batch_size
;
430 for (; key
< end_key
; key
++) {
431 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
433 assert(!bpf_map_update_elem(expected_map_fd
, &key
, value
,
437 assert(map_equal(lru_map_fd
, expected_map_fd
));
439 close(expected_map_fd
);
446 static void test_lru_sanity4(int map_type
, int map_flags
, unsigned int tgt_free
)
448 int lru_map_fd
, expected_map_fd
;
449 unsigned long long key
, value
[nr_cpus
];
450 unsigned long long end_key
;
453 printf("%s (map_type:%d map_flags:0x%X): ", __func__
, map_type
,
456 assert(sched_next_online(0, &next_cpu
) != -1);
458 if (map_flags
& BPF_F_NO_COMMON_LRU
)
459 lru_map_fd
= create_map(map_type
, map_flags
,
460 3 * tgt_free
* nr_cpus
);
462 lru_map_fd
= create_map(map_type
, map_flags
, 3 * tgt_free
);
463 assert(lru_map_fd
!= -1);
465 expected_map_fd
= create_map(BPF_MAP_TYPE_HASH
, 0,
467 assert(expected_map_fd
!= -1);
471 for (key
= 1; key
<= 2 * tgt_free
; key
++)
472 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
476 assert(bpf_map_update_elem(lru_map_fd
, &key
, value
, BPF_NOEXIST
));
478 for (key
= 1; key
<= tgt_free
; key
++) {
479 assert(!bpf_map_lookup_elem(lru_map_fd
, &key
, value
));
480 assert(!bpf_map_update_elem(expected_map_fd
, &key
, value
,
484 for (; key
<= 2 * tgt_free
; key
++) {
485 assert(!bpf_map_delete_elem(lru_map_fd
, &key
));
486 assert(bpf_map_delete_elem(lru_map_fd
, &key
));
489 end_key
= key
+ 2 * tgt_free
;
490 for (; key
< end_key
; key
++) {
491 assert(!bpf_map_update_elem(lru_map_fd
, &key
, value
,
493 assert(!bpf_map_update_elem(expected_map_fd
, &key
, value
,
497 assert(map_equal(lru_map_fd
, expected_map_fd
));
499 close(expected_map_fd
);
505 static void do_test_lru_sanity5(unsigned long long last_key
, int map_fd
)
507 unsigned long long key
, value
[nr_cpus
];
509 /* Ensure the last key inserted by previous CPU can be found */
510 assert(!bpf_map_lookup_elem(map_fd
, &last_key
, value
));
515 assert(!bpf_map_update_elem(map_fd
, &key
, value
, BPF_NOEXIST
));
516 assert(!bpf_map_lookup_elem(map_fd
, &key
, value
));
518 /* Cannot find the last key because it was removed by LRU */
519 assert(bpf_map_lookup_elem(map_fd
, &last_key
, value
));
522 /* Test map with only one element */
523 static void test_lru_sanity5(int map_type
, int map_flags
)
525 unsigned long long key
, value
[nr_cpus
];
529 if (map_flags
& BPF_F_NO_COMMON_LRU
)
532 printf("%s (map_type:%d map_flags:0x%X): ", __func__
, map_type
,
535 map_fd
= create_map(map_type
, map_flags
, 1);
536 assert(map_fd
!= -1);
540 assert(!bpf_map_update_elem(map_fd
, &key
, value
, BPF_NOEXIST
));
542 while (sched_next_online(0, &next_cpu
) != -1) {
547 do_test_lru_sanity5(key
, map_fd
);
549 } else if (pid
== -1) {
550 printf("couldn't spawn process to test key:%llu\n",
556 assert(waitpid(pid
, &status
, 0) == pid
);
563 /* At least one key should be tested */
569 int main(int argc
, char **argv
)
571 struct rlimit r
= {RLIM_INFINITY
, RLIM_INFINITY
};
572 int map_types
[] = {BPF_MAP_TYPE_LRU_HASH
,
573 BPF_MAP_TYPE_LRU_PERCPU_HASH
};
574 int map_flags
[] = {0, BPF_F_NO_COMMON_LRU
};
577 setbuf(stdout
, NULL
);
579 assert(!setrlimit(RLIMIT_MEMLOCK
, &r
));
581 nr_cpus
= bpf_num_possible_cpus();
582 assert(nr_cpus
!= -1);
583 printf("nr_cpus:%d\n\n", nr_cpus
);
585 for (f
= 0; f
< sizeof(map_flags
) / sizeof(*map_flags
); f
++) {
586 unsigned int tgt_free
= (map_flags
[f
] & BPF_F_NO_COMMON_LRU
) ?
587 PERCPU_FREE_TARGET
: LOCAL_FREE_TARGET
;
589 for (t
= 0; t
< sizeof(map_types
) / sizeof(*map_types
); t
++) {
590 test_lru_sanity0(map_types
[t
], map_flags
[f
]);
591 test_lru_sanity1(map_types
[t
], map_flags
[f
], tgt_free
);
592 test_lru_sanity2(map_types
[t
], map_flags
[f
], tgt_free
);
593 test_lru_sanity3(map_types
[t
], map_flags
[f
], tgt_free
);
594 test_lru_sanity4(map_types
[t
], map_flags
[f
], tgt_free
);
595 test_lru_sanity5(map_types
[t
], map_flags
[f
]);