2016-03-18 04:21:45 +07:00
|
|
|
|
2016-12-15 06:07:56 +07:00
|
|
|
CFLAGS += -I. -I../../include -g -O2 -Wall -D_LGPL_SOURCE
|
2016-03-18 04:21:45 +07:00
|
|
|
LDFLAGS += -lpthread -lurcu
|
|
|
|
TARGETS = main
|
2016-12-29 10:53:46 +07:00
|
|
|
OFILES = main.o radix-tree.o linux.o test.o tag_check.o find_bit.o \
|
radix-tree tests: add iteration test
There are four cases I can see where we could end up with a NULL 'slot' in
radix_tree_next_slot(). This unit test exercises all four of them, making
sure that if in the future we have an unsafe path through
radix_tree_next_slot(), we'll catch it.
Here are details on the four cases:
1) radix_tree_iter_retry() via a non-tagged iteration like
radix_tree_for_each_slot(). In this case we currently aren't seeing a bug
because radix_tree_iter_retry() sets
iter->next_index = iter->index;
which means that in in the else case in radix_tree_next_slot(), 'count' is
zero, so we skip over the while() loop and effectively just return NULL
without ever dereferencing 'slot'.
2) radix_tree_iter_retry() via tagged iteration like
radix_tree_for_each_tagged(). This case was giving us NULL pointer
dereferences in testing, and was fixed with this commit:
commit 3cb9185c6730 ("radix-tree: fix radix_tree_iter_retry() for tagged
iterators.")
This fix doesn't explicitly check for 'slot' being NULL, though, it works
around the NULL pointer dereference by instead zeroing iter->tags in
radix_tree_iter_retry(), which makes us bail out of the if() case in
radix_tree_next_slot() before we dereference 'slot'.
3) radix_tree_iter_next() via via a non-tagged iteration like
radix_tree_for_each_slot(). This currently happens in shmem_tag_pins()
and shmem_partial_swap_usage().
As with non-tagged iteration, 'count' in the else case of
radix_tree_next_slot() is zero, so we skip over the while() loop and
effectively just return NULL without ever dereferencing 'slot'.
4) radix_tree_iter_next() via tagged iteration like
radix_tree_for_each_tagged(). This happens in shmem_wait_for_pins().
radix_tree_iter_next() zeros out iter->tags, so we end up exiting
radix_tree_next_slot() here:
if (flags & RADIX_TREE_ITER_TAGGED) {
void *canon = slot;
iter->tags >>= 1;
if (unlikely(!iter->tags))
return NULL;
Link: http://lkml.kernel.org/r/20160815194237.25967-3-ross.zwisler@linux.intel.com
Signed-off-by: Ross Zwisler <ross.zwisler@linux.intel.com>
Cc: Konstantin Khlebnikov <koct9i@gmail.com>
Cc: Andrey Ryabinin <aryabinin@virtuozzo.com>
Cc: Dmitry Vyukov <dvyukov@google.com>
Cc: Shuah Khan <shuahkh@osg.samsung.com>
Signed-off-by: Andrew Morton <akpm@linux-foundation.org>
Signed-off-by: Linus Torvalds <torvalds@linux-foundation.org>
2016-10-12 03:51:21 +07:00
|
|
|
regression1.o regression2.o regression3.o multiorder.o \
|
2016-12-15 06:08:14 +07:00
|
|
|
iteration_check.o benchmark.o
|
|
|
|
|
|
|
|
ifdef BENCHMARK
|
|
|
|
CFLAGS += -DBENCHMARK=1
|
|
|
|
endif
|
2016-03-18 04:21:45 +07:00
|
|
|
|
|
|
|
targets: $(TARGETS)
|
|
|
|
|
|
|
|
main: $(OFILES)
|
|
|
|
$(CC) $(CFLAGS) $(LDFLAGS) $(OFILES) -o main
|
|
|
|
|
|
|
|
clean:
|
|
|
|
$(RM) -f $(TARGETS) *.o radix-tree.c
|
|
|
|
|
2016-12-29 10:53:46 +07:00
|
|
|
vpath %.c ../../lib
|
2016-12-15 06:08:29 +07:00
|
|
|
|
|
|
|
$(OFILES): *.h */*.h \
|
|
|
|
../../include/linux/*.h \
|
2016-12-19 23:09:34 +07:00
|
|
|
../../include/asm/*.h \
|
2016-12-15 06:08:29 +07:00
|
|
|
../../../include/linux/radix-tree.h
|
2016-03-18 04:21:45 +07:00
|
|
|
|
|
|
|
radix-tree.c: ../../../lib/radix-tree.c
|
|
|
|
sed -e 's/^static //' -e 's/__always_inline //' -e 's/inline //' < $< > $@
|