1
0
mirror of https://github.com/samba-team/samba.git synced 2024-12-22 13:34:15 +03:00

s4: torture: Add an async SMB2_OP_FLUSH + SMB2_OP_FLUSH test to smb2.compound_async.

Shows we fail sending an SMB2_OP_FLUSH + SMB2_OP_FLUSH
compound if we immediately close the file afterward.

Internally the flushes go async and we free the req, then
we process the close. When the flushes complete they try to access
already freed data.

Extra test which will allow me to test when the final
component (flush) of the compound goes async and returns
NT_STATUS_PENDING.

Add knownfail.

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

Signed-off-by: Jeremy Allison <jra@samba.org>
Reviewed-by: Ralph Boehme <slow@samba.org>
This commit is contained in:
Jeremy Allison 2022-10-20 14:22:25 -07:00 committed by Ralph Boehme
parent 17a110c1b5
commit 6f149dfd9d
2 changed files with 116 additions and 0 deletions

View File

@ -1 +1,2 @@
^samba3.smb2.compound_async.flush_close\(fileserver\)
^samba3.smb2.compound_async.flush_flush\(fileserver\)

View File

@ -2161,6 +2161,119 @@ static bool test_compound_async_flush_close(struct torture_context *tctx,
return ret;
}
static bool test_compound_async_flush_flush(struct torture_context *tctx,
struct smb2_tree *tree)
{
struct smb2_handle fhandle = { .data = { 0, 0 } };
struct smb2_handle relhandle = { .data = { UINT64_MAX, UINT64_MAX } };
struct smb2_flush fl1;
struct smb2_flush fl2;
const char *fname = "compound_async_flush_flush";
struct smb2_request *req[2];
NTSTATUS status;
bool ret = false;
/* Start clean. */
smb2_util_unlink(tree, fname);
/* Create a file. */
status = torture_smb2_testfile_access(tree,
fname,
&fhandle,
SEC_RIGHTS_FILE_ALL);
CHECK_STATUS(status, NT_STATUS_OK);
/* Now do a compound flush + flush handle. */
smb2_transport_compound_start(tree->session->transport, 2);
ZERO_STRUCT(fl1);
fl1.in.file.handle = fhandle;
req[0] = smb2_flush_send(tree, &fl1);
torture_assert_not_null_goto(tctx, req[0], ret, done,
"smb2_flush_send (1) failed\n");
smb2_transport_compound_set_related(tree->session->transport, true);
ZERO_STRUCT(fl2);
fl2.in.file.handle = relhandle;
req[1] = smb2_flush_send(tree, &fl2);
torture_assert_not_null_goto(tctx, req[1], ret, done,
"smb2_flush_send (2) failed\n");
status = smb2_flush_recv(req[0], &fl1);
/*
* On Windows, this flush will usually
* succeed as we have nothing to flush,
* so allow NT_STATUS_OK. Once bug #15172
* is fixed Samba will do the flush synchronously
* so allow NT_STATUS_OK.
*/
if (!NT_STATUS_IS_OK(status)) {
/*
* If we didn't get NT_STATUS_OK, we *must*
* get NT_STATUS_INTERNAL_ERROR if the flush
* goes async.
*
* For pre-bugfix #15172 Samba, the flush goes async and
* we should get NT_STATUS_INTERNAL_ERROR.
*/
torture_assert_ntstatus_equal_goto(tctx,
status,
NT_STATUS_INTERNAL_ERROR,
ret,
done,
"smb2_flush_recv (1) didn't return "
"NT_STATUS_INTERNAL_ERROR.\n");
}
/*
* If the flush is the last entry in a compound,
* it should always succeed even if it goes async.
*/
status = smb2_flush_recv(req[1], &fl2);
torture_assert_ntstatus_ok_goto(tctx, status, ret, done,
"smb2_flush_recv (2) failed.");
status = smb2_util_close(tree, fhandle);
torture_assert_ntstatus_ok_goto(tctx, status, ret, done,
"smb2_util_close failed.");
ZERO_STRUCT(fhandle);
/*
* Do several more operations on the tree, spaced
* out by 1 sec sleeps to make sure the server didn't
* crash on the close. The sleeps are required to
* make test test for a crash reliable, as we ensure
* the pthread fsync internally finishes and accesses
* freed memory. Without them the test occassionally
* passes as we disconnect before the pthread fsync
* finishes.
*/
status = smb2_util_unlink(tree, fname);
CHECK_STATUS(status, NT_STATUS_OK);
sleep(1);
status = smb2_util_unlink(tree, fname);
CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
sleep(1);
status = smb2_util_unlink(tree, fname);
CHECK_STATUS(status, NT_STATUS_OBJECT_NAME_NOT_FOUND);
ret = true;
done:
if (fhandle.data[0] != 0) {
smb2_util_close(tree, fhandle);
}
smb2_util_unlink(tree, fname);
return ret;
}
struct torture_suite *torture_smb2_compound_init(TALLOC_CTX *ctx)
{
struct torture_suite *suite = torture_suite_create(ctx, "compound");
@ -2219,6 +2332,8 @@ struct torture_suite *torture_smb2_compound_async_init(TALLOC_CTX *ctx)
torture_suite_add_1smb2_test(suite, "flush_close",
test_compound_async_flush_close);
torture_suite_add_1smb2_test(suite, "flush_flush",
test_compound_async_flush_flush);
suite->description = talloc_strdup(suite, "SMB2-COMPOUND-ASYNC tests");