@ -18,6 +18,7 @@ import (
"time"
"time"
"golang.org/x/mod/module"
"golang.org/x/mod/module"
"golang.org/x/mod/semver"
)
)
//go:generate ./scripts/gen-go-std-tables.sh
//go:generate ./scripts/gen-go-std-tables.sh
@ -151,12 +152,10 @@ type listedPackage struct {
Dir string
Dir string
CompiledGoFiles [ ] string
CompiledGoFiles [ ] string
IgnoredGoFiles [ ] string
Imports [ ] string
Imports [ ] string
Incomplete bool
Error * packageError // to report package loading errors to the user
// These two exist to report package loading errors to the user.
Error * packageError
DepsErrors [ ] * packageError
// The fields below are not part of 'go list', but are still reused
// The fields below are not part of 'go list', but are still reused
// between garble processes. Use "Garble" as a prefix to ensure no
// between garble processes. Use "Garble" as a prefix to ensure no
@ -174,6 +173,7 @@ type listedPackage struct {
}
}
type packageError struct {
type packageError struct {
Pos string
Err string
Err string
}
}
@ -254,58 +254,30 @@ func appendListedPackages(packages []string, mainBuild bool) error {
if sharedCache . ListedPackages == nil {
if sharedCache . ListedPackages == nil {
sharedCache . ListedPackages = make ( map [ string ] * listedPackage )
sharedCache . ListedPackages = make ( map [ string ] * listedPackage )
}
}
var pkgErrors [ ] string
var pkgErrors strings . Builder
for dec . More ( ) {
for dec . More ( ) {
var pkg listedPackage
var pkg listedPackage
if err := dec . Decode ( & pkg ) ; err != nil {
if err := dec . Decode ( & pkg ) ; err != nil {
return err
return err
}
}
// Sometimes cmd/go sets Error without setting Incomplete per the docs.
// TODO: remove the workaround once https://go.dev/issue/57724 is fixed.
if pkg . Error != nil || pkg . DepsErrors != nil {
pkg . Incomplete = true
}
if perr := pkg . Error ; perr != nil {
if perr := pkg . Error ; perr != nil {
switch {
if pkg . Standard && len ( pkg . CompiledGoFiles ) == 0 && len ( pkg . IgnoredGoFiles ) > 0 {
// All errors in non-std packages are fatal,
// Some packages in runtimeLinknamed need a build tag to be importable,
// but only some errors in std packages are.
// like crypto/internal/boring/fipstls with boringcrypto,
case strings . Contains ( pkg . ImportPath , "." ) :
// so any pkg.Error should be ignored when the build tag isn't set.
fallthrough
} else if pkg . ImportPath == "maps" && semver . Compare ( sharedCache . GoVersionSemver , "v1.21" ) < 0 {
default :
// "maps" was added in Go 1.21, so Go 1.20 runs into a "not found" error.
// Error messages sometimes include a trailing newline.
} else {
pkgErrors = append ( pkgErrors , strings . TrimSpace ( perr . Err ) )
if pkgErrors . Len ( ) > 0 {
pkgErrors . WriteString ( "\n" )
// Some packages in runtimeLinknamed are OS-specific,
}
// like crypto/internal/boring/fipstls, so "no Go files"
if perr . Pos != "" {
// for the current OS can be ignored safely as an error.
pkgErrors . WriteString ( perr . Pos )
case pkg . Standard && strings . Contains ( perr . Err , "build constraints exclude all Go files" ) :
pkgErrors . WriteString ( ": " )
// Some packages in runtimeLinknamed are recent,
// like "arena", so older Go versions that we support
// do not yet have them and that's OK.
// Note that pkg.Standard is false for them.
// Note that Go 1.21 is swapping "GOROOT" for "std".
// TODO(mvdan): We try to list test packages like runtime/metrics_test, which always fail.
case strings . Contains ( perr . Err , "is not in GOROOT" ) :
case strings . Contains ( perr . Err , "is not in std" ) :
case strings . Contains ( perr . Err , "cannot find package" ) :
}
}
if len ( pkg . DepsErrors ) > 0 {
for i , derr := range pkg . DepsErrors {
// When an error in DepsErrors starts with a "# pkg/path" line,
// it's an error that we're already printing via that package's Error field.
// Otherwise, the error is that we couldn't find that package at all,
// so we do need to print it here as that package won't be listed.
if i == 0 {
if strings . HasPrefix ( derr . Err , "# " ) {
break
}
pkgErrors = append ( pkgErrors , "# " + pkg . ImportPath )
}
}
// Error messages sometimes include a trailing newline.
// Error messages sometimes include a trailing newline.
pkgErrors = append ( pkgErrors , strings . TrimSpace ( derr . Err ) )
pkgErrors . WriteString ( strings . TrimRight ( perr . Err , "\n" ) )
}
}
}
}
@ -329,8 +301,8 @@ func appendListedPackages(packages []string, mainBuild bool) error {
if err := cmd . Wait ( ) ; err != nil {
if err := cmd . Wait ( ) ; err != nil {
return fmt . Errorf ( "go list error: %v:\nargs: %q\n%s" , err , args , stderr . Bytes ( ) )
return fmt . Errorf ( "go list error: %v:\nargs: %q\n%s" , err , args , stderr . Bytes ( ) )
}
}
if len ( pkgErrors ) > 0 {
if pkgErrors . Len ( ) > 0 {
return errors . New ( strings. Join ( pkgErrors , "\n" ) )
return errors . New ( pkgErrors. String ( ) )
}
}
anyToObfuscate := false
anyToObfuscate := false
@ -345,11 +317,7 @@ func appendListedPackages(packages []string, mainBuild bool) error {
// "unknown pc" crashes on windows in the cgo test otherwise.
// "unknown pc" crashes on windows in the cgo test otherwise.
path == "runtime/cgo" :
path == "runtime/cgo" :
// We can't obfuscate packages which weren't loaded.
// No point in obfuscating empty packages, like OS-specific ones that don't match.
// This can happen since we ignore some pkg.Error messages above.
case pkg . Incomplete :
// No point in obfuscating empty packages.
case len ( pkg . CompiledGoFiles ) == 0 :
case len ( pkg . CompiledGoFiles ) == 0 :
// Test main packages like "foo/bar.test" are always obfuscated,
// Test main packages like "foo/bar.test" are always obfuscated,