Fix adts producer for VLC player support #1643
This commit is contained in:
+13
-2
@@ -8,8 +8,19 @@ import (
|
|||||||
"github.com/pion/rtp"
|
"github.com/pion/rtp"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const ADTSHeaderSize = 7
|
||||||
|
|
||||||
func IsADTS(b []byte) bool {
|
func IsADTS(b []byte) bool {
|
||||||
return len(b) > 7 && b[0] == 0xFF && b[1]&0xF6 == 0xF0
|
// AAAAAAAA AAAABCCD EEFFFFGH HHIJKLMM MMMMMMMM MMMOOOOO OOOOOOPP (QQQQQQQQ QQQQQQQQ)
|
||||||
|
// A 12 Syncword, all bits must be set to 1.
|
||||||
|
// C 2 Layer, always set to 0.
|
||||||
|
return len(b) >= ADTSHeaderSize && b[0] == 0xFF && b[1]&0b1111_0110 == 0xF0
|
||||||
|
}
|
||||||
|
|
||||||
|
func HasCRC(b []byte) bool {
|
||||||
|
// AAAAAAAA AAAABCCD EEFFFFGH HHIJKLMM MMMMMMMM MMMOOOOO OOOOOOPP (QQQQQQQQ QQQQQQQQ)
|
||||||
|
// D 1 Protection absence, set to 1 if there is no CRC and 0 if there is CRC.
|
||||||
|
return b[1]&0b1 == 0
|
||||||
}
|
}
|
||||||
|
|
||||||
func ADTSToCodec(b []byte) *core.Codec {
|
func ADTSToCodec(b []byte) *core.Codec {
|
||||||
@@ -58,7 +69,7 @@ func ADTSToCodec(b []byte) *core.Codec {
|
|||||||
func ReadADTSSize(b []byte) uint16 {
|
func ReadADTSSize(b []byte) uint16 {
|
||||||
// AAAAAAAA AAAABCCD EEFFFFGH HHIJKLMM MMMMMMMM MMMOOOOO OOOOOOPP (QQQQQQQQ QQQQQQQQ)
|
// AAAAAAAA AAAABCCD EEFFFFGH HHIJKLMM MMMMMMMM MMMOOOOO OOOOOOPP (QQQQQQQQ QQQQQQQQ)
|
||||||
_ = b[5] // bounds
|
_ = b[5] // bounds
|
||||||
return uint16(b[3]&0x03)<<(8+3) | uint16(b[4])<<3 | uint16(b[5]>>5)
|
return uint16(b[3]&0b11)<<11 | uint16(b[4])<<3 | uint16(b[5]>>5)
|
||||||
}
|
}
|
||||||
|
|
||||||
func WriteADTSSize(b []byte, size uint16) {
|
func WriteADTSSize(b []byte, size uint16) {
|
||||||
|
|||||||
+25
-11
@@ -2,7 +2,7 @@ package aac
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"encoding/binary"
|
"errors"
|
||||||
"io"
|
"io"
|
||||||
|
|
||||||
"github.com/AlexxIT/go2rtc/pkg/core"
|
"github.com/AlexxIT/go2rtc/pkg/core"
|
||||||
@@ -17,16 +17,22 @@ type Producer struct {
|
|||||||
func Open(r io.Reader) (*Producer, error) {
|
func Open(r io.Reader) (*Producer, error) {
|
||||||
rd := bufio.NewReader(r)
|
rd := bufio.NewReader(r)
|
||||||
|
|
||||||
b, err := rd.Peek(8)
|
b, err := rd.Peek(ADTSHeaderSize)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
codec := ADTSToCodec(b)
|
||||||
|
if codec == nil {
|
||||||
|
return nil, errors.New("adts: wrong header")
|
||||||
|
}
|
||||||
|
codec.PayloadType = core.PayloadTypeRAW
|
||||||
|
|
||||||
medias := []*core.Media{
|
medias := []*core.Media{
|
||||||
{
|
{
|
||||||
Kind: core.KindAudio,
|
Kind: core.KindAudio,
|
||||||
Direction: core.DirectionRecvonly,
|
Direction: core.DirectionRecvonly,
|
||||||
Codecs: []*core.Codec{ADTSToCodec(b)},
|
Codecs: []*core.Codec{codec},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
return &Producer{
|
return &Producer{
|
||||||
@@ -42,14 +48,25 @@ func Open(r io.Reader) (*Producer, error) {
|
|||||||
|
|
||||||
func (c *Producer) Start() error {
|
func (c *Producer) Start() error {
|
||||||
for {
|
for {
|
||||||
b, err := c.rd.Peek(6)
|
// read ADTS header
|
||||||
if err != nil {
|
adts := make([]byte, ADTSHeaderSize)
|
||||||
|
if _, err := io.ReadFull(c.rd, adts); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
auSize := ReadADTSSize(b)
|
auSize := ReadADTSSize(adts) - ADTSHeaderSize
|
||||||
payload := make([]byte, 2+2+auSize)
|
|
||||||
if _, err = io.ReadFull(c.rd, payload[4:]); err != nil {
|
if HasCRC(adts) {
|
||||||
|
// skip CRC after header
|
||||||
|
if _, err := c.rd.Discard(2); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
auSize -= 2
|
||||||
|
}
|
||||||
|
|
||||||
|
// read AAC payload after header
|
||||||
|
payload := make([]byte, auSize)
|
||||||
|
if _, err := io.ReadFull(c.rd, payload); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -59,9 +76,6 @@ func (c *Producer) Start() error {
|
|||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
payload[1] = 16 // header size in bits
|
|
||||||
binary.BigEndian.PutUint16(payload[2:], auSize<<3)
|
|
||||||
|
|
||||||
pkt := &rtp.Packet{
|
pkt := &rtp.Packet{
|
||||||
Header: rtp.Header{Timestamp: core.Now90000()},
|
Header: rtp.Header{Timestamp: core.Now90000()},
|
||||||
Payload: payload,
|
Payload: payload,
|
||||||
|
|||||||
+7
-4
@@ -8,7 +8,6 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
const RTPPacketVersionAAC = 0
|
const RTPPacketVersionAAC = 0
|
||||||
const ADTSHeaderSize = 7
|
|
||||||
|
|
||||||
func RTPDepay(handler core.HandlerFunc) core.HandlerFunc {
|
func RTPDepay(handler core.HandlerFunc) core.HandlerFunc {
|
||||||
var timestamp uint32
|
var timestamp uint32
|
||||||
@@ -65,7 +64,8 @@ func RTPDepay(handler core.HandlerFunc) core.HandlerFunc {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func RTPPay(handler core.HandlerFunc) core.HandlerFunc {
|
func RTPPay(handler core.HandlerFunc) core.HandlerFunc {
|
||||||
sequencer := rtp.NewRandomSequencer()
|
var seq uint16
|
||||||
|
var ts uint32
|
||||||
|
|
||||||
return func(packet *rtp.Packet) {
|
return func(packet *rtp.Packet) {
|
||||||
if packet.Version != RTPPacketVersionAAC {
|
if packet.Version != RTPPacketVersionAAC {
|
||||||
@@ -85,12 +85,15 @@ func RTPPay(handler core.HandlerFunc) core.HandlerFunc {
|
|||||||
Header: rtp.Header{
|
Header: rtp.Header{
|
||||||
Version: 2,
|
Version: 2,
|
||||||
Marker: true,
|
Marker: true,
|
||||||
SequenceNumber: sequencer.NextSequenceNumber(),
|
SequenceNumber: seq,
|
||||||
Timestamp: packet.Timestamp,
|
Timestamp: ts,
|
||||||
},
|
},
|
||||||
Payload: payload,
|
Payload: payload,
|
||||||
}
|
}
|
||||||
handler(&clone)
|
handler(&clone)
|
||||||
|
|
||||||
|
seq++
|
||||||
|
ts += AUTime
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user