2016-04-10 15:32:26 -06:00
|
|
|
// Copyright 2014 The Go Authors. All rights reserved.
|
2014-12-19 14:16:17 -07:00
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
// Test that SIGSETXID runs on signal stack, since it's likely to
|
|
|
|
// overflow if it runs on the Go stack.
|
|
|
|
|
|
|
|
package cgotest
|
|
|
|
|
|
|
|
/*
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
*/
|
|
|
|
import "C"
|
|
|
|
|
|
|
|
import (
|
|
|
|
"runtime"
|
2021-11-19 15:33:42 -07:00
|
|
|
"runtime/debug"
|
2014-12-19 14:16:17 -07:00
|
|
|
"sync/atomic"
|
|
|
|
"testing"
|
|
|
|
|
2019-02-22 08:17:22 -07:00
|
|
|
"cgotest/issue9400"
|
2014-12-19 14:16:17 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
func test9400(t *testing.T) {
|
|
|
|
// We synchronize through a shared variable, so we need two procs
|
|
|
|
defer runtime.GOMAXPROCS(runtime.GOMAXPROCS(2))
|
|
|
|
|
|
|
|
// Start signaller
|
|
|
|
atomic.StoreInt32(&issue9400.Baton, 0)
|
|
|
|
go func() {
|
|
|
|
// Wait for RewindAndSetgid
|
|
|
|
for atomic.LoadInt32(&issue9400.Baton) == 0 {
|
|
|
|
runtime.Gosched()
|
|
|
|
}
|
|
|
|
// Broadcast SIGSETXID
|
|
|
|
runtime.LockOSThread()
|
|
|
|
C.setgid(0)
|
|
|
|
// Indicate that signalling is done
|
|
|
|
atomic.StoreInt32(&issue9400.Baton, 0)
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Grow the stack and put down a test pattern
|
|
|
|
const pattern = 0x123456789abcdef
|
2018-08-22 23:06:47 -06:00
|
|
|
var big [1024]uint64 // len must match assembly
|
2014-12-19 14:16:17 -07:00
|
|
|
for i := range big {
|
|
|
|
big[i] = pattern
|
|
|
|
}
|
|
|
|
|
2021-11-19 15:33:42 -07:00
|
|
|
// Disable GC for the duration of the test.
|
|
|
|
// This avoids a potential GC deadlock when spinning in uninterruptable ASM below #49695.
|
|
|
|
defer debug.SetGCPercent(debug.SetGCPercent(-1))
|
runtime: fix comments on the behavior of SetGCPercent
Fixes for #49680, #49695, #45867, and #49370 all assumed that
SetGCPercent(-1) doesn't block until the GC's mark phase is done, but
it actually does. The cause of 3 of those 4 failures comes from the fact
that at the beginning of the sweep phase, the GC does try to preempt
every P once, and this may run concurrently with test code. In the
fourth case, the issue was likely that only *one* of the debug_test.go
tests was missing a call to SetGCPercent(-1). Just to be safe, leave a
TODO there for now to remove the extraneous runtime.GC calls, but leave
the calls in.
Updates #49680, #49695, #45867, and #49370.
Change-Id: Ibf4e64addfba18312526968bcf40f1f5d54eb3f1
Reviewed-on: https://go-review.googlesource.com/c/go/+/369815
Reviewed-by: Austin Clements <austin@google.com>
Trust: Michael Knyszek <mknyszek@google.com>
Run-TryBot: Michael Knyszek <mknyszek@google.com>
TryBot-Result: Gopher Robot <gobot@golang.org>
2021-12-06 22:24:54 -07:00
|
|
|
// SetGCPercent waits until the mark phase is over, but the runtime
|
|
|
|
// also preempts at the start of the sweep phase, so make sure that's
|
|
|
|
// done too. See #49695.
|
2021-11-19 15:33:42 -07:00
|
|
|
runtime.GC()
|
2021-11-19 15:33:42 -07:00
|
|
|
|
2014-12-19 14:16:17 -07:00
|
|
|
// Temporarily rewind the stack and trigger SIGSETXID
|
|
|
|
issue9400.RewindAndSetgid()
|
|
|
|
|
|
|
|
// Check test pattern
|
|
|
|
for i := range big {
|
|
|
|
if big[i] != pattern {
|
2014-12-22 19:18:09 -07:00
|
|
|
t.Fatalf("entry %d of test pattern is wrong; %#x != %#x", i, big[i], uint64(pattern))
|
2014-12-19 14:16:17 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|