add SafeList destructor which calls maybe_cleanup() to prevent mem leak
This commit is contained in:
parent
ac591d9904
commit
4f3769fd75
1 changed files with 26 additions and 4 deletions
|
@ -203,7 +203,7 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calling this will cause zero to many deallocations.
|
// Calling this will cause zero to many deallocations.
|
||||||
void maybe_cleanup() {
|
bool maybe_cleanup() {
|
||||||
SafeListNode *cursor = nullptr;
|
SafeListNode *cursor = nullptr;
|
||||||
SafeListNode *new_graveyard_head = nullptr;
|
SafeListNode *new_graveyard_head = nullptr;
|
||||||
do {
|
do {
|
||||||
|
@ -212,7 +212,7 @@ public:
|
||||||
if (active_iterator_count.load() != 0) {
|
if (active_iterator_count.load() != 0) {
|
||||||
// It's not safe to clean up with an active iterator, because that iterator
|
// It's not safe to clean up with an active iterator, because that iterator
|
||||||
// could be pointing to an element that we want to delete.
|
// could be pointing to an element that we want to delete.
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
// Any iterator created after this point will never point to a deleted node.
|
// Any iterator created after this point will never point to a deleted node.
|
||||||
// Swap it out with the current graveyard head.
|
// Swap it out with the current graveyard head.
|
||||||
|
@ -225,6 +225,17 @@ public:
|
||||||
tmp->deletion_fn(tmp->val);
|
tmp->deletion_fn(tmp->val);
|
||||||
memdelete_allocator<SafeListNode, A>(tmp);
|
memdelete_allocator<SafeListNode, A>(tmp);
|
||||||
}
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
~SafeList() {
|
||||||
|
#ifdef DEBUG_ENABLED
|
||||||
|
if (!maybe_cleanup()) {
|
||||||
|
ERR_PRINT("There are still iterators around when destructing a SafeList. Memory will be leaked. This is a bug.");
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
maybe_cleanup();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -353,11 +364,11 @@ public:
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calling this will cause zero to many deallocations.
|
// Calling this will cause zero to many deallocations.
|
||||||
void maybe_cleanup() {
|
bool maybe_cleanup() {
|
||||||
SafeListNode *cursor = graveyard_head;
|
SafeListNode *cursor = graveyard_head;
|
||||||
if (active_iterator_count != 0) {
|
if (active_iterator_count != 0) {
|
||||||
// It's not safe to clean up with an active iterator, because that iterator could be pointing to an element that we want to delete.
|
// It's not safe to clean up with an active iterator, because that iterator could be pointing to an element that we want to delete.
|
||||||
return;
|
return false;
|
||||||
}
|
}
|
||||||
graveyard_head = nullptr;
|
graveyard_head = nullptr;
|
||||||
// Our graveyard list is now unreachable by any active iterators, detached from the main graveyard head and ready for deletion.
|
// Our graveyard list is now unreachable by any active iterators, detached from the main graveyard head and ready for deletion.
|
||||||
|
@ -367,6 +378,17 @@ public:
|
||||||
tmp->deletion_fn(tmp->val);
|
tmp->deletion_fn(tmp->val);
|
||||||
memdelete_allocator<SafeListNode, A>(tmp);
|
memdelete_allocator<SafeListNode, A>(tmp);
|
||||||
}
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
~SafeList() {
|
||||||
|
#ifdef DEBUG_ENABLED
|
||||||
|
if (!maybe_cleanup()) {
|
||||||
|
ERR_PRINT("There are still iterators around when destructing a SafeList. Memory will be leaked. This is a bug.");
|
||||||
|
}
|
||||||
|
#else
|
||||||
|
maybe_cleanup();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue