mirror of
https://github.com/bluenviron/mediamtx
synced 2025-01-10 17:09:56 +00:00
110 lines
1.9 KiB
Go
110 lines
1.9 KiB
Go
package hls
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/aler9/gortsplib"
|
|
"github.com/aler9/gortsplib/pkg/aac"
|
|
)
|
|
|
|
type clientAudioProcessorData struct {
|
|
data []byte
|
|
pts time.Duration
|
|
}
|
|
|
|
type clientAudioProcessor struct {
|
|
ctx context.Context
|
|
onTrack func(gortsplib.Track) error
|
|
onData func(time.Duration, [][]byte)
|
|
|
|
trackInitialized bool
|
|
queue chan clientAudioProcessorData
|
|
clockStartRTC time.Time
|
|
}
|
|
|
|
func newClientAudioProcessor(
|
|
ctx context.Context,
|
|
onTrack func(gortsplib.Track) error,
|
|
onData func(time.Duration, [][]byte),
|
|
) *clientAudioProcessor {
|
|
p := &clientAudioProcessor{
|
|
ctx: ctx,
|
|
onTrack: onTrack,
|
|
onData: onData,
|
|
queue: make(chan clientAudioProcessorData, clientQueueSize),
|
|
}
|
|
|
|
return p
|
|
}
|
|
|
|
func (p *clientAudioProcessor) run() error {
|
|
for {
|
|
select {
|
|
case item := <-p.queue:
|
|
err := p.doProcess(item.data, item.pts)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
case <-p.ctx.Done():
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
|
|
func (p *clientAudioProcessor) doProcess(
|
|
data []byte,
|
|
pts time.Duration) error {
|
|
adtsPkts, err := aac.DecodeADTS(data)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
aus := make([][]byte, 0, len(adtsPkts))
|
|
pktPts := pts
|
|
now := time.Now()
|
|
|
|
for _, pkt := range adtsPkts {
|
|
elapsed := now.Sub(p.clockStartRTC)
|
|
|
|
if pktPts > elapsed {
|
|
select {
|
|
case <-p.ctx.Done():
|
|
return fmt.Errorf("terminated")
|
|
case <-time.After(pktPts - elapsed):
|
|
}
|
|
}
|
|
|
|
if !p.trackInitialized {
|
|
p.trackInitialized = true
|
|
|
|
track, err := gortsplib.NewTrackAAC(97, pkt.Type, pkt.SampleRate, pkt.ChannelCount, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
err = p.onTrack(track)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
aus = append(aus, pkt.AU)
|
|
pktPts += 1000 * time.Second / time.Duration(pkt.SampleRate)
|
|
}
|
|
|
|
p.onData(pts, aus)
|
|
return nil
|
|
}
|
|
|
|
func (p *clientAudioProcessor) process(
|
|
data []byte,
|
|
pts time.Duration) {
|
|
select {
|
|
case p.queue <- clientAudioProcessorData{data, pts}:
|
|
case <-p.ctx.Done():
|
|
}
|
|
}
|