📄 indirect.h
字号:
#ifndef BZ_ARRAY_INDIRECT_H
#define BZ_ARRAY_INDIRECT_H
#include <blitz/array/asexpr.h>
#include <blitz/array/cartesian.h>
BZ_NAMESPACE(blitz)
template<class T_array, class T_index>
class IndirectArray {
public:
IndirectArray(T_array& array, T_index& index)
: array_(array), index_(index)
{ }
template<class T_expr>
void operator=(T_expr expr);
protected:
T_array& array_;
T_index& index_;
};
// Forward declarations
template<class T_array, class T_arrayiter, class T_subdomain, class T_expr>
inline void applyOverSubdomain(const T_array& array, T_arrayiter& arrayIter,
T_subdomain subdomain, T_expr expr);
template<class T_array, class T_arrayiter, int N_rank, class T_expr>
inline void applyOverSubdomain(const T_array& array, T_arrayiter& arrayIter,
RectDomain<N_rank> subdomain,
T_expr expr);
template<class T_array, class T_index> template<class T_rhs>
void IndirectArray<T_array, T_index>::operator=(T_rhs rhs)
{
typedef _bz_typename asExpr<T_rhs>::T_expr T_expr;
T_expr expr(rhs);
_bz_typename T_array::T_iterator arrayIter(array_);
_bz_typename T_index::iterator iter = index_.begin(),
end = index_.end();
for (; iter != end; ++iter)
{
_bz_typename T_index::value_type subdomain = *iter;
applyOverSubdomain(array_, arrayIter, subdomain, expr);
}
}
template<class T_array, class T_arrayiter, class T_subdomain, class T_expr>
inline void applyOverSubdomain(const T_array& array, T_arrayiter& arrayIter,
T_subdomain subdomain, T_expr expr)
{
BZPRECHECK(array.isInRange(subdomain),
"In indirection using an STL container of TinyVector<int,"
<< array.rank() << ">, one of the" << endl << "positions is out of"
" range: " << endl << subdomain << endl
<< "Array lower bounds: " << array.lbound() << endl
<< "Array upper bounds: " << array.ubound() << endl)
arrayIter.moveTo(subdomain);
expr.moveTo(subdomain);
*const_cast<_bz_typename T_arrayiter::T_numtype*>(arrayIter.data()) = *expr;
}
// Specialization for RectDomain<N>
template<class T_array, class T_arrayiter, int N_rank, class T_expr>
inline void applyOverSubdomain(const T_array& array, T_arrayiter& arrayIter,
RectDomain<N_rank> subdomain,
T_expr expr)
{
typedef _bz_typename T_array::T_numtype T_numtype;
// Assume that the RectDomain<N_rank> is a 1-D strip.
// Find the dimension in which the strip is oriented. This
// variable is static so that we cache the value; likely to be
// the same for all strips within a container.
static int stripDim = 0;
if (subdomain.lbound(stripDim) == subdomain.ubound(stripDim))
{
// Cached value was wrong, find the correct value of stripDim
for (stripDim=0; stripDim < N_rank; ++stripDim)
if (subdomain.lbound(stripDim) != subdomain.ubound(stripDim))
break;
// Handle case where the strip is just a single point
if (stripDim == N_rank)
stripDim = 0;
}
#ifdef BZ_DEBUG
// Check that this is in fact a 1D strip
for (int i=0; i < N_rank; ++i)
if ((i != stripDim) && (subdomain.lbound(i) != subdomain.ubound(i)))
BZPRECHECK(0, "In indirection using an STL container of RectDomain<"
<< N_rank << ">, one of" << endl << "the RectDomain objects was not"
" a one-dimensional strip:" << endl << "RectDomain<" << N_rank
<< ">::lbound() = " << subdomain.lbound() << endl
<< "RectDomain<" << N_rank << ">::ubound() = " << subdomain.ubound())
#endif
// Check that the start and end position are in range
BZPRECHECK(array.isInRange(subdomain.lbound()),
"In indirection using an STL container of RectDomain<"
<< N_rank << ">, one of" << endl << "the RectDomain objects has a"
" lbound which is out of range:" << endl
<< subdomain.lbound() << endl
<< "Array lower bounds: " << array.lbound() << endl
<< "Array upper bounds: " << array.ubound() << endl)
BZPRECHECK(array.isInRange(subdomain.ubound()),
"In indirection using an STL container of RectDomain<"
<< N_rank << ">, one of" << endl << "the RectDomain objects has a"
" ubound which is out of range:" << endl
<< subdomain.lbound() << endl
<< "Array lower bounds: " << array.lbound() << endl
<< "Array upper bounds: " << array.ubound() << endl)
// Position at the beginning of the strip
arrayIter.moveTo(subdomain.lbound());
expr.moveTo(subdomain.lbound());
// Loop through the strip
#ifdef BZ_USE_FAST_READ_ARRAY_EXPR
_bz_bool useUnitStride = arrayIter.isUnitStride(stripDim)
&& expr.isUnitStride(stripDim);
int lbound = subdomain.lbound(stripDim);
int ubound = subdomain.ubound(stripDim);
if (useUnitStride)
{
T_numtype* _bz_restrict data = const_cast<T_numtype*>(arrayIter.data());
int length = ubound - lbound + 1;
for (int i=0; i < length; ++i)
data[i] = expr.fastRead(i);
}
else {
#endif
arrayIter.loadStride(stripDim);
expr.loadStride(stripDim);
for (int i=lbound; i <= ubound; ++i)
{
*const_cast<_bz_typename T_arrayiter::T_numtype*>(arrayIter.data())
= *expr;
expr.advance();
arrayIter.advance();
}
#ifdef BZ_USE_FAST_READ_ARRAY_EXPR
}
#endif
}
// Global functions for cartesian product of index sets
template<class T_container>
CartesianProduct<TinyVector<int,2>,T_container,2>
indexSet(const T_container& container0, const T_container& container1)
{
return CartesianProduct<TinyVector<int,2>,T_container,2>(
const_cast<T_container&>(container0),
const_cast<T_container&>(container1));
}
template<class T_container>
CartesianProduct<TinyVector<int,3>,T_container,3>
indexSet(const T_container& container0, const T_container& container1,
const T_container& container2)
{
return CartesianProduct<TinyVector<int,3>,T_container,3>(
const_cast<T_container&>(container0),
const_cast<T_container&>(container1),
const_cast<T_container&>(container2));
}
// Mixture of singletons and containers, e.g. A[indexSet(I,3,K)]
// cp_findContainerType<T1,T2,T3,...,Tn>::T_container
// The set of parameters T1, T2, T3, ... Tn is a mixture of
// int and T_container. This traits class finds the container
// type, and sets T_container.
//
// e.g. cp_findContainerType<int,int,list<int>,int>::T_container is list<int>
// cp_findContainerType<int,deque<int>,deque<int>>::T_container
// is deque<int>
template<class T1, class T2, class T3=int, class T4=int>
struct cp_findContainerType {
typedef T1 T_container;
};
template<class T2, class T3, class T4>
struct cp_findContainerType<int,T2,T3,T4> {
typedef _bz_typename cp_findContainerType<T2,T3,T4>::T_container T_container;
};
// The cp_traits class handles promotion of singleton integers to
// containers. It takes two template parameters:
// T = argument type
// T2 = container type
// If T is an integer, then a container of type T2 is created and the
// integer is inserted. This container is returned.
// Otherwise, T is assumed to be the same type as T2, and the original
// container is returned.
template<class T, class T2>
struct cp_traits {
typedef T T_container;
static const T_container& make(const T& x)
{ return x; }
};
template<class T2>
struct cp_traits<int,T2> {
typedef T2 T_container;
static T2 make(int x)
{
T2 singleton;
singleton.push_back(x);
return singleton;
}
};
// These versions of indexSet() allow mixtures of integer
// and container arguments. At least one integer must be
// specified.
template<class T1, class T2>
CartesianProduct<TinyVector<int,2>, _bz_typename
cp_findContainerType<T1,T2>::T_container,2>
indexSet(const T1& c1, const T2& c2)
{
typedef _bz_typename cp_findContainerType<T1,T2>::T_container
T_container;
return CartesianProduct<TinyVector<int,2>, T_container, 2>(
cp_traits<T1,T_container>::make(c1),
cp_traits<T2,T_container>::make(c2));
}
template<class T1, class T2, class T3>
CartesianProduct<TinyVector<int,3>, _bz_typename
cp_findContainerType<T1,T2,T3>::T_container, 3>
indexSet(const T1& c1, const T2& c2, const T3& c3)
{
typedef _bz_typename cp_findContainerType<T1,T2,T3>::T_container
T_container;
return CartesianProduct<TinyVector<int,3>, T_container, 3>(
cp_traits<T1,T_container>::make(c1),
cp_traits<T2,T_container>::make(c2),
cp_traits<T3,T_container>::make(c3));
}
BZ_NAMESPACE_END
#endif // BZ_ARRAY_INDIRECT_H
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -