Rewrite magic source

This commit is contained in:
Alexey Khit
2023-08-19 15:19:09 +03:00
parent a9118562a9
commit bc60cbefb8
11 changed files with 124 additions and 114 deletions

View File

@@ -40,7 +40,7 @@ func ADTSToCodec(b []byte) *core.Codec {
//_ = rd.ReadBits(16) // CRC check //_ = rd.ReadBits(16) // CRC check
// 3. Encode RTP config // 3. Encode RTP config
wr := bits.NewWriter() wr := bits.NewWriter(nil)
wr.WriteBits8(objType, 5) wr.WriteBits8(objType, 5)
wr.WriteBits8(sampleRateIdx, 4) wr.WriteBits8(sampleRateIdx, 4)
wr.WriteBits16(channels, 4) wr.WriteBits16(channels, 4)

View File

@@ -47,7 +47,10 @@ type Producer interface {
// GetTrack - return Receiver, that can only produce rtp.Packet(s) // GetTrack - return Receiver, that can only produce rtp.Packet(s)
GetTrack(media *Media, codec *Codec) (*Receiver, error) GetTrack(media *Media, codec *Codec) (*Receiver, error)
// Deprecated: rename to Run()
Start() error Start() error
// Deprecated: rename to Close()
Stop() error Stop() error
} }
@@ -59,6 +62,7 @@ type Consumer interface {
AddTrack(media *Media, codec *Codec, track *Receiver) error AddTrack(media *Media, codec *Codec, track *Receiver) error
// Deprecated: rename to Close()
Stop() error Stop() error
} }
@@ -102,3 +106,60 @@ const (
UnsupportedCodec = "unsupported codec" UnsupportedCodec = "unsupported codec"
WrongMediaDirection = "wrong media direction" WrongMediaDirection = "wrong media direction"
) )
type SuperProducer struct {
Type string `json:"type,omitempty"`
URL string `json:"url,omitempty"`
Medias []*Media `json:"medias,omitempty"`
Receivers []*Receiver `json:"receivers,omitempty"`
Recv int `json:"recv,omitempty"`
}
func (s *SuperProducer) GetMedias() []*Media {
return s.Medias
}
func (s *SuperProducer) GetTrack(media *Media, codec *Codec) (*Receiver, error) {
for _, receiver := range s.Receivers {
if receiver.Codec == codec {
return receiver, nil
}
}
receiver := NewReceiver(media, codec)
s.Receivers = append(s.Receivers, receiver)
return receiver, nil
}
func (s *SuperProducer) Close() error {
for _, receiver := range s.Receivers {
receiver.Close()
}
return nil
}
type SuperConsumer struct {
Type string `json:"type,omitempty"`
URL string `json:"url,omitempty"`
Medias []*Media `json:"medias,omitempty"`
Senders []*Sender `json:"receivers,omitempty"`
Send int `json:"recv,omitempty"`
}
func (s *SuperConsumer) GetMedias() []*Media {
return s.Medias
}
func (s *SuperConsumer) AddTrack(media *Media, codec *Codec, track *Receiver) error {
return nil
}
//func (b *SuperConsumer) WriteTo(w io.Writer) (n int64, err error) {
// return 0, nil
//}
func (s *SuperConsumer) Close() error {
for _, sender := range s.Senders {
sender.Close()
}
return nil
}

View File

@@ -12,13 +12,13 @@ const (
BufferDrainAndClear = -1 BufferDrainAndClear = -1
) )
// ReadSeeker support buffering and Seek over buffer // ReadBuffer support buffering and Seek over buffer
// positive BufferSize will enable buffering mode // positive BufferSize will enable buffering mode
// Seek to negative offset will clear buffer // Seek to negative offset will clear buffer
// Seek with a positive BufferSize will continue buffering after the last read from the buffer // Seek with a positive BufferSize will continue buffering after the last read from the buffer
// Seek with a negative BufferSize will clear buffer after the last read from the buffer // Seek with a negative BufferSize will clear buffer after the last read from the buffer
// Read more than BufferSize will raise error // Read more than BufferSize will raise error
type ReadSeeker struct { type ReadBuffer struct {
io.Reader io.Reader
BufferSize int BufferSize int
@@ -27,14 +27,14 @@ type ReadSeeker struct {
pos int pos int
} }
func NewReadSeeker(rd io.Reader) *ReadSeeker { func NewReadBuffer(rd io.Reader) *ReadBuffer {
if rs, ok := rd.(*ReadSeeker); ok { if rs, ok := rd.(*ReadBuffer); ok {
return rs return rs
} }
return &ReadSeeker{Reader: rd} return &ReadBuffer{Reader: rd}
} }
func (r *ReadSeeker) Read(p []byte) (n int, err error) { func (r *ReadBuffer) Read(p []byte) (n int, err error) {
// with zero buffer - read as usual // with zero buffer - read as usual
if r.BufferSize == BufferDisable { if r.BufferSize == BufferDisable {
return r.Reader.Read(p) return r.Reader.Read(p)
@@ -65,7 +65,14 @@ func (r *ReadSeeker) Read(p []byte) (n int, err error) {
return return
} }
func (r *ReadSeeker) Seek(offset int64, whence int) (int64, error) { func (r *ReadBuffer) Close() error {
if closer, ok := r.Reader.(io.Closer); ok {
return closer.Close()
}
return nil
}
func (r *ReadBuffer) Seek(offset int64, whence int) (int64, error) {
var pos int var pos int
switch whence { switch whence {
case io.SeekStart: case io.SeekStart:
@@ -89,17 +96,17 @@ func (r *ReadSeeker) Seek(offset int64, whence int) (int64, error) {
return int64(r.pos), nil return int64(r.pos), nil
} }
func (r *ReadSeeker) Peek(n int) ([]byte, error) { func (r *ReadBuffer) Peek(n int) ([]byte, error) {
r.BufferSize = n r.BufferSize = n
b := make([]byte, n) b := make([]byte, n)
if _, err := io.ReadAtLeast(r, b, n); err != nil { if _, err := io.ReadAtLeast(r, b, n); err != nil {
return nil, err return nil, err
} }
r.Rewind() r.Reset()
return b, nil return b, nil
} }
func (r *ReadSeeker) Rewind() { func (r *ReadBuffer) Reset() {
r.BufferSize = BufferDrainAndClear r.BufferSize = BufferDrainAndClear
r.pos = 0 r.pos = 0
} }

View File

@@ -12,7 +12,7 @@ func TestReadSeeker(t *testing.T) {
b := []byte{0, 1, 2, 3, 4, 5, 6, 7, 8, 9} b := []byte{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
buf := bytes.NewReader(b) buf := bytes.NewReader(b)
rd := NewReadSeeker(buf) rd := NewReadBuffer(buf)
rd.BufferSize = ProbeSize rd.BufferSize = ProbeSize
// 1. Read to buffer // 1. Read to buffer

View File

@@ -18,7 +18,7 @@ const Signature = "FLV"
type Client struct { type Client struct {
URL string URL string
rd *core.ReadSeeker rd *core.ReadBuffer
medias []*core.Media medias []*core.Media
receivers []*core.Receiver receivers []*core.Receiver
@@ -30,7 +30,7 @@ type Client struct {
func Open(rd io.Reader) (*Client, error) { func Open(rd io.Reader) (*Client, error) {
client := &Client{ client := &Client{
rd: core.NewReadSeeker(rd), rd: core.NewReadBuffer(rd),
} }
if err := client.describe(); err != nil { if err := client.describe(); err != nil {
return nil, err return nil, err
@@ -53,7 +53,7 @@ func (c *Client) describe() error {
} }
c.rd.BufferSize = core.ProbeSize c.rd.BufferSize = core.ProbeSize
defer c.rd.Rewind() defer c.rd.Reset()
// Normal software sends: // Normal software sends:
// 1. Video/audio flag in header // 1. Video/audio flag in header

View File

@@ -70,8 +70,10 @@ func EncodeToAVCC(b []byte, safeAppend bool) []byte {
return b return b
} }
func DecodeAVCC(b []byte) []byte { func DecodeAVCC(b []byte, safeClone bool) []byte {
if safeClone {
b = bytes.Clone(b) b = bytes.Clone(b)
}
for i := 0; i < len(b); { for i := 0; i < len(b); {
size := int(binary.BigEndian.Uint32(b[i:])) size := int(binary.BigEndian.Uint32(b[i:]))
b[i] = 0 b[i] = 0

View File

@@ -2,7 +2,6 @@ package bitstream
import ( import (
"encoding/hex" "encoding/hex"
"encoding/json"
"errors" "errors"
"io" "io"
@@ -13,17 +12,13 @@ import (
"github.com/pion/rtp" "github.com/pion/rtp"
) )
type Client struct { type Producer struct {
rd *core.ReadSeeker core.SuperProducer
rd *core.ReadBuffer
media *core.Media
receiver *core.Receiver
recv int
} }
func Open(r io.Reader) (*Client, error) { func Open(r io.Reader) (*Producer, error) {
rd := core.NewReadSeeker(r) rd := core.NewReadBuffer(r)
buf, err := rd.Peek(256) buf, err := rd.Peek(256)
if err != nil { if err != nil {
@@ -43,30 +38,19 @@ func Open(r io.Reader) (*Client, error) {
return nil, errors.New("bitstream: unsupported header: " + hex.EncodeToString(buf[:8])) return nil, errors.New("bitstream: unsupported header: " + hex.EncodeToString(buf[:8]))
} }
client := &Client{ prod := &Producer{rd: rd}
rd: rd, prod.Type = "Bitstream producer"
media: &core.Media{ prod.Medias = []*core.Media{
{
Kind: core.KindVideo, Kind: core.KindVideo,
Direction: core.DirectionRecvonly, Direction: core.DirectionRecvonly,
Codecs: []*core.Codec{codec}, Codecs: []*core.Codec{codec},
}, },
} }
return prod, nil
return client, nil
} }
func (c *Client) GetMedias() []*core.Media { func (c *Producer) Start() error {
return []*core.Media{c.media}
}
func (c *Client) GetTrack(media *core.Media, codec *core.Codec) (*core.Receiver, error) {
if c.receiver == nil {
c.receiver = core.NewReceiver(media, codec)
}
return c.receiver, nil
}
func (c *Client) Start() error {
var buf []byte var buf []byte
b := make([]byte, core.BufferSize) b := make([]byte, core.BufferSize)
@@ -76,7 +60,7 @@ func (c *Client) Start() error {
return err return err
} }
c.recv += n c.Recv += n
buf = append(buf, b[:n]...) buf = append(buf, b[:n]...)
@@ -89,7 +73,7 @@ func (c *Client) Start() error {
Header: rtp.Header{Timestamp: core.Now90000()}, Header: rtp.Header{Timestamp: core.Now90000()},
Payload: annexb.EncodeToAVCC(buf[:i], true), Payload: annexb.EncodeToAVCC(buf[:i], true),
} }
c.receiver.WriteRTP(pkt) c.Receivers[0].WriteRTP(pkt)
//log.Printf("[AVC] %v, len: %d", h264.Types(pkt.Payload), len(pkt.Payload)) //log.Printf("[AVC] %v, len: %d", h264.Types(pkt.Payload), len(pkt.Payload))
@@ -97,22 +81,7 @@ func (c *Client) Start() error {
} }
} }
func (c *Client) Stop() error { func (c *Producer) Stop() error {
if c.receiver != nil { _ = c.SuperProducer.Close()
c.receiver.Close() return c.rd.Close()
}
if closer, ok := c.rd.Reader.(io.Closer); ok {
return closer.Close()
}
return nil
}
func (c *Client) MarshalJSON() ([]byte, error) {
info := &core.Info{
Type: "Bitstream active producer",
Medias: []*core.Media{c.media},
Receivers: []*core.Receiver{c.receiver},
Recv: c.recv,
}
return json.Marshal(info)
} }

View File

@@ -43,7 +43,7 @@ func (k *Keyframe) AddTrack(media *core.Media, _ *core.Codec, track *core.Receiv
if !h264.IsKeyframe(packet.Payload) { if !h264.IsKeyframe(packet.Payload) {
return return
} }
b := annexb.DecodeAVCC(packet.Payload) b := annexb.DecodeAVCC(packet.Payload, true)
k.Fire(b) k.Fire(b)
} }

View File

@@ -2,26 +2,22 @@ package mjpeg
import ( import (
"bytes" "bytes"
"encoding/json"
"io" "io"
"github.com/AlexxIT/go2rtc/pkg/core" "github.com/AlexxIT/go2rtc/pkg/core"
"github.com/pion/rtp" "github.com/pion/rtp"
) )
type Client struct { type Producer struct {
rd *core.ReadSeeker core.SuperProducer
rd *core.ReadBuffer
media *core.Media
receiver *core.Receiver
recv int
} }
func NewClient(rd io.Reader) *Client { func Open(rd io.Reader) (*Producer, error) {
return &Client{ prod := &Producer{rd: core.NewReadBuffer(rd)}
rd: core.NewReadSeeker(rd), prod.Type = "MJPEG producer"
media: &core.Media{ prod.Medias = []*core.Media{
{
Kind: core.KindVideo, Kind: core.KindVideo,
Direction: core.DirectionRecvonly, Direction: core.DirectionRecvonly,
Codecs: []*core.Codec{ Codecs: []*core.Codec{
@@ -33,20 +29,10 @@ func NewClient(rd io.Reader) *Client {
}, },
}, },
} }
return prod, nil
} }
func (c *Client) GetMedias() []*core.Media { func (c *Producer) Start() error {
return []*core.Media{c.media}
}
func (c *Client) GetTrack(media *core.Media, codec *core.Codec) (*core.Receiver, error) {
if c.receiver == nil {
c.receiver = core.NewReceiver(media, codec)
}
return c.receiver, nil
}
func (c *Client) Start() error {
var buf []byte // total bufer var buf []byte // total bufer
b := make([]byte, core.BufferSize) // reading buffer b := make([]byte, core.BufferSize) // reading buffer
@@ -59,7 +45,7 @@ func (c *Client) Start() error {
return err return err
} }
c.recv += n c.Recv += n
buf = append(buf, b[:n]...) buf = append(buf, b[:n]...)
@@ -77,7 +63,7 @@ func (c *Client) Start() error {
Header: rtp.Header{Timestamp: core.Now90000()}, Header: rtp.Header{Timestamp: core.Now90000()},
Payload: buf[:i], Payload: buf[:i],
} }
c.receiver.WriteRTP(pkt) c.Receivers[0].WriteRTP(pkt)
//log.Printf("[mjpeg] ts=%d size=%d", pkt.Header.Timestamp, len(pkt.Payload)) //log.Printf("[mjpeg] ts=%d size=%d", pkt.Header.Timestamp, len(pkt.Payload))
@@ -85,22 +71,7 @@ func (c *Client) Start() error {
} }
} }
func (c *Client) Stop() error { func (c *Producer) Stop() error {
if c.receiver != nil { _ = c.SuperProducer.Close()
c.receiver.Close() return c.rd.Close()
}
if closer, ok := c.rd.Reader.(io.Closer); ok {
return closer.Close()
}
return nil
}
func (c *Client) MarshalJSON() ([]byte, error) {
info := &core.Info{
Type: "MJPEG active producer",
Medias: []*core.Media{c.media},
Receivers: []*core.Receiver{c.receiver},
Recv: c.recv,
}
return json.Marshal(info)
} }

View File

@@ -15,7 +15,7 @@ import (
) )
func Open(r io.Reader) (core.Producer, error) { func Open(r io.Reader) (core.Producer, error) {
rd := core.NewReadSeeker(r) rd := core.NewReadBuffer(r)
b, err := rd.Peek(4) b, err := rd.Peek(4)
if err != nil { if err != nil {
@@ -27,7 +27,7 @@ func Open(r io.Reader) (core.Producer, error) {
return bitstream.Open(rd) return bitstream.Open(rd)
case bytes.HasPrefix(b, []byte{0xFF, 0xD8}): case bytes.HasPrefix(b, []byte{0xFF, 0xD8}):
return mjpeg.NewClient(rd), nil return mjpeg.Open(rd)
case bytes.HasPrefix(b, []byte(flv.Signature)): case bytes.HasPrefix(b, []byte(flv.Signature)):
return flv.Open(rd) return flv.Open(rd)

View File

@@ -14,7 +14,7 @@ import (
type Client struct { type Client struct {
URL string URL string
rd *core.ReadSeeker rd *core.ReadBuffer
medias []*core.Media medias []*core.Media
receivers []*core.Receiver receivers []*core.Receiver
@@ -23,7 +23,7 @@ type Client struct {
} }
func Open(rd io.Reader) (*Client, error) { func Open(rd io.Reader) (*Client, error) {
client := &Client{rd: core.NewReadSeeker(rd)} client := &Client{rd: core.NewReadBuffer(rd)}
if err := client.describe(); err != nil { if err := client.describe(); err != nil {
return nil, err return nil, err
} }
@@ -32,7 +32,7 @@ func Open(rd io.Reader) (*Client, error) {
func (c *Client) describe() error { func (c *Client) describe() error {
c.rd.BufferSize = core.ProbeSize c.rd.BufferSize = core.ProbeSize
defer c.rd.Rewind() defer c.rd.Reset()
rd := NewReader() rd := NewReader()