Skip to content

Commit 00f508d

Browse files
authored
Fix leak due to cid queue never getting cleaned up (#756)
* Fix leak due to cid queue never getting cleaned up This removes entries from the queue portion of CidQueue that were removed from the map portion. * Remove unused cidQueue function. Do not export cidQueue functions.
1 parent e11e4ed commit 00f508d

File tree

3 files changed

+33
-38
lines changed

3 files changed

+33
-38
lines changed

bitswap/client/internal/blockpresencemanager/blockpresencemanager.go

+4
Original file line numberDiff line numberDiff line change
@@ -111,6 +111,10 @@ func (bpm *BlockPresenceManager) allDontHave(peers []peer.ID, c cid.Cid) bool {
111111

112112
// RemoveKeys cleans up the given keys from the block presence map
113113
func (bpm *BlockPresenceManager) RemoveKeys(ks []cid.Cid) {
114+
if len(ks) == 0 {
115+
return
116+
}
117+
114118
bpm.Lock()
115119
defer bpm.Unlock()
116120

bitswap/client/internal/session/cidqueue.go

+16-28
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ func newCidQueue() *cidQueue {
1414
return &cidQueue{eset: cid.NewSet()}
1515
}
1616

17-
func (cq *cidQueue) Pop() cid.Cid {
17+
func (cq *cidQueue) pop() cid.Cid {
1818
for {
1919
if cq.elems.Len() == 0 {
2020
return cid.Cid{}
@@ -29,43 +29,31 @@ func (cq *cidQueue) Pop() cid.Cid {
2929
}
3030
}
3131

32-
func (cq *cidQueue) Cids() []cid.Cid {
33-
// Lazily delete from the list any cids that were removed from the set
34-
if cq.elems.Len() > cq.eset.Len() {
35-
for i := 0; i < cq.elems.Len(); i++ {
36-
c := cq.elems.PopFront()
37-
if cq.eset.Has(c) {
38-
cq.elems.PushBack(c)
39-
}
40-
}
41-
}
42-
43-
if cq.elems.Len() == 0 {
44-
return nil
45-
}
46-
47-
// Make a copy of the cids
48-
cids := make([]cid.Cid, cq.elems.Len())
49-
for i := 0; i < cq.elems.Len(); i++ {
50-
cids[i] = cq.elems.At(i)
51-
}
52-
return cids
53-
}
54-
55-
func (cq *cidQueue) Push(c cid.Cid) {
32+
func (cq *cidQueue) push(c cid.Cid) {
5633
if cq.eset.Visit(c) {
5734
cq.elems.PushBack(c)
5835
}
5936
}
6037

61-
func (cq *cidQueue) Remove(c cid.Cid) {
38+
func (cq *cidQueue) remove(c cid.Cid) {
6239
cq.eset.Remove(c)
6340
}
6441

65-
func (cq *cidQueue) Has(c cid.Cid) bool {
42+
func (cq *cidQueue) has(c cid.Cid) bool {
6643
return cq.eset.Has(c)
6744
}
6845

69-
func (cq *cidQueue) Len() int {
46+
func (cq *cidQueue) len() int {
7047
return cq.eset.Len()
7148
}
49+
50+
func (cq *cidQueue) gc() {
51+
if cq.elems.Len() > cq.eset.Len() {
52+
for i := 0; i < cq.elems.Len(); i++ {
53+
c := cq.elems.PopFront()
54+
if cq.eset.Has(c) {
55+
cq.elems.PushBack(c)
56+
}
57+
}
58+
}
59+
}

bitswap/client/internal/session/sessionwants.go

+13-10
Original file line numberDiff line numberDiff line change
@@ -35,13 +35,13 @@ func newSessionWants(broadcastLimit int) sessionWants {
3535
}
3636

3737
func (sw *sessionWants) String() string {
38-
return fmt.Sprintf("%d pending / %d live", sw.toFetch.Len(), len(sw.liveWants))
38+
return fmt.Sprintf("%d pending / %d live", sw.toFetch.len(), len(sw.liveWants))
3939
}
4040

4141
// BlocksRequested is called when the client makes a request for blocks
4242
func (sw *sessionWants) BlocksRequested(newWants []cid.Cid) {
4343
for _, k := range newWants {
44-
sw.toFetch.Push(k)
44+
sw.toFetch.push(k)
4545
}
4646
}
4747

@@ -56,14 +56,14 @@ func (sw *sessionWants) GetNextWants() []cid.Cid {
5656
// limit)
5757
currentLiveCount := len(sw.liveWants)
5858
toAdd := sw.broadcastLimit - currentLiveCount
59-
liveSize := min(toAdd, sw.toFetch.Len())
59+
liveSize := min(toAdd, sw.toFetch.len())
6060
if liveSize <= 0 {
6161
return nil
6262
}
6363

6464
live := make([]cid.Cid, 0, liveSize)
65-
for ; toAdd > 0 && sw.toFetch.Len() > 0; toAdd-- {
66-
c := sw.toFetch.Pop()
65+
for ; toAdd > 0 && sw.toFetch.len() > 0; toAdd-- {
66+
c := sw.toFetch.pop()
6767
live = append(live, c)
6868
sw.liveWantsOrder = append(sw.liveWantsOrder, c)
6969
sw.liveWants[c] = now
@@ -76,12 +76,13 @@ func (sw *sessionWants) GetNextWants() []cid.Cid {
7676
func (sw *sessionWants) WantsSent(ks []cid.Cid) {
7777
now := time.Now()
7878
for _, c := range ks {
79-
if _, ok := sw.liveWants[c]; !ok && sw.toFetch.Has(c) {
80-
sw.toFetch.Remove(c)
79+
if _, ok := sw.liveWants[c]; !ok && sw.toFetch.has(c) {
80+
sw.toFetch.remove(c)
8181
sw.liveWantsOrder = append(sw.liveWantsOrder, c)
8282
sw.liveWants[c] = now
8383
}
8484
}
85+
sw.toFetch.gc()
8586
}
8687

8788
// BlocksReceived removes received block CIDs from the live wants list and
@@ -108,9 +109,10 @@ func (sw *sessionWants) BlocksReceived(ks []cid.Cid) ([]cid.Cid, time.Duration)
108109

109110
// Remove the CID from the live wants / toFetch queue
110111
delete(sw.liveWants, c)
111-
sw.toFetch.Remove(c)
112+
sw.toFetch.remove(c)
112113
}
113114
}
115+
sw.toFetch.gc()
114116

115117
// If the live wants ordering array is a long way out of sync with the
116118
// live wants map, clean up the ordering array
@@ -152,9 +154,10 @@ func (sw *sessionWants) PrepareBroadcast() []cid.Cid {
152154
// CancelPending removes the given CIDs from the fetch queue.
153155
func (sw *sessionWants) CancelPending(keys []cid.Cid) {
154156
for _, k := range keys {
155-
sw.toFetch.Remove(k)
157+
sw.toFetch.remove(k)
156158
delete(sw.liveWants, k)
157159
}
160+
sw.toFetch.gc()
158161
}
159162

160163
// LiveWants returns a list of live wants
@@ -193,7 +196,7 @@ func (sw *sessionWants) HasLiveWants() bool {
193196
func (sw *sessionWants) isWanted(c cid.Cid) bool {
194197
_, ok := sw.liveWants[c]
195198
if !ok {
196-
ok = sw.toFetch.Has(c)
199+
ok = sw.toFetch.has(c)
197200
}
198201
return ok
199202
}

0 commit comments

Comments
 (0)