Commit a718f939 authored by Cherry Zhang's avatar Cherry Zhang Committed by Ian Lance Taylor

[release-branch.go1.12] cmd/compile: guard against loads with negative offset...

[release-branch.go1.12] cmd/compile: guard against loads with negative offset from readonly constants

CL 154057 adds guards agaist out-of-bound reads from readonly
constants. It turns out that in dead code, the offset can also
be negative. Guard against negative offset as well.

Fixes #30257.

Change-Id: I47c2a2e434dd466c08ae6f50f213999a358c796e
Reviewed-on: https://go-review.googlesource.com/c/162819Reviewed-by: default avatarKeith Randall <khr@golang.org>
(cherry picked from commit dca707b2)
Reviewed-on: https://go-review.googlesource.com/c/162827
Run-TryBot: Cherry Zhang <cherryyz@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarIan Lance Taylor <iant@golang.org>
parent da1f5d37
...@@ -1141,7 +1141,7 @@ func symIsRO(sym interface{}) bool { ...@@ -1141,7 +1141,7 @@ func symIsRO(sym interface{}) bool {
// read8 reads one byte from the read-only global sym at offset off. // read8 reads one byte from the read-only global sym at offset off.
func read8(sym interface{}, off int64) uint8 { func read8(sym interface{}, off int64) uint8 {
lsym := sym.(*obj.LSym) lsym := sym.(*obj.LSym)
if off >= int64(len(lsym.P)) { if off >= int64(len(lsym.P)) || off < 0 {
// Invalid index into the global sym. // Invalid index into the global sym.
// This can happen in dead code, so we don't want to panic. // This can happen in dead code, so we don't want to panic.
// Just return any value, it will eventually get ignored. // Just return any value, it will eventually get ignored.
...@@ -1154,7 +1154,7 @@ func read8(sym interface{}, off int64) uint8 { ...@@ -1154,7 +1154,7 @@ func read8(sym interface{}, off int64) uint8 {
// read16 reads two bytes from the read-only global sym at offset off. // read16 reads two bytes from the read-only global sym at offset off.
func read16(sym interface{}, off int64, bigEndian bool) uint16 { func read16(sym interface{}, off int64, bigEndian bool) uint16 {
lsym := sym.(*obj.LSym) lsym := sym.(*obj.LSym)
if off >= int64(len(lsym.P))-1 { if off >= int64(len(lsym.P))-1 || off < 0 {
return 0 return 0
} }
if bigEndian { if bigEndian {
...@@ -1167,7 +1167,7 @@ func read16(sym interface{}, off int64, bigEndian bool) uint16 { ...@@ -1167,7 +1167,7 @@ func read16(sym interface{}, off int64, bigEndian bool) uint16 {
// read32 reads four bytes from the read-only global sym at offset off. // read32 reads four bytes from the read-only global sym at offset off.
func read32(sym interface{}, off int64, bigEndian bool) uint32 { func read32(sym interface{}, off int64, bigEndian bool) uint32 {
lsym := sym.(*obj.LSym) lsym := sym.(*obj.LSym)
if off >= int64(len(lsym.P))-3 { if off >= int64(len(lsym.P))-3 || off < 0 {
return 0 return 0
} }
if bigEndian { if bigEndian {
...@@ -1180,7 +1180,7 @@ func read32(sym interface{}, off int64, bigEndian bool) uint32 { ...@@ -1180,7 +1180,7 @@ func read32(sym interface{}, off int64, bigEndian bool) uint32 {
// read64 reads eight bytes from the read-only global sym at offset off. // read64 reads eight bytes from the read-only global sym at offset off.
func read64(sym interface{}, off int64, bigEndian bool) uint64 { func read64(sym interface{}, off int64, bigEndian bool) uint64 {
lsym := sym.(*obj.LSym) lsym := sym.(*obj.LSym)
if off >= int64(len(lsym.P))-7 { if off >= int64(len(lsym.P))-7 || off < 0 {
return 0 return 0
} }
if bigEndian { if bigEndian {
......
...@@ -16,3 +16,20 @@ func f() { ...@@ -16,3 +16,20 @@ func f() {
} }
_ = s == "bbb" _ = s == "bbb"
} }
// Another case: load from negative offset of a symbol
// in dead code (issue 30257).
func g() {
var i int
var s string
if true {
s = "a"
}
if f := 0.0; -f < 0 {
i = len(s[:4])
}
_ = s[i-1:0] != "bb" && true
}
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