2010-12-03 04:34:57 +00:00
|
|
|
// Copyright 2009 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.
|
|
|
|
|
2012-01-13 05:11:45 +00:00
|
|
|
// +build darwin freebsd linux netbsd openbsd windows
|
2011-10-26 23:57:58 +00:00
|
|
|
|
2010-12-03 04:34:57 +00:00
|
|
|
// Sockets
|
|
|
|
|
|
|
|
package net
|
|
|
|
|
|
|
|
import (
|
2011-09-16 15:47:21 +00:00
|
|
|
"io"
|
2010-12-03 04:34:57 +00:00
|
|
|
"syscall"
|
|
|
|
)
|
|
|
|
|
2012-01-25 20:56:26 +00:00
|
|
|
var listenerBacklog = maxListenerBacklog()
|
2010-12-03 04:34:57 +00:00
|
|
|
|
|
|
|
// Generic socket creation.
|
2012-03-30 21:27:11 +00:00
|
|
|
func socket(net string, f, t, p int, ipv6only bool, la, ra syscall.Sockaddr, toAddr func(syscall.Sockaddr) Addr) (fd *netFD, err error) {
|
2010-12-03 04:34:57 +00:00
|
|
|
// See ../syscall/exec.go for description of ForkLock.
|
|
|
|
syscall.ForkLock.RLock()
|
2012-01-25 21:54:22 +00:00
|
|
|
s, err := syscall.Socket(f, t, p)
|
2011-12-12 23:40:51 +00:00
|
|
|
if err != nil {
|
2010-12-03 04:34:57 +00:00
|
|
|
syscall.ForkLock.RUnlock()
|
2011-12-12 23:40:51 +00:00
|
|
|
return nil, err
|
2010-12-03 04:34:57 +00:00
|
|
|
}
|
|
|
|
syscall.CloseOnExec(s)
|
|
|
|
syscall.ForkLock.RUnlock()
|
|
|
|
|
2012-03-30 21:27:11 +00:00
|
|
|
err = setDefaultSockopts(s, f, t, ipv6only)
|
2012-02-09 08:19:58 +00:00
|
|
|
if err != nil {
|
|
|
|
closesocket(s)
|
|
|
|
return nil, err
|
|
|
|
}
|
2010-12-03 04:34:57 +00:00
|
|
|
|
2012-03-02 20:01:37 +00:00
|
|
|
var bla syscall.Sockaddr
|
2010-12-03 04:34:57 +00:00
|
|
|
if la != nil {
|
2012-03-02 20:01:37 +00:00
|
|
|
bla, err = listenerSockaddr(s, f, la, toAddr)
|
2012-02-09 08:19:58 +00:00
|
|
|
if err != nil {
|
|
|
|
closesocket(s)
|
|
|
|
return nil, err
|
|
|
|
}
|
2012-03-02 20:01:37 +00:00
|
|
|
err = syscall.Bind(s, bla)
|
2012-01-25 21:54:22 +00:00
|
|
|
if err != nil {
|
2011-01-21 18:19:03 +00:00
|
|
|
closesocket(s)
|
2012-01-25 21:54:22 +00:00
|
|
|
return nil, err
|
2010-12-03 04:34:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-01-25 21:54:22 +00:00
|
|
|
if fd, err = newFD(s, f, t, net); err != nil {
|
2011-03-30 15:33:16 +00:00
|
|
|
closesocket(s)
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2010-12-03 04:34:57 +00:00
|
|
|
if ra != nil {
|
2011-03-30 15:33:16 +00:00
|
|
|
if err = fd.connect(ra); err != nil {
|
2011-10-26 23:57:58 +00:00
|
|
|
closesocket(s)
|
2011-09-16 15:47:21 +00:00
|
|
|
fd.Close()
|
2011-03-30 15:33:16 +00:00
|
|
|
return nil, err
|
2010-12-03 04:34:57 +00:00
|
|
|
}
|
2012-02-01 19:26:59 +00:00
|
|
|
fd.isConnected = true
|
2010-12-03 04:34:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
sa, _ := syscall.Getsockname(s)
|
2012-03-02 20:01:37 +00:00
|
|
|
var laddr Addr
|
|
|
|
if la != nil && bla != la {
|
|
|
|
laddr = toAddr(la)
|
|
|
|
} else {
|
|
|
|
laddr = toAddr(sa)
|
|
|
|
}
|
2010-12-03 04:34:57 +00:00
|
|
|
sa, _ = syscall.Getpeername(s)
|
|
|
|
raddr := toAddr(sa)
|
|
|
|
|
2011-03-30 15:33:16 +00:00
|
|
|
fd.setAddr(laddr, raddr)
|
2010-12-03 04:34:57 +00:00
|
|
|
return fd, nil
|
|
|
|
}
|
|
|
|
|
2011-09-16 15:47:21 +00:00
|
|
|
type writerOnly struct {
|
|
|
|
io.Writer
|
|
|
|
}
|
2010-12-03 04:34:57 +00:00
|
|
|
|
2011-09-16 15:47:21 +00:00
|
|
|
// Fallback implementation of io.ReaderFrom's ReadFrom, when sendfile isn't
|
|
|
|
// applicable.
|
2011-12-03 02:17:34 +00:00
|
|
|
func genericReadFrom(w io.Writer, r io.Reader) (n int64, err error) {
|
2011-09-16 15:47:21 +00:00
|
|
|
// Use wrapper to hide existing r.ReadFrom from io.Copy.
|
|
|
|
return io.Copy(writerOnly{w}, r)
|
2010-12-03 04:34:57 +00:00
|
|
|
}
|