data[1] = uint8(ctlSetPeerID)
binary.BigEndian.PutUint16(data[2:4], uint16(clt.ID()))
if _, err := clt.sendRaw(rawPkt{Data: data}); err != nil {
+ if errors.Is(err, net.ErrClosed) {
+ return nil
+ }
return fmt.Errorf("can't set client peer id: %w", err)
}
select {
case clt.pkts <- pkt:
default:
- return fmt.Errorf("ignoring net pkt from %s because buf is full", addrstr)
+ // It's OK to drop packets if the buffer is full
+ // because MT RUDP can cope with packet loss.
}
}
package rudp
import (
+ "errors"
"fmt"
"net"
"sync"
select {
case <-ping:
if _, err := p.sendRaw(pkt); err != nil {
+ if errors.Is(err, net.ErrClosed) {
+ return
+ }
p.errs <- fmt.Errorf("can't send ping: %w", err)
}
case <-p.Disco():
}
// Connect connects to the server on conn
-// and closes conn when the Peer disconnects.
+// and closes conn when the returned *Peer disconnects.
func Connect(conn net.PacketConn, addr net.Addr) *Peer {
srv := newPeer(conn, addr, PeerIDSrv, PeerIDNil)
"errors"
"fmt"
"io"
+ "net"
)
// A PktError is an error that occured while processing a packet.
case ctlDisco:
defer errWrap("disco: %w")
- if err := p.Close(); err != nil {
- return fmt.Errorf("can't close: %w", err)
- }
+ p.Close()
if len(pkt.Data) > 1+1 {
return TrailingDataError(pkt.Data[1+1:])
Unrel: true,
}
if _, err := p.sendRaw(ack); err != nil {
+ if errors.Is(err, net.ErrClosed) {
+ return nil
+ }
return fmt.Errorf("can't ack %d: %w", sn, err)
}
package main
import (
+ "errors"
"fmt"
"log"
"net"
"os"
- "mt/rudp"
+ "github.com/anon55555/mt/rudp"
)
func main() {
for {
pkt, err := src.Recv()
if err != nil {
- if err == net.ErrClosed {
+ if errors.Is(err, net.ErrClosed) {
msg := src.Addr().String() + " disconnected"
if src.TimedOut() {
msg += " (timed out)"
select {
case <-time.After(500 * time.Millisecond):
if _, err := p.sendRaw(relpkt); err != nil {
+ if errors.Is(err, net.ErrClosed) {
+ return
+ }
p.errs <- fmt.Errorf("failed to re-send timed out reliable seqnum: %d: %w", sn, err)
}
case <-ack: