centralize logic dealing with main package import paths

obfuscatedImportPath already handled ToObfuscate, so the callers
don't have to do that as well. Handle main packages too, whose logic
was sprinkled and repeated throughout the project.
pull/908/head
Daniel Martí 3 months ago committed by Paul Scheduikat
parent 066771481b
commit 3c742dac65

@ -678,9 +678,7 @@ func (tf *transformer) transformAsm(args []string) ([]string, error) {
flags, paths := splitFlagsFromFiles(args, ".s") flags, paths := splitFlagsFromFiles(args, ".s")
// When assembling, the import path can make its way into the output object file. // When assembling, the import path can make its way into the output object file.
if tf.curPkg.Name != "main" && tf.curPkg.ToObfuscate { flags = flagSetValue(flags, "-p", tf.curPkg.obfuscatedImportPath())
flags = flagSetValue(flags, "-p", tf.curPkg.obfuscatedImportPath())
}
flags = alterTrimpath(flags) flags = alterTrimpath(flags)
@ -692,7 +690,7 @@ func (tf *transformer) transformAsm(args []string) ([]string, error) {
if !slices.Contains(args, "-gensymabis") { if !slices.Contains(args, "-gensymabis") {
for _, path := range paths { for _, path := range paths {
name := hashWithPackage(tf.curPkg, filepath.Base(path)) + ".s" name := hashWithPackage(tf.curPkg, filepath.Base(path)) + ".s"
pkgDir := filepath.Join(sharedTempDir, tf.curPkg.obfuscatedImportPath()) pkgDir := filepath.Join(sharedTempDir, tf.curPkg.obfuscatedSourceDir())
newPath := filepath.Join(pkgDir, name) newPath := filepath.Join(pkgDir, name)
newPaths = append(newPaths, newPath) newPaths = append(newPaths, newPath)
} }
@ -857,9 +855,8 @@ func (tf *transformer) replaceAsmNames(buf *bytes.Buffer, remaining []byte) {
// If the name was qualified, fetch the package, and write the // If the name was qualified, fetch the package, and write the
// obfuscated import path if needed. // obfuscated import path if needed.
// Note that we don't obfuscate the package path "main".
lpkg := tf.curPkg lpkg := tf.curPkg
if asmPkgPath != "" && asmPkgPath != "main" { if asmPkgPath != "" {
if asmPkgPath != tf.curPkg.Name { if asmPkgPath != tf.curPkg.Name {
goPkgPath := asmPkgPath goPkgPath := asmPkgPath
goPkgPath = strings.ReplaceAll(goPkgPath, string(asmPeriod), string(goPeriod)) goPkgPath = strings.ReplaceAll(goPkgPath, string(asmPeriod), string(goPeriod))
@ -930,7 +927,7 @@ func (tf *transformer) writeSourceFile(basename, obfuscated string, content []by
// We use the obfuscated import path to hold the temporary files. // We use the obfuscated import path to hold the temporary files.
// Assembly files do not support line directives to set positions, // Assembly files do not support line directives to set positions,
// so the only way to not leak the import path is to replace it. // so the only way to not leak the import path is to replace it.
pkgDir := filepath.Join(sharedTempDir, tf.curPkg.obfuscatedImportPath()) pkgDir := filepath.Join(sharedTempDir, tf.curPkg.obfuscatedSourceDir())
if err := os.MkdirAll(pkgDir, 0o777); err != nil { if err := os.MkdirAll(pkgDir, 0o777); err != nil {
return "", err return "", err
} }
@ -1062,13 +1059,8 @@ func (tf *transformer) transformCompile(args []string) ([]string, error) {
return nil, err return nil, err
} }
// If this is a package to obfuscate, swap the -p flag with the new package path. // Note that the main package always uses `-p main`, even though it's not an import path.
// We don't if it's the main package, as that just uses "-p main". flags = flagSetValue(flags, "-p", tf.curPkg.obfuscatedImportPath())
// We only set newPkgPath if we're obfuscating the import path,
// to replace the original package name in the package clause below.
if tf.curPkg.Name != "main" && tf.curPkg.ToObfuscate {
flags = flagSetValue(flags, "-p", tf.curPkg.obfuscatedImportPath())
}
newPaths := make([]string, 0, len(files)) newPaths := make([]string, 0, len(files))
@ -1244,11 +1236,7 @@ func (tf *transformer) transformLinkname(localName, newName string) (string, str
newForeignName = hashWithPackage(lpkg, foreignName) newForeignName = hashWithPackage(lpkg, foreignName)
} }
newPkgPath := lpkg.ImportPath newName = lpkg.obfuscatedImportPath() + "." + newForeignName
if newPkgPath != "main" {
newPkgPath = lpkg.obfuscatedImportPath()
}
newName = newPkgPath + "." + newForeignName
return localName, newName return localName, newName
} }
@ -1379,9 +1367,7 @@ func (tf *transformer) processImportCfg(flags []string, requiredPkgs []string) (
} }
return "", err return "", err
} }
if lpkg.Name != "main" { impPath = lpkg.obfuscatedImportPath()
impPath = lpkg.obfuscatedImportPath()
}
fmt.Fprintf(newCfg, "packagefile %s=%s\n", impPath, pkgfile) fmt.Fprintf(newCfg, "packagefile %s=%s\n", impPath, pkgfile)
} }
@ -1643,7 +1629,7 @@ func computeLinkerVariableStrings(pkg *types.Package) (map[*types.Var]string, er
i := strings.LastIndexByte(fullName, '.') i := strings.LastIndexByte(fullName, '.')
path, name := fullName[:i], fullName[i+1:] path, name := fullName[:i], fullName[i+1:]
// -X represents the main package as "main", not its import path. // Note that package main always has import path "main" as part of a build.
if path != pkg.Path() && (path != "main" || pkg.Name() != "main") { if path != pkg.Path() && (path != "main" || pkg.Name() != "main") {
return // not the current package return // not the current package
} }
@ -2091,16 +2077,14 @@ func (tf *transformer) transformGoFile(file *ast.File) *ast.File {
} }
// We're importing an obfuscated package. // We're importing an obfuscated package.
// Replace the import path with its obfuscated version. // Replace the import path with its obfuscated version.
// If the import was unnamed, give it the name of the
// original package name, to keep references working.
lpkg, err := listPackage(tf.curPkg, path) lpkg, err := listPackage(tf.curPkg, path)
if err != nil { if err != nil {
panic(err) // should never happen panic(err) // should never happen
} }
if lpkg.Name != "main" { // Note that a main package is imported via its original import path.
newPath := lpkg.obfuscatedImportPath() imp.Path.Value = strconv.Quote(lpkg.obfuscatedImportPath())
imp.Path.Value = strconv.Quote(newPath) // If the import was unnamed, give it the name of the
} // original package name, to keep references working.
if imp.Name == nil { if imp.Name == nil {
imp.Name = &ast.Ident{ imp.Name = &ast.Ident{
NamePos: imp.Path.ValuePos, // ensure it ends up on the same line NamePos: imp.Path.ValuePos, // ensure it ends up on the same line
@ -2174,8 +2158,7 @@ func (tf *transformer) transformLink(args []string) ([]string, error) {
path, name := fullName[:i], fullName[i+1:] path, name := fullName[:i], fullName[i+1:]
// If the package path is "main", it's the current top-level // If the package path is "main", it's the current top-level
// package we are linking. // package we are linking. Otherwise, find it in the cache.
// Otherwise, find it in the cache.
lpkg := tf.curPkg lpkg := tf.curPkg
if path != "main" { if path != "main" {
lpkg = sharedCache.ListedPackages[path] lpkg = sharedCache.ListedPackages[path]
@ -2186,13 +2169,8 @@ func (tf *transformer) transformLink(args []string) ([]string, error) {
// cmd/link ignores those, so we should too. // cmd/link ignores those, so we should too.
return return
} }
// As before, the main package must remain as "main".
newPath := path
if path != "main" {
newPath = lpkg.obfuscatedImportPath()
}
newName := hashWithPackage(lpkg, name) newName := hashWithPackage(lpkg, name)
flags = append(flags, fmt.Sprintf("-X=%s.%s=%s", newPath, newName, stringValue)) flags = append(flags, fmt.Sprintf("-X=%s.%s=%s", lpkg.obfuscatedImportPath(), newName, stringValue))
}) })
// Starting in Go 1.17, Go's version is implicitly injected by the linker. // Starting in Go 1.17, Go's version is implicitly injected by the linker.

@ -205,8 +205,10 @@ type packageError struct {
Err string Err string
} }
// obfuscatedPackageName returns a package's obfuscated package name,
// which may be unchanged in some cases where we cannot obfuscate it.
// Note that package main is unchanged as it is treated in a special way by the toolchain.
func (p *listedPackage) obfuscatedPackageName() string { func (p *listedPackage) obfuscatedPackageName() string {
// Note that package main is treated in a special way by the toolchain.
if p.Name == "main" || !p.ToObfuscate { if p.Name == "main" || !p.ToObfuscate {
return p.Name return p.Name
} }
@ -214,7 +216,24 @@ func (p *listedPackage) obfuscatedPackageName() string {
return hashWithPackage(p, p.Name) return hashWithPackage(p, p.Name)
} }
// obfuscatedSourceDir returns an obfuscated directory name which can be used
// to write obfuscated source files to. This directory name should be unique per package,
// even when building many main packages at once, such as in `go test ./...`.
func (p *listedPackage) obfuscatedSourceDir() string {
return hashWithPackage(p, p.ImportPath)
}
// obfuscatedImportPath returns a package's obfuscated import path,
// which may be unchanged in some cases where we cannot obfuscate it.
// Note that package main always has the unchanged import path "main" as part of a build,
// but not if it's a main package as part of a test, which can be imported.
func (p *listedPackage) obfuscatedImportPath() string { func (p *listedPackage) obfuscatedImportPath() string {
if p.Name == "main" && p.ForTest == "" {
return "main"
}
if !p.ToObfuscate {
return p.ImportPath
}
// We can't obfuscate these standard library import paths, // We can't obfuscate these standard library import paths,
// as the toolchain expects to recognize the packages by them: // as the toolchain expects to recognize the packages by them:
// //
@ -233,9 +252,6 @@ func (p *listedPackage) obfuscatedImportPath() string {
if _, ok := compilerIntrinsics[p.ImportPath]; ok { if _, ok := compilerIntrinsics[p.ImportPath]; ok {
return p.ImportPath return p.ImportPath
} }
if !p.ToObfuscate {
return p.ImportPath
}
newPath := hashWithPackage(p, p.ImportPath) newPath := hashWithPackage(p, p.ImportPath)
log.Printf("import path %q hashed with %x to %q", p.ImportPath, p.GarbleActionID, newPath) log.Printf("import path %q hashed with %x to %q", p.ImportPath, p.GarbleActionID, newPath)
return newPath return newPath

Loading…
Cancel
Save