mirror of
git://gcc.gnu.org/git/gcc.git
synced 2024-12-13 00:01:03 +08:00
8dc2499aa6
gotools/ * Makefile.am (go_cmd_cgo_files): Add ast_go118.go (check-go-tool): Copy golang.org/x/tools directories. * Makefile.in: Regenerate. Reviewed-on: https://go-review.googlesource.com/c/gofrontend/+/384695
200 lines
4.9 KiB
Go
200 lines
4.9 KiB
Go
// Copyright 2016 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
// Package testdeps provides access to dependencies needed by test execution.
|
|
//
|
|
// This package is imported by the generated main package, which passes
|
|
// TestDeps into testing.Main. This allows tests to use packages at run time
|
|
// without making those packages direct dependencies of package testing.
|
|
// Direct dependencies of package testing are harder to write tests for.
|
|
package testdeps
|
|
|
|
import (
|
|
"bufio"
|
|
"context"
|
|
"internal/fuzz"
|
|
"internal/testlog"
|
|
"io"
|
|
"os"
|
|
"os/signal"
|
|
"reflect"
|
|
"regexp"
|
|
"runtime/pprof"
|
|
"strings"
|
|
"sync"
|
|
"time"
|
|
)
|
|
|
|
// TestDeps is an implementation of the testing.testDeps interface,
|
|
// suitable for passing to testing.MainStart.
|
|
type TestDeps struct{}
|
|
|
|
var matchPat string
|
|
var matchRe *regexp.Regexp
|
|
|
|
func (TestDeps) MatchString(pat, str string) (result bool, err error) {
|
|
if matchRe == nil || matchPat != pat {
|
|
matchPat = pat
|
|
matchRe, err = regexp.Compile(matchPat)
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
return matchRe.MatchString(str), nil
|
|
}
|
|
|
|
func (TestDeps) StartCPUProfile(w io.Writer) error {
|
|
return pprof.StartCPUProfile(w)
|
|
}
|
|
|
|
func (TestDeps) StopCPUProfile() {
|
|
pprof.StopCPUProfile()
|
|
}
|
|
|
|
func (TestDeps) WriteProfileTo(name string, w io.Writer, debug int) error {
|
|
return pprof.Lookup(name).WriteTo(w, debug)
|
|
}
|
|
|
|
// ImportPath is the import path of the testing binary, set by the generated main function.
|
|
var ImportPath string
|
|
|
|
func (TestDeps) ImportPath() string {
|
|
return ImportPath
|
|
}
|
|
|
|
// testLog implements testlog.Interface, logging actions by package os.
|
|
type testLog struct {
|
|
mu sync.Mutex
|
|
w *bufio.Writer
|
|
set bool
|
|
}
|
|
|
|
func (l *testLog) Getenv(key string) {
|
|
l.add("getenv", key)
|
|
}
|
|
|
|
func (l *testLog) Open(name string) {
|
|
l.add("open", name)
|
|
}
|
|
|
|
func (l *testLog) Stat(name string) {
|
|
l.add("stat", name)
|
|
}
|
|
|
|
func (l *testLog) Chdir(name string) {
|
|
l.add("chdir", name)
|
|
}
|
|
|
|
// add adds the (op, name) pair to the test log.
|
|
func (l *testLog) add(op, name string) {
|
|
if strings.Contains(name, "\n") || name == "" {
|
|
return
|
|
}
|
|
|
|
l.mu.Lock()
|
|
defer l.mu.Unlock()
|
|
if l.w == nil {
|
|
return
|
|
}
|
|
l.w.WriteString(op)
|
|
l.w.WriteByte(' ')
|
|
l.w.WriteString(name)
|
|
l.w.WriteByte('\n')
|
|
}
|
|
|
|
var log testLog
|
|
|
|
func (TestDeps) StartTestLog(w io.Writer) {
|
|
log.mu.Lock()
|
|
log.w = bufio.NewWriter(w)
|
|
if !log.set {
|
|
// Tests that define TestMain and then run m.Run multiple times
|
|
// will call StartTestLog/StopTestLog multiple times.
|
|
// Checking log.set avoids calling testlog.SetLogger multiple times
|
|
// (which will panic) and also avoids writing the header multiple times.
|
|
log.set = true
|
|
testlog.SetLogger(&log)
|
|
log.w.WriteString("# test log\n") // known to cmd/go/internal/test/test.go
|
|
}
|
|
log.mu.Unlock()
|
|
}
|
|
|
|
func (TestDeps) StopTestLog() error {
|
|
log.mu.Lock()
|
|
defer log.mu.Unlock()
|
|
err := log.w.Flush()
|
|
log.w = nil
|
|
return err
|
|
}
|
|
|
|
// SetPanicOnExit0 tells the os package whether to panic on os.Exit(0).
|
|
func (TestDeps) SetPanicOnExit0(v bool) {
|
|
testlog.SetPanicOnExit0(v)
|
|
}
|
|
|
|
func (TestDeps) CoordinateFuzzing(
|
|
timeout time.Duration,
|
|
limit int64,
|
|
minimizeTimeout time.Duration,
|
|
minimizeLimit int64,
|
|
parallel int,
|
|
seed []fuzz.CorpusEntry,
|
|
types []reflect.Type,
|
|
corpusDir,
|
|
cacheDir string) (err error) {
|
|
// Fuzzing may be interrupted with a timeout or if the user presses ^C.
|
|
// In either case, we'll stop worker processes gracefully and save
|
|
// crashers and interesting values.
|
|
ctx, cancel := signal.NotifyContext(context.Background(), os.Interrupt)
|
|
defer cancel()
|
|
err = fuzz.CoordinateFuzzing(ctx, fuzz.CoordinateFuzzingOpts{
|
|
Log: os.Stderr,
|
|
Timeout: timeout,
|
|
Limit: limit,
|
|
MinimizeTimeout: minimizeTimeout,
|
|
MinimizeLimit: minimizeLimit,
|
|
Parallel: parallel,
|
|
Seed: seed,
|
|
Types: types,
|
|
CorpusDir: corpusDir,
|
|
CacheDir: cacheDir,
|
|
})
|
|
if err == ctx.Err() {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (TestDeps) RunFuzzWorker(fn func(fuzz.CorpusEntry) error) error {
|
|
// Worker processes may or may not receive a signal when the user presses ^C
|
|
// On POSIX operating systems, a signal sent to a process group is delivered
|
|
// to all processes in that group. This is not the case on Windows.
|
|
// If the worker is interrupted, return quickly and without error.
|
|
// If only the coordinator process is interrupted, it tells each worker
|
|
// process to stop by closing its "fuzz_in" pipe.
|
|
ctx, cancel := signal.NotifyContext(context.Background(), os.Interrupt)
|
|
defer cancel()
|
|
err := fuzz.RunFuzzWorker(ctx, fn)
|
|
if err == ctx.Err() {
|
|
return nil
|
|
}
|
|
return err
|
|
}
|
|
|
|
func (TestDeps) ReadCorpus(dir string, types []reflect.Type) ([]fuzz.CorpusEntry, error) {
|
|
return fuzz.ReadCorpus(dir, types)
|
|
}
|
|
|
|
func (TestDeps) CheckCorpus(vals []any, types []reflect.Type) error {
|
|
return fuzz.CheckCorpus(vals, types)
|
|
}
|
|
|
|
func (TestDeps) ResetCoverage() {
|
|
fuzz.ResetCoverage()
|
|
}
|
|
|
|
func (TestDeps) SnapshotCoverage() {
|
|
fuzz.SnapshotCoverage()
|
|
}
|