mirror of
https://github.com/MetaCubeX/mihomo.git
synced 2024-12-26 20:26:06 +08:00
97 lines
2.4 KiB
Go
97 lines
2.4 KiB
Go
|
package deadline
|
||
|
|
||
|
import (
|
||
|
"os"
|
||
|
|
||
|
"github.com/Dreamacro/clash/common/net/packet"
|
||
|
"github.com/sagernet/sing/common/buf"
|
||
|
M "github.com/sagernet/sing/common/metadata"
|
||
|
)
|
||
|
|
||
|
type SingPacketConn struct {
|
||
|
*NetPacketConn
|
||
|
singPacketConn
|
||
|
}
|
||
|
|
||
|
var _ packet.SingPacketConn = (*SingPacketConn)(nil)
|
||
|
|
||
|
func NewSingPacketConn(pc packet.SingPacketConn) packet.SingPacketConn {
|
||
|
return NewNetPacketConn(pc).(packet.SingPacketConn)
|
||
|
}
|
||
|
|
||
|
type EnhanceSingPacketConn struct {
|
||
|
*EnhancePacketConn
|
||
|
singPacketConn
|
||
|
}
|
||
|
|
||
|
func NewEnhanceSingPacketConn(pc packet.EnhanceSingPacketConn) packet.EnhanceSingPacketConn {
|
||
|
return NewNetPacketConn(pc).(packet.EnhanceSingPacketConn)
|
||
|
}
|
||
|
|
||
|
var _ packet.EnhanceSingPacketConn = (*EnhanceSingPacketConn)(nil)
|
||
|
|
||
|
type singReadResult struct {
|
||
|
buffer *buf.Buffer
|
||
|
destination M.Socksaddr
|
||
|
}
|
||
|
|
||
|
type singPacketConn struct {
|
||
|
netPacketConn *NetPacketConn
|
||
|
singPacketConn packet.SingPacketConn
|
||
|
}
|
||
|
|
||
|
func (c *singPacketConn) ReadPacket(buffer *buf.Buffer) (destination M.Socksaddr, err error) {
|
||
|
select {
|
||
|
case result := <-c.netPacketConn.resultCh:
|
||
|
if result != nil {
|
||
|
destination = result.destination
|
||
|
err = result.err
|
||
|
buffer.Resize(result.buffer.Start(), 0)
|
||
|
n := copy(buffer.FreeBytes(), result.buffer.Bytes())
|
||
|
buffer.Truncate(n)
|
||
|
result.buffer.Advance(n)
|
||
|
if result.buffer.IsEmpty() {
|
||
|
result.buffer.Release()
|
||
|
}
|
||
|
c.netPacketConn.resultCh <- nil // finish cache read
|
||
|
return
|
||
|
} else {
|
||
|
c.netPacketConn.resultCh <- nil
|
||
|
break
|
||
|
}
|
||
|
case <-c.netPacketConn.pipeDeadline.wait():
|
||
|
return M.Socksaddr{}, os.ErrDeadlineExceeded
|
||
|
}
|
||
|
|
||
|
if c.netPacketConn.disablePipe.Load() {
|
||
|
return c.singPacketConn.ReadPacket(buffer)
|
||
|
} else if c.netPacketConn.deadline.Load().IsZero() {
|
||
|
c.netPacketConn.inRead.Store(true)
|
||
|
defer c.netPacketConn.inRead.Store(false)
|
||
|
destination, err = c.singPacketConn.ReadPacket(buffer)
|
||
|
return
|
||
|
}
|
||
|
|
||
|
<-c.netPacketConn.resultCh
|
||
|
go c.pipeReadPacket(buffer.Cap(), buffer.Start())
|
||
|
|
||
|
return c.ReadPacket(buffer)
|
||
|
}
|
||
|
|
||
|
func (c *singPacketConn) pipeReadPacket(bufLen int, bufStart int) {
|
||
|
buffer := buf.NewSize(bufLen)
|
||
|
buffer.Advance(bufStart)
|
||
|
destination, err := c.singPacketConn.ReadPacket(buffer)
|
||
|
c.netPacketConn.resultCh <- &readResult{
|
||
|
singReadResult: singReadResult{
|
||
|
buffer: buffer,
|
||
|
destination: destination,
|
||
|
},
|
||
|
err: err,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func (c *singPacketConn) WritePacket(buffer *buf.Buffer, destination M.Socksaddr) error {
|
||
|
return c.singPacketConn.WritePacket(buffer, destination)
|
||
|
}
|