You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1244 lines
32 KiB
1244 lines
32 KiB
//***************************************************************************
|
|
//
|
|
// Copyright © Microsoft Corporation. All rights reserved.
|
|
//
|
|
// wbemtime.cpp
|
|
//
|
|
// Purpose: Defines the WBEMTime and WBEMTimeSpan objects which are
|
|
// similar to the MFC CTime and CTimeSpan objects. The WBEM versions
|
|
// are capable of storing down to the nsec and also have functions for
|
|
// Creating from and getting BSTRs.
|
|
//
|
|
// Note; The current implementation of WBEMTime does not support dates
|
|
// before 1/1/1601;
|
|
//
|
|
// WBEMTime::m_uTime is stored in GMT as 100 nsecs since 1/1/1601
|
|
//
|
|
//***************************************************************************
|
|
|
|
#include "precomp.h"
|
|
#include <stdio.h>
|
|
#pragma warning( disable : 4290 )
|
|
|
|
#ifdef UTILLIB
|
|
#include <assertbreak.h>
|
|
#else
|
|
#define ASSERT_BREAK(a)
|
|
#endif //UTILLIB
|
|
|
|
#include <WbemTime.h>
|
|
#include <comdef.h>
|
|
|
|
|
|
// These are here rather than wbemtime.h so we don't have to doc/support
|
|
#define DECPOS 14
|
|
#define SGNPOS 21
|
|
#define DMTFLEN 25
|
|
|
|
#define DEPRECATED 0
|
|
#define INVALID_TIME_FORMAT 0
|
|
#define INVALID_TIME_ARITHMETIC 0
|
|
#define BAD_TIMEZONE 0
|
|
|
|
// ****************************************************************
|
|
// Static functions and variables. These can't be called/referenced
|
|
// outside of wbemtime.cpp
|
|
|
|
static WBEMTime g_Jan1970((time_t)0);
|
|
|
|
//***************************************************************************
|
|
//
|
|
// StructtmToSystemTime
|
|
//
|
|
// Description: General utility for converting between the two common
|
|
// data structures.
|
|
//
|
|
// Return values: TRUE if OK;
|
|
//
|
|
//***************************************************************************
|
|
|
|
static BOOL StructtmToSystemTime(const struct tm *ptm, SYSTEMTIME * pst)
|
|
{
|
|
if (pst && ptm)
|
|
{
|
|
pst->wYear = ptm->tm_year + 1900;
|
|
pst->wMonth = ptm->tm_mon + 1;
|
|
pst->wDay = (WORD)ptm->tm_mday;
|
|
pst->wHour = (WORD)ptm->tm_hour;
|
|
pst->wMinute = (WORD)ptm->tm_min;
|
|
pst->wSecond = (WORD)ptm->tm_sec;
|
|
pst->wDayOfWeek = (WORD)ptm->tm_wday;
|
|
pst->wMilliseconds = 0;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static BOOL SystemTimeToStructtm(const SYSTEMTIME *pst, struct tm *ptm)
|
|
{
|
|
if (pst && ptm && pst->wYear >= 1900)
|
|
{
|
|
ptm->tm_year = pst->wYear - 1900;
|
|
ptm->tm_mon = pst->wMonth - 1;
|
|
ptm->tm_mday = pst->wDay;
|
|
ptm->tm_hour = pst->wHour;
|
|
ptm->tm_min = pst->wMinute;
|
|
ptm->tm_sec = pst->wSecond;
|
|
ptm->tm_wday = pst->wDayOfWeek;
|
|
ptm->tm_isdst = 0; // Since we are working in gmt...
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// FileTimeToui64
|
|
// ui64ToFileTime
|
|
//
|
|
// Description: Conversion routines for going between FILETIME structures
|
|
// and __int64.
|
|
//
|
|
//***************************************************************************
|
|
|
|
static void FileTimeToui64(const FILETIME *pft, ULONGLONG *p64)
|
|
{
|
|
*p64 = pft->dwHighDateTime;
|
|
*p64 = *p64 << 32;
|
|
*p64 |= pft->dwLowDateTime;
|
|
}
|
|
|
|
static void ui64ToFileTime(const ULONGLONG *p64,FILETIME *pft)
|
|
{
|
|
unsigned __int64 uTemp = *p64;
|
|
pft->dwLowDateTime = (DWORD)uTemp;
|
|
uTemp = uTemp >> 32;
|
|
pft->dwHighDateTime = (DWORD)uTemp;
|
|
}
|
|
|
|
static int CompareSYSTEMTIME(const SYSTEMTIME *pst1, const SYSTEMTIME *pst2)
|
|
{
|
|
FILETIME ft1, ft2;
|
|
|
|
SystemTimeToFileTime(pst1, &ft1);
|
|
SystemTimeToFileTime(pst2, &ft2);
|
|
|
|
return CompareFileTime(&ft1, &ft2);
|
|
}
|
|
|
|
// This function is used to convert the relative values that come
|
|
// back from GetTimeZoneInformation into an actual date for the year
|
|
// in question. The system time structure that is passed in is updated
|
|
// to contain the absolute values.
|
|
static void DayInMonthToAbsolute(SYSTEMTIME *pst, const WORD wYear)
|
|
{
|
|
const static int _lpdays[] = {
|
|
-1, 30, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365
|
|
};
|
|
|
|
const static int _days[] = {
|
|
-1, 30, 58, 89, 119, 150, 180, 211, 242, 272, 303, 333, 364
|
|
};
|
|
|
|
SHORT shYearDay;
|
|
|
|
// If this is not 0, this is not a relative date
|
|
if (pst->wYear == 0)
|
|
{
|
|
// Was that year a leap year?
|
|
BOOL bLeap = ( (( wYear % 400) == 0) || ((( wYear % 4) == 0) && (( wYear % 100) != 0)));
|
|
|
|
// Figure out the day of the year for the first day of the month in question
|
|
if (bLeap)
|
|
shYearDay = 1 + _lpdays[pst->wMonth - 1];
|
|
else
|
|
shYearDay = 1 + _days[pst->wMonth - 1];
|
|
|
|
// Now, figure out how many leap days there have been since 1/1/1601
|
|
WORD yc = wYear - 1601;
|
|
WORD y4 = (yc) / 4;
|
|
WORD y100 = (yc) / 100;
|
|
WORD y400 = (yc) / 400;
|
|
|
|
// This will tell us the day of the week for the first day of the month in question.
|
|
// The '1 +' reflects the fact that 1/1/1601 was a monday (figures). You might ask,
|
|
// 'why do we care what day of the week this is?' Well, I'll tell you. The way
|
|
// daylight savings time is defined is with things like 'the last sunday of the month
|
|
// of october.' Kinda helps to know what day that is.
|
|
SHORT monthdow = (1 + (yc * 365 + y4 + y400 - y100) + shYearDay) % 7;
|
|
|
|
if ( monthdow < pst->wDayOfWeek )
|
|
shYearDay += (pst->wDayOfWeek - monthdow) + (pst->wDay - 1) * 7;
|
|
else
|
|
shYearDay += (pst->wDayOfWeek - monthdow) + pst->wDay * 7;
|
|
|
|
/*
|
|
* May have to adjust the calculation above if week == 5 (meaning
|
|
* the last instance of the day in the month). Check if yearday falls
|
|
* beyond month and adjust accordingly.
|
|
*/
|
|
if ( (pst->wDay == 5) &&
|
|
(shYearDay > (bLeap ? _lpdays[pst->wMonth] :
|
|
_days[pst->wMonth])) )
|
|
{
|
|
shYearDay -= 7;
|
|
}
|
|
|
|
// Now update the structure.
|
|
pst->wYear = wYear;
|
|
pst->wDay = shYearDay - (bLeap ? _lpdays[pst->wMonth - 1] :
|
|
_days[pst->wMonth - 1]);
|
|
}
|
|
|
|
}
|
|
|
|
// **************************************************************************
|
|
// These are static to WBEMTIME, which means they CAN be called from outside
|
|
// wbemtime
|
|
|
|
LONG WBEMTime::GetLocalOffsetForDate(const time_t &t)
|
|
{
|
|
FILETIME ft;
|
|
ULONGLONG ull = Int32x32To64(t, 10000000) + 116444736000000000;
|
|
|
|
ui64ToFileTime(&ull, &ft);
|
|
|
|
return GetLocalOffsetForDate(&ft);
|
|
}
|
|
|
|
LONG WBEMTime::GetLocalOffsetForDate(const struct tm *ptmin)
|
|
{
|
|
SYSTEMTIME st;
|
|
|
|
StructtmToSystemTime(ptmin, &st);
|
|
|
|
return GetLocalOffsetForDate(&st);
|
|
}
|
|
|
|
LONG WBEMTime::GetLocalOffsetForDate(const FILETIME *pft)
|
|
{
|
|
SYSTEMTIME st;
|
|
|
|
FileTimeToSystemTime(pft, &st);
|
|
|
|
return GetLocalOffsetForDate(&st);
|
|
}
|
|
|
|
LONG WBEMTime::GetLocalOffsetForDate(const SYSTEMTIME *pst)
|
|
{
|
|
TIME_ZONE_INFORMATION tzTime;
|
|
DWORD dwRes = GetTimeZoneInformation(&tzTime);
|
|
LONG lRes = 0xffffffff;
|
|
|
|
switch (dwRes)
|
|
{
|
|
case TIME_ZONE_ID_UNKNOWN:
|
|
{
|
|
// Read tz, but no dst defined in this zone
|
|
lRes = tzTime.Bias * -1;
|
|
break;
|
|
}
|
|
case TIME_ZONE_ID_STANDARD:
|
|
case TIME_ZONE_ID_DAYLIGHT:
|
|
{
|
|
|
|
// Convert the relative dates to absolute dates
|
|
DayInMonthToAbsolute(&tzTime.DaylightDate, pst->wYear);
|
|
DayInMonthToAbsolute(&tzTime.StandardDate, pst->wYear);
|
|
|
|
if ( CompareSYSTEMTIME(&tzTime.DaylightDate, &tzTime.StandardDate) < 0 )
|
|
{
|
|
/*
|
|
* Northern hemisphere ordering
|
|
*/
|
|
if ( CompareSYSTEMTIME(pst, &tzTime.DaylightDate) < 0 || CompareSYSTEMTIME(pst, &tzTime.StandardDate) > 0)
|
|
{
|
|
lRes = tzTime.Bias * -1;
|
|
}
|
|
else
|
|
{
|
|
lRes = (tzTime.Bias + tzTime.DaylightBias) * -1;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
/*
|
|
* Southern hemisphere ordering
|
|
*/
|
|
if ( CompareSYSTEMTIME(pst, &tzTime.StandardDate) < 0 || CompareSYSTEMTIME(pst, &tzTime.DaylightDate) > 0)
|
|
{
|
|
lRes = (tzTime.Bias + tzTime.DaylightBias) * -1;
|
|
}
|
|
else
|
|
{
|
|
lRes = tzTime.Bias * -1;
|
|
}
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
case TIME_ZONE_ID_INVALID:
|
|
default:
|
|
{
|
|
// Can't read the timezone info
|
|
ASSERT_BREAK(BAD_TIMEZONE);
|
|
break;
|
|
}
|
|
}
|
|
|
|
return lRes;
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// WBEMTime - This class holds time values.
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::operator=(BSTR bstrWbemFormat)
|
|
//
|
|
// Description: Assignment operator which is also used by the constructor.
|
|
// The string must have the format:
|
|
// YYYYMMDDHHSS.123456789 So 3:04 am, 1/1/96 would be 199601010304.0
|
|
//
|
|
// or the format yyyymmddhhmmss.mmmmmmsuuu.
|
|
//
|
|
// Note that the fractional part can be between 1 and nine digits.
|
|
//
|
|
// Return: WBEMTime object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
const WBEMTime & WBEMTime::operator=(const BSTR bstrWbemFormat)
|
|
{
|
|
Clear(); // set when properly assigned
|
|
|
|
if((NULL == bstrWbemFormat) ||
|
|
wcslen(bstrWbemFormat) != DMTFLEN ||
|
|
bstrWbemFormat[DECPOS] != L'.'
|
|
)
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_FORMAT);
|
|
return *this;
|
|
}
|
|
|
|
if ( (wcslen(bstrWbemFormat) == DMTFLEN) &&
|
|
(bstrWbemFormat[SGNPOS] == L'+' || bstrWbemFormat[SGNPOS] == L'-') )
|
|
{
|
|
SetDMTF(bstrWbemFormat);
|
|
}
|
|
else
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_FORMAT);
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::operator=(const SYSTEMTIME)
|
|
//
|
|
// Description: Assignment operator which is also used by the constructor.
|
|
// This takes a standard WIN32 SYSTEMTIME stucture.
|
|
//
|
|
// Return: WBEMTime object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
const WBEMTime & WBEMTime::operator=(const SYSTEMTIME & st)
|
|
{
|
|
Clear(); // set when properly assigned
|
|
FILETIME t_ft;
|
|
|
|
if ( SystemTimeToFileTime(&st, &t_ft) )
|
|
{
|
|
// now assign using a FILETIME.
|
|
*this = t_ft;
|
|
}
|
|
else
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_FORMAT);
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::operator=(const FILETIME)
|
|
//
|
|
// Description: Assignment operator which is also used by the constructor.
|
|
// This takes a standard WIN32 FILETIME stucture.
|
|
//
|
|
// Return: WBEMTime object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
const WBEMTime & WBEMTime::operator=(const FILETIME & ft)
|
|
{
|
|
FileTimeToui64(&ft, &m_uTime);
|
|
return *this;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::operator=(struct tm tmin)
|
|
//
|
|
// Description: Assignment operator which is also used by the constructor.
|
|
// This takes a standard c runtine struct tm stucture.
|
|
//
|
|
// Return: WBEMTime object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
const WBEMTime & WBEMTime::operator=(const struct tm &a_tmin)
|
|
{
|
|
Clear(); // set when properly assigned
|
|
|
|
SYSTEMTIME systemTime;
|
|
if (StructtmToSystemTime(&a_tmin, &systemTime))
|
|
{
|
|
*this = systemTime;
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::operator=(struct time_t t)
|
|
//
|
|
// Description: Assignment operator which is also used by the constructor.
|
|
// This takes a standard c runtine time_t stucture.
|
|
//
|
|
// Return: WBEMTime object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
const WBEMTime & WBEMTime::operator=(const time_t & t)
|
|
{
|
|
if (t >= 0)
|
|
{
|
|
m_uTime = Int32x32To64(t, 10000000) + 116444736000000000;
|
|
}
|
|
else
|
|
{
|
|
Clear();
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::operator+(const WBEMTime &uAdd)
|
|
//
|
|
// Description: dummy function for adding two WBEMTime. It doesnt really
|
|
// make sense to add two date, but this is here for Tomas's template.
|
|
//
|
|
// Return: WBEMTime object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
WBEMTime WBEMTime::operator+(const WBEMTimeSpan &uAdd) const
|
|
{
|
|
WBEMTime ret;
|
|
|
|
if (IsOk() && uAdd.IsOk())
|
|
{
|
|
ret.m_uTime = m_uTime + uAdd.m_Time;
|
|
}
|
|
else
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
const WBEMTime &WBEMTime::operator+=( const WBEMTimeSpan &ts )
|
|
{
|
|
if (IsOk() && ts.IsOk())
|
|
{
|
|
m_uTime += ts.m_Time ;
|
|
}
|
|
else
|
|
{
|
|
Clear();
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
}
|
|
|
|
return *this ;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::operator-(const WBEMTime & sub)
|
|
//
|
|
// Description: returns a WBEMTimeSpan object as the difference between
|
|
// two WBEMTime objects.
|
|
//
|
|
// Return: WBEMTimeSpan object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
WBEMTimeSpan WBEMTime::operator-(const WBEMTime & sub)
|
|
{
|
|
WBEMTimeSpan ret;
|
|
|
|
if (IsOk() && sub.IsOk() && (m_uTime >= sub.m_uTime))
|
|
{
|
|
ret.m_Time = m_uTime-sub.m_uTime;
|
|
}
|
|
else
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
WBEMTime WBEMTime::operator-(const WBEMTimeSpan & sub) const
|
|
{
|
|
WBEMTime ret;
|
|
|
|
if (IsOk() && sub.IsOk() && (m_uTime >= sub.m_Time))
|
|
{
|
|
ret.m_uTime = m_uTime - sub.m_Time;
|
|
}
|
|
else
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
const WBEMTime &WBEMTime::operator-=(const WBEMTimeSpan & sub)
|
|
{
|
|
if (IsOk() && sub.IsOk() && (m_uTime >= sub.m_Time))
|
|
{
|
|
m_uTime -= sub.m_Time;
|
|
}
|
|
else
|
|
{
|
|
Clear();
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::GetBSTR(void)
|
|
//
|
|
// This function used to CLAIM to do this:
|
|
//
|
|
// WRONG Description: Converts the time which is stored as the number of
|
|
// nano seconds since 1970 into a bstr with this format.
|
|
// YYYYMMDDHHSS.123456789 So 3:04 am, 1/1/96 would be 199601010304.000000000
|
|
//
|
|
// What it really did was return some bastardized form of a dmtf string. Now
|
|
// it returns a dmtf string in gmt form (which is what the docs claim).
|
|
//
|
|
// Return: BSTR representation of time, or NULL if error. Note that the
|
|
// caller should free up this string!
|
|
//
|
|
//***************************************************************************
|
|
|
|
BSTR WBEMTime::GetBSTR(void) const
|
|
{
|
|
return GetDMTF(false) ;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::GetDMTFNonNtfs(void)
|
|
//
|
|
//***************************************************************************
|
|
|
|
BSTR WBEMTime::GetDMTFNonNtfs(void) const
|
|
{
|
|
FILETIME t_ft1, t_ft2;
|
|
BSTR t_Date = NULL;
|
|
|
|
if (GetFILETIME(&t_ft1) && FileTimeToLocalFileTime(&t_ft1, &t_ft2))
|
|
{
|
|
t_Date = WBEMTime(t_ft2).GetDMTF();
|
|
|
|
if (t_Date != NULL)
|
|
{
|
|
t_Date[21] = L'+';
|
|
t_Date[22] = L'*';
|
|
t_Date[23] = L'*';
|
|
t_Date[24] = L'*';
|
|
}
|
|
}
|
|
|
|
return t_Date;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::time_t(time_t * ptm)
|
|
//
|
|
// Return: TRUE if OK.
|
|
//
|
|
//***************************************************************************
|
|
|
|
BOOL WBEMTime::Gettime_t(time_t * ptm) const
|
|
{
|
|
if( (!IsOk()) || (ptm == NULL) || (*this < g_Jan1970))
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
return FALSE;
|
|
}
|
|
|
|
if (g_Jan1970 != *this)
|
|
{
|
|
LONGLONG t_tmp = ( (m_uTime - g_Jan1970.m_uTime) / 10000000);
|
|
|
|
if (t_tmp <= (LONGLONG)0xffffffff)
|
|
{
|
|
*ptm = (time_t)t_tmp;
|
|
}
|
|
else
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
return FALSE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
*ptm = 0;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::GetStructtm(struct tm * ptm)
|
|
//
|
|
// Return: TRUE if OK.
|
|
//
|
|
//***************************************************************************
|
|
|
|
BOOL WBEMTime::GetStructtm(struct tm * ptm) const
|
|
{
|
|
SYSTEMTIME systemTime;
|
|
|
|
return (GetSYSTEMTIME(&systemTime) && SystemTimeToStructtm(&systemTime, ptm));
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::GetSYSTEMTIME(SYSTEMTIME * pst)
|
|
//
|
|
// Return: TRUE if OK.
|
|
//
|
|
//***************************************************************************
|
|
|
|
BOOL WBEMTime::GetSYSTEMTIME(SYSTEMTIME * pst) const
|
|
{
|
|
if ((pst == NULL) || (!IsOk()))
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
return FALSE;
|
|
}
|
|
|
|
FILETIME t_ft;
|
|
|
|
if (GetFILETIME(&t_ft))
|
|
{
|
|
if (!FileTimeToSystemTime(&t_ft, pst))
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
return FALSE;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTime::GetFILETIME(FILETIME * pst)
|
|
//
|
|
// Return: TRUE if OK.
|
|
//
|
|
//***************************************************************************
|
|
|
|
BOOL WBEMTime::GetFILETIME(FILETIME * pft) const
|
|
{
|
|
if ((pft == NULL) || (!IsOk()))
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
return FALSE;
|
|
}
|
|
|
|
ui64ToFileTime(&m_uTime, pft);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// CWbemTime::SetDMTF(BSTR wszText)
|
|
//
|
|
// Description: Sets the time value to the DMTF string datetime value
|
|
// passed as the parameter
|
|
//
|
|
// Return: TRUE if OK.
|
|
//
|
|
//***************************************************************************
|
|
BOOL WBEMTime::SetDMTF( const BSTR a_wszText )
|
|
{
|
|
|
|
wchar_t t_DefaultBuffer[] = {L"16010101000000.000000+000"} ;
|
|
wchar_t t_DateBuffer[ DMTFLEN + 1 ] ;
|
|
t_DateBuffer[ DMTFLEN ] = NULL ;
|
|
|
|
bstr_t t_bstrDate( a_wszText ) ;
|
|
|
|
// wildcard cleanup and validation
|
|
// ===============================
|
|
|
|
if( DMTFLEN != t_bstrDate.length() )
|
|
{
|
|
ASSERT_BREAK( INVALID_TIME_FORMAT ) ;
|
|
return FALSE ;
|
|
}
|
|
|
|
wchar_t *t_pwBuffer = (wchar_t*)t_bstrDate ;
|
|
|
|
for( int t_i = 0; t_i < DMTFLEN; t_i++ )
|
|
{
|
|
switch( t_pwBuffer[ t_i ] )
|
|
{
|
|
case '0':
|
|
case '1':
|
|
case '2':
|
|
case '3':
|
|
case '4':
|
|
case '5':
|
|
case '6':
|
|
case '7':
|
|
case '8':
|
|
case '9':
|
|
{
|
|
// stepping on separator or sign
|
|
if( DECPOS == t_i || SGNPOS == t_i )
|
|
{
|
|
ASSERT_BREAK( INVALID_TIME_FORMAT ) ;
|
|
return FALSE ;
|
|
}
|
|
t_DateBuffer[ t_i ] = t_pwBuffer[ t_i ] ;
|
|
|
|
break ;
|
|
}
|
|
case '*':
|
|
{
|
|
// stepping on separator or sign
|
|
if( DECPOS == t_i || SGNPOS == t_i )
|
|
{
|
|
ASSERT_BREAK( INVALID_TIME_FORMAT ) ;
|
|
return FALSE ;
|
|
}
|
|
else
|
|
{
|
|
// replace with default stamp
|
|
t_DateBuffer[ t_i ] = t_DefaultBuffer[ t_i ] ;
|
|
}
|
|
break ;
|
|
}
|
|
case '.':
|
|
{
|
|
if( DECPOS != t_i )
|
|
{
|
|
ASSERT_BREAK( INVALID_TIME_FORMAT ) ;
|
|
return FALSE ;
|
|
}
|
|
t_DateBuffer[ t_i ] = t_pwBuffer[ t_i ] ;
|
|
|
|
break ;
|
|
}
|
|
case '+':
|
|
case '-':
|
|
{
|
|
if( SGNPOS != t_i )
|
|
{
|
|
ASSERT_BREAK( INVALID_TIME_FORMAT ) ;
|
|
return FALSE ;
|
|
}
|
|
t_DateBuffer[ t_i ] = t_pwBuffer[ t_i ] ;
|
|
break ;
|
|
}
|
|
default:
|
|
{
|
|
ASSERT_BREAK( INVALID_TIME_FORMAT ) ;
|
|
return FALSE ;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Parse it
|
|
// ========
|
|
|
|
int nYear, nMonth, nDay, nHour, nMinute, nSecond, nMicro, nOffset;
|
|
WCHAR wchSep;
|
|
|
|
int nRes = swscanf (
|
|
|
|
(LPCWSTR)&t_DateBuffer,
|
|
L"%4d%2d%2d%2d%2d%2d.%6d%c%3d",
|
|
&nYear,
|
|
&nMonth,
|
|
&nDay,
|
|
&nHour,
|
|
&nMinute,
|
|
&nSecond,
|
|
&nMicro,
|
|
&wchSep,
|
|
&nOffset
|
|
);
|
|
|
|
if ( ( 9 != nRes ) || ( 1601 > nYear) )
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_FORMAT);
|
|
return FALSE;
|
|
}
|
|
|
|
// Convert it to SYSTEMTIME
|
|
// ========================
|
|
|
|
SYSTEMTIME st;
|
|
st.wYear = (WORD)nYear;
|
|
st.wMonth = (WORD)nMonth;
|
|
st.wDay = (WORD)nDay;
|
|
st.wHour = (WORD)nHour;
|
|
st.wMinute = (WORD)nMinute;
|
|
st.wSecond = (WORD)nSecond;
|
|
st.wMilliseconds = 0;
|
|
st.wDayOfWeek = 0;
|
|
|
|
*this = st;
|
|
|
|
// we need to add microseconds and nanoseconds now!
|
|
// it was basically cut when assigned to SYSTEMTIME
|
|
// ================================================
|
|
m_uTime += (LONGLONG)nMicro * 10;
|
|
|
|
// NOW we adjust for the offset
|
|
// ============================
|
|
|
|
if ( IsOk() )
|
|
{
|
|
int nSign = (wchSep == L'+') ? 1 : -1 ;
|
|
|
|
m_uTime -= (LONGLONG)nSign * (LONGLONG)nOffset * 60 * 10000000;
|
|
}
|
|
else
|
|
{
|
|
ASSERT_BREAK( INVALID_TIME_ARITHMETIC ) ;
|
|
return FALSE ;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// BSTR WBEMTime::GetDMTF(void)
|
|
//
|
|
// Description: Gets the time in DMTF string datetime format. User must call
|
|
// SysFreeString with the result. If bLocal is true, then the time is given
|
|
// in the local timezone, else the time is given in GMT.
|
|
//
|
|
// Return: NULL if not OK.
|
|
//
|
|
//***************************************************************************
|
|
|
|
|
|
BSTR WBEMTime::GetDMTF(BOOL bLocal) const
|
|
{
|
|
|
|
if (!IsOk())
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
return NULL;
|
|
}
|
|
|
|
SYSTEMTIME t_Systime;
|
|
wchar_t chsign = L'-';
|
|
int offset = 0;
|
|
|
|
// If the date to be converted is within 12 hours of
|
|
// 1/1/1601, return the greenwich time
|
|
ULONGLONG t_ConversionZone = 12L * 60L * 60L ;
|
|
t_ConversionZone = t_ConversionZone * 10000000L ;
|
|
if ( !bLocal || ( m_uTime < t_ConversionZone ) )
|
|
{
|
|
if(!GetSYSTEMTIME(&t_Systime))
|
|
{
|
|
return NULL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (GetSYSTEMTIME(&t_Systime))
|
|
{
|
|
offset = GetLocalOffsetForDate(&t_Systime);
|
|
|
|
WBEMTime wt;
|
|
if (offset >= 0)
|
|
{
|
|
chsign = '+';
|
|
wt = *this + WBEMTimeSpan(0, 0, offset, 0);
|
|
}
|
|
else
|
|
{
|
|
offset *= -1;
|
|
wt = *this - WBEMTimeSpan(0, 0, offset, 0);
|
|
}
|
|
wt.GetSYSTEMTIME(&t_Systime);
|
|
}
|
|
else
|
|
{
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
LONGLONG tmpMicros = m_uTime%10000000;
|
|
LONG micros = (LONG)(tmpMicros / 10);
|
|
|
|
BSTR t_String = SysAllocStringLen(NULL, DMTFLEN + 1);
|
|
if ( ! t_String )
|
|
{
|
|
throw CHeap_Exception ( CHeap_Exception :: E_ALLOCATION_ERROR ) ;
|
|
}
|
|
|
|
swprintf(
|
|
|
|
t_String,
|
|
L"%04.4d%02.2d%02.2d%02.2d%02.2d%02.2d.%06.6d%c%03.3ld",
|
|
t_Systime.wYear,
|
|
t_Systime.wMonth,
|
|
t_Systime.wDay,
|
|
t_Systime.wHour,
|
|
t_Systime.wMinute,
|
|
t_Systime.wSecond,
|
|
micros,
|
|
chsign,
|
|
offset
|
|
);
|
|
|
|
return t_String ;
|
|
|
|
}
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
// WBEMTimeSpan - This class holds timespan values. The data is stored
|
|
// in 100 nanosecond units (like FILETIME).
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTimeSpan::WBEMTimeSpan(int iDays, int iHours, int iMinutes, int iSeconds,
|
|
// int iMSec, int iUSec, int iNSec)
|
|
//
|
|
// Description: Constructor.
|
|
//
|
|
//***************************************************************************
|
|
|
|
WBEMTimeSpan::WBEMTimeSpan(int iDays, int iHours, int iMinutes, int iSeconds,
|
|
int iMSec, int iUSec, int iNSec)
|
|
{
|
|
m_Time = 0; //todo, check values!!!
|
|
m_Time += iSeconds;
|
|
m_Time += iMinutes * 60;
|
|
m_Time += iHours * 60 * 60;
|
|
m_Time += iDays * 24 * 60 * 60;
|
|
m_Time *= 10000000;
|
|
m_Time += iNSec / 100; // Nanoseconds
|
|
m_Time += iUSec*10; // Microseconds
|
|
m_Time += iMSec*10000; // Milliseconds
|
|
}
|
|
|
|
WBEMTimeSpan::WBEMTimeSpan ( const FILETIME &ft )
|
|
{
|
|
ASSERT_BREAK(DEPRECATED);
|
|
*this = ft ;
|
|
}
|
|
|
|
WBEMTimeSpan::WBEMTimeSpan ( const time_t & t )
|
|
{
|
|
ASSERT_BREAK(DEPRECATED);
|
|
*this = t ;
|
|
} ;
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTimeSpan::operator=(const BSTR bstrWbemFormat)
|
|
//
|
|
// Return: WBEMTimeSpan object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
const WBEMTimeSpan & WBEMTimeSpan::operator=(const BSTR bstrWbemFormat)
|
|
{
|
|
Clear();
|
|
|
|
// all characters should be digits except for one which
|
|
// must be a period
|
|
|
|
if ((bstrWbemFormat == NULL) || (bstrWbemFormat[DECPOS] != L'.') ||
|
|
(wcslen(bstrWbemFormat) != DMTFLEN) || (bstrWbemFormat[SGNPOS] != L':') )
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_FORMAT);
|
|
return *this;
|
|
}
|
|
|
|
int nDays, nHours, nMinutes, nSeconds, nMicros, nOffset;
|
|
WCHAR wchSep;
|
|
|
|
int nRes = swscanf (
|
|
|
|
bstrWbemFormat,
|
|
L"%8d%2d%2d%2d.%6d%c%3d",
|
|
&nDays,
|
|
&nHours,
|
|
&nMinutes,
|
|
&nSeconds,
|
|
&nMicros,
|
|
&wchSep,
|
|
&nOffset
|
|
);
|
|
|
|
if ( (nRes != 7) || ( nOffset != 0) )
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_FORMAT);
|
|
return *this;
|
|
}
|
|
|
|
*this = WBEMTimeSpan(nDays, nHours, nMinutes, nSeconds, 0, nMicros, 0);
|
|
|
|
return *this;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTimeSpan::operator=(const FILETIME &)
|
|
// WBEMTimeSpan::operator=(const time_t &)
|
|
//
|
|
// Description: Assignment operator which is also used by the constructor.
|
|
//
|
|
// Return: WBEMTimeSpan object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
const WBEMTimeSpan & WBEMTimeSpan::operator=(const FILETIME &ft)
|
|
{
|
|
ASSERT_BREAK(DEPRECATED);
|
|
|
|
ULONGLONG uTemp;
|
|
FileTimeToui64(&ft, &uTemp);
|
|
m_Time = uTemp;
|
|
|
|
return *this;
|
|
}
|
|
|
|
const WBEMTimeSpan & WBEMTimeSpan::operator=(const time_t & t)
|
|
{
|
|
ASSERT_BREAK(DEPRECATED);
|
|
|
|
ULONGLONG uTemp = 0;
|
|
|
|
uTemp = t;
|
|
if (t >= 0)
|
|
{
|
|
m_Time = uTemp * 10000000;
|
|
}
|
|
else
|
|
{
|
|
Clear();
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTimeSpan::operator +(const WBEMTimeSpan &uAdd)
|
|
//
|
|
// Description: function for adding two WBEMTimeSpan objects.
|
|
//
|
|
// Return: WBEMTimeSpan object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
WBEMTimeSpan WBEMTimeSpan::operator+(const WBEMTimeSpan &uAdd) const
|
|
{
|
|
WBEMTimeSpan ret;
|
|
|
|
if (IsOk() && uAdd.IsOk())
|
|
{
|
|
ret.m_Time = m_Time + uAdd.m_Time;
|
|
}
|
|
else
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
const WBEMTimeSpan &WBEMTimeSpan::operator+= ( const WBEMTimeSpan &uAdd )
|
|
{
|
|
if (IsOk() && uAdd.IsOk())
|
|
{
|
|
m_Time += uAdd.m_Time ;
|
|
}
|
|
else
|
|
{
|
|
Clear();
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
}
|
|
|
|
return *this ;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTimeSpan::operator -(const WBEMTimeSpan &uAdd)
|
|
//
|
|
// Description: function for adding two WBEMTimeSpan objects.
|
|
//
|
|
// Return: WBEMTimeSpan object.
|
|
//
|
|
//***************************************************************************
|
|
|
|
WBEMTimeSpan WBEMTimeSpan::operator-(const WBEMTimeSpan &uSub) const
|
|
{
|
|
WBEMTimeSpan ret;
|
|
|
|
if (IsOk() && uSub.IsOk() && (m_Time >= uSub.m_Time))
|
|
{
|
|
ret.m_Time = m_Time - uSub.m_Time;
|
|
}
|
|
else
|
|
{
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
const WBEMTimeSpan &WBEMTimeSpan::operator-= ( const WBEMTimeSpan &uSub )
|
|
{
|
|
if (IsOk() && uSub.IsOk() && (m_Time >= uSub.m_Time))
|
|
{
|
|
m_Time -= uSub.m_Time;
|
|
}
|
|
else
|
|
{
|
|
Clear();
|
|
ASSERT_BREAK(INVALID_TIME_ARITHMETIC);
|
|
}
|
|
|
|
return *this;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTimeSpan::GetBSTR(void)
|
|
//
|
|
// Description: Converts the time which is stored as the number of
|
|
// 100 nano second units into a dmtf formatted string
|
|
// ddddddddhhmmss.mmmmmm:000
|
|
//
|
|
// Return: BSTR representation of time, or NULL if error. Note that the
|
|
// caller should free up this string!
|
|
//
|
|
//***************************************************************************
|
|
|
|
BSTR WBEMTimeSpan::GetBSTR(void) const
|
|
{
|
|
if(!IsOk())
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
ULONGLONG Time = m_Time;
|
|
|
|
// The /10 is to convert from 100ns to microseconds
|
|
long iMicro = (long)((Time % 10000000) / 10);
|
|
Time /= 10000000;
|
|
int iSec = (int)(Time % 60);
|
|
Time /= 60;
|
|
int iMin = (int)(Time % 60);
|
|
Time /= 60;
|
|
int iHour = (int)(Time % 24);
|
|
Time /= 24;
|
|
|
|
BSTR t_String = SysAllocStringLen(NULL, DMTFLEN + 1);
|
|
if ( ! t_String )
|
|
{
|
|
throw CHeap_Exception ( CHeap_Exception :: E_ALLOCATION_ERROR ) ;
|
|
}
|
|
|
|
swprintf(t_String, L"%08I64i%02d%02d%02d.%06ld:000",
|
|
Time, iHour, iMin, iSec, iMicro);
|
|
|
|
return t_String ;
|
|
}
|
|
|
|
//***************************************************************************
|
|
//
|
|
// WBEMTimeSpan::Gettime_t(void)
|
|
// WBEMTimeSpan::GetFILETIME(void)
|
|
//
|
|
// Description: Converts the time span which is stored as the number of
|
|
// nano seconds into common stuctures.
|
|
//
|
|
// Return: TRUE if OK.
|
|
//
|
|
//***************************************************************************
|
|
|
|
BOOL WBEMTimeSpan::Gettime_t(time_t * ptime_t) const
|
|
{
|
|
ASSERT_BREAK(DEPRECATED);
|
|
|
|
if(!IsOk())
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
*ptime_t = (DWORD)(m_Time / 10000000);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
BOOL WBEMTimeSpan::GetFILETIME(FILETIME * pst) const
|
|
{
|
|
ASSERT_BREAK(DEPRECATED);
|
|
|
|
if(!IsOk())
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
ULONGLONG uTemp;
|
|
uTemp = m_Time;
|
|
ui64ToFileTime(&uTemp,pst);
|
|
return TRUE;
|
|
}
|