111 lines
1.9 KiB
Go
111 lines
1.9 KiB
Go
package y4m
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"errors"
|
|
"io"
|
|
|
|
"github.com/AlexxIT/go2rtc/pkg/core"
|
|
"github.com/pion/rtp"
|
|
)
|
|
|
|
func Open(r io.Reader) (*Producer, error) {
|
|
rd := bufio.NewReaderSize(r, core.BufferSize)
|
|
b, err := rd.ReadBytes('\n')
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
b = b[:len(b)-1] // remove \n
|
|
|
|
sdp := string(b)
|
|
var fmtp string
|
|
|
|
for b != nil {
|
|
// YUV4MPEG2 W1280 H720 F24:1 Ip A1:1 C420mpeg2 XYSCSS=420MPEG2
|
|
// https://manned.org/yuv4mpeg.5
|
|
// https://github.com/FFmpeg/FFmpeg/blob/master/libavformat/yuv4mpegenc.c
|
|
key := b[0]
|
|
var value string
|
|
if i := bytes.IndexByte(b, ' '); i > 0 {
|
|
value = string(b[1:i])
|
|
b = b[i+1:]
|
|
} else {
|
|
value = string(b[1:])
|
|
b = nil
|
|
}
|
|
|
|
switch key {
|
|
case 'W':
|
|
fmtp = "width=" + value
|
|
case 'H':
|
|
fmtp += ";height=" + value
|
|
case 'C':
|
|
fmtp += ";colorspace=" + value
|
|
}
|
|
}
|
|
|
|
if GetSize(fmtp) == 0 {
|
|
return nil, errors.New("y4m: unsupported format: " + sdp)
|
|
}
|
|
|
|
prod := &Producer{rd: rd, cl: r.(io.Closer)}
|
|
prod.Type = "YUV4MPEG2 producer"
|
|
prod.SDP = sdp
|
|
prod.Medias = []*core.Media{
|
|
{
|
|
Kind: core.KindVideo,
|
|
Direction: core.DirectionRecvonly,
|
|
Codecs: []*core.Codec{
|
|
{
|
|
Name: core.CodecRAW,
|
|
ClockRate: 90000,
|
|
FmtpLine: fmtp,
|
|
PayloadType: core.PayloadTypeRAW,
|
|
},
|
|
},
|
|
},
|
|
}
|
|
|
|
return prod, nil
|
|
}
|
|
|
|
type Producer struct {
|
|
core.SuperProducer
|
|
rd *bufio.Reader
|
|
cl io.Closer
|
|
}
|
|
|
|
func (c *Producer) Start() error {
|
|
size := GetSize(c.Medias[0].Codecs[0].FmtpLine)
|
|
|
|
for {
|
|
if _, err := c.rd.Discard(len(frameHdr)); err != nil {
|
|
return err
|
|
}
|
|
|
|
frame := make([]byte, size)
|
|
if _, err := io.ReadFull(c.rd, frame); err != nil {
|
|
return err
|
|
}
|
|
|
|
c.Recv += size
|
|
|
|
if len(c.Receivers) == 0 {
|
|
continue
|
|
}
|
|
|
|
pkt := &rtp.Packet{
|
|
Header: rtp.Header{Timestamp: core.Now90000()},
|
|
Payload: frame,
|
|
}
|
|
c.Receivers[0].WriteRTP(pkt)
|
|
}
|
|
}
|
|
|
|
func (c *Producer) Stop() error {
|
|
_ = c.SuperProducer.Close()
|
|
return c.cl.Close()
|
|
}
|