Commit c9fbe0f2 authored by Aliaksandr Valialkin's avatar Aliaksandr Valialkin Committed by Rob Pike

cmd/vet: properly handle indexed arguments in printf

Fixes #15884

Change-Id: I33d98db861d74e3c37a546efaf83ce6f2f76d335
Reviewed-on: https://go-review.googlesource.com/24391
Run-TryBot: Rob Pike <r@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarRob Pike <r@golang.org>
parent 49551472
...@@ -194,7 +194,6 @@ type formatState struct { ...@@ -194,7 +194,6 @@ type formatState struct {
name string // Printf, Sprintf etc. name string // Printf, Sprintf etc.
flags []byte // the list of # + etc. flags []byte // the list of # + etc.
argNums []int // the successive argument numbers that are consumed, adjusted to refer to actual arg in call argNums []int // the successive argument numbers that are consumed, adjusted to refer to actual arg in call
indexed bool // whether an indexing expression appears: %[1]d.
firstArg int // Index of first argument after the format in the Printf call. firstArg int // Index of first argument after the format in the Printf call.
// Used only during parse. // Used only during parse.
file *File file *File
...@@ -223,7 +222,7 @@ func (f *File) checkPrintf(call *ast.CallExpr, name string) { ...@@ -223,7 +222,7 @@ func (f *File) checkPrintf(call *ast.CallExpr, name string) {
} }
// Hard part: check formats against args. // Hard part: check formats against args.
argNum := firstArg argNum := firstArg
indexed := false maxArgNum := firstArg
for i, w := 0, 0; i < len(format); i += w { for i, w := 0, 0; i < len(format); i += w {
w = 1 w = 1
if format[i] == '%' { if format[i] == '%' {
...@@ -232,9 +231,6 @@ func (f *File) checkPrintf(call *ast.CallExpr, name string) { ...@@ -232,9 +231,6 @@ func (f *File) checkPrintf(call *ast.CallExpr, name string) {
return return
} }
w = len(state.format) w = len(state.format)
if state.indexed {
indexed = true
}
if !f.okPrintfArg(call, state) { // One error per format is enough. if !f.okPrintfArg(call, state) { // One error per format is enough.
return return
} }
...@@ -242,16 +238,20 @@ func (f *File) checkPrintf(call *ast.CallExpr, name string) { ...@@ -242,16 +238,20 @@ func (f *File) checkPrintf(call *ast.CallExpr, name string) {
// Continue with the next sequential argument. // Continue with the next sequential argument.
argNum = state.argNums[len(state.argNums)-1] + 1 argNum = state.argNums[len(state.argNums)-1] + 1
} }
for _, n := range state.argNums {
if n >= maxArgNum {
maxArgNum = n + 1
}
}
} }
} }
// Dotdotdot is hard. // Dotdotdot is hard.
if call.Ellipsis.IsValid() && argNum >= len(call.Args)-1 { if call.Ellipsis.IsValid() && maxArgNum >= len(call.Args)-1 {
return return
} }
// If the arguments were direct indexed, we assume the programmer knows what's up. // There should be no leftover arguments.
// Otherwise, there should be no leftover arguments. if maxArgNum != len(call.Args) {
if !indexed && argNum != len(call.Args) { expect := maxArgNum - firstArg
expect := argNum - firstArg
numArgs := len(call.Args) - firstArg numArgs := len(call.Args) - firstArg
f.Badf(call.Pos(), "wrong number of args for format in %s call: %d needed but %d args", name, expect, numArgs) f.Badf(call.Pos(), "wrong number of args for format in %s call: %d needed but %d args", name, expect, numArgs)
} }
...@@ -286,17 +286,20 @@ func (s *formatState) parseIndex() bool { ...@@ -286,17 +286,20 @@ func (s *formatState) parseIndex() bool {
return true return true
} }
// Argument index present. // Argument index present.
s.indexed = true
s.nbytes++ // skip '[' s.nbytes++ // skip '['
start := s.nbytes start := s.nbytes
s.scanNum() s.scanNum()
if s.nbytes == len(s.format) || s.nbytes == start || s.format[s.nbytes] != ']' { if s.nbytes == len(s.format) || s.nbytes == start || s.format[s.nbytes] != ']' {
s.file.Badf(s.call.Pos(), "illegal syntax for printf argument index") end := strings.Index(s.format, "]")
if end < 0 {
end = len(s.format)
}
s.file.Badf(s.call.Pos(), "bad syntax for printf argument index: [%s]", s.format[start:end])
return false return false
} }
arg32, err := strconv.ParseInt(s.format[start:s.nbytes], 10, 32) arg32, err := strconv.ParseInt(s.format[start:s.nbytes], 10, 32)
if err != nil { if err != nil {
s.file.Badf(s.call.Pos(), "illegal syntax for printf argument index: %s", err) s.file.Badf(s.call.Pos(), "bad syntax for printf argument index: %s", err)
return false return false
} }
s.nbytes++ // skip ']' s.nbytes++ // skip ']'
...@@ -349,14 +352,12 @@ func (f *File) parsePrintfVerb(call *ast.CallExpr, name, format string, firstArg ...@@ -349,14 +352,12 @@ func (f *File) parsePrintfVerb(call *ast.CallExpr, name, format string, firstArg
argNum: argNum, argNum: argNum,
argNums: make([]int, 0, 1), argNums: make([]int, 0, 1),
nbytes: 1, // There's guaranteed to be a percent sign. nbytes: 1, // There's guaranteed to be a percent sign.
indexed: false,
firstArg: firstArg, firstArg: firstArg,
file: f, file: f,
call: call, call: call,
} }
// There may be flags. // There may be flags.
state.parseFlags() state.parseFlags()
indexPending := false
// There may be an index. // There may be an index.
if !state.parseIndex() { if !state.parseIndex() {
return nil return nil
...@@ -370,7 +371,7 @@ func (f *File) parsePrintfVerb(call *ast.CallExpr, name, format string, firstArg ...@@ -370,7 +371,7 @@ func (f *File) parsePrintfVerb(call *ast.CallExpr, name, format string, firstArg
return nil return nil
} }
// Now a verb, possibly prefixed by an index (which we may already have). // Now a verb, possibly prefixed by an index (which we may already have).
if !indexPending && !state.parseIndex() { if !state.indexPending && !state.parseIndex() {
return nil return nil
} }
if state.nbytes == len(state.format) { if state.nbytes == len(state.format) {
......
...@@ -174,8 +174,8 @@ func PrintfTests() { ...@@ -174,8 +174,8 @@ func PrintfTests() {
Printf("%[2]*.[1]*[3]d", 2, 3, 4) Printf("%[2]*.[1]*[3]d", 2, 3, 4)
fmt.Fprintf(os.Stderr, "%[2]*.[1]*[3]d", 2, 3, 4) // Use Fprintf to make sure we count arguments correctly. fmt.Fprintf(os.Stderr, "%[2]*.[1]*[3]d", 2, 3, 4) // Use Fprintf to make sure we count arguments correctly.
// Bad argument reorderings. // Bad argument reorderings.
Printf("%[xd", 3) // ERROR "illegal syntax for printf argument index" Printf("%[xd", 3) // ERROR "bad syntax for printf argument index: \[xd\]"
Printf("%[x]d", 3) // ERROR "illegal syntax for printf argument index" Printf("%[x]d", 3) // ERROR "bad syntax for printf argument index: \[x\]"
Printf("%[3]*s", "hi", 2) // ERROR "missing argument for Printf.* reads arg 3, have only 2" Printf("%[3]*s", "hi", 2) // ERROR "missing argument for Printf.* reads arg 3, have only 2"
_ = fmt.Sprintf("%[3]d", 2) // ERROR "missing argument for Sprintf.* reads arg 3, have only 1" _ = fmt.Sprintf("%[3]d", 2) // ERROR "missing argument for Sprintf.* reads arg 3, have only 1"
Printf("%[2]*.[1]*[3]d", 2, "hi", 4) // ERROR "arg .hi. for \* in printf format not of type int" Printf("%[2]*.[1]*[3]d", 2, "hi", 4) // ERROR "arg .hi. for \* in printf format not of type int"
...@@ -249,6 +249,15 @@ func PrintfTests() { ...@@ -249,6 +249,15 @@ func PrintfTests() {
ss.log(someFunction) // OK ss.log(someFunction) // OK
ss.log(someFunction, "bar", 1.33) // OK ss.log(someFunction, "bar", 1.33) // OK
ss.log(someFunction, someFunction) // ERROR "arg someFunction in log call is a function value, not a function call" ss.log(someFunction, someFunction) // ERROR "arg someFunction in log call is a function value, not a function call"
// indexed arguments
Printf("%d %[3]d %d %[2]d", 1, 2, 3, 4) // OK
Printf("%d %[0]d %d %[2]d", 1, 2, 3, 4) // ERROR "indexes start at 1"
Printf("%d %[3]d %d %[-2]d", 1, 2, 3, 4) // ERROR "bad syntax for printf argument index: \[-2\]"
Printf("%d %[3]d %d %[2234234234234]d", 1, 2, 3, 4) // ERROR "bad syntax for printf argument index: .+ value out of range"
Printf("%d %[3]d %d %[2]d", 1, 2, 3) // ERROR "format reads arg 4, have only 3 args"
Printf("%d %[3]d %d %[2]d", 1, 2, 3, 4, 5) // ERROR "wrong number of args for format in Printf call: 4 needed but 5 args"
Printf("%[1][3]d", 1, 2) // ERROR "unrecognized printf verb '\['"
} }
type someStruct struct{} type someStruct struct{}
......
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