// Copyright 2023 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 loong64 import ( "bytes" "fmt" "internal/testenv" "os" "path/filepath" "regexp" "runtime" "testing" ) const genBufSize = (1024 * 1024 * 32) // 32MB // TestLargeBranch generates a large function with a very far conditional // branch, in order to ensure that it assembles successfully. func TestLargeBranch(t *testing.T) { if testing.Short() { t.Skip("Skipping test in short mode") } testenv.MustHaveGoBuild(t) dir := t.TempDir() // Generate a very large function. buf := bytes.NewBuffer(make([]byte, 0, genBufSize)) genLargeBranch(buf) tmpfile := filepath.Join(dir, "x.s") if err := os.WriteFile(tmpfile, buf.Bytes(), 0644); err != nil { t.Fatalf("Failed to write file: %v", err) } // Assemble generated file. cmd := testenv.Command(t, testenv.GoToolPath(t), "tool", "asm", "-o", filepath.Join(dir, "x.o"), tmpfile) cmd.Env = append(os.Environ(), "GOARCH=loong64", "GOOS=linux") out, err := cmd.CombinedOutput() if err != nil { t.Errorf("Build failed: %v, output: %s", err, out) } } func genLargeBranch(buf *bytes.Buffer) { genSize1 := (1 << 16) + 16 genSize2 := (1 << 21) + 16 fmt.Fprintln(buf, "TEXT f(SB),0,$0-0") fmt.Fprintln(buf, "BEQ R5, R6, label18") fmt.Fprintln(buf, "BNE R5, R6, label18") fmt.Fprintln(buf, "BGE R5, R6, label18") fmt.Fprintln(buf, "BGEU R5, R6, label18") fmt.Fprintln(buf, "BLTU R5, R6, label18") fmt.Fprintln(buf, "BLEZ R5, label18") fmt.Fprintln(buf, "BGEZ R5, label18") fmt.Fprintln(buf, "BLTZ R5, label18") fmt.Fprintln(buf, "BGTZ R5, label18") fmt.Fprintln(buf, "BFPT label23") fmt.Fprintln(buf, "BFPF label23") fmt.Fprintln(buf, "BEQ R5, label23") fmt.Fprintln(buf, "BNE R5, label23") for i := 0; i <= genSize1; i++ { fmt.Fprintln(buf, "ADDV $0, R0, R0") } fmt.Fprintln(buf, "label18:") for i := 0; i <= (genSize2 - genSize1); i++ { fmt.Fprintln(buf, "ADDV $0, R0, R0") } fmt.Fprintln(buf, "label23:") fmt.Fprintln(buf, "ADDV $0, R0, R0") fmt.Fprintln(buf, "RET") } // 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 := t.TempDir() tmpfile := filepath.Join(dir, "testpcalign.s") tmpout := filepath.Join(dir, "testpcalign.o") code1 := []byte("TEXT ·foo(SB),$0-0\nMOVW $0, R0\nPCALIGN $8\nADDV $8, R0\nRET\n") code2 := []byte("TEXT ·foo(SB),$0-0\nMOVW $0, R0\nPCALIGN $16\nADDV $16, R0\nRET\n") code3 := []byte("TEXT ·foo(SB),$0-0\nMOVW $0, R0\nPCALIGN $32\nADDV $32, R0\nRET\n") out1 := `0x0008\s00008\s\(.*\)\s*ADDV\s\$8,\sR0` out2 := `0x0010\s00016\s\(.*\)\s*ADDV\s\$16,\sR0` out3 := `0x0020\s00032\s\(.*\)\s*ADDV\s\$32,\sR0` var testCases = []struct { name string source []byte want string }{ {"pcalign8", code1, out1}, {"pcalign16", code2, out2}, {"pcalign32", code3, out3}, } for _, test := range testCases { if err := os.WriteFile(tmpfile, test.source, 0644); err != nil { t.Fatal(err) } cmd := testenv.Command(t, testenv.GoToolPath(t), "tool", "asm", "-S", "-o", tmpout, tmpfile) cmd.Env = append(os.Environ(), "GOARCH=loong64", "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.want, string(out)) if err != nil { t.Fatal(err) } if !matched { t.Errorf("The %s testing failed!\ninput: %s\noutput: %s\n", test.name, test.source, out) } } } func TestNoRet(t *testing.T) { dir := t.TempDir() tmpfile := filepath.Join(dir, "testnoret.s") tmpout := filepath.Join(dir, "testnoret.o") if err := os.WriteFile(tmpfile, []byte("TEXT ·foo(SB),$0-0\nNOP\n"), 0644); err != nil { t.Fatal(err) } cmd := testenv.Command(t, testenv.GoToolPath(t), "tool", "asm", "-o", tmpout, tmpfile) cmd.Env = append(os.Environ(), "GOARCH=loong64", "GOOS=linux") if out, err := cmd.CombinedOutput(); err != nil { t.Errorf("%v\n%s", err, out) } } func TestLargeCall(t *testing.T) { if testing.Short() { t.Skip("Skipping test in short mode") } if runtime.GOARCH != "loong64" { t.Skip("Require loong64 to run") } testenv.MustHaveGoBuild(t) dir := t.TempDir() if err := os.WriteFile(filepath.Join(dir, "go.mod"), []byte("module largecall"), 0644); err != nil { t.Fatalf("Failed to write file: %v\n", err) } main := `package main func main() { a() } func a() ` if err := os.WriteFile(filepath.Join(dir, "largecall.go"), []byte(main), 0644); err != nil { t.Fatalf("failed to write main: %v\n", err) } // Generate a very large call instruction. buf := bytes.NewBuffer(make([]byte, 0, 7000000)) genLargeCall(buf) if err := os.WriteFile(filepath.Join(dir, "largecall.s"), buf.Bytes(), 0644); err != nil { t.Fatalf("Failed to write file: %v\n", err) } // Build generated files. cmd := testenv.Command(t, testenv.GoToolPath(t), "build") cmd.Dir = dir out, err := cmd.CombinedOutput() if err != nil { t.Errorf("Build failed: %v, output: %s", err, out) } } func genLargeCall(buf *bytes.Buffer) { fmt.Fprintln(buf, "TEXT main·a(SB),0,$0-8") fmt.Fprintln(buf, "CALL b(SB)") for i := 0; i <= ((1 << 26) + 26); i++ { fmt.Fprintln(buf, "ADDV $0, R0, R0") } fmt.Fprintln(buf, "RET") fmt.Fprintln(buf, "TEXT b(SB),0,$0-8") fmt.Fprintln(buf, "ADDV $0, R0, R0") fmt.Fprintln(buf, "RET") }