2004-04-22 15:30:42 +00:00
/*
* implementation of an Shadow Copy module
*
* Copyright ( C ) Stefan Metzmacher 2003 - 2004
*
* 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
2004-04-22 15:30:42 +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 05:23:25 +00:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2004-04-22 15:30:42 +00:00
*/
# include "includes.h"
2011-03-22 22:34:22 +01:00
# include "smbd/smbd.h"
2011-03-25 13:42:42 +01:00
# include "ntioctl.h"
2004-04-22 15:30:42 +00:00
/*
Please read the VFS module Samba - HowTo - Collection .
there ' s a chapter about this module
For this share
Z : \
the ShadowCopies are in this directories
Z : \ @ GMT - 2003.08 .05 - 12.00 .00 \
Z : \ @ GMT - 2003.08 .05 - 12.01 .00 \
Z : \ @ GMT - 2003.08 .05 - 12.02 .00 \
e . g .
Z : \ testfile . txt
Z : \ @ GMT - 2003.08 .05 - 12.02 .00 \ testfile . txt
or :
Z : \ testdir \ testfile . txt
Z : \ @ GMT - 2003.08 .05 - 12.02 .00 \ testdir \ testfile . txt
Note : Files must differ to be displayed via Windows Explorer !
Directories are always displayed . . .
*/
static int vfs_shadow_copy_debug_level = DBGC_VFS ;
# undef DBGC_CLASS
# define DBGC_CLASS vfs_shadow_copy_debug_level
# define SHADOW_COPY_PREFIX "@GMT-"
# define SHADOW_COPY_SAMPLE "@GMT-2004.02.18-15.44.00"
typedef struct {
int pos ;
int num ;
2012-03-28 13:18:14 +11:00
struct dirent * dirs ;
2004-04-22 15:30:42 +00:00
} shadow_copy_Dir ;
2007-10-18 17:40:25 -07:00
static bool shadow_copy_match_name ( const char * name )
2004-04-22 15:30:42 +00:00
{
if ( strncmp ( SHADOW_COPY_PREFIX , name , sizeof ( SHADOW_COPY_PREFIX ) - 1 ) = = 0 & &
( strlen ( SHADOW_COPY_SAMPLE ) = = strlen ( name ) ) ) {
return True ;
}
return False ;
}
2015-04-30 20:16:18 -07:00
static DIR * shadow_copy_fdopendir ( vfs_handle_struct * handle , files_struct * fsp , const char * mask , uint32_t attr )
2011-02-08 15:07:48 -08:00
{
shadow_copy_Dir * dirp ;
2012-03-28 13:22:03 +11:00
DIR * p = SMB_VFS_NEXT_FDOPENDIR ( handle , fsp , mask , attr ) ;
2011-02-08 15:07:48 -08:00
if ( ! p ) {
DEBUG ( 10 , ( " shadow_copy_opendir: SMB_VFS_NEXT_FDOPENDIR() failed for [%s] \n " ,
smb_fname_str_dbg ( fsp - > fsp_name ) ) ) ;
return NULL ;
}
dirp = SMB_MALLOC_P ( shadow_copy_Dir ) ;
if ( ! dirp ) {
DEBUG ( 0 , ( " shadow_copy_fdopendir: Out of memory \n " ) ) ;
SMB_VFS_NEXT_CLOSEDIR ( handle , p ) ;
/* We have now closed the fd in fsp. */
2020-09-26 21:46:51 +02:00
fsp_set_fd ( fsp , - 1 ) ;
2011-02-08 15:07:48 -08:00
return NULL ;
}
ZERO_STRUCTP ( dirp ) ;
while ( True ) {
2012-03-28 13:18:14 +11:00
struct dirent * d ;
2011-02-08 15:07:48 -08:00
2023-06-20 12:42:52 +02:00
d = SMB_VFS_NEXT_READDIR ( handle , fsp , p ) ;
2011-02-08 15:07:48 -08:00
if ( d = = NULL ) {
break ;
}
if ( shadow_copy_match_name ( d - > d_name ) ) {
DEBUG ( 8 , ( " shadow_copy_fdopendir: hide [%s] \n " , d - > d_name ) ) ;
continue ;
}
DEBUG ( 10 , ( " shadow_copy_fdopendir: not hide [%s] \n " , d - > d_name ) ) ;
2012-03-28 13:18:14 +11:00
dirp - > dirs = SMB_REALLOC_ARRAY ( dirp - > dirs , struct dirent , dirp - > num + 1 ) ;
2011-02-08 15:07:48 -08:00
if ( ! dirp - > dirs ) {
DEBUG ( 0 , ( " shadow_copy_fdopendir: Out of memory \n " ) ) ;
break ;
}
dirp - > dirs [ dirp - > num + + ] = * d ;
}
SMB_VFS_NEXT_CLOSEDIR ( handle , p ) ;
/* We have now closed the fd in fsp. */
2020-09-26 21:46:51 +02:00
fsp_set_fd ( fsp , - 1 ) ;
2012-03-28 13:22:03 +11:00
return ( ( DIR * ) dirp ) ;
2011-02-08 15:07:48 -08:00
}
2012-03-28 13:18:14 +11:00
static struct dirent * shadow_copy_readdir ( vfs_handle_struct * handle ,
2020-11-22 13:57:27 +01:00
struct files_struct * dirfsp ,
2023-06-20 12:42:52 +02:00
DIR * _dirp )
2004-04-22 15:30:42 +00:00
{
shadow_copy_Dir * dirp = ( shadow_copy_Dir * ) _dirp ;
if ( dirp - > pos < dirp - > num ) {
return & ( dirp - > dirs [ dirp - > pos + + ] ) ;
}
return NULL ;
}
2012-03-28 13:22:03 +11:00
static void shadow_copy_rewinddir ( struct vfs_handle_struct * handle , DIR * _dirp )
2005-06-13 16:49:57 +00:00
{
shadow_copy_Dir * dirp = ( shadow_copy_Dir * ) _dirp ;
dirp - > pos = 0 ;
}
2012-03-28 13:22:03 +11:00
static int shadow_copy_closedir ( vfs_handle_struct * handle , DIR * _dirp )
2004-04-22 15:30:42 +00:00
{
shadow_copy_Dir * dirp = ( shadow_copy_Dir * ) _dirp ;
2005-01-14 20:23:22 +00:00
SAFE_FREE ( dirp - > dirs ) ;
2004-04-22 15:30:42 +00:00
SAFE_FREE ( dirp ) ;
return 0 ;
}
2011-05-30 12:06:31 +02:00
static int shadow_copy_get_shadow_copy_data ( vfs_handle_struct * handle ,
files_struct * fsp ,
struct shadow_copy_data * shadow_copy_data ,
bool labels )
2004-04-22 15:30:42 +00:00
{
2020-03-19 12:03:27 +01:00
struct smb_Dir * dir_hnd = NULL ;
const char * dname = NULL ;
char * talloced = NULL ;
2022-02-28 14:11:04 -08:00
NTSTATUS status ;
2016-02-26 14:53:12 -08:00
struct smb_filename * smb_fname = synthetic_smb_fname ( talloc_tos ( ) ,
fsp - > conn - > connectpath ,
NULL ,
2016-03-18 21:19:38 -07:00
NULL ,
2020-04-30 11:48:32 +02:00
0 ,
2016-03-18 21:19:38 -07:00
0 ) ;
2016-02-26 14:53:12 -08:00
if ( smb_fname = = NULL ) {
errno = ENOMEM ;
return - 1 ;
}
2022-02-28 14:34:48 -08:00
status = OpenDir ( talloc_tos ( ) ,
handle - > conn ,
smb_fname ,
NULL ,
0 ,
& dir_hnd ) ;
2016-02-26 14:53:12 -08:00
TALLOC_FREE ( smb_fname ) ;
2022-02-28 14:11:04 -08:00
if ( ! NT_STATUS_IS_OK ( status ) ) {
2020-06-20 17:13:49 +05:30
DBG_ERR ( " OpenDir() failed for [%s] \n " , fsp - > conn - > connectpath ) ;
2022-02-28 14:11:04 -08:00
errno = map_errno_from_nt_status ( status ) ;
2020-03-19 12:03:27 +01:00
return - 1 ;
}
2004-04-22 15:30:42 +00:00
shadow_copy_data - > num_volumes = 0 ;
shadow_copy_data - > labels = NULL ;
while ( True ) {
SHADOW_COPY_LABEL * tlabels ;
2017-02-09 13:02:52 +13:00
int ret ;
2004-04-22 15:30:42 +00:00
2023-06-20 12:25:45 +02:00
dname = ReadDirName ( dir_hnd , & talloced ) ;
2020-03-19 12:03:27 +01:00
if ( dname = = NULL ) {
2004-04-22 15:30:42 +00:00
break ;
}
/* */
2020-03-19 12:03:27 +01:00
if ( ! shadow_copy_match_name ( dname ) ) {
DBG_DEBUG ( " ignore [%s] \n " , dname ) ;
TALLOC_FREE ( talloced ) ;
2004-04-22 15:30:42 +00:00
continue ;
}
2020-03-19 12:03:27 +01:00
DBG_DEBUG ( " not ignore [%s] \n " , dname ) ;
2004-04-22 15:30:42 +00:00
if ( ! labels ) {
shadow_copy_data - > num_volumes + + ;
2020-03-19 12:03:27 +01:00
TALLOC_FREE ( talloced ) ;
2004-04-22 15:30:42 +00:00
continue ;
}
2011-05-30 12:11:53 +02:00
tlabels = ( SHADOW_COPY_LABEL * ) TALLOC_REALLOC ( shadow_copy_data ,
2004-04-22 15:30:42 +00:00
shadow_copy_data - > labels ,
( shadow_copy_data - > num_volumes + 1 ) * sizeof ( SHADOW_COPY_LABEL ) ) ;
if ( tlabels = = NULL ) {
DEBUG ( 0 , ( " shadow_copy_get_shadow_copy_data: Out of memory \n " ) ) ;
2020-03-19 12:03:27 +01:00
TALLOC_FREE ( talloced ) ;
TALLOC_FREE ( dir_hnd ) ;
2004-04-22 15:30:42 +00:00
return - 1 ;
}
2020-03-19 12:03:27 +01:00
ret = strlcpy ( tlabels [ shadow_copy_data - > num_volumes ] , dname ,
2017-02-09 13:02:52 +13:00
sizeof ( tlabels [ shadow_copy_data - > num_volumes ] ) ) ;
if ( ret ! = sizeof ( tlabels [ shadow_copy_data - > num_volumes ] ) - 1 ) {
2020-03-19 12:03:27 +01:00
DBG_ERR ( " malformed label %s \n " , dname ) ;
TALLOC_FREE ( talloced ) ;
TALLOC_FREE ( dir_hnd ) ;
2017-02-09 13:02:52 +13:00
return - 1 ;
}
shadow_copy_data - > num_volumes + + ;
2004-04-22 15:30:42 +00:00
shadow_copy_data - > labels = tlabels ;
2020-03-19 12:03:27 +01:00
TALLOC_FREE ( talloced ) ;
2004-04-22 15:30:42 +00:00
}
2020-03-19 12:03:27 +01:00
TALLOC_FREE ( dir_hnd ) ;
2004-04-22 15:30:42 +00:00
return 0 ;
}
2009-07-23 20:28:58 -04:00
static struct vfs_fn_pointers vfs_shadow_copy_fns = {
2011-12-03 20:45:04 -08:00
. fdopendir_fn = shadow_copy_fdopendir ,
. readdir_fn = shadow_copy_readdir ,
. rewind_dir_fn = shadow_copy_rewinddir ,
. closedir_fn = shadow_copy_closedir ,
. get_shadow_copy_data_fn = shadow_copy_get_shadow_copy_data ,
2004-04-22 15:30:42 +00:00
} ;
2017-12-15 15:32:12 -07:00
static_decl_vfs ;
2017-04-20 12:24:43 -07:00
NTSTATUS vfs_shadow_copy_init ( TALLOC_CTX * ctx )
2004-04-22 15:30:42 +00:00
{
2009-07-23 20:28:58 -04:00
NTSTATUS ret = smb_register_vfs ( SMB_VFS_INTERFACE_VERSION ,
" shadow_copy " , & vfs_shadow_copy_fns ) ;
2004-04-22 15:30:42 +00:00
if ( ! NT_STATUS_IS_OK ( ret ) )
return ret ;
vfs_shadow_copy_debug_level = debug_add_class ( " shadow_copy " ) ;
if ( vfs_shadow_copy_debug_level = = - 1 ) {
vfs_shadow_copy_debug_level = DBGC_VFS ;
DEBUG ( 0 , ( " %s: Couldn't register custom debugging class! \n " ,
" vfs_shadow_copy_init " ) ) ;
} else {
DEBUG ( 10 , ( " %s: Debug class number of '%s': %d \n " ,
" vfs_shadow_copy_init " , " shadow_copy " , vfs_shadow_copy_debug_level ) ) ;
}
return ret ;
}