Fix adts producer for VLC player support #1643

This commit is contained in:
Alex X
2025-11-20 17:52:08 +03:00
parent 68036b68c1
commit aa0ece2d1e
3 changed files with 45 additions and 17 deletions
+13 -2
View File
@@ -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
View File
@@ -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
View File
@@ -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
} }
} }