1998-07-29 07:08:05 +04:00
/*
2002-01-30 09:08:46 +03:00
Unix SMB / CIFS implementation .
1998-07-29 07:08:05 +04:00
functions to calculate the free disk space
Copyright ( C ) Andrew Tridgell 1998
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 23:25:36 +04:00
the Free Software Foundation ; either version 3 of the License , or
1998-07-29 07:08:05 +04: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 04:52:41 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
1998-07-29 07:08:05 +04:00
*/
# include "includes.h"
2011-03-22 18:57:01 +03:00
# include "smbd/smbd.h"
2009-01-08 14:03:45 +03:00
# include "smbd/globals.h"
2016-02-16 19:09:43 +03:00
# include "lib/util_file.h"
2018-05-16 23:17:52 +03:00
# include "lib/util/memcache.h"
1998-07-29 07:08:05 +04:00
/****************************************************************************
2005-10-20 00:02:12 +04:00
Normalise for DOS usage .
1998-07-29 07:08:05 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-10-20 00:02:12 +04:00
2016-01-10 20:54:06 +03:00
static void disk_norm ( uint64_t * bsize , uint64_t * dfree , uint64_t * dsize )
1998-07-29 07:08:05 +04:00
{
/* check if the disk is beyond the max disk size */
2014-02-04 06:09:12 +04:00
uint64_t maxdisksize = lp_max_disk_size ( ) ;
1998-07-29 07:08:05 +04:00
if ( maxdisksize ) {
/* convert to blocks - and don't overflow */
maxdisksize = ( ( maxdisksize * 1024 ) / ( * bsize ) ) * 1024 ;
2015-02-16 21:26:24 +03:00
if ( * dsize > maxdisksize ) {
* dsize = maxdisksize ;
}
if ( * dfree > maxdisksize ) {
* dfree = maxdisksize - 1 ;
}
1998-07-29 07:08:05 +04:00
/* the -1 should stop applications getting div by 0
errors */
}
}
/****************************************************************************
2005-10-20 00:02:12 +04:00
Return number of 1 K blocks available on a path and total number .
1998-07-29 07:08:05 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1999-12-13 16:27:58 +03:00
2016-01-14 01:09:36 +03:00
uint64_t sys_disk_free ( connection_struct * conn , struct smb_filename * fname ,
2015-02-16 21:26:24 +03:00
uint64_t * bsize , uint64_t * dfree , uint64_t * dsize )
1998-07-29 07:08:05 +04:00
{
2008-10-14 03:59:36 +04:00
uint64_t dfree_retval ;
uint64_t dfree_q = 0 ;
uint64_t bsize_q = 0 ;
uint64_t dsize_q = 0 ;
2005-10-20 00:02:12 +04:00
const char * dfree_command ;
2015-07-17 10:37:52 +03:00
static bool dfree_broken = false ;
2016-01-14 01:09:36 +03:00
const char * path = fname - > base_name ;
1998-07-29 07:08:05 +04:00
( * dfree ) = ( * dsize ) = 0 ;
( * bsize ) = 512 ;
2000-01-06 04:41:27 +03:00
/*
* If external disk calculation specified , use it .
*/
2012-07-18 09:37:23 +04:00
dfree_command = lp_dfree_command ( talloc_tos ( ) , SNUM ( conn ) ) ;
2000-01-06 04:41:27 +03:00
if ( dfree_command & & * dfree_command ) {
2003-05-12 05:20:17 +04:00
const char * p ;
2007-09-13 03:50:21 +04:00
char * * lines = NULL ;
char * syscmd = NULL ;
syscmd = talloc_asprintf ( talloc_tos ( ) ,
" %s %s " ,
dfree_command ,
path ) ;
if ( ! syscmd ) {
2008-10-14 03:59:36 +04:00
return ( uint64_t ) - 1 ;
2007-09-13 03:50:21 +04:00
}
2000-01-06 04:41:27 +03:00
2012-09-04 17:01:18 +04:00
DEBUG ( 3 , ( " disk_free: Running command '%s' \n " , syscmd ) ) ;
2000-02-15 22:36:47 +03:00
2016-02-16 18:29:01 +03:00
lines = file_lines_pload ( talloc_tos ( ) , syscmd , NULL ) ;
2015-07-17 10:35:11 +03:00
if ( lines ! = NULL ) {
2000-04-16 15:00:21 +04:00
char * line = lines [ 0 ] ;
2000-01-06 04:41:27 +03:00
DEBUG ( 3 , ( " Read input from dfree, \" %s \" \n " , line ) ) ;
2003-05-12 05:20:17 +04:00
* dsize = STR_TO_SMB_BIG_UINT ( line , & p ) ;
while ( p & & * p & & isspace ( * p ) )
2000-01-06 04:41:27 +03:00
p + + ;
if ( p & & * p )
2003-05-12 05:20:17 +04:00
* dfree = STR_TO_SMB_BIG_UINT ( p , & p ) ;
while ( p & & * p & & isspace ( * p ) )
2000-01-06 04:41:27 +03:00
p + + ;
if ( p & & * p )
2003-05-12 05:20:17 +04:00
* bsize = STR_TO_SMB_BIG_UINT ( p , NULL ) ;
2000-01-06 04:41:27 +03:00
else
* bsize = 1024 ;
2008-10-12 19:34:43 +04:00
TALLOC_FREE ( lines ) ;
2000-01-06 04:41:27 +03:00
DEBUG ( 3 , ( " Parsed output of dfree, dsize=%u, dfree=%u, bsize=%u \n " ,
( unsigned int ) * dsize , ( unsigned int ) * dfree , ( unsigned int ) * bsize ) ) ;
if ( ! * dsize )
* dsize = 2048 ;
if ( ! * dfree )
* dfree = 1024 ;
2015-07-17 10:35:11 +03:00
goto dfree_done ;
2000-01-06 04:41:27 +03:00
}
2015-07-17 10:35:11 +03:00
DEBUG ( 0 , ( " disk_free: file_lines_load() failed for "
" command '%s'. Error was : %s \n " ,
syscmd , strerror ( errno ) ) ) ;
}
2017-05-23 20:40:47 +03:00
if ( SMB_VFS_DISK_FREE ( conn , fname , bsize , dfree , dsize ) = =
2016-01-10 16:15:41 +03:00
( uint64_t ) - 1 ) {
DBG_ERR ( " VFS disk_free failed. Error was : %s \n " ,
strerror ( errno ) ) ;
2015-07-17 10:35:11 +03:00
return ( uint64_t ) - 1 ;
2005-03-16 04:41:21 +03:00
}
1998-07-29 07:08:05 +04:00
2016-01-14 01:09:36 +03:00
if ( disk_quotas ( conn , fname , & bsize_q , & dfree_q , & dsize_q ) ) {
2016-01-19 15:57:16 +03:00
uint64_t min_bsize = MIN ( * bsize , bsize_q ) ;
( * dfree ) = ( * dfree ) * ( * bsize ) / min_bsize ;
( * dsize ) = ( * dsize ) * ( * bsize ) / min_bsize ;
dfree_q = dfree_q * bsize_q / min_bsize ;
dsize_q = dsize_q * bsize_q / min_bsize ;
( * bsize ) = min_bsize ;
1998-11-06 21:40:51 +03:00
( * dfree ) = MIN ( * dfree , dfree_q ) ;
( * dsize ) = MIN ( * dsize , dsize_q ) ;
}
/* FIXME : Any reason for this assumption ? */
1998-07-29 07:08:05 +04:00
if ( * bsize < 256 ) {
1998-11-09 06:45:49 +03:00
DEBUG ( 5 , ( " disk_free:Warning: bsize == %d < 256 . Changing to assumed correct bsize = 512 \n " , ( int ) * bsize ) ) ;
1998-07-29 07:08:05 +04:00
* bsize = 512 ;
}
if ( ( * dsize ) < 1 ) {
2009-01-08 14:03:45 +03:00
if ( ! dfree_broken ) {
1998-07-29 07:08:05 +04:00
DEBUG ( 0 , ( " WARNING: dfree is broken on this system \n " ) ) ;
2009-01-08 14:03:45 +03:00
dfree_broken = true ;
1998-07-29 07:08:05 +04:00
}
* dsize = 20 * 1024 * 1024 / ( * bsize ) ;
* dfree = MAX ( 1 , * dfree ) ;
}
2015-07-17 10:35:11 +03:00
dfree_done :
2015-02-16 21:26:24 +03:00
disk_norm ( bsize , dfree , dsize ) ;
1998-07-29 07:08:05 +04:00
if ( ( * bsize ) < 1024 ) {
dfree_retval = ( * dfree ) / ( 1024 / ( * bsize ) ) ;
} else {
dfree_retval = ( ( * bsize ) / 1024 ) * ( * dfree ) ;
}
return ( dfree_retval ) ;
}
/****************************************************************************
2005-10-20 00:02:12 +04:00
Potentially returned cached dfree info .
2018-05-16 23:17:52 +03:00
Depending on the file system layout and file system features , the free space
information can be different for different sub directories underneath a SMB
share . Store the cache information in memcache using the query path as the
key to accomodate this .
1998-07-29 07:08:05 +04:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-10-20 00:02:12 +04:00
2018-05-16 23:25:54 +03:00
struct dfree_cached_info {
time_t last_dfree_time ;
uint64_t dfree_ret ;
uint64_t bsize ;
uint64_t dfree ;
uint64_t dsize ;
} ;
2016-01-14 01:09:36 +03:00
uint64_t get_dfree_info ( connection_struct * conn , struct smb_filename * fname ,
uint64_t * bsize , uint64_t * dfree , uint64_t * dsize )
1998-07-29 07:08:05 +04:00
{
2005-10-20 00:02:12 +04:00
int dfree_cache_time = lp_dfree_cache_time ( SNUM ( conn ) ) ;
2018-05-16 23:17:52 +03:00
struct dfree_cached_info * dfc = NULL ;
struct dfree_cached_info dfc_new = { 0 } ;
2008-10-14 03:59:36 +04:00
uint64_t dfree_ret ;
2018-05-16 23:17:52 +03:00
char tmpbuf [ PATH_MAX ] ;
char * full_path = NULL ;
char * to_free = NULL ;
char * key_path = NULL ;
size_t len ;
DATA_BLOB key , value ;
bool found ;
2005-10-20 00:02:12 +04:00
if ( ! dfree_cache_time ) {
2016-01-14 01:09:36 +03:00
return sys_disk_free ( conn , fname , bsize , dfree , dsize ) ;
2005-10-20 00:02:12 +04:00
}
2018-05-16 23:17:52 +03:00
len = full_path_tos ( conn - > connectpath ,
fname - > base_name ,
tmpbuf ,
sizeof ( tmpbuf ) ,
& full_path ,
& to_free ) ;
if ( len = = - 1 ) {
errno = ENOMEM ;
return - 1 ;
}
if ( VALID_STAT ( fname - > st ) & & S_ISREG ( fname - > st . st_ex_mode ) ) {
/*
* In case of a file use the parent directory to reduce number
* of cache entries .
*/
bool ok ;
ok = parent_dirname ( talloc_tos ( ) ,
full_path ,
& key_path ,
NULL ) ;
TALLOC_FREE ( to_free ) ; /* We're done with full_path */
if ( ! ok ) {
errno = ENOMEM ;
return - 1 ;
}
/*
* key_path is always a talloced object .
*/
to_free = key_path ;
} else {
/*
* key_path might not be a talloced object ; rely on
* to_free set from full_path_tos .
*/
key_path = full_path ;
}
key = data_blob_const ( key_path , strlen ( key_path ) ) ;
found = memcache_lookup ( smbd_memcache ( ) ,
DFREE_CACHE ,
key ,
& value ) ;
dfc = found ? ( struct dfree_cached_info * ) value . data : NULL ;
2005-10-20 00:02:12 +04:00
if ( dfc & & ( conn - > lastused - dfc - > last_dfree_time < dfree_cache_time ) ) {
2018-05-16 23:17:52 +03:00
DBG_DEBUG ( " Returning dfree cache entry for %s \n " , key_path ) ;
2005-10-20 00:02:12 +04:00
* bsize = dfc - > bsize ;
* dfree = dfc - > dfree ;
* dsize = dfc - > dsize ;
2018-05-16 23:17:52 +03:00
dfree_ret = dfc - > dfree_ret ;
goto out ;
2005-10-20 00:02:12 +04:00
}
2016-01-14 01:09:36 +03:00
dfree_ret = sys_disk_free ( conn , fname , bsize , dfree , dsize ) ;
2005-10-20 00:02:12 +04:00
2008-10-14 03:59:36 +04:00
if ( dfree_ret = = ( uint64_t ) - 1 ) {
2005-10-20 00:02:12 +04:00
/* Don't cache bad data. */
2018-05-16 23:17:52 +03:00
goto out ;
2005-10-20 00:02:12 +04:00
}
2018-05-16 23:17:52 +03:00
DBG_DEBUG ( " Creating dfree cache entry for %s \n " , key_path ) ;
dfc_new . bsize = * bsize ;
dfc_new . dfree = * dfree ;
dfc_new . dsize = * dsize ;
dfc_new . dfree_ret = dfree_ret ;
dfc_new . last_dfree_time = conn - > lastused ;
memcache_add ( smbd_memcache ( ) ,
DFREE_CACHE ,
key ,
data_blob_const ( & dfc_new , sizeof ( dfc_new ) ) ) ;
out :
TALLOC_FREE ( to_free ) ;
2005-10-20 00:02:12 +04:00
return dfree_ret ;
1998-07-29 07:08:05 +04:00
}
2018-05-19 06:51:58 +03:00
void flush_dfree_cache ( void )
{
memcache_flush ( smbd_memcache ( ) , DFREE_CACHE ) ;
}