/*---------------------------------------------------------------------------*\
========= |
\\ / F ield | OpenFOAM: The Open Source CFD Toolbox
\\ / O peration |
\\ / A nd | www.openfoam.com
\\/ M anipulation |
-------------------------------------------------------------------------------
Copyright (C) 2011-2016 OpenFOAM Foundation
Copyright (C) 2015-2025 OpenCFD Ltd.
-------------------------------------------------------------------------------
License
This file is part of OpenFOAM.
OpenFOAM 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.
OpenFOAM 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 OpenFOAM. If not, see .
Class
Foam::mapDistribute
Description
Class containing processor-to-processor mapping information.
We store mapping from the bits-to-send to the complete starting list
(subXXXMap) and from the received bits to their location in the new
list (constructXXXMap).
Note:
Schedule is a list of processor pairs (one send, one receive. One of
them will be myself) which forms a scheduled (i.e. non-buffered) exchange.
See distribute on how to use it.
Note2: number of items sent on one processor have to equal the number
of items received on the other processor.
To aid constructing these maps there are the constructors from global
numbering, either with or without transforms.
- without transforms:
Constructors using compact numbering: layout is
- all my own elements first (whether used or not)
- followed by used-only remote elements sorted by remote processor.
So e.g 4 procs and on proc 1 the compact
table will first have all globalIndex.localSize() elements from proc1
followed by used-only elements of proc0, proc2, proc3.
The constructed mapDistribute sends the local elements from and
receives the remote elements into their compact position.
compactMap[proci] is the position of elements from proci in the compact
map. compactMap[myProcNo()] is empty since trivial addressing.
It rewrites the input global indices into indices into the constructed
data.
- with transforms:
This requires the precalculated set of possible transforms
(globalIndexAndTransform). These are given as permutations (+, -, or none)
of up to 3 independent transforms.
The layout of the data is
- all my own elements first (whether used or not)
- followed by used-only remote elements sorted by remote processor.
- followed by - for each transformation index - the set of local or
remote elements with that transformation.
The inputs for the constructor are
- the set of untransformed local or remote indices in globalIndex
numbering. These get rewritten to be indices into the layout of the data.
- the set of transformed local or remote indices in globalIndexAndTransform
encoding. These are labelPairs.
Any distribute with transforms is now done as:
1. exchange data with other processors and receive these into the
slots for that processor
2. for all transformations transform a subset of the data according
to transformElements_[transformI] and store this starting from
transformStart_[transformI]
In the same way a reverse distribute will
1. apply the inverse transform to the data starting at
transformStart_[transformI] and copy the result back into the
transformElements_[transformI]. These might be local or remote slots.
2. the data in the remote slots will now be sent back to the correct
location in the originating processor.
E.g. a map to handle
- mesh points on a mesh with
- 1 cyclic so 3 permutations (+,-,none) will have layout
- on e.g. processor 1 out of 2:
+------+ <- transformStart[2]
| |
| | <- transform2 applied to data in local or remote slots
| |
+------+ <- transformStart[1]
| |
| | <- transform1 applied to data in local or remote slots
| |
+------+ <- transformStart[1]
| |
| | <- transform0 applied to data in local or remote slots
| |
+------+ <- transformStart[0]
| |
| | <- data from proc2
| |
+------+
| |
| | <- data from proc0
| |
+------+ <- mesh.nPoints()
| |
| |
| |
+------+ 0
When constructing from components optionally a 'flip' on
the maps can be specified. This will interpret the map
values as index+flip, similar to e.g. faceProcAddressing. The flip
will only be applied to fieldTypes (scalar, vector, .. triad)
SourceFiles
mapDistribute.C
mapDistributeIO.C
mapDistributeTemplates.C
\*---------------------------------------------------------------------------*/
#ifndef Foam_mapDistribute_H
#define Foam_mapDistribute_H
#include "mapDistributeBase.H"
#include "transformList.H"
#include "vectorTensorTransform.H"
#include "coupledPolyPatch.H"
// * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * //
namespace Foam
{
// Forward Declarations
class globalIndexAndTransform;
class mapDistribute;
Istream& operator>>(Istream&, mapDistribute&);
Ostream& operator<<(Ostream&, const mapDistribute&);
/*---------------------------------------------------------------------------*\
Class mapDistribute Declaration
\*---------------------------------------------------------------------------*/
class mapDistribute
:
public mapDistributeBase
{
// Private Data
//- For every globalIndexAndTransform::transformPermutations
//- gives the elements that need to be transformed
labelListList transformElements_;
//- Destination in constructMap for transformed elements
labelList transformStart_;
// Private Member Functions
//- Helper function: copy transformElements without transformation
template
void applyDummyTransforms(UList& field) const;
template
void applyTransforms
(
const globalIndexAndTransform& globalTransforms,
UList& field,
const TransformOp& top
) const;
//- Helper function: copy transformElements without transformation
template
void applyDummyInverseTransforms(UList& field) const;
template
void applyInverseTransforms
(
const globalIndexAndTransform& globalTransforms,
UList& field,
const TransformOp& top
) const;
//- Helper: convert mapDistribute to mapDistributeBase
static UPtrList extractBase
(
const UPtrList& maps
);
public:
// Public classes
//- Default transformation behaviour
class transform
{
public:
template
void operator()
(
const vectorTensorTransform& vt,
const bool forward,
UList& fld
) const
{
if constexpr (is_rotational_vectorspace_v)
{
const tensor rot(forward ? vt.R() : vt.R().T());
transformList(rot, fld);
}
}
template
void operator()
(
const vectorTensorTransform& vt,
const bool forward,
List>& flds
) const
{
if constexpr (is_rotational_vectorspace_v)
{
const tensor rot(forward ? vt.R() : vt.R().T());
for (auto& fld : flds)
{
transformList(rot, fld);
}
}
}
//- Transform patch-based field
template
void operator()(const coupledPolyPatch& cpp, UList& fld) const
{
if constexpr (is_rotational_vectorspace_v)
{
if (!cpp.parallel())
{
transformList(cpp.forwardT(), fld);
}
}
}
//- Transform sparse field
template class Container>
void operator()(const coupledPolyPatch& cpp, Container& map)
const
{
if constexpr (is_rotational_vectorspace_v)
{
if (!cpp.parallel())
{
transformList(cpp.forwardT(), map);
}
}
}
};
//- Default transformation behaviour for position
class transformPosition
{
public:
void operator()
(
const vectorTensorTransform& vt,
const bool forward,
UList& fld
) const
{
if (forward)
{
vt.transformPositionList(fld);
}
else
{
vt.invTransformPositionList(fld);
}
}
void operator()
(
const vectorTensorTransform& vt,
const bool forward,
List>& flds
) const
{
if (forward)
{
for (auto& fld : flds)
{
vt.transformPositionList(fld);
}
}
else
{
for (auto& fld : flds)
{
vt.invTransformPositionList(fld);
}
}
}
//- Transform patch-based field
void operator()(const coupledPolyPatch& cpp, pointField& fld) const
{
cpp.transformPosition(fld);
}
template class Container>
void operator()(const coupledPolyPatch& cpp, Container& map)
const
{
Field fld(map.size());
label i = 0;
forAllConstIters(map, iter)
{
fld[i++] = *iter;
}
cpp.transformPosition(fld);
i = 0;
forAllIters(map, iter)
{
*iter = fld[i++];
}
}
};
// Declare name of the class and its debug switch
ClassName("mapDistribute");
// Constructors
//- Inherit constructors
using mapDistributeBase::mapDistributeBase;
//- Default construct - uses worldComm
mapDistribute() noexcept;
//- Default construct with specified communicator
explicit mapDistribute(const label comm) noexcept;
//- Move construct from base, no transforms
explicit mapDistribute(mapDistributeBase&& map);
//- Copy construct
explicit mapDistribute(const mapDistribute& map);
//- Move construct
explicit mapDistribute(mapDistribute&& map);
//- Read construct from dictionary
explicit mapDistribute
(
const dictionary& dict,
const label comm = UPstream::worldComm
);
//- Move construct from components
mapDistribute
(
const label constructSize,
labelListList&& subMap,
labelListList&& constructMap,
labelListList&& transformElements,
labelList&& transformStart,
const bool subHasFlip = false,
const bool constructHasFlip = false,
const label comm = UPstream::worldComm
);
//- Construct from list of (possibly remote) untransformed elements
//- in globalIndex numbering (or -1) and (possibly remote)
//- transformed elements in globalIndexAndTransform numbering.
// Determines compact numbering (see above) and
// distribute map to get data into this ordering and renumbers the
// elements to be in compact numbering.
mapDistribute
(
const globalIndex&,
labelList& untransformedElements,
const globalIndexAndTransform&,
const labelPairList& transformedElements,
labelList& transformedIndices,
List