2010-01-05 10:42:38 +01:00
/*
* Module to work around a bug in Linux XFS :
* http : //oss.sgi.com/bugzilla/show_bug.cgi?id=280
*
* Copyright ( c ) Volker Lendecke 2010
*
* 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 3 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 , see < http : //www.gnu.org/licenses/>.
*/
# include "includes.h"
2011-02-25 23:20:06 +01:00
# include "system/filesys.h"
2011-03-22 22:34:22 +01:00
# include "smbd/smbd.h"
2010-01-05 10:42:38 +01:00
2019-09-05 10:15:18 -07:00
static int linux_xfs_sgid_mkdirat ( vfs_handle_struct * handle ,
struct files_struct * dirfsp ,
const struct smb_filename * smb_fname ,
mode_t mode )
{
struct smb_filename fname = { 0 , } ;
int mkdir_res ;
int res ;
DEBUG ( 10 , ( " Calling linux_xfs_sgid_mkdirat(%s) \n " ,
smb_fname - > base_name ) ) ;
mkdir_res = SMB_VFS_NEXT_MKDIRAT ( handle ,
dirfsp ,
smb_fname ,
mode ) ;
if ( mkdir_res = = - 1 ) {
DEBUG ( 10 , ( " SMB_VFS_NEXT_MKDIRAT returned error: %s \n " ,
strerror ( errno ) ) ) ;
return mkdir_res ;
}
if ( ! parent_dirname ( talloc_tos ( ) ,
smb_fname - > base_name ,
& fname . base_name ,
NULL ) ) {
DEBUG ( 1 , ( " parent_dirname failed \n " ) ) ;
/* return success, we did the mkdir */
return mkdir_res ;
}
res = SMB_VFS_NEXT_STAT ( handle , & fname ) ;
if ( res = = - 1 ) {
DEBUG ( 10 , ( " NEXT_STAT(%s) failed: %s \n " , fname . base_name ,
strerror ( errno ) ) ) ;
/* return success, we did the mkdir */
return mkdir_res ;
}
TALLOC_FREE ( fname . base_name ) ;
if ( ( fname . st . st_ex_mode & S_ISGID ) = = 0 ) {
/* No SGID to inherit */
DEBUG ( 10 , ( " No SGID to inherit \n " ) ) ;
return mkdir_res ;
}
fname . base_name = discard_const_p ( char , smb_fname - > base_name ) ;
res = SMB_VFS_NEXT_STAT ( handle , & fname ) ;
if ( res = = - 1 ) {
DEBUG ( 2 , ( " Could not stat just created dir %s: %s \n " ,
smb_fname - > base_name ,
strerror ( errno ) ) ) ;
/* return success, we did the mkdir */
return mkdir_res ;
}
fname . st . st_ex_mode | = S_ISGID ;
fname . st . st_ex_mode & = ~ S_IFDIR ;
/*
* Yes , we have to do this as root . If you do it as
* non - privileged user , XFS on Linux will just ignore us and
* return success . What can you do . . .
*/
become_root ( ) ;
res = SMB_VFS_NEXT_CHMOD ( handle ,
smb_fname ,
fname . st . st_ex_mode ) ;
unbecome_root ( ) ;
if ( res = = - 1 ) {
DEBUG ( 2 , ( " CHMOD(%s, %o) failed: %s \n " , smb_fname - > base_name ,
( int ) fname . st . st_ex_mode , strerror ( errno ) ) ) ;
/* return success, we did the mkdir */
return mkdir_res ;
}
return mkdir_res ;
}
2010-01-05 10:42:38 +01:00
static struct vfs_fn_pointers linux_xfs_sgid_fns = {
2019-09-05 10:15:18 -07:00
. mkdirat_fn = linux_xfs_sgid_mkdirat ,
2010-01-05 10:42:38 +01:00
} ;
2017-12-15 15:32:12 -07:00
static_decl_vfs ;
2017-04-20 12:24:43 -07:00
NTSTATUS vfs_linux_xfs_sgid_init ( TALLOC_CTX * ctx )
2010-01-05 10:42:38 +01:00
{
return smb_register_vfs ( SMB_VFS_INTERFACE_VERSION ,
" linux_xfs_sgid " , & linux_xfs_sgid_fns ) ;
}