| // Copyright 2016 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 arm64 |
| |
| import ( |
| "bytes" |
| "fmt" |
| "internal/testenv" |
| "io/ioutil" |
| "os" |
| "os/exec" |
| "path/filepath" |
| "regexp" |
| "testing" |
| ) |
| |
| // TestLarge generates a very large file to verify that large |
| // program builds successfully, in particular, too-far |
| // conditional branches are fixed, and also verify that the |
| // instruction's pc can be correctly aligned even when branches |
| // need to be fixed. |
| func TestLarge(t *testing.T) { |
| if testing.Short() { |
| t.Skip("Skip in short mode") |
| } |
| testenv.MustHaveGoBuild(t) |
| |
| dir, err := ioutil.TempDir("", "testlarge") |
| if err != nil { |
| t.Fatalf("could not create directory: %v", err) |
| } |
| defer os.RemoveAll(dir) |
| |
| // generate a very large function |
| buf := bytes.NewBuffer(make([]byte, 0, 7000000)) |
| gen(buf) |
| |
| tmpfile := filepath.Join(dir, "x.s") |
| err = ioutil.WriteFile(tmpfile, buf.Bytes(), 0644) |
| if err != nil { |
| t.Fatalf("can't write output: %v\n", err) |
| } |
| |
| pattern := `0x0080\s00128\s\(.*\)\tMOVD\t\$3,\sR3` |
| |
| // assemble generated file |
| cmd := exec.Command(testenv.GoToolPath(t), "tool", "asm", "-S", "-o", filepath.Join(dir, "test.o"), tmpfile) |
| cmd.Env = append(os.Environ(), "GOOS=linux") |
| out, err := cmd.CombinedOutput() |
| if err != nil { |
| t.Errorf("Assemble failed: %v, output: %s", err, out) |
| } |
| matched, err := regexp.MatchString(pattern, string(out)) |
| if err != nil { |
| t.Fatal(err) |
| } |
| if !matched { |
| t.Errorf("The alignment is not correct: %t, output:%s\n", matched, out) |
| } |
| |
| // build generated file |
| cmd = exec.Command(testenv.GoToolPath(t), "tool", "asm", "-o", filepath.Join(dir, "x.o"), tmpfile) |
| cmd.Env = append(os.Environ(), "GOOS=linux") |
| out, err = cmd.CombinedOutput() |
| if err != nil { |
| t.Errorf("Build failed: %v, output: %s", err, out) |
| } |
| } |
| |
| // gen generates a very large program, with a very far conditional branch. |
| func gen(buf *bytes.Buffer) { |
| fmt.Fprintln(buf, "TEXT f(SB),0,$0-0") |
| fmt.Fprintln(buf, "TBZ $5, R0, label") |
| fmt.Fprintln(buf, "CBZ R0, label") |
| fmt.Fprintln(buf, "BEQ label") |
| fmt.Fprintln(buf, "PCALIGN $128") |
| fmt.Fprintln(buf, "MOVD $3, R3") |
| for i := 0; i < 1<<19; i++ { |
| fmt.Fprintln(buf, "MOVD R0, R1") |
| } |
| fmt.Fprintln(buf, "label:") |
| fmt.Fprintln(buf, "RET") |
| } |
| |
| // Issue 20348. |
| func TestNoRet(t *testing.T) { |
| dir, err := ioutil.TempDir("", "testnoret") |
| if err != nil { |
| t.Fatal(err) |
| } |
| defer os.RemoveAll(dir) |
| tmpfile := filepath.Join(dir, "x.s") |
| if err := ioutil.WriteFile(tmpfile, []byte("TEXT ·stub(SB),$0-0\nNOP\n"), 0644); err != nil { |
| t.Fatal(err) |
| } |
| cmd := exec.Command(testenv.GoToolPath(t), "tool", "asm", "-o", filepath.Join(dir, "x.o"), tmpfile) |
| cmd.Env = append(os.Environ(), "GOOS=linux") |
| if out, err := cmd.CombinedOutput(); err != nil { |
| t.Errorf("%v\n%s", err, out) |
| } |
| } |
| |
| // TestPCALIGN verifies the correctness of the PCALIGN by checking if the |
| // code can be aligned to the alignment value. |
| func TestPCALIGN(t *testing.T) { |
| testenv.MustHaveGoBuild(t) |
| dir, err := ioutil.TempDir("", "testpcalign") |
| if err != nil { |
| t.Fatal(err) |
| } |
| defer os.RemoveAll(dir) |
| tmpfile := filepath.Join(dir, "test.s") |
| tmpout := filepath.Join(dir, "test.o") |
| |
| code1 := []byte("TEXT ·foo(SB),$0-0\nMOVD $0, R0\nPCALIGN $8\nMOVD $1, R1\nRET\n") |
| code2 := []byte("TEXT ·foo(SB),$0-0\nMOVD $0, R0\nPCALIGN $16\nMOVD $2, R2\nRET\n") |
| // If the output contains this pattern, the pc-offsite of "MOVD $1, R1" is 8 bytes aligned. |
| out1 := `0x0008\s00008\s\(.*\)\tMOVD\t\$1,\sR1` |
| // If the output contains this pattern, the pc-offsite of "MOVD $2, R2" is 16 bytes aligned. |
| out2 := `0x0010\s00016\s\(.*\)\tMOVD\t\$2,\sR2` |
| var testCases = []struct { |
| name string |
| code []byte |
| out string |
| }{ |
| {"8-byte alignment", code1, out1}, |
| {"16-byte alignment", code2, out2}, |
| } |
| |
| for _, test := range testCases { |
| if err := ioutil.WriteFile(tmpfile, test.code, 0644); err != nil { |
| t.Fatal(err) |
| } |
| cmd := exec.Command(testenv.GoToolPath(t), "tool", "asm", "-S", "-o", tmpout, tmpfile) |
| cmd.Env = append(os.Environ(), "GOOS=linux") |
| out, err := cmd.CombinedOutput() |
| if err != nil { |
| t.Errorf("The %s build failed: %v, output: %s", test.name, err, out) |
| continue |
| } |
| |
| matched, err := regexp.MatchString(test.out, string(out)) |
| if err != nil { |
| t.Fatal(err) |
| } |
| if !matched { |
| t.Errorf("The %s testing failed!\ninput: %s\noutput: %s\n", test.name, test.code, out) |
| } |
| } |
| } |
| |
| func testvmovq() (r1, r2 uint64) |
| |
| // TestVMOVQ checks if the arm64 VMOVQ instruction is working properly. |
| func TestVMOVQ(t *testing.T) { |
| a, b := testvmovq() |
| if a != 0x7040201008040201 || b != 0x3040201008040201 { |
| t.Errorf("TestVMOVQ got: a=0x%x, b=0x%x, want: a=0x7040201008040201, b=0x3040201008040201", a, b) |
| } |
| } |
| |
| func testmovk() uint64 |
| |
| // TestMOVK makes sure MOVK with a very large constant works. See issue 52261. |
| func TestMOVK(t *testing.T) { |
| x := testmovk() |
| want := uint64(40000 << 48) |
| if x != want { |
| t.Errorf("TestMOVK got %x want %x\n", x, want) |
| } |
| } |