Add an AT_NO_AUTOMOUNT flag to suppress terminal automounting of directories
with follow_link semantics.  This can be used by fstatat()/xstat() users to
permit the gathering of attributes on an automount point and also prevent
mass-automounting of a directory of automount points by ls.

Signed-off-by: David Howells <dhowe...@redhat.com>
---

 fs/namei.c            |   15 ++++++++++-----
 fs/stat.c             |    4 +++-
 include/linux/fcntl.h |    1 +
 include/linux/namei.h |    2 ++
 4 files changed, 16 insertions(+), 6 deletions(-)

diff --git a/fs/namei.c b/fs/namei.c
index 86068a2..056427e 100644
--- a/fs/namei.c
+++ b/fs/namei.c
@@ -654,7 +654,8 @@ static int follow_automount(struct path *path, int res)
 /* no need for dcache_lock, as serialization is taken care in
  * namespace.c
  */
-static int __follow_mount(struct path *path, unsigned nofollow)
+static int __follow_mount(struct path *path, unsigned nofollow,
+                         struct nameidata *nd)
 {
        struct vfsmount *mounted;
        int ret, res = 0;
@@ -674,8 +675,12 @@ static int __follow_mount(struct path *path, unsigned 
nofollow)
                }
                if (!d_automount_point(path->dentry))
                        break;
-               if (nofollow)
-                       return -ELOOP;
+               if (!(nd->flags & LOOKUP_CONTINUE)) {
+                       if (nofollow)
+                               return -ELOOP;
+                       if (nd->flags & LOOKUP_NO_AUTOMOUNT)
+                               break;
+               }
                ret = follow_automount(path, res);
                if (ret < 0)
                        return ret;
@@ -769,7 +774,7 @@ static int do_lookup(struct nameidata *nd, struct qstr 
*name,
 done:
        path->mnt = mnt;
        path->dentry = dentry;
-       ret = __follow_mount(path, 0);
+       ret = __follow_mount(path, 0, nd);
        if (unlikely(ret < 0))
                path_put(path);
        return ret;
@@ -1762,7 +1767,7 @@ static struct file *do_last(struct nameidata *nd, struct 
path *path,
        if (open_flag & O_EXCL)
                goto exit_dput;
 
-       error = __follow_mount(path, open_flag & O_NOFOLLOW);
+       error = __follow_mount(path, open_flag & O_NOFOLLOW, nd);
        if (error < 0)
                goto exit_dput;
 
diff --git a/fs/stat.c b/fs/stat.c
index bb0f538..3f2ab5f 100644
--- a/fs/stat.c
+++ b/fs/stat.c
@@ -182,11 +182,13 @@ int vfs_xstat(int dfd, const char __user *filename, int 
flags,
        struct path path;
        int error, lookup_flags;
 
-       if (flags & ~(AT_SYMLINK_NOFOLLOW | KSTAT_QUERY_FLAGS))
+       if (flags & ~(AT_SYMLINK_NOFOLLOW | AT_NO_AUTOMOUNT |
+                     KSTAT_QUERY_FLAGS))
                return -EINVAL;
 
        stat->query_flags = flags & KSTAT_QUERY_FLAGS;
        lookup_flags = (flags & AT_SYMLINK_NOFOLLOW) ? 0 : LOOKUP_FOLLOW;
+       lookup_flags |= (flags & AT_NO_AUTOMOUNT) ? LOOKUP_NO_AUTOMOUNT : 0;
 
        error = user_path_at(dfd, filename, lookup_flags, &path);
        if (!error) {
diff --git a/include/linux/fcntl.h b/include/linux/fcntl.h
index bcf8083..768b0fd 100644
--- a/include/linux/fcntl.h
+++ b/include/linux/fcntl.h
@@ -46,6 +46,7 @@
                                            unlinking file.  */
 #define AT_SYMLINK_FOLLOW      0x400   /* Follow symbolic links.  */
 #define AT_FORCE_ATTR_SYNC     0x800   /* Force the attributes to be sync'd 
with the server */
+#define AT_NO_AUTOMOUNT                0x1000  /* Suppress terminal automount 
traversal */
 
 #ifdef __KERNEL__
 
diff --git a/include/linux/namei.h b/include/linux/namei.h
index 05b441d..1e1febf 100644
--- a/include/linux/namei.h
+++ b/include/linux/namei.h
@@ -43,12 +43,14 @@ enum {LAST_NORM, LAST_ROOT, LAST_DOT, LAST_DOTDOT, 
LAST_BIND};
  *  - internal "there are more path components" flag
  *  - locked when lookup done with dcache_lock held
  *  - dentry cache is untrusted; force a real lookup
+ *  - suppress terminal automount
  */
 #define LOOKUP_FOLLOW           1
 #define LOOKUP_DIRECTORY        2
 #define LOOKUP_CONTINUE                 4
 #define LOOKUP_PARENT          16
 #define LOOKUP_REVAL           64
+#define LOOKUP_NO_AUTOMOUNT    128
 /*
  * Intent data
  */

--
To unsubscribe from this list: send the line "unsubscribe linux-cifs" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

Reply via email to