Commit e7f6d8b2 authored by Caio Marcelo de Oliveira Filho's avatar Caio Marcelo de Oliveira Filho Committed by Rob Pike

cmd/cover: don't overskip children nodes when adding counters

When visiting the AST to add counters, there are special cases in which
the code calls cuts the walking short by returning nil. In some cases
certain nodes are ignored, e.g. Init and Cond inside IfStmt.

The fix is to explicitly walk all the children nodes (not only
Body and Else) when cutting the current walk. Similar approach
was taken with SwitchStmt and TypeSwitchStmt.

While the existing test code doesn't handle different counters in the
same line, the generated HTML report does it correctly (because it takes
column into account).

The previous behavior caused lines in function literals to not be
tracked when those literals were inside Init or Cond of an IfStmt for
example.

Fixes #14039.

Change-Id: Iad591363330843ad833bd79a0388d709c8d0c8aa
Reviewed-on: https://go-review.googlesource.com/19775Reviewed-by: default avatarRob Pike <r@golang.org>
parent f39cca94
...@@ -181,6 +181,10 @@ func (f *File) Visit(node ast.Node) ast.Visitor { ...@@ -181,6 +181,10 @@ func (f *File) Visit(node ast.Node) ast.Visitor {
} }
n.List = f.addCounters(n.Lbrace, n.Rbrace+1, n.List, true) // +1 to step past closing brace. n.List = f.addCounters(n.Lbrace, n.Rbrace+1, n.List, true) // +1 to step past closing brace.
case *ast.IfStmt: case *ast.IfStmt:
if n.Init != nil {
ast.Walk(f, n.Init)
}
ast.Walk(f, n.Cond)
ast.Walk(f, n.Body) ast.Walk(f, n.Body)
if n.Else == nil { if n.Else == nil {
return nil return nil
...@@ -219,11 +223,21 @@ func (f *File) Visit(node ast.Node) ast.Visitor { ...@@ -219,11 +223,21 @@ func (f *File) Visit(node ast.Node) ast.Visitor {
case *ast.SwitchStmt: case *ast.SwitchStmt:
// Don't annotate an empty switch - creates a syntax error. // Don't annotate an empty switch - creates a syntax error.
if n.Body == nil || len(n.Body.List) == 0 { if n.Body == nil || len(n.Body.List) == 0 {
if n.Init != nil {
ast.Walk(f, n.Init)
}
if n.Tag != nil {
ast.Walk(f, n.Tag)
}
return nil return nil
} }
case *ast.TypeSwitchStmt: case *ast.TypeSwitchStmt:
// Don't annotate an empty type switch - creates a syntax error. // Don't annotate an empty type switch - creates a syntax error.
if n.Body == nil || len(n.Body.List) == 0 { if n.Body == nil || len(n.Body.List) == 0 {
if n.Init != nil {
ast.Walk(f, n.Init)
}
ast.Walk(f, n.Assign)
return nil return nil
} }
} }
......
...@@ -24,6 +24,7 @@ func testAll() { ...@@ -24,6 +24,7 @@ func testAll() {
testSelect2() testSelect2()
testPanic() testPanic()
testEmptySwitches() testEmptySwitches()
testFunctionLiteral()
} }
// The indexes of the counters in testPanic are known to main.go // The indexes of the counters in testPanic are known to main.go
...@@ -216,3 +217,32 @@ func testEmptySwitches() { ...@@ -216,3 +217,32 @@ func testEmptySwitches() {
<-c <-c
check(LINE, 1) check(LINE, 1)
} }
func testFunctionLiteral() {
a := func(f func()) error {
f()
f()
return nil
}
b := func(f func()) bool {
f()
f()
return true
}
check(LINE, 1)
a(func() {
check(LINE, 2)
})
if err := a(func() {
check(LINE, 2)
}); err != nil {
}
switch b(func() {
check(LINE, 2)
}) {
}
}
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