mirror of
https://github.com/aler9/gortsplib
synced 2025-10-21 06:09:33 +08:00
164 lines
3.6 KiB
Go
164 lines
3.6 KiB
Go
package gortsplib
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"time"
|
|
|
|
"github.com/pion/rtcp"
|
|
"github.com/pion/rtp"
|
|
|
|
"github.com/bluenviron/gortsplib/v3/pkg/formats"
|
|
"github.com/bluenviron/gortsplib/v3/pkg/rtcpreceiver"
|
|
"github.com/bluenviron/gortsplib/v3/pkg/rtcpsender"
|
|
"github.com/bluenviron/gortsplib/v3/pkg/rtplossdetector"
|
|
"github.com/bluenviron/gortsplib/v3/pkg/rtpreorderer"
|
|
)
|
|
|
|
// workaround until this gets tagged:
|
|
// https://github.com/pion/rtp/pull/234
|
|
func rtpPacketMarshalToSafe(p *rtp.Packet, buf []byte) (n int, err error) {
|
|
n, err = p.Header.MarshalTo(buf)
|
|
if err != nil {
|
|
return 0, err
|
|
}
|
|
|
|
// Make sure the buffer is large enough to hold the packet.
|
|
if n+len(p.Payload)+int(p.PaddingSize) > len(buf) {
|
|
return 0, io.ErrShortBuffer
|
|
}
|
|
|
|
m := copy(buf[n:], p.Payload)
|
|
|
|
if p.Header.Padding {
|
|
buf[n+m+int(p.PaddingSize-1)] = p.PaddingSize
|
|
}
|
|
|
|
return n + m + int(p.PaddingSize), nil
|
|
}
|
|
|
|
type clientFormat struct {
|
|
c *Client
|
|
cm *clientMedia
|
|
format formats.Format
|
|
udpReorderer *rtpreorderer.Reorderer // play
|
|
udpRTCPReceiver *rtcpreceiver.RTCPReceiver // play
|
|
tcpLossDetector *rtplossdetector.LossDetector // play
|
|
rtcpSender *rtcpsender.RTCPSender // record
|
|
onPacketRTP OnPacketRTPFunc
|
|
}
|
|
|
|
func newClientFormat(cm *clientMedia, forma formats.Format) *clientFormat {
|
|
return &clientFormat{
|
|
c: cm.c,
|
|
cm: cm,
|
|
format: forma,
|
|
onPacketRTP: func(*rtp.Packet) {},
|
|
}
|
|
}
|
|
|
|
func (ct *clientFormat) start() {
|
|
if ct.cm.c.state == clientStatePlay {
|
|
if ct.cm.udpRTPListener != nil {
|
|
ct.udpReorderer = rtpreorderer.New()
|
|
var err error
|
|
ct.udpRTCPReceiver, err = rtcpreceiver.New(
|
|
ct.cm.c.udpReceiverReportPeriod,
|
|
nil,
|
|
ct.format.ClockRate(), func(pkt rtcp.Packet) {
|
|
ct.cm.writePacketRTCP(pkt)
|
|
})
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
} else {
|
|
ct.tcpLossDetector = rtplossdetector.New()
|
|
}
|
|
} else {
|
|
ct.rtcpSender = rtcpsender.New(
|
|
ct.format.ClockRate(),
|
|
func(pkt rtcp.Packet) {
|
|
ct.cm.writePacketRTCP(pkt)
|
|
})
|
|
}
|
|
}
|
|
|
|
// start writing after write*() has been allocated in order to avoid a crash
|
|
func (ct *clientFormat) startWriting() {
|
|
if ct.c.state != clientStatePlay && !ct.c.DisableRTCPSenderReports {
|
|
ct.rtcpSender.Start(ct.c.senderReportPeriod)
|
|
}
|
|
}
|
|
|
|
func (ct *clientFormat) stop() {
|
|
if ct.udpRTCPReceiver != nil {
|
|
ct.udpRTCPReceiver.Close()
|
|
ct.udpRTCPReceiver = nil
|
|
}
|
|
|
|
if ct.rtcpSender != nil {
|
|
ct.rtcpSender.Close()
|
|
}
|
|
}
|
|
|
|
func (ct *clientFormat) writePacketRTPWithNTP(pkt *rtp.Packet, ntp time.Time) error {
|
|
byts := make([]byte, udpMaxPayloadSize)
|
|
n, err := rtpPacketMarshalToSafe(pkt, byts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
byts = byts[:n]
|
|
|
|
select {
|
|
case <-ct.c.done:
|
|
return ct.c.closeError
|
|
default:
|
|
}
|
|
|
|
ct.c.writer.queue(func() {
|
|
ct.cm.writePacketRTPInQueue(byts)
|
|
})
|
|
|
|
ct.rtcpSender.ProcessPacket(pkt, ntp, ct.format.PTSEqualsDTS(pkt))
|
|
return nil
|
|
}
|
|
|
|
func (ct *clientFormat) readRTPUDP(pkt *rtp.Packet) {
|
|
packets, lost := ct.udpReorderer.Process(pkt)
|
|
if lost != 0 {
|
|
ct.c.OnPacketLost(fmt.Errorf("%d RTP %s lost",
|
|
lost,
|
|
func() string {
|
|
if lost == 1 {
|
|
return "packet"
|
|
}
|
|
return "packets"
|
|
}()))
|
|
// do not return
|
|
}
|
|
|
|
now := time.Now()
|
|
|
|
for _, pkt := range packets {
|
|
ct.udpRTCPReceiver.ProcessPacket(pkt, now, ct.format.PTSEqualsDTS(pkt))
|
|
ct.onPacketRTP(pkt)
|
|
}
|
|
}
|
|
|
|
func (ct *clientFormat) readRTPTCP(pkt *rtp.Packet) {
|
|
lost := ct.tcpLossDetector.Process(pkt)
|
|
if lost != 0 {
|
|
ct.c.OnPacketLost(fmt.Errorf("%d RTP %s lost",
|
|
lost,
|
|
func() string {
|
|
if lost == 1 {
|
|
return "packet"
|
|
}
|
|
return "packets"
|
|
}()))
|
|
// do not return
|
|
}
|
|
|
|
ct.onPacketRTP(pkt)
|
|
}
|