slightly improve code thanks to Go 1.18 APIs

strings.Cut makes some string handling code more intuitive.
Note that we can't use it everywhere, as some places need LastIndexByte.

Start using x/exp/slices, too, which is our first use of generics.
Note that its API is experimental and may still change,
but since we are not a library, we can control its version updates.

I also noticed that we were using TrimSpace for importcfg files.
It's actually unnecessary if we swap strings.SplitAfter for Split,
as the only whitespace present was the trailing newline.

While here, I noticed an unused copy of printfWithoutPackage.
pull/505/head
Daniel Martí 3 years ago committed by lu4p
parent 237e0b7b7c
commit 1c564ef091

@ -6,6 +6,7 @@ require (
github.com/frankban/quicktest v1.14.2 github.com/frankban/quicktest v1.14.2
github.com/google/go-cmp v0.5.7 github.com/google/go-cmp v0.5.7
github.com/rogpeppe/go-internal v1.8.1 github.com/rogpeppe/go-internal v1.8.1
golang.org/x/exp v0.0.0-20220325121720-054d8573a5d8
golang.org/x/mod v0.6.0-dev.0.20220106191415-9b9b3d81d5e3 golang.org/x/mod v0.6.0-dev.0.20220106191415-9b9b3d81d5e3
golang.org/x/tools v0.1.10 golang.org/x/tools v0.1.10
) )

@ -15,6 +15,8 @@ github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsK
github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc=
github.com/rogpeppe/go-internal v1.8.1 h1:geMPLpDpQOgVyCg5z5GoRwLHepNdb71NXb67XFkP+Eg= github.com/rogpeppe/go-internal v1.8.1 h1:geMPLpDpQOgVyCg5z5GoRwLHepNdb71NXb67XFkP+Eg=
github.com/rogpeppe/go-internal v1.8.1/go.mod h1:JeRgkft04UBgHMgCIwADu4Pn6Mtm5d4nPKWu0nJ5d+o= github.com/rogpeppe/go-internal v1.8.1/go.mod h1:JeRgkft04UBgHMgCIwADu4Pn6Mtm5d4nPKWu0nJ5d+o=
golang.org/x/exp v0.0.0-20220325121720-054d8573a5d8 h1:Xt4/LzbTwfocTk9ZLEu4onjeFucl88iW+v4j4PWbQuE=
golang.org/x/exp v0.0.0-20220325121720-054d8573a5d8/go.mod h1:lgLbSvA5ygNOMpwM/9anMpWVlVJ7Z+cHWq/eFuinpGE=
golang.org/x/mod v0.6.0-dev.0.20220106191415-9b9b3d81d5e3 h1:kQgndtyPBW/JIYERgdxfwMYh3AVStj88WQTlNDi2a+o= golang.org/x/mod v0.6.0-dev.0.20220106191415-9b9b3d81d5e3 h1:kQgndtyPBW/JIYERgdxfwMYh3AVStj88WQTlNDi2a+o=
golang.org/x/mod v0.6.0-dev.0.20220106191415-9b9b3d81d5e3/go.mod h1:3p9vT2HGsQu2K1YbXdKPJLVgG5VJdoTa1poYQBtP1AY= golang.org/x/mod v0.6.0-dev.0.20220106191415-9b9b3d81d5e3/go.mod h1:3p9vT2HGsQu2K1YbXdKPJLVgG5VJdoTa1poYQBtP1AY=
golang.org/x/sys v0.0.0-20211019181941-9d821ace8654 h1:id054HUawV2/6IGm2IV8KZQjqtwAOo2CYlOToYqa0d0= golang.org/x/sys v0.0.0-20211019181941-9d821ace8654 h1:id054HUawV2/6IGm2IV8KZQjqtwAOo2CYlOToYqa0d0=

@ -34,6 +34,7 @@ import (
"unicode" "unicode"
"unicode/utf8" "unicode/utf8"
"golang.org/x/exp/slices"
"golang.org/x/mod/modfile" "golang.org/x/mod/modfile"
"golang.org/x/mod/semver" "golang.org/x/mod/semver"
"golang.org/x/tools/go/ast/astutil" "golang.org/x/tools/go/ast/astutil"
@ -549,16 +550,8 @@ func transformAsm(args []string) ([]string, error) {
// If the assembler is running just for -gensymabis, // If the assembler is running just for -gensymabis,
// don't obfuscate the source, as we are not assembling yet. // don't obfuscate the source, as we are not assembling yet.
// The assembler will run again later; obfuscating twice is just wasteful. // The assembler will run again later; obfuscating twice is just wasteful.
symabis := false
for _, arg := range args {
if arg == "-gensymabis" {
symabis = true
break
}
}
newPaths := make([]string, 0, len(paths)) newPaths := make([]string, 0, len(paths))
if !symabis { if !slices.Contains(args, "-gensymabis") {
var newPaths []string
for _, path := range paths { for _, path := range paths {
name := filepath.Base(path) name := filepath.Base(path)
pkgDir := filepath.Join(sharedTempDir, filepath.FromSlash(curPkg.ImportPath)) pkgDir := filepath.Join(sharedTempDir, filepath.FromSlash(curPkg.ImportPath))
@ -578,7 +571,6 @@ func transformAsm(args []string) ([]string, error) {
middleDotLen := utf8.RuneLen(middleDot) middleDotLen := utf8.RuneLen(middleDot)
for _, path := range paths { for _, path := range paths {
// Read the entire file into memory. // Read the entire file into memory.
// If we find issues with large files, we can use bufio. // If we find issues with large files, we can use bufio.
content, err := os.ReadFile(path) content, err := os.ReadFile(path)
@ -899,33 +891,26 @@ func processImportCfg(flags []string) (newImportCfg string, _ error) {
var packagefiles, importmaps [][2]string var packagefiles, importmaps [][2]string
for _, line := range strings.SplitAfter(string(data), "\n") { for _, line := range strings.Split(string(data), "\n") {
line = strings.TrimSpace(line)
if line == "" || strings.HasPrefix(line, "#") { if line == "" || strings.HasPrefix(line, "#") {
continue continue
} }
i := strings.IndexByte(line, ' ') verb, args, found := strings.Cut(line, " ")
if i < 0 { if !found {
continue continue
} }
verb := line[:i]
switch verb { switch verb {
case "importmap": case "importmap":
args := strings.TrimSpace(line[i+1:]) beforePath, afterPath, found := strings.Cut(args, "=")
j := strings.IndexByte(args, '=') if !found {
if j < 0 {
continue continue
} }
beforePath, afterPath := args[:j], args[j+1:]
importmaps = append(importmaps, [2]string{beforePath, afterPath}) importmaps = append(importmaps, [2]string{beforePath, afterPath})
case "packagefile": case "packagefile":
args := strings.TrimSpace(line[i+1:]) importPath, objectPath, found := strings.Cut(args, "=")
j := strings.IndexByte(args, '=') if !found {
if j < 0 {
continue continue
} }
importPath, objectPath := args[:j], args[j+1:]
packagefiles = append(packagefiles, [2]string{importPath, objectPath}) packagefiles = append(packagefiles, [2]string{importPath, objectPath})
} }
} }
@ -1187,16 +1172,15 @@ func (tf *transformer) prefillObjectMaps(files []*ast.File) error {
return err return err
} }
flagValueIter(ldflags, "-X", func(val string) { flagValueIter(ldflags, "-X", func(val string) {
// val is in the form of "importpath.name=value". // val is in the form of "foo.com/bar.name=value".
i := strings.IndexByte(val, '=') fullName, stringValue, found := strings.Cut(val, "=")
if i < 0 { if !found {
return // invalid return // invalid
} }
stringValue := val[i+1:]
val = val[:i] // "importpath.name" // fullName is "foo.com/bar.name"
i = strings.LastIndexByte(val, '.') i := strings.LastIndexByte(fullName, '.')
path, name := val[:i], val[i+1:] path, name := fullName[:i], fullName[i+1:]
// -X represents the main package as "main", not its import path. // -X represents the main package as "main", not its import path.
if path != curPkg.ImportPath && !(path == "main" && curPkg.Name == "main") { if path != curPkg.ImportPath && !(path == "main" && curPkg.Name == "main") {
@ -1793,26 +1777,22 @@ func transformLink(args []string) ([]string, error) {
// To cover both obfuscated and non-obfuscated names, // To cover both obfuscated and non-obfuscated names,
// duplicate each flag with a obfuscated version. // duplicate each flag with a obfuscated version.
flagValueIter(flags, "-X", func(val string) { flagValueIter(flags, "-X", func(val string) {
// val is in the form of "pkg.name=str" // val is in the form of "foo.com/bar.name=value".
i := strings.IndexByte(val, '=') fullName, stringValue, found := strings.Cut(val, "=")
if i <= 0 { if !found {
return return // invalid
}
name := val[:i]
str := val[i+1:]
j := strings.LastIndexByte(name, '.')
if j <= 0 {
return
} }
pkg := name[:j]
name = name[j+1:] // fullName is "foo.com/bar.name"
i := strings.LastIndexByte(fullName, '.')
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 := curPkg lpkg := curPkg
if pkg != "main" { if path != "main" {
lpkg = cache.ListedPackages[pkg] lpkg = cache.ListedPackages[path]
} }
if lpkg == nil { if lpkg == nil {
// We couldn't find the package. // We couldn't find the package.
@ -1821,12 +1801,12 @@ func transformLink(args []string) ([]string, error) {
return return
} }
// As before, the main package must remain as "main". // As before, the main package must remain as "main".
newPkg := pkg newPath := path
if pkg != "main" { if path != "main" {
newPkg = lpkg.obfuscatedImportPath() newPath = lpkg.obfuscatedImportPath()
} }
newName := hashWithPackage(lpkg, name) newName := hashWithPackage(lpkg, name)
flags = append(flags, fmt.Sprintf("-X=%s.%s=%s", newPkg, newName, str)) flags = append(flags, fmt.Sprintf("-X=%s.%s=%s", newPath, 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.
@ -1938,10 +1918,7 @@ func filterForwardBuildFlags(flags []string) (filtered []string, firstUnknown st
arg = arg[1:] // "--name" to "-name"; keep the short form arg = arg[1:] // "--name" to "-name"; keep the short form
} }
name := arg name, _, _ := strings.Cut(arg, "=") // "-name=value" to "-name"
if i := strings.IndexByte(arg, '='); i > 0 {
name = arg[:i] // "-name=value" to "-name"
}
buildFlag := forwardBuildFlags[name] buildFlag := forwardBuildFlags[name]
if buildFlag { if buildFlag {

@ -10,8 +10,9 @@ import (
"go/parser" "go/parser"
"go/printer" "go/printer"
"path/filepath" "path/filepath"
"sort"
"strings" "strings"
"golang.org/x/exp/slices"
) )
func isDirective(text string) bool { func isDirective(text string) bool {
@ -120,8 +121,8 @@ func printFile(file1 *ast.File) ([]byte, error) {
}) })
// We add comments in order. // We add comments in order.
sort.Slice(toAdd, func(i, j int) bool { slices.SortFunc(toAdd, func(a, b commentToAdd) bool {
return toAdd[i].offset < toAdd[j].offset return a.offset < b.offset
}) })
copied := 0 copied := 0

@ -162,11 +162,11 @@ One can reverse a captured panic stack trace as follows:
return err return err
} }
defer f.Close() defer f.Close()
any, err := reverseContent(os.Stdout, f, repl) modified, err := reverseContent(os.Stdout, f, repl)
if err != nil { if err != nil {
return err return err
} }
anyModified = anyModified || any anyModified = anyModified || modified
f.Close() // since we're in a loop f.Close() // since we're in a loop
} }
if !anyModified { if !anyModified {

@ -73,7 +73,6 @@ package main
import ( import (
"fmt" "fmt"
"strings"
"test/main/importedpkg" "test/main/importedpkg"
@ -92,10 +91,6 @@ func main() {
fmt.Println(quote.Go()) fmt.Println(quote.Go())
garbletest.Test() garbletest.Test()
} }
func printfWithoutPackage(format string, v any) {
fmt.Print(strings.Split(fmt.Sprintf(format, v), ".")[1])
}
-- notag_fail.go -- -- notag_fail.go --
// +build !buildtag // +build !buildtag

@ -113,8 +113,8 @@ type EmbeddingIndirect struct {
func printfWithoutPackage(format string, v any) { func printfWithoutPackage(format string, v any) {
s := fmt.Sprintf(format, v) s := fmt.Sprintf(format, v)
if i := strings.IndexByte(s, '.'); i > 0 { if _, without, found := strings.Cut(s, "."); found {
s = s[i+1:] s = without
} }
fmt.Print(s) fmt.Print(s)
} }

Loading…
Cancel
Save