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"
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"
39#include "arcane/std/internal/IosFile.h"
40#include "arcane/std/internal/IosGmsh.h"
79class MshParallelMeshReader
112 Int64 entity_tag = -1;
130 items_infos.add(type_id);
131 items_infos.add(unique_id);
132 items_infos.addRange(nodes_uid);
160 explicit MshParallelMeshReader(
ITraceMng* tm)
168 IMesh* m_mesh =
nullptr;
170 Int32 m_master_io_rank = A_NULL_RANK;
171 bool m_is_parallel =
false;
174 MshMeshAllocateInfo m_mesh_allocate_info;
185 void _readNodesOneEntity(
Int32 entity_index);
187 void _readMeshFromFile();
190 void _allocateCells();
191 void _allocateGroups();
199 void _readPhysicalNames();
202 void _readOneEntity(
Int32 entity_dim,
Int32 entity_index_in_dim);
205 Int32 _getIntegerAndBroadcast();
206 Int64 _getInt64AndBroadcast();
216 void _goToNextLine();
231 inline std::pair<Int64, Int64>
234 Int64 isize = size / nb_interval;
235 Int64 ibegin = index * isize;
237 if ((index + 1) == nb_interval)
238 isize = size - ibegin;
239 return { ibegin, isize };
254 s = f->getNextLine();
258 info() <<
"BroadcastNextLine: " << s;
259 m_parallel_mng->broadcastString(s, m_master_io_rank);
261 info() <<
"GetNextLine: " << s;
274 Int32 is_end_int = 0;
276 is_end_int = f->isEnd() ? 1 : 0;
277 info() <<
"IsEndOfFile_Master: " << is_end_int;
281 info() <<
"IsEndOfFile: " << is_end_int;
282 m_parallel_mng->broadcast(
ArrayView<Int32>(1, &is_end_int), m_master_io_rank);
284 bool is_end = (is_end_int != 0);
285 info() <<
"IsEnd: " << is_end;
294_getIntegerAndBroadcast()
299 v[0] = f->getInteger();
301 m_parallel_mng->broadcast(v.
view(), m_master_io_rank);
309Int64 MshParallelMeshReader::
310_getInt64AndBroadcast()
312 IosFile* f = m_ios_file.get();
313 FixedArray<Int64, 1> v;
317 m_parallel_mng->broadcast(v.view(), m_master_io_rank);
325void MshParallelMeshReader::
328 IosFile* f = m_ios_file.get();
331 f->binaryRead(values);
334 for (
Int64& v : values)
339 m_parallel_mng->broadcast(values, m_master_io_rank);
345void MshParallelMeshReader::
348 IosFile* f = m_ios_file.get();
351 f->binaryRead(values);
354 for (
Int32& v : values)
359 m_parallel_mng->broadcast(values, m_master_io_rank);
365void MshParallelMeshReader::
368 IosFile* f = m_ios_file.get();
371 f->binaryRead(values);
374 for (
double& v : values)
379 m_parallel_mng->broadcast(values, m_master_io_rank);
385Real3 MshParallelMeshReader::
388 IosFile* f = m_ios_file.get();
392 f->binaryRead(SmallSpan<Real3>(&v, 1));
395 Real x = f->getReal();
396 Real y = f->getReal();
397 Real z = f->getReal();
406Int32 MshParallelMeshReader::
409 IosFile* f = m_ios_file.get();
413 f->binaryRead(SmallSpan<Int32>(&v, 1));
422Int64 MshParallelMeshReader::
425 IosFile* f = m_ios_file.get();
429 f->binaryRead(SmallSpan<Int64>(&v, 1));
438void MshParallelMeshReader::
441 if (m_ios_file.get())
442 m_ios_file->getNextLine();
448Int16 MshParallelMeshReader::
449_switchMshType(
Int64 mshElemType,
Int32& nNodes)
const
451 switch (mshElemType) {
457 info() <<
"Could not decode IT_NullType with nNodes=" << nNodes;
458 throw IOException(
"_convertToMshType",
"Could not decode IT_NullType with nNodes");
475 return IT_Tetraedron4;
478 return IT_Hexaedron8;
481 return IT_Pentaedron6;
487 return IT_Heptaedron10;
490 return IT_Octaedron12;
496 return IT_Tetraedron10;
510 ARCANE_THROW(NotSupportedException,
"Unknown GMSH element type '{0}'", mshElemType);
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) {
541 Real min_x = min_box.
x;
542 Real max_x = max_box.
x;
546 info() <<
"MIN_MAX_X=" << 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;
560 for (
Int64 i = 0; i < nb_node; ++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));
567 info() <<
"NB_NODE_PER_RANK=" << nb_node_per_rank;
600 _getInt64ArrayAndBroadcast(nodes_info.
view());
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];
610 if (total_nb_node < 0)
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
618 <<
" nb_rank=" << m_parallel_mng->commSize();
620 for (
Integer i_entity = 0; i_entity < nb_entity; ++i_entity) {
621 _readNodesOneEntity(i_entity);
633void MshParallelMeshReader::
634_readNodesOneEntity(
Int32 entity_index)
636 IosFile* ios_file = m_ios_file.get();
644 _getInt32ArrayAndBroadcast(entity_infos.
view());
645 Int64 nb_node2 = _getInt64AndBroadcast();
651 [[maybe_unused]]
Int32 entity_dim = entity_infos[0];
653 [[maybe_unused]]
Int32 entity_tag = entity_infos[1];
654 Int32 parametric_coordinates = entity_infos[2];
657 info() <<
"[Nodes] index=" << entity_index <<
" entity_dim=" << entity_dim <<
" entity_tag=" << entity_tag
658 <<
" parametric=" << parametric_coordinates
659 <<
" nb_node2=" << nb_node2;
661 if (parametric_coordinates != 0)
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);
684 ios_file->binaryRead(nodes_uids.
view());
687 for (
Integer i = 0; i < nb_to_read; ++i) {
689 nodes_uids[i] = ios_file->getInt64();
693 if (dest_rank != m_master_io_rank) {
694 pm->send(nodes_uids, dest_rank);
697 else if (my_rank == dest_rank) {
698 pm->
recv(nodes_uids, m_master_io_rank);
702 if (my_rank == dest_rank) {
703 m_mesh_allocate_info.nodes_unique_id.addRange(nodes_uids);
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);
720 ios_file->binaryRead(nodes_coordinates.
view());
723 for (
Integer i = 0; i < nb_to_read; ++i) {
724 nodes_coordinates[i] = _getReal3();
728 if (dest_rank != m_master_io_rank) {
729 pm->send(nodes_coordinates, dest_rank);
732 else if (my_rank == dest_rank) {
733 pm->
recv(nodes_coordinates, m_master_io_rank);
737 if (my_rank == dest_rank) {
738 m_mesh_allocate_info.nodes_coordinates.addRange(nodes_coordinates);
754 IosFile* ios_file = m_ios_file.get();
760 info() <<
"Reading block nb_entity=" << nb_entity_in_block <<
" item_nb_node=" << item_nb_node;
768 const Int64 nb_to_read = _interval(i_part,
m_nb_part, nb_entity_in_block).second;
771 info() <<
"Reading block part i_part=" << i_part
772 <<
" nb_to_read=" << nb_to_read <<
" dest_rank=" << dest_rank;
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) {
778 connectivities.
resize(nb_connectivity);
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());
786 for (
Int64 i = 0; i < nb_uid; ++i) {
787 Int64 item_unique_id = tmp_uid_and_connectivities[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];
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();
805 if (dest_rank != m_master_io_rank) {
806 pm->send(uids, dest_rank);
807 pm->send(connectivities, dest_rank);
810 else if (my_rank == dest_rank) {
811 pm->
recv(uids, m_master_io_rank);
812 pm->
recv(connectivities, m_master_io_rank);
814 if (my_rank == dest_rank) {
854 IosFile* ios_file = m_ios_file.get();
858 _getInt64ArrayAndBroadcast(elements_info.
view());
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];
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;
873 if (number_of_elements < 0)
891 _getInt32ArrayAndBroadcast(block_info.
view());
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();
899 ItemTypeId item_type(_switchMshType(entity_type, item_nb_node));
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;
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;
912 if (entity_type == MSH_PNT) {
917 Int64 item_unique_id = NULL_ITEM_UNIQUE_ID;
919 [[maybe_unused]]
Int64 unused_id = _getInt64();
920 item_unique_id = _getInt64();
921 info() <<
"Adding unique node uid=" << item_unique_id;
925 block.uids.add(item_unique_id);
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)
946 info() <<
"Computed mesh dimension = " << mesh_dimension;
948 bool allow_multi_dim_cell = m_mesh->meshKind().isNonManifold();
949 bool use_experimental_type_for_cell =
false;
950 if (allow_multi_dim_cell) {
952 use_experimental_type_for_cell =
true;
954 use_experimental_type_for_cell = (v.value() != 0);
956 info() <<
"Use experimental cell type?=" << use_experimental_type_for_cell;
957 ItemTypeMng* item_type_mng = m_mesh->itemTypeMng();
959 const Int32 block_dim = block.dimension;
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) {
967 bool use_sub_dim_cell =
false;
968 if (mesh_dimension == 3 && (block_dim == 2 || block_dim == 1))
970 use_sub_dim_cell =
true;
971 else if (mesh_dimension == 2 && block_dim == 1)
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) {
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);
987 ARCANE_FATAL(
"Not supported sub dimension cell type={0} for 3D mesh", item_type_name);
989 else if (mesh_dimension == 2) {
990 if (block.item_type == IT_Line2)
993 ARCANE_FATAL(
"Not supported sub dimension cell type={0} for 2D mesh", item_type_name);
995 block.is_built_as_cells =
true;
996 _computeOwnItems(block, m_mesh_allocate_info.cells_infos,
false);
1001 return mesh_dimension;
1015 if (my_rank != dest_rank) {
1016 work_values.
resize(size);
1017 view = work_values.
view();
1019 pm->broadcast(view, dest_rank);
1033 const Int32 my_rank = pm->commRank();
1036 if (dest_rank == my_rank)
1037 size = values.size();
1039 return _broadcastArrayWithSize(pm, values, work_values, dest_rank, size);
1047void MshParallelMeshReader::
1048_computeOwnItems(MshElementsBlock& block, MshItemKindInfo& item_kind_info,
bool is_generate_uid)
1052 IParallelMng* pm = m_parallel_mng;
1053 const Int32 my_rank = pm->commRank();
1055 const ItemTypeId item_type = block.item_type;
1056 const Int32 item_nb_node = block.item_nb_node;
1058 UniqueArray<Int64> connectivities;
1059 UniqueArray<Int64> uids;
1060 UniqueArray<Int32> nodes_rank;
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) {
1067 ArrayView<Int64> connectivities_view = _broadcastArray(pm, block.connectivities, connectivities, dest_rank);
1068 ArrayView<Int64> uids_view = _broadcastArray(pm, block.uids, uids, dest_rank);
1070 Int32 nb_item = uids_view.size();
1071 nodes_rank.resize(nb_item);
1072 nodes_rank.fill(-1);
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())
1083 Int32 rank = x->second;
1084 nodes_rank[i] = rank;
1087 for (
Int32 i = 0; i < nb_item; ++i) {
1088 const Int32 rank = nodes_rank[i];
1089 if (rank != my_rank)
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)
1098 uid = NULL_ITEM_UNIQUE_ID;
1099 item_kind_info.addItem(item_type, uid, v);
1129 const IItemFamily* node_family = m_mesh->nodeFamily();
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);
1137 local_ids.
resize(nb_item);
1143 for (
Int32 i = 0; i < nb_item; ++i) {
1144 NodeLocalId nid(local_ids[i]);
1146 nodes_coord_var[nid] = coords[i];
1154void MshParallelMeshReader::
1158 Integer nb_item = m_mesh_allocate_info.faces_infos.nb_item;
1159 info() <<
"Adding faces direct nb_face=" << nb_item;
1161 mesh->modifier()->addFaces(nb_item, m_mesh_allocate_info.faces_infos.items_infos);
1167void MshParallelMeshReader::
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;
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 ##";
1183 if (m_mesh->
meshKind().isNonManifold())
1185 pmesh->endAllocate();
1186 info() <<
"## Done ##";
1192void MshParallelMeshReader::
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;
1201 entity_list.clear();
1202 physical_name_list.clear();
1203 Int32 block_index = block.index;
1204 Int32 block_dim = block.dimension;
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";
1212 if (block_dim == 0) {
1213 const MshEntitiesNodes* entity = m_mesh_info->findNodeEntities(block_entity_tag);
1215 info(5) <<
"[Groups] Skipping block index=" << block_index
1216 <<
" because entity tag is invalid";
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);
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);
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";
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) {
1243 else if (block_dim == face_dim) {
1267 _addFaceGroupOnePart(connectivities_view, item_nb_node, group_name, block.
index);
1274void MshParallelMeshReader::
1279 const Int32 nb_entity = connectivities.
size() / item_nb_node;
1283 FaceGroup face_group =
mesh->faceFamily()->findGroup(group_name,
true);
1288 const Int32 face_nb_node = nb_entity * item_nb_node;
1293 Integer faces_nodes_unique_id_index = 0;
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];
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;
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);
1330 if (!m_is_parallel) {
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());
1341 faces_id.
add(face.localId());
1344 faces_nodes_unique_id_index += n;
1346 info(4) <<
"Adding " << faces_id.
size() <<
" faces from block index=" << block_index
1347 <<
" to group '" << face_group.
name() <<
"'";
1363 ArrayView<Int64> uids_view = _broadcastArray(pm, block.uids, uids, dest_rank);
1364 _addCellOrNodeGroupOnePart(uids_view, group_name, block.
index, family);
1371void MshParallelMeshReader::
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);
1393 info() <<
"Adding " << items_lid.size() <<
" items from block index=" << block_index
1394 <<
" to group '" << group_name <<
"' for family=" << family->
name();
1408void MshParallelMeshReader::
1412 String quote_mark =
"\"";
1413 Int32 nb_name = _getIntegerAndBroadcast();
1414 info() <<
"nb_physical_name=" << nb_name;
1418 for (
Int32 i = 0; i < nb_name; ++i) {
1419 Int32 dim = _getIntegerAndBroadcast();
1420 Int32 tag = _getIntegerAndBroadcast();
1422 if (dim < 0 || dim > 3)
1423 ARCANE_FATAL(
"Invalid value for physical name dimension dim={0}", dim);
1427 if (s.startsWith(quote_mark))
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 <<
"'";
1436 if (s !=
"$EndPhysicalNames")
1437 ARCANE_FATAL(
"found '{0}' and expected '$EndPhysicalNames'", s);
1475 IosFile* ios_file = m_ios_file.get();
1478 _getInt64ArrayAndBroadcast(nb_dim_item.
view());
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];
1488 for (
Int64 i = 0; i < nb_dim_item[0]; ++i) {
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);
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;
1504 tag_info[1] = physical_tag;
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]));
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);
1526void MshParallelMeshReader::
1527_readOneEntity(
Int32 entity_dim,
Int32 entity_index_in_dim)
1529 IosFile* ios_file = m_ios_file.get();
1542 info() <<
"[Entities] Reading entity dim=" << entity_dim <<
" index_in_dim=" << entity_index_in_dim;
1544 dim_and_tag_info.
add(entity_dim);
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;
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;
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;
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);
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));
1612 Int64 nb_link = _getInt64AndBroadcast();
1613 info() <<
"[Periodic] nb_link=" << nb_link;
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];
1623 _getInt32ArrayAndBroadcast(entity_info.
view());
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];
1631 Int64 num_affine = _getInt64AndBroadcast();
1632 info() <<
"[Periodic] num_affine=" << num_affine;
1636 info() <<
"[Periodic] nb_corresponding_node=" << one_info.m_nb_corresponding_node;
1645 if (s !=
"$EndPeriodic")
1646 ARCANE_FATAL(
"found '{0}' and expected '$EndPeriodic'", s);
1659 constexpr Int32 MAX_SIZE = 128;
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();
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);
1680 if (s != expected_value)
1681 ARCANE_FATAL(
"found '{0}' and expected '{1}'", s, expected_value);
1688void MshParallelMeshReader::
1691 IosFile* ios_file = m_ios_file.get();
1693 info() <<
"Reading 'msh' file in parallel";
1695 const int MSH_BINARY_TYPE = 1;
1698 Real version = ios_file->getReal();
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();
1702 if (file_type == MSH_BINARY_TYPE)
1706 pwarning() <<
"MSH reader for binary format is experimental";
1707 Int32 data_size = ios_file->getInteger();
1708 ARCANE_UNUSED(data_size);
1710 ARCANE_FATAL(
"Only 'size_t' of size '8' is allowed (current size is '{0}')", data_size);
1715 Int32 int_value_one = 0;
1717 if (int_value_one != 1)
1718 ARCANE_FATAL(
"Bad endianess for file. Read int as value '{0}' (expected=1)", int_value_one);
1721 ios_file->getNextLine();
1724 if (!ios_file->lookForString(
"$EndMeshFormat"))
1734 if (next_line ==
"$PhysicalNames") {
1735 _readPhysicalNames();
1741 if (next_line ==
"$Entities") {
1746 if (next_line !=
"$Nodes")
1747 ARCANE_THROW(IOException,
"Unexpected string '{0}'. Valid values are '$Nodes'", next_line);
1753 if (ios_file && !ios_file->lookForString(
"$EndNodes"))
1757 if (ios_file && !ios_file->lookForString(
"$Elements"))
1763 if (ios_file && !ios_file->lookForString(
"$EndElements"))
1766 info() <<
"Computed mesh dimension = " << mesh_dimension;
1771 info() <<
"NextLine=" << next_line;
1777 if (next_line ==
"$Periodic") {
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;
1801 if (!use_internal_partition)
1803 m_parallel_mng = pm;
1824 m_master_io_rank = master_io_rank;
1830 info() <<
"Is file readable ?=" << is_readable;
1831 file_readable[0] = is_readable ? 1 : 0;
1833 error() <<
"Unable to read file '" << filename <<
"'";
1835 pm->broadcast(file_readable.
view(), master_io_rank);
1836 if (file_readable[0] == 0) {
1840 std::ifstream ifile;
1846 m_ios_file = ios_file;
1848 if (IosFile::isEqualString(mesh_format_str,
"$MeshFormat")) {
1849 _readMeshFromFile();
1850 if (!use_internal_partition) {
1851 info() <<
"Synchronize groups and variables";
1852 mesh->synchronizeGroupsAndVariables();
1857 info() <<
"The file does not begin with '$MeshFormat' returning RTError";
1865createMshParallelMeshReader(
ITraceMng* tm)
#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.
constexpr __host__ __device__ ArrayView< T > view()
Vue modifiable sur le tableau.
constexpr __host__ __device__ SmallSpan< T, NbElement > span()
Vue modifiable sur le tableau.
Informations sur le type flottant.
Interface d'une famille d'entités.
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.
@ RTError
Erreur lors de l'opération.
@ RTOk
Opération effectuée avec succès.
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.
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.
Groupe d'entités de maillage.
const String & name() const
Nom du groupe.
void addItems(Int32ConstArrayView items_local_id, bool check_if_present=true)
Ajoute des entités.
Type d'une entité (Item).
String typeName() const
Nom du type.
Gestionnaire des types d'entités de maillage.
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 _computeNodesPartition()
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.
Référence à une instance.
Tableau 1D de données avec buffer pré-alloué sur la pile.
Vue d'un tableau d'éléments de type T.
constexpr __host__ __device__ SizeType size() const noexcept
Retourne la taille du tableau.
Vue d'un tableau d'éléments de type T.
Chaîne de caractères unicode.
const char * localstr() const
Retourne la conversion de l'instance dans l'encodage UTF-8.
Span< const Byte > bytes() const
Retourne la conversion de l'instance dans l'encodage UTF-8.
static String collapseWhiteSpace(const String &rhs)
Effectue une normalisation des caractères espaces.
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).
Infos pour les entités 0D.
Infos pour les entités 1D, 2D et 3D.
Informations sur la périodicité
UniqueArray< double > m_affine_values
Liste des valeurs affines.
UniqueArray< Int64 > m_corresponding_nodes
Liste de couples (uniqueId noeud esclave, unique() noeud maître)
Infos du bloc '$PhysicalNames'.
Infos sur un nom physique.
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.
T max(const T &a, const T &b, const T &c)
Retourne le maximum de trois éléments.
ItemGroupT< Face > FaceGroup
Groupe de faces.
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...
-*- 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.
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.
Int64 nb_entity
Nombre d'entités du bloc.
ItemTypeId item_type
Type Arcane de l'entité
bool is_built_as_cells
Indique si les entités du bloc sont des mailles.
Int32 item_nb_node
Nombre de noeuds de l'entité.
Int32 index
Index du bloc dans la liste.
Int32 dimension
Dimension de l'entité
UniqueArray< Int64 > connectivities
< Liste des uniqueId() du bloc
Real x
première composante du triplet