diff options
| author | David Chase <drchase@google.com> | 2016-11-14 18:00:17 -0500 |
|---|---|---|
| committer | David Chase <drchase@google.com> | 2016-11-16 22:05:19 +0000 |
| commit | 9f5673d9307572ff3b435a845470e3b7fd3c6a43 (patch) | |
| tree | f4c0a4db2896b4c8776ec944d550d2cc772423fe /src | |
| parent | 1e3c57c2cc1500b12a35a859f3d6e8aa27aeebc5 (diff) | |
| download | go-9f5673d9307572ff3b435a845470e3b7fd3c6a43.tar.xz | |
cmd/compile: ensure necessary types appear in .debug_info
Autotmp filtering was too aggressive and excluded types
necessary to make debuggers work properly. Restore the
"late filter" in dwarf.go based on names to exclude autotmps,
and remove the "early filter" in pgen.go based on how the
name was introduced. However, the updated naming scheme
with a dot prefix is retained to prevent accidental clashes
with legal Go identifier names.
Includes test (grouped with runtime gdb tests),
verified to fail without the fix.
Updates #17644.
Fixes #17830.
Change-Id: I7ec3f7230083889660236e5f6bc77ba5fe434e93
Reviewed-on: https://go-review.googlesource.com/33233
Run-TryBot: David Chase <drchase@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Matthew Dempsky <mdempsky@google.com>
Diffstat (limited to 'src')
| -rw-r--r-- | src/cmd/compile/internal/gc/pgen.go | 5 | ||||
| -rw-r--r-- | src/cmd/internal/dwarf/dwarf.go | 4 | ||||
| -rw-r--r-- | src/runtime/runtime-gdb_test.go | 70 |
3 files changed, 76 insertions, 3 deletions
diff --git a/src/cmd/compile/internal/gc/pgen.go b/src/cmd/compile/internal/gc/pgen.go index 91256229a7..acea790498 100644 --- a/src/cmd/compile/internal/gc/pgen.go +++ b/src/cmd/compile/internal/gc/pgen.go @@ -427,9 +427,8 @@ func compile(fn *Node) { } fallthrough case PPARAM, PPARAMOUT: - if n.IsAutoTmp() { // skip debugging info for temporaries - continue - } + // The symbol is excluded later from debugging info if its name begins ".autotmp_", but the type is still necessary. + // See bugs #17644 and #17830 and cmd/internal/dwarf/dwarf.go p := Gins(obj.ATYPE, n, nil) p.From.Sym = obj.Linklookup(Ctxt, n.Sym.Name, 0) p.To.Type = obj.TYPE_MEM diff --git a/src/cmd/internal/dwarf/dwarf.go b/src/cmd/internal/dwarf/dwarf.go index c72ef5b0d4..725f5027bb 100644 --- a/src/cmd/internal/dwarf/dwarf.go +++ b/src/cmd/internal/dwarf/dwarf.go @@ -9,6 +9,7 @@ package dwarf import ( "fmt" + "strings" ) // InfoPrefix is the prefix for all the symbols containing DWARF info entries. @@ -576,6 +577,9 @@ func PutFunc(ctxt Context, s Sym, name string, external bool, startPC Sym, size putattr(ctxt, s, DW_ABRV_FUNCTION, DW_FORM_flag, DW_CLS_FLAG, ev, 0) names := make(map[string]bool) for v := vars; v != nil; v = v.Link { + if strings.Contains(v.Name, ".autotmp_") { + continue + } var n string if names[v.Name] { n = fmt.Sprintf("%s#%d", v.Name, len(names)) diff --git a/src/runtime/runtime-gdb_test.go b/src/runtime/runtime-gdb_test.go index 06e61e9ced..94ba879ed1 100644 --- a/src/runtime/runtime-gdb_test.go +++ b/src/runtime/runtime-gdb_test.go @@ -15,6 +15,7 @@ import ( "regexp" "runtime" "strconv" + "strings" "testing" ) @@ -275,3 +276,72 @@ func TestGdbBacktrace(t *testing.T) { } } } + +const autotmpTypeSource = ` +package main + +type astruct struct { + a, b int +} + +func main() { + var iface interface{} = map[string]astruct{} + var iface2 interface{} = []astruct{} + println(iface, iface2) +} +` + +// TestGdbAutotmpTypes ensures that types of autotmp variables appear in .debug_info +// See bug #17830. +func TestGdbAutotmpTypes(t *testing.T) { + t.Parallel() + checkGdbEnvironment(t) + checkGdbVersion(t) + + dir, err := ioutil.TempDir("", "go-build") + if err != nil { + t.Fatalf("failed to create temp directory: %v", err) + } + defer os.RemoveAll(dir) + + // Build the source code. + src := filepath.Join(dir, "main.go") + err = ioutil.WriteFile(src, []byte(autotmpTypeSource), 0644) + if err != nil { + t.Fatalf("failed to create file: %v", err) + } + cmd := exec.Command(testenv.GoToolPath(t), "build", "-gcflags=-N -l", "-o", "a.exe") + cmd.Dir = dir + out, err := testEnv(cmd).CombinedOutput() + if err != nil { + t.Fatalf("building source %v\n%s", err, out) + } + + // Execute gdb commands. + args := []string{"-nx", "-batch", + "-ex", "set startup-with-shell off", + "-ex", "break main.main", + "-ex", "run", + "-ex", "step", + "-ex", "info types astruct", + filepath.Join(dir, "a.exe"), + } + got, _ := exec.Command("gdb", args...).CombinedOutput() + + sgot := string(got) + + // Check that the backtrace matches the source code. + types := []string{ + "struct []main.astruct;", + "struct bucket<string,main.astruct>;", + "struct hash<string,main.astruct>;", + "struct main.astruct;", + "typedef struct hash<string,main.astruct> * map[string]main.astruct;", + } + for _, name := range types { + if !strings.Contains(sgot, name) { + t.Errorf("could not find %s in 'info typrs astruct' output", name) + t.Fatalf("gdb output:\n%v", sgot) + } + } +} |
