mirror of
https://github.com/aler9/gortsplib
synced 2025-10-05 15:16:51 +08:00
133 lines
2.6 KiB
Go
133 lines
2.6 KiB
Go
package base
|
|
|
|
import (
|
|
"bufio"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"io"
|
|
)
|
|
|
|
const (
|
|
interleavedFrameMagicByte = 0x24
|
|
)
|
|
|
|
// ReadInterleavedFrameOrRequest reads an InterleavedFrame or a Response.
|
|
func ReadInterleavedFrameOrRequest(
|
|
frame *InterleavedFrame,
|
|
maxPayloadSize int,
|
|
req *Request,
|
|
br *bufio.Reader,
|
|
) (interface{}, error) {
|
|
b, err := br.ReadByte()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
br.UnreadByte()
|
|
|
|
if b == interleavedFrameMagicByte {
|
|
err := frame.Read(maxPayloadSize, br)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return frame, err
|
|
}
|
|
|
|
err = req.Read(br)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return req, nil
|
|
}
|
|
|
|
// ReadInterleavedFrameOrResponse reads an InterleavedFrame or a Response.
|
|
func ReadInterleavedFrameOrResponse(
|
|
frame *InterleavedFrame,
|
|
maxPayloadSize int,
|
|
res *Response,
|
|
br *bufio.Reader,
|
|
) (interface{}, error) {
|
|
b, err := br.ReadByte()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
br.UnreadByte()
|
|
|
|
if b == interleavedFrameMagicByte {
|
|
err := frame.Read(maxPayloadSize, br)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return frame, err
|
|
}
|
|
|
|
err = res.Read(br)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return res, nil
|
|
}
|
|
|
|
// InterleavedFrame is an interleaved frame, and allows to transfer binary data
|
|
// within RTSP/TCP connections. It is used to send and receive RTP and RTCP packets with TCP.
|
|
type InterleavedFrame struct {
|
|
// channel ID
|
|
Channel int
|
|
|
|
// payload
|
|
Payload []byte
|
|
}
|
|
|
|
// Read reads an interleaved frame.
|
|
func (f *InterleavedFrame) Read(maxPayloadSize int, br *bufio.Reader) error {
|
|
var header [4]byte
|
|
_, err := io.ReadFull(br, header[:])
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if header[0] != interleavedFrameMagicByte {
|
|
return fmt.Errorf("invalid magic byte (0x%.2x)", header[0])
|
|
}
|
|
|
|
payloadLen := int(binary.BigEndian.Uint16(header[2:]))
|
|
if payloadLen > maxPayloadSize {
|
|
return fmt.Errorf("payload size (%d) greater than maximum allowed (%d)",
|
|
payloadLen, maxPayloadSize)
|
|
}
|
|
|
|
f.Channel = int(header[1])
|
|
f.Payload = make([]byte, payloadLen)
|
|
|
|
_, err = io.ReadFull(br, f.Payload)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// WriteSize returns the size of an InterleavedFrame.
|
|
func (f InterleavedFrame) WriteSize() int {
|
|
return 4 + len(f.Payload)
|
|
}
|
|
|
|
// WriteTo writes an InterleavedFrame.
|
|
func (f InterleavedFrame) WriteTo(buf []byte) (int, error) {
|
|
pos := 0
|
|
|
|
pos += copy(buf[pos:], []byte{0x24, byte(f.Channel)})
|
|
|
|
binary.BigEndian.PutUint16(buf[pos:], uint16(len(f.Payload)))
|
|
pos += 2
|
|
|
|
pos += copy(buf[pos:], f.Payload)
|
|
|
|
return pos, nil
|
|
}
|
|
|
|
// Write writes an InterleavedFrame.
|
|
func (f InterleavedFrame) Write() ([]byte, error) {
|
|
buf := make([]byte, f.WriteSize())
|
|
_, err := f.WriteTo(buf)
|
|
return buf, err
|
|
}
|