From 7f80dfb59d2d5932a5d3c081bc57284f87fc2b79 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Daniel=20Mart=C3=AD?= Date: Sun, 30 Mar 2025 00:57:31 +0000 Subject: [PATCH] rebuild cmd/link with the correct toolchain under GOTOOLCHAIN=auto When we build the patched cmd/link binary for use by garble, we perform this build in a temporary directory so that the Go module from the user does not get in the way. When the user module made us upgrade the toolchain per GOTOOLCHAIN, leaving that module's directory stops upgrading the toolchain, so we patch a newer toolchain and build it with an older toolchain. This is largely harmless, but it makes the newer toolchain think it is actually an older toolchain, which leads to those pesky "linker object header mismatch" version errors. Updates #934. --- internal/linker/linker.go | 14 +++++++------- testdata/script/gotoolchain.txtar | 2 +- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/internal/linker/linker.go b/internal/linker/linker.go index e043433..0a51ae6 100644 --- a/internal/linker/linker.go +++ b/internal/linker/linker.go @@ -27,9 +27,6 @@ const ( MagicValueEnv = "GARBLE_LINK_MAGIC" TinyEnv = "GARBLE_LINK_TINY" EntryOffKeyEnv = "GARBLE_LINK_ENTRYOFF_KEY" - - versionExt = ".version" - baseSrcSubdir = "src" ) //go:embed patches/*/*.patch @@ -163,6 +160,8 @@ func getCurrentVersion(goVersion, patchesVer string) string { return goVersion + " " + patchesVer + "\n" } +const versionExt = ".version" + func checkVersion(linkerPath, goVersion, patchesVer string) (bool, error) { versionPath := linkerPath + versionExt version, err := os.ReadFile(versionPath) @@ -181,7 +180,7 @@ func writeVersion(linkerPath, goVersion, patchesVer string) error { return os.WriteFile(versionPath, []byte(getCurrentVersion(goVersion, patchesVer)), 0o777) } -func buildLinker(workingDir string, overlay map[string]string, outputLinkPath string) error { +func buildLinker(goRoot, workingDir string, overlay map[string]string, outputLinkPath string) error { file, err := json.Marshal(&struct{ Replace map[string]string }{overlay}) if err != nil { return err @@ -191,7 +190,8 @@ func buildLinker(workingDir string, overlay map[string]string, outputLinkPath st return err } - cmd := exec.Command("go", "build", "-overlay", overlayPath, "-o", outputLinkPath, "cmd/link") + goCmd := filepath.Join(goRoot, "bin", "go") + cmd := exec.Command(goCmd, "build", "-overlay", overlayPath, "-o", outputLinkPath, "cmd/link") // Ignore any build settings from the environment or GOENV. // We want to build cmd/link like the rest of the toolchain, // regardless of what build options are set for the current build. @@ -250,14 +250,14 @@ func PatchLinker(goRoot, goVersion, cacheDir, tempDir string) (string, func(), e return outputLinkPath, unlock, nil } - srcDir := filepath.Join(goRoot, baseSrcSubdir) + srcDir := filepath.Join(goRoot, "src") workingDir := filepath.Join(tempDir, "linker-src") overlay, err := applyPatches(srcDir, workingDir, modFiles, patches) if err != nil { return "", nil, err } - if err := buildLinker(workingDir, overlay, outputLinkPath); err != nil { + if err := buildLinker(goRoot, workingDir, overlay, outputLinkPath); err != nil { return "", nil, err } if err := writeVersion(outputLinkPath, goVersion, patchesVer); err != nil { diff --git a/testdata/script/gotoolchain.txtar b/testdata/script/gotoolchain.txtar index e4dee89..f62bab3 100644 --- a/testdata/script/gotoolchain.txtar +++ b/testdata/script/gotoolchain.txtar @@ -21,7 +21,7 @@ stderr 'hello from go1\.24\.1' # TODO: fix this. ! exec garble run . -stderr 'linked object header mismatch' +stderr 'gcMarkTermination: relocation target .* not defined' -- mod/go.mod -- module test