// Retry.
                case syscall.ENOSYS, syscall.EOPNOTSUPP, syscall.EINVAL:
                        // ENOSYS indicates no kernel support for sendfile.
-                       // EINVAL indicates a FD type which does not support sendfile.
+                       // EINVAL indicates a FD type that does not support sendfile.
                        //
                        // On Linux, copy_file_range can return EOPNOTSUPP when copying
                        // to a NFS file (issue #40731); check for it here just in case.
                        return written, err, written > 0
                default:
+                       // We want to handle ENOTSUP like EOPNOTSUPP.
+                       // It's a pain to put it as a switch case
+                       // because on Linux systems ENOTSUP == EOPNOTSUPP,
+                       // so the compiler complains about a duplicate case.
+                       if err == syscall.ENOTSUP {
+                               return written, err, written > 0
+                       }
+
                        // Not a retryable error.
                        return written, err, true
                }
 
--- /dev/null
+// 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)
+       }
+}