// 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. package net import ( "io" "os" "syscall" "time" ) // Network file descritor. type netFD struct { // locking/lifetime of sysfd + serialize access to Read and Write methods fdmu fdMutex // immutable until Close proto string n string dir string ctl, data *os.File laddr, raddr Addr } var ( netdir string // default network ) func sysInit() { netdir = "/net" } func dial(net string, ra Addr, dialer func(time.Time) (Conn, error), deadline time.Time) (Conn, error) { // On plan9, use the relatively inefficient // goroutine-racing implementation. return dialChannel(net, ra, dialer, deadline) } func newFD(proto, name string, ctl, data *os.File, laddr, raddr Addr) (*netFD, error) { return &netFD{proto: proto, n: name, dir: netdir + "/" + proto + "/" + name, ctl: ctl, data: data, laddr: laddr, raddr: raddr}, nil } func (fd *netFD) init() error { // stub for future fd.pd.Init(fd) return nil } func (fd *netFD) name() string { var ls, rs string if fd.laddr != nil { ls = fd.laddr.String() } if fd.raddr != nil { rs = fd.raddr.String() } return fd.proto + ":" + ls + "->" + rs } func (fd *netFD) ok() bool { return fd != nil && fd.ctl != nil } func (fd *netFD) destroy() { if !fd.ok() { return } err := fd.ctl.Close() if fd.data != nil { if err1 := fd.data.Close(); err1 != nil && err == nil { err = err1 } } fd.ctl = nil fd.data = nil } // Add a reference to this fd. // Returns an error if the fd cannot be used. func (fd *netFD) incref() error { if !fd.fdmu.Incref() { return errClosing } return nil } // Remove a reference to this FD and close if we've been asked to do so // (and there are no references left). func (fd *netFD) decref() { if fd.fdmu.Decref() { fd.destroy() } } // Add a reference to this fd and lock for reading. // Returns an error if the fd cannot be used. func (fd *netFD) readLock() error { if !fd.fdmu.RWLock(true) { return errClosing } return nil } // Unlock for reading and remove a reference to this FD. func (fd *netFD) readUnlock() { if fd.fdmu.RWUnlock(true) { fd.destroy() } } // Add a reference to this fd and lock for writing. // Returns an error if the fd cannot be used. func (fd *netFD) writeLock() error { if !fd.fdmu.RWLock(false) { return errClosing } return nil } // Unlock for writing and remove a reference to this FD. func (fd *netFD) writeUnlock() { if fd.fdmu.RWUnlock(false) { fd.destroy() } } func (fd *netFD) Read(b []byte) (n int, err error) { if !fd.ok() || fd.data == nil { return 0, syscall.EINVAL } if err := fd.readLock(); err != nil { return 0, err } defer fd.readUnlock() n, err = fd.data.Read(b) if fd.proto == "udp" && err == io.EOF { n = 0 err = nil } return } func (fd *netFD) Write(b []byte) (n int, err error) { if !fd.ok() || fd.data == nil { return 0, syscall.EINVAL } if err := fd.writeLock(); err != nil { return 0, err } defer fd.writeUnlock() return fd.data.Write(b) } func (fd *netFD) closeRead() error { if !fd.ok() { return syscall.EINVAL } return syscall.EPLAN9 } func (fd *netFD) closeWrite() error { if !fd.ok() { return syscall.EINVAL } return syscall.EPLAN9 } func (fd *netFD) Close() error { if !fd.fdmu.IncrefAndClose() { return errClosing } if !fd.ok() { return syscall.EINVAL } err := fd.ctl.Close() if fd.data != nil { if err1 := fd.data.Close(); err1 != nil && err == nil { err = err1 } } fd.ctl = nil fd.data = nil return err } // This method is only called via Conn. func (fd *netFD) dup() (*os.File, error) { if !fd.ok() || fd.data == nil { return nil, syscall.EINVAL } return fd.file(fd.data, fd.dir+"/data") } func (l *TCPListener) dup() (*os.File, error) { if !l.fd.ok() { return nil, syscall.EINVAL } return l.fd.file(l.fd.ctl, l.fd.dir+"/ctl") } func (fd *netFD) file(f *os.File, s string) (*os.File, error) { syscall.ForkLock.RLock() dfd, err := syscall.Dup(int(f.Fd()), -1) syscall.ForkLock.RUnlock() if err != nil { return nil, &OpError{"dup", s, fd.laddr, err} } return os.NewFile(uintptr(dfd), s), nil } func (fd *netFD) setDeadline(t time.Time) error { return syscall.EPLAN9 } func (fd *netFD) setReadDeadline(t time.Time) error { return syscall.EPLAN9 } func (fd *netFD) setWriteDeadline(t time.Time) error { return syscall.EPLAN9 } func setReadBuffer(fd *netFD, bytes int) error { return syscall.EPLAN9 } func setWriteBuffer(fd *netFD, bytes int) error { return syscall.EPLAN9 } func skipRawSocketTests() (skip bool, skipmsg string, err error) { return true, "skipping test on plan9", nil }