2000-11-14 21:56:32 +00:00
/*
2002-03-19 02:32:39 +00:00
Unix SMB / Netbios implementation .
Version 1.9 .
2000-02-03 05:10:09 +00:00
VFS initialisation and support functions
Copyright ( C ) Tim Potter 1999
2002-08-17 15:27:10 +00:00
Copyright ( C ) Alexander Bokovoy 2002
2000-11-14 21:56:32 +00:00
2000-02-03 05:10:09 +00:00
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 .
2000-11-14 21:56:32 +00:00
2000-02-03 05:10:09 +00:00
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 .
2000-11-14 21:56:32 +00:00
2000-02-03 05:10:09 +00:00
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 .
2002-08-17 15:27:10 +00:00
This work was sponsored by Optifacio Software Services , Inc .
2000-02-03 05:10:09 +00:00
*/
# include "includes.h"
2002-09-25 15:19:00 +00:00
# undef DBGC_CLASS
# define DBGC_CLASS DBGC_VFS
2003-04-16 14:45:11 +00:00
struct vfs_init_function_entry {
2003-05-11 23:34:18 +00:00
char * name ;
vfs_op_tuple * vfs_op_tuples ;
struct vfs_init_function_entry * prev , * next ;
2003-04-16 14:45:11 +00:00
} ;
static struct vfs_init_function_entry * backends = NULL ;
2002-09-25 15:19:00 +00:00
2000-02-03 05:10:09 +00:00
/* Some structures to help us initialise the vfs operations table */
struct vfs_syminfo {
2000-10-06 18:13:52 +00:00
char * name ;
void * fptr ;
2000-02-03 05:10:09 +00:00
} ;
/* Default vfs hooks. WARNING: The order of these initialisers is
very important . They must be in the same order as defined in
vfs . h . Change at your own peril . */
2003-05-11 23:34:18 +00:00
static struct vfs_ops default_vfs = {
2003-05-12 09:01:50 +00:00
{
2003-05-11 23:34:18 +00:00
/* Disk operations */
vfswrap_dummy_connect ,
vfswrap_dummy_disconnect ,
vfswrap_disk_free ,
vfswrap_get_quota ,
vfswrap_set_quota ,
2003-08-07 21:47:46 +00:00
vfswrap_get_shadow_copy_data ,
2003-05-11 23:34:18 +00:00
/* Directory operations */
vfswrap_opendir ,
vfswrap_readdir ,
vfswrap_mkdir ,
vfswrap_rmdir ,
vfswrap_closedir ,
/* File operations */
vfswrap_open ,
vfswrap_close ,
vfswrap_read ,
2004-01-06 01:22:14 +00:00
vfswrap_pread ,
2003-05-11 23:34:18 +00:00
vfswrap_write ,
2004-01-06 01:22:14 +00:00
vfswrap_pwrite ,
2003-05-11 23:34:18 +00:00
vfswrap_lseek ,
vfswrap_sendfile ,
vfswrap_rename ,
vfswrap_fsync ,
vfswrap_stat ,
vfswrap_fstat ,
vfswrap_lstat ,
vfswrap_unlink ,
vfswrap_chmod ,
vfswrap_fchmod ,
vfswrap_chown ,
vfswrap_fchown ,
vfswrap_chdir ,
vfswrap_getwd ,
vfswrap_utime ,
vfswrap_ftruncate ,
vfswrap_lock ,
vfswrap_symlink ,
vfswrap_readlink ,
vfswrap_link ,
vfswrap_mknod ,
vfswrap_realpath ,
2003-06-06 00:04:27 +00:00
/* Windows ACL operations. */
2003-05-11 23:34:18 +00:00
vfswrap_fget_nt_acl ,
vfswrap_get_nt_acl ,
vfswrap_fset_nt_acl ,
vfswrap_set_nt_acl ,
/* POSIX ACL operations. */
vfswrap_chmod_acl ,
vfswrap_fchmod_acl ,
2003-05-12 16:03:16 +00:00
2003-05-11 23:34:18 +00:00
vfswrap_sys_acl_get_entry ,
vfswrap_sys_acl_get_tag_type ,
vfswrap_sys_acl_get_permset ,
vfswrap_sys_acl_get_qualifier ,
vfswrap_sys_acl_get_file ,
vfswrap_sys_acl_get_fd ,
vfswrap_sys_acl_clear_perms ,
vfswrap_sys_acl_add_perm ,
vfswrap_sys_acl_to_text ,
vfswrap_sys_acl_init ,
vfswrap_sys_acl_create_entry ,
vfswrap_sys_acl_set_tag_type ,
vfswrap_sys_acl_set_qualifier ,
vfswrap_sys_acl_set_permset ,
vfswrap_sys_acl_valid ,
vfswrap_sys_acl_set_file ,
vfswrap_sys_acl_set_fd ,
vfswrap_sys_acl_delete_def_file ,
vfswrap_sys_acl_get_perm ,
vfswrap_sys_acl_free_text ,
vfswrap_sys_acl_free_acl ,
2003-06-06 00:04:27 +00:00
vfswrap_sys_acl_free_qualifier ,
/* EA operations. */
vfswrap_getxattr ,
vfswrap_lgetxattr ,
vfswrap_fgetxattr ,
vfswrap_listxattr ,
vfswrap_llistxattr ,
vfswrap_flistxattr ,
vfswrap_removexattr ,
vfswrap_lremovexattr ,
vfswrap_fremovexattr ,
vfswrap_setxattr ,
vfswrap_lsetxattr ,
vfswrap_fsetxattr
2003-05-11 23:34:18 +00:00
}
2000-02-03 05:10:09 +00:00
} ;
2003-04-16 14:45:11 +00:00
/****************************************************************************
maintain the list of available backends
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static struct vfs_init_function_entry * vfs_find_backend_entry ( const char * name )
{
2003-05-11 23:34:18 +00:00
struct vfs_init_function_entry * entry = backends ;
while ( entry ) {
2003-05-16 06:20:57 +00:00
if ( strcmp ( entry - > name , name ) = = 0 ) return entry ;
2003-05-11 23:34:18 +00:00
entry = entry - > next ;
}
2003-04-16 14:45:11 +00:00
2003-05-11 23:34:18 +00:00
return NULL ;
2003-04-16 14:45:11 +00:00
}
2003-05-11 23:34:18 +00:00
NTSTATUS smb_register_vfs ( int version , const char * name , vfs_op_tuple * vfs_op_tuples )
2003-04-16 14:45:11 +00:00
{
2003-05-11 23:34:18 +00:00
struct vfs_init_function_entry * entry = backends ;
if ( ( version ! = SMB_VFS_INTERFACE_VERSION ) ) {
DEBUG ( 0 , ( " Failed to register vfs module. \n "
" The module was compiled against SMB_VFS_INTERFACE_VERSION %d, \n "
" current SMB_VFS_INTERFACE_VERSION is %d. \n "
" Please recompile against the current Samba Version! \n " ,
version , SMB_VFS_INTERFACE_VERSION ) ) ;
return NT_STATUS_OBJECT_TYPE_MISMATCH ;
}
if ( ! name | | ! name [ 0 ] | | ! vfs_op_tuples ) {
DEBUG ( 0 , ( " smb_register_vfs() called with NULL pointer or empty name! \n " ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
if ( vfs_find_backend_entry ( name ) ) {
DEBUG ( 0 , ( " VFS module %s already loaded! \n " , name ) ) ;
return NT_STATUS_OBJECT_NAME_COLLISION ;
}
entry = smb_xmalloc ( sizeof ( struct vfs_init_function_entry ) ) ;
entry - > name = smb_xstrdup ( name ) ;
entry - > vfs_op_tuples = vfs_op_tuples ;
DLIST_ADD ( backends , entry ) ;
DEBUG ( 5 , ( " Successfully added vfs backend '%s' \n " , name ) ) ;
return NT_STATUS_OK ;
2003-04-16 14:45:11 +00:00
}
2000-02-03 05:10:09 +00:00
/****************************************************************************
initialise default vfs hooks
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-06-29 22:32:24 +00:00
2002-08-17 15:27:10 +00:00
static void vfs_init_default ( connection_struct * conn )
2000-02-03 05:10:09 +00:00
{
2002-03-27 03:00:39 +00:00
DEBUG ( 3 , ( " Initialising default vfs hooks \n " ) ) ;
2000-02-03 05:10:09 +00:00
2003-05-11 23:34:18 +00:00
memcpy ( & conn - > vfs . ops , & default_vfs . ops , sizeof ( default_vfs . ops ) ) ;
memcpy ( & conn - > vfs_opaque . ops , & default_vfs . ops , sizeof ( default_vfs . ops ) ) ;
2003-04-16 14:45:11 +00:00
}
/****************************************************************************
initialise custom vfs hooks
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
BOOL vfs_init_custom ( connection_struct * conn , const char * vfs_object )
{
vfs_op_tuple * ops ;
2003-05-11 23:34:18 +00:00
char * module_name = NULL ;
char * module_param = NULL , * p ;
2003-04-16 14:45:11 +00:00
int i ;
2003-05-11 23:34:18 +00:00
vfs_handle_struct * handle ;
2003-04-16 14:45:11 +00:00
struct vfs_init_function_entry * entry ;
2003-05-11 23:34:18 +00:00
if ( ! conn | | ! vfs_object | | ! vfs_object [ 0 ] ) {
DEBUG ( 0 , ( " vfs_init_custon() called with NULL pointer or emtpy vfs_object! \n " ) ) ;
return False ;
}
2003-04-16 14:45:11 +00:00
if ( ! backends ) static_init_vfs ;
2003-05-11 23:34:18 +00:00
DEBUG ( 3 , ( " Initialising custom vfs hooks from [%s] \n " , vfs_object ) ) ;
module_name = smb_xstrdup ( vfs_object ) ;
p = strchr ( module_name , ' : ' ) ;
if ( p ) {
* p = 0 ;
module_param = p + 1 ;
2003-09-05 19:59:55 +00:00
trim_char ( module_param , ' ' , ' ' ) ;
2003-05-11 23:34:18 +00:00
}
2003-09-05 19:59:55 +00:00
trim_char ( module_name , ' ' , ' ' ) ;
2003-05-11 23:34:18 +00:00
2003-04-16 14:45:11 +00:00
/* First, try to load the module with the new module system */
2003-05-11 23:34:18 +00:00
if ( ( entry = vfs_find_backend_entry ( module_name ) ) | |
( NT_STATUS_IS_OK ( smb_probe_module ( " vfs " , module_name ) ) & &
( entry = vfs_find_backend_entry ( module_name ) ) ) ) {
2003-04-16 14:45:11 +00:00
2003-05-11 23:34:18 +00:00
DEBUGADD ( 5 , ( " Successfully loaded vfs module [%s] with the new modules system \n " , vfs_object ) ) ;
2003-04-16 14:45:11 +00:00
2003-05-11 23:34:18 +00:00
if ( ( ops = entry - > vfs_op_tuples ) = = NULL ) {
DEBUG ( 0 , ( " entry->vfs_op_tuples==NULL for [%s] failed \n " , vfs_object ) ) ;
SAFE_FREE ( module_name ) ;
return False ;
}
2003-04-16 14:45:11 +00:00
} else {
2003-05-11 23:34:18 +00:00
DEBUG ( 0 , ( " Can't find a vfs module [%s] \n " , vfs_object ) ) ;
SAFE_FREE ( module_name ) ;
return False ;
}
handle = ( vfs_handle_struct * ) talloc_zero ( conn - > mem_ctx , sizeof ( vfs_handle_struct ) ) ;
if ( ! handle ) {
DEBUG ( 0 , ( " talloc_zero() failed! \n " ) ) ;
SAFE_FREE ( module_name ) ;
return False ;
}
memcpy ( & handle - > vfs_next , & conn - > vfs , sizeof ( struct vfs_ops ) ) ;
handle - > conn = conn ;
if ( module_param ) {
handle - > param = talloc_strdup ( conn - > mem_ctx , module_param ) ;
2003-04-16 14:45:11 +00:00
}
2003-05-11 23:34:18 +00:00
DLIST_ADD ( conn - > vfs_handles , handle ) ;
2003-04-16 14:45:11 +00:00
2002-08-17 15:27:10 +00:00
for ( i = 0 ; ops [ i ] . op ! = NULL ; i + + ) {
2003-05-11 23:34:18 +00:00
DEBUG ( 5 , ( " Checking operation #%d (type %d, layer %d) \n " , i , ops [ i ] . type , ops [ i ] . layer ) ) ;
2002-08-17 15:27:10 +00:00
if ( ops [ i ] . layer = = SMB_VFS_LAYER_OPAQUE ) {
/* Check whether this operation was already made opaque by different module */
2003-05-11 23:34:18 +00:00
if ( ( ( void * * ) & conn - > vfs_opaque . ops ) [ ops [ i ] . type ] = = ( ( void * * ) & default_vfs . ops ) [ ops [ i ] . type ] ) {
2002-08-17 15:27:10 +00:00
/* No, it isn't overloaded yet. Overload. */
2003-05-11 23:34:18 +00:00
DEBUGADD ( 5 , ( " Making operation type %d opaque [module %s] \n " , ops [ i ] . type , vfs_object ) ) ;
( ( void * * ) & conn - > vfs_opaque . ops ) [ ops [ i ] . type ] = ops [ i ] . op ;
( ( vfs_handle_struct * * ) & conn - > vfs_opaque . handles ) [ ops [ i ] . type ] = handle ;
2002-08-17 15:27:10 +00:00
}
}
/* Change current VFS disposition*/
2003-05-11 23:34:18 +00:00
DEBUGADD ( 5 , ( " Accepting operation type %d from module %s \n " , ops [ i ] . type , vfs_object ) ) ;
( ( void * * ) & conn - > vfs . ops ) [ ops [ i ] . type ] = ops [ i ] . op ;
( ( vfs_handle_struct * * ) & conn - > vfs . handles ) [ ops [ i ] . type ] = handle ;
2001-07-04 07:15:53 +00:00
}
2000-11-06 21:44:33 +00:00
2003-05-11 23:34:18 +00:00
SAFE_FREE ( module_name ) ;
2002-03-27 03:00:39 +00:00
return True ;
2000-02-03 05:10:09 +00:00
}
2001-06-29 22:32:24 +00:00
/*****************************************************************
Generic VFS init .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2001-10-18 00:27:20 +00:00
BOOL smbd_vfs_init ( connection_struct * conn )
2001-06-29 22:32:24 +00:00
{
2003-04-16 14:45:11 +00:00
const char * * vfs_objects ;
unsigned int i = 0 ;
int j = 0 ;
2002-08-17 15:27:10 +00:00
/* Normal share - initialise with disk access functions */
vfs_init_default ( conn ) ;
2003-05-11 23:34:18 +00:00
vfs_objects = lp_vfs_objects ( SNUM ( conn ) ) ;
2002-08-17 15:27:10 +00:00
2003-04-21 19:43:25 +00:00
/* Override VFS functions if 'vfs object' was not specified*/
if ( ! vfs_objects | | ! vfs_objects [ 0 ] )
2003-04-16 14:45:11 +00:00
return True ;
2003-05-11 23:34:18 +00:00
for ( i = 0 ; vfs_objects [ i ] ; ) {
i + + ;
2003-04-16 14:45:11 +00:00
}
for ( j = i - 1 ; j > = 0 ; j - - ) {
2003-05-11 23:34:18 +00:00
if ( ! vfs_init_custom ( conn , vfs_objects [ j ] ) ) {
DEBUG ( 0 , ( " smbd_vfs_init: vfs_init_custom failed for %s \n " , vfs_objects [ j ] ) ) ;
2003-04-16 14:45:11 +00:00
return False ;
2001-06-29 22:32:24 +00:00
}
}
2002-08-17 15:27:10 +00:00
return True ;
}
2000-09-27 19:09:59 +00:00
/*******************************************************************
Check if directory exists .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-02-03 05:10:09 +00:00
2001-07-04 07:15:53 +00:00
BOOL vfs_directory_exist ( connection_struct * conn , const char * dname , SMB_STRUCT_STAT * st )
2000-09-27 19:09:59 +00:00
{
SMB_STRUCT_STAT st2 ;
BOOL ret ;
if ( ! st )
st = & st2 ;
2000-02-03 05:10:09 +00:00
2003-05-14 10:59:01 +00:00
if ( SMB_VFS_STAT ( conn , dname , st ) ! = 0 )
2000-09-27 19:09:59 +00:00
return ( False ) ;
ret = S_ISDIR ( st - > st_mode ) ;
if ( ! ret )
errno = ENOTDIR ;
2000-02-03 05:10:09 +00:00
2001-07-04 07:15:53 +00:00
return ret ;
}
2000-10-03 02:12:14 +00:00
/*******************************************************************
2001-07-04 07:15:53 +00:00
vfs mkdir wrapper
2000-10-03 02:12:14 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-05-11 23:34:18 +00:00
int vfs_MkDir ( connection_struct * conn , const char * name , mode_t mode )
2000-10-03 02:12:14 +00:00
{
int ret ;
SMB_STRUCT_STAT sbuf ;
2003-05-14 10:59:01 +00:00
if ( ! ( ret = SMB_VFS_MKDIR ( conn , name , mode ) ) ) {
2002-10-23 01:22:32 +00:00
inherit_access_acl ( conn , name , mode ) ;
2000-11-14 21:56:32 +00:00
/*
2000-10-03 02:12:14 +00:00
* Check if high bits should have been set ,
* then ( if bits are missing ) : add them .
* Consider bits automagically set by UNIX , i . e . SGID bit from parent dir .
*/
if ( mode & ~ ( S_IRWXU | S_IRWXG | S_IRWXO ) & &
2003-05-14 10:59:01 +00:00
! SMB_VFS_STAT ( conn , name , & sbuf ) & & ( mode & ~ sbuf . st_mode ) )
SMB_VFS_CHMOD ( conn , name , sbuf . st_mode | ( mode & ~ sbuf . st_mode ) ) ;
2000-10-03 02:12:14 +00:00
}
return ret ;
}
2000-09-27 19:09:59 +00:00
/*******************************************************************
2002-01-04 21:11:35 +00:00
Check if an object exists in the vfs .
2000-02-03 05:10:09 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-09-27 19:09:59 +00:00
2002-01-04 21:11:35 +00:00
BOOL vfs_object_exist ( connection_struct * conn , const char * fname , SMB_STRUCT_STAT * sbuf )
2000-02-03 05:10:09 +00:00
{
2000-09-27 19:09:59 +00:00
SMB_STRUCT_STAT st ;
2000-02-03 05:10:09 +00:00
2000-09-27 19:09:59 +00:00
if ( ! sbuf )
sbuf = & st ;
2000-10-19 02:58:24 +00:00
ZERO_STRUCTP ( sbuf ) ;
2000-11-14 21:56:32 +00:00
2003-05-14 10:59:01 +00:00
if ( SMB_VFS_STAT ( conn , fname , sbuf ) = = - 1 )
2000-09-27 19:09:59 +00:00
return ( False ) ;
2002-01-04 21:11:35 +00:00
return True ;
}
/*******************************************************************
Check if a file exists in the vfs .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-03-19 02:32:39 +00:00
BOOL vfs_file_exist ( connection_struct * conn , const char * fname , SMB_STRUCT_STAT * sbuf )
2002-01-04 21:11:35 +00:00
{
SMB_STRUCT_STAT st ;
if ( ! sbuf )
sbuf = & st ;
2000-09-27 19:09:59 +00:00
2002-01-04 21:11:35 +00:00
ZERO_STRUCTP ( sbuf ) ;
2003-05-14 10:59:01 +00:00
if ( SMB_VFS_STAT ( conn , fname , sbuf ) = = - 1 )
2002-01-04 21:11:35 +00:00
return False ;
2000-09-27 19:09:59 +00:00
return ( S_ISREG ( sbuf - > st_mode ) ) ;
2000-02-03 05:10:09 +00:00
}
2000-11-14 21:56:32 +00:00
/****************************************************************************
Read data from fsp on the vfs . ( note : EINTR re - read differs from vfs_write_data )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
ssize_t vfs_read_data ( files_struct * fsp , char * buf , size_t byte_count )
{
size_t total = 0 ;
while ( total < byte_count )
{
2003-05-14 10:59:01 +00:00
ssize_t ret = SMB_VFS_READ ( fsp , fsp - > fd , buf + total ,
2002-03-27 03:00:39 +00:00
byte_count - total ) ;
2000-11-14 21:56:32 +00:00
if ( ret = = 0 ) return total ;
if ( ret = = - 1 ) {
if ( errno = = EINTR )
continue ;
else
return - 1 ;
}
total + = ret ;
}
return ( ssize_t ) total ;
}
2004-01-06 01:22:14 +00:00
ssize_t vfs_pread_data ( files_struct * fsp , char * buf ,
size_t byte_count , SMB_OFF_T offset )
{
size_t total = 0 ;
while ( total < byte_count )
{
ssize_t ret = SMB_VFS_PREAD ( fsp , fsp - > fd , buf + total ,
byte_count - total , offset + total ) ;
if ( ret = = 0 ) return total ;
if ( ret = = - 1 ) {
if ( errno = = EINTR )
continue ;
else
return - 1 ;
}
total + = ret ;
}
return ( ssize_t ) total ;
}
2000-02-03 05:10:09 +00:00
/****************************************************************************
2000-09-27 19:09:59 +00:00
Write data to a fd on the vfs .
2000-02-03 05:10:09 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-09-27 19:09:59 +00:00
2001-07-04 07:15:53 +00:00
ssize_t vfs_write_data ( files_struct * fsp , const char * buffer , size_t N )
2000-02-03 05:10:09 +00:00
{
2001-11-12 01:00:54 +00:00
size_t total = 0 ;
ssize_t ret ;
2000-02-03 05:10:09 +00:00
2001-11-12 01:00:54 +00:00
while ( total < N ) {
2003-05-14 10:59:01 +00:00
ret = SMB_VFS_WRITE ( fsp , fsp - > fd , buffer + total , N - total ) ;
2000-02-03 05:10:09 +00:00
2001-11-12 01:00:54 +00:00
if ( ret = = - 1 )
return - 1 ;
if ( ret = = 0 )
return total ;
2000-02-03 05:10:09 +00:00
2001-11-12 01:00:54 +00:00
total + = ret ;
}
return ( ssize_t ) total ;
2000-02-03 05:10:09 +00:00
}
2004-01-06 01:22:14 +00:00
ssize_t vfs_pwrite_data ( files_struct * fsp , const char * buffer ,
size_t N , SMB_OFF_T offset )
{
size_t total = 0 ;
ssize_t ret ;
while ( total < N ) {
ret = SMB_VFS_PWRITE ( fsp , fsp - > fd , buffer + total ,
N - total , offset + total ) ;
if ( ret = = - 1 )
return - 1 ;
if ( ret = = 0 )
return total ;
total + = ret ;
}
return ( ssize_t ) total ;
}
2001-08-01 22:13:50 +00:00
/****************************************************************************
An allocate file space call using the vfs interface .
Allocates space for a file from a filedescriptor .
Returns 0 on success , - 1 on failure .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-12-04 03:12:09 +00:00
int vfs_allocate_file_space ( files_struct * fsp , SMB_BIG_UINT len )
2001-08-01 22:13:50 +00:00
{
int ret ;
SMB_STRUCT_STAT st ;
2001-11-12 01:00:54 +00:00
connection_struct * conn = fsp - > conn ;
2002-12-04 03:12:09 +00:00
SMB_BIG_UINT space_avail ;
2001-11-12 01:00:54 +00:00
SMB_BIG_UINT bsize , dfree , dsize ;
2001-08-01 22:13:50 +00:00
release_level_2_oplocks_on_change ( fsp ) ;
/*
* Actually try and commit the space on disk . . . .
*/
DEBUG ( 10 , ( " vfs_allocate_file_space: file %s, len %.0f \n " , fsp - > fsp_name , ( double ) len ) ) ;
2002-12-04 03:12:09 +00:00
if ( ( ( SMB_OFF_T ) len ) < 0 ) {
DEBUG ( 0 , ( " vfs_allocate_file_space: %s negative len requested. \n " , fsp - > fsp_name ) ) ;
return - 1 ;
}
2003-05-14 10:59:01 +00:00
ret = SMB_VFS_FSTAT ( fsp , fsp - > fd , & st ) ;
2001-08-01 22:13:50 +00:00
if ( ret = = - 1 )
return ret ;
2002-12-04 03:12:09 +00:00
if ( len = = ( SMB_BIG_UINT ) st . st_size )
2001-08-01 22:13:50 +00:00
return 0 ;
2002-12-04 03:12:09 +00:00
if ( len < ( SMB_BIG_UINT ) st . st_size ) {
2001-08-01 22:13:50 +00:00
/* Shrink - use ftruncate. */
DEBUG ( 10 , ( " vfs_allocate_file_space: file %s, shrink. Current size %.0f \n " ,
fsp - > fsp_name , ( double ) st . st_size ) ) ;
2002-01-20 00:04:15 +00:00
flush_write_cache ( fsp , SIZECHANGE_FLUSH ) ;
2003-05-14 10:59:01 +00:00
if ( ( ret = SMB_VFS_FTRUNCATE ( fsp , fsp - > fd , ( SMB_OFF_T ) len ) ) ! = - 1 ) {
2001-08-01 22:13:50 +00:00
set_filelen_write_cache ( fsp , len ) ;
}
return ret ;
}
2001-11-12 01:00:54 +00:00
/* Grow - we need to test if we have enough space. */
2001-08-01 22:13:50 +00:00
2001-11-12 02:03:44 +00:00
if ( ! lp_strict_allocate ( SNUM ( fsp - > conn ) ) )
return 0 ;
2001-11-12 01:00:54 +00:00
len - = st . st_size ;
len / = 1024 ; /* Len is now number of 1k blocks needed. */
2003-05-14 10:59:01 +00:00
space_avail = SMB_VFS_DISK_FREE ( conn , fsp - > fsp_name , False , & bsize , & dfree , & dsize ) ;
2001-08-01 22:13:50 +00:00
2002-12-04 03:12:09 +00:00
DEBUG ( 10 , ( " vfs_allocate_file_space: file %s, grow. Current size %.0f, needed blocks = %.0f, space avail = %.0f \n " ,
fsp - > fsp_name , ( double ) st . st_size , ( double ) len , ( double ) space_avail ) ) ;
2001-08-01 22:13:50 +00:00
2001-11-12 01:00:54 +00:00
if ( len > space_avail ) {
errno = ENOSPC ;
return - 1 ;
2001-08-01 22:13:50 +00:00
}
2001-11-12 01:00:54 +00:00
2001-08-01 22:13:50 +00:00
return 0 ;
}
2000-11-16 00:59:18 +00:00
/****************************************************************************
A vfs set_filelen call .
set the length of a file from a filedescriptor .
Returns 0 on success , - 1 on failure .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int vfs_set_filelen ( files_struct * fsp , SMB_OFF_T len )
{
int ret ;
release_level_2_oplocks_on_change ( fsp ) ;
2001-11-07 23:47:20 +00:00
DEBUG ( 10 , ( " vfs_set_filelen: ftruncate %s to len %.0f \n " , fsp - > fsp_name , ( double ) len ) ) ;
2002-01-20 00:04:15 +00:00
flush_write_cache ( fsp , SIZECHANGE_FLUSH ) ;
2003-05-14 10:59:01 +00:00
if ( ( ret = SMB_VFS_FTRUNCATE ( fsp , fsp - > fd , len ) ) ! = - 1 )
2000-11-16 00:59:18 +00:00
set_filelen_write_cache ( fsp , len ) ;
return ret ;
}
2000-02-03 05:10:09 +00:00
/****************************************************************************
2001-09-04 19:10:30 +00:00
Transfer some data ( n bytes ) between two file_struct ' s .
2000-02-03 05:10:09 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-09-27 19:09:59 +00:00
2001-09-04 19:10:30 +00:00
static files_struct * in_fsp ;
static files_struct * out_fsp ;
2000-02-03 05:10:09 +00:00
2001-09-04 19:10:30 +00:00
static ssize_t read_fn ( int fd , void * buf , size_t len )
{
2003-05-14 10:59:01 +00:00
return SMB_VFS_READ ( in_fsp , fd , buf , len ) ;
2001-09-04 19:10:30 +00:00
}
2000-02-03 05:10:09 +00:00
2001-09-04 19:10:30 +00:00
static ssize_t write_fn ( int fd , const void * buf , size_t len )
{
2003-05-14 10:59:01 +00:00
return SMB_VFS_WRITE ( out_fsp , fd , buf , len ) ;
2001-09-04 19:10:30 +00:00
}
2000-02-03 05:10:09 +00:00
2001-09-04 19:10:30 +00:00
SMB_OFF_T vfs_transfer_file ( files_struct * in , files_struct * out , SMB_OFF_T n )
{
in_fsp = in ;
out_fsp = out ;
2000-02-03 05:10:09 +00:00
2001-09-04 19:10:30 +00:00
return transfer_file_internal ( in_fsp - > fd , out_fsp - > fd , n , read_fn , write_fn ) ;
2000-02-03 05:10:09 +00:00
}
/*******************************************************************
2000-09-27 19:09:59 +00:00
A vfs_readdir wrapper which just returns the file name .
2000-02-03 05:10:09 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-09-27 19:09:59 +00:00
2000-02-03 05:10:09 +00:00
char * vfs_readdirname ( connection_struct * conn , void * p )
{
2003-05-11 23:34:18 +00:00
struct dirent * ptr = NULL ;
2000-02-03 05:10:09 +00:00
char * dname ;
2000-09-27 19:09:59 +00:00
if ( ! p )
return ( NULL ) ;
2000-11-14 21:56:32 +00:00
2003-05-14 10:59:01 +00:00
ptr = ( struct dirent * ) SMB_VFS_READDIR ( conn , p ) ;
2000-09-27 19:09:59 +00:00
if ( ! ptr )
return ( NULL ) ;
2000-02-03 05:10:09 +00:00
dname = ptr - > d_name ;
# ifdef NEXT2
2000-09-27 19:09:59 +00:00
if ( telldir ( p ) < 0 )
return ( NULL ) ;
2000-02-03 05:10:09 +00:00
# endif
# ifdef HAVE_BROKEN_READDIR
/* using /usr/ucb/cc is BAD */
dname = dname - 2 ;
# endif
return ( dname ) ;
}
2000-09-27 19:09:59 +00:00
/*******************************************************************
A wrapper for vfs_chdir ( ) .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-03-03 20:12:37 +00:00
int vfs_ChDir ( connection_struct * conn , const char * path )
2000-09-27 19:09:59 +00:00
{
int res ;
static pstring LastDir = " " ;
if ( strcsequal ( path , " . " ) )
return ( 0 ) ;
if ( * path = = ' / ' & & strcsequal ( LastDir , path ) )
return ( 0 ) ;
2003-05-11 23:34:18 +00:00
DEBUG ( 4 , ( " vfs_ChDir to %s \n " , path ) ) ;
2000-09-27 19:09:59 +00:00
2003-05-14 10:59:01 +00:00
res = SMB_VFS_CHDIR ( conn , path ) ;
2000-09-27 19:09:59 +00:00
if ( ! res )
pstrcpy ( LastDir , path ) ;
return ( res ) ;
}
/* number of list structures for a caching GetWd function. */
# define MAX_GETWDCACHE (50)
2002-07-15 10:35:28 +00:00
static struct {
2002-03-27 03:00:39 +00:00
SMB_DEV_T dev ; /* These *must* be compatible with the types returned in a stat() call. */
SMB_INO_T inode ; /* These *must* be compatible with the types returned in a stat() call. */
char * dos_path ; /* The pathname in DOS format. */
BOOL valid ;
2000-09-27 19:09:59 +00:00
} ino_list [ MAX_GETWDCACHE ] ;
extern BOOL use_getwd_cache ;
/****************************************************************************
Prompte a ptr ( to make it recently used )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void array_promote ( char * array , int elsize , int element )
{
char * p ;
if ( element = = 0 )
return ;
p = ( char * ) malloc ( elsize ) ;
if ( ! p ) {
DEBUG ( 5 , ( " array_promote: malloc fail \n " ) ) ;
return ;
}
memcpy ( p , array + element * elsize , elsize ) ;
memmove ( array + elsize , array , elsize * element ) ;
memcpy ( array , p , elsize ) ;
2001-09-17 11:25:41 +00:00
SAFE_FREE ( p ) ;
2000-09-27 19:09:59 +00:00
}
/*******************************************************************
Return the absolute current directory path - given a UNIX pathname .
Note that this path is returned in DOS format , not UNIX
format . Note this can be called with conn = = NULL .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
char * vfs_GetWd ( connection_struct * conn , char * path )
{
2002-03-27 03:00:39 +00:00
pstring s ;
static BOOL getwd_cache_init = False ;
SMB_STRUCT_STAT st , st2 ;
int i ;
* s = 0 ;
if ( ! use_getwd_cache )
2003-05-14 10:59:01 +00:00
return ( SMB_VFS_GETWD ( conn , path ) ) ;
2002-03-27 03:00:39 +00:00
/* init the cache */
if ( ! getwd_cache_init ) {
getwd_cache_init = True ;
for ( i = 0 ; i < MAX_GETWDCACHE ; i + + ) {
string_set ( & ino_list [ i ] . dos_path , " " ) ;
ino_list [ i ] . valid = False ;
}
}
/* Get the inode of the current directory, if this doesn't work we're
in trouble : - ) */
2000-09-27 19:09:59 +00:00
2003-05-14 10:59:01 +00:00
if ( SMB_VFS_STAT ( conn , " . " , & st ) = = - 1 ) {
2002-03-27 03:00:39 +00:00
DEBUG ( 0 , ( " Very strange, couldn't stat \" . \" path=%s \n " , path ) ) ;
2003-05-14 10:59:01 +00:00
return ( SMB_VFS_GETWD ( conn , path ) ) ;
2002-03-27 03:00:39 +00:00
}
2000-09-27 19:09:59 +00:00
2002-03-27 03:00:39 +00:00
for ( i = 0 ; i < MAX_GETWDCACHE ; i + + ) {
if ( ino_list [ i ] . valid ) {
/* If we have found an entry with a matching inode and dev number
then find the inode number for the directory in the cached string .
If this agrees with that returned by the stat for the current
directory then all is o . k . ( but make sure it is a directory all
the same . . . ) */
if ( st . st_ino = = ino_list [ i ] . inode & & st . st_dev = = ino_list [ i ] . dev ) {
2003-05-14 10:59:01 +00:00
if ( SMB_VFS_STAT ( conn , ino_list [ i ] . dos_path , & st2 ) = = 0 ) {
2002-03-27 03:00:39 +00:00
if ( st . st_ino = = st2 . st_ino & & st . st_dev = = st2 . st_dev & &
( st2 . st_mode & S_IFMT ) = = S_IFDIR ) {
pstrcpy ( path , ino_list [ i ] . dos_path ) ;
/* promote it for future use */
array_promote ( ( char * ) & ino_list [ 0 ] , sizeof ( ino_list [ 0 ] ) , i ) ;
return ( path ) ;
} else {
/* If the inode is different then something's changed,
scrub the entry and start from scratch . */
ino_list [ i ] . valid = False ;
}
}
}
}
}
2000-09-27 19:09:59 +00:00
2002-03-27 03:00:39 +00:00
/* We don't have the information to hand so rely on traditional methods.
The very slow getcwd , which spawns a process on some systems , or the
not quite so bad getwd . */
2000-09-27 19:09:59 +00:00
2003-05-14 10:59:01 +00:00
if ( ! SMB_VFS_GETWD ( conn , s ) ) {
DEBUG ( 0 , ( " vfs_GetWd: SMB_VFS_GETWD call failed, errno %s \n " , strerror ( errno ) ) ) ;
2002-03-27 03:00:39 +00:00
return ( NULL ) ;
}
2000-09-27 19:09:59 +00:00
2002-03-27 03:00:39 +00:00
pstrcpy ( path , s ) ;
2000-09-27 19:09:59 +00:00
2002-03-27 03:00:39 +00:00
DEBUG ( 5 , ( " vfs_GetWd %s, inode %.0f, dev %.0f \n " , s , ( double ) st . st_ino , ( double ) st . st_dev ) ) ;
2000-09-27 19:09:59 +00:00
2002-03-27 03:00:39 +00:00
/* add it to the cache */
i = MAX_GETWDCACHE - 1 ;
string_set ( & ino_list [ i ] . dos_path , s ) ;
ino_list [ i ] . dev = st . st_dev ;
ino_list [ i ] . inode = st . st_ino ;
ino_list [ i ] . valid = True ;
/* put it at the top of the list */
array_promote ( ( char * ) & ino_list [ 0 ] , sizeof ( ino_list [ 0 ] ) , i ) ;
return ( path ) ;
2000-09-27 19:09:59 +00:00
}
2003-02-02 18:08:37 +00:00
/* check if the file 'nmae' is a symlink, in that case check that it point to
a file that reside under the ' dir ' tree */
2003-03-03 20:12:37 +00:00
static BOOL readlink_check ( connection_struct * conn , const char * dir , char * name )
2003-02-02 18:08:37 +00:00
{
BOOL ret = True ;
pstring flink ;
pstring cleanlink ;
pstring savedir ;
pstring realdir ;
size_t reallen ;
if ( ! vfs_GetWd ( conn , savedir ) ) {
DEBUG ( 0 , ( " couldn't vfs_GetWd for %s %s \n " , name , dir ) ) ;
return False ;
}
if ( vfs_ChDir ( conn , dir ) ! = 0 ) {
DEBUG ( 0 , ( " couldn't vfs_ChDir to %s \n " , dir ) ) ;
return False ;
}
if ( ! vfs_GetWd ( conn , realdir ) ) {
DEBUG ( 0 , ( " couldn't vfs_GetWd for %s \n " , dir ) ) ;
vfs_ChDir ( conn , savedir ) ;
return ( False ) ;
}
reallen = strlen ( realdir ) ;
if ( realdir [ reallen - 1 ] = = ' / ' ) {
reallen - - ;
realdir [ reallen ] = 0 ;
}
2003-05-14 10:59:01 +00:00
if ( SMB_VFS_READLINK ( conn , name , flink , sizeof ( pstring ) - 1 ) ! = - 1 ) {
2003-02-02 18:08:37 +00:00
DEBUG ( 3 , ( " reduce_name: file path name %s is a symlink \n Checking it's path \n " , name ) ) ;
if ( * flink = = ' / ' ) {
pstrcpy ( cleanlink , flink ) ;
} else {
pstrcpy ( cleanlink , realdir ) ;
pstrcat ( cleanlink , " / " ) ;
pstrcat ( cleanlink , flink ) ;
}
unix_clean_name ( cleanlink ) ;
if ( strncmp ( cleanlink , realdir , reallen ) ! = 0 ) {
DEBUG ( 2 , ( " Bad access attempt? s=%s dir=%s newname=%s l=%d \n " , name , realdir , cleanlink , ( int ) reallen ) ) ;
ret = False ;
}
}
vfs_ChDir ( conn , savedir ) ;
return ret ;
}
2000-09-27 19:09:59 +00:00
/*******************************************************************
2000-11-14 21:56:32 +00:00
Reduce a file name , removing . . elements and checking that
2000-09-27 19:09:59 +00:00
it is below dir in the heirachy . This uses vfs_GetWd ( ) and so must be run
on the system that has the referenced file system .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-03-05 01:37:12 +00:00
BOOL reduce_name ( connection_struct * conn , pstring s , const char * dir )
2000-09-27 19:09:59 +00:00
{
# ifndef REDUCE_PATHS
2002-03-27 03:00:39 +00:00
return True ;
2000-09-27 19:09:59 +00:00
# else
2002-03-27 03:00:39 +00:00
pstring dir2 ;
pstring wd ;
pstring base_name ;
pstring newname ;
char * p = NULL ;
BOOL relative = ( * s ! = ' / ' ) ;
* dir2 = * wd = * base_name = * newname = 0 ;
DEBUG ( 3 , ( " reduce_name [%s] [%s] \n " , s , dir ) ) ;
2004-03-05 22:32:45 +00:00
/* We know there are no double slashes as this comes from srvstr_get_path().
and has gone through check_path_syntax ( ) . JRA */
2002-03-27 03:00:39 +00:00
pstrcpy ( base_name , s ) ;
p = strrchr_m ( base_name , ' / ' ) ;
if ( ! p )
2003-02-02 18:08:37 +00:00
return readlink_check ( conn , dir , s ) ;
2002-03-27 03:00:39 +00:00
if ( ! vfs_GetWd ( conn , wd ) ) {
DEBUG ( 0 , ( " couldn't vfs_GetWd for %s %s \n " , s , dir ) ) ;
return ( False ) ;
}
if ( vfs_ChDir ( conn , dir ) ! = 0 ) {
DEBUG ( 0 , ( " couldn't vfs_ChDir to %s \n " , dir ) ) ;
return ( False ) ;
}
if ( ! vfs_GetWd ( conn , dir2 ) ) {
DEBUG ( 0 , ( " couldn't vfs_GetWd for %s \n " , dir ) ) ;
vfs_ChDir ( conn , wd ) ;
return ( False ) ;
}
if ( p & & ( p ! = base_name ) ) {
* p = 0 ;
if ( strcmp ( p + 1 , " . " ) = = 0 )
p [ 1 ] = 0 ;
if ( strcmp ( p + 1 , " .. " ) = = 0 )
* p = ' / ' ;
}
if ( vfs_ChDir ( conn , base_name ) ! = 0 ) {
vfs_ChDir ( conn , wd ) ;
DEBUG ( 3 , ( " couldn't vfs_ChDir for %s %s basename=%s \n " , s , dir , base_name ) ) ;
return ( False ) ;
}
if ( ! vfs_GetWd ( conn , newname ) ) {
vfs_ChDir ( conn , wd ) ;
2003-02-02 18:08:37 +00:00
DEBUG ( 2 , ( " couldn't get vfs_GetWd for %s %s \n " , s , base_name ) ) ;
2002-03-27 03:00:39 +00:00
return ( False ) ;
}
if ( p & & ( p ! = base_name ) ) {
pstrcat ( newname , " / " ) ;
pstrcat ( newname , p + 1 ) ;
}
{
size_t l = strlen ( dir2 ) ;
2004-03-05 22:32:45 +00:00
char * last_slash = strrchr_m ( dir2 , ' / ' ) ;
if ( last_slash & & ( last_slash [ 1 ] = = ' \0 ' ) )
2002-03-27 03:00:39 +00:00
l - - ;
if ( strncmp ( newname , dir2 , l ) ! = 0 ) {
vfs_ChDir ( conn , wd ) ;
2004-03-05 22:32:45 +00:00
DEBUG ( 2 , ( " Bad access attempt: s=%s dir=%s newname=%s l=%d \n " , s , dir2 , newname , ( int ) l ) ) ;
2002-03-27 03:00:39 +00:00
return ( False ) ;
}
2003-02-02 18:08:37 +00:00
if ( ! readlink_check ( conn , dir , newname ) ) {
2004-03-05 22:32:45 +00:00
DEBUG ( 2 , ( " Bad access attemt: %s is a symlink outside the share path " , s ) ) ;
2003-02-02 18:08:37 +00:00
return ( False ) ;
}
2002-03-27 03:00:39 +00:00
if ( relative ) {
if ( newname [ l ] = = ' / ' )
pstrcpy ( s , newname + l + 1 ) ;
else
pstrcpy ( s , newname + l ) ;
} else
pstrcpy ( s , newname ) ;
}
2000-09-27 19:09:59 +00:00
2002-03-27 03:00:39 +00:00
vfs_ChDir ( conn , wd ) ;
2000-09-27 19:09:59 +00:00
2002-03-27 03:00:39 +00:00
if ( strlen ( s ) = = 0 )
pstrcpy ( s , " ./ " ) ;
2000-09-27 19:09:59 +00:00
2002-03-27 03:00:39 +00:00
DEBUG ( 3 , ( " reduced to %s \n " , s ) ) ;
return ( True ) ;
2000-09-27 19:09:59 +00:00
# endif
}