1998-08-17 07:40:06 +00:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
1998-08-17 07:40:06 +00:00
dos mode handling functions
Copyright ( C ) Andrew Tridgell 1992 - 1998
2006-03-22 23:49:09 +00:00
Copyright ( C ) James Peach 2006
1998-08-17 07:40:06 +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
2007-07-09 19:25:36 +00:00
the Free Software Foundation ; either version 3 of the License , or
1998-08-17 07:40:06 +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/>.
1998-08-17 07:40:06 +00:00
*/
# include "includes.h"
2005-06-10 00:31:59 +00:00
static int set_sparse_flag ( const SMB_STRUCT_STAT * const sbuf )
{
# if defined (HAVE_STAT_ST_BLOCKS) && defined(STAT_ST_BLOCKSIZE)
if ( sbuf - > st_size > sbuf - > st_blocks * ( SMB_OFF_T ) STAT_ST_BLOCKSIZE ) {
return FILE_ATTRIBUTE_SPARSE ;
}
# endif
return 0 ;
}
1998-08-17 07:40:06 +00:00
/****************************************************************************
2004-03-31 22:46:15 +00:00
Change a dos mode to a unix mode .
Base permission for files :
2006-12-27 10:57:59 +00:00
if creating file and inheriting ( i . e . parent_dir ! = NULL )
2000-01-14 01:41:04 +00:00
apply read / write bits from parent directory .
else
everybody gets read bit set
1998-08-17 07:40:06 +00:00
dos readonly is represented in unix by removing everyone ' s write bit
dos archive is represented in unix by the user ' s execute bit
dos system is represented in unix by the group ' s execute bit
dos hidden is represented in unix by the other ' s execute bit
2000-01-14 01:41:04 +00:00
if ! inheriting {
Then apply create mask ,
then add force bits .
}
2004-03-31 22:46:15 +00:00
Base permission for directories :
1998-08-17 07:40:06 +00:00
dos directory is represented in unix by unix ' s dir bit and the exec bit
2000-01-14 01:41:04 +00:00
if ! inheriting {
Then apply create mask ,
then add force bits .
}
1998-08-17 07:40:06 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-03-31 22:46:15 +00:00
2006-12-27 10:57:59 +00:00
mode_t unix_mode ( connection_struct * conn , int dosmode , const char * fname ,
const char * inherit_from_dir )
1998-08-17 07:40:06 +00:00
{
2004-04-02 18:46:19 +00:00
mode_t result = ( S_IRUSR | S_IRGRP | S_IROTH | S_IWUSR | S_IWGRP | S_IWOTH ) ;
2006-12-27 10:57:59 +00:00
mode_t dir_mode = 0 ; /* Mode of the inherit_from directory if
* inheriting . */
2004-03-31 22:46:15 +00:00
2004-04-02 18:46:19 +00:00
if ( ! lp_store_dos_attributes ( SNUM ( conn ) ) & & IS_DOS_READONLY ( dosmode ) ) {
result & = ~ ( S_IWUSR | S_IWGRP | S_IWOTH ) ;
}
2004-03-31 22:46:15 +00:00
2006-12-27 10:57:59 +00:00
if ( fname & & ( inherit_from_dir ! = NULL )
& & lp_inherit_perms ( SNUM ( conn ) ) ) {
2004-03-31 22:46:15 +00:00
SMB_STRUCT_STAT sbuf ;
2006-12-27 10:57:59 +00:00
DEBUG ( 2 , ( " unix_mode(%s) inheriting from %s \n " , fname ,
inherit_from_dir ) ) ;
if ( SMB_VFS_STAT ( conn , inherit_from_dir , & sbuf ) ! = 0 ) {
DEBUG ( 4 , ( " unix_mode(%s) failed, [dir %s]: %s \n " , fname ,
inherit_from_dir , strerror ( errno ) ) ) ;
2004-03-31 22:46:15 +00:00
return ( 0 ) ; /* *** shouldn't happen! *** */
}
/* Save for later - but explicitly remove setuid bit for safety. */
dir_mode = sbuf . st_mode & ~ S_ISUID ;
DEBUG ( 2 , ( " unix_mode(%s) inherit mode %o \n " , fname , ( int ) dir_mode ) ) ;
/* Clear "result" */
result = 0 ;
}
if ( IS_DOS_DIR ( dosmode ) ) {
/* We never make directories read only for the owner as under DOS a user
can always create a file in a read - only directory . */
result | = ( S_IFDIR | S_IWUSR ) ;
if ( dir_mode ) {
/* Inherit mode of parent directory. */
result | = dir_mode ;
} else {
/* Provisionally add all 'x' bits */
result | = ( S_IXUSR | S_IXGRP | S_IXOTH ) ;
/* Apply directory mask */
result & = lp_dir_mask ( SNUM ( conn ) ) ;
/* Add in force bits */
result | = lp_force_dir_mode ( SNUM ( conn ) ) ;
}
} else {
if ( lp_map_archive ( SNUM ( conn ) ) & & IS_DOS_ARCHIVE ( dosmode ) )
result | = S_IXUSR ;
if ( lp_map_system ( SNUM ( conn ) ) & & IS_DOS_SYSTEM ( dosmode ) )
result | = S_IXGRP ;
1998-08-17 07:40:06 +00:00
2004-03-31 22:46:15 +00:00
if ( lp_map_hidden ( SNUM ( conn ) ) & & IS_DOS_HIDDEN ( dosmode ) )
result | = S_IXOTH ;
if ( dir_mode ) {
/* Inherit 666 component of parent directory mode */
result | = dir_mode & ( S_IRUSR | S_IRGRP | S_IROTH | S_IWUSR | S_IWGRP | S_IWOTH ) ;
} else {
/* Apply mode mask */
result & = lp_create_mask ( SNUM ( conn ) ) ;
/* Add in force bits */
result | = lp_force_create_mode ( SNUM ( conn ) ) ;
}
}
DEBUG ( 3 , ( " unix_mode(%s) returning 0%o \n " , fname , ( int ) result ) ) ;
return ( result ) ;
1998-08-17 07:40:06 +00:00
}
/****************************************************************************
2004-03-31 22:46:15 +00:00
Change a unix mode to a dos mode .
1998-08-17 07:40:06 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-03-19 23:46:48 +00:00
2005-11-17 22:39:00 +00:00
static uint32 dos_mode_from_sbuf ( connection_struct * conn , const char * path , SMB_STRUCT_STAT * sbuf )
1998-08-17 07:40:06 +00:00
{
2002-08-17 15:27:10 +00:00
int result = 0 ;
2005-10-14 01:09:37 +00:00
enum mapreadonly_options ro_opts = ( enum mapreadonly_options ) lp_map_readonly ( SNUM ( conn ) ) ;
1998-08-17 07:40:06 +00:00
2005-10-14 01:09:37 +00:00
if ( ro_opts = = MAP_READONLY_YES ) {
/* Original Samba method - map inverse of user "w" bit. */
if ( ( sbuf - > st_mode & S_IWUSR ) = = 0 ) {
result | = aRONLY ;
}
} else if ( ro_opts = = MAP_READONLY_PERMISSIONS ) {
/* Check actual permissions for read-only. */
2005-05-18 23:37:35 +00:00
if ( ! can_write_to_file ( conn , path , sbuf ) ) {
result | = aRONLY ;
}
2005-10-14 01:09:37 +00:00
} /* Else never set the readonly bit. */
2005-05-18 23:37:35 +00:00
2002-08-17 15:27:10 +00:00
if ( MAP_ARCHIVE ( conn ) & & ( ( sbuf - > st_mode & S_IXUSR ) ! = 0 ) )
result | = aARCH ;
1998-08-17 07:40:06 +00:00
2002-08-17 15:27:10 +00:00
if ( MAP_SYSTEM ( conn ) & & ( ( sbuf - > st_mode & S_IXGRP ) ! = 0 ) )
result | = aSYSTEM ;
if ( MAP_HIDDEN ( conn ) & & ( ( sbuf - > st_mode & S_IXOTH ) ! = 0 ) )
result | = aHIDDEN ;
1998-08-17 07:40:06 +00:00
2002-08-17 15:27:10 +00:00
if ( S_ISDIR ( sbuf - > st_mode ) )
result = aDIR | ( result & aRONLY ) ;
2005-06-10 00:31:59 +00:00
result | = set_sparse_flag ( sbuf ) ;
2000-05-16 01:13:16 +00:00
1998-08-17 07:40:06 +00:00
# ifdef S_ISLNK
# if LINKS_READ_ONLY
2002-08-17 15:27:10 +00:00
if ( S_ISLNK ( sbuf - > st_mode ) & & S_ISDIR ( sbuf - > st_mode ) )
result | = aRONLY ;
1998-08-17 07:40:06 +00:00
# endif
# endif
2004-03-19 23:46:48 +00:00
DEBUG ( 8 , ( " dos_mode_from_sbuf returning " ) ) ;
if ( result & aHIDDEN ) DEBUG ( 8 , ( " h " ) ) ;
if ( result & aRONLY ) DEBUG ( 8 , ( " r " ) ) ;
if ( result & aSYSTEM ) DEBUG ( 8 , ( " s " ) ) ;
if ( result & aDIR ) DEBUG ( 8 , ( " d " ) ) ;
if ( result & aARCH ) DEBUG ( 8 , ( " a " ) ) ;
DEBUG ( 8 , ( " \n " ) ) ;
return result ;
}
2004-04-02 18:46:19 +00:00
/****************************************************************************
Get DOS attributes from an EA .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
static bool get_ea_dos_attribute ( connection_struct * conn , const char * path , SMB_STRUCT_STAT * sbuf , uint32 * pattr )
2004-04-02 18:46:19 +00:00
{
ssize_t sizeret ;
fstring attrstr ;
unsigned int dosattr ;
if ( ! lp_store_dos_attributes ( SNUM ( conn ) ) ) {
return False ;
}
2005-11-28 17:03:50 +00:00
/* Don't reset pattr to zero as we may already have filename-based attributes we
need to preserve . */
2004-04-02 18:46:19 +00:00
sizeret = SMB_VFS_GETXATTR ( conn , path , SAMBA_XATTR_DOS_ATTRIB , attrstr , sizeof ( attrstr ) ) ;
if ( sizeret = = - 1 ) {
# if defined(ENOTSUP) && defined(ENOATTR)
2006-04-08 05:09:12 +00:00
if ( ( errno ! = ENOTSUP ) & & ( errno ! = ENOATTR ) & & ( errno ! = EACCES ) & & ( errno ! = EPERM ) ) {
2004-04-02 18:46:19 +00:00
DEBUG ( 1 , ( " get_ea_dos_attributes: Cannot get attribute from EA on file %s: Error = %s \n " ,
path , strerror ( errno ) ) ) ;
2004-06-01 20:43:32 +00:00
set_store_dos_attributes ( SNUM ( conn ) , False ) ;
2004-04-02 18:46:19 +00:00
}
# endif
return False ;
}
/* Null terminate string. */
attrstr [ sizeret ] = 0 ;
DEBUG ( 10 , ( " get_ea_dos_attribute: %s attrstr = %s \n " , path , attrstr ) ) ;
if ( sizeret < 2 | | attrstr [ 0 ] ! = ' 0 ' | | attrstr [ 1 ] ! = ' x ' | |
sscanf ( attrstr , " %x " , & dosattr ) ! = 1 ) {
DEBUG ( 1 , ( " get_ea_dos_attributes: Badly formed DOSATTRIB on file %s - %s \n " , path , attrstr ) ) ;
return False ;
}
if ( S_ISDIR ( sbuf - > st_mode ) ) {
dosattr | = aDIR ;
}
* pattr = ( uint32 ) ( dosattr & SAMBA_ATTRIBUTES_MASK ) ;
DEBUG ( 8 , ( " get_ea_dos_attribute returning (0x%x) " , dosattr ) ) ;
if ( dosattr & aHIDDEN ) DEBUG ( 8 , ( " h " ) ) ;
if ( dosattr & aRONLY ) DEBUG ( 8 , ( " r " ) ) ;
if ( dosattr & aSYSTEM ) DEBUG ( 8 , ( " s " ) ) ;
if ( dosattr & aDIR ) DEBUG ( 8 , ( " d " ) ) ;
if ( dosattr & aARCH ) DEBUG ( 8 , ( " a " ) ) ;
DEBUG ( 8 , ( " \n " ) ) ;
return True ;
}
/****************************************************************************
Set DOS attributes in an EA .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
static bool set_ea_dos_attribute ( connection_struct * conn , const char * path , SMB_STRUCT_STAT * sbuf , uint32 dosmode )
2004-04-02 18:46:19 +00:00
{
fstring attrstr ;
files_struct * fsp = NULL ;
2007-10-18 17:40:25 -07:00
bool ret = False ;
2004-04-02 18:46:19 +00:00
2004-06-01 20:43:32 +00:00
if ( ! lp_store_dos_attributes ( SNUM ( conn ) ) ) {
return False ;
}
2004-04-02 18:46:19 +00:00
snprintf ( attrstr , sizeof ( attrstr ) - 1 , " 0x%x " , dosmode & SAMBA_ATTRIBUTES_MASK ) ;
if ( SMB_VFS_SETXATTR ( conn , path , SAMBA_XATTR_DOS_ATTRIB , attrstr , strlen ( attrstr ) , 0 ) = = - 1 ) {
if ( ( errno ! = EPERM ) & & ( errno ! = EACCES ) ) {
2004-06-01 20:43:32 +00:00
if ( errno = = ENOSYS
# if defined(ENOTSUP)
| | errno = = ENOTSUP ) {
# else
) {
# endif
set_store_dos_attributes ( SNUM ( conn ) , False ) ;
}
2004-04-02 18:46:19 +00:00
return False ;
}
/* We want DOS semantics, ie allow non owner with write permission to change the
2007-03-05 23:40:03 +00:00
bits on a file . Just like file_ntimes below .
2004-04-02 18:46:19 +00:00
*/
/* Check if we have write access. */
if ( ! CAN_WRITE ( conn ) | | ! lp_dos_filemode ( SNUM ( conn ) ) )
return False ;
/*
* We need to open the file with write access whilst
* still in our current user context . This ensures we
* are not violating security in doing the setxattr .
*/
2008-10-09 15:38:53 +02:00
if ( ! NT_STATUS_IS_OK ( open_file_fchmod ( NULL , conn , path , sbuf ,
& fsp ) ) )
2004-04-02 18:46:19 +00:00
return ret ;
become_root ( ) ;
if ( SMB_VFS_SETXATTR ( conn , path , SAMBA_XATTR_DOS_ATTRIB , attrstr , strlen ( attrstr ) , 0 ) = = 0 ) {
ret = True ;
}
unbecome_root ( ) ;
2008-10-09 16:27:49 +02:00
close_file_fchmod ( NULL , fsp ) ;
2004-04-02 18:46:19 +00:00
return ret ;
}
DEBUG ( 10 , ( " set_ea_dos_attribute: set EA %s on file %s \n " , attrstr , path ) ) ;
return True ;
}
2006-07-03 21:07:46 +00:00
/****************************************************************************
Change a unix mode to a dos mode for an ms dfs link .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
uint32 dos_mode_msdfs ( connection_struct * conn , const char * path , SMB_STRUCT_STAT * sbuf )
{
uint32 result = 0 ;
DEBUG ( 8 , ( " dos_mode_msdfs: %s \n " , path ) ) ;
if ( ! VALID_STAT ( * sbuf ) ) {
return 0 ;
}
/* First do any modifications that depend on the path name. */
/* hide files with a name starting with a . */
if ( lp_hide_dot_files ( SNUM ( conn ) ) ) {
const char * p = strrchr_m ( path , ' / ' ) ;
if ( p ) {
p + + ;
} else {
p = path ;
}
if ( p [ 0 ] = = ' . ' & & p [ 1 ] ! = ' . ' & & p [ 1 ] ! = 0 ) {
result | = aHIDDEN ;
}
}
result | = dos_mode_from_sbuf ( conn , path , sbuf ) ;
/* Optimization : Only call is_hidden_path if it's not already
hidden . */
if ( ! ( result & aHIDDEN ) & & IS_HIDDEN_PATH ( conn , path ) ) {
result | = aHIDDEN ;
}
DEBUG ( 8 , ( " dos_mode_msdfs returning " ) ) ;
if ( result & aHIDDEN ) DEBUG ( 8 , ( " h " ) ) ;
if ( result & aRONLY ) DEBUG ( 8 , ( " r " ) ) ;
if ( result & aSYSTEM ) DEBUG ( 8 , ( " s " ) ) ;
if ( result & aDIR ) DEBUG ( 8 , ( " d " ) ) ;
if ( result & aARCH ) DEBUG ( 8 , ( " a " ) ) ;
if ( result & FILE_ATTRIBUTE_SPARSE ) DEBUG ( 8 , ( " [sparse] " ) ) ;
DEBUG ( 8 , ( " \n " ) ) ;
return ( result ) ;
}
2004-03-19 23:46:48 +00:00
/****************************************************************************
2004-03-31 22:46:15 +00:00
Change a unix mode to a dos mode .
2004-03-19 23:46:48 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-03-31 22:46:15 +00:00
2004-04-02 18:46:19 +00:00
uint32 dos_mode ( connection_struct * conn , const char * path , SMB_STRUCT_STAT * sbuf )
2004-03-19 23:46:48 +00:00
{
2004-04-02 18:46:19 +00:00
uint32 result = 0 ;
2008-01-16 12:17:03 +03:00
bool offline ;
2004-03-19 23:46:48 +00:00
DEBUG ( 8 , ( " dos_mode: %s \n " , path ) ) ;
2004-04-02 18:46:19 +00:00
if ( ! VALID_STAT ( * sbuf ) ) {
return 0 ;
}
2005-09-20 20:20:51 +00:00
/* First do any modifications that depend on the path name. */
2002-08-17 15:27:10 +00:00
/* hide files with a name starting with a . */
if ( lp_hide_dot_files ( SNUM ( conn ) ) ) {
2004-04-02 18:46:19 +00:00
const char * p = strrchr_m ( path , ' / ' ) ;
2005-09-20 20:20:51 +00:00
if ( p ) {
2002-08-17 15:27:10 +00:00
p + + ;
2005-09-20 20:20:51 +00:00
} else {
2002-08-17 15:27:10 +00:00
p = path ;
2005-09-20 20:20:51 +00:00
}
2002-08-17 15:27:10 +00:00
2005-09-20 20:20:51 +00:00
if ( p [ 0 ] = = ' . ' & & p [ 1 ] ! = ' . ' & & p [ 1 ] ! = 0 ) {
2002-08-17 15:27:10 +00:00
result | = aHIDDEN ;
2005-09-20 20:20:51 +00:00
}
2002-08-17 15:27:10 +00:00
}
2005-09-20 20:20:51 +00:00
/* Get the DOS attributes from an EA by preference. */
if ( get_ea_dos_attribute ( conn , path , sbuf , & result ) ) {
result | = set_sparse_flag ( sbuf ) ;
} else {
result | = dos_mode_from_sbuf ( conn , path , sbuf ) ;
}
2008-01-16 12:17:03 +03:00
2008-01-17 14:57:35 +03:00
offline = SMB_VFS_IS_OFFLINE ( conn , path , sbuf ) ;
if ( S_ISREG ( sbuf - > st_mode ) & & offline ) {
2008-01-16 12:17:03 +03:00
result | = FILE_ATTRIBUTE_OFFLINE ;
2006-03-22 23:49:09 +00:00
}
2002-08-17 15:27:10 +00:00
/* Optimization : Only call is_hidden_path if it's not already
hidden . */
if ( ! ( result & aHIDDEN ) & & IS_HIDDEN_PATH ( conn , path ) ) {
result | = aHIDDEN ;
}
DEBUG ( 8 , ( " dos_mode returning " ) ) ;
if ( result & aHIDDEN ) DEBUG ( 8 , ( " h " ) ) ;
if ( result & aRONLY ) DEBUG ( 8 , ( " r " ) ) ;
if ( result & aSYSTEM ) DEBUG ( 8 , ( " s " ) ) ;
if ( result & aDIR ) DEBUG ( 8 , ( " d " ) ) ;
if ( result & aARCH ) DEBUG ( 8 , ( " a " ) ) ;
2005-11-14 06:29:48 +00:00
if ( result & FILE_ATTRIBUTE_SPARSE ) DEBUG ( 8 , ( " [sparse] " ) ) ;
2002-08-17 15:27:10 +00:00
DEBUG ( 8 , ( " \n " ) ) ;
return ( result ) ;
1998-08-17 07:40:06 +00:00
}
/*******************************************************************
2004-03-31 22:46:15 +00:00
chmod a file - but preserve some bits .
1998-08-17 07:40:06 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2003-11-25 23:25:42 +00:00
2006-12-27 10:57:59 +00:00
int file_set_dosmode ( connection_struct * conn , const char * fname ,
uint32 dosmode , SMB_STRUCT_STAT * st ,
2007-10-31 15:45:45 -07:00
const char * parent_dir ,
bool newfile )
1998-08-17 07:40:06 +00:00
{
2001-01-09 20:34:37 +00:00
SMB_STRUCT_STAT st1 ;
int mask = 0 ;
mode_t tmp ;
mode_t unixmode ;
2008-01-16 12:17:03 +03:00
int ret = - 1 , lret = - 1 ;
2008-02-06 09:09:23 +03:00
uint32_t old_mode ;
2001-01-09 20:34:37 +00:00
2005-11-14 06:29:48 +00:00
/* We only allow READONLY|HIDDEN|SYSTEM|DIRECTORY|ARCHIVE here. */
2008-02-06 10:31:22 -08:00
dosmode & = ( SAMBA_ATTRIBUTES_MASK | FILE_ATTRIBUTE_OFFLINE ) ;
2005-11-14 06:29:48 +00:00
2004-04-02 18:46:19 +00:00
DEBUG ( 10 , ( " file_set_dosmode: setting dos mode 0x%x on file %s \n " , dosmode , fname ) ) ;
2007-12-27 21:31:08 +01:00
if ( st = = NULL ) {
SET_STAT_INVALID ( st1 ) ;
2001-01-09 20:34:37 +00:00
st = & st1 ;
2007-12-27 21:31:08 +01:00
}
if ( ! VALID_STAT ( * st ) ) {
2003-05-14 10:59:01 +00:00
if ( SMB_VFS_STAT ( conn , fname , st ) )
2001-01-09 20:34:37 +00:00
return ( - 1 ) ;
}
1998-08-17 07:40:06 +00:00
2007-12-27 21:31:08 +01:00
unixmode = st - > st_mode ;
2004-11-10 23:12:02 +00:00
get_acl_group_bits ( conn , fname , & st - > st_mode ) ;
2003-11-25 23:25:42 +00:00
2001-01-09 20:34:37 +00:00
if ( S_ISDIR ( st - > st_mode ) )
dosmode | = aDIR ;
2002-03-05 01:43:50 +00:00
else
dosmode & = ~ aDIR ;
1998-08-17 07:40:06 +00:00
2008-02-06 09:09:23 +03:00
old_mode = dos_mode ( conn , fname , st ) ;
if ( dosmode & FILE_ATTRIBUTE_OFFLINE ) {
if ( ! ( old_mode & FILE_ATTRIBUTE_OFFLINE ) ) {
lret = SMB_VFS_SET_OFFLINE ( conn , fname ) ;
if ( lret = = - 1 ) {
DEBUG ( 0 , ( " set_dos_mode: client has asked to set "
" FILE_ATTRIBUTE_OFFLINE to %s/%s but there was "
" an error while setting it or it is not supported. \n " ,
parent_dir , fname ) ) ;
}
}
}
dosmode & = ~ FILE_ATTRIBUTE_OFFLINE ;
old_mode & = ~ FILE_ATTRIBUTE_OFFLINE ;
if ( old_mode = = dosmode ) {
2007-12-27 21:31:08 +01:00
st - > st_mode = unixmode ;
2001-01-09 20:34:37 +00:00
return ( 0 ) ;
2007-12-27 21:31:08 +01:00
}
1998-08-17 07:40:06 +00:00
2004-04-02 18:46:19 +00:00
/* Store the DOS attributes in an EA by preference. */
if ( set_ea_dos_attribute ( conn , fname , st , dosmode ) ) {
2007-10-31 15:45:45 -07:00
if ( ! newfile ) {
notify_fname ( conn , NOTIFY_ACTION_MODIFIED ,
FILE_NOTIFY_CHANGE_ATTRIBUTES , fname ) ;
}
2007-12-27 21:31:08 +01:00
st - > st_mode = unixmode ;
2004-04-02 18:46:19 +00:00
return 0 ;
}
2006-12-27 10:57:59 +00:00
unixmode = unix_mode ( conn , dosmode , fname , parent_dir ) ;
1998-08-17 07:40:06 +00:00
2001-01-09 20:34:37 +00:00
/* preserve the s bits */
mask | = ( S_ISUID | S_ISGID ) ;
1998-08-17 07:40:06 +00:00
2001-01-09 20:34:37 +00:00
/* preserve the t bit */
1998-08-17 07:40:06 +00:00
# ifdef S_ISVTX
2001-01-09 20:34:37 +00:00
mask | = S_ISVTX ;
1998-08-17 07:40:06 +00:00
# endif
2001-01-09 20:34:37 +00:00
/* possibly preserve the x bits */
if ( ! MAP_ARCHIVE ( conn ) )
mask | = S_IXUSR ;
if ( ! MAP_SYSTEM ( conn ) )
mask | = S_IXGRP ;
if ( ! MAP_HIDDEN ( conn ) )
mask | = S_IXOTH ;
unixmode | = ( st - > st_mode & mask ) ;
/* if we previously had any r bits set then leave them alone */
if ( ( tmp = st - > st_mode & ( S_IRUSR | S_IRGRP | S_IROTH ) ) ) {
unixmode & = ~ ( S_IRUSR | S_IRGRP | S_IROTH ) ;
unixmode | = tmp ;
}
/* if we previously had any w bits set then leave them alone
whilst adding in the new w bits , if the new mode is not rdonly */
if ( ! IS_DOS_READONLY ( dosmode ) ) {
unixmode | = ( st - > st_mode & ( S_IWUSR | S_IWGRP | S_IWOTH ) ) ;
}
2008-01-16 12:17:03 +03:00
ret = SMB_VFS_CHMOD ( conn , fname , unixmode ) ;
if ( ret = = 0 ) {
if ( ! newfile | | ( lret ! = - 1 ) ) {
2007-10-31 15:45:45 -07:00
notify_fname ( conn , NOTIFY_ACTION_MODIFIED ,
2008-01-16 12:17:03 +03:00
FILE_NOTIFY_CHANGE_ATTRIBUTES , fname ) ;
2007-10-31 15:45:45 -07:00
}
2007-12-27 21:31:08 +01:00
st - > st_mode = unixmode ;
2001-01-17 23:47:08 +00:00
return 0 ;
2007-01-17 16:23:45 +00:00
}
2001-01-09 20:34:37 +00:00
2001-01-17 23:47:08 +00:00
if ( ( errno ! = EPERM ) & & ( errno ! = EACCES ) )
2001-01-09 20:34:37 +00:00
return - 1 ;
if ( ! lp_dos_filemode ( SNUM ( conn ) ) )
return - 1 ;
/* We want DOS semantics, ie allow non owner with write permission to change the
2007-03-05 23:40:03 +00:00
bits on a file . Just like file_ntimes below .
2001-01-09 20:34:37 +00:00
*/
/* Check if we have write access. */
if ( CAN_WRITE ( conn ) ) {
2001-04-14 00:19:12 +00:00
/*
* We need to open the file with write access whilst
* still in our current user context . This ensures we
* are not violating security in doing the fchmod .
* This file open does * not * break any oplocks we are
* holding . We need to review this . . . . may need to
* break batch oplocks open by others . JRA .
*/
2006-07-11 18:01:26 +00:00
files_struct * fsp ;
2008-10-09 15:38:53 +02:00
if ( ! NT_STATUS_IS_OK ( open_file_fchmod ( NULL , conn , fname , st ,
& fsp ) ) )
2001-04-14 00:19:12 +00:00
return - 1 ;
become_root ( ) ;
2008-01-07 13:44:37 +01:00
ret = SMB_VFS_FCHMOD ( fsp , unixmode ) ;
2001-04-14 00:19:12 +00:00
unbecome_root ( ) ;
2008-10-09 16:27:49 +02:00
close_file_fchmod ( NULL , fsp ) ;
2007-10-31 15:45:45 -07:00
if ( ! newfile ) {
notify_fname ( conn , NOTIFY_ACTION_MODIFIED ,
FILE_NOTIFY_CHANGE_ATTRIBUTES , fname ) ;
}
2007-12-27 21:31:08 +01:00
if ( ret = = 0 ) {
st - > st_mode = unixmode ;
}
2001-01-09 20:34:37 +00:00
}
return ( ret ) ;
1998-08-17 07:40:06 +00:00
}
1998-08-17 13:11:34 +00:00
/*******************************************************************
2007-03-05 23:40:03 +00:00
Wrapper around the VFS ntimes that possibly allows DOS semantics rather
2004-03-31 22:46:15 +00:00
than POSIX .
1998-08-17 13:11:34 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-03-31 22:46:15 +00:00
2007-03-05 23:40:03 +00:00
int file_ntimes ( connection_struct * conn , const char * fname , const struct timespec ts [ 2 ] )
1998-08-17 13:11:34 +00:00
{
2005-05-18 23:37:35 +00:00
SMB_STRUCT_STAT sbuf ;
2004-03-31 22:46:15 +00:00
int ret = - 1 ;
errno = 0 ;
2005-05-18 23:37:35 +00:00
ZERO_STRUCT ( sbuf ) ;
2004-03-31 22:46:15 +00:00
2008-03-12 15:39:38 +01:00
DEBUG ( 6 , ( " file_ntime: actime: %s " ,
time_to_asc ( convert_timespec_to_time_t ( ts [ 0 ] ) ) ) ) ;
DEBUG ( 6 , ( " file_ntime: modtime: %s " ,
time_to_asc ( convert_timespec_to_time_t ( ts [ 1 ] ) ) ) ) ;
2005-09-16 20:06:10 +00:00
/* Don't update the time on read-only shares */
/* We need this as set_filetime (which can be called on
close and other paths ) can end up calling this function
without the NEED_WRITE protection . Found by :
Leo Weppelman < leo @ wau . mis . ah . nl >
*/
if ( ! CAN_WRITE ( conn ) ) {
return 0 ;
}
2007-03-05 23:40:03 +00:00
if ( SMB_VFS_NTIMES ( conn , fname , ts ) = = 0 ) {
2004-03-31 22:46:15 +00:00
return 0 ;
2007-03-05 23:40:03 +00:00
}
2004-03-31 22:46:15 +00:00
2007-03-05 23:40:03 +00:00
if ( ( errno ! = EPERM ) & & ( errno ! = EACCES ) ) {
2004-03-31 22:46:15 +00:00
return - 1 ;
2007-03-05 23:40:03 +00:00
}
2004-03-31 22:46:15 +00:00
2007-03-05 23:40:03 +00:00
if ( ! lp_dos_filetimes ( SNUM ( conn ) ) ) {
2004-03-31 22:46:15 +00:00
return - 1 ;
2007-03-05 23:40:03 +00:00
}
2004-03-31 22:46:15 +00:00
/* We have permission (given by the Samba admin) to
break POSIX semantics and allow a user to change
the time on a file they don ' t own but can write to
( as DOS does ) .
*/
/* Check if we have write access. */
2005-10-11 04:28:46 +00:00
if ( can_write_to_file ( conn , fname , & sbuf ) ) {
2005-03-24 22:34:28 +00:00
/* We are allowed to become root and change the filetime. */
become_root ( ) ;
2007-03-05 23:40:03 +00:00
ret = SMB_VFS_NTIMES ( conn , fname , ts ) ;
2005-03-24 22:34:28 +00:00
unbecome_root ( ) ;
2004-03-31 22:46:15 +00:00
}
return ret ;
1998-08-17 13:11:34 +00:00
}
2007-10-31 16:32:48 -07:00
2008-09-05 19:00:48 -07:00
/******************************************************************
Force a " sticky " write time on a pathname . This will always be
returned on all future write time queries and set on close .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2004-03-31 22:46:15 +00:00
2008-09-05 19:00:48 -07:00
bool set_sticky_write_time_path ( connection_struct * conn , const char * fname ,
struct file_id fileid , const struct timespec mtime )
1998-08-17 13:11:34 +00:00
{
2007-03-05 23:40:03 +00:00
if ( null_timespec ( mtime ) ) {
2008-03-12 15:39:38 +01:00
return true ;
2007-03-05 23:40:03 +00:00
}
1998-08-17 13:11:34 +00:00
2008-09-05 19:00:48 -07:00
if ( ! set_sticky_write_time ( fileid , mtime ) ) {
2008-03-12 15:39:38 +01:00
return false ;
2004-03-31 22:46:15 +00:00
}
2007-01-17 16:23:45 +00:00
2007-10-31 16:32:48 -07:00
return true ;
2007-03-05 23:40:03 +00:00
}
2008-03-12 15:39:38 +01:00
2008-09-05 19:00:48 -07:00
/******************************************************************
Force a " sticky " write time on an fsp . This will always be
returned on all future write time queries and set on close .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
bool set_sticky_write_time_fsp ( struct files_struct * fsp , const struct timespec mtime )
2008-03-12 15:39:38 +01:00
{
2008-09-05 19:00:48 -07:00
fsp - > write_time_forced = true ;
TALLOC_FREE ( fsp - > update_write_time_event ) ;
return set_sticky_write_time_path ( fsp - > conn , fsp - > fsp_name ,
fsp - > file_id , mtime ) ;
}
/******************************************************************
Update a write time immediately , without the 2 second delay .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
bool update_write_time ( struct files_struct * fsp )
{
if ( ! set_write_time ( fsp - > file_id , timespec_current ( ) ) ) {
return false ;
2008-03-12 15:39:38 +01:00
}
2008-09-05 19:00:48 -07:00
notify_fname ( fsp - > conn , NOTIFY_ACTION_MODIFIED ,
FILE_NOTIFY_CHANGE_LAST_WRITE , fsp - > fsp_name ) ;
return true ;
2008-03-12 15:39:38 +01:00
}