[libgee] Fix memory leak in Gee.LinkedList
- From: Maciej Marcin Piechotka <mpiechotka src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [libgee] Fix memory leak in Gee.LinkedList
- Date: Tue, 3 Aug 2010 09:45:52 +0000 (UTC)
commit da07618ad145a1e4928e29b24bfc7dcf639591f6
Author: Maciej Piechotka <uzytkownik2 gmail com>
Date: Tue Aug 3 11:44:06 2010 +0200
Fix memory leak in Gee.LinkedList
gee/linkedlist.vala | 75 +++++++++++++++++++++++++++++---------------------
1 files changed, 43 insertions(+), 32 deletions(-)
---
diff --git a/gee/linkedlist.vala b/gee/linkedlist.vala
index 020ef77..8fbf4c8 100644
--- a/gee/linkedlist.vala
+++ b/gee/linkedlist.vala
@@ -36,7 +36,7 @@ public class Gee.LinkedList<G> : AbstractList<G>, Queue<G>, Deque<G> {
private int _size = 0;
private int _stamp = 0;
private Node<G>? _head = null;
- private Node<G>? _tail = null;
+ private weak Node<G>? _tail = null;
/**
* The elements' equality testing function.
@@ -92,11 +92,12 @@ public class Gee.LinkedList<G> : AbstractList<G>, Queue<G>, Deque<G> {
public override bool add (G item) {
Node<G> n = new Node<G> (item);
if (this._head == null && this._tail == null) {
- this._head = this._tail = n;
+ this._tail = n;
+ this._head = (owned) n;
} else {
- this._tail.next = n;
n.prev = this._tail;
- this._tail = n;
+ this._tail.next = (owned) n;
+ this._tail = this._tail.next;
}
// Adding items to the list during iterations is allowed.
@@ -110,7 +111,7 @@ public class Gee.LinkedList<G> : AbstractList<G>, Queue<G>, Deque<G> {
* { inheritDoc}
*/
public override bool remove (G item) { // Should remove only the first occurence (a test should be added)
- for (Node<G> n = this._head; n != null; n = n.next) {
+ for (weak Node<G> n = this._head; n != null; n = n.next) {
if (this.equal_func (item, n.data)) {
this._remove_node (n);
return true;
@@ -124,7 +125,8 @@ public class Gee.LinkedList<G> : AbstractList<G>, Queue<G>, Deque<G> {
*/
public override void clear () {
++this._stamp;
- this._head = this._tail = null;
+ this._head = null;
+ this._tail = null;
this._size = 0;
}
@@ -181,18 +183,18 @@ public class Gee.LinkedList<G> : AbstractList<G>, Queue<G>, Deque<G> {
} else {
Node<G> n = new Node<G> (item);
if (index == 0) {
- n.next = this._head;
- this._head.prev = n;
+ n.next = (owned) this._head;
+ n.next.prev = n;
this._head = (owned)n;
} else {
- Node prev = this._head;
+ weak Node prev = this._head;
for (int i = 0; i < index - 1; i++) {
prev = prev.next;
}
n.prev = prev;
- n.next = prev.next;
+ n.next = (owned) prev.next;
n.next.prev = n;
- prev.next = n;
+ prev.next = (owned) n;
}
// Adding items to the list during iterations is allowed.
@@ -225,7 +227,7 @@ public class Gee.LinkedList<G> : AbstractList<G>, Queue<G>, Deque<G> {
return_val_if_fail (stop <= this._size, null);
List<G> slice = new LinkedList<G> (this.equal_func);
- Node<G> n = this._get_node_at (start);
+ weak Node<G> n = this._get_node_at (start);
for (int i = start; i < stop; i++) {
slice.add (n.data);
n = n.next;
@@ -386,11 +388,12 @@ public class Gee.LinkedList<G> : AbstractList<G>, Queue<G>, Deque<G> {
return amount;
}
+ [Compact]
private class Node<G> { // Maybe a compact class should be used?
public G data;
- public Node<G>? prev = null;
+ public weak Node<G>? prev = null;
public Node<G>? next = null;
- public Node (G data) {
+ public Node (owned G data) {
this.data = data;
}
}
@@ -525,13 +528,18 @@ public class Gee.LinkedList<G> : AbstractList<G>, Queue<G>, Deque<G> {
Node<G> n = new Node<G> (item);
if (this.position.prev != null) {
- this.position.prev.next = n;
- n.prev = this.position.prev;
+ Node<G> position = (owned) this.position.prev.next;
+ n.prev = position.prev;
+ position.prev = n;
+ n.next = (owned) position;
+ weak Node<G> _n = n;
+ _n.prev.next = (owned) n;
} else {
- this._list._head = n;
+ Node<G> position = (owned) this._list._head;
+ position.prev = n;
+ n.next = (owned) position;
+ this._list._head = (owned) n;
}
- this.position.prev = n;
- n.next = this.position;
this._list._size++;
this._index++;
_stamp = _list._stamp;
@@ -544,13 +552,13 @@ public class Gee.LinkedList<G> : AbstractList<G>, Queue<G>, Deque<G> {
Node<G> n = new Node<G> (item);
if (this.position.next != null) {
this.position.next.prev = n;
- n.next = this.position.next;
+ n.next = (owned) this.position.next;
} else {
this._list._tail = n;
}
- this.position.next = n;
- n.prev = this.position;
- this.position = n;
+ this.position.next = (owned) n;
+ this.position.next.prev = this.position;
+ this.position = this.position.next;
this._list._size++;
this._index++;
_stamp = _list._stamp;
@@ -584,21 +592,24 @@ public class Gee.LinkedList<G> : AbstractList<G>, Queue<G>, Deque<G> {
return n;
}
- private void _remove_node (owned Node<G> n) {
- if (n == this._head) {
- this._head = n.next;
+ private void _remove_node (Node<G> _n) {
+ Node<G> n;
+ weak Node<G> next;
+ if (_n == this._head) {
+ n = (owned) this._head;
+ next = this._head = (owned) n.next;
+ } else {
+ n = (owned) _n.prev.next;
+ next = n.prev.next = (owned) n.next;
}
if (n == this._tail) {
this._tail = n.prev;
- }
- if (n.prev != null) {
- n.prev.next = n.next;
- }
- if (n.next != null) {
- n.next.prev = n.prev;
+ } else {
+ next.prev = n.prev;
}
n.prev = null;
n.next = null;
+ n.data = null;
++this._stamp;
this._size--;
}
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]