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: Namjae Jeon <linkinjeon@kernel.org>
Signed-off-by: Steve French <stfrench@microsoft.com>
This commit is contained in:
Namjae Jeon 2023-11-20 23:39:39 +09:00 committed by Steve French
parent 4274a9dc6a
commit 2e450920d5

View File

@ -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)
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)
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)
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)
(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)
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)
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)
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)
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)
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)
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)
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)
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,80 +3192,6 @@ int smb2_open(struct ksmbd_work *work)
goto err_out; goto err_out;
} }
share_ret = ksmbd_smb_check_shared_mode(fp->filp, fp);
if (!test_share_config_flag(work->tcon->share_conf, KSMBD_SHARE_FLAG_OPLOCKS) ||
(req_op_level == SMB2_OPLOCK_LEVEL_LEASE &&
!(conn->vals->capabilities & SMB2_GLOBAL_CAP_LEASING))) {
if (share_ret < 0 && !S_ISDIR(file_inode(fp->filp)->i_mode)) {
rc = share_ret;
goto err_out;
}
} else {
if (req_op_level == SMB2_OPLOCK_LEVEL_LEASE) {
req_op_level = smb2_map_lease_to_oplock(lc->req_state);
ksmbd_debug(SMB,
"lease req for(%s) req oplock state 0x%x, lease state 0x%x\n",
name, req_op_level, lc->req_state);
rc = find_same_lease_key(sess, fp->f_ci, lc);
if (rc)
goto err_out;
} else if (open_flags == O_RDONLY &&
(req_op_level == SMB2_OPLOCK_LEVEL_BATCH ||
req_op_level == SMB2_OPLOCK_LEVEL_EXCLUSIVE))
req_op_level = SMB2_OPLOCK_LEVEL_II;
rc = smb_grant_oplock(work, req_op_level,
fp->persistent_id, fp,
le32_to_cpu(req->hdr.Id.SyncId.TreeId),
lc, share_ret);
if (rc < 0)
goto err_out;
}
if (req->CreateOptions & FILE_DELETE_ON_CLOSE_LE)
ksmbd_fd_set_delete_on_close(fp, file_info);
if (req->CreateContextsOffset) {
struct create_alloc_size_req *az_req;
az_req = (struct create_alloc_size_req *)smb2_find_context_vals(req,
SMB2_CREATE_ALLOCATION_SIZE, 4);
if (IS_ERR(az_req)) {
rc = PTR_ERR(az_req);
goto err_out;
} else if (az_req) {
loff_t alloc_size;
int err;
if (le16_to_cpu(az_req->ccontext.DataOffset) +
le32_to_cpu(az_req->ccontext.DataLength) <
sizeof(struct create_alloc_size_req)) {
rc = -EINVAL;
goto err_out;
}
alloc_size = le64_to_cpu(az_req->AllocationSize);
ksmbd_debug(SMB,
"request smb2 create allocate size : %llu\n",
alloc_size);
smb_break_all_levII_oplock(work, fp, 1);
err = vfs_fallocate(fp->filp, FALLOC_FL_KEEP_SIZE, 0,
alloc_size);
if (err < 0)
ksmbd_debug(SMB,
"vfs_fallocate is failed : %d\n",
err);
}
context = smb2_find_context_vals(req, SMB2_CREATE_QUERY_ON_DISK_ID, 4);
if (IS_ERR(context)) {
rc = PTR_ERR(context);
goto err_out;
} else if (context) {
ksmbd_debug(SMB, "get query on disk id context\n");
query_disk_id = 1;
}
}
rc = ksmbd_vfs_getattr(&path, &stat); rc = ksmbd_vfs_getattr(&path, &stat);
if (rc) if (rc)
goto err_out; goto err_out;
@ -3288,6 +3209,95 @@ int smb2_open(struct ksmbd_work *work)
else else
smb2_new_xattrs(tcon, &path, fp); 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);
if (!test_share_config_flag(work->tcon->share_conf, KSMBD_SHARE_FLAG_OPLOCKS) ||
(req_op_level == SMB2_OPLOCK_LEVEL_LEASE &&
!(conn->vals->capabilities & SMB2_GLOBAL_CAP_LEASING))) {
if (share_ret < 0 && !S_ISDIR(file_inode(fp->filp)->i_mode)) {
rc = share_ret;
goto err_out1;
}
} else {
if (req_op_level == SMB2_OPLOCK_LEVEL_LEASE) {
req_op_level = smb2_map_lease_to_oplock(lc->req_state);
ksmbd_debug(SMB,
"lease req for(%s) req oplock state 0x%x, lease state 0x%x\n",
name, req_op_level, lc->req_state);
rc = find_same_lease_key(sess, fp->f_ci, lc);
if (rc)
goto err_out1;
} else if (open_flags == O_RDONLY &&
(req_op_level == SMB2_OPLOCK_LEVEL_BATCH ||
req_op_level == SMB2_OPLOCK_LEVEL_EXCLUSIVE))
req_op_level = SMB2_OPLOCK_LEVEL_II;
rc = smb_grant_oplock(work, req_op_level,
fp->persistent_id, fp,
le32_to_cpu(req->hdr.Id.SyncId.TreeId),
lc, share_ret);
if (rc < 0)
goto err_out1;
}
if (req->CreateOptions & FILE_DELETE_ON_CLOSE_LE)
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) {
struct create_alloc_size_req *az_req;
az_req = (struct create_alloc_size_req *)smb2_find_context_vals(req,
SMB2_CREATE_ALLOCATION_SIZE, 4);
if (IS_ERR(az_req)) {
rc = PTR_ERR(az_req);
goto err_out1;
} else if (az_req) {
loff_t alloc_size;
int err;
if (le16_to_cpu(az_req->ccontext.DataOffset) +
le32_to_cpu(az_req->ccontext.DataLength) <
sizeof(struct create_alloc_size_req)) {
rc = -EINVAL;
goto err_out1;
}
alloc_size = le64_to_cpu(az_req->AllocationSize);
ksmbd_debug(SMB,
"request smb2 create allocate size : %llu\n",
alloc_size);
smb_break_all_levII_oplock(work, fp, 1);
err = vfs_fallocate(fp->filp, FALLOC_FL_KEEP_SIZE, 0,
alloc_size);
if (err < 0)
ksmbd_debug(SMB,
"vfs_fallocate is failed : %d\n",
err);
}
context = smb2_find_context_vals(req, SMB2_CREATE_QUERY_ON_DISK_ID, 4);
if (IS_ERR(context)) {
rc = PTR_ERR(context);
goto err_out1;
} else if (context) {
ksmbd_debug(SMB, "get query on disk id context\n");
query_disk_id = 1;
}
}
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)
} }
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);