mirror of
https://github.com/aler9/gortsplib
synced 2025-09-30 21:12:18 +08:00
110 lines
2.2 KiB
Go
110 lines
2.2 KiB
Go
package base
|
|
|
|
import (
|
|
"bufio"
|
|
"bytes"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"io"
|
|
)
|
|
|
|
const (
|
|
interleavedFrameMagicByte = 0x24
|
|
)
|
|
|
|
// ReadInterleavedFrameOrRequest reads an InterleavedFrame or a Response.
|
|
func ReadInterleavedFrameOrRequest(frame *InterleavedFrame, 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(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, 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(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
|
|
|
|
// frame payload
|
|
Payload []byte
|
|
}
|
|
|
|
// Read reads an interleaved frame.
|
|
func (f *InterleavedFrame) Read(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])
|
|
}
|
|
|
|
framelen := int(binary.BigEndian.Uint16(header[2:]))
|
|
if framelen > len(f.Payload) {
|
|
return fmt.Errorf("payload size greater than maximum allowed (%d vs %d)",
|
|
framelen, len(f.Payload))
|
|
}
|
|
|
|
f.Channel = int(header[1])
|
|
f.Payload = f.Payload[:framelen]
|
|
|
|
_, err = io.ReadFull(br, f.Payload)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Write writes an InterleavedFrame into a buffered writer.
|
|
func (f InterleavedFrame) Write(bb *bytes.Buffer) {
|
|
bb.Reset()
|
|
|
|
buf := []byte{0x24, byte(f.Channel), 0x00, 0x00}
|
|
binary.BigEndian.PutUint16(buf[2:], uint16(len(f.Payload)))
|
|
|
|
bb.Write(buf)
|
|
|
|
bb.Write(f.Payload)
|
|
}
|