Commit 2e450920 authored by Namjae Jeon's avatar Namjae Jeon Committed by Steve French

ksmbd: move oplock handling after unlock parent dir

ksmbd should process secound parallel smb2 create request during waiting
oplock break ack. parent lock range that is too large in smb2_open() causes
smb2_open() to be serialized. Move the oplock handling to the bottom of
smb2_open() and make it called after parent unlock. This fixes the failure
of smb2.lease.breaking1 testcase.
Signed-off-by: default avatarNamjae Jeon <linkinjeon@kernel.org>
Signed-off-by: default avatarSteve French <stfrench@microsoft.com>
parent 4274a9dc
...@@ -2691,7 +2691,7 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2691,7 +2691,7 @@ int smb2_open(struct ksmbd_work *work)
*(char *)req->Buffer == '\\') { *(char *)req->Buffer == '\\') {
pr_err("not allow directory name included leading slash\n"); pr_err("not allow directory name included leading slash\n");
rc = -EINVAL; rc = -EINVAL;
goto err_out1; goto err_out2;
} }
name = smb2_get_name(req->Buffer, name = smb2_get_name(req->Buffer,
...@@ -2702,7 +2702,7 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2702,7 +2702,7 @@ int smb2_open(struct ksmbd_work *work)
if (rc != -ENOMEM) if (rc != -ENOMEM)
rc = -ENOENT; rc = -ENOENT;
name = NULL; name = NULL;
goto err_out1; goto err_out2;
} }
ksmbd_debug(SMB, "converted name = %s\n", name); ksmbd_debug(SMB, "converted name = %s\n", name);
...@@ -2710,28 +2710,28 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2710,28 +2710,28 @@ int smb2_open(struct ksmbd_work *work)
if (!test_share_config_flag(work->tcon->share_conf, if (!test_share_config_flag(work->tcon->share_conf,
KSMBD_SHARE_FLAG_STREAMS)) { KSMBD_SHARE_FLAG_STREAMS)) {
rc = -EBADF; rc = -EBADF;
goto err_out1; goto err_out2;
} }
rc = parse_stream_name(name, &stream_name, &s_type); rc = parse_stream_name(name, &stream_name, &s_type);
if (rc < 0) if (rc < 0)
goto err_out1; goto err_out2;
} }
rc = ksmbd_validate_filename(name); rc = ksmbd_validate_filename(name);
if (rc < 0) if (rc < 0)
goto err_out1; goto err_out2;
if (ksmbd_share_veto_filename(share, name)) { if (ksmbd_share_veto_filename(share, name)) {
rc = -ENOENT; rc = -ENOENT;
ksmbd_debug(SMB, "Reject open(), vetoed file: %s\n", ksmbd_debug(SMB, "Reject open(), vetoed file: %s\n",
name); name);
goto err_out1; goto err_out2;
} }
} else { } else {
name = kstrdup("", GFP_KERNEL); name = kstrdup("", GFP_KERNEL);
if (!name) { if (!name) {
rc = -ENOMEM; rc = -ENOMEM;
goto err_out1; goto err_out2;
} }
} }
...@@ -2744,14 +2744,14 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2744,14 +2744,14 @@ int smb2_open(struct ksmbd_work *work)
le32_to_cpu(req->ImpersonationLevel)); le32_to_cpu(req->ImpersonationLevel));
rc = -EIO; rc = -EIO;
rsp->hdr.Status = STATUS_BAD_IMPERSONATION_LEVEL; rsp->hdr.Status = STATUS_BAD_IMPERSONATION_LEVEL;
goto err_out1; goto err_out2;
} }
if (req->CreateOptions && !(req->CreateOptions & CREATE_OPTIONS_MASK_LE)) { if (req->CreateOptions && !(req->CreateOptions & CREATE_OPTIONS_MASK_LE)) {
pr_err("Invalid create options : 0x%x\n", pr_err("Invalid create options : 0x%x\n",
le32_to_cpu(req->CreateOptions)); le32_to_cpu(req->CreateOptions));
rc = -EINVAL; rc = -EINVAL;
goto err_out1; goto err_out2;
} else { } else {
if (req->CreateOptions & FILE_SEQUENTIAL_ONLY_LE && if (req->CreateOptions & FILE_SEQUENTIAL_ONLY_LE &&
req->CreateOptions & FILE_RANDOM_ACCESS_LE) req->CreateOptions & FILE_RANDOM_ACCESS_LE)
...@@ -2761,13 +2761,13 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2761,13 +2761,13 @@ int smb2_open(struct ksmbd_work *work)
(FILE_OPEN_BY_FILE_ID_LE | CREATE_TREE_CONNECTION | (FILE_OPEN_BY_FILE_ID_LE | CREATE_TREE_CONNECTION |
FILE_RESERVE_OPFILTER_LE)) { FILE_RESERVE_OPFILTER_LE)) {
rc = -EOPNOTSUPP; rc = -EOPNOTSUPP;
goto err_out1; goto err_out2;
} }
if (req->CreateOptions & FILE_DIRECTORY_FILE_LE) { if (req->CreateOptions & FILE_DIRECTORY_FILE_LE) {
if (req->CreateOptions & FILE_NON_DIRECTORY_FILE_LE) { if (req->CreateOptions & FILE_NON_DIRECTORY_FILE_LE) {
rc = -EINVAL; rc = -EINVAL;
goto err_out1; goto err_out2;
} else if (req->CreateOptions & FILE_NO_COMPRESSION_LE) { } else if (req->CreateOptions & FILE_NO_COMPRESSION_LE) {
req->CreateOptions = ~(FILE_NO_COMPRESSION_LE); req->CreateOptions = ~(FILE_NO_COMPRESSION_LE);
} }
...@@ -2779,21 +2779,21 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2779,21 +2779,21 @@ int smb2_open(struct ksmbd_work *work)
pr_err("Invalid create disposition : 0x%x\n", pr_err("Invalid create disposition : 0x%x\n",
le32_to_cpu(req->CreateDisposition)); le32_to_cpu(req->CreateDisposition));
rc = -EINVAL; rc = -EINVAL;
goto err_out1; goto err_out2;
} }
if (!(req->DesiredAccess & DESIRED_ACCESS_MASK)) { if (!(req->DesiredAccess & DESIRED_ACCESS_MASK)) {
pr_err("Invalid desired access : 0x%x\n", pr_err("Invalid desired access : 0x%x\n",
le32_to_cpu(req->DesiredAccess)); le32_to_cpu(req->DesiredAccess));
rc = -EACCES; rc = -EACCES;
goto err_out1; goto err_out2;
} }
if (req->FileAttributes && !(req->FileAttributes & FILE_ATTRIBUTE_MASK_LE)) { if (req->FileAttributes && !(req->FileAttributes & FILE_ATTRIBUTE_MASK_LE)) {
pr_err("Invalid file attribute : 0x%x\n", pr_err("Invalid file attribute : 0x%x\n",
le32_to_cpu(req->FileAttributes)); le32_to_cpu(req->FileAttributes));
rc = -EINVAL; rc = -EINVAL;
goto err_out1; goto err_out2;
} }
if (req->CreateContextsOffset) { if (req->CreateContextsOffset) {
...@@ -2801,19 +2801,19 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2801,19 +2801,19 @@ int smb2_open(struct ksmbd_work *work)
context = smb2_find_context_vals(req, SMB2_CREATE_EA_BUFFER, 4); context = smb2_find_context_vals(req, SMB2_CREATE_EA_BUFFER, 4);
if (IS_ERR(context)) { if (IS_ERR(context)) {
rc = PTR_ERR(context); rc = PTR_ERR(context);
goto err_out1; goto err_out2;
} else if (context) { } else if (context) {
ea_buf = (struct create_ea_buf_req *)context; ea_buf = (struct create_ea_buf_req *)context;
if (le16_to_cpu(context->DataOffset) + if (le16_to_cpu(context->DataOffset) +
le32_to_cpu(context->DataLength) < le32_to_cpu(context->DataLength) <
sizeof(struct create_ea_buf_req)) { sizeof(struct create_ea_buf_req)) {
rc = -EINVAL; rc = -EINVAL;
goto err_out1; goto err_out2;
} }
if (req->CreateOptions & FILE_NO_EA_KNOWLEDGE_LE) { if (req->CreateOptions & FILE_NO_EA_KNOWLEDGE_LE) {
rsp->hdr.Status = STATUS_ACCESS_DENIED; rsp->hdr.Status = STATUS_ACCESS_DENIED;
rc = -EACCES; rc = -EACCES;
goto err_out1; goto err_out2;
} }
} }
...@@ -2821,7 +2821,7 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2821,7 +2821,7 @@ int smb2_open(struct ksmbd_work *work)
SMB2_CREATE_QUERY_MAXIMAL_ACCESS_REQUEST, 4); SMB2_CREATE_QUERY_MAXIMAL_ACCESS_REQUEST, 4);
if (IS_ERR(context)) { if (IS_ERR(context)) {
rc = PTR_ERR(context); rc = PTR_ERR(context);
goto err_out1; goto err_out2;
} else if (context) { } else if (context) {
ksmbd_debug(SMB, ksmbd_debug(SMB,
"get query maximal access context\n"); "get query maximal access context\n");
...@@ -2832,11 +2832,11 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2832,11 +2832,11 @@ int smb2_open(struct ksmbd_work *work)
SMB2_CREATE_TIMEWARP_REQUEST, 4); SMB2_CREATE_TIMEWARP_REQUEST, 4);
if (IS_ERR(context)) { if (IS_ERR(context)) {
rc = PTR_ERR(context); rc = PTR_ERR(context);
goto err_out1; goto err_out2;
} else if (context) { } else if (context) {
ksmbd_debug(SMB, "get timewarp context\n"); ksmbd_debug(SMB, "get timewarp context\n");
rc = -EBADF; rc = -EBADF;
goto err_out1; goto err_out2;
} }
if (tcon->posix_extensions) { if (tcon->posix_extensions) {
...@@ -2844,7 +2844,7 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2844,7 +2844,7 @@ int smb2_open(struct ksmbd_work *work)
SMB2_CREATE_TAG_POSIX, 16); SMB2_CREATE_TAG_POSIX, 16);
if (IS_ERR(context)) { if (IS_ERR(context)) {
rc = PTR_ERR(context); rc = PTR_ERR(context);
goto err_out1; goto err_out2;
} else if (context) { } else if (context) {
struct create_posix *posix = struct create_posix *posix =
(struct create_posix *)context; (struct create_posix *)context;
...@@ -2852,7 +2852,7 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2852,7 +2852,7 @@ int smb2_open(struct ksmbd_work *work)
le32_to_cpu(context->DataLength) < le32_to_cpu(context->DataLength) <
sizeof(struct create_posix) - 4) { sizeof(struct create_posix) - 4) {
rc = -EINVAL; rc = -EINVAL;
goto err_out1; goto err_out2;
} }
ksmbd_debug(SMB, "get posix context\n"); ksmbd_debug(SMB, "get posix context\n");
...@@ -2864,7 +2864,7 @@ int smb2_open(struct ksmbd_work *work) ...@@ -2864,7 +2864,7 @@ int smb2_open(struct ksmbd_work *work)
if (ksmbd_override_fsids(work)) { if (ksmbd_override_fsids(work)) {
rc = -ENOMEM; rc = -ENOMEM;
goto err_out1; goto err_out2;
} }
rc = ksmbd_vfs_kern_path_locked(work, name, LOOKUP_NO_SYMLINKS, rc = ksmbd_vfs_kern_path_locked(work, name, LOOKUP_NO_SYMLINKS,
...@@ -3177,11 +3177,6 @@ int smb2_open(struct ksmbd_work *work) ...@@ -3177,11 +3177,6 @@ int smb2_open(struct ksmbd_work *work)
fp->attrib_only = !(req->DesiredAccess & ~(FILE_READ_ATTRIBUTES_LE | fp->attrib_only = !(req->DesiredAccess & ~(FILE_READ_ATTRIBUTES_LE |
FILE_WRITE_ATTRIBUTES_LE | FILE_SYNCHRONIZE_LE)); FILE_WRITE_ATTRIBUTES_LE | FILE_SYNCHRONIZE_LE));
if (!S_ISDIR(file_inode(filp)->i_mode) && open_flags & O_TRUNC &&
!fp->attrib_only && !stream_name) {
smb_break_all_oplock(work, fp);
need_truncate = 1;
}
/* fp should be searchable through ksmbd_inode.m_fp_list /* fp should be searchable through ksmbd_inode.m_fp_list
* after daccess, saccess, attrib_only, and stream are * after daccess, saccess, attrib_only, and stream are
...@@ -3197,13 +3192,39 @@ int smb2_open(struct ksmbd_work *work) ...@@ -3197,13 +3192,39 @@ int smb2_open(struct ksmbd_work *work)
goto err_out; goto err_out;
} }
rc = ksmbd_vfs_getattr(&path, &stat);
if (rc)
goto err_out;
if (stat.result_mask & STATX_BTIME)
fp->create_time = ksmbd_UnixTimeToNT(stat.btime);
else
fp->create_time = ksmbd_UnixTimeToNT(stat.ctime);
if (req->FileAttributes || fp->f_ci->m_fattr == 0)
fp->f_ci->m_fattr =
cpu_to_le32(smb2_get_dos_mode(&stat, le32_to_cpu(req->FileAttributes)));
if (!created)
smb2_update_xattrs(tcon, &path, fp);
else
smb2_new_xattrs(tcon, &path, fp);
if (file_present || created)
ksmbd_vfs_kern_path_unlock(&parent_path, &path);
if (!S_ISDIR(file_inode(filp)->i_mode) && open_flags & O_TRUNC &&
!fp->attrib_only && !stream_name) {
smb_break_all_oplock(work, fp);
need_truncate = 1;
}
share_ret = ksmbd_smb_check_shared_mode(fp->filp, fp); share_ret = ksmbd_smb_check_shared_mode(fp->filp, fp);
if (!test_share_config_flag(work->tcon->share_conf, KSMBD_SHARE_FLAG_OPLOCKS) || if (!test_share_config_flag(work->tcon->share_conf, KSMBD_SHARE_FLAG_OPLOCKS) ||
(req_op_level == SMB2_OPLOCK_LEVEL_LEASE && (req_op_level == SMB2_OPLOCK_LEVEL_LEASE &&
!(conn->vals->capabilities & SMB2_GLOBAL_CAP_LEASING))) { !(conn->vals->capabilities & SMB2_GLOBAL_CAP_LEASING))) {
if (share_ret < 0 && !S_ISDIR(file_inode(fp->filp)->i_mode)) { if (share_ret < 0 && !S_ISDIR(file_inode(fp->filp)->i_mode)) {
rc = share_ret; rc = share_ret;
goto err_out; goto err_out1;
} }
} else { } else {
if (req_op_level == SMB2_OPLOCK_LEVEL_LEASE) { if (req_op_level == SMB2_OPLOCK_LEVEL_LEASE) {
...@@ -3213,7 +3234,7 @@ int smb2_open(struct ksmbd_work *work) ...@@ -3213,7 +3234,7 @@ int smb2_open(struct ksmbd_work *work)
name, req_op_level, lc->req_state); name, req_op_level, lc->req_state);
rc = find_same_lease_key(sess, fp->f_ci, lc); rc = find_same_lease_key(sess, fp->f_ci, lc);
if (rc) if (rc)
goto err_out; goto err_out1;
} else if (open_flags == O_RDONLY && } else if (open_flags == O_RDONLY &&
(req_op_level == SMB2_OPLOCK_LEVEL_BATCH || (req_op_level == SMB2_OPLOCK_LEVEL_BATCH ||
req_op_level == SMB2_OPLOCK_LEVEL_EXCLUSIVE)) req_op_level == SMB2_OPLOCK_LEVEL_EXCLUSIVE))
...@@ -3224,12 +3245,18 @@ int smb2_open(struct ksmbd_work *work) ...@@ -3224,12 +3245,18 @@ int smb2_open(struct ksmbd_work *work)
le32_to_cpu(req->hdr.Id.SyncId.TreeId), le32_to_cpu(req->hdr.Id.SyncId.TreeId),
lc, share_ret); lc, share_ret);
if (rc < 0) if (rc < 0)
goto err_out; goto err_out1;
} }
if (req->CreateOptions & FILE_DELETE_ON_CLOSE_LE) if (req->CreateOptions & FILE_DELETE_ON_CLOSE_LE)
ksmbd_fd_set_delete_on_close(fp, file_info); ksmbd_fd_set_delete_on_close(fp, file_info);
if (need_truncate) {
rc = smb2_create_truncate(&fp->filp->f_path);
if (rc)
goto err_out1;
}
if (req->CreateContextsOffset) { if (req->CreateContextsOffset) {
struct create_alloc_size_req *az_req; struct create_alloc_size_req *az_req;
...@@ -3237,7 +3264,7 @@ int smb2_open(struct ksmbd_work *work) ...@@ -3237,7 +3264,7 @@ int smb2_open(struct ksmbd_work *work)
SMB2_CREATE_ALLOCATION_SIZE, 4); SMB2_CREATE_ALLOCATION_SIZE, 4);
if (IS_ERR(az_req)) { if (IS_ERR(az_req)) {
rc = PTR_ERR(az_req); rc = PTR_ERR(az_req);
goto err_out; goto err_out1;
} else if (az_req) { } else if (az_req) {
loff_t alloc_size; loff_t alloc_size;
int err; int err;
...@@ -3246,7 +3273,7 @@ int smb2_open(struct ksmbd_work *work) ...@@ -3246,7 +3273,7 @@ int smb2_open(struct ksmbd_work *work)
le32_to_cpu(az_req->ccontext.DataLength) < le32_to_cpu(az_req->ccontext.DataLength) <
sizeof(struct create_alloc_size_req)) { sizeof(struct create_alloc_size_req)) {
rc = -EINVAL; rc = -EINVAL;
goto err_out; goto err_out1;
} }
alloc_size = le64_to_cpu(az_req->AllocationSize); alloc_size = le64_to_cpu(az_req->AllocationSize);
ksmbd_debug(SMB, ksmbd_debug(SMB,
...@@ -3264,30 +3291,13 @@ int smb2_open(struct ksmbd_work *work) ...@@ -3264,30 +3291,13 @@ int smb2_open(struct ksmbd_work *work)
context = smb2_find_context_vals(req, SMB2_CREATE_QUERY_ON_DISK_ID, 4); context = smb2_find_context_vals(req, SMB2_CREATE_QUERY_ON_DISK_ID, 4);
if (IS_ERR(context)) { if (IS_ERR(context)) {
rc = PTR_ERR(context); rc = PTR_ERR(context);
goto err_out; goto err_out1;
} else if (context) { } else if (context) {
ksmbd_debug(SMB, "get query on disk id context\n"); ksmbd_debug(SMB, "get query on disk id context\n");
query_disk_id = 1; query_disk_id = 1;
} }
} }
rc = ksmbd_vfs_getattr(&path, &stat);
if (rc)
goto err_out;
if (stat.result_mask & STATX_BTIME)
fp->create_time = ksmbd_UnixTimeToNT(stat.btime);
else
fp->create_time = ksmbd_UnixTimeToNT(stat.ctime);
if (req->FileAttributes || fp->f_ci->m_fattr == 0)
fp->f_ci->m_fattr =
cpu_to_le32(smb2_get_dos_mode(&stat, le32_to_cpu(req->FileAttributes)));
if (!created)
smb2_update_xattrs(tcon, &path, fp);
else
smb2_new_xattrs(tcon, &path, fp);
memcpy(fp->client_guid, conn->ClientGUID, SMB2_CLIENT_GUID_SIZE); memcpy(fp->client_guid, conn->ClientGUID, SMB2_CLIENT_GUID_SIZE);
rsp->StructureSize = cpu_to_le16(89); rsp->StructureSize = cpu_to_le16(89);
...@@ -3394,14 +3404,13 @@ int smb2_open(struct ksmbd_work *work) ...@@ -3394,14 +3404,13 @@ int smb2_open(struct ksmbd_work *work)
} }
err_out: err_out:
if (file_present || created) if (rc && (file_present || created))
ksmbd_vfs_kern_path_unlock(&parent_path, &path); ksmbd_vfs_kern_path_unlock(&parent_path, &path);
if (fp && need_truncate)
rc = smb2_create_truncate(&fp->filp->f_path);
ksmbd_revert_fsids(work);
err_out1: err_out1:
ksmbd_revert_fsids(work);
err_out2:
if (!rc) { if (!rc) {
ksmbd_update_fstate(&work->sess->file_table, fp, FP_INITED); ksmbd_update_fstate(&work->sess->file_table, fp, FP_INITED);
rc = ksmbd_iov_pin_rsp(work, (void *)rsp, iov_len); rc = ksmbd_iov_pin_rsp(work, (void *)rsp, iov_len);
......
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