|  | // 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. | 
|  |  | 
|  | // This program generates a test to verify that a program can be | 
|  | // successfully linked even when there are very large text | 
|  | // sections present. | 
|  |  | 
|  | package main | 
|  |  | 
|  | import ( | 
|  | "bytes" | 
|  | "fmt" | 
|  | "internal/buildcfg" | 
|  | "internal/testenv" | 
|  | "os" | 
|  | "testing" | 
|  | ) | 
|  |  | 
|  | func TestLargeText(t *testing.T) { | 
|  | if testing.Short() || (buildcfg.GOARCH != "ppc64le" && buildcfg.GOARCH != "ppc64" && buildcfg.GOARCH != "arm") { | 
|  | t.Skipf("Skipping large text section test in short mode or on %s", buildcfg.GOARCH) | 
|  | } | 
|  | testenv.MustHaveGoBuild(t) | 
|  |  | 
|  | var w bytes.Buffer | 
|  | const FN = 4 | 
|  | tmpdir := t.TempDir() | 
|  |  | 
|  | if err := os.WriteFile(tmpdir+"/go.mod", []byte("module big_test\n"), 0666); err != nil { | 
|  | t.Fatal(err) | 
|  | } | 
|  |  | 
|  | // Generate the scenario where the total amount of text exceeds the | 
|  | // limit for the jmp/call instruction, on RISC architectures like ppc64le, | 
|  | // which is 2^26.  When that happens the call requires special trampolines or | 
|  | // long branches inserted by the linker where supported. | 
|  | // Multiple .s files are generated instead of one. | 
|  | instOnArch := map[string]string{ | 
|  | "ppc64":   "\tMOVD\tR0,R3\n", | 
|  | "ppc64le": "\tMOVD\tR0,R3\n", | 
|  | "arm":     "\tMOVW\tR0,R1\n", | 
|  | } | 
|  | inst := instOnArch[buildcfg.GOARCH] | 
|  | for j := 0; j < FN; j++ { | 
|  | testname := fmt.Sprintf("bigfn%d", j) | 
|  | fmt.Fprintf(&w, "TEXT ยท%s(SB),$0\n", testname) | 
|  | for i := 0; i < 2200000; i++ { | 
|  | fmt.Fprintf(&w, inst) | 
|  | } | 
|  | fmt.Fprintf(&w, "\tRET\n") | 
|  | err := os.WriteFile(tmpdir+"/"+testname+".s", w.Bytes(), 0666) | 
|  | if err != nil { | 
|  | t.Fatalf("can't write output: %v\n", err) | 
|  | } | 
|  | w.Reset() | 
|  | } | 
|  | fmt.Fprintf(&w, "package main\n") | 
|  | fmt.Fprintf(&w, "\nimport (\n") | 
|  | fmt.Fprintf(&w, "\t\"os\"\n") | 
|  | fmt.Fprintf(&w, "\t\"fmt\"\n") | 
|  | fmt.Fprintf(&w, ")\n\n") | 
|  |  | 
|  | for i := 0; i < FN; i++ { | 
|  | fmt.Fprintf(&w, "func bigfn%d()\n", i) | 
|  | } | 
|  | fmt.Fprintf(&w, "\nfunc main() {\n") | 
|  |  | 
|  | // There are lots of dummy code generated in the .s files just to generate a lot | 
|  | // of text. Link them in but guard their call so their code is not executed but | 
|  | // the main part of the program can be run. | 
|  | fmt.Fprintf(&w, "\tif os.Getenv(\"LINKTESTARG\") != \"\" {\n") | 
|  | for i := 0; i < FN; i++ { | 
|  | fmt.Fprintf(&w, "\t\tbigfn%d()\n", i) | 
|  | } | 
|  | fmt.Fprintf(&w, "\t}\n") | 
|  | fmt.Fprintf(&w, "\tfmt.Printf(\"PASS\\n\")\n") | 
|  | fmt.Fprintf(&w, "}") | 
|  | err := os.WriteFile(tmpdir+"/bigfn.go", w.Bytes(), 0666) | 
|  | if err != nil { | 
|  | t.Fatalf("can't write output: %v\n", err) | 
|  | } | 
|  |  | 
|  | // Build and run with internal linking. | 
|  | cmd := testenv.Command(t, testenv.GoToolPath(t), "build", "-o", "bigtext") | 
|  | cmd.Dir = tmpdir | 
|  | out, err := cmd.CombinedOutput() | 
|  | if err != nil { | 
|  | t.Fatalf("Build failed for big text program with internal linking: %v, output: %s", err, out) | 
|  | } | 
|  | cmd = testenv.Command(t, "./bigtext") | 
|  | cmd.Dir = tmpdir | 
|  | out, err = cmd.CombinedOutput() | 
|  | if err != nil { | 
|  | t.Fatalf("Program built with internal linking failed to run with err %v, output: %s", err, out) | 
|  | } | 
|  |  | 
|  | // Build and run with external linking | 
|  | cmd = testenv.Command(t, testenv.GoToolPath(t), "build", "-o", "bigtext", "-ldflags", "-linkmode=external") | 
|  | cmd.Dir = tmpdir | 
|  | out, err = cmd.CombinedOutput() | 
|  | if err != nil { | 
|  | t.Fatalf("Build failed for big text program with external linking: %v, output: %s", err, out) | 
|  | } | 
|  | cmd = testenv.Command(t, "./bigtext") | 
|  | cmd.Dir = tmpdir | 
|  | out, err = cmd.CombinedOutput() | 
|  | if err != nil { | 
|  | t.Fatalf("Program built with external linking failed to run with err %v, output: %s", err, out) | 
|  | } | 
|  | } |