2020-07-15 23:14:50 -04:00
|
|
|
// Copyright 2020 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.
|
|
|
|
|
|
|
|
|
|
package obj
|
|
|
|
|
|
|
|
|
|
import (
|
2020-10-19 10:40:24 -04:00
|
|
|
"bytes"
|
|
|
|
|
"internal/testenv"
|
|
|
|
|
"os"
|
|
|
|
|
"path/filepath"
|
2020-07-15 23:14:50 -04:00
|
|
|
"testing"
|
2020-10-19 10:40:24 -04:00
|
|
|
"unsafe"
|
2022-08-28 03:38:00 +08:00
|
|
|
|
|
|
|
|
"cmd/internal/goobj"
|
|
|
|
|
"cmd/internal/sys"
|
2020-07-15 23:14:50 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
var dummyArch = LinkArch{Arch: sys.ArchAMD64}
|
|
|
|
|
|
|
|
|
|
func TestContentHash64(t *testing.T) {
|
|
|
|
|
s1 := &LSym{P: []byte("A")}
|
|
|
|
|
s2 := &LSym{P: []byte("A\x00\x00\x00")}
|
|
|
|
|
s1.Set(AttrContentAddressable, true)
|
|
|
|
|
s2.Set(AttrContentAddressable, true)
|
|
|
|
|
h1 := contentHash64(s1)
|
|
|
|
|
h2 := contentHash64(s2)
|
|
|
|
|
if h1 != h2 {
|
|
|
|
|
t.Errorf("contentHash64(s1)=%x, contentHash64(s2)=%x, expect equal", h1, h2)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ctxt := Linknew(&dummyArch) // little endian
|
|
|
|
|
s3 := ctxt.Int64Sym(int64('A'))
|
|
|
|
|
h3 := contentHash64(s3)
|
|
|
|
|
if h1 != h3 {
|
|
|
|
|
t.Errorf("contentHash64(s1)=%x, contentHash64(s3)=%x, expect equal", h1, h3)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func TestContentHash(t *testing.T) {
|
|
|
|
|
syms := []*LSym{
|
|
|
|
|
&LSym{P: []byte("TestSymbol")}, // 0
|
|
|
|
|
&LSym{P: []byte("TestSymbol")}, // 1
|
|
|
|
|
&LSym{P: []byte("TestSymbol2")}, // 2
|
|
|
|
|
&LSym{P: []byte("")}, // 3
|
|
|
|
|
&LSym{P: []byte("")}, // 4
|
|
|
|
|
&LSym{P: []byte("")}, // 5
|
|
|
|
|
&LSym{P: []byte("")}, // 6
|
|
|
|
|
}
|
|
|
|
|
for _, s := range syms {
|
|
|
|
|
s.Set(AttrContentAddressable, true)
|
2020-08-02 19:36:28 -04:00
|
|
|
s.PkgIdx = goobj.PkgIdxHashed
|
2020-07-15 23:14:50 -04:00
|
|
|
}
|
|
|
|
|
// s3 references s0
|
|
|
|
|
r := Addrel(syms[3])
|
|
|
|
|
r.Sym = syms[0]
|
|
|
|
|
// s4 references s0
|
|
|
|
|
r = Addrel(syms[4])
|
|
|
|
|
r.Sym = syms[0]
|
|
|
|
|
// s5 references s1
|
|
|
|
|
r = Addrel(syms[5])
|
|
|
|
|
r.Sym = syms[1]
|
|
|
|
|
// s6 references s2
|
|
|
|
|
r = Addrel(syms[6])
|
|
|
|
|
r.Sym = syms[2]
|
|
|
|
|
|
|
|
|
|
// compute hashes
|
2020-08-02 19:36:28 -04:00
|
|
|
h := make([]goobj.HashType, len(syms))
|
2020-07-15 23:14:50 -04:00
|
|
|
w := &writer{}
|
|
|
|
|
for i := range h {
|
|
|
|
|
h[i] = w.contentHash(syms[i])
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
tests := []struct {
|
|
|
|
|
a, b int
|
|
|
|
|
equal bool
|
|
|
|
|
}{
|
|
|
|
|
{0, 1, true}, // same contents, no relocs
|
|
|
|
|
{0, 2, false}, // different contents
|
|
|
|
|
{3, 4, true}, // same contents, same relocs
|
|
|
|
|
{3, 5, true}, // recursively same contents
|
|
|
|
|
{3, 6, false}, // same contents, different relocs
|
|
|
|
|
}
|
|
|
|
|
for _, test := range tests {
|
|
|
|
|
if (h[test.a] == h[test.b]) != test.equal {
|
|
|
|
|
eq := "equal"
|
|
|
|
|
if !test.equal {
|
|
|
|
|
eq = "not equal"
|
|
|
|
|
}
|
|
|
|
|
t.Errorf("h%d=%x, h%d=%x, expect %s", test.a, h[test.a], test.b, h[test.b], eq)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-10-19 10:40:24 -04:00
|
|
|
|
|
|
|
|
func TestSymbolTooLarge(t *testing.T) { // Issue 42054
|
|
|
|
|
testenv.MustHaveGoBuild(t)
|
|
|
|
|
if unsafe.Sizeof(uintptr(0)) < 8 {
|
|
|
|
|
t.Skip("skip on 32-bit architectures")
|
|
|
|
|
}
|
|
|
|
|
|
2024-09-04 18:59:49 -07:00
|
|
|
tmpdir := t.TempDir()
|
2020-10-19 10:40:24 -04:00
|
|
|
|
|
|
|
|
src := filepath.Join(tmpdir, "p.go")
|
2024-09-04 18:59:49 -07:00
|
|
|
err := os.WriteFile(src, []byte("package p; var x [1<<32]byte"), 0666)
|
2020-10-19 10:40:24 -04:00
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("failed to write source file: %v\n", err)
|
|
|
|
|
}
|
|
|
|
|
obj := filepath.Join(tmpdir, "p.o")
|
2022-11-15 10:14:16 -05:00
|
|
|
cmd := testenv.Command(t, testenv.GoToolPath(t), "tool", "compile", "-p=p", "-o", obj, src)
|
2020-10-19 10:40:24 -04:00
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
|
if err == nil {
|
|
|
|
|
t.Fatalf("did not fail\noutput: %s", out)
|
|
|
|
|
}
|
|
|
|
|
const want = "symbol too large"
|
|
|
|
|
if !bytes.Contains(out, []byte(want)) {
|
|
|
|
|
t.Errorf("unexpected error message: want: %q, got: %s", want, out)
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-05-05 17:22:17 -04:00
|
|
|
|
|
|
|
|
func TestNoRefName(t *testing.T) {
|
|
|
|
|
// Test that the norefname flag works.
|
|
|
|
|
testenv.MustHaveGoBuild(t)
|
|
|
|
|
|
|
|
|
|
tmpdir := t.TempDir()
|
|
|
|
|
|
|
|
|
|
src := filepath.Join(tmpdir, "x.go")
|
2022-08-28 03:38:00 +08:00
|
|
|
err := os.WriteFile(src, []byte("package main; import \"fmt\"; func main() { fmt.Println(123) }\n"), 0666)
|
2022-05-05 17:22:17 -04:00
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("failed to write source file: %v\n", err)
|
|
|
|
|
}
|
|
|
|
|
exe := filepath.Join(tmpdir, "x.exe")
|
|
|
|
|
|
|
|
|
|
// Build the fmt package with norefname. Not rebuilding all packages to save time.
|
|
|
|
|
// Also testing that norefname and non-norefname packages can link together.
|
2022-11-15 10:14:16 -05:00
|
|
|
cmd := testenv.Command(t, testenv.GoToolPath(t), "build", "-gcflags=fmt=-d=norefname", "-o", exe, src)
|
2022-05-05 17:22:17 -04:00
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("build failed: %v, output:\n%s", err, out)
|
|
|
|
|
}
|
|
|
|
|
}
|