// run //go:build !nacl && !js && gc && !wasip1 // 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. // Test the compiler -linkobj flag. package main import ( "fmt" "io/ioutil" "log" "os" "os/exec" "strings" ) var pwd, tmpdir string func main() { dir, err := ioutil.TempDir("", "go-test-linkobj-") if err != nil { log.Fatal(err) } pwd, err = os.Getwd() if err != nil { log.Fatal(err) } if err := os.Chdir(dir); err != nil { os.RemoveAll(dir) log.Fatal(err) } tmpdir = dir writeFile("p1.go", ` package p1 func F() { println("hello from p1") } `) writeFile("p2.go", ` package p2 import "./p1" func F() { p1.F() println("hello from p2") } func main() {} `) writeFile("p3.go", ` package main import "./p2" func main() { p2.F() println("hello from main") } `) stdlibimportcfg, err := os.ReadFile(os.Getenv("STDLIB_IMPORTCFG")) if err != nil { fatalf("listing stdlib export files: %v", err) } // two rounds: once using normal objects, again using .a files (compile -pack). for round := 0; round < 2; round++ { pkg := "-pack=" + fmt.Sprint(round) // The compiler expects the files being read to have the right suffix. o := "o" if round == 1 { o = "a" } importcfg := string(stdlibimportcfg) + "\npackagefile p1=p1." + o + "\npackagefile p2=p2." + o os.WriteFile("importcfg", []byte(importcfg), 0644) // inlining is disabled to make sure that the link objects contain needed code. run("go", "tool", "compile", "-p=p1", pkg, "-D", ".", "-importcfg=importcfg", "-l", "-o", "p1."+o, "-linkobj", "p1.lo", "p1.go") run("go", "tool", "compile", "-p=p2", pkg, "-D", ".", "-importcfg=importcfg", "-l", "-o", "p2."+o, "-linkobj", "p2.lo", "p2.go") run("go", "tool", "compile", "-p=main", pkg, "-D", ".", "-importcfg=importcfg", "-l", "-o", "p3."+o, "-linkobj", "p3.lo", "p3.go") cp("p1."+o, "p1.oo") cp("p2."+o, "p2.oo") cp("p3."+o, "p3.oo") cp("p1.lo", "p1."+o) cp("p2.lo", "p2."+o) cp("p3.lo", "p3."+o) out := runFail("go", "tool", "link", "p2."+o) if !strings.Contains(out, "not package main") { fatalf("link p2.o failed but not for package main:\n%s", out) } run("go", "tool", "link", "-importcfg=importcfg", "-o", "a.out.exe", "p3."+o) out = run("./a.out.exe") if !strings.Contains(out, "hello from p1\nhello from p2\nhello from main\n") { fatalf("running main, incorrect output:\n%s", out) } // ensure that mistaken future round can't use these os.Remove("p1.o") os.Remove("a.out.exe") } cleanup() } func run(args ...string) string { out, err := exec.Command(args[0], args[1:]...).CombinedOutput() if err != nil { fatalf("run %v: %s\n%s", args, err, out) } return string(out) } func runFail(args ...string) string { out, err := exec.Command(args[0], args[1:]...).CombinedOutput() if err == nil { fatalf("runFail %v: unexpected success!\n%s", args, err, out) } return string(out) } func cp(src, dst string) { data, err := ioutil.ReadFile(src) if err != nil { fatalf("%v", err) } err = ioutil.WriteFile(dst, data, 0666) if err != nil { fatalf("%v", err) } } func writeFile(name, data string) { err := ioutil.WriteFile(name, []byte(data), 0666) if err != nil { fatalf("%v", err) } } func cleanup() { const debug = false if debug { println("TMPDIR:", tmpdir) return } os.Chdir(pwd) // get out of tmpdir before removing it os.RemoveAll(tmpdir) } func fatalf(format string, args ...interface{}) { cleanup() log.Fatalf(format, args...) }