Commit 4d02b124 authored by Keith Randall's avatar Keith Randall

runtime: don't expose stack buffer in stringto{byte,rune}slice

When using a stack-allocated buffer for the result, don't
expose the uninitialized portion of it by restricting its
capacity to its length.

The other option is to zero the portion between len and cap.
That seems like more work, but might be worth it if the caller
then appends some stuff to the result.  But this close to 1.6,
I'm inclined to do the simplest fix possible.

Fixes #14232

Change-Id: I21c50d3cda02fd2df4d60ba5e2cfe2efe272f333
Reviewed-on: https://go-review.googlesource.com/19231Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
Reviewed-by: default avatarIan Lance Taylor <iant@golang.org>
parent 39304eb6
...@@ -139,7 +139,7 @@ func slicebytetostringtmp(b []byte) string { ...@@ -139,7 +139,7 @@ func slicebytetostringtmp(b []byte) string {
func stringtoslicebyte(buf *tmpBuf, s string) []byte { func stringtoslicebyte(buf *tmpBuf, s string) []byte {
var b []byte var b []byte
if buf != nil && len(s) <= len(buf) { if buf != nil && len(s) <= len(buf) {
b = buf[:len(s)] b = buf[:len(s):len(s)]
} else { } else {
b = rawbyteslice(len(s)) b = rawbyteslice(len(s))
} }
...@@ -171,7 +171,7 @@ func stringtoslicerune(buf *[tmpStringBufSize]rune, s string) []rune { ...@@ -171,7 +171,7 @@ func stringtoslicerune(buf *[tmpStringBufSize]rune, s string) []rune {
} }
var a []rune var a []rune
if buf != nil && n <= len(buf) { if buf != nil && n <= len(buf) {
a = buf[:n] a = buf[:n:n]
} else { } else {
a = rawruneslice(n) a = rawruneslice(n)
} }
......
...@@ -222,3 +222,18 @@ func TestRangeStringCast(t *testing.T) { ...@@ -222,3 +222,18 @@ func TestRangeStringCast(t *testing.T) {
t.Fatalf("want 0 allocs, got %v", n) t.Fatalf("want 0 allocs, got %v", n)
} }
} }
func TestString2Slice(t *testing.T) {
// Make sure we don't return slices that expose
// an unzeroed section of stack-allocated temp buf
// between len and cap. See issue 14232.
s := "foož"
b := ([]byte)(s)
if cap(b) != 5 {
t.Errorf("want cap of 5, got %d", cap(b))
}
r := ([]rune)(s)
if cap(r) != 4 {
t.Errorf("want cap of 4, got %d", cap(r))
}
}
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment