Skip to content

Commit ea5825b

Browse files
npat-efaultianlancetaylor
authored andcommittedApr 11, 2018
os: use poller when NewFile is called with a blocking descriptor.
If NewFile is called with a file descriptor that is already set to non-blocking mode, it tries to return a pollable file (one for which SetDeadline methods work) by adding the filedes to the poll/netpoll mechanism. If called with a filedes in blocking mode, it returns a non-pollable file, as it always did. Fixes #22939 Updates #24331 Change-Id: Id54c8be1b83e6d35e14e76d7df0e57a9fd64e176 Reviewed-on: https://go-review.googlesource.com/100077 Run-TryBot: Ian Lance Taylor <iant@golang.org> TryBot-Result: Gobot Gobot <gobot@golang.org> Reviewed-by: Ian Lance Taylor <iant@golang.org>
1 parent ab48574 commit ea5825b

File tree

7 files changed

+114
-4
lines changed

7 files changed

+114
-4
lines changed
 

‎src/go/build/deps_test.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -158,7 +158,7 @@ var pkgDeps = map[string][]string{
158158

159159
"internal/poll": {"L0", "internal/race", "syscall", "time", "unicode/utf16", "unicode/utf8", "internal/syscall/windows"},
160160
"internal/testlog": {"L0"},
161-
"os": {"L1", "os", "syscall", "time", "internal/poll", "internal/syscall/windows", "internal/testlog"},
161+
"os": {"L1", "os", "syscall", "time", "internal/poll", "internal/syscall/windows", "internal/syscall/unix", "internal/testlog"},
162162
"path/filepath": {"L2", "os", "syscall", "internal/syscall/windows"},
163163
"io/ioutil": {"L2", "os", "path/filepath", "time"},
164164
"os/exec": {"L2", "os", "context", "path/filepath", "syscall"},

‎src/internal/syscall/unix/empty.s

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
// Copyright 2018 The Go Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
// This file is here just to make the go tool happy. It allows
6+
// empty function declarations (no function body).
7+
// It is used with "go:linkname".
+23
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
// Copyright 2018 The Go Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
// +build darwin dragonfly freebsd linux netbsd openbsd solaris
6+
7+
package unix
8+
9+
import (
10+
"syscall"
11+
_ "unsafe" // for go:linkname
12+
)
13+
14+
//go:linkname syscall_fcntl syscall.fcntl
15+
func syscall_fcntl(fd int, cmd int, arg int) (val int, err error)
16+
17+
func IsNonblock(fd int) (nonblocking bool, err error) {
18+
flag, err := syscall_fcntl(fd, syscall.F_GETFL, 0)
19+
if err != nil {
20+
return false, err
21+
}
22+
return flag&syscall.O_NONBLOCK != 0, nil
23+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
// Copyright 2018 The Go Authors. All rights reserved.
2+
// Use of this source code is governed by a BSD-style
3+
// license that can be found in the LICENSE file.
4+
5+
package unix
6+
7+
func IsNonblock(fd int) (nonblocking bool, err error) {
8+
return false, nil
9+
}

‎src/os/exec/exec_test.go

+6
Original file line numberDiff line numberDiff line change
@@ -404,6 +404,12 @@ var testedAlreadyLeaked = false
404404
// stdin, stdout, stderr, epoll/kqueue, maybe testlog
405405
func basefds() uintptr {
406406
n := os.Stderr.Fd() + 1
407+
// The poll (epoll/kqueue) descriptor can be numerically
408+
// either between stderr and the testlog-fd, or after
409+
// testlog-fd.
410+
if poll.PollDescriptor() == n {
411+
n++
412+
}
407413
for _, arg := range os.Args {
408414
if strings.HasPrefix(arg, "-test.testlogfile=") {
409415
n++

‎src/os/file_unix.go

+14-3
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ package os
88

99
import (
1010
"internal/poll"
11+
"internal/syscall/unix"
1112
"runtime"
1213
"syscall"
1314
)
@@ -74,9 +75,15 @@ func (f *File) Fd() uintptr {
7475

7576
// NewFile returns a new File with the given file descriptor and
7677
// name. The returned value will be nil if fd is not a valid file
77-
// descriptor.
78+
// descriptor. On Unix systems, if the file descriptor is in
79+
// non-blocking mode, NewFile will attempt to return a pollable File
80+
// (one for which the SetDeadline methods work).
7881
func NewFile(fd uintptr, name string) *File {
79-
return newFile(fd, name, kindNewFile)
82+
kind := kindNewFile
83+
if nb, err := unix.IsNonblock(int(fd)); err == nil && nb {
84+
kind = kindNonBlock
85+
}
86+
return newFile(fd, name, kind)
8087
}
8188

8289
// newFileKind describes the kind of file to newFile.
@@ -86,6 +93,7 @@ const (
8693
kindNewFile newFileKind = iota
8794
kindOpenFile
8895
kindPipe
96+
kindNonBlock
8997
)
9098

9199
// newFile is like NewFile, but if called from OpenFile or Pipe
@@ -109,11 +117,14 @@ func newFile(fd uintptr, name string, kind newFileKind) *File {
109117
// Don't try to use kqueue with regular files on FreeBSD.
110118
// It crashes the system unpredictably while running all.bash.
111119
// Issue 19093.
120+
// If the caller passed a non-blocking filedes (kindNonBlock),
121+
// we assume they know what they are doing so we allow it to be
122+
// used with kqueue.
112123
if runtime.GOOS == "freebsd" && kind == kindOpenFile {
113124
kind = kindNewFile
114125
}
115126

116-
pollable := kind == kindOpenFile || kind == kindPipe
127+
pollable := kind == kindOpenFile || kind == kindPipe || kind == kindNonBlock
117128
if err := f.pfd.Init("file", pollable); err != nil {
118129
// An error here indicates a failure to register
119130
// with the netpoll system. That can happen for

‎src/os/os_unix_test.go

+54
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ import (
1515
"strings"
1616
"syscall"
1717
"testing"
18+
"time"
1819
)
1920

2021
func init() {
@@ -224,3 +225,56 @@ func TestMkdirStickyUmask(t *testing.T) {
224225
t.Errorf("unexpected mode %s", mode)
225226
}
226227
}
228+
229+
// See also issues: 22939, 24331
230+
func newFileTest(t *testing.T, blocking bool) {
231+
p := make([]int, 2)
232+
if err := syscall.Pipe(p); err != nil {
233+
t.Fatalf("pipe: %v", err)
234+
}
235+
defer syscall.Close(p[1])
236+
237+
// Set the the read-side to non-blocking.
238+
if !blocking {
239+
if err := syscall.SetNonblock(p[0], true); err != nil {
240+
syscall.Close(p[0])
241+
t.Fatalf("SetNonblock: %v", err)
242+
}
243+
}
244+
// Convert it to a file.
245+
file := NewFile(uintptr(p[0]), "notapipe")
246+
if file == nil {
247+
syscall.Close(p[0])
248+
t.Fatalf("failed to convert fd to file!")
249+
}
250+
defer file.Close()
251+
252+
// Try to read with deadline (but don't block forever).
253+
b := make([]byte, 1)
254+
// Send something after 100ms.
255+
timer := time.AfterFunc(100*time.Millisecond, func() { syscall.Write(p[1], []byte("a")) })
256+
defer timer.Stop()
257+
file.SetReadDeadline(time.Now().Add(10 * time.Millisecond))
258+
_, err := file.Read(b)
259+
if !blocking {
260+
// We want it to fail with a timeout.
261+
if !IsTimeout(err) {
262+
t.Fatalf("No timeout reading from file: %v", err)
263+
}
264+
} else {
265+
// We want it to succeed after 100ms
266+
if err != nil {
267+
t.Fatalf("Error reading from file: %v", err)
268+
}
269+
}
270+
}
271+
272+
func TestNewFileBlock(t *testing.T) {
273+
t.Parallel()
274+
newFileTest(t, true)
275+
}
276+
277+
func TestNewFileNonBlock(t *testing.T) {
278+
t.Parallel()
279+
newFileTest(t, false)
280+
}

0 commit comments

Comments
 (0)
Please sign in to comment.