Commit 478b594d authored by Alexandru Moșoi's avatar Alexandru Moșoi Committed by Alexandru Moșoi

encoding/binary: fix bound check

The inserted early bound checks cause the slice
to expand beyond the original length of the slice.

Change-Id: Ib38891605f4a9a12d3b9e2071a5f77640b083d2d
Reviewed-on: https://go-review.googlesource.com/20981Reviewed-by: default avatarKeith Randall <khr@golang.org>
Reviewed-by: default avatarMinux Ma <minux@golang.org>
parent 6a33f776
...@@ -49,23 +49,23 @@ var BigEndian bigEndian ...@@ -49,23 +49,23 @@ var BigEndian bigEndian
type littleEndian struct{} type littleEndian struct{}
func (littleEndian) Uint16(b []byte) uint16 { func (littleEndian) Uint16(b []byte) uint16 {
b = b[:2] // bounds check hint to compiler; see golang.org/issue/14808 b = b[:2:len(b)] // bounds check hint to compiler; see golang.org/issue/14808
return uint16(b[0]) | uint16(b[1])<<8 return uint16(b[0]) | uint16(b[1])<<8
} }
func (littleEndian) PutUint16(b []byte, v uint16) { func (littleEndian) PutUint16(b []byte, v uint16) {
b = b[:2] // early bounds check to guarantee safety of writes below b = b[:2:len(b)] // early bounds check to guarantee safety of writes below
b[0] = byte(v) b[0] = byte(v)
b[1] = byte(v >> 8) b[1] = byte(v >> 8)
} }
func (littleEndian) Uint32(b []byte) uint32 { func (littleEndian) Uint32(b []byte) uint32 {
b = b[:4] // bounds check hint to compiler; see golang.org/issue/14808 b = b[:4:len(b)] // bounds check hint to compiler; see golang.org/issue/14808
return uint32(b[0]) | uint32(b[1])<<8 | uint32(b[2])<<16 | uint32(b[3])<<24 return uint32(b[0]) | uint32(b[1])<<8 | uint32(b[2])<<16 | uint32(b[3])<<24
} }
func (littleEndian) PutUint32(b []byte, v uint32) { func (littleEndian) PutUint32(b []byte, v uint32) {
b = b[:4] // early bounds check to guarantee safety of writes below b = b[:4:len(b)] // early bounds check to guarantee safety of writes below
b[0] = byte(v) b[0] = byte(v)
b[1] = byte(v >> 8) b[1] = byte(v >> 8)
b[2] = byte(v >> 16) b[2] = byte(v >> 16)
...@@ -73,13 +73,13 @@ func (littleEndian) PutUint32(b []byte, v uint32) { ...@@ -73,13 +73,13 @@ func (littleEndian) PutUint32(b []byte, v uint32) {
} }
func (littleEndian) Uint64(b []byte) uint64 { func (littleEndian) Uint64(b []byte) uint64 {
b = b[:8] // bounds check hint to compiler; see golang.org/issue/14808 b = b[:8:len(b)] // bounds check hint to compiler; see golang.org/issue/14808
return uint64(b[0]) | uint64(b[1])<<8 | uint64(b[2])<<16 | uint64(b[3])<<24 | return uint64(b[0]) | uint64(b[1])<<8 | uint64(b[2])<<16 | uint64(b[3])<<24 |
uint64(b[4])<<32 | uint64(b[5])<<40 | uint64(b[6])<<48 | uint64(b[7])<<56 uint64(b[4])<<32 | uint64(b[5])<<40 | uint64(b[6])<<48 | uint64(b[7])<<56
} }
func (littleEndian) PutUint64(b []byte, v uint64) { func (littleEndian) PutUint64(b []byte, v uint64) {
b = b[:8] // early bounds check to guarantee safety of writes below b = b[:8:len(b)] // early bounds check to guarantee safety of writes below
b[0] = byte(v) b[0] = byte(v)
b[1] = byte(v >> 8) b[1] = byte(v >> 8)
b[2] = byte(v >> 16) b[2] = byte(v >> 16)
...@@ -97,23 +97,23 @@ func (littleEndian) GoString() string { return "binary.LittleEndian" } ...@@ -97,23 +97,23 @@ func (littleEndian) GoString() string { return "binary.LittleEndian" }
type bigEndian struct{} type bigEndian struct{}
func (bigEndian) Uint16(b []byte) uint16 { func (bigEndian) Uint16(b []byte) uint16 {
b = b[:2] // bounds check hint to compiler; see golang.org/issue/14808 b = b[:2:len(b)] // bounds check hint to compiler; see golang.org/issue/14808
return uint16(b[1]) | uint16(b[0])<<8 return uint16(b[1]) | uint16(b[0])<<8
} }
func (bigEndian) PutUint16(b []byte, v uint16) { func (bigEndian) PutUint16(b []byte, v uint16) {
b = b[:2] // early bounds check to guarantee safety of writes below b = b[:2:len(b)] // early bounds check to guarantee safety of writes below
b[0] = byte(v >> 8) b[0] = byte(v >> 8)
b[1] = byte(v) b[1] = byte(v)
} }
func (bigEndian) Uint32(b []byte) uint32 { func (bigEndian) Uint32(b []byte) uint32 {
b = b[:4] // bounds check hint to compiler; see golang.org/issue/14808 b = b[:4:len(b)] // bounds check hint to compiler; see golang.org/issue/14808
return uint32(b[3]) | uint32(b[2])<<8 | uint32(b[1])<<16 | uint32(b[0])<<24 return uint32(b[3]) | uint32(b[2])<<8 | uint32(b[1])<<16 | uint32(b[0])<<24
} }
func (bigEndian) PutUint32(b []byte, v uint32) { func (bigEndian) PutUint32(b []byte, v uint32) {
b = b[:4] // early bounds check to guarantee safety of writes below b = b[:4:len(b)] // early bounds check to guarantee safety of writes below
b[0] = byte(v >> 24) b[0] = byte(v >> 24)
b[1] = byte(v >> 16) b[1] = byte(v >> 16)
b[2] = byte(v >> 8) b[2] = byte(v >> 8)
...@@ -121,13 +121,13 @@ func (bigEndian) PutUint32(b []byte, v uint32) { ...@@ -121,13 +121,13 @@ func (bigEndian) PutUint32(b []byte, v uint32) {
} }
func (bigEndian) Uint64(b []byte) uint64 { func (bigEndian) Uint64(b []byte) uint64 {
b = b[:8] // bounds check hint to compiler; see golang.org/issue/14808 b = b[:8:len(b)] // bounds check hint to compiler; see golang.org/issue/14808
return uint64(b[7]) | uint64(b[6])<<8 | uint64(b[5])<<16 | uint64(b[4])<<24 | return uint64(b[7]) | uint64(b[6])<<8 | uint64(b[5])<<16 | uint64(b[4])<<24 |
uint64(b[3])<<32 | uint64(b[2])<<40 | uint64(b[1])<<48 | uint64(b[0])<<56 uint64(b[3])<<32 | uint64(b[2])<<40 | uint64(b[1])<<48 | uint64(b[0])<<56
} }
func (bigEndian) PutUint64(b []byte, v uint64) { func (bigEndian) PutUint64(b []byte, v uint64) {
b = b[:8] // early bounds check to guarantee safety of writes below b = b[:8:len(b)] // early bounds check to guarantee safety of writes below
b[0] = byte(v >> 56) b[0] = byte(v >> 56)
b[1] = byte(v >> 48) b[1] = byte(v >> 48)
b[2] = byte(v >> 40) b[2] = byte(v >> 40)
......
...@@ -339,6 +339,33 @@ func TestReadTruncated(t *testing.T) { ...@@ -339,6 +339,33 @@ func TestReadTruncated(t *testing.T) {
} }
} }
func testUint64SmallSliceLengthPanics() (panicked bool) {
defer func() {
panicked = recover() != nil
}()
b := [8]byte{1, 2, 3, 4, 5, 6, 7, 8}
LittleEndian.Uint64(b[:4])
return false
}
func testPutUint64SmallSliceLengthPanics() (panicked bool) {
defer func() {
panicked = recover() != nil
}()
b := [8]byte{}
LittleEndian.PutUint64(b[:4], 0x0102030405060708)
return false
}
func TestEarlyBoundsChecks(t *testing.T) {
if testUint64SmallSliceLengthPanics() != true {
t.Errorf("binary.LittleEndian.Uint64 expected to panic for small slices, but didn't")
}
if testPutUint64SmallSliceLengthPanics() != true {
t.Errorf("binary.LittleEndian.PutUint64 expected to panic for small slices, but didn't")
}
}
type byteSliceReader struct { type byteSliceReader struct {
remain []byte remain []byte
} }
......
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