14#include "arcane/utils/CheckedConvert.h"
15#include "arcane/utils/PlatformUtils.h"
16#include "arcane/utils/StringBuilder.h"
21#include "arcane/core/ITimeLoopMng.h"
22#include "arcane/core/ITimeLoopService.h"
23#include "arcane/core/ITimeLoop.h"
24#include "arcane/core/TimeLoopEntryPointInfo.h"
25#include "arcane/core/IMesh.h"
26#include "arcane/core/IItemFamily.h"
27#include "arcane/core/ItemPrinter.h"
28#include "arcane/core/IParallelMng.h"
29#include "arcane/core/IMeshWriter.h"
30#include "arcane/core/MeshKind.h"
32#include "arcane/core/ICaseDocument.h"
33#include "arcane/core/IInitialPartitioner.h"
34#include "arcane/core/IMesh.h"
35#include "arcane/core/IItemFamily.h"
36#include "arcane/core/IMeshModifier.h"
37#include "arcane/core/IMeshUtilities.h"
38#include "arcane/core/ServiceBuilder.h"
40#include "arcane/core/IMeshPartitionerBase.h"
41#include "arcane/core/BasicService.h"
42#include "arcane/core/MeshReaderMng.h"
43#include "arcane/core/IGridMeshPartitioner.h"
44#include "arcane/core/ICartesianMeshGenerationInfo.h"
45#include "arcane/core/AbstractItemOperationByBasicType.h"
47#include "arcane/core/Connectivity.h"
49#include "arcane/cartesianmesh/CartesianMeshCoarsening.h"
50#include "arcane/cartesianmesh/CartesianMeshCoarsening2.h"
52#include "arcane/cartesianmesh/CartesianMeshUtils.h"
53#include "arcane/cartesianmesh/ICartesianMesh.h"
54#include "arcane/cartesianmesh/CellDirectionMng.h"
55#include "arcane/cartesianmesh/NodeDirectionMng.h"
56#include "arcane/cartesianmesh/CartesianConnectivity.h"
58#include "arcane/cartesianmesh/ICartesianMeshPatch.h"
60#include "arcane/tests/ArcaneTestGlobal.h"
61#include "arcane/tests/CartesianMeshTester_axl.h"
62#include "arcane/tests/CartesianMeshTestUtils.h"
63#include "arcane/tests/CartesianMeshV2TestUtils.h"
71using namespace Arcane;
78class CartesianMeshTesterModule
79:
public ArcaneCartesianMeshTesterObject
98 ~CartesianMeshTesterModule();
106 void buildInit()
override;
107 void compute()
override;
108 void init()
override;
128 void _testXmlInfos();
130 void _printCartesianMeshInfos();
131 void _checkFaceUniqueIdsAreContiguous();
133 void _testCoarsening();
134 void _checkSpecificApplyOperator();
140class CartesianMeshPartitionerService
146 : BasicService(sbi){}
153 if (!initial_partition)
162 Int64 nb_bloc = nb_rank * 3;
163 Int64 cell_index = 0;
164 info() <<
"Partitioning with 'CartesianMeshPartitionerService' nb_rank=" << nb_rank;
168 Int64 new_owner = ((cell_index * nb_bloc) / nb_cell) % nb_rank;
172 cells_new_owner.synchronize();
173 mesh->utilities()->changeOwnersFromCells();
187class CartesianMeshInitialPartitioner
193 CartesianMeshInitialPartitioner(
ISubDomain* sd)
197 void build()
override {}
201 info() <<
"Partitioning mesh name=" <<
mesh->name();
209 String service_name =
"CartesianMeshPartitionerTester";
210 auto mesh_partitioner(sbuilder.createReference(service_name,
mesh));
212 bool is_dynamic =
mesh->isDynamic();
213 mesh->modifier()->setDynamic(
true);
214 mesh->utilities()->partitionAndExchangeMeshWithReplication(mesh_partitioner.get(),
true);
215 mesh->modifier()->setDynamic(is_dynamic);
227CartesianMeshTesterModule::
229: ArcaneCartesianMeshTesterObject(mbi)
236, m_cartesian_mesh(nullptr)
237, m_initial_partitioner(nullptr)
242 m_initial_partitioner =
new CartesianMeshInitialPartitioner(sd);
243 info() <<
"SETTING INITIAL PARTITIONER";
253CartesianMeshTesterModule::
254~CartesianMeshTesterModule()
261void CartesianMeshTesterModule::
264 String time_loop_name(
"CartesianMeshTestLoop");
270 List<TimeLoopEntryPointInfo> clist;
271 clist.add(TimeLoopEntryPointInfo(
"CartesianMeshTester.buildInit"));
276 List<TimeLoopEntryPointInfo> clist;
277 clist.add(TimeLoopEntryPointInfo(
"CartesianMeshTester.init"));
282 List<TimeLoopEntryPointInfo> clist;
283 clist.add(TimeLoopEntryPointInfo(
"CartesianMeshTester.compute"));
289 clist.add(
"CartesianMeshTester");
292 clist.add(
"ArcanePostProcessing");
293 clist.add(
"ArcaneCheckpoint");
303void CartesianMeshTesterModule::
306 bool has_edge = options()->hasEdges();
307 info() <<
"Adding edge connectivity?=" << has_edge;
309 Connectivity c(mesh()->connectivity());
310 c.enableConnectivity(Connectivity::CT_HasEdge);
313 m_global_deltat.assign(1.0);
315 IItemFamily* cell_family = defaultMesh()->cellFamily();
319 IItemFamily* face_family = defaultMesh()->faceFamily();
334void CartesianMeshTesterModule::
337 IMesh* mesh = defaultMesh();
339 IItemFamily* cell_family = mesh->
cellFamily();
347 IItemFamily* face_family = defaultMesh()->faceFamily();
367 for( NodeLocalId inode : cell.
nodeIds() )
368 center += nodes_coord[inode];
370 m_cell_center[icell] = center;
380 for( NodeLocalId inode : face.
nodeIds() )
381 center += nodes_coord[inode];
383 m_face_center[iface] = center;
388 m_cartesian_mesh->computeDirections();
390 m_utils =
makeRef(
new CartesianMeshTestUtils(m_cartesian_mesh,acceleratorMng()));
391 m_utils_v2 =
makeRef(
new CartesianMeshV2TestUtils(m_cartesian_mesh));
398 Integer nb_dir = defaultMesh()->dimension();
399 for(
Integer idir=0; idir<nb_dir; ++idir){
400 CellDirectionMng cdm(m_cartesian_mesh->cellDirection(idir));
404 DirCell cc(cdm.cell(*icell));
405 Cell next = cc.next();
406 Cell prev = cc.previous();
411 m_density[icell] += 5.0;
416 DirCell cc(cdm[icell]);
418 info() <<
"CELL: cell=" << ItemPrinter(*icell)
419 <<
" next=" << ItemPrinter(cc.next())
420 <<
" previous=" << ItemPrinter(cc.previous());
423 m_density[icell] += 5.0;
426 info() <<
"NB_BOUNDARY1=" << nb_boundary1 <<
" NB_BOUNDARY2=" << nb_boundary2;
429 m_utils->testAll(
false);
430 m_utils_v2->testAll();
431 _checkFaceUniqueIdsAreContiguous();
434 _printCartesianMeshInfos();
435 _checkSpecificApplyOperator();
441void CartesianMeshTesterModule::
444 Int32 coarse_version = options()->coarseCartesianMesh();
445 IMesh* mesh = m_cartesian_mesh->mesh();
447 const Int32 coarse_factor = 1 << mesh_dim;
448 if (coarse_version==1){
449 info() <<
"Test CartesianCoarsening V1";
450 Ref<CartesianMeshCoarsening> coarser = m_cartesian_mesh->createCartesianMeshCoarsening();
451 IItemFamily* cell_family = mesh->
cellFamily();
452 CellInfoListView cells(cell_family);
453 coarser->createCoarseCells();
455 for(
Int32 cell_lid : coarser->coarseCells()){
456 Cell cell = cells[cell_lid];
457 info() <<
"Test1: CoarseCell= " << ItemPrinter(cell);
458 ConstArrayView<Int32> sub_cells(coarser->refinedCells(index));
460 for(
Int32 sub_lid : sub_cells )
461 info() <<
"SubCell=" << ItemPrinter(cells[sub_lid]);
463 coarser->removeRefinedCells();
466 if (coarse_version==2){
467 info() <<
"Test CartesianCoarsening V2";
468 const Int32 nb_orig_cell = ownCells().size();
470 coarser->createCoarseCells();
475 info() <<
"Test2: CoarseCell= " << ItemPrinter(cell);
477 info() <<
"SubCell=" << ItemPrinter(cell.
hChild(i));
480 Int32 nb_patch = m_cartesian_mesh->nbPatch();
481 info() <<
"NB_PATCH=" << nb_patch;
482 for(
Int32 i=0; i<nb_patch; ++i ){
483 ICartesianMeshPatch* p = m_cartesian_mesh->patch(i);
484 info() <<
"Patch i=" << i <<
" nb_cell=" << p->
cells().
size();
486 coarser->removeRefinedCells();
489 const Int32 nb_final_cell = ownCells().size();
490 info() <<
"nb_orig_cell=" << nb_orig_cell <<
" nb_final_cell=" << nb_final_cell
491 <<
" coarse_factor=" << coarse_factor;
492 const Int32 nb_computed = nb_final_cell * coarse_factor;
493 if (nb_computed != nb_orig_cell)
494 ARCANE_FATAL(
"Bad number of cells orig={0} computed={1}", nb_orig_cell, nb_computed);
501void CartesianMeshTesterModule::
504 if (m_global_iteration()==1)
513void CartesianMeshTesterModule::
524 Integer nb_dir = defaultMesh()->dimension();
525 for(
Integer idir=0; idir<nb_dir; ++idir){
526 m_old_density.copy(m_density);
527 CellDirectionMng cdm(m_cartesian_mesh->cellDirection(idir));
530 DirCell cc(cdm.cell(*icell));
531 Cell next = cc.next();
532 Cell prev = cc.previous();
533 Real d = m_old_density[icell] + m_old_density[next] + m_old_density[prev];
534 m_density[icell] = d / 3.0;
539 DirCell cc(cdm[icell]);
540 Cell next = cc.next();
541 Cell prev = cc.previous();
542 Real d = m_old_density[icell];
545 d += m_old_density[next];
549 d += m_old_density[prev];
552 m_density[icell] = d / n;
557 Int64 to_add = m_global_iteration();
559 Int64 uid = iface->uniqueId();
560 m_faces_uid[iface] = uid + to_add;
562 m_faces_uid.synchronize();
566 Int64 expected_value = uid + to_add;
567 if (expected_value!=m_faces_uid[iface])
568 ARCANE_FATAL(
"Bad FaceUid face={0} expected={1} value={2}",ItemPrinter(face),expected_value,m_faces_uid[iface]);
576void CartesianMeshTesterModule::
586 Integer nb_dir = defaultMesh()->dimension();
587 for(
Integer idir=0; idir<nb_dir; ++idir){
588 CellDirectionMng cdm(m_cartesian_mesh->cellDirection(idir));
591 DirCell cc(cdm[icell]);
592 Cell next = cc.next();
593 Cell prev = cc.previous();
594 Real d = m_density[icell];
597 d += m_density[next];
601 d += m_density[prev];
604 m_density[icell] = d / n;
632void CartesianMeshTesterModule::
633_checkFaceUniqueIdsAreContiguous()
635 if (!options()->checkContiguousFaceUniqueIds())
637 info() <<
"Test " << A_FUNCINFO;
640 Int64 total_nb_face = allFaces().own().size();
642 info() <<
"TotalNbFace=" << total_nb_face;
646 ARCANE_FATAL(
"FaceUniqueId is too big: uid={0} total_nb_face={1}",
654void CartesianMeshTesterModule::
657 info() <<
"PRINT Xml infos for <cartesian> mesh generator";
658 ICaseDocument* cd = subDomain()->caseDocument();
660 if (mesh_elements.size()==0)
662 XmlNode mesh_generator_element = mesh_elements[0].child(
"meshgenerator");
664 if (mesh_generator_element.
null()){
665 info() <<
"No element <meshgenerator> found";
668 XmlNode cartesian_node = mesh_generator_element.
child(
"cartesian");
669 if (cartesian_node.
null()){
670 info() <<
"No element <cartesian> found";
673 XmlNode origine_node = cartesian_node.
child(
"origine");
674 XmlNode nsd_node = cartesian_node.
child(
"nsd");
677 XmlNodeList lx_node_list = cartesian_node.
children(
"lx");
678 info() <<
"NB_X=" << lx_node_list.size();
679 for( XmlNode lx_node : lx_node_list ){
680 Real lx_value = lx_node.valueAsReal(
true);
681 Integer nx_value = lx_node.attr(
"nx",
true).valueAsInteger(
true);
682 Real px_value = lx_node.attr(
"prx").valueAsReal(
true);
683 info() <<
"V=" << lx_value <<
" nx=" << nx_value <<
" px=" << px_value;
693 if (!options()->unstructuredMeshFile.isPresent())
703 info() <<
"UnstructuredMeshFileName=" << file_name;
712 info() <<
"MESH=" << new_mesh;
716 auto partitioner_ref = sbuilder.
createReference(
"SimpleGridMeshPartitioner",new_mesh);
728 Real min_value = -max_value;
735 Real3 coord = nodes_coord[node];
748 const bool is_debug =
false;
756 mesh_writer->writeMeshToFile(new_mesh,fname);
762 mesh_writer->writeMeshToFile(current_mesh,fname);
770void CartesianMeshTesterModule::
776 ARCANE_FATAL(
"Bad value X expected={0} value={1} message={2}",a.x,b.
x,message);
778 ARCANE_FATAL(
"Bad value Y expected={0} value={1} message={2}",a.y,b.
y,message);
780 ARCANE_FATAL(
"Bad value Z expected={0} value={1} message={2}",a.z,b.
z,message);
786void CartesianMeshTesterModule::
787_printCartesianMeshInfos()
789 auto* cartesian_info = ICartesianMeshGenerationInfo::getReference(defaultMesh(),
false);
793 info() <<
"Test: _printCartesianMeshInfos()";
794 info() <<
" Origin=" << cartesian_info->globalOrigin();
795 info() <<
" Length=" << cartesian_info->globalLength();
797 if (options()->expectedMeshOrigin.isPresent())
798 _checkNearlyEqual(cartesian_info->globalOrigin(),options()->expectedMeshOrigin(),
"Origin");
799 if (options()->expectedMeshLength.isPresent())
800 _checkNearlyEqual(cartesian_info->globalLength(),options()->expectedMeshLength(),
"Length");
806void CartesianMeshTesterModule::
807_checkSpecificApplyOperator()
815 info() <<
"MeshStructure=" << mk;
816 if (mk != eMeshStructure::Cartesian)
817 ARCANE_FATAL(
"Invalid mesh structure v={0} (expected 'Cartesian')", mk);
819 if (nb_item != op.m_nb_hexa8)
820 ARCANE_FATAL(
"Bad number of Hexa8 n={0} expected={1}", op.m_nb_hexa8, nb_item);
823 if (nb_item != op.m_nb_quad4)
824 ARCANE_FATAL(
"Bad number of Quad8 n={0} expected={1}", op.m_nb_quad4, nb_item);
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
Fonctions mathématiques diverses.
Fonctions utilitaires sur le maillage.
Ce fichier contient les différentes fabriques de services et macro pour enregistrer les services.
#define ARCANE_SERVICE_INTERFACE(ainterface)
Macro pour déclarer une interface lors de l'enregistrement d'un service.
void partitionAndDistributeMeshes(ConstArrayView< IMesh * > meshes) override
Partitionne les maillages.
void notifyEndPartition() override
Notification lors de la fin d'un repartionnement (après échange des entités)
void build() override
Construction de niveau build du service.
void partitionMesh(bool initial_partition) override
IPrimaryMesh * primaryMesh() override
Maillage associé
Module de test pour les infos sur les maillages cartésiens.
void _testGridPartitioning()
void _sample(ICartesianMesh *cartesian_mesh)
Opérateur abstrait sur des entités rangées par type.
Informations de connectivité d'un maillage cartésien.
Cell lowerRight(Node n) const
Maille en bas à droite du noeud n.
Cell upperLeft(Node n) const
Maille en haut à gauche du noeud n.
Cell lowerLeft(Node n) const
Maille en bas à gauche du noeud n.
Cell upperRight(Node n) const
Maille en haut à droite du noeud n.
Int32 subDomainOffset() const
Offset dans cette direction du sous-domaine.
Int32 nbHChildren() const
Nombre d'enfants pour l'AMR.
Cell hChild(Int32 i) const
i-ème enfant AMR
void clear()
Supprime tous les éléments de la collection.
Vue constante d'un tableau de type T.
Informations sur le type flottant.
virtual ITraceMng * traceMng() const =0
Gestionnaire de traces.
virtual CellGroup cells()=0
Groupe de mailles du patch.
Interface d'un maillage cartésien.
static ICartesianMesh * getReference(const MeshHandleOrMesh &mesh, bool create=true)
Récupère ou créé la référence associée à mesh.
virtual CellDirectionMng cellDirection(eMeshDirection dir)=0
Liste des mailles dans la direction dir.
virtual CartesianConnectivity connectivity()=0
Informations sur la connectivité
virtual IMesh * mesh() const =0
Maillage associé à ce maillage cartésien.
virtual const XmlNodeList & meshElements()=0
Retourne l'élément racine des informations de maillage.
Interface d'un partitionneur de maillage sur une grille.
virtual void setBoundingBox(Real3 min_val, Real3 max_val)=0
Positionne la bounding box de notre sous-domaine.
virtual void applyMeshPartitioning(IMesh *mesh)=0
Applique le repartitionnement sur le maillage mesh.
virtual void setPartIndex(Int32 i, Int32 j, Int32 k)=0
Indice (i,j,k) de la partie.
Interface d'un partitionneur initial.
Interface d'une famille d'entités.
virtual ItemGroup allItems() const =0
Groupe de toutes les entités.
virtual ItemGroup createGroup(const String &name, Int32ConstArrayView local_ids, bool do_override=false)=0
Créé un groupe d'entités de nom name contenant les entités local_ids.
virtual VariableItemInt32 & itemsNewOwner()=0
Variable contenant le numéro du nouveau sous-domaine propriétaire de l'entité.
virtual CellGroup ownCells()=0
Groupe de toutes les mailles propres au domaine.
virtual Integer dimension()=0
Dimension du maillage (1D, 2D ou 3D).
virtual IItemFamily * cellFamily()=0
Retourne la famille des mailles.
Interface d'un partitionneur de maillage.
virtual VariableNodeReal3 & nodesCoordinates()=0
Coordonnées des noeuds.
virtual IParallelMng * parallelMng()=0
Gestionnaire de parallèlisme.
virtual const MeshKind meshKind() const =0
Caractéristiques 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 Int32 commSize() const =0
Nombre d'instance dans le communicateur.
Interface du gestionnaire d'un sous-domaine.
virtual ITimeLoopMng * timeLoopMng()=0
Retourne le gestionnaire de la boucle en temps.
virtual void setInitialPartitioner(IInitialPartitioner *partitioner)=0
Positionne le partitionneur initial.
virtual ITimeLoop * createTimeLoop(const String &name)=0
Crée une boucle en temps de nom name.
virtual void registerTimeLoop(ITimeLoop *time_loop)=0
Enregistrement et choix de la boucle en temps.
virtual void setRequiredModulesName(const StringCollection &)=0
Positionne la liste des des modules obligatoires.
virtual void setOptionalModulesName(const StringCollection &)=0
Positionne la liste des des modules facultatifs.
static const char * WBuild
appelé lors de la lecture du jeu de données
static const char * WComputeLoop
appelé pendant la boucle de calcul
virtual void setEntryPoints(const String &where, const TimeLoopEntryPointInfoCollection &)=0
Positionne la liste des noms des points d'entrée pour le point d'appel where.
static const char * WInit
appelé pendant l'initialisation, l'initialisation d'une reprise ou d'un nouveau cas
Groupe d'entités de maillage.
Integer size() const
Nombre d'éléments du groupe.
void applyOperation(IItemOperationByBasicType *operation) const
Applique l'opération operation sur les entités du groupe.
ItemGroup own() const
Groupe équivalent à celui-ci mais contenant uniquement les éléments propres au sous-domaine.
Classe utilitaire pour imprimer les infos sur une entité.
Vue sur un vecteur d'entités.
Int32 size() const
Nombre d'éléments du vecteur.
NodeConnectedListViewType nodes() const
Liste des noeuds de l'entité
Int32 nbNode() const
Nombre de noeuds de l'entité
NodeLocalIdView nodeIds() const
Liste des noeuds 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)
Gestionnaire de lecteurs de maillage.
IMesh * readMesh(const String &mesh_name, const String &file_name)
Lit le maillage dont le nom de fichier est file_name.
Informations pour construire un module.
ISubDomain * subDomain() const
Accès au sous-domaine associé
Classe gérant un vecteur de réel de dimension 3.
Référence à une instance.
Structure contenant les informations pour créer un service.
Classe utilitaire pour instantier un service d'une interface donnée.
Ref< InterfaceType > createReference(const String &name, eServiceBuilderProperties properties=SB_None)
Créé une instance implémentant l'interface InterfaceType.
Propriétés de création d'un service.
Constructeur de chaîne de caractère unicode.
Chaîne de caractères unicode.
TraceAccessor(ITraceMng *m)
Construit un accesseur via le gestionnaire de trace m.
TraceMessage info() const
Flot pour un message d'information.
Paramètres nécessaires à la construction d'une variable.
XmlNode child(const String &name) const
Noeud fils de celui-ci de nom name.
XmlNodeList children(const String &name) const
Ensemble des noeuds fils de ce noeud ayant pour nom name.
bool null() const
Vrai si le noeud est nul.
__host__ __device__ Real2 min(Real2 a, Real2 b)
Retourne le minimum de deux Real2.
T max(const T &a, const T &b, const T &c)
Retourne le maximum de trois éléments.
ItemGroupT< Cell > CellGroup
Groupe de mailles.
#define ARCANE_REGISTER_SERVICE(aclass, a_service_property,...)
Macro pour enregistrer un service.
MeshVariableScalarRefT< Cell, Real > VariableCellReal
Grandeur au centre des mailles de type réel.
MeshVariableScalarRefT< Node, Real > VariableNodeReal
Grandeur au noeud de type réel.
MeshVariableScalarRefT< Cell, Real3 > VariableCellReal3
Grandeur au centre des mailles de type coordonnées.
MeshVariableScalarRefT< Node, Real3 > VariableNodeReal3
Grandeur au noeud de type coordonnées.
ItemVariableScalarRefT< Int32 > VariableItemInt32
Grandeur de type entier 32 bits.
MeshVariableScalarRefT< Face, Real3 > VariableFaceReal3
Grandeur aux faces de type coordonnées.
MeshVariableScalarRefT< Face, Int64 > VariableFaceInt64
Grandeur aux faces de type entier 64 bits.
ARCANE_CARTESIANMESH_EXPORT Ref< CartesianMeshCoarsening2 > createCartesianMeshCoarsening2(ICartesianMesh *cm)
Créé une instance pour gérer le déraffinement du maillage (V2).
Int32 toInt32(Int64 v)
Converti un Int64 en un Int32.
@ ReduceSum
Somme des valeurs.
constexpr __host__ __device__ bool isNearlyEqual(const _Type &a, const _Type &b)
Teste si deux valeurs sont à un peu près égales. Pour les types entiers, cette fonction est équivalen...
@ SB_Collective
Indique que tous les processus font la même opération.
std::int64_t Int64
Type entier signé sur 64 bits.
Int32 Integer
Type représentant un entier.
eMeshStructure
Structure du maillage.
@ ST_SubDomain
Le service s'utilise au niveau du sous-domaine.
UniqueArray< Int32 > Int32UniqueArray
Tableau dynamique à une dimension d'entiers 32 bits.
List< String > StringList
Tableau de chaînes de caractères unicode.
double Real
Type représentant un réel.
auto makeRef(InstanceType *t) -> Ref< InstanceType >
Créé une référence sur un pointeur.
std::int32_t Int32
Type entier signé sur 32 bits.
Real y
deuxième composante du triplet
Real z
troisième composante du triplet
Real x
première composante du triplet