|  | // 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...) | 
|  | } |