Arcane  v3.16.0.0
Documentation développeur
Chargement...
Recherche...
Aucune correspondance
DynamicMeshKindInfos.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/* DynamicMeshKindInfos.cc (C) 2000-2025 */
9/* */
10/* Infos de maillage pour un genre d'entité donnée. */
11/*---------------------------------------------------------------------------*/
12/*---------------------------------------------------------------------------*/
13
14#include "arcane/utils/FatalErrorException.h"
15#include "arcane/utils/PlatformUtils.h"
16#include "arcane/utils/NotImplementedException.h"
17#include "arcane/utils/StringBuilder.h"
18
19#include "arcane/core/IMesh.h"
20#include "arcane/core/IItemFamily.h"
22#include "arcane/core/IParallelMng.h"
23#include "arcane/core/ItemPrinter.h"
24#include "arcane/core/VariableTypes.h"
25#include "arcane/core/Timer.h"
26#include "arcane/core/IItemInternalSortFunction.h"
27#include "arcane/core/ItemFamilyCompactInfos.h"
28#include "arcane/core/IMeshCompacter.h"
29#include "arcane/core/MeshPartInfo.h"
30#include "arcane/core/ItemFamilyItemListChangedEventArgs.h"
31
32#include "arcane/mesh/DynamicMeshKindInfos.h"
33#include "arcane/mesh/ItemFamily.h"
34
35#include <algorithm>
36#include <set>
37
38// #define ARCANE_DEBUG_MESH
39
40/*---------------------------------------------------------------------------*/
41/*---------------------------------------------------------------------------*/
42
43namespace Arcane::mesh
44{
45
46/*---------------------------------------------------------------------------*/
47/*---------------------------------------------------------------------------*/
48
50DynamicMeshKindInfos(IMesh* mesh,eItemKind kind,const String& kind_name)
52, m_mesh(mesh)
53, m_item_family(nullptr)
54, m_kind(kind)
55, m_kind_name(kind_name)
56, m_nb_item(0)
57, m_is_verbose(false)
58, m_use_new_finalize(true)
59, m_is_first_finalize(true)
60, m_has_unique_id_map(true)
61, m_compact_infos(nullptr)
63{
64}
65
66/*---------------------------------------------------------------------------*/
67/*---------------------------------------------------------------------------*/
68
74
75/*---------------------------------------------------------------------------*/
76/*---------------------------------------------------------------------------*/
77
78void DynamicMeshKindInfos::
79build()
80{
81 // Créé le groupe contenant toutes les entités de ce genre
82 {
83 // Si le nom de ce groupe change, il faut mettre a jour
84 // le nom equivalent dans VariableInfo.cc
85 StringBuilder str("All");
86 str += m_kind_name;
87 str += "s";
88 m_all_group_name = str.toString();
90 m_all_group.internal()->setIsAllItems();
91 m_all_group.setLocalToSubDomain(true);
92 }
93 if (!platform::getEnvironmentVariable("ARCANE_OLD_FINALIZE").null()){
94 m_use_new_finalize = false;
95 pwarning() << "USE OLD FINALIZE";
96 }
97 m_is_first_finalize = true;
98}
99
100/*---------------------------------------------------------------------------*/
101/*---------------------------------------------------------------------------*/
106{
107 Trace::Setter mci(traceMng(),"Mesh");
108
109 ItemGroupImpl& impl = *m_all_group.internal();
110 Integer internal_size = m_internals.size();
111
112 {
113 info(5) << A_FUNCNAME << " " << m_kind_name
114 << " nb_item=" << m_nb_item
115 << " internal size= " << internal_size
116 << " max_local_id=" << maxUsedLocalId();
117#if 0
118 for( Integer i=0; i<internal_size; ++i ){
119 ItemInternal* item = m_internals[i];
120 info() << "Item " << i << " lib=" << item->localId()
121 << " uid=" << item->uniqueId() << " owner=" << item->owner()
122 << " item=" << item
123 << " removed?=" << item->isSuppressed();
124 }
125#endif
126
127 }
128
129 Integer nb_added_items = m_added_items.size();
130 Integer nb_removed_items = m_removed_items.size();
131 bool test_mode = m_use_new_finalize;
132
133 info(5) << A_FUNCNAME << " " << m_kind_name
134 << " nb_item " << m_nb_item
135 << " internal size " << internal_size
136 << " nb_add=" << nb_added_items
137 << " nb_remove=" << nb_removed_items;
138
139 // GG: passage en OLD le 11/2018. Si tout est OK du côté IFPEN
140 // on pourra définitivement supprimer cela.
141#if OLD
142 IParallelMng* pm = m_mesh->parallelMng();
143 // En séquentiel, positionne le champs m_owner de chaque entité.
144 // TODO: ne doit pas être fait ici, mais dans le lecteur de maillage...
145 if (!pm->isParallel() && nb_added_items!=0){
146 const Integer sid = pm->commRank();
147 for( Integer i=0; i<nb_added_items; ++i )
148 m_internals[ m_added_items[i] ]->setOwner(sid,sid);
149 }
150#endif
151 // (HP) TODO: On peut surement optimiser la réorganisation via des
152 // addItems, removeItems, changeIds...
153 // Le problème est qu'ici il ne faut pas impacter les sous-groupes calculés
154 impl.beginTransaction();
155 if (nb_added_items==0 && nb_removed_items!=0 && test_mode){
156 impl.removeItems(m_removed_items,true);
157 } else if (nb_added_items!=0 && nb_removed_items==0 && test_mode){
158 impl.addItems(m_added_items,true);
159 } else {
160 impl.removeAddItems(m_removed_items,m_added_items,true);
161 }
162 impl.endTransaction();
163
164 // restore integrity of allItems group after this agressive modification
165 impl.checkNeedUpdate();
166
167#if defined(ARCANE_DEBUG_MESH)
168 if (arcaneIsCheck())
169 m_all_group.checkValid();
170#endif
171 // Il faut aussi changer les groupes des entités propres
172 // (A faire uniquement en parallele ?)
173 m_item_family->notifyItemsOwnerChanged();
174
175#if defined(ARCANE_DEBUG_MESH)
176 if (arcaneIsCheck())
177 checkValid();
178#endif
179
180 m_removed_items.clear();
181 m_added_items.clear();
182}
183
184/*---------------------------------------------------------------------------*/
185/*---------------------------------------------------------------------------*/
186
187void DynamicMeshKindInfos::
188itemsUniqueIdToLocalId(ArrayView<Int64> ids,bool do_fatal) const
189{
190 if (!m_has_unique_id_map)
191 _badUniqueIdMap();
192 if (!arcaneIsCheck()){
193 if (do_fatal){
194 for( Integer i=0, s=ids.size(); i<s; ++i ){
195 Int64 unique_id = ids[i];
196 ids[i] = m_items_map.findLocalId(unique_id);
197 }
198 }
199 else{
200 for( Integer i=0, s=ids.size(); i<s; ++i ){
201 Int64 unique_id = ids[i];
202 ids[i] = m_items_map.tryFindLocalId(unique_id);
203 }
204 }
205 }
206 else{
207 Integer nb_error = 0;
208 for( Integer i=0, s=ids.size(); i<s; ++i ){
209 Int64 unique_id = ids[i];
210 ids[i] = m_items_map.tryFindLocalId(unique_id);
211 if ((ids[i] == NULL_ITEM_LOCAL_ID) && do_fatal) {
212 if (nb_error<10){
213 error() << "DynamicMeshKindInfos::itemsUniqueIdToLocalId() can't find "
214 << "entity " << m_kind_name << " with global id "
215 << unique_id << " in the subdomain.";
216 }
217 ++nb_error;
218 }
219 }
220 if (nb_error!=0){
221 if (do_fatal)
222 ARCANE_FATAL("{0} entities not found",nb_error);
223 }
224 }
225}
226
227/*---------------------------------------------------------------------------*/
228/*---------------------------------------------------------------------------*/
229
230void DynamicMeshKindInfos::
231itemsUniqueIdToLocalId(Int32ArrayView local_ids,
232 Int64ConstArrayView unique_ids,bool do_fatal) const
233{
234 if (!m_has_unique_id_map)
235 _badUniqueIdMap();
236 if (!arcaneIsCheck()){
237 if (do_fatal){
238 for( Integer i=0, s=unique_ids.size(); i<s; ++i ){
239 Int64 unique_id = unique_ids[i];
240 local_ids[i] = (unique_id == NULL_ITEM_UNIQUE_ID) ? NULL_ITEM_LOCAL_ID : m_items_map.findLocalId(unique_id);
241 }
242 }
243 else{
244 for( Integer i=0, s=unique_ids.size(); i<s; ++i ){
245 Int64 unique_id = unique_ids[i];
246 local_ids[i] = m_items_map.tryFindLocalId(unique_id);
247 }
248 }
249 }
250 else{
251 Integer nb_error = 0;
252 for( Integer i=0, s=unique_ids.size(); i<s; ++i ){
253 Int64 unique_id = unique_ids[i];
254 local_ids[i] = m_items_map.tryFindLocalId(unique_id);
255 if ((local_ids[i] == NULL_ITEM_LOCAL_ID) && do_fatal && unique_id != NULL_ITEM_UNIQUE_ID) {
256 if (nb_error<10){
257 error() << "DynamicMeshKindInfos::itemsUniqueIdToLocalId() can't find "
258 << "entity " << m_kind_name << " with global id "
259 << unique_id << " in the subdomain.";
260 }
261 ++nb_error;
262 }
263 }
264 if (nb_error!=0){
265 if (do_fatal)
266 ARCANE_FATAL("{0} entities not found",nb_error);
267 }
268 }
269}
270
271/*---------------------------------------------------------------------------*/
272/*---------------------------------------------------------------------------*/
273
274void DynamicMeshKindInfos::
275itemsUniqueIdToLocalId(Int32ArrayView local_ids,
276 ConstArrayView<ItemUniqueId> unique_ids,bool do_fatal) const
277{
278 if (!m_has_unique_id_map)
279 _badUniqueIdMap();
280 if (!arcaneIsCheck()){
281 if (do_fatal){
282 for( Integer i=0, s=unique_ids.size(); i<s; ++i ){
283 Int64 unique_id = unique_ids[i];
284 local_ids[i] = m_items_map.findLocalId(unique_id);
285 }
286 }
287 else{
288 for( Integer i=0, s=unique_ids.size(); i<s; ++i ){
289 Int64 unique_id = unique_ids[i];
290 local_ids[i] = m_items_map.tryFindLocalId(unique_id);
291 }
292 }
293 }
294 else{
295 Integer nb_error = 0;
296 for( Integer i=0, s=unique_ids.size(); i<s; ++i ){
297 Int64 unique_id = unique_ids[i];
298 local_ids[i] = m_items_map.tryFindLocalId(unique_id);
299 if ((local_ids[i] == NULL_ITEM_LOCAL_ID) && do_fatal) {
300 if (nb_error<10){
301 error() << "DynamicMeshKindInfos::itemsUniqueIdToLocalId() can't find "
302 << "entity " << m_kind_name << " with global id "
303 << unique_id << " in the subdomain.";
304 }
305 ++nb_error;
306 }
307 }
308 if (nb_error!=0){
309 if (do_fatal)
310 ARCANE_FATAL("{0} entities not found",nb_error);
311 }
312 }
313}
314
315/*---------------------------------------------------------------------------*/
316/*---------------------------------------------------------------------------*/
317
322
323/*---------------------------------------------------------------------------*/
324/*---------------------------------------------------------------------------*/
325
328{
329 m_all_group = m_mesh->findGroup(m_all_group_name);
330 // Supprime toutes les entités
331 m_nb_item = 0;
332 m_internals.clear();
333 _updateItemSharedInfoInternalView();
334 m_free_internals.clear();
335 m_items_map.clear();
338 m_free_internals_in_multi_buffer.clear();
339}
340
341/*---------------------------------------------------------------------------*/
342/*---------------------------------------------------------------------------*/
343
346{
347 Int32 sid = m_mesh->meshPartInfo().partRank();
348
349 debug() << "DynamicMeshKindInfos::checkValid(): " << m_kind_name;
350
351 Integer nb_killed_item = 0;
352 Integer nb_internal = m_internals.size();
353 Integer nb_error = 0;
354
355 // Premièrement, le item->localId() doit correspondre à l'indice
356 // dans le tableau m_internal
357 for( Integer i=0; i<nb_internal; ++i ){
358 ItemInternal* item = m_internals[i];
359 if (item->localId()!=i && nb_error<10){
360 error() << "The local id (" << item->localId() << ") of the entity "
361 << m_kind_name << ':' << item->uniqueId() << " is not "
362 << "consistent with its internal value (" << i << ")";
363 ++nb_error;
364 }
365 if (item->isSuppressed()){
366 ++nb_killed_item;
367 continue;
368 }
369 if (item->owner()==NULL_SUB_DOMAIN_ID && nb_error<10){
370 error() << "entity " << m_kind_name << ":lid=" << item->localId()
371 << ":uid=" << item->uniqueId() << " belongs to no subdomain";
372 ++nb_error;
373 }
374
375 if (item->owner()!=sid && item->isOwn() && nb_error<10){
376 error() << "entity " << m_kind_name << ":lid=" << item->localId()
377 << ":uid=" << item->uniqueId() << " incoherence between isOwn() and owner()"
378 << " " << item->owner() << ' ' << sid;
379 ++nb_error;
380 }
381
382 }
383 Integer nb_free = m_free_internals.size();
384 if (nb_killed_item!=nb_free){
385 error() << "DynamicMeshKindInfos::checkValid(): " << m_kind_name
386 << ": incoherence between killed and free entities"
387 << " free=" << nb_free
388 << " killed=" << nb_killed_item
389 << " internal=" << nb_internal
390 << " count=" << m_nb_item;
391 ++nb_error;
392 }
393
394 if (nb_error!=0)
395 ARCANE_FATAL("Internal error in the mesh structure mesh={0} part={1}",
396 m_mesh->name(),sid);
397}
398
399/*---------------------------------------------------------------------------*/
400/*---------------------------------------------------------------------------*/
401
402#ifdef OLD
403class ItemParticleCompareWithSuppression
404{
405 public:
406 bool operator()(const ItemInternal* item1,const ItemInternal* item2) const
407 {
408 // Il faut mettre les entités détruites en fin de liste
409 //cout << "Compare: " << item1->uniqueId() << " " << item2->uniqueId() << '\n';
410 bool s1 = item1->isSuppressed();
411 bool s2 = item2->isSuppressed();
412 if (s1 && !s2)
413 return false;
414 if (!s1 && s2)
415 return true;
416 Int64 uid1 = Particle(item1).cell(0)->uniqueId();
417 Int64 uid2 = Particle(item2).cell(0)->uniqueId();
418 if (uid1==uid2)
419 return item1->uniqueId() < item2->uniqueId();
420 return uid1<uid2;
421 }
422};
423#endif
424
425/*---------------------------------------------------------------------------*/
426/*---------------------------------------------------------------------------*/
427
429{
430 public:
431 bool operator()(const ItemLocalIdAndUniqueId& item1,const ItemLocalIdAndUniqueId item2) const
432 {
433 return item1.m_unique_id < item2.m_unique_id;
434 }
435 Integer m_local_id;
436 Integer m_unique_id;
437};
438
439/*---------------------------------------------------------------------------*/
440/*---------------------------------------------------------------------------*/
441
443oldToNewLocalIds() const
444{
445 if (!m_compact_infos)
446 return Int32ConstArrayView();
447 return m_compact_infos->oldToNewLocalIds();
448}
449
450/*---------------------------------------------------------------------------*/
451/*---------------------------------------------------------------------------*/
452
454newToOldLocalIds() const
455{
456 if (!m_compact_infos)
457 return Int32ConstArrayView();
458 return m_compact_infos->newToOldLocalIds();
459}
460
461/*---------------------------------------------------------------------------*/
462/*---------------------------------------------------------------------------*/
478{
479 m_compact_infos = &compact_infos;
480 bool do_sort = compact_infos.compacter()->isSorted();
481
482 if (arcaneIsCheck())
483 checkValid();
484
485 Integer nb_internal = m_internals.size();
486 info(4) << m_kind_name << " beginCompactItems() Family Internal Size=" << nb_internal
487 << " max_id=" << maxUsedLocalId()
488 << " nb_item=" << m_nb_item
489 << " do_sort=" << do_sort;
490
491 if (nb_internal==0)
492 return;
493
494 UniqueArray<Int32> old_to_new_local_ids(maxUsedLocalId());
495 UniqueArray<Int32> new_to_old_local_ids(m_nb_item);
496 // m_new_to_old_local_ids[lid] contient l'ancien numéro local de l'entité dont le nouveau numéro local est \a lid
497 // m_old_to_new_local_ids[lid] contient le nouveau numéro local de l'entité dont l'ancien numéro local est \a lid
498 old_to_new_local_ids.fill(NULL_ITEM_LOCAL_ID);
499 // après compactage, il ne peut y avoir que m_nb_item items actifs (et forcément en tête de numérotation)
500 new_to_old_local_ids.fill(NULL_ITEM_LOCAL_ID);
501
502 if (do_sort){
503 // Fait une copie temporaire des ItemInternal pour le tri
505
506 debug() << "BEGIN = " << items.data() << " " << m_internals.data()
507 << " S1=" << items.size() << " S2=" << m_internals.size();
508 const bool print_infos = false;
509 {
510 ITimerMng* tm = m_mesh->parallelMng()->timerMng();
511 Timer timer(tm,"DynamicMeshKindInfos::beginCompactItems()",Timer::TimerReal);
512 IItemInternalSortFunction* sort_func = m_item_family->itemSortFunction();
513 if (print_infos){
514 for( Integer i=0, is=items.size(); i<is; ++i ){
515 info() << "Before Sort: " << i << " uid=" << items[i]->uniqueId()
516 << " lid=" << items[i]->localId()
517 << " destroyed?=" << items[i]->isSuppressed();
518 }
519 }
520 {
521 Timer::Sentry sentry(&timer);
522 sort_func->sortItems(items);
523 }
524 info(4) << "Temps pour trier les entités <" << m_kind_name << "> "
525 << timer.lastActivationTime()
526 << " sort_func_name=" << sort_func->name();
527 info(4) << "Sort Infos: size=" << items.size() << " nb_item=" << m_nb_item;
528 if (print_infos){
529 for( Integer i=0, n=items.size(); i<n; ++i ){
530 info() << "After Sort: " << i << " uid=" << items[i]->uniqueId()
531 << " lid=" << items[i]->localId()
532 << " destroyed?=" << items[i]->isSuppressed();
533 }
534 }
535 }
536
537#ifdef ARCANE_TEST_ADD_MESH
538 typedef std::set<ItemInternal*,ItemCompare> ItemSet;
539 ItemSet items_set;
540 items_set.insert(m_internals.begin(),m_internals.end());
542 {
543 ItemSet::const_iterator b = items_set.begin();
544 ItemSet::const_iterator e = items_set.end();
545 for( Integer z=0; b!=e; ++b, ++z)
546 items[z] = *b;
547 }
548#endif
549
550 for( Integer i=0, n=m_nb_item; i<n; ++i ){
551 Integer current_local_id = items[i]->localId();
552 //info() << "Item: " << " uid=" << items[i]->uniqueId() << " lid="
553 //<< current_local_id << " newid=" << i;
554 old_to_new_local_ids[ current_local_id ] = i;
555 new_to_old_local_ids[ i ] = current_local_id;
556 }
557 }
558 else{
559 const bool use_old = true;
560 if (use_old){
561 info(4) << "USE_OLD_METHOD";
562 Int32ConstArrayView existing_items = m_all_group.internal()->itemsLocalId();
563
564 for( Integer i=0; i<m_nb_item; ++i ){
565 Integer current_local_id = existing_items[i];
566 old_to_new_local_ids[ current_local_id ] = i;
567 new_to_old_local_ids[ i ] = current_local_id;
568 }
569
570 // Place les éléments détruits en fin de liste et vérifie
571 // que le nombre de trous + le nombre d'entités est égal à #nb_internal
572 {
573 Integer old_index = m_nb_item;
574 Integer total_nb = maxUsedLocalId();
575 for( Integer i=0, n=total_nb; i<n; ++i ){
576 //bool is_destroyed = false;
577 if (old_to_new_local_ids[i]==NULL_ITEM_ID){
578 //is_destroyed = true;
579 old_to_new_local_ids[i] = old_index;
580 //m_new_to_old_local_ids[old_index] = i;
581 ++old_index;
582 }
583 //debug() << "OLD_TO_NEW i=" << i
584 // << " old=" << m_old_to_new_local_ids[i]
585 // << " destroyed=" << is_destroyed;
586 }
587 if (old_index!=total_nb){
588 ARCANE_FATAL("Family '{0}' bad indices: expected={1} found={2} (nb_internal={3})",
589 itemFamily()->name(),old_index,total_nb,nb_internal);
590 }
591 }
592 }
593 else{
594 // Compactage sans triage, ce qui revient à déplacer les trous à la fin
595
596 // Place les éléments détruits en fin de liste et vérifie
597 // que l'on a bien tout rangé (on conserve l'ordre relatif des items)
598 Integer new_index = 0;
599 Integer free_index = m_nb_item;
600 for( Integer i=0, n=maxUsedLocalId(); i<n; ++i ){
601 if (m_internals[i]->isSuppressed()) {
602 old_to_new_local_ids[i] = free_index;
603 // m_new_to_old_local_ids[free_index] = i; // non utilisé quand m_new_to_old_local_ids restreint à m_nb_item
604 ++free_index;
605 } else {
606 old_to_new_local_ids[i] = new_index;
607 new_to_old_local_ids[new_index] = i;
608 ++new_index;
609 }
610 //debug() << "OLD_TO_NEW i=" << i
611 // << " old=" << m_old_to_new_local_ids[i]
612 // << " destroyed=" << m_internals[i]->isSuppressed();
613 }
614 if (new_index!=m_nb_item){
615 ARCANE_FATAL("Family '{0}' bad indices: expected={1} found={2}",
616 itemFamily()->name(),m_nb_item,new_index);
617 }
618 }
619 }
620 compact_infos.setOldToNewLocalIds(std::move(old_to_new_local_ids));
621 compact_infos.setNewToOldLocalIds(std::move(new_to_old_local_ids));
622}
623
624/*---------------------------------------------------------------------------*/
625/*---------------------------------------------------------------------------*/
626
627void DynamicMeshKindInfos::
628finishCompactItems(ItemFamilyCompactInfos& compact_infos)
629{
630 if (arcaneIsCheck())
631 checkValid();
632
633 Int32ConstArrayView old_to_new_local_ids = compact_infos.oldToNewLocalIds();
634 Int32ConstArrayView new_to_old_local_ids = compact_infos.newToOldLocalIds();
635
636 Integer nb_internal = m_internals.size();
637
638 info(4) << m_kind_name << " Family Compression: nb_item=" << m_nb_item
639 << " nb_internal=" << nb_internal
640 << " buf_size=" << m_item_internals_buffer->bufferSize()
641 << " nb_allocated_buf=" << m_item_internals_buffer->nbAllocatedBuffer()
642 << " nb_free" << m_free_internals_in_multi_buffer.size();
643
644 if (nb_internal==0)
645 return;
646
647 // Finalise la réorganisation des structures après un tri
648 // 1. Il faut mettre à jour la structure m_items_map pour référencer
649 // le nouveau local_id
650 // 2. Il faut recopier les valeurs de chaque ItemInternal pour qu'il
651 // soit bien placé dans la nouvelle numérotation.
652 // IMPORTANT: Cette opération doit toujours être la dernière car ensuite
653 // on perd la relation entre les anciens local_ids et les nouveaux à
654 // travers cette structure
655 m_items_map._changeLocalIds(m_internals, old_to_new_local_ids);
656
657 if (m_is_verbose){
658 info() << "DumpItemsBefore:";
659 _dumpList();
660 }
661
662 // Pour l'instant, utilise un tableau temporaire
663 // TODO: tableau a supprimer
666 UniqueArray<Int32> new_owners(m_nb_item);
667 UniqueArray<Int32> new_flags(m_nb_item);
668 UniqueArray<Int16> new_typeids(m_nb_item);
669
670 for( Integer i=0, n=m_nb_item; i<n; ++i ) {
671 ItemInternal* old_item = m_internals[ new_to_old_local_ids[ i ] ];
672 new_uids[i] = old_item->uniqueId().asInt64();
673 new_owners[i] = old_item->owner();
674 new_flags[i] = old_item->flags();
675 new_typeids[i] = old_item->typeId();
676 new_items[i] = *old_item;
677 }
678
679 Integer nb_error = 0;
680 Int32 sid = m_mesh->meshPartInfo().partRank();
681
682 for( Integer i=0; i<m_nb_item; ++i ){
683 ItemInternal* ii = m_internals[i];
684 *ii = new_items[i];
685 ii->setLocalId(i);
686 ii->setFlags(new_flags[i]);
687 ii->setOwner(new_owners[i],sid);
688 ii->setUniqueId(new_uids[i]);
689 ItemSharedInfo* isi = ItemInternalCompatibility::_getSharedInfo(ii);
690 ii->_setSharedInfo(isi,ItemTypeId(new_typeids[i]));
691 // L'entité est marqué comme créée
692 //_setAdded(ii);
693#ifdef ARCANE_CHECK
694 if (ii->isSuppressed()){
695 if (nb_error<10)
696 error() << "Entity deleted from the list of created entities "
697 << ItemPrinter(ii) << " index=" << i;
698 ++nb_error;
699 }
700#endif /* ARCANE_CHECK */
701 }
702
703 // Il faut remplir les m_free_internals_in_multi_buffer avec les entités supprimées
704 // pour qu'elles puissent être réutilisées sinon on va réallouer
705 // des blocs dans m_item_internals_buffer à chaque recréation des
706 // entités
707 for( Integer i=m_nb_item, n=m_internals.size(); i<n; ++i ){
708 // Empile dans l'ordre inverse pour que allocOne()
709 // dépile dans l'ordre des m_internals et profite des
710 // effets de cache.
711 ItemInternal* removed_item = m_internals[m_nb_item + n-(i+1)];
712 m_free_internals_in_multi_buffer.add(removed_item);
713 }
714
715 // Retaillage optimal des tableaux
716 m_internals.resize(m_nb_item);
717 _updateItemSharedInfoInternalView();
718 m_free_internals.clear();
719
720 if (m_is_verbose){
721 info() << "DumpItemsAfter:";
722 _dumpList();
723 }
724
725 if (nb_error!=0)
726 ARCANE_FATAL("Error in compacting nb_error={0}",nb_error);
727
728 debug() << "Compression: old=" << nb_internal << " new=" << m_nb_item
729 << " internal=" << m_internals.size()
730 << " free=" << m_free_internals.size();
731
732 // Vide les tableaux de 'compact_infos' car ils ne sont plus valides.
733 compact_infos.clear();
734
735 m_compact_infos = nullptr;
737}
738
739/*---------------------------------------------------------------------------*/
740/*---------------------------------------------------------------------------*/
741
742namespace{
743 class _Inverse
744 {
745 public:
746 bool operator()(Int32 i1,Int32 i2) const
747 {
748 return i1>i2;
749 }
750 };
751}
752
753/*---------------------------------------------------------------------------*/
754/*---------------------------------------------------------------------------*/
755
757clear()
758{
759 Int32ConstArrayView items_id = m_all_group.internal()->itemsLocalId();
760
761 // Pour réutiliser au moins les m_free_internals afin
762 // d'avoir des localId() consécutifs, il faut trier
763 // les ids par ordre décroissant.
764 Int32UniqueArray ids(items_id);
765 std::sort(std::begin(ids),std::end(ids),_Inverse());
766
767 removeMany(ids);
768 //printFreeInternals(10000);
769}
770
771/*---------------------------------------------------------------------------*/
772/*---------------------------------------------------------------------------*/
773
774void DynamicMeshKindInfos::
775printFreeInternals(Integer max_print)
776{
777 Integer nb_free = m_free_internals.size();
778 max_print = math::min(nb_free,max_print);
779 for( Integer i=0; i<max_print; ++i ){
780 Integer pos = (nb_free-1)-i;
781 Int32 index = m_free_internals[pos];
782 ItemInternal* iitem = m_internals[index];
783 info() << "I=" << i << " pos=" << pos << " index=" << index
784 << " local_id=" << iitem->localId();
785 }
786}
787
788/*---------------------------------------------------------------------------*/
789/*---------------------------------------------------------------------------*/
790
793{
794 Int64ArrayView uids = *(m_item_family->uniqueIds());
795 Integer nb_item = local_ids.size();
796
797 if (m_has_unique_id_map){
798 for( Integer i=0; i<nb_item; ++i ){
799 Int32 lid = local_ids[i];
800 ItemInternal* item = m_internals[lid];
801#ifdef ARCANE_CHECK
802 _checkActiveItem(item);
803#endif
804 _setSuppressed(item);
805 if (uids[lid] != NULL_ITEM_UNIQUE_ID)
806 m_items_map.remove(uids[lid]);
807 }
808 }
809 else{
810 for( Integer i=0; i<nb_item; ++i ){
811 Int32 lid = local_ids[i];
812 ItemInternal* item = m_internals[lid];
813#ifdef ARCANE_CHECK
814 _checkActiveItem(item);
815#endif
816 _setSuppressed(item);
817 }
818 }
819
820 m_nb_item -= nb_item;
821 Integer nb_removed = m_removed_items.size();
822 m_removed_items.resize(nb_removed + nb_item);
823 memcpy(m_removed_items.data()+nb_removed,local_ids.data(),sizeof(Int32)*nb_item);
824 Integer nb_free = m_free_internals.size();
825 m_free_internals.resize(nb_free + nb_item);
826 memcpy(m_free_internals.data()+nb_free,local_ids.data(),sizeof(Int32)*nb_item);
827}
828
829/*---------------------------------------------------------------------------*/
830/*---------------------------------------------------------------------------*/
831
832void DynamicMeshKindInfos::
833setHasUniqueIdMap(bool v)
834{
835 if (m_nb_item!=0)
836 ARCANE_FATAL("family is not empty");
837 m_has_unique_id_map = v;
838}
839
840/*---------------------------------------------------------------------------*/
841/*---------------------------------------------------------------------------*/
842
843void DynamicMeshKindInfos::
844_checkActiveItem(ItemInternal* item)
845{
846 if (item->isSuppressed())
847 ARCANE_FATAL("Attempting to remove an entity already deleted item={0}",
848 ItemPrinter(item));
849}
850
851/*---------------------------------------------------------------------------*/
852/*---------------------------------------------------------------------------*/
853
854void DynamicMeshKindInfos::
855_dumpList()
856{
857 Integer nb_internal = m_internals.size();
858 Integer computed_nb_item = 0;
859 for( Integer i=0; i<nb_internal; ++i ){
860 ItemInternal* item = m_internals[i];
861 bool is_suppressed = item->isSuppressed();
862 if (is_suppressed)
863 ++computed_nb_item;
864 info() << "Item: INDEX=" << i
865 << " LID=" << item->localId()
866 << " UID=" << item->uniqueId()
867 << " KILLED=" << is_suppressed;
868 }
869 info() << "EndOfDump: "
870 << " nb_internal=" << nb_internal
871 << " nb_item=" << m_nb_item
872 << " computed_nb_item= " << (nb_internal-computed_nb_item);
873}
874
875/*---------------------------------------------------------------------------*/
876/*---------------------------------------------------------------------------*/
877
878void DynamicMeshKindInfos::
879_badSameUniqueId(Int64 unique_id) const
880{
881 ARCANE_FATAL("duplicate unique id family={0} uid={1}",
882 m_item_family->name(),unique_id);
883}
884
885/*---------------------------------------------------------------------------*/
886/*---------------------------------------------------------------------------*/
887
888void DynamicMeshKindInfos::
889_badUniqueIdMap() const
890{
891 if (m_has_unique_id_map)
892 ARCANE_FATAL("family have unique id map");
893 else
894 ARCANE_FATAL("family does not have unique id map");
895}
896
897/*---------------------------------------------------------------------------*/
898/*---------------------------------------------------------------------------*/
899
900void DynamicMeshKindInfos::
901setItemFamily(ItemFamily* item_family)
902{
903 m_item_family = item_family;
904 m_common_item_shared_info = item_family->commonItemSharedInfo();
905}
906
907/*---------------------------------------------------------------------------*/
908/*---------------------------------------------------------------------------*/
909
910void DynamicMeshKindInfos::
911_updateItemSharedInfoInternalView()
912{
913 if (m_common_item_shared_info)
914 m_common_item_shared_info->m_items_internal = m_internals.constView();
915}
916
917/*---------------------------------------------------------------------------*/
918/*---------------------------------------------------------------------------*/
919
920EventObservableView<const ItemFamilyItemListChangedEventArgs&> DynamicMeshKindInfos::
921itemListChangedEvent()
922{
923 return EventObservableView<const ItemFamilyItemListChangedEventArgs&>(m_item_list_change_event);
924}
925
926/*---------------------------------------------------------------------------*/
927/*---------------------------------------------------------------------------*/
928
929void DynamicMeshKindInfos::
930_notifyRemove2(ItemInternal* item)
931{
932 ItemFamilyItemListChangedEventArgs args(m_item_family,item->localId(),item->uniqueId());
933 m_item_list_change_event.notify(args);
934 args.setIsAdd(false);
935}
936
937/*---------------------------------------------------------------------------*/
938/*---------------------------------------------------------------------------*/
939
940void DynamicMeshKindInfos::
941_notifyAdd2(ItemInternal* item,Int64 uid)
942{
943 ItemFamilyItemListChangedEventArgs args(m_item_family,item->localId(),uid);
944 args.setIsAdd(true);
945 m_item_list_change_event.notify(args);
946}
947
948/*---------------------------------------------------------------------------*/
949/*---------------------------------------------------------------------------*/
950
951} // End namespace Arcane::mesh
952
953/*---------------------------------------------------------------------------*/
954/*---------------------------------------------------------------------------*/
955
#define ARCANE_FATAL(...)
Macro envoyant une exception FatalErrorException.
Types et macros pour itérer sur les entités du maillage.
Integer size() const
Nombre d'éléments du vecteur.
Vue modifiable d'un tableau d'un type T.
constexpr Integer size() const noexcept
Retourne la taille du tableau.
void fill(const DataType &data)
Remplissage du tableau.
const T * data() const
Accès à la racine du tableau hors toute protection.
constexpr const_pointer data() const noexcept
Pointeur sur la mémoire allouée.
constexpr Integer size() const noexcept
Nombre d'éléments du tableau.
Interface d'une fonction de tri des entités.
virtual void sortItems(ItemInternalMutableArrayView items)=0
Trie les entités du tableau items.
virtual const String & name() const =0
Nom de la fonction de tri.
virtual bool isSorted() const =0
Indique si souhaite trier les entités en plus de les compacter.
virtual const MeshPartInfo & meshPartInfo() const =0
Informations sur les parties du maillage.
Interface du gestionnaire de parallélisme pour un sous-domaine.
virtual Int32 commRank() const =0
Rang de cette instance dans le communicateur.
virtual bool isParallel() const =0
Retourne true si l'exécution est parallèle.
Interface d'un gestionnaire de timer.
Definition ITimerMng.h:53
bool isSuppressed() const
Vrai si l'entité est supprimée.
ItemUniqueId uniqueId() const
Numéro unique de l'entité
Int32 owner() const
Numéro du sous-domaine propriétaire de l'entité
Int32 flags() const
Flags de l'entité
Int16 typeId() const
Type de l'entité
bool isOwn() const
Vrai si l'entité appartient au sous-domaine.
Int32 localId() const
Numéro local (au sous-domaine) de l'entité
Informations pour gérer le compactage des entités d'une famille.
Int32ConstArrayView newToOldLocalIds() const
Conversion entre les nouveaux et les anciens id locaux.
Int32ConstArrayView oldToNewLocalIds() const
Conversion entre les anciens et les nouveaux id locaux.
Implémentation d'un groupe d'entités de maillage.
void setIsAllItems()
Indique à ce groupe qu'il s'agit du groupe de toutes les entités de la famille.
void setLocalToSubDomain(bool v)
Positionne le booléen indiquant si le groupe est local au sous-domaine.
Definition ItemGroup.h:238
ItemGroupImpl * internal() const
Retourne l'implémentation du groupe.
Definition ItemGroup.h:120
Structure interne d'une entité de maillage.
Classe utilitaire pour imprimer les infos sur une entité.
Definition ItemPrinter.h:35
ItemUniqueId uniqueId() const
Identifiant unique sur tous les domaines.
Definition Item.h:225
Tampon pour allocation multiple.
Definition MultiBuffer.h:44
Particule.
Definition Item.h:1397
Cell cell() const
Maille à laquelle appartient la particule. Il faut appeler setCell() avant d'appeler cette fonction....
Definition Item.h:1459
Constructeur de chaîne de caractère unicode.
Chaîne de caractères unicode.
Sentinelle pour le timer. La sentinelle associée à un timer permet de déclancher celui-ci au moment d...
Definition Timer.h:89
Gestion d'un timer.
Definition Timer.h:62
@ TimerReal
Timer utilisant le temps réel.
Definition Timer.h:76
Real lastActivationTime() const
Retourne le temps (en secondes) passé lors de la dernière activation du timer.
Definition Timer.h:219
TraceAccessor(ITraceMng *m)
Construit un accesseur via le gestionnaire de trace m.
TraceMessageDbg debug(Trace::eDebugLevel=Trace::Medium) const
Flot pour un message de debug.
TraceMessage info() const
Flot pour un message d'information.
TraceMessage error() const
Flot pour un message d'erreur.
ITraceMng * traceMng() const
Gestionnaire de trace.
TraceMessage pwarning() const
Positionne une classe de message.
Vecteur 1D de données avec sémantique par valeur (style STL).
Integer m_nb_item
Groupe de toutes les entités.
String m_kind_name
Nom du genre des entités (Node, Cell, ...)
Integer maxUsedLocalId() const
Numéro local le plus grand utilisé.
MultiBufferT< ItemInternal > * m_item_internals_buffer
Tampon pour stocker une instance de ItemInternal.
void finalizeMeshChanged()
Réalloue et recalcule les structures après modification du maillage.
void removeMany(Int32ConstArrayView local_ids)
Supprime une liste d'entités.
void prepareForDump()
Prépare les variables pour les sauvegardes.
void beginCompactItems(ItemFamilyCompactInfos &compact_infos)
Calcule les nouveaux id locaux des entités pour qu'ils soient consécutifs.
ARCANE_DEPRECATED_240 Int32ConstArrayView oldToNewLocalIds() const
Conversion entre les anciens et les nouveaux id locaux.
UniqueArray< ItemInternal * > m_internals
ItemInternal des entités.
void readFromDump()
Restaure les infos à partir des dumps.
DynamicMeshKindInfos(IMesh *mesh, eItemKind kind, const String &kind_name)
Créé une instance pour un maillage et un genre donnée.
ARCANE_DEPRECATED_240 Int32ConstArrayView newToOldLocalIds() const
Conversion entre les nouveaux et les anciens id locaux.
ItemFamilyCompactInfos * m_compact_infos
Temporaire tant que oldToNewLocalIds() et newToOldLocalIds() existent.
void checkValid()
Vérifie si les structures internes de l'instance sont valides.
~DynamicMeshKindInfos()
Libère les ressources.
String m_all_group_name
Nom du groupe contenant toutes les entités.
ItemInternalMap m_items_map
Table de hachage conversion uniqueId() -> ItemInternal*.
Int32UniqueArray m_free_internals
Liste des ItemInternal de m_item_internals_buffer libres.
void clear()
Supprime toutes les entités.
eItemKind m_kind
Genre correspondant.
ItemFamily * m_item_family
Famille de maillage associée.
Famille d'entités.
Definition ItemFamily.h:76
ItemGroup createGroup(const String &name, Int32ConstArrayView elements, bool do_override=false) override
Créé un groupe d'entités de nom name contenant les entités local_ids.
Int32 tryFindLocalId(Int64 key) const
Retourne le localId() associé à key si trouvé ou NULL_ITEM_LOCAL_ID sinon aucun.
Int32 findLocalId(Int64 uid) const
Retourne le numéro local de l'entité de numéro unique uid.
void _changeLocalIds(ArrayView< ItemInternal * > items_internal, ConstArrayView< Int32 > old_to_new_local_ids)
Change la valeurs des localId()
__host__ __device__ Real2 min(Real2 a, Real2 b)
Retourne le minimum de deux Real2.
Definition MathUtils.h:336
ARCCORE_BASE_EXPORT String getEnvironmentVariable(const String &name)
Variable d'environnement du nom name.
ArrayView< Int64 > Int64ArrayView
Equivalent C d'un tableau à une dimension d'entiers 64 bits.
Definition UtilsTypes.h:538
bool arcaneIsCheck()
Vrai si on est en mode vérification.
Definition Misc.cc:68
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
UniqueArray< Int32 > Int32UniqueArray
Tableau dynamique à une dimension d'entiers 32 bits.
Definition UtilsTypes.h:428
ArrayView< Int32 > Int32ArrayView
Equivalent C d'un tableau à une dimension d'entiers 32 bits.
Definition UtilsTypes.h:540
eItemKind
Genre d'entité de maillage.
std::int32_t Int32
Type entier signé sur 32 bits.