Ready-to-use SRT / WebRTC / RTSP / RTMP / LL-HLS media server and media proxy that allows to read, publish, proxy, record and playback video and audio streams.
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

262 lines
5.0 KiB

package fmp4
import (
"bytes"
"fmt"
gomp4 "github.com/abema/go-mp4"
"github.com/aler9/gortsplib/v2/pkg/codecs/mpeg4audio"
"github.com/aler9/gortsplib/v2/pkg/format"
)
// Init is a FMP4 initialization file.
type Init struct {
Tracks []*InitTrack
}
// Unmarshal decodes a FMP4 initialization file.
func (i *Init) Unmarshal(byts []byte) error {
type readState int
const (
waitingTrak readState = iota
waitingTkhd
waitingMdhd
waitingCodec
waitingAvcc
waitingEsds
)
state := waitingTrak
var curTrack *InitTrack
_, err := gomp4.ReadBoxStructure(bytes.NewReader(byts), func(h *gomp4.ReadHandle) (interface{}, error) {
switch h.BoxInfo.Type.String() {
case "trak":
if state != waitingTrak {
return nil, fmt.Errorf("parse error")
}
curTrack = &InitTrack{}
i.Tracks = append(i.Tracks, curTrack)
state = waitingTkhd
case "tkhd":
if state != waitingTkhd {
return nil, fmt.Errorf("parse error")
}
box, _, err := h.ReadPayload()
if err != nil {
return nil, err
}
tkhd := box.(*gomp4.Tkhd)
curTrack.ID = int(tkhd.TrackID)
state = waitingMdhd
case "mdhd":
if state != waitingMdhd {
return nil, fmt.Errorf("parse error")
}
box, _, err := h.ReadPayload()
if err != nil {
return nil, err
}
mdhd := box.(*gomp4.Mdhd)
curTrack.TimeScale = mdhd.Timescale
state = waitingCodec
case "avc1":
if state != waitingCodec {
return nil, fmt.Errorf("parse error")
}
state = waitingAvcc
case "avcC":
if state != waitingAvcc {
return nil, fmt.Errorf("parse error")
}
box, _, err := h.ReadPayload()
if err != nil {
return nil, err
}
conf := box.(*gomp4.AVCDecoderConfiguration)
if len(conf.SequenceParameterSets) > 1 {
return nil, fmt.Errorf("multiple SPS are not supported")
}
var sps []byte
if len(conf.SequenceParameterSets) == 1 {
sps = conf.SequenceParameterSets[0].NALUnit
}
if len(conf.PictureParameterSets) > 1 {
return nil, fmt.Errorf("multiple PPS are not supported")
}
var pps []byte
if len(conf.PictureParameterSets) == 1 {
pps = conf.PictureParameterSets[0].NALUnit
}
curTrack.Format = &format.H264{
PayloadTyp: 96,
SPS: sps,
PPS: pps,
PacketizationMode: 1,
}
state = waitingTrak
case "mp4a":
if state != waitingCodec {
return nil, fmt.Errorf("parse error")
}
state = waitingEsds
case "esds":
if state != waitingEsds {
return nil, fmt.Errorf("parse error")
}
box, _, err := h.ReadPayload()
if err != nil {
return nil, err
}
esds := box.(*gomp4.Esds)
encodedConf := func() []byte {
for _, desc := range esds.Descriptors {
if desc.Tag == gomp4.DecSpecificInfoTag {
return desc.Data
}
}
return nil
}()
if encodedConf == nil {
return nil, fmt.Errorf("unable to find MPEG4-audio configuration")
}
var c mpeg4audio.Config
err = c.Unmarshal(encodedConf)
if err != nil {
return nil, fmt.Errorf("invalid MPEG4-audio configuration: %s", err)
}
curTrack.Format = &format.MPEG4Audio{
PayloadTyp: 96,
Config: &c,
SizeLength: 13,
IndexLength: 3,
IndexDeltaLength: 3,
}
state = waitingTrak
case "ac-3":
return nil, fmt.Errorf("AC-3 codec is not supported (yet)")
}
return h.Expand()
})
if err != nil {
return err
}
if state != waitingTrak {
return fmt.Errorf("parse error")
}
if i.Tracks == nil {
return fmt.Errorf("no tracks found")
}
return nil
}
// Marshal encodes a FMP4 initialization file.
func (i *Init) Marshal() ([]byte, error) {
/*
- ftyp
- moov
- mvhd
- trak
- trak
- ...
- mvex
- trex
- trex
- ...
*/
w := newMP4Writer()
_, err := w.WriteBox(&gomp4.Ftyp{ // <ftyp/>
MajorBrand: [4]byte{'m', 'p', '4', '2'},
MinorVersion: 1,
CompatibleBrands: []gomp4.CompatibleBrandElem{
{CompatibleBrand: [4]byte{'m', 'p', '4', '1'}},
{CompatibleBrand: [4]byte{'m', 'p', '4', '2'}},
{CompatibleBrand: [4]byte{'i', 's', 'o', 'm'}},
{CompatibleBrand: [4]byte{'h', 'l', 's', 'f'}},
},
})
if err != nil {
return nil, err
}
_, err = w.writeBoxStart(&gomp4.Moov{}) // <moov>
if err != nil {
return nil, err
}
_, err = w.WriteBox(&gomp4.Mvhd{ // <mvhd/>
Timescale: 1000,
Rate: 65536,
Volume: 256,
Matrix: [9]int32{0x00010000, 0, 0, 0, 0x00010000, 0, 0, 0, 0x40000000},
NextTrackID: 4294967295,
})
if err != nil {
return nil, err
}
for _, track := range i.Tracks {
err := track.marshal(w)
if err != nil {
return nil, err
}
}
_, err = w.writeBoxStart(&gomp4.Mvex{}) // <mvex>
if err != nil {
return nil, err
}
for _, track := range i.Tracks {
_, err = w.WriteBox(&gomp4.Trex{ // <trex/>
TrackID: uint32(track.ID),
DefaultSampleDescriptionIndex: 1,
})
if err != nil {
return nil, err
}
}
err = w.writeBoxEnd() // </mvex>
if err != nil {
return nil, err
}
err = w.writeBoxEnd() // </moov>
if err != nil {
return nil, err
}
return w.bytes(), nil
}