2018-10-30 15:06:29 +01:00
|
|
|
|
|
|
|
#include "helper_tools.h"
|
|
|
|
|
|
|
|
namespace MultiArrayTools
|
|
|
|
{
|
|
|
|
template <typename... T>
|
|
|
|
std::ostream& operator<<(std::ostream& out, const std::tuple<T...>& tp)
|
|
|
|
{
|
|
|
|
PackNum<sizeof...(T)-1>::printTuple(out, tp);
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class RangeType>
|
|
|
|
auto getIndex(std::shared_ptr<RangeType> range)
|
|
|
|
-> std::shared_ptr<typename RangeType::IndexType>
|
|
|
|
{
|
|
|
|
return std::make_shared<typename RangeType::IndexType>(range);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class RangeType>
|
|
|
|
auto getIndex()
|
|
|
|
-> std::shared_ptr<typename RangeType::IndexType>
|
|
|
|
{
|
|
|
|
static_assert( RangeType::defaultable,
|
|
|
|
/*typeid(typename RangeType).name() + */" is not defaultable" );
|
|
|
|
static auto f = RangeType::factory();
|
|
|
|
static auto r = std::dynamic_pointer_cast<RangeType>( f.create() );
|
|
|
|
return std::make_shared<typename RangeType::IndexType>(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class... RangeTypes>
|
|
|
|
auto mkMulti(std::shared_ptr<RangeTypes>... ranges)
|
|
|
|
-> std::shared_ptr<MultiRange<RangeTypes...> >
|
|
|
|
{
|
|
|
|
MultiRangeFactory<RangeTypes...> mrf( ranges... );
|
|
|
|
return std::dynamic_pointer_cast<MultiRange<RangeTypes...> >( mrf.create() );
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class... IndexTypes>
|
|
|
|
auto mkMIndex(std::shared_ptr<IndexTypes>... indices)
|
|
|
|
-> decltype( getIndex( mkMulti( indices->range()... ) ) )
|
|
|
|
{
|
|
|
|
auto mi = getIndex( mkMulti( indices->range()... ) );
|
|
|
|
(*mi)( indices... );
|
|
|
|
return mi;
|
|
|
|
}
|
|
|
|
|
2019-03-29 12:58:39 +01:00
|
|
|
template <SpaceType STYPE, class Op, class MA, class... RangeTypes>
|
|
|
|
auto mkGenMapR(const std::tuple<Op,MA>& f, std::shared_ptr<RangeTypes>... ranges)
|
|
|
|
-> std::shared_ptr<GenMapRange<Op,STYPE,RangeTypes...> >
|
2018-10-30 15:06:29 +01:00
|
|
|
{
|
2019-03-29 12:58:39 +01:00
|
|
|
GenMapRangeFactory<Op,STYPE,RangeTypes...> mrf(f, ranges... );
|
|
|
|
return createExplicit( mrf );
|
2018-10-30 15:06:29 +01:00
|
|
|
}
|
|
|
|
|
2019-03-29 12:58:39 +01:00
|
|
|
template <SpaceType STYPE, class Op, class MA, class... IndexTypes>
|
|
|
|
auto mkGenMapI(const std::tuple<Op,MA>& f, std::shared_ptr<IndexTypes>... indices)
|
2019-02-06 14:11:17 +01:00
|
|
|
-> decltype( getIndex( mkGenMapR<STYPE>( f, indices->range()... ) ) )
|
2018-10-30 15:06:29 +01:00
|
|
|
{
|
2019-02-06 14:11:17 +01:00
|
|
|
auto mi = getIndex( mkGenMapR<STYPE>( f, indices->range()... ) );
|
2018-10-30 15:06:29 +01:00
|
|
|
(*mi)(indices...);
|
|
|
|
return mi;
|
|
|
|
}
|
2019-02-06 13:01:14 +01:00
|
|
|
|
2019-03-29 12:58:39 +01:00
|
|
|
template <class Op, class MA, class... RangeTypes>
|
|
|
|
auto mkMapR(const std::tuple<Op,MA>& f, std::shared_ptr<RangeTypes>... ranges)
|
2019-02-06 13:01:14 +01:00
|
|
|
-> decltype( mkGenMapR<SpaceType::ANY>(f, ranges... ) )
|
|
|
|
{
|
|
|
|
return mkGenMapR<SpaceType::ANY>(f, ranges... );
|
|
|
|
}
|
|
|
|
|
2019-03-29 12:58:39 +01:00
|
|
|
template <class Op, class MA, class... IndexTypes>
|
|
|
|
auto mkMapI(const std::tuple<Op,MA>& f, std::shared_ptr<IndexTypes>... indices)
|
2019-02-06 13:01:14 +01:00
|
|
|
-> decltype( mkGenMapI<SpaceType::ANY>(f, indices... ) )
|
|
|
|
{
|
|
|
|
return mkGenMapI<SpaceType::ANY>(f, indices... );
|
|
|
|
}
|
2018-10-30 15:06:29 +01:00
|
|
|
|
|
|
|
template <class... RangeTypes>
|
|
|
|
auto mkMulti(std::tuple<std::shared_ptr<RangeTypes>...> rangesTuple)
|
|
|
|
-> MultiRange<RangeTypes...>
|
|
|
|
{
|
|
|
|
MultiRangeFactory<RangeTypes...> mrf( rangesTuple );
|
|
|
|
return std::dynamic_pointer_cast<MultiRange<RangeTypes...> >( mrf.create() );
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class RangeFactory>
|
|
|
|
auto createExplicit(RangeFactory& rf)
|
|
|
|
-> std::shared_ptr<typename RangeFactory::oType>
|
|
|
|
{
|
|
|
|
return std::dynamic_pointer_cast<typename RangeFactory::oType>( rf.create() );
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class RangeFactory>
|
|
|
|
auto createExplicit(std::shared_ptr<RangeFactory> rfp)
|
|
|
|
-> std::shared_ptr<typename RangeFactory::oType>
|
|
|
|
{
|
|
|
|
return std::dynamic_pointer_cast<typename RangeFactory::oType>( rfp->create() );
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class Range>
|
2019-02-13 21:59:13 +01:00
|
|
|
auto createRange(const vector<char>& cvec)
|
2018-10-30 15:06:29 +01:00
|
|
|
-> std::shared_ptr<Range>
|
|
|
|
{
|
|
|
|
const char* dp = cvec.data();
|
|
|
|
auto ff = createRangeFactory(&dp);
|
|
|
|
auto rbptr = ff->create();
|
|
|
|
assert(rbptr->spaceType() == Range::STYPE);
|
|
|
|
// CATCH CAST ERROR HERE !!!
|
|
|
|
return std::dynamic_pointer_cast<Range>( rbptr );
|
|
|
|
}
|
|
|
|
|
2019-03-27 19:00:16 +01:00
|
|
|
inline auto createRange(const vector<char>* cvec, int metaType, size_t size)
|
|
|
|
-> std::shared_ptr<RangeBase>
|
|
|
|
{
|
|
|
|
auto f = createSingleRangeFactory(cvec, metaType, size);
|
|
|
|
return f->create();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline auto createRangeA(const vector<char>* cvec, int metaType, size_t size)
|
|
|
|
-> std::shared_ptr<AnonymousRange>
|
|
|
|
{
|
|
|
|
AnonymousRangeFactory arf(createRange(cvec, metaType, size));
|
|
|
|
return createExplicit(arf);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline auto cvecMetaCast(const std::shared_ptr<SingleRange<vector<char>,SpaceType::ANY>>& r, int metaType)
|
|
|
|
-> std::shared_ptr<RangeBase>
|
|
|
|
{
|
|
|
|
return createRange(&r->get(0), metaType, r->size());
|
|
|
|
}
|
|
|
|
|
|
|
|
inline auto cvecMetaCastA(const std::shared_ptr<SingleRange<vector<char>,SpaceType::ANY>>& r, int metaType)
|
|
|
|
-> std::shared_ptr<AnonymousRange>
|
|
|
|
{
|
|
|
|
return createRangeA(&r->get(0), metaType, r->size());
|
|
|
|
}
|
|
|
|
|
2019-03-13 19:17:22 +01:00
|
|
|
template <class Range, typename... Args>
|
2019-03-14 19:10:06 +01:00
|
|
|
auto createRangeE(Args&&... args)
|
2019-03-13 19:17:22 +01:00
|
|
|
-> std::shared_ptr<Range>
|
|
|
|
{
|
2019-03-14 19:10:06 +01:00
|
|
|
typename Range::FType f(args...);
|
2019-03-13 19:17:22 +01:00
|
|
|
return createExplicit(f);
|
|
|
|
}
|
|
|
|
|
2018-10-30 15:06:29 +01:00
|
|
|
template <size_t N, class MArray>
|
|
|
|
auto rptr(const MArray& ma)
|
|
|
|
-> decltype(ma.template getRangePtr<N>())
|
|
|
|
{
|
|
|
|
return ma.template getRangePtr<N>();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-11-01 22:11:08 +01:00
|
|
|
template <class EC, class MArray>
|
|
|
|
auto dynamic(const MArray& ma, bool slice)
|
|
|
|
-> std::shared_ptr<MultiArrayBase<typename MArray::value_type,DynamicRange<EC>>>
|
|
|
|
{
|
|
|
|
DynamicRangeFactory<EC> drf(ma.range()->space());
|
|
|
|
if(slice){
|
|
|
|
return std::make_shared<ConstSlice<typename MArray::value_type,DynamicRange<EC>>>
|
|
|
|
( std::dynamic_pointer_cast<DynamicRange<EC>>( drf.create() ),
|
|
|
|
ma.data() );
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return std::make_shared<MultiArray<typename MArray::value_type,DynamicRange<EC>>>
|
|
|
|
( std::dynamic_pointer_cast<DynamicRange<EC>>( drf.create() ),
|
|
|
|
ma.vdata() );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-12 18:11:22 +02:00
|
|
|
template <class EC, class MArray>
|
|
|
|
auto mdynamic(MArray& ma, bool slice)
|
|
|
|
-> std::shared_ptr<MutableMultiArrayBase<typename MArray::value_type,DynamicRange<EC>>>
|
|
|
|
{
|
|
|
|
DynamicRangeFactory<EC> drf(ma.range()->space());
|
|
|
|
if(slice){
|
|
|
|
return std::make_shared<Slice<typename MArray::value_type,DynamicRange<EC>>>
|
|
|
|
( std::dynamic_pointer_cast<DynamicRange<EC>>( drf.create() ),
|
|
|
|
ma.data() );
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return std::make_shared<MultiArray<typename MArray::value_type,DynamicRange<EC>>>
|
|
|
|
( std::dynamic_pointer_cast<DynamicRange<EC>>( drf.create() ),
|
|
|
|
ma.vdata() );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-03 22:03:35 +01:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
template <size_t N>
|
|
|
|
struct CopyRanges
|
|
|
|
{
|
|
|
|
template <class Space1, class Space2>
|
|
|
|
static inline void exec(const Space1& space1, Space2& space2)
|
|
|
|
{
|
|
|
|
std::get<N>(space2) = std::get<N>(space1);
|
|
|
|
CopyRanges<N-1>::exec(space1,space2);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
template <>
|
|
|
|
struct CopyRanges<0>
|
|
|
|
{
|
|
|
|
template <class Space1, class Space2>
|
|
|
|
static inline void exec(const Space1& space1, Space2& space2)
|
|
|
|
{
|
|
|
|
std::get<0>(space2) = std::get<0>(space1);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, class EC, class Range1, class... RangeTypes>
|
2019-05-15 20:37:00 +02:00
|
|
|
auto anonToDynView(const MultiArrayBase<T,Range1,RangeTypes...,AnonymousRange>& ma)
|
2018-11-03 22:03:35 +01:00
|
|
|
-> ConstSlice<T,Range1,RangeTypes...,DynamicRange<EC>>
|
|
|
|
{
|
|
|
|
constexpr size_t LAST = sizeof...(RangeTypes)+1;
|
|
|
|
DynamicRangeFactory<EC> drf(rptr<LAST>(ma)->orig());
|
|
|
|
std::tuple<std::shared_ptr<Range1>,std::shared_ptr<RangeTypes>...,
|
|
|
|
std::shared_ptr<DynamicRange<EC>>> mNSpace;
|
|
|
|
CopyRanges<LAST-1>::exec(ma.range()->space(),mNSpace);
|
|
|
|
std::get<LAST>(mNSpace) = createExplicit( drf );
|
|
|
|
return ConstSlice<T,Range1,RangeTypes...,DynamicRange<EC>>(mNSpace, ma.data());
|
|
|
|
}
|
|
|
|
|
2019-09-13 18:08:23 +02:00
|
|
|
template <typename T, class EC, class Range1, class... RangeTypes>
|
|
|
|
auto anonToDynView(MutableMultiArrayBase<T,Range1,RangeTypes...,AnonymousRange>& ma)
|
|
|
|
-> Slice<T,Range1,RangeTypes...,DynamicRange<EC>>
|
|
|
|
{
|
|
|
|
constexpr size_t LAST = sizeof...(RangeTypes)+1;
|
|
|
|
DynamicRangeFactory<EC> drf(rptr<LAST>(ma)->orig());
|
|
|
|
std::tuple<std::shared_ptr<Range1>,std::shared_ptr<RangeTypes>...,
|
|
|
|
std::shared_ptr<DynamicRange<EC>>> mNSpace;
|
|
|
|
CopyRanges<LAST-1>::exec(ma.range()->space(),mNSpace);
|
|
|
|
std::get<LAST>(mNSpace) = createExplicit( drf );
|
|
|
|
return Slice<T,Range1,RangeTypes...,DynamicRange<EC>>(mNSpace, ma.data());
|
|
|
|
}
|
|
|
|
|
2018-11-03 22:03:35 +01:00
|
|
|
template <typename T, class EC, class Range1, class... RangeTypes>
|
|
|
|
auto dynToAnonMove(MultiArray<T,Range1,RangeTypes...,DynamicRange<EC>>&& ma)
|
|
|
|
-> MultiArray<T,Range1,RangeTypes...,AnonymousRange>
|
|
|
|
{
|
|
|
|
constexpr size_t LAST = sizeof...(RangeTypes)+1;
|
|
|
|
AnonymousRangeFactory arf(rptr<LAST>(ma)->orig());
|
|
|
|
std::tuple<std::shared_ptr<Range1>,std::shared_ptr<RangeTypes>...,
|
|
|
|
std::shared_ptr<AnonymousRange>> mNSpace;
|
|
|
|
CopyRanges<LAST-1>::exec(ma.range()->space(),mNSpace);
|
|
|
|
std::get<LAST>(mNSpace) = createExplicit( arf );
|
|
|
|
return ma.format(mNSpace);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, class EC>
|
2019-05-15 20:37:00 +02:00
|
|
|
auto anonToDynView(const MultiArrayBase<T,AnonymousRange>& ma)
|
2018-11-03 22:03:35 +01:00
|
|
|
-> ConstSlice<T,DynamicRange<EC>>
|
|
|
|
{
|
|
|
|
DynamicRangeFactory<EC> drf(rptr<0>(ma)->orig());
|
|
|
|
auto mNSpace = std::make_tuple( createExplicit( drf ) );
|
|
|
|
return ConstSlice<T,DynamicRange<EC>>(mNSpace, ma.data());
|
|
|
|
}
|
|
|
|
|
2019-09-13 18:08:23 +02:00
|
|
|
template <typename T, class EC>
|
|
|
|
auto anonToDynView(MutableMultiArrayBase<T,AnonymousRange>& ma)
|
|
|
|
-> Slice<T,DynamicRange<EC>>
|
|
|
|
{
|
|
|
|
DynamicRangeFactory<EC> drf(rptr<0>(ma)->orig());
|
|
|
|
auto mNSpace = std::make_tuple( createExplicit( drf ) );
|
|
|
|
return Slice<T,DynamicRange<EC>>(mNSpace, ma.data());
|
|
|
|
}
|
|
|
|
|
2018-11-03 22:03:35 +01:00
|
|
|
template <typename T, class EC>
|
|
|
|
auto dynToAnonMove(MultiArray<T,DynamicRange<EC>>&& ma)
|
|
|
|
-> MultiArray<T,AnonymousRange>
|
|
|
|
{
|
|
|
|
AnonymousRangeFactory arf(rptr<0>(ma)->orig());
|
|
|
|
auto mNSpace = std::make_tuple( createExplicit( arf ) );
|
|
|
|
return ma.format(mNSpace);
|
|
|
|
}
|
|
|
|
|
2019-03-08 13:09:10 +01:00
|
|
|
template <class Range>
|
|
|
|
auto metaSlice(const std::shared_ptr<Range>& r)
|
2019-03-11 19:24:34 +01:00
|
|
|
-> ConstSlice<typename Range::MetaType,ClassicRange>
|
2019-03-08 13:09:10 +01:00
|
|
|
{
|
|
|
|
ClassicRF crf(r->size());
|
2019-03-11 19:24:34 +01:00
|
|
|
return ConstSlice<typename Range::MetaType,ClassicRange>( createExplicit(crf), &r->get(0) );
|
2019-03-08 13:09:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
template <class Range, class ORange>
|
|
|
|
auto metaSlice(const std::shared_ptr<Range>& r, const std::shared_ptr<ORange>& ro)
|
2019-03-11 19:24:34 +01:00
|
|
|
-> ConstSlice<typename Range::MetaType,ORange>
|
2019-03-08 13:09:10 +01:00
|
|
|
{
|
2019-03-11 19:24:34 +01:00
|
|
|
return ConstSlice<typename Range::MetaType,ORange>( ro, &r->get(0) );
|
2019-03-08 13:09:10 +01:00
|
|
|
}
|
|
|
|
|
2019-03-13 19:17:22 +01:00
|
|
|
template <typename T, class... Ranges>
|
|
|
|
auto mkArray(const std::shared_ptr<Ranges>&... rs)
|
|
|
|
-> MultiArray<T,Ranges...>
|
|
|
|
{
|
|
|
|
return MultiArray<T,Ranges...>(rs...);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename T, class... Ranges>
|
|
|
|
auto mkArray(const std::shared_ptr<Ranges>&... rs, const T& val)
|
|
|
|
-> MultiArray<T,Ranges...>
|
|
|
|
{
|
|
|
|
return MultiArray<T,Ranges...>(rs..., val);
|
|
|
|
}
|
|
|
|
|
2018-11-03 22:03:35 +01:00
|
|
|
|
2018-10-30 15:06:29 +01:00
|
|
|
}
|