mirror of https://github.com/v2ray/v2ray-core
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
146 lines
3.0 KiB
146 lines
3.0 KiB
package udp
|
|
|
|
import (
|
|
"v2ray.com/core/common/buf"
|
|
"v2ray.com/core/common/net"
|
|
)
|
|
|
|
// Payload represents a single UDP payload.
|
|
type Payload struct {
|
|
payload *buf.Buffer
|
|
source net.Destination
|
|
originalDest net.Destination
|
|
}
|
|
|
|
// PayloadHandler is function to handle Payload.
|
|
type PayloadHandler func(payload *buf.Buffer, source net.Destination, originalDest net.Destination)
|
|
|
|
type HubOption func(h *Hub)
|
|
|
|
func HubCapacity(cap int) HubOption {
|
|
return func(h *Hub) {
|
|
h.capacity = cap
|
|
}
|
|
}
|
|
|
|
func HubReceiveOriginalDestination(r bool) HubOption {
|
|
return func(h *Hub) {
|
|
h.recvOrigDest = r
|
|
}
|
|
}
|
|
|
|
type Hub struct {
|
|
conn *net.UDPConn
|
|
callback PayloadHandler
|
|
capacity int
|
|
recvOrigDest bool
|
|
}
|
|
|
|
func ListenUDP(address net.Address, port net.Port, callback PayloadHandler, options ...HubOption) (*Hub, error) {
|
|
udpConn, err := net.ListenUDP("udp", &net.UDPAddr{
|
|
IP: address.IP(),
|
|
Port: int(port),
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
newError("listening UDP on ", address, ":", port).WriteToLog()
|
|
hub := &Hub{
|
|
conn: udpConn,
|
|
capacity: 256,
|
|
callback: callback,
|
|
recvOrigDest: false,
|
|
}
|
|
for _, opt := range options {
|
|
opt(hub)
|
|
}
|
|
|
|
if hub.recvOrigDest {
|
|
rawConn, err := udpConn.SyscallConn()
|
|
if err != nil {
|
|
return nil, newError("failed to get fd").Base(err)
|
|
}
|
|
err = rawConn.Control(func(fd uintptr) {
|
|
if err := SetOriginalDestOptions(int(fd)); err != nil {
|
|
newError("failed to set socket options").Base(err).WriteToLog()
|
|
}
|
|
})
|
|
if err != nil {
|
|
return nil, newError("failed to control socket").Base(err)
|
|
}
|
|
}
|
|
|
|
c := make(chan *Payload, hub.capacity)
|
|
|
|
go hub.start(c)
|
|
go hub.process(c)
|
|
return hub, nil
|
|
}
|
|
|
|
// Close implements net.Listener.
|
|
func (h *Hub) Close() error {
|
|
h.conn.Close()
|
|
return nil
|
|
}
|
|
|
|
func (h *Hub) WriteTo(payload []byte, dest net.Destination) (int, error) {
|
|
return h.conn.WriteToUDP(payload, &net.UDPAddr{
|
|
IP: dest.Address.IP(),
|
|
Port: int(dest.Port),
|
|
})
|
|
}
|
|
|
|
func (h *Hub) process(c <-chan *Payload) {
|
|
for p := range c {
|
|
h.callback(p.payload, p.source, p.originalDest)
|
|
}
|
|
}
|
|
|
|
func (h *Hub) start(c chan<- *Payload) {
|
|
defer close(c)
|
|
|
|
oobBytes := make([]byte, 256)
|
|
|
|
for {
|
|
buffer := buf.New()
|
|
var noob int
|
|
var addr *net.UDPAddr
|
|
err := buffer.AppendSupplier(func(b []byte) (int, error) {
|
|
n, nb, _, a, e := ReadUDPMsg(h.conn, b, oobBytes)
|
|
noob = nb
|
|
addr = a
|
|
return n, e
|
|
})
|
|
|
|
if err != nil {
|
|
newError("failed to read UDP msg").Base(err).WriteToLog()
|
|
buffer.Release()
|
|
break
|
|
}
|
|
|
|
payload := &Payload{
|
|
payload: buffer,
|
|
}
|
|
payload.source = net.UDPDestination(net.IPAddress(addr.IP), net.Port(addr.Port))
|
|
if h.recvOrigDest && noob > 0 {
|
|
payload.originalDest = RetrieveOriginalDest(oobBytes[:noob])
|
|
if payload.originalDest.IsValid() {
|
|
newError("UDP original destination: ", payload.originalDest).AtDebug().WriteToLog()
|
|
} else {
|
|
newError("failed to read UDP original destination").WriteToLog()
|
|
}
|
|
}
|
|
|
|
select {
|
|
case c <- payload:
|
|
default:
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
// Addr implements net.Listener.
|
|
func (h *Hub) Addr() net.Addr {
|
|
return h.conn.LocalAddr()
|
|
}
|