mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
runtime: use pipe rather than note in TestSignalM
At least on Darwin notewakeup is not async-signal-safe. Fixes #35276 Change-Id: I1d7523715e8e77dbd7f21d9b1ed131e52d46cc41 Reviewed-on: https://go-review.googlesource.com/c/go/+/206078 Reviewed-by: Austin Clements <austin@google.com>
This commit is contained in:
parent
d22b5735e7
commit
99957b6930
2 changed files with 53 additions and 12 deletions
|
|
@ -18,6 +18,7 @@ import (
|
||||||
"sync"
|
"sync"
|
||||||
"syscall"
|
"syscall"
|
||||||
"testing"
|
"testing"
|
||||||
|
"time"
|
||||||
"unsafe"
|
"unsafe"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
@ -308,20 +309,45 @@ func TestSignalDuringExec(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestSignalM(t *testing.T) {
|
func TestSignalM(t *testing.T) {
|
||||||
|
r, w, errno := runtime.Pipe()
|
||||||
|
if errno != 0 {
|
||||||
|
t.Fatal(syscall.Errno(errno))
|
||||||
|
}
|
||||||
|
defer func() {
|
||||||
|
runtime.Close(r)
|
||||||
|
runtime.Close(w)
|
||||||
|
}()
|
||||||
|
runtime.Closeonexec(r)
|
||||||
|
runtime.Closeonexec(w)
|
||||||
|
|
||||||
var want, got int64
|
var want, got int64
|
||||||
var wg sync.WaitGroup
|
var wg sync.WaitGroup
|
||||||
ready := make(chan *runtime.M)
|
ready := make(chan *runtime.M)
|
||||||
wg.Add(1)
|
wg.Add(1)
|
||||||
go func() {
|
go func() {
|
||||||
runtime.LockOSThread()
|
runtime.LockOSThread()
|
||||||
want, got = runtime.WaitForSigusr1(func(mp *runtime.M) {
|
var errno int32
|
||||||
|
want, got = runtime.WaitForSigusr1(r, w, func(mp *runtime.M) {
|
||||||
ready <- mp
|
ready <- mp
|
||||||
}, 1e9)
|
})
|
||||||
|
if errno != 0 {
|
||||||
|
t.Error(syscall.Errno(errno))
|
||||||
|
}
|
||||||
runtime.UnlockOSThread()
|
runtime.UnlockOSThread()
|
||||||
wg.Done()
|
wg.Done()
|
||||||
}()
|
}()
|
||||||
waitingM := <-ready
|
waitingM := <-ready
|
||||||
runtime.SendSigusr1(waitingM)
|
runtime.SendSigusr1(waitingM)
|
||||||
|
|
||||||
|
timer := time.AfterFunc(time.Second, func() {
|
||||||
|
// Write 1 to tell WaitForSigusr1 that we timed out.
|
||||||
|
bw := byte(1)
|
||||||
|
if n := runtime.Write(uintptr(w), unsafe.Pointer(&bw), 1); n != 1 {
|
||||||
|
t.Errorf("pipe write failed: %d", n)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
defer timer.Stop()
|
||||||
|
|
||||||
wg.Wait()
|
wg.Wait()
|
||||||
if got == -1 {
|
if got == -1 {
|
||||||
t.Fatal("signalM signal not received")
|
t.Fatal("signalM signal not received")
|
||||||
|
|
|
||||||
|
|
@ -6,6 +6,8 @@
|
||||||
|
|
||||||
package runtime
|
package runtime
|
||||||
|
|
||||||
|
import "unsafe"
|
||||||
|
|
||||||
var NonblockingPipe = nonblockingPipe
|
var NonblockingPipe = nonblockingPipe
|
||||||
var Pipe = pipe
|
var Pipe = pipe
|
||||||
var SetNonblock = setNonblock
|
var SetNonblock = setNonblock
|
||||||
|
|
@ -26,33 +28,45 @@ func Sigisblocked(i int) bool {
|
||||||
type M = m
|
type M = m
|
||||||
|
|
||||||
var waitForSigusr1 struct {
|
var waitForSigusr1 struct {
|
||||||
park note
|
rdpipe int32
|
||||||
|
wrpipe int32
|
||||||
mID int64
|
mID int64
|
||||||
}
|
}
|
||||||
|
|
||||||
// WaitForSigusr1 blocks until a SIGUSR1 is received. It calls ready
|
// WaitForSigusr1 blocks until a SIGUSR1 is received. It calls ready
|
||||||
// when it is set up to receive SIGUSR1. The ready function should
|
// when it is set up to receive SIGUSR1. The ready function should
|
||||||
// cause a SIGUSR1 to be sent.
|
// cause a SIGUSR1 to be sent. The r and w arguments are a pipe that
|
||||||
|
// the signal handler can use to report when the signal is received.
|
||||||
//
|
//
|
||||||
// Once SIGUSR1 is received, it returns the ID of the current M and
|
// Once SIGUSR1 is received, it returns the ID of the current M and
|
||||||
// the ID of the M the SIGUSR1 was received on. If no SIGUSR1 is
|
// the ID of the M the SIGUSR1 was received on. If the caller writes
|
||||||
// received for timeoutNS nanoseconds, it returns -1.
|
// a non-zero byte to w, WaitForSigusr1 returns immediately with -1, -1.
|
||||||
func WaitForSigusr1(ready func(mp *M), timeoutNS int64) (int64, int64) {
|
func WaitForSigusr1(r, w int32, ready func(mp *M)) (int64, int64) {
|
||||||
lockOSThread()
|
lockOSThread()
|
||||||
// Make sure we can receive SIGUSR1.
|
// Make sure we can receive SIGUSR1.
|
||||||
unblocksig(_SIGUSR1)
|
unblocksig(_SIGUSR1)
|
||||||
|
|
||||||
|
waitForSigusr1.rdpipe = r
|
||||||
|
waitForSigusr1.wrpipe = w
|
||||||
|
|
||||||
mp := getg().m
|
mp := getg().m
|
||||||
testSigusr1 = waitForSigusr1Callback
|
testSigusr1 = waitForSigusr1Callback
|
||||||
ready(mp)
|
ready(mp)
|
||||||
ok := notetsleepg(&waitForSigusr1.park, timeoutNS)
|
|
||||||
noteclear(&waitForSigusr1.park)
|
// Wait for the signal. We use a pipe rather than a note
|
||||||
|
// because write is always async-signal-safe.
|
||||||
|
entersyscallblock()
|
||||||
|
var b byte
|
||||||
|
read(waitForSigusr1.rdpipe, noescape(unsafe.Pointer(&b)), 1)
|
||||||
|
exitsyscall()
|
||||||
|
|
||||||
gotM := waitForSigusr1.mID
|
gotM := waitForSigusr1.mID
|
||||||
testSigusr1 = nil
|
testSigusr1 = nil
|
||||||
|
|
||||||
unlockOSThread()
|
unlockOSThread()
|
||||||
|
|
||||||
if !ok {
|
if b != 0 {
|
||||||
|
// timeout signal from caller
|
||||||
return -1, -1
|
return -1, -1
|
||||||
}
|
}
|
||||||
return mp.id, gotM
|
return mp.id, gotM
|
||||||
|
|
@ -69,7 +83,8 @@ func waitForSigusr1Callback(gp *g) bool {
|
||||||
} else {
|
} else {
|
||||||
waitForSigusr1.mID = gp.m.id
|
waitForSigusr1.mID = gp.m.id
|
||||||
}
|
}
|
||||||
notewakeup(&waitForSigusr1.park)
|
b := byte(0)
|
||||||
|
write(uintptr(waitForSigusr1.wrpipe), noescape(unsafe.Pointer(&b)), 1)
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue