Arcane  v3.16.0.0
Documentation développeur
Chargement...
Recherche...
Aucune correspondance
AMRCartesianMeshTesterModule.cc
1// -*- tab-width: 2; indent-tabs-mode: nil; coding: utf-8-with-signature -*-
2//-----------------------------------------------------------------------------
3// Copyright 2000-2025 CEA (www.cea.fr) IFPEN (www.ifpenergiesnouvelles.com)
4// See the top-level COPYRIGHT file for details.
5// SPDX-License-Identifier: Apache-2.0
6//-----------------------------------------------------------------------------
7/*---------------------------------------------------------------------------*/
8/* AMRCartesianMeshTesterModule.cc (C) 2000-2025 */
9/* */
10/* Module de test du gestionnaire de maillages cartésiens AMR. */
11/*---------------------------------------------------------------------------*/
12/*---------------------------------------------------------------------------*/
13
14#include "arcane/utils/CheckedConvert.h"
15#include "arcane/utils/PlatformUtils.h"
16#include "arcane/utils/Real2.h"
17#include "arcane/utils/MD5HashAlgorithm.h"
18
20#include "arcane/core/MeshKind.h"
21#include "arcane/core/Directory.h"
22
23#include "arcane/core/ITimeLoopMng.h"
24#include "arcane/core/ITimeLoopService.h"
25#include "arcane/core/ITimeLoop.h"
26#include "arcane/core/TimeLoopEntryPointInfo.h"
27#include "arcane/core/IMesh.h"
28#include "arcane/core/IItemFamily.h"
29#include "arcane/core/ItemPrinter.h"
30#include "arcane/core/IParallelMng.h"
31
32#include "arcane/core/IMesh.h"
33#include "arcane/core/IItemFamily.h"
34#include "arcane/core/IMeshModifier.h"
35#include "arcane/core/IMeshUtilities.h"
36#include "arcane/core/ServiceBuilder.h"
38#include "arcane/core/MeshStats.h"
39#include "arcane/core/IPostProcessorWriter.h"
40#include "arcane/core/IVariableMng.h"
41#include "arcane/core/SimpleSVGMeshExporter.h"
42#include "arcane/core/IGhostLayerMng.h"
43
44#include "arcane/cartesianmesh/ICartesianMesh.h"
45#include "arcane/cartesianmesh/CellDirectionMng.h"
46#include "arcane/cartesianmesh/FaceDirectionMng.h"
47#include "arcane/cartesianmesh/NodeDirectionMng.h"
48#include "arcane/cartesianmesh/CartesianConnectivity.h"
49#include "arcane/cartesianmesh/CartesianMeshRenumberingInfo.h"
50#include "arcane/cartesianmesh/ICartesianMeshPatch.h"
51#include "arcane/cartesianmesh/CartesianMeshUtils.h"
52#include "arcane/cartesianmesh/CartesianMeshCoarsening2.h"
53#include "arcane/cartesianmesh/CartesianMeshPatchListView.h"
54
55#include "arcane/tests/ArcaneTestGlobal.h"
56#include "arcane/tests/AMRCartesianMeshTester_axl.h"
57#include "arcane/tests/CartesianMeshTestUtils.h"
58
59/*---------------------------------------------------------------------------*/
60/*---------------------------------------------------------------------------*/
61
62namespace ArcaneTest
63{
64
65using namespace Arcane;
66
67/*---------------------------------------------------------------------------*/
68/*---------------------------------------------------------------------------*/
72class AMRCartesianMeshTesterModule
73: public ArcaneAMRCartesianMeshTesterObject
74{
75 public:
76
77 explicit AMRCartesianMeshTesterModule(const ModuleBuildInfo& mbi);
78 ~AMRCartesianMeshTesterModule();
79
80 public:
81
82 static void staticInitialize(ISubDomain* sd);
83
84 public:
85
86 void buildInit() override;
87 void compute() override;
88 void init() override;
89
90 private:
91
92 VariableCellReal m_density;
93 VariableCellReal m_old_density;
94 VariableCellReal3 m_cell_center;
95 VariableFaceReal3 m_face_center;
96 VariableNodeReal m_node_density;
97 ICartesianMesh* m_cartesian_mesh;
99 UniqueArray<VariableCellReal*> m_cell_patch_variables;
100 Int32 m_nb_expected_patch = 0;
101
102 private:
103
104 void _compute1();
105 void _compute2();
106 void _initAMR();
107 void _coarseZone();
108 void _reduceNbGhostLayers();
109 void _computeSubCellDensity(Cell cell);
110 void _computeCenters();
111 void _processPatches();
112 void _writePostProcessing();
113 void _checkUniqueIds();
114 void _testDirections();
115 void _checkDirections();
116 String _checkDirectionUniqueIdsHashCollective(ArrayView<Int64> own_items_uid_around, Integer nb_items_around);
117 Integer _cellsUidAroundCells(UniqueArray<Int64>& own_cells_uid_around_cells);
118 Integer _cellsUidAroundFaces(UniqueArray<Int64>& own_cells_uid_around_faces);
119 Integer _nodesUidAroundNodes(UniqueArray<Int64>& own_nodes_uid_around_nodes);
120 void _checkSync();
121 void _cellsInPatch(Real3 position, Real3 length, bool is_3d, Int32 level, UniqueArray<Int32>& cells_in_patch);
122};
123
124/*---------------------------------------------------------------------------*/
125/*---------------------------------------------------------------------------*/
126
127AMRCartesianMeshTesterModule::
128AMRCartesianMeshTesterModule(const ModuleBuildInfo& mbi)
129: ArcaneAMRCartesianMeshTesterObject(mbi)
130, m_density(VariableBuildInfo(this,"Density"))
131, m_old_density(VariableBuildInfo(this,"OldDensity"))
132, m_cell_center(VariableBuildInfo(this,"CellCenter"))
133, m_face_center(VariableBuildInfo(this,"FaceCenter"))
134, m_node_density(VariableBuildInfo(this,"NodeDensity"))
135, m_cartesian_mesh(nullptr)
136{
137}
138
139/*---------------------------------------------------------------------------*/
140/*---------------------------------------------------------------------------*/
141
142AMRCartesianMeshTesterModule::
143~AMRCartesianMeshTesterModule()
144{
145 for (VariableCellReal* v : m_cell_patch_variables)
146 delete v;
147}
148
149/*---------------------------------------------------------------------------*/
150/*---------------------------------------------------------------------------*/
151
152void AMRCartesianMeshTesterModule::
153staticInitialize(ISubDomain* sd)
154{
155 String time_loop_name("AMRCartesianMeshTestLoop");
156
157 ITimeLoopMng* tlm = sd->timeLoopMng();
158 ITimeLoop* time_loop = tlm->createTimeLoop(time_loop_name);
159
160 {
161 List<TimeLoopEntryPointInfo> clist;
162 clist.add(TimeLoopEntryPointInfo("AMRCartesianMeshTester.buildInit"));
163 time_loop->setEntryPoints(ITimeLoop::WBuild,clist);
164 }
165
166 {
167 List<TimeLoopEntryPointInfo> clist;
168 clist.add(TimeLoopEntryPointInfo("AMRCartesianMeshTester.init"));
169 time_loop->setEntryPoints(ITimeLoop::WInit,clist);
170 }
171
172 {
173 List<TimeLoopEntryPointInfo> clist;
174 clist.add(TimeLoopEntryPointInfo("AMRCartesianMeshTester.compute"));
175 time_loop->setEntryPoints(ITimeLoop::WComputeLoop,clist);
176 }
177
178 {
179 StringList clist;
180 clist.add("AMRCartesianMeshTester");
181 time_loop->setRequiredModulesName(clist);
182 clist.clear();
183 clist.add("ArcanePostProcessing");
184 clist.add("ArcaneCheckpoint");
185 clist.add("ArcaneLoadBalance");
186 time_loop->setOptionalModulesName(clist);
187 }
188
189 tlm->registerTimeLoop(time_loop);
190}
191
192/*---------------------------------------------------------------------------*/
193/*---------------------------------------------------------------------------*/
194
195void AMRCartesianMeshTesterModule::
196buildInit()
197{
198 if (subDomain()->isContinue())
199 return;
200
201 m_global_deltat.assign(1.0);
202
203 IItemFamily* cell_family = defaultMesh()->cellFamily();
204 cell_family->createGroup("CELL0");
205 cell_family->createGroup("CELL1");
206 cell_family->createGroup("CELL2");
207 cell_family->createGroup("AMRPatchCells0");
208 cell_family->createGroup("AMRPatchCells1");
209 cell_family->createGroup("AMRPatchCells2");
210 cell_family->createGroup("AMRPatchCells3");
211 cell_family->createGroup("AMRPatchCells4");
212 cell_family->createGroup("AMRPatchCells5");
213
214 IItemFamily* face_family = defaultMesh()->faceFamily();
215 face_family->createGroup("FACE0");
216 face_family->createGroup("FACE1");
217 face_family->createGroup("FACE2");
218 face_family->createGroup("FACE3");
219 face_family->createGroup("FACE4");
220 face_family->createGroup("FACE5");
221
222 face_family->createGroup("AllFacesDirection0");
223 face_family->createGroup("AllFacesDirection1");
224}
225
226/*---------------------------------------------------------------------------*/
227/*---------------------------------------------------------------------------*/
228
229void AMRCartesianMeshTesterModule::
230init()
231{
232 info() << "AMR Init";
233
234 IMesh* mesh = defaultMesh();
235
236 IItemFamily* cell_family = mesh->cellFamily();
237 Int32UniqueArray ids(1);
238 ids[0] = 0;
239 cell_family->createGroup("CELL0",ids,true);
240 ids[0] = 1;
241 cell_family->createGroup("CELL1",ids,true);
242 ids[0] = 2;
243 cell_family->createGroup("CELL2",ids,true);
244 IItemFamily* face_family = defaultMesh()->faceFamily();
245 ids[0] = 0;
246 face_family->createGroup("FACE0",ids,true);
247 ids[0] = 1;
248 face_family->createGroup("FACE1",ids,true);
249 ids[0] = 2;
250 face_family->createGroup("FACE2",ids,true);
251 ids[0] = 3;
252 face_family->createGroup("FACE3",ids,true);
253 ids[0] = 4;
254 face_family->createGroup("FACE4",ids,true);
255 ids[0] = 5;
256 face_family->createGroup("FACE5",ids,true);
257
258 m_cartesian_mesh = ICartesianMesh::getReference(mesh);
259 m_utils = makeRef(new CartesianMeshTestUtils(m_cartesian_mesh,acceleratorMng()));
260
261 if (!subDomain()->isContinue()) {
262 _initAMR();
263 _coarseZone();
264 _reduceNbGhostLayers();
265 }
266
267 _computeCenters();
268
269
270 const bool do_coarse_at_init = options()->coarseAtInit();
271
272 const Integer dimension = defaultMesh()->dimension();
273 if (dimension==2)
274 m_nb_expected_patch = 1 + options()->refinement2d().size();
275 else if (dimension==3)
276 m_nb_expected_patch = 1 + options()->refinement3d().size();
277
278 // Si on dé-raffine à l'init, on aura un patch de plus
279 if (do_coarse_at_init)
280 ++m_nb_expected_patch;
281
282 if (subDomain()->isContinue())
283 m_cartesian_mesh->recreateFromDump();
284 else{
285 m_cartesian_mesh->computeDirections();
286 CartesianMeshRenumberingInfo renumbering_info;
287 renumbering_info.setRenumberPatchMethod(options()->renumberPatchMethod());
288 renumbering_info.setSortAfterRenumbering(true);
289 if (options()->coarseAtInit())
290 renumbering_info.setParentPatch(m_cartesian_mesh->amrPatch(1));
291 m_cartesian_mesh->renumberItemsUniqueId(renumbering_info);
292 _checkUniqueIds();
293 _processPatches();
294 info() << "MaxUid for mesh=" << MeshUtils::getMaxItemUniqueIdCollective(m_cartesian_mesh->mesh());
295 }
296
297 // Initialise la densité.
298 // On met une densité de 1.0 à l'intérieur
299 // et on ajoute une densité de 5.0 pour chaque direction dans les
300 // mailles de bord.
301 m_density.fill(1.0);
302 for( Integer idir=0, nb_dir=dimension; idir<nb_dir; ++idir){
303 CellDirectionMng cdm(m_cartesian_mesh->cellDirection(idir));
304 Integer nb_boundary1 = 0;
305 Integer nb_boundary2 = 0;
306 ENUMERATE_CELL(icell,cdm.innerCells()){
307 DirCell cc(cdm.cell(*icell));
308 Cell next = cc.next();
309 Cell prev = cc.previous();
310 if (next.null() || prev.null()){
311 // Maille au bord. J'ajoute de la densité.
312 // Ne devrait pas arriver car on est sur les innerCells()
313 ++nb_boundary1;
314 m_density[icell] += 5.0;
315 }
316 }
317 // Parcours les mailles frontières pour la direction
318 ENUMERATE_CELL(icell,cdm.outerCells()){
319 DirCell cc(cdm[icell]);
320 if (icell.index()<5)
321 info() << "CELL: cell=" << ItemPrinter(*icell)
322 << " next=" << ItemPrinter(cc.next())
323 << " previous=" << ItemPrinter(cc.previous());
324 // Maille au bord. J'ajoute de la densité.
325 ++nb_boundary2;
326 m_density[icell] += 5.0;
327 }
328
329 info() << "NB_BOUNDARY1=" << nb_boundary1 << " NB_BOUNDARY2=" << nb_boundary2;
330 }
331 bool is_amr = m_nb_expected_patch!=1;
332 if (options()->verbosityLevel()==0)
333 m_utils->setNbPrint(5);
334 m_utils->testAll(is_amr);
335 _writePostProcessing();
336 _testDirections();
337 _checkDirections();
338 _checkSync();
339}
340
341/*---------------------------------------------------------------------------*/
342/*---------------------------------------------------------------------------*/
343
344void AMRCartesianMeshTesterModule::
345_checkUniqueIds()
346{
347 IMesh* mesh = m_cartesian_mesh->mesh();
348 bool print_hash = true;
349 bool with_ghost = options()->hashWithGhost();
350 MD5HashAlgorithm hash_algo;
351 MeshUtils::checkUniqueIdsHashCollective(mesh->nodeFamily(), &hash_algo, options()->nodesUidHash(),
352 print_hash, with_ghost);
353 MeshUtils::checkUniqueIdsHashCollective(mesh->faceFamily(), &hash_algo, options()->facesUidHash(),
354 print_hash, with_ghost);
355 MeshUtils::checkUniqueIdsHashCollective(mesh->cellFamily(), &hash_algo, options()->cellsUidHash(),
356 print_hash, with_ghost);
357}
358
359/*---------------------------------------------------------------------------*/
360/*---------------------------------------------------------------------------*/
361
362void AMRCartesianMeshTesterModule::
363_processPatches()
364{
365 const bool do_check = true;
366 const bool is_verbose = options()->verbosityLevel()>=1;
367
368 const Int32 dimension = defaultMesh()->dimension();
369
370 bool without_coarse_zone = true;
371 if (dimension == 2)
372 without_coarse_zone = options()->coarseZone2d().empty();
373 else if (dimension == 3)
374 without_coarse_zone = options()->coarseZone3d().empty();
375
376 // Vérifie qu'il y a autant de patchs que d'options raffinement dans
377 // le jeu de données (en comptant le patch 0 qui est le maillage cartésien).
378 // Cela permet de vérifier que les appels successifs
379 // à computeDirections() n'ajoutent pas de patchs.
380 // Cette vérification ne s'applique que s'il n'y a pas de zone de dé-raffinement.
381 // En effet, dé-raffiner un patch complet le supprime de la liste des patchs.
382 Integer nb_expected_patch = m_nb_expected_patch;
383
384 Integer nb_patch = m_cartesian_mesh->nbPatch();
385 if (without_coarse_zone && nb_expected_patch != nb_patch)
386 ARCANE_FATAL("Bad number of patchs expected={0} value={1}",nb_expected_patch,nb_patch);
387
388 IParallelMng* pm = parallelMng();
389 Int32 comm_rank = pm->commRank();
390 Int32 comm_size = pm->commSize();
391
392 UniqueArray<Int32> nb_cells_expected(options()->expectedNumberOfCellsInPatchs);
393 if (nb_cells_expected.size()!=nb_patch)
394 ARCANE_FATAL("Bad size ({0}, expected={1}) for option '{2}'",
395 nb_cells_expected.size(),nb_patch,options()->expectedNumberOfCellsInPatchs.name());
396
397 // Nombre de mailles fantômes attendu. Utilisé uniquement en parallèle
398 bool has_expected_ghost_cells = options()->expectedNumberOfGhostCellsInPatchs.isPresent();
399 if (!pm->isParallel())
400 has_expected_ghost_cells = false;
401
402 UniqueArray<Int32> nb_ghost_cells_expected(options()->expectedNumberOfGhostCellsInPatchs);
403 if (has_expected_ghost_cells && (nb_ghost_cells_expected.size()!=nb_patch))
404 ARCANE_FATAL("Bad size ({0}, expected={1}) for option '{2}'",
405 nb_ghost_cells_expected.size(), nb_patch, options()->expectedNumberOfGhostCellsInPatchs.name());
406 // Affiche les informations sur les patchs
407 for( Integer i=0; i<nb_patch; ++i ){
408 ICartesianMeshPatch* p = m_cartesian_mesh->patch(i);
409 CellGroup patch_cells(p->cells());
410 info() << "Patch cell_group=" << patch_cells.name() << " nb_cell=" << patch_cells.size();
411 VariableCellReal* cellv = new VariableCellReal(VariableBuildInfo(defaultMesh(),String("CellPatch")+i));
412 m_cell_patch_variables.add(cellv);
413 cellv->fill(0.0);
414 ENUMERATE_CELL(icell,patch_cells){
415 (*cellv)[icell] = 2.0;
416 }
417
418 CellGroup patch_own_cell = patch_cells.own();
419 UniqueArray<Int64> own_cells_uid;
420 ENUMERATE_(Cell,icell,patch_own_cell){
421 Cell cell{*icell};
422 if (is_verbose)
423 info() << "Patch i=" << i << " cell=" << ItemPrinter(*icell);
424 own_cells_uid.add(cell.uniqueId());
425 }
426 // Affiche la liste globales des uniqueId() des mailles.
427 {
428 UniqueArray<Int64> global_cells_uid;
429 pm->allGatherVariable(own_cells_uid,global_cells_uid);
430 std::sort(global_cells_uid.begin(),global_cells_uid.end());
431 Integer nb_global_uid = global_cells_uid.size();
432 info() << "GlobalUids Patch=" << i << " NB=" << nb_global_uid
433 << " expected=" << nb_cells_expected[i];
434 // Vérifie que le nombre de mailles par patch est le bon.
435 if (do_check && nb_cells_expected[i]!=nb_global_uid)
436 ARCANE_FATAL("Bad number of cells for patch I={0} N={1} expected={2}",
437 i,nb_global_uid,nb_cells_expected[i]);
438 if (is_verbose)
439 for( Integer c=0; c<nb_global_uid; ++c )
440 info() << "GlobalUid Patch=" << i << " I=" << c << " cell_uid=" << global_cells_uid[c];
441 }
442 // Teste le nombre de mailles fantômes
443 if (has_expected_ghost_cells){
444 Int32 local_nb_ghost_cell = patch_cells.size() - patch_own_cell.size();
445 Int32 total = pm->reduce(Parallel::ReduceSum,local_nb_ghost_cell);
446 pinfo() << "NbGhostCells my_rank=" << comm_rank << " local=" << local_nb_ghost_cell << " total=" << total;
447 if (total!=nb_ghost_cells_expected[i])
448 ARCANE_FATAL("Bad number of ghost cells for patch I={0} N={1} expected={2}",
449 i,total,nb_ghost_cells_expected[i]);
450 }
451
452 // Exporte le patch au format SVG
453 if (dimension==2 && options()->dumpSvg()){
454 String filename = String::format("Patch{0}-{1}-{2}.svg",i,comm_rank,comm_size);
455 Directory directory = subDomain()->exportDirectory();
456 String full_filename = directory.file(filename);
457 std::ofstream ofile(full_filename.localstr());
458 SimpleSVGMeshExporter exporter(ofile);
459 exporter.write(patch_cells);
460 }
461 }
462}
463
464/*---------------------------------------------------------------------------*/
465/*---------------------------------------------------------------------------*/
466
467void AMRCartesianMeshTesterModule::
468_computeCenters()
469{
470 IMesh* mesh = defaultMesh();
471
472 // Calcule le centre des mailles
473 {
474 VariableNodeReal3& nodes_coord = mesh->nodesCoordinates();
475 ENUMERATE_CELL(icell,allCells()){
476 Cell cell = *icell;
477 Real3 center;
478 for( NodeLocalId inode : cell.nodes() )
479 center += nodes_coord[inode];
480 center /= cell.nbNode();
481 m_cell_center[icell] = center;
482 }
483 }
484
485 // Calcule le centre des faces
486 {
487 VariableNodeReal3& nodes_coord = mesh->nodesCoordinates();
488 ENUMERATE_FACE(iface,allFaces()){
489 Face face = *iface;
490 Real3 center;
491 for( NodeLocalId inode : face.nodes() )
492 center += nodes_coord[inode];
493 center /= face.nbNode();
494 m_face_center[iface] = center;
495 }
496 }
497}
498
499/*---------------------------------------------------------------------------*/
500/*---------------------------------------------------------------------------*/
501
502void AMRCartesianMeshTesterModule::
503_initAMR()
504{
505 // Regarde si on dé-raffine le maillage initial
506 if (options()->coarseAtInit()){
507 // Il faut que les directions aient été calculées avant d'appeler le dé-raffinement
508 m_cartesian_mesh->computeDirections();
509
510 info() << "Doint initial coarsening";
511
512 if (m_cartesian_mesh->mesh()->meshKind().meshAMRKind() == eMeshAMRKind::PatchCartesianMeshOnly) {
513 debug() << "Coarse with specific coarser (for cartesian mesh only)";
514 Ref<ICartesianMeshAMRPatchMng> coarser = CartesianMeshUtils::cartesianMeshAMRPatchMng(m_cartesian_mesh);
515 coarser->createSubLevel();
516 }
517 else {
518 Ref<CartesianMeshCoarsening2> coarser = CartesianMeshUtils::createCartesianMeshCoarsening2(m_cartesian_mesh);
519 coarser->createCoarseCells();
520 }
521
522 CartesianMeshPatchListView patches = m_cartesian_mesh->patches();
523 Int32 nb_patch = patches.size();
524 {
525 Int32 index = 0;
526 info() << "NB_PATCH=" << nb_patch;
527 for( CartesianPatch p : patches){
528 info() << "Patch i=" << index << " nb_cell=" << p.cells().size();
529 ++index;
530 }
531 }
532 }
533 // Parcours les mailles actives et ajoute dans la liste des mailles
534 // à raffiner celles qui sont contenues dans le boîte englobante
535 // spécifiée dans le jeu de données.
536 Int32 dim = defaultMesh()->dimension();
537 if (dim==2){
538 for( const auto& x : options()->refinement2d() ){
539 m_cartesian_mesh->refinePatch({x->position(), x->length()});
540 m_cartesian_mesh->computeDirections();
541 }
542 }
543 if (dim==3){
544 for( const auto& x : options()->refinement3d() ){
545 m_cartesian_mesh->refinePatch({x->position(), x->length()});
546 m_cartesian_mesh->computeDirections();
547 }
548 }
549}
550
551/*---------------------------------------------------------------------------*/
552/*---------------------------------------------------------------------------*/
553
554void AMRCartesianMeshTesterModule::
555_coarseZone()
556{
557 Int32 dim = defaultMesh()->dimension();
558
559 if (dim == 2) {
560 //UniqueArray<Int32> cells_in_patchs;
561 for (auto& x : options()->coarseZone2d()) {
562 // _cellsInPatch(Real3(x->position()), Real3(x->length()), false, x->level(), cells_in_patchs);
563 // defaultMesh()->modifier()->flagCellToCoarsen(cells_in_patchs);
564 // defaultMesh()->modifier()->coarsenItemsV2(true);
565 // cells_in_patchs.clear();
566 m_cartesian_mesh->coarseZone({{x->position()}, {x->length()}});
567 m_cartesian_mesh->computeDirections();
568 }
569 }
570 if (dim == 3) {
571 // UniqueArray<Int32> cells_in_patchs;
572 for (auto& x : options()->coarseZone3d()) {
573 // _cellsInPatch(x->position(), x->length(), true, x->level(), cells_in_patchs);
574 // defaultMesh()->modifier()->flagCellToCoarsen(cells_in_patchs);
575 // defaultMesh()->modifier()->coarsenItemsV2(true);
576 // cells_in_patchs.clear();
577 m_cartesian_mesh->coarseZone({{x->position()}, {x->length()}});
578 m_cartesian_mesh->computeDirections();
579 }
580 }
581}
582
583/*---------------------------------------------------------------------------*/
584/*---------------------------------------------------------------------------*/
585
586void AMRCartesianMeshTesterModule::
587_reduceNbGhostLayers()
588{
589 for (auto& x : options()->reduceNbGhostLayers()) {
590 Integer final_nb_ghost_layer{ m_cartesian_mesh->reduceNbGhostLayers(x->level(), x->nbGhostLayers()) };
591
592 if (parallelMng()->commSize() != 1 && final_nb_ghost_layer != x->nbGhostLayers()) {
593 ARCANE_FATAL("Bad nb ghost layers");
594 }
595 //info() << "Final nb ghost layer : " << final_nb_ghost_layer;
596 }
597}
598
599/*---------------------------------------------------------------------------*/
600/*---------------------------------------------------------------------------*/
601
602void AMRCartesianMeshTesterModule::
603compute()
604{
605 _compute1();
606}
607
608/*---------------------------------------------------------------------------*/
609/*---------------------------------------------------------------------------*/
615{
616 Int32 nb_children = cell.nbHChildren();
617 if (nb_children==0)
618 return;
619 // Pour les mailles AMR, la densité est la moyenne des noeuds qui la compose.
620 for( Int32 j=0; j<nb_children; ++j ) {
621 Real sub_density = 0.0;
622 Cell sub_cell = cell.hChild(j);
623 Integer sub_cell_nb_node = sub_cell.nbNode();
624 for( Integer k=0; k<sub_cell_nb_node; ++k )
625 sub_density += m_node_density[sub_cell.node(k)];
626 sub_density /= (Real)sub_cell_nb_node;
627 m_density[sub_cell] =sub_density;
628 _computeSubCellDensity(sub_cell);
629 }
630}
631
632/*---------------------------------------------------------------------------*/
633/*---------------------------------------------------------------------------*/
634
635void AMRCartesianMeshTesterModule::
636_compute1()
637{
638 // Pour test, on parcours les N directions
639 // et pour chaque maille, on modifie sa densité
640 // par la formule new_density = (density+density_next+density_prev) / 3.0.
641
642 // Effectue l'operation en deux fois. Une premiere sur les
643 // mailles internes, et une deuxieme sur les mailles externes.
644 // Du coup, il faut passer par une variable intermediaire (m_old_density)
645 // mais on evite un test dans la boucle principale
646 IMesh* mesh = defaultMesh();
647 Integer nb_dir = mesh->dimension();
648 for( Integer idir=0; idir<nb_dir; ++idir){
649 m_old_density.copy(m_density);
650 CellDirectionMng cdm(m_cartesian_mesh->cellDirection(idir));
651 // Travail sur les mailles internes
652 info() << "Direction=" << idir << " cells=" << cdm.innerCells().name()
653 << " n=" << cdm.innerCells().size();
654 ENUMERATE_CELL(icell,cdm.innerCells()){
655 Cell cell = *icell;
656 DirCell cc(cdm.cell(cell));
657 Cell next = cc.next();
658 Cell prev = cc.previous();
659 Real d = m_old_density[icell] + m_old_density[next] + m_old_density[prev];
660 m_density[icell] = d / 3.0;
662 }
663 // Travail sur les mailles externes
664 // Test si la maille avant ou apres est nulle.
665 ENUMERATE_CELL(icell,cdm.outerCells()){
666 Cell cell = *icell;
667 DirCell cc(cdm[icell]);
668 Cell next = cc.next();
669 Cell prev = cc.previous();
670 Real d = m_old_density[icell];
671 Integer n = 1;
672 if (!next.null()){
673 d += m_old_density[next];
674 ++n;
675 }
676 if (!prev.null()){
677 d += m_old_density[prev];
678 ++n;
679 }
680 m_density[icell] = d / n;
682 }
683 }
684 // Modifie la densité aux noeuds.
685 // Elle sera égale à la moyenne des densités des mailles entourant ce noeud
686 ENUMERATE_NODE(inode,mesh->allNodes()){
687 Node node = *inode;
688 Integer nb_cell = node.nbCell();
689 Real density = 0.0;
690 for( Integer i=0; i<nb_cell; ++i )
691 density += m_density[node.cell(i)];
692 density /= (Real)nb_cell;
693 m_node_density[inode] = density;
694 }
695}
696
697/*---------------------------------------------------------------------------*/
698/*---------------------------------------------------------------------------*/
699
700void AMRCartesianMeshTesterModule::
701_compute2()
702{
703 // Pour test, on parcours les N directions
704 // et pour chaque maille, on modifie sa densité
705 // par la formule new_density = (density+density_next+density_prev) / 3.0.
706
707 // A noter que cette methode ne donne pas le meme comportement que
708 // _compute1() car les mailles de bord et internes sont mises à jour
709 // dans un ordre différent.
710 Integer nb_dir = defaultMesh()->dimension();
711 for( Integer idir=0; idir<nb_dir; ++idir){
712 CellDirectionMng cdm(m_cartesian_mesh->cellDirection(idir));
713 // Travail sur toutes les mailles
714 ENUMERATE_CELL(icell,cdm.allCells()){
715 DirCell cc(cdm[icell]);
716 Cell next = cc.next();
717 Cell prev = cc.previous();
718 Real d = m_density[icell];
719 Integer n = 1;
720 if (!next.null()){
721 d += m_density[next];
722 ++n;
723 }
724 if (!prev.null()){
725 d += m_density[prev];
726 ++n;
727 }
728 m_density[icell] = d / n;
729 }
730 }
731}
732
733/*---------------------------------------------------------------------------*/
734/*---------------------------------------------------------------------------*/
735
736void AMRCartesianMeshTesterModule::
737_writePostProcessing()
738{
739 info() << "Post-process AMR";
740 IPostProcessorWriter* post_processor = options()->postProcessor();
741 Directory output_directory = Directory(subDomain()->exportDirectory(),"amrtestpost1");
742 output_directory.createDirectory();
743 info() << "Creating output dir '" << output_directory.path() << "' for export";
744 UniqueArray<Real> times;
745 times.add(m_global_time());
746 post_processor->setTimes(times);
747 post_processor->setMesh(defaultMesh());
748 post_processor->setBaseDirectoryName(output_directory.path());
749
750 VariableList variables;
751 //variables.add(m_density.variable());
752 //variables.add(m_node_density.variable());
753 for( VariableCellReal* v : m_cell_patch_variables )
754 variables.add(v->variable());
755 post_processor->setVariables(variables);
756 ItemGroupList groups;
757 groups.add(allCells());
758 for( CartesianPatch p : m_cartesian_mesh->patches() )
759 groups.add(p.cells());
760 post_processor->setGroups(groups);
761 IVariableMng* vm = subDomain()->variableMng();
762 vm->writePostProcessing(post_processor);
763}
764
765/*---------------------------------------------------------------------------*/
766/*---------------------------------------------------------------------------*/
767
768void AMRCartesianMeshTesterModule::
769_testDirections()
770{
771 Integer nb_patch = m_cartesian_mesh->nbPatch();
772 Integer nb_dir = m_cartesian_mesh->mesh()->dimension();
773 NodeDirectionMng node_dm2;
774 for( Integer ipatch=0; ipatch<nb_patch; ++ipatch ){
775 ICartesianMeshPatch* p = m_cartesian_mesh->patch(ipatch);
776 for( Integer idir=0; idir<nb_dir; ++idir ){
777 NodeDirectionMng node_dm(p->nodeDirection(idir));
778 node_dm2 = p->nodeDirection(idir);
779 NodeGroup dm_all_nodes = node_dm.allNodes();
780 ENUMERATE_NODE(inode,dm_all_nodes){
781 DirNode dir_node(node_dm[inode]);
782 DirNode dir_node2(node_dm2[inode]);
783 Node prev_node = dir_node.previous();
784 Node next_node = dir_node.next();
785 Node prev_node2 = dir_node2.previous();
786 Node next_node2 = dir_node2.next();
787 m_utils->checkSameId(prev_node, prev_node2);
788 m_utils->checkSameId(next_node, next_node2);
789 }
790 }
791 }
792}
793
794void AMRCartesianMeshTesterModule::
795_checkDirections()
796{
797 m_cartesian_mesh->computeDirections();
798 IMesh* mesh = m_cartesian_mesh->mesh();
799 bool print_hash = true;
800
801 auto check_hash = [&](const IItemFamily* item_family, const String& expected_hash, ArrayView<Int64> own_items_uid_around, Integer nb_items_around) {
802 String cell_hash = _checkDirectionUniqueIdsHashCollective(own_items_uid_around, nb_items_around);
803
804 if (print_hash) {
805 info() << "HASH_RESULT direction items of family=" << item_family->name()
806 << " v= " << cell_hash << " expected= " << expected_hash;
807 }
808
809 if (cell_hash != expected_hash)
810 ARCANE_FATAL("Bad hash for uniqueId() for direction items of family '{0}' v= {1} expected='{2}'",
811 item_family->fullName(), cell_hash, expected_hash);
812 };
813
814 if (!options()->cellsDirectionHash().empty()) {
815 debug() << "Check cells direction hash";
816 UniqueArray<Int64> own_cells_uid_around_cells;
817 Integer nb_items_around = _cellsUidAroundCells(own_cells_uid_around_cells);
818 check_hash(mesh->cellFamily(), options()->cellsDirectionHash(), own_cells_uid_around_cells, nb_items_around);
819 }
820 if (!options()->facesDirectionHash().empty()) {
821 debug() << "Check faces direction hash";
822 UniqueArray<Int64> own_cells_uid_around_faces;
823 Integer nb_items_around = _cellsUidAroundFaces(own_cells_uid_around_faces);
824 check_hash(mesh->faceFamily(), options()->facesDirectionHash(), own_cells_uid_around_faces, nb_items_around);
825 }
826
827 if (!options()->nodesDirectionHash().empty()) {
828 debug() << "Check nodes direction hash";
829 UniqueArray<Int64> own_nodes_uid_around_nodes;
830 Integer nb_items_around = _nodesUidAroundNodes(own_nodes_uid_around_nodes);
831 check_hash(mesh->nodeFamily(), options()->nodesDirectionHash(), own_nodes_uid_around_nodes, nb_items_around);
832 }
833}
834
846_checkDirectionUniqueIdsHashCollective(ArrayView<Int64> own_items_uid_around, Integer nb_items_around)
847{
848 // +1 car on a le uid dedans.
849 Integer size_of_once_case_around = nb_items_around + 1;
850
851 UniqueArray<Int64> final_all_items_uid;
852 {
853 UniqueArray<Int64> global_items_uid_around;
854 parallelMng()->allGatherVariable(own_items_uid_around, global_items_uid_around);
855
856 UniqueArray<Int64*> global_items_uid(global_items_uid_around.size() / size_of_once_case_around);
857 {
858 Int64 index = 0;
859 for (Int64 i = 0; i < global_items_uid_around.size(); i += size_of_once_case_around) {
860 Int64 uid = global_items_uid_around[i];
861 ARCANE_ASSERT((uid != -1), ("Un uid dans le tableau est = -1"));
862 global_items_uid[index++] = &(global_items_uid_around[i]);
863 }
864 }
865
866 std::sort(global_items_uid.begin(), global_items_uid.end(),
867 [](const Int64* a, const Int64* b) {
868 return *a < *b;
869 });
870
871 final_all_items_uid.resize(global_items_uid_around.size());
872
873 Int64 index = 0;
874 Int64 previous_uid = -1;
875
876 for (Int64* ptr_uid : global_items_uid) {
877 if (*ptr_uid == previous_uid) {
878 ARCANE_FATAL("Le uid {0} est dupliqué", *ptr_uid);
879 }
880 previous_uid = *ptr_uid;
881 for (Integer iaround = 0; iaround < size_of_once_case_around; ++iaround) {
882 final_all_items_uid[index++] = ptr_uid[iaround];
883 }
884 }
885 }
886
887 // info() << "final_all_items_uid : " << final_all_items_uid;
888
889 UniqueArray<Byte> hash_result;
890 MD5HashAlgorithm hash_algo;
891 hash_algo.computeHash64(asBytes(final_all_items_uid.constSpan()), hash_result);
892 return Convert::toHexaString(hash_result);
893}
894
905_cellsUidAroundCells(UniqueArray<Int64>& own_cells_uid_around_cells)
906{
907 IParallelMng* pm = parallelMng();
908 IMesh* mesh = m_cartesian_mesh->mesh();
909
910 if (pm->commSize() != 1 && mesh->ghostLayerMng()->nbGhostLayer() == 0) {
911 ARCANE_FATAL("Pas compatible sans ghost");
912 }
913
914 Integer nb_patch = m_cartesian_mesh->nbPatch();
915 Integer nb_dir = mesh->dimension();
916 Integer nb_items = mesh->cellFamily()->allItems().own().size();
917
918 // On a que pred et succ.
919 constexpr Integer nb_items_per_dir = 2;
920 constexpr Integer ipred = 0;
921 constexpr Integer isucc = 1;
922
923 // +1 car on a le uid dedans.
924 Integer size_of_once_case_around = nb_dir * nb_items_per_dir + 1;
925
926 own_cells_uid_around_cells.resize(nb_items * size_of_once_case_around, -1);
927
928 Integer index = 0;
929 ENUMERATE_ (Cell, icell, mesh->cellFamily()->allItems().own()) {
930 own_cells_uid_around_cells[index] = icell->uniqueId();
931 index += size_of_once_case_around;
932 }
933
934 auto set_value = [&](Integer dir, Int64 uid, Int64 uid_pred, Int64 uid_succ) -> void {
935 // debug() << " -- dir : " << dir
936 // << " -- uid : " << uid
937 // << " -- uid_pred : " << uid_pred
938 // << " -- uid_succ : " << uid_succ;
939
940 ARCANE_ASSERT((uid != -1), ("Uid ne peut pas être égal à -1"));
941
942 for (Integer i = 0; i < own_cells_uid_around_cells.size(); i += size_of_once_case_around) {
943 if (own_cells_uid_around_cells[i] == uid) {
944 Integer pos_final = i + 1 + (dir * nb_items_per_dir);
945 Integer pos_pred = pos_final + ipred;
946 Integer pos_succ = pos_final + isucc;
947 if (own_cells_uid_around_cells[pos_pred] != -1 && own_cells_uid_around_cells[pos_pred] != uid_pred) {
948 ARCANE_FATAL("Problème de cohérence entre les patchs (uid={0} -- old_uid_pred={1} -- new_uid_pred={2})", uid, own_cells_uid_around_cells[pos_pred], uid_pred);
949 }
950 if (own_cells_uid_around_cells[pos_succ] != -1 && own_cells_uid_around_cells[pos_succ] != uid_succ) {
951 ARCANE_FATAL("Problème de cohérence entre les patchs (uid={0} -- old_uid_succ={1} -- new_uid_succ={2})", uid, own_cells_uid_around_cells[pos_succ], uid_succ);
952 }
953 own_cells_uid_around_cells[pos_pred] = uid_pred;
954 own_cells_uid_around_cells[pos_succ] = uid_succ;
955 return;
956 }
957 }
958 };
959
960 for (Integer idir = 0; idir < nb_dir; ++idir) {
961 CellDirectionMng cdm(m_cartesian_mesh->cellDirection(idir));
962 ENUMERATE_ (Cell, icell, cdm.allCells().own()) {
963 DirCell cc(cdm.cell(*icell));
964 Cell next = cc.next();
965 Cell prev = cc.previous();
966 set_value(idir, icell->uniqueId(), prev.uniqueId(), next.uniqueId());
967 }
968 }
969
970 for (Integer ipatch = 0; ipatch < nb_patch; ++ipatch) {
971 ICartesianMeshPatch* p = m_cartesian_mesh->patch(ipatch);
972 for (Integer idir = 0; idir < nb_dir; ++idir) {
973 CellDirectionMng cdm(p->cellDirection(idir));
974 ENUMERATE_ (Cell, icell, cdm.allCells().own()) {
975 DirCell cc(cdm.cell(*icell));
976 Cell next = cc.next();
977 Cell prev = cc.previous();
978 set_value(idir, icell->uniqueId(), prev.uniqueId(), next.uniqueId());
979 }
980 }
981 }
982
983 return size_of_once_case_around-1;
984}
985
996_cellsUidAroundFaces(UniqueArray<Int64>& own_cells_uid_around_faces)
997{
998 IParallelMng* pm = parallelMng();
999 IMesh* mesh = m_cartesian_mesh->mesh();
1000
1001 if (pm->commSize() != 1 && mesh->ghostLayerMng()->nbGhostLayer() == 0) {
1002 ARCANE_FATAL("Pas compatible sans ghost");
1003 }
1004
1005 Integer nb_patch = m_cartesian_mesh->nbPatch();
1006 Integer nb_dir = mesh->dimension();
1007 Integer nb_items = mesh->faceFamily()->allItems().own().size();
1008
1009 // On a que pred et succ.
1010 constexpr Integer nb_items_per_dir = 2;
1011 constexpr Integer ipred = 0;
1012 constexpr Integer isucc = 1;
1013
1014 // +1 car on a le uid dedans.
1015 Integer size_of_once_case_around = nb_items_per_dir * nb_patch + 1;
1016
1017 own_cells_uid_around_faces.resize(nb_items * size_of_once_case_around, -1);
1018
1019 Integer index = 0;
1020 ENUMERATE_ (Item, iitem, mesh->faceFamily()->allItems().own()) {
1021 own_cells_uid_around_faces[index] = iitem->uniqueId();
1022 index += size_of_once_case_around;
1023 }
1024
1025 auto set_value = [&](Integer ipatch, Int64 uid, Int64 uid_pred, Int64 uid_succ) -> void {
1026 // debug() << " -- uid : " << uid
1027 // << " -- uid_pred : " << uid_pred
1028 // << " -- uid_succ : " << uid_succ;
1029
1030 ARCANE_ASSERT((uid != -1), ("Uid ne peut pas être égal à -1"));
1031
1032 for (Integer i = 0; i < own_cells_uid_around_faces.size(); i += size_of_once_case_around) {
1033 if (own_cells_uid_around_faces[i] == uid) {
1034 Integer pos_final = i + 1 + ipatch * nb_items_per_dir;
1035 Integer pos_pred = pos_final + ipred;
1036 Integer pos_succ = pos_final + isucc;
1037 if (own_cells_uid_around_faces[pos_pred] != -1 && own_cells_uid_around_faces[pos_pred] != uid_pred) {
1038 ARCANE_FATAL("Problème de cohérence entre les patchs (uid={0} -- old_uid_pred={1} -- new_uid_pred={2})", uid, own_cells_uid_around_faces[pos_pred], uid_pred);
1039 }
1040 if (own_cells_uid_around_faces[pos_succ] != -1 && own_cells_uid_around_faces[pos_succ] != uid_succ) {
1041 ARCANE_FATAL("Problème de cohérence entre les patchs (uid={0} -- old_uid_succ={1} -- new_uid_succ={2})", uid, own_cells_uid_around_faces[pos_succ], uid_succ);
1042 }
1043 own_cells_uid_around_faces[pos_pred] = uid_pred;
1044 own_cells_uid_around_faces[pos_succ] = uid_succ;
1045 return;
1046 }
1047 }
1048 };
1049
1050 // // TODO : Il faut pouvoir récupérer le patch correspondant.
1051 // for (Integer idir = 0; idir < nb_dir; ++idir) {
1052 // FaceDirectionMng fdm(m_cartesian_mesh->faceDirection(idir));
1053 // ENUMERATE_ (Face, iface, fdm.allFaces().own()) {
1054 // DirFace cc(fdm.face(*iface));
1055 // Cell next = cc.nextCell();
1056 // Cell prev = cc.previousCell();
1057 // set_value(iface->uniqueId(), prev.uniqueId(), next.uniqueId());
1058 // }
1059 // }
1060
1061 for (Integer ipatch = 0; ipatch < nb_patch; ++ipatch) {
1062 ICartesianMeshPatch* p = m_cartesian_mesh->patch(ipatch);
1063 for (Integer idir = 0; idir < nb_dir; ++idir) {
1064 FaceDirectionMng fdm(p->faceDirection(idir));
1065 ENUMERATE_ (Face, iface, fdm.allFaces().own()) {
1066 DirFace cc(fdm.face(*iface));
1067 Cell next = cc.nextCell();
1068 Cell prev = cc.previousCell();
1069 set_value(ipatch, iface->uniqueId(), prev.uniqueId(), next.uniqueId());
1070 }
1071 }
1072 }
1073
1074 return size_of_once_case_around-1;
1075}
1076
1092_nodesUidAroundNodes(UniqueArray<Int64>& own_nodes_uid_around_nodes)
1093{
1094 IParallelMng* pm = parallelMng();
1095 IMesh* mesh = m_cartesian_mesh->mesh();
1096
1097 if (pm->commSize() != 1 && mesh->ghostLayerMng()->nbGhostLayer() == 0) {
1098 ARCANE_FATAL("Pas compatible sans ghost");
1099 }
1100
1101 Integer nb_patch = m_cartesian_mesh->nbPatch();
1102 Integer nb_dir = mesh->dimension();
1103 Integer nb_items = mesh->nodeFamily()->allItems().own().size();
1104
1105 // On a que pred et succ.
1106 constexpr Integer nb_items_per_dir = 2;
1107 constexpr Integer ipred = 0;
1108 constexpr Integer isucc = 1;
1109
1110 // +1 car on a le uid dedans.
1111 Integer size_of_once_case_around = nb_dir * nb_items_per_dir * nb_patch + 1;
1112
1113 own_nodes_uid_around_nodes.resize(nb_items * size_of_once_case_around, -1);
1114
1115 Integer index = 0;
1116 ENUMERATE_ (Item, iitem, mesh->nodeFamily()->allItems().own()) {
1117 own_nodes_uid_around_nodes[index] = iitem->uniqueId();
1118 index += size_of_once_case_around;
1119 }
1120
1121 auto set_value = [&](Integer dir, Integer ipatch, Int64 uid, Int64 uid_pred, Int64 uid_succ) {
1122 // debug() << " -- dir : " << dir
1123 // << " -- ipatch : " << ipatch
1124 // << " -- uid : " << uid
1125 // << " -- uid_pred : " << uid_pred
1126 // << " -- uid_succ : " << uid_succ;
1127
1128 ARCANE_ASSERT((uid != -1), ("Uid ne peut pas être égal à -1"));
1129
1130 for (Integer i = 0; i < own_nodes_uid_around_nodes.size(); i += size_of_once_case_around) {
1131 if (own_nodes_uid_around_nodes[i] == uid) {
1132 Integer pos_final = i + 1 + ipatch * (nb_items_per_dir * nb_dir) + dir * nb_items_per_dir;
1133 Integer pos_pred = pos_final + ipred;
1134 Integer pos_succ = pos_final + isucc;
1135 if (own_nodes_uid_around_nodes[pos_pred] != -1 && own_nodes_uid_around_nodes[pos_pred] != uid_pred) {
1136 ARCANE_FATAL("Problème de cohérence entre les patchs (uid={0} -- old_uid_pred={1} -- new_uid_pred={2})", uid, own_nodes_uid_around_nodes[pos_pred], uid_pred);
1137 }
1138 if (own_nodes_uid_around_nodes[pos_succ] != -1 && own_nodes_uid_around_nodes[pos_succ] != uid_succ) {
1139 ARCANE_FATAL("Problème de cohérence entre les patchs (uid={0} -- old_uid_succ={1} -- new_uid_succ={2})", uid, own_nodes_uid_around_nodes[pos_succ], uid_succ);
1140 }
1141 own_nodes_uid_around_nodes[pos_pred] = uid_pred;
1142 own_nodes_uid_around_nodes[pos_succ] = uid_succ;
1143 return;
1144 }
1145 }
1146 };
1147
1148 // // TODO : Il faut pouvoir récupérer le patch correspondant.
1149 // for (Integer idir = 0; idir < nb_dir; ++idir) {
1150 // NodeDirectionMng ndm(m_cartesian_mesh->nodeDirection(idir));
1151 // ENUMERATE_ (Node, inode, ndm.allNodes().own()) {
1152 // DirNode cc(ndm.node(*inode));
1153 // Node next = cc.next();
1154 // Node prev = cc.previous();
1155 // set_value(idir, 0, inode->uniqueId(), prev.uniqueId(), next.uniqueId());
1156 // }
1157 // }
1158
1159 for (Integer ipatch = 0; ipatch < nb_patch; ++ipatch) {
1160 ICartesianMeshPatch* p = m_cartesian_mesh->patch(ipatch);
1161 for (Integer idir = 0; idir < nb_dir; ++idir) {
1162 NodeDirectionMng ndm(p->nodeDirection(idir));
1163 ENUMERATE_ (Node, inode, ndm.allNodes().own()) {
1164 DirNode cc(ndm.node(*inode));
1165 Node next = cc.next();
1166 Node prev = cc.previous();
1167 set_value(idir, ipatch, inode->uniqueId(), prev.uniqueId(), next.uniqueId());
1168 }
1169 }
1170 }
1171
1172 return size_of_once_case_around-1;
1173}
1174
1175/*---------------------------------------------------------------------------*/
1176/*---------------------------------------------------------------------------*/
1177
1178void AMRCartesianMeshTesterModule::
1179_checkSync()
1180{
1181 IMesh* mesh = m_cartesian_mesh->mesh();
1182 Integer nb_error = 0;
1183
1184 VariableCellInt32 test_var(VariableBuildInfo(mesh, "ArcaneTestAMRCheckSync"));
1185 test_var.fill(0);
1186 ENUMERATE_ (Cell, icell, mesh->ownCells()) {
1187 test_var[icell] = 1;
1188 }
1189 test_var.synchronize();
1190 ENUMERATE_ (Cell, icell, mesh->allCells()) {
1191 if (test_var[icell] != 1) {
1192 nb_error++;
1193 }
1194 }
1195 if (nb_error > 0) {
1196 ARCANE_FATAL("Bad sync -- Nb error : {0}", nb_error);
1197 }
1198}
1199
1200/*---------------------------------------------------------------------------*/
1201/*---------------------------------------------------------------------------*/
1202
1203void AMRCartesianMeshTesterModule::
1204_cellsInPatch(Real3 position, Real3 length, bool is_3d, Int32 level, UniqueArray<Int32>& cells_in_patch)
1205{
1206 // Parcours les mailles actives et ajoute dans la liste des mailles
1207 // à raffiner celles qui sont contenues dans le boîte englobante
1208 // spécifiée dans le jeu de données.
1209 Real3 min_pos = position;
1210 Real3 max_pos = min_pos + length;
1211 ENUMERATE_ (Cell, icell, mesh()->allCells()) {
1212 if ((icell->level() == level) || (level == -1 && icell->nbHChildren() == 0)) {
1213 Real3 center = m_cell_center[icell];
1214 bool is_inside_x = center.x > min_pos.x && center.x < max_pos.x;
1215 bool is_inside_y = center.y > min_pos.y && center.y < max_pos.y;
1216 bool is_inside_z = (center.z > min_pos.z && center.z < max_pos.z) || !is_3d;
1217 if (is_inside_x && is_inside_y && is_inside_z) {
1218 cells_in_patch.add(icell.itemLocalId());
1219 }
1220 }
1221 }
1222}
1223
1224/*---------------------------------------------------------------------------*/
1225/*---------------------------------------------------------------------------*/
1226
1227ARCANE_REGISTER_MODULE_AMRCARTESIANMESHTESTER(AMRCartesianMeshTesterModule);
1228
1229/*---------------------------------------------------------------------------*/
1230/*---------------------------------------------------------------------------*/
1231
1232} // End namespace ArcaneTest
1233
1234/*---------------------------------------------------------------------------*/
1235/*---------------------------------------------------------------------------*/
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
#define ENUMERATE_FACE(name, group)
Enumérateur générique d'un groupe de faces.
#define ENUMERATE_(type, name, group)
Enumérateur générique d'un groupe d'entité
#define ENUMERATE_CELL(name, group)
Enumérateur générique d'un groupe de mailles.
#define ENUMERATE_NODE(name, group)
Enumérateur générique d'un groupe de noeuds.
Fonctions utilitaires sur le maillage.
Ce fichier contient les différentes fabriques de services et macro pour enregistrer les services.
Module de test pour les infos sur les maillages cartésiens.
Integer _cellsUidAroundFaces(UniqueArray< Int64 > &own_cells_uid_around_faces)
Méthode permettant de récupérer un tableau contenant les mailles autour des faces.
Integer _cellsUidAroundCells(UniqueArray< Int64 > &own_cells_uid_around_cells)
Méthode permettant de récupérer un tableau contenant les mailles autour des mailles.
Integer _nodesUidAroundNodes(UniqueArray< Int64 > &own_nodes_uid_around_nodes)
Méthode permettant de récupérer un tableau contenant les noeuds autour des noeuds.
void _computeSubCellDensity(Cell cell)
Calcule la densité d'une maille AMR.
String _checkDirectionUniqueIdsHashCollective(ArrayView< Int64 > own_items_uid_around, Integer nb_items_around)
Méthode permettant de calculer un hash à partir d'un tableau d'items "autour".
Integer size() const
Nombre d'éléments du vecteur.
Vue modifiable d'un tableau d'un type T.
iterator end()
Itérateur sur le premier élément après la fin du tableau.
void resize(Int64 s)
Change le nombre d'éléments du tableau à s.
iterator begin()
Itérateur sur le premier élément du tableau.
Span< const T > constSpan() const
Vue constante sur ce tableau.
void add(ConstReferenceType val)
Ajoute l'élément val à la fin du tableau.
void setSortAfterRenumbering(bool v)
Indique si on retrie les entités après renumérotation.
void setParentPatch(CartesianPatch patch)
Numéro du patch parent pour la renumérotation.
void setRenumberPatchMethod(Int32 v)
Méthode pour renuméroter les patchs.
Infos sur les mailles d'une direction spécifique X,Y ou Z d'un maillage structuré.
CellGroup allCells() const
Groupe de toutes les mailles dans la direction.
DirCell cell(Cell c) const
Maille direction correspondant à la maille c.
Maille d'un maillage.
Definition Item.h:1191
Int32 nbHChildren() const
Nombre d'enfants pour l'AMR.
Definition Item.h:1307
Cell hChild(Int32 i) const
i-ème enfant AMR
Definition Item.h:1310
void clear()
Supprime tous les éléments de la collection.
Definition Collection.h:68
Maille avant et après une maille suivant une direction.
Infos sur maille avant et après une face suivant une direction.
Noeud avant et après un noeud suivant une direction.
virtual String file(const String &file_name) const
Retourne le chemin complet du fichier file_name dans le répertoire.
Definition Directory.cc:138
virtual bool createDirectory() const
Créé le répertoire.
Definition Directory.cc:120
virtual String path() const
Retourne le chemin du répertoire.
Definition Directory.cc:129
Infos sur les face d'une direction spécifique X,Y ou Z d'un maillage structuré.
DirFace face(Face f) const
Face direction correspondant à la face f.
FaceGroup allFaces() const
Groupe de toutes les faces dans la direction.
Face d'une maille.
Definition Item.h:944
Interface d'un patch AMR d'un maillage cartésien.
virtual CellGroup cells()=0
Groupe de mailles du patch.
virtual NodeDirectionMng & nodeDirection(eMeshDirection dir)=0
Liste des noeuds dans la direction dir.
virtual FaceDirectionMng & faceDirection(eMeshDirection dir)=0
Liste des faces dans la direction dir.
virtual CellDirectionMng & cellDirection(eMeshDirection dir)=0
Liste des mailles dans la direction dir.
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 IMesh * mesh() const =0
Maillage associé à ce maillage cartésien.
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 String name() const =0
Nom de la famille.
virtual String fullName() const =0
Nom complet de la famille (avec celui du maillage)
virtual IItemFamily * nodeFamily()=0
Retourne la famille des noeuds.
virtual IItemFamily * faceFamily()=0
Retourne la famille des faces.
virtual NodeGroup allNodes()=0
Groupe de tous les noeuds.
virtual IItemFamily * cellFamily()=0
Retourne la famille des mailles.
virtual VariableNodeReal3 & nodesCoordinates()=0
Coordonnées des noeuds.
Interface du gestionnaire de parallélisme pour un sous-domaine.
virtual Int32 commRank() const =0
Rang de cette instance dans le communicateur.
virtual void allGatherVariable(ConstArrayView< char > send_buf, Array< char > &recv_buf)=0
Effectue un regroupement sur tous les processeurs.
virtual Int32 commSize() const =0
Nombre d'instance dans le communicateur.
virtual bool isParallel() const =0
Retourne true si l'exécution est parallèle.
virtual char reduce(eReduceType rt, char v)=0
Effectue la réduction de type rt sur le réel v et retourne la valeur.
virtual void setVariables(VariableCollection variables)=0
Positionne la liste des variables à sortir.
virtual void setTimes(RealConstArrayView times)=0
Positionne la liste des temps.
virtual void setGroups(ItemGroupCollection groups)=0
Positionne la liste des groupes à sortir.
virtual void setBaseDirectoryName(const String &dirname)=0
Positionne le nom du répertoire de sortie des fichiers. Ce répertoire doit exister.
virtual void setMesh(IMesh *mesh)
Positionne le maillage.
Interface du gestionnaire d'un sous-domaine.
Definition ISubDomain.h:74
virtual ITimeLoopMng * timeLoopMng()=0
Retourne le gestionnaire de la boucle en temps.
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
Definition ITimeLoop.h:50
static const char * WComputeLoop
appelé pendant la boucle de calcul
Definition ITimeLoop.h:48
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
Definition ITimeLoop.h:52
virtual void writePostProcessing(IPostProcessorWriter *writer)=0
Ecrit les variables pour un post-traitement.
Integer size() const
Nombre d'éléments du groupe.
Definition ItemGroup.h:88
Node node(Int32 i) const
i-ème noeud de l'entité
Definition Item.h:779
NodeConnectedListViewType nodes() const
Liste des noeuds de l'entité
Definition Item.h:782
Int32 nbNode() const
Nombre de noeuds de l'entité
Definition Item.h:776
Classe de base d'un élément de maillage.
Definition Item.h:83
constexpr bool null() const
true si l'entité est nul (i.e. non connecté au maillage)
Definition Item.h:216
Calcule la fonction de hashage MD5 d'un tableau.
void computeHash64(Span< const Byte > input, ByteArray &output) override
Calcule la valeur du hash pour le tableau input.
Informations pour construire un module.
Infos sur les noeuds d'une direction spécifique X,Y ou Z d'un maillage structuré.
NodeGroup allNodes() const
Groupe de tous les noeuds dans la direction.
DirNode node(Node n) const
Noeud direction correspondant au noeud n.
Noeud d'un maillage.
Definition Item.h:573
Cell cell(Int32 i) const
i-ème maille du noeud
Definition Item.h:1584
Int32 nbCell() const
Nombre de mailles connectées au noeud.
Definition Item.h:653
Classe gérant un vecteur de réel de dimension 3.
Definition Real3.h:132
Référence à une instance.
Chaîne de caractères unicode.
const char * localstr() const
Retourne la conversion de l'instance dans l'encodage UTF-8.
Definition String.cc:227
Vecteur 1D de données avec sémantique par valeur (style STL).
Paramètres nécessaires à la construction d'une variable.
ItemGroupT< Cell > CellGroup
Groupe de mailles.
Definition ItemTypes.h:183
ItemGroupT< Node > NodeGroup
Groupe de noeuds.
Definition ItemTypes.h:167
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.
MeshVariableScalarRefT< Cell, Int32 > VariableCellInt32
Grandeur au centre des mailles de type entier 32 bits.
MeshVariableScalarRefT< Face, Real3 > VariableFaceReal3
Grandeur aux faces de type coordonnées.
ARCANE_CARTESIANMESH_EXPORT Ref< CartesianMeshCoarsening2 > createCartesianMeshCoarsening2(ICartesianMesh *cm)
Créé une instance pour gérer le déraffinement du maillage (V2).
ARCANE_CARTESIANMESH_EXPORT Ref< ICartesianMeshAMRPatchMng > cartesianMeshAMRPatchMng(ICartesianMesh *cm)
Créé une instance pour gérer le déraffinement du maillage (V3?).
String toHexaString(ByteConstArrayView input)
Converti un tableau d'octet en sa représentation hexadécimale.
Definition Convert.cc:75
@ ReduceSum
Somme des valeurs.
std::int64_t Int64
Type entier signé sur 64 bits.
Int32 Integer
Type représentant un entier.
List< ItemGroup > ItemGroupList
Tableau de groupes d'éléments du maillage.
UniqueArray< Int32 > Int32UniqueArray
Tableau dynamique à une dimension d'entiers 32 bits.
Definition UtilsTypes.h:428
List< String > StringList
Tableau de chaînes de caractères unicode.
Definition UtilsTypes.h:596
impl::SpanTypeFromSize< conststd::byte, SizeType >::SpanType asBytes(const SpanImpl< DataType, SizeType, Extent > &s)
Converti la vue en un tableau d'octets non modifiables.
Definition Span.h:884
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
Definition Real3.h:36
Real z
troisième composante du triplet
Definition Real3.h:37
Real x
première composante du triplet
Definition Real3.h:35