Second attempt at fix for bug 6529 - Offline files conflict with Vista and Office...
[samba.git] / source3 / lib / time.c
index f76a1bdc0d89029c3db7896d46088e958d62dff1..f6ff6d3407480020389dd1911c5e7157e51e63ec 100644 (file)
@@ -1,11 +1,14 @@
 /* 
    Unix SMB/CIFS implementation.
    time handling functions
-   Copyright (C) Andrew Tridgell               1992-1998
+
+   Copyright (C) Andrew Tridgell               1992-2004
    Copyright (C) Stefan (metze) Metzmacher     2002   
+   Copyright (C) Jeremy Allison                        2007
+
    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 2 of the License, or
+   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,
    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, write to the Free Software
-   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+   along with this program.  If not, see <http://www.gnu.org/licenses/>.
 */
 
 #include "includes.h"
 
-/*
-  This stuff was largely rewritten by Paul Eggert <eggert@twinsun.com>
-  in May 1996 
 */
+/**
+ * @file
+ * @brief time handling functions
+ */
 
 
-int extra_time_offset = 0;
-
-#ifndef CHAR_BIT
-#define CHAR_BIT 8
-#endif
-
 #ifndef TIME_T_MIN
 #define TIME_T_MIN ((time_t)0 < (time_t) -1 ? (time_t) 0 \
                    : ~ (time_t) 0 << (sizeof (time_t) * CHAR_BIT - 1))
@@ -40,714 +36,762 @@ int extra_time_offset = 0;
 #define TIME_T_MAX (~ (time_t) 0 - TIME_T_MIN)
 #endif
 
-void get_nttime_max(NTTIME *t)
-{
-       /* FIXME: This is incorrect */
-       unix_to_nt_time(t, get_time_t_max());
-}
+#define NTTIME_INFINITY (NTTIME)0x8000000000000000LL
+
+#if (SIZEOF_LONG == 8)
+#define TIME_FIXUP_CONSTANT_INT 11644473600L
+#elif (SIZEOF_LONG_LONG == 8)
+#define TIME_FIXUP_CONSTANT_INT 11644473600LL
+#endif
 
 /*******************************************************************
- External access to time_t_min and time_t_max.
+  create a 16 bit dos packed date
 ********************************************************************/
-
-time_t get_time_t_max(void)
+static uint16_t make_dos_date1(struct tm *t)
 {
-       return TIME_T_MAX;
+       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;
 }
 
 /*******************************************************************
-a gettimeofday wrapper
+  create a 16 bit dos packed time
 ********************************************************************/
-void GetTimeOfDay(struct timeval *tval)
+static uint16_t make_dos_time1(struct tm *t)
 {
-#ifdef HAVE_GETTIMEOFDAY_TZ
-       gettimeofday(tval,NULL);
-#else
-       gettimeofday(tval);
-#endif
+       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;
 }
 
-#define TM_YEAR_BASE 1900
-
 /*******************************************************************
-yield the difference between *A and *B, in seconds, ignoring leap seconds
+  create a 32 bit dos packed date/time from some parameters
+  This takes a GMT time and returns a packed localtime structure
 ********************************************************************/
-static int tm_diff(struct tm *a, struct tm *b)
+static uint32_t make_dos_date(time_t unixdate, int zone_offset)
 {
-  int ay = a->tm_year + (TM_YEAR_BASE - 1);
-  int by = b->tm_year + (TM_YEAR_BASE - 1);
-  int intervening_leap_days =
-    (ay/4 - by/4) - (ay/100 - by/100) + (ay/400 - by/400);
-  int years = ay - by;
-  int days = 365*years + intervening_leap_days + (a->tm_yday - b->tm_yday);
-  int hours = 24*days + (a->tm_hour - b->tm_hour);
-  int minutes = 60*hours + (a->tm_min - b->tm_min);
-  int seconds = 60*minutes + (a->tm_sec - b->tm_sec);
+       struct tm *t;
+       uint32_t ret=0;
+
+       if (unixdate == 0) {
+               return 0;
+       }
+
+       unixdate -= zone_offset;
 
-  return seconds;
+       t = gmtime(&unixdate);
+       if (!t) {
+               return 0xFFFFFFFF;
+       }
+
+       ret = make_dos_date1(t);
+       ret = ((ret&0xFFFF)<<16) | make_dos_time1(t);
+
+       return ret;
 }
 
-/*******************************************************************
-  return the UTC offset in seconds west of UTC, or 0 if it cannot be determined
-  ******************************************************************/
-static int TimeZone(time_t t)
+/**
+  parse a nttime as a large integer in a string and return a NTTIME
+*/
+NTTIME nttime_from_string(const char *s)
 {
-  struct tm *tm = gmtime(&t);
-  struct tm tm_utc;
-  if (!tm)
-    return 0;
-  tm_utc = *tm;
-  tm = localtime(&t);
-  if (!tm)
-    return 0;
-  return tm_diff(&tm_utc,tm);
+       return strtoull(s, NULL, 0);
+}
 
+/**************************************************************
+ Handle conversions between time_t and uint32, taking care to
+ preserve the "special" values.
+**************************************************************/
+
+uint32_t 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)t;
 }
 
-static BOOL done_serverzone_init;
+time_t convert_uint32_to_time_t(uint32_t 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) {
+               return (time_t)0x7FFFFFFFFFFFFFFFLL;
+       }
+#endif
+       return (time_t)u;
+}
 
-/* Return the smb serverzone value */
+/****************************************************************************
+ Check if NTTIME is 0.
+****************************************************************************/
 
-static int get_serverzone(void)
+bool nt_time_is_zero(const NTTIME *nt)
 {
-        static int serverzone;
+       return (*nt == 0);
+}
 
-        if (!done_serverzone_init) {
-                serverzone = TimeZone(time(NULL));
+/****************************************************************************
+ Convert ASN.1 GeneralizedTime string to unix-time.
+ Returns 0 on failure; Currently ignores timezone. 
+****************************************************************************/
 
-                if ((serverzone % 60) != 0) {
-                        DEBUG(1,("WARNING: Your timezone is not a multiple of 1 minute.\n"));
-                }
+time_t generalized_to_unix_time(const char *str)
+{ 
+       struct tm tm;
 
-                DEBUG(4,("Serverzone is %d\n",serverzone));
+       ZERO_STRUCT(tm);
 
-                done_serverzone_init = True;
-        }
+       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 serverzone;
+       return timegm(&tm);
 }
 
-/* Re-read the smb serverzone value */
+/*******************************************************************
+ Accessor function for the server time zone offset.
+ set_server_zone_offset() must have been called first.
+******************************************************************/
 
-static struct timeval start_time_hires;
+static int server_zone_offset;
 
-void TimeInit(void)
+int get_server_zone_offset(void)
 {
-       done_serverzone_init = False;
-       get_serverzone();
-       /* 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 server_zone_offset;
 }
 
-/**********************************************************************
- 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.
-***********************************************************************/
+/*******************************************************************
+ Initialize the server time zone offset. Called when a client connects.
+******************************************************************/
 
-void get_process_uptime(struct timeval *ret_time)
+int set_server_zone_offset(time_t t)
 {
-       struct timeval time_now_hires;
-
-       GetTimeOfDay(&time_now_hires);
-       ret_time->tv_sec = time_now_hires.tv_sec - start_time_hires.tv_sec;
-       ret_time->tv_usec = time_now_hires.tv_usec - start_time_hires.tv_usec;
-       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;
+       server_zone_offset = get_time_zone(t);
+       return server_zone_offset;
 }
 
-/*******************************************************************
-return the same value as TimeZone, but it should be more efficient.
+/****************************************************************************
+ Return the date and time as a string
+****************************************************************************/
 
-We keep a table of DST offsets to prevent calling localtime() on each 
-call of this function. This saves a LOT of time on many unixes.
+char *current_timestring(TALLOC_CTX *ctx, bool hires)
+{
+       fstring TimeBuf;
+       struct timeval tp;
+       time_t t;
+       struct tm *tm;
 
-Updated by Paul Eggert <eggert@twinsun.com>
-********************************************************************/
-static int TimeZoneFaster(time_t t)
-{
-  static struct dst_table {time_t start,end; int zone;} *tdt, *dst_table = NULL;
-  static int table_size = 0;
-  int i;
-  int zone = 0;
-
-  if (t == 0) t = time(NULL);
-
-  /* Tunis has a 8 day DST region, we need to be careful ... */
-#define MAX_DST_WIDTH (365*24*60*60)
-#define MAX_DST_SKIP (7*24*60*60)
-
-  for (i=0;i<table_size;i++)
-    if (t >= dst_table[i].start && t <= dst_table[i].end) break;
-
-  if (i<table_size) {
-    zone = dst_table[i].zone;
-  } else {
-    time_t low,high;
-
-    zone = TimeZone(t);
-    tdt = (struct dst_table *)Realloc(dst_table,
-                                             sizeof(dst_table[0])*(i+1));
-    if (!tdt) {
-      DEBUG(0,("TimeZoneFaster: out of memory!\n"));
-      SAFE_FREE(dst_table);
-      table_size = 0;
-    } else {
-      dst_table = tdt;
-      table_size++;
-
-      dst_table[i].zone = zone; 
-      dst_table[i].start = dst_table[i].end = t;
-    
-      /* no entry will cover more than 6 months */
-      low = t - MAX_DST_WIDTH/2;
-      if (t < low)
-       low = TIME_T_MIN;
-      
-      high = t + MAX_DST_WIDTH/2;
-      if (high < t)
-       high = TIME_T_MAX;
-      
-      /* widen the new entry using two bisection searches */
-      while (low+60*60 < dst_table[i].start) {
-       if (dst_table[i].start - low > MAX_DST_SKIP*2)
-         t = dst_table[i].start - MAX_DST_SKIP;
-       else
-         t = low + (dst_table[i].start-low)/2;
-       if (TimeZone(t) == zone)
-         dst_table[i].start = t;
-       else
-         low = t;
-      }
-
-      while (high-60*60 > dst_table[i].end) {
-       if (high - dst_table[i].end > MAX_DST_SKIP*2)
-         t = dst_table[i].end + MAX_DST_SKIP;
-       else
-         t = high - (high-dst_table[i].end)/2;
-       if (TimeZone(t) == zone)
-         dst_table[i].end = t;
-       else
-         high = t;
-      }
-#if 0
-      DEBUG(1,("Added DST entry from %s ",
-              asctime(localtime(&dst_table[i].start))));
-      DEBUG(1,("to %s (%d)\n",asctime(localtime(&dst_table[i].end)),
-              dst_table[i].zone));
+       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);
+               }
+#else
+               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");
+               }
 #endif
-    }
-  }
-  return zone;
+       }
+       return talloc_strdup(ctx, TimeBuf);
 }
 
-/****************************************************************************
-  return the UTC offset in seconds west of UTC, adjusted for extra time offset
-  **************************************************************************/
-int TimeDiff(time_t t)
+
+/*******************************************************************
+ 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)
 {
-  return TimeZoneFaster(t) + 60*extra_time_offset;
+       uint32_t x = make_dos_date(unixdate, zone_offset);
+       SIVAL(buf,offset,x);
 }
 
+/*******************************************************************
+ Put a dos date into a buffer (date/time format).
+ This takes GMT time and puts local time in the buffer.
+********************************************************************/
 
-/****************************************************************************
-  return the UTC offset in seconds west of UTC, adjusted for extra time
-  offset, for a local time value.  If ut = lt + LocTimeDiff(lt), then
-  lt = ut - TimeDiff(ut), but the converse does not necessarily hold near
-  daylight savings transitions because some local times are ambiguous.
-  LocTimeDiff(t) equals TimeDiff(t) except near daylight savings transitions.
-  +**************************************************************************/
-static int LocTimeDiff(time_t lte)
+static void put_dos_date2(char *buf,int offset,time_t unixdate, int zone_offset)
 {
-  time_t lt = lte - 60*extra_time_offset;
-  int d = TimeZoneFaster(lt);
-  time_t t = lt + d;
+       uint32_t x = make_dos_date(unixdate, zone_offset);
+       x = ((x&0xFFFF)<<16) | ((x&0xFFFF0000)>>16);
+       SIVAL(buf,offset,x);
+}
 
-  /* if overflow occurred, ignore all the adjustments so far */
-  if (((lte < lt) ^ (extra_time_offset < 0))  |  ((t < lt) ^ (d < 0)))
-    t = lte;
+/*******************************************************************
+ 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)
+********************************************************************/
 
-  /* now t should be close enough to the true UTC to yield the right answer */
-  return TimeDiff(t);
+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);
 }
 
 
-/****************************************************************************
-try to optimise the localtime call, it can be quite expensive on some machines
-****************************************************************************/
-struct tm *LocalTime(time_t *t)
+/***************************************************************************
+ Server versions of the above functions.
+***************************************************************************/
+
+void srv_put_dos_date(char *buf,int offset,time_t unixdate)
 {
-  time_t t2 = *t;
+       put_dos_date(buf, offset, unixdate, server_zone_offset);
+}
 
-  t2 -= TimeDiff(t2);
+void srv_put_dos_date2(char *buf,int offset, time_t unixdate)
+{
+       put_dos_date2(buf, offset, unixdate, server_zone_offset);
+}
 
-  return(gmtime(&t2));
+void srv_put_dos_date3(char *buf,int offset,time_t unixdate)
+{
+       put_dos_date3(buf, offset, unixdate, server_zone_offset);
 }
 
-#define TIME_FIXUP_CONSTANT (369.0*365.25*24*60*60-(3.0*24*60*60+6.0*60*60))
+void round_timespec(enum timestamp_set_resolution res, struct timespec *ts)
+{
+       switch (res) {
+               case TIMESTAMP_SET_SECONDS:
+                       round_timespec_to_sec(ts);
+                       break;
+               case TIMESTAMP_SET_MSEC:
+                       round_timespec_to_usec(ts);
+                       break;
+               case TIMESTAMP_SET_NT_OR_BETTER:
+                       /* No rounding needed. */
+                       break;
+        }
+}
 
 /****************************************************************************
-interpret an 8 byte "filetime" structure to a time_t
-It's originally in "100ns units since jan 1st 1601"
-
-It appears to be kludge-GMT (at least for file listings). This means
-its the GMT you get by taking a localtime and adding the
-serverzone. This is NOT the same as GMT in some cases. This routine
-converts this to real GMT.
+ Take a Unix time and convert to an NTTIME structure and place in buffer 
+ pointed to by p, rounded to the correct resolution.
 ****************************************************************************/
-time_t nt_time_to_unix(NTTIME *nt)
+
+void put_long_date_timespec(enum timestamp_set_resolution res, char *p, struct timespec ts)
 {
-  double d;
-  time_t ret;
-  /* The next two lines are a fix needed for the 
-     broken SCO compiler. JRA. */
-  time_t l_time_min = TIME_T_MIN;
-  time_t l_time_max = TIME_T_MAX;
+       NTTIME nt;
+       round_timespec(res, &ts);
+       unix_timespec_to_nt_time(&nt, ts);
+       SIVAL(p, 0, nt & 0xFFFFFFFF);
+       SIVAL(p, 4, nt >> 32);
+}
 
-  if (nt->high == 0) return(0);
+void put_long_date(char *p, time_t t)
+{
+       struct timespec ts;
+       ts.tv_sec = t;
+       ts.tv_nsec = 0;
+       put_long_date_timespec(TIMESTAMP_SET_SECONDS, p, ts);
+}
 
-  d = ((double)nt->high)*4.0*(double)(1<<30);
-  d += (nt->low&0xFFF00000);
-  d *= 1.0e-7;
-  /* now adjust by 369 years to make the secs since 1970 */
-  d -= TIME_FIXUP_CONSTANT;
+void dos_filetime_timespec(struct timespec *tsp)
+{
+       tsp->tv_sec &= ~1;
+       tsp->tv_nsec = 0;
+}
 
-  if (!(l_time_min <= d && d <= l_time_max))
-    return(0);
+/*******************************************************************
+ Create a unix date (int GMT) from a dos date (which is actually in
+ localtime).
+********************************************************************/
 
-  ret = (time_t)(d+0.5);
+static time_t make_unix_date(const void *date_ptr, int zone_offset)
+{
+       uint32_t dos_date=0;
+       struct tm t;
+       time_t ret;
 
-  /* this takes us from kludge-GMT to real GMT */
-  ret -= get_serverzone();
-  ret += LocTimeDiff(ret);
+       dos_date = IVAL(date_ptr,0);
 
-  return(ret);
-}
+       if (dos_date == 0) {
+               return 0;
+       }
+  
+       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);
 
-/****************************************************************************
- Convert a NTTIME structure to a time_t.
- It's originally in "100ns units".
+       ret += zone_offset;
 
- 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
-****************************************************************************/
+       return(ret);
+}
 
-time_t nt_time_to_unix_abs(NTTIME *nt)
+/*******************************************************************
+ Like make_unix_date() but the words are reversed.
+********************************************************************/
+
+time_t make_unix_date2(const void *date_ptr, int zone_offset)
 {
-       double d;
-       time_t ret;
-       /* The next two lines are a fix needed for the 
-          broken SCO compiler. JRA. */
-       time_t l_time_min = TIME_T_MIN;
-       time_t l_time_max = TIME_T_MAX;
+       uint32_t x,x2;
 
-       if (nt->high == 0)
-               return(0);
+       x = IVAL(date_ptr,0);
+       x2 = ((x&0xFFFF)<<16) | ((x&0xFFFF0000)>>16);
+       SIVAL(&x,0,x2);
 
-       if (nt->high==0x80000000 && nt->low==0)
-               return -1;
+       return(make_unix_date((const void *)&x, zone_offset));
+}
 
-       /* reverse the time */
-       /* it's a negative value, turn it to positive */
-       nt->high=~nt->high;
-       nt->low=~nt->low;
+/*******************************************************************
+ Create a unix GMT date from a dos date in 32 bit "unix like" format
+ these generally arrive as localtimes, with corresponding DST.
+******************************************************************/
 
-       d = ((double)nt->high)*4.0*(double)(1<<30);
-       d += (nt->low&0xFFF00000);
-       d *= 1.0e-7;
-  
-       if (!(l_time_min <= d && d <= l_time_max))
-               return(0);
+time_t make_unix_date3(const void *date_ptr, int zone_offset)
+{
+       time_t t = (time_t)IVAL(date_ptr,0);
+       if (!null_mtime(t)) {
+               t += zone_offset;
+       }
+       return(t);
+}
 
-       ret = (time_t)(d+0.5);
+time_t srv_make_unix_date(const void *date_ptr)
+{
+       return make_unix_date(date_ptr, server_zone_offset);
+}
 
-       return(ret);
+time_t srv_make_unix_date2(const void *date_ptr)
+{
+       return make_unix_date2(date_ptr, server_zone_offset);
 }
 
-/****************************************************************************
-interprets an nt time into a unix time_t
-****************************************************************************/
-time_t interpret_long_date(char *p)
+time_t srv_make_unix_date3(const void *date_ptr)
 {
-       NTTIME nt;
-       nt.low = IVAL(p,0);
-       nt.high = IVAL(p,4);
-       return nt_time_to_unix(&nt);
+       return make_unix_date3(date_ptr, server_zone_offset);
 }
 
 /****************************************************************************
-put a 8 byte filetime from a time_t
-This takes real GMT as input and converts to kludge-GMT
+ Convert a normalized timeval to a timespec.
 ****************************************************************************/
-void unix_to_nt_time(NTTIME *nt, time_t t)
+
+struct timespec convert_timeval_to_timespec(const struct timeval tv)
 {
-       double d;
+       struct timespec ts;
+       ts.tv_sec = tv.tv_sec;
+       ts.tv_nsec = tv.tv_usec * 1000;
+       return ts;
+}
 
-       if (t==0)
-       {
-               nt->low = 0;
-               nt->high = 0;
-               return;
-       }
-       if (t == TIME_T_MAX)
-       {
-               nt->low = 0xffffffff;
-               nt->high = 0x7fffffff;
-               return;
-       }               
-       if (t == -1)
-       {
-               nt->low = 0xffffffff;
-               nt->high = 0xffffffff;
-               return;
-       }               
+/****************************************************************************
+ Convert a normalized timespec to a timeval.
+****************************************************************************/
 
-       /* this converts GMT to kludge-GMT */
-       t -= TimeDiff(t) - get_serverzone(); 
+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;
+}
 
-       d = (double)(t);
-       d += TIME_FIXUP_CONSTANT;
-       d *= 1.0e7;
+/****************************************************************************
+ Return a timespec for the current time
+****************************************************************************/
 
-       nt->high = (uint32)(d * (1.0/(4.0*(double)(1<<30))));
-       nt->low  = (uint32)(d - ((double)nt->high)*4.0*(double)(1<<30));
+struct timespec timespec_current(void)
+{
+       struct timeval tv;
+       struct timespec ts;
+       GetTimeOfDay(&tv);
+       ts.tv_sec = tv.tv_sec;
+       ts.tv_nsec = tv.tv_usec * 1000;
+       return ts;
 }
 
 /****************************************************************************
- 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
- If the nttime_t was 5 seconds, the NTTIME is 5 seconds. JFM
+ Return the lesser of two timespecs.
 ****************************************************************************/
 
-void unix_to_nt_time_abs(NTTIME *nt, time_t t)
+struct timespec timespec_min(const struct timespec *ts1,
+                          const struct timespec *ts2)
 {
-       double d;
-
-       if (t==0) {
-               nt->low = 0;
-               nt->high = 0;
-               return;
-       }
+       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;
+}
 
-       if (t == TIME_T_MAX) {
-               nt->low = 0xffffffff;
-               nt->high = 0x7fffffff;
-               return;
-       }
-               
-       if (t == -1) {
-               /* that's what NT uses for infinite */
-               nt->low = 0x0;
-               nt->high = 0x80000000;
-               return;
-       }               
+/****************************************************************************
+  compare two timespec structures. 
+  Return -1 if ts1 < ts2
+  Return 0 if ts1 == ts2
+  Return 1 if ts1 > ts2
+****************************************************************************/
 
-       d = (double)(t);
-       d *= 1.0e7;
+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;
+}
 
-       nt->high = (uint32)(d * (1.0/(4.0*(double)(1<<30))));
-       nt->low  = (uint32)(d - ((double)nt->high)*4.0*(double)(1<<30));
+/****************************************************************************
+ Round up a timespec if nsec > 500000000, round down if lower,
+ then zero nsec.
+****************************************************************************/
 
-       /* convert to a negative value */
-       nt->high=~nt->high;
-       nt->low=~nt->low;
+void round_timespec_to_sec(struct timespec *ts)
+{
+       ts->tv_sec = convert_timespec_to_time_t(*ts);
+       ts->tv_nsec = 0;
 }
 
-
 /****************************************************************************
-take an NTTIME structure, containing high / low time.  convert to unix time.
-lkclXXXX this may need 2 SIVALs not a memcpy.  we'll see...
+ Round a timespec to usec value.
 ****************************************************************************/
-void put_long_date(char *p,time_t t)
+
+void round_timespec_to_usec(struct timespec *ts)
 {
-       NTTIME nt;
-       unix_to_nt_time(&nt, t);
-       SIVAL(p, 0, nt.low);
-       SIVAL(p, 4, nt.high);
+       struct timeval tv = convert_timespec_to_timeval(*ts);
+       *ts = convert_timeval_to_timespec(tv);
 }
 
 /****************************************************************************
-check if it's a null mtime
+ 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.
 ****************************************************************************/
-BOOL null_mtime(time_t mtime)
+
+struct timespec interpret_long_date(const char *p)
 {
-  if (mtime == 0 || mtime == (time_t)0xFFFFFFFF || mtime == (time_t)-1)
-    return(True);
-  return(False);
+       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);
 }
 
-/*******************************************************************
-  create a 16 bit dos packed date
-********************************************************************/
-static uint16 make_dos_date1(struct tm *t)
+/***************************************************************************
+ Client versions of the above functions.
+***************************************************************************/
+
+void cli_put_dos_date(struct cli_state *cli, char *buf, int offset, time_t unixdate)
 {
-  uint16 ret=0;
-  ret = (((unsigned)(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);
+       put_dos_date(buf, offset, unixdate, cli->serverzone);
 }
 
-/*******************************************************************
-  create a 16 bit dos packed time
-********************************************************************/
-static uint16 make_dos_time1(struct tm *t)
+void cli_put_dos_date2(struct cli_state *cli, char *buf, int offset, time_t unixdate)
 {
-  uint16 ret=0;
-  ret = ((((unsigned)t->tm_min >> 3)&0x7) | (((unsigned)t->tm_hour) << 3));
-  ret = ((ret&0xFF)<<8) | ((t->tm_sec/2) | ((t->tm_min & 0x7) << 5));
-  return(ret);
+       put_dos_date2(buf, offset, unixdate, cli->serverzone);
 }
 
-/*******************************************************************
-  create a 32 bit dos packed date/time from some parameters
-  This takes a GMT time and returns a packed localtime structure
-********************************************************************/
-static uint32 make_dos_date(time_t unixdate)
+void cli_put_dos_date3(struct cli_state *cli, char *buf, int offset, time_t unixdate)
 {
-  struct tm *t;
-  uint32 ret=0;
-
-  t = LocalTime(&unixdate);
-  if (!t)
-    return 0xFFFFFFFF;
-
-  ret = make_dos_date1(t);
-  ret = ((ret&0xFFFF)<<16) | make_dos_time1(t);
+       put_dos_date3(buf, offset, unixdate, cli->serverzone);
+}
 
-  return(ret);
+time_t cli_make_unix_date(struct cli_state *cli, const void *date_ptr)
+{
+       return make_unix_date(date_ptr, cli->serverzone);
 }
 
-/*******************************************************************
-put a dos date into a buffer (time/date format)
-This takes GMT time and puts local time in the buffer
-********************************************************************/
-void put_dos_date(char *buf,int offset,time_t unixdate)
+time_t cli_make_unix_date2(struct cli_state *cli, const void *date_ptr)
 {
-  uint32 x = make_dos_date(unixdate);
-  SIVAL(buf,offset,x);
+       return make_unix_date2(date_ptr, cli->serverzone);
 }
 
-/*******************************************************************
-put a dos date into a buffer (date/time format)
-This takes GMT time and puts local time in the buffer
-********************************************************************/
-void put_dos_date2(char *buf,int offset,time_t unixdate)
+time_t cli_make_unix_date3(struct cli_state *cli, const void *date_ptr)
 {
-  uint32 x = make_dos_date(unixdate);
-  x = ((x&0xFFFF)<<16) | ((x&0xFFFF0000)>>16);
-  SIVAL(buf,offset,x);
+       return make_unix_date3(date_ptr, cli->serverzone);
 }
 
-/*******************************************************************
-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)
-********************************************************************/
-void put_dos_date3(char *buf,int offset,time_t unixdate)
+/****************************************************************************
+ Check if two NTTIMEs are the same.
+****************************************************************************/
+
+bool nt_time_equals(const NTTIME *nt1, const NTTIME *nt2)
 {
-  if (!null_mtime(unixdate))
-    unixdate -= TimeDiff(unixdate);
-  SIVAL(buf,offset,unixdate);
+       return (*nt1 == *nt2);
 }
 
 /*******************************************************************
-  interpret a 32 bit dos packed date/time to some parameters
-********************************************************************/
-static void interpret_dos_date(uint32 date,int *year,int *month,int *day,int *hour,int *minute,int *second)
+ Re-read the smb serverzone value.
+******************************************************************/
+
+static struct timeval start_time_hires;
+
+void TimeInit(void)
 {
-  uint32 p0,p1,p2,p3;
+       set_server_zone_offset(time(NULL));
 
-  p0=date&0xFF; p1=((date&0xFF00)>>8)&0xFF; 
-  p2=((date&0xFF0000)>>16)&0xFF; p3=((date&0xFF000000)>>24)&0xFF;
+       DEBUG(4,("TimeInit: Serverzone is %d\n", server_zone_offset));
 
-  *second = 2*(p0 & 0x1F);
-  *minute = ((p0>>5)&0xFF) + ((p1&0x7)<<3);
-  *hour = (p1>>3)&0xFF;
-  *day = (p2&0x1F);
-  *month = ((p2>>5)&0xFF) + ((p3&0x1)<<3) - 1;
-  *year = ((p3>>1)&0xFF) + 80;
+       /* Save the start time of this process. */
+       if (start_time_hires.tv_sec == 0 && start_time_hires.tv_usec == 0) {
+               GetTimeOfDay(&start_time_hires);
+       }
 }
 
-/*******************************************************************
-  create a unix date (int GMT) from a dos date (which is actually in
-  localtime)
-********************************************************************/
-time_t make_unix_date(void *date_ptr)
+/**********************************************************************
+ 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)
 {
-  uint32 dos_date=0;
-  struct tm t;
-  time_t ret;
+       struct timeval time_now_hires;
 
-  dos_date = IVAL(date_ptr,0);
+       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;
+       }
+}
 
-  if (dos_date == 0) return(0);
-  
-  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;
-  
-  /* mktime() also does the local to GMT time conversion for us */
-  ret = mktime(&t);
+/****************************************************************************
+ Convert a NTTIME structure to a time_t.
+ It's originally in "100ns units".
 
-  return(ret);
-}
+ 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
+****************************************************************************/
 
-/*******************************************************************
-like make_unix_date() but the words are reversed
-********************************************************************/
-time_t make_unix_date2(void *date_ptr)
+time_t nt_time_to_unix_abs(const NTTIME *nt)
 {
-  uint32 x,x2;
+       uint64_t d;
 
-  x = IVAL(date_ptr,0);
-  x2 = ((x&0xFFFF)<<16) | ((x&0xFFFF0000)>>16);
-  SIVAL(&x,0,x2);
+       if (*nt == 0) {
+               return (time_t)0;
+       }
 
-  return(make_unix_date((void *)&x));
-}
+       if (*nt == (uint64_t)-1) {
+               return (time_t)-1;
+       }
 
-/*******************************************************************
-  create a unix GMT date from a dos date in 32 bit "unix like" format
-  these generally arrive as localtimes, with corresponding DST
-  ******************************************************************/
-time_t make_unix_date3(void *date_ptr)
-{
-  time_t t = (time_t)IVAL(date_ptr,0);
-  if (!null_mtime(t))
-    t += LocTimeDiff(t);
-  return(t);
-}
+       if (*nt == NTTIME_INFINITY) {
+               return (time_t)-1;
+       }
 
+       /* reverse the time */
+       /* it's a negative value, turn it to positive */
+       d=~*nt;
 
-/***************************************************************************
-return a HTTP/1.0 time string
-  ***************************************************************************/
-char *http_timestring(time_t t)
-{
-  static fstring buf;
-  struct tm *tm = LocalTime(&t);
+       d += 1000*1000*10/2;
+       d /= 1000*1000*10;
 
-  if (!tm)
-    slprintf(buf,sizeof(buf)-1,"%ld seconds since the Epoch",(long)t);
-  else
-#ifndef HAVE_STRFTIME
-  fstrcpy(buf, asctime(tm));
-  if(buf[strlen(buf)-1] == '\n')
-    buf[strlen(buf)-1] = 0;
-#else /* !HAVE_STRFTIME */
-  strftime(buf, sizeof(buf)-1, "%a, %d %b %Y %H:%M:%S %Z", tm);
-#endif /* !HAVE_STRFTIME */
-  return buf;
-}
+       if (!(TIME_T_MIN <= ((time_t)d) && ((time_t)d) <= TIME_T_MAX)) {
+               return (time_t)0;
+       }
 
+       return (time_t)d;
+}
 
+time_t uint64s_nt_time_to_unix_abs(const uint64_t *src)
+{
+       NTTIME nttime;
+       nttime = *src;
+       return nt_time_to_unix_abs(&nttime);
+}
 
 /****************************************************************************
- Return the date and time as a string
+ Put a 8 byte filetime from a struct timespec. Uses GMT.
 ****************************************************************************/
 
-char *timestring(BOOL hires)
+void unix_timespec_to_nt_time(NTTIME *nt, struct timespec ts)
 {
-       static fstring TimeBuf;
-       struct timeval tp;
-       time_t t;
-       struct tm *tm;
+       uint64_t d;
 
-       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,100,"%Y/%m/%d %H:%M:%S",tm);
-               }
-#else
-               if (hires) {
-                       slprintf(TimeBuf, 
-                                sizeof(TimeBuf)-1, 
-                                "%s.%06ld", 
-                                asctime(tm), 
-                                (long)tp.tv_usec);
-               } else {
-                       fstrcpy(TimeBuf, asctime(tm));
-               }
-#endif
+       if (ts.tv_sec ==0 && ts.tv_nsec == 0) {
+               *nt = 0;
+               return;
        }
-       return(TimeBuf);
+       if (ts.tv_sec == TIME_T_MAX) {
+               *nt = 0x7fffffffffffffffLL;
+               return;
+       }               
+       if (ts.tv_sec == (time_t)-1) {
+               *nt = (uint64_t)-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;
 }
 
 /****************************************************************************
-  return the best approximation to a 'create time' under UNIX from a stat
-  structure.
+ 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
+ If the time_t was 5 seconds, the NTTIME is 5 seconds. JFM
 ****************************************************************************/
 
-time_t get_create_time(SMB_STRUCT_STAT *st,BOOL fake_dirs)
+void unix_to_nt_time_abs(NTTIME *nt, time_t t)
 {
-  time_t ret, ret1;
+       double d;
 
-  if(S_ISDIR(st->st_mode) && fake_dirs)
-    return (time_t)315493200L;          /* 1/1/1980 */
-    
-  ret = MIN(st->st_ctime, st->st_mtime);
-  ret1 = MIN(ret, st->st_atime);
+       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 */
+               *nt = NTTIME_INFINITY;
+               return;
+       }               
+
+       d = (double)(t);
+       d *= 1.0e7;
 
-  if(ret1 != (time_t)0)
-    return ret1;
+       *nt = (NTTIME)d;
 
-  /*
-   * One of ctime, mtime or atime was zero (probably atime).
-   * Just return MIN(ctime, mtime).
-   */
-  return ret;
+       /* convert to a negative value */
+       *nt=~*nt;
 }
 
+
 /****************************************************************************
-initialise an NTTIME to -1, which means "unknown" or "don't expire"
+ Check if it's a null mtime.
 ****************************************************************************/
 
-void init_nt_time(NTTIME *nt)
+bool null_mtime(time_t mtime)
 {
-       nt->high = 0x7FFFFFFF;
-       nt->low = 0xFFFFFFFF;
+       if (mtime == 0 || mtime == (time_t)0xFFFFFFFF || mtime == (time_t)-1)
+               return true;
+       return false;
 }
 
 /****************************************************************************
-check if NTTIME is 0
+ Utility function that always returns a const string even if localtime
+ and asctime fail.
 ****************************************************************************/
-BOOL nt_time_is_zero(NTTIME *nt)
+
+const char *time_to_asc(const time_t t)
 {
-       if(nt->high==0) 
-               return True;
-       return False;
+       const char *asct;
+       struct tm *lt = localtime(&t);
+
+       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";
+
+       if (nttime==NTTIME_INFINITY)
+               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);
+
+       sec=(int)(high+low);
+
+       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);
+
+       return talloc_asprintf(talloc_tos(), "%u days, %u hours, %u minutes, "
+                              "%u seconds", days, hours, mins, secs);
+}
+
+bool nt_time_is_set(const NTTIME *nt)
+{
+       if (*nt == 0x7FFFFFFFFFFFFFFFLL) {
+               return false;
+       }
+
+       if (*nt == NTTIME_INFINITY) {
+               return false;
+       }
+
+       return true;
 }