mirror of
https://github.com/aler9/gortsplib
synced 2025-10-04 23:02:45 +08:00
190 lines
3.7 KiB
Go
190 lines
3.7 KiB
Go
package gortsplib
|
|
|
|
import (
|
|
"encoding/base64"
|
|
"encoding/hex"
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
"sync"
|
|
|
|
psdp "github.com/pion/sdp/v3"
|
|
)
|
|
|
|
// TrackH264 is a H264 track.
|
|
type TrackH264 struct {
|
|
PayloadType uint8
|
|
SPS []byte
|
|
PPS []byte
|
|
Extradata []byte
|
|
|
|
trackBase
|
|
mutex sync.RWMutex
|
|
}
|
|
|
|
func newTrackH264FromMediaDescription(
|
|
control string,
|
|
payloadType uint8,
|
|
md *psdp.MediaDescription,
|
|
) (*TrackH264, error) {
|
|
t := &TrackH264{
|
|
PayloadType: payloadType,
|
|
trackBase: trackBase{
|
|
control: control,
|
|
},
|
|
}
|
|
|
|
t.fillParamsFromMediaDescription(md)
|
|
|
|
return t, nil
|
|
}
|
|
|
|
func (t *TrackH264) fillParamsFromMediaDescription(md *psdp.MediaDescription) error {
|
|
v, ok := md.Attribute("fmtp")
|
|
if !ok {
|
|
return fmt.Errorf("fmtp attribute is missing")
|
|
}
|
|
|
|
tmp := strings.SplitN(v, " ", 2)
|
|
if len(tmp) != 2 {
|
|
return fmt.Errorf("invalid fmtp attribute (%v)", v)
|
|
}
|
|
|
|
for _, kv := range strings.Split(tmp[1], ";") {
|
|
kv = strings.Trim(kv, " ")
|
|
|
|
if len(kv) == 0 {
|
|
continue
|
|
}
|
|
|
|
tmp := strings.SplitN(kv, "=", 2)
|
|
if len(tmp) != 2 {
|
|
return fmt.Errorf("invalid fmtp attribute (%v)", v)
|
|
}
|
|
|
|
if tmp[0] == "sprop-parameter-sets" {
|
|
tmp := strings.Split(tmp[1], ",")
|
|
if len(tmp) < 2 {
|
|
return fmt.Errorf("invalid sprop-parameter-sets (%v)", v)
|
|
}
|
|
|
|
sps, err := base64.StdEncoding.DecodeString(tmp[0])
|
|
if err != nil {
|
|
return fmt.Errorf("invalid sprop-parameter-sets (%v)", v)
|
|
}
|
|
|
|
pps, err := base64.StdEncoding.DecodeString(tmp[1])
|
|
if err != nil {
|
|
return fmt.Errorf("invalid sprop-parameter-sets (%v)", v)
|
|
}
|
|
|
|
var extradata []byte
|
|
if len(tmp) > 2 {
|
|
var err error
|
|
extradata, err = base64.StdEncoding.DecodeString(tmp[1])
|
|
if err != nil {
|
|
return fmt.Errorf("invalid sprop-parameter-sets (%v)", v)
|
|
}
|
|
}
|
|
|
|
t.SPS = sps
|
|
t.PPS = pps
|
|
t.Extradata = extradata
|
|
return nil
|
|
}
|
|
}
|
|
|
|
return fmt.Errorf("sprop-parameter-sets is missing (%v)", v)
|
|
}
|
|
|
|
// ClockRate returns the track clock rate.
|
|
func (t *TrackH264) ClockRate() int {
|
|
return 90000
|
|
}
|
|
|
|
func (t *TrackH264) clone() Track {
|
|
return &TrackH264{
|
|
PayloadType: t.PayloadType,
|
|
SPS: t.SPS,
|
|
PPS: t.PPS,
|
|
Extradata: t.Extradata,
|
|
trackBase: t.trackBase,
|
|
}
|
|
}
|
|
|
|
// SafeSPS returns the track SPS.
|
|
func (t *TrackH264) SafeSPS() []byte {
|
|
t.mutex.RLock()
|
|
defer t.mutex.RUnlock()
|
|
return t.SPS
|
|
}
|
|
|
|
// SafePPS returns the track PPS.
|
|
func (t *TrackH264) SafePPS() []byte {
|
|
t.mutex.RLock()
|
|
defer t.mutex.RUnlock()
|
|
return t.PPS
|
|
}
|
|
|
|
// SafeSetSPS sets the track SPS.
|
|
func (t *TrackH264) SafeSetSPS(v []byte) {
|
|
t.mutex.Lock()
|
|
defer t.mutex.Unlock()
|
|
t.SPS = v
|
|
}
|
|
|
|
// SafeSetPPS sets the track PPS.
|
|
func (t *TrackH264) SafeSetPPS(v []byte) {
|
|
t.mutex.Lock()
|
|
defer t.mutex.Unlock()
|
|
t.PPS = v
|
|
}
|
|
|
|
// MediaDescription returns the track media description in SDP format.
|
|
func (t *TrackH264) MediaDescription() *psdp.MediaDescription {
|
|
t.mutex.RLock()
|
|
defer t.mutex.RUnlock()
|
|
|
|
typ := strconv.FormatInt(int64(t.PayloadType), 10)
|
|
|
|
fmtp := typ + " packetization-mode=1"
|
|
|
|
var tmp []string
|
|
if t.SPS != nil {
|
|
tmp = append(tmp, base64.StdEncoding.EncodeToString(t.SPS))
|
|
}
|
|
if t.PPS != nil {
|
|
tmp = append(tmp, base64.StdEncoding.EncodeToString(t.PPS))
|
|
}
|
|
if t.Extradata != nil {
|
|
tmp = append(tmp, base64.StdEncoding.EncodeToString(t.Extradata))
|
|
}
|
|
fmtp += "; sprop-parameter-sets=" + strings.Join(tmp, ",")
|
|
|
|
if len(t.SPS) >= 4 {
|
|
fmtp += "; profile-level-id=" + strings.ToUpper(hex.EncodeToString(t.SPS[1:4]))
|
|
}
|
|
|
|
return &psdp.MediaDescription{
|
|
MediaName: psdp.MediaName{
|
|
Media: "video",
|
|
Protos: []string{"RTP", "AVP"},
|
|
Formats: []string{typ},
|
|
},
|
|
Attributes: []psdp.Attribute{
|
|
{
|
|
Key: "rtpmap",
|
|
Value: typ + " H264/90000",
|
|
},
|
|
{
|
|
Key: "fmtp",
|
|
Value: fmtp,
|
|
},
|
|
{
|
|
Key: "control",
|
|
Value: t.control,
|
|
},
|
|
},
|
|
}
|
|
}
|