]> Cypherpunks repositories - gostls13.git/commitdiff
[release-branch.go1.23] net: don't use sendfile for irregular files; handle EOPNOTSUP...
authorIan Lance Taylor <iant@golang.org>
Wed, 11 Dec 2024 22:18:42 +0000 (14:18 -0800)
committerGopher Robot <gobot@golang.org>
Wed, 18 Dec 2024 17:42:15 +0000 (09:42 -0800)
This is not a cherry pick, because the code has changed on tip.

For #70763
Fixes #70789

Change-Id: If9fcfee17e86a746cf8c72293dc34f80b430f6e6
Reviewed-on: https://go-review.googlesource.com/c/go/+/635397
Auto-Submit: Ian Lance Taylor <iant@google.com>
Reviewed-by: Dmitri Shuralyov <dmitshur@google.com>
Reviewed-by: Damien Neil <dneil@google.com>
LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>

src/internal/poll/sendfile_bsd.go
src/net/sendfile_unix_alt.go
src/net/sendfile_unix_test.go [new file with mode: 0644]

index 341e07ca1fed6a33545bead6fadd791f70a0c3bf..d1023d4ebb99387f5cfb34acae699040f0baa691 100644 (file)
@@ -72,6 +72,6 @@ func SendFile(dstFD *FD, src int, pos, remain int64) (written int64, err error,
        if err == syscall.EAGAIN {
                err = nil
        }
-       handled = written != 0 || (err != syscall.ENOSYS && err != syscall.EINVAL)
+       handled = written != 0 || (err != syscall.ENOSYS && err != syscall.EINVAL && err != syscall.EOPNOTSUPP && err != syscall.ENOTSUP)
        return
 }
index 9e46c4e607d4d86999ad9ec9808bfbbf18e8a4b8..4056856f306175dff8b0b62a37982402b2aaf7a5 100644 (file)
@@ -53,6 +53,9 @@ func sendFile(c *netFD, r io.Reader) (written int64, err error, handled bool) {
                if err != nil {
                        return 0, err, false
                }
+               if fi.Mode()&(fs.ModeSymlink|fs.ModeDevice|fs.ModeCharDevice|fs.ModeIrregular) != 0 {
+                       return 0, nil, false
+               }
 
                remain = fi.Size()
        }
diff --git a/src/net/sendfile_unix_test.go b/src/net/sendfile_unix_test.go
new file mode 100644 (file)
index 0000000..79fb23b
--- /dev/null
@@ -0,0 +1,86 @@
+// Copyright 2024 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.
+
+//go:build unix
+
+package net
+
+import (
+       "internal/testpty"
+       "io"
+       "os"
+       "sync"
+       "syscall"
+       "testing"
+)
+
+// Issue 70763: test that we don't fail on sendfile from a tty.
+func TestCopyFromTTY(t *testing.T) {
+       pty, ttyName, err := testpty.Open()
+       if err != nil {
+               t.Skipf("skipping test because pty open failed: %v", err)
+       }
+       defer pty.Close()
+
+       // Use syscall.Open so that the tty is blocking.
+       ttyFD, err := syscall.Open(ttyName, syscall.O_RDWR, 0)
+       if err != nil {
+               t.Skipf("skipping test because tty open failed: %v", err)
+       }
+       defer syscall.Close(ttyFD)
+
+       tty := os.NewFile(uintptr(ttyFD), "tty")
+       defer tty.Close()
+
+       ln := newLocalListener(t, "tcp")
+       defer ln.Close()
+
+       ch := make(chan bool)
+
+       const data = "data\n"
+
+       var wg sync.WaitGroup
+       defer wg.Wait()
+
+       wg.Add(1)
+       go func() {
+               defer wg.Done()
+               conn, err := ln.Accept()
+               if err != nil {
+                       t.Error(err)
+                       return
+               }
+               defer conn.Close()
+
+               buf := make([]byte, len(data))
+               if _, err := io.ReadFull(conn, buf); err != nil {
+                       t.Error(err)
+               }
+
+               ch <- true
+       }()
+
+       conn, err := Dial("tcp", ln.Addr().String())
+       if err != nil {
+               t.Fatal(err)
+       }
+       defer conn.Close()
+
+       wg.Add(1)
+       go func() {
+               defer wg.Done()
+               if _, err := pty.Write([]byte(data)); err != nil {
+                       t.Error(err)
+               }
+               <-ch
+               if err := pty.Close(); err != nil {
+                       t.Error(err)
+               }
+       }()
+
+       lr := io.LimitReader(tty, int64(len(data)))
+       if _, err := io.Copy(conn, lr); err != nil {
+               t.Error(err)
+       }
+}