2001-03-11 00:32:10 +00:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
2000-06-10 14:29:31 +00:00
IRIX kernel oplock processing
2000-06-10 13:38:07 +00:00
Copyright ( C ) Andrew Tridgell 1992 - 1998
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 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
2000-06-10 13:38:07 +00: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 00:52:41 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2000-06-10 13:38:07 +00:00
*/
2006-05-05 02:06:37 +00:00
# define DBGC_CLASS DBGC_LOCKING
2000-06-10 13:38:07 +00:00
# include "includes.h"
2011-03-30 16:18:38 +02:00
# include "system/filesys.h"
2011-03-22 16:57:01 +01:00
# include "smbd/smbd.h"
2009-01-08 12:03:45 +01:00
# include "smbd/globals.h"
2000-06-10 13:38:07 +00:00
2000-06-11 05:57:58 +00:00
# if HAVE_KERNEL_OPLOCKS_IRIX
2000-06-10 13:38:07 +00:00
2009-01-09 14:02:18 +01:00
struct irix_oplocks_context {
struct kernel_oplocks * ctx ;
2011-12-13 13:38:41 +01:00
struct smbd_server_connection * sconn ;
2009-01-09 14:02:18 +01:00
int write_fd ;
int read_fd ;
struct fd_event * read_fde ;
bool pending ;
} ;
2000-06-10 13:38:07 +00:00
/****************************************************************************
2001-10-20 21:59:34 +00:00
Test to see if IRIX kernel oplocks work .
2000-06-10 13:38:07 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-20 21:59:34 +00:00
2007-10-18 17:40:25 -07:00
static bool irix_oplocks_available ( void )
2000-06-10 13:38:07 +00:00
{
int fd ;
int pfd [ 2 ] ;
2007-09-14 01:07:57 +00:00
TALLOC_CTX * ctx = talloc_stackframe ( ) ;
char * tmpname = NULL ;
2000-06-10 13:38:07 +00:00
2006-03-21 02:56:49 +00:00
set_effective_capability ( KERNEL_OPLOCK_CAPABILITY ) ;
2000-06-10 13:38:07 +00:00
2007-09-14 01:07:57 +00:00
tmpname = talloc_asprintf ( ctx ,
" %s/koplock.%d " ,
lp_lockdir ( ) ,
( int ) sys_getpid ( ) ) ;
if ( ! tmpname ) {
TALLOC_FREE ( ctx ) ;
return False ;
}
2000-06-10 13:38:07 +00:00
if ( pipe ( pfd ) ! = 0 ) {
2007-01-16 15:50:25 +00:00
DEBUG ( 0 , ( " check_kernel_oplocks: Unable to create pipe. Error "
" was %s \n " ,
2000-06-10 13:38:07 +00:00
strerror ( errno ) ) ) ;
2007-09-14 01:07:57 +00:00
TALLOC_FREE ( ctx ) ;
2000-06-10 13:38:07 +00:00
return False ;
}
if ( ( fd = sys_open ( tmpname , O_RDWR | O_CREAT | O_EXCL | O_TRUNC , 0600 ) ) < 0 ) {
2007-01-16 15:50:25 +00:00
DEBUG ( 0 , ( " check_kernel_oplocks: Unable to open temp test file "
" %s. Error was %s \n " ,
2000-06-10 13:38:07 +00:00
tmpname , strerror ( errno ) ) ) ;
unlink ( tmpname ) ;
close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
2007-09-14 01:07:57 +00:00
TALLOC_FREE ( ctx ) ;
2000-06-10 13:38:07 +00:00
return False ;
}
unlink ( tmpname ) ;
2007-09-14 01:07:57 +00:00
TALLOC_FREE ( ctx ) ;
2002-07-15 10:35:28 +00:00
if ( sys_fcntl_long ( fd , F_OPLKREG , pfd [ 1 ] ) = = - 1 ) {
2007-01-16 15:50:25 +00:00
DEBUG ( 0 , ( " check_kernel_oplocks: Kernel oplocks are not "
" available on this machine. Disabling kernel oplock "
" support. \n " ) ) ;
2000-06-10 13:38:07 +00:00
close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
close ( fd ) ;
return False ;
}
2002-07-15 10:35:28 +00:00
if ( sys_fcntl_long ( fd , F_OPLKACK , OP_REVOKE ) < 0 ) {
2007-01-16 15:50:25 +00:00
DEBUG ( 0 , ( " check_kernel_oplocks: Error when removing kernel "
" oplock. Error was %s. Disabling kernel oplock "
" support. \n " , strerror ( errno ) ) ) ;
2000-06-10 13:38:07 +00:00
close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
close ( fd ) ;
return False ;
}
close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
close ( fd ) ;
return True ;
}
2009-02-15 23:38:53 -08:00
/*
* This is bad because the file_id should always be created through the vfs
* layer ! Unfortunately , a conn struct isn ' t available here .
*/
static struct file_id file_id_create_dev ( SMB_DEV_T dev , SMB_INO_T inode )
{
struct file_id key ;
/* the ZERO_STRUCT ensures padding doesn't break using the key as a
* blob */
ZERO_STRUCT ( key ) ;
key . devid = dev ;
key . inode = inode ;
return key ;
}
2000-06-10 13:38:07 +00:00
/****************************************************************************
* Deal with the IRIX kernel < - - > smbd
* oplock break protocol .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-20 21:59:34 +00:00
2009-01-09 14:02:18 +01:00
static files_struct * irix_oplock_receive_message ( struct kernel_oplocks * _ctx )
2000-06-10 13:38:07 +00:00
{
2009-01-09 14:02:18 +01:00
struct irix_oplocks_context * ctx = talloc_get_type ( _ctx - > private_data ,
struct irix_oplocks_context ) ;
2001-10-20 21:59:34 +00:00
oplock_stat_t os ;
char dummy ;
2007-08-07 14:06:27 +00:00
struct file_id fileid ;
2001-10-20 21:59:34 +00:00
files_struct * fsp ;
2009-01-09 14:02:18 +01:00
/*
* TODO : is it correct to assume we only get one
* oplock break , for each byte we read from the pipe ?
*/
ctx - > pending = false ;
2006-02-14 23:00:39 +00:00
2001-10-20 21:59:34 +00:00
/*
* Read one byte of zero to clear the
* kernel break notify message .
*/
2009-01-09 14:02:18 +01:00
if ( read ( ctx - > read_fd , & dummy , 1 ) ! = 1 ) {
2005-10-01 10:00:07 +00:00
DEBUG ( 0 , ( " irix_oplock_receive_message: read of kernel "
" notification failed. Error was %s. \n " ,
strerror ( errno ) ) ) ;
2005-09-30 17:13:37 +00:00
return NULL ;
2001-10-20 21:59:34 +00:00
}
/*
* Do a query to get the
* device and inode of the file that has the break
* request outstanding .
*/
2009-01-09 14:02:18 +01:00
if ( sys_fcntl_ptr ( ctx - > read_fd , F_OPLKSTAT , & os ) < 0 ) {
2005-10-01 10:00:07 +00:00
DEBUG ( 0 , ( " irix_oplock_receive_message: fcntl of kernel "
" notification failed. Error was %s. \n " ,
strerror ( errno ) ) ) ;
2001-10-20 21:59:34 +00:00
if ( errno = = EAGAIN ) {
/*
* Duplicate kernel break message - ignore .
*/
2005-10-01 10:00:07 +00:00
return NULL ;
2001-10-20 21:59:34 +00:00
}
2005-09-30 17:13:37 +00:00
return NULL ;
2001-10-20 21:59:34 +00:00
}
/*
* We only have device and inode info here - we have to guess that this
* is the first fsp open with this dev , ino pair .
2007-08-07 14:06:27 +00:00
*
* NOTE : this doesn ' t work if any VFS modules overloads
* the file_id_create ( ) hook !
2001-10-20 21:59:34 +00:00
*/
2007-08-07 14:06:27 +00:00
fileid = file_id_create_dev ( ( SMB_DEV_T ) os . os_dev ,
( SMB_INO_T ) os . os_ino ) ;
2011-12-13 13:38:41 +01:00
if ( ( fsp = file_find_di_first ( ctx - > sconn , fileid ) ) = = NULL ) {
2005-10-01 10:00:07 +00:00
DEBUG ( 0 , ( " irix_oplock_receive_message: unable to find open "
" file with dev = %x, inode = %.0f \n " ,
( unsigned int ) os . os_dev , ( double ) os . os_ino ) ) ;
return NULL ;
2001-10-20 21:59:34 +00:00
}
2000-06-10 13:38:07 +00:00
2005-10-01 10:00:07 +00:00
DEBUG ( 5 , ( " irix_oplock_receive_message: kernel oplock break request "
2007-05-29 09:30:34 +00:00
" received for file_id %s gen_id = %ul " ,
2007-09-10 10:56:07 +00:00
file_id_string_tos ( & fsp - > file_id ) ,
2007-05-29 09:30:34 +00:00
fsp - > fh - > gen_id ) ) ;
2005-09-30 17:13:37 +00:00
return fsp ;
2000-06-10 13:38:07 +00:00
}
/****************************************************************************
Attempt to set an kernel oplock on a file .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-20 21:59:34 +00:00
2009-01-09 14:02:18 +01:00
static bool irix_set_kernel_oplock ( struct kernel_oplocks * _ctx ,
files_struct * fsp , int oplock_type )
2000-06-10 13:38:07 +00:00
{
2009-01-09 14:02:18 +01:00
struct irix_oplocks_context * ctx = talloc_get_type ( _ctx - > private_data ,
struct irix_oplocks_context ) ;
if ( sys_fcntl_long ( fsp - > fh - > fd , F_OPLKREG , ctx - > write_fd ) = = - 1 ) {
2000-06-10 13:38:07 +00:00
if ( errno ! = EAGAIN ) {
2007-01-16 15:50:25 +00:00
DEBUG ( 0 , ( " irix_set_kernel_oplock: Unable to get "
2007-05-29 09:30:34 +00:00
" kernel oplock on file %s, file_id %s "
" gen_id = %ul. Error was %s \n " ,
2009-07-10 14:50:37 -07:00
fsp_str_dbg ( fsp ) ,
file_id_string_tos ( & fsp - > file_id ) ,
2007-05-29 09:30:34 +00:00
fsp - > fh - > gen_id ,
2000-06-10 13:38:07 +00:00
strerror ( errno ) ) ) ;
} else {
2007-01-16 15:50:25 +00:00
DEBUG ( 5 , ( " irix_set_kernel_oplock: Refused oplock on "
2008-07-01 15:39:41 -07:00
" file %s, fd = %d, file_id = %s, "
2007-05-29 09:30:34 +00:00
" gen_id = %ul. Another process had the file "
2007-01-16 15:50:25 +00:00
" open. \n " ,
2009-07-10 14:50:37 -07:00
fsp_str_dbg ( fsp ) , fsp - > fh - > fd ,
2007-09-10 10:56:07 +00:00
file_id_string_tos ( & fsp - > file_id ) ,
2007-05-29 09:30:34 +00:00
fsp - > fh - > gen_id ) ) ;
2000-06-10 13:38:07 +00:00
}
return False ;
}
2007-05-29 09:30:34 +00:00
DEBUG ( 10 , ( " irix_set_kernel_oplock: got kernel oplock on file %s, file_id = %s "
" gen_id = %ul \n " ,
2009-07-10 14:50:37 -07:00
fsp_str_dbg ( fsp ) , file_id_string_tos ( & fsp - > file_id ) ,
2007-05-29 09:30:34 +00:00
fsp - > fh - > gen_id ) ) ;
2000-06-10 13:38:07 +00:00
return True ;
}
/****************************************************************************
Release a kernel oplock on a file .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-20 21:59:34 +00:00
2009-01-09 14:02:18 +01:00
static void irix_release_kernel_oplock ( struct kernel_oplocks * _ctx ,
2009-01-09 13:07:58 -08:00
files_struct * fsp , int oplock_type )
2000-06-10 13:38:07 +00:00
{
if ( DEBUGLVL ( 10 ) ) {
/*
* Check and print out the current kernel
* oplock state of this file .
*/
2005-07-08 07:54:28 +00:00
int state = sys_fcntl_long ( fsp - > fh - > fd , F_OPLKACK , - 1 ) ;
2007-05-29 09:30:34 +00:00
dbgtext ( " irix_release_kernel_oplock: file %s, file_id = %s "
" gen_id = %ul, has kernel oplock state "
2009-07-10 14:50:37 -07:00
" of %x. \n " , fsp_str_dbg ( fsp ) ,
file_id_string_tos ( & fsp - > file_id ) ,
2007-05-29 09:30:34 +00:00
fsp - > fh - > gen_id , state ) ;
2000-06-10 13:38:07 +00:00
}
/*
* Remove the kernel oplock on this file .
*/
2005-07-08 07:54:28 +00:00
if ( sys_fcntl_long ( fsp - > fh - > fd , F_OPLKACK , OP_REVOKE ) < 0 ) {
2000-06-10 13:38:07 +00:00
if ( DEBUGLVL ( 0 ) ) {
2007-01-16 15:50:25 +00:00
dbgtext ( " irix_release_kernel_oplock: Error when "
" removing kernel oplock on file " ) ;
2007-05-29 09:30:34 +00:00
dbgtext ( " %s, file_id = %s gen_id = %ul. "
2007-01-16 15:50:25 +00:00
" Error was %s \n " ,
2009-07-10 14:50:37 -07:00
fsp_str_dbg ( fsp ) ,
file_id_string_tos ( & fsp - > file_id ) ,
2007-05-29 09:30:34 +00:00
fsp - > fh - > gen_id ,
2007-01-16 15:50:25 +00:00
strerror ( errno ) ) ;
2000-06-10 13:38:07 +00:00
}
}
}
2009-01-09 14:02:18 +01:00
static void irix_oplocks_read_fde_handler ( struct event_context * ev ,
struct fd_event * fde ,
uint16_t flags ,
void * private_data )
{
struct irix_oplocks_context * ctx = talloc_get_type ( private_data ,
struct irix_oplocks_context ) ;
2009-01-23 10:08:44 +01:00
files_struct * fsp ;
2006-02-15 01:05:06 +00:00
2009-01-23 10:08:44 +01:00
fsp = irix_oplock_receive_message ( ctx - > ctx ) ;
2011-12-13 13:38:41 +01:00
break_kernel_oplock ( ctx - > sconn - > msg_ctx , fsp ) ;
2000-06-10 13:38:07 +00:00
}
/****************************************************************************
2001-10-20 21:59:34 +00:00
Setup kernel oplocks .
2000-06-10 13:38:07 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-20 21:59:34 +00:00
2009-01-09 14:02:18 +01:00
static const struct kernel_oplocks_ops irix_koplocks = {
2009-02-03 11:56:35 -08:00
. set_oplock = irix_set_kernel_oplock ,
. release_oplock = irix_release_kernel_oplock ,
. contend_level2_oplocks_begin = NULL ,
. contend_level2_oplocks_end = NULL ,
2009-01-09 14:02:18 +01:00
} ;
2011-12-13 13:38:41 +01:00
struct kernel_oplocks * irix_init_kernel_oplocks ( struct smbd_server_connection * sconn )
2000-06-10 13:38:07 +00:00
{
2009-01-09 14:02:18 +01:00
struct kernel_oplocks * _ctx ;
struct irix_oplocks_context * ctx ;
2000-06-10 13:38:07 +00:00
int pfd [ 2 ] ;
2001-10-20 21:59:34 +00:00
if ( ! irix_oplocks_available ( ) )
return NULL ;
2000-06-10 13:38:07 +00:00
2011-12-13 13:38:41 +01:00
_ctx = talloc_zero ( sconn , struct kernel_oplocks ) ;
2009-01-09 14:02:18 +01:00
if ( ! _ctx ) {
return NULL ;
}
ctx = talloc_zero ( _ctx , struct irix_oplocks_context ) ;
if ( ! ctx ) {
talloc_free ( _ctx ) ;
return NULL ;
}
_ctx - > ops = & irix_koplocks ;
_ctx - > private_data = ctx ;
ctx - > ctx = _ctx ;
2011-12-13 13:38:41 +01:00
ctx - > sconn = sconn ;
2009-01-09 14:02:18 +01:00
2000-06-10 13:38:07 +00:00
if ( pipe ( pfd ) ! = 0 ) {
2009-01-09 14:02:18 +01:00
talloc_free ( _ctx ) ;
2007-01-16 15:50:25 +00:00
DEBUG ( 0 , ( " setup_kernel_oplock_pipe: Unable to create pipe. "
" Error was %s \n " , strerror ( errno ) ) ) ;
2000-06-10 13:38:07 +00:00
return False ;
}
2009-01-09 14:02:18 +01:00
ctx - > read_fd = pfd [ 0 ] ;
ctx - > write_fd = pfd [ 1 ] ;
2000-06-10 13:38:07 +00:00
2011-12-13 13:38:41 +01:00
ctx - > read_fde = event_add_fd ( sconn - > ev_ctx ,
2009-01-09 14:02:18 +01:00
ctx ,
ctx - > read_fd ,
EVENT_FD_READ ,
irix_oplocks_read_fde_handler ,
ctx ) ;
return _ctx ;
2000-06-10 13:38:07 +00:00
}
# else
2006-12-12 20:15:47 +00:00
void oplock_irix_dummy ( void ) ;
2000-06-10 13:38:07 +00:00
void oplock_irix_dummy ( void ) { }
# endif /* HAVE_KERNEL_OPLOCKS_IRIX */