14#include "arcane/cartesianmesh/CartesianMeshCoarsening2.h"
16#include "arcane/utils/FatalErrorException.h"
17#include "arcane/utils/ValueConvert.h"
18#include "arcane/utils/SmallArray.h"
20#include "arcane/core/IMesh.h"
21#include "arcane/core/ItemGroup.h"
22#include "arcane/core/IParallelMng.h"
23#include "arcane/core/CartesianGridDimension.h"
24#include "arcane/core/IMeshModifier.h"
25#include "arcane/core/SimpleSVGMeshExporter.h"
26#include "arcane/core/ItemPrinter.h"
27#include "arcane/core/MeshStats.h"
28#include "arcane/core/IGhostLayerMng.h"
30#include "arcane/mesh/CellFamily.h"
32#include "arcane/cartesianmesh/ICartesianMesh.h"
33#include "arcane/cartesianmesh/CellDirectionMng.h"
34#include "arcane/cartesianmesh/internal/ICartesianMeshInternal.h"
36#include <unordered_set>
47CartesianMeshCoarsening2::
52 if (
auto v = Convert::Type<Int32>::tryParseFromEnvironment(
"ARCANE_CARTESIANMESH_COARSENING_VERBOSITY_LEVEL",
true))
53 m_verbosity_level = v.value();
75void CartesianMeshCoarsening2::
76_writeMeshSVG(
const String& name)
78 if (m_verbosity_level <= 0)
81 if (
mesh->dimension() != 2)
85 String filename = String::format(
"mesh_{0}_{1}.svg", name, mesh_rank);
86 info() <<
"WriteMesh name=" << filename;
87 std::ofstream ofile(filename.
localstr());
89 writer.write(
mesh->allCells());
100void CartesianMeshCoarsening2::
101_doDoubleGhostLayers()
119 m_cartesian_mesh->computeDirections();
122 _writeMeshSVG(
"double_ghost");
128void CartesianMeshCoarsening2::
131 if (m_is_create_coarse_called)
133 m_is_create_coarse_called =
true;
135 const bool is_verbose = m_verbosity_level > 0;
138 Integer nb_patch = m_cartesian_mesh->nbPatch();
140 ARCANE_FATAL(
"This method is only valid for 1 patch (nb_patch={0})", nb_patch);
142 if (!
mesh->isAmrActivated())
146 if (nb_dir != 2 && nb_dir != 3)
147 ARCANE_FATAL(
"This method is only valid for 2D or 3D mesh (dim={0})", nb_dir);
149 info() <<
"CoarseCartesianMesh nb_direction=" << nb_dir;
151 for (
Integer idir = 0; idir < nb_dir; ++idir) {
154 info() <<
"NB_OWN_CELL dir=" << idir <<
" n=" << nb_own_cell;
155 if ((nb_own_cell % 2) != 0)
156 ARCANE_FATAL(
"Invalid number of cells ({0}) for direction {1}. Should be a multiple of 2",
160 _writeMeshSVG(
"orig");
173 for (
Int32 i = 0; i < nb_node; ++i)
175 info() <<
"Orig cell_uid=" << cell.
uniqueId() <<
" Nodes=" << uids;
178 for (
Int32 i = 0; i < nb_face; ++i)
180 info() <<
"Orig cell_uid=" << cell.
uniqueId() <<
" Faces=" << uids;
191 m_first_own_cell_unique_id_offset = coarse_grid_cell_offset;
192 info() <<
"FirstCellUniqueIdOffset=" << m_first_own_cell_unique_id_offset;
195 else if (nb_dir == 2)
200 mesh->modifier()->endUpdate();
221 m_cartesian_mesh->_internalApi()->addPatchFromExistingChildren(parent_cells.
view().
localIds());
228 mesh->computeSynchronizeInfos();
232 m_cartesian_mesh->computeDirections();
234 _writeMeshSVG(
"coarse");
240void CartesianMeshCoarsening2::
241_createCoarseCells2D()
243 const bool is_verbose = m_verbosity_level > 0;
268 Int32 nb_coarse_face = 0;
269 Int32 nb_coarse_cell = 0;
280 Int64x3 cell_xy = refined_cell_uid_computer.compute(cell_uid);
281 const Int64 cell_x = cell_xy.x;
282 const Int64 cell_y = cell_xy.y;
288 if ((cell_x % 2) != 0 || (cell_y % 2) != 0)
291 info() <<
"CellToCoarse refined_uid=" << cell_uid <<
" x=" << cell_x <<
" y=" << cell_y;
292 coarse_cells_owner.
add(cell.
owner());
293 const Int64 coarse_cell_x = cell_x / 2;
294 const Int64 coarse_cell_y = cell_y / 2;
295 std::array<Int64, 4> node_uids_container;
297 node_uids[0] = refined_node_uid_computer.compute(cell_x + 0, cell_y + 0);
298 node_uids[1] = refined_node_uid_computer.compute(cell_x + 2, cell_y + 0);
299 node_uids[2] = refined_node_uid_computer.compute(cell_x + 2, cell_y + 2);
300 node_uids[3] = refined_node_uid_computer.compute(cell_x + 0, cell_y + 2);
302 info() <<
"CELLNodes uid=" << node_uids;
303 std::array<Int64, 4> coarse_face_uids = coarse_face_uid_computer.
computeForCell(coarse_cell_x, coarse_cell_y);
306 for (
Int32 z = 0; z < 4; ++z) {
308 faces_infos.
add(IT_Line2);
309 faces_infos.
add(coarse_face_uids[z]);
310 Int64 node_uid0 = node_uids[lface.
node(0)];
311 Int64 node_uid1 = node_uids[lface.
node(1)];
313 if (node_uid0 > node_uid1)
314 std::swap(node_uid0, node_uid1);
316 info() <<
"ADD_FACE coarse_uid=" << coarse_face_uids[z] <<
" n0=" << node_uid0 <<
" n1=" << node_uid1;
317 faces_infos.
add(node_uid0);
318 faces_infos.
add(node_uid1);
323 cells_infos.
add(IT_Quad4);
324 Int64 coarse_cell_uid = coarse_cell_uid_computer.compute(coarse_cell_x, coarse_cell_y);
325 cells_infos.
add(coarse_cell_uid);
327 info() <<
"CoarseCellUid=" << coarse_cell_uid;
329 for (
Int32 z = 0; z < 4; ++z)
330 cells_infos.
add(node_uids[z]);
332 first_child_cells.
add(cell);
337 std::array<Int32, 4> sub_cell_lids_container;
339 Cell cell1 = cdm_x[cell].next();
346 Cell cell2 = cdm_y[cell1].next();
349 Cell cell3 = cdm_y[cell].next();
364 for (
Int32 i = 0; i < 4; ++i)
365 refined_cells_lids.
add(sub_lids[i]);
371 mesh->modifier()->addFaces(nb_coarse_face, faces_infos, faces_local_ids);
379 mesh->modifier()->addCells(add_cells_args);
388 std::array<Int32, 4> sub_cell_lids_container;
390 for (
Int32 i = 0; i < nb_coarse_cell; ++i) {
391 Int32 coarse_cell_lid = cells_local_ids[i];
392 Cell coarse_cell = cells[coarse_cell_lid];
393 Cell first_child_cell = first_child_cells[i];
396 sub_cell_lids[0] = first_child_cell.
localId();
397 sub_cell_lids[1] = cdm_x[first_child_cell].next().localId();
398 sub_cell_lids[2] = cdm_y[CellLocalId(sub_cell_lids[1])].next().localId();
399 sub_cell_lids[3] = cdm_y[first_child_cell].next().localId();
401 info() <<
"AddChildForCoarseCell i=" << i <<
" coarse=" <<
ItemPrinter(coarse_cell)
402 <<
" children_lid=" << sub_cell_lids;
403 for (
Int32 z = 0; z < 4; ++z) {
404 Cell child_cell = cells[sub_cell_lids[z]];
406 info() <<
" AddParentCellToCell: z=" << z <<
" child=" <<
ItemPrinter(child_cell);
407 true_cell_family->_addParentCellToCell(child_cell, coarse_cell);
409 true_cell_family->_addChildrenCellsToCell(coarse_cell, sub_cell_lids);
418 Int32 owner = coarse_cells_owner[index];
420 const Int64 sub_cell_index = index * 4;
421 for (
Int32 z = 0; z < 4; ++z) {
434void CartesianMeshCoarsening2::
435_createCoarseCells3D()
437 const bool is_verbose = m_verbosity_level > 0;
464 Int32 nb_coarse_face = 0;
465 Int32 nb_coarse_cell = 0;
473 static constexpr Int32 const_cell_nb_node = 8;
474 static constexpr Int32 const_cell_nb_face = 6;
475 static constexpr Int32 const_cell_nb_sub_cell = 8;
476 static constexpr Int32 const_face_nb_node = 4;
485 Int64x3 cell_xyz = refined_cell_uid_computer.
compute(cell_uid);
486 const Int64 cell_x = cell_xyz.x;
487 const Int64 cell_y = cell_xyz.y;
488 const Int64 cell_z = cell_xyz.z;
494 if ((cell_x % 2) != 0 || (cell_y % 2) != 0 || (cell_z % 2) != 0)
497 info() <<
"CellToCoarse refined_uid=" << cell_uid <<
" x=" << cell_x <<
" y=" << cell_y <<
" z=" << cell_z;
498 coarse_cells_owner.
add(cell.
owner());
499 const Int64 coarse_cell_x = cell_x / 2;
500 const Int64 coarse_cell_y = cell_y / 2;
501 const Int64 coarse_cell_z = cell_z / 2;
502 std::array<Int64, const_cell_nb_node> node_uids_container;
504 node_uids[0] = refined_node_uid_computer.compute(cell_x + 0, cell_y + 0, cell_z + 0);
505 node_uids[1] = refined_node_uid_computer.compute(cell_x + 2, cell_y + 0, cell_z + 0);
506 node_uids[2] = refined_node_uid_computer.compute(cell_x + 2, cell_y + 2, cell_z + 0);
507 node_uids[3] = refined_node_uid_computer.compute(cell_x + 0, cell_y + 2, cell_z + 0);
508 node_uids[4] = refined_node_uid_computer.compute(cell_x + 0, cell_y + 0, cell_z + 2);
509 node_uids[5] = refined_node_uid_computer.compute(cell_x + 2, cell_y + 0, cell_z + 2);
510 node_uids[6] = refined_node_uid_computer.compute(cell_x + 2, cell_y + 2, cell_z + 2);
511 node_uids[7] = refined_node_uid_computer.compute(cell_x + 0, cell_y + 2, cell_z + 2);
513 info() <<
"CELLNodes uid=" << node_uids;
514 std::array<Int64, const_cell_nb_face> coarse_face_uids = coarse_face_uid_computer.
computeForCell(coarse_cell_x, coarse_cell_y, coarse_cell_z);
518 for (
Int32 z = 0; z < const_cell_nb_face; ++z) {
520 faces_infos.
add(IT_Quad4);
521 faces_infos.
add(coarse_face_uids[z]);
522 for (
Int32 knode = 0; knode < const_face_nb_node; ++knode)
523 face_node_uids[knode] = node_uids[lface.
node(knode)];
526 info() <<
"ADD_FACE coarse_uid=" << coarse_face_uids[z] <<
" n=" << face_sorted_node_uids;
527 faces_infos.
addRange(face_sorted_node_uids);
533 cells_infos.
add(IT_Hexaedron8);
534 Int64 coarse_cell_uid = coarse_cell_uid_computer.
compute(coarse_cell_x, coarse_cell_y, coarse_cell_z);
535 cells_infos.
add(coarse_cell_uid);
537 info() <<
"CoarseCellUid=" << coarse_cell_uid;
539 for (
Int32 z = 0; z < const_cell_nb_node; ++z)
540 cells_infos.
add(node_uids[z]);
542 first_child_cells.
add(cell);
549 std::array<Int32, const_cell_nb_sub_cell> sub_cell_lids_container;
551 Cell cell1 = cdm_x[cell].next();
558 Cell cell2 = cdm_y[cell1].next();
561 Cell cell3 = cdm_y[cell].next();
565 Cell cell4 = cdm_z[cell].next();
569 Cell cell5 = cdm_x[cell4].next();
572 Cell cell6 = cdm_y[cell5].next();
575 Cell cell7 = cdm_y[cell4].next();
599 for (
Int32 i = 0; i < const_cell_nb_sub_cell; ++i)
600 refined_cells_lids.
add(sub_lids[i]);
606 mesh->modifier()->addFaces(nb_coarse_face, faces_infos, faces_local_ids);
614 mesh->modifier()->addCells(add_cells_args);
623 std::array<Int32, const_cell_nb_sub_cell> sub_cell_lids_container;
625 for (
Int32 i = 0; i < nb_coarse_cell; ++i) {
626 Int32 coarse_cell_lid = cells_local_ids[i];
627 Cell coarse_cell = cells[coarse_cell_lid];
628 Cell first_child_cell = first_child_cells[i];
631 sub_cell_lids[0] = first_child_cell.
localId();
632 sub_cell_lids[1] = cdm_x[first_child_cell].next().localId();
633 sub_cell_lids[2] = cdm_y[CellLocalId(sub_cell_lids[1])].next().localId();
634 sub_cell_lids[3] = cdm_y[first_child_cell].next().localId();
636 Cell top_first_child_cell = cdm_z[first_child_cell].next();
637 sub_cell_lids[4] = top_first_child_cell.
localId();
638 sub_cell_lids[5] = cdm_x[top_first_child_cell].next().localId();
639 sub_cell_lids[6] = cdm_y[CellLocalId(sub_cell_lids[5])].next().localId();
640 sub_cell_lids[7] = cdm_y[top_first_child_cell].next().localId();
643 info() <<
"AddChildForCoarseCell i=" << i <<
" coarse=" <<
ItemPrinter(coarse_cell)
644 <<
" children_lid=" << sub_cell_lids;
645 for (
Int32 z = 0; z < const_cell_nb_sub_cell; ++z) {
646 Cell child_cell = cells[sub_cell_lids[z]];
648 info() <<
" AddParentCellToCell: z=" << z <<
" child=" <<
ItemPrinter(child_cell);
649 true_cell_family->_addParentCellToCell(child_cell, coarse_cell);
651 true_cell_family->_addChildrenCellsToCell(coarse_cell, sub_cell_lids);
660 Int32 owner = coarse_cells_owner[index];
662 const Int64 sub_cell_index = index * const_cell_nb_face;
663 for (
Int32 z = 0; z < const_cell_nb_face; ++z) {
678void CartesianMeshCoarsening2::
679_recomputeMeshGenerationInfo()
682 auto* cmgi = ICartesianMeshGenerationInfo::getReference(
mesh,
false);
691 cmgi->setOwnCellOffsets(v[0] / cf, v[1] / cf, v[2] / cf);
695 cmgi->setGlobalNbCells(v[0] / cf, v[1] / cf, v[2] / cf);
699 cmgi->setOwnNbCells(v[0] / cf, v[1] / cf, v[2] / cf);
701 cmgi->setFirstOwnCellUniqueId(m_first_own_cell_unique_id_offset);
707void CartesianMeshCoarsening2::
710 if (!m_is_create_coarse_called)
711 ARCANE_FATAL(
"You need to call createCoarseCells() before");
712 if (m_is_remove_refined_called)
714 m_is_remove_refined_called =
true;
716 const bool is_verbose = m_verbosity_level > 0;
721 info() <<
"RemoveRefinedCells nb_coarse_cell=" << m_coarse_cells_uid.size();
723 info() <<
"CoarseCells=" << m_coarse_cells_uid;
727 std::unordered_set<Int64> coarse_cells_set;
728 for (
Int64 cell_uid : m_coarse_cells_uid)
729 coarse_cells_set.insert(cell_uid);
732 Int32 local_id = icell.itemLocalId();
734 if (coarse_cells_set.find(cell.
uniqueId()) == coarse_cells_set.end())
735 cells_to_remove.
add(local_id);
738 info() <<
"CellsToRemove n=" << cells_to_remove.
size() <<
" list=" << cells_to_remove;
749 MeshStats ms(traceMng(), mesh, mesh->parallelMng());
754 m_cartesian_mesh->computeDirections();
#define ARCANE_CHECK_POINTER(ptr)
Macro retournant le pointeur ptr s'il est non nul ou lancant une exception s'il est nul.
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
bool reorderNodesOfFace(Int64ConstArrayView before_ids, Int64ArrayView after_ids)
Réordonne les noeuds d'une face.
Integer size() const
Nombre d'éléments du vecteur.
Vue modifiable d'un tableau d'un type T.
void resize(Int64 s)
Change le nombre d'éléments du tableau à s.
void addRange(ConstReferenceType val, Int64 n)
Ajoute n élément de valeur val à la fin du tableau.
void add(ConstReferenceType val)
Ajoute l'élément val à la fin du tableau.
Classe pour calculer en 2D le uniqueId() d'une maille en fonction de sa position dans la grille.
Classe pour calculer en 3D le uniqueId() d'une maille en fonction de sa position dans la grille.
Int64 compute(Int64 x, Int64 y, Int64 z)
Calcul le uniqueId() en fonction des coordonnées.
Classe pour calculer en 2D le uniqueId() d'une face en fonction de sa position dans la grille.
std::array< Int64, 4 > computeForCell(Int64 x, Int64 y)
Calcule les uniqueIds() des 4 faces de la mailles de coordonnées topologique (x,y)
Classe pour calculer en 2D le uniqueId() d'une face en fonction de sa position dans la grille.
std::array< Int64, 6 > computeForCell(Int64 x, Int64 y, Int64 z)
Calcule les uniqueIds() des 6 faces de la mailles de coordonnées topologique (x,y,...
Classe pour calculer en 2D le uniqueId() d'un noeud en fonction de sa position dans la grille.
Classe pour calculer en 3D le uniqueId() d'un noeud en fonction de sa position dans la grille.
Informations sur les dimensions d'une grille cartésienne.
CellUniqueIdComputer2D getCellComputer2D(Int64 offset) const
Instance pour calculer les uniqueId() des mailles pour cette grille.
CellUniqueIdComputer3D getCellComputer3D(Int64 offset) const
Instance pour calculer les uniqueId() des mailles pour cette grille.
FaceUniqueIdComputer3D getFaceComputer3D(Int64 offset) const
Instance pour calculer les uniqueId() des faces pour cette grille.
NodeUniqueIdComputer3D getNodeComputer3D(Int64 offset) const
Instance pour calculer les uniqueId() des noeuds pour cette grille.
NodeUniqueIdComputer2D getNodeComputer2D(Int64 offset) const
Instance pour calculer les uniqueId() des noeuds pour cette grille.
FaceUniqueIdComputer2D getFaceComputer2D(Int64 offset) const
Instance pour calculer les uniqueId() des faces pour cette grille.
void _doDoubleGhostLayers()
Double la couche de mailles fantômes du maillage initial.
void _createCoarseCells3D()
Int64 _getMaxUniqueId(const ItemGroup &group)
Retourne le max des uniqueId() des entités de group.
void _recomputeMeshGenerationInfo()
Recalcule les informations sur le nombre de mailles par direction.
UniqueArray< Int64 > m_coarse_cells_uid
uniqueId() des mailles grossières
void _createCoarseCells2D()
Infos sur les mailles d'une direction spécifique X,Y ou Z d'un maillage structuré.
Int64 globalNbCell() const
Nombre global de mailles dans cette direction.
Int32 ownNbCell() const
Nombre de mailles propres dans cette direction.
Vue sur les informations des mailles.
Face face(Int32 i) const
i-ème face de la maille
Int32 nbFace() const
Nombre de faces de la maille.
Vue constante d'un tableau de type T.
Interface d'un maillage cartésien.
virtual void setBuilderVersion(Integer n)=0
Positionne la version du constructeur de mailles fantômes. Pour l'instant (version 3....
virtual Integer nbGhostLayer() const =0
Nombre de couches fantômes.
virtual Integer builderVersion() const =0
Version du constructeur de mailles fantômes.
virtual void setNbGhostLayer(Integer n)=0
Positionne le nombre de couches fantômes.
Interface d'une famille d'entités.
virtual void notifyItemsOwnerChanged()=0
Notifie que les entités propres au sous-domaine de la famille ont été modifiées.
virtual ItemVectorView view(Int32ConstArrayView local_ids)=0
Vue sur les entités.
Interface de modification du maillage.
virtual void updateGhostLayers()=0
Mise à jour de la couche fantôme.
virtual void setDynamic(bool v)=0
Positionne la propriété indiquant si le maillage peut évoluer.
virtual void removeCells(Int32ConstArrayView cells_local_id)=0
Supprime des mailles.
virtual void endUpdate()=0
Notifie l'instance de la fin de la modification du maillage.
Interface du gestionnaire de parallélisme pour un sous-domaine.
virtual Int32 commRank() const =0
Rang de cette instance dans le communicateur.
virtual char reduce(eReduceType rt, char v)=0
Effectue la réduction de type rt sur le réel v et retourne la valeur.
@ II_JustRefined
L'entité vient d'être raffinée.
Groupe d'entités de maillage.
ItemVectorView view() const
Vue sur les entités du groupe.
Classe utilitaire pour imprimer les infos sur une entité.
Informations locales sur une face d'une maille.
Integer node(Integer i) const
Indice locale dans la maille du i-ème noeud de la face.
Infos sur un type d'entité du maillage.
LocalFace localFace(Integer id) const
Connectivité locale de la i-ème face de la maille.
Int32ConstArrayView localIds() const
Tableau des numéros locaux des entités.
Node node(Int32 i) const
i-ème noeud de l'entité
Int32 nbNode() const
Nombre de noeuds de l'entité
Classe de base d'un élément de maillage.
const ItemTypeInfo * typeInfo() const
Infos sur le type de l'entité.
impl::MutableItemBase mutableItemBase() const
Partie interne modifiable de l'entité.
constexpr Int32 localId() const
Identifiant local de l'entité dans le sous-domaine du processeur.
Int32 owner() const
Numéro du sous-domaine propriétaire de l'entité
ItemUniqueId uniqueId() const
Identifiant unique sur tous les domaines.
constexpr bool null() const
true si l'entité est nul (i.e. non connecté au maillage)
Arguments pour IMeshModifier::addCells().
void setAllowBuildFaces(bool v)
Indique si on autorise la création des faces associées.
void dumpStats() override
Imprime des infos sur le maillage.
void setOwner(Integer suid, Int32 current_sub_domain)
Positionne le numéro du sous-domaine propriétaire de l'entité.
void addFlags(Int32 added_flags)
Ajoute les flags \added_flags à ceux de l'entité
Exportation d'un maillage au format SVG.
Tableau 1D de données avec buffer pré-alloué sur la pile.
Chaîne de caractères unicode.
const char * localstr() const
Retourne la conversion de l'instance dans l'encodage UTF-8.
Classe d'accès aux traces.
TraceMessage info() const
Flot pour un message d'information.
ITraceMng * traceMng() const
Gestionnaire de trace.
Vecteur 1D de données avec sémantique par valeur (style STL).
T max(const T &a, const T &b, const T &c)
Retourne le maximum de trois éléments.
ItemGroupT< Cell > CellGroup
Groupe de mailles.
@ ReduceMax
Maximum des valeurs.
-*- tab-width: 2; indent-tabs-mode: nil; coding: utf-8-with-signature -*-
std::int64_t Int64
Type entier signé sur 64 bits.
Int32 Integer
Type représentant un entier.
std::int32_t Int32
Type entier signé sur 32 bits.