mirror of
https://github.com/MetaCubeX/mihomo.git
synced 2024-07-02 19:09:59 +00:00
Some checks are pending
Build / build (map[goamd64:v1 goarch:amd64 goos:linux output:amd64-compatible test:test]) (push) Waiting to run
Build / build (map[goamd64:v1 goarch:amd64 goos:windows goversion:1.20 output:amd64-compatible-go120]) (push) Waiting to run
Build / build (map[goamd64:v1 goarch:amd64 goos:windows output:amd64-compatible]) (push) Waiting to run
Build / build (map[abi:1 goarch:loong64 goos:linux output:loong64-abi1]) (push) Waiting to run
Build / build (map[abi:2 goarch:loong64 goos:linux output:loong64-abi2]) (push) Waiting to run
Build / build (map[goamd64:v1 goarch:amd64 goos:darwin goversion:1.20 output:amd64-compatible-go120]) (push) Waiting to run
Build / build (map[goamd64:v1 goarch:amd64 goos:darwin output:amd64-compatible]) (push) Waiting to run
Build / build (map[goamd64:v1 goarch:amd64 goos:freebsd output:amd64-compatible]) (push) Waiting to run
Build / build (map[goamd64:v1 goarch:amd64 goos:linux goversion:1.20 output:amd64-compatible-go120 test:test]) (push) Waiting to run
Build / build (map[goamd64:v3 goarch:amd64 goos:darwin goversion:1.20 output:amd64-go120]) (push) Waiting to run
Build / build (map[goamd64:v3 goarch:amd64 goos:darwin output:amd64]) (push) Waiting to run
Build / build (map[goamd64:v3 goarch:amd64 goos:freebsd output:amd64]) (push) Waiting to run
Build / build (map[goamd64:v3 goarch:amd64 goos:linux goversion:1.20 output:amd64-go120]) (push) Waiting to run
Build / build (map[goamd64:v3 goarch:amd64 goos:linux output:amd64]) (push) Waiting to run
Build / build (map[goamd64:v3 goarch:amd64 goos:windows goversion:1.20 output:amd64-go120]) (push) Waiting to run
Build / build (map[goamd64:v3 goarch:amd64 goos:windows output:amd64]) (push) Waiting to run
Build / build (map[goarch:386 goos:android ndk:i686-linux-android34 output:386]) (push) Waiting to run
Build / build (map[goarch:386 goos:freebsd output:386]) (push) Waiting to run
Build / build (map[goarch:386 goos:linux goversion:1.20 output:386-go120]) (push) Waiting to run
Build / build (map[goarch:386 goos:linux output:386]) (push) Waiting to run
Build / build (map[goarch:386 goos:windows goversion:1.20 output:386-go120]) (push) Waiting to run
Build / build (map[goarch:386 goos:windows output:386]) (push) Waiting to run
Build / build (map[goarch:amd64 goos:android ndk:x86_64-linux-android34 output:amd64]) (push) Waiting to run
Build / build (map[goarch:arm goarm:7 goos:linux output:armv7]) (push) Waiting to run
Build / build (map[goarch:arm goarm:7 goos:windows output:armv7]) (push) Waiting to run
Build / build (map[goarch:arm goos:android ndk:armv7a-linux-androideabi34 output:armv7]) (push) Waiting to run
Build / build (map[goarch:arm64 goos:android ndk:aarch64-linux-android34 output:arm64-v8]) (push) Waiting to run
Build / build (map[goarch:arm64 goos:darwin goversion:1.20 output:arm64-go120]) (push) Waiting to run
Build / build (map[goarch:arm64 goos:darwin output:arm64]) (push) Waiting to run
Build / build (map[goarch:arm64 goos:freebsd output:arm64]) (push) Waiting to run
Build / build (map[goarch:arm64 goos:linux output:arm64]) (push) Waiting to run
Build / build (map[goarch:arm64 goos:windows output:arm64]) (push) Waiting to run
Build / build (map[goarch:mips goos:linux mips:hardfloat output:mips-hardfloat]) (push) Waiting to run
Build / build (map[goarch:mips goos:linux mips:softfloat output:mips-softfloat]) (push) Waiting to run
Build / build (map[goarch:mips64 goos:linux output:mips64]) (push) Waiting to run
Build / build (map[goarch:mips64le goos:linux output:mips64le]) (push) Waiting to run
Build / build (map[goarch:mipsle goos:linux mips:hardfloat output:mipsle-hardfloat]) (push) Waiting to run
Build / build (map[goarch:mipsle goos:linux mips:softfloat output:mipsle-softfloat]) (push) Waiting to run
Build / build (map[goarch:riscv64 goos:linux output:riscv64]) (push) Waiting to run
Build / build (map[goarch:s390x goos:linux output:s390x]) (push) Waiting to run
Build / Upload-Prerelease (push) Blocked by required conditions
Build / Upload-Release (push) Blocked by required conditions
Build / Docker (push) Blocked by required conditions
Trigger CMFA Update / trigger-CMFA-update (push) Waiting to run
127 lines
3.6 KiB
Go
127 lines
3.6 KiB
Go
package sing_tun
|
|
|
|
import (
|
|
"context"
|
|
"net"
|
|
"net/netip"
|
|
"sync"
|
|
"time"
|
|
|
|
"github.com/metacubex/mihomo/component/resolver"
|
|
"github.com/metacubex/mihomo/listener/sing"
|
|
"github.com/metacubex/mihomo/log"
|
|
|
|
"github.com/sagernet/sing/common/buf"
|
|
"github.com/sagernet/sing/common/bufio"
|
|
M "github.com/sagernet/sing/common/metadata"
|
|
"github.com/sagernet/sing/common/network"
|
|
)
|
|
|
|
type ListenerHandler struct {
|
|
*sing.ListenerHandler
|
|
DnsAdds []netip.AddrPort
|
|
}
|
|
|
|
func (h *ListenerHandler) ShouldHijackDns(targetAddr netip.AddrPort) bool {
|
|
if targetAddr.Addr().IsLoopback() && targetAddr.Port() == 53 { // cause by system stack
|
|
return true
|
|
}
|
|
for _, addrPort := range h.DnsAdds {
|
|
if addrPort == targetAddr || (addrPort.Addr().IsUnspecified() && targetAddr.Port() == 53) {
|
|
return true
|
|
}
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (h *ListenerHandler) NewConnection(ctx context.Context, conn net.Conn, metadata M.Metadata) error {
|
|
if h.ShouldHijackDns(metadata.Destination.AddrPort()) {
|
|
log.Debugln("[DNS] hijack tcp:%s", metadata.Destination.String())
|
|
return resolver.RelayDnsConn(ctx, conn, resolver.DefaultDnsReadTimeout)
|
|
}
|
|
return h.ListenerHandler.NewConnection(ctx, conn, metadata)
|
|
}
|
|
|
|
func (h *ListenerHandler) NewPacketConnection(ctx context.Context, conn network.PacketConn, metadata M.Metadata) error {
|
|
if h.ShouldHijackDns(metadata.Destination.AddrPort()) {
|
|
log.Debugln("[DNS] hijack udp:%s from %s", metadata.Destination.String(), metadata.Source.String())
|
|
defer func() { _ = conn.Close() }()
|
|
mutex := sync.Mutex{}
|
|
conn2 := conn // a new interface to set nil in defer
|
|
defer func() {
|
|
mutex.Lock() // this goroutine must exit after all conn.WritePacket() is not running
|
|
defer mutex.Unlock()
|
|
conn2 = nil
|
|
}()
|
|
rwOptions := network.ReadWaitOptions{
|
|
FrontHeadroom: network.CalculateFrontHeadroom(conn),
|
|
RearHeadroom: network.CalculateRearHeadroom(conn),
|
|
MTU: resolver.SafeDnsPacketSize,
|
|
}
|
|
readWaiter, isReadWaiter := bufio.CreatePacketReadWaiter(conn)
|
|
if isReadWaiter {
|
|
readWaiter.InitializeReadWaiter(rwOptions)
|
|
}
|
|
for {
|
|
var (
|
|
readBuff *buf.Buffer
|
|
dest M.Socksaddr
|
|
err error
|
|
)
|
|
_ = conn.SetReadDeadline(time.Now().Add(resolver.DefaultDnsReadTimeout))
|
|
readBuff = nil // clear last loop status, avoid repeat release
|
|
if isReadWaiter {
|
|
readBuff, dest, err = readWaiter.WaitReadPacket()
|
|
} else {
|
|
readBuff = rwOptions.NewPacketBuffer()
|
|
dest, err = conn.ReadPacket(readBuff)
|
|
if readBuff != nil {
|
|
rwOptions.PostReturn(readBuff)
|
|
}
|
|
}
|
|
if err != nil {
|
|
if readBuff != nil {
|
|
readBuff.Release()
|
|
}
|
|
if sing.ShouldIgnorePacketError(err) {
|
|
break
|
|
}
|
|
return err
|
|
}
|
|
go func() {
|
|
ctx, cancel := context.WithTimeout(ctx, resolver.DefaultDnsRelayTimeout)
|
|
defer cancel()
|
|
inData := readBuff.Bytes()
|
|
writeBuff := readBuff
|
|
writeBuff.Resize(writeBuff.Start(), 0)
|
|
if len(writeBuff.FreeBytes()) < resolver.SafeDnsPacketSize { // only create a new buffer when space don't enough
|
|
writeBuff = rwOptions.NewPacketBuffer()
|
|
}
|
|
msg, err := resolver.RelayDnsPacket(ctx, inData, writeBuff.FreeBytes())
|
|
if writeBuff != readBuff {
|
|
readBuff.Release()
|
|
}
|
|
if err != nil {
|
|
writeBuff.Release()
|
|
return
|
|
}
|
|
writeBuff.Truncate(len(msg))
|
|
mutex.Lock()
|
|
defer mutex.Unlock()
|
|
conn := conn2
|
|
if conn == nil {
|
|
writeBuff.Release()
|
|
return
|
|
}
|
|
err = conn.WritePacket(writeBuff, dest) // WritePacket will release writeBuff
|
|
if err != nil {
|
|
writeBuff.Release()
|
|
return
|
|
}
|
|
}()
|
|
}
|
|
return nil
|
|
}
|
|
return h.ListenerHandler.NewPacketConnection(ctx, conn, metadata)
|
|
}
|