2016-08-25 11:07:33 -05:00
|
|
|
// 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"
|
2021-04-15 23:05:49 -04:00
|
|
|
"internal/buildcfg"
|
2016-10-17 13:40:18 -04:00
|
|
|
"internal/testenv"
|
2022-08-28 03:38:00 +08:00
|
|
|
"os"
|
2016-08-25 11:07:33 -05:00
|
|
|
"testing"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func TestLargeText(t *testing.T) {
|
2021-04-15 23:05:49 -04:00
|
|
|
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)
|
2016-10-17 13:40:18 -04:00
|
|
|
}
|
|
|
|
|
testenv.MustHaveGoBuild(t)
|
2016-08-25 11:07:33 -05:00
|
|
|
|
|
|
|
|
var w bytes.Buffer
|
|
|
|
|
const FN = 4
|
2021-03-07 20:52:39 -08:00
|
|
|
tmpdir := t.TempDir()
|
2016-08-25 11:07:33 -05:00
|
|
|
|
2022-08-28 03:38:00 +08:00
|
|
|
if err := os.WriteFile(tmpdir+"/go.mod", []byte("module big_test\n"), 0666); err != nil {
|
2021-04-09 10:41:27 -05:00
|
|
|
t.Fatal(err)
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-25 11:07:33 -05:00
|
|
|
// Generate the scenario where the total amount of text exceeds the
|
2016-10-17 13:40:18 -04:00
|
|
|
// limit for the jmp/call instruction, on RISC architectures like ppc64le,
|
2016-08-25 11:07:33 -05:00
|
|
|
// 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.
|
2016-10-17 13:40:18 -04:00
|
|
|
instOnArch := map[string]string{
|
|
|
|
|
"ppc64": "\tMOVD\tR0,R3\n",
|
|
|
|
|
"ppc64le": "\tMOVD\tR0,R3\n",
|
|
|
|
|
"arm": "\tMOVW\tR0,R1\n",
|
|
|
|
|
}
|
2021-04-15 23:05:49 -04:00
|
|
|
inst := instOnArch[buildcfg.GOARCH]
|
2016-08-25 11:07:33 -05:00
|
|
|
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++ {
|
2024-09-04 13:14:17 -04:00
|
|
|
w.WriteString(inst)
|
2016-08-25 11:07:33 -05:00
|
|
|
}
|
|
|
|
|
fmt.Fprintf(&w, "\tRET\n")
|
2022-08-28 03:38:00 +08:00
|
|
|
err := os.WriteFile(tmpdir+"/"+testname+".s", w.Bytes(), 0666)
|
2016-08-25 11:07:33 -05:00
|
|
|
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, "}")
|
2022-08-28 03:38:00 +08:00
|
|
|
err := os.WriteFile(tmpdir+"/bigfn.go", w.Bytes(), 0666)
|
2016-08-25 11:07:33 -05:00
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("can't write output: %v\n", err)
|
|
|
|
|
}
|
|
|
|
|
|
2016-10-11 09:26:40 -05:00
|
|
|
// Build and run with internal linking.
|
2025-11-17 11:54:48 -08:00
|
|
|
cmd := goCmd(t, "build", "-o", "bigtext")
|
2021-04-09 10:41:27 -05:00
|
|
|
cmd.Dir = tmpdir
|
2016-08-25 11:07:33 -05:00
|
|
|
out, err := cmd.CombinedOutput()
|
|
|
|
|
if err != nil {
|
2016-10-11 09:26:40 -05:00
|
|
|
t.Fatalf("Build failed for big text program with internal linking: %v, output: %s", err, out)
|
|
|
|
|
}
|
2022-11-15 10:21:21 -05:00
|
|
|
cmd = testenv.Command(t, "./bigtext")
|
2021-04-09 10:41:27 -05:00
|
|
|
cmd.Dir = tmpdir
|
2016-10-11 09:26:40 -05:00
|
|
|
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
|
2025-11-17 11:54:48 -08:00
|
|
|
cmd = goCmd(t, "build", "-o", "bigtext", "-ldflags", "-linkmode=external")
|
2021-04-09 10:41:27 -05:00
|
|
|
cmd.Dir = tmpdir
|
2016-10-11 09:26:40 -05:00
|
|
|
out, err = cmd.CombinedOutput()
|
|
|
|
|
if err != nil {
|
|
|
|
|
t.Fatalf("Build failed for big text program with external linking: %v, output: %s", err, out)
|
2016-08-25 11:07:33 -05:00
|
|
|
}
|
2022-11-15 10:21:21 -05:00
|
|
|
cmd = testenv.Command(t, "./bigtext")
|
2021-04-09 10:41:27 -05:00
|
|
|
cmd.Dir = tmpdir
|
2016-08-25 11:07:33 -05:00
|
|
|
out, err = cmd.CombinedOutput()
|
|
|
|
|
if err != nil {
|
2016-10-11 09:26:40 -05:00
|
|
|
t.Fatalf("Program built with external linking failed to run with err %v, output: %s", err, out)
|
2016-08-25 11:07:33 -05:00
|
|
|
}
|
|
|
|
|
}
|