Arcane  v3.16.0.0
Documentation développeur
Chargement...
Recherche...
Aucune correspondance
ArcaneCaseMeshService.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/* ArcaneCaseMeshService.cc (C) 2000-2025 */
9/* */
10/* Service Arcane gérant un maillage du jeu de données. */
11/*---------------------------------------------------------------------------*/
12/*---------------------------------------------------------------------------*/
13
14#include "arcane/utils/ApplicationInfo.h"
15#include "arcane/utils/CommandLineArguments.h"
16
18#include "arcane/core/ServiceBuilder.h"
19#include "arcane/core/ICaseMeshService.h"
20#include "arcane/core/ICaseMeshReader.h"
21#include "arcane/core/IMeshBuilder.h"
22#include "arcane/core/IPrimaryMesh.h"
23#include "arcane/core/IItemFamily.h"
24#include "arcane/core/IMeshPartitionerBase.h"
25#include "arcane/core/IVariableMng.h"
26#include "arcane/core/IMeshModifier.h"
27#include "arcane/core/IMeshUtilities.h"
28#include "arcane/core/IParallelMng.h"
29#include "arcane/core/MeshBuildInfo.h"
30#include "arcane/core/IMeshMng.h"
31#include "arcane/core/IMeshFactoryMng.h"
32#include "arcane/core/IGhostLayerMng.h"
33#include "arcane/core/MeshPartInfo.h"
34#include "arcane/core/IMeshSubdivider.h"
35#include "arcane/core/IMeshUniqueIdMng.h"
36#include "arcane/core/internal/StringVariableReplace.h"
37
38#include "arcane/impl/ArcaneCaseMeshService_axl.h"
39
40/*---------------------------------------------------------------------------*/
41/*---------------------------------------------------------------------------*/
42
43namespace Arcane
44{
45
46/*---------------------------------------------------------------------------*/
47/*---------------------------------------------------------------------------*/
51class ArcaneCaseMeshService
53{
54 public:
55
56 explicit ArcaneCaseMeshService(const ServiceBuildInfo& sbi);
57
58 public:
59
60 void createMesh(const String& default_name) override;
61 void allocateMeshItems() override;
62 void partitionMesh() override;
63 void applyAdditionalOperations() override;
64
65 private:
66
67 ISubDomain* m_sub_domain = nullptr;
68 IPrimaryMesh* m_mesh = nullptr;
69 IMeshBuilder* m_mesh_builder = nullptr;
70 Ref<IMeshBuilder> m_mesh_builder_ref;
71 String m_mesh_file_name;
72 String m_partitioner_name;
73
74 private:
75
76 void _fillReadInfo(CaseMeshReaderReadInfo& read_info);
77 Ref<IMeshBuilder> _createBuilderFromFile(const CaseMeshReaderReadInfo& read_info);
78 void _initializeVariables();
79 void _doInitialPartition();
80 void _doInitialPartition2(const String& name);
81 void _setGhostLayerInfos();
82 void _checkMeshCreationAndAllocation(bool is_check_allocated);
83 void _setUniqueIdNumberingVersion();
84};
85
86/*---------------------------------------------------------------------------*/
87/*---------------------------------------------------------------------------*/
88
89ArcaneCaseMeshService::
90ArcaneCaseMeshService(const ServiceBuildInfo& sbi)
92, m_sub_domain(sbi.subDomain())
93{
94}
95
96/*---------------------------------------------------------------------------*/
97/*---------------------------------------------------------------------------*/
98
100createMesh(const String& default_name)
101{
102 if (m_mesh)
103 ARCANE_FATAL("Mesh is already created");
104
105 info() << "Creating mesh from 'ArcaneCaseMeshService'";
106 MeshBuildInfo build_info(default_name);
107 ISubDomain* sd = m_sub_domain;
108 String filename = options()->filename();
109 bool has_filename = options()->filename.isPresent();
110 bool has_generator = options()->generator.isPresent();
111
112 // Il faut spécifier soit le nom du fichier, soit le générateur mais pas les deux
113 if ((has_filename && has_generator) || (!has_filename && !has_generator))
114 ARCANE_FATAL("In '{0}': one and one only of <{1}> or <{2}> has to be specified",
115 options()->configList()->xpathFullName(),
116 options()->generator.rootTagName(),
117 options()->filename.name());
118 if (has_filename) {
119 m_mesh_file_name = options()->filename();
120 if (m_mesh_file_name.empty())
121 ARCANE_FATAL("Invalid filename '{0}' in option '{1}'",
122 m_mesh_file_name,options()->filename.xpathFullName());
123 CaseMeshReaderReadInfo read_info;
124 _fillReadInfo(read_info);
125 auto& specific_reader = options()->specificReader;
126 if (specific_reader.isPresent()){
127 m_mesh_builder_ref = specific_reader()->createBuilder(read_info);
128 if (m_mesh_builder_ref.isNull())
129 ARCANE_FATAL("No 'IMeshBuilder' created by specific reader");
130 }
131 else{
132 m_mesh_builder_ref = _createBuilderFromFile(read_info);
133 }
134 m_mesh_builder = m_mesh_builder_ref.get();
135 }
136 else if (has_generator)
137 m_mesh_builder = options()->generator();
138 else
139 ARCANE_FATAL("Invalid operation");
140
141 ARCANE_CHECK_POINTER(m_mesh_builder);
142
143 // Indique si les entités multi-dimension sont autorisées
144 bool is_non_manifold = options()->nonManifoldMesh;
145 if (is_non_manifold) {
146 MeshKind mesh_kind = build_info.meshKind();
147 mesh_kind.setIsNonManifold(true);
148 build_info.addMeshKind(mesh_kind);
149 }
150
151 m_mesh_builder->fillMeshBuildInfo(build_info);
152 // Le générateur peut forcer l'utilisation du partitionnement
153 if (build_info.isNeedPartitioning())
154 m_partitioner_name = options()->partitioner();
155
156 // Positionne avec des valeurs par défaut les champs non remplit.
157 if (build_info.factoryName().empty())
158 build_info.addFactoryName("ArcaneDynamicMeshFactory");
159 if (build_info.parallelMngRef().isNull())
160 build_info.addParallelMng(makeRef(sd->parallelMng()));
161 IPrimaryMesh* pm = sd->meshMng()->meshFactoryMng()->createMesh(build_info);
162 m_mesh = pm;
163}
164
165/*---------------------------------------------------------------------------*/
166/*---------------------------------------------------------------------------*/
167
170{
171 _checkMeshCreationAndAllocation(false);
172
173 ARCANE_CHECK_POINTER(m_mesh_builder);
174
175 _setGhostLayerInfos();
176 _setUniqueIdNumberingVersion();
177
178 m_mesh_builder->allocateMeshItems(m_mesh);
179}
180
181/*---------------------------------------------------------------------------*/
182/*---------------------------------------------------------------------------*/
183
186{
187 _checkMeshCreationAndAllocation(true);
188
189 if (m_mesh->meshPartInfo().nbPart() > 1)
190 if (!m_partitioner_name.empty())
191 _doInitialPartition();
192}
193
194/*---------------------------------------------------------------------------*/
195/*---------------------------------------------------------------------------*/
196
199{
200 _checkMeshCreationAndAllocation(true);
201
202 IMeshSubdivider* subdivider = options()->subdivider();
203 if (subdivider)
204 subdivider->subdivideMesh(m_mesh);
205
206 _initializeVariables();
207}
208
209/*---------------------------------------------------------------------------*/
210/*---------------------------------------------------------------------------*/
211
212void ArcaneCaseMeshService::
213_checkMeshCreationAndAllocation(bool is_check_allocated)
214{
215 if (!m_mesh)
216 ARCANE_FATAL("Mesh is not created. You should call createMesh() before");
217 if (is_check_allocated && !m_mesh->isAllocated())
218 ARCANE_FATAL("Mesh is not allocated. You should call initializeMesh() before");
219}
220
221/*---------------------------------------------------------------------------*/
222/*---------------------------------------------------------------------------*/
223
224void ArcaneCaseMeshService::
225_fillReadInfo(CaseMeshReaderReadInfo& read_info)
226{
227 // Cherche l'extension du fichier.
228 String file_extension;
229 {
230 std::string_view fview = m_mesh_file_name.toStdStringView();
231 std::size_t extension_pos = fview.find_last_of('.');
232 if (extension_pos!=std::string_view::npos){
233 fview.remove_prefix(extension_pos+1);
234 file_extension = fview;
235 }
236 read_info.setFormat(file_extension);
237 }
238
239 // Nom du maillage à utiliser pour la lecture
240 // Normalement c'est le nom du maillage dans le jeu de données
241 // sauf en cas d'utilisation du partitionneur externe auquel cas
242 // c'est 'CPU%05d.$file_extension'.
243 String mesh_file_name = m_mesh_file_name;
244
245 String partitioner_name = options()->partitioner();
246 bool use_internal_partitioner = partitioner_name != "External";
247 if (use_internal_partitioner){
248 m_partitioner_name = partitioner_name;
249 }
250 else{
251 info() << "Using external partitioner";
252 int mesh_rank = m_sub_domain->parallelMng()->commRank();
253 char buf[128];
254 sprintf(buf,"CPU%05d.%s",mesh_rank,file_extension.localstr());
255 mesh_file_name = String(std::string_view(buf));
256 }
257
258 read_info.setFileName(mesh_file_name);
259
260 info() << "Mesh filename=" << mesh_file_name
261 << " extension=" << read_info.format() << " partitioner=" << partitioner_name;
262
263 read_info.setParallelRead(use_internal_partitioner);
264}
265
266/*---------------------------------------------------------------------------*/
267/*---------------------------------------------------------------------------*/
268
269Ref<IMeshBuilder> ArcaneCaseMeshService::
270_createBuilderFromFile(const CaseMeshReaderReadInfo& read_info)
271{
272 // Construit les services potentiels de lecture de maillage. Ce sont
273 // ceux qui implémentent ICaseMeshReader. Construit une instance
274 // de chaque service et appelle la méthode createBuilder(). Dès que l'une
275 // de ces méthodes renvoie une référence non nulle, on l'utilise
276 // pour générer le maillage.
277 ServiceBuilder<ICaseMeshReader> builder(m_sub_domain);
278 UniqueArray<Ref<ICaseMeshReader>> mesh_readers(builder.createAllInstances());
279
280 for( auto& mesh_reader_ref : mesh_readers ){
281 ICaseMeshReader* mesh_reader = mesh_reader_ref.get();
282 Ref<IMeshBuilder> builder = mesh_reader->createBuilder(read_info);
283 if (!builder.isNull())
284 return builder;
285 }
286
287 // Pas de service trouvé pour ce format de fichier.
288 // Affiche la liste des services disponibles et fait un fatal.
289 StringUniqueArray valid_names;
290 builder.getServicesNames(valid_names);
291 String available_readers = String::join(", ",valid_names);
292 ARCANE_FATAL("The mesh reader required for format '{0}' is not available."
293 "The following reader services are available: {1}",
294 read_info.format(),available_readers);
295}
296
297/*---------------------------------------------------------------------------*/
298/*---------------------------------------------------------------------------*/
299
300void ArcaneCaseMeshService::
301_doInitialPartition()
302{
303 // N'utilise plus le service de partitionnement de test pour garantir
304 // avec ParMetis qu'on n'a pas de partitions vides car cela est maintenant
305 // normalement supporté.
306 const bool use_partitioner_tester = false;
307 String test_service = "MeshPartitionerTester";
308 if (use_partitioner_tester) {
309 Int64 nb_cell = m_mesh->nbCell();
310 Int64 min_nb_cell = m_mesh->parallelMng()->reduce(Parallel::ReduceMin,nb_cell);
311 info() << "Min nb cell=" << min_nb_cell;
312 if (min_nb_cell==0)
313 _doInitialPartition2(test_service);
314 else
315 info() << "Mesh name=" << m_mesh->name() << " have cells. Do not use " << test_service;
316 }
317 else
318 info() << "No basic partition first needed";
319 _doInitialPartition2(m_partitioner_name);
320}
321
322/*---------------------------------------------------------------------------*/
323/*---------------------------------------------------------------------------*/
324
325void ArcaneCaseMeshService::
326_doInitialPartition2(const String& partitioner_name)
327{
328 info() << "Doing initial partitioning service=" << partitioner_name;
329 // NOTE: Ce service n'utilise que les partionneurs qui implémentent
330 // IMeshPartitionerBase et pas ceux (historiques) qui n'implémentent que
331 // IMeshPartitioner.
332 ServiceBuilder<IMeshPartitionerBase> sbuilder(m_sub_domain);
333 auto mesh_partitioner = sbuilder.createReference(partitioner_name,m_mesh);
334
335 IMesh* mesh = m_mesh;
336 bool is_dynamic = mesh->isDynamic();
337 mesh->modifier()->setDynamic(true);
338 mesh->utilities()->partitionAndExchangeMeshWithReplication(mesh_partitioner.get(),true);
339 mesh->modifier()->setDynamic(is_dynamic);
340}
341
342/*---------------------------------------------------------------------------*/
343/*---------------------------------------------------------------------------*/
344
345void ArcaneCaseMeshService::
346_initializeVariables()
347{
348 IVariableMng* vm = m_sub_domain->variableMng();
349 const auto& vars_opt = options()->initialization().variable;
350 UniqueArray<String> errors;
351 for( Integer i=0, n=vars_opt.size(); i<n; ++i ){
352 const auto& o = vars_opt[i];
353 String var_name = o.name;
354 String group_name = o.group;
355 String value = o.value;
356 info() << "Initialize variable=" << var_name << " group=" << group_name << " value=" << value;
357 IVariable* var = vm->findMeshVariable(m_mesh,var_name);
358 if (!var){
359 errors.add(String::format("No variable named '{0}' exists",var_name));
360 continue;
361 }
362
363 // Alloue la variable si besoin.
364 if (!var->isUsed())
365 var->setUsed(true);
366 IItemFamily* var_family = var->itemFamily();
367 if (!var_family){
368 errors.add(String::format("Variable '{0}' has no family",var->fullName()));
369 continue;
370 }
371
372 ItemGroup group = var_family->findGroup(group_name);
373 if (group.null()){
374 errors.add(String::format("No group named '{0}' exists in family '{1}'",
375 group_name,var_family->name()));
376 continue;
377 }
378
379 bool ret = var->initialize(group,value);
380 if (ret){
381 errors.add(String::format("Bad value '{0}' for initializing variable '{1}'",
382 value,var->fullName()));
383 continue;
384 }
385 }
386 if (!errors.empty()){
387 for( String s : errors )
388 pinfo() << "ERROR: " << s;
389 ARCANE_FATAL("Variable initialization failed for option '{0}'",
390 vars_opt.xpathFullName());
391 }
392}
393
394/*---------------------------------------------------------------------------*/
395/*---------------------------------------------------------------------------*/
396
397void ArcaneCaseMeshService::
398_setGhostLayerInfos()
399{
400 IGhostLayerMng* gm = m_mesh->ghostLayerMng();
401 if (!gm)
402 return;
403
404 // Positionne les infos sur les mailles fantômes.
405 // TODO Cela est fait pour rester compatible avec le mode historique mais
406 // il faudrait pouvoir gérer cela autrement (via un service par exemple)
407 Integer nb_ghost_layer = options()->nbGhostLayer();
408 if (nb_ghost_layer >= 0) {
409 info() << "Set number of ghost layers to '" << nb_ghost_layer << "' from caseoption";
410 gm->setNbGhostLayer(nb_ghost_layer);
411 }
412
413 Integer builder_version = options()->ghostLayerBuilderVersion();
414 if (builder_version >= 0) {
415 info() << "Set ghostlayer builder version to '" << builder_version << "' from caseoption";
416 gm->setBuilderVersion(builder_version);
417 }
418}
419
420/*---------------------------------------------------------------------------*/
421/*---------------------------------------------------------------------------*/
422
423void ArcaneCaseMeshService::
424_setUniqueIdNumberingVersion()
425{
426 // NOTE: actuellement (12/2024) l'implémentation 'PolyedralMesh' lève une
427 // exception si on appelle meshUniqueIdMng(). On ne le fait que si
428 // l'option est présente.
429 if (options()->faceNumberingVersion.isPresent()) {
430 Int32 v = options()->faceNumberingVersion.value();
431 info() << "Set face uniqueId numbering version to '" << v << "' from caseoption";
432 IMeshUniqueIdMng* mum = m_mesh->meshUniqueIdMng();
433 mum->setFaceBuilderVersion(v);
434 }
435}
436
437/*---------------------------------------------------------------------------*/
438/*---------------------------------------------------------------------------*/
439
440ARCANE_REGISTER_SERVICE_ARCANECASEMESHSERVICE(ArcaneCaseMeshService,
442
443/*---------------------------------------------------------------------------*/
444/*---------------------------------------------------------------------------*/
445
446} // End namespace Arcane
447
448/*---------------------------------------------------------------------------*/
449/*---------------------------------------------------------------------------*/
#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_FATAL(...)
Macro envoyant une exception FatalErrorException.
Ce fichier contient les différentes fabriques de services et macro pour enregistrer les services.
Generation de la classe de base du Service.
ArcaneArcaneCaseMeshServiceObject(const Arcane::ServiceBuildInfo &sbi)
Constructeur.
CaseOptionsArcaneCaseMeshService * options() const
Options du jeu de données du service.
Service Arcane un maillage du jeu de données.
void createMesh(const String &default_name) override
Créé le maillage avec le nom name.
void allocateMeshItems() override
Alloue les éléments du maillage.
void partitionMesh() override
Partitionne le maillage.
void applyAdditionalOperations() override
Applique les opérations après tout le reste.
Informations nécessaires pour la lecture d'un fichier de maillage.
const Type & value() const
Retourne la valeur de l'option.
bool isPresent() const
Retourne true si l'option est présente.
Interface d'un service de création/lecture du maillage.
virtual IPrimaryMesh * createMesh(const MeshBuildInfo &build_info)=0
Créé un maillage ou un sous-maillage.
virtual IMeshFactoryMng * meshFactoryMng() const =0
Fabrique de maillages associée à ce gestionnaire.
virtual void subdivideMesh(IPrimaryMesh *mesh)=0
Subdivise le maillage mesh.
virtual bool isAllocated()=0
Vrai si le maillage est allouée.
Interface du gestionnaire d'un sous-domaine.
Definition ISubDomain.h:74
virtual IParallelMng * parallelMng()=0
Retourne le gestionnaire de parallélisme.
virtual IMeshMng * meshMng() const =0
Retourne le gestionnaire de maillage.
Paramètres nécessaires à la construction d'un maillage.
const MeshKind meshKind() const
Caractéristiques du maillage.
MeshBuildInfo & addParallelMng(Ref< IParallelMng > pm)
Positionne le gestionnaire de parallélisme pour créér la maillage.
bool isNeedPartitioning() const
Indique si le lecteur/générateur nécessite un partitionnement.
Ref< IParallelMng > parallelMngRef() const
Gestionnaire de parallélisme dans le cas d'un nouveau maillage.
const String & factoryName() const
Nom de la fabrique pour créer le maillage (via IMeshFactory)
MeshBuildInfo & addMeshKind(const MeshKind &v)
Positionne les caractéristiques du maillage.
MeshBuildInfo & addFactoryName(const String &factory_name)
Positionne le nom de la fabrique pour créer ce maillage.
Caractéristiques d'un maillage.
Definition MeshKind.h:64
Référence à une instance.
Structure contenant les informations pour créer un service.
Chaîne de caractères unicode.
bool empty() const
Vrai si la chaîne est vide (nulle ou "")
Definition String.cc:315
std::string_view toStdStringView() const
Retourne une vue de la STL sur la chaîne actuelle.
Definition String.cc:348
TraceMessage pinfo() const
Flot pour un message d'information en parallèle.
TraceMessage info() const
Flot pour un message d'information.
@ ReduceMin
Minimum des valeurs.
-*- 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.
auto makeRef(InstanceType *t) -> Ref< InstanceType >
Créé une référence sur un pointeur.
UniqueArray< String > StringUniqueArray
Tableau dynamique à une dimension de chaînes de caractères.
Definition UtilsTypes.h:446
std::int32_t Int32
Type entier signé sur 32 bits.