2005-06-28 02:53:56 +04:00
/*
Unix SMB / Netbios implementation .
Version 3.0
async_io read handling using POSIX async io .
Copyright ( C ) Jeremy Allison 2005.
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-09 23:25:36 +04:00
the Free Software Foundation ; either version 3 of the License , or
2005-06-28 02:53:56 +04:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 04:52:41 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2005-06-28 02:53:56 +04:00
*/
# include "includes.h"
2011-03-22 18:57:01 +03:00
# include "smbd/smbd.h"
2009-01-08 14:03:45 +03:00
# include "smbd/globals.h"
2011-04-28 19:38:09 +04:00
# include "../lib/util/tevent_ntstatus.h"
2005-06-28 02:53:56 +04:00
# if defined(WITH_AIO)
/* The signal we'll use to signify aio done. */
# ifndef RT_SIGNAL_AIO
2008-09-16 05:45:10 +04:00
# define RT_SIGNAL_AIO (SIGRTMIN+3)
# endif
# ifndef HAVE_STRUCT_SIGEVENT_SIGEV_VALUE_SIVAL_PTR
# ifdef HAVE_STRUCT_SIGEVENT_SIGEV_VALUE_SIGVAL_PTR
# define sival_int sigval_int
# define sival_ptr sigval_ptr
# endif
2005-06-28 02:53:56 +04:00
# endif
2010-06-02 03:05:44 +04:00
/****************************************************************************
The buffer we keep around whilst an aio request is in process .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct aio_extra {
struct aio_extra * next , * prev ;
SMB_STRUCT_AIOCB acb ;
files_struct * fsp ;
2010-06-04 22:41:38 +04:00
struct smb_request * smbreq ;
2010-06-02 21:25:56 +04:00
DATA_BLOB outbuf ;
2010-06-02 03:05:44 +04:00
struct lock_struct lock ;
2010-06-08 03:25:18 +04:00
bool write_through ;
2011-11-14 12:53:25 +04:00
bool pass_cancel ;
2010-06-02 03:05:44 +04:00
int ( * handle_completion ) ( struct aio_extra * ex , int errcode ) ;
} ;
2010-04-02 03:47:59 +04:00
/****************************************************************************
Initialize the signal handler for aio read / write .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void smbd_aio_signal_handler ( struct tevent_context * ev_ctx ,
struct tevent_signal * se ,
int signum , int count ,
void * _info , void * private_data )
{
siginfo_t * info = ( siginfo_t * ) _info ;
2010-06-02 03:05:44 +04:00
struct aio_extra * aio_ex = ( struct aio_extra * )
info - > si_value . sival_ptr ;
2010-04-02 03:47:59 +04:00
2010-06-04 22:30:46 +04:00
smbd_aio_complete_aio_ex ( aio_ex ) ;
2012-01-12 04:37:48 +04:00
TALLOC_FREE ( aio_ex ) ;
2010-04-02 03:47:59 +04:00
}
2010-07-18 16:40:43 +04:00
static bool initialize_async_io_handler ( void )
2010-04-02 03:47:59 +04:00
{
2010-07-18 16:40:43 +04:00
static bool tried_signal_setup = false ;
2010-04-02 03:47:59 +04:00
if ( aio_signal_event ) {
2010-07-18 16:40:43 +04:00
return true ;
}
if ( tried_signal_setup ) {
return false ;
2010-04-02 03:47:59 +04:00
}
2010-07-18 16:40:43 +04:00
tried_signal_setup = true ;
2010-04-02 03:47:59 +04:00
2011-05-25 12:51:56 +04:00
aio_signal_event = tevent_add_signal ( server_event_context ( ) ,
server_event_context ( ) ,
2010-04-02 03:47:59 +04:00
RT_SIGNAL_AIO , SA_SIGINFO ,
smbd_aio_signal_handler ,
NULL ) ;
if ( ! aio_signal_event ) {
2010-07-18 16:40:43 +04:00
DEBUG ( 10 , ( " Failed to setup RT_SIGNAL_AIO handler \n " ) ) ;
return false ;
2010-04-02 03:47:59 +04:00
}
/* tevent supports 100 signal with SA_SIGINFO */
aio_pending_size = 100 ;
2010-07-20 20:34:32 +04:00
return true ;
2010-04-02 03:47:59 +04:00
}
2009-10-13 17:48:19 +04:00
static int handle_aio_read_complete ( struct aio_extra * aio_ex , int errcode ) ;
static int handle_aio_write_complete ( struct aio_extra * aio_ex , int errcode ) ;
2010-06-11 00:20:37 +04:00
static int handle_aio_smb2_read_complete ( struct aio_extra * aio_ex , int errcode ) ;
static int handle_aio_smb2_write_complete ( struct aio_extra * aio_ex , int errcode ) ;
2008-11-03 23:56:02 +03:00
static int aio_extra_destructor ( struct aio_extra * aio_ex )
{
DLIST_REMOVE ( aio_list_head , aio_ex ) ;
return 0 ;
}
2005-06-28 02:53:56 +04:00
/****************************************************************************
Create the extended aio struct we must keep around for the lifetime
2008-11-03 23:56:02 +03:00
of the aio call .
2005-06-28 02:53:56 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-06-08 03:26:30 +04:00
static struct aio_extra * create_aio_extra ( TALLOC_CTX * mem_ctx ,
files_struct * fsp ,
size_t buflen )
2005-06-28 02:53:56 +04:00
{
2011-06-07 05:44:43 +04:00
struct aio_extra * aio_ex = talloc_zero ( mem_ctx , struct aio_extra ) ;
2005-06-28 02:53:56 +04:00
if ( ! aio_ex ) {
return NULL ;
}
2008-11-03 21:59:11 +03:00
2005-06-28 02:53:56 +04:00
/* The output buffer stored in the aio_ex is the start of
the smb return buffer . The buffer used in the acb
is the start of the reply data portion of that buffer . */
2010-06-08 03:26:30 +04:00
if ( buflen ) {
aio_ex - > outbuf = data_blob_talloc ( aio_ex , NULL , buflen ) ;
if ( ! aio_ex - > outbuf . data ) {
TALLOC_FREE ( aio_ex ) ;
return NULL ;
}
2005-06-28 02:53:56 +04:00
}
DLIST_ADD ( aio_list_head , aio_ex ) ;
2008-11-03 23:56:02 +03:00
talloc_set_destructor ( aio_ex , aio_extra_destructor ) ;
2005-06-28 02:53:56 +04:00
aio_ex - > fsp = fsp ;
return aio_ex ;
}
/****************************************************************************
Set up an aio request from a SMBreadX call .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-04-06 01:16:21 +04:00
NTSTATUS schedule_aio_read_and_X ( connection_struct * conn ,
2010-06-04 22:41:38 +04:00
struct smb_request * smbreq ,
2005-06-28 02:53:56 +04:00
files_struct * fsp , SMB_OFF_T startpos ,
size_t smb_maxcnt )
{
struct aio_extra * aio_ex ;
SMB_STRUCT_AIOCB * a ;
size_t bufsize ;
size_t min_aio_read_size = lp_aio_read_size ( SNUM ( conn ) ) ;
2008-11-02 23:52:16 +03:00
int ret ;
2005-06-28 02:53:56 +04:00
2010-06-02 03:05:44 +04:00
/* Ensure aio is initialized. */
2010-07-18 16:40:43 +04:00
if ( ! initialize_async_io_handler ( ) ) {
return NT_STATUS_RETRY ;
}
2010-06-02 03:05:44 +04:00
2008-02-24 02:01:07 +03:00
if ( fsp - > base_fsp ! = NULL ) {
/* No AIO on streams yet */
DEBUG ( 10 , ( " AIO on streams not yet supported \n " ) ) ;
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2008-02-24 02:01:07 +03:00
}
2008-01-30 13:11:27 +03:00
if ( ( ! min_aio_read_size | | ( smb_maxcnt < min_aio_read_size ) )
& & ! SMB_VFS_AIO_FORCE ( fsp ) ) {
2005-06-28 02:53:56 +04:00
/* Too small a read for aio request. */
DEBUG ( 10 , ( " schedule_aio_read_and_X: read size (%u) too small "
" for minimum aio_read of %u \n " ,
( unsigned int ) smb_maxcnt ,
( unsigned int ) min_aio_read_size ) ) ;
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2005-06-28 02:53:56 +04:00
}
2007-02-10 16:15:58 +03:00
/* Only do this on non-chained and non-chaining reads not using the
* write cache . */
2010-06-04 22:41:38 +04:00
if ( req_is_in_chain ( smbreq ) | | ( lp_write_cache_size ( SNUM ( conn ) ) ! = 0 ) ) {
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2005-06-28 02:53:56 +04:00
}
2008-02-24 13:32:22 +03:00
if ( outstanding_aio_calls > = aio_pending_size ) {
2007-02-10 16:15:58 +03:00
DEBUG ( 10 , ( " schedule_aio_read_and_X: Already have %d aio "
" activities outstanding. \n " ,
2005-06-28 02:53:56 +04:00
outstanding_aio_calls ) ) ;
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2005-06-28 02:53:56 +04:00
}
2007-08-11 20:28:10 +04:00
/* The following is safe from integer wrap as we've already checked
smb_maxcnt is 128 k or less . Wct is 12 for read replies */
bufsize = smb_size + 12 * 2 + smb_maxcnt ;
2005-06-28 02:53:56 +04:00
2010-06-08 03:26:30 +04:00
if ( ( aio_ex = create_aio_extra ( NULL , fsp , bufsize ) ) = = NULL ) {
2005-06-28 02:53:56 +04:00
DEBUG ( 10 , ( " schedule_aio_read_and_X: malloc fail. \n " ) ) ;
2010-04-06 01:16:21 +04:00
return NT_STATUS_NO_MEMORY ;
2005-06-28 02:53:56 +04:00
}
2008-11-03 23:56:02 +03:00
aio_ex - > handle_completion = handle_aio_read_complete ;
2005-06-28 02:53:56 +04:00
2010-06-04 22:41:38 +04:00
construct_reply_common_req ( smbreq , ( char * ) aio_ex - > outbuf . data ) ;
2010-06-02 21:25:56 +04:00
srv_set_message ( ( char * ) aio_ex - > outbuf . data , 12 , 0 , True ) ;
SCVAL ( aio_ex - > outbuf . data , smb_vwv0 , 0xFF ) ; /* Never a chained reply. */
2005-06-28 02:53:56 +04:00
2010-06-04 22:41:38 +04:00
init_strict_lock_struct ( fsp , ( uint64_t ) smbreq - > smbpid ,
2010-04-06 01:16:21 +04:00
( uint64_t ) startpos , ( uint64_t ) smb_maxcnt , READ_LOCK ,
& aio_ex - > lock ) ;
/* Take the lock until the AIO completes. */
if ( ! SMB_VFS_STRICT_LOCK ( conn , fsp , & aio_ex - > lock ) ) {
TALLOC_FREE ( aio_ex ) ;
return NT_STATUS_FILE_LOCK_CONFLICT ;
}
2005-06-28 02:53:56 +04:00
a = & aio_ex - > acb ;
/* Now set up the aio record for the read call. */
2008-11-03 22:07:11 +03:00
2005-07-13 04:26:52 +04:00
a - > aio_fildes = fsp - > fh - > fd ;
2010-06-02 21:25:56 +04:00
a - > aio_buf = smb_buf ( aio_ex - > outbuf . data ) ;
2005-06-28 02:53:56 +04:00
a - > aio_nbytes = smb_maxcnt ;
a - > aio_offset = startpos ;
a - > aio_sigevent . sigev_notify = SIGEV_SIGNAL ;
a - > aio_sigevent . sigev_signo = RT_SIGNAL_AIO ;
2010-06-02 03:05:44 +04:00
a - > aio_sigevent . sigev_value . sival_ptr = aio_ex ;
2005-06-28 02:53:56 +04:00
2008-11-02 23:52:16 +03:00
ret = SMB_VFS_AIO_READ ( fsp , a ) ;
if ( ret = = - 1 ) {
2007-02-10 16:15:58 +03:00
DEBUG ( 0 , ( " schedule_aio_read_and_X: aio_read failed. "
" Error %s \n " , strerror ( errno ) ) ) ;
2010-04-06 01:16:21 +04:00
SMB_VFS_STRICT_UNLOCK ( conn , fsp , & aio_ex - > lock ) ;
2008-11-03 21:59:11 +03:00
TALLOC_FREE ( aio_ex ) ;
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2005-06-28 02:53:56 +04:00
}
2009-10-01 02:20:09 +04:00
outstanding_aio_calls + + ;
2010-06-04 22:41:38 +04:00
aio_ex - > smbreq = talloc_move ( aio_ex , & smbreq ) ;
2008-11-03 23:56:02 +03:00
2007-02-10 16:15:58 +03:00
DEBUG ( 10 , ( " schedule_aio_read_and_X: scheduled aio_read for file %s, "
" offset %.0f, len = %u (mid = %u) \n " ,
2009-07-11 01:50:37 +04:00
fsp_str_dbg ( fsp ) , ( double ) startpos , ( unsigned int ) smb_maxcnt ,
2010-06-04 22:41:38 +04:00
( unsigned int ) aio_ex - > smbreq - > mid ) ) ;
2005-06-28 02:53:56 +04:00
2010-04-06 01:16:21 +04:00
return NT_STATUS_OK ;
2005-06-28 02:53:56 +04:00
}
/****************************************************************************
Set up an aio request from a SMBwriteX call .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-04-06 01:16:21 +04:00
NTSTATUS schedule_aio_write_and_X ( connection_struct * conn ,
2010-06-04 22:41:38 +04:00
struct smb_request * smbreq ,
2011-05-05 21:41:59 +04:00
files_struct * fsp , const char * data ,
2007-08-08 23:05:30 +04:00
SMB_OFF_T startpos ,
size_t numtowrite )
2005-06-28 02:53:56 +04:00
{
struct aio_extra * aio_ex ;
SMB_STRUCT_AIOCB * a ;
2008-11-03 23:56:02 +03:00
size_t bufsize ;
2005-06-28 02:53:56 +04:00
size_t min_aio_write_size = lp_aio_write_size ( SNUM ( conn ) ) ;
2008-11-02 23:52:16 +03:00
int ret ;
2005-06-28 02:53:56 +04:00
2010-06-02 03:05:44 +04:00
/* Ensure aio is initialized. */
2010-07-18 16:40:43 +04:00
if ( ! initialize_async_io_handler ( ) ) {
return NT_STATUS_RETRY ;
}
2010-06-02 03:05:44 +04:00
2008-02-24 02:01:07 +03:00
if ( fsp - > base_fsp ! = NULL ) {
/* No AIO on streams yet */
DEBUG ( 10 , ( " AIO on streams not yet supported \n " ) ) ;
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2008-02-24 02:01:07 +03:00
}
2008-01-30 13:11:27 +03:00
if ( ( ! min_aio_write_size | | ( numtowrite < min_aio_write_size ) )
& & ! SMB_VFS_AIO_FORCE ( fsp ) ) {
2005-06-28 02:53:56 +04:00
/* Too small a write for aio request. */
2007-02-10 16:15:58 +03:00
DEBUG ( 10 , ( " schedule_aio_write_and_X: write size (%u) too "
" small for minimum aio_write of %u \n " ,
2005-06-28 02:53:56 +04:00
( unsigned int ) numtowrite ,
( unsigned int ) min_aio_write_size ) ) ;
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2005-06-28 02:53:56 +04:00
}
2010-06-08 03:25:18 +04:00
/* Only do this on non-chained and non-chaining writes not using the
2007-02-10 16:15:58 +03:00
* write cache . */
2010-06-04 22:41:38 +04:00
if ( req_is_in_chain ( smbreq ) | | ( lp_write_cache_size ( SNUM ( conn ) ) ! = 0 ) ) {
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2005-06-28 02:53:56 +04:00
}
2008-02-24 13:32:22 +03:00
if ( outstanding_aio_calls > = aio_pending_size ) {
2007-02-10 16:15:58 +03:00
DEBUG ( 3 , ( " schedule_aio_write_and_X: Already have %d aio "
" activities outstanding. \n " ,
2005-06-28 02:53:56 +04:00
outstanding_aio_calls ) ) ;
2007-02-10 16:15:58 +03:00
DEBUG ( 10 , ( " schedule_aio_write_and_X: failed to schedule "
" aio_write for file %s, offset %.0f, len = %u "
" (mid = %u) \n " ,
2009-07-11 01:50:37 +04:00
fsp_str_dbg ( fsp ) , ( double ) startpos ,
2007-02-10 16:15:58 +03:00
( unsigned int ) numtowrite ,
2010-06-04 22:41:38 +04:00
( unsigned int ) smbreq - > mid ) ) ;
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2005-06-28 02:53:56 +04:00
}
2008-11-03 23:56:02 +03:00
bufsize = smb_size + 6 * 2 ;
2010-06-08 03:26:30 +04:00
if ( ! ( aio_ex = create_aio_extra ( NULL , fsp , bufsize ) ) ) {
2005-06-28 02:53:56 +04:00
DEBUG ( 0 , ( " schedule_aio_write_and_X: malloc fail. \n " ) ) ;
2010-04-06 01:16:21 +04:00
return NT_STATUS_NO_MEMORY ;
2005-06-28 02:53:56 +04:00
}
2008-11-03 23:56:02 +03:00
aio_ex - > handle_completion = handle_aio_write_complete ;
2010-06-08 03:25:18 +04:00
aio_ex - > write_through = BITSETW ( smbreq - > vwv + 7 , 0 ) ;
2005-06-28 02:53:56 +04:00
2010-06-04 22:41:38 +04:00
construct_reply_common_req ( smbreq , ( char * ) aio_ex - > outbuf . data ) ;
2010-06-02 21:25:56 +04:00
srv_set_message ( ( char * ) aio_ex - > outbuf . data , 6 , 0 , True ) ;
SCVAL ( aio_ex - > outbuf . data , smb_vwv0 , 0xFF ) ; /* Never a chained reply. */
2005-06-28 02:53:56 +04:00
2010-06-04 22:41:38 +04:00
init_strict_lock_struct ( fsp , ( uint64_t ) smbreq - > smbpid ,
2010-04-06 01:16:21 +04:00
( uint64_t ) startpos , ( uint64_t ) numtowrite , WRITE_LOCK ,
& aio_ex - > lock ) ;
/* Take the lock until the AIO completes. */
if ( ! SMB_VFS_STRICT_LOCK ( conn , fsp , & aio_ex - > lock ) ) {
TALLOC_FREE ( aio_ex ) ;
return NT_STATUS_FILE_LOCK_CONFLICT ;
}
2005-06-28 02:53:56 +04:00
a = & aio_ex - > acb ;
/* Now set up the aio record for the write call. */
2008-11-03 22:07:11 +03:00
2005-07-13 04:26:52 +04:00
a - > aio_fildes = fsp - > fh - > fd ;
2011-05-05 21:41:59 +04:00
a - > aio_buf = discard_const_p ( char , data ) ;
2005-06-28 02:53:56 +04:00
a - > aio_nbytes = numtowrite ;
a - > aio_offset = startpos ;
a - > aio_sigevent . sigev_notify = SIGEV_SIGNAL ;
a - > aio_sigevent . sigev_signo = RT_SIGNAL_AIO ;
2010-06-02 03:05:44 +04:00
a - > aio_sigevent . sigev_value . sival_ptr = aio_ex ;
2005-06-28 02:53:56 +04:00
2008-11-02 23:52:16 +03:00
ret = SMB_VFS_AIO_WRITE ( fsp , a ) ;
if ( ret = = - 1 ) {
2007-02-10 16:15:58 +03:00
DEBUG ( 3 , ( " schedule_aio_wrote_and_X: aio_write failed. "
" Error %s \n " , strerror ( errno ) ) ) ;
2010-04-06 01:16:21 +04:00
SMB_VFS_STRICT_UNLOCK ( conn , fsp , & aio_ex - > lock ) ;
2008-11-03 21:59:11 +03:00
TALLOC_FREE ( aio_ex ) ;
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2005-06-28 02:53:56 +04:00
}
2008-11-02 23:52:16 +03:00
2009-10-01 02:20:09 +04:00
outstanding_aio_calls + + ;
2010-06-04 22:41:38 +04:00
aio_ex - > smbreq = talloc_move ( aio_ex , & smbreq ) ;
2008-11-03 23:56:02 +03:00
2009-02-03 22:56:35 +03:00
/* This should actually be improved to span the write. */
contend_level2_oplocks_begin ( fsp , LEVEL2_CONTEND_WRITE ) ;
contend_level2_oplocks_end ( fsp , LEVEL2_CONTEND_WRITE ) ;
2008-02-19 17:53:57 +03:00
2010-06-08 03:25:18 +04:00
if ( ! aio_ex - > write_through & & ! lp_syncalways ( SNUM ( fsp - > conn ) )
2007-10-11 00:34:30 +04:00
& & fsp - > aio_write_behind ) {
/* Lie to the client and immediately claim we finished the
* write . */
2010-06-02 21:25:56 +04:00
SSVAL ( aio_ex - > outbuf . data , smb_vwv2 , numtowrite ) ;
SSVAL ( aio_ex - > outbuf . data , smb_vwv4 , ( numtowrite > > 16 ) & 1 ) ;
show_msg ( ( char * ) aio_ex - > outbuf . data ) ;
2010-08-24 22:10:20 +04:00
if ( ! srv_send_smb ( aio_ex - > smbreq - > sconn ,
2010-08-14 16:53:45 +04:00
( char * ) aio_ex - > outbuf . data ,
2010-06-04 22:41:38 +04:00
true , aio_ex - > smbreq - > seqnum + 1 ,
2009-02-09 10:10:34 +03:00
IS_CONN_ENCRYPTED ( fsp - > conn ) ,
2010-06-04 22:41:38 +04:00
& aio_ex - > smbreq - > pcd ) ) {
2010-08-14 00:51:29 +04:00
exit_server_cleanly ( " schedule_aio_write_and_X: "
" srv_send_smb failed. " ) ;
2007-10-11 00:34:30 +04:00
}
DEBUG ( 10 , ( " schedule_aio_write_and_X: scheduled aio_write "
2009-07-11 01:50:37 +04:00
" behind for file %s \n " , fsp_str_dbg ( fsp ) ) ) ;
2007-10-11 00:34:30 +04:00
}
2005-06-28 02:53:56 +04:00
2007-02-10 16:15:58 +03:00
DEBUG ( 10 , ( " schedule_aio_write_and_X: scheduled aio_write for file "
" %s, offset %.0f, len = %u (mid = %u) "
" outstanding_aio_calls = %d \n " ,
2009-07-11 01:50:37 +04:00
fsp_str_dbg ( fsp ) , ( double ) startpos , ( unsigned int ) numtowrite ,
2010-06-04 22:41:38 +04:00
( unsigned int ) aio_ex - > smbreq - > mid , outstanding_aio_calls ) ) ;
2005-06-28 02:53:56 +04:00
2010-04-06 01:16:21 +04:00
return NT_STATUS_OK ;
2005-06-28 02:53:56 +04:00
}
2011-11-14 12:52:47 +04:00
bool cancel_smb2_aio ( struct smb_request * smbreq )
{
struct smbd_smb2_request * smb2req = smbreq - > smb2req ;
struct aio_extra * aio_ex = NULL ;
int ret ;
if ( smb2req ) {
aio_ex = talloc_get_type ( smbreq - > async_priv ,
struct aio_extra ) ;
}
if ( aio_ex = = NULL ) {
return false ;
}
if ( aio_ex - > fsp = = NULL ) {
return false ;
}
ret = SMB_VFS_AIO_CANCEL ( aio_ex - > fsp , & aio_ex - > acb ) ;
if ( ret ! = AIO_CANCELED ) {
return false ;
}
return true ;
}
2010-06-11 00:20:37 +04:00
/****************************************************************************
Set up an aio request from a SMB2 read call .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
NTSTATUS schedule_smb2_aio_read ( connection_struct * conn ,
struct smb_request * smbreq ,
files_struct * fsp ,
2010-12-15 03:32:10 +03:00
TALLOC_CTX * ctx ,
DATA_BLOB * preadbuf ,
2010-06-11 00:20:37 +04:00
SMB_OFF_T startpos ,
size_t smb_maxcnt )
{
struct aio_extra * aio_ex ;
SMB_STRUCT_AIOCB * a ;
size_t min_aio_read_size = lp_aio_read_size ( SNUM ( conn ) ) ;
int ret ;
/* Ensure aio is initialized. */
2010-07-18 16:40:43 +04:00
if ( ! initialize_async_io_handler ( ) ) {
return NT_STATUS_RETRY ;
}
2010-06-11 00:20:37 +04:00
if ( fsp - > base_fsp ! = NULL ) {
/* No AIO on streams yet */
DEBUG ( 10 , ( " AIO on streams not yet supported \n " ) ) ;
return NT_STATUS_RETRY ;
}
if ( ( ! min_aio_read_size | | ( smb_maxcnt < min_aio_read_size ) )
& & ! SMB_VFS_AIO_FORCE ( fsp ) ) {
/* Too small a read for aio request. */
DEBUG ( 10 , ( " smb2: read size (%u) too small "
" for minimum aio_read of %u \n " ,
( unsigned int ) smb_maxcnt ,
( unsigned int ) min_aio_read_size ) ) ;
return NT_STATUS_RETRY ;
}
/* Only do this on reads not using the write cache. */
if ( lp_write_cache_size ( SNUM ( conn ) ) ! = 0 ) {
return NT_STATUS_RETRY ;
}
if ( outstanding_aio_calls > = aio_pending_size ) {
DEBUG ( 10 , ( " smb2: Already have %d aio "
" activities outstanding. \n " ,
outstanding_aio_calls ) ) ;
return NT_STATUS_RETRY ;
}
2010-12-15 03:32:10 +03:00
/* Create the out buffer. */
* preadbuf = data_blob_talloc ( ctx , NULL , smb_maxcnt ) ;
if ( preadbuf - > data = = NULL ) {
return NT_STATUS_NO_MEMORY ;
}
2010-06-11 00:20:37 +04:00
if ( ! ( aio_ex = create_aio_extra ( smbreq - > smb2req , fsp , 0 ) ) ) {
return NT_STATUS_NO_MEMORY ;
}
aio_ex - > handle_completion = handle_aio_smb2_read_complete ;
2011-11-14 12:53:25 +04:00
aio_ex - > pass_cancel = true ;
2010-06-11 00:20:37 +04:00
init_strict_lock_struct ( fsp , ( uint64_t ) smbreq - > smbpid ,
( uint64_t ) startpos , ( uint64_t ) smb_maxcnt , READ_LOCK ,
& aio_ex - > lock ) ;
/* Take the lock until the AIO completes. */
if ( ! SMB_VFS_STRICT_LOCK ( conn , fsp , & aio_ex - > lock ) ) {
TALLOC_FREE ( aio_ex ) ;
return NT_STATUS_FILE_LOCK_CONFLICT ;
}
a = & aio_ex - > acb ;
/* Now set up the aio record for the read call. */
a - > aio_fildes = fsp - > fh - > fd ;
2010-12-15 03:32:10 +03:00
a - > aio_buf = preadbuf - > data ;
2010-06-11 00:20:37 +04:00
a - > aio_nbytes = smb_maxcnt ;
a - > aio_offset = startpos ;
a - > aio_sigevent . sigev_notify = SIGEV_SIGNAL ;
a - > aio_sigevent . sigev_signo = RT_SIGNAL_AIO ;
a - > aio_sigevent . sigev_value . sival_ptr = aio_ex ;
ret = SMB_VFS_AIO_READ ( fsp , a ) ;
if ( ret = = - 1 ) {
DEBUG ( 0 , ( " smb2: aio_read failed. "
" Error %s \n " , strerror ( errno ) ) ) ;
SMB_VFS_STRICT_UNLOCK ( conn , fsp , & aio_ex - > lock ) ;
TALLOC_FREE ( aio_ex ) ;
return NT_STATUS_RETRY ;
}
outstanding_aio_calls + + ;
/* We don't need talloc_move here as both aio_ex and
* smbreq are children of smbreq - > smb2req . */
aio_ex - > smbreq = smbreq ;
2011-11-14 12:52:47 +04:00
smbreq - > async_priv = aio_ex ;
2010-06-11 00:20:37 +04:00
DEBUG ( 10 , ( " smb2: scheduled aio_read for file %s, "
" offset %.0f, len = %u (mid = %u) \n " ,
fsp_str_dbg ( fsp ) , ( double ) startpos , ( unsigned int ) smb_maxcnt ,
( unsigned int ) aio_ex - > smbreq - > mid ) ) ;
return NT_STATUS_OK ;
}
/****************************************************************************
Set up an aio request from a SMB2write call .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
NTSTATUS schedule_aio_smb2_write ( connection_struct * conn ,
struct smb_request * smbreq ,
files_struct * fsp ,
uint64_t in_offset ,
DATA_BLOB in_data ,
bool write_through )
{
struct aio_extra * aio_ex = NULL ;
SMB_STRUCT_AIOCB * a = NULL ;
size_t min_aio_write_size = lp_aio_write_size ( SNUM ( conn ) ) ;
int ret ;
/* Ensure aio is initialized. */
2010-07-18 16:40:43 +04:00
if ( ! initialize_async_io_handler ( ) ) {
return NT_STATUS_RETRY ;
}
2010-06-11 00:20:37 +04:00
if ( fsp - > base_fsp ! = NULL ) {
/* No AIO on streams yet */
DEBUG ( 10 , ( " AIO on streams not yet supported \n " ) ) ;
return NT_STATUS_RETRY ;
}
if ( ( ! min_aio_write_size | | ( in_data . length < min_aio_write_size ) )
& & ! SMB_VFS_AIO_FORCE ( fsp ) ) {
/* Too small a write for aio request. */
DEBUG ( 10 , ( " smb2: write size (%u) too "
" small for minimum aio_write of %u \n " ,
( unsigned int ) in_data . length ,
( unsigned int ) min_aio_write_size ) ) ;
return NT_STATUS_RETRY ;
}
/* Only do this on writes not using the write cache. */
if ( lp_write_cache_size ( SNUM ( conn ) ) ! = 0 ) {
return NT_STATUS_RETRY ;
}
if ( outstanding_aio_calls > = aio_pending_size ) {
DEBUG ( 3 , ( " smb2: Already have %d aio "
" activities outstanding. \n " ,
outstanding_aio_calls ) ) ;
return NT_STATUS_RETRY ;
}
if ( ! ( aio_ex = create_aio_extra ( smbreq - > smb2req , fsp , 0 ) ) ) {
return NT_STATUS_NO_MEMORY ;
}
aio_ex - > handle_completion = handle_aio_smb2_write_complete ;
aio_ex - > write_through = write_through ;
2011-11-14 12:53:25 +04:00
aio_ex - > pass_cancel = true ;
2010-06-11 00:20:37 +04:00
init_strict_lock_struct ( fsp , ( uint64_t ) smbreq - > smbpid ,
in_offset , ( uint64_t ) in_data . length , WRITE_LOCK ,
& aio_ex - > lock ) ;
/* Take the lock until the AIO completes. */
if ( ! SMB_VFS_STRICT_LOCK ( conn , fsp , & aio_ex - > lock ) ) {
TALLOC_FREE ( aio_ex ) ;
return NT_STATUS_FILE_LOCK_CONFLICT ;
}
a = & aio_ex - > acb ;
/* Now set up the aio record for the write call. */
a - > aio_fildes = fsp - > fh - > fd ;
a - > aio_buf = in_data . data ;
a - > aio_nbytes = in_data . length ;
a - > aio_offset = in_offset ;
a - > aio_sigevent . sigev_notify = SIGEV_SIGNAL ;
a - > aio_sigevent . sigev_signo = RT_SIGNAL_AIO ;
a - > aio_sigevent . sigev_value . sival_ptr = aio_ex ;
ret = SMB_VFS_AIO_WRITE ( fsp , a ) ;
if ( ret = = - 1 ) {
DEBUG ( 3 , ( " smb2: aio_write failed. "
" Error %s \n " , strerror ( errno ) ) ) ;
SMB_VFS_STRICT_UNLOCK ( conn , fsp , & aio_ex - > lock ) ;
TALLOC_FREE ( aio_ex ) ;
return NT_STATUS_RETRY ;
}
outstanding_aio_calls + + ;
/* We don't need talloc_move here as both aio_ex and
* smbreq are children of smbreq - > smb2req . */
aio_ex - > smbreq = smbreq ;
2011-11-14 12:52:47 +04:00
smbreq - > async_priv = aio_ex ;
2010-06-11 00:20:37 +04:00
/* This should actually be improved to span the write. */
contend_level2_oplocks_begin ( fsp , LEVEL2_CONTEND_WRITE ) ;
contend_level2_oplocks_end ( fsp , LEVEL2_CONTEND_WRITE ) ;
/*
* We don ' t want to do write behind due to ownership
* issues of the request structs . Maybe add it if I
* figure those out . JRA .
*/
DEBUG ( 10 , ( " smb2: scheduled aio_write for file "
" %s, offset %.0f, len = %u (mid = %u) "
" outstanding_aio_calls = %d \n " ,
fsp_str_dbg ( fsp ) ,
( double ) in_offset ,
( unsigned int ) in_data . length ,
( unsigned int ) aio_ex - > smbreq - > mid ,
outstanding_aio_calls ) ) ;
return NT_STATUS_OK ;
}
2005-06-28 02:53:56 +04:00
/****************************************************************************
Complete the read and return the data or error back to the client .
Returns errno or zero if all ok .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-10-13 17:48:19 +04:00
static int handle_aio_read_complete ( struct aio_extra * aio_ex , int errcode )
2005-06-28 02:53:56 +04:00
{
int outsize ;
2010-06-02 21:25:56 +04:00
char * outbuf = ( char * ) aio_ex - > outbuf . data ;
2005-06-28 02:53:56 +04:00
char * data = smb_buf ( outbuf ) ;
ssize_t nread = SMB_VFS_AIO_RETURN ( aio_ex - > fsp , & aio_ex - > acb ) ;
if ( nread < 0 ) {
/* We're relying here on the fact that if the fd is
closed then the aio will complete and aio_return
will return an error . Hopefully this is
true . . . . JRA . */
2009-10-13 17:48:19 +04:00
DEBUG ( 3 , ( " handle_aio_read_complete: file %s nread == %d. "
2007-02-10 16:15:58 +03:00
" Error = %s \n " ,
2009-10-14 00:28:57 +04:00
fsp_str_dbg ( aio_ex - > fsp ) , ( int ) nread , strerror ( errcode ) ) ) ;
2005-06-28 02:53:56 +04:00
2009-10-14 00:28:57 +04:00
ERROR_NT ( map_nt_error_from_unix ( errcode ) ) ;
2008-01-04 23:56:23 +03:00
outsize = srv_set_message ( outbuf , 0 , 0 , true ) ;
2005-06-28 02:53:56 +04:00
} else {
2008-01-04 23:56:23 +03:00
outsize = srv_set_message ( outbuf , 12 , nread , False ) ;
2005-06-28 02:53:56 +04:00
SSVAL ( outbuf , smb_vwv2 , 0xFFFF ) ; /* Remaining - must be * -1. */
SSVAL ( outbuf , smb_vwv5 , nread ) ;
SSVAL ( outbuf , smb_vwv6 , smb_offset ( data , outbuf ) ) ;
SSVAL ( outbuf , smb_vwv7 , ( ( nread > > 16 ) & 1 ) ) ;
SSVAL ( smb_buf ( outbuf ) , - 2 , nread ) ;
2008-02-19 15:27:08 +03:00
aio_ex - > fsp - > fh - > pos = aio_ex - > acb . aio_offset + nread ;
aio_ex - > fsp - > fh - > position_information = aio_ex - > fsp - > fh - > pos ;
2007-02-10 16:15:58 +03:00
DEBUG ( 3 , ( " handle_aio_read_complete file %s max=%d "
" nread=%d \n " ,
2009-07-11 01:50:37 +04:00
fsp_str_dbg ( aio_ex - > fsp ) ,
2007-07-25 22:53:16 +04:00
( int ) aio_ex - > acb . aio_nbytes , ( int ) nread ) ) ;
2005-06-28 02:53:56 +04:00
}
2008-01-04 23:56:23 +03:00
smb_setlen ( outbuf , outsize - 4 ) ;
2005-06-28 02:53:56 +04:00
show_msg ( outbuf ) ;
2010-08-24 22:10:20 +04:00
if ( ! srv_send_smb ( aio_ex - > smbreq - > sconn , outbuf ,
2010-06-04 22:41:38 +04:00
true , aio_ex - > smbreq - > seqnum + 1 ,
2009-02-09 10:10:34 +03:00
IS_CONN_ENCRYPTED ( aio_ex - > fsp - > conn ) , NULL ) ) {
2008-01-04 23:56:23 +03:00
exit_server_cleanly ( " handle_aio_read_complete: srv_send_smb "
2007-02-10 16:15:58 +03:00
" failed. " ) ;
2005-06-28 02:53:56 +04:00
}
2007-02-10 16:15:58 +03:00
DEBUG ( 10 , ( " handle_aio_read_complete: scheduled aio_read completed "
" for file %s, offset %.0f, len = %u \n " ,
2009-07-11 01:50:37 +04:00
fsp_str_dbg ( aio_ex - > fsp ) , ( double ) aio_ex - > acb . aio_offset ,
2007-02-10 16:15:58 +03:00
( unsigned int ) nread ) ) ;
2005-06-28 02:53:56 +04:00
2009-10-14 00:28:57 +04:00
return errcode ;
2005-06-28 02:53:56 +04:00
}
/****************************************************************************
Complete the write and return the data or error back to the client .
2009-10-13 17:48:19 +04:00
Returns error code or zero if all ok .
2005-06-28 02:53:56 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2009-10-13 17:48:19 +04:00
static int handle_aio_write_complete ( struct aio_extra * aio_ex , int errcode )
2005-06-28 02:53:56 +04:00
{
files_struct * fsp = aio_ex - > fsp ;
2010-06-02 21:25:56 +04:00
char * outbuf = ( char * ) aio_ex - > outbuf . data ;
2005-06-28 02:53:56 +04:00
ssize_t numtowrite = aio_ex - > acb . aio_nbytes ;
ssize_t nwritten = SMB_VFS_AIO_RETURN ( fsp , & aio_ex - > acb ) ;
2007-10-11 00:34:30 +04:00
if ( fsp - > aio_write_behind ) {
if ( nwritten ! = numtowrite ) {
if ( nwritten = = - 1 ) {
DEBUG ( 5 , ( " handle_aio_write_complete: "
" aio_write_behind failed ! File %s "
" is corrupt ! Error %s \n " ,
2009-10-13 17:48:19 +04:00
fsp_str_dbg ( fsp ) , strerror ( errcode ) ) ) ;
2007-10-11 00:34:30 +04:00
} else {
DEBUG ( 0 , ( " handle_aio_write_complete: "
" aio_write_behind failed ! File %s "
" is corrupt ! Wanted %u bytes but "
2009-07-11 01:50:37 +04:00
" only wrote %d \n " , fsp_str_dbg ( fsp ) ,
2007-10-11 00:34:30 +04:00
( unsigned int ) numtowrite ,
( int ) nwritten ) ) ;
2009-10-14 00:28:57 +04:00
errcode = EIO ;
2007-10-11 00:34:30 +04:00
}
} else {
DEBUG ( 10 , ( " handle_aio_write_complete: "
" aio_write_behind completed for file %s \n " ,
2009-07-11 01:50:37 +04:00
fsp_str_dbg ( fsp ) ) ) ;
2007-10-11 00:34:30 +04:00
}
2009-10-13 17:48:19 +04:00
/* TODO: should no return 0 in case of an error !!! */
2007-10-11 00:34:30 +04:00
return 0 ;
}
2005-06-28 02:53:56 +04:00
/* We don't need outsize or set_message here as we've already set the
fixed size length when we set up the aio call . */
if ( nwritten = = - 1 ) {
2007-02-10 16:15:58 +03:00
DEBUG ( 3 , ( " handle_aio_write: file %s wanted %u bytes. "
" nwritten == %d. Error = %s \n " ,
2009-07-11 01:50:37 +04:00
fsp_str_dbg ( fsp ) , ( unsigned int ) numtowrite ,
2009-10-14 00:03:39 +04:00
( int ) nwritten , strerror ( errcode ) ) ) ;
2005-06-28 02:53:56 +04:00
2009-10-14 00:28:57 +04:00
ERROR_NT ( map_nt_error_from_unix ( errcode ) ) ;
2008-01-04 23:56:23 +03:00
srv_set_message ( outbuf , 0 , 0 , true ) ;
2005-06-28 02:53:56 +04:00
} else {
2007-06-15 23:24:04 +04:00
NTSTATUS status ;
2005-06-28 02:53:56 +04:00
SSVAL ( outbuf , smb_vwv2 , nwritten ) ;
SSVAL ( outbuf , smb_vwv4 , ( nwritten > > 16 ) & 1 ) ;
if ( nwritten < ( ssize_t ) numtowrite ) {
SCVAL ( outbuf , smb_rcls , ERRHRD ) ;
SSVAL ( outbuf , smb_err , ERRdiskfull ) ;
}
2007-02-10 16:15:58 +03:00
DEBUG ( 3 , ( " handle_aio_write: fnum=%d num=%d wrote=%d \n " ,
fsp - > fnum , ( int ) numtowrite , ( int ) nwritten ) ) ;
2010-06-08 03:25:18 +04:00
status = sync_file ( fsp - > conn , fsp , aio_ex - > write_through ) ;
2007-06-15 23:24:04 +04:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2009-10-14 00:28:57 +04:00
errcode = errno ;
ERROR_BOTH ( map_nt_error_from_unix ( errcode ) ,
2007-12-27 21:56:44 +03:00
ERRHRD , ERRdiskfull ) ;
2008-01-04 23:56:23 +03:00
srv_set_message ( outbuf , 0 , 0 , true ) ;
2007-06-15 23:24:04 +04:00
DEBUG ( 5 , ( " handle_aio_write: sync_file for %s returned %s \n " ,
2009-07-11 01:50:37 +04:00
fsp_str_dbg ( fsp ) , nt_errstr ( status ) ) ) ;
2007-06-15 23:24:04 +04:00
}
2008-02-19 15:27:08 +03:00
aio_ex - > fsp - > fh - > pos = aio_ex - > acb . aio_offset + nwritten ;
2005-06-28 02:53:56 +04:00
}
show_msg ( outbuf ) ;
2010-08-24 22:10:20 +04:00
if ( ! srv_send_smb ( aio_ex - > smbreq - > sconn , outbuf ,
2010-06-04 22:41:38 +04:00
true , aio_ex - > smbreq - > seqnum + 1 ,
2009-03-09 11:47:59 +03:00
IS_CONN_ENCRYPTED ( fsp - > conn ) ,
2009-02-09 10:10:34 +03:00
NULL ) ) {
2010-08-14 00:51:29 +04:00
exit_server_cleanly ( " handle_aio_write_complete: "
" srv_send_smb failed. " ) ;
2005-06-28 02:53:56 +04:00
}
2007-02-10 16:15:58 +03:00
DEBUG ( 10 , ( " handle_aio_write_complete: scheduled aio_write completed "
" for file %s, offset %.0f, requested %u, written = %u \n " ,
2009-07-11 01:50:37 +04:00
fsp_str_dbg ( fsp ) , ( double ) aio_ex - > acb . aio_offset ,
2007-02-10 16:15:58 +03:00
( unsigned int ) numtowrite , ( unsigned int ) nwritten ) ) ;
2005-06-28 02:53:56 +04:00
2009-10-14 00:28:57 +04:00
return errcode ;
2005-06-28 02:53:56 +04:00
}
2010-06-11 00:20:37 +04:00
/****************************************************************************
Complete the read and return the data or error back to the client .
Returns errno or zero if all ok .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int handle_aio_smb2_read_complete ( struct aio_extra * aio_ex , int errcode )
{
NTSTATUS status ;
struct tevent_req * subreq = aio_ex - > smbreq - > smb2req - > subreq ;
ssize_t nread = SMB_VFS_AIO_RETURN ( aio_ex - > fsp , & aio_ex - > acb ) ;
/* Common error or success code processing for async or sync
read returns . */
status = smb2_read_complete ( subreq , nread , errcode ) ;
if ( nread > 0 ) {
aio_ex - > fsp - > fh - > pos = aio_ex - > acb . aio_offset + nread ;
aio_ex - > fsp - > fh - > position_information = aio_ex - > fsp - > fh - > pos ;
}
DEBUG ( 10 , ( " smb2: scheduled aio_read completed "
" for file %s, offset %.0f, len = %u "
" (errcode = %d, NTSTATUS = %s) \n " ,
fsp_str_dbg ( aio_ex - > fsp ) ,
( double ) aio_ex - > acb . aio_offset ,
( unsigned int ) nread ,
errcode ,
nt_errstr ( status ) ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
tevent_req_nterror ( subreq , status ) ;
2011-08-02 23:16:26 +04:00
return errcode ;
2010-06-11 00:20:37 +04:00
}
tevent_req_done ( subreq ) ;
return errcode ;
}
/****************************************************************************
Complete the SMB2 write and return the data or error back to the client .
Returns error code or zero if all ok .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int handle_aio_smb2_write_complete ( struct aio_extra * aio_ex , int errcode )
{
files_struct * fsp = aio_ex - > fsp ;
ssize_t numtowrite = aio_ex - > acb . aio_nbytes ;
ssize_t nwritten = SMB_VFS_AIO_RETURN ( fsp , & aio_ex - > acb ) ;
struct tevent_req * subreq = aio_ex - > smbreq - > smb2req - > subreq ;
NTSTATUS status ;
status = smb2_write_complete ( subreq , nwritten , errcode ) ;
DEBUG ( 10 , ( " smb2: scheduled aio_write completed "
" for file %s, offset %.0f, requested %u, "
" written = %u (errcode = %d, NTSTATUS = %s) \n " ,
fsp_str_dbg ( fsp ) ,
( double ) aio_ex - > acb . aio_offset ,
( unsigned int ) numtowrite ,
( unsigned int ) nwritten ,
errcode ,
nt_errstr ( status ) ) ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
tevent_req_nterror ( subreq , status ) ;
2011-08-02 23:16:26 +04:00
return errcode ;
2010-06-11 00:20:37 +04:00
}
tevent_req_done ( subreq ) ;
return errcode ;
}
2005-06-28 02:53:56 +04:00
/****************************************************************************
2007-02-10 16:15:58 +03:00
Handle any aio completion . Returns True if finished ( and sets * perr if err
was non - zero ) , False if not .
2005-06-28 02:53:56 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-19 04:40:25 +04:00
static bool handle_aio_completed ( struct aio_extra * aio_ex , int * perr )
2005-06-28 02:53:56 +04:00
{
2010-04-06 01:16:21 +04:00
files_struct * fsp = NULL ;
2005-06-28 02:53:56 +04:00
int err ;
2008-09-16 05:45:10 +04:00
if ( ! aio_ex ) {
DEBUG ( 3 , ( " handle_aio_completed: Non-existing aio_ex passed \n " ) ) ;
return false ;
}
2011-11-14 12:54:05 +04:00
if ( ! aio_ex - > fsp ) {
DEBUG ( 3 , ( " handle_aio_completed: aio_ex->fsp == NULL \n " ) ) ;
return false ;
}
2010-04-06 01:16:21 +04:00
fsp = aio_ex - > fsp ;
2005-06-28 02:53:56 +04:00
/* Ensure the operation has really completed. */
2010-04-06 01:16:21 +04:00
err = SMB_VFS_AIO_ERROR ( fsp , & aio_ex - > acb ) ;
2009-10-13 17:48:19 +04:00
if ( err = = EINPROGRESS ) {
2010-04-13 08:40:28 +04:00
DEBUG ( 10 , ( " handle_aio_completed: operation mid %llu still in "
" process for file %s \n " ,
2010-06-04 22:41:38 +04:00
( unsigned long long ) aio_ex - > smbreq - > mid ,
2010-04-13 08:40:28 +04:00
fsp_str_dbg ( aio_ex - > fsp ) ) ) ;
2005-06-28 02:53:56 +04:00
return False ;
2010-04-06 01:16:21 +04:00
}
/* Unlock now we're done. */
SMB_VFS_STRICT_UNLOCK ( fsp - > conn , fsp , & aio_ex - > lock ) ;
2011-11-14 12:53:25 +04:00
if ( ! aio_ex - > pass_cancel & & err = = ECANCELED ) {
2009-10-13 17:48:19 +04:00
/* If error is ECANCELED then don't return anything to the
* client . */
2010-04-13 08:40:28 +04:00
DEBUG ( 10 , ( " handle_aio_completed: operation mid %llu "
" canceled \n " ,
2010-06-04 22:41:38 +04:00
( unsigned long long ) aio_ex - > smbreq - > mid ) ) ;
2009-10-13 17:48:19 +04:00
return True ;
}
2005-06-28 02:53:56 +04:00
2009-10-13 17:48:19 +04:00
err = aio_ex - > handle_completion ( aio_ex , err ) ;
2005-06-28 02:53:56 +04:00
if ( err ) {
* perr = err ; /* Only save non-zero errors. */
}
return True ;
}
/****************************************************************************
Handle any aio completion inline .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-06-04 22:30:46 +04:00
void smbd_aio_complete_aio_ex ( struct aio_extra * aio_ex )
2005-06-28 02:53:56 +04:00
{
2009-01-22 20:04:17 +03:00
files_struct * fsp = NULL ;
2005-06-28 02:53:56 +04:00
int ret = 0 ;
2009-10-01 02:20:09 +04:00
outstanding_aio_calls - - ;
2010-04-13 08:40:28 +04:00
DEBUG ( 10 , ( " smbd_aio_complete_mid: mid[%llu] \n " ,
2010-06-04 22:41:38 +04:00
( unsigned long long ) aio_ex - > smbreq - > mid ) ) ;
2005-06-28 02:53:56 +04:00
2009-01-22 20:04:17 +03:00
fsp = aio_ex - > fsp ;
if ( fsp = = NULL ) {
/* file was closed whilst I/O was outstanding. Just
* ignore . */
DEBUG ( 3 , ( " smbd_aio_complete_mid: file closed whilst "
2010-04-13 08:40:28 +04:00
" aio outstanding (mid[%llu]). \n " ,
2010-06-04 22:41:38 +04:00
( unsigned long long ) aio_ex - > smbreq - > mid ) ) ;
2009-01-22 20:04:17 +03:00
return ;
}
2005-06-28 02:53:56 +04:00
2009-01-22 20:04:17 +03:00
if ( ! handle_aio_completed ( aio_ex , & ret ) ) {
return ;
}
}
2005-06-28 02:53:56 +04:00
2007-10-11 00:34:30 +04:00
/****************************************************************************
We ' re doing write behind and the client closed the file . Wait up to 30
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 29
int wait_for_aio_completion ( files_struct * fsp )
{
struct aio_extra * aio_ex ;
const SMB_STRUCT_AIOCB * * aiocb_list ;
int aio_completion_count = 0 ;
2010-09-07 05:29:19 +04:00
time_t start_time = time_mono ( NULL ) ;
2007-10-11 00:34:30 +04:00
int seconds_left ;
for ( seconds_left = SMB_TIME_FOR_AIO_COMPLETE_WAIT ;
seconds_left > = 0 ; ) {
int err = 0 ;
int i ;
struct timespec ts ;
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 ) ) ) ;
2008-11-03 22:07:11 +03:00
2007-10-11 00:34:30 +04:00
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. */
cancel_aio_by_fsp ( fsp ) ;
SAFE_FREE ( aiocb_list ) ;
return EIO ;
}
/* One or more events might have completed - process them if
* so . */
for ( i = 0 ; i < aio_completion_count ; i + + ) {
2010-06-02 03:05:44 +04:00
aio_ex = ( struct aio_extra * ) aiocb_list [ i ] - > aio_sigevent . sigev_value . sival_ptr ;
2007-10-11 00:34:30 +04:00
if ( ! handle_aio_completed ( aio_ex , & err ) ) {
continue ;
}
2008-11-03 21:59:11 +03:00
TALLOC_FREE ( aio_ex ) ;
2007-10-11 00:34:30 +04:00
}
SAFE_FREE ( aiocb_list ) ;
seconds_left = SMB_TIME_FOR_AIO_COMPLETE_WAIT
2010-09-07 05:29:19 +04:00
- ( time_mono ( NULL ) - start_time ) ;
2007-10-11 00:34:30 +04:00
}
/* 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 ;
}
2005-06-28 02:53:56 +04:00
/****************************************************************************
Cancel any outstanding aio requests . The client doesn ' t care about the reply .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void cancel_aio_by_fsp ( files_struct * fsp )
{
struct aio_extra * aio_ex ;
for ( aio_ex = aio_list_head ; aio_ex ; aio_ex = aio_ex - > next ) {
if ( aio_ex - > fsp = = fsp ) {
2010-04-06 01:16:21 +04:00
/* Unlock now we're done. */
SMB_VFS_STRICT_UNLOCK ( fsp - > conn , fsp , & aio_ex - > lock ) ;
2007-02-10 16:15:58 +03:00
/* Don't delete the aio_extra record as we may have
completed and don ' t yet know it . Just do the
aio_cancel call and return . */
2008-01-08 14:20:51 +03:00
SMB_VFS_AIO_CANCEL ( fsp , & aio_ex - > acb ) ;
2007-02-10 16:15:58 +03:00
aio_ex - > fsp = NULL ; /* fsp will be closed when we
* return . */
2005-06-28 02:53:56 +04:00
}
}
}
2009-01-22 20:04:17 +03:00
# else
2010-04-06 01:16:21 +04:00
NTSTATUS schedule_aio_read_and_X ( connection_struct * conn ,
2010-06-04 22:41:38 +04:00
struct smb_request * smbreq ,
2005-06-28 02:53:56 +04:00
files_struct * fsp , SMB_OFF_T startpos ,
size_t smb_maxcnt )
{
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2005-06-28 02:53:56 +04:00
}
2010-04-06 01:16:21 +04:00
NTSTATUS schedule_aio_write_and_X ( connection_struct * conn ,
2010-06-04 22:41:38 +04:00
struct smb_request * smbreq ,
2011-05-05 22:38:06 +04:00
files_struct * fsp , const char * data ,
2007-08-08 23:05:30 +04:00
SMB_OFF_T startpos ,
size_t numtowrite )
2005-06-28 02:53:56 +04:00
{
2010-04-06 01:16:21 +04:00
return NT_STATUS_RETRY ;
2005-06-28 02:53:56 +04:00
}
2011-11-14 12:52:47 +04:00
bool cancel_smb2_aio ( struct smb_request * smbreq )
{
return false ;
}
2010-06-11 01:28:55 +04:00
NTSTATUS schedule_smb2_aio_read ( connection_struct * conn ,
struct smb_request * smbreq ,
files_struct * fsp ,
2010-12-15 03:32:10 +03:00
TALLOC_CTX * ctx ,
DATA_BLOB * preadbuf ,
2010-06-11 01:28:55 +04:00
SMB_OFF_T startpos ,
size_t smb_maxcnt )
{
return NT_STATUS_RETRY ;
}
NTSTATUS schedule_aio_smb2_write ( connection_struct * conn ,
struct smb_request * smbreq ,
files_struct * fsp ,
uint64_t in_offset ,
DATA_BLOB in_data ,
bool write_through )
{
return NT_STATUS_RETRY ;
}
2005-06-28 02:53:56 +04:00
void cancel_aio_by_fsp ( files_struct * fsp )
{
}
2007-10-19 04:40:25 +04:00
int wait_for_aio_completion ( files_struct * fsp )
2007-10-11 00:34:30 +04:00
{
2010-06-04 22:41:57 +04:00
return 0 ;
2007-10-11 00:34:30 +04:00
}
2009-01-22 20:04:17 +03:00
2010-04-13 08:40:28 +04:00
void smbd_aio_complete_mid ( uint64_t mid ) ;
2009-01-22 20:04:17 +03:00
2005-06-28 02:53:56 +04:00
# endif