1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-22 22:04:08 +03:00

s3:blocking: fix the fsp->blocked_smb1_lock_reqs handling

A new request is first checks against all pending
requests before checking the already granted locks.

Before we retried the lock array of another request
(the first in the list), but then finished current request,
which is wrong.

BUG: https://bugzilla.samba.org/show_bug.cgi?id=14113

Signed-off-by: Stefan Metzmacher <metze@samba.org>
Reviewed-by: Volker Lendecke <vl@samba.org>
This commit is contained in:
Stefan Metzmacher 2019-08-16 14:55:13 +02:00
parent 997548a5f1
commit 3123271062
2 changed files with 125 additions and 13 deletions

View File

@ -1,4 +0,0 @@
^samba3.raw.lock.multilock3.*nt4_dc
^samba3.raw.lock.multilock4.*nt4_dc
^samba3.raw.lock.multilock5.*nt4_dc
^samba3.raw.lock.multilock6.*nt4_dc

View File

@ -116,6 +116,14 @@ static void smbd_smb1_do_locks_try(struct tevent_req *req);
static void smbd_smb1_do_locks_retry(struct tevent_req *subreq);
static void smbd_smb1_blocked_locks_cleanup(
struct tevent_req *req, enum tevent_req_state req_state);
static NTSTATUS smbd_smb1_do_locks_check(
struct files_struct *fsp,
enum brl_flavour lock_flav,
uint16_t num_locks,
struct smbd_lock_element *locks,
uint16_t *blocker_idx,
struct server_id *blocking_pid,
uint64_t *blocking_smblctx);
static void smbd_smb1_do_locks_setup_timeout(
struct smbd_smb1_do_locks_state *state,
@ -264,7 +272,7 @@ struct tevent_req *smbd_smb1_do_locks_send(
return tevent_req_post(req, ev);
}
status = smbd_do_locks_try(
status = smbd_smb1_do_locks_check(
state->fsp,
state->lock_flav,
state->num_locks,
@ -390,15 +398,123 @@ static void smbd_smb1_blocked_locks_cleanup(
fsp, blocked, struct tevent_req *, num_blocked-1);
}
static NTSTATUS smbd_smb1_do_locks_check_blocked(
uint16_t num_blocked,
struct smbd_lock_element *blocked,
uint16_t num_locks,
struct smbd_lock_element *locks,
uint16_t *blocker_idx,
uint64_t *blocking_smblctx)
{
uint16_t li;
for (li=0; li < num_locks; li++) {
struct smbd_lock_element *l = &locks[li];
uint16_t bi;
bool valid;
valid = byte_range_valid(l->offset, l->count);
if (!valid) {
return NT_STATUS_INVALID_LOCK_RANGE;
}
for (bi = 0; bi < num_blocked; bi++) {
struct smbd_lock_element *b = &blocked[li];
bool overlap;
/* Read locks never conflict. */
if (l->brltype == READ_LOCK && b->brltype == READ_LOCK) {
continue;
}
overlap = byte_range_overlap(l->offset,
l->count,
b->offset,
b->count);
if (!overlap) {
continue;
}
*blocker_idx = li;
*blocking_smblctx = b->smblctx;
return NT_STATUS_LOCK_NOT_GRANTED;
}
}
return NT_STATUS_OK;
}
static NTSTATUS smbd_smb1_do_locks_check(
struct files_struct *fsp,
enum brl_flavour lock_flav,
uint16_t num_locks,
struct smbd_lock_element *locks,
uint16_t *blocker_idx,
struct server_id *blocking_pid,
uint64_t *blocking_smblctx)
{
struct tevent_req **blocked = fsp->blocked_smb1_lock_reqs;
size_t num_blocked = talloc_array_length(blocked);
NTSTATUS status;
size_t bi;
/*
* We check the pending/blocked requests
* from the oldest to the youngest request.
*
* Note due to the retry logic the current request
* might already be in the list.
*/
for (bi = 0; bi < num_blocked; bi++) {
struct smbd_smb1_do_locks_state *blocked_state =
tevent_req_data(blocked[bi],
struct smbd_smb1_do_locks_state);
if (blocked_state->locks == locks) {
SMB_ASSERT(blocked_state->num_locks == num_locks);
SMB_ASSERT(blocked_state->lock_flav == lock_flav);
/*
* We found ourself...
*/
break;
}
status = smbd_smb1_do_locks_check_blocked(
blocked_state->num_locks,
blocked_state->locks,
num_locks,
locks,
blocker_idx,
blocking_smblctx);
if (!NT_STATUS_IS_OK(status)) {
*blocking_pid = messaging_server_id(
fsp->conn->sconn->msg_ctx);
return status;
}
}
status = smbd_do_locks_try(
fsp,
lock_flav,
num_locks,
locks,
blocker_idx,
blocking_pid,
blocking_smblctx);
if (!NT_STATUS_IS_OK(status)) {
return status;
}
return NT_STATUS_OK;
}
static void smbd_smb1_do_locks_try(struct tevent_req *req)
{
struct smbd_smb1_do_locks_state *state = tevent_req_data(
req, struct smbd_smb1_do_locks_state);
struct files_struct *fsp = state->fsp;
struct tevent_req **blocked = fsp->blocked_smb1_lock_reqs;
struct tevent_req *retry_req = blocked[0];
struct smbd_smb1_do_locks_state *retry_state = tevent_req_data(
retry_req, struct smbd_smb1_do_locks_state);
struct share_mode_lock *lck;
struct timeval endtime = { 0 };
struct server_id blocking_pid = { 0 };
@ -414,11 +530,11 @@ static void smbd_smb1_do_locks_try(struct tevent_req *req)
return;
}
status = smbd_do_locks_try(
status = smbd_smb1_do_locks_check(
fsp,
retry_state->lock_flav,
retry_state->num_locks,
retry_state->locks,
state->lock_flav,
state->num_locks,
state->locks,
&state->blocker,
&blocking_pid,
&blocking_smblctx);