14#include "arcane/utils/PlatformUtils.h"
15#include "arcane/utils/ArcanePrecomp.h"
16#include "arcane/utils/Convert.h"
17#include "arcane/utils/CheckedConvert.h"
18#include "arcane/utils/Array.h"
19#include "arcane/utils/StringBuilder.h"
20#include "arcane/utils/Iostream.h"
21#include "arcane/utils/ScopedPtr.h"
22#include "arcane/utils/NotImplementedException.h"
23#include "arcane/utils/ArgumentException.h"
25#include "arcane/ISubDomain.h"
26#include "arcane/IParallelMng.h"
27#include "arcane/ItemEnumerator.h"
28#include "arcane/IMeshSubMeshTransition.h"
29#include "arcane/ItemGroup.h"
30#include "arcane/Service.h"
31#include "arcane/Timer.h"
32#include "arcane/FactoryService.h"
33#include "arcane/ItemPrinter.h"
34#include "arcane/IItemFamily.h"
35#include "arcane/MeshVariable.h"
36#include "arcane/VariableBuildInfo.h"
37#include "arcane/CommonVariables.h"
38#include "arcane/utils/HashTableMap.h"
41#define MPICH_SKIP_MPICXX
42#define OMPI_SKIP_MPICXX
45#include "arcane/std/MeshPartitionerBase.h"
46#include "arcane/std/ZoltanMeshPartitioner_axl.h"
55#define ARCANE_DEBUG_ZOLTAN
72 ZOLTAN_GRAPH = (1<<1),
85 ZoltanInfo(
MeshPartitionerBase* basePartitionner,ostream* ofile,
int model=1,
const Real edgeWeightMultiplier = 1.)
87 , m_mesh_partitionner_base(basePartitionner)
93 , m_edge_weight_multiplier(edgeWeightMultiplier)
95 m_own_cells = m_mesh_partitionner_base->mesh()->ownCells();
106 Real m_edge_weight_multiplier;
107 std::set<std::pair<Int64, Int64> > m_weight_set;
111 info() <<
"ZoltanInfo::build()";
112 if (m_model & ZOLTAN_GEOM)
115 Integer nbOwnCells = m_own_cells.size();
118 if (m_model & ZOLTAN_HG) {
119 nbOwnEdges += nbOwnCells;
121 if (m_model & ZOLTAN_GRAPH) {
123 nbOwnEdges += nbOwnCells * 6;
127 IParallelMng* pm = m_mesh_partitionner_base->mesh()->parallelMng();
130 scanouille[pm->
commRank()] = nbOwnEdges;
133 for (
int i = 0 ; i < pm->
commRank() ; ++i) {
134 m_edgeGIDStart += scanouille[i];
141 if (!m_mesh_partitionner_base->cellUsedWithConstraints(*iCell))
144 int nbNgb = m_mesh_partitionner_base->nbNeighbourCellsWithConstraints(*iCell);
145 if (m_model & ZOLTAN_HG) {
147 m_nbPins += nbNgb + 1;
149 if (m_model & ZOLTAN_GRAPH) {
151 m_nbPins += (nbNgb*2);
155 info() <<
"nbEdges=" << m_nbEdges <<
" ; nbPins=" << m_nbPins;
157 if(m_mesh_partitionner_base->haveWeakConstraints())
160 ENUMERATE_FACE(iface,m_mesh_partitionner_base->mesh()->ownFaces())
162 const Face& face = *iface;
163 const Cell& bCell = iface->backCell();
164 const Cell& fCell = iface->frontCell();
167 if(weak_constraint[face]==2)
178 static int getHgNumVertices(
void *data,
int *ierr)
182 ZoltanInfo* zi = (ZoltanInfo*)data;
193 return zi->m_mesh_partitionner_base->nbOwnCellsWithConstraints();
196 static void getHgVerticesAndWeights(
void *data,
int num_gid_entries,
197 int num_lid_entries, ZOLTAN_ID_PTR gids, ZOLTAN_ID_PTR lids,
198 int wgt_dim,
float *obj_weights,
int *ierr)
200 ARCANE_UNUSED(num_gid_entries);
201 ARCANE_UNUSED(num_lid_entries);
203 ZoltanInfo* zi = (ZoltanInfo*)data;
216 if (zi->m_mesh_partitionner_base->nbCellWeight() <= wgt_dim) {
217 cells_weights = zi->m_mesh_partitionner_base->cellsWeightsWithConstraints(wgt_dim);
221 cells_weights = zi->m_mesh_partitionner_base->cellsSizeWithConstraints();
224 view_weights.
copy(cells_weights);
230 if (!zi->m_mesh_partitionner_base->cellUsedWithConstraints(*icell))
233 gids[index] = (*icell).uniqueId().asInt32();
234 lids[index] = zi->m_mesh_partitionner_base->localIdWithConstraints(*icell);
237 for(
int j=0; j< wgt_dim; ++j ){
238 float weight = cells_weights[lids[index]*wgt_dim+j];
239 *obj_weights++ = weight;
241 (*zi->m_ofile) <<
" Weight uid=" << gids[index] <<
" w=" << weight <<
'\n';
249 static void getHgSizeAndFormat(
void *data,
int *num_lists,
int *num_pins,
int *format,
int *ierr)
251 ZoltanInfo* zi = (ZoltanInfo*)data;
253 zi->
info() <<
" ZoltanInfo::getHgSizeAndFormat() ";
262 *format = ZOLTAN_COMPRESSED_EDGE;
263 *num_pins = zi->m_nbPins;
264 *num_lists = zi->m_nbEdges;
269 zi->
info() <<
" ZoltanInfo::getHgSizeAndFormat() " <<
" num_list= " << *num_lists <<
" num_pins= " << *num_pins;
273 static void getHg(
void *data,
int num_gid_entries,
274 int nrowcol,
int npins,
int format,
275 ZOLTAN_ID_PTR z_vtxedge_GID,
int *z_vtxedge_ptr, ZOLTAN_ID_PTR z_pin_GID,
int *ierr)
277 ZoltanInfo* zi = (ZoltanInfo*)data;
279 zi->
info() <<
" ZoltanInfo::getHg() "
280 <<
" num_gid_entries= " << num_gid_entries
281 <<
" nrowcol= " << nrowcol
282 <<
" npins= " << npins
283 <<
" format= " << format;
290 int gid = zi->m_edgeGIDStart;
291 z_vtxedge_ptr[0] = 0;
293 const Cell& item = *i_item;
295 if (!zi->m_mesh_partitionner_base->cellUsedWithConstraints(item))
298 neighbour_cells.
resize(0);
299 zi->m_mesh_partitionner_base->getNeighbourCellsUidWithConstraints(item,
302 if (zi->m_model & ZOLTAN_HG)
306 z_pin_GID[indexPin++] = item.
uniqueId().asInt32();
307 z_vtxedge_GID[indexEdge] = gid++;
309 z_vtxedge_ptr[indexEdge+1] = z_vtxedge_ptr[indexEdge] + neighbour_cells.
size() + 1;
313 if (zi->m_model & ZOLTAN_GRAPH)
317 for(
Integer z=0; z<neighbour_cells.
size(); ++z ) {
319 z_pin_GID[indexPin++] = (item.
uniqueId().asInt32());
320 z_vtxedge_GID[indexEdge] = gid++;
321 z_vtxedge_ptr[indexEdge+1] = z_vtxedge_ptr[indexEdge] + 2;
327 for (
int i = 0 ; i < nrowcol ; ++i) {
328 (*zi->m_ofile) <<
"*topo GID " << z_vtxedge_GID[i]
329 <<
" : " << z_vtxedge_ptr[i+1] - z_vtxedge_ptr[i];
330 for (
int j = z_vtxedge_ptr[i] ; j < z_vtxedge_ptr[i+1] ; ++j) {
331 (*zi->m_ofile) <<
'\t' << z_pin_GID[j];
333 (*zi->m_ofile) <<
'\n';
346 static void getHgVertexSizes (
void *data,
int num_gid_entries,
int num_lid_entries,
347 int num_ids, ZOLTAN_ID_PTR global_ids, ZOLTAN_ID_PTR local_ids,
348 int *sizes,
int *ierr)
350 ARCANE_UNUSED(num_gid_entries);
351 ARCANE_UNUSED(num_lid_entries);
352 ARCANE_UNUSED(global_ids);
354 ZoltanInfo* zi = (ZoltanInfo*)data;
356 SharedArray<float> cellSizes = zi->m_mesh_partitionner_base->cellsSizeWithConstraints();
358 for (
int i = 0 ; i < num_ids ; ++i)
366 static void getHgEdgeWeightSize(
void *data,
int *num_edges,
int *ierr)
368 ZoltanInfo* zi = (ZoltanInfo*)data;
379 *num_edges = zi->m_nbEdges;
383 static void getHgEdgeWeights(
void *data,
int num_gid_entries,
384 int num_lid_entries,
int nedges,
int edge_weight_dim,
385 ZOLTAN_ID_PTR edge_GID, ZOLTAN_ID_PTR edge_LID,
float *edge_weight,
int *ierr)
387 ARCANE_UNUSED(num_lid_entries);
389 ZoltanInfo* zi = (ZoltanInfo*)data;
395 zi->
info() <<
" ZoltanInfo::getHgEdgeWeights() "
396 <<
" num_gid_entries= " << num_gid_entries
397 <<
" nedges= " << nedges
398 <<
" edge_weight_dim= " << edge_weight_dim;
403 connectivityWeights.
reserve(6);
409 if (!zi->m_mesh_partitionner_base->cellUsedWithConstraints(item))
412 connectivityWeights.
resize(0);
413 neighbour_cells.
resize(0);
414 bool hg_model=(zi->m_model & ZOLTAN_HG);
417 zi->m_mesh_partitionner_base->getNeighbourCellsUidWithConstraints(item,
418 neighbour_cells, &connectivityWeights, hg_model);
420 if (zi->m_model & ZOLTAN_HG) {
421 if (!(zi->m_model & ZOLTAN_GRAPH)) {
423 for(
Integer z=0; z<neighbour_cells.
size(); ++z ) {
424 he_weight += connectivityWeights[z];
427 edge_GID[indexEdge] = indexEdge + zi->m_edgeGIDStart;
428 edge_LID[indexEdge] = indexEdge;
429 edge_weight[indexEdge++] = (float)he_weight;
432 if (zi->m_model & ZOLTAN_GRAPH) {
436 edge_GID[indexEdge] = indexEdge + zi->m_edgeGIDStart;
437 edge_LID[indexEdge] = indexEdge;
438 Real w = connectivityWeights[z];
439 edge_weight[indexEdge] = (float)w;
440 if (zi->m_mesh_partitionner_base->haveWeakConstraints()){
441 std::pair<Int64,Int64> items(item.
uniqueId(), neighbour_cells[z]);
442 if(zi->m_mesh_partitionner_base->cellUsedWithWeakConstraints(items)){
443 edge_weight[indexEdge] *= (float)zi->m_edge_weight_multiplier;
454 static int get_num_geometry(
void *data,
int *ierr)
462 static void get_geometry_list(
void *data,
int sizeGID,
int sizeLID,
464 ZOLTAN_ID_PTR global_ids, ZOLTAN_ID_PTR local_ids,
465 int num_dim,
double *geom_vec,
int *ierr)
467 ARCANE_UNUSED(global_ids);
468 ARCANE_UNUSED(local_ids);
470 ZoltanInfo* zi = (ZoltanInfo*)data;
472 if ( (sizeGID != 1) || (sizeLID != 1) || (num_dim > 3)){
473 *ierr = ZOLTAN_FATAL;
482 if (!zi->m_mesh_partitionner_base->cellUsedWithConstraints(*icell))
489 for(
Integer z=0, zs = (*icell).nbNode(); z<zs; ++z ){
490 const Node& node = (*icell).node(z);
495 geom_vec[num_dim*i ] = bar.x;
496 geom_vec[num_dim*i+1] = bar.y;
497 geom_vec[num_dim*i+2] = bar.z;
502 if (!s.
null() && (s ==
"MYGEOM")) {
503 for (
int i = 0 ; i < num_obj ; ++i) {
504 geom_vec[num_dim*i+num_dim-1] = 0.0;
518class ZoltanMeshPartitioner
542ZoltanMeshPartitioner::
543ZoltanMeshPartitioner(
const ServiceBuildInfo& sbi)
544 : ArcaneZoltanMeshPartitionerObject(sbi)
554 Int32 nb_part =
mesh()->parallelMng()->commSize();
564 info() <<
"Load balancing with Zoltan\n";
568 int rc = ::Zoltan_Initialize(0,0,&ver);
569 Zoltan_Memory_Debug(2);
571 fatal() <<
"Can not initialize zoltan (r=" << rc <<
")";
580 warning() <<
"Unable to test load balancing on a single sub-domain";
584 Integer nb_weight = nbCellWeight();
586 struct Zoltan_Struct *zz;
591 ZOLTAN_ID_PTR importGlobalIds;
592 ZOLTAN_ID_PTR importLocalIds;
596 ZOLTAN_ID_PTR exportGlobalIds;
607 zz = Zoltan_Create(*(MPI_Comm*)getCommunicator());
609 Zoltan_Set_Param(zz,
"RCB_REUSE",
"1");
618 if (!s.null() &&
options()->model.isPresent())
619 fatal() <<
"Conflicting configuration between ZOLTAN_MODEL environment variable and user data set";
620 usePHG =
options()->useHypergraphe();
625 }
else if (s.null()) {
631 if (subDomain()->commonVariables().globalIteration() <= 2)
637 String algo =
"HYPERGRAPH";
638 if (s ==
"MYHG" || s ==
"OLDHG" || s ==
"DUALHG" || s ==
"GRAPH") {
641 }
else if (s ==
"RIB" || s==
"HSFC") {
644 }
else if (s ==
"RCB" || s ==
"MYGEOM") {
648 fatal() <<
"Undefined zoltan model '" << s <<
"'";
651 int model = ZOLTAN_HG;
652 if (usePHG ==
false) {
657 model |= ZOLTAN_GRAPH;
658 }
else if (s ==
"GRAPH") {
659 model = ZOLTAN_GRAPH;
666 Real edgeWeightMultiplier = 1;
667 Integer repartitionFrequency = 10;
668 Real imbalanceTol = 1.05;
669 Real phgRepartMultiplier = 10;
675 edgeWeightMultiplier =
options()->edgeWeightMultiplier();
676 repartitionFrequency =
options()->repartFrequency();
677 imbalanceTol =
options()->imbalanceTol();
678 phgRepartMultiplier =
options()->phgRepartMultiplier();
679 phgOutputLevel =
options()->phgOutputLevel();
680 debugLevel =
options()->debugLevel();
684 info() <<
"Zoltan: utilise un repartitionnement " << algo <<
" (" << s <<
").";
685 Zoltan_Set_Param(zz,
"LB_METHOD", algo.localstr());
686 Zoltan_Set_Param(zz,
"HYPERGRAPH_PACKAGE",
"PHG");
688 if(
mesh->subDomain()->commonVariables().globalIteration()==1)
690 Zoltan_Set_Param(zz,
"LB_APPROACH",
"PARTITION");
691 info() <<
"Zoltan: Partition";
693 else if(
mesh->subDomain()->commonVariables().globalIteration()%repartitionFrequency==0 && repartitionFrequency!=-1)
695 Zoltan_Set_Param(zz,
"LB_APPROACH",
"REPARTITION");
696 info() <<
"Zoltan: Repartition";
700 Zoltan_Set_Param(zz,
"LB_APPROACH",
"REFINE");
701 info() <<
"Zoltan: Refine";
704 String s_imbalaceTol(String::fromNumber(imbalanceTol));
705 Zoltan_Set_Param(zz,
"IMBALANCE_TOL", s_imbalaceTol.localstr());
707 String s_nb_part(String::fromNumber(nb_part));
708 Zoltan_Set_Param(zz,
"NUM_GLOBAL_PARTS", s_nb_part.localstr());
710 Zoltan_Set_Param(zz,
"NUM_GID_ENTRIES",
"1");
711 Zoltan_Set_Param(zz,
"NUM_LID_ENTRIES",
"1");
712 Zoltan_Set_Param(zz,
"RETURN_LISTS",
"EXPORT");
713 String s_nb_weight(String::fromNumber(nb_weight));
715 Zoltan_Set_Param(zz,
"OBJ_WEIGHT_DIM", s_nb_weight.localstr());
718 Zoltan_Set_Param(zz,
"ADD_OBJ_WEIGHT",
"NONE");
722 Zoltan_Set_Param(zz,
"FINAL_OUTPUT",
"0");
723 Zoltan_Set_Param(zz,
"PHG_USE_TIMERS",
"1");
725 String s_phgOutputLevel(String::fromNumber(phgOutputLevel));
726 Zoltan_Set_Param(zz,
"PHG_OUTPUT_LEVEL", s_phgOutputLevel.localstr());
729 Zoltan_Set_Param(zz,
"CHECK_HYPERGRAPH",
"0");
730 String s_debugLevel(String::fromNumber(debugLevel));
731 Zoltan_Set_Param(zz,
"DEBUG_LEVEL", s_debugLevel.localstr());
736 Zoltan_Set_Param(zz,
"PHG_COARSENING_METHOD",
"IPM");
740 Zoltan_Set_Param(zz,
"EDGE_WEIGHT_DIM",
"1");
741 Zoltan_Set_Param(zz,
"PHG_EDGE_WEIGHT_OPERATION",
"add");
743 if(!initial_partition)
747 String s_phgRepartMultiplier(String::fromNumber(phgRepartMultiplier));
748 Zoltan_Set_Param(zz,
"PHG_REPART_MULTIPLIER", s_phgRepartMultiplier.localstr());
753 Zoltan_Set_Param(zz,
"KEEP_CUTS",
"0");
754 Zoltan_Set_Param(zz,
"RCB_OUTPUT_LEVEL",
"0");
755 Zoltan_Set_Param(zz,
"RCB_RECTILINEAR_BLOCKS",
"0");
758 Zoltan_Set_Param(zz,
"OBJ_WEIGHTS_COMPARABLE",
"0");
762 Zoltan_Set_Param(zz,
"RCB_MULTICRITERIA_NORM",
"3");
763 Zoltan_Set_Param(zz,
"RCB_MAX_ASPECT_RATIO",
"10");
766 bool dump_infos =
false;
772 Integer iteration =
mesh->subDomain()->commonVariables().globalIteration();
778 ofile.open(f.localstr());
790 ScopedPtrT<ZoltanInfo> zoltan_info(
new ZoltanInfo(
this,zofile, model, edgeWeightMultiplier));
791 Zoltan_Set_Num_Obj_Fn(zz,&ZoltanInfo::getHgNumVertices, zoltan_info.get());
792 Zoltan_Set_Obj_List_Fn(zz, &ZoltanInfo::getHgVerticesAndWeights, zoltan_info.get());
793 if (!initial_partition) {
794 Zoltan_Set_Obj_Size_Multi_Fn(zz, &ZoltanInfo::getHgVertexSizes,
798 info() <<
"Setting up HG Callbacks";
799 Zoltan_Set_HG_Size_CS_Fn(zz, &ZoltanInfo::getHgSizeAndFormat, zoltan_info.get());
800 Zoltan_Set_HG_CS_Fn(zz, &ZoltanInfo::getHg, zoltan_info.get());
801 Zoltan_Set_HG_Size_Edge_Weights_Fn(zz, &ZoltanInfo::getHgEdgeWeightSize, zoltan_info.get());
802 Zoltan_Set_HG_Edge_Weights_Fn(zz, &ZoltanInfo::getHgEdgeWeights, zoltan_info.get());
806 info() <<
"Setting up Geom Callbacks";
807 Zoltan_Set_Num_Geom_Fn(zz, ZoltanInfo::get_num_geometry, zoltan_info.get());
808 Zoltan_Set_Geom_Multi_Fn(zz, ZoltanInfo::get_geometry_list, zoltan_info.get());
813 int* export_partitions = 0;
814 ZOLTAN_ID_PTR export_local_ids = 0;
816 info() <<
"Doing partition";
817 rc = Zoltan_LB_Partition(zz, &changes, &numGidEntries, &numLidEntries,
818 &numImport, &importGlobalIds, &importLocalIds,
819 &importProcs, &importToPart,
820 &numExport, &exportGlobalIds, &export_local_ids,
821 &exportProcs, &export_partitions);
826 cells_new_owner[icell] = (*icell).owner();
829 invertArrayLid2LidCompacted();
832 CellInfoListView items_internal(m_cell_family);
834 info() <<
"numExport = "<<numExport;
835 for(
Integer i=0; i<nb_export; ++i ){
836 Item item = items_internal[ localIdWithConstraints(export_local_ids[i]) ];
838 changeCellOwner(item, cells_new_owner, export_partitions[i]);
840 ofile <<
"EXPORT: uid=" << ItemPrinter(item) <<
" OLD=" << item.owner()
841 <<
" NEW=" << cells_new_owner[item] <<
" PROC=" << exportProcs[i] <<
'\n';
855 MPI_Abort(MPI_COMM_WORLD, EXIT_FAILURE);
858 Zoltan_LB_Free_Part(&exportGlobalIds, &export_local_ids,
859 &exportProcs, &export_partitions);
866 cells_new_owner.synchronize();
887#if ARCANE_DEFAULT_PARTITIONER == ZOLTAN_DEFAULT_PARTITIONER
#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.
CaseOptionsZoltanMeshPartitioner * options() const
Options du jeu de données du service.
ArcaneZoltanMeshPartitionerObject(const Arcane::ServiceBuildInfo &sbi)
Constructeur.
Exception lorsqu'un argument est invalide.
Vue modifiable d'un tableau d'un type T.
void copy(const U ©_array)
Recopie le tableau copy_array dans l'instance.
void fill(const DataType &data)
Remplissage du tableau.
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.
Cell frontCell() const
Maille devant la face (maille nulle si aucune)
Cell backCell() const
Maille derrière la face (maille nulle si aucune)
virtual ITraceMng * traceMng()=0
Gestionnaire de message associé
Interface d'un partitionneur de maillage.
Interface d'un partitionneur de maillage.
virtual IMesh * mesh() const =0
Maillage associé au partitionneur.
virtual VariableNodeReal3 & nodesCoordinates()=0
Coordonnées des noeuds.
Interface du gestionnaire de parallélisme pour un sous-domaine.
virtual Int32 commRank() const =0
Rang de cette instance dans le communicateur.
virtual void scan(eReduceType rt, ArrayView< char > v)=0
Applique un algorithme de prefix-um sur les valeurs de v via l'opération rt.
virtual Int32 commSize() const =0
Nombre d'instance dans le communicateur.
virtual void barrier()=0
Effectue une barière.
ItemUniqueId uniqueId() const
Identifiant unique sur tous les domaines.
constexpr bool null() const
true si l'entité est nul (i.e. non connecté au maillage)
Classe de base d'un service d'équilibrage de charge.
IMesh * mesh() const override
Maillage associé au partitionneur.
virtual void changeOwnersFromCells()
Positionne les nouveaux propriétaires des noeuds, arêtes et faces à partir des mailles.
Variable scalaire sur un type d'entité du maillage.
Classe gérant un vecteur de réel de dimension 3.
Structure contenant les informations pour créer un service.
Propriétés de création d'un service.
Vecteur 1D de données avec sémantique par référence.
Chaîne de caractères unicode.
bool null() const
Retourne true si la chaîne est nulle.
TraceAccessor(ITraceMng *m)
Construit un accesseur via le gestionnaire de trace m.
TraceMessage fatal() const
Flot pour un message d'erreur fatale.
TraceMessage info() const
Flot pour un message d'information.
TraceMessage warning() const
Flot pour un message d'avertissement.
Vecteur 1D de données avec sémantique par valeur (style STL).
Paramètres nécessaires à la construction d'une variable.
Partitioneur de maillage utilisant la bibliotheque Zoltan.
virtual void notifyEndPartition()
Notification lors de la fin d'un repartionnement (après échange des entités)
virtual void build()
Construction de niveau build du service.
virtual void partitionMesh(bool initial_partition)
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.
ItemVariableScalarRefT< Int32 > VariableItemInt32
Grandeur de type entier 32 bits.
Integer toInteger(Real r)
Converti un Int64 en un Integer.
Integer toInteger(Real r)
Converti un Real en Integer.
double toDouble(Real r)
Converti un Real en double.
@ ReduceSum
Somme des valeurs.
-*- tab-width: 2; indent-tabs-mode: nil; coding: utf-8-with-signature -*-
UniqueArray< Int64 > Int64UniqueArray
Tableau dynamique à une dimension d'entiers 64 bits.
Int32 Integer
Type représentant un entier.
@ ST_SubDomain
Le service s'utilise au niveau du sous-domaine.
UniqueArray< Int32 > Int32UniqueArray
Tableau dynamique à une dimension d'entiers 32 bits.
@ IK_Cell
Entité de maillage de genre maille.
double Real
Type représentant un réel.
std::int32_t Int32
Type entier signé sur 32 bits.