mirror of
https://github.com/aler9/gortsplib
synced 2025-10-06 07:37:07 +08:00
use mediacommon/pkg/mpegts in examples
This commit is contained in:
@@ -2,15 +2,18 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"context"
|
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/asticode/go-astits"
|
|
||||||
"github.com/bluenviron/mediacommon/pkg/codecs/h264"
|
"github.com/bluenviron/mediacommon/pkg/codecs/h264"
|
||||||
|
"github.com/bluenviron/mediacommon/pkg/formats/mpegts"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func durationGoToMPEGTS(v time.Duration) int64 {
|
||||||
|
return int64(v.Seconds() * 90000)
|
||||||
|
}
|
||||||
|
|
||||||
// mpegtsMuxer allows to save a H264 stream into a MPEG-TS file.
|
// mpegtsMuxer allows to save a H264 stream into a MPEG-TS file.
|
||||||
type mpegtsMuxer struct {
|
type mpegtsMuxer struct {
|
||||||
sps []byte
|
sps []byte
|
||||||
@@ -18,7 +21,8 @@ type mpegtsMuxer struct {
|
|||||||
|
|
||||||
f *os.File
|
f *os.File
|
||||||
b *bufio.Writer
|
b *bufio.Writer
|
||||||
mux *astits.Muxer
|
w *mpegts.Writer
|
||||||
|
track *mpegts.Track
|
||||||
dtsExtractor *h264.DTSExtractor
|
dtsExtractor *h264.DTSExtractor
|
||||||
firstIDRReceived bool
|
firstIDRReceived bool
|
||||||
startDTS time.Duration
|
startDTS time.Duration
|
||||||
@@ -32,19 +36,20 @@ func newMPEGTSMuxer(sps []byte, pps []byte) (*mpegtsMuxer, error) {
|
|||||||
}
|
}
|
||||||
b := bufio.NewWriter(f)
|
b := bufio.NewWriter(f)
|
||||||
|
|
||||||
mux := astits.NewMuxer(context.Background(), b)
|
track := &mpegts.Track{
|
||||||
mux.AddElementaryStream(astits.PMTElementaryStream{
|
PID: 256,
|
||||||
ElementaryPID: 256,
|
Codec: &mpegts.CodecH264{},
|
||||||
StreamType: astits.StreamTypeH264Video,
|
}
|
||||||
})
|
|
||||||
mux.SetPCRPID(256)
|
w := mpegts.NewWriter(b, []*mpegts.Track{track})
|
||||||
|
|
||||||
return &mpegtsMuxer{
|
return &mpegtsMuxer{
|
||||||
sps: sps,
|
sps: sps,
|
||||||
pps: pps,
|
pps: pps,
|
||||||
f: f,
|
f: f,
|
||||||
b: b,
|
b: b,
|
||||||
mux: mux,
|
w: w,
|
||||||
|
track: track,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -131,39 +136,8 @@ func (e *mpegtsMuxer) encode(au [][]byte, pts time.Duration) error {
|
|||||||
pts -= e.startDTS
|
pts -= e.startDTS
|
||||||
}
|
}
|
||||||
|
|
||||||
oh := &astits.PESOptionalHeader{
|
// encode into MPEG-TS
|
||||||
MarkerBits: 2,
|
err := e.w.WriteH26x(e.track, durationGoToMPEGTS(pts), durationGoToMPEGTS(dts), idrPresent, au)
|
||||||
}
|
|
||||||
|
|
||||||
if dts == pts {
|
|
||||||
oh.PTSDTSIndicator = astits.PTSDTSIndicatorOnlyPTS
|
|
||||||
oh.PTS = &astits.ClockReference{Base: int64(pts.Seconds() * 90000)}
|
|
||||||
} else {
|
|
||||||
oh.PTSDTSIndicator = astits.PTSDTSIndicatorBothPresent
|
|
||||||
oh.DTS = &astits.ClockReference{Base: int64(dts.Seconds() * 90000)}
|
|
||||||
oh.PTS = &astits.ClockReference{Base: int64(pts.Seconds() * 90000)}
|
|
||||||
}
|
|
||||||
|
|
||||||
// encode into Annex-B
|
|
||||||
annexb, err := h264.AnnexBMarshal(au)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// write TS packet
|
|
||||||
_, err = e.mux.WriteData(&astits.MuxerData{
|
|
||||||
PID: 256,
|
|
||||||
AdaptationField: &astits.PacketAdaptationField{
|
|
||||||
RandomAccessIndicator: idrPresent,
|
|
||||||
},
|
|
||||||
PES: &astits.PESData{
|
|
||||||
Header: &astits.PESHeader{
|
|
||||||
OptionalHeader: oh,
|
|
||||||
StreamID: 224, // video
|
|
||||||
},
|
|
||||||
Data: annexb,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
@@ -0,0 +1,68 @@
|
|||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bufio"
|
||||||
|
"log"
|
||||||
|
"os"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/bluenviron/mediacommon/pkg/codecs/mpeg4audio"
|
||||||
|
"github.com/bluenviron/mediacommon/pkg/formats/mpegts"
|
||||||
|
)
|
||||||
|
|
||||||
|
func durationGoToMPEGTS(v time.Duration) int64 {
|
||||||
|
return int64(v.Seconds() * 90000)
|
||||||
|
}
|
||||||
|
|
||||||
|
// mpegtsMuxer allows to save a MPEG4-audio stream into a MPEG-TS file.
|
||||||
|
type mpegtsMuxer struct {
|
||||||
|
config *mpeg4audio.Config
|
||||||
|
f *os.File
|
||||||
|
b *bufio.Writer
|
||||||
|
w *mpegts.Writer
|
||||||
|
track *mpegts.Track
|
||||||
|
}
|
||||||
|
|
||||||
|
// newMPEGTSMuxer allocates a mpegtsMuxer.
|
||||||
|
func newMPEGTSMuxer(config *mpeg4audio.Config) (*mpegtsMuxer, error) {
|
||||||
|
f, err := os.Create("mystream.ts")
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
b := bufio.NewWriter(f)
|
||||||
|
|
||||||
|
track := &mpegts.Track{
|
||||||
|
PID: 256,
|
||||||
|
Codec: &mpegts.CodecMPEG4Audio{
|
||||||
|
Config: *config,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
w := mpegts.NewWriter(b, []*mpegts.Track{track})
|
||||||
|
|
||||||
|
return &mpegtsMuxer{
|
||||||
|
config: config,
|
||||||
|
f: f,
|
||||||
|
b: b,
|
||||||
|
w: w,
|
||||||
|
track: track,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// close closes all the mpegtsMuxer resources.
|
||||||
|
func (e *mpegtsMuxer) close() {
|
||||||
|
e.b.Flush()
|
||||||
|
e.f.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
// encode encodes a MPEG4-audio access unit into MPEG-TS.
|
||||||
|
func (e *mpegtsMuxer) encode(au []byte, pts time.Duration) error {
|
||||||
|
// encode into MPEG-TS
|
||||||
|
err := e.w.WriteMPEG4Audio(e.track, durationGoToMPEGTS(pts), [][]byte{au})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Println("wrote TS packet")
|
||||||
|
return nil
|
||||||
|
}
|
@@ -1,91 +0,0 @@
|
|||||||
package main
|
|
||||||
|
|
||||||
import (
|
|
||||||
"bufio"
|
|
||||||
"context"
|
|
||||||
"log"
|
|
||||||
"os"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/asticode/go-astits"
|
|
||||||
"github.com/bluenviron/mediacommon/pkg/codecs/mpeg4audio"
|
|
||||||
)
|
|
||||||
|
|
||||||
// mpegtsMuxer allows to save a MPEG4-audio stream into a MPEG-TS file.
|
|
||||||
type mpegtsMuxer struct {
|
|
||||||
config *mpeg4audio.Config
|
|
||||||
f *os.File
|
|
||||||
b *bufio.Writer
|
|
||||||
mux *astits.Muxer
|
|
||||||
}
|
|
||||||
|
|
||||||
// newMPEGTSMuxer allocates a mpegtsMuxer.
|
|
||||||
func newMPEGTSMuxer(config *mpeg4audio.Config) (*mpegtsMuxer, error) {
|
|
||||||
f, err := os.Create("mystream.ts")
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
b := bufio.NewWriter(f)
|
|
||||||
|
|
||||||
mux := astits.NewMuxer(context.Background(), b)
|
|
||||||
mux.AddElementaryStream(astits.PMTElementaryStream{
|
|
||||||
ElementaryPID: 257,
|
|
||||||
StreamType: astits.StreamTypeAACAudio,
|
|
||||||
})
|
|
||||||
mux.SetPCRPID(257)
|
|
||||||
|
|
||||||
return &mpegtsMuxer{
|
|
||||||
config: config,
|
|
||||||
f: f,
|
|
||||||
b: b,
|
|
||||||
mux: mux,
|
|
||||||
}, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
// close closes all the mpegtsMuxer resources.
|
|
||||||
func (e *mpegtsMuxer) close() {
|
|
||||||
e.b.Flush()
|
|
||||||
e.f.Close()
|
|
||||||
}
|
|
||||||
|
|
||||||
// encode encodes a MPEG4-audio access unit into MPEG-TS.
|
|
||||||
func (e *mpegtsMuxer) encode(au []byte, pts time.Duration) error {
|
|
||||||
// wrap access unit inside an ADTS packet
|
|
||||||
pkts := mpeg4audio.ADTSPackets{
|
|
||||||
{
|
|
||||||
Type: e.config.Type,
|
|
||||||
SampleRate: e.config.SampleRate,
|
|
||||||
ChannelCount: e.config.ChannelCount,
|
|
||||||
AU: au,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
enc, err := pkts.Marshal()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
_, err = e.mux.WriteData(&astits.MuxerData{
|
|
||||||
PID: 257,
|
|
||||||
AdaptationField: &astits.PacketAdaptationField{
|
|
||||||
RandomAccessIndicator: true,
|
|
||||||
},
|
|
||||||
PES: &astits.PESData{
|
|
||||||
Header: &astits.PESHeader{
|
|
||||||
OptionalHeader: &astits.PESOptionalHeader{
|
|
||||||
MarkerBits: 2,
|
|
||||||
PTSDTSIndicator: astits.PTSDTSIndicatorOnlyPTS,
|
|
||||||
PTS: &astits.ClockReference{Base: int64(pts.Seconds() * 90000)},
|
|
||||||
},
|
|
||||||
PacketLength: uint16(len(enc) + 8),
|
|
||||||
StreamID: 192, // audio
|
|
||||||
},
|
|
||||||
Data: enc,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Println("wrote TS packet")
|
|
||||||
return nil
|
|
||||||
}
|
|
@@ -2,15 +2,18 @@ package main
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"context"
|
|
||||||
"log"
|
"log"
|
||||||
"os"
|
"os"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/asticode/go-astits"
|
|
||||||
"github.com/bluenviron/mediacommon/pkg/codecs/h264"
|
"github.com/bluenviron/mediacommon/pkg/codecs/h264"
|
||||||
|
"github.com/bluenviron/mediacommon/pkg/formats/mpegts"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
func durationGoToMPEGTS(v time.Duration) int64 {
|
||||||
|
return int64(v.Seconds() * 90000)
|
||||||
|
}
|
||||||
|
|
||||||
// mpegtsMuxer allows to save a H264 stream into a MPEG-TS file.
|
// mpegtsMuxer allows to save a H264 stream into a MPEG-TS file.
|
||||||
type mpegtsMuxer struct {
|
type mpegtsMuxer struct {
|
||||||
sps []byte
|
sps []byte
|
||||||
@@ -18,7 +21,8 @@ type mpegtsMuxer struct {
|
|||||||
|
|
||||||
f *os.File
|
f *os.File
|
||||||
b *bufio.Writer
|
b *bufio.Writer
|
||||||
mux *astits.Muxer
|
w *mpegts.Writer
|
||||||
|
track *mpegts.Track
|
||||||
dtsExtractor *h264.DTSExtractor
|
dtsExtractor *h264.DTSExtractor
|
||||||
firstIDRReceived bool
|
firstIDRReceived bool
|
||||||
startDTS time.Duration
|
startDTS time.Duration
|
||||||
@@ -32,19 +36,20 @@ func newMPEGTSMuxer(sps []byte, pps []byte) (*mpegtsMuxer, error) {
|
|||||||
}
|
}
|
||||||
b := bufio.NewWriter(f)
|
b := bufio.NewWriter(f)
|
||||||
|
|
||||||
mux := astits.NewMuxer(context.Background(), b)
|
track := &mpegts.Track{
|
||||||
mux.AddElementaryStream(astits.PMTElementaryStream{
|
PID: 256,
|
||||||
ElementaryPID: 256,
|
Codec: &mpegts.CodecH264{},
|
||||||
StreamType: astits.StreamTypeH264Video,
|
}
|
||||||
})
|
|
||||||
mux.SetPCRPID(256)
|
w := mpegts.NewWriter(b, []*mpegts.Track{track})
|
||||||
|
|
||||||
return &mpegtsMuxer{
|
return &mpegtsMuxer{
|
||||||
sps: sps,
|
sps: sps,
|
||||||
pps: pps,
|
pps: pps,
|
||||||
f: f,
|
f: f,
|
||||||
b: b,
|
b: b,
|
||||||
mux: mux,
|
w: w,
|
||||||
|
track: track,
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -55,7 +60,7 @@ func (e *mpegtsMuxer) close() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// encode encodes H264 NALUs into MPEG-TS.
|
// encode encodes H264 NALUs into MPEG-TS.
|
||||||
func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
func (e *mpegtsMuxer) encode(au [][]byte, pts time.Duration) error {
|
||||||
// prepend an AUD. This is required by some players
|
// prepend an AUD. This is required by some players
|
||||||
filteredNALUs := [][]byte{
|
filteredNALUs := [][]byte{
|
||||||
{byte(h264.NALUTypeAccessUnitDelimiter), 240},
|
{byte(h264.NALUTypeAccessUnitDelimiter), 240},
|
||||||
@@ -64,7 +69,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
nonIDRPresent := false
|
nonIDRPresent := false
|
||||||
idrPresent := false
|
idrPresent := false
|
||||||
|
|
||||||
for _, nalu := range nalus {
|
for _, nalu := range au {
|
||||||
typ := h264.NALUType(nalu[0] & 0x1F)
|
typ := h264.NALUType(nalu[0] & 0x1F)
|
||||||
switch typ {
|
switch typ {
|
||||||
case h264.NALUTypeSPS:
|
case h264.NALUTypeSPS:
|
||||||
@@ -88,7 +93,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
filteredNALUs = append(filteredNALUs, nalu)
|
filteredNALUs = append(filteredNALUs, nalu)
|
||||||
}
|
}
|
||||||
|
|
||||||
nalus = filteredNALUs
|
au = filteredNALUs
|
||||||
|
|
||||||
if !nonIDRPresent && !idrPresent {
|
if !nonIDRPresent && !idrPresent {
|
||||||
return nil
|
return nil
|
||||||
@@ -96,7 +101,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
|
|
||||||
// add SPS and PPS before every group that contains an IDR
|
// add SPS and PPS before every group that contains an IDR
|
||||||
if idrPresent {
|
if idrPresent {
|
||||||
nalus = append([][]byte{e.sps, e.pps}, nalus...)
|
au = append([][]byte{e.sps, e.pps}, au...)
|
||||||
}
|
}
|
||||||
|
|
||||||
var dts time.Duration
|
var dts time.Duration
|
||||||
@@ -111,7 +116,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
e.dtsExtractor = h264.NewDTSExtractor()
|
e.dtsExtractor = h264.NewDTSExtractor()
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
dts, err = e.dtsExtractor.Extract(nalus, pts)
|
dts, err = e.dtsExtractor.Extract(au, pts)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -122,7 +127,7 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
|
|
||||||
} else {
|
} else {
|
||||||
var err error
|
var err error
|
||||||
dts, err = e.dtsExtractor.Extract(nalus, pts)
|
dts, err = e.dtsExtractor.Extract(au, pts)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@@ -131,39 +136,8 @@ func (e *mpegtsMuxer) encode(nalus [][]byte, pts time.Duration) error {
|
|||||||
pts -= e.startDTS
|
pts -= e.startDTS
|
||||||
}
|
}
|
||||||
|
|
||||||
oh := &astits.PESOptionalHeader{
|
// encode into MPEG-TS
|
||||||
MarkerBits: 2,
|
err := e.w.WriteH26x(e.track, durationGoToMPEGTS(pts), durationGoToMPEGTS(dts), idrPresent, au)
|
||||||
}
|
|
||||||
|
|
||||||
if dts == pts {
|
|
||||||
oh.PTSDTSIndicator = astits.PTSDTSIndicatorOnlyPTS
|
|
||||||
oh.PTS = &astits.ClockReference{Base: int64(pts.Seconds() * 90000)}
|
|
||||||
} else {
|
|
||||||
oh.PTSDTSIndicator = astits.PTSDTSIndicatorBothPresent
|
|
||||||
oh.DTS = &astits.ClockReference{Base: int64(dts.Seconds() * 90000)}
|
|
||||||
oh.PTS = &astits.ClockReference{Base: int64(pts.Seconds() * 90000)}
|
|
||||||
}
|
|
||||||
|
|
||||||
// encode into Annex-B
|
|
||||||
annexb, err := h264.AnnexBMarshal(nalus)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
// write TS packet
|
|
||||||
_, err = e.mux.WriteData(&astits.MuxerData{
|
|
||||||
PID: 256,
|
|
||||||
AdaptationField: &astits.PacketAdaptationField{
|
|
||||||
RandomAccessIndicator: idrPresent,
|
|
||||||
},
|
|
||||||
PES: &astits.PESData{
|
|
||||||
Header: &astits.PESHeader{
|
|
||||||
OptionalHeader: oh,
|
|
||||||
StreamID: 224, // video
|
|
||||||
},
|
|
||||||
Data: annexb,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
2
go.mod
2
go.mod
@@ -3,7 +3,6 @@ module github.com/bluenviron/gortsplib/v3
|
|||||||
go 1.18
|
go 1.18
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/asticode/go-astits v1.11.1-0.20230727094110-0df190a2dd87
|
|
||||||
github.com/bluenviron/mediacommon v0.7.1-0.20230730144331-10b74a4f6eda
|
github.com/bluenviron/mediacommon v0.7.1-0.20230730144331-10b74a4f6eda
|
||||||
github.com/google/uuid v1.3.0
|
github.com/google/uuid v1.3.0
|
||||||
github.com/pion/rtcp v1.2.10
|
github.com/pion/rtcp v1.2.10
|
||||||
@@ -15,6 +14,7 @@ require (
|
|||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/asticode/go-astikit v0.30.0 // indirect
|
github.com/asticode/go-astikit v0.30.0 // indirect
|
||||||
|
github.com/asticode/go-astits v1.11.1-0.20230727094110-0df190a2dd87 // indirect
|
||||||
github.com/davecgh/go-spew v1.1.1 // indirect
|
github.com/davecgh/go-spew v1.1.1 // indirect
|
||||||
github.com/pion/randutil v0.1.0 // indirect
|
github.com/pion/randutil v0.1.0 // indirect
|
||||||
github.com/pmezard/go-difflib v1.0.0 // indirect
|
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||||
|
Reference in New Issue
Block a user