Commit 4d913b33 authored by Daniel Martí's avatar Daniel Martí

cmd/vet: fix printf false negative with nested pointers

Pointers to compound objects (structs, slices, arrays, maps) are only
followed by fmt if the pointer is at the top level of an argument. This
is to minimise the chances of fmt running into loops.

However, vet did not follow this rule. It likely doesn't help that fmt
does not document that restriction well, which is being tracked in
 #28625.

Updates #27672.

Change-Id: Ie9bbd9b974eda5ab9a285986d207ef92fca4453e
Reviewed-on: https://go-review.googlesource.com/c/147997
Run-TryBot: Daniel Martí <mvdan@mvdan.cc>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarAlan Donovan <adonovan@google.com>
parent edb2d1cb
...@@ -234,7 +234,7 @@ func PrintfTests() { ...@@ -234,7 +234,7 @@ func PrintfTests() {
Printf("%T", someFunction) // ok: maybe someone wants to see the type Printf("%T", someFunction) // ok: maybe someone wants to see the type
// Bug: used to recur forever. // Bug: used to recur forever.
Printf("%p %x", recursiveStructV, recursiveStructV.next) Printf("%p %x", recursiveStructV, recursiveStructV.next)
Printf("%p %x", recursiveStruct1V, recursiveStruct1V.next) Printf("%p %x", recursiveStruct1V, recursiveStruct1V.next) // ERROR "Printf format %x has arg recursiveStruct1V\.next of wrong type \*testdata\.RecursiveStruct2"
Printf("%p %x", recursiveSliceV, recursiveSliceV) Printf("%p %x", recursiveSliceV, recursiveSliceV)
Printf("%p %x", recursiveMapV, recursiveMapV) Printf("%p %x", recursiveMapV, recursiveMapV)
// Special handling for Log. // Special handling for Log.
...@@ -670,4 +670,12 @@ func PointersToCompoundTypes() { ...@@ -670,4 +670,12 @@ func PointersToCompoundTypes() {
intMap := map[int]int{3: 4} intMap := map[int]int{3: 4}
fmt.Printf("%s", &intMap) // ERROR "Printf format %s has arg &intMap of wrong type \*map\[int\]int" fmt.Printf("%s", &intMap) // ERROR "Printf format %s has arg &intMap of wrong type \*map\[int\]int"
type T2 struct {
X string
}
type T1 struct {
X *T2
}
fmt.Printf("%s\n", T1{&T2{"x"}}) // ERROR "Printf format %s has arg T1{&T2{.x.}} of wrong type testdata\.T1"
} }
...@@ -227,9 +227,13 @@ func (f *File) matchArgTypeInternal(t printfArgType, typ types.Type, arg ast.Exp ...@@ -227,9 +227,13 @@ func (f *File) matchArgTypeInternal(t printfArgType, typ types.Type, arg ast.Exp
// Check whether the rest can print pointers. // Check whether the rest can print pointers.
return t&argPointer != 0 return t&argPointer != 0
} }
// If it's a pointer to a struct, array, slice, or map, that's // If it's a top-level pointer to a struct, array, slice, or
// equivalent in our analysis to whether we can print the type // map, that's equivalent in our analysis to whether we can
// being pointed to. // print the type being pointed to. Pointers in nested levels
// are not supported to minimize fmt running into loops.
if len(inProgress) > 1 {
return false
}
return f.matchArgTypeInternal(t, under, arg, inProgress) return f.matchArgTypeInternal(t, under, arg, inProgress)
case *types.Struct: case *types.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