|
|
|
# This test stopped working on Go 1.23, as linknaming into the runtime
|
|
|
|
# is now forbidden since https://go.dev/issue/67401.
|
|
|
|
# Perhaps rewrite this test by searching for the pcheader magic number
|
|
|
|
# by scanning the binary, like debug/buildinfo/buildinfo.go does in searchMagic.
|
|
|
|
skip 'no longer works on Go 1.23'
|
|
|
|
|
|
|
|
# Past garble versions might not properly patch cmd/link with "git apply"
|
|
|
|
# when running inside a git repository. Skip the extra check with -short.
|
|
|
|
[!short] [exec:git] exec git init -q
|
|
|
|
[!short] [exec:git] env GARBLE_CACHE=${WORK}/garble-cache
|
use fewer build flags when building std or cmd
When we use `go list` on the standard library, we need to be careful
about what flags are passed from the top-level build command,
because some flags are not going to be appropriate.
In particular, GOFLAGS=-modfile=... resulted in a failure,
reproduced via the GOFLAGS variable added to linker.txtar:
go: inconsistent vendoring in /home/mvdan/tip/src:
golang.org/x/crypto@v0.5.1-0.20230203195927-310bfa40f1e4: is marked as explicit in vendor/modules.txt, but not explicitly required in go.mod
golang.org/x/net@v0.7.0: is marked as explicit in vendor/modules.txt, but not explicitly required in go.mod
golang.org/x/sys@v0.5.1-0.20230208141308-4fee21c92339: is marked as explicit in vendor/modules.txt, but not explicitly required in go.mod
golang.org/x/text@v0.7.1-0.20230207171107-30dadde3188b: is marked as explicit in vendor/modules.txt, but not explicitly required in go.mod
To ignore the vendor directory, use -mod=readonly or -mod=mod.
To sync the vendor directory, run:
go mod vendor
To work around this problem, reset the -mod and -modfile flags when
calling "go list" on the standard library, as those are the only two
flags which alter how we load the main module in a build.
The code which builds a modified cmd/link has a similar problem;
it already reset GOOS and GOARCH, but it could similarly run into
problems if other env vars like GOFLAGS were set.
To be on the safe side, we also disable GOENV and GOEXPERIMENT,
which we borrow from Go's bootstrapping commands.
2 years ago
|
|
|
|
|
|
|
# Any build settings for the main build shouldn't affect building the linker.
|
|
|
|
# If this flag makes it through when using build commands on std or cmd,
|
|
|
|
# those commands are likely to fail as std and cmd are their own modules.
|
|
|
|
env GOFLAGS=-modfile=${WORK}/go.mod
|
|
|
|
|
|
|
|
exec garble build
|
patch and rebuild cmd/link to modify the magic value in pclntab
This value is hard-coded in the linker and written in a header.
We could rewrite the final binary, like we used to do with import paths,
but that would require once again maintaining libraries to do so.
Instead, we're now modifying the linker to do what we want.
It's not particularly hard, as every Go install has its source code,
and rebuilding a slightly modified linker only takes a few seconds at most.
Thanks to `go build -overlay`, we only need to copy the files we modify,
and right now we're just modifying one file in the toolchain.
We use a git patch, as the change is fairly static and small,
and the patch is easier to understand and maintain.
The other side of this change is in the runtime,
as it also hard-codes the magic value when loading information.
We modify the code via syntax trees in that case, like `-tiny` does,
because the change is tiny (one literal) and the affected lines of code
are modified regularly between major Go releases.
Since rebuilding a slightly modified linker can take a few seconds,
and Go's build cache does not cache linked binaries,
we keep our own cached version of the rebuilt binary in `os.UserCacheDir`.
The feature isn't perfect, and will be improved in the future.
See the TODOs about the added dependency on `git`,
or how we are currently only able to cache one linker binary at once.
Fixes #622.
2 years ago
|
|
|
exec ./main
|
|
|
|
! cmp stderr main.stderr
|
|
|
|
|
|
|
|
[short] stop # no need to verify this with -short
|
|
|
|
|
|
|
|
# The rebuilt linker should use the executable extension for the host GOOS,
|
|
|
|
# not the target one. Not doing so might be harmless, but can result in
|
|
|
|
# building the linker twice, wasting CPU and disk.
|
|
|
|
[!windows] env GOOS=windows
|
|
|
|
[windows] env GOOS=linux
|
|
|
|
exec garble build
|
|
|
|
[!windows] [exec:git] exists ${GARBLE_CACHE}/tool/link
|
|
|
|
[!windows] [exec:git] ! exists ${GARBLE_CACHE}/tool/link.exe
|
|
|
|
[windows] [exec:git] ! exists ${GARBLE_CACHE}/tool/link
|
|
|
|
[windows] [exec:git] exists ${GARBLE_CACHE}/tool/link.exe
|
|
|
|
env GOOS=
|
|
|
|
|
|
|
|
# Verify a build without garble.
|
patch and rebuild cmd/link to modify the magic value in pclntab
This value is hard-coded in the linker and written in a header.
We could rewrite the final binary, like we used to do with import paths,
but that would require once again maintaining libraries to do so.
Instead, we're now modifying the linker to do what we want.
It's not particularly hard, as every Go install has its source code,
and rebuilding a slightly modified linker only takes a few seconds at most.
Thanks to `go build -overlay`, we only need to copy the files we modify,
and right now we're just modifying one file in the toolchain.
We use a git patch, as the change is fairly static and small,
and the patch is easier to understand and maintain.
The other side of this change is in the runtime,
as it also hard-codes the magic value when loading information.
We modify the code via syntax trees in that case, like `-tiny` does,
because the change is tiny (one literal) and the affected lines of code
are modified regularly between major Go releases.
Since rebuilding a slightly modified linker can take a few seconds,
and Go's build cache does not cache linked binaries,
we keep our own cached version of the rebuilt binary in `os.UserCacheDir`.
The feature isn't perfect, and will be improved in the future.
See the TODOs about the added dependency on `git`,
or how we are currently only able to cache one linker binary at once.
Fixes #622.
2 years ago
|
|
|
go build
|
|
|
|
exec ./main
|
|
|
|
cmp stderr main.stderr
|
|
|
|
|
patch and rebuild cmd/link to modify the magic value in pclntab
This value is hard-coded in the linker and written in a header.
We could rewrite the final binary, like we used to do with import paths,
but that would require once again maintaining libraries to do so.
Instead, we're now modifying the linker to do what we want.
It's not particularly hard, as every Go install has its source code,
and rebuilding a slightly modified linker only takes a few seconds at most.
Thanks to `go build -overlay`, we only need to copy the files we modify,
and right now we're just modifying one file in the toolchain.
We use a git patch, as the change is fairly static and small,
and the patch is easier to understand and maintain.
The other side of this change is in the runtime,
as it also hard-codes the magic value when loading information.
We modify the code via syntax trees in that case, like `-tiny` does,
because the change is tiny (one literal) and the affected lines of code
are modified regularly between major Go releases.
Since rebuilding a slightly modified linker can take a few seconds,
and Go's build cache does not cache linked binaries,
we keep our own cached version of the rebuilt binary in `os.UserCacheDir`.
The feature isn't perfect, and will be improved in the future.
See the TODOs about the added dependency on `git`,
or how we are currently only able to cache one linker binary at once.
Fixes #622.
2 years ago
|
|
|
-- go.mod --
|
|
|
|
module test/main
|
|
|
|
|
|
|
|
go 1.22
|
patch and rebuild cmd/link to modify the magic value in pclntab
This value is hard-coded in the linker and written in a header.
We could rewrite the final binary, like we used to do with import paths,
but that would require once again maintaining libraries to do so.
Instead, we're now modifying the linker to do what we want.
It's not particularly hard, as every Go install has its source code,
and rebuilding a slightly modified linker only takes a few seconds at most.
Thanks to `go build -overlay`, we only need to copy the files we modify,
and right now we're just modifying one file in the toolchain.
We use a git patch, as the change is fairly static and small,
and the patch is easier to understand and maintain.
The other side of this change is in the runtime,
as it also hard-codes the magic value when loading information.
We modify the code via syntax trees in that case, like `-tiny` does,
because the change is tiny (one literal) and the affected lines of code
are modified regularly between major Go releases.
Since rebuilding a slightly modified linker can take a few seconds,
and Go's build cache does not cache linked binaries,
we keep our own cached version of the rebuilt binary in `os.UserCacheDir`.
The feature isn't perfect, and will be improved in the future.
See the TODOs about the added dependency on `git`,
or how we are currently only able to cache one linker binary at once.
Fixes #622.
2 years ago
|
|
|
-- main.go --
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
_ "unsafe"
|
|
|
|
)
|
|
|
|
|
|
|
|
type fakeModuleData struct {
|
|
|
|
pcHeader *struct {
|
|
|
|
magic uint32
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//go:linkname activeModules runtime.activeModules
|
|
|
|
func activeModules() []*fakeModuleData
|
|
|
|
|
|
|
|
// genericMagicValue returns magic value without last digit
|
|
|
|
func genericMagicValue() string {
|
|
|
|
mod := activeModules()[0]
|
|
|
|
magicValHex := strings.ToUpper(strconv.FormatUint(uint64(mod.pcHeader.magic), 16))
|
|
|
|
return "0x" + magicValHex[:len(magicValHex)-1] + "?"
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
println(genericMagicValue())
|
|
|
|
}
|
|
|
|
|
|
|
|
-- main.stderr --
|
|
|
|
0xFFFFFFF?
|