mirror of
https://github.com/livekit/livekit.git
synced 2026-04-01 06:35:57 +00:00
* Address another panic. If receiver is not resolved yet, prevent race of access dummy receiver. * reset callback only if there is receiver * missed return
490 lines
12 KiB
Go
490 lines
12 KiB
Go
// Copyright 2023 LiveKit, Inc.
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package rtc
|
|
|
|
import (
|
|
"errors"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/pion/webrtc/v4"
|
|
"go.uber.org/atomic"
|
|
"golang.org/x/exp/slices"
|
|
|
|
"github.com/livekit/protocol/livekit"
|
|
"github.com/livekit/protocol/logger"
|
|
|
|
"github.com/livekit/livekit-server/pkg/sfu"
|
|
)
|
|
|
|
// wrapper around WebRTC receiver, overriding its ID
|
|
|
|
type WrappedReceiverParams struct {
|
|
Receivers []*simulcastReceiver
|
|
TrackID livekit.TrackID
|
|
StreamId string
|
|
UpstreamCodecs []webrtc.RTPCodecParameters
|
|
Logger logger.Logger
|
|
DisableRed bool
|
|
}
|
|
|
|
type WrappedReceiver struct {
|
|
lock sync.Mutex
|
|
|
|
sfu.TrackReceiver
|
|
params WrappedReceiverParams
|
|
receivers []sfu.TrackReceiver
|
|
codecs []webrtc.RTPCodecParameters
|
|
determinedCodec webrtc.RTPCodecCapability
|
|
onReadyCallbacks []func()
|
|
}
|
|
|
|
func NewWrappedReceiver(params WrappedReceiverParams) *WrappedReceiver {
|
|
sfuReceivers := make([]sfu.TrackReceiver, 0, len(params.Receivers))
|
|
for _, r := range params.Receivers {
|
|
sfuReceivers = append(sfuReceivers, r.TrackReceiver)
|
|
}
|
|
|
|
codecs := params.UpstreamCodecs
|
|
if len(codecs) == 1 {
|
|
if strings.EqualFold(codecs[0].MimeType, sfu.MimeTypeAudioRed) {
|
|
// if upstream is opus/red, then add opus to match clients that don't support red
|
|
codecs = append(codecs, webrtc.RTPCodecParameters{
|
|
RTPCodecCapability: OpusCodecCapability,
|
|
PayloadType: 111,
|
|
})
|
|
} else if !params.DisableRed && strings.EqualFold(codecs[0].MimeType, webrtc.MimeTypeOpus) {
|
|
// if upstream is opus only and red enabled, add red to match clients that support red
|
|
codecs = append(codecs, webrtc.RTPCodecParameters{
|
|
RTPCodecCapability: RedCodecCapability,
|
|
PayloadType: 63,
|
|
})
|
|
// prefer red codec
|
|
codecs[0], codecs[1] = codecs[1], codecs[0]
|
|
}
|
|
}
|
|
|
|
return &WrappedReceiver{
|
|
params: params,
|
|
receivers: sfuReceivers,
|
|
codecs: codecs,
|
|
}
|
|
}
|
|
|
|
func (r *WrappedReceiver) TrackID() livekit.TrackID {
|
|
return r.params.TrackID
|
|
}
|
|
|
|
func (r *WrappedReceiver) StreamID() string {
|
|
return r.params.StreamId
|
|
}
|
|
|
|
// DetermineReceiver determines the receiver of negotiated codec and return ready state of the receiver
|
|
func (r *WrappedReceiver) DetermineReceiver(codec webrtc.RTPCodecCapability) bool {
|
|
r.lock.Lock()
|
|
r.determinedCodec = codec
|
|
|
|
var trackReceiver sfu.TrackReceiver
|
|
for _, receiver := range r.receivers {
|
|
if c := receiver.Codec(); strings.EqualFold(c.MimeType, codec.MimeType) {
|
|
trackReceiver = receiver
|
|
break
|
|
} else if strings.EqualFold(c.MimeType, sfu.MimeTypeAudioRed) && strings.EqualFold(codec.MimeType, webrtc.MimeTypeOpus) {
|
|
// audio opus/red can match opus only
|
|
trackReceiver = receiver.GetPrimaryReceiverForRed()
|
|
break
|
|
} else if strings.EqualFold(c.MimeType, webrtc.MimeTypeOpus) && strings.EqualFold(codec.MimeType, sfu.MimeTypeAudioRed) {
|
|
trackReceiver = receiver.GetRedReceiver()
|
|
break
|
|
}
|
|
}
|
|
if trackReceiver == nil {
|
|
r.params.Logger.Errorw("can't determine receiver for codec", nil, "codec", codec.MimeType)
|
|
if len(r.receivers) > 0 {
|
|
trackReceiver = r.receivers[0]
|
|
}
|
|
}
|
|
r.TrackReceiver = trackReceiver
|
|
|
|
var onReadyCallbacks []func()
|
|
if trackReceiver != nil {
|
|
onReadyCallbacks = r.onReadyCallbacks
|
|
r.onReadyCallbacks = nil
|
|
}
|
|
r.lock.Unlock()
|
|
|
|
if trackReceiver != nil {
|
|
for _, f := range onReadyCallbacks {
|
|
trackReceiver.AddOnReady(f)
|
|
}
|
|
|
|
if d, ok := trackReceiver.(*DummyReceiver); ok {
|
|
return d.IsReady()
|
|
}
|
|
return true
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (r *WrappedReceiver) Codecs() []webrtc.RTPCodecParameters {
|
|
return slices.Clone(r.codecs)
|
|
}
|
|
|
|
func (r *WrappedReceiver) DeleteDownTrack(participantID livekit.ParticipantID) {
|
|
r.lock.Lock()
|
|
trackReceiver := r.TrackReceiver
|
|
r.lock.Unlock()
|
|
|
|
if trackReceiver != nil {
|
|
trackReceiver.DeleteDownTrack(participantID)
|
|
}
|
|
}
|
|
|
|
func (r *WrappedReceiver) AddOnReady(f func()) {
|
|
r.lock.Lock()
|
|
trackReceiver := r.TrackReceiver
|
|
if trackReceiver == nil {
|
|
r.onReadyCallbacks = append(r.onReadyCallbacks, f)
|
|
r.lock.Unlock()
|
|
return
|
|
}
|
|
r.lock.Unlock()
|
|
|
|
trackReceiver.AddOnReady(f)
|
|
}
|
|
|
|
// --------------------------------------------
|
|
|
|
type DummyReceiver struct {
|
|
receiver atomic.Value
|
|
trackID livekit.TrackID
|
|
streamId string
|
|
codec webrtc.RTPCodecParameters
|
|
headerExtensions []webrtc.RTPHeaderExtensionParameter
|
|
|
|
downTrackLock sync.Mutex
|
|
downTracks map[livekit.ParticipantID]sfu.TrackSender
|
|
onReadyCallbacks []func()
|
|
|
|
settingsLock sync.Mutex
|
|
maxExpectedLayerValid bool
|
|
maxExpectedLayer int32
|
|
pausedValid bool
|
|
paused bool
|
|
|
|
redReceiver, primaryReceiver *DummyRedReceiver
|
|
}
|
|
|
|
func NewDummyReceiver(trackID livekit.TrackID, streamId string, codec webrtc.RTPCodecParameters, headerExtensions []webrtc.RTPHeaderExtensionParameter) *DummyReceiver {
|
|
return &DummyReceiver{
|
|
trackID: trackID,
|
|
streamId: streamId,
|
|
codec: codec,
|
|
headerExtensions: headerExtensions,
|
|
downTracks: make(map[livekit.ParticipantID]sfu.TrackSender),
|
|
}
|
|
}
|
|
|
|
func (d *DummyReceiver) Receiver() sfu.TrackReceiver {
|
|
r, _ := d.receiver.Load().(sfu.TrackReceiver)
|
|
return r
|
|
}
|
|
|
|
func (d *DummyReceiver) Upgrade(receiver sfu.TrackReceiver) {
|
|
if !d.receiver.CompareAndSwap(nil, receiver) {
|
|
return
|
|
}
|
|
|
|
d.downTrackLock.Lock()
|
|
for _, t := range d.downTracks {
|
|
receiver.AddDownTrack(t)
|
|
}
|
|
d.downTracks = make(map[livekit.ParticipantID]sfu.TrackSender)
|
|
onReadyCallbacks := d.onReadyCallbacks
|
|
d.onReadyCallbacks = nil
|
|
d.downTrackLock.Unlock()
|
|
|
|
for _, f := range onReadyCallbacks {
|
|
receiver.AddOnReady(f)
|
|
}
|
|
|
|
d.settingsLock.Lock()
|
|
if d.maxExpectedLayerValid {
|
|
receiver.SetMaxExpectedSpatialLayer(d.maxExpectedLayer)
|
|
}
|
|
d.maxExpectedLayerValid = false
|
|
|
|
if d.pausedValid {
|
|
receiver.SetUpTrackPaused(d.paused)
|
|
}
|
|
d.pausedValid = false
|
|
|
|
if d.primaryReceiver != nil {
|
|
d.primaryReceiver.upgrade(receiver)
|
|
}
|
|
if d.redReceiver != nil {
|
|
d.redReceiver.upgrade(receiver)
|
|
}
|
|
d.settingsLock.Unlock()
|
|
}
|
|
|
|
func (d *DummyReceiver) TrackID() livekit.TrackID {
|
|
return d.trackID
|
|
}
|
|
|
|
func (d *DummyReceiver) StreamID() string {
|
|
return d.streamId
|
|
}
|
|
|
|
func (d *DummyReceiver) Codec() webrtc.RTPCodecParameters {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.Codec()
|
|
}
|
|
return d.codec
|
|
}
|
|
|
|
func (d *DummyReceiver) HeaderExtensions() []webrtc.RTPHeaderExtensionParameter {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.HeaderExtensions()
|
|
}
|
|
return d.headerExtensions
|
|
}
|
|
|
|
func (d *DummyReceiver) ReadRTP(buf []byte, layer uint8, esn uint64) (int, error) {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.ReadRTP(buf, layer, esn)
|
|
}
|
|
return 0, errors.New("no receiver")
|
|
}
|
|
|
|
func (d *DummyReceiver) GetLayeredBitrate() ([]int32, sfu.Bitrates) {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.GetLayeredBitrate()
|
|
}
|
|
return nil, sfu.Bitrates{}
|
|
}
|
|
|
|
func (d *DummyReceiver) GetAudioLevel() (float64, bool) {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.GetAudioLevel()
|
|
}
|
|
return 0, false
|
|
}
|
|
|
|
func (d *DummyReceiver) SendPLI(layer int32, force bool) {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
r.SendPLI(layer, force)
|
|
}
|
|
}
|
|
|
|
func (d *DummyReceiver) SetUpTrackPaused(paused bool) {
|
|
d.settingsLock.Lock()
|
|
defer d.settingsLock.Unlock()
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
d.pausedValid = false
|
|
r.SetUpTrackPaused(paused)
|
|
} else {
|
|
d.pausedValid = true
|
|
d.paused = paused
|
|
}
|
|
}
|
|
|
|
func (d *DummyReceiver) SetMaxExpectedSpatialLayer(layer int32) {
|
|
d.settingsLock.Lock()
|
|
defer d.settingsLock.Unlock()
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
d.maxExpectedLayerValid = false
|
|
r.SetMaxExpectedSpatialLayer(layer)
|
|
} else {
|
|
d.maxExpectedLayerValid = true
|
|
d.maxExpectedLayer = layer
|
|
}
|
|
}
|
|
|
|
func (d *DummyReceiver) AddDownTrack(track sfu.TrackSender) error {
|
|
d.downTrackLock.Lock()
|
|
defer d.downTrackLock.Unlock()
|
|
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
r.AddDownTrack(track)
|
|
} else {
|
|
d.downTracks[track.SubscriberID()] = track
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (d *DummyReceiver) DeleteDownTrack(subscriberID livekit.ParticipantID) {
|
|
d.downTrackLock.Lock()
|
|
defer d.downTrackLock.Unlock()
|
|
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
r.DeleteDownTrack(subscriberID)
|
|
} else {
|
|
delete(d.downTracks, subscriberID)
|
|
}
|
|
}
|
|
|
|
func (d *DummyReceiver) DebugInfo() map[string]interface{} {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.DebugInfo()
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (d *DummyReceiver) GetTemporalLayerFpsForSpatial(spatial int32) []float32 {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.GetTemporalLayerFpsForSpatial(spatial)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (d *DummyReceiver) TrackInfo() *livekit.TrackInfo {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.TrackInfo()
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (d *DummyReceiver) UpdateTrackInfo(ti *livekit.TrackInfo) {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
r.UpdateTrackInfo(ti)
|
|
}
|
|
}
|
|
|
|
func (d *DummyReceiver) IsClosed() bool {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.IsClosed()
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (d *DummyReceiver) GetPrimaryReceiverForRed() sfu.TrackReceiver {
|
|
d.settingsLock.Lock()
|
|
defer d.settingsLock.Unlock()
|
|
|
|
if d.primaryReceiver == nil {
|
|
d.primaryReceiver = NewDummyRedReceiver(d, false)
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
d.primaryReceiver.upgrade(r)
|
|
}
|
|
}
|
|
return d.primaryReceiver
|
|
}
|
|
|
|
func (d *DummyReceiver) GetRedReceiver() sfu.TrackReceiver {
|
|
d.settingsLock.Lock()
|
|
defer d.settingsLock.Unlock()
|
|
if d.redReceiver == nil {
|
|
d.redReceiver = NewDummyRedReceiver(d, true)
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
d.redReceiver.upgrade(r)
|
|
}
|
|
}
|
|
return d.redReceiver
|
|
}
|
|
|
|
func (d *DummyReceiver) GetTrackStats() *livekit.RTPStats {
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.GetTrackStats()
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (d *DummyReceiver) AddOnReady(f func()) {
|
|
var receiver sfu.TrackReceiver
|
|
d.downTrackLock.Lock()
|
|
if r, ok := d.receiver.Load().(sfu.TrackReceiver); ok {
|
|
receiver = r
|
|
} else {
|
|
d.onReadyCallbacks = append(d.onReadyCallbacks, f)
|
|
}
|
|
d.downTrackLock.Unlock()
|
|
if receiver != nil {
|
|
receiver.AddOnReady(f)
|
|
}
|
|
}
|
|
|
|
func (d *DummyReceiver) IsReady() bool {
|
|
return d.receiver.Load() != nil
|
|
}
|
|
|
|
// --------------------------------------------
|
|
|
|
type DummyRedReceiver struct {
|
|
*DummyReceiver
|
|
redReceiver atomic.Value // sfu.TrackReceiver
|
|
// indicates this receiver is for RED encoding receiver of primary codec OR
|
|
// primary decoding receiver of RED codec
|
|
isRedEncoding bool
|
|
|
|
downTrackLock sync.Mutex
|
|
downTracks map[livekit.ParticipantID]sfu.TrackSender
|
|
}
|
|
|
|
func NewDummyRedReceiver(d *DummyReceiver, isRedEncoding bool) *DummyRedReceiver {
|
|
return &DummyRedReceiver{
|
|
DummyReceiver: d,
|
|
isRedEncoding: isRedEncoding,
|
|
downTracks: make(map[livekit.ParticipantID]sfu.TrackSender),
|
|
}
|
|
}
|
|
|
|
func (d *DummyRedReceiver) AddDownTrack(track sfu.TrackSender) error {
|
|
d.downTrackLock.Lock()
|
|
defer d.downTrackLock.Unlock()
|
|
|
|
if r, ok := d.redReceiver.Load().(sfu.TrackReceiver); ok {
|
|
r.AddDownTrack(track)
|
|
} else {
|
|
d.downTracks[track.SubscriberID()] = track
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (d *DummyRedReceiver) DeleteDownTrack(subscriberID livekit.ParticipantID) {
|
|
d.downTrackLock.Lock()
|
|
defer d.downTrackLock.Unlock()
|
|
|
|
if r, ok := d.redReceiver.Load().(sfu.TrackReceiver); ok {
|
|
r.DeleteDownTrack(subscriberID)
|
|
} else {
|
|
delete(d.downTracks, subscriberID)
|
|
}
|
|
}
|
|
|
|
func (d *DummyRedReceiver) ReadRTP(buf []byte, layer uint8, esn uint64) (int, error) {
|
|
if r, ok := d.redReceiver.Load().(sfu.TrackReceiver); ok {
|
|
return r.ReadRTP(buf, layer, esn)
|
|
}
|
|
return 0, errors.New("no receiver")
|
|
}
|
|
|
|
func (d *DummyRedReceiver) upgrade(receiver sfu.TrackReceiver) {
|
|
var redReceiver sfu.TrackReceiver
|
|
if d.isRedEncoding {
|
|
redReceiver = receiver.GetRedReceiver()
|
|
} else {
|
|
redReceiver = receiver.GetPrimaryReceiverForRed()
|
|
}
|
|
d.redReceiver.Store(redReceiver)
|
|
|
|
d.downTrackLock.Lock()
|
|
for _, t := range d.downTracks {
|
|
redReceiver.AddDownTrack(t)
|
|
}
|
|
d.downTracks = make(map[livekit.ParticipantID]sfu.TrackSender)
|
|
d.downTrackLock.Unlock()
|
|
}
|