Commit f40125c0 authored by Linus Torvalds's avatar Linus Torvalds

Merge tag '6.5-rc3-ksmbd-server-fixes' of git://git.samba.org/ksmbd

Pull ksmbd server fixes from Steve French:

 - fixes for two possible out of bounds access (in negotiate, and in
   decrypt msg)

 - fix unsigned compared to zero warning

 - fix path lookup crossing a mountpoint

 - fix case when first compound request is a tree connect

 - fix memory leak if reads are compounded

* tag '6.5-rc3-ksmbd-server-fixes' of git://git.samba.org/ksmbd:
  ksmbd: fix out of bounds in init_smb2_rsp_hdr()
  ksmbd: no response from compound read
  ksmbd: validate session id and tree id in compound request
  ksmbd: fix out of bounds in smb3_decrypt_req()
  ksmbd: check if a mount point is crossed during path lookup
  ksmbd: Fix unsigned expression compared with zero
parents 5f0bc0b0 536bb492
...@@ -352,7 +352,8 @@ enum KSMBD_TREE_CONN_STATUS { ...@@ -352,7 +352,8 @@ enum KSMBD_TREE_CONN_STATUS {
#define KSMBD_SHARE_FLAG_STREAMS BIT(11) #define KSMBD_SHARE_FLAG_STREAMS BIT(11)
#define KSMBD_SHARE_FLAG_FOLLOW_SYMLINKS BIT(12) #define KSMBD_SHARE_FLAG_FOLLOW_SYMLINKS BIT(12)
#define KSMBD_SHARE_FLAG_ACL_XATTR BIT(13) #define KSMBD_SHARE_FLAG_ACL_XATTR BIT(13)
#define KSMBD_SHARE_FLAG_UPDATE BIT(14) #define KSMBD_SHARE_FLAG_UPDATE BIT(14)
#define KSMBD_SHARE_FLAG_CROSSMNT BIT(15)
/* /*
* Tree connect request flags. * Tree connect request flags.
......
...@@ -286,6 +286,7 @@ static void handle_ksmbd_work(struct work_struct *wk) ...@@ -286,6 +286,7 @@ static void handle_ksmbd_work(struct work_struct *wk)
static int queue_ksmbd_work(struct ksmbd_conn *conn) static int queue_ksmbd_work(struct ksmbd_conn *conn)
{ {
struct ksmbd_work *work; struct ksmbd_work *work;
int err;
work = ksmbd_alloc_work_struct(); work = ksmbd_alloc_work_struct();
if (!work) { if (!work) {
...@@ -297,7 +298,11 @@ static int queue_ksmbd_work(struct ksmbd_conn *conn) ...@@ -297,7 +298,11 @@ static int queue_ksmbd_work(struct ksmbd_conn *conn)
work->request_buf = conn->request_buf; work->request_buf = conn->request_buf;
conn->request_buf = NULL; conn->request_buf = NULL;
ksmbd_init_smb_server(work); err = ksmbd_init_smb_server(work);
if (err) {
ksmbd_free_work_struct(work);
return 0;
}
ksmbd_conn_enqueue_request(work); ksmbd_conn_enqueue_request(work);
atomic_inc(&conn->r_count); atomic_inc(&conn->r_count);
......
...@@ -87,9 +87,9 @@ struct channel *lookup_chann_list(struct ksmbd_session *sess, struct ksmbd_conn ...@@ -87,9 +87,9 @@ struct channel *lookup_chann_list(struct ksmbd_session *sess, struct ksmbd_conn
*/ */
int smb2_get_ksmbd_tcon(struct ksmbd_work *work) int smb2_get_ksmbd_tcon(struct ksmbd_work *work)
{ {
struct smb2_hdr *req_hdr = smb2_get_msg(work->request_buf); struct smb2_hdr *req_hdr = ksmbd_req_buf_next(work);
unsigned int cmd = le16_to_cpu(req_hdr->Command); unsigned int cmd = le16_to_cpu(req_hdr->Command);
int tree_id; unsigned int tree_id;
if (cmd == SMB2_TREE_CONNECT_HE || if (cmd == SMB2_TREE_CONNECT_HE ||
cmd == SMB2_CANCEL_HE || cmd == SMB2_CANCEL_HE ||
...@@ -114,7 +114,7 @@ int smb2_get_ksmbd_tcon(struct ksmbd_work *work) ...@@ -114,7 +114,7 @@ int smb2_get_ksmbd_tcon(struct ksmbd_work *work)
pr_err("The first operation in the compound does not have tcon\n"); pr_err("The first operation in the compound does not have tcon\n");
return -EINVAL; return -EINVAL;
} }
if (work->tcon->id != tree_id) { if (tree_id != UINT_MAX && work->tcon->id != tree_id) {
pr_err("tree id(%u) is different with id(%u) in first operation\n", pr_err("tree id(%u) is different with id(%u) in first operation\n",
tree_id, work->tcon->id); tree_id, work->tcon->id);
return -EINVAL; return -EINVAL;
...@@ -559,9 +559,9 @@ int smb2_allocate_rsp_buf(struct ksmbd_work *work) ...@@ -559,9 +559,9 @@ int smb2_allocate_rsp_buf(struct ksmbd_work *work)
*/ */
int smb2_check_user_session(struct ksmbd_work *work) int smb2_check_user_session(struct ksmbd_work *work)
{ {
struct smb2_hdr *req_hdr = smb2_get_msg(work->request_buf); struct smb2_hdr *req_hdr = ksmbd_req_buf_next(work);
struct ksmbd_conn *conn = work->conn; struct ksmbd_conn *conn = work->conn;
unsigned int cmd = conn->ops->get_cmd_val(work); unsigned int cmd = le16_to_cpu(req_hdr->Command);
unsigned long long sess_id; unsigned long long sess_id;
/* /*
...@@ -587,7 +587,7 @@ int smb2_check_user_session(struct ksmbd_work *work) ...@@ -587,7 +587,7 @@ int smb2_check_user_session(struct ksmbd_work *work)
pr_err("The first operation in the compound does not have sess\n"); pr_err("The first operation in the compound does not have sess\n");
return -EINVAL; return -EINVAL;
} }
if (work->sess->id != sess_id) { if (sess_id != ULLONG_MAX && work->sess->id != sess_id) {
pr_err("session id(%llu) is different with the first operation(%lld)\n", pr_err("session id(%llu) is different with the first operation(%lld)\n",
sess_id, work->sess->id); sess_id, work->sess->id);
return -EINVAL; return -EINVAL;
...@@ -2467,8 +2467,9 @@ static void smb2_update_xattrs(struct ksmbd_tree_connect *tcon, ...@@ -2467,8 +2467,9 @@ static void smb2_update_xattrs(struct ksmbd_tree_connect *tcon,
} }
} }
static int smb2_creat(struct ksmbd_work *work, struct path *path, char *name, static int smb2_creat(struct ksmbd_work *work, struct path *parent_path,
int open_flags, umode_t posix_mode, bool is_dir) struct path *path, char *name, int open_flags,
umode_t posix_mode, bool is_dir)
{ {
struct ksmbd_tree_connect *tcon = work->tcon; struct ksmbd_tree_connect *tcon = work->tcon;
struct ksmbd_share_config *share = tcon->share_conf; struct ksmbd_share_config *share = tcon->share_conf;
...@@ -2495,7 +2496,7 @@ static int smb2_creat(struct ksmbd_work *work, struct path *path, char *name, ...@@ -2495,7 +2496,7 @@ static int smb2_creat(struct ksmbd_work *work, struct path *path, char *name,
return rc; return rc;
} }
rc = ksmbd_vfs_kern_path_locked(work, name, 0, path, 0); rc = ksmbd_vfs_kern_path_locked(work, name, 0, parent_path, path, 0);
if (rc) { if (rc) {
pr_err("cannot get linux path (%s), err = %d\n", pr_err("cannot get linux path (%s), err = %d\n",
name, rc); name, rc);
...@@ -2565,7 +2566,7 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2565,7 +2566,7 @@ int smb2_open(struct ksmbd_work *work)
struct ksmbd_tree_connect *tcon = work->tcon; struct ksmbd_tree_connect *tcon = work->tcon;
struct smb2_create_req *req; struct smb2_create_req *req;
struct smb2_create_rsp *rsp; struct smb2_create_rsp *rsp;
struct path path; struct path path, parent_path;
struct ksmbd_share_config *share = tcon->share_conf; struct ksmbd_share_config *share = tcon->share_conf;
struct ksmbd_file *fp = NULL; struct ksmbd_file *fp = NULL;
struct file *filp = NULL; struct file *filp = NULL;
...@@ -2786,7 +2787,8 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2786,7 +2787,8 @@ int smb2_open(struct ksmbd_work *work)
goto err_out1; goto err_out1;
} }
rc = ksmbd_vfs_kern_path_locked(work, name, LOOKUP_NO_SYMLINKS, &path, 1); rc = ksmbd_vfs_kern_path_locked(work, name, LOOKUP_NO_SYMLINKS,
&parent_path, &path, 1);
if (!rc) { if (!rc) {
file_present = true; file_present = true;
...@@ -2906,7 +2908,8 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2906,7 +2908,8 @@ int smb2_open(struct ksmbd_work *work)
/*create file if not present */ /*create file if not present */
if (!file_present) { if (!file_present) {
rc = smb2_creat(work, &path, name, open_flags, posix_mode, rc = smb2_creat(work, &parent_path, &path, name, open_flags,
posix_mode,
req->CreateOptions & FILE_DIRECTORY_FILE_LE); req->CreateOptions & FILE_DIRECTORY_FILE_LE);
if (rc) { if (rc) {
if (rc == -ENOENT) { if (rc == -ENOENT) {
...@@ -3321,8 +3324,9 @@ int smb2_open(struct ksmbd_work *work) ...@@ -3321,8 +3324,9 @@ int smb2_open(struct ksmbd_work *work)
err_out: err_out:
if (file_present || created) { if (file_present || created) {
inode_unlock(d_inode(path.dentry->d_parent)); inode_unlock(d_inode(parent_path.dentry));
dput(path.dentry); path_put(&path);
path_put(&parent_path);
} }
ksmbd_revert_fsids(work); ksmbd_revert_fsids(work);
err_out1: err_out1:
...@@ -5545,7 +5549,7 @@ static int smb2_create_link(struct ksmbd_work *work, ...@@ -5545,7 +5549,7 @@ static int smb2_create_link(struct ksmbd_work *work,
struct nls_table *local_nls) struct nls_table *local_nls)
{ {
char *link_name = NULL, *target_name = NULL, *pathname = NULL; char *link_name = NULL, *target_name = NULL, *pathname = NULL;
struct path path; struct path path, parent_path;
bool file_present = false; bool file_present = false;
int rc; int rc;
...@@ -5575,7 +5579,7 @@ static int smb2_create_link(struct ksmbd_work *work, ...@@ -5575,7 +5579,7 @@ static int smb2_create_link(struct ksmbd_work *work,
ksmbd_debug(SMB, "target name is %s\n", target_name); ksmbd_debug(SMB, "target name is %s\n", target_name);
rc = ksmbd_vfs_kern_path_locked(work, link_name, LOOKUP_NO_SYMLINKS, rc = ksmbd_vfs_kern_path_locked(work, link_name, LOOKUP_NO_SYMLINKS,
&path, 0); &parent_path, &path, 0);
if (rc) { if (rc) {
if (rc != -ENOENT) if (rc != -ENOENT)
goto out; goto out;
...@@ -5605,8 +5609,9 @@ static int smb2_create_link(struct ksmbd_work *work, ...@@ -5605,8 +5609,9 @@ static int smb2_create_link(struct ksmbd_work *work,
rc = -EINVAL; rc = -EINVAL;
out: out:
if (file_present) { if (file_present) {
inode_unlock(d_inode(path.dentry->d_parent)); inode_unlock(d_inode(parent_path.dentry));
path_put(&path); path_put(&path);
path_put(&parent_path);
} }
if (!IS_ERR(link_name)) if (!IS_ERR(link_name))
kfree(link_name); kfree(link_name);
...@@ -6209,6 +6214,11 @@ int smb2_read(struct ksmbd_work *work) ...@@ -6209,6 +6214,11 @@ int smb2_read(struct ksmbd_work *work)
unsigned int max_read_size = conn->vals->max_read_size; unsigned int max_read_size = conn->vals->max_read_size;
WORK_BUFFERS(work, req, rsp); WORK_BUFFERS(work, req, rsp);
if (work->next_smb2_rcv_hdr_off) {
work->send_no_response = 1;
err = -EOPNOTSUPP;
goto out;
}
if (test_share_config_flag(work->tcon->share_conf, if (test_share_config_flag(work->tcon->share_conf,
KSMBD_SHARE_FLAG_PIPE)) { KSMBD_SHARE_FLAG_PIPE)) {
...@@ -8609,7 +8619,8 @@ int smb3_decrypt_req(struct ksmbd_work *work) ...@@ -8609,7 +8619,8 @@ int smb3_decrypt_req(struct ksmbd_work *work)
struct smb2_transform_hdr *tr_hdr = smb2_get_msg(buf); struct smb2_transform_hdr *tr_hdr = smb2_get_msg(buf);
int rc = 0; int rc = 0;
if (buf_data_size < sizeof(struct smb2_hdr)) { if (pdu_length < sizeof(struct smb2_transform_hdr) ||
buf_data_size < sizeof(struct smb2_hdr)) {
pr_err("Transform message is too small (%u)\n", pr_err("Transform message is too small (%u)\n",
pdu_length); pdu_length);
return -ECONNABORTED; return -ECONNABORTED;
......
...@@ -388,26 +388,29 @@ static struct smb_version_cmds smb1_server_cmds[1] = { ...@@ -388,26 +388,29 @@ static struct smb_version_cmds smb1_server_cmds[1] = {
[SMB_COM_NEGOTIATE_EX] = { .proc = smb1_negotiate, }, [SMB_COM_NEGOTIATE_EX] = { .proc = smb1_negotiate, },
}; };
static void init_smb1_server(struct ksmbd_conn *conn) static int init_smb1_server(struct ksmbd_conn *conn)
{ {
conn->ops = &smb1_server_ops; conn->ops = &smb1_server_ops;
conn->cmds = smb1_server_cmds; conn->cmds = smb1_server_cmds;
conn->max_cmds = ARRAY_SIZE(smb1_server_cmds); conn->max_cmds = ARRAY_SIZE(smb1_server_cmds);
return 0;
} }
void ksmbd_init_smb_server(struct ksmbd_work *work) int ksmbd_init_smb_server(struct ksmbd_work *work)
{ {
struct ksmbd_conn *conn = work->conn; struct ksmbd_conn *conn = work->conn;
__le32 proto; __le32 proto;
if (conn->need_neg == false)
return;
proto = *(__le32 *)((struct smb_hdr *)work->request_buf)->Protocol; proto = *(__le32 *)((struct smb_hdr *)work->request_buf)->Protocol;
if (conn->need_neg == false) {
if (proto == SMB1_PROTO_NUMBER)
return -EINVAL;
return 0;
}
if (proto == SMB1_PROTO_NUMBER) if (proto == SMB1_PROTO_NUMBER)
init_smb1_server(conn); return init_smb1_server(conn);
else return init_smb3_11_server(conn);
init_smb3_11_server(conn);
} }
int ksmbd_populate_dot_dotdot_entries(struct ksmbd_work *work, int info_level, int ksmbd_populate_dot_dotdot_entries(struct ksmbd_work *work, int info_level,
......
...@@ -427,7 +427,7 @@ bool ksmbd_smb_request(struct ksmbd_conn *conn); ...@@ -427,7 +427,7 @@ bool ksmbd_smb_request(struct ksmbd_conn *conn);
int ksmbd_lookup_dialect_by_id(__le16 *cli_dialects, __le16 dialects_count); int ksmbd_lookup_dialect_by_id(__le16 *cli_dialects, __le16 dialects_count);
void ksmbd_init_smb_server(struct ksmbd_work *work); int ksmbd_init_smb_server(struct ksmbd_work *work);
struct ksmbd_kstat; struct ksmbd_kstat;
int ksmbd_populate_dot_dotdot_entries(struct ksmbd_work *work, int ksmbd_populate_dot_dotdot_entries(struct ksmbd_work *work,
......
...@@ -63,13 +63,13 @@ int ksmbd_vfs_lock_parent(struct dentry *parent, struct dentry *child) ...@@ -63,13 +63,13 @@ int ksmbd_vfs_lock_parent(struct dentry *parent, struct dentry *child)
static int ksmbd_vfs_path_lookup_locked(struct ksmbd_share_config *share_conf, static int ksmbd_vfs_path_lookup_locked(struct ksmbd_share_config *share_conf,
char *pathname, unsigned int flags, char *pathname, unsigned int flags,
struct path *parent_path,
struct path *path) struct path *path)
{ {
struct qstr last; struct qstr last;
struct filename *filename; struct filename *filename;
struct path *root_share_path = &share_conf->vfs_path; struct path *root_share_path = &share_conf->vfs_path;
int err, type; int err, type;
struct path parent_path;
struct dentry *d; struct dentry *d;
if (pathname[0] == '\0') { if (pathname[0] == '\0') {
...@@ -84,7 +84,7 @@ static int ksmbd_vfs_path_lookup_locked(struct ksmbd_share_config *share_conf, ...@@ -84,7 +84,7 @@ static int ksmbd_vfs_path_lookup_locked(struct ksmbd_share_config *share_conf,
return PTR_ERR(filename); return PTR_ERR(filename);
err = vfs_path_parent_lookup(filename, flags, err = vfs_path_parent_lookup(filename, flags,
&parent_path, &last, &type, parent_path, &last, &type,
root_share_path); root_share_path);
if (err) { if (err) {
putname(filename); putname(filename);
...@@ -92,13 +92,13 @@ static int ksmbd_vfs_path_lookup_locked(struct ksmbd_share_config *share_conf, ...@@ -92,13 +92,13 @@ static int ksmbd_vfs_path_lookup_locked(struct ksmbd_share_config *share_conf,
} }
if (unlikely(type != LAST_NORM)) { if (unlikely(type != LAST_NORM)) {
path_put(&parent_path); path_put(parent_path);
putname(filename); putname(filename);
return -ENOENT; return -ENOENT;
} }
inode_lock_nested(parent_path.dentry->d_inode, I_MUTEX_PARENT); inode_lock_nested(parent_path->dentry->d_inode, I_MUTEX_PARENT);
d = lookup_one_qstr_excl(&last, parent_path.dentry, 0); d = lookup_one_qstr_excl(&last, parent_path->dentry, 0);
if (IS_ERR(d)) if (IS_ERR(d))
goto err_out; goto err_out;
...@@ -108,15 +108,22 @@ static int ksmbd_vfs_path_lookup_locked(struct ksmbd_share_config *share_conf, ...@@ -108,15 +108,22 @@ static int ksmbd_vfs_path_lookup_locked(struct ksmbd_share_config *share_conf,
} }
path->dentry = d; path->dentry = d;
path->mnt = share_conf->vfs_path.mnt; path->mnt = mntget(parent_path->mnt);
path_put(&parent_path);
putname(filename);
if (test_share_config_flag(share_conf, KSMBD_SHARE_FLAG_CROSSMNT)) {
err = follow_down(path, 0);
if (err < 0) {
path_put(path);
goto err_out;
}
}
putname(filename);
return 0; return 0;
err_out: err_out:
inode_unlock(parent_path.dentry->d_inode); inode_unlock(d_inode(parent_path->dentry));
path_put(&parent_path); path_put(parent_path);
putname(filename); putname(filename);
return -ENOENT; return -ENOENT;
} }
...@@ -412,7 +419,8 @@ static int ksmbd_vfs_stream_write(struct ksmbd_file *fp, char *buf, loff_t *pos, ...@@ -412,7 +419,8 @@ static int ksmbd_vfs_stream_write(struct ksmbd_file *fp, char *buf, loff_t *pos,
{ {
char *stream_buf = NULL, *wbuf; char *stream_buf = NULL, *wbuf;
struct mnt_idmap *idmap = file_mnt_idmap(fp->filp); struct mnt_idmap *idmap = file_mnt_idmap(fp->filp);
size_t size, v_len; size_t size;
ssize_t v_len;
int err = 0; int err = 0;
ksmbd_debug(VFS, "write stream data pos : %llu, count : %zd\n", ksmbd_debug(VFS, "write stream data pos : %llu, count : %zd\n",
...@@ -429,9 +437,9 @@ static int ksmbd_vfs_stream_write(struct ksmbd_file *fp, char *buf, loff_t *pos, ...@@ -429,9 +437,9 @@ static int ksmbd_vfs_stream_write(struct ksmbd_file *fp, char *buf, loff_t *pos,
fp->stream.name, fp->stream.name,
fp->stream.size, fp->stream.size,
&stream_buf); &stream_buf);
if ((int)v_len < 0) { if (v_len < 0) {
pr_err("not found stream in xattr : %zd\n", v_len); pr_err("not found stream in xattr : %zd\n", v_len);
err = (int)v_len; err = v_len;
goto out; goto out;
} }
...@@ -1194,14 +1202,14 @@ static int ksmbd_vfs_lookup_in_dir(const struct path *dir, char *name, ...@@ -1194,14 +1202,14 @@ static int ksmbd_vfs_lookup_in_dir(const struct path *dir, char *name,
* Return: 0 on success, otherwise error * Return: 0 on success, otherwise error
*/ */
int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name, int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name,
unsigned int flags, struct path *path, unsigned int flags, struct path *parent_path,
bool caseless) struct path *path, bool caseless)
{ {
struct ksmbd_share_config *share_conf = work->tcon->share_conf; struct ksmbd_share_config *share_conf = work->tcon->share_conf;
int err; int err;
struct path parent_path;
err = ksmbd_vfs_path_lookup_locked(share_conf, name, flags, path); err = ksmbd_vfs_path_lookup_locked(share_conf, name, flags, parent_path,
path);
if (!err) if (!err)
return 0; return 0;
...@@ -1216,10 +1224,10 @@ int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name, ...@@ -1216,10 +1224,10 @@ int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name,
path_len = strlen(filepath); path_len = strlen(filepath);
remain_len = path_len; remain_len = path_len;
parent_path = share_conf->vfs_path; *parent_path = share_conf->vfs_path;
path_get(&parent_path); path_get(parent_path);
while (d_can_lookup(parent_path.dentry)) { while (d_can_lookup(parent_path->dentry)) {
char *filename = filepath + path_len - remain_len; char *filename = filepath + path_len - remain_len;
char *next = strchrnul(filename, '/'); char *next = strchrnul(filename, '/');
size_t filename_len = next - filename; size_t filename_len = next - filename;
...@@ -1228,7 +1236,7 @@ int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name, ...@@ -1228,7 +1236,7 @@ int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name,
if (filename_len == 0) if (filename_len == 0)
break; break;
err = ksmbd_vfs_lookup_in_dir(&parent_path, filename, err = ksmbd_vfs_lookup_in_dir(parent_path, filename,
filename_len, filename_len,
work->conn->um); work->conn->um);
if (err) if (err)
...@@ -1245,8 +1253,8 @@ int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name, ...@@ -1245,8 +1253,8 @@ int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name,
goto out2; goto out2;
else if (is_last) else if (is_last)
goto out1; goto out1;
path_put(&parent_path); path_put(parent_path);
parent_path = *path; *parent_path = *path;
next[0] = '/'; next[0] = '/';
remain_len -= filename_len + 1; remain_len -= filename_len + 1;
...@@ -1254,16 +1262,17 @@ int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name, ...@@ -1254,16 +1262,17 @@ int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name,
err = -EINVAL; err = -EINVAL;
out2: out2:
path_put(&parent_path); path_put(parent_path);
out1: out1:
kfree(filepath); kfree(filepath);
} }
if (!err) { if (!err) {
err = ksmbd_vfs_lock_parent(parent_path.dentry, path->dentry); err = ksmbd_vfs_lock_parent(parent_path->dentry, path->dentry);
if (err) if (err) {
dput(path->dentry); path_put(path);
path_put(&parent_path); path_put(parent_path);
}
} }
return err; return err;
} }
......
...@@ -115,8 +115,8 @@ int ksmbd_vfs_xattr_stream_name(char *stream_name, char **xattr_stream_name, ...@@ -115,8 +115,8 @@ int ksmbd_vfs_xattr_stream_name(char *stream_name, char **xattr_stream_name,
int ksmbd_vfs_remove_xattr(struct mnt_idmap *idmap, int ksmbd_vfs_remove_xattr(struct mnt_idmap *idmap,
const struct path *path, char *attr_name); const struct path *path, char *attr_name);
int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name, int ksmbd_vfs_kern_path_locked(struct ksmbd_work *work, char *name,
unsigned int flags, struct path *path, unsigned int flags, struct path *parent_path,
bool caseless); struct path *path, bool caseless);
struct dentry *ksmbd_vfs_kern_path_create(struct ksmbd_work *work, struct dentry *ksmbd_vfs_kern_path_create(struct ksmbd_work *work,
const char *name, const char *name,
unsigned int flags, unsigned int flags,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment