Skip to content

Commit

Permalink
cifs: ensure correct super block for DFS reconnect
Browse files Browse the repository at this point in the history
This patch is basically fixing the lookup of tcons (DFS specific) during
reconnect (smb2pdu.c:__smb2_reconnect) to update their prefix paths.

Previously, we relied on the TCP_Server_Info pointer
(misc.c:tcp_super_cb) to determine which tcon to update the prefix path

We could not rely on TCP server pointer to determine which super block
to update the prefix path when reconnecting tcons since it might map
to different tcons that share same TCP connection.

Instead, walk through all cifs super blocks and compare their DFS full
paths with the tcon being updated to.

Signed-off-by: Paulo Alcantara (SUSE) <[email protected]>
Signed-off-by: Steve French <[email protected]>
Reviewed-by: Ronnie Sahlberg <[email protected]>
  • Loading branch information
pcacjr authored and Steve French committed Apr 23, 2020
1 parent 65303de commit 3786f4b
Showing 1 changed file with 65 additions and 17 deletions.
82 changes: 65 additions & 17 deletions fs/cifs/misc.c
Original file line number Diff line number Diff line change
Expand Up @@ -1025,59 +1025,107 @@ int copy_path_name(char *dst, const char *src)
}

struct super_cb_data {
struct TCP_Server_Info *server;
void *data;
struct super_block *sb;
};

static void super_cb(struct super_block *sb, void *arg)
static void tcp_super_cb(struct super_block *sb, void *arg)
{
struct super_cb_data *d = arg;
struct super_cb_data *sd = arg;
struct TCP_Server_Info *server = sd->data;
struct cifs_sb_info *cifs_sb;
struct cifs_tcon *tcon;

if (d->sb)
if (sd->sb)
return;

cifs_sb = CIFS_SB(sb);
tcon = cifs_sb_master_tcon(cifs_sb);
if (tcon->ses->server == d->server)
d->sb = sb;
if (tcon->ses->server == server)
sd->sb = sb;
}

struct super_block *cifs_get_tcp_super(struct TCP_Server_Info *server)
static struct super_block *__cifs_get_super(void (*f)(struct super_block *, void *),
void *data)
{
struct super_cb_data d = {
.server = server,
struct super_cb_data sd = {
.data = data,
.sb = NULL,
};

iterate_supers_type(&cifs_fs_type, super_cb, &d);
iterate_supers_type(&cifs_fs_type, f, &sd);

if (unlikely(!d.sb))
return ERR_PTR(-ENOENT);
if (!sd.sb)
return ERR_PTR(-EINVAL);
/*
* Grab an active reference in order to prevent automounts (DFS links)
* of expiring and then freeing up our cifs superblock pointer while
* we're doing failover.
*/
cifs_sb_active(d.sb);
return d.sb;
cifs_sb_active(sd.sb);
return sd.sb;
}

void cifs_put_tcp_super(struct super_block *sb)
static void __cifs_put_super(struct super_block *sb)
{
if (!IS_ERR_OR_NULL(sb))
cifs_sb_deactive(sb);
}

struct super_block *cifs_get_tcp_super(struct TCP_Server_Info *server)
{
return __cifs_get_super(tcp_super_cb, server);
}

void cifs_put_tcp_super(struct super_block *sb)
{
__cifs_put_super(sb);
}

#ifdef CONFIG_CIFS_DFS_UPCALL
static void tcon_super_cb(struct super_block *sb, void *arg)
{
struct super_cb_data *sd = arg;
struct cifs_tcon *tcon = sd->data;
struct cifs_sb_info *cifs_sb;

if (sd->sb)
return;

cifs_sb = CIFS_SB(sb);
if (tcon->dfs_path && cifs_sb->origin_fullpath &&
!strcasecmp(tcon->dfs_path, cifs_sb->origin_fullpath))
sd->sb = sb;
}

static inline struct super_block *cifs_get_tcon_super(struct cifs_tcon *tcon)
{
return __cifs_get_super(tcon_super_cb, tcon);
}

static inline void cifs_put_tcon_super(struct super_block *sb)
{
__cifs_put_super(sb);
}
#else
static inline struct super_block *cifs_get_tcon_super(struct cifs_tcon *tcon)
{
return ERR_PTR(-EOPNOTSUPP);
}

static inline void cifs_put_tcon_super(struct super_block *sb)
{
}
#endif

int update_super_prepath(struct cifs_tcon *tcon, const char *prefix,
size_t prefix_len)
{
struct super_block *sb;
struct cifs_sb_info *cifs_sb;
int rc = 0;

sb = cifs_get_tcp_super(tcon->ses->server);
sb = cifs_get_tcon_super(tcon);
if (IS_ERR(sb))
return PTR_ERR(sb);

Expand All @@ -1099,6 +1147,6 @@ int update_super_prepath(struct cifs_tcon *tcon, const char *prefix,
cifs_sb->mnt_cifs_flags |= CIFS_MOUNT_USE_PREFIX_PATH;

out:
cifs_put_tcp_super(sb);
cifs_put_tcon_super(sb);
return rc;
}

0 comments on commit 3786f4b

Please sign in to comment.