Skip to content

Commit 1b97071

Browse files
Deleplacestapelberg
authored andcommitted
slices: zero the slice elements discarded by Delete, DeleteFunc, Compact, CompactFunc, Replace.
Backport from stdlib: to avoid memory leaks in slices that contain pointers, clear the elements between the new length and the original length. Fixes golang/go#63393 Change-Id: I38bf64b27619d8067f2e95ce3c7952ec95ca55b8 Reviewed-on: https://go-review.googlesource.com/c/exp/+/543335 LUCI-TryBot-Result: Go LUCI <[email protected]> Reviewed-by: Eli Bendersky <[email protected]>
1 parent db7319d commit 1b97071

File tree

2 files changed

+160
-14
lines changed

2 files changed

+160
-14
lines changed

slices/slices.go

+30-14
Original file line numberDiff line numberDiff line change
@@ -209,25 +209,37 @@ func Insert[S ~[]E, E any](s S, i int, v ...E) S {
209209
return s
210210
}
211211

212+
// clearSlice sets all elements up to the length of s to the zero value of E.
213+
// We may use the builtin clear func instead, and remove clearSlice, when upgrading
214+
// to Go 1.21+.
215+
func clearSlice[S ~[]E, E any](s S) {
216+
var zero E
217+
for i := range s {
218+
s[i] = zero
219+
}
220+
}
221+
212222
// Delete removes the elements s[i:j] from s, returning the modified slice.
213-
// Delete panics if s[i:j] is not a valid slice of s.
214-
// Delete is O(len(s)-j), so if many items must be deleted, it is better to
223+
// Delete panics if j > len(s) or s[i:j] is not a valid slice of s.
224+
// Delete is O(len(s)-i), so if many items must be deleted, it is better to
215225
// make a single call deleting them all together than to delete one at a time.
216-
// Delete might not modify the elements s[len(s)-(j-i):len(s)]. If those
217-
// elements contain pointers you might consider zeroing those elements so that
218-
// objects they reference can be garbage collected.
226+
// Delete zeroes the elements s[len(s)-(j-i):len(s)].
219227
func Delete[S ~[]E, E any](s S, i, j int) S {
220-
_ = s[i:j] // bounds check
228+
_ = s[i:j:len(s)] // bounds check
221229

222-
return append(s[:i], s[j:]...)
230+
if i == j {
231+
return s
232+
}
233+
234+
oldlen := len(s)
235+
s = append(s[:i], s[j:]...)
236+
clearSlice(s[len(s):oldlen]) // zero/nil out the obsolete elements, for GC
237+
return s
223238
}
224239

225240
// DeleteFunc removes any elements from s for which del returns true,
226241
// returning the modified slice.
227-
// When DeleteFunc removes m elements, it might not modify the elements
228-
// s[len(s)-m:len(s)]. If those elements contain pointers you might consider
229-
// zeroing those elements so that objects they reference can be garbage
230-
// collected.
242+
// DeleteFunc zeroes the elements between the new length and the original length.
231243
func DeleteFunc[S ~[]E, E any](s S, del func(E) bool) S {
232244
i := IndexFunc(s, del)
233245
if i == -1 {
@@ -240,11 +252,13 @@ func DeleteFunc[S ~[]E, E any](s S, del func(E) bool) S {
240252
i++
241253
}
242254
}
255+
clearSlice(s[i:]) // zero/nil out the obsolete elements, for GC
243256
return s[:i]
244257
}
245258

246259
// Replace replaces the elements s[i:j] by the given v, and returns the
247260
// modified slice. Replace panics if s[i:j] is not a valid slice of s.
261+
// When len(v) < (j-i), Replace zeroes the elements between the new length and the original length.
248262
func Replace[S ~[]E, E any](s S, i, j int, v ...E) S {
249263
_ = s[i:j] // verify that i:j is a valid subslice
250264

@@ -272,6 +286,7 @@ func Replace[S ~[]E, E any](s S, i, j int, v ...E) S {
272286
if i+len(v) != j {
273287
copy(r[i+len(v):], s[j:])
274288
}
289+
clearSlice(s[tot:]) // zero/nil out the obsolete elements, for GC
275290
return r
276291
}
277292

@@ -345,9 +360,7 @@ func Clone[S ~[]E, E any](s S) S {
345360
// This is like the uniq command found on Unix.
346361
// Compact modifies the contents of the slice s and returns the modified slice,
347362
// which may have a smaller length.
348-
// When Compact discards m elements in total, it might not modify the elements
349-
// s[len(s)-m:len(s)]. If those elements contain pointers you might consider
350-
// zeroing those elements so that objects they reference can be garbage collected.
363+
// Compact zeroes the elements between the new length and the original length.
351364
func Compact[S ~[]E, E comparable](s S) S {
352365
if len(s) < 2 {
353366
return s
@@ -361,11 +374,13 @@ func Compact[S ~[]E, E comparable](s S) S {
361374
i++
362375
}
363376
}
377+
clearSlice(s[i:]) // zero/nil out the obsolete elements, for GC
364378
return s[:i]
365379
}
366380

367381
// CompactFunc is like [Compact] but uses an equality function to compare elements.
368382
// For runs of elements that compare equal, CompactFunc keeps the first one.
383+
// CompactFunc zeroes the elements between the new length and the original length.
369384
func CompactFunc[S ~[]E, E any](s S, eq func(E, E) bool) S {
370385
if len(s) < 2 {
371386
return s
@@ -379,6 +394,7 @@ func CompactFunc[S ~[]E, E any](s S, eq func(E, E) bool) S {
379394
i++
380395
}
381396
}
397+
clearSlice(s[i:]) // zero/nil out the obsolete elements, for GC
382398
return s[:i]
383399
}
384400

slices/slices_test.go

+130
Original file line numberDiff line numberDiff line change
@@ -654,6 +654,39 @@ func TestDeletePanics(t *testing.T) {
654654
}
655655
}
656656

657+
func TestDeleteClearTail(t *testing.T) {
658+
mem := []*int{new(int), new(int), new(int), new(int), new(int), new(int)}
659+
s := mem[0:5] // there is 1 element beyond len(s), within cap(s)
660+
661+
s = Delete(s, 2, 4)
662+
663+
if mem[3] != nil || mem[4] != nil {
664+
// Check that potential memory leak is avoided
665+
t.Errorf("Delete: want nil discarded elements, got %v, %v", mem[3], mem[4])
666+
}
667+
if mem[5] == nil {
668+
t.Errorf("Delete: want unchanged elements beyond original len, got nil")
669+
}
670+
}
671+
672+
func TestDeleteFuncClearTail(t *testing.T) {
673+
mem := []*int{new(int), new(int), new(int), new(int), new(int), new(int)}
674+
*mem[2], *mem[3] = 42, 42
675+
s := mem[0:5] // there is 1 element beyond len(s), within cap(s)
676+
677+
s = DeleteFunc(s, func(i *int) bool {
678+
return i != nil && *i == 42
679+
})
680+
681+
if mem[3] != nil || mem[4] != nil {
682+
// Check that potential memory leak is avoided
683+
t.Errorf("DeleteFunc: want nil discarded elements, got %v, %v", mem[3], mem[4])
684+
}
685+
if mem[5] == nil {
686+
t.Errorf("DeleteFunc: want unchanged elements beyond original len, got nil")
687+
}
688+
}
689+
657690
func TestClone(t *testing.T) {
658691
s1 := []int{1, 2, 3}
659692
s2 := Clone(s1)
@@ -757,6 +790,53 @@ func TestCompactFunc(t *testing.T) {
757790
}
758791
}
759792

793+
func TestCompactClearTail(t *testing.T) {
794+
one, two, three, four := 1, 2, 3, 4
795+
mem := []*int{&one, &one, &two, &two, &three, &four}
796+
s := mem[0:5] // there is 1 element beyond len(s), within cap(s)
797+
copy := Clone(s)
798+
799+
s = Compact(s)
800+
801+
if want := []*int{&one, &two, &three}; !Equal(s, want) {
802+
t.Errorf("Compact(%v) = %v, want %v", copy, s, want)
803+
}
804+
805+
if mem[3] != nil || mem[4] != nil {
806+
// Check that potential memory leak is avoided
807+
t.Errorf("Compact: want nil discarded elements, got %v, %v", mem[3], mem[4])
808+
}
809+
if mem[5] != &four {
810+
t.Errorf("Compact: want unchanged element beyond original len, got %v", mem[5])
811+
}
812+
}
813+
814+
func TestCompactFuncClearTail(t *testing.T) {
815+
a, b, c, d, e, f := 1, 1, 2, 2, 3, 4
816+
mem := []*int{&a, &b, &c, &d, &e, &f}
817+
s := mem[0:5] // there is 1 element beyond len(s), within cap(s)
818+
copy := Clone(s)
819+
820+
s = CompactFunc(s, func(x, y *int) bool {
821+
if x == nil || y == nil {
822+
return x == y
823+
}
824+
return *x == *y
825+
})
826+
827+
if want := []*int{&a, &c, &e}; !Equal(s, want) {
828+
t.Errorf("CompactFunc(%v) = %v, want %v", copy, s, want)
829+
}
830+
831+
if mem[3] != nil || mem[4] != nil {
832+
// Check that potential memory leak is avoided
833+
t.Errorf("CompactFunc: want nil discarded elements, got %v, %v", mem[3], mem[4])
834+
}
835+
if mem[5] != &f {
836+
t.Errorf("CompactFunc: want unchanged elements beyond original len, got %v", mem[5])
837+
}
838+
}
839+
760840
func BenchmarkCompactFunc_Large(b *testing.B) {
761841
type Large [4 * 1024]byte
762842

@@ -922,6 +1002,56 @@ func TestReplacePanics(t *testing.T) {
9221002
}
9231003
}
9241004

1005+
func TestReplaceGrow(t *testing.T) {
1006+
// When Replace needs to allocate a new slice, we want the original slice
1007+
// to not be changed.
1008+
a, b, c, d, e, f := 1, 2, 3, 4, 5, 6
1009+
mem := []*int{&a, &b, &c, &d, &e, &f}
1010+
memcopy := Clone(mem)
1011+
s := mem[0:5] // there is 1 element beyond len(s), within cap(s)
1012+
copy := Clone(s)
1013+
original := s
1014+
1015+
// The new elements don't fit within cap(s), so Replace will allocate.
1016+
z := 99
1017+
s = Replace(s, 1, 3, &z, &z, &z, &z)
1018+
1019+
if want := []*int{&a, &z, &z, &z, &z, &d, &e}; !Equal(s, want) {
1020+
t.Errorf("Replace(%v, 1, 3, %v, %v, %v, %v) = %v, want %v", copy, &z, &z, &z, &z, s, want)
1021+
}
1022+
1023+
if !Equal(original, copy) {
1024+
t.Errorf("original slice has changed, got %v, want %v", original, copy)
1025+
}
1026+
1027+
if !Equal(mem, memcopy) {
1028+
// Changing the original tail s[len(s):cap(s)] is unwanted
1029+
t.Errorf("original backing memory has changed, got %v, want %v", mem, memcopy)
1030+
}
1031+
}
1032+
1033+
func TestReplaceClearTail(t *testing.T) {
1034+
a, b, c, d, e, f := 1, 2, 3, 4, 5, 6
1035+
mem := []*int{&a, &b, &c, &d, &e, &f}
1036+
s := mem[0:5] // there is 1 element beyond len(s), within cap(s)
1037+
copy := Clone(s)
1038+
1039+
y, z := 8, 9
1040+
s = Replace(s, 1, 4, &y, &z)
1041+
1042+
if want := []*int{&a, &y, &z, &e}; !Equal(s, want) {
1043+
t.Errorf("Replace(%v) = %v, want %v", copy, s, want)
1044+
}
1045+
1046+
if mem[4] != nil {
1047+
// Check that potential memory leak is avoided
1048+
t.Errorf("Replace: want nil discarded element, got %v", mem[4])
1049+
}
1050+
if mem[5] != &f {
1051+
t.Errorf("Replace: want unchanged elements beyond original len, got %v", mem[5])
1052+
}
1053+
}
1054+
9251055
func TestReplaceOverlap(t *testing.T) {
9261056
const N = 10
9271057
a := make([]int, N)

0 commit comments

Comments
 (0)