| | |
| | |
| | |
| |
|
| | package obj |
| |
|
| | import ( |
| | "bytes" |
| | "internal/testenv" |
| | "os" |
| | "path/filepath" |
| | "testing" |
| | "unsafe" |
| |
|
| | "cmd/internal/goobj" |
| | "cmd/internal/sys" |
| | ) |
| |
|
| | 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) |
| | 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")}, |
| | &LSym{P: []byte("TestSymbol")}, |
| | &LSym{P: []byte("TestSymbol2")}, |
| | &LSym{P: []byte("")}, |
| | &LSym{P: []byte("")}, |
| | &LSym{P: []byte("")}, |
| | &LSym{P: []byte("")}, |
| | } |
| | for _, s := range syms { |
| | s.Set(AttrContentAddressable, true) |
| | s.PkgIdx = goobj.PkgIdxHashed |
| | } |
| | |
| | syms[3].R = []Reloc{{Sym: syms[0]}} |
| | |
| | syms[4].R = []Reloc{{Sym: syms[0]}} |
| | |
| | syms[5].R = []Reloc{{Sym: syms[1]}} |
| | |
| | syms[6].R = []Reloc{{Sym: syms[2]}} |
| |
|
| | |
| | h := make([]goobj.HashType, len(syms)) |
| | w := &writer{} |
| | for i := range h { |
| | h[i] = w.contentHash(syms[i]) |
| | } |
| |
|
| | tests := []struct { |
| | a, b int |
| | equal bool |
| | }{ |
| | {0, 1, true}, |
| | {0, 2, false}, |
| | {3, 4, true}, |
| | {3, 5, true}, |
| | {3, 6, false}, |
| | } |
| | 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) |
| | } |
| | } |
| | } |
| |
|
| | func TestSymbolTooLarge(t *testing.T) { |
| | testenv.MustHaveGoBuild(t) |
| | if unsafe.Sizeof(uintptr(0)) < 8 { |
| | t.Skip("skip on 32-bit architectures") |
| | } |
| |
|
| | tmpdir := t.TempDir() |
| |
|
| | src := filepath.Join(tmpdir, "p.go") |
| | err := os.WriteFile(src, []byte("package p; var x [1<<32]byte"), 0666) |
| | if err != nil { |
| | t.Fatalf("failed to write source file: %v\n", err) |
| | } |
| | obj := filepath.Join(tmpdir, "p.o") |
| | cmd := testenv.Command(t, testenv.GoToolPath(t), "tool", "compile", "-p=p", "-o", obj, src) |
| | 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) |
| | } |
| | } |
| |
|
| | func TestNoRefName(t *testing.T) { |
| | |
| | testenv.MustHaveGoBuild(t) |
| |
|
| | tmpdir := t.TempDir() |
| |
|
| | src := filepath.Join(tmpdir, "x.go") |
| | err := os.WriteFile(src, []byte("package main; import \"fmt\"; func main() { fmt.Println(123) }\n"), 0666) |
| | if err != nil { |
| | t.Fatalf("failed to write source file: %v\n", err) |
| | } |
| | exe := filepath.Join(tmpdir, "x.exe") |
| |
|
| | |
| | |
| | cmd := testenv.Command(t, testenv.GoToolPath(t), "build", "-gcflags=fmt=-d=norefname", "-o", exe, src) |
| | out, err := cmd.CombinedOutput() |
| | if err != nil { |
| | t.Fatalf("build failed: %v, output:\n%s", err, out) |
| | } |
| | } |
| |
|