Commit b62d90d8 authored by Kenton Varda's avatar Kenton Varda

Port SimpleEventLoop to pthreads for non-Linux platforms.

parent 853c9af0
...@@ -25,10 +25,11 @@ ...@@ -25,10 +25,11 @@
#include "debug.h" #include "debug.h"
#include <exception> #include <exception>
// TODO(now): Encapsulate in mutex.h, with portable implementation. #if __linux__
#include <unistd.h> #include <unistd.h>
#include <sys/syscall.h> #include <sys/syscall.h>
#include <linux/futex.h> #include <linux/futex.h>
#endif
namespace kj { namespace kj {
...@@ -210,6 +211,8 @@ void EventLoop::Event::disarm() { ...@@ -210,6 +211,8 @@ void EventLoop::Event::disarm() {
// ======================================================================================= // =======================================================================================
#if __linux__
SimpleEventLoop::SimpleEventLoop() {} SimpleEventLoop::SimpleEventLoop() {}
SimpleEventLoop::~SimpleEventLoop() noexcept(false) {} SimpleEventLoop::~SimpleEventLoop() noexcept(false) {}
...@@ -230,6 +233,57 @@ void SimpleEventLoop::wake() const { ...@@ -230,6 +233,57 @@ void SimpleEventLoop::wake() const {
} }
} }
#else
#define KJ_PTHREAD_CALL(code) \
{ \
int pthreadError = code; \
if (pthreadError != 0) { \
KJ_FAIL_SYSCALL(#code, pthreadError); \
} \
}
#define KJ_PTHREAD_CLEANUP(code) \
{ \
int pthreadError = code; \
if (pthreadError != 0) { \
KJ_LOG(ERROR, #code, strerror(pthreadError)); \
} \
}
SimpleEventLoop::SimpleEventLoop() {
KJ_PTHREAD_CALL(pthread_mutex_init(&mutex, nullptr));
KJ_PTHREAD_CALL(pthread_cond_init(&condvar, nullptr));
}
SimpleEventLoop::~SimpleEventLoop() noexcept(false) {
KJ_PTHREAD_CLEANUP(pthread_cond_destroy(&condvar));
KJ_PTHREAD_CLEANUP(pthread_mutex_destroy(&mutex));
}
void SimpleEventLoop::prepareToSleep() noexcept {
pthread_mutex_lock(&mutex);
preparedToSleep = 1;
}
void SimpleEventLoop::sleep() {
while (preparedToSleep == 1) {
pthread_cond_wait(&condvar, &mutex);
}
pthread_mutex_unlock(&mutex);
}
void SimpleEventLoop::wake() const {
pthread_mutex_lock(&mutex);
if (preparedToSleep != 0) {
// preparedToSleep was 1 before the exchange, so a sleep must be in progress in another thread.
preparedToSleep = 0;
pthread_cond_signal(&condvar);
}
pthread_mutex_unlock(&mutex);
}
#endif
// ======================================================================================= // =======================================================================================
void PromiseBase::absolve() { void PromiseBase::absolve() {
......
...@@ -310,8 +310,9 @@ protected: ...@@ -310,8 +310,9 @@ protected:
// Subclasses should implement these. // Subclasses should implement these.
virtual void prepareToSleep() noexcept = 0; virtual void prepareToSleep() noexcept = 0;
// Called just before `sleep()`. `sleep()` may or may not actually be called after this -- it's // Called just before `sleep()`. After calling this, the caller checks if any events are
// possible that some other work will be done and then `prepareToSleep()` will be called again. // scheduled. If so, it calls `wake()`. Then, whether or not events were scheduled, it calls
// `sleep()`. Thus, `prepareToSleep()` is always followed by exactly one call to `sleep()`.
virtual void sleep() = 0; virtual void sleep() = 0;
// Do not return until `wake()` is called. Always preceded by a call to `prepareToSleep()`. // Do not return until `wake()` is called. Always preceded by a call to `prepareToSleep()`.
...@@ -359,6 +360,10 @@ protected: ...@@ -359,6 +360,10 @@ protected:
private: private:
mutable int preparedToSleep = 0; mutable int preparedToSleep = 0;
#if !__linux__
mutable pthread_mutex_t mutex;
mutable pthread_cond_t condvar;
#endif
}; };
// ------------------------------------------------------------------- // -------------------------------------------------------------------
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment