273 lignes
5,1 Kio
Go
273 lignes
5,1 Kio
Go
|
package utp
|
||
|
|
||
|
/*
|
||
|
#include "utp.h"
|
||
|
*/
|
||
|
import "C"
|
||
|
import (
|
||
|
"bytes"
|
||
|
"context"
|
||
|
"errors"
|
||
|
"io"
|
||
|
"net"
|
||
|
"sync"
|
||
|
"syscall"
|
||
|
"time"
|
||
|
"unsafe"
|
||
|
)
|
||
|
|
||
|
type Conn struct {
|
||
|
s *C.utp_socket
|
||
|
cond sync.Cond
|
||
|
readBuf bytes.Buffer
|
||
|
gotEOF bool
|
||
|
gotConnect bool
|
||
|
// Set on state changed to UTP_STATE_DESTROYING. Not valid to refer to the
|
||
|
// socket after getting this.
|
||
|
destroyed bool
|
||
|
// Conn.Close was called.
|
||
|
closed bool
|
||
|
libError error
|
||
|
|
||
|
writeDeadline time.Time
|
||
|
writeDeadlineTimer *time.Timer
|
||
|
readDeadline time.Time
|
||
|
readDeadlineTimer *time.Timer
|
||
|
|
||
|
numBytesRead int64
|
||
|
numBytesWritten int64
|
||
|
}
|
||
|
|
||
|
func (c *Conn) onLibError(codeName string) {
|
||
|
c.libError = errors.New(codeName)
|
||
|
c.cond.Broadcast()
|
||
|
}
|
||
|
|
||
|
func (c *Conn) setConnected() {
|
||
|
c.gotConnect = true
|
||
|
c.cond.Broadcast()
|
||
|
}
|
||
|
|
||
|
func (c *Conn) waitForConnect(ctx context.Context) error {
|
||
|
ctx, cancel := context.WithCancel(ctx)
|
||
|
defer cancel()
|
||
|
go func() {
|
||
|
<-ctx.Done()
|
||
|
c.cond.Broadcast()
|
||
|
}()
|
||
|
for {
|
||
|
if c.libError != nil {
|
||
|
return c.libError
|
||
|
}
|
||
|
if c.gotConnect {
|
||
|
return nil
|
||
|
}
|
||
|
if ctx.Err() != nil {
|
||
|
return ctx.Err()
|
||
|
}
|
||
|
c.cond.Wait()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (c *Conn) Close() (err error) {
|
||
|
mu.Lock()
|
||
|
defer mu.Unlock()
|
||
|
c.close()
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (c *Conn) close() {
|
||
|
// log.Printf("conn %p: closed", c)
|
||
|
if c.closed {
|
||
|
return
|
||
|
}
|
||
|
if !c.destroyed {
|
||
|
C.utp_close(c.s)
|
||
|
// C.utp_issue_deferred_acks(C.utp_get_context(c.s))
|
||
|
}
|
||
|
c.closed = true
|
||
|
c.cond.Broadcast()
|
||
|
}
|
||
|
|
||
|
func (c *Conn) LocalAddr() net.Addr {
|
||
|
mu.Lock()
|
||
|
defer mu.Unlock()
|
||
|
return getSocketForLibContext(C.utp_get_context(c.s)).pc.LocalAddr()
|
||
|
}
|
||
|
|
||
|
func (c *Conn) readNoWait(b []byte) (n int, err error) {
|
||
|
n, _ = c.readBuf.Read(b)
|
||
|
if n != 0 && c.readBuf.Len() == 0 {
|
||
|
// Can we call this if the utp_socket is closed, destroyed or errored?
|
||
|
if c.s != nil {
|
||
|
C.utp_read_drained(c.s)
|
||
|
// C.utp_issue_deferred_acks(C.utp_get_context(c.s))
|
||
|
}
|
||
|
}
|
||
|
if c.readBuf.Len() != 0 {
|
||
|
return
|
||
|
}
|
||
|
err = func() error {
|
||
|
switch {
|
||
|
case c.gotEOF:
|
||
|
return io.EOF
|
||
|
case c.libError != nil:
|
||
|
return c.libError
|
||
|
case c.destroyed:
|
||
|
return errors.New("destroyed")
|
||
|
case c.closed:
|
||
|
return errors.New("closed")
|
||
|
case !c.readDeadline.IsZero() && !time.Now().Before(c.readDeadline):
|
||
|
return errDeadlineExceeded{}
|
||
|
default:
|
||
|
return nil
|
||
|
}
|
||
|
}()
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func (c *Conn) Read(b []byte) (int, error) {
|
||
|
mu.Lock()
|
||
|
defer mu.Unlock()
|
||
|
for {
|
||
|
n, err := c.readNoWait(b)
|
||
|
c.numBytesRead += int64(n)
|
||
|
// log.Printf("read %d bytes", c.numBytesRead)
|
||
|
if n != 0 || len(b) == 0 || err != nil {
|
||
|
// log.Printf("conn %p: read %d bytes: %s", c, n, err)
|
||
|
return n, err
|
||
|
}
|
||
|
c.cond.Wait()
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (c *Conn) writeNoWait(b []byte) (n int, err error) {
|
||
|
err = func() error {
|
||
|
switch {
|
||
|
case c.libError != nil:
|
||
|
return c.libError
|
||
|
case c.closed:
|
||
|
return errors.New("closed")
|
||
|
case c.destroyed:
|
||
|
return errors.New("destroyed")
|
||
|
case !c.writeDeadline.IsZero() && !time.Now().Before(c.writeDeadline):
|
||
|
return errDeadlineExceeded{}
|
||
|
default:
|
||
|
return nil
|
||
|
}
|
||
|
}()
|
||
|
if err != nil {
|
||
|
return
|
||
|
}
|
||
|
n = int(C.utp_write(c.s, unsafe.Pointer(&b[0]), C.size_t(len(b))))
|
||
|
if n < 0 {
|
||
|
panic(n)
|
||
|
}
|
||
|
// log.Print(n)
|
||
|
// C.utp_issue_deferred_acks(C.utp_get_context(c.s))
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func (c *Conn) Write(b []byte) (n int, err error) {
|
||
|
// defer func() { log.Printf("wrote %d bytes: %s", n, err) }()
|
||
|
// log.Print(len(b))
|
||
|
mu.Lock()
|
||
|
defer mu.Unlock()
|
||
|
for len(b) != 0 {
|
||
|
var n1 int
|
||
|
n1, err = c.writeNoWait(b)
|
||
|
b = b[n1:]
|
||
|
n += n1
|
||
|
if err != nil {
|
||
|
break
|
||
|
}
|
||
|
if n1 != 0 {
|
||
|
continue
|
||
|
}
|
||
|
c.cond.Wait()
|
||
|
}
|
||
|
c.numBytesWritten += int64(n)
|
||
|
// log.Printf("wrote %d bytes", c.numBytesWritten)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
func (c *Conn) RemoteAddr() net.Addr {
|
||
|
var rsa syscall.RawSockaddrAny
|
||
|
var addrlen C.socklen_t = C.socklen_t(unsafe.Sizeof(rsa))
|
||
|
C.utp_getpeername(c.s, (*C.struct_sockaddr)(unsafe.Pointer(&rsa)), &addrlen)
|
||
|
sa, err := anyToSockaddr(&rsa)
|
||
|
if err != nil {
|
||
|
panic(err)
|
||
|
}
|
||
|
return sockaddrToUDP(sa)
|
||
|
}
|
||
|
|
||
|
func (c *Conn) SetDeadline(t time.Time) error {
|
||
|
mu.Lock()
|
||
|
defer mu.Unlock()
|
||
|
c.readDeadline = t
|
||
|
c.writeDeadline = t
|
||
|
if t.IsZero() {
|
||
|
c.readDeadlineTimer.Stop()
|
||
|
c.writeDeadlineTimer.Stop()
|
||
|
} else {
|
||
|
d := t.Sub(time.Now())
|
||
|
c.readDeadlineTimer.Reset(d)
|
||
|
c.writeDeadlineTimer.Reset(d)
|
||
|
}
|
||
|
c.cond.Broadcast()
|
||
|
return nil
|
||
|
}
|
||
|
func (c *Conn) SetReadDeadline(t time.Time) error {
|
||
|
mu.Lock()
|
||
|
defer mu.Unlock()
|
||
|
c.readDeadline = t
|
||
|
if t.IsZero() {
|
||
|
c.readDeadlineTimer.Stop()
|
||
|
} else {
|
||
|
d := t.Sub(time.Now())
|
||
|
c.readDeadlineTimer.Reset(d)
|
||
|
}
|
||
|
c.cond.Broadcast()
|
||
|
return nil
|
||
|
}
|
||
|
func (c *Conn) SetWriteDeadline(t time.Time) error {
|
||
|
mu.Lock()
|
||
|
defer mu.Unlock()
|
||
|
c.writeDeadline = t
|
||
|
if t.IsZero() {
|
||
|
c.writeDeadlineTimer.Stop()
|
||
|
} else {
|
||
|
d := t.Sub(time.Now())
|
||
|
c.writeDeadlineTimer.Reset(d)
|
||
|
}
|
||
|
c.cond.Broadcast()
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (c *Conn) setGotEOF() {
|
||
|
c.gotEOF = true
|
||
|
c.cond.Broadcast()
|
||
|
}
|
||
|
|
||
|
func (c *Conn) onDestroyed() {
|
||
|
c.destroyed = true
|
||
|
c.s = nil
|
||
|
c.cond.Broadcast()
|
||
|
}
|
||
|
|
||
|
func (c *Conn) WriteBufferLen() int {
|
||
|
mu.Lock()
|
||
|
defer mu.Unlock()
|
||
|
return int(C.utp_getsockopt(c.s, C.UTP_SNDBUF))
|
||
|
}
|
||
|
|
||
|
func (c *Conn) SetWriteBufferLen(len int) {
|
||
|
mu.Lock()
|
||
|
defer mu.Unlock()
|
||
|
i := C.utp_setsockopt(c.s, C.UTP_SNDBUF, C.int(len))
|
||
|
if i != 0 {
|
||
|
panic(i)
|
||
|
}
|
||
|
}
|