mirror of
				https://github.com/aler9/gortsplib
				synced 2025-10-31 18:42:40 +08:00 
			
		
		
		
	
		
			
				
	
	
		
			198 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			198 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package gortsplib
 | |
| 
 | |
| import (
 | |
| 	"context"
 | |
| 	"net"
 | |
| 	"sync"
 | |
| 	"sync/atomic"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/aler9/gortsplib/pkg/multibuffer"
 | |
| 	"github.com/aler9/gortsplib/pkg/ringbuffer"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	serverConnUDPListenerKernelReadBufferSize = 0x80000 // same as gstreamer's rtspsrc
 | |
| )
 | |
| 
 | |
| type bufAddrPair struct {
 | |
| 	buf  []byte
 | |
| 	addr *net.UDPAddr
 | |
| }
 | |
| 
 | |
| type clientData struct {
 | |
| 	ss           *ServerSession
 | |
| 	trackID      int
 | |
| 	isPublishing bool
 | |
| }
 | |
| 
 | |
| type clientAddr struct {
 | |
| 	ip   [net.IPv6len]byte // use a fixed-size array to enable the equality operator
 | |
| 	port int
 | |
| }
 | |
| 
 | |
| func (p *clientAddr) fill(ip net.IP, port int) {
 | |
| 	p.port = port
 | |
| 
 | |
| 	if len(ip) == net.IPv4len {
 | |
| 		copy(p.ip[0:], []byte{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0xff, 0xff}) // v4InV6Prefix
 | |
| 		copy(p.ip[12:], ip)
 | |
| 	} else {
 | |
| 		copy(p.ip[:], ip)
 | |
| 	}
 | |
| }
 | |
| 
 | |
| type serverUDPListener struct {
 | |
| 	s *Server
 | |
| 
 | |
| 	ctx          context.Context
 | |
| 	ctxCancel    func()
 | |
| 	wg           sync.WaitGroup
 | |
| 	pc           *net.UDPConn
 | |
| 	streamType   StreamType
 | |
| 	writeTimeout time.Duration
 | |
| 	readBuf      *multibuffer.MultiBuffer
 | |
| 	clientsMutex sync.RWMutex
 | |
| 	clients      map[clientAddr]*clientData
 | |
| 	ringBuffer   *ringbuffer.RingBuffer
 | |
| }
 | |
| 
 | |
| func newServerUDPListener(
 | |
| 	s *Server,
 | |
| 	address string,
 | |
| 	streamType StreamType) (*serverUDPListener, error) {
 | |
| 	tmp, err := s.ListenPacket("udp", address)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	pc := tmp.(*net.UDPConn)
 | |
| 
 | |
| 	err = pc.SetReadBuffer(serverConnUDPListenerKernelReadBufferSize)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	ctx, ctxCancel := context.WithCancel(context.Background())
 | |
| 
 | |
| 	u := &serverUDPListener{
 | |
| 		s:         s,
 | |
| 		ctx:       ctx,
 | |
| 		ctxCancel: ctxCancel,
 | |
| 		pc:        pc,
 | |
| 		clients:   make(map[clientAddr]*clientData),
 | |
| 	}
 | |
| 
 | |
| 	u.streamType = streamType
 | |
| 	u.writeTimeout = s.WriteTimeout
 | |
| 	u.readBuf = multibuffer.New(uint64(s.ReadBufferCount), uint64(s.ReadBufferSize))
 | |
| 	u.ringBuffer = ringbuffer.New(uint64(s.ReadBufferCount))
 | |
| 
 | |
| 	u.wg.Add(1)
 | |
| 	go u.run()
 | |
| 
 | |
| 	return u, nil
 | |
| }
 | |
| 
 | |
| func (u *serverUDPListener) close() {
 | |
| 	u.ctxCancel()
 | |
| 	u.wg.Wait()
 | |
| }
 | |
| 
 | |
| func (u *serverUDPListener) run() {
 | |
| 	defer u.wg.Done()
 | |
| 
 | |
| 	u.wg.Add(1)
 | |
| 	go func() {
 | |
| 		defer u.wg.Done()
 | |
| 
 | |
| 		for {
 | |
| 			buf := u.readBuf.Next()
 | |
| 			n, addr, err := u.pc.ReadFromUDP(buf)
 | |
| 			if err != nil {
 | |
| 				break
 | |
| 			}
 | |
| 
 | |
| 			func() {
 | |
| 				u.clientsMutex.RLock()
 | |
| 				defer u.clientsMutex.RUnlock()
 | |
| 
 | |
| 				var clientAddr clientAddr
 | |
| 				clientAddr.fill(addr.IP, addr.Port)
 | |
| 				clientData, ok := u.clients[clientAddr]
 | |
| 				if !ok {
 | |
| 					return
 | |
| 				}
 | |
| 
 | |
| 				if clientData.isPublishing {
 | |
| 					now := time.Now()
 | |
| 					atomic.StoreInt64(clientData.ss.udpLastFrameTime, now.Unix())
 | |
| 					clientData.ss.announcedTracks[clientData.trackID].rtcpReceiver.ProcessFrame(now, u.streamType, buf[:n])
 | |
| 				}
 | |
| 
 | |
| 				if h, ok := u.s.Handler.(ServerHandlerOnFrame); ok {
 | |
| 					h.OnFrame(&ServerHandlerOnFrameCtx{
 | |
| 						Session:    clientData.ss,
 | |
| 						TrackID:    clientData.trackID,
 | |
| 						StreamType: u.streamType,
 | |
| 						Payload:    buf[:n],
 | |
| 					})
 | |
| 				}
 | |
| 			}()
 | |
| 		}
 | |
| 	}()
 | |
| 
 | |
| 	u.wg.Add(1)
 | |
| 	go func() {
 | |
| 		defer u.wg.Done()
 | |
| 
 | |
| 		for {
 | |
| 			tmp, ok := u.ringBuffer.Pull()
 | |
| 			if !ok {
 | |
| 				return
 | |
| 			}
 | |
| 			pair := tmp.(bufAddrPair)
 | |
| 
 | |
| 			u.pc.SetWriteDeadline(time.Now().Add(u.writeTimeout))
 | |
| 			u.pc.WriteTo(pair.buf, pair.addr)
 | |
| 		}
 | |
| 	}()
 | |
| 
 | |
| 	<-u.ctx.Done()
 | |
| 
 | |
| 	u.pc.Close()
 | |
| 	u.ringBuffer.Close()
 | |
| }
 | |
| 
 | |
| func (u *serverUDPListener) port() int {
 | |
| 	return u.pc.LocalAddr().(*net.UDPAddr).Port
 | |
| }
 | |
| 
 | |
| func (u *serverUDPListener) write(buf []byte, addr *net.UDPAddr) {
 | |
| 	u.ringBuffer.Push(bufAddrPair{buf, addr})
 | |
| }
 | |
| 
 | |
| func (u *serverUDPListener) addClient(ip net.IP, port int, ss *ServerSession, trackID int, isPublishing bool) {
 | |
| 	u.clientsMutex.Lock()
 | |
| 	defer u.clientsMutex.Unlock()
 | |
| 
 | |
| 	var addr clientAddr
 | |
| 	addr.fill(ip, port)
 | |
| 
 | |
| 	u.clients[addr] = &clientData{
 | |
| 		ss:           ss,
 | |
| 		trackID:      trackID,
 | |
| 		isPublishing: isPublishing,
 | |
| 	}
 | |
| }
 | |
| 
 | |
| func (u *serverUDPListener) removeClient(ss *ServerSession) {
 | |
| 	u.clientsMutex.Lock()
 | |
| 	defer u.clientsMutex.Unlock()
 | |
| 
 | |
| 	for addr, data := range u.clients {
 | |
| 		if data.ss == ss {
 | |
| 			delete(u.clients, addr)
 | |
| 		}
 | |
| 	}
 | |
| }
 | 
