mirror of
https://github.com/triqs/dft_tools
synced 2025-01-12 14:08:24 +01:00
47cb8a03f7
- Simplify group_indices - Only for C ordered, remove complex compile time. - Could be generalized to non C ordered, but no need. - Fix slice for custom orders. - Generalize the group_indices for the custom order. - Add c_ordered_transposed_view (useful ?) - Improve slice, special for ellipsis (quicker). - Simplify TraversalOrder - Assignement. Specialize one case for speed. - use FORCEINLINE in foreach, according to speed test for clang - add one speed test - Modify iterators for better speed. - along the lines decided for the foreach - update doc.
288 lines
9.6 KiB
C++
288 lines
9.6 KiB
C++
/*******************************************************************************
|
|
*
|
|
* TRIQS: a Toolbox for Research in Interacting Quantum Systems
|
|
*
|
|
* Copyright (C) 2011-2014 by O. Parcollet
|
|
*
|
|
* TRIQS 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.
|
|
*
|
|
* TRIQS is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
|
|
* details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along with
|
|
* TRIQS. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
******************************************************************************/
|
|
#ifndef TRIQS_ARRAYS_MINI_VECTOR_H
|
|
#define TRIQS_ARRAYS_MINI_VECTOR_H
|
|
#include "./first_include.hpp"
|
|
#include <iostream>
|
|
#include "./macros.hpp"
|
|
#include "./c14.hpp"
|
|
#include "./compiler_details.hpp"
|
|
#include "./exceptions.hpp"
|
|
#include <boost/serialization/utility.hpp>
|
|
#include <boost/preprocessor/repetition/repeat.hpp>
|
|
#include <boost/preprocessor/repetition/enum_params.hpp>
|
|
#include <vector>
|
|
|
|
#define TRIQS_MINI_VECTOR_NRANK_MAX 10
|
|
|
|
#define TRIQS_MAKE_NVP(NAME,X) X
|
|
//#define TRIQS_MAKE_NVP(NAME,X) boost::serialization::make_nvp(NAME,X)
|
|
namespace triqs { namespace utility {
|
|
|
|
struct no_init_tag{};
|
|
|
|
template <typename T, int Rank>
|
|
class mini_vector {
|
|
T _data[(Rank!=0 ? Rank : 1)];
|
|
friend class boost::serialization::access;
|
|
template<class Archive> void serialize(Archive & ar, const unsigned int version) { ar & TRIQS_MAKE_NVP("_data",_data); }
|
|
void init() { for (int i=0;i<Rank; ++i) _data[i] = 0;}
|
|
public :
|
|
|
|
typedef T value_type;
|
|
|
|
mini_vector(){init();}
|
|
|
|
mini_vector(no_init_tag){}
|
|
|
|
mini_vector(T x_0) {
|
|
static_assert(Rank == 1, "mini_vector : incorrect number of variables in constructor");
|
|
_data[0] = x_0;
|
|
}
|
|
mini_vector(T x_0, T x_1) {
|
|
static_assert(Rank == 2, "mini_vector : incorrect number of variables in constructor");
|
|
_data[0] = x_0;
|
|
_data[1] = x_1;
|
|
}
|
|
mini_vector(T x_0, T x_1, T x_2) {
|
|
static_assert(Rank == 3, "mini_vector : incorrect number of variables in constructor");
|
|
_data[0] = x_0;
|
|
_data[1] = x_1;
|
|
_data[2] = x_2;
|
|
}
|
|
mini_vector(T x_0, T x_1, T x_2, T x_3) {
|
|
static_assert(Rank == 4, "mini_vector : incorrect number of variables in constructor");
|
|
_data[0] = x_0;
|
|
_data[1] = x_1;
|
|
_data[2] = x_2;
|
|
_data[3] = x_3;
|
|
}
|
|
mini_vector(T x_0, T x_1, T x_2, T x_3, T x_4) {
|
|
static_assert(Rank == 5, "mini_vector : incorrect number of variables in constructor");
|
|
_data[0] = x_0;
|
|
_data[1] = x_1;
|
|
_data[2] = x_2;
|
|
_data[3] = x_3;
|
|
_data[4] = x_4;
|
|
}
|
|
mini_vector(T x_0, T x_1, T x_2, T x_3, T x_4, T x_5) {
|
|
static_assert(Rank == 6, "mini_vector : incorrect number of variables in constructor");
|
|
_data[0] = x_0;
|
|
_data[1] = x_1;
|
|
_data[2] = x_2;
|
|
_data[3] = x_3;
|
|
_data[4] = x_4;
|
|
_data[5] = x_5;
|
|
}
|
|
mini_vector(T x_0, T x_1, T x_2, T x_3, T x_4, T x_5, T x_6) {
|
|
static_assert(Rank == 7, "mini_vector : incorrect number of variables in constructor");
|
|
_data[0] = x_0;
|
|
_data[1] = x_1;
|
|
_data[2] = x_2;
|
|
_data[3] = x_3;
|
|
_data[4] = x_4;
|
|
_data[5] = x_5;
|
|
_data[6] = x_6;
|
|
}
|
|
mini_vector(T x_0, T x_1, T x_2, T x_3, T x_4, T x_5, T x_6, T x_7) {
|
|
static_assert(Rank == 8, "mini_vector : incorrect number of variables in constructor");
|
|
_data[0] = x_0;
|
|
_data[1] = x_1;
|
|
_data[2] = x_2;
|
|
_data[3] = x_3;
|
|
_data[4] = x_4;
|
|
_data[5] = x_5;
|
|
_data[6] = x_6;
|
|
_data[7] = x_7;
|
|
}
|
|
mini_vector(T x_0, T x_1, T x_2, T x_3, T x_4, T x_5, T x_6, T x_7, T x_8) {
|
|
static_assert(Rank == 9, "mini_vector : incorrect number of variables in constructor");
|
|
_data[0] = x_0;
|
|
_data[1] = x_1;
|
|
_data[2] = x_2;
|
|
_data[3] = x_3;
|
|
_data[4] = x_4;
|
|
_data[5] = x_5;
|
|
_data[6] = x_6;
|
|
_data[7] = x_7;
|
|
_data[8] = x_8;
|
|
}
|
|
mini_vector(T x_0, T x_1, T x_2, T x_3, T x_4, T x_5, T x_6, T x_7, T x_8, T x_9) {
|
|
static_assert(Rank == 10, "mini_vector : incorrect number of variables in constructor");
|
|
_data[0] = x_0;
|
|
_data[1] = x_1;
|
|
_data[2] = x_2;
|
|
_data[3] = x_3;
|
|
_data[4] = x_4;
|
|
_data[5] = x_5;
|
|
_data[6] = x_6;
|
|
_data[7] = x_7;
|
|
_data[8] = x_8;
|
|
_data[9] = x_9;
|
|
};
|
|
|
|
/*
|
|
#define AUX(z,p,unused) _data[p] = x_##p;
|
|
#define IMPL(z, NN, unused) \
|
|
mini_vector (BOOST_PP_ENUM_PARAMS(BOOST_PP_INC(NN), T x_)){ \
|
|
static_assert(Rank-1==NN,"mini_vector : incorrect number of variables in constructor");\
|
|
BOOST_PP_REPEAT(BOOST_PP_INC(NN),AUX,nil) }
|
|
BOOST_PP_REPEAT(TRIQS_MINI_VECTOR_NRANK_MAX , IMPL, nil);
|
|
#undef IMPL
|
|
#undef AUX
|
|
*/
|
|
|
|
mini_vector(const mini_vector & x){ *this = x; }
|
|
mini_vector(mini_vector && x){ *this = std::move(x); }
|
|
|
|
template<typename T2> mini_vector(const mini_vector<T2,Rank> & x){ *this = x; }
|
|
|
|
mini_vector(const std::vector<T> & v){
|
|
if (v.size()!=Rank)
|
|
TRIQS_RUNTIME_ERROR<< "mini_vector construction : vector size incorrect : expected "<<Rank<<" got : "<< v.size();
|
|
for (int i=0;i<Rank; ++i) _data[i] = v[i];
|
|
}
|
|
|
|
mini_vector & operator=(const mini_vector & x){ for (int i=0;i<Rank; ++i) _data[i] = x._data[i]; return *this;}
|
|
mini_vector & operator=(mini_vector && x) = default;
|
|
|
|
friend void swap(mini_vector & a, mini_vector & b) { std::swap(a._data, b._data);}
|
|
|
|
int size() const { return Rank;}
|
|
|
|
template<typename T2>
|
|
mini_vector & operator=(const mini_vector<T2,Rank> & x){ for (int i=0;i<Rank; ++i) _data[i] = x[i]; return *this;}
|
|
|
|
mini_vector &operator=(T x) {
|
|
for (int i = 0; i < Rank; ++i) _data[i] = x;
|
|
return *this;
|
|
}
|
|
|
|
T & operator[](size_t i) { return _data[i];}
|
|
const T & operator[](size_t i) const { return _data[i];}
|
|
|
|
std::vector<T> to_vector () const { std::vector<T> V(Rank); for (int i=0;i<Rank; ++i) V[i] = _data[i]; return V; }
|
|
|
|
T product_of_elements () const { T res=1; for (int i=0;i<Rank; ++i) res *= _data[i]; return res; }
|
|
|
|
T * restrict ptr() { return _data;}
|
|
const T * restrict ptr() const { return _data;}
|
|
|
|
std::string to_string() const {
|
|
std::stringstream fs;
|
|
fs <<"(";
|
|
for (int i=0;i<Rank; ++i) fs<<(i==0? "" : " ") << (*this)[i];
|
|
fs<<")";
|
|
return fs.str();
|
|
}
|
|
|
|
mini_vector<T, Rank+1> append (T const & x) const {
|
|
mini_vector<T, Rank+1> res;
|
|
for (int i=0;i<Rank; ++i) res[i]=_data[i];
|
|
res[Rank] = x;
|
|
return res;
|
|
}
|
|
|
|
mini_vector<T, Rank+1> front_append (T const & x) const {
|
|
mini_vector<T, Rank+1> res;
|
|
for (int i=0;i<Rank; ++i) res[i+1]=_data[i];
|
|
res[0] = x;
|
|
return res;
|
|
}
|
|
|
|
mini_vector<T, Rank-1> pop () const {
|
|
mini_vector<T, Rank-1> res;
|
|
for (int i=0;i<Rank-1; ++i) res[i]=_data[i];
|
|
return res;
|
|
}
|
|
|
|
mini_vector<T, Rank-1> front_pop () const {
|
|
mini_vector<T, Rank-1> res;
|
|
for (int i=1;i<Rank; ++i) res[i-1]=_data[i];
|
|
return res;
|
|
}
|
|
|
|
friend std::ostream & operator << ( std::ostream & out, mini_vector const & v ) {return out<<v.to_string();}
|
|
friend std::stringstream & operator << ( std::stringstream & out, mini_vector const & v ) { out<<v.to_string(); return out;}
|
|
}; // class mini_vector
|
|
|
|
// ------------ specialization for size 0 -------------
|
|
template <typename T> class mini_vector<T, 0> {
|
|
T _data[1];
|
|
public:
|
|
mini_vector(){}
|
|
mini_vector(no_init_tag){}
|
|
|
|
T & operator[](size_t i) { return _data[i];}
|
|
const T & operator[](size_t i) const { return _data[i];}
|
|
T * restrict ptr() { return _data;}
|
|
const T * restrict ptr() const { return _data;}
|
|
friend std::ostream & operator << ( std::ostream & out, mini_vector const & v ) {return out<<"[]";}
|
|
};
|
|
|
|
// ------------- Comparison --------------------------------------
|
|
|
|
template <typename T, int R>
|
|
bool operator ==(mini_vector<T,R> const & v1, mini_vector<T,R> const & v2) {
|
|
for (int i=0;i<R; ++i) { if (v1[i]!=v2[i]) return false;}
|
|
return true;
|
|
}
|
|
|
|
template <typename T, int R> bool operator !=(mini_vector<T,R> const & v1, mini_vector<T,R> const & v2) { return (!(v1==v2));}
|
|
|
|
// ------------- join --------------------------------------
|
|
template <typename T1, typename T2, int R1, int R2>
|
|
mini_vector<T1, R1+R2> join (mini_vector<T1,R1> const & v1, mini_vector<T2,R2> const & v2) {
|
|
mini_vector<T1, R1+R2> res;
|
|
for (int i=0;i<R1; ++i) res[i]=v1[i];
|
|
for (int i=0;i<R2; ++i) res[R1+i]=v2[i];
|
|
return res;
|
|
}
|
|
|
|
// ------------- dot --------------------------------------
|
|
template <typename T1, typename T2, int Rank>
|
|
T1 dot_product(mini_vector<T1,Rank> const & v1, mini_vector<T2,Rank> const & v2) {
|
|
T1 res=0;
|
|
for (int i=0;i<Rank; ++i) res += v1[i]*v2[i];
|
|
return res;
|
|
}
|
|
|
|
//#ifndef TRIQS_C11
|
|
// // ------------- transform a tuple into a minivector --------------------------------------
|
|
//
|
|
// template <typename T, typename TU> mini_vector<T, std::tuple_size<TU>::value> tuple_to_mini_vector(TU const &t) {
|
|
// return tuple::apply_construct_parenthesis<mini_vector<T, std::tuple_size<TU>::value>>(t);
|
|
// }
|
|
//#endif
|
|
|
|
}}//namespace triqs::arrays
|
|
|
|
namespace std { // overload std::get to work with it
|
|
|
|
template <int i, typename T, int R> AUTO_DECL get(triqs::utility::mini_vector<T, R> &v) RETURN(v[i]);
|
|
template <int i, typename T, int R> AUTO_DECL get(triqs::utility::mini_vector<T, R> const &v) RETURN(v[i]);
|
|
template<typename T, int R> struct tuple_size<triqs::utility::mini_vector<T, R>>: std::integral_constant<size_t,R>{};
|
|
|
|
}
|
|
|
|
#endif
|
|
|