Fix readdir() bug when a non-zero offset is specified in filler
authorRostislav Skudnov <rostislav@tuxera.com>
Wed, 25 Jul 2018 10:36:38 +0000 (10:36 +0000)
committerNikolaus Rath <Nikolaus@rath.org>
Wed, 25 Jul 2018 11:28:59 +0000 (12:28 +0100)
commit06fc40705f23cb7e9af4df2febae8e6889b1a95d
tree4393aa9a9da4c5fe6906950b7f6a422c1ab6295c
parent6172ece7177fec28f3a771bbd445d5e9b9d5acd0
Fix readdir() bug when a non-zero offset is specified in filler

The bug occurs when a filesystem client reads a directory until the end,
seeks using seekdir() to some valid non-zero position and calls
readdir(). A valid 'struct dirent *' is expected, but NULL is returned
instead. Pseudocode demonstrating the bug:

DIR *dp = opendir("some_dir");
struct dirent *de = readdir(dp);

/* Get offset of the second entry */
long offset = telldir(dp);

/* Read directory until the end */
while (de)
de = readdir(de);

seekdir(dp, offset);
de = readdir(dp);
/* de must contain the second entry, but NULL is returned instead */

The reason of the bug is that when the end of directory is reached, the
kernel calls FUSE_READDIR op with an offset at the end of directory, so
the filesystem's .readdir callback never calls the filler function, and
we end up with dh->filled set to 1. After seekdir(), FUSE_READDIR is
called again with a new offset, but this time the filesystem's .readdir
callback is never called, and an empty reply is returned.

Fix by setting dh->filled to 1 only when zero offsets are given to
filler function.

This commit is backported from the following commit in 'master' branch:

commit 5f125c5e6be24c8d216a4d3c623dc73d742c8c86
Author: Rostislav <rostislav@users.noreply.github.com>
Date:   Sat Jul 21 12:57:09 2018 +0300

    Fix readdir() bug when a non-zero offset is specified in filler (#269)
ChangeLog
lib/fuse.c
test/test.c