// Copyright 2023 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 js || wasip1 package net import ( "internal/poll" "runtime" "time" ) const ( readSyscallName = "fd_read" writeSyscallName = "fd_write" ) // Network file descriptor. type netFD struct { pfd poll.FD // immutable until Close family int sotype int isConnected bool // handshake completed or use of association with peer net string laddr Addr raddr Addr // The only networking available in WASI preview 1 is the ability to // sock_accept on a pre-opened socket, and then fd_read, fd_write, // fd_close, and sock_shutdown on the resulting connection. We // intercept applicable netFD calls on this instance, and then pass // the remainder of the netFD calls to fakeNetFD. *fakeNetFD } func newFD(net string, sysfd int) *netFD { return newPollFD(net, poll.FD{ Sysfd: sysfd, IsStream: true, ZeroReadIsEOF: true, }) } func newPollFD(net string, pfd poll.FD) *netFD { var laddr Addr var raddr Addr // WASI preview 1 does not have functions like getsockname/getpeername, // so we cannot get access to the underlying IP address used by connections. // // However, listeners created by FileListener are of type *TCPListener, // which can be asserted by a Go program. The (*TCPListener).Addr method // documents that the returned value will be of type *TCPAddr, we satisfy // the documented behavior by creating addresses of the expected type here. switch net { case "tcp": laddr = new(TCPAddr) raddr = new(TCPAddr) case "udp": laddr = new(UDPAddr) raddr = new(UDPAddr) default: laddr = unknownAddr{} raddr = unknownAddr{} } return &netFD{ pfd: pfd, net: net, laddr: laddr, raddr: raddr, } } func (fd *netFD) init() error { return fd.pfd.Init(fd.net, true) } func (fd *netFD) name() string { return "unknown" } func (fd *netFD) accept() (netfd *netFD, err error) { if fd.fakeNetFD != nil { return fd.fakeNetFD.accept(fd.laddr) } d, _, errcall, err := fd.pfd.Accept() if err != nil { if errcall != "" { err = wrapSyscallError(errcall, err) } return nil, err } netfd = newFD("tcp", d) if err = netfd.init(); err != nil { netfd.Close() return nil, err } return netfd, nil } func (fd *netFD) setAddr(laddr, raddr Addr) { fd.laddr = laddr fd.raddr = raddr runtime.SetFinalizer(fd, (*netFD).Close) } func (fd *netFD) Close() error { if fd.fakeNetFD != nil { return fd.fakeNetFD.Close() } runtime.SetFinalizer(fd, nil) return fd.pfd.Close() } func (fd *netFD) shutdown(how int) error { if fd.fakeNetFD != nil { return nil } err := fd.pfd.Shutdown(how) runtime.KeepAlive(fd) return wrapSyscallError("shutdown", err) } func (fd *netFD) Read(p []byte) (n int, err error) { if fd.fakeNetFD != nil { return fd.fakeNetFD.Read(p) } n, err = fd.pfd.Read(p) runtime.KeepAlive(fd) return n, wrapSyscallError(readSyscallName, err) } func (fd *netFD) Write(p []byte) (nn int, err error) { if fd.fakeNetFD != nil { return fd.fakeNetFD.Write(p) } nn, err = fd.pfd.Write(p) runtime.KeepAlive(fd) return nn, wrapSyscallError(writeSyscallName, err) } func (fd *netFD) SetDeadline(t time.Time) error { if fd.fakeNetFD != nil { return fd.fakeNetFD.SetDeadline(t) } return fd.pfd.SetDeadline(t) } func (fd *netFD) SetReadDeadline(t time.Time) error { if fd.fakeNetFD != nil { return fd.fakeNetFD.SetReadDeadline(t) } return fd.pfd.SetReadDeadline(t) } func (fd *netFD) SetWriteDeadline(t time.Time) error { if fd.fakeNetFD != nil { return fd.fakeNetFD.SetWriteDeadline(t) } return fd.pfd.SetWriteDeadline(t) } type unknownAddr struct{} func (unknownAddr) Network() string { return "unknown" } func (unknownAddr) String() string { return "unknown" }