123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183 |
- // -*- mode: c++; coding: utf-8 -*-
- /// @file pick.H
- /// @brief Expression template that picks one of several arguments.
- // (c) Daniel Llorens - 2016-2017, 2019
- // This library is free software; you can redistribute it and/or modify it under
- // the terms of the GNU General Public License as published by the Free
- // Software Foundation; either version 3 of the License, or (at your option) any
- // later version.
- // This class is needed because Expr evaluates its arguments before calling its
- // operator. Note that pick() is a normal function, so its *arguments* will always
- // be evaluated; it is the individual array elements that will not be.
- #pragma once
- #include "ra/ply.H"
- #include "ra/match.H"
- #if defined(RA_CHECK_BOUNDS) && RA_CHECK_BOUNDS==0
- #define CHECK_BOUNDS( cond )
- #else
- #define CHECK_BOUNDS( cond ) RA_ASSERT( cond, 0 )
- #endif
- namespace ra {
- // -----------------
- // return type of pick expression, otherwise compiler complains of ambiguity.
- // TODO & is crude, maybe is_assignable?
- // -----------------
- template <class T, class Enable=void>
- struct pick_type
- {
- using type = mp::apply<std::common_type_t, T>;
- };
- // lvalue
- template <class P0, class ... P>
- struct pick_type<std::tuple<P0 &, P ...>,
- std::enable_if_t<!std::is_const_v<P0> && (std::is_same_v<P0 &, P> && ...)>>
- {
- using type = P0 &;
- };
- // const lvalue
- template <class P0, class ... P>
- struct pick_type<std::tuple<P0 &, P & ...>,
- std::enable_if_t<(std::is_same_v<std::decay_t<P0>, std::decay_t<P>> && ...)
- && (std::is_const_v<P0> || (std::is_const_v<P> || ...))>>
- {
- using type = P0 const &;
- };
- // -----------------
- // runtime to compile time conversion for Pick::at() and PickFlat::operator*()
- // -----------------
- template <class T, class J> struct pick_at_type;
- template <class ... P, class J> struct pick_at_type<std::tuple<P ...>, J>
- {
- using type = typename pick_type<std::tuple<decltype(std::declval<P>().at(std::declval<J>())) ...>>::type;
- };
- template <class T, class J> using pick_at_t = typename pick_at_type<mp::drop1<std::decay_t<T>>, J>::type;
- template <size_t I, class T, class J> inline constexpr
- pick_at_t<T, J>
- pick_at(size_t p0, T && t, J const & j)
- {
- if constexpr (I+2<std::tuple_size_v<std::decay_t<T>>) {
- if (p0==I) {
- return std::get<I+1>(t).at(j);
- } else {
- return pick_at<I+1>(p0, t, j);
- }
- } else {
- CHECK_BOUNDS(p0==I);
- return std::get<I+1>(t).at(j);
- }
- }
- template <class T> struct pick_star_type;
- template <class ... P> struct pick_star_type<std::tuple<P ...>>
- {
- using type = typename pick_type<std::tuple<decltype(*std::declval<P>()) ...>>::type;
- };
- template <class T> using pick_star_t = typename pick_star_type<mp::drop1<std::decay_t<T>>>::type;
- template <size_t I, class T> inline constexpr
- pick_star_t<T>
- pick_star(size_t p0, T && t)
- {
- if constexpr (I+2<std::tuple_size_v<std::decay_t<T>>) {
- if (p0==I) {
- return *(std::get<I+1>(t));
- } else {
- return pick_star<I+1>(p0, t);
- }
- } else {
- CHECK_BOUNDS(p0==I);
- return *(std::get<I+1>(t));
- }
- }
- // -----------------
- // follows closely Flat, Expr
- // -----------------
- // Manipulate ET through flat (raw pointer-like) iterators P ...
- template <class T, class I=mp::iota<mp::len<T>>>
- struct PickFlat;
- template <class P0, class ... P, int ... I>
- struct PickFlat<std::tuple<P0, P ...>, mp::int_list<I ...>>
- {
- std::tuple<P0, P ...> t;
- template <class S> void operator+=(S const & s) { ((std::get<I>(t) += std::get<I>(s)), ...); }
- decltype(auto) operator*() { return pick_star<0>(*std::get<0>(t), t); }
- };
- template <class P0, class ... P> inline constexpr auto
- pick_flat(P0 && p0, P && ... p)
- {
- return PickFlat<std::tuple<P0, P ...>> { std::tuple<P0, P ...> { std::forward<P0>(p0), std::forward<P>(p) ... } };
- }
- template <class P0, class ... P> inline constexpr auto
- pick_in(P0 && p0, P && ... p)
- {
- return Pick<std::tuple<P0, P ...>> { std::forward<P0>(p0), std::forward<P>(p) ... };
- }
- template <class P0, class ... P> inline constexpr auto
- pick(P0 && p0, P && ... p)
- {
- return pick_in(start(std::forward<P0>(p0)), start(std::forward<P>(p)) ...);
- }
- // forward decl in atom.H
- template <class P0, class ... P, int ... I>
- struct Pick<std::tuple<P0, P ...>, mp::int_list<I ...>>: public Match<std::tuple<P0, P ...>>
- {
- using Match_ = Match<std::tuple<P0, P ...>>;
- // see test/ra-9.C [ra01] for forward() here.
- constexpr Pick(P0 p0_, P ... p_): Match_(std::forward<P0>(p0_), std::forward<P>(p_) ...) {}
- template <class J> constexpr decltype(auto)
- at(J const & j)
- {
- return pick_at<0>(std::get<0>(this->t).at(j), this->t, j);
- }
- constexpr auto
- flat()
- {
- return pick_flat(std::get<I>(this->t).flat() ...);
- }
- // needed for xpr with rank_s()==RANK_ANY, which don't decay to scalar when used as operator arguments.
- using scalar = decltype(*(pick_flat(std::get<I>(Match_::t).flat() ...)));
- operator scalar()
- {
- if constexpr (this->rank_s()!=1 || size_s(*this)!=1) { // for coord types; so fixed only
- if constexpr (this->rank_s()!=0) {
- static_assert(this->rank_s()==RANK_ANY);
- assert(this->rank()==0);
- }
- }
- return *flat();
- }
- // forward to make sure value y is not misused as ref [ra05]
- #define DEF_ASSIGNOPS(OP) template <class X> constexpr void operator OP(X && x) \
- { for_each([](auto && y, auto && x) { std::forward<decltype(y)>(y) OP x; }, *this, x); }
- FOR_EACH(DEF_ASSIGNOPS, =, *=, +=, -=, /=)
- #undef DEF_ASSIGNOPS
- };
- } // namespace ra
- #undef CHECK_BOUNDS
|