123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792 |
- // This file is distributed under the BSD License.
- // See "license.txt" for details.
- // Copyright 2009-2012, Jonathan Turner ([email protected])
- // Copyright 2009-2017, Jason Turner ([email protected])
- // http://www.chaiscript.com
- // This is an open source non-commercial project. Dear PVS-Studio, please check it.
- // PVS-Studio Static Code Analyzer for C, C++ and C#: http://www.viva64.com
- /// \file
- /// This file contains utility functions for registration of STL container
- /// classes. The methodology used is based on the SGI STL concepts.
- /// http://www.sgi.com/tech/stl/table_of_contents.html
- #ifndef CHAISCRIPT_BOOTSTRAP_STL_HPP_
- #define CHAISCRIPT_BOOTSTRAP_STL_HPP_
- #include <functional>
- #include <memory>
- #include <stdexcept>
- #include <typeinfo>
- #include <vector>
- #include <algorithm>
- #include "bootstrap.hpp"
- #include "boxed_value.hpp"
- #include "dispatchkit.hpp"
- #include "operators.hpp"
- #include "proxy_constructors.hpp"
- #include "register_function.hpp"
- #include "type_info.hpp"
- namespace chaiscript
- {
- namespace bootstrap
- {
- namespace standard_library
- {
- /// Bidir_Range, based on the D concept of ranges.
- /// \todo Update the Range code to base its capabilities on
- /// the user_typetraits of the iterator passed in
- template<typename Container, typename IterType>
- struct Bidir_Range
- {
- typedef Container container_type;
- Bidir_Range(Container &c)
- : m_begin(c.begin()), m_end(c.end())
- {
- }
- bool empty() const
- {
- return m_begin == m_end;
- }
- void pop_front()
- {
- if (empty())
- {
- throw std::range_error("Range empty");
- }
- ++m_begin;
- }
- void pop_back()
- {
- if (empty())
- {
- throw std::range_error("Range empty");
- }
- --m_end;
- }
- decltype(auto) front() const
- {
- if (empty())
- {
- throw std::range_error("Range empty");
- }
- return (*m_begin);
- }
- decltype(auto) back() const
- {
- if (empty())
- {
- throw std::range_error("Range empty");
- }
- auto pos = m_end;
- --pos;
- return (*(pos));
- }
- IterType m_begin;
- IterType m_end;
- };
- namespace detail {
- template<typename T>
- size_t count(const T &t_target, const typename T::key_type &t_key)
- {
- return t_target.count(t_key);
- }
- template<typename T>
- void insert(T &t_target, const T &t_other)
- {
- t_target.insert(t_other.begin(), t_other.end());
- }
- template<typename T>
- void insert_ref(T &t_target, const typename T::value_type &t_val)
- {
- t_target.insert(t_val);
- }
- /// Add Bidir_Range support for the given ContainerType
- template<typename Bidir_Type>
- void input_range_type_impl(const std::string &type, Module& m)
- {
- m.add(user_type<Bidir_Type>(), type + "_Range");
- copy_constructor<Bidir_Type>(type + "_Range", m);
- m.add(constructor<Bidir_Type (typename Bidir_Type::container_type &)>(), "range_internal");
- m.add(fun(&Bidir_Type::empty), "empty");
- m.add(fun(&Bidir_Type::pop_front), "pop_front");
- m.add(fun(&Bidir_Type::front), "front");
- m.add(fun(&Bidir_Type::pop_back), "pop_back");
- m.add(fun(&Bidir_Type::back), "back");
- }
- /// Algorithm for inserting at a specific position into a container
- template<typename Type>
- void insert_at(Type &container, int pos, const typename Type::value_type &v)
- {
- auto itr = container.begin();
- auto end = container.end();
- if (pos < 0 || std::distance(itr, end) < pos)
- {
- throw std::range_error("Cannot insert past end of range");
- }
- std::advance(itr, pos);
- container.insert(itr, v);
- }
- /// Algorithm for erasing a specific position from a container
- template<typename Type>
- void erase_at(Type &container, int pos)
- {
- auto itr = container.begin();
- auto end = container.end();
- if (pos < 0 || std::distance(itr, end) < (pos-1))
- {
- throw std::range_error("Cannot erase past end of range");
- }
- std::advance(itr, pos);
- container.erase(itr);
- }
- }
- template<typename ContainerType>
- void input_range_type(const std::string &type, Module& m)
- {
- detail::input_range_type_impl<Bidir_Range<ContainerType, typename ContainerType::iterator> >(type,m);
- detail::input_range_type_impl<Bidir_Range<const ContainerType, typename ContainerType::const_iterator> >("Const_" + type,m);
- }
- template<typename ContainerType>
- ModulePtr input_range_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- input_range_type<ContainerType>(type, *m);
- return m;
- }
- /// Add random_access_container concept to the given ContainerType
- /// http://www.sgi.com/tech/stl/RandomAccessContainer.html
- template<typename ContainerType>
- void random_access_container_type(const std::string &/*type*/, Module& m)
- {
- //In the interest of runtime safety for the m, we prefer the at() method for [] access,
- //to throw an exception in an out of bounds condition.
- m.add(
- fun(
- [](ContainerType &c, int index) -> typename ContainerType::reference {
- /// \todo we are prefering to keep the key as 'int' to avoid runtime conversions
- /// during dispatch. reevaluate
- return c.at(static_cast<typename ContainerType::size_type>(index));
- }), "[]");
- m.add(
- fun(
- [](const ContainerType &c, int index) -> typename ContainerType::const_reference {
- /// \todo we are prefering to keep the key as 'int' to avoid runtime conversions
- /// during dispatch. reevaluate
- return c.at(static_cast<typename ContainerType::size_type>(index));
- }), "[]");
- }
- template<typename ContainerType>
- ModulePtr random_access_container_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- random_access_container_type<ContainerType>(type, *m);
- return m;
- }
- /// Add assignable concept to the given ContainerType
- /// http://www.sgi.com/tech/stl/Assignable.html
- template<typename ContainerType>
- void assignable_type(const std::string &type, Module& m)
- {
- copy_constructor<ContainerType>(type, m);
- operators::assign<ContainerType>(m);
- }
- template<typename ContainerType>
- ModulePtr assignable_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- assignable_type<ContainerType>(type, *m);
- return m;
- }
- /// Add container resize concept to the given ContainerType
- /// http://www.cplusplus.com/reference/stl/
- template<typename ContainerType>
- void resizable_type(const std::string &/*type*/, Module& m)
- {
- m.add(fun([](ContainerType *a, typename ContainerType::size_type n, const typename ContainerType::value_type& val) { return a->resize(n, val); } ), "resize");
- m.add(fun([](ContainerType *a, typename ContainerType::size_type n) { return a->resize(n); } ), "resize");
- }
- template<typename ContainerType>
- ModulePtr resizable_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- resizable_type<ContainerType>(type, *m);
- return m;
- }
- /// Add container reserve concept to the given ContainerType
- /// http://www.cplusplus.com/reference/stl/
- template<typename ContainerType>
- void reservable_type(const std::string &/*type*/, Module& m)
- {
- m.add(fun([](ContainerType *a, typename ContainerType::size_type n) { return a->reserve(n); } ), "reserve");
- m.add(fun([](const ContainerType *a) { return a->capacity(); } ), "capacity");
- }
- template<typename ContainerType>
- ModulePtr reservable_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- reservable_type<ContainerType>(type, *m);
- return m;
- }
- /// Add container concept to the given ContainerType
- /// http://www.sgi.com/tech/stl/Container.html
- template<typename ContainerType>
- void container_type(const std::string &/*type*/, Module& m)
- {
- m.add(fun([](const ContainerType *a) { return a->size(); } ), "size");
- m.add(fun([](const ContainerType *a) { return a->empty(); } ), "empty");
- m.add(fun([](ContainerType *a) { a->clear(); } ), "clear");
- }
- template <typename ContainerType>
- ModulePtr container_type(const std::string& type)
- {
- auto m = std::make_shared<Module>();
- container_type<ContainerType>(type, *m);
- return m;
- }
- /// Add default constructable concept to the given Type
- /// http://www.sgi.com/tech/stl/DefaultConstructible.html
- template<typename Type>
- void default_constructible_type(const std::string &type, Module& m)
- {
- m.add(constructor<Type ()>(), type);
- }
- template <typename Type>
- ModulePtr default_constructible_type(const std::string& type)
- {
- auto m = std::make_shared<Module>();
- default_constructible_type<Type>(type, *m);
- return m;
- }
- /// Add sequence concept to the given ContainerType
- /// http://www.sgi.com/tech/stl/Sequence.html
- template<typename ContainerType>
- void sequence_type(const std::string &/*type*/, Module& m)
- {
- m.add(fun(&detail::insert_at<ContainerType>),
- []()->std::string{
- if (typeid(typename ContainerType::value_type) == typeid(Boxed_Value)) {
- return "insert_ref_at";
- } else {
- return "insert_at";
- }
- }());
- m.add(fun(&detail::erase_at<ContainerType>), "erase_at");
- }
- template <typename ContainerType>
- ModulePtr sequence_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- sequence_type<ContainerType>(type, *m);
- return m;
- }
- /// Add back insertion sequence concept to the given ContainerType
- /// http://www.sgi.com/tech/stl/BackInsertionSequence.html
- template<typename ContainerType>
- void back_insertion_sequence_type(const std::string &type, Module& m)
- {
- m.add(fun([](ContainerType &container)->decltype(auto){
- if (container.empty()) {
- throw std::range_error("Container empty");
- } else {
- return (container.back());
- }
- }
- )
- , "back");
- m.add(fun([](const ContainerType &container)->decltype(auto){
- if (container.empty()) {
- throw std::range_error("Container empty");
- } else {
- return (container.back());
- }
- }
- )
- , "back");
- typedef void (ContainerType::*push_back)(const typename ContainerType::value_type &);
- m.add(fun(static_cast<push_back>(&ContainerType::push_back)),
- [&]()->std::string{
- if (typeid(typename ContainerType::value_type) == typeid(Boxed_Value)) {
- m.eval(
- "# Pushes the second value onto the container while making a clone of the value\n"
- "def push_back(" + type + " container, x)\n"
- "{ \n"
- " if (x.is_var_return_value()) {\n"
- " x.reset_var_return_value() \n"
- " container.push_back_ref(x) \n"
- " } else { \n"
- " container.push_back_ref(clone(x)); \n"
- " }\n"
- "} \n"
- );
- return "push_back_ref";
- } else {
- return "push_back";
- }
- }());
- m.add(fun(&ContainerType::pop_back), "pop_back");
- }
- template<typename ContainerType>
- ModulePtr back_insertion_sequence_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- back_insertion_sequence_type<ContainerType>(type, *m);
- return m;
- }
- /// Front insertion sequence
- /// http://www.sgi.com/tech/stl/FrontInsertionSequence.html
- template<typename ContainerType>
- void front_insertion_sequence_type(const std::string &type, Module& m)
- {
- typedef void (ContainerType::*push_ptr)(typename ContainerType::const_reference);
- typedef void (ContainerType::*pop_ptr)();
- m.add(fun([](ContainerType &container)->decltype(auto){
- if (container.empty()) {
- throw std::range_error("Container empty");
- } else {
- return (container.front());
- }
- }
- )
- , "front");
- m.add(fun([](const ContainerType &container)->decltype(auto){
- if (container.empty()) {
- throw std::range_error("Container empty");
- } else {
- return (container.front());
- }
- }
- )
- , "front");
- m.add(fun(static_cast<push_ptr>(&ContainerType::push_front)),
- [&]()->std::string{
- if (typeid(typename ContainerType::value_type) == typeid(Boxed_Value)) {
- m.eval(
- "# Pushes the second value onto the front of container while making a clone of the value\n"
- "def push_front(" + type + " container, x)\n"
- "{ \n"
- " if (x.is_var_return_value()) {\n"
- " x.reset_var_return_value() \n"
- " container.push_front_ref(x) \n"
- " } else { \n"
- " container.push_front_ref(clone(x)); \n"
- " }\n"
- "} \n"
- );
- return "push_front_ref";
- } else {
- return "push_front";
- }
- }());
- m.add(fun(static_cast<pop_ptr>(&ContainerType::pop_front)), "pop_front");
- }
- template<typename ContainerType>
- ModulePtr front_insertion_sequence_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- front_insertion_sequence_type<ContainerType>(type, *m);
- return m;
- }
- /// bootstrap a given PairType
- /// http://www.sgi.com/tech/stl/pair.html
- template<typename PairType>
- void pair_type(const std::string &type, Module& m)
- {
- m.add(user_type<PairType>(), type);
- m.add(fun(&PairType::first), "first");
- m.add(fun(&PairType::second), "second");
- basic_constructors<PairType>(type, m);
- m.add(constructor<PairType (const typename PairType::first_type &, const typename PairType::second_type &)>(), type);
- }
- template<typename PairType>
- ModulePtr pair_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- pair_type<PairType>(type, *m);
- return m;
- }
- /// Add pair associative container concept to the given ContainerType
- /// http://www.sgi.com/tech/stl/PairAssociativeContainer.html
- template<typename ContainerType>
- void pair_associative_container_type(const std::string &type, Module& m)
- {
- pair_type<typename ContainerType::value_type>(type + "_Pair", m);
- }
- template<typename ContainerType>
- ModulePtr pair_associative_container_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- pair_associative_container_type<ContainerType>(type, *m);
- return m;
- }
- /// Add unique associative container concept to the given ContainerType
- /// http://www.sgi.com/tech/stl/UniqueAssociativeContainer.html
- template<typename ContainerType>
- void unique_associative_container_type(const std::string &/*type*/, Module& m)
- {
- m.add(fun(detail::count<ContainerType>), "count");
- typedef size_t (ContainerType::*erase_ptr)(const typename ContainerType::key_type &);
- m.add(fun(static_cast<erase_ptr>(&ContainerType::erase)), "erase");
- m.add(fun(&detail::insert<ContainerType>), "insert");
- m.add(fun(&detail::insert_ref<ContainerType>),
- []()->std::string{
- if (typeid(typename ContainerType::mapped_type) == typeid(Boxed_Value)) {
- return "insert_ref";
- } else {
- return "insert";
- }
- }());
- }
- template<typename ContainerType>
- ModulePtr unique_associative_container_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- unique_associative_container_type<ContainerType>(type, *m);
- return m;
- }
- /// Add a MapType container
- /// http://www.sgi.com/tech/stl/Map.html
- template<typename MapType>
- void map_type(const std::string &type, Module& m)
- {
- m.add(user_type<MapType>(), type);
- typedef typename MapType::mapped_type &(MapType::*elem_access)(const typename MapType::key_type &);
- typedef const typename MapType::mapped_type &(MapType::*const_elem_access)(const typename MapType::key_type &) const;
- m.add(fun(static_cast<elem_access>(&MapType::operator[])), "[]");
- m.add(fun(static_cast<elem_access>(&MapType::at)), "at");
- m.add(fun(static_cast<const_elem_access>(&MapType::at)), "at");
- if (typeid(MapType) == typeid(std::map<std::string, Boxed_Value>))
- {
- m.eval(R"(
- def Map::`==`(Map rhs) {
- if ( rhs.size() != this.size() ) {
- return false;
- } else {
- auto r1 = range(this);
- auto r2 = range(rhs);
- while (!r1.empty())
- {
- if (!eq(r1.front().first, r2.front().first) || !eq(r1.front().second, r2.front().second))
- {
- return false;
- }
- r1.pop_front();
- r2.pop_front();
- }
- true;
- }
- } )"
- );
- }
- container_type<MapType>(type, m);
- default_constructible_type<MapType>(type, m);
- assignable_type<MapType>(type, m);
- unique_associative_container_type<MapType>(type, m);
- pair_associative_container_type<MapType>(type, m);
- input_range_type<MapType>(type, m);
- }
- template<typename MapType>
- ModulePtr map_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- map_type<MapType>(type, *m);
- return m;
- }
- /// http://www.sgi.com/tech/stl/List.html
- template<typename ListType>
- void list_type(const std::string &type, Module& m)
- {
- m.add(user_type<ListType>(), type);
- front_insertion_sequence_type<ListType>(type, m);
- back_insertion_sequence_type<ListType>(type, m);
- sequence_type<ListType>(type, m);
- resizable_type<ListType>(type, m);
- container_type<ListType>(type, m);
- default_constructible_type<ListType>(type, m);
- assignable_type<ListType>(type, m);
- input_range_type<ListType>(type, m);
- }
- template<typename ListType>
- ModulePtr list_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- list_type<ListType>(type, m);
- return m;
- }
- /// Create a vector type with associated concepts
- /// http://www.sgi.com/tech/stl/Vector.html
- template<typename VectorType>
- void vector_type(const std::string &type, Module& m)
- {
- m.add(user_type<VectorType>(), type);
- m.add(fun([](VectorType &container)->decltype(auto){
- if (container.empty()) {
- throw std::range_error("Container empty");
- } else {
- return (container.front());
- }
- }
- )
- , "front");
- m.add(fun([](const VectorType &container)->decltype(auto){
- if (container.empty()) {
- throw std::range_error("Container empty");
- } else {
- return (container.front());
- }
- }
- )
- , "front");
- back_insertion_sequence_type<VectorType>(type, m);
- sequence_type<VectorType>(type, m);
- random_access_container_type<VectorType>(type, m);
- resizable_type<VectorType>(type, m);
- reservable_type<VectorType>(type, m);
- container_type<VectorType>(type, m);
- default_constructible_type<VectorType>(type, m);
- assignable_type<VectorType>(type, m);
- input_range_type<VectorType>(type, m);
- if (typeid(VectorType) == typeid(std::vector<Boxed_Value>))
- {
- m.eval(R"(
- def Vector::`==`(Vector rhs) {
- if ( rhs.size() != this.size() ) {
- return false;
- } else {
- auto r1 = range(this);
- auto r2 = range(rhs);
- while (!r1.empty())
- {
- if (!eq(r1.front(), r2.front()))
- {
- return false;
- }
- r1.pop_front();
- r2.pop_front();
- }
- true;
- }
- } )"
- );
- }
- }
- template<typename VectorType>
- ModulePtr vector_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- vector_type<VectorType>(type, *m);
- return m;
- }
- /// Add a String container
- /// http://www.sgi.com/tech/stl/basic_string.html
- template<typename String>
- void string_type(const std::string &type, Module& m)
- {
- m.add(user_type<String>(), type);
- operators::addition<String>(m);
- operators::assign_sum<String>(m);
- opers_comparison<String>(m);
- random_access_container_type<String>(type, m);
- sequence_type<String>(type, m);
- default_constructible_type<String>(type, m);
- // container_type<String>(type, m);
- assignable_type<String>(type, m);
- input_range_type<String>(type, m);
- //Special case: add push_back to string (which doesn't support other back_insertion operations
- m.add(fun(&String::push_back),
- []()->std::string{
- if (typeid(typename String::value_type) == typeid(Boxed_Value)) {
- return "push_back_ref";
- } else {
- return "push_back";
- }
- }());
- m.add(fun([](const String *s, const String &f, size_t pos) { return s->find(f, pos); } ), "find");
- m.add(fun([](const String *s, typename String::value_type c, size_t pos) { return s->find(c, pos); } ), "find");
-
- m.add(fun([](const String *s, const String &f, size_t pos) { return s->rfind(f, pos); } ), "rfind");
- m.add(fun([](const String *s, typename String::value_type c, size_t pos) { return s->rfind(c, pos); } ), "rfind");
-
- m.add(fun([](const String *s, const String &f, size_t pos) { return s->find_first_of(f, pos); } ), "find_first_of");
- m.add(fun([](const String *s, typename String::value_type c, size_t pos) { return s->find_first_of(c, pos); } ), "find_first_of");
-
- m.add(fun([](const String *s, const String &f, size_t pos) { return s->find_last_of(f, pos); } ), "find_last_of");
- m.add(fun([](const String *s, typename String::value_type c, size_t pos) { return s->find_last_of(c, pos); } ), "find_last_of");
-
- m.add(fun([](const String *s, const String &f, size_t pos) { return s->find_last_not_of(f, pos); } ), "find_last_not_of");
- m.add(fun([](const String *s, typename String::value_type c, size_t pos) { return s->find_last_not_of(c, pos); } ), "find_last_not_of");
-
- m.add(fun([](const String *s, const String &f, size_t pos) { return s->find_first_not_of(f, pos); } ), "find_first_not_of");
- m.add(fun([](const String *s, typename String::value_type c, size_t pos) { return s->find_first_not_of(c, pos); } ), "find_first_not_of");
- m.add(fun([](String *s, typename String::value_type c) -> decltype(auto) { return (*s += c); } ), "+=");
- m.add(fun([](String *s, typename String::value_type c) -> decltype(auto) { return (*s + c); } ), "+");
-
- m.add(fun([](String *s) { std::reverse(s->begin(), s->end()); } ), "reverse");
- m.add(fun([](String *s, const String &from, const String &to)
- {
- std::string newString;
- newString.reserve(s->length()); // avoids a few memory allocations
- std::string::size_type lastPos = 0;
- std::string::size_type findPos;
- while (std::string::npos != (findPos = s->find(from, lastPos)))
- {
- newString.append(*s, lastPos, findPos - lastPos);
- newString += to;
- lastPos = findPos + from.length();
- }
- // Care for the rest after last occurrence
- newString += s->substr(lastPos);
- return newString;
- }), "replace");
- m.add(fun([](String *s) { s->clear(); } ), "clear");
- m.add(fun([](const String *s) { return s->empty(); } ), "empty");
- m.add(fun([](const String *s) { return s->size(); } ), "size");
- m.add(fun([](const String *s) { return s->c_str(); } ), "c_str");
- m.add(fun([](const String *s) { return s->data(); } ), "data");
- m.add(fun([](const String *s, size_t pos, size_t len) { return s->substr(pos, len); } ), "substr");
- }
- template<typename String>
- ModulePtr string_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- string_type<String>(type, *m);
- return m;
- }
- /// Add a MapType container
- /// http://www.sgi.com/tech/stl/Map.html
- template<typename FutureType>
- void future_type(const std::string &type, Module& m)
- {
- m.add(user_type<FutureType>(), type);
- m.add(fun([](const FutureType &t) { return t.valid(); }), "valid");
- m.add(fun(&FutureType::get), "get");
- m.add(fun(&FutureType::wait), "wait");
- }
- template<typename FutureType>
- ModulePtr future_type(const std::string &type)
- {
- auto m = std::make_shared<Module>();
- future_type<FutureType>(type, *m);
- return m;
- }
- }
- }
- }
- #endif
|