Arcane  v3.16.0.0
Documentation développeur
Chargement...
Recherche...
Aucune correspondance
MshParallelMeshReader.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/* MshParallelMeshReader.cc (C) 2000-2025 */
9/* */
10/* Lecture parallèle d'un fichier au format MSH. */
11/*---------------------------------------------------------------------------*/
12/*---------------------------------------------------------------------------*/
13
14#include "arcane/utils/TraceAccessor.h"
15#include "arcane/utils/Array.h"
16#include "arcane/utils/IOException.h"
17#include "arcane/utils/FatalErrorException.h"
18#include "arcane/utils/Ref.h"
19#include "arcane/utils/NotSupportedException.h"
20#include "arcane/utils/Real3.h"
21#include "arcane/utils/OStringStream.h"
22#include "arcane/utils/FixedArray.h"
23#include "arcane/utils/SmallArray.h"
24#include "arcane/utils/PlatformUtils.h"
25#include "arcane/utils/CheckedConvert.h"
26
27#include "arcane/core/IMeshReader.h"
28#include "arcane/core/IPrimaryMesh.h"
29#include "arcane/core/IItemFamily.h"
30#include "arcane/core/ItemGroup.h"
31#include "arcane/core/VariableTypes.h"
32#include "arcane/core/IParallelMng.h"
34#include "arcane/core/IMeshModifier.h"
35#include "arcane/core/MeshKind.h"
36#include "arcane/core/internal/MshMeshGenerationInfo.h"
37
38// Element types in .msh file format, found in gmsh-2.0.4/Common/GmshDefines.h
39#include "arcane/std/internal/IosFile.h"
40#include "arcane/std/internal/IosGmsh.h"
41
42/*---------------------------------------------------------------------------*/
43/*---------------------------------------------------------------------------*/
44
45/*
46 * NOTES:
47 * - La bibliothèque `gmsh` fournit un script 'open.py' dans le répertoire
48 * 'demos/api' qui permet de générer un fichier '.msh' à partir d'un '.geo'.
49 * - Il est aussi possible d'utiliser directement l'exécutable 'gmsh' avec
50 * l'option '-save-all' pour sauver un fichier '.msh' à partir d'un '.geo'
51 *
52 * TODO:
53 * - supporter les partitions
54 * - pouvoir utiliser la bibliothèque 'gmsh' directement.
55 */
56
57/*---------------------------------------------------------------------------*/
58/*---------------------------------------------------------------------------*/
59
60namespace Arcane
61{
62
63/*---------------------------------------------------------------------------*/
64/*---------------------------------------------------------------------------*/
79class MshParallelMeshReader
80: public TraceAccessor
81, public IMshMeshReader
82{
83 public:
84
86 using MshPhysicalNameList = impl::MshMeshGenerationInfo::MshPhysicalNameList;
87 using MshEntitiesNodes = impl::MshMeshGenerationInfo::MshEntitiesNodes;
88 using MshEntitiesWithNodes = impl::MshMeshGenerationInfo::MshEntitiesWithNodes;
89 using MshPeriodicOneInfo = impl::MshMeshGenerationInfo::MshPeriodicOneInfo;
91
92 public:
93
94 using eReturnType = typename IMeshReader::eReturnType;
95
117
122 {
123 // TODO: Allouer la connectivité par bloc pour éviter de trop grosses allocations
124
125 public:
126
127 void addItem(Int16 type_id, Int64 unique_id, SmallSpan<const Int64> nodes_uid)
128 {
129 ++nb_item;
130 items_infos.add(type_id);
131 items_infos.add(unique_id);
132 items_infos.addRange(nodes_uid);
133 }
134
135 public:
136
137 Int32 nb_item = 0;
138 UniqueArray<Int64> items_infos;
139 };
140
143 {
144 public:
145
146 MshItemKindInfo faces_infos;
147 MshItemKindInfo cells_infos;
148
154 std::unordered_map<Int64, Int32> nodes_rank_map;
156 };
157
158 public:
159
160 explicit MshParallelMeshReader(ITraceMng* tm)
161 : TraceAccessor(tm)
162 {}
163
164 eReturnType readMeshFromMshFile(IMesh* mesh, const String& filename, bool use_internal_partition) override;
165
166 private:
167
168 IMesh* m_mesh = nullptr;
169 IParallelMng* m_parallel_mng = nullptr;
170 Int32 m_master_io_rank = A_NULL_RANK;
171 bool m_is_parallel = false;
172 Ref<IosFile> m_ios_file; // nullptr sauf pour le rang maitre.
173 impl::MshMeshGenerationInfo* m_mesh_info = nullptr;
174 MshMeshAllocateInfo m_mesh_allocate_info;
180 bool m_is_binary = false;
181
182 private:
183
184 void _readNodesFromFile();
185 void _readNodesOneEntity(Int32 entity_index);
187 void _readMeshFromFile();
189 void _addFaces();
190 void _allocateCells();
191 void _allocateGroups();
192 void _addFaceGroup(MshElementsBlock& block, const String& group_name);
193 void _addFaceGroupOnePart(ConstArrayView<Int64> connectivities, Int32 item_nb_node,
194 const String& group_name, Int32 block_index);
195 void _addCellOrNodeGroup(MshElementsBlock& block, const String& group_name, IItemFamily* family);
196 void _addCellOrNodeGroupOnePart(ConstArrayView<Int64> uids, const String& group_name,
197 Int32 block_index, IItemFamily* family);
198 Int16 _switchMshType(Int64, Int32&) const;
199 void _readPhysicalNames();
200 void _readEntities();
201 void _readPeriodic();
202 void _readOneEntity(Int32 entity_dim, Int32 entity_index_in_dim);
205 Int32 _getIntegerAndBroadcast();
206 Int64 _getInt64AndBroadcast();
207 void _getInt64ArrayAndBroadcast(ArrayView<Int64> values);
208 void _getInt32ArrayAndBroadcast(ArrayView<Int32> values);
209 void _getDoubleArrayAndBroadcast(ArrayView<double> values);
212 void _computeOwnItems(MshElementsBlock& block, MshItemKindInfo& item_kind_info, bool is_generate_uid);
213 Real3 _getReal3();
214 Int32 _getInt32();
215 Int64 _getInt64();
216 void _goToNextLine();
217 void _readAndCheck(const String& expected_value);
218};
219
220/*---------------------------------------------------------------------------*/
221/*---------------------------------------------------------------------------*/
222
223namespace
224{
231 inline std::pair<Int64, Int64>
232 _interval(Int32 index, Int32 nb_interval, Int64 size)
233 {
234 Int64 isize = size / nb_interval;
235 Int64 ibegin = index * isize;
236 // Pour le dernier interval, prend les elements restants
237 if ((index + 1) == nb_interval)
238 isize = size - ibegin;
239 return { ibegin, isize };
240 }
241} // namespace
242
243/*---------------------------------------------------------------------------*/
244/*---------------------------------------------------------------------------*/
250{
251 IosFile* f = m_ios_file.get();
252 String s;
253 if (f) {
254 s = f->getNextLine();
255 }
256 if (m_is_parallel) {
257 if (f)
258 info() << "BroadcastNextLine: " << s;
259 m_parallel_mng->broadcastString(s, m_master_io_rank);
260 }
261 info() << "GetNextLine: " << s;
262 return s;
263}
264
265/*---------------------------------------------------------------------------*/
266/*---------------------------------------------------------------------------*/
272{
273 IosFile* f = m_ios_file.get();
274 Int32 is_end_int = 0;
275 if (f) {
276 is_end_int = f->isEnd() ? 1 : 0;
277 info() << "IsEndOfFile_Master: " << is_end_int;
278 }
279 if (m_is_parallel) {
280 if (f)
281 info() << "IsEndOfFile: " << is_end_int;
282 m_parallel_mng->broadcast(ArrayView<Int32>(1, &is_end_int), m_master_io_rank);
283 }
284 bool is_end = (is_end_int != 0);
285 info() << "IsEnd: " << is_end;
286 return is_end;
287}
288
289/*---------------------------------------------------------------------------*/
290/*---------------------------------------------------------------------------*/
291
293//_getASCIIIntegerAndBroadcast()
294_getIntegerAndBroadcast()
295{
296 IosFile* f = m_ios_file.get();
298 if (f)
299 v[0] = f->getInteger();
300 if (m_is_parallel) {
301 m_parallel_mng->broadcast(v.view(), m_master_io_rank);
302 }
303 return v[0];
304}
305
306/*---------------------------------------------------------------------------*/
307/*---------------------------------------------------------------------------*/
308
309Int64 MshParallelMeshReader::
310_getInt64AndBroadcast()
311{
312 IosFile* f = m_ios_file.get();
313 FixedArray<Int64, 1> v;
314 if (f)
315 v[0] = _getInt64();
316 if (m_is_parallel) {
317 m_parallel_mng->broadcast(v.view(), m_master_io_rank);
318 }
319 return v[0];
320}
321
322/*---------------------------------------------------------------------------*/
323/*---------------------------------------------------------------------------*/
324
325void MshParallelMeshReader::
326_getInt64ArrayAndBroadcast(ArrayView<Int64> values)
327{
328 IosFile* f = m_ios_file.get();
329 if (f) {
330 if (m_is_binary) {
331 f->binaryRead(values);
332 }
333 else {
334 for (Int64& v : values)
335 v = f->getInt64();
336 }
337 }
338 if (m_is_parallel)
339 m_parallel_mng->broadcast(values, m_master_io_rank);
340}
341
342/*---------------------------------------------------------------------------*/
343/*---------------------------------------------------------------------------*/
344
345void MshParallelMeshReader::
346_getInt32ArrayAndBroadcast(ArrayView<Int32> values)
347{
348 IosFile* f = m_ios_file.get();
349 if (f) {
350 if (m_is_binary) {
351 f->binaryRead(values);
352 }
353 else {
354 for (Int32& v : values)
355 v = f->getInteger();
356 }
357 }
358 if (m_is_parallel)
359 m_parallel_mng->broadcast(values, m_master_io_rank);
360}
361
362/*---------------------------------------------------------------------------*/
363/*---------------------------------------------------------------------------*/
364
365void MshParallelMeshReader::
366_getDoubleArrayAndBroadcast(ArrayView<double> values)
367{
368 IosFile* f = m_ios_file.get();
369 if (f) {
370 if (m_is_binary) {
371 f->binaryRead(values);
372 }
373 else {
374 for (double& v : values)
375 v = f->getReal();
376 }
377 }
378 if (m_is_parallel)
379 m_parallel_mng->broadcast(values, m_master_io_rank);
380}
381
382/*---------------------------------------------------------------------------*/
383/*---------------------------------------------------------------------------*/
384
385Real3 MshParallelMeshReader::
386_getReal3()
387{
388 IosFile* f = m_ios_file.get();
390 Real3 v;
391 if (m_is_binary) {
392 f->binaryRead(SmallSpan<Real3>(&v, 1));
393 }
394 else {
395 Real x = f->getReal();
396 Real y = f->getReal();
397 Real z = f->getReal();
398 v = Real3(x, y, z);
399 }
400 return v;
401}
402
403/*---------------------------------------------------------------------------*/
404/*---------------------------------------------------------------------------*/
405
406Int32 MshParallelMeshReader::
407_getInt32()
408{
409 IosFile* f = m_ios_file.get();
411 Int32 v = 0;
412 if (m_is_binary)
413 f->binaryRead(SmallSpan<Int32>(&v, 1));
414 else
415 v = f->getInteger();
416 return v;
417}
418
419/*---------------------------------------------------------------------------*/
420/*---------------------------------------------------------------------------*/
421
422Int64 MshParallelMeshReader::
423_getInt64()
424{
425 IosFile* f = m_ios_file.get();
427 Int64 v = 0;
428 if (m_is_binary)
429 f->binaryRead(SmallSpan<Int64>(&v, 1));
430 else
431 v = f->getInt64();
432 return v;
433}
434
435/*---------------------------------------------------------------------------*/
436/*---------------------------------------------------------------------------*/
437
438void MshParallelMeshReader::
439_goToNextLine()
440{
441 if (m_ios_file.get())
442 m_ios_file->getNextLine();
443}
444
445/*---------------------------------------------------------------------------*/
446/*---------------------------------------------------------------------------*/
447
448Int16 MshParallelMeshReader::
449_switchMshType(Int64 mshElemType, Int32& nNodes) const
450{
451 switch (mshElemType) {
452 case IT_NullType: // used to decode IT_NullType: IT_HemiHexa7|IT_Line9
453 switch (nNodes) {
454 case 7:
455 return IT_HemiHexa7;
456 default:
457 info() << "Could not decode IT_NullType with nNodes=" << nNodes;
458 throw IOException("_convertToMshType", "Could not decode IT_NullType with nNodes");
459 }
460 break;
461 case MSH_PNT:
462 nNodes = 1;
463 return IT_Vertex;
464 case MSH_LIN_2:
465 nNodes = 2;
466 return IT_Line2;
467 case MSH_TRI_3:
468 nNodes = 3;
469 return IT_Triangle3;
470 case MSH_QUA_4:
471 nNodes = 4;
472 return IT_Quad4;
473 case MSH_TET_4:
474 nNodes = 4;
475 return IT_Tetraedron4;
476 case MSH_HEX_8:
477 nNodes = 8;
478 return IT_Hexaedron8;
479 case MSH_PRI_6:
480 nNodes = 6;
481 return IT_Pentaedron6;
482 case MSH_PYR_5:
483 nNodes = 5;
484 return IT_Pyramid5;
485 case MSH_TRI_10:
486 nNodes = 10;
487 return IT_Heptaedron10;
488 case MSH_TRI_12:
489 nNodes = 12;
490 return IT_Octaedron12;
491 case MSH_TRI_6:
492 nNodes = 6;
493 return IT_Triangle6;
494 case MSH_TET_10:
495 nNodes = 10;
496 return IT_Tetraedron10;
497 case MSH_QUA_9:
498 case MSH_HEX_27:
499 case MSH_PRI_18:
500 case MSH_PYR_14:
501 case MSH_QUA_8:
502 case MSH_HEX_20:
503 case MSH_PRI_15:
504 case MSH_PYR_13:
505 case MSH_TRI_9:
506 case MSH_TRI_15:
507 case MSH_TRI_15I:
508 case MSH_TRI_21:
509 default:
510 ARCANE_THROW(NotSupportedException, "Unknown GMSH element type '{0}'", mshElemType);
511 }
512 return IT_NullType;
513}
514
515/*---------------------------------------------------------------------------*/
516/*---------------------------------------------------------------------------*/
517
520{
521 // Détermine la bounding box de la maille
522 Real max_value = FloatInfo<Real>::maxValue();
523 Real min_value = -max_value;
524 Real3 min_box(max_value, max_value, max_value);
525 Real3 max_box(min_value, min_value, min_value);
526 const Int64 nb_node = m_mesh_allocate_info.nodes_coordinates.largeSize();
527 for (Real3 pos : m_mesh_allocate_info.nodes_coordinates) {
528 min_box = math::min(min_box, pos);
529 max_box = math::max(max_box, pos);
530 }
531
533 UniqueArray<Int32> nodes_part(nb_node, 0);
534
535 // Pour la partition, on ne prend en compte que la coordonnée X.
536 // On est sur qu'elle est valable pour toutes les dimensions du maillage.
537 // On partitionne avec des intervalles de même longueur.
538 // NOTE: Cela fonctionne bien si l'ensemble des noeuds est bien réparti.
539 // Si ce n'est pas le cas on pourrait utiliser une bisection en coupant
540 // à chaque fois sur la moyenne.
541 Real min_x = min_box.x;
542 Real max_x = max_box.x;
543 IParallelMng* pm = m_parallel_mng;
544 Real global_min_x = pm->reduce(Parallel::ReduceMin, min_x);
545 Real global_max_x = pm->reduce(Parallel::ReduceMax, max_x);
546 info() << "MIN_MAX_X=" << global_min_x << " " << global_max_x;
547
548 Real diff_v = (global_max_x - global_min_x) / static_cast<Real>(m_nb_part);
549 // Ne devrait pas arriver mais c'est nécessaire pour éviter d'éventuelles
550 // divisions par zéro.
551 if (!math::isNearlyEqual(global_min_x, global_max_x)) {
552 for (Int64 i = 0; i < nb_node; ++i) {
553 Int32 part = static_cast<Int32>((m_mesh_allocate_info.nodes_coordinates[i].x - global_min_x) / diff_v);
554 part = std::clamp(part, 0, m_nb_part - 1);
555 nodes_part[i] = part;
556 }
557 }
558 UniqueArray<Int32> nb_node_per_rank(m_nb_part, 0);
559 // Construit la table de hashage des rangs
560 for (Int64 i = 0; i < nb_node; ++i) {
561 Int32 rank = m_parts_rank[nodes_part[i]];
562 Int64 uid = m_mesh_allocate_info.nodes_unique_id[i];
563 ++nb_node_per_rank[rank];
564 m_mesh_allocate_info.nodes_rank_map.insert(std::make_pair(uid, rank));
565 }
566 pm->reduce(Parallel::ReduceSum, nb_node_per_rank);
567 info() << "NB_NODE_PER_RANK=" << nb_node_per_rank;
568}
569
570/*---------------------------------------------------------------------------*/
571/*---------------------------------------------------------------------------*/
598{
599 FixedArray<Int64, 4> nodes_info;
600 _getInt64ArrayAndBroadcast(nodes_info.view());
601
602 Int64 nb_entity = nodes_info[0];
603 Int64 total_nb_node = nodes_info[1];
604 Int64 min_node_tag = nodes_info[2];
605 Int64 max_node_tag = nodes_info[3];
606
607 if (!m_is_binary)
608 _goToNextLine();
609
610 if (total_nb_node < 0)
611 ARCANE_THROW(IOException, "Invalid number of nodes : '{0}'", total_nb_node);
612
613 info() << "[Nodes] nb_entity=" << nb_entity
614 << " total_nb_node=" << total_nb_node
615 << " min_tag=" << min_node_tag
616 << " max_tag=" << max_node_tag
617 << " read_nb_part=" << m_nb_part
618 << " nb_rank=" << m_parallel_mng->commSize();
619
620 for (Integer i_entity = 0; i_entity < nb_entity; ++i_entity) {
621 _readNodesOneEntity(i_entity);
622 }
623
625
626 if (m_is_binary)
627 _goToNextLine();
628}
629
630/*---------------------------------------------------------------------------*/
631/*---------------------------------------------------------------------------*/
632
633void MshParallelMeshReader::
634_readNodesOneEntity(Int32 entity_index)
635{
636 IosFile* ios_file = m_ios_file.get();
637 IParallelMng* pm = m_parallel_mng;
638 const Int32 my_rank = pm->commRank();
639
640 UniqueArray<Int64> nodes_uids;
641 UniqueArray<Real3> nodes_coordinates;
642
643 FixedArray<Int32, 3> entity_infos;
644 _getInt32ArrayAndBroadcast(entity_infos.view());
645 Int64 nb_node2 = _getInt64AndBroadcast();
646
647 if (!m_is_binary)
648 _goToNextLine();
649
650 // Dimension de l'entité (pas utile)
651 [[maybe_unused]] Int32 entity_dim = entity_infos[0];
652 // Tag de l'entité (pas utile)
653 [[maybe_unused]] Int32 entity_tag = entity_infos[1];
654 Int32 parametric_coordinates = entity_infos[2];
655 //Int64 nb_node2 = entity_infos[3];
656
657 info() << "[Nodes] index=" << entity_index << " entity_dim=" << entity_dim << " entity_tag=" << entity_tag
658 << " parametric=" << parametric_coordinates
659 << " nb_node2=" << nb_node2;
660
661 if (parametric_coordinates != 0)
662 ARCANE_THROW(NotSupportedException, "Only 'parametric coordinates' value of '0' is supported (current={0})", parametric_coordinates);
663
664 // Il est possible que le nombre de noeuds soit 0.
665 // Dans ce cas, il faut directement passer à la ligne suivante
666 if (nb_node2 == 0)
667 return;
668
669 // Partitionne la lecture en \a m_nb_part
670 // Pour chaque i_entity , on a d'abord la liste des identifiants puis la liste des coordonnées
671
672 for (Int32 i_part = 0; i_part < m_nb_part; ++i_part) {
673 Int64 nb_to_read = _interval(i_part, m_nb_part, nb_node2).second;
674 Int32 dest_rank = m_parts_rank[i_part];
675 info() << "Reading UIDS part i=" << i_part << " dest_rank=" << dest_rank << " nb_to_read=" << nb_to_read;
676 if (my_rank == dest_rank || my_rank == m_master_io_rank) {
677 nodes_uids.resize(nb_to_read);
678 }
679
680 // Le rang maitre lit les informations des noeuds pour la partie concernée
681 // et les transfère au rang destination
682 if (ios_file) {
683 if (m_is_binary) {
684 ios_file->binaryRead(nodes_uids.view());
685 }
686 else {
687 for (Integer i = 0; i < nb_to_read; ++i) {
688 // Conserve le uniqueId() du noeuds.
689 nodes_uids[i] = ios_file->getInt64();
690 //info() << "I=" << i << " ID=" << nodes_uids[i];
691 }
692 }
693 if (dest_rank != m_master_io_rank) {
694 pm->send(nodes_uids, dest_rank);
695 }
696 }
697 else if (my_rank == dest_rank) {
698 pm->recv(nodes_uids, m_master_io_rank);
699 }
700
701 // Conserve les informations de ma partie
702 if (my_rank == dest_rank) {
703 m_mesh_allocate_info.nodes_unique_id.addRange(nodes_uids);
704 }
705 }
706
707 // Lecture par partie des coordonnées
708 for (Int32 i_part = 0; i_part < m_nb_part; ++i_part) {
709 Int64 nb_to_read = _interval(i_part, m_nb_part, nb_node2).second;
710 Int32 dest_rank = m_parts_rank[i_part];
711 info() << "Reading COORDS part i=" << i_part << " dest_rank=" << dest_rank << " nb_to_read=" << nb_to_read;
712 if (my_rank == dest_rank || my_rank == m_master_io_rank) {
713 nodes_coordinates.resize(nb_to_read);
714 }
715
716 // Le rang maitre lit les informations des noeuds pour la partie concernée
717 // et les transfère au rang destination
718 if (ios_file) {
719 if (m_is_binary) {
720 ios_file->binaryRead(nodes_coordinates.view());
721 }
722 else {
723 for (Integer i = 0; i < nb_to_read; ++i) {
724 nodes_coordinates[i] = _getReal3();
725 //info() << "I=" << i << " ID=" << nodes_uids[i] << " COORD=" << Real3(nx, ny, nz);
726 }
727 }
728 if (dest_rank != m_master_io_rank) {
729 pm->send(nodes_coordinates, dest_rank);
730 }
731 }
732 else if (my_rank == dest_rank) {
733 pm->recv(nodes_coordinates, m_master_io_rank);
734 }
735
736 // Conserve les informations de ma partie
737 if (my_rank == dest_rank) {
738 m_mesh_allocate_info.nodes_coordinates.addRange(nodes_coordinates);
739 }
740 }
741
742 if (!m_is_binary)
743 _goToNextLine();
744}
745
746/*---------------------------------------------------------------------------*/
747/*---------------------------------------------------------------------------*/
753{
754 IosFile* ios_file = m_ios_file.get();
755 IParallelMng* pm = m_parallel_mng;
756 const Int32 my_rank = pm->commRank();
757 const Int64 nb_entity_in_block = block.nb_entity;
758 const Int32 item_nb_node = block.item_nb_node;
759
760 info() << "Reading block nb_entity=" << nb_entity_in_block << " item_nb_node=" << item_nb_node;
761
763 UniqueArray<Int64> connectivities;
764
765 UniqueArray<Int64> tmp_uid_and_connectivities;
766
767 for (Int32 i_part = 0; i_part < m_nb_part; ++i_part) {
768 const Int64 nb_to_read = _interval(i_part, m_nb_part, nb_entity_in_block).second;
769 const Int32 dest_rank = m_parts_rank[i_part];
770
771 info() << "Reading block part i_part=" << i_part
772 << " nb_to_read=" << nb_to_read << " dest_rank=" << dest_rank;
773
774 const Int64 nb_uid = nb_to_read;
775 const Int64 nb_connectivity = nb_uid * item_nb_node;
776 if (my_rank == dest_rank || my_rank == m_master_io_rank) {
777 uids.resize(nb_uid);
778 connectivities.resize(nb_connectivity);
779 }
780 if (ios_file) {
781 if (m_is_binary) {
782 Int64 nb_to_read = nb_uid * (item_nb_node + 1);
783 tmp_uid_and_connectivities.resize(nb_to_read);
784 ios_file->binaryRead(tmp_uid_and_connectivities.view());
785 Int64 index = 0;
786 for (Int64 i = 0; i < nb_uid; ++i) {
787 Int64 item_unique_id = tmp_uid_and_connectivities[index];
788 ++index;
789 uids[i] = item_unique_id;
790 for (Int32 j = 0; j < item_nb_node; ++j) {
791 connectivities[(i * item_nb_node) + j] = tmp_uid_and_connectivities[index];
792 ++index;
793 }
794 }
795 }
796 else {
797 // Utilise des Int64 pour garantir qu'on ne déborde pas.
798 for (Int64 i = 0; i < nb_uid; ++i) {
799 Int64 item_unique_id = ios_file->getInt64();
800 uids[i] = item_unique_id;
801 for (Int32 j = 0; j < item_nb_node; ++j)
802 connectivities[(i * item_nb_node) + j] = ios_file->getInt64();
803 }
804 }
805 if (dest_rank != m_master_io_rank) {
806 pm->send(uids, dest_rank);
807 pm->send(connectivities, dest_rank);
808 }
809 }
810 else if (my_rank == dest_rank) {
811 pm->recv(uids, m_master_io_rank);
812 pm->recv(connectivities, m_master_io_rank);
813 }
814 if (my_rank == dest_rank) {
815 block.uids.addRange(uids);
816 block.connectivities.addRange(connectivities);
817 }
818 }
819}
820
821/*---------------------------------------------------------------------------*/
822/*---------------------------------------------------------------------------*/
853{
854 IosFile* ios_file = m_ios_file.get();
855 IParallelMng* pm = m_parallel_mng;
856
857 FixedArray<Int64, 4> elements_info;
858 _getInt64ArrayAndBroadcast(elements_info.view());
859
860 Int64 nb_block = elements_info[0];
861 Int64 number_of_elements = elements_info[1];
862 Int64 min_element_tag = elements_info[2];
863 Int64 max_element_tag = elements_info[3];
864
865 if (!m_is_binary)
866 _goToNextLine();
867
868 info() << "[Elements] nb_block=" << nb_block
869 << " nb_elements=" << number_of_elements
870 << " min_element_tag=" << min_element_tag
871 << " max_element_tag=" << max_element_tag;
872
873 if (number_of_elements < 0)
874 ARCANE_THROW(IOException, "Invalid number of elements: {0}", number_of_elements);
875
876 UniqueArray<MshElementsBlock>& blocks = m_mesh_allocate_info.blocks;
877 blocks.resize(nb_block);
878
879 {
880 // Numérote les blocs (pour le débug)
881 Integer index = 0;
882 for (MshElementsBlock& block : blocks) {
883 block.index = index;
884 ++index;
885 }
886 }
887
888 for (MshElementsBlock& block : blocks) {
889
890 FixedArray<Int32, 3> block_info;
891 _getInt32ArrayAndBroadcast(block_info.view());
892
893 Int32 entity_dim = block_info[0];
894 Int32 entity_tag = block_info[1];
895 Int32 entity_type = block_info[2];
896 Int64 nb_entity_in_block = _getInt64AndBroadcast();
897
898 Integer item_nb_node = 0;
899 ItemTypeId item_type(_switchMshType(entity_type, item_nb_node));
900
901 info() << "[Elements] index=" << block.index << " entity_dim=" << entity_dim
902 << " entity_tag=" << entity_tag
903 << " entity_type=" << entity_type << " nb_in_block=" << nb_entity_in_block
904 << " item_type=" << item_type << " item_nb_node=" << item_nb_node;
905
906 block.nb_entity = nb_entity_in_block;
907 block.item_type = item_type;
908 block.item_nb_node = item_nb_node;
909 block.dimension = entity_dim;
910 block.entity_tag = entity_tag;
911
912 if (entity_type == MSH_PNT) {
913 // Si le type est un point, le traitement semble un peu particulier.
914 // Il y a dans ce cas deux entiers dans la ligne suivante:
915 // - un entier qui ne semble pas être utilisé
916 // - le numéro unique du noeud qui nous intéresse
917 Int64 item_unique_id = NULL_ITEM_UNIQUE_ID;
918 if (ios_file) {
919 [[maybe_unused]] Int64 unused_id = _getInt64();
920 item_unique_id = _getInt64();
921 info() << "Adding unique node uid=" << item_unique_id;
922 }
923 if (m_is_parallel)
924 pm->broadcast(ArrayView<Int64>(1, &item_unique_id), m_master_io_rank);
925 block.uids.add(item_unique_id);
926 }
927 else {
929 }
930 if (!m_is_binary)
931 _goToNextLine();
932 }
933
934 if (m_is_binary)
935 _goToNextLine();
936
937 // Maintenant qu'on a tous les blocs, la dimension du maillage est
938 // la plus grande dimension des blocs
939 Integer mesh_dimension = -1;
940 for (const MshElementsBlock& block : blocks)
941 mesh_dimension = math::max(mesh_dimension, block.dimension);
942 if (mesh_dimension < 0)
943 ARCANE_FATAL("Invalid computed mesh dimension '{0}'", mesh_dimension);
944 if (mesh_dimension != 2 && mesh_dimension != 3)
945 ARCANE_THROW(NotSupportedException, "mesh dimension '{0}'. Only 2D or 3D meshes are supported", mesh_dimension);
946 info() << "Computed mesh dimension = " << mesh_dimension;
947
948 bool allow_multi_dim_cell = m_mesh->meshKind().isNonManifold();
949 bool use_experimental_type_for_cell = false;
950 if (allow_multi_dim_cell) {
951 // Par défaut utilise les nouveaux types.
952 use_experimental_type_for_cell = true;
953 if (auto v = Convert::Type<Int32>::tryParseFromEnvironment("ARCANE_USE_EXPERIMENTAL_CELL_TYPE", true))
954 use_experimental_type_for_cell = (v.value() != 0);
955 }
956 info() << "Use experimental cell type?=" << use_experimental_type_for_cell;
957 ItemTypeMng* item_type_mng = m_mesh->itemTypeMng();
958 for (MshElementsBlock& block : blocks) {
959 const Int32 block_dim = block.dimension;
960 String item_type_name = item_type_mng->typeFromId(block.item_type)->typeName();
961 info() << "Reading block dim=" << block_dim << " type_name=" << item_type_name;
962 if (block_dim == mesh_dimension)
963 _computeOwnItems(block, m_mesh_allocate_info.cells_infos, false);
964 else if (allow_multi_dim_cell) {
965 // Regarde si on peut créé des mailles de dimension inférieures à celles
966 // du maillage.
967 bool use_sub_dim_cell = false;
968 if (mesh_dimension == 3 && (block_dim == 2 || block_dim == 1))
969 // Maille 1D ou 2D dans un maillage 3D
970 use_sub_dim_cell = true;
971 else if (mesh_dimension == 2 && block_dim == 1)
972 // Maille 1D dans un maillage 2D
973 use_sub_dim_cell = true;
974 if (!use_experimental_type_for_cell)
975 use_sub_dim_cell = false;
976 if (use_sub_dim_cell) {
977 // Ici on va créer des mailles 2D dans un maillage 3D.
978 // On converti le type de base en un type équivalent pour les mailles.
979 if (mesh_dimension == 3) {
980 if (block.item_type == IT_Triangle3)
981 block.item_type = ItemTypeId(IT_Cell3D_Triangle3);
982 else if (block.item_type == IT_Quad4)
983 block.item_type = ItemTypeId(IT_Cell3D_Quad4);
984 else if (block.item_type == IT_Line2)
985 block.item_type = ItemTypeId(IT_Cell3D_Line2);
986 else
987 ARCANE_FATAL("Not supported sub dimension cell type={0} for 3D mesh", item_type_name);
988 }
989 else if (mesh_dimension == 2) {
990 if (block.item_type == IT_Line2)
991 block.item_type = ItemTypeId(IT_CellLine2);
992 else
993 ARCANE_FATAL("Not supported sub dimension cell type={0} for 2D mesh", item_type_name);
994 }
995 block.is_built_as_cells = true;
996 _computeOwnItems(block, m_mesh_allocate_info.cells_infos, false);
997 }
998 }
999 }
1000
1001 return mesh_dimension;
1002}
1003
1004/*---------------------------------------------------------------------------*/
1005/*---------------------------------------------------------------------------*/
1006
1007namespace
1008{
1009 template <typename DataType> inline ArrayView<DataType>
1010 _broadcastArrayWithSize(IParallelMng* pm, UniqueArray<DataType>& values,
1011 UniqueArray<DataType>& work_values, Int32 dest_rank, Int64 size)
1012 {
1013 const Int32 my_rank = pm->commRank();
1014 ArrayView<DataType> view = values.view();
1015 if (my_rank != dest_rank) {
1016 work_values.resize(size);
1017 view = work_values.view();
1018 }
1019 pm->broadcast(view, dest_rank);
1020 return view;
1021 }
1029 template <typename DataType> inline ArrayView<DataType>
1030 _broadcastArray(IParallelMng* pm, UniqueArray<DataType>& values,
1031 UniqueArray<DataType>& work_values, Int32 dest_rank)
1032 {
1033 const Int32 my_rank = pm->commRank();
1034 Int64 size = 0;
1035 // Envoie la taille
1036 if (dest_rank == my_rank)
1037 size = values.size();
1038 pm->broadcast(ArrayView<Int64>(1, &size), dest_rank);
1039 return _broadcastArrayWithSize(pm, values, work_values, dest_rank, size);
1040 }
1041
1042} // namespace
1043
1044/*---------------------------------------------------------------------------*/
1045/*---------------------------------------------------------------------------*/
1046
1047void MshParallelMeshReader::
1048_computeOwnItems(MshElementsBlock& block, MshItemKindInfo& item_kind_info, bool is_generate_uid)
1049{
1050 // On ne conserve que les entités dont le premier noeud appartient à notre rang.
1051
1052 IParallelMng* pm = m_parallel_mng;
1053 const Int32 my_rank = pm->commRank();
1054
1055 const ItemTypeId item_type = block.item_type;
1056 const Int32 item_nb_node = block.item_nb_node;
1057
1058 UniqueArray<Int64> connectivities;
1059 UniqueArray<Int64> uids;
1060 UniqueArray<Int32> nodes_rank;
1061
1062 const Int32 nb_part = m_parts_rank.size();
1063 info() << "Compute own items block_index=" << block.index << " nb_part=" << nb_part;
1064 for (Int32 i_part = 0; i_part < nb_part; ++i_part) {
1065 const Int32 dest_rank = m_parts_rank[i_part];
1066 // Broadcast la i_part-ème partie des uids et connectivités des mailles
1067 ArrayView<Int64> connectivities_view = _broadcastArray(pm, block.connectivities, connectivities, dest_rank);
1068 ArrayView<Int64> uids_view = _broadcastArray(pm, block.uids, uids, dest_rank);
1069
1070 Int32 nb_item = uids_view.size();
1071 nodes_rank.resize(nb_item);
1072 nodes_rank.fill(-1);
1073
1074 // Parcours les entités. Chaque entité appartiendra au rang
1075 // de son premier noeud. Si cette partie correspond à mon rang, alors
1076 // on conserve la maille.
1077 for (Int32 i = 0; i < nb_item; ++i) {
1078 Int64 first_node_uid = connectivities_view[i * item_nb_node];
1079 auto x = m_mesh_allocate_info.nodes_rank_map.find(first_node_uid);
1080 if (x == m_mesh_allocate_info.nodes_rank_map.end())
1081 // Le noeud n'est pas dans ma liste
1082 continue;
1083 Int32 rank = x->second;
1084 nodes_rank[i] = rank;
1085 }
1086 pm->reduce(Parallel::ReduceMax, nodes_rank);
1087 for (Int32 i = 0; i < nb_item; ++i) {
1088 const Int32 rank = nodes_rank[i];
1089 if (rank != my_rank)
1090 // Le noeud n'est pas dans ma partie
1091 continue;
1092 // Le noeud est chez moi, j'ajoute l'entité à la liste des
1093 // entités que je vais créer.
1094 ConstArrayView<Int64> v = connectivities_view.subView(i * item_nb_node, item_nb_node);
1095 Int64 uid = uids_view[i];
1096 if (is_generate_uid)
1097 // Le uniqueId() sera généré automatiquement
1098 uid = NULL_ITEM_UNIQUE_ID;
1099 item_kind_info.addItem(item_type, uid, v);
1100 }
1101 }
1102}
1103
1104/*---------------------------------------------------------------------------*/
1105/*---------------------------------------------------------------------------*/
1122{
1123 UniqueArray<Int64> uids_storage;
1124 UniqueArray<Real3> coords_storage;
1125 UniqueArray<Int32> local_ids;
1126
1127 IParallelMng* pm = m_parallel_mng;
1128
1129 const IItemFamily* node_family = m_mesh->nodeFamily();
1130 VariableNodeReal3& nodes_coord_var(m_mesh->nodesCoordinates());
1131
1132 for (Int32 dest_rank : m_parts_rank) {
1133 ConstArrayView<Int64> uids = _broadcastArray(pm, m_mesh_allocate_info.nodes_unique_id, uids_storage, dest_rank);
1134 ConstArrayView<Real3> coords = _broadcastArray(pm, m_mesh_allocate_info.nodes_coordinates, coords_storage, dest_rank);
1135
1136 Int32 nb_item = uids.size();
1137 local_ids.resize(nb_item);
1138
1139 // Converti les uniqueId() en localId(). S'ils sont non nuls
1140 // c'est que l'entité est dans mon sous-domaine et donc on peut
1141 // positionner sa coordonnée
1142 node_family->itemsUniqueIdToLocalId(local_ids, uids, false);
1143 for (Int32 i = 0; i < nb_item; ++i) {
1144 NodeLocalId nid(local_ids[i]);
1145 if (!nid.isNull())
1146 nodes_coord_var[nid] = coords[i];
1147 }
1148 }
1149}
1150
1151/*---------------------------------------------------------------------------*/
1152/*---------------------------------------------------------------------------*/
1153
1154void MshParallelMeshReader::
1155_addFaces()
1156{
1157 IMesh* mesh = m_mesh;
1158 Integer nb_item = m_mesh_allocate_info.faces_infos.nb_item;
1159 info() << "Adding faces direct nb_face=" << nb_item;
1160 if (nb_item != 0)
1161 mesh->modifier()->addFaces(nb_item, m_mesh_allocate_info.faces_infos.items_infos);
1162}
1163
1164/*---------------------------------------------------------------------------*/
1165/*---------------------------------------------------------------------------*/
1166
1167void MshParallelMeshReader::
1168_allocateCells()
1169{
1170 IMesh* mesh = m_mesh;
1171 Integer nb_elements = m_mesh_allocate_info.cells_infos.nb_item;
1172 info() << "nb_of_elements=cells_type.size()=" << nb_elements;
1173 Integer nb_cell_node = m_mesh_allocate_info.cells_infos.items_infos.size();
1174 info() << "nb_cell_node=cells_connectivity.size()=" << nb_cell_node;
1175
1176 // Création des mailles
1177 info() << "Building cells, nb_cell=" << nb_elements << " nb_cell_node=" << nb_cell_node;
1178 IPrimaryMesh* pmesh = mesh->toPrimaryMesh();
1179 info() << "## Allocating ##";
1180 pmesh->allocateCells(nb_elements, m_mesh_allocate_info.cells_infos.items_infos, false);
1181 info() << "## Ending ##";
1182 // Ajoute ensuite faces si elles peuvent ne pas être connectées au maillage.
1183 if (m_mesh->meshKind().isNonManifold())
1184 _addFaces();
1185 pmesh->endAllocate();
1186 info() << "## Done ##";
1187}
1188
1189/*---------------------------------------------------------------------------*/
1190/*---------------------------------------------------------------------------*/
1191
1192void MshParallelMeshReader::
1193_allocateGroups()
1194{
1195 IMesh* mesh = m_mesh;
1196 Int32 mesh_dim = mesh->dimension();
1197 Int32 face_dim = mesh_dim - 1;
1198 UniqueArray<MshEntitiesWithNodes> entity_list;
1199 UniqueArray<MshPhysicalName> physical_name_list;
1200 for (MshElementsBlock& block : m_mesh_allocate_info.blocks) {
1201 entity_list.clear();
1202 physical_name_list.clear();
1203 Int32 block_index = block.index;
1204 Int32 block_dim = block.dimension;
1205 // On alloue un groupe s'il a un nom physique associé.
1206 // Pour cela, il faut déjà qu'il soit associé à une entité.
1207 Int64 block_entity_tag = block.entity_tag;
1208 if (block_entity_tag < 0) {
1209 info(5) << "[Groups] Skipping block index=" << block_index << " because it has no entity";
1210 continue;
1211 }
1212 if (block_dim == 0) {
1213 const MshEntitiesNodes* entity = m_mesh_info->findNodeEntities(block_entity_tag);
1214 if (!entity) {
1215 info(5) << "[Groups] Skipping block index=" << block_index
1216 << " because entity tag is invalid";
1217 continue;
1218 }
1219 Int64 entity_physical_tag = entity->physicalTag();
1220 MshPhysicalName physical_name = m_mesh_info->physical_name_list.find(block_dim, entity_physical_tag);
1221 physical_name_list.add(physical_name);
1222 }
1223 else {
1224 m_mesh_info->findEntities(block_dim, block_entity_tag, entity_list);
1225 for (const MshEntitiesWithNodes& x : entity_list) {
1226 Int64 entity_physical_tag = x.physicalTag();
1227 MshPhysicalName physical_name = m_mesh_info->physical_name_list.find(block_dim, entity_physical_tag);
1228 physical_name_list.add(physical_name);
1229 }
1230 }
1231 for (const MshPhysicalName& physical_name : physical_name_list) {
1232 if (physical_name.isNull()) {
1233 info(5) << "[Groups] Skipping block index=" << block_index
1234 << " because entity physical tag is invalid";
1235 continue;
1236 }
1237 String group_name = physical_name.name();
1238 info(4) << "[Groups] Block index=" << block_index << " dim=" << block_dim
1239 << " name='" << group_name << "' built_as_cells=" << block.is_built_as_cells;
1240 if (block_dim == mesh_dim || block.is_built_as_cells) {
1241 _addCellOrNodeGroup(block, group_name, mesh->cellFamily());
1242 }
1243 else if (block_dim == face_dim) {
1244 _addFaceGroup(block, group_name);
1245 }
1246 else {
1247 _addCellOrNodeGroup(block, group_name, mesh->nodeFamily());
1248 }
1249 }
1250 }
1251}
1252
1253/*---------------------------------------------------------------------------*/
1254/*---------------------------------------------------------------------------*/
1259_addFaceGroup(MshElementsBlock& block, const String& group_name)
1260{
1261 IParallelMng* pm = m_parallel_mng;
1262 const Int32 item_nb_node = block.item_nb_node;
1263
1264 UniqueArray<Int64> connectivities;
1265 for (Int32 dest_rank : m_parts_rank) {
1266 ArrayView<Int64> connectivities_view = _broadcastArray(pm, block.connectivities, connectivities, dest_rank);
1267 _addFaceGroupOnePart(connectivities_view, item_nb_node, group_name, block.index);
1268 }
1269}
1270
1271/*---------------------------------------------------------------------------*/
1272/*---------------------------------------------------------------------------*/
1273
1274void MshParallelMeshReader::
1275_addFaceGroupOnePart(ConstArrayView<Int64> connectivities, Int32 item_nb_node,
1276 const String& group_name, Int32 block_index)
1277{
1278 IMesh* mesh = m_mesh;
1279 const Int32 nb_entity = connectivities.size() / item_nb_node;
1280
1281 // Il peut y avoir plusieurs blocs pour le même groupe.
1282 // On récupère le groupe s'il existe déjà.
1283 FaceGroup face_group = mesh->faceFamily()->findGroup(group_name, true);
1284
1285 UniqueArray<Int32> faces_id; // Numéro de la face dans le maillage \a mesh
1286 faces_id.reserve(nb_entity);
1287
1288 const Int32 face_nb_node = nb_entity * item_nb_node;
1289
1290 UniqueArray<Int64> faces_first_node_unique_id(nb_entity);
1291 UniqueArray<Int32> faces_first_node_local_id(nb_entity);
1292 UniqueArray<Int64> faces_nodes_unique_id(face_nb_node);
1293 Integer faces_nodes_unique_id_index = 0;
1294
1295 UniqueArray<Int64> orig_nodes_id(item_nb_node);
1296 UniqueArray<Integer> face_nodes_index(item_nb_node);
1297
1298 IItemFamily* node_family = mesh->nodeFamily();
1299 NodeInfoListView mesh_nodes(node_family);
1300
1301 // Réordonne les identifiants des faces retrouver la face dans le maillage.
1302 // Pour cela, on récupère le premier noeud de la face et on regarde s'il
1303 // se trouve dans notre sous-domaine. Si oui, la face sera ajoutée à notre
1304 // partie du maillage
1305 for (Integer i_face = 0; i_face < nb_entity; ++i_face) {
1306 for (Integer z = 0; z < item_nb_node; ++z)
1307 orig_nodes_id[z] = connectivities[faces_nodes_unique_id_index + z];
1308
1309 MeshUtils::reorderNodesOfFace2(orig_nodes_id, face_nodes_index);
1310 for (Integer z = 0; z < item_nb_node; ++z)
1311 faces_nodes_unique_id[faces_nodes_unique_id_index + z] = orig_nodes_id[face_nodes_index[z]];
1312 faces_first_node_unique_id[i_face] = orig_nodes_id[face_nodes_index[0]];
1313 faces_nodes_unique_id_index += item_nb_node;
1314 }
1315
1316 node_family->itemsUniqueIdToLocalId(faces_first_node_local_id, faces_first_node_unique_id, false);
1317
1318 faces_nodes_unique_id_index = 0;
1319 for (Integer i_face = 0; i_face < nb_entity; ++i_face) {
1320 const Integer n = item_nb_node;
1321 Int32 face_first_node_lid = faces_first_node_local_id[i_face];
1322 if (face_first_node_lid != NULL_ITEM_LOCAL_ID) {
1323 Int64ConstArrayView face_nodes_id(item_nb_node, &faces_nodes_unique_id[faces_nodes_unique_id_index]);
1324 Node current_node(mesh_nodes[faces_first_node_local_id[i_face]]);
1325 Face face = MeshUtils::getFaceFromNodesUniqueId(current_node, face_nodes_id);
1326
1327 // En parallèle, il est possible que la face ne soit pas dans notre sous-domaine
1328 // même si un de ses noeuds l'est
1329 if (face.null()) {
1330 if (!m_is_parallel) {
1331 OStringStream ostr;
1332 ostr() << "(Nodes:";
1333 for (Integer z = 0; z < n; ++z)
1334 ostr() << ' ' << face_nodes_id[z];
1335 ostr() << " - " << current_node.localId() << ")";
1336 ARCANE_FATAL("INTERNAL: MeshMeshReader face index={0} with nodes '{1}' is not in node/face connectivity",
1337 i_face, ostr.str());
1338 }
1339 }
1340 else
1341 faces_id.add(face.localId());
1342 }
1343
1344 faces_nodes_unique_id_index += n;
1345 }
1346 info(4) << "Adding " << faces_id.size() << " faces from block index=" << block_index
1347 << " to group '" << face_group.name() << "'";
1348 face_group.addItems(faces_id);
1349}
1350
1351/*---------------------------------------------------------------------------*/
1352/*---------------------------------------------------------------------------*/
1357_addCellOrNodeGroup(MshElementsBlock& block, const String& group_name, IItemFamily* family)
1358{
1359 IParallelMng* pm = m_parallel_mng;
1360
1361 UniqueArray<Int64> uids;
1362 for (Int32 dest_rank : m_parts_rank) {
1363 ArrayView<Int64> uids_view = _broadcastArray(pm, block.uids, uids, dest_rank);
1364 _addCellOrNodeGroupOnePart(uids_view, group_name, block.index, family);
1365 }
1366}
1367
1368/*---------------------------------------------------------------------------*/
1369/*---------------------------------------------------------------------------*/
1370
1371void MshParallelMeshReader::
1372_addCellOrNodeGroupOnePart(ConstArrayView<Int64> uids, const String& group_name,
1373 Int32 block_index, IItemFamily* family)
1374{
1375 const Int32 nb_entity = uids.size();
1376
1377 // Il peut y avoir plusieurs blocs pour le même groupe.
1378 // On récupère le groupe s'il existe déjà.
1379 ItemGroup group = family->findGroup(group_name, true);
1380
1381 UniqueArray<Int32> items_lid(nb_entity);
1382
1383 family->itemsUniqueIdToLocalId(items_lid, uids, false);
1384
1385 // En parallèle, il est possible que certaines entités du groupe ne soient
1386 // pas dans notre sous-domaine. Il faut les filtrer.
1387 if (m_is_parallel) {
1388 auto items_begin = items_lid.begin();
1389 Int64 new_size = std::remove(items_begin, items_lid.end(), NULL_ITEM_LOCAL_ID) - items_begin;
1390 items_lid.resize(new_size);
1391 }
1392
1393 info() << "Adding " << items_lid.size() << " items from block index=" << block_index
1394 << " to group '" << group_name << "' for family=" << family->name();
1395
1396 group.addItems(items_lid);
1397}
1398
1399/*---------------------------------------------------------------------------*/
1400/*---------------------------------------------------------------------------*/
1401/*
1402 * $PhysicalNames // same as MSH version 2
1403 * numPhysicalNames(ASCII int)
1404 * dimension(ASCII int) physicalTag(ASCII int) "name"(127 characters max)
1405 * ...
1406 * $EndPhysicalNames
1407 */
1408void MshParallelMeshReader::
1409_readPhysicalNames()
1410{
1411 // NOTE: même en binaire, la partie $PhysicalNames est écrite en ASCII.
1412 String quote_mark = "\"";
1413 Int32 nb_name = _getIntegerAndBroadcast();
1414 info() << "nb_physical_name=" << nb_name;
1415
1416 _goToNextLine();
1417
1418 for (Int32 i = 0; i < nb_name; ++i) {
1419 Int32 dim = _getIntegerAndBroadcast();
1420 Int32 tag = _getIntegerAndBroadcast();
1421 String s = _getNextLineAndBroadcast();
1422 if (dim < 0 || dim > 3)
1423 ARCANE_FATAL("Invalid value for physical name dimension dim={0}", dim);
1424 // Les noms des groupes peuvent commencer par des espaces et contiennent
1425 // des guillemets qu'il faut supprimer.
1427 if (s.startsWith(quote_mark))
1428 s = s.substring(1);
1429 if (s.endsWith(quote_mark))
1430 s = s.substring(0, s.length() - 1);
1431 m_mesh_info->physical_name_list.add(dim, tag, s);
1432 info(4) << "[PhysicalName] index=" << i << " dim=" << dim << " tag=" << tag << " name='" << s << "'";
1433 }
1434
1435 String s = _getNextLineAndBroadcast();
1436 if (s != "$EndPhysicalNames")
1437 ARCANE_FATAL("found '{0}' and expected '$EndPhysicalNames'", s);
1438}
1439
1440/*---------------------------------------------------------------------------*/
1441/*---------------------------------------------------------------------------*/
1474{
1475 IosFile* ios_file = m_ios_file.get();
1476
1477 FixedArray<Int64, 4> nb_dim_item;
1478 _getInt64ArrayAndBroadcast(nb_dim_item.view());
1479
1480 info() << "[Entities] nb_0d=" << nb_dim_item[0] << " nb_1d=" << nb_dim_item[1]
1481 << " nb_2d=" << nb_dim_item[2] << " nb_3d=" << nb_dim_item[3];
1482 // Après le format, on peut avoir les entités mais cela est optionnel
1483 // Si elles sont présentes, on lit le fichier jusqu'à la fin de cette section.
1484 if (!m_is_binary)
1485 _goToNextLine();
1486
1487 // Lecture des entités associées à des points
1488 for (Int64 i = 0; i < nb_dim_item[0]; ++i) {
1489 FixedArray<Int64, 2> tag_info;
1490 if (ios_file) {
1491 Int32 tag = _getInt32();
1492 Real3 xyz = _getReal3();
1493 Int64 num_physical_tag = _getInt64();
1494 if (num_physical_tag > 1)
1495 ARCANE_FATAL("NotImplemented numPhysicalTag>1 (n={0}, index={1} xyz={2})",
1496 num_physical_tag, i, xyz);
1497
1498 Int32 physical_tag = -1;
1499 if (num_physical_tag == 1)
1500 physical_tag = _getInt32();
1501 info() << "[Entities] point tag=" << tag << " pos=" << xyz << " phys_tag=" << physical_tag;
1502
1503 tag_info[0] = tag;
1504 tag_info[1] = physical_tag;
1505 }
1506 m_parallel_mng->broadcast(tag_info.view(), m_master_io_rank);
1507 m_mesh_info->entities_nodes_list.add(MshEntitiesNodes(tag_info[0], tag_info[1]));
1508 if (!m_is_binary)
1509 _goToNextLine();
1510 }
1511
1512 // Lecture des entités de dimensions 1, 2 et 3.
1513 for (Int32 i_dim = 1; i_dim <= 3; ++i_dim)
1514 for (Int32 i = 0; i < nb_dim_item[i_dim]; ++i)
1515 _readOneEntity(i_dim, i);
1516
1517 // En binaire, il faut aller au début de la ligne suivante.
1518 if (m_is_binary)
1519 _goToNextLine();
1520 _readAndCheck("$EndEntities");
1521}
1522
1523/*---------------------------------------------------------------------------*/
1524/*---------------------------------------------------------------------------*/
1525
1526void MshParallelMeshReader::
1527_readOneEntity(Int32 entity_dim, Int32 entity_index_in_dim)
1528{
1529 IosFile* ios_file = m_ios_file.get();
1530
1531 // Infos pour les tags
1532 // [0] entity_dim
1533 // [1] tag
1534 // [2] nb_physical_tag
1535 // [2+1] physical_tag1
1536 // [2+N] physical_tagN
1537 // ...
1538 // [3+nb_physical_tag] nb_boundary
1539 // [3+nb_physical_tag+1] boundary_tag1
1540 // [3+nb_physical_tag+n] boundary_tagN
1541 // ...
1542 info() << "[Entities] Reading entity dim=" << entity_dim << " index_in_dim=" << entity_index_in_dim;
1543 SmallArray<Int64, 128> dim_and_tag_info;
1544 dim_and_tag_info.add(entity_dim);
1545 if (ios_file) {
1546 Int32 tag = _getInt32();
1547 dim_and_tag_info.add(tag);
1548 Real3 min_pos = _getReal3();
1549 Real3 max_pos = _getReal3();
1550 Int64 nb_physical_tag = _getInt64();
1551 dim_and_tag_info.add(nb_physical_tag);
1552 for (Int32 z = 0; z < nb_physical_tag; ++z) {
1553 Int32 physical_tag = _getInt32();
1554 dim_and_tag_info.add(physical_tag);
1555 info(4) << "[Entities] z=" << z << " physical_tag=" << physical_tag;
1556 }
1557 // TODO: Lire les informations numBounding...
1558 Int64 nb_bounding_group = _getInt64();
1559 dim_and_tag_info.add(nb_bounding_group);
1560 for (Int64 z = 0; z < nb_bounding_group; ++z) {
1561 Int32 boundary_tag = _getInt32();
1562 info(4) << "[Entities] z=" << z << " boundary_tag=" << boundary_tag;
1563 }
1564 info(4) << "[Entities] dim=" << entity_dim << " tag=" << tag
1565 << " min_pos=" << min_pos << " max_pos=" << max_pos
1566 << " nb_phys_tag=" << nb_physical_tag
1567 << " nb_bounding=" << nb_bounding_group;
1568 }
1569 Int32 info_size = dim_and_tag_info.size();
1570 m_parallel_mng->broadcast(ArrayView<Int32>(1, &info_size), m_master_io_rank);
1571 dim_and_tag_info.resize(info_size);
1572 m_parallel_mng->broadcast(dim_and_tag_info.view(), m_master_io_rank);
1573
1574 {
1575 Int32 dim = CheckedConvert::toInt32(dim_and_tag_info[0]);
1576 Int64 tag = dim_and_tag_info[1];
1577 Int64 nb_physical_tag = dim_and_tag_info[2];
1578 for (Int32 z = 0; z < nb_physical_tag; ++z) {
1579 Int64 physical_tag = dim_and_tag_info[3 + z];
1580 info(4) << "[Entities] adding info dim=" << entity_dim << " tag=" << tag
1581 << " physical_tag=" << physical_tag;
1582 m_mesh_info->entities_with_nodes_list[dim - 1].add(MshEntitiesWithNodes(dim, tag, physical_tag));
1583 }
1584 }
1585
1586 if (!m_is_binary)
1587 _goToNextLine();
1588}
1589
1590/*---------------------------------------------------------------------------*/
1591/*---------------------------------------------------------------------------*/
1611{
1612 Int64 nb_link = _getInt64AndBroadcast();
1613 info() << "[Periodic] nb_link=" << nb_link;
1614
1615 // TODO: pour l'instant, tous les rangs conservent les
1616 // données car on suppose qu'il n'y en a pas beaucoup.
1617 // A terme, il faudra aussi distribuer ces informations.
1618 MshPeriodicInfo& periodic_info = m_mesh_info->m_periodic_info;
1619 periodic_info.m_periodic_list.resize(nb_link);
1620 for (Int64 ilink = 0; ilink < nb_link; ++ilink) {
1621 MshPeriodicOneInfo& one_info = periodic_info.m_periodic_list[ilink];
1622 FixedArray<Int32, 3> entity_info;
1623 _getInt32ArrayAndBroadcast(entity_info.view());
1624
1625 info() << "[Periodic] link_index=" << ilink << " dim=" << entity_info[0] << " entity_tag=" << entity_info[1]
1626 << " entity_tag_master=" << entity_info[2];
1627 one_info.m_entity_dim = entity_info[0];
1628 one_info.m_entity_tag = entity_info[1];
1629 one_info.m_entity_tag_master = entity_info[2];
1630
1631 Int64 num_affine = _getInt64AndBroadcast();
1632 info() << "[Periodic] num_affine=" << num_affine;
1633 one_info.m_affine_values.resize(num_affine);
1634 _getDoubleArrayAndBroadcast(one_info.m_affine_values);
1635 one_info.m_nb_corresponding_node = CheckedConvert::toInt32(_getInt64AndBroadcast());
1636 info() << "[Periodic] nb_corresponding_node=" << one_info.m_nb_corresponding_node;
1637 one_info.m_corresponding_nodes.resize(one_info.m_nb_corresponding_node * 2);
1638 _getInt64ArrayAndBroadcast(one_info.m_corresponding_nodes);
1639 info() << "[Periodic] corresponding_nodes=" << one_info.m_corresponding_nodes;
1640 }
1641
1642 _goToNextLine();
1643
1645 if (s != "$EndPeriodic")
1646 ARCANE_FATAL("found '{0}' and expected '$EndPeriodic'", s);
1647}
1648
1649/*---------------------------------------------------------------------------*/
1650/*---------------------------------------------------------------------------*/
1655_readAndCheck(const String& expected_value)
1656{
1657 String s;
1658 if (m_is_binary) {
1659 constexpr Int32 MAX_SIZE = 128;
1661 Span<const Byte> expected_bytes = expected_value.bytes();
1662 Int32 read_size = CheckedConvert::toInt32(expected_bytes.size());
1663 SmallSpan<Byte> bytes_to_read = buf_bytes.span().subSpan(0, read_size);
1664 if (read_size >= MAX_SIZE)
1665 ARCANE_FATAL("local buffer is too small (size={0} max={1})", read_size, MAX_SIZE);
1666 IosFile* f = m_ios_file.get();
1667 if (f) {
1668 f->binaryRead(bytes_to_read);
1669 s = String(bytes_to_read);
1670 info() << "S=" << s;
1671 if (m_is_parallel) {
1672 m_parallel_mng->broadcastString(s, m_master_io_rank);
1673 }
1674 }
1675 _goToNextLine();
1676 }
1677 else {
1679 }
1680 if (s != expected_value)
1681 ARCANE_FATAL("found '{0}' and expected '{1}'", s, expected_value);
1682}
1683
1684/*---------------------------------------------------------------------------*/
1685/*---------------------------------------------------------------------------*/
1688void MshParallelMeshReader::
1689_readMeshFromFile()
1690{
1691 IosFile* ios_file = m_ios_file.get();
1692 IMesh* mesh = m_mesh;
1693 info() << "Reading 'msh' file in parallel";
1694
1695 const int MSH_BINARY_TYPE = 1;
1696
1697 if (ios_file) {
1698 Real version = ios_file->getReal();
1699 if (version != 4.1)
1700 ARCANE_THROW(IOException, "Wrong msh file version '{0}'. Only version '4.1' is supported in parallel", version);
1701 Integer file_type = ios_file->getInteger(); // is an integer equal to 0 in the ASCII file format, equal to 1 for the binary format
1702 if (file_type == MSH_BINARY_TYPE)
1703 m_is_binary = true;
1704 info() << "IsBinary?=" << m_is_binary;
1705 if (m_is_binary)
1706 pwarning() << "MSH reader for binary format is experimental";
1707 Int32 data_size = ios_file->getInteger(); // is an integer equal to the size of the floating point numbers used in the file
1708 ARCANE_UNUSED(data_size);
1709 if (data_size != 8)
1710 ARCANE_FATAL("Only 'size_t' of size '8' is allowed (current size is '{0}')", data_size);
1711 // En binaire, il a un entier à lire qui vaut 1 et qui permet de savoir si on n'est en big/little endian.
1712 if (m_is_binary) {
1713 // Il faut lire jusqu'à la fin de la ligne pour le binaire.
1714 _goToNextLine();
1715 Int32 int_value_one = 0;
1716 ios_file->binaryRead(SmallSpan<Int32>(&int_value_one, 1));
1717 if (int_value_one != 1)
1718 ARCANE_FATAL("Bad endianess for file. Read int as value '{0}' (expected=1)", int_value_one);
1719 }
1720
1721 ios_file->getNextLine(); // Skip current \n\r
1722
1723 // $EndMeshFormat
1724 if (!ios_file->lookForString("$EndMeshFormat"))
1725 ARCANE_THROW(IOException, "$EndMeshFormat not found");
1726 }
1727
1728 // TODO: Les différentes sections ($Nodes, $Entitites, ...) peuvent
1729 // être dans n'importe quel ordre (à part $Nodes qui doit être avant $Elements)
1730 // Faire un méthode qui gère cela.
1731
1732 String next_line = _getNextLineAndBroadcast();
1733 // Noms des groupes
1734 if (next_line == "$PhysicalNames") {
1735 _readPhysicalNames();
1736 next_line = _getNextLineAndBroadcast();
1737 }
1738
1739 // Après le format, on peut avoir les entités mais cela est optionnel
1740 // Si elles sont présentes, on lit le fichier jusqu'à la fin de cette section.
1741 if (next_line == "$Entities") {
1742 _readEntities();
1743 next_line = _getNextLineAndBroadcast();
1744 }
1745 // $Nodes
1746 if (next_line != "$Nodes")
1747 ARCANE_THROW(IOException, "Unexpected string '{0}'. Valid values are '$Nodes'", next_line);
1748
1749 // Fetch nodes number and the coordinates
1751
1752 // $EndNodes
1753 if (ios_file && !ios_file->lookForString("$EndNodes"))
1754 ARCANE_THROW(IOException, "$EndNodes not found");
1755
1756 // $Elements
1757 if (ios_file && !ios_file->lookForString("$Elements"))
1758 ARCANE_THROW(IOException, "$Elements not found");
1759
1760 Int32 mesh_dimension = _readElementsFromFile();
1761
1762 // $EndElements
1763 if (ios_file && !ios_file->lookForString("$EndElements"))
1764 ARCANE_THROW(IOException, "$EndElements not found");
1765
1766 info() << "Computed mesh dimension = " << mesh_dimension;
1767
1768 IPrimaryMesh* pmesh = mesh->toPrimaryMesh();
1769 pmesh->setDimension(mesh_dimension);
1770
1771 info() << "NextLine=" << next_line;
1772
1773 bool is_end = _getIsEndOfFileAndBroadcast();
1774 if (!is_end) {
1775 next_line = _getNextLineAndBroadcast();
1776 // $Periodic
1777 if (next_line == "$Periodic") {
1778 _readPeriodic();
1779 }
1780 }
1781
1782 _allocateCells();
1784 _allocateGroups();
1785}
1786
1787/*---------------------------------------------------------------------------*/
1788/*---------------------------------------------------------------------------*/
1793readMeshFromMshFile(IMesh* mesh, const String& filename, bool use_internal_partition)
1794{
1795 m_mesh_info = impl::MshMeshGenerationInfo::getReference(mesh, true);
1796 info() << "Trying to read in parallel 'msh' file '" << filename << "'"
1797 << " use_internal_partition=" << use_internal_partition;
1798 m_mesh = mesh;
1799 IParallelMng* pm = mesh->parallelMng();
1800 // Lit en séquentiel si les fichiers sont déjà découpés
1801 if (!use_internal_partition)
1802 pm = pm->sequentialParallelMng();
1803 m_parallel_mng = pm;
1804 const Int32 nb_rank = pm->commSize();
1805
1806 // Détermine les rangs qui vont conserver les données.
1807 // Il n'est pas obligatoire que tous les rangs participent
1808 // à la conservation des données. L'idéal avec un
1809 // grand nombre de rangs serait qu'un rang sur 2 ou 4 participent.
1810 // Cependant, cela génère alors des partitions vides (pour
1811 // les rangs qui ne participent pas) et cela peut
1812 // faire planter certains partitionneurs (comme ParMetis)
1813 // lorsqu'il y a des partitions vides. Il faudrait d'abord
1814 // corriger ce problème.
1815 m_nb_part = nb_rank;
1816 m_parts_rank.resize(m_nb_part);
1817 for (Int32 i = 0; i < m_nb_part; ++i) {
1818 m_parts_rank[i] = i % nb_rank;
1819 }
1820
1821 bool is_master_io = pm->isMasterIO();
1822 Int32 master_io_rank = pm->masterIORank();
1823 m_is_parallel = pm->isParallel();
1824 m_master_io_rank = master_io_rank;
1825 FixedArray<Int32, 1> file_readable;
1826 // Seul le rang maître va lire le fichier.
1827 // On vérifie d'abord qu'il est lisible
1828 if (is_master_io) {
1829 bool is_readable = platform::isFileReadable(filename);
1830 info() << "Is file readable ?=" << is_readable;
1831 file_readable[0] = is_readable ? 1 : 0;
1832 if (!is_readable)
1833 error() << "Unable to read file '" << filename << "'";
1834 }
1835 pm->broadcast(file_readable.view(), master_io_rank);
1836 if (file_readable[0] == 0) {
1837 return IMeshReader::RTError;
1838 }
1839
1840 std::ifstream ifile;
1841 Ref<IosFile> ios_file;
1842 if (is_master_io) {
1843 ifile.open(filename.localstr());
1844 ios_file = makeRef<IosFile>(new IosFile(&ifile));
1845 }
1846 m_ios_file = ios_file;
1847 String mesh_format_str = _getNextLineAndBroadcast();
1848 if (IosFile::isEqualString(mesh_format_str, "$MeshFormat")) {
1849 _readMeshFromFile();
1850 if (!use_internal_partition) {
1851 info() << "Synchronize groups and variables";
1852 mesh->synchronizeGroupsAndVariables();
1853 }
1854 return IMeshReader::RTOk;
1855 }
1856
1857 info() << "The file does not begin with '$MeshFormat' returning RTError";
1858 return IMeshReader::RTError;
1859}
1860
1861/*---------------------------------------------------------------------------*/
1862/*---------------------------------------------------------------------------*/
1863
1864extern "C++" Ref<IMshMeshReader>
1865createMshParallelMeshReader(ITraceMng* tm)
1866{
1868}
1869
1870/*---------------------------------------------------------------------------*/
1871/*---------------------------------------------------------------------------*/
1872
1873} // End namespace Arcane
1874
1875/*---------------------------------------------------------------------------*/
1876/*---------------------------------------------------------------------------*/
#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_THROW(exception_class,...)
Macro pour envoyer une exception avec formattage.
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
Fonctions utilitaires sur le maillage.
bool reorderNodesOfFace2(Int64ConstArrayView nodes_unique_id, IntegerArrayView new_index)
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 reserve(Int64 new_capacity)
Réserve le mémoire pour new_capacity éléments.
void addRange(ConstReferenceType val, Int64 n)
Ajoute n élément de valeur val à la fin du tableau.
ArrayView< T > view() const
Vue mutable sur ce tableau.
void add(ConstReferenceType val)
Ajoute l'élément val à la fin du tableau.
Vue constante d'un tableau de type T.
constexpr Integer size() const noexcept
Nombre d'éléments du tableau.
Tableau 1D de taille fixe.
Definition FixedArray.h:45
constexpr __host__ __device__ ArrayView< T > view()
Vue modifiable sur le tableau.
Definition FixedArray.h:97
constexpr __host__ __device__ SmallSpan< T, NbElement > span()
Vue modifiable sur le tableau.
Definition FixedArray.h:93
Informations sur le type flottant.
Definition Limits.h:48
Interface d'une famille d'entités.
Definition IItemFamily.h:84
virtual ItemGroup findGroup(const String &name) const =0
Recherche un groupe.
virtual String name() const =0
Nom de la famille.
virtual void itemsUniqueIdToLocalId(Int32ArrayView local_ids, Int64ConstArrayView unique_ids, bool do_fatal=true) const =0
Converti un tableau de numéros uniques en numéros locaux.
eReturnType
Types des codes de retour d'une lecture ou écriture.
Definition IMeshReader.h:42
@ RTError
Erreur lors de l'opération.
Definition IMeshReader.h:44
@ RTOk
Opération effectuée avec succès.
Definition IMeshReader.h:43
virtual const MeshKind meshKind() const =0
Caractéristiques du maillage.
virtual IPrimaryMesh * toPrimaryMesh()=0
Retourne l'instance sous la forme d'un IPrimaryMesh.
Exception lorsqu'une erreur d'entrée/sortie est détectée.
Definition IOException.h:32
Interface du gestionnaire de parallélisme pour un sous-domaine.
virtual Int32 commRank() const =0
Rang de cette instance dans le communicateur.
virtual IParallelMng * sequentialParallelMng()=0
Retourne un gestionnaire de parallélisme séquentiel.
virtual void recv(ArrayView< char > values, Int32 rank)=0
virtual bool isMasterIO() const =0
true si l'instance est un gestionnaire maître des entrées/sorties.
virtual Int32 commSize() const =0
Nombre d'instance dans le communicateur.
virtual Integer masterIORank() const =0
Rang de l'instance gérant les entrées/sorties (pour laquelle isMasterIO() est vrai)
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 setDimension(Integer dim)=0
Positionne la dimension du maillage (1D, 2D ou 3D).
Interface du gestionnaire de traces.
Routines des Lecture/Ecriture d'un fichier.
Definition IosFile.h:33
Groupe d'entités de maillage.
Definition ItemGroup.h:49
const String & name() const
Nom du groupe.
Definition ItemGroup.h:76
void addItems(Int32ConstArrayView items_local_id, bool check_if_present=true)
Ajoute des entités.
Definition ItemGroup.cc:445
Type d'une entité (Item).
Definition ItemTypeId.h:32
String typeName() const
Nom du type.
Gestionnaire des types d'entités de maillage.
Definition ItemTypeMng.h:66
ItemTypeInfo * typeFromId(Integer id) const
Type correspondant au numéro id.
Informations pour créer les entités d'un genre.
Informations pour créer les entités Arcane.
UniqueArray< Int64 > nodes_unique_id
UniqueId() des noeuds de ma partie.
std::unordered_map< Int64, Int32 > nodes_rank_map
Tableau associatif (uniqueId(),rang) auquel le noeud appartiendra.
UniqueArray< Real3 > nodes_coordinates
Coordonnées des noeuds de ma partie.
Lecteur de fichiers de maillage au format msh.
bool _getIsEndOfFileAndBroadcast()
Retourne true si on est à la fin du fichier.
void _readPeriodic()
Lecture des informations périodiques.
Integer _readElementsFromFile()
Lecture des éléments (mailles,faces,...)
void _readAndCheck(const String &expected_value)
Tente de lire la valeur value.
UniqueArray< Int32 > m_parts_rank
Liste des rangs qui participent à la conservation des données.
void _readEntities()
Lecture des entités.
Int32 m_nb_part
Nombre de partitions pour la lecture des noeuds et blocs.
eReturnType readMeshFromMshFile(IMesh *mesh, const String &filename, bool use_internal_partition) override
Lit le maillage contenu dans le fichier filename et le construit dans mesh.
void _readNodesFromFile()
Lecture des noeuds du maillage.
void _setNodesCoordinates()
Positionne les coordonnées des noeuds.
bool m_is_binary
Vrai si le format est binaire.
void _addCellOrNodeGroup(MshElementsBlock &block, const String &group_name, IItemFamily *family)
Ajoute des faces au groupe group_name.
void _readOneElementBlock(MshElementsBlock &block)
Lit un bloc d'entité de type 'Element'.
void _addFaceGroup(MshElementsBlock &block, const String &group_name)
Ajoute des faces au groupe group_name.
String _getNextLineAndBroadcast()
Lis la valeur de la prochaine ligne et la broadcast aux autres rangs.
Vue sur les informations des noeuds.
Exception lorsqu'une opération n'est pas supportée.
Classe gérant un vecteur de réel de dimension 3.
Definition Real3.h:132
Référence à une instance.
Tableau 1D de données avec buffer pré-alloué sur la pile.
Definition SmallArray.h:89
Vue d'un tableau d'éléments de type T.
Definition Span.h:673
constexpr __host__ __device__ SizeType size() const noexcept
Retourne la taille du tableau.
Definition Span.h:212
Vue d'un tableau d'éléments de type T.
Definition Span.h:513
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
Span< const Byte > bytes() const
Retourne la conversion de l'instance dans l'encodage UTF-8.
Definition String.cc:291
static String collapseWhiteSpace(const String &rhs)
Effectue une normalisation des caractères espaces.
Definition String.cc:452
TraceAccessor(ITraceMng *m)
Construit un accesseur via le gestionnaire de trace m.
TraceMessage info() const
Flot pour un message d'information.
TraceMessage error() const
Flot pour un message d'erreur.
TraceMessage pwarning() const
Vecteur 1D de données avec sémantique par valeur (style STL).
UniqueArray< double > m_affine_values
Liste des valeurs affines.
UniqueArray< Int64 > m_corresponding_nodes
Liste de couples (uniqueId noeud esclave, unique() noeud maître)
Informations d'un maillage issu du format 'msh'.
void resize(Int64 s)
Change le nombre d'éléments du tableau à s.
__host__ __device__ Real2 min(Real2 a, Real2 b)
Retourne le minimum de deux Real2.
Definition MathUtils.h:336
T max(const T &a, const T &b, const T &c)
Retourne le maximum de trois éléments.
Definition MathUtils.h:392
ItemGroupT< Face > FaceGroup
Groupe de faces.
Definition ItemTypes.h:178
MeshVariableScalarRefT< Node, Real3 > VariableNodeReal3
Grandeur au noeud de type coordonnées.
Int32 toInt32(Int64 v)
Converti un Int64 en un Int32.
@ ReduceSum
Somme des valeurs.
@ ReduceMin
Minimum des valeurs.
@ ReduceMax
Maximum 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...
Definition Numeric.h:212
ARCCORE_BASE_EXPORT bool isFileReadable(const String &file_name)
Vérifie que le fichier file_name est accessible et lisible.
-*- 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.
ConstArrayView< Int64 > Int64ConstArrayView
Equivalent C d'un tableau à une dimension d'entiers 64 bits.
Definition UtilsTypes.h:567
std::int16_t Int16
Type entier signé sur 16 bits.
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.
Infos d'un bloc pour $Elements pour la version 4.
bool is_built_as_cells
Indique si les entités du bloc sont des mailles.
UniqueArray< Int64 > connectivities
< Liste des uniqueId() du bloc
Real x
première composante du triplet
Definition Real3.h:35