mirror of
https://github.com/bluenviron/mediamtx.git
synced 2025-12-20 02:00:05 -08:00
write stream ID, segment number, DTS, NTP in a dedicated box. This allows to improve the merge algorithm in the playback server.
88 lines
2 KiB
Go
88 lines
2 KiB
Go
package recorder
|
|
|
|
import (
|
|
"fmt"
|
|
"io"
|
|
"time"
|
|
|
|
"github.com/bluenviron/mediacommon/v2/pkg/formats/fmp4"
|
|
"github.com/bluenviron/mediacommon/v2/pkg/formats/fmp4/seekablebuffer"
|
|
"github.com/bluenviron/mediamtx/internal/conf"
|
|
)
|
|
|
|
func writePart(
|
|
f io.Writer,
|
|
sequenceNumber uint32,
|
|
partTracks map[*formatFMP4Track]*fmp4.PartTrack,
|
|
) error {
|
|
fmp4PartTracks := make([]*fmp4.PartTrack, len(partTracks))
|
|
i := 0
|
|
for _, partTrack := range partTracks {
|
|
fmp4PartTracks[i] = partTrack
|
|
i++
|
|
}
|
|
|
|
part := &fmp4.Part{
|
|
SequenceNumber: sequenceNumber,
|
|
Tracks: fmp4PartTracks,
|
|
}
|
|
|
|
var buf seekablebuffer.Buffer
|
|
err := part.Marshal(&buf)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
_, err = f.Write(buf.Bytes())
|
|
return err
|
|
}
|
|
|
|
type formatFMP4Part struct {
|
|
maxPartSize conf.StringSize
|
|
segmentStartDTS time.Duration
|
|
number uint32
|
|
startDTS time.Duration
|
|
|
|
partTracks map[*formatFMP4Track]*fmp4.PartTrack
|
|
size uint64
|
|
endDTS time.Duration
|
|
}
|
|
|
|
func (p *formatFMP4Part) initialize() {
|
|
p.partTracks = make(map[*formatFMP4Track]*fmp4.PartTrack)
|
|
}
|
|
|
|
func (p *formatFMP4Part) close(w io.Writer) error {
|
|
return writePart(w, p.number, p.partTracks)
|
|
}
|
|
|
|
func (p *formatFMP4Part) write(track *formatFMP4Track, sample *sample, dts time.Duration) error {
|
|
size := uint64(len(sample.Payload))
|
|
if (p.size + size) > uint64(p.maxPartSize) {
|
|
return fmt.Errorf("reached maximum part size")
|
|
}
|
|
p.size += size
|
|
|
|
partTrack, ok := p.partTracks[track]
|
|
if !ok {
|
|
partTrack = &fmp4.PartTrack{
|
|
ID: track.initTrack.ID,
|
|
BaseTime: uint64(multiplyAndDivide(int64(dts-p.segmentStartDTS),
|
|
int64(track.initTrack.TimeScale), int64(time.Second))),
|
|
}
|
|
p.partTracks[track] = partTrack
|
|
}
|
|
|
|
partTrack.Samples = append(partTrack.Samples, sample.Sample)
|
|
|
|
endDTS := dts + timestampToDuration(int64(sample.Duration), int(track.initTrack.TimeScale))
|
|
if endDTS > p.endDTS {
|
|
p.endDTS = endDTS
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (p *formatFMP4Part) duration() time.Duration {
|
|
return p.endDTS - p.startDTS
|
|
}
|