Files
gortsplib/servermulticasthandler.go

93 lines
1.6 KiB
Go

package gortsplib
import (
"net"
"github.com/aler9/gortsplib/pkg/ringbuffer"
)
type trackTypePayload struct {
trackID int
isRTP bool
payload []byte
}
type serverMulticastHandler struct {
rtpl *serverUDPListener
rtcpl *serverUDPListener
writeBuffer *ringbuffer.RingBuffer
writerDone chan struct{}
}
func newServerMulticastHandler(s *Server) (*serverMulticastHandler, error) {
rtpl, rtcpl, err := newServerUDPListenerMulticastPair(s)
if err != nil {
return nil, err
}
h := &serverMulticastHandler{
rtpl: rtpl,
rtcpl: rtcpl,
writeBuffer: ringbuffer.New(uint64(s.ReadBufferCount)),
writerDone: make(chan struct{}),
}
go h.runWriter()
return h, nil
}
func (h *serverMulticastHandler) close() {
h.rtpl.close()
h.rtcpl.close()
h.writeBuffer.Close()
<-h.writerDone
}
func (h *serverMulticastHandler) ip() net.IP {
return h.rtpl.ip()
}
func (h *serverMulticastHandler) runWriter() {
defer close(h.writerDone)
rtpAddr := &net.UDPAddr{
IP: h.rtpl.ip(),
Port: h.rtpl.port(),
}
rtcpAddr := &net.UDPAddr{
IP: h.rtcpl.ip(),
Port: h.rtcpl.port(),
}
for {
tmp, ok := h.writeBuffer.Pull()
if !ok {
return
}
data := tmp.(trackTypePayload)
if data.isRTP {
h.rtpl.write(data.payload, rtpAddr)
} else {
h.rtcpl.write(data.payload, rtcpAddr)
}
}
}
func (h *serverMulticastHandler) writeRTP(payload []byte) {
h.writeBuffer.Push(trackTypePayload{
isRTP: true,
payload: payload,
})
}
func (h *serverMulticastHandler) writeRTCP(payload []byte) {
h.writeBuffer.Push(trackTypePayload{
isRTP: false,
payload: payload,
})
}