2001-03-11 03:32:10 +03:00
/*
2002-01-30 09:08:46 +03:00
Unix SMB / CIFS implementation .
2000-06-10 18:29:31 +04:00
IRIX kernel oplock processing
2000-06-10 17:38:07 +04: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
the Free Software Foundation ; either version 2 of the License , or
( 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
along with this program ; if not , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "includes.h"
2000-06-11 09:57:58 +04:00
# if HAVE_KERNEL_OPLOCKS_IRIX
2000-06-10 17:38:07 +04:00
static int oplock_pipe_write = - 1 ;
static int oplock_pipe_read = - 1 ;
/****************************************************************************
2001-10-21 01:59:34 +04:00
Test to see if IRIX kernel oplocks work .
2000-06-10 17:38:07 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-21 01:59:34 +04:00
2000-06-10 17:38:07 +04:00
static BOOL irix_oplocks_available ( void )
{
int fd ;
int pfd [ 2 ] ;
pstring tmpname ;
oplock_set_capability ( True , False ) ;
slprintf ( tmpname , sizeof ( tmpname ) - 1 , " %s/koplock.%d " , lp_lockdir ( ) , ( int ) sys_getpid ( ) ) ;
if ( pipe ( pfd ) ! = 0 ) {
DEBUG ( 0 , ( " check_kernel_oplocks: Unable to create pipe. Error was %s \n " ,
strerror ( errno ) ) ) ;
return False ;
}
if ( ( fd = sys_open ( tmpname , O_RDWR | O_CREAT | O_EXCL | O_TRUNC , 0600 ) ) < 0 ) {
DEBUG ( 0 , ( " check_kernel_oplocks: Unable to open temp test file %s. Error was %s \n " ,
tmpname , strerror ( errno ) ) ) ;
unlink ( tmpname ) ;
close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
return False ;
}
unlink ( tmpname ) ;
2002-07-15 14:35:28 +04:00
if ( sys_fcntl_long ( fd , F_OPLKREG , pfd [ 1 ] ) = = - 1 ) {
2000-06-10 17:38:07 +04:00
DEBUG ( 0 , ( " check_kernel_oplocks: Kernel oplocks are not available on this machine. \
Disabling kernel oplock support . \ n " ));
close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
close ( fd ) ;
return False ;
}
2002-07-15 14:35:28 +04:00
if ( sys_fcntl_long ( fd , F_OPLKACK , OP_REVOKE ) < 0 ) {
2000-06-10 17:38:07 +04:00
DEBUG ( 0 , ( " check_kernel_oplocks: Error when removing kernel oplock. Error was %s. \
Disabling kernel oplock support . \ n " , strerror(errno) ));
close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
close ( fd ) ;
return False ;
}
close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
close ( fd ) ;
return True ;
}
/****************************************************************************
* Deal with the IRIX kernel < - - > smbd
* oplock break protocol .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-21 01:59:34 +04:00
2000-06-10 17:38:07 +04:00
static BOOL irix_oplock_receive_message ( fd_set * fds , char * buffer , int buffer_len )
{
2000-06-20 04:32:32 +04:00
extern int smb_read_error ;
2001-10-21 01:59:34 +04:00
oplock_stat_t os ;
char dummy ;
files_struct * fsp ;
/*
* Read one byte of zero to clear the
* kernel break notify message .
*/
if ( read ( oplock_pipe_read , & dummy , 1 ) ! = 1 ) {
2002-07-15 14:35:28 +04:00
DEBUG ( 0 , ( " irix_oplock_receive_message: read of kernel notification failed. \
2000-06-10 17:38:07 +04:00
Error was % s . \ n " , strerror(errno) ));
2001-10-21 01:59:34 +04:00
smb_read_error = READ_ERROR ;
return False ;
}
/*
* Do a query to get the
* device and inode of the file that has the break
* request outstanding .
*/
2002-07-15 14:35:28 +04:00
if ( sys_fcntl_ptr ( oplock_pipe_read , F_OPLKSTAT , & os ) < 0 ) {
DEBUG ( 0 , ( " irix_oplock_receive_message: fcntl of kernel notification failed. \
2000-06-10 17:38:07 +04:00
Error was % s . \ n " , strerror(errno) ));
2001-10-21 01:59:34 +04:00
if ( errno = = EAGAIN ) {
/*
* Duplicate kernel break message - ignore .
*/
memset ( buffer , ' \0 ' , KERNEL_OPLOCK_BREAK_MSG_LEN ) ;
return True ;
}
smb_read_error = READ_ERROR ;
return False ;
}
/*
* We only have device and inode info here - we have to guess that this
* is the first fsp open with this dev , ino pair .
*/
if ( ( fsp = file_find_di_first ( ( SMB_DEV_T ) os . os_dev , ( SMB_INO_T ) os . os_ino ) ) = = NULL ) {
2002-07-15 14:35:28 +04:00
DEBUG ( 0 , ( " irix_oplock_receive_message: unable to find open file with dev = %x, inode = %.0f \n " ,
2001-10-21 01:59:34 +04:00
( unsigned int ) os . os_dev , ( double ) os . os_ino ) ) ;
return False ;
}
2000-06-10 17:38:07 +04:00
2002-07-15 14:35:28 +04:00
DEBUG ( 5 , ( " irix_oplock_receive_message: kernel oplock break request received for \
2001-10-21 01:59:34 +04:00
dev = % x , inode = % .0f \ n , file_id = % ul " , (unsigned int)fsp->dev, (double)fsp->inode, fsp->file_id ));
2000-06-10 17:38:07 +04:00
2001-10-21 01:59:34 +04:00
/*
* Create a kernel oplock break message .
*/
/* Setup the message header */
SIVAL ( buffer , OPBRK_CMD_LEN_OFFSET , KERNEL_OPLOCK_BREAK_MSG_LEN ) ;
SSVAL ( buffer , OPBRK_CMD_PORT_OFFSET , 0 ) ;
buffer + = OPBRK_CMD_HEADER_LEN ;
2000-06-10 17:38:07 +04:00
2001-10-21 01:59:34 +04:00
SSVAL ( buffer , OPBRK_MESSAGE_CMD_OFFSET , KERNEL_OPLOCK_BREAK_CMD ) ;
memcpy ( buffer + KERNEL_OPLOCK_BREAK_DEV_OFFSET , ( char * ) & fsp - > dev , sizeof ( fsp - > dev ) ) ;
memcpy ( buffer + KERNEL_OPLOCK_BREAK_INODE_OFFSET , ( char * ) & fsp - > inode , sizeof ( fsp - > inode ) ) ;
memcpy ( buffer + KERNEL_OPLOCK_BREAK_FILEID_OFFSET , ( char * ) & fsp - > file_id , sizeof ( fsp - > file_id ) ) ;
return True ;
2000-06-10 17:38:07 +04:00
}
/****************************************************************************
Attempt to set an kernel oplock on a file .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-21 01:59:34 +04:00
2000-06-10 17:38:07 +04:00
static BOOL irix_set_kernel_oplock ( files_struct * fsp , int oplock_type )
{
2002-07-15 14:35:28 +04:00
if ( sys_fcntl_long ( fsp - > fd , F_OPLKREG , oplock_pipe_write ) = = - 1 ) {
2000-06-10 17:38:07 +04:00
if ( errno ! = EAGAIN ) {
2002-07-15 14:35:28 +04:00
DEBUG ( 0 , ( " irix_set_kernel_oplock: Unable to get kernel oplock on file %s, dev = %x, \
2001-10-21 01:59:34 +04:00
inode = % .0f , file_id = % ul . Error was % s \ n " ,
fsp - > fsp_name , ( unsigned int ) fsp - > dev , ( double ) fsp - > inode , fsp - > file_id ,
2000-06-10 17:38:07 +04:00
strerror ( errno ) ) ) ;
} else {
2002-07-15 14:35:28 +04:00
DEBUG ( 5 , ( " irix_set_kernel_oplock: Refused oplock on file %s, fd = %d, dev = %x, \
2001-10-21 01:59:34 +04:00
inode = % .0f , file_id = % ul . Another process had the file open . \ n " ,
fsp - > fsp_name , fsp - > fd , ( unsigned int ) fsp - > dev , ( double ) fsp - > inode , fsp - > file_id ) ) ;
2000-06-10 17:38:07 +04:00
}
return False ;
}
2002-07-15 14:35:28 +04:00
DEBUG ( 10 , ( " irix_set_kernel_oplock: got kernel oplock on file %s, dev = %x, inode = %.0f, file_id = %ul \n " ,
2001-10-21 01:59:34 +04:00
fsp - > fsp_name , ( unsigned int ) fsp - > dev , ( double ) fsp - > inode , fsp - > file_id ) ) ;
2000-06-10 17:38:07 +04:00
return True ;
}
/****************************************************************************
Release a kernel oplock on a file .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-21 01:59:34 +04:00
2000-06-10 17:38:07 +04:00
static void irix_release_kernel_oplock ( files_struct * fsp )
{
if ( DEBUGLVL ( 10 ) ) {
/*
* Check and print out the current kernel
* oplock state of this file .
*/
2002-07-15 14:35:28 +04:00
int state = sys_fcntl_long ( fsp - > fd , F_OPLKACK , - 1 ) ;
dbgtext ( " irix_release_kernel_oplock: file %s, dev = %x, inode = %.0f file_id = %ul, has kernel \
2000-06-10 17:38:07 +04:00
oplock state of % x . \ n " , fsp->fsp_name, (unsigned int)fsp->dev,
2001-10-21 01:59:34 +04:00
( double ) fsp - > inode , fsp - > file_id , state ) ;
2000-06-10 17:38:07 +04:00
}
/*
* Remove the kernel oplock on this file .
*/
2002-07-15 14:35:28 +04:00
if ( sys_fcntl_long ( fsp - > fd , F_OPLKACK , OP_REVOKE ) < 0 ) {
2000-06-10 17:38:07 +04:00
if ( DEBUGLVL ( 0 ) ) {
2002-07-15 14:35:28 +04:00
dbgtext ( " irix_release_kernel_oplock: Error when removing kernel oplock on file " ) ;
2001-10-21 01:59:34 +04:00
dbgtext ( " %s, dev = %x, inode = %.0f, file_id = %ul. Error was %s \n " ,
2000-06-10 17:38:07 +04:00
fsp - > fsp_name , ( unsigned int ) fsp - > dev ,
2001-10-21 01:59:34 +04:00
( double ) fsp - > inode , fsp - > file_id , strerror ( errno ) ) ;
2000-06-10 17:38:07 +04:00
}
}
}
/****************************************************************************
2001-10-21 01:59:34 +04:00
Parse a kernel oplock message .
2000-06-10 17:38:07 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-21 01:59:34 +04:00
static BOOL irix_kernel_oplock_parse ( char * msg_start , int msg_len ,
SMB_INO_T * inode , SMB_DEV_T * dev , unsigned long * file_id )
2000-06-10 17:38:07 +04:00
{
/* Ensure that the msg length is correct. */
if ( msg_len ! = KERNEL_OPLOCK_BREAK_MSG_LEN ) {
totally rewrote the async signal, notification and oplock notification
handling in Samba. This was needed due to several limitations and
races in the previous code - as a side effect the new code is much
cleaner :)
in summary:
- changed sys_select() to avoid a signal/select race condition. It is a
rare race but once we have signals doing notification and oplocks it
is important.
- changed our main processing loop to take advantage of the new
sys_select semantics
- split the notify code into implementaion dependent and general
parts. Added the following structure that defines an implementation:
struct cnotify_fns {
void * (*register_notify)(connection_struct *conn, char *path, uint32 flags);
BOOL (*check_notify)(connection_struct *conn, uint16 vuid, char *path, uint32 flags, void *data, time_t t);
void (*remove_notify)(void *data);
};
then I wrote two implementations, one using hash/poll (like our old
code) and the other using the new Linux kernel change notify. It
should be easy to add other change notify implementations by creating
a sructure of the above type.
- fixed a bug in change notify where we were returning the wrong error
code.
- rewrote the core change notify code to be much simpler
- moved to real-time signals for leases and change notify
Amazingly, it all seems to work. I was very surprised!
(This used to be commit 44766c39e0027c762bee8b33b12c621c109a3267)
2000-06-12 19:53:31 +04:00
DEBUG ( 0 , ( " incorrect length for KERNEL_OPLOCK_BREAK_CMD (was %d, should be %d). \n " ,
msg_len , KERNEL_OPLOCK_BREAK_MSG_LEN ) ) ;
2000-06-10 17:38:07 +04:00
return False ;
}
2001-10-21 01:59:34 +04:00
memcpy ( ( char * ) inode , msg_start + KERNEL_OPLOCK_BREAK_INODE_OFFSET , sizeof ( * inode ) ) ;
memcpy ( ( char * ) dev , msg_start + KERNEL_OPLOCK_BREAK_DEV_OFFSET , sizeof ( * dev ) ) ;
memcpy ( ( char * ) file_id , msg_start + KERNEL_OPLOCK_BREAK_FILEID_OFFSET , sizeof ( * file_id ) ) ;
2000-06-10 17:38:07 +04:00
2001-10-21 01:59:34 +04:00
DEBUG ( 5 , ( " kernel oplock break request for file dev = %x, inode = %.0f, file_id = %ul \n " ,
( unsigned int ) * dev , ( double ) * inode , * file_id ) ) ;
2000-06-10 17:38:07 +04:00
return True ;
}
/****************************************************************************
2001-10-21 01:59:34 +04:00
Set * maxfd to include oplock read pipe .
2000-06-10 17:38:07 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-21 01:59:34 +04:00
2000-06-10 17:38:07 +04:00
static BOOL irix_oplock_msg_waiting ( fd_set * fds )
{
2001-10-21 01:59:34 +04:00
if ( oplock_pipe_read = = - 1 )
return False ;
2000-06-10 17:38:07 +04:00
return FD_ISSET ( oplock_pipe_read , fds ) ;
}
/****************************************************************************
2001-10-21 01:59:34 +04:00
Setup kernel oplocks .
2000-06-10 17:38:07 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-21 01:59:34 +04:00
2000-06-10 17:38:07 +04:00
struct kernel_oplocks * irix_init_kernel_oplocks ( void )
{
int pfd [ 2 ] ;
static struct kernel_oplocks koplocks ;
2001-10-21 01:59:34 +04:00
if ( ! irix_oplocks_available ( ) )
return NULL ;
2000-06-10 17:38:07 +04:00
if ( pipe ( pfd ) ! = 0 ) {
DEBUG ( 0 , ( " setup_kernel_oplock_pipe: Unable to create pipe. Error was %s \n " ,
strerror ( errno ) ) ) ;
return False ;
}
oplock_pipe_read = pfd [ 0 ] ;
oplock_pipe_write = pfd [ 1 ] ;
koplocks . receive_message = irix_oplock_receive_message ;
koplocks . set_oplock = irix_set_kernel_oplock ;
koplocks . release_oplock = irix_release_kernel_oplock ;
koplocks . parse_message = irix_kernel_oplock_parse ;
koplocks . msg_waiting = irix_oplock_msg_waiting ;
koplocks . notification_fd = oplock_pipe_read ;
return & koplocks ;
}
# else
void oplock_irix_dummy ( void ) { }
# endif /* HAVE_KERNEL_OPLOCKS_IRIX */