diff --git a/istl/ownercopy.hh b/istl/ownercopy.hh new file mode 100644 index 0000000000000000000000000000000000000000..d2b1597d488ae14f482409b4cd9e4f6deb057c90 --- /dev/null +++ b/istl/ownercopy.hh @@ -0,0 +1,162 @@ +// -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*- +// vi: set et ts=4 sw=2 sts=2: +#ifndef DUNE_OWNERCOPY_HH +#define DUNE_OWNERCOPY_HH + +#include <new> +#include <iostream> +#include <vector> +#include <list> +#include <map> +#include <set> + +#include "mpi.h" + +#include "dune/common/tripel.hh" +#include <dune/common/enumset.hh> + +#include "indexset.hh" +#include "communicator.hh" +#include "remoteindices.hh" +#include "istlexception.hh" + +namespace Dune { + + /** + @addtogroup ISTL + @{ + */ + + /** + * @brief A class setting up standard communication for a two-valued + attribute set with owner/copy semantics. + */ + + // set up communication from known distribution with owner/copy semantics + template <class GlobalIdType, class LocalIdType, int ownerattribute, int copyattribute> + class OwnerCopyCommunication + { + // used types + typedef tripel<GlobalIdType,LocalIdType,int> IndexTripel; + typedef tripel<int,GlobalIdType,int> RemoteIndexTripel; + enum AttributeSet { owner=ownerattribute, copy=copyattribute }; + typedef ParallelLocalIndex<AttributeSet> LI; + typedef ParallelIndexSet<GlobalIdType,LI,512> PIS; + typedef RemoteIndices<PIS> RI; + typedef RemoteIndexListModifier<PIS,false> RILM; + typedef typename RI::RemoteIndex RX; + typedef BufferedCommunicator<PIS> BC; + typedef Interface<PIS> IF; + + // gather/scatter callback for communcation + template<typename T> + struct ForwardGatherScatter + { + typedef typename T::value_type V; + + static V gather(const T& a, int i) + { + return a[i]; + } + + static void scatter(T& a, V v, int i) + { + a[i] = v; + } + }; + + public: + + // send owner value to + template<class T> + void ownerToCopy (T& source, T& dest) + { + BC o2c; + o2c.template build<T>(cif); + o2c.template forward<ForwardGatherScatter<T> >(source,dest); + o2c.free(); + } + + // Constructor + // containers of IndexTripel and RemoteIndexTripel sorted appropriately + template<class C1, class C2> + OwnerCopyCommunication (C1& ownindices, C2& othersindices, MPI_Comm comm) + { + // Process configuration + int procs, rank; + MPI_Comm_size(comm, &procs); + MPI_Comm_rank(comm, &rank); + + // set up an ISTL index set + pis.beginResize(); + for (typename C1::iterator i=ownindices.begin(); i!=ownindices.end(); ++i) + { + if (i->third==owner) + pis.add(i->first,LI(i->second,owner,true)); + if (i->third==copy) + pis.add(i->first,LI(i->second,copy,true)); + std::cout << rank << ": adding index " << i->first << " " << i->second << " " << i->third << std::endl; + } + pis.endResize(); + + // build remote indices WITHOUT communication + std::cout << rank << ": build remote indices" << std::endl; + ri.setIndexSets(pis,pis,comm); + if (othersindices.size()>0) + { + typename C2::iterator i=othersindices.begin(); + int p = i->first; + RILM modifier = ri.template getModifier<false,true>(p); + typename PIS::const_iterator pi=pis.begin(); + for ( ; i!=othersindices.end(); ++i) + { + // handle processor change + if (p!=i->first) + { + p = i->first; + modifier = ri.template getModifier<false,true>(p); + pi=pis.begin(); + } + + // position to correct entry in parallel index set + while (pi->global()!=i->second && pi!=pis.end()) + ++pi; + if (pi==pis.end()) + DUNE_THROW(ISTLError,"OwnerOverlapCommunication: global index not in index set"); + + // insert entry + std::cout << rank << ": adding remote index " << i->first << " " << i->second << " " << i->third << std::endl; + if (i->third==owner) + modifier.insert(RX(owner,&(*pi))); + if (i->third==copy) + modifier.insert(RX(copy,&(*pi))); + } + } + + // now set up a communication interface + EnumItem<AttributeSet,owner> sourceFlags; + EnumItem<AttributeSet,copy> destFlags; + cif.build(ri,sourceFlags,destFlags); + } + + // destructor: free memory in some objects + ~OwnerCopyCommunication () + { + ri.free(); + cif.free(); + } + + private: + PIS pis; // parallel index set + RI ri; // remote indices + IF cif; // interface + }; + + + + + /** @} end documentation */ + +} // end namespace + +#endif