| /* |
| * Win32 kernel time functions |
| * |
| * Copyright 1995 Martin von Loewis and Cameron Heide |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Lesser General Public |
| * License as published by the Free Software Foundation; either |
| * version 2.1 of the License, or (at your option) any later version. |
| * |
| * This library 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 |
| * Lesser General Public License for more details. |
| * |
| * You should have received a copy of the GNU Lesser General Public |
| * License along with this library; if not, write to the Free Software |
| * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA |
| */ |
| |
| #include "config.h" |
| |
| #include <string.h> |
| #ifdef HAVE_UNISTD_H |
| # include <unistd.h> |
| #endif |
| #include <stdarg.h> |
| #include <stdlib.h> |
| #include <time.h> |
| #ifdef HAVE_SYS_TIME_H |
| # include <sys/time.h> |
| #endif |
| #ifdef HAVE_SYS_TIMES_H |
| # include <sys/times.h> |
| #endif |
| #ifdef HAVE_SYS_LIMITS_H |
| #include <sys/limits.h> |
| #elif defined(HAVE_MACHINE_LIMITS_H) |
| #include <machine/limits.h> |
| #endif |
| |
| #define NONAMELESSUNION |
| #define NONAMELESSSTRUCT |
| #include "ntstatus.h" |
| #define WIN32_NO_STATUS |
| #include "windef.h" |
| #include "winbase.h" |
| #include "winternl.h" |
| #include "kernel_private.h" |
| #include "wine/unicode.h" |
| #include "wine/debug.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(time); |
| |
| /* maximum time adjustment in seconds for SetLocalTime and SetSystemTime */ |
| #define SETTIME_MAX_ADJUST 120 |
| #define CALINFO_MAX_YEAR 2029 |
| |
| #define LL2FILETIME( ll, pft )\ |
| (pft)->dwLowDateTime = (UINT)(ll); \ |
| (pft)->dwHighDateTime = (UINT)((ll) >> 32); |
| #define FILETIME2LL( pft, ll) \ |
| ll = (((LONGLONG)((pft)->dwHighDateTime))<<32) + (pft)-> dwLowDateTime ; |
| |
| |
| static const int MonthLengths[2][12] = |
| { |
| { 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 }, |
| { 31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 } |
| }; |
| |
| static inline int IsLeapYear(int Year) |
| { |
| return Year % 4 == 0 && (Year % 100 != 0 || Year % 400 == 0) ? 1 : 0; |
| } |
| |
| /*********************************************************************** |
| * TIME_DayLightCompareDate |
| * |
| * Compares two dates without looking at the year. |
| * |
| * PARAMS |
| * date [in] The local time to compare. |
| * compareDate [in] The daylight saving begin or end date. |
| * |
| * RETURNS |
| * |
| * -1 if date < compareDate |
| * 0 if date == compareDate |
| * 1 if date > compareDate |
| * -2 if an error occurs |
| */ |
| static int TIME_DayLightCompareDate( const SYSTEMTIME *date, |
| const SYSTEMTIME *compareDate ) |
| { |
| int limit_day, dayinsecs; |
| |
| if (date->wMonth < compareDate->wMonth) |
| return -1; /* We are in a month before the date limit. */ |
| |
| if (date->wMonth > compareDate->wMonth) |
| return 1; /* We are in a month after the date limit. */ |
| |
| if (compareDate->wDayOfWeek <= 6) |
| { |
| WORD First; |
| /* compareDate->wDay is interpreted as number of the week in the month |
| * 5 means: the last week in the month */ |
| int weekofmonth = compareDate->wDay; |
| /* calculate the day of the first DayOfWeek in the month */ |
| First = ( 6 + compareDate->wDayOfWeek - date->wDayOfWeek + date->wDay |
| ) % 7 + 1; |
| limit_day = First + 7 * (weekofmonth - 1); |
| /* check needed for the 5th weekday of the month */ |
| if(limit_day > MonthLengths[date->wMonth==2 && IsLeapYear(date->wYear)] |
| [date->wMonth - 1]) |
| limit_day -= 7; |
| } |
| else |
| { |
| limit_day = compareDate->wDay; |
| } |
| |
| /* convert to seconds */ |
| limit_day = ((limit_day * 24 + compareDate->wHour) * 60 + |
| compareDate->wMinute ) * 60; |
| dayinsecs = ((date->wDay * 24 + date->wHour) * 60 + |
| date->wMinute ) * 60 + date->wSecond; |
| /* and compare */ |
| return dayinsecs < limit_day ? -1 : |
| dayinsecs > limit_day ? 1 : |
| 0; /* date is equal to the date limit. */ |
| } |
| |
| /*********************************************************************** |
| * TIME_CompTimeZoneID |
| * |
| * Computes the local time bias for a given time and time zone. |
| * |
| * PARAMS |
| * pTZinfo [in] The time zone data. |
| * lpFileTime [in] The system or local time. |
| * islocal [in] it is local time. |
| * |
| * RETURNS |
| * TIME_ZONE_ID_INVALID An error occurred |
| * TIME_ZONE_ID_UNKNOWN There are no transition time known |
| * TIME_ZONE_ID_STANDARD Current time is standard time |
| * TIME_ZONE_ID_DAYLIGHT Current time is dayligh saving time |
| */ |
| static BOOL TIME_CompTimeZoneID ( const TIME_ZONE_INFORMATION *pTZinfo, |
| FILETIME *lpFileTime, BOOL islocal ) |
| { |
| int ret; |
| BOOL beforeStandardDate, afterDaylightDate; |
| DWORD retval = TIME_ZONE_ID_INVALID; |
| LONGLONG llTime = 0; /* initialized to prevent gcc complaining */ |
| SYSTEMTIME SysTime; |
| FILETIME ftTemp; |
| |
| if (pTZinfo->DaylightDate.wMonth != 0) |
| { |
| if (pTZinfo->StandardDate.wMonth == 0 || |
| pTZinfo->StandardDate.wDay<1 || |
| pTZinfo->StandardDate.wDay>5 || |
| pTZinfo->DaylightDate.wDay<1 || |
| pTZinfo->DaylightDate.wDay>5) |
| { |
| SetLastError(ERROR_INVALID_PARAMETER); |
| return TIME_ZONE_ID_INVALID; |
| } |
| |
| if (!islocal) { |
| FILETIME2LL( lpFileTime, llTime ); |
| llTime -= ( pTZinfo->Bias + pTZinfo->DaylightBias ) |
| * (LONGLONG)600000000; |
| LL2FILETIME( llTime, &ftTemp) |
| lpFileTime = &ftTemp; |
| } |
| |
| FileTimeToSystemTime(lpFileTime, &SysTime); |
| |
| /* check for daylight saving */ |
| ret = TIME_DayLightCompareDate( &SysTime, &pTZinfo->StandardDate); |
| if (ret == -2) |
| return TIME_ZONE_ID_INVALID; |
| |
| beforeStandardDate = ret < 0; |
| |
| if (!islocal) { |
| llTime -= ( pTZinfo->StandardBias - pTZinfo->DaylightBias ) |
| * (LONGLONG)600000000; |
| LL2FILETIME( llTime, &ftTemp) |
| FileTimeToSystemTime(lpFileTime, &SysTime); |
| } |
| |
| ret = TIME_DayLightCompareDate( &SysTime, &pTZinfo->DaylightDate); |
| if (ret == -2) |
| return TIME_ZONE_ID_INVALID; |
| |
| afterDaylightDate = ret >= 0; |
| |
| retval = TIME_ZONE_ID_STANDARD; |
| if( pTZinfo->DaylightDate.wMonth < pTZinfo->StandardDate.wMonth ) { |
| /* Northern hemisphere */ |
| if( beforeStandardDate && afterDaylightDate ) |
| retval = TIME_ZONE_ID_DAYLIGHT; |
| } else /* Down south */ |
| if( beforeStandardDate || afterDaylightDate ) |
| retval = TIME_ZONE_ID_DAYLIGHT; |
| } else |
| /* No transition date */ |
| retval = TIME_ZONE_ID_UNKNOWN; |
| |
| return retval; |
| } |
| |
| /*********************************************************************** |
| * TIME_TimeZoneID |
| * |
| * Calculates whether daylight saving is on now. |
| * |
| * PARAMS |
| * pTzi [in] Timezone info. |
| * |
| * RETURNS |
| * TIME_ZONE_ID_INVALID An error occurred |
| * TIME_ZONE_ID_UNKNOWN There are no transition time known |
| * TIME_ZONE_ID_STANDARD Current time is standard time |
| * TIME_ZONE_ID_DAYLIGHT Current time is dayligh saving time |
| */ |
| static DWORD TIME_ZoneID( const TIME_ZONE_INFORMATION *pTzi ) |
| { |
| FILETIME ftTime; |
| GetSystemTimeAsFileTime( &ftTime); |
| return TIME_CompTimeZoneID( pTzi, &ftTime, FALSE); |
| } |
| |
| /*********************************************************************** |
| * TIME_GetTimezoneBias |
| * |
| * Calculates the local time bias for a given time zone. |
| * |
| * PARAMS |
| * pTZinfo [in] The time zone data. |
| * lpFileTime [in] The system or local time. |
| * islocal [in] It is local time. |
| * pBias [out] The calculated bias in minutes. |
| * |
| * RETURNS |
| * TRUE when the time zone bias was calculated. |
| */ |
| static BOOL TIME_GetTimezoneBias( const TIME_ZONE_INFORMATION *pTZinfo, |
| FILETIME *lpFileTime, BOOL islocal, LONG *pBias ) |
| { |
| LONG bias = pTZinfo->Bias; |
| DWORD tzid = TIME_CompTimeZoneID( pTZinfo, lpFileTime, islocal); |
| |
| if( tzid == TIME_ZONE_ID_INVALID) |
| return FALSE; |
| if (tzid == TIME_ZONE_ID_DAYLIGHT) |
| bias += pTZinfo->DaylightBias; |
| else if (tzid == TIME_ZONE_ID_STANDARD) |
| bias += pTZinfo->StandardBias; |
| *pBias = bias; |
| return TRUE; |
| } |
| |
| |
| /*********************************************************************** |
| * SetLocalTime (KERNEL32.@) |
| * |
| * Set the local time using current time zone and daylight |
| * savings settings. |
| * |
| * PARAMS |
| * systime [in] The desired local time. |
| * |
| * RETURNS |
| * Success: TRUE. The time was set. |
| * Failure: FALSE, if the time was invalid or caller does not have |
| * permission to change the time. |
| */ |
| BOOL WINAPI SetLocalTime( const SYSTEMTIME *systime ) |
| { |
| FILETIME ft; |
| LARGE_INTEGER st, st2; |
| NTSTATUS status; |
| |
| if( !SystemTimeToFileTime( systime, &ft )) |
| return FALSE; |
| st.u.LowPart = ft.dwLowDateTime; |
| st.u.HighPart = ft.dwHighDateTime; |
| RtlLocalTimeToSystemTime( &st, &st2 ); |
| |
| if ((status = NtSetSystemTime(&st2, NULL))) |
| SetLastError( RtlNtStatusToDosError(status) ); |
| return !status; |
| } |
| |
| |
| /*********************************************************************** |
| * GetSystemTimeAdjustment (KERNEL32.@) |
| * |
| * Get the period between clock interrupts and the amount the clock |
| * is adjusted each interrupt so as to keep it in sync with an external source. |
| * |
| * PARAMS |
| * lpTimeAdjustment [out] The clock adjustment per interrupt in 100's of nanoseconds. |
| * lpTimeIncrement [out] The time between clock interrupts in 100's of nanoseconds. |
| * lpTimeAdjustmentDisabled [out] The clock synchronisation has been disabled. |
| * |
| * RETURNS |
| * TRUE. |
| * |
| * BUGS |
| * Only the special case of disabled time adjustments is supported. |
| */ |
| BOOL WINAPI GetSystemTimeAdjustment( PDWORD lpTimeAdjustment, PDWORD lpTimeIncrement, |
| PBOOL lpTimeAdjustmentDisabled ) |
| { |
| *lpTimeAdjustment = 0; |
| *lpTimeIncrement = 0; |
| *lpTimeAdjustmentDisabled = TRUE; |
| return TRUE; |
| } |
| |
| |
| /*********************************************************************** |
| * SetSystemTime (KERNEL32.@) |
| * |
| * Set the system time in utc. |
| * |
| * PARAMS |
| * systime [in] The desired system time. |
| * |
| * RETURNS |
| * Success: TRUE. The time was set. |
| * Failure: FALSE, if the time was invalid or caller does not have |
| * permission to change the time. |
| */ |
| BOOL WINAPI SetSystemTime( const SYSTEMTIME *systime ) |
| { |
| FILETIME ft; |
| LARGE_INTEGER t; |
| NTSTATUS status; |
| |
| if( !SystemTimeToFileTime( systime, &ft )) |
| return FALSE; |
| t.u.LowPart = ft.dwLowDateTime; |
| t.u.HighPart = ft.dwHighDateTime; |
| if ((status = NtSetSystemTime(&t, NULL))) |
| SetLastError( RtlNtStatusToDosError(status) ); |
| return !status; |
| } |
| |
| /*********************************************************************** |
| * SetSystemTimeAdjustment (KERNEL32.@) |
| * |
| * Enables or disables the timing adjustments to the system's clock. |
| * |
| * PARAMS |
| * dwTimeAdjustment [in] Number of units to add per clock interrupt. |
| * bTimeAdjustmentDisabled [in] Adjustment mode. |
| * |
| * RETURNS |
| * Success: TRUE. |
| * Failure: FALSE. |
| */ |
| BOOL WINAPI SetSystemTimeAdjustment( DWORD dwTimeAdjustment, BOOL bTimeAdjustmentDisabled ) |
| { |
| /* Fake function for now... */ |
| FIXME("(%08lx,%d): stub !\n", dwTimeAdjustment, bTimeAdjustmentDisabled); |
| return TRUE; |
| } |
| |
| /*********************************************************************** |
| * GetTimeZoneInformation (KERNEL32.@) |
| * |
| * Get information about the current local time zone. |
| * |
| * PARAMS |
| * tzinfo [out] Destination for time zone information. |
| * |
| * RETURNS |
| * TIME_ZONE_ID_INVALID An error occurred |
| * TIME_ZONE_ID_UNKNOWN There are no transition time known |
| * TIME_ZONE_ID_STANDARD Current time is standard time |
| * TIME_ZONE_ID_DAYLIGHT Current time is dayligh saving time |
| */ |
| DWORD WINAPI GetTimeZoneInformation( LPTIME_ZONE_INFORMATION tzinfo ) |
| { |
| NTSTATUS status; |
| |
| status = RtlQueryTimeZoneInformation( (RTL_TIME_ZONE_INFORMATION*)tzinfo ); |
| if ( status != STATUS_SUCCESS ) |
| { |
| SetLastError( RtlNtStatusToDosError(status) ); |
| return TIME_ZONE_ID_INVALID; |
| } |
| return TIME_ZoneID( tzinfo ); |
| } |
| |
| /*********************************************************************** |
| * SetTimeZoneInformation (KERNEL32.@) |
| * |
| * Change the settings of the current local time zone. |
| * |
| * PARAMS |
| * tzinfo [in] The new time zone. |
| * |
| * RETURNS |
| * Success: TRUE. The time zone was updated with the settings from tzinfo. |
| * Failure: FALSE. |
| */ |
| BOOL WINAPI SetTimeZoneInformation( const TIME_ZONE_INFORMATION *tzinfo ) |
| { |
| NTSTATUS status; |
| status = RtlSetTimeZoneInformation( (RTL_TIME_ZONE_INFORMATION*) tzinfo ); |
| if ( status != STATUS_SUCCESS ) |
| SetLastError( RtlNtStatusToDosError(status) ); |
| return !status; |
| } |
| |
| /*********************************************************************** |
| * SystemTimeToTzSpecificLocalTime (KERNEL32.@) |
| * |
| * Convert a utc system time to a local time in a given time zone. |
| * |
| * PARAMS |
| * lpTimeZoneInformation [in] The desired time zone. |
| * lpUniversalTime [in] The utc time to base local time on. |
| * lpLocalTime [out] The local time in the time zone. |
| * |
| * RETURNS |
| * Success: TRUE. lpLocalTime contains the converted time |
| * Failure: FALSE. |
| */ |
| |
| BOOL WINAPI SystemTimeToTzSpecificLocalTime( |
| LPTIME_ZONE_INFORMATION lpTimeZoneInformation, |
| LPSYSTEMTIME lpUniversalTime, LPSYSTEMTIME lpLocalTime ) |
| { |
| FILETIME ft; |
| LONG lBias; |
| LONGLONG llTime; |
| TIME_ZONE_INFORMATION tzinfo; |
| |
| if (lpTimeZoneInformation != NULL) |
| { |
| memcpy(&tzinfo, lpTimeZoneInformation, sizeof(TIME_ZONE_INFORMATION)); |
| } |
| else |
| { |
| if (GetTimeZoneInformation(&tzinfo) == TIME_ZONE_ID_INVALID) |
| return FALSE; |
| } |
| |
| if (!SystemTimeToFileTime(lpUniversalTime, &ft)) |
| return FALSE; |
| FILETIME2LL( &ft, llTime) |
| if (!TIME_GetTimezoneBias(&tzinfo, &ft, FALSE, &lBias)) |
| return FALSE; |
| /* convert minutes to 100-nanoseconds-ticks */ |
| llTime -= (LONGLONG)lBias * 600000000; |
| LL2FILETIME( llTime, &ft) |
| |
| return FileTimeToSystemTime(&ft, lpLocalTime); |
| } |
| |
| |
| /*********************************************************************** |
| * TzSpecificLocalTimeToSystemTime (KERNEL32.@) |
| * |
| * Converts a local time to a time in utc. |
| * |
| * PARAMS |
| * lpTimeZoneInformation [in] The desired time zone. |
| * lpLocalTime [in] The local time. |
| * lpUniversalTime [out] The calculated utc time. |
| * |
| * RETURNS |
| * Success: TRUE. lpUniversalTime contains the converted time. |
| * Failure: FALSE. |
| */ |
| BOOL WINAPI TzSpecificLocalTimeToSystemTime( |
| LPTIME_ZONE_INFORMATION lpTimeZoneInformation, |
| LPSYSTEMTIME lpLocalTime, LPSYSTEMTIME lpUniversalTime) |
| { |
| FILETIME ft; |
| LONG lBias; |
| LONGLONG t; |
| TIME_ZONE_INFORMATION tzinfo; |
| |
| if (lpTimeZoneInformation != NULL) |
| { |
| memcpy(&tzinfo, lpTimeZoneInformation, sizeof(TIME_ZONE_INFORMATION)); |
| } |
| else |
| { |
| if (GetTimeZoneInformation(&tzinfo) == TIME_ZONE_ID_INVALID) |
| return FALSE; |
| } |
| |
| if (!SystemTimeToFileTime(lpLocalTime, &ft)) |
| return FALSE; |
| FILETIME2LL( &ft, t) |
| if (!TIME_GetTimezoneBias(&tzinfo, &ft, TRUE, &lBias)) |
| return FALSE; |
| /* convert minutes to 100-nanoseconds-ticks */ |
| t += (LONGLONG)lBias * 600000000; |
| LL2FILETIME( t, &ft) |
| return FileTimeToSystemTime(&ft, lpUniversalTime); |
| } |
| |
| |
| /*********************************************************************** |
| * GetSystemTimeAsFileTime (KERNEL32.@) |
| * |
| * Get the current time in utc format. |
| * |
| * RETURNS |
| * Nothing. |
| */ |
| VOID WINAPI GetSystemTimeAsFileTime( |
| LPFILETIME time) /* [out] Destination for the current utc time */ |
| { |
| LARGE_INTEGER t; |
| NtQuerySystemTime( &t ); |
| time->dwLowDateTime = t.u.LowPart; |
| time->dwHighDateTime = t.u.HighPart; |
| } |
| |
| |
| /********************************************************************* |
| * TIME_ClockTimeToFileTime (olorin@fandra.org, 20-Sep-1998) |
| * |
| * Used by GetProcessTimes to convert clock_t into FILETIME. |
| * |
| * Differences to UnixTimeToFileTime: |
| * 1) Divided by CLK_TCK |
| * 2) Time is relative. There is no 'starting date', so there is |
| * no need for offset correction, like in UnixTimeToFileTime |
| */ |
| #ifndef CLK_TCK |
| # define CLK_TCK CLOCKS_PER_SEC |
| #endif |
| static void TIME_ClockTimeToFileTime(clock_t unix_time, LPFILETIME filetime) |
| { |
| ULONGLONG secs = RtlEnlargedUnsignedMultiply( unix_time, 10000000 ); |
| secs = RtlExtendedLargeIntegerDivide( secs, CLK_TCK, NULL ); |
| filetime->dwLowDateTime = (DWORD)secs; |
| filetime->dwHighDateTime = (DWORD)(secs >> 32); |
| } |
| |
| /********************************************************************* |
| * GetProcessTimes (KERNEL32.@) |
| * |
| * Get the user and kernel execution times of a process, |
| * along with the creation and exit times if known. |
| * |
| * PARAMS |
| * hprocess [in] The process to be queried. |
| * lpCreationTime [out] The creation time of the process. |
| * lpExitTime [out] The exit time of the process if exited. |
| * lpKernelTime [out] The time spent in kernel routines in 100's of nanoseconds. |
| * lpUserTime [out] The time spent in user routines in 100's of nanoseconds. |
| * |
| * RETURNS |
| * TRUE. |
| * |
| * NOTES |
| * olorin@fandra.org: |
| * Would be nice to subtract the cpu time used by Wine at startup. |
| * Also, there is a need to separate times used by different applications. |
| * |
| * BUGS |
| * KernelTime and UserTime are always for the current process |
| */ |
| BOOL WINAPI GetProcessTimes( HANDLE hprocess, LPFILETIME lpCreationTime, |
| LPFILETIME lpExitTime, LPFILETIME lpKernelTime, LPFILETIME lpUserTime ) |
| { |
| struct tms tms; |
| KERNEL_USER_TIMES pti; |
| |
| times(&tms); |
| TIME_ClockTimeToFileTime(tms.tms_utime,lpUserTime); |
| TIME_ClockTimeToFileTime(tms.tms_stime,lpKernelTime); |
| if (NtQueryInformationProcess( hprocess, ProcessTimes, &pti, sizeof(pti), NULL)) |
| return FALSE; |
| LL2FILETIME( pti.CreateTime.QuadPart, lpCreationTime); |
| LL2FILETIME( pti.ExitTime.QuadPart, lpExitTime); |
| return TRUE; |
| } |
| |
| /********************************************************************* |
| * GetCalendarInfoA (KERNEL32.@) |
| * |
| */ |
| int WINAPI GetCalendarInfoA(LCID lcid, CALID Calendar, CALTYPE CalType, |
| LPSTR lpCalData, int cchData, LPDWORD lpValue) |
| { |
| int ret; |
| LPWSTR lpCalDataW = NULL; |
| |
| lcid = ConvertDefaultLocale(lcid); |
| |
| if (NLS_IsUnicodeOnlyLcid(lcid)) |
| { |
| SetLastError(ERROR_INVALID_PARAMETER); |
| return 0; |
| } |
| |
| if (cchData && |
| !(lpCalDataW = HeapAlloc(GetProcessHeap(), 0, cchData*sizeof(WCHAR)))) |
| return 0; |
| |
| ret = GetCalendarInfoW(lcid, Calendar, CalType, lpCalDataW, cchData, lpValue); |
| if(ret && lpCalDataW && lpCalData) |
| WideCharToMultiByte(CP_ACP, 0, lpCalDataW, cchData, lpCalData, cchData, NULL, NULL); |
| HeapFree(GetProcessHeap(), 0, lpCalDataW); |
| |
| return ret; |
| } |
| |
| /********************************************************************* |
| * GetCalendarInfoW (KERNEL32.@) |
| * |
| */ |
| int WINAPI GetCalendarInfoW(LCID Locale, CALID Calendar, CALTYPE CalType, |
| LPWSTR lpCalData, int cchData, LPDWORD lpValue) |
| { |
| if (CalType & CAL_NOUSEROVERRIDE) |
| FIXME("flag CAL_NOUSEROVERRIDE used, not fully implemented\n"); |
| if (CalType & CAL_USE_CP_ACP) |
| FIXME("flag CAL_USE_CP_ACP used, not fully implemented\n"); |
| |
| if (CalType & CAL_RETURN_NUMBER) { |
| if (lpCalData != NULL) |
| WARN("lpCalData not NULL (%p) when it should!\n", lpCalData); |
| if (cchData != 0) |
| WARN("cchData not 0 (%d) when it should!\n", cchData); |
| } else { |
| if (lpValue != NULL) |
| WARN("lpValue not NULL (%p) when it should!\n", lpValue); |
| } |
| |
| /* FIXME: No verification is made yet wrt Locale |
| * for the CALTYPES not requiring GetLocaleInfoA */ |
| switch (CalType & ~(CAL_NOUSEROVERRIDE|CAL_RETURN_NUMBER|CAL_USE_CP_ACP)) { |
| case CAL_ICALINTVALUE: |
| FIXME("Unimplemented caltype %ld\n", CalType & 0xffff); |
| return E_FAIL; |
| case CAL_SCALNAME: |
| FIXME("Unimplemented caltype %ld\n", CalType & 0xffff); |
| return E_FAIL; |
| case CAL_IYEAROFFSETRANGE: |
| FIXME("Unimplemented caltype %ld\n", CalType & 0xffff); |
| return E_FAIL; |
| case CAL_SERASTRING: |
| FIXME("Unimplemented caltype %ld\n", CalType & 0xffff); |
| return E_FAIL; |
| case CAL_SSHORTDATE: |
| return GetLocaleInfoW(Locale, LOCALE_SSHORTDATE, lpCalData, cchData); |
| case CAL_SLONGDATE: |
| return GetLocaleInfoW(Locale, LOCALE_SLONGDATE, lpCalData, cchData); |
| case CAL_SDAYNAME1: |
| return GetLocaleInfoW(Locale, LOCALE_SDAYNAME1, lpCalData, cchData); |
| case CAL_SDAYNAME2: |
| return GetLocaleInfoW(Locale, LOCALE_SDAYNAME2, lpCalData, cchData); |
| case CAL_SDAYNAME3: |
| return GetLocaleInfoW(Locale, LOCALE_SDAYNAME3, lpCalData, cchData); |
| case CAL_SDAYNAME4: |
| return GetLocaleInfoW(Locale, LOCALE_SDAYNAME4, lpCalData, cchData); |
| case CAL_SDAYNAME5: |
| return GetLocaleInfoW(Locale, LOCALE_SDAYNAME5, lpCalData, cchData); |
| case CAL_SDAYNAME6: |
| return GetLocaleInfoW(Locale, LOCALE_SDAYNAME6, lpCalData, cchData); |
| case CAL_SDAYNAME7: |
| return GetLocaleInfoW(Locale, LOCALE_SDAYNAME7, lpCalData, cchData); |
| case CAL_SABBREVDAYNAME1: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVDAYNAME1, lpCalData, cchData); |
| case CAL_SABBREVDAYNAME2: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVDAYNAME2, lpCalData, cchData); |
| case CAL_SABBREVDAYNAME3: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVDAYNAME3, lpCalData, cchData); |
| case CAL_SABBREVDAYNAME4: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVDAYNAME4, lpCalData, cchData); |
| case CAL_SABBREVDAYNAME5: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVDAYNAME5, lpCalData, cchData); |
| case CAL_SABBREVDAYNAME6: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVDAYNAME6, lpCalData, cchData); |
| case CAL_SABBREVDAYNAME7: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVDAYNAME7, lpCalData, cchData); |
| case CAL_SMONTHNAME1: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME1, lpCalData, cchData); |
| case CAL_SMONTHNAME2: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME2, lpCalData, cchData); |
| case CAL_SMONTHNAME3: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME3, lpCalData, cchData); |
| case CAL_SMONTHNAME4: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME4, lpCalData, cchData); |
| case CAL_SMONTHNAME5: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME5, lpCalData, cchData); |
| case CAL_SMONTHNAME6: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME6, lpCalData, cchData); |
| case CAL_SMONTHNAME7: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME7, lpCalData, cchData); |
| case CAL_SMONTHNAME8: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME8, lpCalData, cchData); |
| case CAL_SMONTHNAME9: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME9, lpCalData, cchData); |
| case CAL_SMONTHNAME10: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME10, lpCalData, cchData); |
| case CAL_SMONTHNAME11: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME11, lpCalData, cchData); |
| case CAL_SMONTHNAME12: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME12, lpCalData, cchData); |
| case CAL_SMONTHNAME13: |
| return GetLocaleInfoW(Locale, LOCALE_SMONTHNAME13, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME1: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME1, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME2: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME2, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME3: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME3, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME4: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME4, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME5: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME5, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME6: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME6, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME7: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME7, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME8: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME8, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME9: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME9, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME10: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME10, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME11: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME11, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME12: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME12, lpCalData, cchData); |
| case CAL_SABBREVMONTHNAME13: |
| return GetLocaleInfoW(Locale, LOCALE_SABBREVMONTHNAME13, lpCalData, cchData); |
| case CAL_SYEARMONTH: |
| return GetLocaleInfoW(Locale, LOCALE_SYEARMONTH, lpCalData, cchData); |
| case CAL_ITWODIGITYEARMAX: |
| if (lpValue) *lpValue = CALINFO_MAX_YEAR; |
| break; |
| default: MESSAGE("Unknown caltype %ld\n",CalType & 0xffff); |
| return E_FAIL; |
| } |
| return 0; |
| } |
| |
| /********************************************************************* |
| * SetCalendarInfoA (KERNEL32.@) |
| * |
| */ |
| int WINAPI SetCalendarInfoA(LCID Locale, CALID Calendar, CALTYPE CalType, LPCSTR lpCalData) |
| { |
| FIXME("(%08lx,%08lx,%08lx,%s): stub\n", |
| Locale, Calendar, CalType, debugstr_a(lpCalData)); |
| return 0; |
| } |
| |
| /********************************************************************* |
| * SetCalendarInfoW (KERNEL32.@) |
| * |
| * |
| */ |
| int WINAPI SetCalendarInfoW(LCID Locale, CALID Calendar, CALTYPE CalType, LPCWSTR lpCalData) |
| { |
| FIXME("(%08lx,%08lx,%08lx,%s): stub\n", |
| Locale, Calendar, CalType, debugstr_w(lpCalData)); |
| return 0; |
| } |
| |
| /********************************************************************* |
| * LocalFileTimeToFileTime (KERNEL32.@) |
| */ |
| BOOL WINAPI LocalFileTimeToFileTime( const FILETIME *localft, LPFILETIME utcft ) |
| { |
| NTSTATUS status; |
| LARGE_INTEGER local, utc; |
| |
| local.u.LowPart = localft->dwLowDateTime; |
| local.u.HighPart = localft->dwHighDateTime; |
| if (!(status = RtlLocalTimeToSystemTime( &local, &utc ))) |
| { |
| utcft->dwLowDateTime = utc.u.LowPart; |
| utcft->dwHighDateTime = utc.u.HighPart; |
| } |
| else SetLastError( RtlNtStatusToDosError(status) ); |
| |
| return !status; |
| } |
| |
| /********************************************************************* |
| * FileTimeToLocalFileTime (KERNEL32.@) |
| */ |
| BOOL WINAPI FileTimeToLocalFileTime( const FILETIME *utcft, LPFILETIME localft ) |
| { |
| NTSTATUS status; |
| LARGE_INTEGER local, utc; |
| |
| utc.u.LowPart = utcft->dwLowDateTime; |
| utc.u.HighPart = utcft->dwHighDateTime; |
| if (!(status = RtlSystemTimeToLocalTime( &utc, &local ))) |
| { |
| localft->dwLowDateTime = local.u.LowPart; |
| localft->dwHighDateTime = local.u.HighPart; |
| } |
| else SetLastError( RtlNtStatusToDosError(status) ); |
| |
| return !status; |
| } |
| |
| /********************************************************************* |
| * FileTimeToSystemTime (KERNEL32.@) |
| */ |
| BOOL WINAPI FileTimeToSystemTime( const FILETIME *ft, LPSYSTEMTIME syst ) |
| { |
| TIME_FIELDS tf; |
| LARGE_INTEGER t; |
| |
| t.u.LowPart = ft->dwLowDateTime; |
| t.u.HighPart = ft->dwHighDateTime; |
| RtlTimeToTimeFields(&t, &tf); |
| |
| syst->wYear = tf.Year; |
| syst->wMonth = tf.Month; |
| syst->wDay = tf.Day; |
| syst->wHour = tf.Hour; |
| syst->wMinute = tf.Minute; |
| syst->wSecond = tf.Second; |
| syst->wMilliseconds = tf.Milliseconds; |
| syst->wDayOfWeek = tf.Weekday; |
| return TRUE; |
| } |
| |
| /********************************************************************* |
| * SystemTimeToFileTime (KERNEL32.@) |
| */ |
| BOOL WINAPI SystemTimeToFileTime( const SYSTEMTIME *syst, LPFILETIME ft ) |
| { |
| TIME_FIELDS tf; |
| LARGE_INTEGER t; |
| |
| tf.Year = syst->wYear; |
| tf.Month = syst->wMonth; |
| tf.Day = syst->wDay; |
| tf.Hour = syst->wHour; |
| tf.Minute = syst->wMinute; |
| tf.Second = syst->wSecond; |
| tf.Milliseconds = syst->wMilliseconds; |
| |
| if( !RtlTimeFieldsToTime(&tf, &t)) { |
| SetLastError( ERROR_INVALID_PARAMETER); |
| return FALSE; |
| } |
| ft->dwLowDateTime = t.u.LowPart; |
| ft->dwHighDateTime = t.u.HighPart; |
| return TRUE; |
| } |
| |
| /********************************************************************* |
| * CompareFileTime (KERNEL32.@) |
| * |
| * Compare two FILETIME's to each other. |
| * |
| * PARAMS |
| * x [I] First time |
| * y [I] time to compare to x |
| * |
| * RETURNS |
| * -1, 0, or 1 indicating that x is less than, equal to, or greater |
| * than y respectively. |
| */ |
| INT WINAPI CompareFileTime( const FILETIME *x, const FILETIME *y ) |
| { |
| if (!x || !y) return -1; |
| |
| if (x->dwHighDateTime > y->dwHighDateTime) |
| return 1; |
| if (x->dwHighDateTime < y->dwHighDateTime) |
| return -1; |
| if (x->dwLowDateTime > y->dwLowDateTime) |
| return 1; |
| if (x->dwLowDateTime < y->dwLowDateTime) |
| return -1; |
| return 0; |
| } |
| |
| /********************************************************************* |
| * GetLocalTime (KERNEL32.@) |
| * |
| * Get the current local time. |
| * |
| * PARAMS |
| * systime [O] Destination for current time. |
| * |
| * RETURNS |
| * Nothing. |
| */ |
| VOID WINAPI GetLocalTime(LPSYSTEMTIME systime) |
| { |
| FILETIME lft; |
| LARGE_INTEGER ft, ft2; |
| |
| NtQuerySystemTime(&ft); |
| RtlSystemTimeToLocalTime(&ft, &ft2); |
| lft.dwLowDateTime = ft2.u.LowPart; |
| lft.dwHighDateTime = ft2.u.HighPart; |
| FileTimeToSystemTime(&lft, systime); |
| } |
| |
| /********************************************************************* |
| * GetSystemTime (KERNEL32.@) |
| * |
| * Get the current system time. |
| * |
| * PARAMS |
| * systime [O] Destination for current time. |
| * |
| * RETURNS |
| * Nothing. |
| */ |
| VOID WINAPI GetSystemTime(LPSYSTEMTIME systime) |
| { |
| FILETIME ft; |
| LARGE_INTEGER t; |
| |
| NtQuerySystemTime(&t); |
| ft.dwLowDateTime = t.u.LowPart; |
| ft.dwHighDateTime = t.u.HighPart; |
| FileTimeToSystemTime(&ft, systime); |
| } |
| |
| /********************************************************************* |
| * GetDaylightFlag (KERNEL32.@) |
| * |
| * Specifies if daylight saving time is in operation. |
| * |
| * NOTES |
| * This function is called from the Win98's control applet timedate.cpl. |
| * |
| * RETURNS |
| * TRUE if daylight saving time is in operation. |
| * FALSE otherwise. |
| */ |
| BOOL WINAPI GetDaylightFlag(void) |
| { |
| TIME_ZONE_INFORMATION tzinfo; |
| return GetTimeZoneInformation( &tzinfo) == TIME_ZONE_ID_DAYLIGHT; |
| } |
| |
| /*********************************************************************** |
| * DosDateTimeToFileTime (KERNEL32.@) |
| */ |
| BOOL WINAPI DosDateTimeToFileTime( WORD fatdate, WORD fattime, LPFILETIME ft) |
| { |
| struct tm newtm; |
| #ifndef HAVE_TIMEGM |
| struct tm *gtm; |
| time_t time1, time2; |
| #endif |
| |
| newtm.tm_sec = (fattime & 0x1f) * 2; |
| newtm.tm_min = (fattime >> 5) & 0x3f; |
| newtm.tm_hour = (fattime >> 11); |
| newtm.tm_mday = (fatdate & 0x1f); |
| newtm.tm_mon = ((fatdate >> 5) & 0x0f) - 1; |
| newtm.tm_year = (fatdate >> 9) + 80; |
| #ifdef HAVE_TIMEGM |
| RtlSecondsSince1970ToTime( timegm(&newtm), (LARGE_INTEGER *)ft ); |
| #else |
| time1 = mktime(&newtm); |
| gtm = gmtime(&time1); |
| time2 = mktime(gtm); |
| RtlSecondsSince1970ToTime( 2*time1-time2, (LARGE_INTEGER *)ft ); |
| #endif |
| return TRUE; |
| } |
| |
| |
| /*********************************************************************** |
| * FileTimeToDosDateTime (KERNEL32.@) |
| */ |
| BOOL WINAPI FileTimeToDosDateTime( const FILETIME *ft, LPWORD fatdate, |
| LPWORD fattime ) |
| { |
| LARGE_INTEGER li; |
| ULONG t; |
| time_t unixtime; |
| struct tm* tm; |
| |
| li.u.LowPart = ft->dwLowDateTime; |
| li.u.HighPart = ft->dwHighDateTime; |
| RtlTimeToSecondsSince1970( &li, &t ); |
| unixtime = t; |
| tm = gmtime( &unixtime ); |
| if (fattime) |
| *fattime = (tm->tm_hour << 11) + (tm->tm_min << 5) + (tm->tm_sec / 2); |
| if (fatdate) |
| *fatdate = ((tm->tm_year - 80) << 9) + ((tm->tm_mon + 1) << 5) |
| + tm->tm_mday; |
| return TRUE; |
| } |