1996-05-31 15:13:29 +00:00
/*
2002-01-30 06:08:46 +00:00
Unix SMB / CIFS implementation .
1996-05-31 15:13:29 +00:00
time handling functions
2006-09-06 18:32:20 +00:00
Copyright ( C ) Andrew Tridgell 1992 - 2004
2002-09-26 18:58:34 +00:00
Copyright ( C ) Stefan ( metze ) Metzmacher 2002
2007-03-05 23:40:03 +00:00
Copyright ( C ) Jeremy Allison 2007
2006-09-06 18:32:20 +00:00
1996-05-31 15:13:29 +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
1996-05-31 15:13:29 +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/>.
1996-05-31 15:13:29 +00:00
*/
# include "includes.h"
2006-09-06 18:32:20 +00:00
/**
* @ file
* @ brief time handling functions
*/
1996-05-31 15:13:29 +00:00
# ifndef TIME_T_MIN
1997-03-07 17:22:27 +00:00
# define TIME_T_MIN ((time_t)0 < (time_t) -1 ? (time_t) 0 \
1996-05-31 15:13:29 +00:00
: ~ ( time_t ) 0 < < ( sizeof ( time_t ) * CHAR_BIT - 1 ) )
# endif
# ifndef TIME_T_MAX
# define TIME_T_MAX (~ (time_t) 0 - TIME_T_MIN)
# endif
2007-03-01 20:52:14 +00:00
# define NTTIME_INFINITY (NTTIME)0x8000000000000000LL
2006-08-24 20:51:57 +00:00
# if (SIZEOF_LONG == 8)
# define TIME_FIXUP_CONSTANT_INT 11644473600L
# elif (SIZEOF_LONG_LONG == 8)
# define TIME_FIXUP_CONSTANT_INT 11644473600LL
# endif
1996-05-31 15:13:29 +00:00
2006-09-06 18:32:20 +00:00
/*******************************************************************
create a 16 bit dos packed date
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static uint16_t make_dos_date1 ( struct tm * t )
{
uint16_t ret = 0 ;
ret = ( ( ( unsigned int ) ( t - > tm_mon + 1 ) ) > > 3 ) | ( ( t - > tm_year - 80 ) < < 1 ) ;
ret = ( ( ret & 0xFF ) < < 8 ) | ( t - > tm_mday | ( ( ( t - > tm_mon + 1 ) & 0x7 ) < < 5 ) ) ;
return ret ;
}
2006-08-24 20:51:57 +00:00
2006-09-06 18:32:20 +00:00
/*******************************************************************
create a 16 bit dos packed time
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static uint16_t make_dos_time1 ( struct tm * t )
{
uint16_t ret = 0 ;
ret = ( ( ( ( unsigned int ) t - > tm_min > > 3 ) & 0x7 ) | ( ( ( unsigned int ) t - > tm_hour ) < < 3 ) ) ;
ret = ( ( ret & 0xFF ) < < 8 ) | ( ( t - > tm_sec / 2 ) | ( ( t - > tm_min & 0x7 ) < < 5 ) ) ;
return ret ;
}
2006-08-24 20:51:57 +00:00
2006-09-06 18:32:20 +00:00
/*******************************************************************
create a 32 bit dos packed date / time from some parameters
This takes a GMT time and returns a packed localtime structure
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static uint32_t make_dos_date ( time_t unixdate , int zone_offset )
1996-05-31 15:13:29 +00:00
{
2006-09-06 18:32:20 +00:00
struct tm * t ;
uint32_t ret = 0 ;
1998-06-30 21:19:40 +00:00
2006-09-06 18:32:20 +00:00
if ( unixdate = = 0 ) {
return 0 ;
2005-11-05 04:21:55 +00:00
}
1996-05-31 15:13:29 +00:00
2006-09-06 18:32:20 +00:00
unixdate - = zone_offset ;
1996-05-31 15:13:29 +00:00
2006-09-06 18:32:20 +00:00
t = gmtime ( & unixdate ) ;
if ( ! t ) {
return 0xFFFFFFFF ;
2005-11-05 04:21:55 +00:00
}
2003-10-12 20:48:56 +00:00
2006-09-06 18:32:20 +00:00
ret = make_dos_date1 ( t ) ;
ret = ( ( ret & 0xFFFF ) < < 16 ) | make_dos_time1 ( t ) ;
1996-05-31 15:13:29 +00:00
2006-08-24 16:44:00 +00:00
return ret ;
}
2006-09-06 18:32:20 +00:00
/**
parse a nttime as a large integer in a string and return a NTTIME
*/
NTTIME nttime_from_string ( const char * s )
2005-09-30 17:13:37 +00:00
{
2006-09-06 18:32:20 +00:00
return strtoull ( s , NULL , 0 ) ;
2005-09-30 17:13:37 +00:00
}
2007-05-18 23:38:56 +00:00
/**************************************************************
Handle conversions between time_t and uint32 , taking care to
preserve the " special " values .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
uint32 convert_time_t_to_uint32 ( time_t t )
{
# if (defined(SIZEOF_TIME_T) && (SIZEOF_TIME_T == 8))
/* time_t is 64-bit. */
if ( t = = 0x8000000000000000LL ) {
return 0x80000000 ;
} else if ( t = = 0x7FFFFFFFFFFFFFFFLL ) {
return 0x7FFFFFFF ;
}
# endif
return ( uint32 ) t ;
}
time_t convert_uint32_to_time_t ( uint32 u )
{
# if (defined(SIZEOF_TIME_T) && (SIZEOF_TIME_T == 8))
/* time_t is 64-bit. */
if ( u = = 0x80000000 ) {
return ( time_t ) 0x8000000000000000LL ;
} else if ( u = = 0x7FFFFFFF ) {
2007-05-18 23:56:34 +00:00
return ( time_t ) 0x7FFFFFFFFFFFFFFFLL ;
2007-05-18 23:38:56 +00:00
}
# endif
return ( time_t ) u ;
}
2006-09-06 18:32:20 +00:00
int extra_time_offset = 0 ;
2005-11-05 04:21:55 +00:00
/****************************************************************************
2006-09-06 18:32:20 +00:00
Check if NTTIME is 0.
2005-11-05 04:21:55 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
bool nt_time_is_zero ( const NTTIME * nt )
2005-09-30 17:13:37 +00:00
{
2006-09-06 18:32:20 +00:00
return ( * nt = = 0 ) ;
2005-09-30 17:13:37 +00:00
}
2005-03-23 23:26:33 +00:00
/****************************************************************************
2005-11-05 04:21:55 +00:00
Convert ASN .1 GeneralizedTime string to unix - time .
Returns 0 on failure ; Currently ignores timezone .
2005-03-23 23:26:33 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2005-11-05 04:21:55 +00:00
2005-03-23 23:26:33 +00:00
time_t generalized_to_unix_time ( const char * str )
{
struct tm tm ;
ZERO_STRUCT ( tm ) ;
if ( sscanf ( str , " %4d%2d%2d%2d%2d%2d " ,
& tm . tm_year , & tm . tm_mon , & tm . tm_mday ,
& tm . tm_hour , & tm . tm_min , & tm . tm_sec ) ! = 6 ) {
return 0 ;
}
tm . tm_year - = 1900 ;
tm . tm_mon - = 1 ;
return timegm ( & tm ) ;
}
2006-02-10 01:43:33 +00:00
2006-09-06 18:32:20 +00:00
/*******************************************************************
Accessor function for the server time zone offset .
set_server_zone_offset ( ) must have been called first .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-02-10 01:43:33 +00:00
2006-09-06 18:32:20 +00:00
static int server_zone_offset ;
2006-02-10 01:43:33 +00:00
2006-09-06 18:32:20 +00:00
int get_server_zone_offset ( void )
{
return server_zone_offset ;
2006-02-10 01:43:33 +00:00
}
2006-09-06 18:32:20 +00:00
/*******************************************************************
Initialize the server time zone offset . Called when a client connects .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int set_server_zone_offset ( time_t t )
2006-08-24 16:44:00 +00:00
{
2006-09-06 18:32:20 +00:00
server_zone_offset = get_time_zone ( t ) ;
return server_zone_offset ;
2006-08-24 16:44:00 +00:00
}
2006-09-06 18:32:20 +00:00
/****************************************************************************
Return the date and time as a string
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-03-28 15:49:13 +01:00
char * current_timestring ( TALLOC_CTX * ctx , bool hires )
2006-02-10 01:43:33 +00:00
{
2007-09-04 10:15:04 +00:00
fstring TimeBuf ;
2006-09-06 18:32:20 +00:00
struct timeval tp ;
time_t t ;
struct tm * tm ;
2006-02-10 01:43:33 +00:00
2006-09-06 18:32:20 +00:00
if ( hires ) {
GetTimeOfDay ( & tp ) ;
t = ( time_t ) tp . tv_sec ;
} else {
t = time ( NULL ) ;
}
tm = localtime ( & t ) ;
if ( ! tm ) {
if ( hires ) {
slprintf ( TimeBuf ,
sizeof ( TimeBuf ) - 1 ,
" %ld.%06ld seconds since the Epoch " ,
( long ) tp . tv_sec ,
( long ) tp . tv_usec ) ;
} else {
slprintf ( TimeBuf ,
sizeof ( TimeBuf ) - 1 ,
" %ld seconds since the Epoch " ,
( long ) t ) ;
}
} else {
# ifdef HAVE_STRFTIME
if ( hires ) {
strftime ( TimeBuf , sizeof ( TimeBuf ) - 1 , " %Y/%m/%d %H:%M:%S " , tm ) ;
slprintf ( TimeBuf + strlen ( TimeBuf ) ,
sizeof ( TimeBuf ) - 1 - strlen ( TimeBuf ) ,
" .%06ld " ,
( long ) tp . tv_usec ) ;
} else {
strftime ( TimeBuf , sizeof ( TimeBuf ) - 1 , " %Y/%m/%d %H:%M:%S " , tm ) ;
}
2006-02-10 01:43:33 +00:00
# else
2006-09-06 18:32:20 +00:00
if ( hires ) {
const char * asct = asctime ( tm ) ;
slprintf ( TimeBuf ,
sizeof ( TimeBuf ) - 1 ,
" %s.%06ld " ,
asct ? asct : " unknown " ,
( long ) tp . tv_usec ) ;
} else {
const char * asct = asctime ( tm ) ;
fstrcpy ( TimeBuf , asct ? asct : " unknown " ) ;
}
2006-02-10 01:43:33 +00:00
# endif
2006-09-06 18:32:20 +00:00
}
2008-03-28 15:49:13 +01:00
return talloc_strdup ( ctx , TimeBuf ) ;
2006-02-10 01:43:33 +00:00
}
2006-09-06 18:32:20 +00:00
/*******************************************************************
Put a dos date into a buffer ( time / date format ) .
This takes GMT time and puts local time in the buffer .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void put_dos_date ( char * buf , int offset , time_t unixdate , int zone_offset )
2006-08-24 16:44:00 +00:00
{
2006-09-06 18:32:20 +00:00
uint32 x = make_dos_date ( unixdate , zone_offset ) ;
SIVAL ( buf , offset , x ) ;
2006-08-24 16:44:00 +00:00
}
2006-09-06 18:32:20 +00:00
/*******************************************************************
Put a dos date into a buffer ( date / time format ) .
This takes GMT time and puts local time in the buffer .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void put_dos_date2 ( char * buf , int offset , time_t unixdate , int zone_offset )
2006-02-10 01:43:33 +00:00
{
2006-09-06 18:32:20 +00:00
uint32 x = make_dos_date ( unixdate , zone_offset ) ;
x = ( ( x & 0xFFFF ) < < 16 ) | ( ( x & 0xFFFF0000 ) > > 16 ) ;
SIVAL ( buf , offset , x ) ;
}
2006-02-10 01:43:33 +00:00
2006-09-06 18:32:20 +00:00
/*******************************************************************
Put a dos 32 bit " unix like " date into a buffer . This routine takes
GMT and converts it to LOCAL time before putting it ( most SMBs assume
localtime for this sort of date )
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void put_dos_date3 ( char * buf , int offset , time_t unixdate , int zone_offset )
{
if ( ! null_mtime ( unixdate ) ) {
unixdate - = zone_offset ;
}
SIVAL ( buf , offset , unixdate ) ;
2006-02-10 01:43:33 +00:00
}
2006-09-06 18:32:20 +00:00
/***************************************************************************
Server versions of the above functions .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void srv_put_dos_date ( char * buf , int offset , time_t unixdate )
2006-08-24 16:44:00 +00:00
{
2006-09-06 18:32:20 +00:00
put_dos_date ( buf , offset , unixdate , server_zone_offset ) ;
}
void srv_put_dos_date2 ( char * buf , int offset , time_t unixdate )
{
put_dos_date2 ( buf , offset , unixdate , server_zone_offset ) ;
}
void srv_put_dos_date3 ( char * buf , int offset , time_t unixdate )
{
put_dos_date3 ( buf , offset , unixdate , server_zone_offset ) ;
}
/****************************************************************************
Take a Unix time and convert to an NTTIME structure and place in buffer
pointed to by p .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void put_long_date_timespec ( char * p , struct timespec ts )
{
NTTIME nt ;
unix_timespec_to_nt_time ( & nt , ts ) ;
SIVAL ( p , 0 , nt & 0xFFFFFFFF ) ;
SIVAL ( p , 4 , nt > > 32 ) ;
}
void put_long_date ( char * p , time_t t )
{
struct timespec ts ;
ts . tv_sec = t ;
ts . tv_nsec = 0 ;
put_long_date_timespec ( p , ts ) ;
2006-08-24 16:44:00 +00:00
}
2006-02-10 01:43:33 +00:00
/****************************************************************************
Return the best approximation to a ' create time ' under UNIX from a stat
structure .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2008-08-27 15:06:14 -07:00
static time_t calc_create_time ( const SMB_STRUCT_STAT * st )
2006-02-10 01:43:33 +00:00
{
time_t ret , ret1 ;
ret = MIN ( st - > st_ctime , st - > st_mtime ) ;
ret1 = MIN ( ret , st - > st_atime ) ;
if ( ret1 ! = ( time_t ) 0 ) {
return ret1 ;
}
/*
* One of ctime , mtime or atime was zero ( probably atime ) .
* Just return MIN ( ctime , mtime ) .
*/
return ret ;
}
2006-09-06 18:32:20 +00:00
2008-08-27 15:06:14 -07:00
/****************************************************************************
Return the ' create time ' from a stat struct if it exists ( birthtime ) or else
use the best approximation .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct timespec get_create_timespec ( const SMB_STRUCT_STAT * pst , bool fake_dirs )
2006-09-06 18:32:20 +00:00
{
2008-08-27 15:06:14 -07:00
struct timespec ret ;
if ( S_ISDIR ( pst - > st_mode ) & & fake_dirs ) {
ret . tv_sec = 315493200L ; /* 1/1/1980 */
ret . tv_nsec = 0 ;
return ret ;
}
# if defined(HAVE_STAT_ST_BIRTHTIMESPEC)
2008-08-29 09:29:07 -07:00
ret = pst - > st_birthtimespec ;
2008-08-27 15:06:14 -07:00
# elif defined(HAVE_STAT_ST_BIRTHTIMENSEC)
ret . tv_sec = pst - > st_birthtime ;
ret . tv_nsec = pst - > st_birthtimenspec ;
# elif defined(HAVE_STAT_ST_BIRTHTIME)
ret . tv_sec = pst - > st_birthtime ;
ret . tv_nsec = 0 ;
# else
ret . tv_sec = calc_create_time ( pst ) ;
ret . tv_nsec = 0 ;
# endif
2008-08-29 09:29:07 -07:00
/* Deal with systems that don't initialize birthtime correctly.
* Pointed out by SATOH Fumiyasu < fumiyas @ osstech . jp > .
*/
if ( null_timespec ( ret ) ) {
ret . tv_sec = calc_create_time ( pst ) ;
ret . tv_nsec = 0 ;
}
return ret ;
2006-09-06 18:32:20 +00:00
}
/****************************************************************************
Get / Set all the possible time fields from a stat struct as a timespec .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-03-05 23:40:03 +00:00
struct timespec get_atimespec ( const SMB_STRUCT_STAT * pst )
2006-09-06 18:32:20 +00:00
{
# if !defined(HAVE_STAT_HIRES_TIMESTAMPS)
struct timespec ret ;
/* Old system - no ns timestamp. */
ret . tv_sec = pst - > st_atime ;
ret . tv_nsec = 0 ;
return ret ;
# else
# if defined(HAVE_STAT_ST_ATIM)
return pst - > st_atim ;
# elif defined(HAVE_STAT_ST_ATIMENSEC)
struct timespec ret ;
ret . tv_sec = pst - > st_atime ;
ret . tv_nsec = pst - > st_atimensec ;
return ret ;
# else
# error CONFIGURE_ERROR_IN_DETECTING_TIMESPEC_IN_STAT
2006-02-10 01:43:33 +00:00
# endif
2006-09-06 18:32:20 +00:00
# endif
}
2006-06-14 21:36:49 +00:00
2006-09-06 18:32:20 +00:00
void set_atimespec ( SMB_STRUCT_STAT * pst , struct timespec ts )
2006-08-24 16:44:00 +00:00
{
2006-09-06 18:32:20 +00:00
# if !defined(HAVE_STAT_HIRES_TIMESTAMPS)
/* Old system - no ns timestamp. */
pst - > st_atime = ts . tv_sec ;
# else
# if defined(HAVE_STAT_ST_ATIM)
pst - > st_atim = ts ;
# elif defined(HAVE_STAT_ST_ATIMENSEC)
pst - > st_atime = ts . tv_sec ;
pst - > st_atimensec = ts . tv_nsec
# else
# error CONFIGURE_ERROR_IN_DETECTING_TIMESPEC_IN_STAT
# endif
# endif
2006-08-24 16:44:00 +00:00
}
2006-07-11 18:01:26 +00:00
2007-03-05 23:40:03 +00:00
struct timespec get_mtimespec ( const SMB_STRUCT_STAT * pst )
2006-07-11 18:01:26 +00:00
{
2006-09-06 18:32:20 +00:00
# if !defined(HAVE_STAT_HIRES_TIMESTAMPS)
struct timespec ret ;
2006-07-11 18:01:26 +00:00
2006-09-06 18:32:20 +00:00
/* Old system - no ns timestamp. */
ret . tv_sec = pst - > st_mtime ;
ret . tv_nsec = 0 ;
return ret ;
# else
# if defined(HAVE_STAT_ST_MTIM)
return pst - > st_mtim ;
# elif defined(HAVE_STAT_ST_MTIMENSEC)
struct timespec ret ;
ret . tv_sec = pst - > st_mtime ;
ret . tv_nsec = pst - > st_mtimensec ;
return ret ;
# else
# error CONFIGURE_ERROR_IN_DETECTING_TIMESPEC_IN_STAT
# endif
# endif
}
2006-07-11 18:01:26 +00:00
2006-09-06 18:32:20 +00:00
void set_mtimespec ( SMB_STRUCT_STAT * pst , struct timespec ts )
{
# if !defined(HAVE_STAT_HIRES_TIMESTAMPS)
/* Old system - no ns timestamp. */
pst - > st_mtime = ts . tv_sec ;
2006-07-11 18:01:26 +00:00
# else
2006-09-06 18:32:20 +00:00
# if defined(HAVE_STAT_ST_MTIM)
pst - > st_mtim = ts ;
# elif defined(HAVE_STAT_ST_MTIMENSEC)
pst - > st_mtime = ts . tv_sec ;
pst - > st_mtimensec = ts . tv_nsec
# else
# error CONFIGURE_ERROR_IN_DETECTING_TIMESPEC_IN_STAT
2006-07-11 18:01:26 +00:00
# endif
2006-09-06 18:32:20 +00:00
# endif
}
2006-07-11 18:01:26 +00:00
2007-03-05 23:40:03 +00:00
struct timespec get_ctimespec ( const SMB_STRUCT_STAT * pst )
2006-09-06 18:32:20 +00:00
{
# if !defined(HAVE_STAT_HIRES_TIMESTAMPS)
struct timespec ret ;
/* Old system - no ns timestamp. */
ret . tv_sec = pst - > st_ctime ;
ret . tv_nsec = 0 ;
return ret ;
# else
# if defined(HAVE_STAT_ST_CTIM)
return pst - > st_ctim ;
# elif defined(HAVE_STAT_ST_CTIMENSEC)
struct timespec ret ;
ret . tv_sec = pst - > st_ctime ;
ret . tv_nsec = pst - > st_ctimensec ;
return ret ;
# else
# error CONFIGURE_ERROR_IN_DETECTING_TIMESPEC_IN_STAT
# endif
# endif
2006-07-11 18:01:26 +00:00
}
2006-09-06 18:32:20 +00:00
void set_ctimespec ( SMB_STRUCT_STAT * pst , struct timespec ts )
{
# if !defined(HAVE_STAT_HIRES_TIMESTAMPS)
/* Old system - no ns timestamp. */
pst - > st_ctime = ts . tv_sec ;
# else
# if defined(HAVE_STAT_ST_CTIM)
pst - > st_ctim = ts ;
# elif defined(HAVE_STAT_ST_CTIMENSEC)
pst - > st_ctime = ts . tv_sec ;
pst - > st_ctimensec = ts . tv_nsec
# else
# error CONFIGURE_ERROR_IN_DETECTING_TIMESPEC_IN_STAT
# endif
# endif
}
2006-07-11 18:01:26 +00:00
2006-09-06 18:32:20 +00:00
void dos_filetime_timespec ( struct timespec * tsp )
2006-07-11 18:01:26 +00:00
{
2006-09-06 18:32:20 +00:00
tsp - > tv_sec & = ~ 1 ;
tsp - > tv_nsec = 0 ;
2006-07-11 18:01:26 +00:00
}
2006-09-06 18:32:20 +00:00
/*******************************************************************
Create a unix date ( int GMT ) from a dos date ( which is actually in
localtime ) .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2006-06-14 21:36:49 +00:00
2007-01-30 19:26:01 +00:00
static time_t make_unix_date ( const void * date_ptr , int zone_offset )
2006-06-14 21:36:49 +00:00
{
2006-09-06 18:32:20 +00:00
uint32 dos_date = 0 ;
struct tm t ;
time_t ret ;
2006-06-14 21:36:49 +00:00
2006-09-06 18:32:20 +00:00
dos_date = IVAL ( date_ptr , 0 ) ;
2006-06-14 21:36:49 +00:00
2006-09-06 18:32:20 +00:00
if ( dos_date = = 0 ) {
return 0 ;
2006-06-14 21:36:49 +00:00
}
2006-09-06 18:32:20 +00:00
interpret_dos_date ( dos_date , & t . tm_year , & t . tm_mon ,
& t . tm_mday , & t . tm_hour , & t . tm_min , & t . tm_sec ) ;
t . tm_isdst = - 1 ;
ret = timegm ( & t ) ;
ret + = zone_offset ;
return ( ret ) ;
}
/*******************************************************************
Like make_unix_date ( ) but the words are reversed .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-01-30 19:26:01 +00:00
static time_t make_unix_date2 ( const void * date_ptr , int zone_offset )
2006-09-06 18:32:20 +00:00
{
uint32 x , x2 ;
x = IVAL ( date_ptr , 0 ) ;
x2 = ( ( x & 0xFFFF ) < < 16 ) | ( ( x & 0xFFFF0000 ) > > 16 ) ;
SIVAL ( & x , 0 , x2 ) ;
2007-01-30 19:26:01 +00:00
return ( make_unix_date ( ( const void * ) & x , zone_offset ) ) ;
2006-09-06 18:32:20 +00:00
}
/*******************************************************************
Create a unix GMT date from a dos date in 32 bit " unix like " format
these generally arrive as localtimes , with corresponding DST .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-01-30 19:26:01 +00:00
static time_t make_unix_date3 ( const void * date_ptr , int zone_offset )
2006-09-06 18:32:20 +00:00
{
time_t t = ( time_t ) IVAL ( date_ptr , 0 ) ;
if ( ! null_mtime ( t ) ) {
t + = zone_offset ;
}
return ( t ) ;
}
2007-01-30 19:26:01 +00:00
time_t srv_make_unix_date ( const void * date_ptr )
2006-09-06 18:32:20 +00:00
{
return make_unix_date ( date_ptr , server_zone_offset ) ;
}
2007-01-30 19:26:01 +00:00
time_t srv_make_unix_date2 ( const void * date_ptr )
2006-09-06 18:32:20 +00:00
{
return make_unix_date2 ( date_ptr , server_zone_offset ) ;
}
2007-01-30 19:26:01 +00:00
time_t srv_make_unix_date3 ( const void * date_ptr )
2006-09-06 18:32:20 +00:00
{
return make_unix_date3 ( date_ptr , server_zone_offset ) ;
}
2007-03-05 23:40:03 +00:00
/****************************************************************************
Convert a normalized timeval to a timespec .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct timespec convert_timeval_to_timespec ( const struct timeval tv )
{
struct timespec ts ;
ts . tv_sec = tv . tv_sec ;
ts . tv_nsec = tv . tv_usec * 1000 ;
return ts ;
}
/****************************************************************************
Convert a normalized timespec to a timeval .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct timeval convert_timespec_to_timeval ( const struct timespec ts )
{
struct timeval tv ;
tv . tv_sec = ts . tv_sec ;
tv . tv_usec = ts . tv_nsec / 1000 ;
return tv ;
}
/****************************************************************************
Return a timespec for the current time
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct timespec timespec_current ( void )
{
struct timeval tv ;
struct timespec ts ;
GetTimeOfDay ( & tv ) ;
ts . tv_sec = tv . tv_sec ;
2007-06-05 19:17:05 +00:00
ts . tv_nsec = tv . tv_usec * 1000 ;
2007-03-05 23:40:03 +00:00
return ts ;
}
/****************************************************************************
Return the lesser of two timespecs .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
struct timespec timespec_min ( const struct timespec * ts1 ,
const struct timespec * ts2 )
{
if ( ts1 - > tv_sec < ts2 - > tv_sec ) return * ts1 ;
if ( ts1 - > tv_sec > ts2 - > tv_sec ) return * ts2 ;
if ( ts1 - > tv_nsec < ts2 - > tv_nsec ) return * ts1 ;
return * ts2 ;
}
/****************************************************************************
compare two timespec structures .
Return - 1 if ts1 < ts2
Return 0 if ts1 = = ts2
Return 1 if ts1 > ts2
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int timespec_compare ( const struct timespec * ts1 , const struct timespec * ts2 )
{
if ( ts1 - > tv_sec > ts2 - > tv_sec ) return 1 ;
if ( ts1 - > tv_sec < ts2 - > tv_sec ) return - 1 ;
if ( ts1 - > tv_nsec > ts2 - > tv_nsec ) return 1 ;
if ( ts1 - > tv_nsec < ts2 - > tv_nsec ) return - 1 ;
return 0 ;
}
2006-09-06 18:32:20 +00:00
/****************************************************************************
Interprets an nt time into a unix struct timespec .
Differs from nt_time_to_unix in that an 8 byte value of 0xffffffffffffffff
will be returned as ( time_t ) - 1 , whereas nt_time_to_unix returns 0 in this case .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-01-30 19:26:01 +00:00
struct timespec interpret_long_date ( const char * p )
2006-09-06 18:32:20 +00:00
{
NTTIME nt ;
nt = IVAL ( p , 0 ) + ( ( uint64_t ) IVAL ( p , 4 ) < < 32 ) ;
if ( nt = = ( uint64_t ) - 1 ) {
struct timespec ret ;
ret . tv_sec = ( time_t ) - 1 ;
ret . tv_nsec = 0 ;
return ret ;
}
return nt_time_to_unix_timespec ( & nt ) ;
}
/***************************************************************************
Client versions of the above functions .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void cli_put_dos_date ( struct cli_state * cli , char * buf , int offset , time_t unixdate )
{
put_dos_date ( buf , offset , unixdate , cli - > serverzone ) ;
}
void cli_put_dos_date2 ( struct cli_state * cli , char * buf , int offset , time_t unixdate )
{
put_dos_date2 ( buf , offset , unixdate , cli - > serverzone ) ;
}
void cli_put_dos_date3 ( struct cli_state * cli , char * buf , int offset , time_t unixdate )
{
put_dos_date3 ( buf , offset , unixdate , cli - > serverzone ) ;
}
2007-11-29 13:24:54 -08:00
time_t cli_make_unix_date ( struct cli_state * cli , const void * date_ptr )
2006-09-06 18:32:20 +00:00
{
return make_unix_date ( date_ptr , cli - > serverzone ) ;
}
2007-11-29 13:24:54 -08:00
time_t cli_make_unix_date2 ( struct cli_state * cli , const void * date_ptr )
2006-09-06 18:32:20 +00:00
{
return make_unix_date2 ( date_ptr , cli - > serverzone ) ;
}
2007-11-29 13:24:54 -08:00
time_t cli_make_unix_date3 ( struct cli_state * cli , const void * date_ptr )
2006-09-06 18:32:20 +00:00
{
return make_unix_date3 ( date_ptr , cli - > serverzone ) ;
}
/****************************************************************************
Check if two NTTIMEs are the same .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
bool nt_time_equals ( const NTTIME * nt1 , const NTTIME * nt2 )
2006-09-06 18:32:20 +00:00
{
return ( * nt1 = = * nt2 ) ;
}
/*******************************************************************
Re - read the smb serverzone value .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static struct timeval start_time_hires ;
void TimeInit ( void )
{
set_server_zone_offset ( time ( NULL ) ) ;
DEBUG ( 4 , ( " TimeInit: Serverzone is %d \n " , server_zone_offset ) ) ;
/* Save the start time of this process. */
if ( start_time_hires . tv_sec = = 0 & & start_time_hires . tv_usec = = 0 ) {
GetTimeOfDay ( & start_time_hires ) ;
}
}
/**********************************************************************
Return a timeval struct of the uptime of this process . As TimeInit is
done before a daemon fork then this is the start time from the parent
daemon start . JRA .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void get_process_uptime ( struct timeval * ret_time )
{
struct timeval time_now_hires ;
GetTimeOfDay ( & time_now_hires ) ;
ret_time - > tv_sec = time_now_hires . tv_sec - start_time_hires . tv_sec ;
if ( time_now_hires . tv_usec < start_time_hires . tv_usec ) {
ret_time - > tv_sec - = 1 ;
ret_time - > tv_usec = 1000000 + ( time_now_hires . tv_usec - start_time_hires . tv_usec ) ;
} else {
ret_time - > tv_usec = time_now_hires . tv_usec - start_time_hires . tv_usec ;
}
}
/****************************************************************************
Convert a NTTIME structure to a time_t .
It ' s originally in " 100ns units " .
This is an absolute version of the one above .
By absolute I mean , it doesn ' t adjust from 1 / 1 / 1601 to 1 / 1 / 1970
if the NTTIME was 5 seconds , the time_t is 5 seconds . JFM
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
time_t nt_time_to_unix_abs ( const NTTIME * nt )
{
uint64 d ;
if ( * nt = = 0 ) {
return ( time_t ) 0 ;
}
if ( * nt = = ( uint64 ) - 1 ) {
return ( time_t ) - 1 ;
}
2007-03-01 20:52:14 +00:00
if ( * nt = = NTTIME_INFINITY ) {
return ( time_t ) - 1 ;
}
2006-09-06 18:32:20 +00:00
/* reverse the time */
/* it's a negative value, turn it to positive */
d = ~ * nt ;
d + = 1000 * 1000 * 10 / 2 ;
d / = 1000 * 1000 * 10 ;
if ( ! ( TIME_T_MIN < = ( ( time_t ) d ) & & ( ( time_t ) d ) < = TIME_T_MAX ) ) {
return ( time_t ) 0 ;
}
return ( time_t ) d ;
}
2008-06-16 13:36:53 +02:00
time_t uint64s_nt_time_to_unix_abs ( const uint64_t * src )
{
NTTIME nttime ;
nttime = * src ;
return nt_time_to_unix_abs ( & nttime ) ;
}
2006-09-06 18:32:20 +00:00
/****************************************************************************
Put a 8 byte filetime from a struct timespec . Uses GMT .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void unix_timespec_to_nt_time ( NTTIME * nt , struct timespec ts )
{
uint64 d ;
if ( ts . tv_sec = = 0 & & ts . tv_nsec = = 0 ) {
* nt = 0 ;
return ;
}
if ( ts . tv_sec = = TIME_T_MAX ) {
* nt = 0x7fffffffffffffffLL ;
return ;
}
if ( ts . tv_sec = = ( time_t ) - 1 ) {
* nt = ( uint64 ) - 1 ;
return ;
}
d = ts . tv_sec ;
d + = TIME_FIXUP_CONSTANT_INT ;
d * = 1000 * 1000 * 10 ;
/* d is now in 100ns units. */
d + = ( ts . tv_nsec / 100 ) ;
* nt = d ;
}
/****************************************************************************
Convert a time_t to a NTTIME structure
This is an absolute version of the one above .
By absolute I mean , it doesn ' t adjust from 1 / 1 / 1970 to 1 / 1 / 1601
2007-01-12 02:48:37 +00:00
If the time_t was 5 seconds , the NTTIME is 5 seconds . JFM
2006-09-06 18:32:20 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void unix_to_nt_time_abs ( NTTIME * nt , time_t t )
{
double d ;
if ( t = = 0 ) {
* nt = 0 ;
return ;
}
if ( t = = TIME_T_MAX ) {
* nt = 0x7fffffffffffffffLL ;
return ;
}
if ( t = = ( time_t ) - 1 ) {
/* that's what NT uses for infinite */
2007-03-01 20:52:14 +00:00
* nt = NTTIME_INFINITY ;
2006-09-06 18:32:20 +00:00
return ;
}
d = ( double ) ( t ) ;
d * = 1.0e7 ;
2007-12-20 22:17:16 +01:00
* nt = ( NTTIME ) d ;
2006-09-06 18:32:20 +00:00
/* convert to a negative value */
* nt = ~ * nt ;
}
/****************************************************************************
Check if it ' s a null mtime .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
bool null_mtime ( time_t mtime )
2006-09-06 18:32:20 +00:00
{
if ( mtime = = 0 | | mtime = = ( time_t ) 0xFFFFFFFF | | mtime = = ( time_t ) - 1 )
return ( True ) ;
return ( False ) ;
}
/****************************************************************************
Utility function that always returns a const string even if localtime
and asctime fail .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-03-05 23:40:03 +00:00
const char * time_to_asc ( const time_t t )
2006-09-06 18:32:20 +00:00
{
const char * asct ;
2007-03-05 23:40:03 +00:00
struct tm * lt = localtime ( & t ) ;
2006-09-06 18:32:20 +00:00
if ( ! lt ) {
return " unknown time " ;
}
asct = asctime ( lt ) ;
if ( ! asct ) {
return " unknown time " ;
}
return asct ;
}
const char * display_time ( NTTIME nttime )
{
float high ;
float low ;
int sec ;
int days , hours , mins , secs ;
if ( nttime = = 0 )
return " Now " ;
2007-03-01 20:52:14 +00:00
if ( nttime = = NTTIME_INFINITY )
2006-09-06 18:32:20 +00:00
return " Never " ;
high = 65536 ;
high = high / 10000 ;
high = high * 65536 ;
high = high / 1000 ;
high = high * ( ~ ( nttime > > 32 ) ) ;
low = ~ ( nttime & 0xFFFFFFFF ) ;
low = low / ( 1000 * 1000 * 10 ) ;
2007-12-20 22:17:16 +01:00
sec = ( int ) ( high + low ) ;
2006-09-06 18:32:20 +00:00
days = sec / ( 60 * 60 * 24 ) ;
hours = ( sec - ( days * 60 * 60 * 24 ) ) / ( 60 * 60 ) ;
mins = ( sec - ( days * 60 * 60 * 24 ) - ( hours * 60 * 60 ) ) / 60 ;
secs = sec - ( days * 60 * 60 * 24 ) - ( hours * 60 * 60 ) - ( mins * 60 ) ;
2007-09-04 10:15:04 +00:00
return talloc_asprintf ( talloc_tos ( ) , " %u days, %u hours, %u minutes, "
" %u seconds " , days , hours , mins , secs ) ;
2006-09-06 18:32:20 +00:00
}
2007-10-18 17:40:25 -07:00
bool nt_time_is_set ( const NTTIME * nt )
2006-09-06 18:32:20 +00:00
{
if ( * nt = = 0x7FFFFFFFFFFFFFFFLL ) {
return False ;
}
2007-03-01 20:52:14 +00:00
if ( * nt = = NTTIME_INFINITY ) {
2006-09-06 18:32:20 +00:00
return False ;
}
return True ;
}