| |
| |
| |
|
|
| package main |
|
|
| import ( |
| "bytes" |
| "encoding/json" |
| "flag" |
| "fmt" |
| "io" |
| "io/fs" |
| "log" |
| "os" |
| "os/exec" |
| "path/filepath" |
| "reflect" |
| "regexp" |
| "runtime" |
| "slices" |
| "strconv" |
| "strings" |
| "time" |
| ) |
|
|
| func cmdtest() { |
| gogcflags = os.Getenv("GO_GCFLAGS") |
| setNoOpt() |
|
|
| var t tester |
|
|
| var noRebuild bool |
| flag.BoolVar(&t.listMode, "list", false, "list available tests") |
| flag.BoolVar(&t.rebuild, "rebuild", false, "rebuild everything first") |
| flag.BoolVar(&noRebuild, "no-rebuild", false, "overrides -rebuild (historical dreg)") |
| flag.BoolVar(&t.keepGoing, "k", false, "keep going even when error occurred") |
| flag.BoolVar(&t.race, "race", false, "run in race builder mode (different set of tests)") |
| flag.BoolVar(&t.compileOnly, "compile-only", false, "compile tests, but don't run them") |
| flag.StringVar(&t.banner, "banner", "##### ", "banner prefix; blank means no section banners") |
| flag.StringVar(&t.runRxStr, "run", "", |
| "run only those tests matching the regular expression; empty means to run all. "+ |
| "Special exception: if the string begins with '!', the match is inverted.") |
| flag.BoolVar(&t.msan, "msan", false, "run in memory sanitizer builder mode") |
| flag.BoolVar(&t.asan, "asan", false, "run in address sanitizer builder mode") |
| flag.BoolVar(&t.json, "json", false, "report test results in JSON") |
|
|
| xflagparse(-1) |
| if noRebuild { |
| t.rebuild = false |
| } |
|
|
| t.run() |
| } |
|
|
| |
| type tester struct { |
| race bool |
| msan bool |
| asan bool |
| listMode bool |
| rebuild bool |
| failed bool |
| keepGoing bool |
| compileOnly bool |
| runRxStr string |
| runRx *regexp.Regexp |
| runRxWant bool |
| runNames []string |
| banner string |
| lastHeading string |
|
|
| short bool |
| cgoEnabled bool |
| json bool |
|
|
| tests []distTest |
| testNames map[string]bool |
| timeoutScale int |
|
|
| worklist []*work |
| } |
|
|
| |
| type work struct { |
| dt *distTest |
| cmd *exec.Cmd |
| flush func() |
| start chan bool |
| out bytes.Buffer |
| err error |
| end chan struct{} |
| } |
|
|
| |
| func (w *work) printSkip(t *tester, msg string) { |
| if t.json { |
| synthesizeSkipEvent(json.NewEncoder(&w.out), w.dt.name, msg) |
| return |
| } |
| fmt.Fprintln(&w.out, msg) |
| } |
|
|
| |
| |
| type distTest struct { |
| name string |
| heading string |
| fn func(*distTest) error |
| } |
|
|
| func (t *tester) run() { |
| timelog("start", "dist test") |
|
|
| os.Setenv("PATH", fmt.Sprintf("%s%c%s", gorootBin, os.PathListSeparator, os.Getenv("PATH"))) |
|
|
| t.short = true |
| if v := os.Getenv("GO_TEST_SHORT"); v != "" { |
| short, err := strconv.ParseBool(v) |
| if err != nil { |
| fatalf("invalid GO_TEST_SHORT %q: %v", v, err) |
| } |
| t.short = short |
| } |
|
|
| cmd := exec.Command(gorootBinGo, "env", "CGO_ENABLED") |
| cmd.Stderr = new(bytes.Buffer) |
| slurp, err := cmd.Output() |
| if err != nil { |
| fatalf("Error running %s: %v\n%s", cmd, err, cmd.Stderr) |
| } |
| parts := strings.Split(string(slurp), "\n") |
| if nlines := len(parts) - 1; nlines < 1 { |
| fatalf("Error running %s: output contains <1 lines\n%s", cmd, cmd.Stderr) |
| } |
| t.cgoEnabled, _ = strconv.ParseBool(parts[0]) |
|
|
| if flag.NArg() > 0 && t.runRxStr != "" { |
| fatalf("the -run regular expression flag is mutually exclusive with test name arguments") |
| } |
|
|
| t.runNames = flag.Args() |
|
|
| |
| |
| |
| |
| |
| if ok := isEnvSet("GOTRACEBACK"); !ok { |
| if err := os.Setenv("GOTRACEBACK", "system"); err != nil { |
| if t.keepGoing { |
| log.Printf("Failed to set GOTRACEBACK: %v", err) |
| } else { |
| fatalf("Failed to set GOTRACEBACK: %v", err) |
| } |
| } |
| } |
|
|
| if t.rebuild { |
| t.out("Building packages and commands.") |
| |
| goInstall(toolenv(), gorootBinGo, append([]string{"-a"}, toolchain...)...) |
| } |
|
|
| if !t.listMode { |
| if builder := os.Getenv("GO_BUILDER_NAME"); builder == "" { |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| goInstall(toolenv(), gorootBinGo, toolchain...) |
| goInstall(toolenv(), gorootBinGo, toolchain...) |
| goInstall(toolenv(), gorootBinGo, toolsToInstall...) |
| } |
| } |
|
|
| t.timeoutScale = 1 |
| if s := os.Getenv("GO_TEST_TIMEOUT_SCALE"); s != "" { |
| t.timeoutScale, err = strconv.Atoi(s) |
| if err != nil { |
| fatalf("failed to parse $GO_TEST_TIMEOUT_SCALE = %q as integer: %v", s, err) |
| } |
| } |
|
|
| if t.runRxStr != "" { |
| if t.runRxStr[0] == '!' { |
| t.runRxWant = false |
| t.runRxStr = t.runRxStr[1:] |
| } else { |
| t.runRxWant = true |
| } |
| t.runRx = regexp.MustCompile(t.runRxStr) |
| } |
|
|
| t.registerTests() |
| if t.listMode { |
| for _, tt := range t.tests { |
| fmt.Println(tt.name) |
| } |
| return |
| } |
|
|
| for _, name := range t.runNames { |
| if !t.testNames[name] { |
| fatalf("unknown test %q", name) |
| } |
| } |
|
|
| |
| if strings.HasPrefix(os.Getenv("GO_BUILDER_NAME"), "linux-") { |
| if os.Getuid() == 0 { |
| |
| |
| } else { |
| xatexit(t.makeGOROOTUnwritable()) |
| } |
| } |
|
|
| if !t.json { |
| if err := t.maybeLogMetadata(); err != nil { |
| t.failed = true |
| if t.keepGoing { |
| log.Printf("Failed logging metadata: %v", err) |
| } else { |
| fatalf("Failed logging metadata: %v", err) |
| } |
| } |
| } |
|
|
| var anyIncluded, someExcluded bool |
| for _, dt := range t.tests { |
| if !t.shouldRunTest(dt.name) { |
| someExcluded = true |
| continue |
| } |
| anyIncluded = true |
| dt := dt |
| if err := dt.fn(&dt); err != nil { |
| t.runPending(&dt) |
| t.failed = true |
| if t.keepGoing { |
| log.Printf("Failed: %v", err) |
| } else { |
| fatalf("Failed: %v", err) |
| } |
| } |
| } |
| t.runPending(nil) |
| timelog("end", "dist test") |
|
|
| if !t.json { |
| if t.failed { |
| fmt.Println("\nFAILED") |
| } else if !anyIncluded { |
| fmt.Println() |
| errprintf("go tool dist: warning: %q matched no tests; use the -list flag to list available tests\n", t.runRxStr) |
| fmt.Println("NO TESTS TO RUN") |
| } else if someExcluded { |
| fmt.Println("\nALL TESTS PASSED (some were excluded)") |
| } else { |
| fmt.Println("\nALL TESTS PASSED") |
| } |
| } |
| if t.failed { |
| xexit(1) |
| } |
| } |
|
|
| func (t *tester) shouldRunTest(name string) bool { |
| if t.runRx != nil { |
| return t.runRx.MatchString(name) == t.runRxWant |
| } |
| if len(t.runNames) == 0 { |
| return true |
| } |
| return slices.Contains(t.runNames, name) |
| } |
|
|
| func (t *tester) maybeLogMetadata() error { |
| if t.compileOnly { |
| |
| |
| return nil |
| } |
| t.out("Test execution environment.") |
| |
| |
| |
| |
| |
| |
| return t.dirCmd(filepath.Join(goroot, "src/cmd/internal/metadata"), gorootBinGo, []string{"run", "main.go"}).Run() |
| } |
|
|
| |
| func testName(pkg, variant string) string { |
| name := pkg |
| if variant != "" { |
| name += ":" + variant |
| } |
| return name |
| } |
|
|
| |
| |
| type goTest struct { |
| timeout time.Duration |
| short bool |
| tags []string |
| race bool |
| bench bool |
| runTests string |
| cpu string |
| skip string |
|
|
| gcflags string |
| ldflags string |
| buildmode string |
|
|
| env []string |
|
|
| runOnHost bool |
|
|
| |
| |
| |
| variant string |
| |
| |
| |
| |
| |
| |
| |
| omitVariant bool |
|
|
| |
| |
| pkgs []string |
| pkg string |
|
|
| testFlags []string |
| } |
|
|
| |
| |
| func (opts *goTest) compileOnly() bool { |
| return opts.runTests == "^$" && !opts.bench |
| } |
|
|
| |
| |
| |
| func (opts *goTest) bgCommand(t *tester, stdout, stderr io.Writer) (cmd *exec.Cmd, flush func()) { |
| build, run, pkgs, testFlags, setupCmd := opts.buildArgs(t) |
|
|
| |
| args := append([]string{"test"}, build...) |
| if t.compileOnly || opts.compileOnly() { |
| args = append(args, "-c", "-o", os.DevNull) |
| } else { |
| args = append(args, run...) |
| } |
| args = append(args, pkgs...) |
| if !t.compileOnly && !opts.compileOnly() { |
| args = append(args, testFlags...) |
| } |
|
|
| cmd = exec.Command(gorootBinGo, args...) |
| setupCmd(cmd) |
| if t.json && opts.variant != "" && !opts.omitVariant { |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| if stdout == stderr { |
| stdout = &lockedWriter{w: stdout} |
| stderr = stdout |
| } |
| f := &testJSONFilter{w: stdout, variant: opts.variant} |
| cmd.Stdout = f |
| flush = f.Flush |
| } else { |
| cmd.Stdout = stdout |
| flush = func() {} |
| } |
| cmd.Stderr = stderr |
|
|
| return cmd, flush |
| } |
|
|
| |
| func (opts *goTest) run(t *tester) error { |
| cmd, flush := opts.bgCommand(t, os.Stdout, os.Stderr) |
| err := cmd.Run() |
| flush() |
| return err |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| func (opts *goTest) buildArgs(t *tester) (build, run, pkgs, testFlags []string, setupCmd func(*exec.Cmd)) { |
| run = append(run, "-count=1") |
| if opts.timeout != 0 { |
| d := opts.timeout * time.Duration(t.timeoutScale) |
| run = append(run, "-timeout="+d.String()) |
| } else if t.timeoutScale != 1 { |
| const goTestDefaultTimeout = 10 * time.Minute |
| run = append(run, "-timeout="+(goTestDefaultTimeout*time.Duration(t.timeoutScale)).String()) |
| } |
| if opts.short || t.short { |
| run = append(run, "-short") |
| } |
| var tags []string |
| if t.iOS() { |
| tags = append(tags, "lldb") |
| } |
| if noOpt { |
| tags = append(tags, "noopt") |
| } |
| tags = append(tags, opts.tags...) |
| if len(tags) > 0 { |
| build = append(build, "-tags="+strings.Join(tags, ",")) |
| } |
| if t.race || opts.race { |
| build = append(build, "-race") |
| } |
| if t.msan { |
| build = append(build, "-msan") |
| } |
| if t.asan { |
| build = append(build, "-asan") |
| } |
| if opts.bench { |
| |
| run = append(run, "-run=^$") |
| |
| run = append(run, "-bench=.*", "-benchtime=.1s") |
| } else if opts.runTests != "" { |
| run = append(run, "-run="+opts.runTests) |
| } |
| if opts.cpu != "" { |
| run = append(run, "-cpu="+opts.cpu) |
| } |
| if opts.skip != "" { |
| run = append(run, "-skip="+opts.skip) |
| } |
| if t.json { |
| run = append(run, "-json") |
| } |
|
|
| if opts.gcflags != "" { |
| build = append(build, "-gcflags=all="+opts.gcflags) |
| } |
| if opts.ldflags != "" { |
| build = append(build, "-ldflags="+opts.ldflags) |
| } |
| if opts.buildmode != "" { |
| build = append(build, "-buildmode="+opts.buildmode) |
| } |
|
|
| pkgs = opts.packages() |
|
|
| runOnHost := opts.runOnHost && (goarch != gohostarch || goos != gohostos) |
| needTestFlags := len(opts.testFlags) > 0 || runOnHost |
| if needTestFlags { |
| testFlags = append([]string{"-args"}, opts.testFlags...) |
| } |
| if runOnHost { |
| |
| testFlags = append(testFlags, "-target="+goos+"/"+goarch) |
| } |
|
|
| setupCmd = func(cmd *exec.Cmd) { |
| setDir(cmd, filepath.Join(goroot, "src")) |
| if len(opts.env) != 0 { |
| for _, kv := range opts.env { |
| if i := strings.Index(kv, "="); i < 0 { |
| unsetEnv(cmd, kv[:len(kv)-1]) |
| } else { |
| setEnv(cmd, kv[:i], kv[i+1:]) |
| } |
| } |
| } |
| if runOnHost { |
| setEnv(cmd, "GOARCH", gohostarch) |
| setEnv(cmd, "GOOS", gohostos) |
| } |
| } |
|
|
| return |
| } |
|
|
| |
| |
| func (opts *goTest) packages() []string { |
| pkgs := opts.pkgs |
| if opts.pkg != "" { |
| pkgs = append(pkgs[:len(pkgs):len(pkgs)], opts.pkg) |
| } |
| if len(pkgs) == 0 { |
| panic("no packages") |
| } |
| return pkgs |
| } |
|
|
| |
| func (opts *goTest) printSkip(t *tester, msg string) { |
| if t.json { |
| enc := json.NewEncoder(os.Stdout) |
| for _, pkg := range opts.packages() { |
| synthesizeSkipEvent(enc, pkg, msg) |
| } |
| return |
| } |
| fmt.Println(msg) |
| } |
|
|
| |
| |
| |
| |
| |
| |
| var ( |
| ranGoTest bool |
| stdMatches []string |
|
|
| ranGoBench bool |
| benchMatches []string |
| ) |
|
|
| func (t *tester) registerStdTest(pkg string) { |
| const stdTestHeading = "Testing packages." |
| gcflags := gogcflags |
| name := testName(pkg, "") |
| if t.runRx == nil || t.runRx.MatchString(name) == t.runRxWant { |
| stdMatches = append(stdMatches, pkg) |
| } |
| t.addTest(name, stdTestHeading, func(dt *distTest) error { |
| if ranGoTest { |
| return nil |
| } |
| t.runPending(dt) |
| timelog("start", dt.name) |
| defer timelog("end", dt.name) |
| ranGoTest = true |
|
|
| timeoutSec := 180 * time.Second |
| for _, pkg := range stdMatches { |
| if pkg == "cmd/go" { |
| timeoutSec *= 3 |
| break |
| } |
| } |
| return (&goTest{ |
| timeout: timeoutSec, |
| gcflags: gcflags, |
| pkgs: stdMatches, |
| }).run(t) |
| }) |
| } |
|
|
| func (t *tester) registerRaceBenchTest(pkg string) { |
| const raceBenchHeading = "Running benchmarks briefly." |
| name := testName(pkg, "racebench") |
| if t.runRx == nil || t.runRx.MatchString(name) == t.runRxWant { |
| benchMatches = append(benchMatches, pkg) |
| } |
| t.addTest(name, raceBenchHeading, func(dt *distTest) error { |
| if ranGoBench { |
| return nil |
| } |
| t.runPending(dt) |
| timelog("start", dt.name) |
| defer timelog("end", dt.name) |
| ranGoBench = true |
| return (&goTest{ |
| variant: "racebench", |
| |
| |
| |
| omitVariant: false, |
| timeout: 1200 * time.Second, |
| race: true, |
| bench: true, |
| cpu: "4", |
| pkgs: benchMatches, |
| }).run(t) |
| }) |
| } |
|
|
| func (t *tester) registerTests() { |
| |
| |
| |
| |
| |
| |
| |
| registerStdTestSpecially := map[string]bool{ |
| |
| |
| |
| |
| "cmd/internal/testdir": true, |
| } |
|
|
| |
| |
| |
| if len(t.runNames) > 0 { |
| for _, name := range t.runNames { |
| if !strings.Contains(name, ":") { |
| t.registerStdTest(name) |
| } else if strings.HasSuffix(name, ":racebench") { |
| t.registerRaceBenchTest(strings.TrimSuffix(name, ":racebench")) |
| } |
| } |
| } else { |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| cmd := exec.Command(gorootBinGo, "list") |
| if t.race { |
| cmd.Args = append(cmd.Args, "-tags=race") |
| } |
| cmd.Args = append(cmd.Args, "std", "cmd") |
| cmd.Stderr = new(bytes.Buffer) |
| all, err := cmd.Output() |
| if err != nil { |
| fatalf("Error running go list std cmd: %v:\n%s", err, cmd.Stderr) |
| } |
| pkgs := strings.Fields(string(all)) |
| for _, pkg := range pkgs { |
| if registerStdTestSpecially[pkg] { |
| continue |
| } |
| if t.short && (strings.HasPrefix(pkg, "vendor/") || strings.HasPrefix(pkg, "cmd/vendor/")) { |
| |
| |
| |
| continue |
| } |
| t.registerStdTest(pkg) |
| } |
| if t.race && !t.short { |
| for _, pkg := range pkgs { |
| if t.packageHasBenchmarks(pkg) { |
| t.registerRaceBenchTest(pkg) |
| } |
| } |
| } |
| } |
|
|
| if t.race { |
| return |
| } |
|
|
| |
| if !t.compileOnly { |
| t.registerTest("os/user with tag osusergo", |
| &goTest{ |
| variant: "osusergo", |
| timeout: 300 * time.Second, |
| tags: []string{"osusergo"}, |
| pkg: "os/user", |
| }) |
| t.registerTest("hash/maphash purego implementation", |
| &goTest{ |
| variant: "purego", |
| timeout: 300 * time.Second, |
| tags: []string{"purego"}, |
| pkg: "hash/maphash", |
| env: []string{"GODEBUG=fips140=off"}, |
| }) |
| } |
|
|
| |
| t.registerTest("crypto with tag purego (build and vet only)", &goTest{ |
| variant: "purego", |
| tags: []string{"purego"}, |
| pkg: "crypto/...", |
| runTests: "^$", |
| }) |
|
|
| |
| if t.fipsSupported() { |
| |
| t.registerTest("GOFIPS140=latest go test crypto/...", &goTest{ |
| variant: "gofips140", |
| env: []string{"GOFIPS140=latest"}, |
| pkg: "crypto/...", |
| }) |
|
|
| |
| |
| for _, version := range fipsVersions(t.short) { |
| suffix := " # (build and vet only)" |
| run := "^$" |
| if !t.short { |
| suffix = "" |
| run = "" |
| } |
| t.registerTest("GOFIPS140="+version+" go test crypto/..."+suffix, &goTest{ |
| variant: "gofips140-" + version, |
| pkg: "crypto/...", |
| runTests: run, |
| env: []string{"GOFIPS140=" + version, "GOMODCACHE=" + filepath.Join(workdir, "fips-"+version)}, |
| }) |
| } |
| } |
|
|
| |
| if !strings.Contains(goexperiment, "jsonv2") { |
| t.registerTest("GOEXPERIMENT=jsonv2 go test encoding/json/...", &goTest{ |
| variant: "jsonv2", |
| env: []string{"GOEXPERIMENT=" + goexperiments("jsonv2")}, |
| pkg: "encoding/json/...", |
| }) |
| } |
|
|
| |
| if !strings.Contains(goexperiment, "runtimesecret") { |
| t.registerTest("GOEXPERIMENT=runtimesecret go test runtime/secret/...", &goTest{ |
| variant: "runtimesecret", |
| env: []string{"GOEXPERIMENT=" + goexperiments("runtimesecret")}, |
| pkg: "runtime/secret/...", |
| }) |
| } |
|
|
| |
| if goarch == "amd64" && !strings.Contains(goexperiment, "simd") { |
| t.registerTest("GOEXPERIMENT=simd go test simd/archsimd/...", &goTest{ |
| variant: "simd", |
| env: []string{"GOEXPERIMENT=" + goexperiments("simd")}, |
| pkg: "simd/archsimd/...", |
| }) |
| } |
|
|
| |
| if goos == "darwin" && goarch == "amd64" && t.cgoEnabled { |
| t.registerTest("GOOS=ios on darwin/amd64", |
| &goTest{ |
| variant: "amd64ios", |
| timeout: 300 * time.Second, |
| runTests: "SystemRoots", |
| env: []string{"GOOS=ios", "CGO_ENABLED=1"}, |
| pkg: "crypto/x509", |
| }) |
| } |
|
|
| |
| |
| |
| if !t.compileOnly && !t.short { |
| t.registerTest("GODEBUG=gcstoptheworld=2 archive/zip", |
| &goTest{ |
| variant: "runtime:gcstoptheworld2", |
| timeout: 300 * time.Second, |
| short: true, |
| env: []string{"GODEBUG=gcstoptheworld=2"}, |
| pkg: "archive/zip", |
| }) |
| t.registerTest("GODEBUG=gccheckmark=1 runtime", |
| &goTest{ |
| variant: "runtime:gccheckmark", |
| timeout: 300 * time.Second, |
| short: true, |
| env: []string{"GODEBUG=gccheckmark=1"}, |
| pkg: "runtime", |
| }) |
| } |
|
|
| |
| |
| |
| |
| if !t.compileOnly && !t.short { |
| |
| hooks := []string{"mayMoreStackPreempt", "mayMoreStackMove"} |
| |
| |
| hookPkgs := []string{"runtime/...", "reflect", "sync"} |
| |
| |
| unhookPkgs := []string{"runtime/testdata/..."} |
| for _, hook := range hooks { |
| |
| |
| |
| |
| |
| |
| goFlagsList := []string{} |
| for _, flag := range []string{"-gcflags", "-asmflags"} { |
| for _, hookPkg := range hookPkgs { |
| goFlagsList = append(goFlagsList, flag+"="+hookPkg+"=-d=maymorestack=runtime."+hook) |
| } |
| for _, unhookPkg := range unhookPkgs { |
| goFlagsList = append(goFlagsList, flag+"="+unhookPkg+"=") |
| } |
| } |
| goFlags := strings.Join(goFlagsList, " ") |
|
|
| t.registerTest("maymorestack="+hook, |
| &goTest{ |
| variant: hook, |
| timeout: 600 * time.Second, |
| short: true, |
| env: []string{"GOFLAGS=" + goFlags}, |
| pkgs: []string{"runtime", "reflect", "sync"}, |
| }) |
| } |
| } |
|
|
| |
| |
| |
| |
| for _, pkg := range cgoPackages { |
| if !t.internalLink() { |
| break |
| } |
|
|
| |
| if goarch == "arm" { |
| break |
| } |
|
|
| |
| |
| run := "^Test[^CS]" |
| if pkg == "net" { |
| run = "TestTCPStress" |
| } |
| t.registerTest("Testing without libgcc.", |
| &goTest{ |
| variant: "nolibgcc", |
| ldflags: "-linkmode=internal -libgcc=none", |
| runTests: run, |
| pkg: pkg, |
| }) |
| } |
|
|
| |
| builderName := os.Getenv("GO_BUILDER_NAME") |
| disablePIE := strings.HasSuffix(builderName, "-alpine") |
|
|
| |
| if t.internalLinkPIE() && !disablePIE { |
| t.registerTest("internal linking, -buildmode=pie", |
| &goTest{ |
| variant: "pie_internal", |
| timeout: 60 * time.Second, |
| buildmode: "pie", |
| ldflags: "-linkmode=internal", |
| env: []string{"CGO_ENABLED=0"}, |
| pkg: "reflect", |
| }) |
| t.registerTest("internal linking, -buildmode=pie", |
| &goTest{ |
| variant: "pie_internal", |
| timeout: 60 * time.Second, |
| buildmode: "pie", |
| ldflags: "-linkmode=internal", |
| env: []string{"CGO_ENABLED=0"}, |
| pkg: "crypto/internal/fips140test", |
| runTests: "TestFIPSCheck", |
| }) |
| |
| if t.cgoEnabled && t.internalLink() && !disablePIE { |
| t.registerTest("internal linking, -buildmode=pie", |
| &goTest{ |
| variant: "pie_internal", |
| timeout: 60 * time.Second, |
| buildmode: "pie", |
| ldflags: "-linkmode=internal", |
| pkg: "os/user", |
| }) |
| } |
| } |
|
|
| if t.extLink() && !t.compileOnly { |
| if goos != "android" { |
| t.registerTest("external linking, -buildmode=exe", |
| &goTest{ |
| variant: "exe_external", |
| timeout: 60 * time.Second, |
| buildmode: "exe", |
| ldflags: "-linkmode=external", |
| env: []string{"CGO_ENABLED=1"}, |
| pkg: "crypto/internal/fips140test", |
| runTests: "TestFIPSCheck", |
| }) |
| } |
| if t.externalLinkPIE() && !disablePIE { |
| t.registerTest("external linking, -buildmode=pie", |
| &goTest{ |
| variant: "pie_external", |
| timeout: 60 * time.Second, |
| buildmode: "pie", |
| ldflags: "-linkmode=external", |
| env: []string{"CGO_ENABLED=1"}, |
| pkg: "crypto/internal/fips140test", |
| runTests: "TestFIPSCheck", |
| }) |
| } |
| } |
|
|
| |
| if t.hasParallelism() { |
| t.registerTest("sync -cpu=10", |
| &goTest{ |
| variant: "cpu10", |
| timeout: 120 * time.Second, |
| cpu: "10", |
| pkg: "sync", |
| }) |
| } |
|
|
| const cgoHeading = "Testing cgo" |
| if t.cgoEnabled { |
| t.registerCgoTests(cgoHeading) |
| } |
|
|
| if goos == "wasip1" { |
| t.registerTest("wasip1 host tests", |
| &goTest{ |
| variant: "host", |
| pkg: "internal/runtime/wasitest", |
| timeout: 1 * time.Minute, |
| runOnHost: true, |
| }) |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| if goos == "darwin" || ((goos == "linux" || goos == "windows") && (goarch == "amd64" && !strings.Contains(goexperiment, "simd"))) { |
| t.registerTest("API release note check", &goTest{variant: "check", pkg: "cmd/relnote", testFlags: []string{"-check"}}) |
| t.registerTest("API check", &goTest{variant: "check", pkg: "cmd/api", timeout: 5 * time.Minute, testFlags: []string{"-check"}}) |
| } |
|
|
| |
| if !t.compileOnly && t.hasParallelism() { |
| for i := 1; i <= 4; i *= 2 { |
| t.registerTest(fmt.Sprintf("GOMAXPROCS=2 runtime -cpu=%d -quick", i), |
| &goTest{ |
| variant: "cpu" + strconv.Itoa(i), |
| timeout: 300 * time.Second, |
| cpu: strconv.Itoa(i), |
| gcflags: gogcflags, |
| short: true, |
| testFlags: []string{"-quick"}, |
| |
| |
| env: []string{"GOMAXPROCS=2"}, |
| pkg: "runtime", |
| }) |
| } |
| } |
|
|
| if t.raceDetectorSupported() && !t.msan && !t.asan { |
| |
| t.registerRaceTests() |
| } |
|
|
| if goos != "android" && !t.iOS() { |
| |
| |
| |
| nShards := 1 |
| if os.Getenv("GO_BUILDER_NAME") != "" { |
| nShards = 10 |
| } |
| if n, err := strconv.Atoi(os.Getenv("GO_TEST_SHARDS")); err == nil { |
| nShards = n |
| } |
| for shard := 0; shard < nShards; shard++ { |
| id := fmt.Sprintf("%d_%d", shard, nShards) |
| t.registerTest("../test", |
| &goTest{ |
| variant: id, |
| |
| |
| |
| omitVariant: false, |
| pkg: "cmd/internal/testdir", |
| testFlags: []string{fmt.Sprintf("-shard=%d", shard), fmt.Sprintf("-shards=%d", nShards)}, |
| runOnHost: true, |
| }, |
| ) |
| } |
| } |
| } |
|
|
| |
| |
| |
| func (t *tester) addTest(name, heading string, fn func(*distTest) error) { |
| if t.testNames[name] { |
| panic("duplicate registered test name " + name) |
| } |
| if heading == "" { |
| panic("empty heading") |
| } |
| |
| if !strings.Contains(name, ":") && heading != "Testing packages." { |
| panic("empty variant is reserved exclusively for registerStdTest") |
| } else if strings.HasSuffix(name, ":racebench") && heading != "Running benchmarks briefly." { |
| panic("racebench variant is reserved exclusively for registerRaceBenchTest") |
| } |
| if t.testNames == nil { |
| t.testNames = make(map[string]bool) |
| } |
| t.testNames[name] = true |
| t.tests = append(t.tests, distTest{ |
| name: name, |
| heading: heading, |
| fn: fn, |
| }) |
| } |
|
|
| type registerTestOpt interface { |
| isRegisterTestOpt() |
| } |
|
|
| |
| |
| type rtSkipFunc struct { |
| skip func(*distTest) (string, bool) |
| } |
|
|
| func (rtSkipFunc) isRegisterTestOpt() {} |
|
|
| |
| |
| |
| |
| |
| |
| func (t *tester) registerTest(heading string, test *goTest, opts ...registerTestOpt) { |
| var skipFunc func(*distTest) (string, bool) |
| for _, opt := range opts { |
| switch opt := opt.(type) { |
| case rtSkipFunc: |
| skipFunc = opt.skip |
| } |
| } |
| |
| register1 := func(test *goTest) { |
| if test.variant == "" { |
| panic("empty variant") |
| } |
| name := testName(test.pkg, test.variant) |
| t.addTest(name, heading, func(dt *distTest) error { |
| if skipFunc != nil { |
| msg, skip := skipFunc(dt) |
| if skip { |
| test.printSkip(t, msg) |
| return nil |
| } |
| } |
| w := &work{dt: dt} |
| w.cmd, w.flush = test.bgCommand(t, &w.out, &w.out) |
| t.worklist = append(t.worklist, w) |
| return nil |
| }) |
| } |
| if test.pkg != "" && len(test.pkgs) == 0 { |
| |
| register1(test) |
| return |
| } |
| |
| |
| |
| |
| |
| |
| for _, pkg := range test.packages() { |
| test1 := *test |
| test1.pkg, test1.pkgs = pkg, nil |
| register1(&test1) |
| } |
| } |
|
|
| |
| |
| |
| func (t *tester) dirCmd(dir string, cmdline ...any) *exec.Cmd { |
| bin, args := flattenCmdline(cmdline) |
| cmd := exec.Command(bin, args...) |
| if filepath.IsAbs(dir) { |
| setDir(cmd, dir) |
| } else { |
| setDir(cmd, filepath.Join(goroot, dir)) |
| } |
| cmd.Stdout = os.Stdout |
| cmd.Stderr = os.Stderr |
| if vflag > 1 { |
| errprintf("%#q\n", cmd) |
| } |
| return cmd |
| } |
|
|
| |
| |
| func flattenCmdline(cmdline []any) (bin string, args []string) { |
| var list []string |
| for _, x := range cmdline { |
| switch x := x.(type) { |
| case string: |
| list = append(list, x) |
| case []string: |
| list = append(list, x...) |
| default: |
| panic("invalid dirCmd argument type: " + reflect.TypeOf(x).String()) |
| } |
| } |
|
|
| bin = list[0] |
| if !filepath.IsAbs(bin) { |
| panic("command is not absolute: " + bin) |
| } |
| return bin, list[1:] |
| } |
|
|
| func (t *tester) iOS() bool { |
| return goos == "ios" |
| } |
|
|
| func (t *tester) out(v string) { |
| if t.json { |
| return |
| } |
| if t.banner == "" { |
| return |
| } |
| fmt.Println("\n" + t.banner + v) |
| } |
|
|
| |
| |
| func (t *tester) extLink() bool { |
| if !cgoEnabled[goos+"/"+goarch] { |
| return false |
| } |
| if goarch == "ppc64" && goos != "aix" { |
| return false |
| } |
| return true |
| } |
|
|
| func (t *tester) internalLink() bool { |
| if gohostos == "dragonfly" { |
| |
| return false |
| } |
| if goos == "android" { |
| return false |
| } |
| if goos == "ios" { |
| return false |
| } |
| |
| |
| |
| if goarch == "mips64" || goarch == "mips64le" || goarch == "mips" || goarch == "mipsle" || goarch == "riscv64" { |
| return false |
| } |
| if goos == "aix" { |
| |
| return false |
| } |
| if t.msan || t.asan { |
| |
| return false |
| } |
| return true |
| } |
|
|
| func (t *tester) internalLinkPIE() bool { |
| if t.msan || t.asan { |
| |
| return false |
| } |
| switch goos + "-" + goarch { |
| case "darwin-amd64", "darwin-arm64", |
| "linux-amd64", "linux-arm64", "linux-loong64", "linux-ppc64le", |
| "android-arm64", |
| "windows-amd64", "windows-386", "windows-arm64": |
| return true |
| } |
| return false |
| } |
|
|
| func (t *tester) externalLinkPIE() bool { |
| |
| |
| switch goos + "-" + goarch { |
| case "linux-s390x": |
| return true |
| } |
| return t.internalLinkPIE() && t.extLink() |
| } |
|
|
| |
| func (t *tester) supportedBuildmode(mode string) bool { |
| switch mode { |
| case "c-archive", "c-shared", "shared", "plugin", "pie": |
| default: |
| fatalf("internal error: unknown buildmode %s", mode) |
| return false |
| } |
|
|
| return buildModeSupported("gc", mode, goos, goarch) |
| } |
|
|
| func (t *tester) registerCgoTests(heading string) { |
| cgoTest := func(variant string, subdir, linkmode, buildmode string, opts ...registerTestOpt) *goTest { |
| gt := &goTest{ |
| variant: variant, |
| pkg: "cmd/cgo/internal/" + subdir, |
| buildmode: buildmode, |
| } |
| var ldflags []string |
| if linkmode != "auto" { |
| |
| ldflags = append(ldflags, "-linkmode="+linkmode) |
| } |
|
|
| if linkmode == "internal" { |
| gt.tags = append(gt.tags, "internal") |
| if buildmode == "pie" { |
| gt.tags = append(gt.tags, "internal_pie") |
| } |
| } |
| if buildmode == "static" { |
| |
| |
| gt.buildmode = "" |
| if linkmode == "external" { |
| ldflags = append(ldflags, `-extldflags "-static -pthread"`) |
| } else if linkmode == "auto" { |
| gt.env = append(gt.env, "CGO_LDFLAGS=-static -pthread") |
| } else { |
| panic("unknown linkmode with static build: " + linkmode) |
| } |
| gt.tags = append(gt.tags, "static") |
| } |
| gt.ldflags = strings.Join(ldflags, " ") |
|
|
| t.registerTest(heading, gt, opts...) |
| return gt |
| } |
|
|
| |
| |
| |
|
|
| |
| builderName := os.Getenv("GO_BUILDER_NAME") |
| disablePIE := strings.HasSuffix(builderName, "-alpine") |
|
|
| if t.internalLink() { |
| cgoTest("internal", "test", "internal", "") |
| } |
|
|
| os := gohostos |
| p := gohostos + "/" + goarch |
| switch { |
| case os == "darwin", os == "windows": |
| if !t.extLink() { |
| break |
| } |
| |
| cgoTest("external", "test", "external", "") |
|
|
| gt := cgoTest("external-s", "test", "external", "") |
| gt.ldflags += " -s" |
|
|
| if t.supportedBuildmode("pie") && !disablePIE { |
| cgoTest("auto-pie", "test", "auto", "pie") |
| if t.internalLink() && t.internalLinkPIE() { |
| cgoTest("internal-pie", "test", "internal", "pie") |
| } |
| } |
|
|
| case os == "aix", os == "android", os == "dragonfly", os == "freebsd", os == "linux", os == "netbsd", os == "openbsd": |
| gt := cgoTest("external-g0", "test", "external", "") |
| gt.env = append(gt.env, "CGO_CFLAGS=-g0 -fdiagnostics-color") |
|
|
| cgoTest("external", "testtls", "external", "") |
| switch { |
| case os == "aix": |
| |
| case p == "freebsd/arm": |
| |
| |
| |
| |
| |
| default: |
| |
| var staticCheck rtSkipFunc |
| ccName := compilerEnvLookup("CC", defaultcc, goos, goarch) |
| cc, err := exec.LookPath(ccName) |
| if err != nil { |
| staticCheck.skip = func(*distTest) (string, bool) { |
| return fmt.Sprintf("$CC (%q) not found, skip cgo static linking test.", ccName), true |
| } |
| } else { |
| cmd := t.dirCmd("src/cmd/cgo/internal/test", cc, "-xc", "-o", "/dev/null", "-static", "-") |
| cmd.Stdin = strings.NewReader("int main() {}") |
| cmd.Stdout, cmd.Stderr = nil, nil |
| if err := cmd.Run(); err != nil { |
| |
| staticCheck.skip = func(*distTest) (string, bool) { |
| return "No support for static linking found (lacks libc.a?), skip cgo static linking test.", true |
| } |
| } |
| } |
|
|
| |
| |
| |
| |
| if staticCheck.skip == nil && goos == "linux" && strings.Contains(goexperiment, "boringcrypto") { |
| staticCheck.skip = func(*distTest) (string, bool) { |
| return "skipping static linking check on Linux when using boringcrypto to avoid C linker warning about getaddrinfo", true |
| } |
| } |
|
|
| |
| if goos != "android" && p != "netbsd/arm" && !t.msan && !t.asan { |
| |
| |
| |
| cgoTest("static", "testtls", "external", "static", staticCheck) |
| } |
| cgoTest("external", "testnocgo", "external", "", staticCheck) |
| if goos != "android" && !t.msan && !t.asan { |
| |
| |
| cgoTest("static", "testnocgo", "external", "static", staticCheck) |
| cgoTest("static", "test", "external", "static", staticCheck) |
| |
| |
| |
| if goarch != "loong64" && !t.msan && !t.asan { |
| |
| cgoTest("auto-static", "test", "auto", "static", staticCheck) |
| } |
| } |
|
|
| |
| if t.supportedBuildmode("pie") && !disablePIE { |
| cgoTest("auto-pie", "test", "auto", "pie") |
| if t.internalLink() && t.internalLinkPIE() { |
| cgoTest("internal-pie", "test", "internal", "pie") |
| } |
| cgoTest("auto-pie", "testtls", "auto", "pie") |
| cgoTest("auto-pie", "testnocgo", "auto", "pie") |
| } |
| } |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| func (t *tester) runPending(nextTest *distTest) { |
| worklist := t.worklist |
| t.worklist = nil |
| for _, w := range worklist { |
| w.start = make(chan bool) |
| w.end = make(chan struct{}) |
| |
| |
| if w.cmd.Stdout == nil || w.cmd.Stdout == os.Stdout || w.cmd.Stderr == nil || w.cmd.Stderr == os.Stderr { |
| panic("work.cmd.Stdout/Stderr must be redirected") |
| } |
| go func(w *work) { |
| if !<-w.start { |
| timelog("skip", w.dt.name) |
| w.printSkip(t, "skipped due to earlier error") |
| } else { |
| timelog("start", w.dt.name) |
| w.err = w.cmd.Run() |
| if w.flush != nil { |
| w.flush() |
| } |
| if w.err != nil { |
| if isUnsupportedVMASize(w) { |
| timelog("skip", w.dt.name) |
| w.out.Reset() |
| w.printSkip(t, "skipped due to unsupported VMA") |
| w.err = nil |
| } |
| } |
| } |
| timelog("end", w.dt.name) |
| w.end <- struct{}{} |
| }(w) |
| } |
|
|
| maxbg := maxbg |
| |
| |
| if runtime.NumCPU() > 4 && runtime.GOMAXPROCS(0) != 1 { |
| for _, w := range worklist { |
| |
| |
| |
| |
| |
| if strings.Contains(w.dt.heading, "GOMAXPROCS=2 runtime") { |
| maxbg = runtime.NumCPU() |
| break |
| } |
| } |
| } |
|
|
| started := 0 |
| ended := 0 |
| var last *distTest |
| for ended < len(worklist) { |
| for started < len(worklist) && started-ended < maxbg { |
| w := worklist[started] |
| started++ |
| w.start <- !t.failed || t.keepGoing |
| } |
| w := worklist[ended] |
| dt := w.dt |
| if t.lastHeading != dt.heading { |
| t.lastHeading = dt.heading |
| t.out(dt.heading) |
| } |
| if dt != last { |
| |
| last = w.dt |
| if vflag > 0 { |
| fmt.Printf("# go tool dist test -run=^%s$\n", dt.name) |
| } |
| } |
| if vflag > 1 { |
| errprintf("%#q\n", w.cmd) |
| } |
| ended++ |
| <-w.end |
| os.Stdout.Write(w.out.Bytes()) |
| |
| w.out = bytes.Buffer{} |
| if w.err != nil { |
| log.Printf("Failed: %v", w.err) |
| t.failed = true |
| } |
| } |
| if t.failed && !t.keepGoing { |
| fatalf("FAILED") |
| } |
|
|
| if dt := nextTest; dt != nil { |
| if t.lastHeading != dt.heading { |
| t.lastHeading = dt.heading |
| t.out(dt.heading) |
| } |
| if vflag > 0 { |
| fmt.Printf("# go tool dist test -run=^%s$\n", dt.name) |
| } |
| } |
| } |
|
|
| func (t *tester) hasBash() bool { |
| switch gohostos { |
| case "windows", "plan9": |
| return false |
| } |
| return true |
| } |
|
|
| |
| |
| |
| func (t *tester) hasParallelism() bool { |
| switch goos { |
| case "js", "wasip1": |
| return false |
| } |
| return true |
| } |
|
|
| func (t *tester) raceDetectorSupported() bool { |
| if gohostos != goos { |
| return false |
| } |
| if !t.cgoEnabled { |
| return false |
| } |
| if !raceDetectorSupported(goos, goarch) { |
| return false |
| } |
| |
| |
| if isAlpineLinux() { |
| return false |
| } |
| |
| |
| if goos == "netbsd" { |
| return false |
| } |
| return true |
| } |
|
|
| func isAlpineLinux() bool { |
| if runtime.GOOS != "linux" { |
| return false |
| } |
| fi, err := os.Lstat("/etc/alpine-release") |
| return err == nil && fi.Mode().IsRegular() |
| } |
|
|
| func (t *tester) registerRaceTests() { |
| hdr := "Testing race detector" |
| t.registerTest(hdr, |
| &goTest{ |
| variant: "race", |
| race: true, |
| runTests: "Output", |
| pkg: "runtime/race", |
| }) |
| t.registerTest(hdr, |
| &goTest{ |
| variant: "race", |
| race: true, |
| runTests: "TestParse|TestEcho|TestStdinCloseRace|TestClosedPipeRace|TestTypeRace|TestFdRace|TestFdReadRace|TestFileCloseRace", |
| pkgs: []string{"flag", "net", "os", "os/exec", "encoding/gob"}, |
| }) |
| |
| |
| |
| |
| |
| if t.cgoEnabled { |
| |
| |
| |
| |
| |
| } |
| if t.extLink() { |
| |
| t.registerTest(hdr, |
| &goTest{ |
| variant: "race-external", |
| race: true, |
| ldflags: "-linkmode=external", |
| runTests: "TestParse|TestEcho|TestStdinCloseRace", |
| pkgs: []string{"flag", "os/exec"}, |
| }) |
| } |
| } |
|
|
| |
| var cgoPackages = []string{ |
| "net", |
| "os/user", |
| } |
|
|
| var funcBenchmark = []byte("\nfunc Benchmark") |
|
|
| |
| |
| |
| |
| |
| |
| |
| func (t *tester) packageHasBenchmarks(pkg string) bool { |
| pkgDir := filepath.Join(goroot, "src", pkg) |
| d, err := os.Open(pkgDir) |
| if err != nil { |
| return true |
| } |
| defer d.Close() |
| names, err := d.Readdirnames(-1) |
| if err != nil { |
| return true |
| } |
| for _, name := range names { |
| if !strings.HasSuffix(name, "_test.go") { |
| continue |
| } |
| slurp, err := os.ReadFile(filepath.Join(pkgDir, name)) |
| if err != nil { |
| return true |
| } |
| if bytes.Contains(slurp, funcBenchmark) { |
| return true |
| } |
| } |
| return false |
| } |
|
|
| |
| |
| func (t *tester) makeGOROOTUnwritable() (undo func()) { |
| dir := os.Getenv("GOROOT") |
| if dir == "" { |
| panic("GOROOT not set") |
| } |
|
|
| type pathMode struct { |
| path string |
| mode os.FileMode |
| } |
| var dirs []pathMode |
|
|
| undo = func() { |
| for i := range dirs { |
| os.Chmod(dirs[i].path, dirs[i].mode) |
| } |
| } |
|
|
| filepath.WalkDir(dir, func(path string, d fs.DirEntry, err error) error { |
| if suffix := strings.TrimPrefix(path, dir+string(filepath.Separator)); suffix != "" { |
| if suffix == ".git" { |
| |
| |
| |
| return filepath.SkipDir |
| } |
| } |
| if err != nil { |
| return nil |
| } |
|
|
| info, err := d.Info() |
| if err != nil { |
| return nil |
| } |
|
|
| mode := info.Mode() |
| if mode&0222 != 0 && (mode.IsDir() || mode.IsRegular()) { |
| dirs = append(dirs, pathMode{path, mode}) |
| } |
| return nil |
| }) |
|
|
| |
| for i := len(dirs) - 1; i >= 0; i-- { |
| err := os.Chmod(dirs[i].path, dirs[i].mode&^0222) |
| if err != nil { |
| dirs = dirs[i:] |
| undo() |
| fatalf("failed to make GOROOT read-only: %v", err) |
| } |
| } |
|
|
| return undo |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| func raceDetectorSupported(goos, goarch string) bool { |
| switch goos { |
| case "linux": |
| return goarch == "amd64" || goarch == "arm64" || goarch == "loong64" || goarch == "ppc64le" || goarch == "riscv64" || goarch == "s390x" |
| case "darwin": |
| return goarch == "amd64" || goarch == "arm64" |
| case "freebsd", "netbsd", "windows": |
| return goarch == "amd64" |
| default: |
| return false |
| } |
| } |
|
|
| |
| |
| |
| func buildModeSupported(compiler, buildmode, goos, goarch string) bool { |
| if compiler == "gccgo" { |
| return true |
| } |
|
|
| platform := goos + "/" + goarch |
|
|
| switch buildmode { |
| case "archive": |
| return true |
|
|
| case "c-archive": |
| switch goos { |
| case "aix", "darwin", "ios", "windows": |
| return true |
| case "linux": |
| switch goarch { |
| case "386", "amd64", "arm", "armbe", "arm64", "arm64be", "loong64", "ppc64le", "riscv64", "s390x": |
| |
| |
| return true |
| default: |
| |
| |
| |
| |
| |
| |
| return false |
| } |
| case "freebsd": |
| return goarch == "amd64" |
| } |
| return false |
|
|
| case "c-shared": |
| switch platform { |
| case "linux/amd64", "linux/arm", "linux/arm64", "linux/loong64", "linux/386", "linux/ppc64le", "linux/riscv64", "linux/s390x", |
| "android/amd64", "android/arm", "android/arm64", "android/386", |
| "freebsd/amd64", |
| "darwin/amd64", "darwin/arm64", |
| "windows/amd64", "windows/386", "windows/arm64", |
| "wasip1/wasm": |
| return true |
| } |
| return false |
|
|
| case "default": |
| return true |
|
|
| case "exe": |
| return true |
|
|
| case "pie": |
| switch platform { |
| case "linux/386", "linux/amd64", "linux/arm", "linux/arm64", "linux/loong64", "linux/ppc64le", "linux/riscv64", "linux/s390x", |
| "android/amd64", "android/arm", "android/arm64", "android/386", |
| "freebsd/amd64", |
| "darwin/amd64", "darwin/arm64", |
| "ios/amd64", "ios/arm64", |
| "aix/ppc64", |
| "openbsd/arm64", |
| "windows/386", "windows/amd64", "windows/arm64": |
| return true |
| } |
| return false |
|
|
| case "shared": |
| switch platform { |
| case "linux/386", "linux/amd64", "linux/arm", "linux/arm64", "linux/ppc64le", "linux/s390x": |
| return true |
| } |
| return false |
|
|
| case "plugin": |
| switch platform { |
| case "linux/amd64", "linux/arm", "linux/arm64", "linux/386", "linux/loong64", "linux/riscv64", "linux/s390x", "linux/ppc64le", |
| "android/amd64", "android/386", |
| "darwin/amd64", "darwin/arm64", |
| "freebsd/amd64": |
| return true |
| } |
| return false |
|
|
| default: |
| return false |
| } |
| } |
|
|
| |
| |
| |
| func isUnsupportedVMASize(w *work) bool { |
| unsupportedVMA := []byte("unsupported VMA range") |
| return strings.Contains(w.dt.name, ":race") && bytes.Contains(w.out.Bytes(), unsupportedVMA) |
| } |
|
|
| |
| |
| func isEnvSet(evar string) bool { |
| evarEq := evar + "=" |
| for _, e := range os.Environ() { |
| if strings.HasPrefix(e, evarEq) { |
| return true |
| } |
| } |
| return false |
| } |
|
|
| func (t *tester) fipsSupported() bool { |
| |
|
|
| |
|
|
| |
| if strings.Contains(goexperiment, "boringcrypto") { |
| return false |
| } |
|
|
| |
| |
| |
| |
| switch { |
| case goarch == "wasm", |
| goos == "windows" && goarch == "386", |
| goos == "openbsd", |
| goos == "aix": |
| return false |
| } |
|
|
| |
| |
| if t.asan { |
| return false |
| } |
|
|
| return true |
| } |
|
|
| |
| func fipsVersions(short bool) []string { |
| var versions []string |
| zips, err := filepath.Glob(filepath.Join(goroot, "lib/fips140/*.zip")) |
| if err != nil { |
| fatalf("%v", err) |
| } |
| for _, zip := range zips { |
| versions = append(versions, strings.TrimSuffix(filepath.Base(zip), ".zip")) |
| } |
| txts, err := filepath.Glob(filepath.Join(goroot, "lib/fips140/*.txt")) |
| if err != nil { |
| fatalf("%v", err) |
| } |
| for _, txt := range txts { |
| versions = append(versions, strings.TrimSuffix(filepath.Base(txt), ".txt")) |
| } |
| return versions |
| } |
|
|
| |
| |
| |
| |
| func goexperiments(exps ...string) string { |
| if len(exps) == 0 { |
| return goexperiment |
| } |
| existing := goexperiment |
| if existing != "" { |
| existing += "," |
| } |
| return existing + strings.Join(exps, ",") |
|
|
| } |
|
|