]> Cypherpunks repositories - gostls13.git/commitdiff
cmd/compile: fix panic in DWARF-gen handling obfuscated code
authorThan McIntosh <thanm@google.com>
Fri, 19 Feb 2021 15:09:15 +0000 (10:09 -0500)
committerThan McIntosh <thanm@google.com>
Sun, 21 Feb 2021 02:25:26 +0000 (02:25 +0000)
DWARF generation uses variable source positions (file/line/col) as a
way to uniquely identify locals and parameters, as part of the process
of matching up post-optimization variables with the corresponding
pre-optimization versions (since the DWARF needs to be in terms of the
original source constructs).

This strategy can run into problems when compiling obfuscated or
machine-generated code, where you can in some circumstances wind up
with two local variables that appear to have the same name, file,
line, and column. This patch changes DWARF generation to skip over
such duplicates as opposed to issuing a fatal error (if an
obfuscation tool is in use, it is unlikely that a human being will be
able to make much sense of DWARF info in any case).

Fixes #44378.

Change-Id: I198022d184701aa9ec3dce42c005d29b72d2e321
Reviewed-on: https://go-review.googlesource.com/c/go/+/294289
TryBot-Result: Go Bot <gobot@golang.org>
Reviewed-by: David Chase <drchase@google.com>
Reviewed-by: Jeremy Faller <jeremy@golang.org>
Reviewed-by: Cherry Zhang <cherryyz@google.com>
Trust: Than McIntosh <thanm@google.com>
Run-TryBot: Than McIntosh <thanm@google.com>

src/cmd/compile/internal/dwarfgen/dwinl.go
test/fixedbugs/issue44378.go [new file with mode: 0644]

index d5687cb1d72bb41465340dfb77c9deb759131254..8adb36fc883da0614f760410fa8f6af8e7ccb25e 100644 (file)
@@ -247,7 +247,8 @@ func makePreinlineDclMap(fnsym *obj.LSym) map[varPos]int {
                        DeclCol:  pos.Col(),
                }
                if _, found := m[vp]; found {
-                       base.Fatalf("child dcl collision on symbol %s within %v\n", n.Sym().Name, fnsym.Name)
+                       // We can see collisions (variables with the same name/file/line/col) in obfuscated or machine-generated code -- see issue 44378 for an example. Skip duplicates in such cases, since it is unlikely that a human will be debugging such code.
+                       continue
                }
                m[vp] = i
        }
diff --git a/test/fixedbugs/issue44378.go b/test/fixedbugs/issue44378.go
new file mode 100644 (file)
index 0000000..58c88d5
--- /dev/null
@@ -0,0 +1,40 @@
+// compile
+
+// Copyright 2021 The Go Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style
+// license that can be found in the LICENSE file.
+
+// This test case caused a panic in the compiler's DWARF gen code.
+
+// Note to future maintainers of this code:
+//
+//    ** Do NOT run gofmt when editing this file **
+//
+// In order for the buggy behavior to be triggered in the compiler,
+// we need to have a the function of interest all on one gigantic line.
+
+package a
+
+type O interface{}
+type IO int
+type OS int
+
+type A struct {
+       x int
+}
+
+// original versions of the two function
+func (p *A) UO(o O) {
+       p.r(o, o)
+}
+func (p *A) r(o1, o2 O) {
+       switch x := o1.(type) {
+       case *IO:
+               p.x = int(*x)
+       case *OS:
+               p.x = int(*x + 2)
+       }
+}
+
+// see note above about the importance of all this code winding up on one line.
+var myverylongname0123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789012345678901234567890123456789 int ; func (p *A) UO2(o O) { p.r2(o, o); }; func (p *A) r2(o1, o2 O) { switch x := o1.(type) { case *IO:      p.x = int(*x);  case *OS:       p.x = int(*x + 2); } }