| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146 | package udpimport (	"v2ray.com/core/common/buf"	"v2ray.com/core/common/net")// Payload represents a single UDP payload.type Payload struct {	Content             *buf.Buffer	Source              net.Destination	OriginalDestination 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	cache        chan *Payload	capacity     int	recvOrigDest bool}func ListenUDP(address net.Address, port net.Port, 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,		recvOrigDest: false,	}	for _, opt := range options {		opt(hub)	}	hub.cache = make(chan *Payload, hub.capacity)	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)		}	}	go hub.start()	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) start() {	c := h.cache	defer close(c)	oobBytes := make([]byte, 256)	for {		buffer := buf.New()		var noob int		var addr *net.UDPAddr		err := buffer.Reset(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		}		if buffer.IsEmpty() {			buffer.Release()			continue		}		payload := &Payload{			Content: buffer,			Source:  net.UDPDestination(net.IPAddress(addr.IP), net.Port(addr.Port)),		}		if h.recvOrigDest && noob > 0 {			payload.OriginalDestination = RetrieveOriginalDest(oobBytes[:noob])			if payload.OriginalDestination.IsValid() {				newError("UDP original destination: ", payload.OriginalDestination).AtDebug().WriteToLog()			} else {				newError("failed to read UDP original destination").WriteToLog()			}		}		select {		case c <- payload:		default:			buffer.Release()			payload.Content = nil		}	}}// Addr implements net.Listener.func (h *Hub) Addr() net.Addr {	return h.conn.LocalAddr()}func (h *Hub) Receive() <-chan *Payload {	return h.cache}
 |