2fd401c8f1
From-SVN: r181964
308 lines
6.1 KiB
Go
308 lines
6.1 KiB
Go
// 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.
|
|
|
|
// IP sockets stubs for Plan 9
|
|
|
|
package net
|
|
|
|
import (
|
|
"errors"
|
|
"io"
|
|
"os"
|
|
)
|
|
|
|
// probeIPv6Stack returns two boolean values. If the first boolean value is
|
|
// true, kernel supports basic IPv6 functionality. If the second
|
|
// boolean value is true, kernel supports IPv6 IPv4-mapping.
|
|
func probeIPv6Stack() (supportsIPv6, supportsIPv4map bool) {
|
|
return false, false
|
|
}
|
|
|
|
// parsePlan9Addr parses address of the form [ip!]port (e.g. 127.0.0.1!80).
|
|
func parsePlan9Addr(s string) (ip IP, iport int, err error) {
|
|
var (
|
|
addr IP
|
|
p, i int
|
|
ok bool
|
|
)
|
|
addr = IPv4zero // address contains port only
|
|
i = byteIndex(s, '!')
|
|
if i >= 0 {
|
|
addr = ParseIP(s[:i])
|
|
if addr == nil {
|
|
err = errors.New("net: parsing IP failed")
|
|
goto Error
|
|
}
|
|
}
|
|
p, _, ok = dtoi(s[i+1:], 0)
|
|
if !ok {
|
|
err = errors.New("net: parsing port failed")
|
|
goto Error
|
|
}
|
|
if p < 0 || p > 0xFFFF {
|
|
err = &AddrError{"invalid port", string(p)}
|
|
goto Error
|
|
}
|
|
return addr, p, nil
|
|
|
|
Error:
|
|
return nil, 0, err
|
|
}
|
|
|
|
func readPlan9Addr(proto, filename string) (addr Addr, err error) {
|
|
var buf [128]byte
|
|
|
|
f, err := os.Open(filename)
|
|
if err != nil {
|
|
return
|
|
}
|
|
n, err := f.Read(buf[:])
|
|
if err != nil {
|
|
return
|
|
}
|
|
ip, port, err := parsePlan9Addr(string(buf[:n]))
|
|
if err != nil {
|
|
return
|
|
}
|
|
switch proto {
|
|
case "tcp":
|
|
addr = &TCPAddr{ip, port}
|
|
case "udp":
|
|
addr = &UDPAddr{ip, port}
|
|
default:
|
|
return nil, errors.New("unknown protocol " + proto)
|
|
}
|
|
return addr, nil
|
|
}
|
|
|
|
type plan9Conn struct {
|
|
proto, name, dir string
|
|
ctl, data *os.File
|
|
laddr, raddr Addr
|
|
}
|
|
|
|
func newPlan9Conn(proto, name string, ctl *os.File, laddr, raddr Addr) *plan9Conn {
|
|
return &plan9Conn{proto, name, "/net/" + proto + "/" + name, ctl, nil, laddr, raddr}
|
|
}
|
|
|
|
func (c *plan9Conn) ok() bool { return c != nil && c.ctl != nil }
|
|
|
|
// Implementation of the Conn interface - see Conn for documentation.
|
|
|
|
// Read implements the net.Conn Read method.
|
|
func (c *plan9Conn) Read(b []byte) (n int, err error) {
|
|
if !c.ok() {
|
|
return 0, os.EINVAL
|
|
}
|
|
if c.data == nil {
|
|
c.data, err = os.OpenFile(c.dir+"/data", os.O_RDWR, 0)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
}
|
|
n, err = c.data.Read(b)
|
|
if c.proto == "udp" && err == io.EOF {
|
|
n = 0
|
|
err = nil
|
|
}
|
|
return
|
|
}
|
|
|
|
// Write implements the net.Conn Write method.
|
|
func (c *plan9Conn) Write(b []byte) (n int, err error) {
|
|
if !c.ok() {
|
|
return 0, os.EINVAL
|
|
}
|
|
if c.data == nil {
|
|
c.data, err = os.OpenFile(c.dir+"/data", os.O_RDWR, 0)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
}
|
|
return c.data.Write(b)
|
|
}
|
|
|
|
// Close closes the connection.
|
|
func (c *plan9Conn) Close() error {
|
|
if !c.ok() {
|
|
return os.EINVAL
|
|
}
|
|
err := c.ctl.Close()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if c.data != nil {
|
|
err = c.data.Close()
|
|
}
|
|
c.ctl = nil
|
|
c.data = nil
|
|
return err
|
|
}
|
|
|
|
// LocalAddr returns the local network address.
|
|
func (c *plan9Conn) LocalAddr() Addr {
|
|
if !c.ok() {
|
|
return nil
|
|
}
|
|
return c.laddr
|
|
}
|
|
|
|
// RemoteAddr returns the remote network address.
|
|
func (c *plan9Conn) RemoteAddr() Addr {
|
|
if !c.ok() {
|
|
return nil
|
|
}
|
|
return c.raddr
|
|
}
|
|
|
|
// SetTimeout implements the net.Conn SetTimeout method.
|
|
func (c *plan9Conn) SetTimeout(nsec int64) error {
|
|
if !c.ok() {
|
|
return os.EINVAL
|
|
}
|
|
return os.EPLAN9
|
|
}
|
|
|
|
// SetReadTimeout implements the net.Conn SetReadTimeout method.
|
|
func (c *plan9Conn) SetReadTimeout(nsec int64) error {
|
|
if !c.ok() {
|
|
return os.EINVAL
|
|
}
|
|
return os.EPLAN9
|
|
}
|
|
|
|
// SetWriteTimeout implements the net.Conn SetWriteTimeout method.
|
|
func (c *plan9Conn) SetWriteTimeout(nsec int64) error {
|
|
if !c.ok() {
|
|
return os.EINVAL
|
|
}
|
|
return os.EPLAN9
|
|
}
|
|
|
|
func startPlan9(net string, addr Addr) (ctl *os.File, dest, proto, name string, err error) {
|
|
var (
|
|
ip IP
|
|
port int
|
|
)
|
|
switch a := addr.(type) {
|
|
case *TCPAddr:
|
|
proto = "tcp"
|
|
ip = a.IP
|
|
port = a.Port
|
|
case *UDPAddr:
|
|
proto = "udp"
|
|
ip = a.IP
|
|
port = a.Port
|
|
default:
|
|
err = UnknownNetworkError(net)
|
|
return
|
|
}
|
|
|
|
clone, dest, err := queryCS1(proto, ip, port)
|
|
if err != nil {
|
|
return
|
|
}
|
|
f, err := os.OpenFile(clone, os.O_RDWR, 0)
|
|
if err != nil {
|
|
return
|
|
}
|
|
var buf [16]byte
|
|
n, err := f.Read(buf[:])
|
|
if err != nil {
|
|
return
|
|
}
|
|
return f, dest, proto, string(buf[:n]), nil
|
|
}
|
|
|
|
func dialPlan9(net string, laddr, raddr Addr) (c *plan9Conn, err error) {
|
|
f, dest, proto, name, err := startPlan9(net, raddr)
|
|
if err != nil {
|
|
return
|
|
}
|
|
_, err = f.WriteString("connect " + dest)
|
|
if err != nil {
|
|
return
|
|
}
|
|
laddr, err = readPlan9Addr(proto, "/net/"+proto+"/"+name+"/local")
|
|
if err != nil {
|
|
return
|
|
}
|
|
raddr, err = readPlan9Addr(proto, "/net/"+proto+"/"+name+"/remote")
|
|
if err != nil {
|
|
return
|
|
}
|
|
return newPlan9Conn(proto, name, f, laddr, raddr), nil
|
|
}
|
|
|
|
type plan9Listener struct {
|
|
proto, name, dir string
|
|
ctl *os.File
|
|
laddr Addr
|
|
}
|
|
|
|
func listenPlan9(net string, laddr Addr) (l *plan9Listener, err error) {
|
|
f, dest, proto, name, err := startPlan9(net, laddr)
|
|
if err != nil {
|
|
return
|
|
}
|
|
_, err = f.WriteString("announce " + dest)
|
|
if err != nil {
|
|
return
|
|
}
|
|
laddr, err = readPlan9Addr(proto, "/net/"+proto+"/"+name+"/local")
|
|
if err != nil {
|
|
return
|
|
}
|
|
l = new(plan9Listener)
|
|
l.proto = proto
|
|
l.name = name
|
|
l.dir = "/net/" + proto + "/" + name
|
|
l.ctl = f
|
|
l.laddr = laddr
|
|
return l, nil
|
|
}
|
|
|
|
func (l *plan9Listener) plan9Conn() *plan9Conn {
|
|
return newPlan9Conn(l.proto, l.name, l.ctl, l.laddr, nil)
|
|
}
|
|
|
|
func (l *plan9Listener) acceptPlan9() (c *plan9Conn, err error) {
|
|
f, err := os.Open(l.dir + "/listen")
|
|
if err != nil {
|
|
return
|
|
}
|
|
var buf [16]byte
|
|
n, err := f.Read(buf[:])
|
|
if err != nil {
|
|
return
|
|
}
|
|
name := string(buf[:n])
|
|
laddr, err := readPlan9Addr(l.proto, l.dir+"/local")
|
|
if err != nil {
|
|
return
|
|
}
|
|
raddr, err := readPlan9Addr(l.proto, l.dir+"/remote")
|
|
if err != nil {
|
|
return
|
|
}
|
|
return newPlan9Conn(l.proto, name, f, laddr, raddr), nil
|
|
}
|
|
|
|
func (l *plan9Listener) Accept() (c Conn, err error) {
|
|
c1, err := l.acceptPlan9()
|
|
if err != nil {
|
|
return
|
|
}
|
|
return c1, nil
|
|
}
|
|
|
|
func (l *plan9Listener) Close() error {
|
|
if l == nil || l.ctl == nil {
|
|
return os.EINVAL
|
|
}
|
|
return l.ctl.Close()
|
|
}
|
|
|
|
func (l *plan9Listener) Addr() Addr { return l.laddr }
|