mirror of
https://github.com/aler9/gortsplib
synced 2025-09-27 11:32:08 +08:00

* switch from v4 to v5 * remove deprecated entities * remove "2" suffix from entities * rename TransportProtocol into Protocol
173 lines
4.2 KiB
Go
173 lines
4.2 KiB
Go
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"math"
|
|
"unsafe"
|
|
)
|
|
|
|
// #cgo pkg-config: libavcodec libavutil
|
|
// #include <libavcodec/avcodec.h>
|
|
// #include <libavutil/opt.h>
|
|
// #include <libavutil/channel_layout.h>
|
|
import "C"
|
|
|
|
func frameData(frame *C.AVFrame) **C.uint8_t {
|
|
return (**C.uint8_t)(unsafe.Pointer(&frame.data[0]))
|
|
}
|
|
|
|
func frameLineSize(frame *C.AVFrame) *C.int {
|
|
return (*C.int)(unsafe.Pointer(&frame.linesize[0]))
|
|
}
|
|
|
|
func bigEndianS16ToFloat3(samples []byte) ([]byte, error) {
|
|
sampleCount := len(samples) / 2
|
|
outSize := len(samples) * 2
|
|
outSamples := make([]byte, outSize)
|
|
|
|
for i := 0; i < sampleCount; i++ {
|
|
s := (int16(samples[(i*2)]) << 8) | int16(samples[(i*2)+1])
|
|
f := float32(s) / 32767
|
|
b := math.Float32bits(f)
|
|
outSamples[(i * 4)] = byte(b)
|
|
outSamples[(i*4)+1] = byte(b >> 8)
|
|
outSamples[(i*4)+2] = byte(b >> 16)
|
|
outSamples[(i*4)+3] = byte(b >> 24)
|
|
}
|
|
|
|
return outSamples, nil
|
|
}
|
|
|
|
// mp4aEncoder is a wrapper around FFmpeg's MPEG-4 Audio encoder.
|
|
type mp4aEncoder struct {
|
|
Width int
|
|
Height int
|
|
FPS int
|
|
|
|
codecCtx *C.AVCodecContext
|
|
frame *C.AVFrame
|
|
pkt *C.AVPacket
|
|
samplesBuffer []byte
|
|
samplesBufferPTS int64
|
|
}
|
|
|
|
// initialize initializes a mp4aEncoder.
|
|
func (d *mp4aEncoder) initialize() error {
|
|
codec := C.avcodec_find_encoder(C.AV_CODEC_ID_AAC)
|
|
if codec == nil {
|
|
return fmt.Errorf("avcodec_find_encoder() failed")
|
|
}
|
|
|
|
d.codecCtx = C.avcodec_alloc_context3(codec)
|
|
if d.codecCtx == nil {
|
|
return fmt.Errorf("avcodec_alloc_context3() failed")
|
|
}
|
|
|
|
d.codecCtx.bit_rate = 64000
|
|
d.codecCtx.sample_fmt = C.AV_SAMPLE_FMT_FLT
|
|
d.codecCtx.sample_rate = 48000
|
|
d.codecCtx.channel_layout = C.AV_CH_LAYOUT_MONO
|
|
d.codecCtx.channels = C.av_get_channel_layout_nb_channels(d.codecCtx.channel_layout)
|
|
|
|
res := C.avcodec_open2(d.codecCtx, codec, nil)
|
|
if res < 0 {
|
|
C.avcodec_close(d.codecCtx)
|
|
return fmt.Errorf("avcodec_open2() failed")
|
|
}
|
|
|
|
d.frame = C.av_frame_alloc()
|
|
if d.frame == nil {
|
|
C.avcodec_close(d.codecCtx)
|
|
return fmt.Errorf("av_frame_alloc() failed")
|
|
}
|
|
|
|
d.frame.nb_samples = d.codecCtx.frame_size
|
|
d.frame.format = (C.int)(d.codecCtx.sample_fmt)
|
|
d.frame.channel_layout = d.codecCtx.channel_layout
|
|
|
|
res = C.av_frame_get_buffer(d.frame, 0)
|
|
if res < 0 {
|
|
C.av_frame_free(&d.frame)
|
|
C.avcodec_close(d.codecCtx)
|
|
return fmt.Errorf("av_frame_get_buffer() failed")
|
|
}
|
|
|
|
d.pkt = C.av_packet_alloc()
|
|
if d.pkt == nil {
|
|
C.av_frame_free(&d.frame)
|
|
C.avcodec_close(d.codecCtx)
|
|
return fmt.Errorf("av_packet_alloc() failed")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// close closes the decoder.
|
|
func (d *mp4aEncoder) close() {
|
|
C.av_packet_free(&d.pkt)
|
|
C.av_frame_free(&d.frame)
|
|
C.avcodec_close(d.codecCtx)
|
|
}
|
|
|
|
// encode encodes LPCM samples into Opus packets.
|
|
func (d *mp4aEncoder) encode(samples []byte) ([][]byte, int64, error) {
|
|
// convert from big-endian signed 16-bit integer to float32
|
|
samples, err := bigEndianS16ToFloat3(samples)
|
|
if err != nil {
|
|
return nil, 0, err
|
|
}
|
|
|
|
// put samples into an internal buffer
|
|
d.samplesBuffer = append(d.samplesBuffer, samples...)
|
|
|
|
// split buffer into AVFrames
|
|
requiredSampleSize := (int)(d.codecCtx.frame_size) * 4
|
|
frameCount := len(d.samplesBuffer) / requiredSampleSize
|
|
if frameCount == 0 {
|
|
return nil, 0, fmt.Errorf("sample buffer is not filled enough")
|
|
}
|
|
|
|
ret := make([][]byte, frameCount)
|
|
var pts int64
|
|
|
|
for i := 0; i < frameCount; i++ {
|
|
samples = d.samplesBuffer[:requiredSampleSize]
|
|
d.samplesBuffer = d.samplesBuffer[requiredSampleSize:]
|
|
|
|
samplePTS := d.samplesBufferPTS
|
|
d.samplesBufferPTS += int64(len(samples) / 4)
|
|
|
|
// pass samples pointer to frame
|
|
d.frame.data[0] = (*C.uint8_t)(&samples[0])
|
|
|
|
// send frame to the encoder
|
|
d.frame.pts = (C.int64_t)(samplePTS)
|
|
res := C.avcodec_send_frame(d.codecCtx, d.frame)
|
|
if res < 0 {
|
|
return nil, 0, fmt.Errorf("avcodec_send_frame() failed")
|
|
}
|
|
|
|
// wait for result
|
|
res = C.avcodec_receive_packet(d.codecCtx, d.pkt)
|
|
if res == -C.EAGAIN {
|
|
return nil, 0, nil
|
|
}
|
|
if res < 0 {
|
|
return nil, 0, fmt.Errorf("avcodec_receive_packet() failed")
|
|
}
|
|
|
|
// perform a deep copy of the data before unreferencing the packet
|
|
data := C.GoBytes(unsafe.Pointer(d.pkt.data), d.pkt.size)
|
|
|
|
if i == 0 {
|
|
pts = (int64)(d.pkt.pts)
|
|
}
|
|
|
|
C.av_packet_unref(d.pkt)
|
|
|
|
ret[i] = data
|
|
}
|
|
|
|
return ret, pts, nil
|
|
}
|