mirror of
https://github.com/MetaCubeX/mihomo.git
synced 2025-01-12 12:02:17 +08:00
208 lines
4.3 KiB
Go
208 lines
4.3 KiB
Go
package v5
|
|
|
|
import (
|
|
"errors"
|
|
"net"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/metacubex/mihomo/common/atomic"
|
|
N "github.com/metacubex/mihomo/common/net"
|
|
"github.com/metacubex/mihomo/common/pool"
|
|
"github.com/metacubex/mihomo/transport/tuic/common"
|
|
|
|
"github.com/metacubex/quic-go"
|
|
"github.com/metacubex/randv2"
|
|
)
|
|
|
|
type quicStreamPacketConn struct {
|
|
connId uint16
|
|
quicConn quic.Connection
|
|
inputConn *N.BufferedConn
|
|
|
|
udpRelayMode common.UdpRelayMode
|
|
maxUdpRelayPacketSize int
|
|
|
|
deferQuicConnFn func(quicConn quic.Connection, err error)
|
|
closeDeferFn func()
|
|
writeClosed *atomic.Bool
|
|
|
|
closeOnce sync.Once
|
|
closeErr error
|
|
closed bool
|
|
|
|
deFragger
|
|
}
|
|
|
|
func (q *quicStreamPacketConn) Close() error {
|
|
q.closeOnce.Do(func() {
|
|
q.closed = true
|
|
q.closeErr = q.close()
|
|
})
|
|
return q.closeErr
|
|
}
|
|
|
|
func (q *quicStreamPacketConn) close() (err error) {
|
|
if q.closeDeferFn != nil {
|
|
defer q.closeDeferFn()
|
|
}
|
|
if q.deferQuicConnFn != nil {
|
|
defer func() {
|
|
q.deferQuicConnFn(q.quicConn, err)
|
|
}()
|
|
}
|
|
if q.inputConn != nil {
|
|
_ = q.inputConn.Close()
|
|
q.inputConn = nil
|
|
|
|
buf := pool.GetBuffer()
|
|
defer pool.PutBuffer(buf)
|
|
err = NewDissociate(q.connId).WriteTo(buf)
|
|
if err != nil {
|
|
return
|
|
}
|
|
var stream quic.SendStream
|
|
stream, err = q.quicConn.OpenUniStream()
|
|
if err != nil {
|
|
return
|
|
}
|
|
_, err = buf.WriteTo(stream)
|
|
if err != nil {
|
|
return
|
|
}
|
|
err = stream.Close()
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func (q *quicStreamPacketConn) SetDeadline(t time.Time) error {
|
|
//TODO implement me
|
|
return nil
|
|
}
|
|
|
|
func (q *quicStreamPacketConn) SetReadDeadline(t time.Time) error {
|
|
if q.inputConn != nil {
|
|
return q.inputConn.SetReadDeadline(t)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (q *quicStreamPacketConn) SetWriteDeadline(t time.Time) error {
|
|
//TODO implement me
|
|
return nil
|
|
}
|
|
|
|
func (q *quicStreamPacketConn) ReadFrom(p []byte) (n int, addr net.Addr, err error) {
|
|
if inputConn := q.inputConn; inputConn != nil { // copy inputConn avoid be nil in for loop
|
|
for {
|
|
var packet Packet
|
|
packet, err = ReadPacket(inputConn)
|
|
if err != nil {
|
|
return
|
|
}
|
|
if packetPtr := q.deFragger.Feed(&packet); packetPtr != nil {
|
|
n = copy(p, packet.DATA)
|
|
addr = packetPtr.ADDR.UDPAddr()
|
|
return
|
|
}
|
|
}
|
|
} else {
|
|
err = net.ErrClosed
|
|
}
|
|
return
|
|
}
|
|
|
|
func (q *quicStreamPacketConn) WaitReadFrom() (data []byte, put func(), addr net.Addr, err error) {
|
|
if inputConn := q.inputConn; inputConn != nil { // copy inputConn avoid be nil in for loop
|
|
for {
|
|
var packet Packet
|
|
packet, err = ReadPacket(inputConn)
|
|
if err != nil {
|
|
return
|
|
}
|
|
if packetPtr := q.deFragger.Feed(&packet); packetPtr != nil {
|
|
data = packetPtr.DATA
|
|
addr = packetPtr.ADDR.UDPAddr()
|
|
return
|
|
}
|
|
}
|
|
} else {
|
|
err = net.ErrClosed
|
|
}
|
|
return
|
|
}
|
|
|
|
func (q *quicStreamPacketConn) WriteTo(p []byte, addr net.Addr) (n int, err error) {
|
|
if len(p) > 0xffff { // uint16 max
|
|
return 0, &quic.DatagramTooLargeError{MaxDatagramPayloadSize: 0xffff}
|
|
}
|
|
if q.closed {
|
|
return 0, net.ErrClosed
|
|
}
|
|
if q.writeClosed != nil && q.writeClosed.Load() {
|
|
_ = q.Close()
|
|
return 0, net.ErrClosed
|
|
}
|
|
if q.deferQuicConnFn != nil {
|
|
defer func() {
|
|
q.deferQuicConnFn(q.quicConn, err)
|
|
}()
|
|
}
|
|
buf := pool.GetBuffer()
|
|
defer pool.PutBuffer(buf)
|
|
address, err := NewAddressNetAddr(addr)
|
|
if err != nil {
|
|
return
|
|
}
|
|
pktId := uint16(randv2.Uint32())
|
|
packet := NewPacket(q.connId, pktId, 1, 0, uint16(len(p)), address, p)
|
|
switch q.udpRelayMode {
|
|
case common.QUIC:
|
|
err = packet.WriteTo(buf)
|
|
if err != nil {
|
|
return
|
|
}
|
|
var stream quic.SendStream
|
|
stream, err = q.quicConn.OpenUniStream()
|
|
if err != nil {
|
|
return
|
|
}
|
|
defer stream.Close()
|
|
_, err = buf.WriteTo(stream)
|
|
if err != nil {
|
|
return
|
|
}
|
|
default: // native
|
|
if len(p) > q.maxUdpRelayPacketSize {
|
|
err = fragWriteNative(q.quicConn, packet, buf, q.maxUdpRelayPacketSize)
|
|
} else {
|
|
err = packet.WriteTo(buf)
|
|
if err != nil {
|
|
return
|
|
}
|
|
data := buf.Bytes()
|
|
err = q.quicConn.SendDatagram(data)
|
|
}
|
|
|
|
var tooLarge *quic.DatagramTooLargeError
|
|
if errors.As(err, &tooLarge) {
|
|
err = fragWriteNative(q.quicConn, packet, buf, int(tooLarge.MaxDatagramPayloadSize)-PacketOverHead)
|
|
}
|
|
if err != nil {
|
|
return
|
|
}
|
|
}
|
|
n = len(p)
|
|
|
|
return
|
|
}
|
|
|
|
func (q *quicStreamPacketConn) LocalAddr() net.Addr {
|
|
return q.quicConn.LocalAddr()
|
|
}
|
|
|
|
var _ net.PacketConn = (*quicStreamPacketConn)(nil)
|