You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
rtpengine/lib/codeclib.c

2169 lines
57 KiB

#include "codeclib.h"
#include <libavcodec/avcodec.h>
#include <libavformat/avformat.h>
#include <libavfilter/avfilter.h>
#include <libavutil/opt.h>
#include <glib.h>
#include <dlfcn.h>
#ifdef HAVE_CODEC_CHAIN
#include <codec-chain/types.h>
#include <codec-chain/client.h>
#endif
#include "str.h"
#include "loglib.h"
#include "resample.h"
#include "rtplib.h"
#include "dtmflib.h"
#include "fix_frame_channel_layout.compat"
#include "codecmod.h"
#define PACKET_SEQ_DUPE_THRES 100
#define PACKET_TS_RESET_THRES 5000 // milliseconds
#define cdbg(x...) ilogs(internals, LOG_DEBUG, x)
static int generic_silence_dtx(decoder_t *, GQueue *, int);
static int generic_cn_dtx_init(decoder_t *);
static void generic_cn_dtx_cleanup(decoder_t *);
static int generic_cn_dtx(decoder_t *, GQueue *, int);
static void *cc_lib_handle;
#ifdef HAVE_CODEC_CHAIN
static __typeof__(codec_chain_client_connect) *cc_client_connect;
static __typeof__(codec_chain_set_thread_funcs) *cc_set_thread_funcs;
static __typeof__(codec_chain_get) *cc_get;
static __typeof__(codec_chain_client_runner_new) *cc_client_runner_new;
static __typeof__(codec_chain_client_runner_free) *cc_client_runner_free;
static __typeof__(codec_chain_client_async_runner_new) *cc_client_async_runner_new;
static __typeof__(codec_chain_client_async_runner_free) *cc_client_async_runner_free;
static __typeof__(codec_chain_runner_do) *cc_runner_do;
static __typeof__(codec_chain_async_runner_do) *cc_async_runner_do;
static __typeof__(codec_chain_client_codec_new) *cc_client_codec_new;
static __typeof__(codec_chain_client_codec_free) *cc_client_codec_free;
static __typeof__(*codec_chain_defs) *cc_defs;
static codec_chain_client *cc_client;
static union {
codec_chain_runner *sync;
codec_chain_async_runner *async;
} cc_runners[CODEC_CHAIN_ID_MAX];
static struct {
unsigned int async_busy;
unsigned int async_blocked;
unsigned int async_retry;
} cc_stats[CODEC_CHAIN_ID_MAX];
typedef enum {
CCC_OK,
CCC_ASYNC,
CCC_ERR,
} codec_cc_state;
struct async_job {
str data;
unsigned long ts;
void *async_cb_obj;
};
TYPED_GQUEUE(async_job, struct async_job);
struct codec_cc_s {
union {
codec_chain_runner *runner;
codec_chain_async_runner *async_runner;
};
__typeof(&cc_stats[0]) stats;
codec_chain_def *def;
codec_chain_codec *codec;
AVPacket *avpkt;
codec_cc_state (*run)(codec_cc_t *c, const str *data, unsigned long ts, void *);
void (*clear)(void *);
void *clear_arg;
mutex_t async_lock;
AVPacket *avpkt_async;
size_t data_len;
bool async_busy; // currently processing a packet
bool async_blocked; // couldn't find context
bool async_shutdown; // shutdown/free happened while busy
async_job_q async_jobs;
unsigned long ts;
void *(*async_init)(void *, void *, void *);
void (*async_callback)(AVPacket *, void *);
void *async_cb_obj;
};
static codec_cc_t *codec_cc_new_sync(codec_def_t *src, format_t *src_format, codec_def_t *dst,
format_t *dst_format, int bitrate, int ptime,
void *(*async_init)(void *, void *, void *),
void (*async_callback)(AVPacket *, void *));
static codec_cc_t *codec_cc_new_async(codec_def_t *src, format_t *src_format, codec_def_t *dst,
format_t *dst_format, int bitrate, int ptime,
void *(*async_init)(void *, void *, void *),
void (*async_callback)(AVPacket *, void *));
static bool __cc_run_async(codec_cc_t *, const str *, unsigned long, void *);
codec_cc_t *(*codec_cc_new)(codec_def_t *src, format_t *src_format, codec_def_t *dst,
format_t *dst_format, int bitrate, int ptime,
void *(*async_init)(void *, void *, void *),
void (*async_callback)(AVPacket *, void *));
#endif
const codec_type_t codec_type_avcodec = {
.def_init = avc_def_init,
.decoder_init = avc_decoder_init,
.decoder_input = avc_decoder_input,
.decoder_close = avc_decoder_close,
.encoder_init = avc_encoder_init,
.encoder_input = avc_encoder_input,
.encoder_close = avc_encoder_close,
};
const dtx_method_t dtx_method_silence = {
.method_id = DTX_SILENCE,
.do_dtx = generic_silence_dtx,
};
const dtx_method_t dtx_method_cn = {
.method_id = DTX_CN,
.do_dtx = generic_cn_dtx,
.init = generic_cn_dtx_init,
.cleanup = generic_cn_dtx_cleanup,
};
static struct codec_def_s *__codec_defs;
static unsigned int __num_codec_defs;
static GQueue __supplemental_codecs = G_QUEUE_INIT;
const GQueue * const codec_supplemental_codecs = &__supplemental_codecs;
static codec_def_t *codec_def_cn;
static codec_def_t *codec_def_pcm16;
void (*codeclib_thread_init)(void);
void (*codeclib_thread_cleanup)(void);
void (*codeclib_thread_loop)(void);
TYPED_GHASHTABLE(codecs_by_name, str, struct codec_def_s, str_case_hash, str_case_equal, NULL, NULL)
TYPED_GHASHTABLE(codecs_by_id_alloc, void, struct codec_def_s, g_direct_hash, g_direct_equal, NULL, g_free)
static codecs_by_name codecs_by_name_ht;
static rwlock_t generic_ffmpeg_codecs_lock = RWLOCK_STATIC_INIT;
static codecs_by_id_alloc generic_ffmpeg_codecs;
codec_def_t *codec_find(const str *name, enum media_type type) {
codec_def_t *ret = t_hash_table_lookup(codecs_by_name_ht, name);
if (!ret)
return NULL;
if (type && type != ret->media_type)
return NULL;
return ret;
}
codec_def_t *codec_get_pcm16(void) {
return codec_def_pcm16;
}
const char *avc_decoder_init(decoder_t *dec, const str *extra_opts) {
const AVCodec *codec = dec->def->decoder;
if (!codec)
return "codec not supported";
dec->avc.avpkt = av_packet_alloc();
dec->avc.avcctx = avcodec_alloc_context3(codec);
if (!dec->avc.avcctx)
return "failed to alloc codec context";
SET_CHANNELS(dec->avc.avcctx, dec->in_format.channels);
DEF_CH_LAYOUT(&dec->avc.avcctx->CH_LAYOUT, dec->in_format.channels);
dec->avc.avcctx->sample_rate = dec->in_format.clockrate;
if (dec->def->set_dec_options)
dec->def->set_dec_options(dec, extra_opts);
int i = avcodec_open2(dec->avc.avcctx, codec, NULL);
if (i) {
ilog(LOG_ERR | LOG_FLAG_LIMIT, "Error returned from libav: %s", av_error(i));
return "failed to open codec context";
}
#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(61, 19, 0)
avcodec_get_supported_config(dec->avc.avcctx, codec, AV_CODEC_CONFIG_SAMPLE_FORMAT, 0, (const void **) &dec->avc.sample_fmts, NULL);
#else
dec->avc.sample_fmts = codec->sample_fmts;
#endif
for (const enum AVSampleFormat *sfmt = dec->avc.sample_fmts; sfmt && *sfmt != -1; sfmt++)
cdbg("supported sample format for input codec %s: %s",
codec->name, av_get_sample_fmt_name(*sfmt));
return NULL;
}
decoder_t *decoder_new_fmt(codec_def_t *def, int clockrate, int channels, int ptime,
const format_t *resample_fmt)
{
return decoder_new_fmtp(def, clockrate, channels, ptime, resample_fmt, NULL, NULL, NULL);
}
bool codec_parse_fmtp(codec_def_t *def, struct rtp_codec_format *fmtp, const str *fmtp_string,
union codec_format_options *copy)
{
struct rtp_codec_format fmtp_store;
if (copy)
ZERO(*copy);
if (!def)
return false;
if (!def->format_parse)
return true;
if (!fmtp_string)
return true;
if (!fmtp) {
ZERO(fmtp_store);
fmtp = &fmtp_store;
}
if (fmtp->fmtp_parsed) {
if (copy)
*copy = fmtp->parsed;
return true;
}
bool ret = def->format_parse(fmtp, fmtp_string);
if (ret) {
fmtp->fmtp_parsed = true;
if (copy)
*copy = fmtp->parsed;
}
return ret;
}
decoder_t *decoder_new_fmtp(codec_def_t *def, int clockrate, int channels, int ptime,
const format_t *resample_fmt,
struct rtp_codec_format *fmtp, const str *fmtp_string,
const str *extra_opts)
{
const char *err;
decoder_t *ret = NULL;
err = "codec not supported";
if (!def->codec_type)
goto err;
ret = g_new0(__typeof(*ret), 1);
ret->def = def;
ret->clockrate_fact = def->default_clockrate_fact;
format_init(&ret->in_format);
ret->in_format.channels = channels;
ret->in_format.clockrate = clockrate;
// output defaults to same as input
ret->dest_format = ret->in_format;
if (resample_fmt)
ret->dest_format = *resample_fmt;
err = "failed to parse \"fmtp\"";
if (!codec_parse_fmtp(def, fmtp, fmtp_string, &ret->format_options))
goto err;
if (def->select_decoder_format)
def->select_decoder_format(ret, fmtp);
ret->in_format.clockrate = fraction_mult(ret->in_format.clockrate, &ret->clockrate_fact);
ret->dec_out_format = ret->in_format;
if (ptime > 0)
ret->ptime = ptime;
else
ret->ptime = def->default_ptime;
// init with first supported DTX method
enum dtx_method dm = -1;
for (int i = 0; i < NUM_DTX_METHODS; i++) {
if (def->dtx_methods[i]) {
dm = i;
break;
}
}
err = def->codec_type->decoder_init(ret, extra_opts);
if (err)
goto err;
ret->pts = (uint64_t) -1LL;
ret->rtp_ts = (unsigned long) -1L;
decoder_switch_dtx(ret, dm);
return ret;
err:
if (ret)
decoder_close(ret);
if (err)
ilog(LOG_ERR | LOG_FLAG_LIMIT, "Error creating media decoder for codec %s: %s", def->rtpname, err);
return NULL;
}
int decoder_switch_dtx(decoder_t *dec, enum dtx_method dm) {
if (dec->dtx.cleanup)
dec->dtx.cleanup(dec);
ZERO(dec->dtx);
unsigned int i = dm;
if (i >= NUM_DTX_METHODS)
return -1;
const dtx_method_t *dmp = dec->def->dtx_methods[i];
if (!dmp)
return -1;
dec->dtx = *dmp;
if (dmp->init) {
if (dmp->init(dec)) {
ilog(LOG_ERR, "Failed to initialise DTX (%u)", i);
decoder_switch_dtx(dec, -1);
return -1;
}
}
return 0;
}
int decoder_set_cn_dtx(decoder_t *dec, const str *cn_pl) {
if (decoder_switch_dtx(dec, DTX_CN))
return -1;
dec->dtx.cn.cn_payload = cn_pl;
return 0;
}
gboolean decoder_has_dtx(decoder_t *dec) {
return dec->dtx.do_dtx == NULL ? FALSE : TRUE;
}
void avc_decoder_close(decoder_t *dec) {
#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(56, 1, 0)
avcodec_free_context(&dec->avc.avcctx);
#else
avcodec_close(dec->avc.avcctx);
av_free(dec->avc.avcctx);
#endif
av_packet_free(&dec->avc.avpkt);
}
void decoder_close(decoder_t *dec) {
if (!dec)
return;
/// XXX drain inputs and outputs
if (dec->def && dec->def->codec_type && dec->def->codec_type->decoder_close)
dec->def->codec_type->decoder_close(dec);
decoder_switch_dtx(dec, -1);
resample_shutdown(&dec->resampler);
g_free(dec);
}
int avc_decoder_input(decoder_t *dec, const str *data, GQueue *out) {
if (!dec->avc.avpkt)
return -1; // decoder shut down
const char *err;
int av_ret = 0;
dec->avc.avpkt->data = (unsigned char *) data->s;
dec->avc.avpkt->size = data->len;
dec->avc.avpkt->pts = dec->pts;
AVFrame *frame = NULL;
// loop until all input is consumed and all available output has been processed
int keep_going;
do {
keep_going = 0;
int got_frame = 0;
err = "failed to alloc av frame";
frame = av_frame_alloc();
if (!frame)
goto err;
#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(57, 36, 0)
if (dec->avc.avpkt->size) {
av_ret = avcodec_send_packet(dec->avc.avcctx, dec->avc.avpkt);
cdbg("send packet ret %i", av_ret);
err = "failed to send packet to avcodec";
if (av_ret == 0) {
// consumed the packet
dec->avc.avpkt->size = 0;
keep_going = 1;
}
else {
if (av_ret == AVERROR(EAGAIN))
; // try again after reading output
else
goto err;
}
}
av_ret = avcodec_receive_frame(dec->avc.avcctx, frame);
cdbg("receive frame ret %i", av_ret);
err = "failed to receive frame from avcodec";
if (av_ret == 0) {
// got a frame
keep_going = 1;
got_frame = 1;
}
else {
if (av_ret == AVERROR(EAGAIN))
; // maybe needs more input now
else
goto err;
}
#else
// only do this if we have any input left
if (dec->avc.avpkt->size == 0)
break;
av_ret = avcodec_decode_audio4(dec->avc.avcctx, frame, &got_frame, dec->avc.avpkt);
cdbg("decode frame ret %i, got frame %i", av_ret, got_frame);
err = "failed to decode audio packet";
if (av_ret < 0)
goto err;
if (av_ret > 0) {
// consumed some input
err = "invalid return value";
if (av_ret > dec->avc.avpkt->size)
goto err;
dec->avc.avpkt->size -= av_ret;
dec->avc.avpkt->data += av_ret;
keep_going = 1;
}
if (got_frame)
keep_going = 1;
#endif
if (got_frame) {
cdbg("raw frame from decoder pts %llu samples %u",
(unsigned long long) frame->pts, frame->nb_samples);
#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(57, 36, 0)
frame->pts = frame->pkt_pts;
#endif
if (G_UNLIKELY(frame->pts == AV_NOPTS_VALUE))
frame->pts = dec->avc.avpkt->pts;
dec->avc.avpkt->pts += frame->nb_samples;
g_queue_push_tail(out, frame);
frame = NULL;
}
} while (keep_going);
av_frame_free(&frame);
return 0;
err:
ilog(LOG_ERR | LOG_FLAG_LIMIT, "Error decoding media packet: %s", err);
if (av_ret)
ilog(LOG_ERR | LOG_FLAG_LIMIT, "Error returned from libav: %s", av_error(av_ret));
av_frame_free(&frame);
return -1;
}
static int __decoder_input_data(decoder_t *dec, const str *data, unsigned long ts, int *ptime,
int (*callback)(decoder_t *, AVFrame *, void *u1, void *u2), void *u1, void *u2)
{
GQueue frames = G_QUEUE_INIT;
if (G_UNLIKELY(!dec))
return -1;
if (!data && (!dec->dtx.do_dtx || !ptime))
return 0;
ts = fraction_mult(ts, &dec->clockrate_fact);
cdbg("%p dec pts %llu rtp_ts %llu incoming ts %lu", dec, (unsigned long long) dec->pts,
(unsigned long long) dec->rtp_ts, (unsigned long) ts);
if (G_UNLIKELY(dec->rtp_ts == (unsigned long) -1L)) {
// initialize pts
dec->pts = 0;
}
else {
// shift pts according to rtp ts shift
uint64_t shift_ts = ts - dec->rtp_ts;
if ((shift_ts * 1000) / dec->in_format.clockrate > PACKET_TS_RESET_THRES) {
ilog(LOG_DEBUG, "Timestamp discontinuity detected, resetting timestamp from "
"%lu to %lu",
dec->rtp_ts, ts);
// XXX handle lost packets here if timestamps don't line up?
}
else
dec->pts += shift_ts;
}
dec->rtp_ts = ts;
if (data)
dec->def->codec_type->decoder_input(dec, data, &frames);
else
dec->dtx.do_dtx(dec, &frames, *ptime);
AVFrame *frame;
int ret = 0;
unsigned long samples = 0;
while ((frame = g_queue_pop_head(&frames))) {
samples += frame->nb_samples;
dec->dec_out_format.format = frame->format;
AVFrame *rsmp_frame = resample_frame(&dec->resampler, frame, &dec->dest_format);
if (!rsmp_frame) {
ilog(LOG_ERR | LOG_FLAG_LIMIT, "Resampling failed");
ret = -1;
}
else {
if (callback(dec, rsmp_frame, u1, u2))
ret = -1;
}
if (rsmp_frame != frame)
av_frame_free(&frame);
}
if (ptime)
*ptime = samples * 1000L / dec->in_format.clockrate;
return ret;
}
int decoder_input_data(decoder_t *dec, const str *data, unsigned long ts,
int (*callback)(decoder_t *, AVFrame *, void *u1, void *u2), void *u1, void *u2)
{
if (!data || !data->s || !data->len)
return 0;
return __decoder_input_data(dec, data, ts, NULL, callback, u1, u2);
}
int decoder_input_data_ptime(decoder_t *dec, const str *data, unsigned long ts, int *ptime,
int (*callback)(decoder_t *, AVFrame *, void *u1, void *u2), void *u1, void *u2)
{
if (!data || !data->s || !data->len)
return 0;
return __decoder_input_data(dec, data, ts, ptime, callback, u1, u2);
}
int decoder_dtx(decoder_t *dec, unsigned long ts, int ptime,
int (*callback)(decoder_t *, AVFrame *, void *u1, void *u2), void *u1, void *u2)
{
return __decoder_input_data(dec, NULL, ts, &ptime, callback, u1, u2);
}
static void avlog_ilog(void *ptr, int loglevel, const char *fmt, va_list ap) {
char *msg;
if (vasprintf(&msg, fmt, ap) <= 0)
ilogs(ffmpeg, LOG_ERR | LOG_FLAG_LIMIT, "av_log message dropped");
else {
#ifdef AV_LOG_PANIC
// translate AV_LOG_ constants to LOG_ levels
if (loglevel >= AV_LOG_VERBOSE)
loglevel = LOG_DEBUG;
else if (loglevel >= AV_LOG_INFO)
loglevel = LOG_NOTICE;
else if (loglevel >= AV_LOG_WARNING)
loglevel = LOG_WARNING;
else if (loglevel >= AV_LOG_ERROR)
loglevel = LOG_ERROR;
else if (loglevel >= AV_LOG_FATAL)
loglevel = LOG_CRIT;
else
loglevel = LOG_ALERT;
#else
// defuse avlog log levels to be either DEBUG or ERR
if (loglevel <= LOG_ERR)
loglevel = LOG_ERR;
else
loglevel = LOG_DEBUG;
#endif
ilogs(ffmpeg, loglevel | LOG_FLAG_LIMIT, "av_log: %s", msg);
free(msg);
}
}
void avc_def_init(struct codec_def_s *def) {
// look up AVCodec structs
if (def->avcodec_name_enc)
def->encoder = avcodec_find_encoder_by_name(def->avcodec_name_enc);
if (def->avcodec_name_dec)
def->decoder = avcodec_find_decoder_by_name(def->avcodec_name_dec);
if (def->avcodec_id >= 0) {
if (!def->encoder)
def->encoder = avcodec_find_encoder(def->avcodec_id);
if (!def->decoder)
def->decoder = avcodec_find_decoder(def->avcodec_id);
}
// check if we have support if we are supposed to
if (def->avcodec_name_enc || def->avcodec_id >= 0) {
if (def->encoder)
def->support_encoding = 1;
}
if (def->avcodec_name_dec || def->avcodec_id >= 0) {
if (def->decoder)
def->support_decoding = 1;
}
}
static void cc_cleanup(void);
void codeclib_free(void) {
t_hash_table_destroy(codecs_by_name_ht);
t_hash_table_destroy(generic_ffmpeg_codecs);
avformat_network_deinit();
cc_cleanup();
if (cc_lib_handle)
dlclose(cc_lib_handle);
}
bool rtpe_has_cpu_flag(enum rtpe_cpu_flag flag) {
static bool done = false;
static bool cpu_flags[__NUM_RTPE_CPU_FLAGS] = {false,};
if (!done) {
#if defined(__x86_64__)
int32_t ebx_7h0h, edx_1h;
__asm (
"mov $1, %%eax" "\n\t"
"cpuid" "\n\t"
"mov %%edx, %1" "\n\t"
"mov $7, %%eax" "\n\t"
"xor %%ecx, %%ecx" "\n\t"
"cpuid" "\n\t"
"mov %%ebx, %0" "\n\t"
: "=rm" (ebx_7h0h), "=rm" (edx_1h)
:
: "eax", "ebx", "ecx", "edx"
);
cpu_flags[RTPE_CPU_FLAG_SSE2] = !!(edx_1h & (1L << 26));
cpu_flags[RTPE_CPU_FLAG_AVX2] = !!(ebx_7h0h & (1L << 5));
cpu_flags[RTPE_CPU_FLAG_AVX512BW] = !!(ebx_7h0h & (1L << 30));
cpu_flags[RTPE_CPU_FLAG_AVX512F] = !!(ebx_7h0h & (1L << 16));
#endif
done = true;
}
if (flag < 0 || flag >= __NUM_RTPE_CPU_FLAGS)
abort();
return cpu_flags[flag];
}
void *dlsym_assert(void *handle, const char *sym, const char *fn) {
void *ret = dlsym(handle, sym);
if (!ret)
die("Failed to resolve symbol '%s' from '%s': %s", sym, fn, dlerror());
return ret;
}
#ifdef HAVE_CODEC_CHAIN
static void cc_dlsym_resolve(const char *fn) {
cc_client_connect = dlsym_assert(cc_lib_handle, "codec_chain_client_connect", fn);
cc_set_thread_funcs = dlsym_assert(cc_lib_handle, "codec_chain_set_thread_funcs", fn);
cc_get = dlsym_assert(cc_lib_handle, "codec_chain_get", fn);
cc_client_runner_new = dlsym_assert(cc_lib_handle,
"codec_chain_client_runner_new", fn);
cc_client_runner_free = dlsym_assert(cc_lib_handle,
"codec_chain_client_runner_free", fn);
cc_client_async_runner_new = dlsym_assert(cc_lib_handle,
"codec_chain_client_async_runner_new", fn);
cc_client_async_runner_free = dlsym_assert(cc_lib_handle,
"codec_chain_client_async_runner_free", fn);
cc_runner_do = dlsym_assert(cc_lib_handle,
"codec_chain_runner_do", fn);
if (!rtpe_common_config_ptr->codec_chain_nonblock) {
cc_async_runner_do = dlsym_assert(cc_lib_handle,
"codec_chain_async_runner_do", fn);
}
else {
__typeof__(codec_chain_async_runner_do_nonblock) *nb = dlsym_assert(cc_lib_handle,
"codec_chain_async_runner_do_nonblock", fn);
cc_async_runner_do = nb;
}
cc_client_codec_new = dlsym_assert(cc_lib_handle,
"codec_chain_client_codec_new", fn);
cc_client_codec_free = dlsym_assert(cc_lib_handle,
"codec_chain_client_codec_free", fn);
cc_defs = dlsym_assert(cc_lib_handle,
"codec_chain_defs", fn);
}
static codec_cc_t *codec_cc_new_dummy(codec_def_t *src, format_t *src_format, codec_def_t *dst,
format_t *dst_format, int bitrate, int ptime,
void *(*async_init)(void *, void *, void *),
void (*async_callback)(AVPacket *, void *))
{
return NULL;
}
static void cc_init(void) {
codec_cc_new = codec_cc_new_dummy;
if (!rtpe_common_config_ptr->codec_chain_lib_path)
return;
cc_lib_handle = dlopen(rtpe_common_config_ptr->codec_chain_lib_path, RTLD_NOW | RTLD_LOCAL);
if (!cc_lib_handle)
die("Failed to load libcodec-chain.so '%s': %s",
rtpe_common_config_ptr->codec_chain_lib_path,
dlerror());
cc_dlsym_resolve(rtpe_common_config_ptr->codec_chain_lib_path);
cc_set_thread_funcs(codeclib_thread_init, codeclib_thread_cleanup, codeclib_thread_loop);
cc_client = cc_client_connect(4);
if (!cc_client)
die("Failed to connect to cudecsd");
if (!rtpe_common_config_ptr->codec_chain_async)
codec_cc_new = codec_cc_new_sync;
else
codec_cc_new = codec_cc_new_async;
ilog(LOG_DEBUG, "CUDA codecs initialised");
}
void cc_init_chain(codec_def_t *src, format_t *src_format, codec_def_t *dst,
format_t *dst_format)
{
if (!cc_get) {
ilog(LOG_WARN, "No codec-chain support loaded");
return;
}
codec_chain_id id = cc_get(
(codec_chain_params) {
.name = src->rtpname,
.clock_rate = src_format->clockrate,
.channels = src_format->channels,
.ptime = 20, // XXX
},
(codec_chain_params) {
.name = dst->rtpname,
.clock_rate = dst_format->clockrate,
.channels = dst_format->channels,
.ptime = 20, // XXX
}
);
if (id == 0) {
ilog(LOG_WARN, "Codec chain %s -> %s not supported by library",
src->rtpname, dst->rtpname);
return;
}
if (id >= CODEC_CHAIN_ID_MAX) {
ilog(LOG_WARN, "Codec chain %s -> %s requires rebuild",
src->rtpname, dst->rtpname);
return;
}
if (rtpe_common_config_ptr->codec_chain_async) {
if (cc_runners[id].async)
return;
cc_runners[id].async = cc_client_async_runner_new(cc_client, id,
rtpe_common_config_ptr->codec_chain_async,
rtpe_common_config_ptr->codec_chain_interval,
rtpe_common_config_ptr->codec_chain_runners,
rtpe_common_config_ptr->codec_chain_concurrency);
if (cc_runners[id].async)
ilog(LOG_DEBUG, "Created async chain runner for %s", cc_defs[id].name);
else
ilog(LOG_WARN, "Failed to create async chain runner for %s", cc_defs[id].name);
}
else {
if (cc_runners[id].sync)
return;
cc_runners[id].sync = cc_client_runner_new(cc_client, id,
rtpe_common_config_ptr->codec_chain_interval,
rtpe_common_config_ptr->codec_chain_runners,
rtpe_common_config_ptr->codec_chain_concurrency);
if (cc_runners[id].sync)
ilog(LOG_DEBUG, "Created chain runner for %s", cc_defs[id].name);
else
ilog(LOG_WARN, "Failed to create chain runner for %s", cc_defs[id].name);
}
}
static void cc_cleanup(void) {
if (!cc_lib_handle)
return;
for (codec_chain_id id = 1; id < CODEC_CHAIN_ID_MAX; id++) {
if (!rtpe_common_config_ptr->codec_chain_async)
cc_client_runner_free(cc_client, &cc_runners[id].sync);
else
cc_client_async_runner_free(cc_client, &cc_runners[id].async);
}
}
#else
static void cc_init(void) { }
static void cc_cleanup(void) { }
#endif
void codeclib_register_codec(const codec_def_t *c) {
struct codec_def_s *n = realloc(__codec_defs, sizeof(*__codec_defs) * (__num_codec_defs + 1));
if (!n) {
fprintf(stderr, "Out of memory initialising codecs\n");
abort();
}
__codec_defs = n;
memcpy(&__codec_defs[__num_codec_defs], c, sizeof(*c));
__num_codec_defs++;
}
void codeclib_init(int print) {
#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(58, 9, 100)
av_register_all();
avcodec_register_all();
avfilter_register_all();
#endif
avformat_network_init();
av_log_set_callback(avlog_ilog);
codecs_by_name_ht = codecs_by_name_new();
generic_ffmpeg_codecs = codecs_by_id_alloc_new();
cc_init();
for (unsigned int i = 0; i < __num_codec_defs; i++) {
// add to hash table
struct codec_def_s *def = &__codec_defs[i];
def->rtpname_str = STR(def->rtpname);
assert(t_hash_table_lookup(codecs_by_name_ht, &def->rtpname_str) == NULL);
t_hash_table_insert(codecs_by_name_ht, &def->rtpname_str, def);
// init undefined member vars
if (!def->default_clockrate_fact.mult)
def->default_clockrate_fact.mult = 1;
if (!def->default_clockrate_fact.div)
def->default_clockrate_fact.div = 1;
if (!def->default_ptime)
def->default_ptime = -1;
if (!def->default_clockrate)
def->default_clockrate = -1;
if (!def->default_channels)
def->default_channels = -1;
// init RFC-related info
const struct rtp_payload_type *pt = rtp_get_rfc_codec(&def->rtpname_str);
if (pt)
def->rfc_payload_type = pt->payload_type;
else {
// special case:
if (!strcmp(def->rtpname, "G729a"))
def->rfc_payload_type = 18;
else
def->rfc_payload_type = -1;
}
if (def->codec_type && def->codec_type->def_init)
def->codec_type->def_init(def);
if (!strcmp(def->rtpname, "CN"))
codec_def_cn = def;
if (def->avcodec_id == AV_CODEC_ID_PCM_S16LE)
codec_def_pcm16 = def;
if (print) {
if (def->support_encoding && def->support_decoding) {
if (def->default_channels > 0 && def->default_clockrate >= 0)
printf("%20s: fully supported\n", def->rtpname);
else
printf("%20s: codec supported but lacks RTP definition\n", def->rtpname);
}
else if (def->support_decoding)
printf("%20s: supported for decoding only\n", def->rtpname);
else if (def->support_encoding)
printf("%20s: supported for encoding only\n", def->rtpname);
else
printf("%20s: not supported\n", def->rtpname);
}
else {
if (!def->support_encoding && !def->support_decoding)
ilog(LOG_DEBUG, "Codec %s is not supported by codec library",
def->rtpname);
else if (!def->support_encoding) {
ilog(LOG_DEBUG, "Codec %s is only supported for decoding "
"by codec library", def->rtpname);
}
else if (!def->support_decoding)
ilog(LOG_DEBUG, "Codec %s is only supported for encoding "
"by codec library", def->rtpname);
}
if (def->supplemental)
g_queue_push_tail(&__supplemental_codecs, def);
if (rtpe_common_config_ptr->mos_type) {
def->mos_type = rtpe_common_config_ptr->mos_type;
if (def->mos_type == MOS_FB && def->default_clockrate != 48000)
def->mos_type = MOS_NB;
}
}
}
static int ptr_cmp(const void *a, const void *b, void *dummy) {
if (a < b)
return -1;
if (a > b)
return 1;
return 0;
}
void __packet_sequencer_init(packet_sequencer_t *ps, GDestroyNotify ffunc) {
ps->packets = g_tree_new_full(ptr_cmp, NULL, NULL, ffunc);
ps->seq = -1;
}
void packet_sequencer_destroy(packet_sequencer_t *ps) {
if (ps->packets)
g_tree_destroy(ps->packets);
ps->packets = NULL;
}
struct tree_searcher {
int find_seq,
found_seq;
};
static int packet_tree_search(const void *testseq_p, const void *ts_p) {
struct tree_searcher *ts = (void *) ts_p;
int testseq = GPOINTER_TO_INT(testseq_p);
// called as a binary search test function. we're looking for the lowest
// seq number that is higher than find_seq. if our test number is too low,
// we proceed with higher numbers. if it's too high, we proceed to the lower
// numbers, but remember the lowest we've seen along that path.
if (G_UNLIKELY(testseq == ts->find_seq)) {
// we've struck gold
ts->found_seq = testseq;
return 0;
}
if (testseq < ts->find_seq)
return 1;
// testseq > ts->find_seq
if (ts->found_seq == -1 || testseq < ts->found_seq)
ts->found_seq = testseq;
return -1;
}
// caller must take care of locking
static void *__packet_sequencer_next_packet(packet_sequencer_t *ps, int num_wait) {
// see if we have a packet with the correct seq nr in the queue
seq_packet_t *packet = g_tree_lookup(ps->packets, GINT_TO_POINTER(ps->seq));
if (G_LIKELY(packet != NULL)) {
cdbg("returning in-sequence packet (seq %i)", ps->seq);
goto out;
}
// why not? do we have anything? (we should)
int nnodes = g_tree_nnodes(ps->packets);
if (G_UNLIKELY(nnodes == 0)) {
cdbg("packet queue empty");
return NULL;
}
if (G_LIKELY(nnodes < num_wait)) {
cdbg("only %i packets in queue - waiting for more", nnodes);
return NULL; // need to wait for more
}
// packet was probably lost. search for the next highest seq
struct tree_searcher ts = { .find_seq = ps->seq + 1, .found_seq = -1 };
packet = g_tree_search(ps->packets, packet_tree_search, &ts);
if (packet) {
// bullseye
cdbg("lost packet - returning packet with next seq %i", packet->seq);
goto out;
}
if (G_UNLIKELY(ts.found_seq == -1)) {
// didn't find anything. seq must have wrapped around. retry
// starting from zero
ts.find_seq = 0;
packet = g_tree_search(ps->packets, packet_tree_search, &ts);
if (packet) {
cdbg("lost packet - returning packet with next seq %i (after wrap)", packet->seq);
goto out;
}
if (G_UNLIKELY(ts.found_seq == -1))
abort();
}
// pull out the packet we found
packet = g_tree_lookup(ps->packets, GINT_TO_POINTER(ts.found_seq));
if (G_UNLIKELY(packet == NULL))
abort();
cdbg("lost multiple packets - returning packet with next highest seq %i", packet->seq);
out:
;
uint16_t l = packet->seq - ps->seq;
ps->lost_count += l;
g_tree_steal(ps->packets, GINT_TO_POINTER(packet->seq));
ps->seq = (packet->seq + 1) & 0xffff;
unsigned int ext_seq = ps->roc << 16 | packet->seq;
while (ext_seq < ps->ext_seq) {
ps->roc++;
ext_seq += 0x10000;
}
ps->ext_seq = ext_seq;
return packet;
}
void *packet_sequencer_next_packet(packet_sequencer_t *ps) {
return __packet_sequencer_next_packet(ps, 10); // arbitrary value
}
void *packet_sequencer_force_next_packet(packet_sequencer_t *ps) {
return __packet_sequencer_next_packet(ps, 0);
}
int packet_sequencer_next_ok(packet_sequencer_t *ps) {
if (g_tree_lookup(ps->packets, GINT_TO_POINTER(ps->seq)))
return 1;
return 0;
}
int packet_sequencer_insert(packet_sequencer_t *ps, seq_packet_t *p) {
int ret = 0;
// check seq for dupes
if (G_UNLIKELY(ps->seq == -1)) {
// first packet we see
ps->seq = p->seq;
goto seq_ok;
}
int diff = p->seq - ps->seq;
// early packet: p->seq = 200, ps->seq = 150, diff = 50
if (G_LIKELY(diff >= 0 && diff < PACKET_SEQ_DUPE_THRES))
goto seq_ok;
// early packet with wrap-around: p->seq = 20, ps->seq = 65530, diff = -65510
if (diff < (-0xffff + PACKET_SEQ_DUPE_THRES))
goto seq_ok;
// recent duplicate: p->seq = 1000, ps->seq = 1080, diff = -80
if (diff < 0 && diff > -PACKET_SEQ_DUPE_THRES)
return -1;
// recent duplicate after wrap-around: p->seq = 65530, ps->seq = 30, diff = 65500
if (diff > (0xffff - PACKET_SEQ_DUPE_THRES))
return -1;
// everything else we consider a seq reset
ilog(LOG_DEBUG, "Seq reset detected: expected seq %i, received seq %i", ps->seq, p->seq);
ps->seq = p->seq;
ret = 1;
// seq ok - fall through
g_tree_clear(ps->packets);
seq_ok:
if (g_tree_lookup(ps->packets, GINT_TO_POINTER(p->seq)))
return -1;
ret = g_tree_nnodes(ps->packets) == 0 ? ret : 2; // indicates an out-of-order packet
g_tree_insert(ps->packets, GINT_TO_POINTER(p->seq), p);
return ret;
}
encoder_t *encoder_new(void) {
encoder_t *ret = g_new0(__typeof(*ret), 1);
format_init(&ret->requested_format);
format_init(&ret->actual_format);
ret->avpkt = av_packet_alloc();
return ret;
}
const char *avc_encoder_init(encoder_t *enc, const str *extra_opts) {
enc->avc.codec = enc->def->encoder;
if (!enc->avc.codec)
return "output codec not found";
enc->avc.avcctx = avcodec_alloc_context3(enc->avc.codec);
if (!enc->avc.avcctx)
return "failed to alloc codec context";
enc->actual_format = enc->requested_format;
#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(61, 19, 0)
avcodec_get_supported_config(enc->avc.avcctx, enc->avc.codec, AV_CODEC_CONFIG_SAMPLE_FORMAT, 0, (const void **) &enc->avc.sample_fmts, NULL);
#else
enc->avc.sample_fmts = enc->avc.codec->sample_fmts;
#endif
enc->actual_format.format = -1;
for (const enum AVSampleFormat *sfmt = enc->avc.sample_fmts; sfmt && *sfmt != -1; sfmt++) {
cdbg("supported sample format for output codec %s: %s",
enc->avc.codec->name, av_get_sample_fmt_name(*sfmt));
if (*sfmt == enc->requested_format.format)
enc->actual_format.format = *sfmt;
}
if (enc->actual_format.format == -1 && enc->avc.sample_fmts)
enc->actual_format.format = enc->avc.sample_fmts[0];
cdbg("using output sample format %s for codec %s",
av_get_sample_fmt_name(enc->actual_format.format), enc->avc.codec->name);
SET_CHANNELS(enc->avc.avcctx, enc->actual_format.channels);
DEF_CH_LAYOUT(&enc->avc.avcctx->CH_LAYOUT, enc->actual_format.channels);
enc->avc.avcctx->sample_rate = enc->actual_format.clockrate;
enc->avc.avcctx->sample_fmt = enc->actual_format.format;
enc->avc.avcctx->time_base = (AVRational){1,enc->actual_format.clockrate};
enc->avc.avcctx->bit_rate = enc->bitrate;
if (enc->def->set_enc_options)
enc->def->set_enc_options(enc, extra_opts);
int i = avcodec_open2(enc->avc.avcctx, enc->avc.codec, NULL);
if (i) {
ilog(LOG_ERR | LOG_FLAG_LIMIT, "Error returned from libav: %s", av_error(i));
return "failed to open output context";
}
if (enc->avc.avcctx->frame_size)
enc->samples_per_frame = enc->avc.avcctx->frame_size;
else
enc->samples_per_frame = enc->actual_format.clockrate * enc->ptime / 1000;
enc->samples_per_packet = enc->samples_per_frame;
return NULL;
}
int encoder_config(encoder_t *enc, codec_def_t *def, int bitrate, int ptime,
const format_t *requested_format, format_t *actual_format)
{
return encoder_config_fmtp(enc, def, bitrate, ptime, NULL, requested_format, actual_format,
NULL, NULL, NULL);
}
int encoder_config_fmtp(encoder_t *enc, codec_def_t *def, int bitrate, int ptime,
const format_t *input_format,
const format_t *requested_format_p, format_t *actual_format,
struct rtp_codec_format *fmtp, const str *fmtp_string,
const str *extra_opts)
{
const char *err;
err = "codec not supported";
if (!def->codec_type)
goto err;
err = "failed to parse \"fmtp\"";
if (!codec_parse_fmtp(def, fmtp, fmtp_string, &enc->format_options))
goto err;
// select encoder format
format_t requested_format = *requested_format_p;
enc->clockrate_fact = def->default_clockrate_fact;
if (def->select_encoder_format)
def->select_encoder_format(enc, &requested_format, input_format, fmtp);
requested_format.clockrate = fraction_mult(requested_format.clockrate, &enc->clockrate_fact);
// anything to do?
if (G_LIKELY(format_eq(&requested_format, &enc->requested_format))) {
if (!input_format)
goto done;
if (G_LIKELY(format_eq(input_format, &enc->input_format)))
goto done;
}
encoder_close(enc);
if (ptime <= 0)
ptime = 20;
if (def->minimum_ptime && ptime < def->minimum_ptime)
ptime = def->minimum_ptime;
enc->requested_format = requested_format;
if (input_format)
enc->input_format = *input_format;
else
format_init(&enc->input_format);
enc->def = def;
enc->ptime = ptime;
enc->bitrate = bitrate;
err = def->codec_type->encoder_init ? def->codec_type->encoder_init(enc, extra_opts) : 0;
if (err)
goto err;
// output frame and fifo
enc->frame = av_frame_alloc();
if (enc->actual_format.format != -1 && enc->actual_format.clockrate > 0) {
enc->frame->nb_samples = enc->samples_per_frame ? : 256;
enc->frame->format = enc->actual_format.format;
enc->frame->sample_rate = enc->actual_format.clockrate;
DEF_CH_LAYOUT(&enc->frame->CH_LAYOUT, enc->actual_format.channels);
if (av_frame_get_buffer(enc->frame, 0) < 0)
abort();
enc->fifo = av_audio_fifo_alloc(enc->frame->format, enc->actual_format.channels,
enc->frame->nb_samples);
ilog(LOG_DEBUG, "Initialized encoder with frame size %u samples", enc->frame->nb_samples);
}
else
ilog(LOG_DEBUG, "Initialized encoder without frame buffer");
done:
if (actual_format)
*actual_format = enc->actual_format;
return 0;
err:
encoder_close(enc);
ilog(LOG_ERR, "Error configuring media output for codec %s: %s", def->rtpname, err);
return -1;
}
void avc_encoder_close(encoder_t *enc) {
if (enc->avc.avcctx) {
#if LIBAVCODEC_VERSION_INT < AV_VERSION_INT(61, 0, 0)
avcodec_close(enc->avc.avcctx);
#endif
avcodec_free_context(&enc->avc.avcctx);
}
enc->avc.avcctx = NULL;
enc->avc.codec = NULL;
}
void encoder_close(encoder_t *enc) {
if (!enc)
return;
if (enc->def && enc->def->codec_type && enc->def->codec_type->encoder_close)
enc->def->codec_type->encoder_close(enc);
format_init(&enc->requested_format);
format_init(&enc->actual_format);
av_audio_fifo_free(enc->fifo);
av_frame_free(&enc->frame);
enc->mux_dts = 0;
enc->fifo = NULL;
enc->fifo_pts = 0;
}
void encoder_free(encoder_t *enc) {
encoder_close(enc);
av_packet_free(&enc->avpkt);
resample_shutdown(&enc->resampler);
g_free(enc);
}
int avc_encoder_input(encoder_t *enc, AVFrame **frame) {
int keep_going = 0;
int got_packet = 0;
int av_ret = 0;
if (!enc->avc.avcctx)
return -1;
#if LIBAVCODEC_VERSION_INT >= AV_VERSION_INT(57, 36, 0)
if (*frame) {
av_ret = avcodec_send_frame(enc->avc.avcctx, *frame);
cdbg("send frame ret %i", av_ret);
if (av_ret == 0) {
// consumed
*frame = NULL;
keep_going = 1;
}
else {
if (av_ret == AVERROR(EAGAIN))
; // check output and maybe try again
else
goto err;
}
}
av_ret = avcodec_receive_packet(enc->avc.avcctx, enc->avpkt);
cdbg("receive packet ret %i", av_ret);
if (av_ret == 0) {
// got some data
keep_going = 1;
got_packet = 1;
}
else {
if (av_ret == AVERROR(EAGAIN))
; // try again if there's still more input
else
goto err;
}
#else
if (!*frame)
return 0;
av_ret = avcodec_encode_audio2(enc->avc.avcctx, enc->avpkt, *frame, &got_packet);
cdbg("encode frame ret %i, got packet %i", av_ret, got_packet);
if (av_ret == 0)
*frame = NULL; // consumed
else
goto err;
if (got_packet)
keep_going = 1;
#endif
if (!got_packet)
return keep_going;
cdbg("output avpkt size is %i", (int) enc->avpkt->size);
cdbg("output pkt pts/dts is %li/%li", (long) enc->avpkt->pts,
(long) enc->avpkt->dts);
// the encoder may return frames with the same dts multiple consecutive times.
// the muxer may not like this, so ensure monotonically increasing dts.
if (enc->mux_dts > enc->avpkt->dts)
enc->avpkt->dts = enc->mux_dts;
if (enc->avpkt->pts < enc->avpkt->dts)
enc->avpkt->pts = enc->avpkt->dts;
return keep_going;
err:
if (av_ret)
ilog(LOG_ERR | LOG_FLAG_LIMIT, "Error returned from libav: %s", av_error(av_ret));
return -1;
}
int encoder_input_data(encoder_t *enc, AVFrame *frame,
int (*callback)(encoder_t *, void *u1, void *u2), void *u1, void *u2)
{
enc->avpkt->size = 0;
while (1) {
if (!enc->def || !enc->def->codec_type)
break;
if (!enc->def->codec_type->encoder_input)
break;
int ret = enc->def->codec_type->encoder_input(enc, &frame);
if (ret < 0)
return -1;
if (enc->avpkt->size) {
// don't rely on the encoder producing steady timestamps,
// instead keep track of them ourselves based on the returned
// frame duration
enc->avpkt->pts = enc->next_pts;
if (enc->def->codec_type->encoder_got_packet)
enc->def->codec_type->encoder_got_packet(enc);
callback(enc, u1, u2);
enc->next_pts += enc->avpkt->duration;
enc->mux_dts = enc->avpkt->dts + 1; // min next expected dts
av_packet_unref(enc->avpkt);
enc->avpkt->size = 0;
}
if (ret == 0)
break;
}
return 0;
}
static int encoder_fifo_flush(encoder_t *enc,
int (*callback)(encoder_t *, void *u1, void *u2), void *u1, void *u2)
{
while (av_audio_fifo_size(enc->fifo) >= enc->frame->nb_samples) {
if (av_audio_fifo_read(enc->fifo, (void **) enc->frame->data,
enc->frame->nb_samples) <= 0)
abort();
cdbg("output fifo pts %lu",(unsigned long) enc->fifo_pts);
enc->frame->pts = enc->fifo_pts;
encoder_input_data(enc, enc->frame, callback, u1, u2);
enc->fifo_pts += enc->frame->nb_samples;
}
return 0;
}
int encoder_input_fifo(encoder_t *enc, AVFrame *frame,
int (*callback)(encoder_t *, void *u1, void *u2), void *u1, void *u2)
{
AVFrame *rsmp_frame = resample_frame(&enc->resampler, frame, &enc->actual_format);
if (!rsmp_frame) {
ilog(LOG_ERR | LOG_FLAG_LIMIT, "Resampling failed");
return -1;
}
if (av_audio_fifo_write(enc->fifo, (void **) rsmp_frame->extended_data, rsmp_frame->nb_samples) < 0)
return -1;
if (rsmp_frame != frame)
av_frame_free(&rsmp_frame);
return encoder_fifo_flush(enc, callback, u1, u2);
}
int packetizer_passthrough(AVPacket *pkt, GString *buf, str *output, size_t num_bytes, encoder_t *enc,
int64_t *__restrict pts, int64_t *__restrict duration)
{
if (!pkt)
return -1;
if (output->len < pkt->size) {
ilog(LOG_WARN | LOG_FLAG_LIMIT, "Output packet size too small (%zu < %i)",
output->len, pkt->size);
return -1;
}
output->len = pkt->size;
memcpy(output->s, pkt->data, pkt->size);
*pts = pkt->pts;
*duration = pkt->duration;
return 0;
}
// returns: -1 = not enough data, nothing returned; 0 = returned a packet;
// 1 = returned a packet and there's more
int packetizer_samplestream(AVPacket *pkt, GString *buf, str *input_output, size_t num_bytes,
encoder_t *enc, int64_t *__restrict pts, int64_t *__restrict duration)
{
// avoid moving buffers around if possible:
// most common case: new input packet has just enough (or more) data as what we need
if (G_LIKELY(pkt && buf->len == 0 && pkt->size >= num_bytes)) {
*pts = pkt->pts;
*duration = pkt->duration;
memcpy(input_output->s, pkt->data, num_bytes);
// any leftovers?
if (pkt->size > num_bytes) {
g_string_append_len(buf, (char *) pkt->data + num_bytes,
pkt->size - num_bytes);
*duration = fraction_mult(num_bytes * 8, &enc->clockrate_fact)
/ enc->def->bits_per_sample;
enc->packet_pts = pkt->pts + *duration;
}
input_output->len = num_bytes;
return buf->len >= num_bytes ? 1 : 0;
}
// we have to move data around. append input packet to buffer if we have one
if (pkt)
g_string_append_len(buf, (char *) pkt->data, pkt->size);
// do we have enough?
if (buf->len < num_bytes)
return -1;
// copy requested data into provided output buffer and remove from interim buffer
memcpy(input_output->s, buf->str, num_bytes);
input_output->len = num_bytes;
g_string_erase(buf, 0, num_bytes);
// adjust output pts
*pts = enc->packet_pts;
*duration = fraction_mult(num_bytes * 8, &enc->clockrate_fact) / enc->def->bits_per_sample;
enc->packet_pts += *duration;
return buf->len >= num_bytes ? 1 : 0;
}
int codeclib_set_av_opt_int(encoder_t *enc, const char *opt, int64_t val) {
ilog(LOG_DEBUG, "Setting ffmpeg '%s' option for '%s' to %" PRId64,
opt, enc->def->rtpname, val);
int ret = av_opt_set_int(enc->avc.avcctx, opt, val, AV_OPT_SEARCH_CHILDREN);
if (!ret)
return 0;
ilog(LOG_WARN, "Failed to set ffmpeg '%s' option for codec '%s' to %" PRId64 ": %s",
opt, enc->def->rtpname, val, av_error(ret));
return -1;
}
void codeclib_key_value_parse(const str *instr, bool need_value,
void (*cb)(str *key, str *value, void *data), void *data)
{
if (!instr || !instr->s)
return;
// semicolon-separated key=value
str s = *instr;
str key, value;
while (str_token_sep(&value, &s, ';')) {
if (!str_token(&key, &value, '=')) {
if (need_value)
continue;
value = STR_NULL;
}
// truncate whitespace
while (key.len && key.s[0] == ' ')
str_shift(&key, 1);
while (key.len && key.s[key.len - 1] == ' ')
key.len--;
while (value.len && value.s[0] == ' ')
str_shift(&value, 1);
while (value.len && value.s[value.len - 1] == ' ')
value.len--;
if (key.len == 0)
continue;
cb(&key, &value, data);
}
}
static int generic_silence_dtx(decoder_t *dec, GQueue *out, int ptime) {
if (dec->dec_out_format.format == -1)
return -1;
if (!dec->avc.avpkt)
return -1;
if (ptime <= 0)
ptime = 20;
int num_samples = ptime * dec->in_format.clockrate / 1000;
ilog(LOG_DEBUG, "pushing %i silence samples into %s decoder", num_samples, dec->def->rtpname);
// create dummy frame, fill with silence, pretend it was returned from the decoder
AVFrame *frame = av_frame_alloc();
frame->nb_samples = num_samples;
frame->format = dec->dec_out_format.format;
frame->sample_rate = dec->dec_out_format.clockrate;
DEF_CH_LAYOUT(&frame->CH_LAYOUT, dec->dec_out_format.channels);
if (av_frame_get_buffer(frame, 0) < 0) {
av_frame_free(&frame);
return -1;
}
memset(frame->extended_data[0], 0, frame->linesize[0]);
// advance PTS
frame->pts = dec->avc.avpkt->pts;
dec->avc.avpkt->pts += frame->nb_samples;
g_queue_push_tail(out, frame);
return 0;
}
static int cn_append_frame(decoder_t *dec, AVFrame *f, void *u1, void *u2) {
GQueue *out = u1;
g_queue_push_tail(out, f);
return 0;
}
static int generic_cn_dtx(decoder_t *dec, GQueue *out, int ptime) {
dec->dtx.cn.cn_dec->ptime = ptime;
return decoder_input_data(dec->dtx.cn.cn_dec, dec->dtx.cn.cn_payload,
dec->rtp_ts, cn_append_frame, out, NULL);
}
static int generic_cn_dtx_init(decoder_t *dec) {
// upsample CN output to same params as output of parent codec
format_t cn_format = dec->dest_format;
cn_format.channels = dec->in_format.channels;
cn_format.clockrate = dec->in_format.clockrate;
dec->dtx.cn.cn_dec = decoder_new_fmt(codec_def_cn, 8000, 1, dec->ptime, &cn_format);
return 0;
}
static void generic_cn_dtx_cleanup(decoder_t *dec) {
decoder_close(dec->dtx.cn.cn_dec);
}
int format_cmp_ignore(const struct rtp_payload_type *a, const struct rtp_payload_type *b) {
return 0;
}
void frame_fill_tone_samples(enum AVSampleFormat fmt, void *samples, unsigned int offset, unsigned int num,
unsigned int freq, unsigned int volume, unsigned int sample_rate, unsigned int channels)
{
switch (fmt) {
case AV_SAMPLE_FMT_S16:
tone_samples_int16_t(samples, offset, num, freq, volume, sample_rate, channels);
break;
case AV_SAMPLE_FMT_S32:
tone_samples_int32_t(samples, offset, num, freq, volume, sample_rate, channels);
break;
case AV_SAMPLE_FMT_DBL:
tone_samples_double(samples, offset, num, freq, volume, sample_rate, channels);
break;
case AV_SAMPLE_FMT_FLT:
tone_samples_float(samples, offset, num, freq, volume, sample_rate, channels);
break;
default:
ilog(LOG_ERR | LOG_FLAG_LIMIT, "Unsupported sample format %u", fmt);
break;
}
}
void frame_fill_dtmf_samples(enum AVSampleFormat fmt, void *samples, unsigned int offset, unsigned int num,
unsigned int event, unsigned int volume, unsigned int sample_rate, unsigned int channels)
{
switch (fmt) {
case AV_SAMPLE_FMT_S16:
dtmf_samples_int16_t(samples, offset, num, event, volume, sample_rate, channels);
break;
case AV_SAMPLE_FMT_S32:
dtmf_samples_int32_t(samples, offset, num, event, volume, sample_rate, channels);
break;
case AV_SAMPLE_FMT_DBL:
dtmf_samples_double(samples, offset, num, event, volume, sample_rate, channels);
break;
case AV_SAMPLE_FMT_FLT:
dtmf_samples_float(samples, offset, num, event, volume, sample_rate, channels);
break;
default:
ilog(LOG_ERR | LOG_FLAG_LIMIT, "Unsupported sample format %u", fmt);
break;
}
}
#ifdef HAVE_CODEC_CHAIN
static codec_cc_state cc_run(codec_cc_t *c, const str *data, unsigned long ts, void *async_cb_obj) {
AVPacket *pkt = c->avpkt;
ssize_t ret = cc_runner_do(c->runner, c->codec,
(unsigned char *) data->s, data->len,
pkt->data, pkt->size);
if (ret <= 0)
return CCC_ERR;
// XXX handle input frame sizes != 160
pkt->size = ret;
pkt->duration = c->def->duration(data->s, data->len);
pkt->pts = c->def->timestamp(ts, c->codec);
return CCC_OK;
}
static void __cc_async_job_free(struct async_job *j) {
g_free(j->data.s);
g_free(j);
}
static void __codec_cc_free(codec_cc_t *c) {
c->clear(c->clear_arg);
while (c->async_jobs.length) {
__auto_type j = t_queue_pop_head(&c->async_jobs);
c->async_callback(NULL, j->async_cb_obj);
__cc_async_job_free(j);
}
av_packet_free(&c->avpkt);
av_packet_free(&c->avpkt_async);
g_free(c);
}
// lock must be held
// append job to queue
static void __cc_async_do_add_queue(codec_cc_t *c, const str *data, unsigned long ts, void *async_cb_obj) {
struct async_job *j = g_new0(__typeof__(*j), 1);
j->data = str_dup_str(data);
j->async_cb_obj = async_cb_obj;
j->ts = ts;
t_queue_push_tail(&c->async_jobs, j);
}
// check busy flag and append to queue if set
// if not busy, sets busy flag
// also check blocked flag if busy: if set, try running first job
static bool __cc_async_check_busy_blocked_queue(codec_cc_t *c, const str *data, unsigned long ts,
void *async_cb_obj, __typeof__(__cc_run_async) run_async)
{
struct async_job *j = NULL;
async_job_q overflow = TYPED_GQUEUE_INIT;
{
LOCK(&c->async_lock);
if (!c->async_busy) {
// we can try running
c->async_busy = true;
return false;
}
atomic_inc_na(&c->stats->async_busy);
// codec is busy (either currently running or was blocked)
// append to queue
__cc_async_do_add_queue(c, data, ts, async_cb_obj);
if (c->async_jobs.length > 20) {
ilog(LOG_WARN | LOG_FLAG_LIMIT, "Async job queue overflow (%u @ %s), dropping frames",
c->async_jobs.length,
c->def->name);
do {
__auto_type jj = t_queue_pop_head(&c->async_jobs);
t_queue_push_tail(&overflow, jj);
} while (c->async_jobs.length > 20);
}
// if we were blocked (not currently running), try running now
if (c->async_blocked)
j = t_queue_pop_head(&c->async_jobs);
}
while (overflow.length) {
__auto_type jj = t_queue_pop_head(&overflow);
c->async_callback(NULL, jj->async_cb_obj);
__cc_async_job_free(jj);
}
if (j) {
atomic_inc_na(&c->stats->async_retry);
if (!run_async(c, &j->data, j->ts, j->async_cb_obj)) {
// still blocked. return to queue
atomic_inc_na(&c->stats->async_blocked);
LOCK(&c->async_lock);
t_queue_push_head(&c->async_jobs, j);
}
else {
// unblocked, running now
__cc_async_job_free(j);
LOCK(&c->async_lock);
c->async_blocked = false;
}
}
return true;
}
// runner failed, needed to block (no available context)
// set blocked flag and append to queue
// queue is guaranteed to be empty
static void __cc_async_blocked_queue(codec_cc_t *c, const str *data, unsigned long ts, void *async_cb_obj) {
LOCK(&c->async_lock);
__cc_async_do_add_queue(c, data, ts, async_cb_obj);
c->async_blocked = true;
// busy == true
}
static codec_cc_state cc_X_run_async(codec_cc_t *c, const str *data, unsigned long ts, void *async_cb_obj,
__typeof__(__cc_run_async) run_async)
{
if (__cc_async_check_busy_blocked_queue(c, data, ts, async_cb_obj, run_async))
return CCC_ASYNC;
if (!run_async(c, data, ts, async_cb_obj))
__cc_async_blocked_queue(c, data, ts, async_cb_obj);
return CCC_ASYNC;
}
codec_cc_state cc_run_async(codec_cc_t *c, const str *data, unsigned long ts, void *async_cb_obj) {
return cc_X_run_async(c, data, ts, async_cb_obj, __cc_run_async);
}
static void cc_X_pkt_callback(codec_cc_t *c, ssize_t size, __typeof__(__cc_run_async) run_async) {
AVPacket *pkt = c->avpkt_async;
void *async_cb_obj = c->async_cb_obj;
c->async_cb_obj = NULL;
c->async_callback(size >= 0 ? pkt : NULL, async_cb_obj);
pkt->size = 0;
struct async_job *j = NULL;
bool shutdown = false;
{
LOCK(&c->async_lock);
j = t_queue_pop_head(&c->async_jobs);
if (!j) {
if (c->async_shutdown)
shutdown = true;
else
c->async_busy = false;
}
}
if (shutdown) {
__codec_cc_free(c);
return;
}
if (j) {
if (!run_async(c, &j->data, j->ts, j->async_cb_obj)) {
LOCK(&c->async_lock);
t_queue_push_head(&c->async_jobs, j);
c->async_blocked = true;
}
else {
g_free(j->data.s);
g_free(j);
LOCK(&c->async_lock);
c->async_blocked = false;
}
}
}
static void cc_run_callback(void *p, ssize_t size) {
codec_cc_t *c = p;
AVPacket *pkt = c->avpkt_async;
if (size >= 0) {
pkt->size = size;
pkt->duration = c->data_len * 6L; // XXX
pkt->pts = c->ts * 6L; // XXX
}
cc_X_pkt_callback(c, size, __cc_run_async);
}
static bool __cc_run_async(codec_cc_t *c, const str *data, unsigned long ts, void *async_cb_obj) {
AVPacket *pkt = c->avpkt_async;
pkt->size = 2048;
c->data_len = data->len;
c->ts = ts;
c->async_cb_obj = async_cb_obj;
return cc_async_runner_do(&c->async_runner->runner,
&c->async_runner->async,
c->codec,
(unsigned char *) data->s, data->len,
pkt->data, pkt->size, cc_run_callback, c);
}
static void cc_clear(void *a) {
codec_chain_codec *c = a;
cc_client_codec_free(cc_client, &c);
}
static codec_cc_t *codec_cc_new_sync(codec_def_t *src, format_t *src_format, codec_def_t *dst,
format_t *dst_format, int bitrate, int ptime,
void *(*async_init)(void *, void *, void *),
void (*async_callback)(AVPacket *, void *))
{
if (!cc_get)
return NULL;
codec_chain_id id = cc_get(
(codec_chain_params) {
.name = src->rtpname,
.clock_rate = src_format->clockrate,
.channels = src_format->channels,
.ptime = 20, // XXX
},
(codec_chain_params) {
.name = dst->rtpname,
.clock_rate = dst_format->clockrate,
.channels = dst_format->channels,
.ptime = 20, // XXX
}
);
if (id == 0)
return NULL;
if (id >= CODEC_CHAIN_ID_MAX)
return NULL;
if (!cc_runners[id].sync)
return NULL;
codec_cc_t *ret = g_new0(codec_cc_t, 1);
codec_chain_codec_args args = {0};
ret->def = &cc_defs[id];
if (ret->def->args == CC_ARGS_OPUS) {
args.opus = (codec_chain_opus_args) {
.bitrate = bitrate,
.complexity = rtpe_common_config_ptr->codec_chain_opus_complexity,
.application = rtpe_common_config_ptr->codec_chain_opus_application,
};
}
ret->codec = cc_client_codec_new(cc_client, id, args);
ret->clear = cc_clear;
ret->clear_arg = ret->codec;
ret->runner = cc_runners[id].sync;
ret->stats = &cc_stats[id];
ret->avpkt = av_packet_alloc();
ret->run = cc_run;
return ret;
}
static codec_cc_t *codec_cc_new_async(codec_def_t *src, format_t *src_format, codec_def_t *dst,
format_t *dst_format, int bitrate, int ptime,
void *(*async_init)(void *, void *, void *),
void (*async_callback)(AVPacket *, void *))
{
if (!cc_get)
return NULL;
codec_chain_id id = cc_get(
(codec_chain_params) {
.name = src->rtpname,
.clock_rate = src_format->clockrate,
.channels = src_format->channels,
.ptime = 20, // XXX
},
(codec_chain_params) {
.name = dst->rtpname,
.clock_rate = dst_format->clockrate,
.channels = dst_format->channels,
.ptime = 20, // XXX
}
);
if (id == 0)
return NULL;
if (id >= CODEC_CHAIN_ID_MAX)
return NULL;
if (!cc_runners[id].async)
return NULL;
codec_cc_t *ret = g_new0(codec_cc_t, 1);
codec_chain_codec_args args = {0};
ret->def = &cc_defs[id];
if (ret->def->args == CC_ARGS_OPUS) {
args.opus = (codec_chain_opus_args) {
.bitrate = bitrate,
.complexity = rtpe_common_config_ptr->codec_chain_opus_complexity,
.application = rtpe_common_config_ptr->codec_chain_opus_application,
};
}
ret->codec = cc_client_codec_new(cc_client, id, args);
ret->clear = cc_clear;
ret->clear_arg = ret->codec;
ret->async_runner = cc_runners[id].async;
ret->stats = &cc_stats[id];
ret->run = cc_run_async;
ret->avpkt_async = av_packet_alloc();
av_new_packet(ret->avpkt_async, 2048);
mutex_init(&ret->async_lock);
t_queue_init(&ret->async_jobs);
ret->async_init = async_init;
ret->async_callback = async_callback;
return ret;
}
void codec_cc_stop(codec_cc_t *c) {
if (!c)
return;
// steal and fire all callbacks to release any references
async_job_q q;
{
LOCK(&c->async_lock);
q = c->async_jobs;
t_queue_init(&c->async_jobs);
}
while (q.length) {
__auto_type j = t_queue_pop_head(&q);
c->async_callback(NULL, j->async_cb_obj);
__cc_async_job_free(j);
}
}
void codec_cc_free(codec_cc_t **ccp) {
codec_cc_t *c = *ccp;
if (!c)
return;
*ccp = NULL;
{
LOCK(&c->async_lock);
if (c->async_busy && !c->async_blocked) {
c->async_shutdown = true;
return; // wait for callback
}
}
__codec_cc_free(c);
}
codec_cc_stats_q codec_cc_stats(void) {
codec_cc_stats_q ret = TYPED_GQUEUE_INIT;
for (unsigned int i = 0; i < CODEC_CHAIN_ID_MAX; i++) {
codec_chain_runner *r;
if (rtpe_common_config_ptr->codec_chain_async) {
if (!cc_runners[i].async)
continue;
r = &cc_runners[i].async->runner;
}
else {
if (!cc_runners[i].sync)
continue;
r = cc_runners[i].sync;
}
__auto_type q = r->queuer;
__auto_type e = g_new0(codec_cc_stats_entry, 1);
t_queue_push_tail(&ret, e);
g_strlcpy(e->name, r->def->name, sizeof(e->name));
#define LA(v) e->v = atomic_get_na(&cc_stats[i].v)
LA(async_busy);
LA(async_blocked);
LA(async_retry);
#undef LA
for (unsigned int j = 0; j < r->num_contexts; j++) {
__auto_type c = g_new0(codec_cc_context_stats, 1);
t_queue_push_tail(&e->contexts, c);
c->ctx_idx = j;
#define LA(v) c->v = atomic_get_na(&q->contexts[j].v)
LA(runs);
LA(slots);
LA(run_wait);
LA(writers_wait);
LA(compute_wait);
LA(readers_wait);
LA(run_busy);
LA(write_busy);
LA(slots_full);
LA(buf_full);
LA(ready_wait);
LA(callbacks_preempt);
LA(callbacks_fetch);
LA(callbacks_run);
LA(loop_barrier);
#undef LA
}
}
return ret;
}
#endif
AVPacket *codec_cc_input_data(codec_cc_t *c, const str *data, unsigned long ts, void *x, void *y, void *z) {
#ifdef HAVE_CODEC_CHAIN
if (c->avpkt)
av_new_packet(c->avpkt, 2048);
void *async_cb_obj = NULL;
if (c->async_init)
async_cb_obj = c->async_init(x, y, z);
codec_cc_state ret = c->run(c, data, ts, async_cb_obj);
if (ret == CCC_ERR) {
ilog(LOG_WARN | LOG_FLAG_LIMIT, "Received error from codec-chain job");
return c->avpkt; // return empty packet in case of error
}
if (ret == CCC_OK)
return c->avpkt;
// CCC_ASYNC
return NULL;
#else
return NULL;
#endif
}
codec_def_t *codec_def_make_generic_av(enum AVCodecID id) {
{
RWLOCK_R(&generic_ffmpeg_codecs_lock);
struct codec_def_s *ret = t_hash_table_lookup(generic_ffmpeg_codecs, GINT_TO_POINTER(id));
if (ret)
return ret;
}
{
RWLOCK_W(&generic_ffmpeg_codecs_lock);
struct codec_def_s *ret = t_hash_table_lookup(generic_ffmpeg_codecs, GINT_TO_POINTER(id));
if (ret)
return ret;
const AVCodec *codec = avcodec_find_decoder(id);
if (!codec)
return NULL;
ret = g_new(__typeof(*ret), 1);
*ret = (__typeof(*ret)) {
.rtpname = "generic ffmpeg codec",
.rtpname_str = STR_CONST("generic ffmpeg codec"),
.avcodec_id = id,
.default_clockrate_fact = {1,1},
.media_type = MT_AUDIO,
.codec_type = &codec_type_avcodec,
.decoder = codec,
.support_decoding = 1,
.support_encoding = 1, // just pretend
};
t_hash_table_insert(generic_ffmpeg_codecs, GINT_TO_POINTER(id), ret);
return ret;
}
}