cannam@148: // Copyright (c) 2013-2014 Sandstorm Development Group, Inc. and contributors cannam@148: // Licensed under the MIT License: cannam@148: // cannam@148: // Permission is hereby granted, free of charge, to any person obtaining a copy cannam@148: // of this software and associated documentation files (the "Software"), to deal cannam@148: // in the Software without restriction, including without limitation the rights cannam@148: // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell cannam@148: // copies of the Software, and to permit persons to whom the Software is cannam@148: // furnished to do so, subject to the following conditions: cannam@148: // cannam@148: // The above copyright notice and this permission notice shall be included in cannam@148: // all copies or substantial portions of the Software. cannam@148: // cannam@148: // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR cannam@148: // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, cannam@148: // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE cannam@148: // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER cannam@148: // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, cannam@148: // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN cannam@148: // THE SOFTWARE. cannam@148: cannam@148: #ifndef KJ_FUNCTION_H_ cannam@148: #define KJ_FUNCTION_H_ cannam@148: cannam@148: #if defined(__GNUC__) && !KJ_HEADER_WARNINGS cannam@148: #pragma GCC system_header cannam@148: #endif cannam@148: cannam@148: #include "memory.h" cannam@148: cannam@148: namespace kj { cannam@148: cannam@148: template cannam@148: class Function; cannam@148: // Function wrapper using virtual-based polymorphism. Use this when template polymorphism is cannam@148: // not possible. You can, for example, accept a Function as a parameter: cannam@148: // cannam@148: // void setFilter(Function filter); cannam@148: // cannam@148: // The caller of `setFilter()` may then pass any callable object as the parameter. The callable cannam@148: // object does not have to have the exact signature specified, just one that is "compatible" -- cannam@148: // i.e. the return type is covariant and the parameters are contravariant. cannam@148: // cannam@148: // Unlike `std::function`, `kj::Function`s are movable but not copyable, just like `kj::Own`. This cannam@148: // is to avoid unexpected heap allocation or slow atomic reference counting. cannam@148: // cannam@148: // When a `Function` is constructed from an lvalue, it captures only a reference to the value. cannam@148: // When constructed from an rvalue, it invokes the value's move constructor. So, for example: cannam@148: // cannam@148: // struct AddN { cannam@148: // int n; cannam@148: // int operator(int i) { return i + n; } cannam@148: // } cannam@148: // cannam@148: // Function f1 = AddN{2}; cannam@148: // // f1 owns an instance of AddN. It may safely be moved out cannam@148: // // of the local scope. cannam@148: // cannam@148: // AddN adder(2); cannam@148: // Function f2 = adder; cannam@148: // // f2 contains a reference to `adder`. Thus, it becomes invalid cannam@148: // // when `adder` goes out-of-scope. cannam@148: // cannam@148: // AddN adder2(2); cannam@148: // Function f3 = kj::mv(adder2); cannam@148: // // f3 owns an insatnce of AddN moved from `adder2`. f3 may safely cannam@148: // // be moved out of the local scope. cannam@148: // cannam@148: // Additionally, a Function may be bound to a class method using KJ_BIND_METHOD(object, methodName). cannam@148: // For example: cannam@148: // cannam@148: // class Printer { cannam@148: // public: cannam@148: // void print(int i); cannam@148: // void print(kj::StringPtr s); cannam@148: // }; cannam@148: // cannam@148: // Printer p; cannam@148: // cannam@148: // Function intPrinter = KJ_BIND_METHOD(p, print); cannam@148: // // Will call Printer::print(int). cannam@148: // cannam@148: // Function strPrinter = KJ_BIND_METHOD(p, print); cannam@148: // // Will call Printer::print(kj::StringPtr). cannam@148: // cannam@148: // Notice how KJ_BIND_METHOD is able to figure out which overload to use depending on the kind of cannam@148: // Function it is binding to. cannam@148: cannam@148: template cannam@148: class ConstFunction; cannam@148: // Like Function, but wraps a "const" (i.e. thread-safe) call. cannam@148: cannam@148: template cannam@148: class Function { cannam@148: public: cannam@148: template cannam@148: inline Function(F&& f): impl(heap>(kj::fwd(f))) {} cannam@148: Function() = default; cannam@148: cannam@148: // Make sure people don't accidentally end up wrapping a reference when they meant to return cannam@148: // a function. cannam@148: KJ_DISALLOW_COPY(Function); cannam@148: Function(Function&) = delete; cannam@148: Function& operator=(Function&) = delete; cannam@148: template Function(const Function&) = delete; cannam@148: template Function& operator=(const Function&) = delete; cannam@148: template Function(const ConstFunction&) = delete; cannam@148: template Function& operator=(const ConstFunction&) = delete; cannam@148: Function(Function&&) = default; cannam@148: Function& operator=(Function&&) = default; cannam@148: cannam@148: inline Return operator()(Params... params) { cannam@148: return (*impl)(kj::fwd(params)...); cannam@148: } cannam@148: cannam@148: Function reference() { cannam@148: // Forms a new Function of the same type that delegates to this Function by reference. cannam@148: // Therefore, this Function must outlive the returned Function, but otherwise they behave cannam@148: // exactly the same. cannam@148: cannam@148: return *impl; cannam@148: } cannam@148: cannam@148: private: cannam@148: class Iface { cannam@148: public: cannam@148: virtual Return operator()(Params... params) = 0; cannam@148: }; cannam@148: cannam@148: template cannam@148: class Impl final: public Iface { cannam@148: public: cannam@148: explicit Impl(F&& f): f(kj::fwd(f)) {} cannam@148: cannam@148: Return operator()(Params... params) override { cannam@148: return f(kj::fwd(params)...); cannam@148: } cannam@148: cannam@148: private: cannam@148: F f; cannam@148: }; cannam@148: cannam@148: Own impl; cannam@148: }; cannam@148: cannam@148: template cannam@148: class ConstFunction { cannam@148: public: cannam@148: template cannam@148: inline ConstFunction(F&& f): impl(heap>(kj::fwd(f))) {} cannam@148: ConstFunction() = default; cannam@148: cannam@148: // Make sure people don't accidentally end up wrapping a reference when they meant to return cannam@148: // a function. cannam@148: KJ_DISALLOW_COPY(ConstFunction); cannam@148: ConstFunction(ConstFunction&) = delete; cannam@148: ConstFunction& operator=(ConstFunction&) = delete; cannam@148: template ConstFunction(const ConstFunction&) = delete; cannam@148: template ConstFunction& operator=(const ConstFunction&) = delete; cannam@148: template ConstFunction(const Function&) = delete; cannam@148: template ConstFunction& operator=(const Function&) = delete; cannam@148: ConstFunction(ConstFunction&&) = default; cannam@148: ConstFunction& operator=(ConstFunction&&) = default; cannam@148: cannam@148: inline Return operator()(Params... params) const { cannam@148: return (*impl)(kj::fwd(params)...); cannam@148: } cannam@148: cannam@148: ConstFunction reference() const { cannam@148: // Forms a new ConstFunction of the same type that delegates to this ConstFunction by reference. cannam@148: // Therefore, this ConstFunction must outlive the returned ConstFunction, but otherwise they cannam@148: // behave exactly the same. cannam@148: cannam@148: return *impl; cannam@148: } cannam@148: cannam@148: private: cannam@148: class Iface { cannam@148: public: cannam@148: virtual Return operator()(Params... params) const = 0; cannam@148: }; cannam@148: cannam@148: template cannam@148: class Impl final: public Iface { cannam@148: public: cannam@148: explicit Impl(F&& f): f(kj::fwd(f)) {} cannam@148: cannam@148: Return operator()(Params... params) const override { cannam@148: return f(kj::fwd(params)...); cannam@148: } cannam@148: cannam@148: private: cannam@148: F f; cannam@148: }; cannam@148: cannam@148: Own impl; cannam@148: }; cannam@148: cannam@148: #if 1 cannam@148: cannam@148: namespace _ { // private cannam@148: cannam@148: template cannam@148: class BoundMethod; cannam@148: cannam@148: template ::*method)(Params...)> cannam@148: class BoundMethod::*)(Params...), method> { cannam@148: public: cannam@148: BoundMethod(T&& t): t(kj::fwd(t)) {} cannam@148: cannam@148: Return operator()(Params&&... params) { cannam@148: return (t.*method)(kj::fwd(params)...); cannam@148: } cannam@148: cannam@148: private: cannam@148: T t; cannam@148: }; cannam@148: cannam@148: template ::*method)(Params...) const> cannam@148: class BoundMethod::*)(Params...) const, method> { cannam@148: public: cannam@148: BoundMethod(T&& t): t(kj::fwd(t)) {} cannam@148: cannam@148: Return operator()(Params&&... params) const { cannam@148: return (t.*method)(kj::fwd(params)...); cannam@148: } cannam@148: cannam@148: private: cannam@148: T t; cannam@148: }; cannam@148: cannam@148: } // namespace _ (private) cannam@148: cannam@148: #define KJ_BIND_METHOD(obj, method) \ cannam@148: ::kj::_::BoundMethod::method), \ cannam@148: &::kj::Decay::method>(obj) cannam@148: // Macro that produces a functor object which forwards to the method `obj.name`. If `obj` is an cannam@148: // lvalue, the functor will hold a reference to it. If `obj` is an rvalue, the functor will cannam@148: // contain a copy (by move) of it. cannam@148: // cannam@148: // The current implementation requires that the method is not overloaded. cannam@148: // cannam@148: // TODO(someday): C++14's generic lambdas may be able to simplify this code considerably, and cannam@148: // probably make it work with overloaded methods. cannam@148: cannam@148: #else cannam@148: // Here's a better implementation of the above that doesn't work with GCC (but does with Clang) cannam@148: // because it uses a local class with a template method. Sigh. This implementation supports cannam@148: // overloaded methods. cannam@148: cannam@148: #define KJ_BIND_METHOD(obj, method) \ cannam@148: ({ \ cannam@148: typedef KJ_DECLTYPE_REF(obj) T; \ cannam@148: class F { \ cannam@148: public: \ cannam@148: inline F(T&& t): t(::kj::fwd(t)) {} \ cannam@148: template \ cannam@148: auto operator()(Params&&... params) \ cannam@148: -> decltype(::kj::instance().method(::kj::fwd(params)...)) { \ cannam@148: return t.method(::kj::fwd(params)...); \ cannam@148: } \ cannam@148: private: \ cannam@148: T t; \ cannam@148: }; \ cannam@148: (F(obj)); \ cannam@148: }) cannam@148: // Macro that produces a functor object which forwards to the method `obj.name`. If `obj` is an cannam@148: // lvalue, the functor will hold a reference to it. If `obj` is an rvalue, the functor will cannam@148: // contain a copy (by move) of it. cannam@148: cannam@148: #endif cannam@148: cannam@148: } // namespace kj cannam@148: cannam@148: #endif // KJ_FUNCTION_H_