Commit d7b34d5f authored by Martin Möhrmann's avatar Martin Möhrmann Committed by Brad Fitzpatrick

runtime: improve atoi implementation

- Adds overflow checks
- Adds parsing of negative integers
- Adds boolean return value to signal parsing errors
- Adds atoi32 for parsing of integers that fit in an int32
- Adds tests

Handling of errors to provide error messages
at the call sites is left to future CLs.

Updates #17718

Change-Id: I3cacd0ab1230b9efc5404c68edae7304d39bcbc0
Reviewed-on: https://go-review.googlesource.com/32390Reviewed-by: default avatarIan Lance Taylor <iant@golang.org>
Reviewed-by: default avatarBrad Fitzpatrick <bradfitz@golang.org>
parent 40aaf283
...@@ -32,6 +32,9 @@ var FuncPC = funcPC ...@@ -32,6 +32,9 @@ var FuncPC = funcPC
var Fastlog2 = fastlog2 var Fastlog2 = fastlog2
var Atoi = atoi
var Atoi32 = atoi32
type LFNode struct { type LFNode struct {
Next uint64 Next uint64
Pushcnt uintptr Pushcnt uintptr
......
...@@ -196,13 +196,13 @@ func gcinit() { ...@@ -196,13 +196,13 @@ func gcinit() {
func readgogc() int32 { func readgogc() int32 {
p := gogetenv("GOGC") p := gogetenv("GOGC")
if p == "" {
return 100
}
if p == "off" { if p == "off" {
return -1 return -1
} }
return int32(atoi(p)) if n, ok := atoi32(p); ok {
return n
}
return 100
} }
// gcenable is called after the bulk of the runtime initialization, // gcenable is called after the bulk of the runtime initialization,
......
...@@ -477,17 +477,14 @@ func schedinit() { ...@@ -477,17 +477,14 @@ func schedinit() {
gcinit() gcinit()
sched.lastpoll = uint64(nanotime()) sched.lastpoll = uint64(nanotime())
procs := int(ncpu) procs := ncpu
if n, ok := atoi32(gogetenv("GOMAXPROCS")); ok && n > 0 {
procs = n
}
if procs > _MaxGomaxprocs { if procs > _MaxGomaxprocs {
procs = _MaxGomaxprocs procs = _MaxGomaxprocs
} }
if n := atoi(gogetenv("GOMAXPROCS")); n > 0 { if procresize(procs) != nil {
if n > _MaxGomaxprocs {
n = _MaxGomaxprocs
}
procs = n
}
if procresize(int32(procs)) != nil {
throw("unknown runnable goroutine during bootstrap") throw("unknown runnable goroutine during bootstrap")
} }
......
...@@ -375,11 +375,15 @@ func parsedebugvars() { ...@@ -375,11 +375,15 @@ func parsedebugvars() {
// is int, not int32, and should only be updated // is int, not int32, and should only be updated
// if specified in GODEBUG. // if specified in GODEBUG.
if key == "memprofilerate" { if key == "memprofilerate" {
MemProfileRate = atoi(value) if n, ok := atoi(value); ok {
MemProfileRate = n
}
} else { } else {
for _, v := range dbgvars { for _, v := range dbgvars {
if v.name == key { if v.name == key {
*v.value = int32(atoi(value)) if n, ok := atoi32(value); ok {
*v.value = n
}
} }
} }
} }
...@@ -422,7 +426,10 @@ func setTraceback(level string) { ...@@ -422,7 +426,10 @@ func setTraceback(level string) {
case "crash": case "crash":
t = 2<<tracebackShift | tracebackAll | tracebackCrash t = 2<<tracebackShift | tracebackAll | tracebackCrash
default: default:
t = uint32(atoi(level))<<tracebackShift | tracebackAll t = tracebackAll
if n, ok := atoi(level); ok && n == int(uint32(n)) {
t |= uint32(n) << tracebackShift
}
} }
// when C owns the process, simply exit'ing the process on fatal errors // when C owns the process, simply exit'ing the process on fatal errors
// and panics is surprising. Be louder and abort instead. // and panics is surprising. Be louder and abort instead.
......
...@@ -320,13 +320,66 @@ func hasprefix(s, t string) bool { ...@@ -320,13 +320,66 @@ func hasprefix(s, t string) bool {
return len(s) >= len(t) && s[:len(t)] == t return len(s) >= len(t) && s[:len(t)] == t
} }
func atoi(s string) int { const (
n := 0 maxUint = ^uint(0)
for len(s) > 0 && '0' <= s[0] && s[0] <= '9' { maxInt = int(maxUint >> 1)
n = n*10 + int(s[0]) - '0' )
// atoi parses an int from a string s.
// The bool result reports whether s is a number
// representable by a value of type int.
func atoi(s string) (int, bool) {
if s == "" {
return 0, false
}
neg := false
if s[0] == '-' {
neg = true
s = s[1:] s = s[1:]
} }
return n
un := uint(0)
for i := 0; i < len(s); i++ {
c := s[i]
if c < '0' || c > '9' {
return 0, false
}
if un > maxUint/10 {
// overflow
return 0, false
}
un *= 10
un1 := un + uint(c) - '0'
if un1 < un {
// overflow
return 0, false
}
un = un1
}
if !neg && un > uint(maxInt) {
return 0, false
}
if neg && un > uint(maxInt)+1 {
return 0, false
}
n := int(un)
if neg {
n = -n
}
return n, true
}
// atoi32 is like atoi but for integers
// that fit into an int32.
func atoi32(s string) (int32, bool) {
if n, ok := atoi(s); n == int(int32(n)) {
return int32(n), ok
}
return 0, false
} }
//go:nosplit //go:nosplit
......
...@@ -279,3 +279,97 @@ func TestString2Slice(t *testing.T) { ...@@ -279,3 +279,97 @@ func TestString2Slice(t *testing.T) {
t.Errorf("extra runes not zeroed") t.Errorf("extra runes not zeroed")
} }
} }
const intSize = 32 << (^uint(0) >> 63)
type atoi64Test struct {
in string
out int64
ok bool
}
var atoi64tests = []atoi64Test{
{"", 0, false},
{"0", 0, true},
{"-0", 0, true},
{"1", 1, true},
{"-1", -1, true},
{"12345", 12345, true},
{"-12345", -12345, true},
{"012345", 12345, true},
{"-012345", -12345, true},
{"12345x", 0, false},
{"-12345x", 0, false},
{"98765432100", 98765432100, true},
{"-98765432100", -98765432100, true},
{"20496382327982653440", 0, false},
{"-20496382327982653440", 0, false},
{"9223372036854775807", 1<<63 - 1, true},
{"-9223372036854775807", -(1<<63 - 1), true},
{"9223372036854775808", 0, false},
{"-9223372036854775808", -1 << 63, true},
{"9223372036854775809", 0, false},
{"-9223372036854775809", 0, false},
}
func TestAtoi(t *testing.T) {
switch intSize {
case 32:
for i := range atoi32tests {
test := &atoi32tests[i]
out, ok := runtime.Atoi(test.in)
if test.out != int32(out) || test.ok != ok {
t.Errorf("atoi(%q) = (%v, %v) want (%v, %v)",
test.in, out, ok, test.out, test.ok)
}
}
case 64:
for i := range atoi64tests {
test := &atoi64tests[i]
out, ok := runtime.Atoi(test.in)
if test.out != int64(out) || test.ok != ok {
t.Errorf("atoi(%q) = (%v, %v) want (%v, %v)",
test.in, out, ok, test.out, test.ok)
}
}
}
}
type atoi32Test struct {
in string
out int32
ok bool
}
var atoi32tests = []atoi32Test{
{"", 0, false},
{"0", 0, true},
{"-0", 0, true},
{"1", 1, true},
{"-1", -1, true},
{"12345", 12345, true},
{"-12345", -12345, true},
{"012345", 12345, true},
{"-012345", -12345, true},
{"12345x", 0, false},
{"-12345x", 0, false},
{"987654321", 987654321, true},
{"-987654321", -987654321, true},
{"2147483647", 1<<31 - 1, true},
{"-2147483647", -(1<<31 - 1), true},
{"2147483648", 0, false},
{"-2147483648", -1 << 31, true},
{"2147483649", 0, false},
{"-2147483649", 0, false},
}
func TestAtoi32(t *testing.T) {
for i := range atoi32tests {
test := &atoi32tests[i]
out, ok := runtime.Atoi32(test.in)
if test.out != out || test.ok != ok {
t.Errorf("atoi32(%q) = (%v, %v) want (%v, %v)",
test.in, out, ok, test.out, test.ok)
}
}
}
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