14#include "arcane/utils/IOException.h"
15#include "arcane/utils/FixedArray.h"
16#include "arcane/utils/Collection.h"
17#include "arcane/utils/ITraceMng.h"
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"
27#include "arcane/std/internal/IosGmsh.h"
48 class ItemFamilyWriteInfo
53 explicit ItemFamilyWriteInfo(
ITraceMng* tm)
63 Int32 m_dimension = -1;
64 Int32 m_physical_tag = -1;
73 , m_item_type(item_type)
74 , m_entity_tag(entity_tag)
83 m_physical_tag_name = name;
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;
103 const ItemGroup& group()
const {
return m_item_group; }
132 std::pair<Int64, Int64> _getFamilyMinMaxUniqueId(
IItemFamily* family);
150MshMeshWriter(
const ServiceBuildInfo& sbi)
151: AbstractService(sbi)
165 switch (arcane_type) {
169 case IT_Cell3D_Line2:
172 case IT_Cell3D_Triangle3:
175 case IT_Cell3D_Quad4:
180 case IT_Tetraedron10:
201 case IT_AntiWedgeLeft6:
203 case IT_AntiWedgeRight6:
206 case IT_Heptaedron10:
230void MshMeshWriter::ItemGroupWriteInfo::
233 m_item_group = group;
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}')",
250 m_items_by_type[item_type].add(item.
localId());
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();
258 m_existing_items_type.
add(i);
259 total_nb_item += nb_type;
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);
268 EntityInfo entity_info(type_dimension, item_type, base_entity_index + type_index);
270 entity_info.setPhysicalTag(base_entity_index + type_index, group_name);
271 m_entities_by_type.add(entity_info);
277 if (nb_existing_type == 0 && !is_all_items) {
280 Int32 entity_dim = -1;
282 entity_dim = mesh_dim;
284 entity_dim = mesh_dim - 1;
286 entity_dim = mesh_dim - 2;
288 ARCANE_FATAL(
"Invalid item kind '{0}' for entity dimension", entity_dim);
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);
300_addGroupsToProcess(IItemFamily* family, Array<ItemGroup>& items_groups)
302 bool has_group =
false;
306 for (ItemGroup group : family->groups()) {
312 items_groups.add(group);
320 if (!has_group && (family->itemKind() ==
IK_Cell))
321 items_groups.add(family->allItems());
337 m_item_type_mng =
mesh->itemTypeMng();
338 String mesh_file_name(file_name);
340 mesh_file_name = mesh_file_name +
".msh";
341 std::ofstream ofile(mesh_file_name.
localstr());
346 info() <<
"writing file '" << mesh_file_name <<
"'";
348 ofile <<
"$MeshFormat\n";
352 ofile <<
"4.1 0 " <<
sizeof(size_t) <<
"\n";
353 ofile <<
"$EndMeshFormat\n";
358 const Int32 mesh_nb_node =
mesh->nbNode();
363 _addGroupsToProcess(cell_family, items_groups);
364 _addGroupsToProcess(face_family, items_groups);
366 std::vector<std::unique_ptr<ItemGroupWriteInfo>> groups_write_info_list;
367 Int32 base_entity_index = 1000;
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;
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;
394 ++nb_entities_by_dim[dim];
396 Int32 item_type = entity_info.m_item_type;
397 Int32 nb_item = ginfo->itemsByType(item_type).size();
398 total_nb_cell += nb_item;
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 });
415 ofile <<
"$PhysicalNames\n";
417 ofile << nb_tag <<
"\n";
420 ofile << tag_info.m_dimension <<
" " << tag_info.m_physical_tag <<
" " << tag_info.m_name <<
"\n";
422 ofile <<
"$EndPhysicalNames\n";
427 Real3 node_min_bounding_box;
428 Real3 node_max_bounding_box;
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);
435 Real3 pos = nodes_coords[inode];
439 node_min_bounding_box = min_box;
440 node_max_bounding_box = max_box;
466 ofile <<
"$Entities\n";
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)
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;
478 Int32 physical_tag = entity_info.m_physical_tag;
479 if (physical_tag > 0) {
480 ofile <<
" 1 " << physical_tag;
490 ofile <<
"$EndEntities\n";
511 auto [node_min_uid, node_max_uid] = _getFamilyMinMaxUniqueId(node_family);
513 ofile <<
"1 " << mesh_nb_node <<
" " << node_min_uid <<
" " << node_max_uid <<
"\n";
515 ofile <<
"0 " <<
"100 " <<
"0 " << mesh_nb_node <<
"\n";
519 Int64 uid = inode->uniqueId();
520 ofile << uid <<
"\n";
525 Real3 coord = nodes_coords[inode];
526 ofile << coord.
x <<
" " << coord.
y <<
" " << coord.
z <<
"\n";
529 ofile <<
"$EndNodes\n";
532 auto [cell_min_uid, cell_max_uid] = _getFamilyMinMaxUniqueId(cell_family);
545 ofile <<
"$Elements\n";
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) {
552 for (
const EntityInfo& entity_info : ginfo->entitiesByType()) {
553 Int32 cell_type = entity_info.m_item_type;
555 ItemTypeInfo* item_type_info = m_item_type_mng->typeFromId(cell_type);
556 Int32 type_dimension = entity_info.m_dim;
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;
567 for (
Int32 i = 0; i < nb_node_for_type; ++i)
573 ofile <<
"$EndElements\n";
581std::pair<Int64, Int64> MshMeshWriter::
584 Int64 min_uid = INT64_MAX;
594 return { min_uid, max_uid };
#define ARCANE_THROW(exception_class,...)
Macro pour envoyer une exception avec formattage.
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
#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.
Informations sur le type flottant.
Interface d'une famille d'entités.
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.
Exception lorsqu'une erreur d'entrée/sortie est détectée.
Interface du gestionnaire de traces.
virtual TraceMessage info()=0
Flot pour un message d'information.
Groupe d'entités de maillage.
const String & name() const
Nom du groupe.
bool isAutoComputed() const
Vrai s'il s'agit d'un groupe calculé automatiquement.
IItemFamily * itemFamily() const
Famille d'entité à laquelle appartient ce groupe (0 pour le group nul)
bool isAllItems() const
Indique si le groupe est celui de toutes les entités.
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.
ItemTypeInfo * typeFromId(Integer id) const
Type correspondant au numéro id.
Elément de maillage s'appuyant sur des noeuds (Edge,Face,Cell).
Node node(Int32 i) const
i-ème noeud de l'entité
Classe de base d'un élément de maillage.
constexpr Int32 localId() const
Identifiant local de l'entité dans le sous-domaine du processeur.
ItemUniqueId uniqueId() const
Identifiant unique sur tous les domaines.
Int16 type() const
Type de l'entité
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.
Exception lorsqu'une opération n'est pas supportée.
Classe gérant un vecteur de réel de dimension 3.
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.
bool endsWith(const String &s) const
Indique si la chaîne se termine par les caractères de s.
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.
T max(const T &a, const T &b, const T &c)
Retourne le maximum de trois éléments.
ItemGroupT< Cell > CellGroup
Groupe de mailles.
#define ARCANE_REGISTER_SERVICE(aclass, a_service_property,...)
Macro pour enregistrer un service.
MeshVariableScalarRefT< 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
Real z
troisième composante du triplet
Real x
première composante du triplet