2017-07-27 14:48:41 +02:00
|
|
|
// -*- C++ -*-
|
|
|
|
|
2017-08-04 11:27:47 +02:00
|
|
|
#ifndef __pack_num_h__
|
|
|
|
#define __pack_num_h__
|
|
|
|
|
2017-07-27 14:48:41 +02:00
|
|
|
#include <cstdlib>
|
2017-12-18 11:19:04 +01:00
|
|
|
//#include <type_traits>
|
2017-07-27 14:48:41 +02:00
|
|
|
#include <tuple>
|
2017-12-18 11:19:04 +01:00
|
|
|
#include <ostream>
|
2017-07-27 14:48:41 +02:00
|
|
|
|
2017-08-04 11:27:47 +02:00
|
|
|
#include "base_def.h"
|
2018-02-12 18:26:56 +01:00
|
|
|
#include "xfor/exttype.h"
|
2017-08-04 11:27:47 +02:00
|
|
|
|
2017-07-27 14:48:41 +02:00
|
|
|
namespace MultiArrayHelper
|
|
|
|
{
|
2018-02-14 00:38:44 +01:00
|
|
|
|
2017-07-27 14:48:41 +02:00
|
|
|
template <size_t N>
|
|
|
|
struct PackNum
|
|
|
|
{
|
2017-09-11 12:54:24 +02:00
|
|
|
template <typename... T>
|
2018-01-09 17:24:10 +01:00
|
|
|
static void printTuple(std::ostream& out, const std::tuple<T...>& tp)
|
|
|
|
{
|
2017-09-11 12:54:24 +02:00
|
|
|
out << std::get<sizeof...(T)-N-1>(tp) << ", ";
|
|
|
|
PackNum<N-1>::printTuple(out, tp);
|
|
|
|
}
|
2017-12-05 17:31:57 +01:00
|
|
|
|
2018-01-09 17:24:10 +01:00
|
|
|
template <class... Ops>
|
2018-02-13 21:36:41 +01:00
|
|
|
static auto mkSteps(std::intptr_t ii, const std::tuple<Ops...>& otp)
|
2018-02-14 00:38:44 +01:00
|
|
|
-> decltype(PackNum<N-1>::mkSteps(ii, otp).extend( std::get<N>(otp).rootSteps(ii)) )
|
2018-01-09 17:24:10 +01:00
|
|
|
{
|
2018-02-14 00:38:44 +01:00
|
|
|
return PackNum<N-1>::mkSteps(ii, otp).extend( std::get<N>(otp).rootSteps(ii));
|
2018-01-09 17:24:10 +01:00
|
|
|
}
|
2018-01-09 22:38:46 +01:00
|
|
|
|
|
|
|
template <class RootStepTuple, class IndexClass, class OpClass>
|
|
|
|
static void mkExt(std::array<RootStepTuple,IndexClass::totalDim()>& out,
|
|
|
|
const std::array<std::intptr_t,IndexClass::totalDim()>& siar,
|
|
|
|
const OpClass& second)
|
2018-01-09 17:24:10 +01:00
|
|
|
{
|
2018-01-09 22:38:46 +01:00
|
|
|
std::get<N>(out) = second.rootSteps( std::get<N>(siar) );
|
|
|
|
PackNum<N-1>::mkExt(out, siar, second);
|
|
|
|
}
|
2018-01-14 19:15:05 +01:00
|
|
|
|
2018-02-13 15:38:03 +01:00
|
|
|
template <size_t LAST, typename T, class ETuple, class OpTuple, class OpFunction, typename... Args>
|
2018-02-12 18:26:56 +01:00
|
|
|
static inline T mkOpExpr(const ETuple& pos, const OpTuple& ops, const Args&... args)
|
2018-01-14 19:15:05 +01:00
|
|
|
{
|
2018-02-13 15:38:03 +01:00
|
|
|
typedef typename std::remove_reference<decltype(std::get<N>(ops))>::type NextOpType;
|
|
|
|
static_assert(LAST > NextOpType::SIZE, "inconsistent array positions");
|
|
|
|
static constexpr size_t NEXT = LAST - NextOpType::SIZE;
|
2018-02-14 00:38:44 +01:00
|
|
|
return PackNum<N-1>::template mkOpExpr<NEXT,T,ETuple,OpTuple,OpFunction,T,Args...>
|
|
|
|
( pos, ops, std::get<N>(ops).get(Getter<NEXT>::template getX<ETuple>( pos )), args...);
|
2018-01-14 19:15:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class OpTuple, class Expr>
|
|
|
|
static auto mkLoop( const OpTuple& ot, Expr&& exp )
|
2018-02-14 00:38:44 +01:00
|
|
|
-> decltype(std::get<N>(ot).loop( PackNum<N-1>::mkLoop(ot,exp) ))
|
2018-01-14 19:15:05 +01:00
|
|
|
{
|
|
|
|
return std::get<N>(ot).loop( PackNum<N-1>::mkLoop(ot,exp) );
|
|
|
|
}
|
2017-07-27 14:48:41 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
template<>
|
|
|
|
struct PackNum<0>
|
|
|
|
{
|
2017-09-11 12:54:24 +02:00
|
|
|
template <typename... T>
|
2018-01-09 17:24:10 +01:00
|
|
|
static void printTuple(std::ostream& out, const std::tuple<T...>& tp)
|
|
|
|
{
|
2017-09-11 12:54:24 +02:00
|
|
|
out << std::get<sizeof...(T)-1>(tp);
|
|
|
|
}
|
2018-01-09 17:24:10 +01:00
|
|
|
|
|
|
|
template <class... Ops>
|
2018-02-13 21:36:41 +01:00
|
|
|
static auto mkSteps(std::intptr_t ii, const std::tuple<Ops...>& otp)
|
2018-01-09 17:24:10 +01:00
|
|
|
-> decltype(std::get<0>(otp).rootSteps(ii))
|
|
|
|
{
|
|
|
|
return std::get<0>(otp).rootSteps(ii);
|
|
|
|
}
|
|
|
|
|
2018-01-09 22:38:46 +01:00
|
|
|
template <class RootStepTuple, class IndexClass, class OpClass>
|
|
|
|
static void mkExt(std::array<RootStepTuple,IndexClass::totalDim()>& out,
|
|
|
|
const std::array<std::intptr_t,IndexClass::totalDim()>& siar,
|
|
|
|
const OpClass& second)
|
|
|
|
{
|
|
|
|
std::get<0>(out) = second.rootSteps( std::get<0>(siar) );
|
|
|
|
}
|
|
|
|
|
2018-02-13 15:38:03 +01:00
|
|
|
template <size_t LAST, typename T, class ETuple, class OpTuple, class OpFunction, typename... Args>
|
|
|
|
static inline T mkOpExpr(const ETuple& pos, const OpTuple& ops, const Args&... args)
|
2018-01-14 19:15:05 +01:00
|
|
|
{
|
2018-02-14 00:38:44 +01:00
|
|
|
typedef typename std::remove_reference<decltype(std::get<0>(ops))>::type NextOpType;
|
|
|
|
static constexpr size_t NEXT = LAST - NextOpType::SIZE;
|
|
|
|
static_assert(NEXT == 0, "inconsistent array positions");
|
|
|
|
return OpFunction::apply(std::get<0>(ops).get(Getter<0>::template getX<ETuple>( pos )), args...);
|
2018-01-14 19:15:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class OpTuple, class Expr>
|
|
|
|
static auto mkLoop( const OpTuple& ot, Expr&& exp )
|
2018-02-14 00:38:44 +01:00
|
|
|
-> decltype(std::get<0>(ot).loop( exp ))
|
2018-01-14 19:15:05 +01:00
|
|
|
{
|
|
|
|
return std::get<0>(ot).loop( exp );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-07-27 14:48:41 +02:00
|
|
|
};
|
2017-09-12 18:36:05 +02:00
|
|
|
|
|
|
|
|
2017-07-27 14:48:41 +02:00
|
|
|
|
|
|
|
} // end namespace MultiArrayHelper
|
2017-08-04 11:27:47 +02:00
|
|
|
|
|
|
|
#endif
|