mirror of
https://github.com/aler9/gortsplib
synced 2025-10-06 07:37:07 +08:00
support decoding H264 units without Marker field (bluenviron/mediamtx#3945) (#759)
(like the ones produced by the FLIR M400)
This commit is contained in:
@@ -62,6 +62,7 @@ type Decoder struct {
|
|||||||
frameBuffer [][]byte
|
frameBuffer [][]byte
|
||||||
frameBufferLen int
|
frameBufferLen int
|
||||||
frameBufferSize int
|
frameBufferSize int
|
||||||
|
frameBufferTimestamp uint32
|
||||||
}
|
}
|
||||||
|
|
||||||
// Init initializes the decoder.
|
// Init initializes the decoder.
|
||||||
@@ -222,12 +223,46 @@ func (d *Decoder) Decode(pkt *rtp.Packet) ([][]byte, error) {
|
|||||||
}
|
}
|
||||||
l := len(nalus)
|
l := len(nalus)
|
||||||
|
|
||||||
if (d.frameBufferLen + l) > h264.MaxNALUsPerAccessUnit {
|
// support splitting access units by timestamp.
|
||||||
errCount := d.frameBufferLen + l
|
// (some cameras do not use the Marker field, like the FLIR M400)
|
||||||
d.frameBuffer = nil
|
if d.frameBuffer != nil && pkt.Timestamp != d.frameBufferTimestamp {
|
||||||
|
ret := d.frameBuffer
|
||||||
|
d.resetFrameBuffer()
|
||||||
|
|
||||||
|
err = d.addToFrameBuffer(nalus, l, pkt.Timestamp)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
err = d.addToFrameBuffer(nalus, l, pkt.Timestamp)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if !pkt.Marker {
|
||||||
|
return nil, ErrMorePacketsNeeded
|
||||||
|
}
|
||||||
|
|
||||||
|
ret := d.frameBuffer
|
||||||
|
d.resetFrameBuffer()
|
||||||
|
|
||||||
|
return ret, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (d *Decoder) resetFrameBuffer() {
|
||||||
|
d.frameBuffer = nil // do not reuse frameBuffer to avoid race conditions
|
||||||
d.frameBufferLen = 0
|
d.frameBufferLen = 0
|
||||||
d.frameBufferSize = 0
|
d.frameBufferSize = 0
|
||||||
return nil, fmt.Errorf("NALU count (%d) exceeds maximum allowed (%d)",
|
}
|
||||||
|
|
||||||
|
func (d *Decoder) addToFrameBuffer(nalus [][]byte, l int, ts uint32) error {
|
||||||
|
if (d.frameBufferLen + l) > h264.MaxNALUsPerAccessUnit {
|
||||||
|
errCount := d.frameBufferLen + l
|
||||||
|
d.resetFrameBuffer()
|
||||||
|
return fmt.Errorf("NALU count (%d) exceeds maximum allowed (%d)",
|
||||||
errCount, h264.MaxNALUsPerAccessUnit)
|
errCount, h264.MaxNALUsPerAccessUnit)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -235,29 +270,16 @@ func (d *Decoder) Decode(pkt *rtp.Packet) ([][]byte, error) {
|
|||||||
|
|
||||||
if (d.frameBufferSize + addSize) > h264.MaxAccessUnitSize {
|
if (d.frameBufferSize + addSize) > h264.MaxAccessUnitSize {
|
||||||
errSize := d.frameBufferSize + addSize
|
errSize := d.frameBufferSize + addSize
|
||||||
d.frameBuffer = nil
|
d.resetFrameBuffer()
|
||||||
d.frameBufferLen = 0
|
return fmt.Errorf("access unit size (%d) is too big, maximum is %d",
|
||||||
d.frameBufferSize = 0
|
|
||||||
return nil, fmt.Errorf("access unit size (%d) is too big, maximum is %d",
|
|
||||||
errSize, h264.MaxAccessUnitSize)
|
errSize, h264.MaxAccessUnitSize)
|
||||||
}
|
}
|
||||||
|
|
||||||
d.frameBuffer = append(d.frameBuffer, nalus...)
|
d.frameBuffer = append(d.frameBuffer, nalus...)
|
||||||
d.frameBufferLen += l
|
d.frameBufferLen += l
|
||||||
d.frameBufferSize += addSize
|
d.frameBufferSize += addSize
|
||||||
|
d.frameBufferTimestamp = ts
|
||||||
if !pkt.Marker {
|
return nil
|
||||||
return nil, ErrMorePacketsNeeded
|
|
||||||
}
|
|
||||||
|
|
||||||
ret := d.frameBuffer
|
|
||||||
|
|
||||||
// do not reuse frameBuffer to avoid race conditions
|
|
||||||
d.frameBuffer = nil
|
|
||||||
d.frameBufferLen = 0
|
|
||||||
d.frameBufferSize = 0
|
|
||||||
|
|
||||||
return ret, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// some cameras / servers wrap NALUs into Annex-B
|
// some cameras / servers wrap NALUs into Annex-B
|
||||||
|
@@ -189,11 +189,15 @@ func TestDecodeAnnexB(t *testing.T) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestDecodeAccessUnit(t *testing.T) {
|
func TestDecodeAccessUnit(t *testing.T) {
|
||||||
d := &Decoder{}
|
for _, ca := range []struct {
|
||||||
err := d.Init()
|
name string
|
||||||
require.NoError(t, err)
|
pkts []*rtp.Packet
|
||||||
|
au [][]byte
|
||||||
nalus, err := d.Decode(&rtp.Packet{
|
}{
|
||||||
|
{
|
||||||
|
"marker-splitted",
|
||||||
|
[]*rtp.Packet{
|
||||||
|
{
|
||||||
Header: rtp.Header{
|
Header: rtp.Header{
|
||||||
Version: 2,
|
Version: 2,
|
||||||
Marker: false,
|
Marker: false,
|
||||||
@@ -202,12 +206,9 @@ func TestDecodeAccessUnit(t *testing.T) {
|
|||||||
Timestamp: 2289531307,
|
Timestamp: 2289531307,
|
||||||
SSRC: 0x9dbb7812,
|
SSRC: 0x9dbb7812,
|
||||||
},
|
},
|
||||||
Payload: []byte{0x01, 0x02},
|
Payload: []byte{1, 2},
|
||||||
})
|
},
|
||||||
require.Equal(t, ErrMorePacketsNeeded, err)
|
{
|
||||||
require.Equal(t, [][]byte(nil), nalus)
|
|
||||||
|
|
||||||
nalus, err = d.Decode(&rtp.Packet{
|
|
||||||
Header: rtp.Header{
|
Header: rtp.Header{
|
||||||
Version: 2,
|
Version: 2,
|
||||||
Marker: true,
|
Marker: true,
|
||||||
@@ -216,10 +217,58 @@ func TestDecodeAccessUnit(t *testing.T) {
|
|||||||
Timestamp: 2289531307,
|
Timestamp: 2289531307,
|
||||||
SSRC: 0x9dbb7812,
|
SSRC: 0x9dbb7812,
|
||||||
},
|
},
|
||||||
Payload: []byte{0x01, 0x02},
|
Payload: []byte{3, 4},
|
||||||
})
|
},
|
||||||
|
},
|
||||||
|
[][]byte{{1, 2}, {3, 4}},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"timestamp-splitted (FLIR M400)",
|
||||||
|
[]*rtp.Packet{
|
||||||
|
{
|
||||||
|
Header: rtp.Header{
|
||||||
|
Version: 2,
|
||||||
|
Marker: false,
|
||||||
|
PayloadType: 96,
|
||||||
|
SequenceNumber: 17647,
|
||||||
|
Timestamp: 2289531307,
|
||||||
|
SSRC: 0x9dbb7812,
|
||||||
|
},
|
||||||
|
Payload: []byte{1, 2},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Header: rtp.Header{
|
||||||
|
Version: 2,
|
||||||
|
Marker: false,
|
||||||
|
PayloadType: 96,
|
||||||
|
SequenceNumber: 17647,
|
||||||
|
Timestamp: 2289531308,
|
||||||
|
SSRC: 0x9dbb7812,
|
||||||
|
},
|
||||||
|
Payload: []byte{3, 4},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
[][]byte{{1, 2}},
|
||||||
|
},
|
||||||
|
} {
|
||||||
|
t.Run(ca.name, func(t *testing.T) {
|
||||||
|
d := &Decoder{}
|
||||||
|
err := d.Init()
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Equal(t, [][]byte{{0x01, 0x02}, {0x01, 0x02}}, nalus)
|
|
||||||
|
var au [][]byte
|
||||||
|
|
||||||
|
for i, pkt := range ca.pkts {
|
||||||
|
au, err = d.Decode(pkt)
|
||||||
|
if i != len(ca.pkts)-1 {
|
||||||
|
require.Equal(t, ErrMorePacketsNeeded, err)
|
||||||
|
} else {
|
||||||
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, ca.au, au)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDecoderErrorNALUSize(t *testing.T) {
|
func TestDecoderErrorNALUSize(t *testing.T) {
|
||||||
|
Reference in New Issue
Block a user