mirror of https://github.com/adamdruppe/arsd.git
on idle handlers
This commit is contained in:
parent
83bcc707fc
commit
223587116c
52
eventloop.d
52
eventloop.d
|
@ -13,6 +13,7 @@ template typehash(T...) {
|
||||||
}
|
}
|
||||||
|
|
||||||
private WrappedListener[][hash_t] listeners;
|
private WrappedListener[][hash_t] listeners;
|
||||||
|
private WrappedListener[] idleHandlers;
|
||||||
|
|
||||||
/// Valid event listeners must be callable and take exactly one argument. The type of argument determines the type of event.
|
/// Valid event listeners must be callable and take exactly one argument. The type of argument determines the type of event.
|
||||||
template isValidEventListener(T) {
|
template isValidEventListener(T) {
|
||||||
|
@ -21,6 +22,22 @@ template isValidEventListener(T) {
|
||||||
|
|
||||||
private enum backingSize = (void*).sizeof + hash_t.sizeof;
|
private enum backingSize = (void*).sizeof + hash_t.sizeof;
|
||||||
|
|
||||||
|
/// Calls this function once every time the event system is idle
|
||||||
|
public void addOnIdle(T)(T t) if(isCallable!T && ParameterTypeTuple!(T).length == 0) {
|
||||||
|
idleHandlers ~= wrap(t);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Removes an idle handler (added with addOnIdle)
|
||||||
|
public void removeOnIdle(T)(T t) if(isCallable!T && ParameterTypeTuple!(T).length == 0) {
|
||||||
|
auto pair = getPtrPair(t);
|
||||||
|
foreach(idx, listener; idleHandlers) {
|
||||||
|
if(listener.matches(pair)) {
|
||||||
|
idleHandlers = idleHandlers[0 .. idx] ~ idleHandlers[idx + 1 .. $];
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Sends an exit event to the loop. The loop will break when it sees this event, ignoring any events after that point.
|
/// Sends an exit event to the loop. The loop will break when it sees this event, ignoring any events after that point.
|
||||||
public void exit() {
|
public void exit() {
|
||||||
ubyte[backingSize] bufferBacking = 0; // a null message means exit...
|
ubyte[backingSize] bufferBacking = 0; // a null message means exit...
|
||||||
|
@ -84,8 +101,13 @@ public void sendSync(T)(T t) {
|
||||||
public void send(T)(T t) {
|
public void send(T)(T t) {
|
||||||
// FIXME: we need to cycle the buffer position back so we can reuse this as the message is received
|
// FIXME: we need to cycle the buffer position back so we can reuse this as the message is received
|
||||||
// (if you want to keep a message, it is your responsibility to make your own copy, unless it is a pointer itself)
|
// (if you want to keep a message, it is your responsibility to make your own copy, unless it is a pointer itself)
|
||||||
static ubyte[1024] copyBuffer;
|
//static ubyte[1024] copyBuffer;
|
||||||
static size_t copyBufferPosition;
|
//static size_t copyBufferPosition;
|
||||||
|
|
||||||
|
// for now we'll use the gc
|
||||||
|
size_t copyBufferPosition = 0;
|
||||||
|
auto copyBuffer = new ubyte[](T.sizeof);
|
||||||
|
|
||||||
|
|
||||||
auto hash = typehash!T;
|
auto hash = typehash!T;
|
||||||
//auto ptr = (cast(void*) &t);
|
//auto ptr = (cast(void*) &t);
|
||||||
|
@ -314,6 +336,17 @@ private void addBackFilesToLoop() {
|
||||||
backFilesForLoop = null;
|
backFilesForLoop = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
addOnIdle(function) is similar to calling setInterval(function, 0)
|
||||||
|
|
||||||
|
auto id = setTimeout(function, wait)
|
||||||
|
clearTimeout(id)
|
||||||
|
|
||||||
|
auto id = setInterval(function, call at least after)
|
||||||
|
clearInterval(0)
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
private bool insideLoop = false;
|
private bool insideLoop = false;
|
||||||
|
|
||||||
version(linux) {
|
version(linux) {
|
||||||
|
@ -365,6 +398,7 @@ version(linux) {
|
||||||
|
|
||||||
outer_loop: for(;;) {
|
outer_loop: for(;;) {
|
||||||
auto nfds = epoll_wait(epoll, events.ptr, events.length, -1 /* wait forever, otherwise in milliseconds */);
|
auto nfds = epoll_wait(epoll, events.ptr, events.length, -1 /* wait forever, otherwise in milliseconds */);
|
||||||
|
moreEvents:
|
||||||
if(nfds == -1)
|
if(nfds == -1)
|
||||||
throw new Exception("epoll_wait");
|
throw new Exception("epoll_wait");
|
||||||
|
|
||||||
|
@ -383,6 +417,13 @@ version(linux) {
|
||||||
sendSync(FileError(fd));
|
sendSync(FileError(fd));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
nfds = epoll_wait(epoll, events.ptr, events.length, 0 /* no wait */);
|
||||||
|
if(nfds != 0)
|
||||||
|
goto moreEvents;
|
||||||
|
// no immediate events means we're idle for now, run those functions
|
||||||
|
foreach(idleHandler; idleHandlers)
|
||||||
|
idleHandler.call(null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -430,7 +471,12 @@ private WrappedListener wrap(T)(T t) {
|
||||||
else {
|
else {
|
||||||
return new class WrappedListener {
|
return new class WrappedListener {
|
||||||
override void call(void* ptr) {
|
override void call(void* ptr) {
|
||||||
t(*(cast(ParameterTypeTuple!(T)[0]*) ptr));
|
enum arity = ParameterTypeTuple!(T).length;
|
||||||
|
static if(arity == 1)
|
||||||
|
t(*(cast(ParameterTypeTuple!(T)[0]*) ptr));
|
||||||
|
else static if(arity == 0)
|
||||||
|
t();
|
||||||
|
else static assert(0, "bad number of arguments");
|
||||||
}
|
}
|
||||||
|
|
||||||
override bool matches(void*[2] pair) {
|
override bool matches(void*[2] pair) {
|
||||||
|
|
Loading…
Reference in New Issue