0001-01-01 02:30:17 +02:30
/*
0001-01-01 02:30:17 +02:30
Unix SMB / CIFS implementation .
0001-01-01 02:30:17 +02:30
change notify handling
Copyright ( C ) Andrew Tridgell 2000
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!
0001-01-01 02:30:17 +02:30
Copyright ( C ) Jeremy Allison 1994 - 1998
2007-01-21 14:49:00 +03:00
Copyright ( C ) Volker Lendecke 2007
0001-01-01 02:30:17 +02:30
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"
2007-01-17 19:23:45 +03:00
static struct notify_mid_map * notify_changes_by_mid ;
0001-01-01 02:30:17 +02:30
2007-01-17 21:23:37 +03:00
/*
* For NTCancel , we need to find the notify_change_request indexed by
* mid . Separate list here .
*/
struct notify_mid_map {
struct notify_mid_map * prev , * next ;
struct notify_change_request * req ;
uint16 mid ;
} ;
2007-01-21 14:49:00 +03:00
static BOOL notify_marshall_changes ( int num_changes ,
2007-01-17 19:23:45 +03:00
struct notify_change * changes ,
prs_struct * ps )
{
int i ;
UNISTR uni_name ;
for ( i = 0 ; i < num_changes ; i + + ) {
struct notify_change * c = & changes [ i ] ;
size_t namelen ;
uint32 u32_tmp ; /* Temp arg to prs_uint32 to avoid
* signed / unsigned issues */
namelen = convert_string_allocate (
NULL , CH_UNIX , CH_UTF16LE , c - > name , strlen ( c - > name ) + 1 ,
& uni_name . buffer , True ) ;
if ( ( namelen = = - 1 ) | | ( uni_name . buffer = = NULL ) ) {
goto fail ;
}
namelen - = 2 ; /* Dump NULL termination */
/*
* Offset to next entry , only if there is one
*/
u32_tmp = ( i = = num_changes - 1 ) ? 0 : namelen + 12 ;
if ( ! prs_uint32 ( " offset " , ps , 1 , & u32_tmp ) ) goto fail ;
u32_tmp = c - > action ;
if ( ! prs_uint32 ( " action " , ps , 1 , & u32_tmp ) ) goto fail ;
u32_tmp = namelen ;
if ( ! prs_uint32 ( " namelen " , ps , 1 , & u32_tmp ) ) goto fail ;
if ( ! prs_unistr ( " name " , ps , 1 , & uni_name ) ) goto fail ;
/*
* Not NULL terminated , decrease by the 2 UCS2 \ 0 chars
*/
prs_set_offset ( ps , prs_offset ( ps ) - 2 ) ;
SAFE_FREE ( uni_name . buffer ) ;
}
return True ;
fail :
SAFE_FREE ( uni_name . buffer ) ;
return False ;
}
0001-01-01 02:30:17 +02:30
/****************************************************************************
Setup the common parts of the return packet and send it .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
0001-01-01 02:30:17 +02:30
2007-01-17 21:26:37 +03:00
static void change_notify_reply_packet ( const char * request_buf ,
NTSTATUS error_code )
0001-01-01 02:30:17 +02:30
{
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!
0001-01-01 02:30:17 +02:30
char outbuf [ smb_size + 38 ] ;
0001-01-01 02:30:17 +02:30
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!
0001-01-01 02:30:17 +02:30
memset ( outbuf , ' \0 ' , sizeof ( outbuf ) ) ;
2007-01-17 19:23:45 +03:00
construct_reply_common ( request_buf , outbuf ) ;
0001-01-01 02:30:17 +02:30
0001-01-01 02:30:17 +02:30
ERROR_NT ( error_code ) ;
0001-01-01 02:30:17 +02:30
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!
0001-01-01 02:30:17 +02:30
/*
* Seems NT needs a transact command with an error code
* in it . This is a longer packet than a simple error .
*/
set_message ( outbuf , 18 , 0 , False ) ;
0001-01-01 02:30:17 +02:30
2005-06-28 02:53:56 +04:00
show_msg ( outbuf ) ;
0001-01-01 02:30:17 +02:30
if ( ! send_smb ( smbd_server_fd ( ) , outbuf ) )
2007-01-17 21:26:37 +03:00
exit_server_cleanly ( " change_notify_reply_packet: send_smb "
" failed. " ) ;
0001-01-01 02:30:17 +02:30
}
2007-01-17 19:23:45 +03:00
void change_notify_reply ( const char * request_buf , uint32 max_param_count ,
2007-01-21 14:49:00 +03:00
int num_changes , struct notify_change * changes )
2007-01-17 19:23:45 +03:00
{
char * outbuf = NULL ;
prs_struct ps ;
size_t buflen = smb_size + 38 + max_param_count ;
2007-01-21 14:49:00 +03:00
if ( num_changes = = - 1 ) {
change_notify_reply_packet ( request_buf , NT_STATUS_OK ) ;
return ;
}
2007-01-17 19:23:45 +03:00
if ( ! prs_init ( & ps , 0 , NULL , False )
| | ! notify_marshall_changes ( num_changes , changes , & ps ) ) {
change_notify_reply_packet ( request_buf , NT_STATUS_NO_MEMORY ) ;
goto done ;
}
if ( prs_offset ( & ps ) > max_param_count ) {
/*
* We exceed what the client is willing to accept . Send
* nothing .
*/
change_notify_reply_packet ( request_buf , NT_STATUS_OK ) ;
goto done ;
}
if ( ! ( outbuf = SMB_MALLOC_ARRAY ( char , buflen ) ) ) {
change_notify_reply_packet ( request_buf , NT_STATUS_NO_MEMORY ) ;
goto done ;
}
construct_reply_common ( request_buf , outbuf ) ;
if ( send_nt_replies ( outbuf , buflen , NT_STATUS_OK , prs_data_p ( & ps ) ,
prs_offset ( & ps ) , NULL , 0 ) = = - 1 ) {
exit_server ( " change_notify_reply_packet: send_smb failed. " ) ;
}
done :
SAFE_FREE ( outbuf ) ;
prs_mem_free ( & ps ) ;
}
NTSTATUS change_notify_add_request ( const char * inbuf , uint32 max_param_count ,
2007-01-31 17:42:56 +03:00
uint32 filter , BOOL recursive ,
struct files_struct * fsp )
2007-01-17 19:23:45 +03:00
{
struct notify_change_request * request = NULL ;
struct notify_mid_map * map = NULL ;
if ( ! ( request = SMB_MALLOC_P ( struct notify_change_request ) )
| | ! ( map = SMB_MALLOC_P ( struct notify_mid_map ) ) ) {
SAFE_FREE ( request ) ;
return NT_STATUS_NO_MEMORY ;
}
request - > mid_map = map ;
map - > req = request ;
memcpy ( request - > request_buf , inbuf , sizeof ( request - > request_buf ) ) ;
request - > max_param_count = max_param_count ;
request - > filter = filter ;
request - > fsp = fsp ;
2007-01-31 17:42:56 +03:00
request - > backend_data = NULL ;
2007-01-21 14:49:00 +03:00
2007-01-17 19:23:45 +03:00
DLIST_ADD_END ( fsp - > notify - > requests , request ,
struct notify_change_request * ) ;
map - > mid = SVAL ( inbuf , smb_mid ) ;
DLIST_ADD ( notify_changes_by_mid , map ) ;
0001-01-01 02:30:17 +02:30
2007-01-17 19:23:45 +03:00
/* Push the MID of this packet on the signing queue. */
srv_defer_sign_response ( SVAL ( inbuf , smb_mid ) ) ;
return NT_STATUS_OK ;
}
static void change_notify_remove_request ( struct notify_change_request * remove_req )
0001-01-01 02:30:17 +02:30
{
2007-01-17 19:23:45 +03:00
files_struct * fsp ;
struct notify_change_request * req ;
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!
0001-01-01 02:30:17 +02:30
2007-01-17 19:23:45 +03:00
/*
* Paranoia checks , the fsp referenced must must have the request in
* its list of pending requests
*/
fsp = remove_req - > fsp ;
SMB_ASSERT ( fsp - > notify ! = NULL ) ;
for ( req = fsp - > notify - > requests ; req ; req = req - > next ) {
if ( req = = remove_req ) {
break ;
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!
0001-01-01 02:30:17 +02:30
}
}
2007-01-17 19:23:45 +03:00
if ( req = = NULL ) {
smb_panic ( " notify_req not found in fsp's requests \n " ) ;
}
DLIST_REMOVE ( fsp - > notify - > requests , req ) ;
DLIST_REMOVE ( notify_changes_by_mid , req - > mid_map ) ;
SAFE_FREE ( req - > mid_map ) ;
2007-01-21 14:49:00 +03:00
TALLOC_FREE ( req - > backend_data ) ;
2007-01-17 19:23:45 +03:00
SAFE_FREE ( req ) ;
2006-12-31 20:52:24 +03:00
}
/****************************************************************************
2007-01-09 19:12:54 +03:00
Delete entries by mid from the change notify pending queue . Always send reply .
2006-12-31 20:52:24 +03:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-01-17 19:23:45 +03:00
void remove_pending_change_notify_requests_by_mid ( uint16 mid )
2006-12-31 20:52:24 +03:00
{
2007-01-17 19:23:45 +03:00
struct notify_mid_map * map ;
2006-12-31 20:52:24 +03:00
2007-01-17 19:23:45 +03:00
for ( map = notify_changes_by_mid ; map ; map = map - > next ) {
if ( map - > mid = = mid ) {
break ;
2007-01-09 19:12:54 +03:00
}
2006-12-31 20:52:24 +03:00
}
2007-01-17 19:23:45 +03:00
if ( map = = NULL ) {
return ;
}
change_notify_reply_packet ( map - > req - > request_buf , NT_STATUS_CANCELLED ) ;
change_notify_remove_request ( map - > req ) ;
}
/****************************************************************************
Delete entries by fnum from the change notify pending queue .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void remove_pending_change_notify_requests_by_fid ( files_struct * fsp ,
NTSTATUS status )
{
if ( fsp - > notify = = NULL ) {
return ;
}
while ( fsp - > notify - > requests ! = NULL ) {
change_notify_reply_packet (
fsp - > notify - > requests - > request_buf , status ) ;
change_notify_remove_request ( fsp - > notify - > requests ) ;
}
0001-01-01 02:30:17 +02:30
}
2007-01-31 17:14:57 +03:00
void notify_fname ( connection_struct * conn , uint32 action , uint32 filter ,
const char * path )
2007-01-17 19:23:45 +03:00
{
2007-01-31 17:42:56 +03:00
char * fullpath ;
2007-01-17 19:23:45 +03:00
2007-01-31 17:42:56 +03:00
if ( asprintf ( & fullpath , " %s/%s " , conn - > connectpath , path ) = = - 1 ) {
DEBUG ( 0 , ( " asprintf failed \n " ) ) ;
2007-01-17 19:23:45 +03:00
return ;
}
2007-01-31 17:42:56 +03:00
notify_trigger ( conn - > notify_ctx , action , filter , fullpath ) ;
SAFE_FREE ( fullpath ) ;
2007-01-17 19:23:45 +03:00
}
2007-01-31 17:42:56 +03:00
void notify_fsp ( files_struct * fsp , uint32 action , const char * name )
2007-01-17 19:23:45 +03:00
{
struct notify_change * change , * changes ;
2007-01-31 17:42:56 +03:00
char * name2 ;
2007-01-17 19:23:45 +03:00
if ( fsp - > notify = = NULL ) {
/*
* Nobody is waiting , don ' t queue
*/
return ;
}
2007-01-31 17:42:56 +03:00
if ( ! ( name2 = talloc_strdup ( fsp - > notify , name ) ) ) {
DEBUG ( 0 , ( " talloc_strdup failed \n " ) ) ;
2007-01-21 14:49:00 +03:00
return ;
}
2007-01-31 17:42:56 +03:00
string_replace ( name2 , ' / ' , ' \\ ' ) ;
2007-01-17 19:23:45 +03:00
/*
* Someone has triggered a notify previously , queue the change for
* later . TODO : Limit the number of changes queued , test how filters
* apply here . Do we have to store them ?
*/
2007-01-21 14:49:00 +03:00
if ( ( fsp - > notify - > num_changes > 30 ) | | ( name = = NULL ) ) {
/*
* W2k3 seems to store at most 30 changes .
*/
TALLOC_FREE ( fsp - > notify - > changes ) ;
2007-01-31 17:42:56 +03:00
TALLOC_FREE ( name2 ) ;
2007-01-21 14:49:00 +03:00
fsp - > notify - > num_changes = - 1 ;
return ;
}
if ( fsp - > notify - > num_changes = = - 1 ) {
return ;
}
2007-01-17 19:23:45 +03:00
if ( ! ( changes = TALLOC_REALLOC_ARRAY (
fsp - > notify , fsp - > notify - > changes ,
struct notify_change , fsp - > notify - > num_changes + 1 ) ) ) {
DEBUG ( 0 , ( " talloc_realloc failed \n " ) ) ;
2007-01-31 17:42:56 +03:00
TALLOC_FREE ( name2 ) ;
2007-01-17 19:23:45 +03:00
return ;
}
fsp - > notify - > changes = changes ;
change = & ( fsp - > notify - > changes [ fsp - > notify - > num_changes ] ) ;
2007-01-31 17:42:56 +03:00
change - > name = talloc_move ( changes , & name2 ) ;
2007-01-21 14:49:00 +03:00
change - > action = action ;
2007-01-17 19:23:45 +03:00
fsp - > notify - > num_changes + = 1 ;
2007-01-31 17:42:56 +03:00
if ( fsp - > notify - > requests = = NULL ) {
/*
* Nobody is waiting , so don ' t send anything . The ot
*/
return ;
}
if ( action = = NOTIFY_ACTION_OLD_NAME ) {
/*
* We have to send the two rename events in one reply . So hold
* the first part back .
*/
2007-01-17 19:23:45 +03:00
return ;
2007-01-31 17:42:56 +03:00
}
/*
* Someone is waiting for the change , trigger the reply immediately .
*
* TODO : do we have to walk the lists of requests pending ?
*/
change_notify_reply ( fsp - > notify - > requests - > request_buf ,
fsp - > notify - > requests - > max_param_count ,
fsp - > notify - > num_changes ,
fsp - > notify - > changes ) ;
change_notify_remove_request ( fsp - > notify - > requests ) ;
TALLOC_FREE ( fsp - > notify - > changes ) ;
fsp - > notify - > num_changes = 0 ;
2007-01-17 19:23:45 +03:00
}
2007-01-31 17:28:08 +03:00
char * notify_filter_string ( TALLOC_CTX * mem_ctx , uint32 filter )
{
char * result = NULL ;
result = talloc_strdup ( mem_ctx , " " ) ;
if ( filter & FILE_NOTIFY_CHANGE_FILE_NAME )
result = talloc_asprintf_append ( result , " FILE_NAME| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_DIR_NAME )
result = talloc_asprintf_append ( result , " DIR_NAME| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_ATTRIBUTES )
result = talloc_asprintf_append ( result , " ATTRIBUTES| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_SIZE )
result = talloc_asprintf_append ( result , " SIZE| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_LAST_WRITE )
result = talloc_asprintf_append ( result , " LAST_WRITE| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_LAST_ACCESS )
result = talloc_asprintf_append ( result , " LAST_ACCESS| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_CREATION )
result = talloc_asprintf_append ( result , " CREATION| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_EA )
result = talloc_asprintf_append ( result , " EA| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_SECURITY )
result = talloc_asprintf_append ( result , " SECURITY| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_STREAM_NAME )
result = talloc_asprintf_append ( result , " STREAM_NAME| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_STREAM_SIZE )
result = talloc_asprintf_append ( result , " STREAM_SIZE| " ) ;
if ( filter & FILE_NOTIFY_CHANGE_STREAM_WRITE )
result = talloc_asprintf_append ( result , " STREAM_WRITE| " ) ;
if ( result = = NULL ) return NULL ;
if ( * result = = ' \0 ' ) return result ;
result [ strlen ( result ) - 1 ] = ' \0 ' ;
return result ;
}
2007-02-01 16:36:02 +03:00
struct sys_notify_context * sys_notify_context_create ( connection_struct * conn ,
2007-01-31 16:47:37 +03:00
TALLOC_CTX * mem_ctx ,
struct event_context * ev )
{
struct sys_notify_context * ctx ;
if ( ! ( ctx = TALLOC_P ( mem_ctx , struct sys_notify_context ) ) ) {
DEBUG ( 0 , ( " talloc failed \n " ) ) ;
return NULL ;
}
ctx - > ev = ev ;
2007-02-01 16:36:02 +03:00
ctx - > conn = conn ;
2007-01-31 16:47:37 +03:00
ctx - > private_data = NULL ;
return ctx ;
}
NTSTATUS sys_notify_watch ( struct sys_notify_context * ctx ,
struct notify_entry * e ,
void ( * callback ) ( struct sys_notify_context * ctx ,
void * private_data ,
struct notify_event * ev ) ,
void * private_data , void * handle )
{
2007-02-01 16:36:02 +03:00
return SMB_VFS_NOTIFY_WATCH ( ctx - > conn , ctx , e , callback , private_data ,
handle ) ;
2007-01-31 16:47:37 +03:00
}