| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145 | package udpimport (	"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:     16,		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()}
 |