2016-01-14 11:51:39 +01:00
/*
2002-07-15 10:35:28 +00:00
Unix SMB / Netbios implementation .
Version 1.9 .
1998-08-17 07:15:54 +00:00
read / write to a files_struct
Copyright ( C ) Andrew Tridgell 1992 - 1998
2002-07-15 10:35:28 +00:00
Copyright ( C ) Jeremy Allison 2000 - 2002. - write cache .
2016-01-14 11:51:39 +01:00
1998-08-17 07:15:54 +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:15:54 +00:00
( at your option ) any later version .
2016-01-14 11:51:39 +01:00
1998-08-17 07:15:54 +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 .
2016-01-14 11:51:39 +01:00
1998-08-17 07:15:54 +00:00
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:15:54 +00:00
*/
# include "includes.h"
2010-05-04 04:28:48 -04:00
# include "printing.h"
2011-03-22 16:57:01 +01:00
# include "smbd/smbd.h"
2009-01-08 12:03:45 +01:00
# include "smbd/globals.h"
2011-04-14 00:36:23 +02:00
# include "smbprofile.h"
1998-08-17 07:15:54 +00:00
/****************************************************************************
2002-07-15 10:35:28 +00:00
Read from a file .
1998-08-17 07:15:54 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-09-11 01:24:30 +00:00
2012-04-05 14:53:08 +10:00
ssize_t read_file ( files_struct * fsp , char * data , off_t pos , size_t n )
1998-08-17 07:15:54 +00:00
{
2020-09-28 10:32:29 +02:00
off_t new_pos ;
2012-06-08 16:06:24 +02:00
ssize_t ret = 0 ;
2020-05-11 18:18:24 +02:00
bool ok ;
1998-08-17 07:15:54 +00:00
2001-11-25 08:26:37 +00:00
/* you can't read from print files */
2005-05-08 23:16:28 +00:00
if ( fsp - > print_file ) {
2009-07-10 18:23:00 -07:00
errno = EBADF ;
2001-11-25 08:26:37 +00:00
return - 1 ;
2005-05-08 23:16:28 +00:00
}
2000-04-16 06:22:31 +00:00
2020-05-11 18:18:24 +02:00
ok = vfs_valid_pread_range ( pos , n ) ;
if ( ! ok ) {
errno = EINVAL ;
return - 1 ;
}
2020-09-28 10:32:29 +02:00
fh_set_pos ( fsp - > fh , pos ) ;
2004-01-06 01:22:14 +00:00
2001-11-25 08:26:37 +00:00
if ( n > 0 ) {
2012-06-08 16:06:24 +02:00
ret = SMB_VFS_PREAD ( fsp , data , n , pos ) ;
2004-01-06 01:22:14 +00:00
2012-06-08 16:06:24 +02:00
if ( ret = = - 1 ) {
2001-11-25 08:26:37 +00:00
return - 1 ;
}
}
2002-07-15 10:35:28 +00:00
DEBUG ( 10 , ( " read_file (%s): pos = %.0f, size = %lu, returned %lu \n " ,
2009-07-10 14:50:37 -07:00
fsp_str_dbg ( fsp ) , ( double ) pos , ( unsigned long ) n , ( long ) ret ) ) ;
2002-07-15 10:35:28 +00:00
2020-09-28 10:32:29 +02:00
new_pos = fh_get_pos ( fsp - > fh ) + ret ;
fh_set_pos ( fsp - > fh , new_pos ) ;
fh_set_position_information ( fsp - > fh , new_pos ) ;
2003-08-14 21:16:06 +00:00
2001-11-25 08:26:37 +00:00
return ( ret ) ;
1998-08-17 07:15:54 +00:00
}
1999-12-13 13:27:58 +00:00
/****************************************************************************
2000-08-25 20:44:33 +00:00
* Really * write to a file .
1999-12-13 13:27:58 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-30 16:22:24 -07:00
static ssize_t real_write_file ( struct smb_request * req ,
files_struct * fsp ,
const char * data ,
2012-04-05 14:53:08 +10:00
off_t pos ,
2007-10-30 16:22:24 -07:00
size_t n )
1999-12-13 13:27:58 +00:00
{
2002-07-15 10:35:28 +00:00
ssize_t ret ;
2020-05-11 18:18:24 +02:00
bool ok ;
ok = vfs_valid_pwrite_range ( pos , n ) ;
if ( ! ok ) {
errno = EINVAL ;
return - 1 ;
}
2002-07-15 10:35:28 +00:00
2020-05-11 18:18:24 +02:00
if ( n = = 0 ) {
return 0 ;
}
2020-09-28 10:32:29 +02:00
fh_set_pos ( fsp - > fh , pos ) ;
2020-04-02 10:29:15 +02:00
if ( pos & &
lp_strict_allocate ( SNUM ( fsp - > conn ) ) & &
2020-04-03 07:52:10 +02:00
! fsp - > fsp_flags . is_sparse )
2020-04-02 10:29:15 +02:00
{
2018-05-02 16:39:16 -07:00
if ( vfs_fill_sparse ( fsp , pos ) = = - 1 ) {
return - 1 ;
2005-05-17 01:04:51 +00:00
}
2004-01-06 01:22:14 +00:00
}
2018-05-02 16:39:16 -07:00
ret = vfs_pwrite_data ( req , fsp , data , n , pos ) ;
2002-07-15 10:35:28 +00:00
DEBUG ( 10 , ( " real_write_file (%s): pos = %.0f, size = %lu, returned %ld \n " ,
2009-07-10 14:50:37 -07:00
fsp_str_dbg ( fsp ) , ( double ) pos , ( unsigned long ) n , ( long ) ret ) ) ;
1999-12-13 13:27:58 +00:00
2003-08-14 21:16:06 +00:00
if ( ret ! = - 1 ) {
2020-09-28 10:32:29 +02:00
off_t new_pos = fh_get_pos ( fsp - > fh ) + ret ;
fh_set_pos ( fsp - > fh , new_pos ) ;
2003-08-14 21:16:06 +00:00
/* Yes - this is correct - writes don't update this. JRA. */
/* Found by Samba4 tests. */
#if 0
fsp - > position_information = fsp - > pos ;
# endif
}
2002-07-15 10:35:28 +00:00
return ret ;
1999-12-13 13:27:58 +00:00
}
1998-08-17 07:15:54 +00:00
2018-05-23 09:41:29 +02:00
void fsp_flush_write_time_update ( struct files_struct * fsp )
2008-03-12 15:39:38 +01:00
{
2018-05-23 09:41:29 +02:00
/*
* Note this won ' t expect any impersonation !
* So don ' t call any SMB_VFS operations here !
*/
2008-03-12 15:39:38 +01:00
2009-07-10 14:50:37 -07:00
DEBUG ( 5 , ( " Update write time on %s \n " , fsp_str_dbg ( fsp ) ) ) ;
2008-03-12 15:39:38 +01:00
2020-03-15 08:30:21 +01:00
trigger_write_time_update_immediate ( fsp ) ;
2008-03-12 15:39:38 +01:00
}
2018-05-23 09:41:29 +02:00
static void update_write_time_handler ( struct tevent_context * ctx ,
struct tevent_timer * te ,
struct timeval now ,
void * private_data )
{
files_struct * fsp = ( files_struct * ) private_data ;
fsp_flush_write_time_update ( fsp ) ;
}
2008-09-05 19:00:48 -07:00
/*********************************************************
Schedule a write time update for WRITE_TIME_UPDATE_USEC_DELAY
in the future .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-03-12 15:39:38 +01:00
void trigger_write_time_update ( struct files_struct * fsp )
{
2008-04-07 09:27:22 +02:00
int delay ;
2015-11-27 18:29:55 +01:00
if ( fsp - > posix_flags & FSP_POSIX_FLAGS_OPEN ) {
2009-07-01 17:39:10 -07:00
/* Don't use delayed writes on POSIX files. */
return ;
}
2020-04-02 15:39:32 +02:00
if ( fsp - > fsp_flags . write_time_forced ) {
2008-09-05 19:00:48 -07:00
/* No point - "sticky" write times
* in effect .
*/
2008-03-12 15:39:38 +01:00
return ;
}
2009-11-05 16:20:11 -08:00
/* We need to remember someone did a write
* and update to current time on close . */
2020-04-02 15:37:15 +02:00
fsp - > fsp_flags . update_write_time_on_close = true ;
2009-11-05 16:20:11 -08:00
2020-04-02 15:31:10 +02:00
if ( fsp - > fsp_flags . update_write_time_triggered ) {
2008-09-05 19:00:48 -07:00
/*
2009-11-05 16:20:11 -08:00
* We only update the write time after 2 seconds
* on the first normal write . After that
* no other writes affect this until close .
2008-09-05 19:00:48 -07:00
*/
2008-03-12 15:39:38 +01:00
return ;
}
2020-04-02 15:31:10 +02:00
fsp - > fsp_flags . update_write_time_triggered = true ;
2008-03-12 15:39:38 +01:00
2008-04-07 09:27:22 +02:00
delay = lp_parm_int ( SNUM ( fsp - > conn ) ,
" smbd " , " writetimeupdatedelay " ,
WRITE_TIME_UPDATE_USEC_DELAY ) ;
2009-11-05 16:20:11 -08:00
DEBUG ( 5 , ( " Update write time %d usec later on %s \n " ,
delay , fsp_str_dbg ( fsp ) ) ) ;
2018-12-27 15:26:03 +01:00
/* trigger the update 2 seconds later */
2008-03-12 15:39:38 +01:00
fsp - > update_write_time_event =
2018-12-27 15:26:03 +01:00
tevent_add_timer ( fsp - > conn - > sconn - > ev_ctx , NULL ,
2011-12-12 13:39:56 +01:00
timeval_current_ofs_usec ( delay ) ,
update_write_time_handler , fsp ) ;
2008-03-12 15:39:38 +01:00
}
2008-09-05 19:00:48 -07:00
void trigger_write_time_update_immediate ( struct files_struct * fsp )
{
2009-11-05 16:20:11 -08:00
struct smb_file_time ft ;
2019-12-02 16:30:50 +01:00
init_smb_file_time ( & ft ) ;
2015-11-27 18:29:55 +01:00
if ( fsp - > posix_flags & FSP_POSIX_FLAGS_OPEN ) {
2009-07-01 17:39:10 -07:00
/* Don't use delayed writes on POSIX files. */
return ;
}
2020-04-02 15:39:32 +02:00
if ( fsp - > fsp_flags . write_time_forced ) {
2008-09-05 19:00:48 -07:00
/*
* No point - " sticky " write times
* in effect .
*/
return ;
}
2008-09-08 15:12:24 +02:00
TALLOC_FREE ( fsp - > update_write_time_event ) ;
2009-07-10 14:50:37 -07:00
DEBUG ( 5 , ( " Update write time immediate on %s \n " ,
fsp_str_dbg ( fsp ) ) ) ;
2008-09-08 15:12:24 +02:00
2009-11-05 16:20:11 -08:00
/* After an immediate update, reset the trigger. */
2020-04-02 15:31:10 +02:00
fsp - > fsp_flags . update_write_time_triggered = true ;
2020-04-02 15:37:15 +02:00
fsp - > fsp_flags . update_write_time_on_close = false ;
2009-11-05 16:20:11 -08:00
2019-12-02 16:30:50 +01:00
ft . mtime = timespec_current ( ) ;
2009-11-05 16:20:11 -08:00
/* Update the time in the open file db. */
( void ) set_write_time ( fsp - > file_id , ft . mtime ) ;
/* Now set on disk - takes care of notify. */
( void ) smb_set_file_time ( fsp - > conn , fsp , fsp - > fsp_name , & ft , false ) ;
2008-09-05 19:00:48 -07:00
}
2012-07-12 16:28:11 +02:00
void mark_file_modified ( files_struct * fsp )
{
int dosmode ;
2020-03-15 08:30:21 +01:00
trigger_write_time_update ( fsp ) ;
2020-04-02 17:37:02 +02:00
if ( fsp - > fsp_flags . modified ) {
2012-07-12 16:28:11 +02:00
return ;
}
2020-04-02 17:37:02 +02:00
fsp - > fsp_flags . modified = true ;
2012-07-12 16:28:11 +02:00
2015-11-27 18:29:55 +01:00
if ( fsp - > posix_flags & FSP_POSIX_FLAGS_OPEN ) {
2012-07-12 16:28:11 +02:00
return ;
}
if ( ! ( lp_store_dos_attributes ( SNUM ( fsp - > conn ) ) | |
MAP_ARCHIVE ( fsp - > conn ) ) ) {
return ;
}
2020-10-19 20:32:49 +02:00
dosmode = fdos_mode ( fsp ) ;
2012-07-12 16:28:11 +02:00
if ( IS_DOS_ARCHIVE ( dosmode ) ) {
return ;
}
file_set_dosmode ( fsp - > conn , fsp - > fsp_name ,
dosmode | FILE_ATTRIBUTE_ARCHIVE , NULL , false ) ;
}
2005-05-08 23:16:28 +00:00
/****************************************************************************
Write to a file .
1998-08-17 07:15:54 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-09-11 01:24:30 +00:00
2007-10-30 16:22:24 -07:00
ssize_t write_file ( struct smb_request * req ,
files_struct * fsp ,
const char * data ,
2012-04-05 14:53:08 +10:00
off_t pos ,
2007-10-30 16:22:24 -07:00
size_t n )
1998-08-17 07:15:54 +00:00
{
2002-07-15 10:35:28 +00:00
ssize_t total_written = 0 ;
2002-12-05 04:00:16 +00:00
if ( fsp - > print_file ) {
2010-05-04 04:28:48 -04:00
uint32_t t ;
int ret ;
2002-12-05 04:00:16 +00:00
2010-05-04 04:28:48 -04:00
ret = print_spool_write ( fsp , data , n , pos , & t ) ;
if ( ret ) {
errno = ret ;
2002-12-05 04:00:16 +00:00
return - 1 ;
}
2010-05-04 04:28:48 -04:00
return t ;
2002-12-05 04:00:16 +00:00
}
2002-07-15 10:35:28 +00:00
2020-04-02 17:28:32 +02:00
if ( ! fsp - > fsp_flags . can_write ) {
2002-07-15 10:35:28 +00:00
errno = EPERM ;
2006-07-25 00:16:45 +00:00
return - 1 ;
2002-07-15 10:35:28 +00:00
}
2012-07-12 16:28:11 +02:00
mark_file_modified ( fsp ) ;
2002-07-15 10:35:28 +00:00
/*
* If this file is level II oplocked then we need
* to grab the shared memory lock and inform all
* other files with a level II lock that they need
* to flush their read caches . We keep the lock over
* the shared memory area whilst doing this .
*/
1999-12-13 13:27:58 +00:00
2009-02-03 11:56:35 -08:00
/* This should actually be improved to span the write. */
contend_level2_oplocks_begin ( fsp , LEVEL2_CONTEND_WRITE ) ;
contend_level2_oplocks_end ( fsp , LEVEL2_CONTEND_WRITE ) ;
1999-12-13 13:27:58 +00:00
2019-05-27 11:24:14 +02:00
total_written = real_write_file ( req , fsp , data , pos , n ) ;
2002-07-15 10:35:28 +00:00
return total_written ;
1998-08-17 07:15:54 +00:00
}
1998-08-17 07:40:06 +00:00
/*******************************************************************
sync a file
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
1998-09-11 01:24:30 +00:00
2007-10-18 17:40:25 -07:00
NTSTATUS sync_file ( connection_struct * conn , files_struct * fsp , bool write_through )
1998-08-17 07:40:06 +00:00
{
2020-09-26 21:52:52 +02:00
if ( fsp_get_io_fd ( fsp ) = = - 1 )
2007-06-15 19:24:04 +00:00
return NT_STATUS_INVALID_HANDLE ;
2005-09-03 07:19:28 +00:00
if ( lp_strict_sync ( SNUM ( conn ) ) & &
2014-02-04 15:09:12 +13:00
( lp_sync_always ( SNUM ( conn ) ) | | write_through ) ) {
2019-05-27 11:24:14 +02:00
int ret ;
2018-04-27 16:59:42 -07:00
ret = smb_vfs_fsync_sync ( fsp ) ;
2007-06-15 19:24:04 +00:00
if ( ret = = - 1 ) {
return map_nt_error_from_unix ( errno ) ;
}
2002-07-15 10:35:28 +00:00
}
2007-06-15 19:24:04 +00:00
return NT_STATUS_OK ;
1998-08-17 07:40:06 +00:00
}