2002-09-12 06:12:52 +04:00
/*
Unix SMB / Netbios implementation .
Version 2.2 . x / 3.0 . x
sendfile implementations .
Copyright ( C ) Jeremy Allison 2002.
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
2002-09-12 06:12:52 +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 09:23:25 +04:00
along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2002-09-12 06:12:52 +04:00
*/
/*
* This file handles the OS dependent sendfile implementations .
* The API is such that it returns - 1 on error , else returns the
* number of bytes written .
*/
# include "includes.h"
# if defined(LINUX_SENDFILE_API)
# include <sys/sendfile.h>
# ifndef MSG_MORE
# define MSG_MORE 0x8000
# endif
2002-09-12 22:18:35 +04:00
ssize_t sys_sendfile ( int tofd , int fromfd , const DATA_BLOB * header , SMB_OFF_T offset , size_t count )
2002-09-12 06:12:52 +04:00
{
size_t total = 0 ;
ssize_t ret ;
2003-01-24 19:34:54 +03:00
size_t hdr_len = 0 ;
2002-09-12 06:12:52 +04:00
/*
* Send the header first .
* Use MSG_MORE to cork the TCP output until sendfile is called .
*/
2002-09-12 10:24:41 +04:00
if ( header ) {
hdr_len = header - > length ;
2002-09-17 23:48:19 +04:00
while ( total < hdr_len ) {
2002-09-12 22:18:35 +04:00
ret = sys_send ( tofd , header - > data + total , hdr_len - total , MSG_MORE ) ;
2002-09-12 10:24:41 +04:00
if ( ret = = - 1 )
return - 1 ;
total + = ret ;
}
2002-09-12 06:12:52 +04:00
}
total = count ;
while ( total ) {
ssize_t nwritten ;
do {
2002-09-14 05:38:55 +04:00
# if defined(HAVE_EXPLICIT_LARGEFILE_SUPPORT) && defined(HAVE_OFF64_T) && defined(HAVE_SENDFILE64)
2002-09-12 22:18:35 +04:00
nwritten = sendfile64 ( tofd , fromfd , & offset , total ) ;
2002-09-12 06:12:52 +04:00
# else
2002-09-12 22:18:35 +04:00
nwritten = sendfile ( tofd , fromfd , & offset , total ) ;
2002-09-12 06:12:52 +04:00
# endif
} while ( nwritten = = - 1 & & errno = = EINTR ) ;
2004-11-25 01:05:59 +03:00
if ( nwritten = = - 1 ) {
2009-01-13 22:19:11 +03:00
if ( errno = = ENOSYS | | errno = = EINVAL ) {
2004-11-25 01:05:59 +03:00
/* Ok - we're in a world of pain here. We just sent
* the header , but the sendfile failed . We have to
* emulate the sendfile at an upper layer before we
* disable it ' s use . So we do something really ugly .
* We set the errno to a strange value so we can detect
* this at the upper level and take care of it without
* layer violation . JRA .
*/
errno = EINTR ; /* Normally we can never return this. */
}
2002-09-12 06:12:52 +04:00
return - 1 ;
2004-11-25 01:05:59 +03:00
}
2009-01-14 00:02:44 +03:00
if ( nwritten = = 0 ) {
/*
* EOF , return a short read
*/
return hdr_len + ( count - total ) ;
}
2002-09-12 06:12:52 +04:00
total - = nwritten ;
}
2002-09-12 10:24:41 +04:00
return count + hdr_len ;
2002-09-12 06:12:52 +04:00
}
2002-09-14 04:34:46 +04:00
# elif defined(LINUX_BROKEN_SENDFILE_API)
2002-09-17 05:00:03 +04:00
/*
* We must use explicit 32 bit types here . This code path means Linux
* won ' t do proper 64 - bit sendfile . JRA .
*/
extern int32 sendfile ( int out_fd , int in_fd , int32 * offset , uint32 count ) ;
2002-09-14 04:34:46 +04:00
# ifndef MSG_MORE
# define MSG_MORE 0x8000
# endif
ssize_t sys_sendfile ( int tofd , int fromfd , const DATA_BLOB * header , SMB_OFF_T offset , size_t count )
{
size_t total = 0 ;
ssize_t ret ;
ssize_t hdr_len = 0 ;
2002-09-17 05:00:03 +04:00
uint32 small_total = 0 ;
int32 small_offset ;
2002-09-14 04:34:46 +04:00
/*
* Fix for broken Linux 2.4 systems with no working sendfile64 ( ) .
* If the offset + count > 2 GB then pretend we don ' t have the
2002-09-17 05:00:03 +04:00
* system call sendfile at all . The upper layer catches this
2002-09-14 04:34:46 +04:00
* and uses a normal read . JRA .
*/
if ( ( sizeof ( SMB_OFF_T ) > = 8 ) & & ( offset + count > ( SMB_OFF_T ) 0x7FFFFFFF ) ) {
errno = ENOSYS ;
return - 1 ;
}
/*
* Send the header first .
* Use MSG_MORE to cork the TCP output until sendfile is called .
*/
if ( header ) {
hdr_len = header - > length ;
2002-09-17 23:48:19 +04:00
while ( total < hdr_len ) {
2002-09-14 04:34:46 +04:00
ret = sys_send ( tofd , header - > data + total , hdr_len - total , MSG_MORE ) ;
if ( ret = = - 1 )
return - 1 ;
total + = ret ;
}
}
2002-09-17 05:00:03 +04:00
small_total = ( uint32 ) count ;
small_offset = ( int32 ) offset ;
while ( small_total ) {
int32 nwritten ;
2002-09-14 04:34:46 +04:00
do {
2002-09-17 05:00:03 +04:00
nwritten = sendfile ( tofd , fromfd , & small_offset , small_total ) ;
2002-09-14 04:34:46 +04:00
} while ( nwritten = = - 1 & & errno = = EINTR ) ;
2004-11-25 01:05:59 +03:00
if ( nwritten = = - 1 ) {
2009-01-13 22:19:11 +03:00
if ( errno = = ENOSYS | | errno = = EINVAL ) {
2004-11-25 01:05:59 +03:00
/* Ok - we're in a world of pain here. We just sent
* the header , but the sendfile failed . We have to
* emulate the sendfile at an upper layer before we
* disable it ' s use . So we do something really ugly .
* We set the errno to a strange value so we can detect
* this at the upper level and take care of it without
* layer violation . JRA .
*/
errno = EINTR ; /* Normally we can never return this. */
}
2002-09-14 04:34:46 +04:00
return - 1 ;
2004-11-25 01:05:59 +03:00
}
2009-01-14 00:02:44 +03:00
if ( nwritten = = 0 ) {
/*
* EOF , return a short read
*/
return hdr_len + ( ( ( uint32 ) count ) - small_total ) ;
}
2002-09-17 05:00:03 +04:00
small_total - = nwritten ;
2002-09-14 04:34:46 +04:00
}
return count + hdr_len ;
}
2002-09-12 06:12:52 +04:00
# elif defined(SOLARIS_SENDFILE_API)
2002-10-01 01:17:18 +04:00
/*
2002-10-01 09:54:40 +04:00
* Solaris sendfile code written by Pierre Belanger < belanger @ pobox . com > .
2002-10-01 01:17:18 +04:00
*/
# include <sys/sendfile.h>
2002-09-12 22:18:35 +04:00
ssize_t sys_sendfile ( int tofd , int fromfd , const DATA_BLOB * header , SMB_OFF_T offset , size_t count )
2002-09-12 06:12:52 +04:00
{
2002-10-01 01:17:18 +04:00
int sfvcnt ;
size_t total , xferred ;
struct sendfilevec vec [ 2 ] ;
ssize_t hdr_len = 0 ;
if ( header ) {
sfvcnt = 2 ;
vec [ 0 ] . sfv_fd = SFV_FD_SELF ;
vec [ 0 ] . sfv_flag = 0 ;
2003-11-02 16:59:37 +03:00
vec [ 0 ] . sfv_off = ( off_t ) header - > data ;
2002-10-01 09:54:40 +04:00
vec [ 0 ] . sfv_len = hdr_len = header - > length ;
2002-10-01 01:17:18 +04:00
vec [ 1 ] . sfv_fd = fromfd ;
vec [ 1 ] . sfv_flag = 0 ;
vec [ 1 ] . sfv_off = offset ;
vec [ 1 ] . sfv_len = count ;
} else {
sfvcnt = 1 ;
vec [ 0 ] . sfv_fd = fromfd ;
vec [ 0 ] . sfv_flag = 0 ;
vec [ 0 ] . sfv_off = offset ;
vec [ 0 ] . sfv_len = count ;
}
total = count + hdr_len ;
while ( total ) {
ssize_t nwritten ;
/*
* Although not listed in the API error returns , this is almost certainly
* a slow system call and will be interrupted by a signal with EINTR . JRA .
*/
xferred = 0 ;
# if defined(HAVE_EXPLICIT_LARGEFILE_SUPPORT) && defined(HAVE_OFF64_T) && defined(HAVE_SENDFILEV64)
nwritten = sendfilev64 ( tofd , vec , sfvcnt , & xferred ) ;
# else
nwritten = sendfilev ( tofd , vec , sfvcnt , & xferred ) ;
# endif
if ( nwritten = = - 1 & & errno = = EINTR ) {
if ( xferred = = 0 )
continue ; /* Nothing written yet. */
else
nwritten = xferred ;
}
if ( nwritten = = - 1 )
return - 1 ;
if ( nwritten = = 0 )
return - 1 ; /* I think we're at EOF here... */
/*
* If this was a short ( signal interrupted ) write we may need
* to subtract it from the header data , or null out the header
* data altogether if we wrote more than vec [ 0 ] . sfv_len bytes .
* We move vec [ 1 ] . * to vec [ 0 ] . * and set sfvcnt to 1
*/
if ( sfvcnt = = 2 & & nwritten > = vec [ 0 ] . sfv_len ) {
vec [ 1 ] . sfv_off + = nwritten - vec [ 0 ] . sfv_len ;
vec [ 1 ] . sfv_len - = nwritten - vec [ 0 ] . sfv_len ;
/* Move vec[1].* to vec[0].* and set sfvcnt to 1 */
vec [ 0 ] = vec [ 1 ] ;
sfvcnt = 1 ;
} else {
vec [ 0 ] . sfv_off + = nwritten ;
vec [ 0 ] . sfv_len - = nwritten ;
}
total - = nwritten ;
}
return count + hdr_len ;
2002-09-12 06:12:52 +04:00
}
# elif defined(HPUX_SENDFILE_API)
# include <sys/socket.h>
# include <sys/uio.h>
2002-09-12 22:18:35 +04:00
ssize_t sys_sendfile ( int tofd , int fromfd , const DATA_BLOB * header , SMB_OFF_T offset , size_t count )
2002-09-12 06:12:52 +04:00
{
size_t total = 0 ;
struct iovec hdtrl [ 2 ] ;
2002-09-12 10:24:41 +04:00
size_t hdr_len = 0 ;
if ( header ) {
/* Set up the header/trailer iovec. */
2009-05-12 22:45:37 +04:00
hdtrl [ 0 ] . iov_base = ( void * ) header - > data ;
2002-09-12 10:24:41 +04:00
hdtrl [ 0 ] . iov_len = hdr_len = header - > length ;
} else {
hdtrl [ 0 ] . iov_base = NULL ;
hdtrl [ 0 ] . iov_len = hdr_len = 0 ;
}
2002-09-12 06:12:52 +04:00
hdtrl [ 1 ] . iov_base = NULL ;
2004-12-21 01:33:37 +03:00
hdtrl [ 1 ] . iov_len = 0 ;
2002-09-12 06:12:52 +04:00
total = count ;
2002-09-12 10:24:41 +04:00
while ( total + hdtrl [ 0 ] . iov_len ) {
2002-09-12 06:12:52 +04:00
ssize_t nwritten ;
2002-09-12 10:24:41 +04:00
/*
* HPUX guarantees that if any data was written before
* a signal interrupt then sendfile returns the number of
* bytes written ( which may be less than requested ) not - 1.
* nwritten includes the header data sent .
*/
2002-09-12 06:12:52 +04:00
do {
2002-09-14 05:38:55 +04:00
# if defined(HAVE_EXPLICIT_LARGEFILE_SUPPORT) && defined(HAVE_OFF64_T) && defined(HAVE_SENDFILE64)
2002-09-12 22:18:35 +04:00
nwritten = sendfile64 ( tofd , fromfd , offset , total , & hdtrl [ 0 ] , 0 ) ;
2002-09-12 06:12:52 +04:00
# else
2002-09-12 22:18:35 +04:00
nwritten = sendfile ( tofd , fromfd , offset , total , & hdtrl [ 0 ] , 0 ) ;
2002-09-12 06:12:52 +04:00
# endif
} while ( nwritten = = - 1 & & errno = = EINTR ) ;
if ( nwritten = = - 1 )
return - 1 ;
2002-09-12 10:24:41 +04:00
if ( nwritten = = 0 )
return - 1 ; /* I think we're at EOF here... */
/*
* If this was a short ( signal interrupted ) write we may need
* to subtract it from the header data , or null out the header
* data altogether if we wrote more than hdtrl [ 0 ] . iov_len bytes .
* We change nwritten to be the number of file bytes written .
*/
if ( hdtrl [ 0 ] . iov_base & & hdtrl [ 0 ] . iov_len ) {
if ( nwritten > = hdtrl [ 0 ] . iov_len ) {
nwritten - = hdtrl [ 0 ] . iov_len ;
hdtrl [ 0 ] . iov_base = NULL ;
hdtrl [ 0 ] . iov_len = 0 ;
} else {
2002-09-19 00:07:56 +04:00
/* iov_base is defined as a void *... */
2009-05-12 22:45:37 +04:00
hdtrl [ 0 ] . iov_base = ( void * ) ( ( ( char * ) hdtrl [ 0 ] . iov_base ) + nwritten ) ;
2002-09-12 10:24:41 +04:00
hdtrl [ 0 ] . iov_len - = nwritten ;
2002-09-30 09:24:12 +04:00
nwritten = 0 ;
2002-09-12 10:24:41 +04:00
}
}
2002-09-12 06:12:52 +04:00
total - = nwritten ;
2002-09-12 10:24:41 +04:00
offset + = nwritten ;
2002-09-12 06:12:52 +04:00
}
2002-09-12 10:24:41 +04:00
return count + hdr_len ;
2002-09-12 06:12:52 +04:00
}
# elif defined(FREEBSD_SENDFILE_API)
# include <sys/types.h>
# include <sys/socket.h>
# include <sys/uio.h>
2002-09-12 22:18:35 +04:00
ssize_t sys_sendfile ( int tofd , int fromfd , const DATA_BLOB * header , SMB_OFF_T offset , size_t count )
2002-09-12 06:12:52 +04:00
{
size_t total = 0 ;
struct sf_hdtr hdr ;
struct iovec hdtrl ;
2002-09-12 22:18:35 +04:00
size_t hdr_len = 0 ;
2002-09-12 06:12:52 +04:00
2002-09-21 12:53:05 +04:00
hdr . headers = & hdtrl ;
hdr . hdr_cnt = 1 ;
hdr . trailers = NULL ;
hdr . trl_cnt = 0 ;
2002-09-12 06:12:52 +04:00
/* Set up the header iovec. */
2002-09-12 22:18:35 +04:00
if ( header ) {
2009-05-12 22:45:37 +04:00
hdtrl . iov_base = ( void * ) header - > data ;
2002-09-12 22:18:35 +04:00
hdtrl . iov_len = hdr_len = header - > length ;
} else {
hdtrl . iov_base = NULL ;
hdtrl . iov_len = 0 ;
}
2002-09-12 06:12:52 +04:00
total = count ;
2002-09-12 22:18:35 +04:00
while ( total + hdtrl . iov_len ) {
SMB_OFF_T nwritten ;
int ret ;
/*
* FreeBSD sendfile returns 0 on success , - 1 on error .
* Remember , the tofd and fromfd are reversed . . . . . : - ) .
* nwritten includes the header data sent .
*/
2002-09-12 06:12:52 +04:00
do {
2002-09-12 22:18:35 +04:00
ret = sendfile ( fromfd , tofd , offset , total , & hdr , & nwritten , 0 ) ;
} while ( ret = = - 1 & & errno = = EINTR ) ;
if ( ret = = - 1 )
2002-09-12 06:12:52 +04:00
return - 1 ;
2002-09-12 22:18:35 +04:00
if ( nwritten = = 0 )
return - 1 ; /* I think we're at EOF here... */
/*
* If this was a short ( signal interrupted ) write we may need
* to subtract it from the header data , or null out the header
* data altogether if we wrote more than hdtrl . iov_len bytes .
* We change nwritten to be the number of file bytes written .
*/
2002-09-21 12:53:05 +04:00
if ( hdtrl . iov_base & & hdtrl . iov_len ) {
2002-09-12 22:18:35 +04:00
if ( nwritten > = hdtrl . iov_len ) {
nwritten - = hdtrl . iov_len ;
hdtrl . iov_base = NULL ;
hdtrl . iov_len = 0 ;
} else {
2008-07-04 09:53:42 +04:00
hdtrl . iov_base =
2009-05-12 22:45:37 +04:00
( void * ) ( ( caddr_t ) hdtrl . iov_base + nwritten ) ;
2002-09-12 22:18:35 +04:00
hdtrl . iov_len - = nwritten ;
2002-09-30 09:24:12 +04:00
nwritten = 0 ;
2002-09-12 22:18:35 +04:00
}
}
2002-09-12 06:12:52 +04:00
total - = nwritten ;
2002-09-12 22:18:35 +04:00
offset + = nwritten ;
2002-09-12 06:12:52 +04:00
}
2002-09-12 22:18:35 +04:00
return count + hdr_len ;
2002-09-12 06:12:52 +04:00
}
2004-12-21 01:33:37 +03:00
# elif defined(AIX_SENDFILE_API)
/* BEGIN AIX SEND_FILE */
/* Contributed by William Jojo <jojowil@hvcc.edu> */
# include <sys/socket.h>
ssize_t sys_sendfile ( int tofd , int fromfd , const DATA_BLOB * header , SMB_OFF_T offset , size_t count )
{
struct sf_parms hdtrl ;
/* Set up the header/trailer struct params. */
if ( header ) {
hdtrl . header_data = header - > data ;
hdtrl . header_length = header - > length ;
} else {
hdtrl . header_data = NULL ;
hdtrl . header_length = 0 ;
}
hdtrl . trailer_data = NULL ;
hdtrl . trailer_length = 0 ;
hdtrl . file_descriptor = fromfd ;
hdtrl . file_offset = offset ;
hdtrl . file_bytes = count ;
while ( hdtrl . file_bytes + hdtrl . header_length ) {
ssize_t ret ;
/*
Return Value
There are three possible return values from send_file :
Value Description
- 1 an error has occurred , errno contains the error code .
0 the command has completed successfully .
1 the command was completed partially , some data has been
transmitted but the command has to return for some reason ,
for example , the command was interrupted by signals .
*/
do {
ret = send_file ( & tofd , & hdtrl , 0 ) ;
} while ( ( ret = = 1 ) | | ( ret = = - 1 & & errno = = EINTR ) ) ;
if ( ret = = - 1 )
return - 1 ;
}
return count + header - > length ;
}
/* END AIX SEND_FILE */
2002-09-12 06:12:52 +04:00
# else /* No sendfile implementation. Return error. */
2002-09-12 22:18:35 +04:00
ssize_t sys_sendfile ( int tofd , int fromfd , const DATA_BLOB * header , SMB_OFF_T offset , size_t count )
2002-09-12 06:12:52 +04:00
{
/* No sendfile syscall. */
errno = ENOSYS ;
return - 1 ;
}
# endif