Arcane  v3.16.0.0
Documentation développeur
Chargement...
Recherche...
Aucune correspondance
ParticleFamily.cc
1// -*- tab-width: 2; indent-tabs-mode: nil; coding: utf-8-with-signature -*-
2//-----------------------------------------------------------------------------
3// Copyright 2000-2024 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/* ParticleFamily.cc (C) 2000-2024 */
9/* */
10/* Famille de particules. */
11/*---------------------------------------------------------------------------*/
12/*---------------------------------------------------------------------------*/
13
14#include "arcane/mesh/ParticleFamily.h"
15
16#include "arcane/utils/FatalErrorException.h"
17#include "arcane/utils/PlatformUtils.h"
18
19#include "arcane/core/ISubDomain.h"
20#include "arcane/core/IMesh.h"
21#include "arcane/core/IVariableMng.h"
22#include "arcane/core/Properties.h"
23#include "arcane/core/ItemPrinter.h"
24
25#include "arcane/mesh/ItemsExchangeInfo2.h"
26#include "arcane/mesh/DynamicMesh.h"
27#include "arcane/mesh/ItemConnectivitySelector.h"
28#include "arcane/mesh/DynamicMeshKindInfos.h"
29
30/*---------------------------------------------------------------------------*/
31/*---------------------------------------------------------------------------*/
32
33namespace Arcane::mesh
34{
35
36/*---------------------------------------------------------------------------*/
37/*---------------------------------------------------------------------------*/
38
39ParticleFamily::
40ParticleFamily(IMesh* mesh, const String& name)
41: ItemFamily(mesh, IK_Particle, name)
42, m_particle_type_info(nullptr)
43, m_particle_shared_info(nullptr)
44, m_sub_domain_id(NULL_SUB_DOMAIN_ID)
45, m_enable_ghost_items(false)
46, m_cell_connectivity(nullptr)
47{
48}
49
50/*---------------------------------------------------------------------------*/
51/*---------------------------------------------------------------------------*/
52
53ParticleFamily::
54~ParticleFamily()
55{
56}
57
58/*---------------------------------------------------------------------------*/
59/*---------------------------------------------------------------------------*/
60
61void ParticleFamily::
62build()
63{
64 ItemFamily::build();
65 ItemTypeMng* itm = m_mesh->itemTypeMng();
66 m_particle_type_info = itm->typeFromId(IT_NullType);
67 m_sub_domain_id = subDomain()->subDomainId();
68
69 // Temporaire: pour désactiver la table de hashage pour uniqueId
70 if (!platform::getEnvironmentVariable("ARCANE_PARTICLE_NO_UNIQUE_ID_MAP").null()) {
71 pwarning() << "TEMPORARY: suppress particule uniqueId map";
72 setHasUniqueIdMap(false);
73 }
74 else {
75 bool has_unique_id_map = !m_properties->getBool("no-unique-id-map");
76 _setHasUniqueIdMap(has_unique_id_map);
77 }
78
79 m_cell_connectivity = new CellConnectivity(this, mesh()->cellFamily(), "ParticleCell");
80
81 _addConnectivitySelector(m_cell_connectivity);
82
83 _buildConnectivitySelectors();
84
85 // Préalloue une maille par particule.
86 m_cell_connectivity->setPreAllocatedSize(1);
87
88 _setSharedInfo();
89
90 auto network = m_mesh->itemFamilyNetwork();
91 if (network) {
92 network->addDependency(this, mesh()->cellFamily(), m_cell_connectivity->customConnectivity(), false);
93 }
94}
95
96/*---------------------------------------------------------------------------*/
97/*---------------------------------------------------------------------------*/
102inline void ParticleFamily::
103_initializeNewlyAllocatedParticle(ItemInternal* particle, Int64 uid)
104{
105 _allocateInfos(particle, uid, m_particle_shared_info);
106 m_cell_connectivity->addConnectedItem(ItemLocalId(particle), ItemLocalId(NULL_ITEM_LOCAL_ID));
107}
108
109/*---------------------------------------------------------------------------*/
110/*---------------------------------------------------------------------------*/
111
112inline ItemInternal* ParticleFamily::
113_allocParticle(Int64 uid, bool& need_alloc)
114{
115 ItemInternal* ii = _allocOne(uid, need_alloc);
116
117 if (!need_alloc)
118 ii->setUniqueId(uid);
119 else
120 _initializeNewlyAllocatedParticle(ii, uid);
121
122 // Une particule appartient toujours au sous-domaine qui l'a créée
123 ii->setOwner(m_sub_domain_id, m_sub_domain_id);
124 return ii;
125}
126
127/*---------------------------------------------------------------------------*/
128/*---------------------------------------------------------------------------*/
129
130inline ItemInternal* ParticleFamily::
131_findOrAllocParticle(Int64 uid, bool& is_alloc)
132{
133 ItemInternal* ii = ItemFamily::_findOrAllocOne(uid, is_alloc);
134 if (is_alloc)
135 _initializeNewlyAllocatedParticle(ii, uid);
136 return ii;
137}
138
139/*---------------------------------------------------------------------------*/
140/*---------------------------------------------------------------------------*/
141
142ParticleVectorView ParticleFamily::
143addParticles(Int64ConstArrayView unique_ids, Int32ArrayView items)
144{
145 _addItems(unique_ids, items);
146 return view(items);
147}
148
149/*---------------------------------------------------------------------------*/
150/*---------------------------------------------------------------------------*/
151
152ParticleVectorView ParticleFamily::
153addParticles2(Int64ConstArrayView unique_ids,
154 Int32ConstArrayView owners,
155 Int32ArrayView items)
156{
157 addItems(unique_ids, owners, items);
158 return view(items);
159}
160
161/*---------------------------------------------------------------------------*/
162/*---------------------------------------------------------------------------*/
163
164void ParticleFamily::
165_setCell(ItemLocalId particle, ItemLocalId cell)
166{
167 m_cell_connectivity->replaceItem(particle, 0, cell);
168}
169
170/*---------------------------------------------------------------------------*/
171/*---------------------------------------------------------------------------*/
172
173ParticleVectorView ParticleFamily::
174addParticles(Int64ConstArrayView unique_ids,
175 Int32ConstArrayView cells_local_id,
176 Int32ArrayView items)
177{
178 _addItems(unique_ids, items);
179 Integer n = items.size();
180 for (Integer i = 0; i < n; ++i) {
181 _setCell(ItemLocalId(items[i]), ItemLocalId(cells_local_id[i]));
182 }
183
184 return view(items);
185}
186
187/*---------------------------------------------------------------------------*/
188/*---------------------------------------------------------------------------*/
189
190void ParticleFamily::
191setParticleCell(Particle particle, Cell new_cell)
192{
193 _setCell(particle, new_cell);
194 m_need_prepare_dump = true;
195}
196
197/*---------------------------------------------------------------------------*/
198/*---------------------------------------------------------------------------*/
199
200void ParticleFamily::
201setParticlesCell(ParticleVectorView particles, CellVectorView new_cells)
202{
203 Int32ConstArrayView cell_ids = new_cells.localIds();
204 Int32ConstArrayView particle_ids = particles.localIds();
205 for (Integer i = 0, n = particle_ids.size(); i < n; ++i)
206 _setCell(ItemLocalId(particle_ids[i]), ItemLocalId(cell_ids[i]));
207 m_need_prepare_dump = true;
208}
209
210/*---------------------------------------------------------------------------*/
211/*---------------------------------------------------------------------------*/
212
213void ParticleFamily::
214_addItems(Int64ConstArrayView unique_ids, Int32ArrayView items)
215{
216 Integer nb_item = unique_ids.size();
217 if (nb_item == 0)
218 return;
219 preAllocate(nb_item);
220
221 bool need_alloc = false;
222 for (Integer i = 0; i < nb_item; ++i) {
223 Int64 uid = unique_ids[i];
224 ItemInternal* ii = _allocParticle(uid, need_alloc);
225 items[i] = ii->localId();
226 }
227
228 m_need_prepare_dump = true;
229 _printInfos(nb_item);
230}
231
232/*---------------------------------------------------------------------------*/
233/*---------------------------------------------------------------------------*/
234
235void ParticleFamily::
236addItems(Int64ConstArrayView unique_ids, Int32ConstArrayView owners, Int32ArrayView items)
237{
238 Integer nb_item = unique_ids.size();
239 if (nb_item == 0)
240 return;
241 preAllocate(nb_item);
242 // La méthode _findOrAlloc nécessite la table de hashage des uniqueId().
243 if (!hasUniqueIdMap())
244 ARCANE_FATAL("Can not add particles with owners when hasUniqueIdMap()==false family={0}",
245 name());
246
247 bool need_alloc = false;
248 for (Integer i = 0; i < nb_item; ++i) {
249 Int64 uid = unique_ids[i];
250 ItemInternal* ii = _findOrAllocParticle(uid, need_alloc);
251 ii->setOwner(owners[i], m_sub_domain_id);
252
253 items[i] = ii->localId();
254 }
255
256 m_need_prepare_dump = true;
257 _printInfos(nb_item);
258}
259
260/*---------------------------------------------------------------------------*/
261/*---------------------------------------------------------------------------*/
262
263void ParticleFamily::
264exchangeParticles()
265{
266 ItemsExchangeInfo2 ex(this);
269 ex.prepareToSend();
270 ex.processExchange();
271 ex.removeSentItems();
274 endUpdate();
275 ex.readGroups();
276 ex.readVariables();
277}
278
279/*---------------------------------------------------------------------------*/
280/*---------------------------------------------------------------------------*/
281
282void ParticleFamily::
283_printInfos(Integer nb_added)
284{
285 ARCANE_UNUSED(nb_added);
286}
287
288/*---------------------------------------------------------------------------*/
289/*---------------------------------------------------------------------------*/
290
291void ParticleFamily::
292internalRemoveItems(Int32ConstArrayView local_ids, bool keep_ghost)
293{
294 ARCANE_UNUSED(keep_ghost);
295
296 InternalConnectivityPolicy policy = mesh()->_connectivityPolicy();
297
298 // A noter que cette boucle n'est pas utile pour le moment si on
299 // n'utilise pas les nouvelles connectivités.
300 // Elle le sera si on souhaite avoir des connectivités inverses
301 // maille->particule.
302 bool want_nullify_cell = (policy != InternalConnectivityPolicy::Legacy);
303 ItemLocalId null_item_lid(NULL_ITEM_LOCAL_ID);
304
305 auto network = m_mesh->itemFamilyNetwork();
306 if (network) {
307 std::set<Int32> remove_lids;
308 if (want_nullify_cell) {
309 for (Integer i = 0, n = local_ids.size(); i < n; ++i) {
310 ItemLocalId lid(local_ids[i]);
311 m_cell_connectivity->replaceItem(lid, 0, null_item_lid);
312 remove_lids.insert(local_ids[i]);
313 }
314 }
315 for (auto parent_connectivity : m_mesh->itemFamilyNetwork()->getParentConnectivities(this)) {
316 //for (auto parent_connectivity : m_mesh->itemFamilyNetwork()->getParentRelations(this)) { // Should be getParentConnectivities, but because legacy connectivity cannot remove a connectivity with a Node as target, we need to restrain to Relations...
317 for (auto source_item : parent_connectivity->sourceFamily()->itemsInternal()) {
318 if (source_item->isSuppressed())
319 continue;
320 ConnectivityItemVector connectivity_accessor(parent_connectivity);
321 ENUMERATE_ITEM (connected_item, connectivity_accessor.connectedItems(ItemLocalId(source_item))) {
322 if (remove_lids.find(connected_item->localId()) != remove_lids.end()) {
323 parent_connectivity->removeConnectedItem(ItemLocalId(source_item), connected_item);
324 }
325 }
326 }
327 }
328 }
329 else {
330 if (want_nullify_cell) {
331 for (Integer i = 0, n = local_ids.size(); i < n; ++i) {
332 ItemLocalId lid(local_ids[i]);
333 m_cell_connectivity->replaceItem(lid, 0, null_item_lid);
334 }
335 }
336 }
337 _removeMany(local_ids);
338
339 m_need_prepare_dump = true;
340}
341
342/*---------------------------------------------------------------------------*/
343/*---------------------------------------------------------------------------*/
344
345void ParticleFamily::
346removeParticles(Int32ConstArrayView items_local_id)
347{
348 internalRemoveItems(items_local_id, false);
349}
350
351/*---------------------------------------------------------------------------*/
352/*---------------------------------------------------------------------------*/
353
354void ParticleFamily::
355prepareForDump()
356{
357 Integer nb_item = nbItem();
358 info(4) << "ParticleFamily::prepareForDump: " << name()
359 << " n=" << nb_item;
361}
362
363/*---------------------------------------------------------------------------*/
364/*---------------------------------------------------------------------------*/
365
366void ParticleFamily::
367readFromDump()
368{
370 // Actualise le shared_info car il peut changer suite à une relecture
371 _setSharedInfo();
372}
373
374/*---------------------------------------------------------------------------*/
375/*---------------------------------------------------------------------------*/
376
377void ParticleFamily::
378preAllocate(Integer nb_item)
379{
380 Integer nb_hash = itemsMap().nbBucket();
381 Integer wanted_size = 2 * (nb_item + nbItem());
382 if (nb_hash < wanted_size)
383 itemsMap().resize(wanted_size, true);
384}
385
386/*---------------------------------------------------------------------------*/
387/*---------------------------------------------------------------------------*/
388
389void ParticleFamily::
390setHasUniqueIdMap(bool v)
391{
392 _setHasUniqueIdMap(v);
393 m_properties->setBool("no-unique-id-map", !v);
394}
395
396/*---------------------------------------------------------------------------*/
397/*---------------------------------------------------------------------------*/
398
399bool ParticleFamily::
400hasUniqueIdMap() const
401{
402 return _infos().hasUniqueIdMap();
403}
404
405/*---------------------------------------------------------------------------*/
406/*---------------------------------------------------------------------------*/
407
408void ParticleFamily::
409_setSharedInfo()
410{
411 m_particle_shared_info = _findSharedInfo(m_particle_type_info);
412}
413
414/*---------------------------------------------------------------------------*/
415/*---------------------------------------------------------------------------*/
416
417void ParticleFamily::
418checkValidConnectivity()
419{
421}
422
423/*---------------------------------------------------------------------------*/
424/*---------------------------------------------------------------------------*/
425
426void ParticleFamily::
427removeNeedRemoveMarkedItems()
428{
429 if (getEnableGhostItems()) {
430 UniqueArray<Integer> lids_to_remove;
431 lids_to_remove.reserve(1000);
432
433 ItemInternalMap& particle_map = itemsMap();
434 particle_map.eachItem([&](Item item) {
435 Integer f = item.itemBase().flags();
436 if (f & ItemFlags::II_NeedRemove) {
437 f &= ~ItemFlags::II_NeedRemove;
438 item.mutableItemBase().setFlags(f);
439 lids_to_remove.add(item.localId());
440 }
441 });
442
443 info() << "Number of particles of family " << name() << " to remove: " << lids_to_remove.size();
444 if (lids_to_remove.size() > 0)
445 removeParticles(lids_to_remove);
446 }
447}
448
449/*---------------------------------------------------------------------------*/
450/*---------------------------------------------------------------------------*/
451
452} // End namespace Arcane::mesh
453
454/*---------------------------------------------------------------------------*/
455/*---------------------------------------------------------------------------*/
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
#define ENUMERATE_ITEM(name, group)
Enumérateur générique d'un groupe de noeuds.
Integer size() const
Nombre d'éléments du vecteur.
constexpr Integer size() const noexcept
Retourne la taille du tableau.
void reserve(Int64 new_capacity)
Réserve le mémoire pour new_capacity éléments.
void add(ConstReferenceType val)
Ajoute l'élément val à la fin du tableau.
Maille d'un maillage.
Definition Item.h:1191
Gère la récupération des informations de connectivité.
ItemVectorView connectedItems(ItemLocalId item)
Retourne les entités connectées à item.
constexpr Integer size() const noexcept
Nombre d'éléments du tableau.
Int32 flags() const
Flags de l'entité
Int32 localId() const
Numéro local (au sous-domaine) de l'entité
Flags pour les caractéristiques des entités.
Definition ItemFlags.h:31
@ II_NeedRemove
L'entité doit être supprimé
Definition ItemFlags.h:55
Structure interne d'une entité de maillage.
Index d'un Item dans une variable.
Definition ItemLocalId.h:41
Int32ConstArrayView localIds() const
Tableau des numéros locaux des entités.
Classe de base d'un élément de maillage.
Definition Item.h:83
impl::MutableItemBase mutableItemBase() const
Partie interne modifiable de l'entité.
Definition Item.h:374
constexpr Int32 localId() const
Identifiant local de l'entité dans le sous-domaine du processeur.
Definition Item.h:219
impl::ItemBase itemBase() const
Partie interne de l'entité.
Definition Item.h:363
void setOwner(Integer suid, Int32 current_sub_domain)
Positionne le numéro du sous-domaine propriétaire de l'entité.
void setFlags(Int32 f)
Positionne les flags de l'entité
Particule.
Definition Item.h:1397
TraceMessage info() const
Flot pour un message d'information.
Vecteur 1D de données avec sémantique par valeur (style STL).
Famille d'entités.
Definition ItemFamily.h:76
IMesh * mesh() const override
Maillage associé
void prepareForDump() override
Prépare les données pour une protection.
void checkValidConnectivity() override
Vérification de la validité des structures internes concernant la connectivité.
ItemVectorView view(Int32ConstArrayView local_ids) override
Vue sur les entités.
void notifyItemsOwnerChanged() override
Notifie que les entités propres au sous-domaine de la famille ont été modifiées.
void readFromDump() override
Relit les données à partir d'une protection.
void eachItem(const Lambda &lambda)
Fonction template pour itérer sur les entités de l'instance.
Informations pour échanger des entités d'une famille donnée et leur caractéristiques.
void readAndAllocItems() override
Après réception des messages, lit et créé les entités transférées.
void readVariables() override
Après réception des messages, lit les valeurs des variables.
void removeSentItems() override
Supprime les entités envoyées.
void computeExchangeItems() override
Détermine la liste des entités à échanger.
void processExchange() override
Envoie les messages d'échange.
bool computeExchangeInfos() override
Détermine les informations nécessaires pour les échanges.
void prepareToSend() override
Prépare les structures d'envoie.
void readGroups() override
Après réception des messages, lit les groupes.
Integer nbItem() const override
Nombre d'entités.
String name() const override
Nom de la famille.
void endUpdate() override
Notifie la fin de modification de la liste des entités.
bool getEnableGhostItems() const override
récupère le flag pour gérer les particules ghost de la famille
ItemVectorViewT< Cell > CellVectorView
Vue sur un vecteur de mailles.
Definition ItemTypes.h:304
ItemVectorViewT< Particle > ParticleVectorView
Vue sur un vecteur de particules.
Definition ItemTypes.h:309
std::int64_t Int64
Type entier signé sur 64 bits.
Int32 Integer
Type représentant un entier.
ConstArrayView< Int32 > Int32ConstArrayView
Equivalent C d'un tableau à une dimension d'entiers 32 bits.
Definition UtilsTypes.h:569
ConstArrayView< Int64 > Int64ConstArrayView
Equivalent C d'un tableau à une dimension d'entiers 64 bits.
Definition UtilsTypes.h:567
ArrayView< Int32 > Int32ArrayView
Equivalent C d'un tableau à une dimension d'entiers 32 bits.
Definition UtilsTypes.h:540
@ IK_Particle
Entité de maillage de genre particule.
InternalConnectivityPolicy
Politique d'utilisation des connectivités.
Definition ItemTypes.h:516
@ Legacy
Connectivités historiques.
Definition ItemTypes.h:525