timer class for simpledisplay and much better pulsing on Windows

This commit is contained in:
Adam D. Ruppe 2015-09-12 00:19:50 -04:00
parent 773fc20999
commit 8350865a4e
2 changed files with 342 additions and 43 deletions

View File

@ -450,7 +450,9 @@ version(linux) {
int epoll = -1; int epoll = -1;
private void addFileToLoopImplementation(int fd, int events, bool edgeTriggered = true) { private void addFileToLoopImplementation(int fd, int events, bool edgeTriggered = true) {
epoll_event ev; epoll_event ev = void;
ev.events = 0;
// I don't remember why I made it edge triggered in the first // I don't remember why I made it edge triggered in the first
// place as that requires a bit more care to do correctly and I don't // place as that requires a bit more care to do correctly and I don't
@ -509,7 +511,7 @@ version(linux) {
addFileToLoop(pipes[0], FileEvents.read, false); addFileToLoop(pipes[0], FileEvents.read, false);
epoll_event[16] events; epoll_event[16] events = void;
timeval tv; timeval tv;

View File

@ -88,6 +88,13 @@ version(html5) {} else {
version = X11; version = X11;
} }
// these are so the static asserts don't trigger unless you want to
// add support to it for an OS
version(Windows)
version = with_timer;
version(linux)
version = with_timer;
// If you have to get down and dirty with implementation details, this helps figure out if X is available // If you have to get down and dirty with implementation details, this helps figure out if X is available
// you can static if(UsingSimpledisplayX11) ... more reliably than version() because version is module-local. // you can static if(UsingSimpledisplayX11) ... more reliably than version() because version is module-local.
version(X11) version(X11)
@ -95,6 +102,157 @@ version(X11)
else else
enum bool UsingSimpledisplayX11 = false; enum bool UsingSimpledisplayX11 = false;
// basic functions to make timers
/**
TIMERS
You create a timer with an interval and a callback. It will continue
to fire on the interval until it is destroyed.
There are currently no one-off timers (instead, just create one and
destroy it when it is triggered) nor are there pause/resume functions -
the timer must again be destroyed and recreated if you want to pause it.
auto timer = new Timer(50, { it happened!; });
timer.destroy();
destroyTimer(handle);
*/
version(with_timer)
class Timer {
// FIXME: I might add overloads for ones that take a count of
// how many elapsed since last time (on Windows, it will divide
// the ticks thing given, on Linux it is just available) and
// maybe one that takes an instance of the Timer itself too
this(int intervalInMilliseconds, void delegate() onPulse) {
assert(onPulse !is null);
this.onPulse = onPulse;
version(Windows) {
handle = SetTimer(null, handle, intervalInMilliseconds, &timerCallback);
if(handle == 0)
throw new Exception("SetTimer fail");
mapping[handle] = this;
} else version(linux) {
static import ep = core.sys.linux.epoll;
// static import tfd = core.sys.linux.timerfd;
static struct tfd {
static:
import core.sys.posix.time;
extern(C):
pragma(mangle, "timerfd_create")
int timerfd_create(int, int);
pragma(mangle, "timerfd_settime")
int timerfd_settime(int, int, const itimerspec*, itimerspec*);
pragma(mangle, "timerfd_gettime")
int timerfd_gettime(int, itimerspec*);
enum TFD_TIMER_ABSTIME = 1 << 0;
enum TFD_CLOEXEC = 0x80000;
enum TFD_NONBLOCK = 0x800;
}
fd = tfd.timerfd_create(tfd.CLOCK_MONOTONIC, 0);
if(fd == -1)
throw new Exception("timer create failed");
mapping[fd] = this;
tfd.itimerspec value;
value.it_value.tv_sec = cast(int) (intervalInMilliseconds / 1000);
value.it_value.tv_nsec = (intervalInMilliseconds % 1000) * 1000_000;
value.it_interval.tv_sec = cast(int) (intervalInMilliseconds / 1000);
value.it_interval.tv_nsec = (intervalInMilliseconds % 1000) * 1000_000;
if(tfd.timerfd_settime(fd, 0, &value, null) == -1)
throw new Exception("couldn't make pulse timer");
version(with_eventloop) {
import arsd.eventloop;
addFileEventListeners(fd, &trigger);
} else {
prepareEventLoop();
ep.epoll_event ev = void;
ev.events = ep.EPOLLIN;
ev.data.fd = fd;
ep.epoll_ctl(epollFd, ep.EPOLL_CTL_ADD, fd, &ev);
}
} else static assert(0);
}
void destroy() {
version(Windows) {
if(handle) {
KillTimer(null, handle);
mapping.remove(handle);
handle = 0;
}
} else version(linux) {
if(fd != -1) {
import unix = core.sys.posix.unistd;
static import ep = core.sys.linux.epoll;
version(with_eventloop) {
import arsd.eventloop;
removeFileEventListeners(fd);
} else {
ep.epoll_event ev = void;
ev.events = ep.EPOLLIN;
ev.data.fd = fd;
ep.epoll_ctl(epollFd, ep.EPOLL_CTL_DEL, fd, &ev);
}
unix.close(fd);
mapping.remove(fd);
fd = -1;
}
} else static assert(0);
}
~this() {
destroy();
}
private:
void delegate() onPulse;
void trigger() {
version(linux) {
import unix = core.sys.posix.unistd;
long val;
unix.read(fd, &val, val.sizeof); // gotta clear the pipe
} else version(Windows) {
} else static assert(0);
onPulse();
}
version(Windows)
extern(Windows)
static void timerCallback(HWND, UINT, UINT_PTR timer, DWORD dwTime) {
if(Timer* t = timer in mapping) {
(*t).trigger();
}
}
version(Windows) {
UINT_PTR handle;
static Timer[UINT_PTR] mapping;
} else version(linux) {
int fd = -1;
static Timer[int] mapping;
} else static assert(0, "timer not supported");
}
// basic functions to access the clipboard // basic functions to access the clipboard
/+ /+
@ -2581,37 +2739,24 @@ version(Windows) {
MSG message; MSG message;
int ret; int ret;
import core.thread; Timer pulser;
scope(exit)
if(pulser !is null)
pulser.destroy();
if(pulseTimeout) { if(pulseTimeout)
bool done = false; pulser = new Timer(cast(int) pulseTimeout, handlePulse);
while(!done) {
if(PeekMessage(&message, null, 0, 0, PM_NOREMOVE)) {
ret = GetMessage(&message, null, 0, 0);
if(ret == 0)
done = true;
// if(!IsDialogMessageA(message.hwnd, &message)) { // if(PeekMessage(&message, null, 0, 0, PM_NOREMOVE))
TranslateMessage(&message); while((ret = GetMessage(&message, null, 0, 0)) != 0) {
DispatchMessage(&message); if(ret == -1)
// } throw new Exception("GetMessage failed");
} // if(!IsDialogMessageA(message.hwnd, &message)) {
TranslateMessage(&message);
DispatchMessage(&message);
// }
if(!done && !closed && handlePulse !is null) SleepEx(0, true); // I call this to give it a chance to do stuff like async io, which never happens when you just block in GetMessage
handlePulse();
SleepEx(cast(DWORD) pulseTimeout, true);
}
} else {
while((ret = GetMessage(&message, null, 0, 0)) != 0) {
if(ret == -1)
throw new Exception("GetMessage failed");
// if(!IsDialogMessageA(message.hwnd, &message)) {
TranslateMessage(&message);
DispatchMessage(&message);
// }
SleepEx(0, true); // I call this to give it a chance to do stuff like async io, which apparently never happens when you just block in GetMessage
}
} }
return message.wParam; return message.wParam;
@ -2719,7 +2864,7 @@ version(Windows) {
enum KEY_ESCAPE = 27; enum KEY_ESCAPE = 27;
} }
version(X11) { version(X11) {
__gshared string xfontstr = "-bitstream-bitstream vera sans-medium-r-*-*-12-*-*-*-*-*-*-*"; __gshared string xfontstr = "-*-dejavu sans-medium-r-*-*-12-*-*-*-*-*-*-*";
alias int delegate(XEvent) NativeEventHandler; alias int delegate(XEvent) NativeEventHandler;
alias Window NativeWindowHandle; alias Window NativeWindowHandle;
@ -3308,20 +3453,159 @@ version(X11) {
bool destroyed = false; bool destroyed = false;
version(with_eventloop)
int eventLoop(long pulseTimeout) {
version(X11)
XFlush(display);
import arsd.eventloop;
auto handle = setInterval(handlePulse, cast(int) pulseTimeout);
scope(exit) clearInterval(handle);
loop();
return 0;
}
else
int eventLoop(long pulseTimeout) { int eventLoop(long pulseTimeout) {
bool done = false; bool done = false;
import core.thread;
while (!done) { version(X11)
while(!done && XFlush(display);
(pulseTimeout == 0 || (XPending(display) > 0)))
{ version(linux) {
done = doXNextEvent(this.display); static import ep = core.sys.linux.epoll;
} static import unix = core.sys.posix.unistd;
if(!done && !closed && pulseTimeout !=0) { static import err = core.stdc.errno;
if(handlePulse !is null)
handlePulse(); // until this is reliably in druntime i'll use my own
Thread.sleep(dur!"msecs"(pulseTimeout)); // so probably like next year i'll switch this over!
// static import tfd = core.sys.linux.timerfd;
static struct tfd {
static:
import core.sys.posix.time;
extern(C):
pragma(mangle, "timerfd_create")
int timerfd_create(int, int);
pragma(mangle, "timerfd_settime")
int timerfd_settime(int, int, const itimerspec*, itimerspec*);
pragma(mangle, "timerfd_gettime")
int timerfd_gettime(int, itimerspec*);
enum TFD_TIMER_ABSTIME = 1 << 0;
enum TFD_CLOEXEC = 0x80000;
enum TFD_NONBLOCK = 0x800;
}
prepareEventLoop();
ep.epoll_event[16] events = void;
int pulseFd = -1;
scope(exit)
if(pulseFd != -1)
unix.close(pulseFd);
{
// adding Xlib file
ep.epoll_event ev = void;
ev.events = ep.EPOLLIN;
ev.data.fd = display.fd;
import std.conv;
if(ep.epoll_ctl(epollFd, ep.EPOLL_CTL_ADD, display.fd, &ev) == -1)
throw new Exception("add x fd" ~ to!string(epollFd));
}
if(pulseTimeout) {
pulseFd = tfd.timerfd_create(tfd.CLOCK_MONOTONIC, 0);
if(pulseFd == -1)
throw new Exception("pulse timer create failed");
tfd.itimerspec value;
value.it_value.tv_sec = cast(int) (pulseTimeout / 1000);
value.it_value.tv_nsec = (pulseTimeout % 1000) * 1000_000;
value.it_interval.tv_sec = cast(int) (pulseTimeout / 1000);
value.it_interval.tv_nsec = (pulseTimeout % 1000) * 1000_000;
if(tfd.timerfd_settime(pulseFd, 0, &value, null) == -1)
throw new Exception("couldn't make pulse timer");
ep.epoll_event ev = void;
ev.events = ep.EPOLLIN;
ev.data.fd = pulseFd;
ep.epoll_ctl(epollFd, ep.EPOLL_CTL_ADD, pulseFd, &ev);
}
while(!done) {
auto nfds = ep.epoll_wait(epollFd, events.ptr, events.length, -1);
if(nfds == -1) {
if(err.errno == err.EINTR) {
continue; // interrupted by signal, just try again
}
throw new Exception("epoll wait failure");
}
foreach(idx; 0 .. nfds) {
if(done) break;
auto fd = events[idx].data.fd;
assert(fd != -1); // should never happen cuz the api doesn't do that but better to assert than assume.
auto flags = events[idx].events;
if(flags & ep.EPOLLIN) {
if(fd == display.fd) {
while(!done && XPending(display))
done = doXNextEvent(this.display);
} else if(fd == pulseFd) {
long expirationCount;
// if we go over the count, I ignore it because i don't want the pulse to go off more often and eat tons of cpu time...
// read just to clear the buffer so poll doesn't trigger again
unix.read(pulseFd, &expirationCount, expirationCount.sizeof);
handlePulse();
} else {
// some other timer
if(Timer* t = fd in Timer.mapping)
(*t).trigger();
// or i might add support for other FDs too
// but for now it is just timer
// (if you want other fds, use arsd.eventloop and compile with -version=with_eventloop), it offers a fuller api for arbitrary stuff.
}
} else {
// not interested in OUT, we are just reading here.
//
// error or hup might also be reported
// but it shouldn't here since we are only
// using a few types of FD and Xlib will report
// if it dies.
// so instead of thoughtfully handling it, I'll
// just throw. for now at least
throw new Exception("epoll did something else");
}
}
}
} else {
// Generic fallback: yes to simple pulse support,
// but NO timer support!
// FIXME: we could probably support the POSIX timer_create
// signal-based option, but I'm in no rush to write it since
// I prefer the fd-based functions.
while (!done) {
while(!done &&
(pulseTimeout == 0 || (XPending(display) > 0)))
{
done = doXNextEvent(this.display);
}
if(!done && !closed && pulseTimeout !=0) {
if(handlePulse !is null)
handlePulse();
import core.thread;
Thread.sleep(dur!"msecs"(pulseTimeout));
}
} }
} }
@ -6179,7 +6463,6 @@ version(html5) {
while(!done && while(!done &&
(pulseTimeout == 0 || socket.recvAvailable())) (pulseTimeout == 0 || socket.recvAvailable()))
{ {
//done = doXNextEvent(this); // FIXME: what about multiple windows? This wasn't originally going to support them but maybe I should
} }
if(!done && pulseTimeout !=0) { if(!done && pulseTimeout !=0) {
if(handlePulse !is null) if(handlePulse !is null)
@ -6319,4 +6602,18 @@ extern(System){
} }
version(linux) {
version(with_eventloop) {} else {
private int epollFd = -1;
void prepareEventLoop() {
if(epollFd != -1)
return; // already initialized, no need to do it again
import ep = core.sys.linux.epoll;
epollFd = ep.epoll_create1(0);
if(epollFd == -1)
throw new Exception("epoll create failure");
}
}
}