internal/linker: place files under GARBLE_CACHE

This means we now have a unified cache directory for garble,
which is now documented in the README.

I considered using the same hash-based cache used for pkgCache,
but decided against it since that cache implementation only stores
regular files without any executable bits set.
We could force the cache package to do what we want here,
but I'm leaning against it for now given that single files work OK.
pull/759/head
Daniel Martí 2 years ago
parent 79376a15f9
commit 6dd5c53a91

@ -93,6 +93,10 @@ Note that the first call to `garble build` may be comparatively slow,
as it has to obfuscate each package for the first time. This is akin to clearing as it has to obfuscate each package for the first time. This is akin to clearing
`GOCACHE` with `go clean -cache` and running a `go build` from scratch. `GOCACHE` with `go clean -cache` and running a `go build` from scratch.
Garble also makes use of its own cache to reuse work, akin to Go's `GOCACHE`.
It defaults to a directory under your user's cache directory,
such as `~/.cache/garble`, and can be placed elsewhere by setting `GARBLE_CACHE`.
### Determinism and seeds ### Determinism and seeds
Just like Go, garble builds are deterministic and reproducible in nature. Just like Go, garble builds are deterministic and reproducible in nature.

@ -27,10 +27,8 @@ const (
TinyEnv = "GARBLE_LINK_TINY" TinyEnv = "GARBLE_LINK_TINY"
EntryOffKeyEnv = "GARBLE_LINK_ENTRYOFF_KEY" EntryOffKeyEnv = "GARBLE_LINK_ENTRYOFF_KEY"
cacheDirName = "garble" versionExt = ".version"
versionExt = ".version" baseSrcSubdir = "src"
garbleCacheDir = "GARBLE_CACHE_DIR"
baseSrcSubdir = "src"
) )
//go:embed patches/*/*.patch //go:embed patches/*/*.patch
@ -137,25 +135,14 @@ func applyPatches(srcDir, workingDir string, modFiles map[string]bool, patches [
return mod, nil return mod, nil
} }
// TODO: put linker binaries into fsCache in the main package func cachePath(cacheDir string) (string, error) {
// Use a subdirectory to clarify what we're using it for.
func cachePath() (string, error) { // Name it "tool", like Go's pkg/tool, as we might want to rebuild
var cacheDir string // other Go toolchain programs like the compiler or assembler in the future.
if val, ok := os.LookupEnv(garbleCacheDir); ok { cacheDir = filepath.Join(cacheDir, "tool")
cacheDir = val
} else {
userCacheDir, err := os.UserCacheDir()
if err != nil {
panic(fmt.Errorf("cannot retreive user cache directory: %v", err))
}
cacheDir = userCacheDir
}
cacheDir = filepath.Join(cacheDir, cacheDirName)
if err := os.MkdirAll(cacheDir, 0o777); err != nil { if err := os.MkdirAll(cacheDir, 0o777); err != nil {
return "", err return "", err
} }
goExe := "" goExe := ""
if runtime.GOOS == "windows" { if runtime.GOOS == "windows" {
goExe = ".exe" goExe = ".exe"
@ -226,7 +213,7 @@ func buildLinker(workingDir string, overlay map[string]string, outputLinkPath st
return nil return nil
} }
func PatchLinker(goRoot, goVersion, tempDir string) (string, func(), error) { func PatchLinker(goRoot, goVersion, cacheDir, tempDir string) (string, func(), error) {
// rxVersion looks for a version like "go1.19" or "go1.20" // rxVersion looks for a version like "go1.19" or "go1.20"
rxVersion := regexp.MustCompile(`go\d+\.\d+`) rxVersion := regexp.MustCompile(`go\d+\.\d+`)
majorGoVersion := rxVersion.FindString(goVersion) majorGoVersion := rxVersion.FindString(goVersion)
@ -236,7 +223,7 @@ func PatchLinker(goRoot, goVersion, tempDir string) (string, func(), error) {
panic(fmt.Errorf("cannot retrieve linker patches: %v", err)) panic(fmt.Errorf("cannot retrieve linker patches: %v", err))
} }
outputLinkPath, err := cachePath() outputLinkPath, err := cachePath(cacheDir)
if err != nil { if err != nil {
return "", nil, err return "", nil, err
} }

@ -400,12 +400,15 @@ func mainErr(args []string) error {
if err := os.RemoveAll(os.Getenv("GARBLE_SHARED")); err != nil { if err := os.RemoveAll(os.Getenv("GARBLE_SHARED")); err != nil {
fmt.Fprintf(os.Stderr, "could not clean up GARBLE_SHARED: %v\n", err) fmt.Fprintf(os.Stderr, "could not clean up GARBLE_SHARED: %v\n", err)
} }
fsCache, err := openCache() // skip the trim if we didn't even start a build
if err == nil { if sharedCache != nil {
err = fsCache.Trim() fsCache, err := openCache()
} if err == nil {
if err != nil { err = fsCache.Trim()
fmt.Fprintf(os.Stderr, "could not trim GARBLE_CACHE: %v\n", err) }
if err != nil {
fmt.Fprintf(os.Stderr, "could not trim GARBLE_CACHE: %v\n", err)
}
} }
}() }()
if err != nil { if err != nil {
@ -455,7 +458,7 @@ func mainErr(args []string) error {
executablePath := args[0] executablePath := args[0]
if tool == "link" { if tool == "link" {
modifiedLinkPath, unlock, err := linker.PatchLinker(sharedCache.GoEnv.GOROOT, sharedCache.GoEnv.GOVERSION, sharedTempDir) modifiedLinkPath, unlock, err := linker.PatchLinker(sharedCache.GoEnv.GOROOT, sharedCache.GoEnv.GOVERSION, sharedCache.CacheDir, sharedTempDir)
if err != nil { if err != nil {
return fmt.Errorf("cannot get modified linker: %v", err) return fmt.Errorf("cannot get modified linker: %v", err)
} }
@ -543,6 +546,20 @@ This command wraps "go %s". Below is its help:
return nil, err return nil, err
} }
// Always an absolute directory; defaults to e.g. "~/.cache/garble".
if dir := os.Getenv("GARBLE_CACHE"); dir != "" {
sharedCache.CacheDir, err = filepath.Abs(dir)
if err != nil {
return nil, err
}
} else {
parentDir, err := os.UserCacheDir()
if err != nil {
return nil, err
}
sharedCache.CacheDir = filepath.Join(parentDir, "garble")
}
binaryBuildID, err := buildidOf(sharedCache.ExecPath) binaryBuildID, err := buildidOf(sharedCache.ExecPath)
if err != nil { if err != nil {
return nil, err return nil, err
@ -1304,18 +1321,9 @@ func (c *pkgCache) CopyFrom(c2 pkgCache) {
} }
func openCache() (*cache.Cache, error) { func openCache() (*cache.Cache, error) {
dir := os.Getenv("GARBLE_CACHE") // e.g. "~/.cache/garble"
if dir == "" {
parentDir, err := os.UserCacheDir()
if err != nil {
return nil, err
}
dir = filepath.Join(parentDir, "garble")
}
// Use a subdirectory for the hashed build cache, to clarify what it is, // Use a subdirectory for the hashed build cache, to clarify what it is,
// and to allow us to have other directories or files later on without mixing. // and to allow us to have other directories or files later on without mixing.
dir = filepath.Join(dir, "build") dir := filepath.Join(sharedCache.CacheDir, "build")
if err := os.MkdirAll(dir, 0o777); err != nil { if err := os.MkdirAll(dir, 0o777); err != nil {
return nil, err return nil, err
} }

@ -101,11 +101,9 @@ func TestScript(t *testing.T) {
// coverage. Otherwise, the coverage info might be incomplete. // coverage. Otherwise, the coverage info might be incomplete.
env.Setenv("GOCACHE", filepath.Join(tempCacheDir, "go-cache")) env.Setenv("GOCACHE", filepath.Join(tempCacheDir, "go-cache"))
env.Setenv("GARBLE_CACHE", filepath.Join(tempCacheDir, "garble-cache")) env.Setenv("GARBLE_CACHE", filepath.Join(tempCacheDir, "garble-cache"))
env.Setenv("GARBLE_CACHE_DIR", filepath.Join(tempCacheDir, "garble-cache-2"))
} else { } else {
// GOCACHE is initialized by gotooltest to use the host's cache. // GOCACHE is initialized by gotooltest to use the host's cache.
env.Setenv("GARBLE_CACHE", filepath.Join(hostCacheDir, "garble")) env.Setenv("GARBLE_CACHE", filepath.Join(hostCacheDir, "garble"))
env.Setenv("GARBLE_CACHE_DIR", hostCacheDir)
} }
return nil return nil
}, },

@ -32,6 +32,8 @@ type sharedCacheType struct {
ExecPath string // absolute path to the garble binary being used ExecPath string // absolute path to the garble binary being used
ForwardBuildFlags []string // build flags fed to the original "garble ..." command ForwardBuildFlags []string // build flags fed to the original "garble ..." command
CacheDir string // absolute path to the GARBLE_CACHE directory being used
// ListedPackages contains data obtained via 'go list -json -export -deps'. // ListedPackages contains data obtained via 'go list -json -export -deps'.
// This allows us to obtain the non-obfuscated export data of all dependencies, // This allows us to obtain the non-obfuscated export data of all dependencies,
// useful for type checking of the packages as we obfuscate them. // useful for type checking of the packages as we obfuscate them.

@ -1,7 +1,7 @@
# Past garble versions might not properly patch cmd/link with "git apply" # Past garble versions might not properly patch cmd/link with "git apply"
# when running inside a git repository. Skip the extra check with -short. # when running inside a git repository. Skip the extra check with -short.
[!short] [exec:git] exec git init -q [!short] [exec:git] exec git init -q
[!short] [exec:git] env GARBLE_CACHE_DIR=${WORK}/linker-cache [!short] [exec:git] env GARBLE_CACHE=${WORK}/garble-cache
# Any build settings for the main build shouldn't affect building the linker. # 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, # If this flag makes it through when using build commands on std or cmd,
@ -20,10 +20,10 @@ exec ./main
[!windows] env GOOS=windows [!windows] env GOOS=windows
[windows] env GOOS=linux [windows] env GOOS=linux
exec garble build exec garble build
[!windows] [exec:git] exists ${GARBLE_CACHE_DIR}/garble/link [!windows] [exec:git] exists ${GARBLE_CACHE}/tool/link
[!windows] [exec:git] ! exists ${GARBLE_CACHE_DIR}/garble/link.exe [!windows] [exec:git] ! exists ${GARBLE_CACHE}/tool/link.exe
[windows] [exec:git] ! exists ${GARBLE_CACHE_DIR}/garble/link [windows] [exec:git] ! exists ${GARBLE_CACHE}/tool/link
[windows] [exec:git] exists ${GARBLE_CACHE_DIR}/garble/link.exe [windows] [exec:git] exists ${GARBLE_CACHE}/tool/link.exe
env GOOS= env GOOS=
# Verify a build without garble. # Verify a build without garble.

Loading…
Cancel
Save