mirror of
https://github.com/aler9/gortsplib
synced 2025-10-04 06:46:42 +08:00
282 lines
5.9 KiB
Go
282 lines
5.9 KiB
Go
package gortsplib
|
|
|
|
import (
|
|
"fmt"
|
|
"net"
|
|
"strconv"
|
|
"sync"
|
|
"sync/atomic"
|
|
"time"
|
|
|
|
"github.com/pion/rtcp"
|
|
"golang.org/x/net/ipv4"
|
|
|
|
"github.com/aler9/gortsplib/pkg/multibuffer"
|
|
)
|
|
|
|
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
|
|
|
|
pc *net.UDPConn
|
|
listenIP net.IP
|
|
isRTP bool
|
|
writeTimeout time.Duration
|
|
readBuffer *multibuffer.MultiBuffer
|
|
rtpPacketBuffer *rtpPacketMultiBuffer
|
|
clientsMutex sync.RWMutex
|
|
clients map[clientAddr]*clientData
|
|
processFunc func(*clientData, []byte)
|
|
|
|
readerDone chan struct{}
|
|
}
|
|
|
|
func newServerUDPListenerMulticastPair(s *Server) (*serverUDPListener, *serverUDPListener, error) {
|
|
res := make(chan net.IP)
|
|
select {
|
|
case s.streamMulticastIP <- streamMulticastIPReq{res: res}:
|
|
case <-s.ctx.Done():
|
|
return nil, nil, fmt.Errorf("terminated")
|
|
}
|
|
ip := <-res
|
|
|
|
rtpl, err := newServerUDPListener(s, true,
|
|
ip.String()+":"+strconv.FormatInt(int64(s.MulticastRTPPort), 10), true)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
|
|
rtcpl, err := newServerUDPListener(s, true,
|
|
ip.String()+":"+strconv.FormatInt(int64(s.MulticastRTCPPort), 10), false)
|
|
if err != nil {
|
|
rtpl.close()
|
|
return nil, nil, err
|
|
}
|
|
|
|
return rtpl, rtcpl, nil
|
|
}
|
|
|
|
func newServerUDPListener(
|
|
s *Server,
|
|
multicast bool,
|
|
address string,
|
|
isRTP bool) (*serverUDPListener, error) {
|
|
var pc *net.UDPConn
|
|
var listenIP net.IP
|
|
if multicast {
|
|
host, port, err := net.SplitHostPort(address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
tmp, err := s.ListenPacket("udp", "224.0.0.0:"+port)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
p := ipv4.NewPacketConn(tmp)
|
|
|
|
err = p.SetTTL(127)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
intfs, err := net.Interfaces()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
listenIP = net.ParseIP(host)
|
|
|
|
for _, intf := range intfs {
|
|
if (intf.Flags & net.FlagMulticast) != 0 {
|
|
err := p.JoinGroup(&intf, &net.UDPAddr{IP: listenIP})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
}
|
|
|
|
pc = tmp.(*net.UDPConn)
|
|
} else {
|
|
tmp, err := s.ListenPacket("udp", address)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
pc = tmp.(*net.UDPConn)
|
|
listenIP = tmp.LocalAddr().(*net.UDPAddr).IP
|
|
}
|
|
|
|
err := pc.SetReadBuffer(serverUDPKernelReadBufferSize)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
u := &serverUDPListener{
|
|
s: s,
|
|
pc: pc,
|
|
listenIP: listenIP,
|
|
clients: make(map[clientAddr]*clientData),
|
|
isRTP: isRTP,
|
|
writeTimeout: s.WriteTimeout,
|
|
readBuffer: multibuffer.New(uint64(s.ReadBufferCount), uint64(s.ReadBufferSize)),
|
|
rtpPacketBuffer: newRTPPacketMultiBuffer(uint64(s.ReadBufferCount)),
|
|
readerDone: make(chan struct{}),
|
|
}
|
|
|
|
if isRTP {
|
|
u.processFunc = u.processRTP
|
|
} else {
|
|
u.processFunc = u.processRTCP
|
|
}
|
|
|
|
go u.runReader()
|
|
|
|
return u, nil
|
|
}
|
|
|
|
func (u *serverUDPListener) close() {
|
|
u.pc.Close()
|
|
<-u.readerDone
|
|
}
|
|
|
|
func (u *serverUDPListener) ip() net.IP {
|
|
return u.listenIP
|
|
}
|
|
|
|
func (u *serverUDPListener) port() int {
|
|
return u.pc.LocalAddr().(*net.UDPAddr).Port
|
|
}
|
|
|
|
func (u *serverUDPListener) runReader() {
|
|
defer close(u.readerDone)
|
|
|
|
for {
|
|
buf := u.readBuffer.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
|
|
}
|
|
|
|
u.processFunc(clientData, buf[:n])
|
|
}()
|
|
}
|
|
}
|
|
|
|
func (u *serverUDPListener) processRTP(clientData *clientData, payload []byte) {
|
|
pkt := u.rtpPacketBuffer.next()
|
|
err := pkt.Unmarshal(payload)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
// remove padding
|
|
pkt.Header.Padding = false
|
|
pkt.PaddingSize = 0
|
|
|
|
now := time.Now()
|
|
atomic.StoreInt64(clientData.ss.udpLastFrameTime, now.Unix())
|
|
clientData.ss.announcedTracks[clientData.trackID].rtcpReceiver.ProcessPacketRTP(now, pkt)
|
|
|
|
if h, ok := u.s.Handler.(ServerHandlerOnPacketRTP); ok {
|
|
h.OnPacketRTP(&ServerHandlerOnPacketRTPCtx{
|
|
Session: clientData.ss,
|
|
TrackID: clientData.trackID,
|
|
Packet: pkt,
|
|
})
|
|
}
|
|
}
|
|
|
|
func (u *serverUDPListener) processRTCP(clientData *clientData, payload []byte) {
|
|
packets, err := rtcp.Unmarshal(payload)
|
|
if err != nil {
|
|
return
|
|
}
|
|
|
|
if clientData.isPublishing {
|
|
now := time.Now()
|
|
atomic.StoreInt64(clientData.ss.udpLastFrameTime, now.Unix())
|
|
|
|
for _, pkt := range packets {
|
|
clientData.ss.announcedTracks[clientData.trackID].rtcpReceiver.ProcessPacketRTCP(now, pkt)
|
|
}
|
|
}
|
|
|
|
if h, ok := u.s.Handler.(ServerHandlerOnPacketRTCP); ok {
|
|
for _, pkt := range packets {
|
|
h.OnPacketRTCP(&ServerHandlerOnPacketRTCPCtx{
|
|
Session: clientData.ss,
|
|
TrackID: clientData.trackID,
|
|
Packet: pkt,
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
func (u *serverUDPListener) write(buf []byte, addr *net.UDPAddr) error {
|
|
// no mutex is needed here since Write() has an internal lock.
|
|
// https://github.com/golang/go/issues/27203#issuecomment-534386117
|
|
|
|
u.pc.SetWriteDeadline(time.Now().Add(u.writeTimeout))
|
|
_, err := u.pc.WriteTo(buf, addr)
|
|
return err
|
|
}
|
|
|
|
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)
|
|
}
|
|
}
|
|
}
|