]> www.infradead.org Git - users/hch/xfs.git/commitdiff
vfs: sanity check the length passed to inode_set_cached_link()
authorMateusz Guzik <mjguzik@gmail.com>
Tue, 4 Feb 2025 21:32:07 +0000 (22:32 +0100)
committerChristian Brauner <brauner@kernel.org>
Fri, 7 Feb 2025 09:29:59 +0000 (10:29 +0100)
This costs a strlen() call when instatianating a symlink.

Preferably it would be hidden behind VFS_WARN_ON (or compatible), but
there is no such facility at the moment. With the facility in place the
call can be patched out in production kernels.

In the meantime, since the cost is being paid unconditionally, use the
result to a fixup the bad caller.

This is not expected to persist in the long run (tm).

Sample splat:
bad length passed for symlink [/tmp/syz-imagegen43743633/file0/file0] (got 131109, expected 37)
[rest of WARN blurp goes here]

Signed-off-by: Mateusz Guzik <mjguzik@gmail.com>
Link: https://lore.kernel.org/r/20250204213207.337980-1-mjguzik@gmail.com
Signed-off-by: Christian Brauner <brauner@kernel.org>
include/linux/fs.h

index 7620547432a84a69d480295177bee6e23524cec3..2c3b2f8a621f76a08706df94c2b90540f7369568 100644 (file)
@@ -790,6 +790,19 @@ struct inode {
 
 static inline void inode_set_cached_link(struct inode *inode, char *link, int linklen)
 {
+       int testlen;
+
+       /*
+        * TODO: patch it into a debug-only check if relevant macros show up.
+        * In the meantime, since we are suffering strlen even on production kernels
+        * to find the right length, do a fixup if the wrong value got passed.
+        */
+       testlen = strlen(link);
+       if (testlen != linklen) {
+               WARN_ONCE(1, "bad length passed for symlink [%s] (got %d, expected %d)",
+                         link, linklen, testlen);
+               linklen = testlen;
+       }
        inode->i_link = link;
        inode->i_linklen = linklen;
        inode->i_opflags |= IOP_CACHED_LINK;