aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorShenghou Ma <minux@golang.org>2015-03-11 12:07:50 -0400
committerChris Broadfoot <cbro@golang.org>2015-09-22 06:36:57 +0000
commite938de22be4ee63d65152024f75d62482d3d69b4 (patch)
tree19b15634d7e7ae9e87f6684fb08245894c9cda22
parenta527bdbda39c48fa772d8d54eb6b849f240442c1 (diff)
downloadgo-e938de22be4ee63d65152024f75d62482d3d69b4.tar.gz
go-e938de22be4ee63d65152024f75d62482d3d69b4.zip
[release-branch.go1.4] runtime: don't return a slice with nil ptr but non-zero len from growslice
Fixes #10135. Change-Id: Ic4c5ab15bcb7b9c3fcc685a788d3b59c60c26e1e Signed-off-by: Shenghou Ma <minux@golang.org> Reviewed-on: https://go-review.googlesource.com/7400 Reviewed-by: Ian Lance Taylor <iant@golang.org> Reviewed-on: https://go-review.googlesource.com/14248 Reviewed-by: Andrew Gerrand <adg@golang.org>
-rw-r--r--src/runtime/slice.go4
-rw-r--r--test/fixedbugs/issue10135.go25
2 files changed, 28 insertions, 1 deletions
diff --git a/src/runtime/slice.go b/src/runtime/slice.go
index 171087d7f6..f457f41429 100644
--- a/src/runtime/slice.go
+++ b/src/runtime/slice.go
@@ -53,7 +53,9 @@ func growslice(t *slicetype, old sliceStruct, n int64) sliceStruct {
et := t.elem
if et.size == 0 {
- return sliceStruct{old.array, old.len, cap}
+ // append should not create a slice with nil pointer but non-zero len.
+ // We assume that append doesn't need to preserve old.array in this case.
+ return sliceStruct{unsafe.Pointer(&zerobase), old.len, cap}
}
newcap := old.cap
diff --git a/test/fixedbugs/issue10135.go b/test/fixedbugs/issue10135.go
new file mode 100644
index 0000000000..9985e5a0ee
--- /dev/null
+++ b/test/fixedbugs/issue10135.go
@@ -0,0 +1,25 @@
+// run
+
+// Copyright 2015 The Go Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style
+// license that can be found in the LICENSE file.
+
+// Issue 10135: append a slice with zero-sized element used
+// to always return a slice with the same data pointer as the
+// old slice, even if it's nil, so this program used to panic
+// with nil pointer dereference because after append, s is a
+// slice with nil data pointer but non-zero len and cap.
+
+package main
+
+type empty struct{}
+
+func main() {
+ var s []empty
+
+ s = append(s, empty{})
+
+ for _, v := range s {
+ _ = v
+ }
+}