Files
gortsplib/server_stream_media.go
2025-07-26 16:28:03 +02:00

158 lines
3.0 KiB
Go

package gortsplib
import (
"crypto/rand"
"fmt"
"sync/atomic"
"github.com/bluenviron/gortsplib/v4/pkg/description"
"github.com/pion/rtcp"
)
type serverStreamMedia struct {
st *ServerStream
media *description.Media
trackID int
srtpOutCtx *wrappedSRTPContext
formats map[uint8]*serverStreamFormat
multicastWriter *serverMulticastWriter
bytesSent *uint64
rtcpPacketsSent *uint64
}
func (sm *serverStreamMedia) initialize() error {
sm.bytesSent = new(uint64)
sm.rtcpPacketsSent = new(uint64)
sm.formats = make(map[uint8]*serverStreamFormat)
for i, forma := range sm.media.Formats {
sf := &serverStreamFormat{
sm: sm,
format: forma,
}
err := sf.initialize()
if err != nil {
for _, forma := range sm.media.Formats[:i] {
sm.formats[forma.PayloadType()].close()
}
return err
}
sm.formats[forma.PayloadType()] = sf
}
if sm.st.Server.TLSConfig != nil {
srtpOutKey := make([]byte, srtpKeyLength)
_, err := rand.Read(srtpOutKey)
if err != nil {
return err
}
ssrcs := make([]uint32, len(sm.formats))
n := 0
for _, cf := range sm.formats {
ssrcs[n] = cf.localSSRC
n++
}
sm.srtpOutCtx = &wrappedSRTPContext{
key: srtpOutKey,
ssrcs: ssrcs,
}
err = sm.srtpOutCtx.initialize()
if err != nil {
return err
}
}
return nil
}
func (sm *serverStreamMedia) close() {
for _, sf := range sm.formats {
sf.close()
}
if sm.multicastWriter != nil {
sm.multicastWriter.close()
}
}
func (sm *serverStreamMedia) writePacketRTCP(pkt rtcp.Packet) error {
plain, err := pkt.Marshal()
if err != nil {
return err
}
maxPlainPacketSize := sm.st.Server.MaxPacketSize
if sm.srtpOutCtx != nil {
maxPlainPacketSize -= srtcpOverhead
}
if len(plain) > maxPlainPacketSize {
return fmt.Errorf("packet is too big")
}
var encr []byte
if sm.srtpOutCtx != nil {
encr = make([]byte, sm.st.Server.MaxPacketSize)
encr, err = sm.srtpOutCtx.encryptRTCP(encr, plain, nil)
if err != nil {
return err
}
}
encrLen := uint64(len(encr))
plainLen := uint64(len(plain))
// send unicast
for r := range sm.st.activeUnicastReaders {
if sm, ok := r.setuppedMedias[sm.media]; ok {
if r.setuppedSecure {
err = sm.writePacketRTCPEncoded(encr)
if err != nil {
r.onStreamWriteError(err)
continue
}
atomic.AddUint64(sm.bytesSent, encrLen)
} else {
err = sm.writePacketRTCPEncoded(plain)
if err != nil {
r.onStreamWriteError(err)
continue
}
atomic.AddUint64(sm.bytesSent, plainLen)
}
atomic.AddUint64(sm.rtcpPacketsSent, 1)
}
}
// send multicast
if sm.multicastWriter != nil {
if sm.srtpOutCtx != nil {
err = sm.multicastWriter.writePacketRTCP(encr)
if err != nil {
return err
}
atomic.AddUint64(sm.bytesSent, encrLen)
} else {
err = sm.multicastWriter.writePacketRTCP(plain)
if err != nil {
return err
}
atomic.AddUint64(sm.bytesSent, plainLen)
}
atomic.AddUint64(sm.rtcpPacketsSent, 1)
}
return nil
}