mirror of
https://github.com/samba-team/samba.git
synced 2025-01-26 10:04:02 +03:00
s3: Properly handle shutdown with the _send/_recv based aio
Signed-off-by: Jeremy Allison <jra@samba.org>
This commit is contained in:
parent
aff6956d35
commit
f5dc8837d9
@ -249,6 +249,8 @@ typedef struct files_struct {
|
|||||||
/* if not NULL, means this is a print file */
|
/* if not NULL, means this is a print file */
|
||||||
struct print_file_data *print_file;
|
struct print_file_data *print_file;
|
||||||
|
|
||||||
|
unsigned num_aio_requests;
|
||||||
|
struct tevent_req **aio_requests;
|
||||||
} files_struct;
|
} files_struct;
|
||||||
|
|
||||||
struct vuid_cache_entry {
|
struct vuid_cache_entry {
|
||||||
|
@ -110,7 +110,6 @@ static int handle_aio_smb2_write_complete(struct aio_extra *aio_ex, int errcode)
|
|||||||
|
|
||||||
static int aio_extra_destructor(struct aio_extra *aio_ex)
|
static int aio_extra_destructor(struct aio_extra *aio_ex)
|
||||||
{
|
{
|
||||||
DLIST_REMOVE(aio_list_head, aio_ex);
|
|
||||||
outstanding_aio_calls--;
|
outstanding_aio_calls--;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -141,13 +140,70 @@ static struct aio_extra *create_aio_extra(TALLOC_CTX *mem_ctx,
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
DLIST_ADD(aio_list_head, aio_ex);
|
|
||||||
talloc_set_destructor(aio_ex, aio_extra_destructor);
|
talloc_set_destructor(aio_ex, aio_extra_destructor);
|
||||||
aio_ex->fsp = fsp;
|
aio_ex->fsp = fsp;
|
||||||
outstanding_aio_calls++;
|
outstanding_aio_calls++;
|
||||||
return aio_ex;
|
return aio_ex;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct aio_req_fsp_link {
|
||||||
|
files_struct *fsp;
|
||||||
|
struct tevent_req *req;
|
||||||
|
};
|
||||||
|
|
||||||
|
static int aio_del_req_from_fsp(struct aio_req_fsp_link *lnk)
|
||||||
|
{
|
||||||
|
unsigned i;
|
||||||
|
files_struct *fsp = lnk->fsp;
|
||||||
|
struct tevent_req *req = lnk->req;
|
||||||
|
|
||||||
|
for (i=0; i<fsp->num_aio_requests; i++) {
|
||||||
|
if (fsp->aio_requests[i] == req) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (i == fsp->num_aio_requests) {
|
||||||
|
DEBUG(1, ("req %p not found in fsp %p\n", req, fsp));
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
fsp->num_aio_requests -= 1;
|
||||||
|
fsp->aio_requests[i] = fsp->aio_requests[fsp->num_aio_requests];
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static bool aio_add_req_to_fsp(files_struct *fsp, struct tevent_req *req)
|
||||||
|
{
|
||||||
|
size_t array_len;
|
||||||
|
struct aio_req_fsp_link *lnk;
|
||||||
|
|
||||||
|
lnk = talloc(req, struct aio_req_fsp_link);
|
||||||
|
if (lnk == NULL) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
array_len = talloc_array_length(fsp->aio_requests);
|
||||||
|
if (array_len <= fsp->num_aio_requests) {
|
||||||
|
struct tevent_req **tmp;
|
||||||
|
|
||||||
|
tmp = talloc_realloc(
|
||||||
|
fsp, fsp->aio_requests, struct tevent_req *,
|
||||||
|
fsp->num_aio_requests+1);
|
||||||
|
if (tmp == NULL) {
|
||||||
|
TALLOC_FREE(lnk);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
fsp->aio_requests = tmp;
|
||||||
|
}
|
||||||
|
fsp->aio_requests[fsp->num_aio_requests] = req;
|
||||||
|
fsp->num_aio_requests += 1;
|
||||||
|
|
||||||
|
lnk->fsp = fsp;
|
||||||
|
lnk->req = req;
|
||||||
|
talloc_set_destructor(lnk, aio_del_req_from_fsp);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
static void aio_pread_smb1_done(struct tevent_req *req);
|
static void aio_pread_smb1_done(struct tevent_req *req);
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@ -243,6 +299,13 @@ NTSTATUS schedule_aio_read_and_X(connection_struct *conn,
|
|||||||
}
|
}
|
||||||
tevent_req_set_callback(req, aio_pread_smb1_done, aio_ex);
|
tevent_req_set_callback(req, aio_pread_smb1_done, aio_ex);
|
||||||
|
|
||||||
|
if (!aio_add_req_to_fsp(fsp, req)) {
|
||||||
|
DEBUG(1, ("Could not add req to fsp\n"));
|
||||||
|
SMB_VFS_STRICT_UNLOCK(conn, fsp, &aio_ex->lock);
|
||||||
|
TALLOC_FREE(aio_ex);
|
||||||
|
return NT_STATUS_RETRY;
|
||||||
|
}
|
||||||
|
|
||||||
aio_ex->smbreq = talloc_move(aio_ex, &smbreq);
|
aio_ex->smbreq = talloc_move(aio_ex, &smbreq);
|
||||||
|
|
||||||
DEBUG(10,("schedule_aio_read_and_X: scheduled aio_read for file %s, "
|
DEBUG(10,("schedule_aio_read_and_X: scheduled aio_read for file %s, "
|
||||||
@ -420,6 +483,13 @@ NTSTATUS schedule_aio_write_and_X(connection_struct *conn,
|
|||||||
}
|
}
|
||||||
tevent_req_set_callback(req, aio_pwrite_smb1_done, aio_ex);
|
tevent_req_set_callback(req, aio_pwrite_smb1_done, aio_ex);
|
||||||
|
|
||||||
|
if (!aio_add_req_to_fsp(fsp, req)) {
|
||||||
|
DEBUG(1, ("Could not add req to fsp\n"));
|
||||||
|
SMB_VFS_STRICT_UNLOCK(conn, fsp, &aio_ex->lock);
|
||||||
|
TALLOC_FREE(aio_ex);
|
||||||
|
return NT_STATUS_RETRY;
|
||||||
|
}
|
||||||
|
|
||||||
aio_ex->smbreq = talloc_move(aio_ex, &smbreq);
|
aio_ex->smbreq = talloc_move(aio_ex, &smbreq);
|
||||||
|
|
||||||
/* This should actually be improved to span the write. */
|
/* This should actually be improved to span the write. */
|
||||||
@ -579,11 +649,12 @@ bool cancel_smb2_aio(struct smb_request *smbreq)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = SMB_VFS_AIO_CANCEL(aio_ex->fsp, &aio_ex->acb);
|
/*
|
||||||
if (ret != AIO_CANCELED) {
|
* We let the aio request run. Setting fsp to NULL has the
|
||||||
return false;
|
* effect that the _done routines don't send anything out.
|
||||||
}
|
*/
|
||||||
|
|
||||||
|
aio_ex->fsp = NULL;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -678,6 +749,13 @@ NTSTATUS schedule_smb2_aio_read(connection_struct *conn,
|
|||||||
}
|
}
|
||||||
tevent_req_set_callback(req, aio_pread_smb2_done, aio_ex);
|
tevent_req_set_callback(req, aio_pread_smb2_done, aio_ex);
|
||||||
|
|
||||||
|
if (!aio_add_req_to_fsp(fsp, req)) {
|
||||||
|
DEBUG(1, ("Could not add req to fsp\n"));
|
||||||
|
SMB_VFS_STRICT_UNLOCK(conn, fsp, &aio_ex->lock);
|
||||||
|
TALLOC_FREE(aio_ex);
|
||||||
|
return NT_STATUS_RETRY;
|
||||||
|
}
|
||||||
|
|
||||||
/* We don't need talloc_move here as both aio_ex and
|
/* We don't need talloc_move here as both aio_ex and
|
||||||
* smbreq are children of smbreq->smb2req. */
|
* smbreq are children of smbreq->smb2req. */
|
||||||
aio_ex->smbreq = smbreq;
|
aio_ex->smbreq = smbreq;
|
||||||
@ -829,6 +907,13 @@ NTSTATUS schedule_aio_smb2_write(connection_struct *conn,
|
|||||||
}
|
}
|
||||||
tevent_req_set_callback(req, aio_pwrite_smb2_done, aio_ex);
|
tevent_req_set_callback(req, aio_pwrite_smb2_done, aio_ex);
|
||||||
|
|
||||||
|
if (!aio_add_req_to_fsp(fsp, req)) {
|
||||||
|
DEBUG(1, ("Could not add req to fsp\n"));
|
||||||
|
SMB_VFS_STRICT_UNLOCK(conn, fsp, &aio_ex->lock);
|
||||||
|
TALLOC_FREE(aio_ex);
|
||||||
|
return NT_STATUS_RETRY;
|
||||||
|
}
|
||||||
|
|
||||||
/* We don't need talloc_move here as both aio_ex and
|
/* We don't need talloc_move here as both aio_ex and
|
||||||
* smbreq are children of smbreq->smb2req. */
|
* smbreq are children of smbreq->smb2req. */
|
||||||
aio_ex->smbreq = smbreq;
|
aio_ex->smbreq = smbreq;
|
||||||
@ -1204,110 +1289,16 @@ void smbd_aio_complete_aio_ex(struct aio_extra *aio_ex)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/****************************************************************************
|
void aio_fsp_close(files_struct *fsp)
|
||||||
We're doing write behind and the client closed the file. Wait up to 45
|
|
||||||
seconds (my arbitrary choice) for the aio to complete. Return 0 if all writes
|
|
||||||
completed, errno to return if not.
|
|
||||||
*****************************************************************************/
|
|
||||||
|
|
||||||
#define SMB_TIME_FOR_AIO_COMPLETE_WAIT 45
|
|
||||||
|
|
||||||
int wait_for_aio_completion(files_struct *fsp)
|
|
||||||
{
|
{
|
||||||
struct aio_extra *aio_ex;
|
unsigned i;
|
||||||
const SMB_STRUCT_AIOCB **aiocb_list;
|
|
||||||
int aio_completion_count = 0;
|
|
||||||
time_t start_time = time_mono(NULL);
|
|
||||||
int seconds_left;
|
|
||||||
|
|
||||||
for (seconds_left = SMB_TIME_FOR_AIO_COMPLETE_WAIT;
|
for (i=0; i<fsp->num_aio_requests; i++) {
|
||||||
seconds_left >= 0;) {
|
struct tevent_req *req = fsp->aio_requests[i];
|
||||||
int err = 0;
|
struct aio_extra *aio_ex = tevent_req_callback_data(
|
||||||
int i;
|
req, struct aio_extra);
|
||||||
struct timespec ts;
|
aio_ex->fsp = NULL;
|
||||||
|
|
||||||
aio_completion_count = 0;
|
|
||||||
for( aio_ex = aio_list_head; aio_ex; aio_ex = aio_ex->next) {
|
|
||||||
if (aio_ex->fsp == fsp) {
|
|
||||||
aio_completion_count++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!aio_completion_count) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
DEBUG(3,("wait_for_aio_completion: waiting for %d aio events "
|
|
||||||
"to complete.\n", aio_completion_count ));
|
|
||||||
|
|
||||||
aiocb_list = SMB_MALLOC_ARRAY(const SMB_STRUCT_AIOCB *,
|
|
||||||
aio_completion_count);
|
|
||||||
if (!aiocb_list) {
|
|
||||||
return ENOMEM;
|
|
||||||
}
|
|
||||||
|
|
||||||
for( i = 0, aio_ex = aio_list_head;
|
|
||||||
aio_ex;
|
|
||||||
aio_ex = aio_ex->next) {
|
|
||||||
if (aio_ex->fsp == fsp) {
|
|
||||||
aiocb_list[i++] = &aio_ex->acb;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Now wait up to seconds_left for completion. */
|
|
||||||
ts.tv_sec = seconds_left;
|
|
||||||
ts.tv_nsec = 0;
|
|
||||||
|
|
||||||
DEBUG(10,("wait_for_aio_completion: %d events, doing a wait "
|
|
||||||
"of %d seconds.\n",
|
|
||||||
aio_completion_count, seconds_left ));
|
|
||||||
|
|
||||||
err = SMB_VFS_AIO_SUSPEND(fsp, aiocb_list,
|
|
||||||
aio_completion_count, &ts);
|
|
||||||
|
|
||||||
DEBUG(10,("wait_for_aio_completion: returned err = %d, "
|
|
||||||
"errno = %s\n", err, strerror(errno) ));
|
|
||||||
|
|
||||||
if (err == -1 && errno == EAGAIN) {
|
|
||||||
DEBUG(0,("wait_for_aio_completion: aio_suspend timed "
|
|
||||||
"out waiting for %d events after a wait of "
|
|
||||||
"%d seconds\n", aio_completion_count,
|
|
||||||
seconds_left));
|
|
||||||
/* Timeout. */
|
|
||||||
SAFE_FREE(aiocb_list);
|
|
||||||
/* We're hosed here - IO may complete
|
|
||||||
and trample over memory if we free
|
|
||||||
the aio_ex struct, but if we don't
|
|
||||||
we leak IO requests. I think smb_panic()
|
|
||||||
if the right thing to do here. JRA.
|
|
||||||
*/
|
|
||||||
smb_panic("AIO suspend timed out - cannot continue.");
|
|
||||||
return EIO;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* One or more events might have completed - process them if
|
|
||||||
* so. */
|
|
||||||
for( i = 0; i < aio_completion_count; i++) {
|
|
||||||
aio_ex = (struct aio_extra *)aiocb_list[i]->aio_sigevent.sigev_value.sival_ptr;
|
|
||||||
|
|
||||||
if (!handle_aio_completed(aio_ex, &err)) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
TALLOC_FREE(aio_ex);
|
|
||||||
}
|
|
||||||
|
|
||||||
SAFE_FREE(aiocb_list);
|
|
||||||
seconds_left = SMB_TIME_FOR_AIO_COMPLETE_WAIT
|
|
||||||
- (time_mono(NULL) - start_time);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* We timed out - we don't know why. Return ret if already an error,
|
|
||||||
* else EIO. */
|
|
||||||
DEBUG(10,("wait_for_aio_completion: aio_suspend timed out waiting "
|
|
||||||
"for %d events\n",
|
|
||||||
aio_completion_count));
|
|
||||||
|
|
||||||
return EIO;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#else
|
#else
|
||||||
@ -1360,6 +1351,11 @@ NTSTATUS schedule_aio_smb2_write(connection_struct *conn,
|
|||||||
return NT_STATUS_RETRY;
|
return NT_STATUS_RETRY;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void aio_fsp_close(files_struct *fsp)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
int wait_for_aio_completion(files_struct *fsp)
|
int wait_for_aio_completion(files_struct *fsp)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -706,17 +706,8 @@ static NTSTATUS close_normal_file(struct smb_request *req, files_struct *fsp,
|
|||||||
NTSTATUS status = NT_STATUS_OK;
|
NTSTATUS status = NT_STATUS_OK;
|
||||||
NTSTATUS tmp;
|
NTSTATUS tmp;
|
||||||
connection_struct *conn = fsp->conn;
|
connection_struct *conn = fsp->conn;
|
||||||
int ret;
|
|
||||||
|
|
||||||
/*
|
aio_fsp_close(fsp);
|
||||||
* If we're finishing async io on a close we can get a write
|
|
||||||
* error here, we must remember this.
|
|
||||||
*/
|
|
||||||
ret = wait_for_aio_completion(fsp);
|
|
||||||
if (ret) {
|
|
||||||
status = ntstatus_keeperror(
|
|
||||||
status, map_nt_error_from_unix(ret));
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* If we're flushing on a close we can get a write
|
* If we're flushing on a close we can get a write
|
||||||
|
@ -26,7 +26,6 @@
|
|||||||
#include "tdb_compat.h"
|
#include "tdb_compat.h"
|
||||||
|
|
||||||
#if defined(HAVE_AIO)
|
#if defined(HAVE_AIO)
|
||||||
struct aio_extra *aio_list_head = NULL;
|
|
||||||
struct tevent_signal *aio_signal_event = NULL;
|
struct tevent_signal *aio_signal_event = NULL;
|
||||||
int aio_pending_size = 100; /* tevent supports 100 signals SA_SIGINFO */
|
int aio_pending_size = 100; /* tevent supports 100 signals SA_SIGINFO */
|
||||||
int outstanding_aio_calls = 0;
|
int outstanding_aio_calls = 0;
|
||||||
|
@ -23,7 +23,6 @@
|
|||||||
|
|
||||||
#if defined(HAVE_AIO)
|
#if defined(HAVE_AIO)
|
||||||
struct aio_extra;
|
struct aio_extra;
|
||||||
extern struct aio_extra *aio_list_head;
|
|
||||||
extern struct tevent_signal *aio_signal_event;
|
extern struct tevent_signal *aio_signal_event;
|
||||||
extern int aio_pending_size;
|
extern int aio_pending_size;
|
||||||
extern int outstanding_aio_calls;
|
extern int outstanding_aio_calls;
|
||||||
|
@ -89,6 +89,7 @@ NTSTATUS schedule_aio_smb2_write(connection_struct *conn,
|
|||||||
DATA_BLOB in_data,
|
DATA_BLOB in_data,
|
||||||
bool write_through);
|
bool write_through);
|
||||||
bool cancel_smb2_aio(struct smb_request *smbreq);
|
bool cancel_smb2_aio(struct smb_request *smbreq);
|
||||||
|
void aio_fsp_close(files_struct *fsp);
|
||||||
int wait_for_aio_completion(files_struct *fsp);
|
int wait_for_aio_completion(files_struct *fsp);
|
||||||
void smbd_aio_complete_aio_ex(struct aio_extra *aio_ex);
|
void smbd_aio_complete_aio_ex(struct aio_extra *aio_ex);
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user