mirror of
https://github.com/golang/go
synced 2024-11-11 18:21:40 -07:00
os/signal: fix a deadlock with syscall.AllThreadsSyscall() use
The syscall.AllThreadsSyscall() fixup mechanism needs to cooperate with signal handling to ensure a notetsleepg() thread can wake up to run the mDoFixup() function. Fixes #43149 Change-Id: I6651b25bc44a4de47d3fb71d0293d51aef8b79c7 Reviewed-on: https://go-review.googlesource.com/c/go/+/277434 Run-TryBot: Ian Lance Taylor <iant@golang.org> TryBot-Result: Go Bot <gobot@golang.org> Trust: Austin Clements <austin@google.com> Reviewed-by: Ian Lance Taylor <iant@golang.org>
This commit is contained in:
parent
b0b0d98283
commit
fd6ba1c8a2
42
src/os/signal/signal_linux_test.go
Normal file
42
src/os/signal/signal_linux_test.go
Normal file
@ -0,0 +1,42 @@
|
||||
// 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.
|
||||
|
||||
// +build linux
|
||||
|
||||
package signal
|
||||
|
||||
import (
|
||||
"os"
|
||||
"syscall"
|
||||
"testing"
|
||||
"time"
|
||||
)
|
||||
|
||||
const prSetKeepCaps = 8
|
||||
|
||||
// This test validates that syscall.AllThreadsSyscall() can reliably
|
||||
// reach all 'm' (threads) of the nocgo runtime even when one thread
|
||||
// is blocked waiting to receive signals from the kernel. This monitors
|
||||
// for a regression vs. the fix for #43149.
|
||||
func TestAllThreadsSyscallSignals(t *testing.T) {
|
||||
if _, _, err := syscall.AllThreadsSyscall(syscall.SYS_PRCTL, prSetKeepCaps, 0, 0); err == syscall.ENOTSUP {
|
||||
t.Skip("AllThreadsSyscall disabled with cgo")
|
||||
}
|
||||
|
||||
sig := make(chan os.Signal, 1)
|
||||
Notify(sig, os.Interrupt)
|
||||
|
||||
for i := 0; i <= 100; i++ {
|
||||
if _, _, errno := syscall.AllThreadsSyscall(syscall.SYS_PRCTL, prSetKeepCaps, uintptr(i&1), 0); errno != 0 {
|
||||
t.Fatalf("[%d] failed to set KEEP_CAPS=%d: %v", i, i&1, errno)
|
||||
}
|
||||
}
|
||||
|
||||
select {
|
||||
case <-time.After(10 * time.Millisecond):
|
||||
case <-sig:
|
||||
t.Fatal("unexpected signal")
|
||||
}
|
||||
Stop(sig)
|
||||
}
|
@ -1515,6 +1515,7 @@ func syscall_runtime_doAllThreadsSyscall(fn func(bool) bool) {
|
||||
if netpollinited() {
|
||||
netpollBreak()
|
||||
}
|
||||
sigRecvPrepareForFixup()
|
||||
_g_ := getg()
|
||||
if raceenabled {
|
||||
// For m's running without racectx, we loan out the
|
||||
|
@ -12,12 +12,16 @@
|
||||
// sigsend is called by the signal handler to queue a new signal.
|
||||
// signal_recv is called by the Go program to receive a newly queued signal.
|
||||
// Synchronization between sigsend and signal_recv is based on the sig.state
|
||||
// variable. It can be in 3 states: sigIdle, sigReceiving and sigSending.
|
||||
// variable. It can be in 4 states: sigIdle, sigReceiving, sigSending and sigFixup.
|
||||
// sigReceiving means that signal_recv is blocked on sig.Note and there are no
|
||||
// new pending signals.
|
||||
// sigSending means that sig.mask *may* contain new pending signals,
|
||||
// signal_recv can't be blocked in this state.
|
||||
// sigIdle means that there are no new pending signals and signal_recv is not blocked.
|
||||
// sigFixup is a transient state that can only exist as a short
|
||||
// transition from sigReceiving and then on to sigIdle: it is
|
||||
// used to ensure the AllThreadsSyscall()'s mDoFixup() operation
|
||||
// occurs on the sleeping m, waiting to receive a signal.
|
||||
// Transitions between states are done atomically with CAS.
|
||||
// When signal_recv is unblocked, it resets sig.Note and rechecks sig.mask.
|
||||
// If several sigsends and signal_recv execute concurrently, it can lead to
|
||||
@ -59,6 +63,7 @@ const (
|
||||
sigIdle = iota
|
||||
sigReceiving
|
||||
sigSending
|
||||
sigFixup
|
||||
)
|
||||
|
||||
// sigsend delivers a signal from sighandler to the internal signal delivery queue.
|
||||
@ -112,6 +117,9 @@ Send:
|
||||
notewakeup(&sig.note)
|
||||
break Send
|
||||
}
|
||||
case sigFixup:
|
||||
// nothing to do - we need to wait for sigIdle.
|
||||
osyield()
|
||||
}
|
||||
}
|
||||
|
||||
@ -119,6 +127,19 @@ Send:
|
||||
return true
|
||||
}
|
||||
|
||||
// sigRecvPrepareForFixup is used to temporarily wake up the
|
||||
// signal_recv() running thread while it is blocked waiting for the
|
||||
// arrival of a signal. If it causes the thread to wake up, the
|
||||
// sig.state travels through this sequence: sigReceiving -> sigFixup
|
||||
// -> sigIdle -> sigReceiving and resumes. (This is only called while
|
||||
// GC is disabled.)
|
||||
//go:nosplit
|
||||
func sigRecvPrepareForFixup() {
|
||||
if atomic.Cas(&sig.state, sigReceiving, sigFixup) {
|
||||
notewakeup(&sig.note)
|
||||
}
|
||||
}
|
||||
|
||||
// Called to receive the next queued signal.
|
||||
// Must only be called from a single goroutine at a time.
|
||||
//go:linkname signal_recv os/signal.signal_recv
|
||||
@ -146,7 +167,16 @@ func signal_recv() uint32 {
|
||||
}
|
||||
notetsleepg(&sig.note, -1)
|
||||
noteclear(&sig.note)
|
||||
break Receive
|
||||
if !atomic.Cas(&sig.state, sigFixup, sigIdle) {
|
||||
break Receive
|
||||
}
|
||||
// Getting here, the code will
|
||||
// loop around again to sleep
|
||||
// in state sigReceiving. This
|
||||
// path is taken when
|
||||
// sigRecvPrepareForFixup()
|
||||
// has been called by another
|
||||
// thread.
|
||||
}
|
||||
case sigSending:
|
||||
if atomic.Cas(&sig.state, sigSending, sigIdle) {
|
||||
|
@ -92,6 +92,13 @@ func sendNote(s *byte) bool {
|
||||
return true
|
||||
}
|
||||
|
||||
// sigRecvPrepareForFixup is a no-op on plan9. (This would only be
|
||||
// called while GC is disabled.)
|
||||
//
|
||||
//go:nosplit
|
||||
func sigRecvPrepareForFixup() {
|
||||
}
|
||||
|
||||
// Called to receive the next queued signal.
|
||||
// Must only be called from a single goroutine at a time.
|
||||
//go:linkname signal_recv os/signal.signal_recv
|
||||
|
Loading…
Reference in New Issue
Block a user