php-src/win32/time.c

203 lines
5.1 KiB
C
Raw Normal View History

1999-04-07 21:05:13 +00:00
/*****************************************************************************
* *
* DH_TIME.C *
* *
* Freely redistributable and modifiable. Use at your own risk. *
* *
* Copyright 1994 The Downhill Project *
2015-01-03 09:22:58 +00:00
*
1999-04-07 21:05:13 +00:00
* Modified by Shane Caraveo for use with PHP
*
*****************************************************************************/
/* $Id$ */
1999-04-07 21:05:13 +00:00
/* Include stuff ************************************************************ */
1999-04-23 20:06:01 +00:00
#include <config.w32.h>
1999-04-07 21:05:13 +00:00
#include "time.h"
#include "unistd.h"
#include "signal.h"
#include <windows.h>
1999-04-07 21:05:13 +00:00
#include <winbase.h>
#include <mmsystem.h>
#include <errno.h>
2004-07-29 02:59:44 +00:00
#include "php_win32_globals.h"
1999-04-07 21:05:13 +00:00
2013-03-15 18:04:40 +00:00
typedef VOID (WINAPI *MyGetSystemTimeAsFileTime)(LPFILETIME lpSystemTimeAsFileTime);
static MyGetSystemTimeAsFileTime timefunc = NULL;
static zend_always_inline MyGetSystemTimeAsFileTime get_time_func(void)
{
2013-03-15 18:04:40 +00:00
MyGetSystemTimeAsFileTime timefunc = NULL;
HMODULE hMod = LoadLibrary("kernel32.dll");
if (hMod) {
/* Max possible resolution <1us, win8/server2012 */
timefunc = (MyGetSystemTimeAsFileTime)GetProcAddress(hMod, "GetSystemTimePreciseAsFileTime");
if(!timefunc) {
/* 100ns blocks since 01-Jan-1641 */
timefunc = (MyGetSystemTimeAsFileTime)GetProcAddress(hMod, "GetSystemTimeAsFileTime");
}
}
return timefunc;
}
static zend_always_inline int getfilesystemtime(struct timeval *tv)
2013-03-15 18:04:40 +00:00
{
FILETIME ft;
unsigned __int64 ff = 0;
ULARGE_INTEGER fft;
2013-03-15 18:04:40 +00:00
if (!timefunc) {
timefunc = get_time_func();
2013-03-15 18:04:40 +00:00
}
timefunc(&ft);
2013-03-15 18:04:40 +00:00
/*
2015-01-03 09:22:58 +00:00
* Do not cast a pointer to a FILETIME structure to either a
* ULARGE_INTEGER* or __int64* value because it can cause alignment faults on 64-bit Windows.
* via http://technet.microsoft.com/en-us/library/ms724284(v=vs.85).aspx
*/
fft.HighPart = ft.dwHighDateTime;
fft.LowPart = ft.dwLowDateTime;
ff = fft.QuadPart;
ff /= 10Ui64; /* convert to microseconds */
2013-03-15 18:04:40 +00:00
ff -= 11644473600000000Ui64; /* convert to unix epoch */
tv->tv_sec = (long)(ff / 1000000Ui64);
tv->tv_usec = (long)(ff % 1000000Ui64);
2013-03-15 18:04:40 +00:00
return 0;
}
PHPAPI int gettimeofday(struct timeval *time_Info, struct timezone *timezone_Info)
1999-04-07 21:05:13 +00:00
{
/* Get the time, if they want it */
if (time_Info != NULL) {
2013-03-15 18:04:40 +00:00
getfilesystemtime(time_Info);
1999-04-07 21:05:13 +00:00
}
/* Get the timezone, if they want it */
if (timezone_Info != NULL) {
_tzset();
timezone_Info->tz_minuteswest = _timezone;
timezone_Info->tz_dsttime = _daylight;
}
/* And return */
return 0;
}
PHPAPI int usleep(unsigned int useconds)
1999-04-07 21:05:13 +00:00
{
2003-11-29 22:48:42 +00:00
HANDLE timer;
LARGE_INTEGER due;
2004-01-13 02:07:04 +00:00
due.QuadPart = -(10 * (__int64)useconds);
1999-04-07 21:05:13 +00:00
2004-01-13 02:07:04 +00:00
timer = CreateWaitableTimer(NULL, TRUE, NULL);
2003-11-29 22:48:42 +00:00
SetWaitableTimer(timer, &due, 0, NULL, NULL, 0);
WaitForSingleObject(timer, INFINITE);
CloseHandle(timer);
return 0;
}
PHPAPI int nanosleep( const struct timespec * rqtp, struct timespec * rmtp )
{
if (rqtp->tv_nsec > 999999999) {
/* The time interval specified 1,000,000 or more microseconds. */
errno = EINVAL;
return -1;
}
return usleep( rqtp->tv_sec * 1000000 + rqtp->tv_nsec / 1000 );
2003-11-29 22:48:42 +00:00
}
1999-04-07 21:05:13 +00:00
2004-07-29 02:59:44 +00:00
#if 0 /* looks pretty ropey in here */
#ifdef HAVE_SETITIMER
1999-04-07 21:05:13 +00:00
#ifndef THREAD_SAFE
unsigned int proftimer, virttimer, realtimer;
extern LPMSG phpmsg;
#endif
struct timer_msg {
int signal;
unsigned int threadid;
};
LPTIMECALLBACK setitimer_timeout(UINT uTimerID, UINT info, DWORD dwUser, DWORD dw1, DWORD dw2)
{
struct timer_msg *msg = (struct timer_msg *) info;
if (msg) {
raise((int) msg->signal);
PostThreadMessage(msg->threadid,
WM_NOTIFY, msg->signal, 0);
free(msg);
}
return 0;
}
PHPAPI int setitimer(int which, const struct itimerval *value, struct itimerval *ovalue)
1999-04-07 21:05:13 +00:00
{
int timeout = value->it_value.tv_sec * 1000 + value->it_value.tv_usec;
int repeat = TIME_ONESHOT;
/*make sure the message queue is initialized */
PeekMessage(phpmsg, NULL, WM_USER, WM_USER, PM_NOREMOVE);
1999-04-07 21:05:13 +00:00
if (timeout > 0) {
struct timer_msg *msg = malloc(sizeof(struct timer_msg));
msg->threadid = GetCurrentThreadId();
if (!ovalue) {
repeat = TIME_PERIODIC;
}
switch (which) {
case ITIMER_REAL:
msg->signal = SIGALRM;
realtimer = timeSetEvent(timeout, 100, (LPTIMECALLBACK) setitimer_timeout, (UINT) msg, repeat);
1999-04-07 21:05:13 +00:00
break;
case ITIMER_VIRT:
msg->signal = SIGVTALRM;
virttimer = timeSetEvent(timeout, 100, (LPTIMECALLBACK) setitimer_timeout, (UINT) msg, repeat);
1999-04-07 21:05:13 +00:00
break;
case ITIMER_PROF:
msg->signal = SIGPROF;
proftimer = timeSetEvent(timeout, 100, (LPTIMECALLBACK) setitimer_timeout, (UINT) msg, repeat);
1999-04-07 21:05:13 +00:00
break;
default:
errno = EINVAL;
return -1;
break;
}
} else {
switch (which) {
case ITIMER_REAL:
timeKillEvent(realtimer);
1999-04-07 21:05:13 +00:00
break;
case ITIMER_VIRT:
timeKillEvent(virttimer);
1999-04-07 21:05:13 +00:00
break;
case ITIMER_PROF:
timeKillEvent(proftimer);
1999-04-07 21:05:13 +00:00
break;
default:
errno = EINVAL;
return -1;
break;
}
}
return 0;
}
#endif
2004-07-29 02:59:44 +00:00
#endif