mirror of
https://github.com/aler9/gortsplib
synced 2025-10-05 07:06:58 +08:00
134 lines
2.7 KiB
Go
134 lines
2.7 KiB
Go
package format
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/bluenviron/mediacommon/v2/pkg/codecs/mpeg4video"
|
|
"github.com/pion/rtp"
|
|
|
|
"github.com/bluenviron/gortsplib/v4/pkg/format/rtpmpeg4video"
|
|
)
|
|
|
|
// MPEG4Video is the RTP format for a MPEG-4 Video codec.
|
|
// Specification: https://datatracker.ietf.org/doc/html/rfc6416#section-7.1
|
|
type MPEG4Video struct {
|
|
PayloadTyp uint8
|
|
ProfileLevelID int
|
|
Config []byte
|
|
|
|
mutex sync.RWMutex
|
|
}
|
|
|
|
func (f *MPEG4Video) unmarshal(ctx *unmarshalContext) error {
|
|
f.PayloadTyp = ctx.payloadType
|
|
f.ProfileLevelID = 1 // default value imposed by specification
|
|
|
|
for key, val := range ctx.fmtp {
|
|
switch key {
|
|
case "profile-level-id":
|
|
tmp, err := strconv.ParseUint(val, 10, 31)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid profile-level-id: %v", val)
|
|
}
|
|
|
|
f.ProfileLevelID = int(tmp)
|
|
|
|
case "config":
|
|
var err error
|
|
f.Config, err = hex.DecodeString(val)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid config: %v", val)
|
|
}
|
|
|
|
err = mpeg4video.IsValidConfig(f.Config)
|
|
if err != nil {
|
|
return fmt.Errorf("invalid config: %w", err)
|
|
}
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// Codec implements Format.
|
|
func (f *MPEG4Video) Codec() string {
|
|
return "MPEG-4 Video"
|
|
}
|
|
|
|
// ClockRate implements Format.
|
|
func (f *MPEG4Video) ClockRate() int {
|
|
return 90000
|
|
}
|
|
|
|
// PayloadType implements Format.
|
|
func (f *MPEG4Video) PayloadType() uint8 {
|
|
return f.PayloadTyp
|
|
}
|
|
|
|
// RTPMap implements Format.
|
|
func (f *MPEG4Video) RTPMap() string {
|
|
return "MP4V-ES/90000"
|
|
}
|
|
|
|
// FMTP implements Format.
|
|
func (f *MPEG4Video) FMTP() map[string]string {
|
|
fmtp := map[string]string{
|
|
"profile-level-id": strconv.FormatInt(int64(f.ProfileLevelID), 10),
|
|
}
|
|
|
|
if f.Config != nil {
|
|
fmtp["config"] = strings.ToUpper(hex.EncodeToString(f.Config))
|
|
}
|
|
|
|
return fmtp
|
|
}
|
|
|
|
// PTSEqualsDTS implements Format.
|
|
func (f *MPEG4Video) PTSEqualsDTS(*rtp.Packet) bool {
|
|
return true
|
|
}
|
|
|
|
// CreateDecoder creates a decoder able to decode the content of the format.
|
|
func (f *MPEG4Video) CreateDecoder() (*rtpmpeg4video.Decoder, error) {
|
|
d := &rtpmpeg4video.Decoder{}
|
|
|
|
err := d.Init()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return d, nil
|
|
}
|
|
|
|
// CreateEncoder creates an encoder able to encode the content of the format.
|
|
func (f *MPEG4Video) CreateEncoder() (*rtpmpeg4video.Encoder, error) {
|
|
e := &rtpmpeg4video.Encoder{
|
|
PayloadType: f.PayloadTyp,
|
|
}
|
|
|
|
err := e.Init()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return e, nil
|
|
}
|
|
|
|
// SafeSetParams sets the codec parameters.
|
|
func (f *MPEG4Video) SafeSetParams(config []byte) {
|
|
f.mutex.Lock()
|
|
defer f.mutex.Unlock()
|
|
f.Config = config
|
|
}
|
|
|
|
// SafeParams returns the codec parameters.
|
|
func (f *MPEG4Video) SafeParams() []byte {
|
|
f.mutex.RLock()
|
|
defer f.mutex.RUnlock()
|
|
return f.Config
|
|
}
|