fix readdirplus when filler is called with zero offset (#896)
authorfarlongsignal <141166749+farlongsignal@users.noreply.github.com>
Sun, 24 Mar 2024 20:54:19 +0000 (21:54 +0100)
committerGitHub <noreply@github.com>
Sun, 24 Mar 2024 20:54:19 +0000 (21:54 +0100)
fixes #235

In fill_dir_plus(), there's a lookup for caching dirent attributes.
However, when offset = 0 the cache metadata from the lookup is lost
as only the entry attributes are passed when added to the list. Kernel
doesn't cache the attributes since .ino = 0.

This change moves the entry lookup to happen just before the relevant
fuse_add_direntry_plus() calls

lib/fuse.c

index 2a88918a4b5da50946a8ecfc84a7cfca1f64f541..f619c0003a9530c549084a1481e211d325893b3c 100644 (file)
@@ -3541,14 +3541,6 @@ static int fill_dir_plus(void *dh_, const char *name, const struct stat *statp,
 
        if (statp && (flags & FUSE_FILL_DIR_PLUS)) {
                e.attr = *statp;
-
-               if (!is_dot_or_dotdot(name)) {
-                       res = do_lookup(f, dh->nodeid, name, &e);
-                       if (res) {
-                               dh->error = res;
-                               return 1;
-                       }
-               }
        } else {
                e.attr.st_ino = FUSE_UNKNOWN_INO;
                if (statp) {
@@ -3577,6 +3569,16 @@ static int fill_dir_plus(void *dh_, const char *name, const struct stat *statp,
                if (extend_contents(dh, dh->needlen) == -1)
                        return 1;
 
+               if (statp && (flags & FUSE_FILL_DIR_PLUS)) {
+                       if (!is_dot_or_dotdot(name)) {
+                               res = do_lookup(f, dh->nodeid, name, &e);
+                               if (res) {
+                                       dh->error = res;
+                                       return 1;
+                               }
+                       }
+               }
+
                newlen = dh->len +
                        fuse_add_direntry_plus(dh->req, dh->contents + dh->len,
                                               dh->needlen - dh->len, name,
@@ -3649,6 +3651,7 @@ static int readdir_fill_from_list(fuse_req_t req, struct fuse_dh *dh,
 {
        off_t pos;
        struct fuse_direntry *de = dh->first;
+       int res;
 
        dh->len = 0;
 
@@ -3673,6 +3676,16 @@ static int readdir_fill_from_list(fuse_req_t req, struct fuse_dh *dh,
                                .ino = 0,
                                .attr = de->stat,
                        };
+
+                       if (!is_dot_or_dotdot(de->name)) {
+                               res = do_lookup(dh->fuse, dh->nodeid,
+                                               de->name, &e);
+                               if (res) {
+                                       dh->error = res;
+                                       return 1;
+                               }
+                       }
+
                        thislen = fuse_add_direntry_plus(req, p, rem,
                                                         de->name, &e, pos);
                } else {