mirror of
git://gcc.gnu.org/git/gcc.git
synced 2024-12-12 11:40:13 +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
131 lines
4.1 KiB
Go
131 lines
4.1 KiB
Go
// Copyright 2020 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.
|
|
|
|
//go:build darwin || freebsd || linux
|
|
|
|
package fuzz
|
|
|
|
import (
|
|
"fmt"
|
|
"os"
|
|
"os/exec"
|
|
"syscall"
|
|
)
|
|
|
|
type sharedMemSys struct{}
|
|
|
|
func sharedMemMapFile(f *os.File, size int, removeOnClose bool) (*sharedMem, error) {
|
|
prot := syscall.PROT_READ | syscall.PROT_WRITE
|
|
flags := syscall.MAP_FILE | syscall.MAP_SHARED
|
|
region, err := syscall.Mmap(int(f.Fd()), 0, size, prot, flags)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return &sharedMem{f: f, region: region, removeOnClose: removeOnClose}, nil
|
|
}
|
|
|
|
// Close unmaps the shared memory and closes the temporary file. If this
|
|
// sharedMem was created with sharedMemTempFile, Close also removes the file.
|
|
func (m *sharedMem) Close() error {
|
|
// Attempt all operations, even if we get an error for an earlier operation.
|
|
// os.File.Close may fail due to I/O errors, but we still want to delete
|
|
// the temporary file.
|
|
var errs []error
|
|
errs = append(errs,
|
|
syscall.Munmap(m.region),
|
|
m.f.Close())
|
|
if m.removeOnClose {
|
|
errs = append(errs, os.Remove(m.f.Name()))
|
|
}
|
|
for _, err := range errs {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// setWorkerComm configures communication channels on the cmd that will
|
|
// run a worker process.
|
|
func setWorkerComm(cmd *exec.Cmd, comm workerComm) {
|
|
mem := <-comm.memMu
|
|
memFile := mem.f
|
|
comm.memMu <- mem
|
|
cmd.ExtraFiles = []*os.File{comm.fuzzIn, comm.fuzzOut, memFile}
|
|
}
|
|
|
|
// getWorkerComm returns communication channels in the worker process.
|
|
func getWorkerComm() (comm workerComm, err error) {
|
|
fuzzIn := os.NewFile(3, "fuzz_in")
|
|
fuzzOut := os.NewFile(4, "fuzz_out")
|
|
memFile := os.NewFile(5, "fuzz_mem")
|
|
fi, err := memFile.Stat()
|
|
if err != nil {
|
|
return workerComm{}, err
|
|
}
|
|
size := int(fi.Size())
|
|
if int64(size) != fi.Size() {
|
|
return workerComm{}, fmt.Errorf("fuzz temp file exceeds maximum size")
|
|
}
|
|
removeOnClose := false
|
|
mem, err := sharedMemMapFile(memFile, size, removeOnClose)
|
|
if err != nil {
|
|
return workerComm{}, err
|
|
}
|
|
memMu := make(chan *sharedMem, 1)
|
|
memMu <- mem
|
|
return workerComm{fuzzIn: fuzzIn, fuzzOut: fuzzOut, memMu: memMu}, nil
|
|
}
|
|
|
|
// isInterruptError returns whether an error was returned by a process that
|
|
// was terminated by an interrupt signal (SIGINT).
|
|
func isInterruptError(err error) bool {
|
|
exitErr, ok := err.(*exec.ExitError)
|
|
if !ok || exitErr.ExitCode() >= 0 {
|
|
return false
|
|
}
|
|
status := exitErr.Sys().(syscall.WaitStatus)
|
|
return status.Signal() == syscall.SIGINT
|
|
}
|
|
|
|
// terminationSignal checks if err is an exec.ExitError with a signal status.
|
|
// If it is, terminationSignal returns the signal and true.
|
|
// If not, -1 and false.
|
|
func terminationSignal(err error) (os.Signal, bool) {
|
|
exitErr, ok := err.(*exec.ExitError)
|
|
if !ok || exitErr.ExitCode() >= 0 {
|
|
return syscall.Signal(-1), false
|
|
}
|
|
status := exitErr.Sys().(syscall.WaitStatus)
|
|
return status.Signal(), status.Signaled()
|
|
}
|
|
|
|
// isCrashSignal returns whether a signal was likely to have been caused by an
|
|
// error in the program that received it, triggered by a fuzz input. For
|
|
// example, SIGSEGV would be received after a nil pointer dereference.
|
|
// Other signals like SIGKILL or SIGHUP are more likely to have been sent by
|
|
// another process, and we shouldn't record a crasher if the worker process
|
|
// receives one of these.
|
|
//
|
|
// Note that Go installs its own signal handlers on startup, so some of these
|
|
// signals may only be received if signal handlers are changed. For example,
|
|
// SIGSEGV is normally transformed into a panic that causes the process to exit
|
|
// with status 2 if not recovered, which we handle as a crash.
|
|
func isCrashSignal(signal os.Signal) bool {
|
|
switch signal {
|
|
case
|
|
syscall.SIGILL, // illegal instruction
|
|
syscall.SIGTRAP, // breakpoint
|
|
syscall.SIGABRT, // abort() called
|
|
syscall.SIGBUS, // invalid memory access (e.g., misaligned address)
|
|
syscall.SIGFPE, // math error, e.g., integer divide by zero
|
|
syscall.SIGSEGV, // invalid memory access (e.g., write to read-only)
|
|
syscall.SIGPIPE: // sent data to closed pipe or socket
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|