lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <20200214185523.GD1734@sasha-vm>
Date:   Fri, 14 Feb 2020 13:55:23 -0500
From:   Sasha Levin <sashal@...nel.org>
To:     Ilya Dryomov <idryomov@...il.com>
Cc:     LKML <linux-kernel@...r.kernel.org>, stable@...r.kernel.org,
        Xiubo Li <xiubli@...hat.com>, Jeff Layton <jlayton@...nel.org>,
        Ceph Development <ceph-devel@...r.kernel.org>
Subject: Re: [PATCH AUTOSEL 5.5 487/542] ceph: remove the extra slashes in
 the server path

On Fri, Feb 14, 2020 at 05:13:21PM +0100, Ilya Dryomov wrote:
>On Fri, Feb 14, 2020 at 4:59 PM Sasha Levin <sashal@...nel.org> wrote:
>>
>> From: Xiubo Li <xiubli@...hat.com>
>>
>> [ Upstream commit 4fbc0c711b2464ee1551850b85002faae0b775d5 ]
>>
>> It's possible to pass the mount helper a server path that has more
>> than one contiguous slash character. For example:
>>
>>   $ mount -t ceph 192.168.195.165:40176:/// /mnt/cephfs/
>>
>> In the MDS server side the extra slashes of the server path will be
>> treated as snap dir, and then we can get the following debug logs:
>>
>>   ceph:  mount opening path //
>>   ceph:  open_root_inode opening '//'
>>   ceph:  fill_trace 0000000059b8a3bc is_dentry 0 is_target 1
>>   ceph:  alloc_inode 00000000dc4ca00b
>>   ceph:  get_inode created new inode 00000000dc4ca00b 1.ffffffffffffffff ino 1
>>   ceph:  get_inode on 1=1.ffffffffffffffff got 00000000dc4ca00b
>>
>> And then when creating any new file or directory under the mount
>> point, we can hit the following BUG_ON in ceph_fill_trace():
>>
>>   BUG_ON(ceph_snap(dir) != dvino.snap);
>>
>> Have the client ignore the extra slashes in the server path when
>> mounting. This will also canonicalize the path, so that identical mounts
>> can be consilidated.
>>
>> 1) "//mydir1///mydir//"
>> 2) "/mydir1/mydir"
>> 3) "/mydir1/mydir/"
>>
>> Regardless of the internal treatment of these paths, the kernel still
>> stores the original string including the leading '/' for presentation
>> to userland.
>>
>> URL: https://tracker.ceph.com/issues/42771
>> Signed-off-by: Xiubo Li <xiubli@...hat.com>
>> Reviewed-by: Jeff Layton <jlayton@...nel.org>
>> Signed-off-by: Ilya Dryomov <idryomov@...il.com>
>> Signed-off-by: Sasha Levin <sashal@...nel.org>
>> ---
>>  fs/ceph/super.c | 122 ++++++++++++++++++++++++++++++++++++++++--------
>>  1 file changed, 102 insertions(+), 20 deletions(-)
>>
>> diff --git a/fs/ceph/super.c b/fs/ceph/super.c
>> index 430dcf329723a..112927dbd2f20 100644
>> --- a/fs/ceph/super.c
>> +++ b/fs/ceph/super.c
>> @@ -107,7 +107,6 @@ static int ceph_statfs(struct dentry *dentry, struct kstatfs *buf)
>>         return 0;
>>  }
>>
>> -
>>  static int ceph_sync_fs(struct super_block *sb, int wait)
>>  {
>>         struct ceph_fs_client *fsc = ceph_sb_to_client(sb);
>> @@ -211,7 +210,6 @@ struct ceph_parse_opts_ctx {
>>
>>  /*
>>   * Parse the source parameter.  Distinguish the server list from the path.
>> - * Internally we do not include the leading '/' in the path.
>>   *
>>   * The source will look like:
>>   *     <server_spec>[,<server_spec>...]:[<path>]
>> @@ -232,12 +230,15 @@ static int ceph_parse_source(struct fs_parameter *param, struct fs_context *fc)
>>
>>         dev_name_end = strchr(dev_name, '/');
>>         if (dev_name_end) {
>> -               if (strlen(dev_name_end) > 1) {
>> -                       kfree(fsopt->server_path);
>> -                       fsopt->server_path = kstrdup(dev_name_end, GFP_KERNEL);
>> -                       if (!fsopt->server_path)
>> -                               return -ENOMEM;
>> -               }
>> +               kfree(fsopt->server_path);
>> +
>> +               /*
>> +                * The server_path will include the whole chars from userland
>> +                * including the leading '/'.
>> +                */
>> +               fsopt->server_path = kstrdup(dev_name_end, GFP_KERNEL);
>> +               if (!fsopt->server_path)
>> +                       return -ENOMEM;
>>         } else {
>>                 dev_name_end = dev_name + strlen(dev_name);
>>         }
>> @@ -461,6 +462,73 @@ static int strcmp_null(const char *s1, const char *s2)
>>         return strcmp(s1, s2);
>>  }
>>
>> +/**
>> + * path_remove_extra_slash - Remove the extra slashes in the server path
>> + * @server_path: the server path and could be NULL
>> + *
>> + * Return NULL if the path is NULL or only consists of "/", or a string
>> + * without any extra slashes including the leading slash(es) and the
>> + * slash(es) at the end of the server path, such as:
>> + * "//dir1////dir2///" --> "dir1/dir2"
>> + */
>> +static char *path_remove_extra_slash(const char *server_path)
>> +{
>> +       const char *path = server_path;
>> +       const char *cur, *end;
>> +       char *buf, *p;
>> +       int len;
>> +
>> +       /* if the server path is omitted */
>> +       if (!path)
>> +               return NULL;
>> +
>> +       /* remove all the leading slashes */
>> +       while (*path == '/')
>> +               path++;
>> +
>> +       /* if the server path only consists of slashes */
>> +       if (*path == '\0')
>> +               return NULL;
>> +
>> +       len = strlen(path);
>> +
>> +       buf = kmalloc(len + 1, GFP_KERNEL);
>> +       if (!buf)
>> +               return ERR_PTR(-ENOMEM);
>> +
>> +       end = path + len;
>> +       p = buf;
>> +       do {
>> +               cur = strchr(path, '/');
>> +               if (!cur)
>> +                       cur = end;
>> +
>> +               len = cur - path;
>> +
>> +               /* including one '/' */
>> +               if (cur != end)
>> +                       len += 1;
>> +
>> +               memcpy(p, path, len);
>> +               p += len;
>> +
>> +               while (cur <= end && *cur == '/')
>> +                       cur++;
>> +               path = cur;
>> +       } while (path < end);
>> +
>> +       *p = '\0';
>> +
>> +       /*
>> +        * remove the last slash if there has and just to make sure that
>> +        * we will get something like "dir1/dir2"
>> +        */
>> +       if (*(--p) == '/')
>> +               *p = '\0';
>> +
>> +       return buf;
>> +}
>> +
>>  static int compare_mount_options(struct ceph_mount_options *new_fsopt,
>>                                  struct ceph_options *new_opt,
>>                                  struct ceph_fs_client *fsc)
>> @@ -468,6 +536,7 @@ static int compare_mount_options(struct ceph_mount_options *new_fsopt,
>>         struct ceph_mount_options *fsopt1 = new_fsopt;
>>         struct ceph_mount_options *fsopt2 = fsc->mount_options;
>>         int ofs = offsetof(struct ceph_mount_options, snapdir_name);
>> +       char *p1, *p2;
>>         int ret;
>>
>>         ret = memcmp(fsopt1, fsopt2, ofs);
>> @@ -480,9 +549,21 @@ static int compare_mount_options(struct ceph_mount_options *new_fsopt,
>>         ret = strcmp_null(fsopt1->mds_namespace, fsopt2->mds_namespace);
>>         if (ret)
>>                 return ret;
>> -       ret = strcmp_null(fsopt1->server_path, fsopt2->server_path);
>> +
>> +       p1 = path_remove_extra_slash(fsopt1->server_path);
>> +       if (IS_ERR(p1))
>> +               return PTR_ERR(p1);
>> +       p2 = path_remove_extra_slash(fsopt2->server_path);
>> +       if (IS_ERR(p2)) {
>> +               kfree(p1);
>> +               return PTR_ERR(p2);
>> +       }
>> +       ret = strcmp_null(p1, p2);
>> +       kfree(p1);
>> +       kfree(p2);
>>         if (ret)
>>                 return ret;
>> +
>>         ret = strcmp_null(fsopt1->fscache_uniq, fsopt2->fscache_uniq);
>>         if (ret)
>>                 return ret;
>> @@ -788,7 +869,6 @@ static void destroy_caches(void)
>>         ceph_fscache_unregister();
>>  }
>>
>> -
>>  /*
>>   * ceph_umount_begin - initiate forced umount.  Tear down down the
>>   * mount, skipping steps that may hang while waiting for server(s).
>> @@ -868,9 +948,6 @@ static struct dentry *open_root_dentry(struct ceph_fs_client *fsc,
>>         return root;
>>  }
>>
>> -
>> -
>> -
>>  /*
>>   * mount: join the ceph cluster, and open root directory.
>>   */
>> @@ -885,7 +962,7 @@ static struct dentry *ceph_real_mount(struct ceph_fs_client *fsc,
>>         mutex_lock(&fsc->client->mount_mutex);
>>
>>         if (!fsc->sb->s_root) {
>> -               const char *path;
>> +               const char *path, *p;
>>                 err = __ceph_open_session(fsc->client, started);
>>                 if (err < 0)
>>                         goto out;
>> @@ -897,17 +974,22 @@ static struct dentry *ceph_real_mount(struct ceph_fs_client *fsc,
>>                                 goto out;
>>                 }
>>
>> -               if (!fsc->mount_options->server_path) {
>> -                       path = "";
>> -                       dout("mount opening path \\t\n");
>> -               } else {
>> -                       path = fsc->mount_options->server_path + 1;
>> -                       dout("mount opening path %s\n", path);
>> +               p = path_remove_extra_slash(fsc->mount_options->server_path);
>> +               if (IS_ERR(p)) {
>> +                       err = PTR_ERR(p);
>> +                       goto out;
>>                 }
>> +               /* if the server path is omitted or just consists of '/' */
>> +               if (!p)
>> +                       path = "";
>> +               else
>> +                       path = p;
>> +               dout("mount opening path '%s'\n", path);
>>
>>                 ceph_fs_debugfs_init(fsc);
>>
>>                 root = open_root_dentry(fsc, path, started);
>> +               kfree(p);
>>                 if (IS_ERR(root)) {
>>                         err = PTR_ERR(root);
>>                         goto out;
>
>Hi Sasha,
>
>This commit is buggy, the fix should land in 5.6-rc2.  Please drop it
>for now -- it would need to be taken together with "ceph: canonicalize
>server path in place" or not backported at all.

I'll try and queue up the fix when it's upstream. This series will
probably be queued closer to -rc3 anyways.

-- 
Thanks,
Sasha

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ