Add support for RTP header extensions, and IVF videos now fully work!

This commit is contained in:
Sean DuBois
2018-06-11 01:42:41 -07:00
parent 882b87b346
commit d1608eda58
3 changed files with 47 additions and 39 deletions

View File

@@ -9,8 +9,6 @@ import (
"github.com/pions/webrtc/rtp"
)
var le = binary.LittleEndian
type IVFWriter struct {
fd *os.File
time time.Time
@@ -50,15 +48,15 @@ func NewIVFWriter(fileName string) (*IVFWriter, error) {
header := make([]byte, 32)
copy(header[0:], []byte("DKIF")) // DKIF
le.PutUint16(header[4:], 0) // Version
le.PutUint16(header[6:], 32) // Header Size
binary.LittleEndian.PutUint16(header[4:], 0) // Version
binary.LittleEndian.PutUint16(header[6:], 32) // Header Size
copy(header[8:], []byte("VP80")) // FOURCC
le.PutUint16(header[12:], 640) // Version
le.PutUint16(header[14:], 480) // Header Size
le.PutUint32(header[16:], 30) // Framerate numerator
le.PutUint32(header[20:], 1) // Framerate Denominator
le.PutUint32(header[24:], 900) // Frame count
le.PutUint32(header[28:], 0) // Unused
binary.LittleEndian.PutUint16(header[12:], 640) // Version
binary.LittleEndian.PutUint16(header[14:], 480) // Header Size
binary.LittleEndian.PutUint32(header[16:], 30) // Framerate numerator
binary.LittleEndian.PutUint32(header[20:], 1) // Framerate Denominator
binary.LittleEndian.PutUint32(header[24:], 900) // Frame count
binary.LittleEndian.PutUint32(header[28:], 0) // Unused
panicWrite(f, header)
@@ -103,7 +101,6 @@ func (i *IVFWriter) DecodeVP8RTPPacket(packet *rtp.Packet) (*VP8RTPPacket, error
payloadIndex++
}
fmt.Print(vp8Packet)
vp8Packet.Payload = p[payloadIndex:]
return vp8Packet, nil
@@ -123,9 +120,8 @@ func (i *IVFWriter) AddPacket(packet *rtp.Packet) {
}
frameHeader := make([]byte, 12)
le.PutUint32(frameHeader[0:], uint32(len(i.currentFrame))) // Frame length
le.PutUint64(frameHeader[4:], i.count) // PTS
binary.LittleEndian.PutUint32(frameHeader[0:], uint32(len(i.currentFrame))) // Frame length
binary.LittleEndian.PutUint64(frameHeader[4:], i.count) // PTS
i.count += 1

View File

@@ -33,4 +33,7 @@ const (
ssrcLength = 4
csrcOffset = 12
csrcLength = 4
extensionHeaderIdLength = 2
extensionLengthFieldLength = 2
extensionHeaderAssumedLength = 4
)

View File

@@ -2,6 +2,7 @@ package rtp
import (
"encoding/binary"
"github.com/pkg/errors"
)
@@ -37,9 +38,9 @@ func (p *Packet) Unmarshal(rawPacket []byte) error {
p.Timestamp = binary.BigEndian.Uint32(rawPacket[timestampOffset : timestampOffset+timestampLength])
p.SSRC = binary.BigEndian.Uint32(rawPacket[ssrcOffset : ssrcOffset+ssrcLength])
headerPlusCsrcs := headerLength + timestampLength + ssrcLength + (len(p.CSRC) * csrcLength)
if len(rawPacket) < headerPlusCsrcs {
return errors.Errorf("RTP header size insufficient; %d < %d", len(rawPacket), headerPlusCsrcs)
currOffset := headerLength + timestampLength + ssrcLength + (len(p.CSRC) * csrcLength)
if len(rawPacket) < currOffset {
return errors.Errorf("RTP header size insufficient; %d < %d", len(rawPacket), currOffset)
}
for i := range p.CSRC {
@@ -47,7 +48,15 @@ func (p *Packet) Unmarshal(rawPacket []byte) error {
p.CSRC[i] = binary.BigEndian.Uint32(rawPacket[offset:offset])
}
p.Payload = rawPacket[headerPlusCsrcs:]
if p.Extension {
currOffset += extensionHeaderIdLength
extensionLength := binary.BigEndian.Uint16(rawPacket[currOffset:])
currOffset += extensionLengthFieldLength
currOffset += (int(extensionLength) * extensionHeaderAssumedLength)
}
p.Payload = rawPacket[currOffset:]
return nil
}