diff options
author | Clyne Sullivan <tullivan99@gmail.com> | 2017-01-10 21:26:13 -0500 |
---|---|---|
committer | Clyne Sullivan <tullivan99@gmail.com> | 2017-01-10 21:26:13 -0500 |
commit | f800dbc034e7a70a613bab8cd9d147be4f6e88b6 (patch) | |
tree | c8714ddc8f6712f07d921c08fd92ae8ef228494b /entityx/3rdparty/simplesignal.h | |
parent | c6651145a4a6e6611b585c1a87c127f38751db4b (diff) |
windows build
Diffstat (limited to 'entityx/3rdparty/simplesignal.h')
-rw-r--r-- | entityx/3rdparty/simplesignal.h | 580 |
1 files changed, 580 insertions, 0 deletions
diff --git a/entityx/3rdparty/simplesignal.h b/entityx/3rdparty/simplesignal.h new file mode 100644 index 0000000..3c60273 --- /dev/null +++ b/entityx/3rdparty/simplesignal.h @@ -0,0 +1,580 @@ +// CC0 Public Domain: http://creativecommons.org/publicdomain/zero/1.0/ +#ifndef SIMPLE_SIGNAL_H__ +#define SIMPLE_SIGNAL_H__ + +#include <assert.h> +#include <stdint.h> +#include <vector> +#include <functional> + +namespace Simple { + +namespace Lib { + +/// ProtoSignal is the template implementation for callback list. +template <typename, typename> +class ProtoSignal; // undefined + +/// CollectorInvocation invokes signal handlers differently depending on return +/// type. +template <typename, typename> +struct CollectorInvocation; + +/// CollectorLast returns the result of the last signal handler from a signal +/// emission. +template <typename Result> +struct CollectorLast { + typedef Result CollectorResult; + explicit CollectorLast() : last_() {} + inline bool operator()(Result r) { + last_ = r; + return true; + } + CollectorResult result() { return last_; } + + private: + Result last_; +}; + +/// CollectorDefault implements the default signal handler collection behaviour. +template <typename Result> +struct CollectorDefault : CollectorLast<Result> {}; + +/// CollectorDefault specialisation for signals with void return type. +template <> +struct CollectorDefault<void> { + typedef void CollectorResult; + void result() {} + inline bool operator()(void) { return true; } +}; + +/// CollectorInvocation specialisation for regular signals. +template <class Collector, class R, class... Args> +struct CollectorInvocation<Collector, R(Args...)> { + inline bool invoke(Collector &collector, const std::function<R(Args...)> &cbf, + Args... args) { + return collector(cbf(args...)); + } +}; + +/// CollectorInvocation specialisation for signals with void return type. +template <class Collector, class... Args> +struct CollectorInvocation<Collector, void(Args...)> { + inline bool invoke(Collector &collector, + const std::function<void(Args...)> &cbf, Args... args) { + cbf(args...); + return collector(); + } +}; + +/// ProtoSignal template specialised for the callback signature and collector. +template <class Collector, class R, class... Args> +class ProtoSignal<R(Args...), Collector> : private CollectorInvocation< + Collector, R(Args...)> { + protected: + typedef std::function<R(Args...)> CbFunction; + typedef typename CbFunction::result_type Result; + typedef typename Collector::CollectorResult CollectorResult; + + private: + /// SignalLink implements a doubly-linked ring with ref-counted nodes + /// containing the signal handlers. + struct SignalLink { + SignalLink *next, *prev; + CbFunction function; + int ref_count; + explicit SignalLink(const CbFunction &cbf) + : next(0), prev(0), function(cbf), ref_count(1) {} + /*dtor*/ ~SignalLink() { assert(ref_count == 0); } + void incref() { + ref_count += 1; + assert(ref_count > 0); + } + void decref() { + ref_count -= 1; + if (!ref_count) + delete this; + else + assert(ref_count > 0); + } + void unlink() { + function = nullptr; + if (next) next->prev = prev; + if (prev) prev->next = next; + decref(); + // leave intact ->next, ->prev for stale iterators + } + size_t add_before(const CbFunction &cb) { + SignalLink *link = new SignalLink(cb); + link->prev = prev; // link to last + link->next = this; + prev->next = link; // link from last + prev = link; + static_assert(sizeof(link) == sizeof(size_t), "sizeof size_t"); + return size_t(link); + } + bool deactivate(const CbFunction &cbf) { + if (cbf == function) { + function = 0; // deactivate static head + return true; + } + for (SignalLink *link = this->next ? this->next : this; link != this; + link = link->next) + if (cbf == link->function) { + link->unlink(); // deactivate and unlink sibling + return true; + } + return false; + } + bool remove_sibling(size_t id) { + for (SignalLink *link = this->next ? this->next : this; link != this; + link = link->next) + if (id == size_t(link)) { + link->unlink(); // deactivate and unlink sibling + return true; + } + return false; + } + }; + SignalLink *callback_ring_; // linked ring of callback nodes + /*copy-ctor*/ ProtoSignal(const ProtoSignal &) = delete; + ProtoSignal &operator=(const ProtoSignal &) = delete; + void ensure_ring() { + if (!callback_ring_) { + callback_ring_ = new SignalLink(CbFunction()); // ref_count = 1 + callback_ring_->incref(); // ref_count = 2, head of ring, can be + // deactivated but not removed + callback_ring_->next = callback_ring_; // ring head initialization + callback_ring_->prev = callback_ring_; // ring tail initialization + } + } + + public: + /// ProtoSignal constructor, connects default callback if non-0. + ProtoSignal(const CbFunction &method) : callback_ring_(0) { + if (method != 0) { + ensure_ring(); + callback_ring_->function = method; + } + } + /// ProtoSignal destructor releases all resources associated with this signal. + ~ProtoSignal() { + if (callback_ring_) { + while (callback_ring_->next != callback_ring_) + callback_ring_->next->unlink(); + assert(callback_ring_->ref_count >= 2); + callback_ring_->decref(); + callback_ring_->decref(); + } + } + /// Operator to add a new function or lambda as signal handler, returns a + /// handler connection ID. + size_t connect(const CbFunction &cb) { + ensure_ring(); + return callback_ring_->add_before(cb); + } + /// Operator to remove a signal handler through it connection ID, returns if a + /// handler was removed. + bool disconnect(size_t connection) { + return callback_ring_ ? callback_ring_->remove_sibling(connection) : false; + } + /// Emit a signal, i.e. invoke all its callbacks and collect return types with + /// the Collector. + CollectorResult emit(Args... args) { + Collector collector; + if (!callback_ring_) return collector.result(); + SignalLink *link = callback_ring_; + link->incref(); + do { + if (link->function != 0) { + const bool continue_emission = + this->invoke(collector, link->function, args...); + if (!continue_emission) break; + } + SignalLink *old = link; + link = old->next; + link->incref(); + old->decref(); + } while (link != callback_ring_); + link->decref(); + return collector.result(); + } + // Number of connected slots. + std::size_t size() { + std::size_t size = 0; + SignalLink *link = callback_ring_; + link->incref(); + do { + if (link->function != 0) { + size++; + } + SignalLink *old = link; + link = old->next; + link->incref(); + old->decref(); + } while (link != callback_ring_); + return size; + } +}; + +} // Lib + // namespace Simple + +/** + * Signal is a template type providing an interface for arbitrary callback + * lists. + * A signal type needs to be declared with the function signature of its + * callbacks, + * and optionally a return result collector class type. + * Signal callbacks can be added with operator+= to a signal and removed with + * operator-=, using + * a callback connection ID return by operator+= as argument. + * The callbacks of a signal are invoked with the emit() method and arguments + * according to the signature. + * The result returned by emit() depends on the signal collector class. By + * default, the result of + * the last callback is returned from emit(). Collectors can be implemented to + * accumulate callback + * results or to halt a running emissions in correspondance to callback results. + * The signal implementation is safe against recursion, so callbacks may be + * removed and + * added during a signal emission and recursive emit() calls are also safe. + * The overhead of an unused signal is intentionally kept very low, around the + * size of a single pointer. + * Note that the Signal template types is non-copyable. + */ +template <typename SignalSignature, + class Collector = Lib::CollectorDefault< + typename std::function<SignalSignature>::result_type>> +struct Signal /*final*/ : Lib::ProtoSignal<SignalSignature, Collector> { + typedef Lib::ProtoSignal<SignalSignature, Collector> ProtoSignal; + typedef typename ProtoSignal::CbFunction CbFunction; + /// Signal constructor, supports a default callback as argument. + Signal(const CbFunction &method = CbFunction()) : ProtoSignal(method) {} +}; + +/// This function creates a std::function by binding @a object to the member +/// function pointer @a method. +template <class Instance, class Class, class R, class... Args> +std::function<R(Args...)> slot(Instance &object, R (Class::*method)(Args...)) { + return [&object, method](Args... args) { return (object.*method)(args...); }; +} + +/// This function creates a std::function by binding @a object to the member +/// function pointer @a method. +template <class Class, class R, class... Args> +std::function<R(Args...)> slot(Class *object, R (Class::*method)(Args...)) { + return [object, method](Args... args) { return (object->*method)(args...); }; +} + +/// Keep signal emissions going while all handlers return !0 (true). +template <typename Result> +struct CollectorUntil0 { + typedef Result CollectorResult; + explicit CollectorUntil0() : result_() {} + const CollectorResult &result() { return result_; } + inline bool operator()(Result r) { + result_ = r; + return result_ ? true : false; + } + + private: + CollectorResult result_; +}; + +/// Keep signal emissions going while all handlers return 0 (false). +template <typename Result> +struct CollectorWhile0 { + typedef Result CollectorResult; + explicit CollectorWhile0() : result_() {} + const CollectorResult &result() { return result_; } + inline bool operator()(Result r) { + result_ = r; + return result_ ? false : true; + } + + private: + CollectorResult result_; +}; + +/// CollectorVector returns the result of the all signal handlers from a signal +/// emission in a std::vector. +template <typename Result> +struct CollectorVector { + typedef std::vector<Result> CollectorResult; + const CollectorResult &result() { return result_; } + inline bool operator()(Result r) { + result_.push_back(r); + return true; + } + + private: + CollectorResult result_; +}; + +} // Simple + +#endif // SIMPLE_SIGNAL_H__ + +#ifdef ENABLE_SIMPLE_SIGNAL_TESTS + +#include <string> +#include <stdarg.h> +#include <time.h> +#include <sys/time.h> + +#ifdef __MACH__ +#include <mach/clock.h> +#include <mach/mach.h> +#endif + +static std::string string_printf(const char *format, ...) + __attribute__((__format__(__printf__, 1, 2))); +static std::string string_printf(const char *format, ...) { + std::string result; + char *str = 0; + va_list args; + va_start(args, format); + if (vasprintf(&str, format, args) >= 0) result = str; + va_end(args); + if (str) free(str); + return result; +} + +static uint64_t timestamp_benchmark() { + struct timespec tp = {0, 0}; + +#ifdef __MACH__ // OS X does not have clock_gettime, use clock_get_time + clock_serv_t cclock; + mach_timespec_t mts; + host_get_clock_service(mach_host_self(), CALENDAR_CLOCK, &cclock); + clock_get_time(cclock, &mts); + mach_port_deallocate(mach_task_self(), cclock); + tp.tv_sec = mts.tv_sec; + tp.tv_nsec = mts.tv_nsec; +#else + clock_gettime(CLOCK_REALTIME, &tp); +#endif + uint64_t stamp = tp.tv_sec * 1000000000ULL + tp.tv_nsec; + return stamp; +} + +struct TestCounter { + static uint64_t get(); + static void set(uint64_t); + static void add2(void *, uint64_t); +}; + +namespace { // Anon +void (*test_counter_add2)(void *, uint64_t) = + TestCounter::add2; // external symbol to prevent easy inlining +static uint64_t test_counter_var = 0; +} // Anon + +class BasicSignalTests { + static std::string accu; + struct Foo { + char foo_bool(float f, int i, std::string s) { + accu += string_printf("Foo: %.2f\n", f + i + s.size()); + return true; + } + }; + static char float_callback(float f, int, std::string) { + accu += string_printf("float: %.2f\n", f); + return 0; + } + + public: + static void run() { + accu = ""; + Simple::Signal<char(float, int, std::string)> sig1; + size_t id1 = sig1.connect(float_callback); + size_t id2 = sig1.connect([](float, int i, std::string) { + accu += string_printf("int: %d\n", i); + return 0; + }); + size_t id3 = sig1.connect([](float, int, const std::string &s) { + accu += string_printf("string: %s\n", s.c_str()); + return 0; + }); + sig1.emit(.3, 4, "huhu"); + bool success; + success = sig1.disconnect(id1); + assert(success == true); + success = sig1.disconnect(id1); + assert(success == false); + success = sig1.disconnect(id2); + assert(success == true); + success = sig1.disconnect(id3); + assert(success == true); + success = sig1.disconnect(id3); + assert(success == false); + success = sig1.disconnect(id2); + assert(success == false); + Foo foo; + sig1.connect(Simple::slot(foo, &Foo::foo_bool)); + sig1.connect(Simple::slot(&foo, &Foo::foo_bool)); + sig1.emit(.5, 1, "12"); + + Simple::Signal<void(std::string, int)> sig2; + sig2.connect([](std::string msg, + int) { accu += string_printf("msg: %s", msg.c_str()); }); + sig2.connect([](std::string, + int d) { accu += string_printf(" *%d*\n", d); }); + sig2.emit("in sig2", 17); + + accu += "DONE"; + + const char *expected = + "float: 0.30\n" + "int: 4\n" + "string: huhu\n" + "Foo: 3.50\n" + "Foo: 3.50\n" + "msg: in sig2 *17*\n" + "DONE"; + assert(accu == expected); + } +}; +std::string BasicSignalTests::accu; + +class TestCollectorVector { + static int handler1() { return 1; } + static int handler42() { return 42; } + static int handler777() { return 777; } + + public: + static void run() { + Simple::Signal<int(), Simple::CollectorVector<int>> sig_vector; + sig_vector.connect(handler777); + sig_vector.connect(handler42); + sig_vector.connect(handler1); + sig_vector.connect(handler42); + sig_vector.connect(handler777); + std::vector<int> results = sig_vector.emit(); + const std::vector<int> reference = {777, 42, 1, 42, 777, }; + assert(results == reference); + } +}; + +class TestCollectorUntil0 { + bool check1, check2; + TestCollectorUntil0() : check1(0), check2(0) {} + bool handler_true() { + check1 = true; + return true; + } + bool handler_false() { + check2 = true; + return false; + } + bool handler_abort() { abort(); } + + public: + static void run() { + TestCollectorUntil0 self; + Simple::Signal<bool(), Simple::CollectorUntil0<bool>> sig_until0; + sig_until0.connect(Simple::slot(self, &TestCollectorUntil0::handler_true)); + sig_until0.connect(Simple::slot(self, &TestCollectorUntil0::handler_false)); + sig_until0.connect(Simple::slot(self, &TestCollectorUntil0::handler_abort)); + assert(!self.check1 && !self.check2); + const bool result = sig_until0.emit(); + assert(!result && self.check1 && self.check2); + } +}; + +class TestCollectorWhile0 { + bool check1, check2; + TestCollectorWhile0() : check1(0), check2(0) {} + bool handler_0() { + check1 = true; + return false; + } + bool handler_1() { + check2 = true; + return true; + } + bool handler_abort() { abort(); } + + public: + static void run() { + TestCollectorWhile0 self; + Simple::Signal<bool(), Simple::CollectorWhile0<bool>> sig_while0; + sig_while0.connect(Simple::slot(self, &TestCollectorWhile0::handler_0)); + sig_while0.connect(Simple::slot(self, &TestCollectorWhile0::handler_1)); + sig_while0.connect(Simple::slot(self, &TestCollectorWhile0::handler_abort)); + assert(!self.check1 && !self.check2); + const bool result = sig_while0.emit(); + assert(result == true && self.check1 && self.check2); + } +}; + +static void bench_simple_signal() { + Simple::Signal<void(void *, uint64_t)> sig_increment; + sig_increment.connect(test_counter_add2); + const uint64_t start_counter = TestCounter::get(); + const uint64_t benchstart = timestamp_benchmark(); + uint64_t i; + for (i = 0; i < 999999; i++) { + sig_increment.emit(0, 1); + } + const uint64_t benchdone = timestamp_benchmark(); + const uint64_t end_counter = TestCounter::get(); + assert(end_counter - start_counter == i); + printf("OK\n Benchmark: Simple::Signal: %fns per emission (size=%zu): ", + size_t(benchdone - benchstart) * 1.0 / size_t(i), + sizeof(sig_increment)); +} + +static void bench_callback_loop() { + void (*counter_increment)(void *, uint64_t) = test_counter_add2; + const uint64_t start_counter = TestCounter::get(); + const uint64_t benchstart = timestamp_benchmark(); + uint64_t i; + for (i = 0; i < 999999; i++) { + counter_increment(0, 1); + } + const uint64_t benchdone = timestamp_benchmark(); + const uint64_t end_counter = TestCounter::get(); + assert(end_counter - start_counter == i); + printf("OK\n Benchmark: callback loop: %fns per round: ", + size_t(benchdone - benchstart) * 1.0 / size_t(i)); +} + +uint64_t TestCounter::get() { return test_counter_var; } + +void TestCounter::set(uint64_t v) { test_counter_var = v; } + +void TestCounter::add2(void *, uint64_t v) { test_counter_var += v; } + +int main(int argc, char *argv[]) { + printf("Signal/Basic Tests: "); + BasicSignalTests::run(); + printf("OK\n"); + + printf("Signal/CollectorVector: "); + TestCollectorVector::run(); + printf("OK\n"); + + printf("Signal/CollectorUntil0: "); + TestCollectorUntil0::run(); + printf("OK\n"); + + printf("Signal/CollectorWhile0: "); + TestCollectorWhile0::run(); + printf("OK\n"); + + printf("Signal/Benchmark: Simple::Signal: "); + bench_simple_signal(); + printf("OK\n"); + + printf("Signal/Benchmark: callback loop: "); + bench_callback_loop(); + printf("OK\n"); + + return 0; +} + +#endif // DISABLE_TESTS + +// g++ -Wall -O2 -std=gnu++0x -pthread simplesignal.cc -lrt && ./a.out |