1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243 |
- #ifndef value_hpp
- #define value_hpp
- #include <algorithm>
- #include <functional>
- #include <initializer_list>
- #include <iomanip>
- #include <iostream>
- #include <limits>
- #include <map>
- #include <sstream>
- #include <tuple>
- #include <typeindex>
- #include <utility>
- #include <vector>
- #include "log.hpp"
- namespace fv{
- namespace detail {
- template<typename T, int N, bool Done, typename... TYPES>
- struct _HomoTuple {
- typedef _HomoTuple<T, N, sizeof...(TYPES)+1==N, TYPES..., T> stype;
- typedef typename stype::type type;
- };
- template<typename T, int N, typename... TYPES>
- struct _HomoTuple<T, N, true, TYPES...> {
- typedef std::tuple<TYPES...> type;
- };
- }
- template<typename T, int N>
- struct HomoTuple {
- typedef detail::_HomoTuple<T, N, N==0> stype;
- typedef typename stype::type type;
- };
- namespace detail {
-
- template<typename Array, std::size_t... I>
- decltype(auto) a2t_impl(const Array& a, std::index_sequence<I...>){
- return std::make_tuple(a[I]...);
- }
- }
- template<typename T, std::size_t N, typename Indices = std::make_index_sequence<N>>
- decltype(auto) a2t(const std::array<T, N>& a){
- return detail::a2t_impl(a, Indices());
- }
- namespace detail {
-
- template<typename R, typename Tuple, std::size_t... Is>
- decltype(auto) t2v_impl(const Tuple& t, std::index_sequence<Is...>){
-
- return std::vector<R>({std::get<Is>(t)...});
- }
- }
- template<typename R, typename... ArgTypes>
- std::vector<R> t2v(const std::tuple<ArgTypes...>& t){
- return detail::t2v_impl<R, std::tuple<ArgTypes...>>(t, std::index_sequence_for<ArgTypes...>{});
- }
- namespace detail {
- template <class F, class Tuple, std::size_t... I>
- constexpr decltype(auto) call_impl(F &&f, Tuple &&t, std::index_sequence<I...>){
- return std::invoke(std::forward<F>(f), std::get<I>(std::forward<Tuple>(t))...);
- }
- }
- template <class F, class Tuple>
- constexpr decltype(auto) call(F &&f, Tuple &&t){
- return detail::call_impl(
- std::forward<F>(f), std::forward<Tuple>(t),
- std::make_index_sequence<std::tuple_size<std::decay_t<Tuple>>::value>{});
- }
- template<typename> class Function;
- class GenFunction {
- private:
- std::string name;
- std::string impl;
- protected:
- inline static bool in_register_function=false;
- public:
-
- inline static std::map<const std::string, GenFunction*> function_registry;
- GenFunction(const std::string& name, const std::string& impl)
- :name(name),
- impl(impl){ }
- virtual ~GenFunction() { };
- std::string& get_name(){
- return name;
- }
- std::string& get_impl(){
- return impl;
- }
-
- static std::string format_code(const std::string& code){
- std::stringstream code_out("");
- std::string command("echo \""+code+"\" | clang-format");
- char buffer[255];
- FILE *stream = popen(command.c_str(), "r");
- while (fgets(buffer, 255, stream) != NULL)
- code_out << buffer;
- if (pclose(stream) == 0)
- return code_out.str();
- else
- return code;
- }
- static std::string summary(){
- std::stringstream ss;
- ss << "The following functions have been registered" << std::endl;
- for(auto p : function_registry){
- if (p.second == nullptr) continue;
- ss << "FUNCTION::" << p.second->name << "@" << p.second << std::endl;
- ss << format_code(p.second->impl);
- }
- return ss.str();
- }
- template <typename T>
- static Function<T>& register_function(const std::string& name, std::function<T> f, const std::string& impl){
- in_register_function = true;
- Function<T>* func;
- if (GenFunction::function_registry[name] != nullptr){
- func = dynamic_cast<Function<T>*>(GenFunction::function_registry[name]);
- if (func == nullptr){
- ERROR("Trying to register function which has already been registered with a different type");
- }
- } else {
- func = new Function<T>(name, impl, f);
- GenFunction::function_registry[name] = func;
- }
- in_register_function = false;
- return *func;
- }
- template <typename T>
- static Function<T>& lookup_function(const std::string& name){
- if (GenFunction::function_registry[name] == nullptr){
- CRITICAL("Function \"" << name << "\" not previously registered", -1);
- } else {
- Function<T>* func = dynamic_cast<Function<T>*>(GenFunction::function_registry[name]);
- if (func == nullptr){
- CRITICAL("Function \"" << name << "\" request and register have mismatched types", -1);
- }
- return *GenFunction::function_registry[name];
- }
- }
- };
- template <typename R, typename... ArgTypes>
- class Function<R(ArgTypes...)> : public GenFunction {
- private:
- std::function<R(ArgTypes...)> f;
- public:
- Function(const std::string& name, const std::string& impl, std::function<R(ArgTypes...)> f)
- :GenFunction(name, impl), f(f){
- if (!in_register_function) {
- WARNING("Don't instantiate Function objects directly! Use GenFunction::register_function instead.");
- }
- }
- Function(const std::string& name, std::function<R(ArgTypes...)> f)
- :Function(name, "N/A", f){ }
- ~Function() { }
- R operator()(ArgTypes ...args){
- return f(args...);
- }
- };
- #define FUNC(f) f, #f
- template <typename T>
- class Value;
- class GenValue;
- typedef std::map<std::string, GenValue*> ValueSet;
- class GenValue{
- private:
-
- std::string name;
- protected:
-
- bool value_valid;
- void _reset(){
- this->value_valid = false;
- }
-
- inline static std::map<std::pair<const std::type_index, const std::string>, GenValue*> values;
-
- inline static std::map<std::pair<const std::type_index, const std::string>, GenValue*> aliases;
- bool logging_enabled;
- public:
- GenValue(const std::type_index&& ti, const std::string& name, const std::string& alias)
- :name(name), value_valid(false), logging_enabled(false){
- if (alias != "")
- INFO("Registered value: \"" << name << "\" with alias: \"" << alias << "\"");
- else
- INFO("Registered value: \"" << name);
- values[std::make_pair(ti,name)] = this;
- if (alias != "")
- GenValue::alias(ti, alias, this);
- }
- const std::string& get_name(){
- return name;
- }
-
- virtual void log() = 0;
- static void reset(){
- for (auto val : values){
- if (val.second != nullptr){
- val.second->_reset();
- }
- }
- }
- template<typename T>
- static Value<T>* get_value(const std::string& name){
- const std::type_index& ti = typeid(T);
- auto lookup_id = std::make_pair(ti,name);
- if (aliases[lookup_id] != nullptr)
- return (Value<T>*)aliases[lookup_id];
- else
- return (Value<T>*)values[lookup_id];
- }
- static void alias(const std::type_index& ti, const std::string& name, GenValue* value){
- auto lookup_id = std::make_pair(ti,name);
- if (aliases[lookup_id] != nullptr){
- WARNING("WARNING: alias \"" << name << "\" overrides previous entry.");
- }
- aliases[lookup_id] = value;
- }
- template<typename T>
- static void alias(const std::string& name, Value<T>* value){
- alias(typeid(T), name, value);
- }
- static std::string summary(){
- std::stringstream ss;
- ss << "The following values have been created:" << std::endl;
- for (auto item : values){
- auto& key = item.first;
- auto& value = item.second;
- if (value == nullptr) continue;
- ss << "\tVALUE::\"" << key.second << "\" at address " << value << std::endl;
- }
- ss << "And these aliases:" << std::endl;
- for (auto item : aliases){
- auto& key = item.first;
- auto& value = item.second;
- std::string orig("VOID");
- if (value == nullptr) continue;
- for (auto v_item : values){
- auto& v_value = v_item.second;
- if (v_value == value){
- orig = v_value->get_name();
- break;
- }
- }
- ss << "\tALIAS::\"" << key.second << "\" referring to \"" << orig << "\"" << std::endl;
- }
- return ss.str();
- }
- friend std::ostream& operator<<(std::ostream& os, const GenValue& gv);
- };
- std::ostream& operator<<(std::ostream& os, GenValue& gv){
- os << gv.get_name();
- return os;
- }
- template <typename T>
- class Value : public GenValue{
- protected:
- std::function<std::string(T)> value_to_string;
- public:
- Value(const std::string& name, const std::string& alias="")
- :value_to_string([](T){return "";}),
- GenValue(typeid(T), name, alias){ }
-
- virtual T& get_value() = 0;
- void enable_logging(const std::function<std::string(T)>& value_to_string = [](T){return "";}){
- logging_enabled = true;
- this->value_to_string = value_to_string;
- }
- void disable_logging(){
- logging_enabled = false;
- }
- };
- template <typename T>
- class ObservedValue : public Value<T>{
- private:
- T *val_ref;
- public:
- ObservedValue(const std::string& name, T* val_ref, const std::string& alias="")
- :Value<T>(name, alias),
- val_ref(val_ref){ }
- void log(){
- if(this->logging_enabled){
- INFO(this->get_name() << ": " << this->value_to_string(*val_ref));
- }
- }
- static std::string fmt_name(const std::string& name){
- return name;
- }
- T& get_value(){
- if (!this->value_valid){
- this->value_valid = true;
- this->log();
- }
- return *val_ref;
- }
- };
- template <typename T>
- class DerivedValue : public Value<T>{
- protected:
- T value;
-
- virtual void update_value() = 0;
- public:
- DerivedValue(const std::string& name, const std::string& alias="")
- :Value<T>(name, alias){ }
- void log(){
- if(this->logging_enabled){
- INFO(this->get_name() << ": " << this->value_to_string(value));
- }
- }
- T& get_value(){
- if (!this->value_valid){
- update_value();
- this->value_valid = true;
- this->log();
- }
- return value;
- }
- };
- template <typename T>
- class WrapperVector : public DerivedValue<std::vector<T> >{
- private:
- Value<int>* size;
- Value<T*>* data;
- void update_value(){
- int n = size->get_value();
- T* data_ref = data->get_value();
- this->value.assign(data_ref, data_ref+n);
- }
- public:
- static std::string fmt_name(Value<int>* size, Value<T*>* data){
- return "wrapper_vector("+size->get_name()+","+data->get_name()+")";
- }
- WrapperVector(Value<int>* size, Value<T*>* data, const std::string& alias="")
- :DerivedValue<std::vector<T> >(fmt_name(size,data), alias),
- size(size), data(data){ }
- };
- template <typename T1, typename T2>
- class Pair : public DerivedValue<std::pair<T1, T2> >{
- protected:
- std::pair<Value<T1>*, Value<T2>* > value_pair;
- void update_value(){
- this->value.first = value_pair.first->get_value();
- this->value.second = value_pair.second->get_value();
- }
- public:
- static std::string fmt_name(Value<T1> *value1, Value<T2> *value2){
- return "pair("+value1->get_name()+","+value2->get_name()+")";
- }
- Pair(Value<T1> *value1, Value<T2> *value2, const std::string alias="")
- :DerivedValue<std::pair<T1, T2> >(fmt_name(value1, value2), alias),
- value_pair(value1, value2){ }
- };
- template<typename... T> class _Zip;
- template<>
- class _Zip<> {
- protected:
- int _get_size(){
- return std::numeric_limits<int>::max();
- }
- std::tuple<> _get_at(int){
- return std::make_tuple();
- }
- std::string _get_name(){
- return "";
- }
- public:
- _Zip() { }
- };
- template<typename Head, typename... Tail>
- class _Zip<Head, Tail...> : private _Zip<Tail...> {
- protected:
- Value<std::vector<Head>>* head;
- int _get_size(){
- int this_size = head->get_value().size();
- int rest_size = _Zip<Tail...>::_get_size();
- return std::min(this_size, rest_size);
- }
- typename std::tuple<Head,Tail...> _get_at(int idx){
- auto tail_tuple = _Zip<Tail...>::_get_at(idx);
- return std::tuple_cat(std::make_tuple(head->get_value()[idx]),tail_tuple);
- }
- std::string _get_name(){
- return head->get_name()+","+_Zip<Tail...>::_get_name();
- }
- public:
- _Zip() { }
- _Zip(Value<std::vector<Head>>* head, Value<std::vector<Tail>>*... tail)
- : _Zip<Tail...>(tail...),
- head(head) { }
- };
- namespace impl {
- std::string zip_fmt_name(){
- return "";
- }
- template<typename Head>
- std::string zip_fmt_name(Value<std::vector<Head>>* head){
- return head->get_name();
- }
- template<typename Head1, typename Head2, typename... Tail>
- std::string zip_fmt_name(Value<std::vector<Head1>>* head1, Value<std::vector<Head2>>* head2, Value<std::vector<Tail>>*... tail){
- return head1->get_name() + "," + zip_fmt_name<Head2, Tail...>(head2, tail...);
- }
- }
- template <typename... ArgTypes>
- class Zip : public DerivedValue<std::vector<std::tuple<ArgTypes...>>>,
- private _Zip<ArgTypes...>{
- protected:
- void update_value(){
- this->value.clear();
- int size = _Zip<ArgTypes...>::_get_size();
- for(int i=0; i<size; i++){
- this->value.push_back(_Zip<ArgTypes...>::_get_at(i));
- }
- }
- public:
- static std::string fmt_name(Value<std::vector<ArgTypes>>*... args){
- return "zip("+zip_fmt_name(args...)+")";
- }
- Zip(Value<std::vector<ArgTypes>>*... args, const std::string& alias)
- :DerivedValue<std::vector<std::tuple<ArgTypes...>>>(fmt_name(args...), alias),
- _Zip<ArgTypes...>(args...) { }
- };
- template<typename> class Map;
- template <typename Ret, typename... ArgTypes>
- class Map<Ret(ArgTypes...)> : public DerivedValue<std::vector<Ret>>{
- private:
- typedef Value<std::vector<std::tuple<ArgTypes...>>> arg_type;
- Function<Ret(ArgTypes...)>& fn;
- arg_type* arg;
- void update_value(){
- this->value.clear();
- for(auto tup : arg->get_value()){
- this->value.push_back(call(fn,tup));
- }
- }
- public:
- static std::string fmt_name(Function<Ret(ArgTypes...)>& fn, arg_type* arg){
- return "map("+fn.get_name()+":"+arg->get_name()+")";
- }
- Map(Function<Ret(ArgTypes...)>& fn, arg_type* arg, const std::string& alias)
- :DerivedValue<std::vector<Ret>>(fmt_name(fn, arg), alias),
- fn(fn), arg(arg){ }
- };
- template<typename... T> class _Tuple;
- template<>
- class _Tuple<> {
- protected:
- std::tuple<> _get_value(){
- return std::make_tuple();
- }
- public:
- _Tuple() { }
- };
- template<typename Head, typename... Tail>
- class _Tuple<Head, Tail...> : private _Tuple<Tail...> {
- protected:
- Value<Head>* head;
- typename std::tuple<Head,Tail...> _get_value(){
- auto tail_tuple = _Tuple<Tail...>::_get_value();
- return std::tuple_cat(std::make_tuple(head->get_value()),tail_tuple);
- }
- public:
- _Tuple() { }
- _Tuple(Value<Head>* head, Value<Tail>*... tail)
- : _Tuple<Tail...>(tail...),
- head(head) { }
- };
- namespace impl {
- std::string tuple_fmt_name(){
- return "";
- }
- template<typename Head>
- std::string tuple_fmt_name(Value<Head>* head){
- return head->get_name();
- }
- template<typename Head1, typename Head2, typename... Tail>
- std::string tuple_fmt_name(Value<Head1>* head1, Value<Head2>* head2, Value<Tail>*... tail){
- return head1->get_name() + "," + tuple_fmt_name<Head2, Tail...>(head2, tail...);
- }
- }
- template <typename... ArgTypes>
- class Tuple : public DerivedValue<std::tuple<ArgTypes...>>,
- private _Tuple<ArgTypes...>{
- protected:
- void update_value(){
- this->value = _Tuple<ArgTypes...>::_get_value();
- }
- public:
- static std::string fmt_name(Value<ArgTypes>*... args){
- return "tuple("+impl::tuple_fmt_name(args...)+")";
- }
- Tuple(Value<ArgTypes>*... args, const std::string& alias)
- :DerivedValue<std::tuple<ArgTypes...>>(fmt_name(args...), alias),
- _Tuple<ArgTypes...>(args...) { }
- };
- template <size_t N, typename... ArgTypes>
- class DeTup : public DerivedValue<typename std::tuple_element<N, std::tuple<ArgTypes...>>::type>{
- Value<std::tuple<ArgTypes...>> tup;
- protected:
- void update_value(){
- this->value = std::get<N>(tup->get_value());
- }
- public:
- static std::string fmt_name(Value<std::tuple<ArgTypes...>>* tup){
- return "detup("+tup->get_name()+")";
- }
- DeTup(Value<std::tuple<ArgTypes...>>* tup, const std::string& alias)
- :DerivedValue<typename std::tuple_element<N, std::tuple<ArgTypes...>>::type>(fmt_name(tup), alias),
- tup(tup) { }
- };
- template <size_t N, typename... ArgTypes>
- class DeTupVector : public DerivedValue<std::vector<typename std::tuple_element<N, std::tuple<ArgTypes...>>::type>>{
- Value<std::vector<std::tuple<ArgTypes...>>>* tup;
- protected:
- void update_value(){
- this->value.clear();
- for( auto& t : tup->get_value()){
- this->value.push_back(std::get<N>(t));
- }
- }
- public:
- static std::string fmt_name(Value<std::vector<std::tuple<ArgTypes...>>>* tup){
- return "detup_vec("+tup->get_name()+")";
- }
- DeTupVector(Value<std::vector<std::tuple<ArgTypes...>>>* tup, const std::string& alias)
- :DerivedValue<std::vector<typename std::tuple_element<N, std::tuple<ArgTypes...>>::type>>(fmt_name(tup), alias),
- tup(tup) { }
- };
- template<typename> class Apply;
- template <typename Ret, typename... ArgTypes>
- class Apply<Ret(ArgTypes...)> : public DerivedValue<Ret>{
- private:
- Function<Ret(ArgTypes...)>& fn;
- Value<std::tuple<ArgTypes...>>* arg;
- void update_value(){
- auto &tup = arg->get_value();
- this->value = call(fn, tup);
- }
- public:
- static std::string fmt_name(Function<Ret(ArgTypes...)>& fn, Value<std::tuple<ArgTypes...>>* arg){
- return "apply("+fn.get_name()+":"+arg->get_name()+")";
- }
- Apply(Function<Ret(ArgTypes...)>& fn, Value<std::tuple<ArgTypes...>>* arg, const std::string& alias)
- :DerivedValue<Ret>(fmt_name(fn,arg), alias),
- fn(fn), arg(arg){ }
- };
- template<typename T>
- class Count : public DerivedValue<int>{
- private:
- Function<bool(T)>& selector;
- Value<std::vector<T> >* v;
- void update_value(){
- value = 0;
- for(auto val : v->get_value()){
- if(selector(val))
- value++;
- }
- }
- public:
- static std::string fmt_name(Function<bool(T)>& selector, Value<std::vector<T>>* v){
- return "count("+selector.get_name()+":"+v->get_name()+")";
- }
- Count(Function<bool(T)>& selector, Value<std::vector<T>>* v, const std::string alias)
- :DerivedValue<int>(fmt_name(selector,v), alias),
- selector(selector), v(v) { }
- };
- template<typename T>
- class Filter : public DerivedValue<std::vector<T>>{
- private:
- Function<bool(T)>& filter;
- Value<std::vector<T> >* v;
- void update_value(){
- this->value.clear();
- for(auto val : v->get_value()){
- if(this->filter(val))
- this->value.push_back(val);
- }
- }
- public:
- static std::string fmt_name(Function<bool(T)>& filter, Value<std::vector<T>>* v){
- return "filter("+filter.get_name()+":"+v->get_name()+")";
- }
- Filter(Function<bool(T)>& filter, Value<std::vector<T>>* v, const std::string alias)
- :DerivedValue<std::vector<T>>(fmt_name(filter,v), alias),
- filter(filter), v(v) { }
- };
- template<typename... ArgTypes>
- class TupFilter : public DerivedValue<std::vector<std::tuple<ArgTypes...>>>{
- private:
- typedef std::vector<std::tuple<ArgTypes...>> value_type;
- Function<bool(ArgTypes...)>& filter;
- Value<value_type>* arg;
- void update_value(){
- this->value.clear();
- for(auto val : arg->get_value()){
- if(call(filter,val))
- this->value.push_back(val);
- }
- }
- public:
- static std::string fmt_name(Function<bool(ArgTypes...)>& filter, Value<value_type>* arg){
- return "tup_filter("+filter.get_name()+":"+arg->get_name()+")";
- }
- TupFilter(Function<bool(ArgTypes...)>& filter, Value<value_type>* arg, const std::string alias)
- :DerivedValue<value_type>(fmt_name(filter, arg), alias),
- filter(filter), arg(arg) { }
- };
- template <typename T>
- class Reduce : public DerivedValue<T>{
- private:
- Function<T(std::vector<T>)>& reduce_fn;
- void update_value(){
- this->value = reduce_fn(v->get_value());
- }
- protected:
- Value<std::vector<T> >* v;
- public:
- static std::string fmt_name(Function<T(std::vector<T>)>& reduce_fn, Value<std::vector<T>>* v){
- return "reduce("+reduce_fn.get_name()+":"+v->get_name()+")";
- }
- Reduce(Function<T(std::vector<T>)>& reduce_fn, Value<std::vector<T> >* v, const std::string alias)
- :DerivedValue<T>(fmt_name(reduce_fn, v), alias),
- reduce_fn(reduce_fn), v(v) { }
- };
- template <typename T>
- class Max : public Reduce<T>{
- public:
- static std::string fmt_name(Value<std::vector<T>>* v){
- return "max("+v->get_name()+")";
- }
- Max(Value<std::vector<T>>* v, const std::string alias)
- :Reduce<T>(GenFunction::register_function<T(std::vector<T>)>("max",
- FUNC(([](std::vector<T> vec){
- return *std::max_element(vec.begin(), vec.end());}))),
- v, alias) { }
- };
- template <typename T>
- class Min : public Reduce<T>{
- public:
- static std::string fmt_name(Value<std::vector<T>>* v){
- return "min("+v->get_name()+")";
- }
- Min(Value<std::vector<T>>* v, const std::string alias)
- :Reduce<T>(GenFunction::register_function<T(std::vector<T>)>("min",
- FUNC(([](std::vector<T> vec){
- return *std::min_element(vec.begin(), vec.end());}))),
- v, alias) { }
- };
- template <typename T>
- class Mean : public Reduce<T>{
- public:
- static std::string fmt_name(Value<std::vector<T>>* v){
- return "mean("+v->get_name()+")";
- }
- Mean(Value<std::vector<T>>* v, const std::string alias)
- :Reduce<T>(GenFunction::register_function<T(std::vector<T>)>("mean",
- FUNC(([](std::vector<T> vec){
- int n = 0; T sum = 0;
- for (T e : vec){ n++; sum += e; }
- return n>0 ? sum / n : 0; }))),
- v, alias) { }
- };
- template <typename T>
- class Range : public Reduce<T>{
- public:
- static std::string fmt_name(Value<std::vector<T>>* v){
- return "range("+v->get_name()+")";
- }
- Range(Value<std::vector<T>>* v, const std::string alias)
- :Reduce<T>(GenFunction::register_function<T(std::vector<T>)>("range",
- FUNC(([](std::vector<T> vec){
- auto minmax = std::minmax_element(vec.begin(), vec.end());
- return (*minmax.second) - (*minmax.first); }))),
- v, alias) { }
- };
- template <typename T>
- class ElementOf : public Reduce<T>{
- public:
- ElementOf(Value<int>* index, Value<std::vector<T>>* v, const std::string alias)
- :Reduce<T>(GenFunction::register_function<T(std::vector<T>)>("elementOf",
- FUNC(([index](std::vector<T> vec){return vec[index->get_value()];}))),
- v, alias) { }
- };
- template <typename T>
- class ReduceIndex : public DerivedValue<std::pair<T, int> >{
- private:
- Function<std::pair<T,int>(std::vector<T>)>& reduce;
- Value<std::vector<T> >* v;
- void update_value(){
- this->value = reduce(v->get_value());
- }
- public:
- ReduceIndex(Function<std::pair<T,int>(std::vector<T>)>& reduce, Value<std::vector<T> >* v, const std::string alias="")
- :DerivedValue<T>("reduceIndexWith("+reduce.get_name()+":"+v->get_name()+")", alias),
- reduce(reduce), v(v) { }
- };
- template <typename T>
- class MaxIndex : public ReduceIndex<T>{
- public:
- MaxIndex(Value<std::vector<T>>* v, const std::string alias="")
- :ReduceIndex<T>(GenFunction::register_function<T(std::vector<T>)>("maxIndex",
- FUNC(([](std::vector<T> vec){
- auto elptr = std::max_element(vec.begin(), vec.end());
- return std::pair<T,int>(*elptr, int(elptr-vec.begin())); }))),
- v, alias) { }
- };
- template <typename T>
- class MinIndex : public ReduceIndex<T>{
- public:
- MinIndex(Value<std::vector<T>>* v, const std::string alias="")
- :ReduceIndex<T>(GenFunction::register_function<T(std::vector<T>)>("minIndex",
- FUNC(([](std::vector<T> vec){
- auto elptr = std::min_element(vec.begin(), vec.end());
- return std::pair<T,int>(*elptr, int(elptr-vec.begin())); }))),
- v, alias) { }
- };
- template <typename T, int Size>
- class Combinations : public DerivedValue<std::vector<typename HomoTuple<T,Size>::type>>{
- private:
- Value<std::vector<T>>* val;
- typedef typename HomoTuple<T,Size>::type tuple_type;
- void update_value(){
- auto& v = val->get_value();
- int data_size = v.size();
- this->value.clear();
- std::vector<bool> selector(data_size);
- std::fill(selector.begin(), selector.begin()+std::min({Size,data_size}), true);
- do {
- std::array<T, Size> perm;
- int idx = 0;
- for (int i=0; i<data_size; i++){
- if (selector[i]){
- perm[idx] = v[i];
- idx++;
- if (idx == Size) break;
- }
- }
- this->value.push_back(a2t(perm));
- } while(std::prev_permutation(selector.begin(), selector.end()));
- }
- public:
- static std::string fmt_name(Value<std::vector<T>>* val){
- std::stringstream ss;
- ss << "combinations(" << Size << "," << val->get_name() << ")";
- return ss.str();
- }
- Combinations(Value<std::vector<T>>* val, const std::string alias="")
- :DerivedValue<std::vector<tuple_type>>(fmt_name(val), alias),
- val(val) { }
- };
- template <typename FST, typename SND>
- class CartProduct : public DerivedValue<std::vector<std::tuple<FST,SND>>>{
- private:
- Value<std::vector<FST>>* val1;
- Value<std::vector<SND>>* val2;
- void update_value(){
- this->value.clear();
- auto& v1 = val1->get_value();
- auto& v2 = val2->get_value();
- for(int i=0; i<v1.size(); i++){
- for(int j=0; j<v2.size(); j++){
- this->value.push_back(std::tuple<FST,SND>(v1[i],v2[j]));
- }
- }
- }
- static std::string calc_name(Value<std::vector<FST>>* val1, Value<std::vector<SND>>* val2){
- std::stringstream ss;
- ss << "cartProduct("
- << val1->get_name() << ", " << val2->get_name()
- << ")";
- return ss.str();
- }
- public:
- static std::string fmt_name(Value<std::vector<FST>>* val1, Value<std::vector<SND>>* val2){
- return "cartProduct("+val1->get_name()+", "+val2->get_name()+")";
- }
- CartProduct(Value<std::vector<FST>>* val1, Value<std::vector<SND>>* val2, const std::string alias="")
- :DerivedValue<std::vector<std::tuple<FST,SND>>>(calc_name(val1, val2), alias),
- val1(val1), val2(val2) { }
- };
- template <typename T>
- class BoundValue : public DerivedValue<T>{
- protected:
- Function<T()>& f;
- void update_value(){
- this->value = f();
- }
- public:
- static std::string fmt_name(Function<T()> f){
- return f.get_name()+"(<bound>)";
- }
- BoundValue(Function<T()>& f, const std::string alias="")
- :DerivedValue<T>(fmt_name(f), alias),
- f(f) { }
- };
- template <typename T>
- class PointerValue : public DerivedValue<T*>{
- protected:
- void update_value(){ }
- public:
- PointerValue(const std::string& name, T* ptr, const std::string alias="")
- :DerivedValue<T*>(name, alias){
- this->value = ptr;
- }
- };
- template <typename T>
- class ConstantValue : public DerivedValue<T>{
- protected:
- void update_value(){ }
- public:
- static std::string fmt_name(const std::string& name){
- return "const::"+name;
- }
- ConstantValue(const std::string& name, T const_value, const std::string alias="")
- :DerivedValue<T>(fmt_name(name), alias) {
- this->value = const_value;
- }
- };
- }
- #endif
|