|
|
|
|
@ -12,7 +12,7 @@
|
|
|
|
|
#include "output.h"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#define NUM_INPUTS 2
|
|
|
|
|
#define NUM_INPUTS 4
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
struct mix_s {
|
|
|
|
|
@ -23,7 +23,7 @@ struct mix_s {
|
|
|
|
|
AVFilterGraph *graph;
|
|
|
|
|
AVFilterContext *src_ctxs[NUM_INPUTS];
|
|
|
|
|
uint64_t pts_offs[NUM_INPUTS]; // initialized at first input seen
|
|
|
|
|
uint64_t in_pts[NUM_INPUTS]; // running counter of last seen adjusted pts
|
|
|
|
|
uint64_t in_pts[NUM_INPUTS]; // running counter of next expected adjusted pts
|
|
|
|
|
AVFilterContext *amix_ctx;
|
|
|
|
|
AVFilterContext *sink_ctx;
|
|
|
|
|
unsigned int next_idx;
|
|
|
|
|
@ -246,14 +246,16 @@ err:
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void mix_silence_fill_idx_off(mix_t *mix, unsigned int idx, unsigned int offset) {
|
|
|
|
|
while (mix->in_pts[idx] + offset < mix->out_pts) {
|
|
|
|
|
static void mix_silence_fill_idx_upto(mix_t *mix, unsigned int idx, uint64_t upto) {
|
|
|
|
|
unsigned int silence_samples = mix->clockrate / 100;
|
|
|
|
|
|
|
|
|
|
while (mix->in_pts[idx] < upto) {
|
|
|
|
|
if (G_UNLIKELY(!mix->silence_frame)) {
|
|
|
|
|
mix->silence_frame = av_frame_alloc();
|
|
|
|
|
mix->silence_frame->format = AV_SAMPLE_FMT_S16;
|
|
|
|
|
mix->silence_frame->channel_layout =
|
|
|
|
|
av_get_default_channel_layout(mix->channels);
|
|
|
|
|
mix->silence_frame->nb_samples = mix->clockrate / 100;
|
|
|
|
|
mix->silence_frame->nb_samples = silence_samples;
|
|
|
|
|
mix->silence_frame->sample_rate = mix->clockrate;
|
|
|
|
|
if (av_frame_get_buffer(mix->silence_frame, 0) < 0) {
|
|
|
|
|
ilog(LOG_ERR, "Failed to get silence frame buffers");
|
|
|
|
|
@ -264,9 +266,10 @@ static void mix_silence_fill_idx_off(mix_t *mix, unsigned int idx, unsigned int
|
|
|
|
|
|
|
|
|
|
dbg("pushing silence frame into stream %i (%lli < %llu)", idx,
|
|
|
|
|
(long long unsigned) mix->in_pts[idx],
|
|
|
|
|
(long long unsigned) mix->out_pts);
|
|
|
|
|
(long long unsigned) upto);
|
|
|
|
|
|
|
|
|
|
mix->silence_frame->pts = mix->in_pts[idx];
|
|
|
|
|
mix->silence_frame->nb_samples = MIN(silence_samples, upto - mix->in_pts[idx]);
|
|
|
|
|
mix->in_pts[idx] += mix->silence_frame->nb_samples;
|
|
|
|
|
|
|
|
|
|
if (av_buffersrc_write_frame(mix->src_ctxs[idx], mix->silence_frame))
|
|
|
|
|
@ -276,11 +279,14 @@ static void mix_silence_fill_idx_off(mix_t *mix, unsigned int idx, unsigned int
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void mix_silence_fill(mix_t *mix) {
|
|
|
|
|
if (mix->out_pts < mix->clockrate)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < NUM_INPUTS; i++) {
|
|
|
|
|
// check the pts of each input and give them max 1 second of delay.
|
|
|
|
|
// if they fall behind too much, fill input with silence. otherwise
|
|
|
|
|
// output stalls and won't produce media
|
|
|
|
|
mix_silence_fill_idx_off(mix, i, mix->clockrate);
|
|
|
|
|
mix_silence_fill_idx_upto(mix, i, mix->out_pts - mix->clockrate);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@ -297,25 +303,33 @@ int mix_add(mix_t *mix, AVFrame *frame, unsigned int idx, output_t *output) {
|
|
|
|
|
if (!mix->src_ctxs[idx])
|
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
|
|
dbg("stream %i pts_off %llu in pts %llu in frame pts %llu samples %u mix out pts %llu",
|
|
|
|
|
idx,
|
|
|
|
|
(unsigned long long) mix->pts_offs[idx],
|
|
|
|
|
(unsigned long long) mix->in_pts[idx],
|
|
|
|
|
(unsigned long long) frame->pts,
|
|
|
|
|
frame->nb_samples,
|
|
|
|
|
(unsigned long long) mix->out_pts);
|
|
|
|
|
|
|
|
|
|
// adjust for media started late
|
|
|
|
|
if (G_UNLIKELY(mix->pts_offs[idx] == (uint64_t) -1LL))
|
|
|
|
|
mix->pts_offs[idx] = mix->out_pts - frame->pts;
|
|
|
|
|
frame->pts += mix->pts_offs[idx];
|
|
|
|
|
|
|
|
|
|
// fill missing time
|
|
|
|
|
mix_silence_fill_idx_off(mix, idx, 0);
|
|
|
|
|
mix_silence_fill_idx_upto(mix, idx, frame->pts);
|
|
|
|
|
|
|
|
|
|
uint64_t frame_pts = frame->pts; // because *_add_frame unrefs the frame and invalidates pts
|
|
|
|
|
uint64_t next_pts = frame->pts + frame->nb_samples;
|
|
|
|
|
|
|
|
|
|
err = "failed to add frame to mixer";
|
|
|
|
|
if (av_buffersrc_add_frame(mix->src_ctxs[idx], frame))
|
|
|
|
|
goto err;
|
|
|
|
|
|
|
|
|
|
// update running counters
|
|
|
|
|
if (frame_pts > mix->out_pts)
|
|
|
|
|
mix->out_pts = frame_pts;
|
|
|
|
|
if (frame_pts > mix->in_pts[idx])
|
|
|
|
|
mix->in_pts[idx] = frame_pts;
|
|
|
|
|
if (next_pts > mix->out_pts)
|
|
|
|
|
mix->out_pts = next_pts;
|
|
|
|
|
if (next_pts > mix->in_pts[idx])
|
|
|
|
|
mix->in_pts[idx] = next_pts;
|
|
|
|
|
|
|
|
|
|
av_frame_free(&frame);
|
|
|
|
|
|
|
|
|
|
|