1
0
mirror of https://github.com/golang/go synced 2024-11-13 20:10:32 -07:00
go/src/net/tcpsock_unix_test.go
Ian Lance Taylor bf0f692202 net: handle spurious netpoll wakeups in connect
In some cases the netpoll code can cause a spurious wakeup. This is
normally harmless, as the woken up code simply retries the operation.
However, for connect, the test we were using to see whether the
connect had succeeded (setsockopt(SO_ERROR)) was not reliable in the
case of a spurious wakeup.  Change to using a reliable test (getpeername).
On Darwin we used a different technique: a second call to connect;
change Darwin to use getpeername as well.

Return the result of getpeername to avoid having to call it twice.

Fixes #19289.

Change-Id: I119ec8e7a41f482f1e590d4c65a37f6103fa22d9
Reviewed-on: https://go-review.googlesource.com/45815
Run-TryBot: Ian Lance Taylor <iant@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Brad Fitzpatrick <bradfitz@golang.org>
2017-06-15 22:39:39 +00:00

117 lines
2.4 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.
// +build !plan9,!windows
package net
import (
"context"
"internal/testenv"
"math/rand"
"runtime"
"sync"
"syscall"
"testing"
"time"
)
// See golang.org/issue/14548.
func TestTCPSpuriousConnSetupCompletion(t *testing.T) {
if testing.Short() {
t.Skip("skipping in short mode")
}
ln, err := newLocalListener("tcp")
if err != nil {
t.Fatal(err)
}
var wg sync.WaitGroup
wg.Add(1)
go func(ln Listener) {
defer wg.Done()
for {
c, err := ln.Accept()
if err != nil {
return
}
wg.Add(1)
go func(c Conn) {
var b [1]byte
c.Read(b[:])
c.Close()
wg.Done()
}(c)
}
}(ln)
attempts := int(1e4) // larger is better
wg.Add(attempts)
throttle := make(chan struct{}, runtime.GOMAXPROCS(-1)*2)
for i := 0; i < attempts; i++ {
throttle <- struct{}{}
go func(i int) {
defer func() {
<-throttle
wg.Done()
}()
d := Dialer{Timeout: 50 * time.Millisecond}
c, err := d.Dial(ln.Addr().Network(), ln.Addr().String())
if err != nil {
if perr := parseDialError(err); perr != nil {
t.Errorf("#%d: %v (original error: %v)", i, perr, err)
}
return
}
var b [1]byte
if _, err := c.Write(b[:]); err != nil {
if perr := parseWriteError(err); perr != nil {
t.Errorf("#%d: %v", i, err)
}
if samePlatformError(err, syscall.ENOTCONN) {
t.Errorf("#%d: %v", i, err)
}
}
c.Close()
}(i)
}
ln.Close()
wg.Wait()
}
// Issue 19289.
// Test that a canceled Dial does not cause a subsequent Dial to succeed.
func TestTCPSpuriousConnSetupCompletionWithCancel(t *testing.T) {
if testenv.Builder() == "" {
testenv.MustHaveExternalNetwork(t)
}
t.Parallel()
const tries = 10000
var wg sync.WaitGroup
wg.Add(tries * 2)
sem := make(chan bool, 5)
for i := 0; i < tries; i++ {
sem <- true
ctx, cancel := context.WithCancel(context.Background())
go func() {
defer wg.Done()
time.Sleep(time.Duration(rand.Int63n(int64(5 * time.Millisecond))))
cancel()
}()
go func(i int) {
defer wg.Done()
var dialer Dialer
// Try to connect to a real host on a port
// that it is not listening on.
_, err := dialer.DialContext(ctx, "tcp", "golang.org:3")
if err == nil {
t.Errorf("Dial to unbound port succeeded on attempt %d", i)
}
<-sem
}(i)
}
wg.Wait()
}