Arcane  v3.16.0.0
Documentation développeur
Chargement...
Recherche...
Aucune correspondance
MshMeshWriter.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/* MshMeshWriter.cc (C) 2000-2025 */
9/* */
10/* Lecture/Écriture d'un fichier au format MSH. */
11/*---------------------------------------------------------------------------*/
12/*---------------------------------------------------------------------------*/
13
14#include "arcane/utils/IOException.h"
15#include "arcane/utils/FixedArray.h"
16#include "arcane/utils/Collection.h"
17#include "arcane/utils/ITraceMng.h"
18
19#include "arcane/core/FactoryService.h"
20#include "arcane/core/IMesh.h"
21#include "arcane/core/VariableTypes.h"
22#include "arcane/core/AbstractService.h"
23#include "arcane/core/IMeshWriter.h"
24#include "arcane/core/ItemTypeMng.h"
25#include "arcane/core/SharedVariable.h"
26
27#include "arcane/std/internal/IosGmsh.h"
28
29#include <tuple>
30
31/*---------------------------------------------------------------------------*/
32/*---------------------------------------------------------------------------*/
33
34namespace Arcane
35{
36
37/*---------------------------------------------------------------------------*/
38/*---------------------------------------------------------------------------*/
42class MshMeshWriter
43: public AbstractService
44, public IMeshWriter
45{
46 public:
47
48 class ItemFamilyWriteInfo
49 : public TraceAccessor
50 {
51 public:
52
53 explicit ItemFamilyWriteInfo(ITraceMng* tm)
54 : TraceAccessor(tm)
55 {
56 }
57 };
58
60 {
61 public:
62
63 Int32 m_dimension = -1;
64 Int32 m_physical_tag = -1;
65 String m_name;
66 };
67 struct EntityInfo
68 {
69 public:
70
71 EntityInfo(Int32 dim, Int32 item_type, Int32 entity_tag)
72 : m_dim(dim)
73 , m_item_type(item_type)
74 , m_entity_tag(entity_tag)
75 {
76 }
77
78 public:
79
80 void setPhysicalTag(Int32 tag, const String& name)
81 {
82 m_physical_tag = tag;
83 m_physical_tag_name = name;
84 }
85
86 public:
87
88 Int32 m_dim = -1;
89 Int32 m_item_type = IT_NullType;
90 Int32 m_entity_tag = -1;
91 Int32 m_physical_tag = -1;
92 String m_physical_tag_name;
93 };
94
96 {
97 public:
98
99 void processGroup(ItemGroup group, Int32 base_entity_index);
100
101 public:
102
103 const ItemGroup& group() const { return m_item_group; }
104 ConstArrayView<EntityInfo> entitiesByType() const { return m_entities_by_type; }
105 ConstArrayView<Int32> itemsByType(Int32 item_type) const { return m_items_by_type[item_type]; }
106
107 private:
108
109 ItemGroup m_item_group;
110 UniqueArray<EntityInfo> m_entities_by_type;
111 FixedArray<UniqueArray<Int32>, NB_BASIC_ITEM_TYPE> m_items_by_type;
112 UniqueArray<Int32> m_existing_items_type;
113 };
114
115 public:
116
117 explicit MshMeshWriter(const ServiceBuildInfo& sbi);
118
119 public:
120
121 void build() override {}
122 bool writeMeshToFile(IMesh* mesh, const String& file_name) override;
123
124 private:
125
126 ItemTypeMng* m_item_type_mng = nullptr;
127
128 private:
129
130 bool _writeMeshToFileV4(IMesh* mesh, const String& file_name);
131 Integer _convertToMshType(Int32 arcane_type);
132 std::pair<Int64, Int64> _getFamilyMinMaxUniqueId(IItemFamily* family);
133 void _addGroupsToProcess(IItemFamily* family, Array<ItemGroup>& items_groups);
134};
135
136/*---------------------------------------------------------------------------*/
137/*---------------------------------------------------------------------------*/
138
139// Obsolète. Utiliser 'MshMeshReader' à la place
140ARCANE_REGISTER_SUB_DOMAIN_FACTORY(MshMeshWriter, IMeshWriter, MshNewMeshWriter);
141
142ARCANE_REGISTER_SERVICE(MshMeshWriter,
143 ServiceProperty("MshMeshWriter", ST_SubDomain),
144 ARCANE_SERVICE_INTERFACE(IMeshWriter));
145
146/*---------------------------------------------------------------------------*/
147/*---------------------------------------------------------------------------*/
148
149MshMeshWriter::
150MshMeshWriter(const ServiceBuildInfo& sbi)
151: AbstractService(sbi)
152{
153}
154
155/*---------------------------------------------------------------------------*/
156/*---------------------------------------------------------------------------*/
163_convertToMshType(Int32 arcane_type)
164{
165 switch (arcane_type) {
166 // case (IT_NullType): return MSH_LIN_2; //case (0) is not used
167 case IT_Vertex:
168 return MSH_PNT; //printf("1-node point");
169 case IT_Cell3D_Line2:
170 case IT_Line2:
171 return MSH_LIN_2; //printf("2-node line");
172 case IT_Cell3D_Triangle3:
173 case IT_Triangle3:
174 return MSH_TRI_3; //printf("3-node triangle");
175 case IT_Cell3D_Quad4:
176 case IT_Quad4:
177 return MSH_QUA_4; //printf("4-node quadrangle");
178 case IT_Tetraedron4:
179 return MSH_TET_4; //printf("4-node tetrahedron");
180 case IT_Tetraedron10:
181 return MSH_TET_10; //printf("4-node tetrahedron");
182 case IT_Hexaedron8:
183 return MSH_HEX_8; //printf("8-node hexahedron");
184 case IT_Pentaedron6:
185 return MSH_PRI_6; //printf("6-node prism");
186 case IT_Pyramid5:
187 return MSH_PYR_5; //printf("5-node pyramid");
188 // Beneath, are some meshes that have been tried to match gmsh's ones
189 // Other 5-nodes
190 case IT_Pentagon5:
191 return MSH_PYR_5; // Could use a tag to encode these
192 case IT_HemiHexa5:
193 return MSH_PYR_5;
194 case IT_DiTetra5:
195 return MSH_PYR_5;
196 // Other 6-nodes
197 case IT_Hexagon6:
198 return MSH_PRI_6;
199 case IT_HemiHexa6:
200 return MSH_PRI_6;
201 case IT_AntiWedgeLeft6:
202 return MSH_PRI_6;
203 case IT_AntiWedgeRight6:
204 return MSH_PRI_6;
205 // Other 10-nodes
206 case IT_Heptaedron10:
207 return MSH_TRI_10;
208 // Other 12-nodes
209 case IT_Octaedron12:
210 return MSH_TRI_12;
211 // Others are still considered as default, rising an exception
212 default:
213 break;
214 }
215 ARCANE_THROW(NotSupportedException, "Arcane type '{0}'", m_item_type_mng->typeFromId(arcane_type)->typeName());
216}
217
218/*---------------------------------------------------------------------------*/
219/*---------------------------------------------------------------------------*/
220
222writeMeshToFile(IMesh* mesh, const String& file_name)
223{
224 return _writeMeshToFileV4(mesh, file_name);
225}
226
227/*---------------------------------------------------------------------------*/
228/*---------------------------------------------------------------------------*/
229
230void MshMeshWriter::ItemGroupWriteInfo::
231processGroup(ItemGroup group, Int32 base_entity_index)
232{
233 m_item_group = group;
234 String group_name = group.name();
235 bool is_all_items = group.isAllItems();
236 IItemFamily* family = group.itemFamily();
237 IMesh* mesh = family->mesh();
238 ItemTypeMng* item_type_mng = mesh->itemTypeMng();
239 ITraceMng* tm = family->traceMng();
240
241 // Pour GMSH, il faut trier les mailles par leur type (Triangle, Quadrangle, ...)
242 // On fait une entity MSH par type d'entité Arcane.
243
244 ENUMERATE_ (Item, iitem, group) {
245 Item item = *iitem;
246 Int16 item_type = item.type();
247 if (item_type >= NB_BASIC_ITEM_TYPE || item_type <= 0)
248 ARCANE_FATAL("Only pre-defined Item type are supported (current item type is '{0}')",
249 item_type_mng->typeFromId(item_type)->typeName());
250 m_items_by_type[item_type].add(item.localId());
251 }
252
253 // Conserve les types pré-définis qui ont des éléments
254 Int64 total_nb_item = 0;
255 for (Int32 i = 0; i < NB_BASIC_ITEM_TYPE; ++i) {
256 Int64 nb_type = m_items_by_type[i].size();
257 if (nb_type > 0)
258 m_existing_items_type.add(i);
259 total_nb_item += nb_type;
260 }
261
262 Int32 nb_existing_type = m_existing_items_type.size();
263 tm->info() << "NbExistingType=" << nb_existing_type;
264 for (Int32 type_index = 0; type_index < nb_existing_type; ++type_index) {
265 Int32 item_type = m_existing_items_type[type_index];
266 ItemTypeInfo* item_type_info = item_type_mng->typeFromId(item_type);
267 Int32 type_dimension = item_type_info->dimension();
268 EntityInfo entity_info(type_dimension, item_type, base_entity_index + type_index);
269 if (!is_all_items)
270 entity_info.setPhysicalTag(base_entity_index + type_index, group_name);
271 m_entities_by_type.add(entity_info);
272 }
273
274 // Si le groupe est vide, il faut quand même un tag physique pour que le
275 // groupe soit créé en lecture et ainsi en parallèle garantir que tous
276 // les sous-domaines ont les mêmes groupes.
277 if (nb_existing_type == 0 && !is_all_items) {
278 Int32 mesh_dim = mesh->dimension();
279 eItemKind ik = family->itemKind();
280 Int32 entity_dim = -1;
281 if (ik == IK_Cell)
282 entity_dim = mesh_dim;
283 else if (ik == IK_Face)
284 entity_dim = mesh_dim - 1;
285 else if (ik == IK_Edge)
286 entity_dim = mesh_dim - 2;
287 else
288 ARCANE_FATAL("Invalid item kind '{0}' for entity dimension", entity_dim);
289 // TODO: prendre un type qui correspond à la dimension
290 EntityInfo entity_info(entity_dim, IT_Tetraedron4, base_entity_index);
291 entity_info.setPhysicalTag(base_entity_index, group_name);
292 m_entities_by_type.add(entity_info);
293 }
294}
295
296/*---------------------------------------------------------------------------*/
297/*---------------------------------------------------------------------------*/
298
299void MshMeshWriter::
300_addGroupsToProcess(IItemFamily* family, Array<ItemGroup>& items_groups)
301{
302 bool has_group = false;
303 // Parcours tous les groupes de la famille
304 // NOTE: Pour l'instant, pour les mailles on suppose que cela
305 // forme une partition du maillage.
306 for (ItemGroup group : family->groups()) {
307 if (group.isAllItems())
308 continue;
309 if (group.isAutoComputed())
310 continue;
311 info() << "Processing ItemGroup group=" << group.name() << " family=" << group.itemFamily()->name();
312 items_groups.add(group);
313 has_group = true;
314 }
315 // Si pas de groupes dans la famille, on prend celui de toutes les entités
316 // si la famille est celle des mailles.
317 // TODO: toujours prendre ce groupe pour les entités qui n'auraient pas
318 // été traitées par les autres groupes lorsqu'on n'est pas sur
319 // que l'ensemble des groupes forme une partition.
320 if (!has_group && (family->itemKind() == IK_Cell))
321 items_groups.add(family->allItems());
322}
323
324/*---------------------------------------------------------------------------*/
325/*---------------------------------------------------------------------------*/
335_writeMeshToFileV4(IMesh* mesh, const String& file_name)
336{
337 m_item_type_mng = mesh->itemTypeMng();
338 String mesh_file_name(file_name);
339 if (!file_name.endsWith(".msh"))
340 mesh_file_name = mesh_file_name + ".msh";
341 std::ofstream ofile(mesh_file_name.localstr());
342 ofile.precision(20);
343 if (!ofile)
344 ARCANE_THROW(IOException, "Unable to open file '{0}' for writing", mesh_file_name);
345
346 info() << "writing file '" << mesh_file_name << "'";
347
348 ofile << "$MeshFormat\n";
349 // 4.1 pour le format
350 // 0 pour ASCII (1 pour binaire)
351 // 8 pour sizeof(size_t)
352 ofile << "4.1 0 " << sizeof(size_t) << "\n";
353 ofile << "$EndMeshFormat\n";
354
355 IItemFamily* cell_family = mesh->cellFamily();
356 IItemFamily* face_family = mesh->faceFamily();
357 CellGroup all_cells = mesh->allCells();
358 const Int32 mesh_nb_node = mesh->nbNode();
359 IItemFamily* node_family = mesh->nodeFamily();
360 ItemGroup all_nodes = mesh->allNodes();
361
362 UniqueArray<ItemGroup> items_groups;
363 _addGroupsToProcess(cell_family, items_groups);
364 _addGroupsToProcess(face_family, items_groups);
365
366 std::vector<std::unique_ptr<ItemGroupWriteInfo>> groups_write_info_list;
367 Int32 base_entity_index = 1000;
368 for (ItemGroup group : items_groups) {
369 auto x(std::make_unique<ItemGroupWriteInfo>());
370 x->processGroup(group, base_entity_index);
371 groups_write_info_list.emplace_back(std::move(x));
372 base_entity_index += 1000;
373 }
374
375 // Pour GMSH, il faut commencer par les 'Entities'.
376 // Il faut une entité par type de maille.
377 // On commence donc par calculer les types de mailles.
378 // Pour les maillages non-manifold les mailles peuvent être de dimension
379 // différentes
380
381 // Liste des tags physiques
382 UniqueArray<PhysicalTagInfo> physical_tags;
383
384 // Calcule le nombre d'entités par dimension
385 FixedArray<Int32, 4> nb_entities_by_dim;
386
387 // Calcule le nombre total d'éléments.
388 // Toutes les entités qui ne sont pas de dimension 0 sont des éléments.
389 Int64 total_nb_cell = 0;
390 for (const auto& ginfo : groups_write_info_list) {
391 for (const EntityInfo& entity_info : ginfo->entitiesByType()) {
392 Int32 dim = entity_info.m_dim;
393 if (dim >= 0)
394 ++nb_entities_by_dim[dim];
395 if (dim > 0) {
396 Int32 item_type = entity_info.m_item_type;
397 Int32 nb_item = ginfo->itemsByType(item_type).size();
398 total_nb_cell += nb_item;
399
400 Int32 physical_tag = entity_info.m_physical_tag;
401 if (physical_tag > 0) {
402 physical_tags.add(PhysicalTagInfo{ dim, physical_tag, entity_info.m_physical_tag_name });
403 }
404 }
405 }
406 }
407
408 // $PhysicalNames // same as MSH version 2
409 // numPhysicalNames(ASCII int)
410 // dimension(ASCII int) physicalTag(ASCII int) "name"(127 characters max)
411 // ...
412 // $EndPhysicalNames
413
414 {
415 ofile << "$PhysicalNames\n";
416 Int32 nb_tag = physical_tags.size();
417 ofile << nb_tag << "\n";
418 for (const PhysicalTagInfo& tag_info : physical_tags) {
419 // TODO: vérifier que le nom ne dépasse pas 127 caractères.
420 ofile << tag_info.m_dimension << " " << tag_info.m_physical_tag << " " << tag_info.m_name << "\n";
421 }
422 ofile << "$EndPhysicalNames\n";
423 }
424
425 // Calcule la bounding box des noeuds
426 VariableNodeReal3 nodes_coords = mesh->nodesCoordinates();
427 Real3 node_min_bounding_box;
428 Real3 node_max_bounding_box;
429 {
430 Real max_value = FloatInfo<Real>::maxValue();
431 Real min_value = -max_value;
432 Real3 min_box(max_value, max_value, max_value);
433 Real3 max_box(min_value, min_value, min_value);
434 ENUMERATE_ (Node, inode, all_nodes) {
435 Real3 pos = nodes_coords[inode];
436 min_box = math::min(min_box, pos);
437 max_box = math::max(max_box, pos);
438 }
439 node_min_bounding_box = min_box;
440 node_max_bounding_box = max_box;
441 }
442
443 // $Entities
444 // numPoints(size_t) numCurves(size_t)
445 // numSurfaces(size_t) numVolumes(size_t)
446 // pointTag(int) X(double) Y(double) Z(double)
447 // numPhysicalTags(size_t) physicalTag(int) ...
448 // ...
449 // curveTag(int) minX(double) minY(double) minZ(double)
450 // maxX(double) maxY(double) maxZ(double)
451 // numPhysicalTags(size_t) physicalTag(int) ...
452 // numBoundingPoints(size_t) pointTag(int; sign encodes orientation) ...
453 // ...
454 // surfaceTag(int) minX(double) minY(double) minZ(double)
455 // maxX(double) maxY(double) maxZ(double)
456 // numPhysicalTags(size_t) physicalTag(int) ...
457 // numBoundingCurves(size_t) curveTag(int; sign encodes orientation) ...
458 // ...
459 // volumeTag(int) minX(double) minY(double) minZ(double)
460 // maxX(double) maxY(double) maxZ(double)
461 // numPhysicalTags(size_t) physicalTag(int) ...
462 // numBoundngSurfaces(size_t) surfaceTag(int; sign encodes orientation) ...
463 // ...
464 // $EndEntities
465 {
466 ofile << "$Entities\n";
467 //nb_entities_by_dim[0] = 0;
468 ofile << nb_entities_by_dim[0] << " " << nb_entities_by_dim[1]
469 << " " << nb_entities_by_dim[2] << " " << nb_entities_by_dim[3] << "\n";
470 for (Int32 idim = 1; idim < 4; ++idim) {
471 for (const auto& ginfo : groups_write_info_list) {
472 for (const EntityInfo& entity_info : ginfo->entitiesByType()) {
473 if (entity_info.m_dim != idim)
474 continue;
475 ofile << entity_info.m_entity_tag << " " << node_min_bounding_box.x << " " << node_min_bounding_box.y << " " << node_min_bounding_box.z
476 << " " << node_max_bounding_box.x << " " << node_max_bounding_box.y << " " << node_max_bounding_box.z;
477 // Pas de tag pour l'instant
478 Int32 physical_tag = entity_info.m_physical_tag;
479 if (physical_tag > 0) {
480 ofile << " 1 " << physical_tag;
481 }
482 else
483 ofile << " 0";
484 // Pas de boundary pour l'instant
485 ofile << " 0";
486 ofile << "\n";
487 }
488 }
489 }
490 ofile << "$EndEntities\n";
491 }
492
493 // $Nodes
494 // numEntityBlocks(size_t) numNodes(size_t)
495 // minNodeTag(size_t) maxNodeTag(size_t)
496 // entityDim(int) entityTag(int) parametric(int; 0 or 1)
497 // numNodesInBlock(size_t)
498 // nodeTag(size_t)
499 // ...
500 // x(double) y(double) z(double)
501 // < u(double; if parametric and entityDim >= 1) >
502 // < v(double; if parametric and entityDim >= 2) >
503 // < w(double; if parametric and entityDim == 3) >
504 // ...
505 // ...
506 // $EndNodes
507
508 // Bloc contenant les noeuds
509 ofile << "$Nodes\n";
510
511 auto [node_min_uid, node_max_uid] = _getFamilyMinMaxUniqueId(node_family);
512
513 ofile << "1 " << mesh_nb_node << " " << node_min_uid << " " << node_max_uid << "\n";
514 // entityDim(int) entityTag(int) parametric(int; 0 or 1) numNodesInBlock(size_t)
515 ofile << "0 " << "100 " << "0 " << mesh_nb_node << "\n";
516
517 // Sauve les uniqueId() des noeuds
518 ENUMERATE_ (Node, inode, all_nodes) {
519 Int64 uid = inode->uniqueId();
520 ofile << uid << "\n";
521 }
522
523 // Sauve les coordonnées
524 ENUMERATE_ (Node, inode, all_nodes) {
525 Real3 coord = nodes_coords[inode];
526 ofile << coord.x << " " << coord.y << " " << coord.z << "\n";
527 }
528
529 ofile << "$EndNodes\n";
530
531 // TODO: regarder s'il faut prendre en compte les uniqueId() des faces.
532 auto [cell_min_uid, cell_max_uid] = _getFamilyMinMaxUniqueId(cell_family);
533
534 // $Elements
535 // numEntityBlocks(size_t) numElements(size_t)
536 // minElementTag(size_t) maxElementTag(size_t)
537 // entityDim(int) entityTag(int) elementType(int; see below)
538 // numElementsInBlock(size_t)
539 // elementTag(size_t) nodeTag(size_t) ...
540 // ...
541 // ...
542 // $EndElements
543
544 // Bloc contenant les mailles
545 ofile << "$Elements\n";
546
547 Int32 nb_existing_type = nb_entities_by_dim[1] + nb_entities_by_dim[2] + nb_entities_by_dim[3];
548 ofile << nb_existing_type << " " << total_nb_cell << " " << cell_min_uid << " " << cell_max_uid << "\n";
549 for (const auto& ginfo : groups_write_info_list) {
550 ItemGroup item_group = ginfo->group();
551 IItemFamily* item_family = item_group.itemFamily();
552 for (const EntityInfo& entity_info : ginfo->entitiesByType()) {
553 Int32 cell_type = entity_info.m_item_type;
554 ConstArrayView<Int32> items_of_current_type = ginfo->itemsByType(cell_type);
555 ItemTypeInfo* item_type_info = m_item_type_mng->typeFromId(cell_type);
556 Int32 type_dimension = entity_info.m_dim;
557 ofile << "\n";
558 ofile << type_dimension << " " << entity_info.m_entity_tag << " " << _convertToMshType(cell_type)
559 << " " << items_of_current_type.size() << "\n";
560 info() << "Writing items family=" << item_family->name() << " type=" << item_type_info->typeName()
561 << " n=" << items_of_current_type.size()
562 << " dimension=" << type_dimension;
563 Int32 nb_node_for_type = item_type_info->nbLocalNode();
564 ENUMERATE_ (ItemWithNodes, iitem, item_family->view(items_of_current_type)) {
565 ItemWithNodes item = *iitem;
566 ofile << item.uniqueId();
567 for (Int32 i = 0; i < nb_node_for_type; ++i)
568 ofile << " " << item.node(i).uniqueId();
569 ofile << "\n";
570 }
571 }
572 }
573 ofile << "$EndElements\n";
574
575 return false;
576}
577
578/*---------------------------------------------------------------------------*/
579/*---------------------------------------------------------------------------*/
580
581std::pair<Int64, Int64> MshMeshWriter::
582_getFamilyMinMaxUniqueId(IItemFamily* family)
583{
584 Int64 min_uid = INT64_MAX;
585 Int64 max_uid = -1;
586 ENUMERATE_ (Item, iitem, family->allItems()) {
587 Item item = *iitem;
588 Int64 uid = item.uniqueId();
589 if (uid < min_uid)
590 min_uid = uid;
591 if (uid > max_uid)
592 max_uid = uid;
593 }
594 return { min_uid, max_uid };
595}
596
597/*---------------------------------------------------------------------------*/
598/*---------------------------------------------------------------------------*/
599
600} // End namespace Arcane
601
602/*---------------------------------------------------------------------------*/
603/*---------------------------------------------------------------------------*/
#define ARCANE_THROW(exception_class,...)
Macro pour envoyer une exception avec formattage.
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
#define ENUMERATE_(type, name, group)
Enumérateur générique d'un groupe d'entité
#define ARCANE_REGISTER_SUB_DOMAIN_FACTORY(aclass, ainterface, aname)
Enregistre un service de fabrique pour la classe aclass.
#define ARCANE_SERVICE_INTERFACE(ainterface)
Macro pour déclarer une interface lors de l'enregistrement d'un service.
Integer size() const
Nombre d'éléments du vecteur.
AbstractService(const ServiceBuildInfo &)
Constructeur à partir d'un ServiceBuildInfo.
Tableau d'items de types quelconques.
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
Informations sur le type flottant.
Definition Limits.h:48
Interface d'une famille d'entités.
Definition IItemFamily.h:84
virtual ItemGroup allItems() const =0
Groupe de toutes les entités.
virtual String name() const =0
Nom de la famille.
virtual eItemKind itemKind() const =0
Genre des entités.
virtual ITraceMng * traceMng() const =0
Gestionnaire de trace associé
virtual IMesh * mesh() const =0
Maillage associé
virtual ItemVectorView view(Int32ConstArrayView local_ids)=0
Vue sur les entités.
virtual Integer dimension()=0
Dimension du maillage (1D, 2D ou 3D).
Interface d'un service d'écriture d'un maillage.
Definition IMeshWriter.h:36
Exception lorsqu'une erreur d'entrée/sortie est détectée.
Definition IOException.h:32
Interface du gestionnaire de traces.
virtual TraceMessage info()=0
Flot pour un message d'information.
Groupe d'entités de maillage.
Definition ItemGroup.h:49
const String & name() const
Nom du groupe.
Definition ItemGroup.h:76
bool isAutoComputed() const
Vrai s'il s'agit d'un groupe calculé automatiquement.
Definition ItemGroup.cc:627
IItemFamily * itemFamily() const
Famille d'entité à laquelle appartient ce groupe (0 pour le group nul)
Definition ItemGroup.h:123
bool isAllItems() const
Indique si le groupe est celui de toutes les entités.
Definition ItemGroup.cc:609
Infos sur un type d'entité du maillage.
Int16 dimension() const
Dimension de l'élément (<0 si inconnu)
Integer nbLocalNode() const
Nombre de noeuds de l'entité
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.
Elément de maillage s'appuyant sur des noeuds (Edge,Face,Cell).
Definition Item.h:724
Node node(Int32 i) const
i-ème noeud de l'entité
Definition Item.h:779
Classe de base d'un élément de maillage.
Definition Item.h:83
constexpr Int32 localId() const
Identifiant local de l'entité dans le sous-domaine du processeur.
Definition Item.h:219
ItemUniqueId uniqueId() const
Identifiant unique sur tous les domaines.
Definition Item.h:225
Int16 type() const
Type de l'entité
Definition Item.h:241
Integer _convertToMshType(Int32 arcane_type)
Converti le type Arcane en le type GMSH.
bool _writeMeshToFileV4(IMesh *mesh, const String &file_name)
Ecrit au format MSH V4.
bool writeMeshToFile(IMesh *mesh, const String &file_name) override
Ecrit un maillage sur un fichier.
void build() override
Construction de niveau build du service.
Noeud d'un maillage.
Definition Item.h:573
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
Structure contenant les informations pour créer un service.
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
bool endsWith(const String &s) const
Indique si la chaîne se termine par les caractères de s.
Definition String.cc:1083
TraceAccessor(ITraceMng *m)
Construit un accesseur via le gestionnaire de trace m.
TraceMessage info() const
Flot pour un message d'information.
Vecteur 1D de données avec sémantique par valeur (style STL).
__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< Cell > CellGroup
Groupe de mailles.
Definition ItemTypes.h:183
#define ARCANE_REGISTER_SERVICE(aclass, a_service_property,...)
Macro pour enregistrer un service.
MeshVariableScalarRefT< Node, Real3 > VariableNodeReal3
Grandeur au noeud de type coordonnées.
-*- 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.
@ ST_SubDomain
Le service s'utilise au niveau du sous-domaine.
eItemKind
Genre d'entité de maillage.
@ IK_Cell
Entité de maillage de genre maille.
@ IK_Face
Entité de maillage de genre face.
@ IK_Edge
Entité de maillage de genre arête.
std::int16_t Int16
Type entier signé sur 16 bits.
double Real
Type représentant un réel.
std::int32_t Int32
Type entier signé sur 32 bits.
Real y
deuxième composante du triplet
Definition Real3.h:36
Real z
troisième composante du triplet
Definition Real3.h:37
Real x
première composante du triplet
Definition Real3.h:35