mirror of
https://github.com/torvalds/linux.git
synced 2024-11-22 12:11:40 +00:00
4508ec1735
When uid, gid and cruid are not specified, we need to dynamically
set them into the filesystem context used for automounting otherwise
they'll end up reusing the values from the parent mount.
Fixes: 9fd29a5bae
("cifs: use fs_context for automounts")
Reported-by: Shane Nehring <snehring@iastate.edu>
Closes: https://bugzilla.redhat.com/show_bug.cgi?id=2259257
Cc: stable@vger.kernel.org # 6.2+
Signed-off-by: Paulo Alcantara (Red Hat) <pc@manguebit.com>
Signed-off-by: Steve French <stfrench@microsoft.com>
279 lines
6.3 KiB
C
279 lines
6.3 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* Contains mounting routines used for handling traversal via SMB junctions.
|
|
*
|
|
* Copyright (c) 2007 Igor Mammedov
|
|
* Copyright (C) International Business Machines Corp., 2008
|
|
* Author(s): Igor Mammedov (niallain@gmail.com)
|
|
* Steve French (sfrench@us.ibm.com)
|
|
* Copyright (c) 2023 Paulo Alcantara <palcantara@suse.de>
|
|
*/
|
|
|
|
#include <linux/dcache.h>
|
|
#include <linux/mount.h>
|
|
#include <linux/namei.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/vfs.h>
|
|
#include <linux/fs.h>
|
|
#include <linux/inet.h>
|
|
#include "cifsglob.h"
|
|
#include "cifsproto.h"
|
|
#include "cifsfs.h"
|
|
#include "cifs_debug.h"
|
|
#include "fs_context.h"
|
|
|
|
static LIST_HEAD(cifs_automount_list);
|
|
|
|
static void cifs_expire_automounts(struct work_struct *work);
|
|
static DECLARE_DELAYED_WORK(cifs_automount_task,
|
|
cifs_expire_automounts);
|
|
static int cifs_mountpoint_expiry_timeout = 500 * HZ;
|
|
|
|
static void cifs_expire_automounts(struct work_struct *work)
|
|
{
|
|
struct list_head *list = &cifs_automount_list;
|
|
|
|
mark_mounts_for_expiry(list);
|
|
if (!list_empty(list))
|
|
schedule_delayed_work(&cifs_automount_task,
|
|
cifs_mountpoint_expiry_timeout);
|
|
}
|
|
|
|
void cifs_release_automount_timer(void)
|
|
{
|
|
if (WARN_ON(!list_empty(&cifs_automount_list)))
|
|
return;
|
|
cancel_delayed_work_sync(&cifs_automount_task);
|
|
}
|
|
|
|
/**
|
|
* cifs_build_devname - build a devicename from a UNC and optional prepath
|
|
* @nodename: pointer to UNC string
|
|
* @prepath: pointer to prefixpath (or NULL if there isn't one)
|
|
*
|
|
* Build a new cifs devicename after chasing a DFS referral. Allocate a buffer
|
|
* big enough to hold the final thing. Copy the UNC from the nodename, and
|
|
* concatenate the prepath onto the end of it if there is one.
|
|
*
|
|
* Returns pointer to the built string, or a ERR_PTR. Caller is responsible
|
|
* for freeing the returned string.
|
|
*/
|
|
char *
|
|
cifs_build_devname(char *nodename, const char *prepath)
|
|
{
|
|
size_t pplen;
|
|
size_t unclen;
|
|
char *dev;
|
|
char *pos;
|
|
|
|
/* skip over any preceding delimiters */
|
|
nodename += strspn(nodename, "\\");
|
|
if (!*nodename)
|
|
return ERR_PTR(-EINVAL);
|
|
|
|
/* get length of UNC and set pos to last char */
|
|
unclen = strlen(nodename);
|
|
pos = nodename + unclen - 1;
|
|
|
|
/* trim off any trailing delimiters */
|
|
while (*pos == '\\') {
|
|
--pos;
|
|
--unclen;
|
|
}
|
|
|
|
/* allocate a buffer:
|
|
* +2 for preceding "//"
|
|
* +1 for delimiter between UNC and prepath
|
|
* +1 for trailing NULL
|
|
*/
|
|
pplen = prepath ? strlen(prepath) : 0;
|
|
dev = kmalloc(2 + unclen + 1 + pplen + 1, GFP_KERNEL);
|
|
if (!dev)
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
pos = dev;
|
|
/* add the initial "//" */
|
|
*pos = '/';
|
|
++pos;
|
|
*pos = '/';
|
|
++pos;
|
|
|
|
/* copy in the UNC portion from referral */
|
|
memcpy(pos, nodename, unclen);
|
|
pos += unclen;
|
|
|
|
/* copy the prefixpath remainder (if there is one) */
|
|
if (pplen) {
|
|
*pos = '/';
|
|
++pos;
|
|
memcpy(pos, prepath, pplen);
|
|
pos += pplen;
|
|
}
|
|
|
|
/* NULL terminator */
|
|
*pos = '\0';
|
|
|
|
convert_delimiter(dev, '/');
|
|
return dev;
|
|
}
|
|
|
|
static bool is_dfs_mount(struct dentry *dentry)
|
|
{
|
|
struct cifs_sb_info *cifs_sb = CIFS_SB(dentry->d_sb);
|
|
struct cifs_tcon *tcon = cifs_sb_master_tcon(cifs_sb);
|
|
bool ret;
|
|
|
|
spin_lock(&tcon->tc_lock);
|
|
ret = !!tcon->origin_fullpath;
|
|
spin_unlock(&tcon->tc_lock);
|
|
return ret;
|
|
}
|
|
|
|
/* Return full path out of a dentry set for automount */
|
|
static char *automount_fullpath(struct dentry *dentry, void *page)
|
|
{
|
|
struct cifs_sb_info *cifs_sb = CIFS_SB(dentry->d_sb);
|
|
struct cifs_tcon *tcon = cifs_sb_master_tcon(cifs_sb);
|
|
size_t len;
|
|
char *s;
|
|
|
|
spin_lock(&tcon->tc_lock);
|
|
if (!tcon->origin_fullpath) {
|
|
spin_unlock(&tcon->tc_lock);
|
|
return build_path_from_dentry_optional_prefix(dentry,
|
|
page,
|
|
true);
|
|
}
|
|
spin_unlock(&tcon->tc_lock);
|
|
|
|
s = dentry_path_raw(dentry, page, PATH_MAX);
|
|
if (IS_ERR(s))
|
|
return s;
|
|
/* for root, we want "" */
|
|
if (!s[1])
|
|
s++;
|
|
|
|
spin_lock(&tcon->tc_lock);
|
|
len = strlen(tcon->origin_fullpath);
|
|
if (s < (char *)page + len) {
|
|
spin_unlock(&tcon->tc_lock);
|
|
return ERR_PTR(-ENAMETOOLONG);
|
|
}
|
|
|
|
s -= len;
|
|
memcpy(s, tcon->origin_fullpath, len);
|
|
spin_unlock(&tcon->tc_lock);
|
|
convert_delimiter(s, '/');
|
|
|
|
return s;
|
|
}
|
|
|
|
static void fs_context_set_ids(struct smb3_fs_context *ctx)
|
|
{
|
|
kuid_t uid = current_fsuid();
|
|
kgid_t gid = current_fsgid();
|
|
|
|
if (ctx->multiuser) {
|
|
if (!ctx->uid_specified)
|
|
ctx->linux_uid = uid;
|
|
if (!ctx->gid_specified)
|
|
ctx->linux_gid = gid;
|
|
}
|
|
if (!ctx->cruid_specified)
|
|
ctx->cred_uid = uid;
|
|
}
|
|
|
|
/*
|
|
* Create a vfsmount that we can automount
|
|
*/
|
|
static struct vfsmount *cifs_do_automount(struct path *path)
|
|
{
|
|
int rc;
|
|
struct dentry *mntpt = path->dentry;
|
|
struct fs_context *fc;
|
|
void *page = NULL;
|
|
struct smb3_fs_context *ctx, *cur_ctx;
|
|
struct smb3_fs_context tmp;
|
|
char *full_path;
|
|
struct vfsmount *mnt;
|
|
|
|
if (IS_ROOT(mntpt))
|
|
return ERR_PTR(-ESTALE);
|
|
|
|
cur_ctx = CIFS_SB(mntpt->d_sb)->ctx;
|
|
|
|
fc = fs_context_for_submount(path->mnt->mnt_sb->s_type, mntpt);
|
|
if (IS_ERR(fc))
|
|
return ERR_CAST(fc);
|
|
|
|
ctx = smb3_fc2context(fc);
|
|
|
|
page = alloc_dentry_path();
|
|
full_path = automount_fullpath(mntpt, page);
|
|
if (IS_ERR(full_path)) {
|
|
mnt = ERR_CAST(full_path);
|
|
goto out;
|
|
}
|
|
|
|
tmp = *cur_ctx;
|
|
tmp.source = NULL;
|
|
tmp.leaf_fullpath = NULL;
|
|
tmp.UNC = tmp.prepath = NULL;
|
|
tmp.dfs_root_ses = NULL;
|
|
fs_context_set_ids(&tmp);
|
|
|
|
rc = smb3_fs_context_dup(ctx, &tmp);
|
|
if (rc) {
|
|
mnt = ERR_PTR(rc);
|
|
goto out;
|
|
}
|
|
|
|
rc = smb3_parse_devname(full_path, ctx);
|
|
if (rc) {
|
|
mnt = ERR_PTR(rc);
|
|
goto out;
|
|
}
|
|
|
|
ctx->source = smb3_fs_context_fullpath(ctx, '/');
|
|
if (IS_ERR(ctx->source)) {
|
|
mnt = ERR_CAST(ctx->source);
|
|
ctx->source = NULL;
|
|
goto out;
|
|
}
|
|
ctx->dfs_automount = is_dfs_mount(mntpt);
|
|
cifs_dbg(FYI, "%s: ctx: source=%s UNC=%s prepath=%s dfs_automount=%d\n",
|
|
__func__, ctx->source, ctx->UNC, ctx->prepath, ctx->dfs_automount);
|
|
|
|
mnt = fc_mount(fc);
|
|
out:
|
|
put_fs_context(fc);
|
|
free_dentry_path(page);
|
|
return mnt;
|
|
}
|
|
|
|
/*
|
|
* Attempt to automount the referral
|
|
*/
|
|
struct vfsmount *cifs_d_automount(struct path *path)
|
|
{
|
|
struct vfsmount *newmnt;
|
|
|
|
cifs_dbg(FYI, "%s: %pd\n", __func__, path->dentry);
|
|
|
|
newmnt = cifs_do_automount(path);
|
|
if (IS_ERR(newmnt)) {
|
|
cifs_dbg(FYI, "leaving %s [automount failed]\n" , __func__);
|
|
return newmnt;
|
|
}
|
|
|
|
mntget(newmnt); /* prevent immediate expiration */
|
|
mnt_set_expiry(newmnt, &cifs_automount_list);
|
|
schedule_delayed_work(&cifs_automount_task,
|
|
cifs_mountpoint_expiry_timeout);
|
|
cifs_dbg(FYI, "leaving %s [ok]\n" , __func__);
|
|
return newmnt;
|
|
}
|
|
|
|
const struct inode_operations cifs_namespace_inode_operations = {
|
|
};
|