mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2024-11-01 17:08:10 +00:00
348a04a8d1
Make namespace.c being built without requiring CONFIG_CIFS_DFS_UPCALL=y by moving set_dest_addr() to dfs.c and call it at the beginning of dfs_mount_share() so it can chase the DFS link starting from the correct server in @ctx->dstaddr. Signed-off-by: Paulo Alcantara (SUSE) <pc@manguebit.com> Signed-off-by: Steve French <stfrench@microsoft.com>
152 lines
3.7 KiB
C
152 lines
3.7 KiB
C
/* SPDX-License-Identifier: GPL-2.0 */
|
|
/*
|
|
* Copyright (c) 2022 Paulo Alcantara <palcantara@suse.de>
|
|
*/
|
|
|
|
#ifndef _CIFS_DFS_H
|
|
#define _CIFS_DFS_H
|
|
|
|
#include "cifsglob.h"
|
|
#include "fs_context.h"
|
|
#include "cifs_unicode.h"
|
|
#include <linux/namei.h>
|
|
|
|
#define DFS_INTERLINK(v) \
|
|
(((v) & DFSREF_REFERRAL_SERVER) && !((v) & DFSREF_STORAGE_SERVER))
|
|
|
|
struct dfs_ref {
|
|
char *path;
|
|
char *full_path;
|
|
struct dfs_cache_tgt_list tl;
|
|
struct dfs_cache_tgt_iterator *tit;
|
|
};
|
|
|
|
struct dfs_ref_walk {
|
|
struct dfs_ref *ref;
|
|
struct dfs_ref refs[MAX_NESTED_LINKS];
|
|
};
|
|
|
|
#define ref_walk_start(w) ((w)->refs)
|
|
#define ref_walk_end(w) (&(w)->refs[ARRAY_SIZE((w)->refs) - 1])
|
|
#define ref_walk_cur(w) ((w)->ref)
|
|
#define ref_walk_descend(w) (--ref_walk_cur(w) >= ref_walk_start(w))
|
|
|
|
#define ref_walk_tit(w) (ref_walk_cur(w)->tit)
|
|
#define ref_walk_empty(w) (!ref_walk_tit(w))
|
|
#define ref_walk_path(w) (ref_walk_cur(w)->path)
|
|
#define ref_walk_fpath(w) (ref_walk_cur(w)->full_path)
|
|
#define ref_walk_tl(w) (&ref_walk_cur(w)->tl)
|
|
|
|
static inline struct dfs_ref_walk *ref_walk_alloc(void)
|
|
{
|
|
struct dfs_ref_walk *rw;
|
|
|
|
rw = kmalloc(sizeof(*rw), GFP_KERNEL);
|
|
if (!rw)
|
|
return ERR_PTR(-ENOMEM);
|
|
return rw;
|
|
}
|
|
|
|
static inline void ref_walk_init(struct dfs_ref_walk *rw)
|
|
{
|
|
memset(rw, 0, sizeof(*rw));
|
|
ref_walk_cur(rw) = ref_walk_start(rw);
|
|
}
|
|
|
|
static inline void __ref_walk_free(struct dfs_ref *ref)
|
|
{
|
|
kfree(ref->path);
|
|
kfree(ref->full_path);
|
|
dfs_cache_free_tgts(&ref->tl);
|
|
memset(ref, 0, sizeof(*ref));
|
|
}
|
|
|
|
static inline void ref_walk_free(struct dfs_ref_walk *rw)
|
|
{
|
|
struct dfs_ref *ref = ref_walk_start(rw);
|
|
|
|
for (; ref <= ref_walk_end(rw); ref++)
|
|
__ref_walk_free(ref);
|
|
kfree(rw);
|
|
}
|
|
|
|
static inline int ref_walk_advance(struct dfs_ref_walk *rw)
|
|
{
|
|
struct dfs_ref *ref = ref_walk_cur(rw) + 1;
|
|
|
|
if (ref > ref_walk_end(rw))
|
|
return -ELOOP;
|
|
__ref_walk_free(ref);
|
|
ref_walk_cur(rw) = ref;
|
|
return 0;
|
|
}
|
|
|
|
static inline struct dfs_cache_tgt_iterator *
|
|
ref_walk_next_tgt(struct dfs_ref_walk *rw)
|
|
{
|
|
struct dfs_cache_tgt_iterator *tit;
|
|
struct dfs_ref *ref = ref_walk_cur(rw);
|
|
|
|
if (!ref->tit)
|
|
tit = dfs_cache_get_tgt_iterator(&ref->tl);
|
|
else
|
|
tit = dfs_cache_get_next_tgt(&ref->tl, ref->tit);
|
|
ref->tit = tit;
|
|
return tit;
|
|
}
|
|
|
|
static inline int ref_walk_get_tgt(struct dfs_ref_walk *rw,
|
|
struct dfs_info3_param *tgt)
|
|
{
|
|
zfree_dfs_info_param(tgt);
|
|
return dfs_cache_get_tgt_referral(ref_walk_path(rw) + 1,
|
|
ref_walk_tit(rw), tgt);
|
|
}
|
|
|
|
static inline int ref_walk_num_tgts(struct dfs_ref_walk *rw)
|
|
{
|
|
return dfs_cache_get_nr_tgts(ref_walk_tl(rw));
|
|
}
|
|
|
|
static inline void ref_walk_set_tgt_hint(struct dfs_ref_walk *rw)
|
|
{
|
|
dfs_cache_noreq_update_tgthint(ref_walk_path(rw) + 1,
|
|
ref_walk_tit(rw));
|
|
}
|
|
|
|
struct dfs_root_ses {
|
|
struct list_head list;
|
|
struct cifs_ses *ses;
|
|
};
|
|
|
|
int dfs_parse_target_referral(const char *full_path, const struct dfs_info3_param *ref,
|
|
struct smb3_fs_context *ctx);
|
|
int dfs_mount_share(struct cifs_mount_ctx *mnt_ctx, bool *isdfs);
|
|
|
|
static inline char *dfs_get_path(struct cifs_sb_info *cifs_sb, const char *path)
|
|
{
|
|
return dfs_cache_canonical_path(path, cifs_sb->local_nls, cifs_remap(cifs_sb));
|
|
}
|
|
|
|
static inline int dfs_get_referral(struct cifs_mount_ctx *mnt_ctx, const char *path,
|
|
struct dfs_info3_param *ref, struct dfs_cache_tgt_list *tl)
|
|
{
|
|
struct smb3_fs_context *ctx = mnt_ctx->fs_ctx;
|
|
struct cifs_sb_info *cifs_sb = mnt_ctx->cifs_sb;
|
|
|
|
return dfs_cache_find(mnt_ctx->xid, ctx->dfs_root_ses, cifs_sb->local_nls,
|
|
cifs_remap(cifs_sb), path, ref, tl);
|
|
}
|
|
|
|
static inline void dfs_put_root_smb_sessions(struct list_head *head)
|
|
{
|
|
struct dfs_root_ses *root, *tmp;
|
|
|
|
list_for_each_entry_safe(root, tmp, head, list) {
|
|
list_del_init(&root->list);
|
|
cifs_put_smb_ses(root->ses);
|
|
kfree(root);
|
|
}
|
|
}
|
|
|
|
#endif /* _CIFS_DFS_H */
|