00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00026 #include "libavutil/avassert.h"
00027 #include "libavutil/channel_layout.h"
00028 #include "libavutil/common.h"
00029 #include "libavutil/mathematics.h"
00030 #include "libavutil/samplefmt.h"
00031
00032 #include "audio.h"
00033 #include "avfilter.h"
00034 #include "internal.h"
00035 #include "video.h"
00036
00037 typedef struct Buf {
00038 AVFilterBufferRef *buf;
00039 struct Buf *next;
00040 } Buf;
00041
00042 typedef struct {
00043 Buf root;
00044 Buf *last;
00045
00050 AVFilterBufferRef *buf_out;
00051 int allocated_samples;
00052 } FifoContext;
00053
00054 static av_cold int init(AVFilterContext *ctx, const char *args)
00055 {
00056 FifoContext *fifo = ctx->priv;
00057 fifo->last = &fifo->root;
00058
00059 return 0;
00060 }
00061
00062 static av_cold void uninit(AVFilterContext *ctx)
00063 {
00064 FifoContext *fifo = ctx->priv;
00065 Buf *buf, *tmp;
00066
00067 for (buf = fifo->root.next; buf; buf = tmp) {
00068 tmp = buf->next;
00069 avfilter_unref_bufferp(&buf->buf);
00070 av_free(buf);
00071 }
00072
00073 avfilter_unref_bufferp(&fifo->buf_out);
00074 }
00075
00076 static int add_to_queue(AVFilterLink *inlink, AVFilterBufferRef *buf)
00077 {
00078 FifoContext *fifo = inlink->dst->priv;
00079
00080 fifo->last->next = av_mallocz(sizeof(Buf));
00081 if (!fifo->last->next) {
00082 avfilter_unref_buffer(buf);
00083 return AVERROR(ENOMEM);
00084 }
00085
00086 fifo->last = fifo->last->next;
00087 fifo->last->buf = buf;
00088
00089 return 0;
00090 }
00091
00092 static void queue_pop(FifoContext *s)
00093 {
00094 Buf *tmp = s->root.next->next;
00095 if (s->last == s->root.next)
00096 s->last = &s->root;
00097 av_freep(&s->root.next);
00098 s->root.next = tmp;
00099 }
00100
00104 static void buffer_offset(AVFilterLink *link, AVFilterBufferRef *buf,
00105 int offset)
00106 {
00107 int nb_channels = av_get_channel_layout_nb_channels(link->channel_layout);
00108 int planar = av_sample_fmt_is_planar(link->format);
00109 int planes = planar ? nb_channels : 1;
00110 int block_align = av_get_bytes_per_sample(link->format) * (planar ? 1 : nb_channels);
00111 int i;
00112
00113 av_assert0(buf->audio->nb_samples > offset);
00114
00115 for (i = 0; i < planes; i++)
00116 buf->extended_data[i] += block_align*offset;
00117 if (buf->data != buf->extended_data)
00118 memcpy(buf->data, buf->extended_data,
00119 FFMIN(planes, FF_ARRAY_ELEMS(buf->data)) * sizeof(*buf->data));
00120 buf->linesize[0] -= block_align*offset;
00121 buf->audio->nb_samples -= offset;
00122
00123 if (buf->pts != AV_NOPTS_VALUE) {
00124 buf->pts += av_rescale_q(offset, (AVRational){1, link->sample_rate},
00125 link->time_base);
00126 }
00127 }
00128
00129 static int calc_ptr_alignment(AVFilterBufferRef *buf)
00130 {
00131 int planes = av_sample_fmt_is_planar(buf->format) ?
00132 av_get_channel_layout_nb_channels(buf->audio->channel_layout) : 1;
00133 int min_align = 128;
00134 int p;
00135
00136 for (p = 0; p < planes; p++) {
00137 int cur_align = 128;
00138 while ((intptr_t)buf->extended_data[p] % cur_align)
00139 cur_align >>= 1;
00140 if (cur_align < min_align)
00141 min_align = cur_align;
00142 }
00143 return min_align;
00144 }
00145
00146 static int return_audio_frame(AVFilterContext *ctx)
00147 {
00148 AVFilterLink *link = ctx->outputs[0];
00149 FifoContext *s = ctx->priv;
00150 AVFilterBufferRef *head = s->root.next->buf;
00151 AVFilterBufferRef *buf_out;
00152 int ret;
00153
00154 if (!s->buf_out &&
00155 head->audio->nb_samples >= link->request_samples &&
00156 calc_ptr_alignment(head) >= 32) {
00157 if (head->audio->nb_samples == link->request_samples) {
00158 buf_out = head;
00159 queue_pop(s);
00160 } else {
00161 buf_out = avfilter_ref_buffer(head, AV_PERM_READ);
00162 if (!buf_out)
00163 return AVERROR(ENOMEM);
00164
00165 buf_out->audio->nb_samples = link->request_samples;
00166 buffer_offset(link, head, link->request_samples);
00167 }
00168 } else {
00169 int nb_channels = av_get_channel_layout_nb_channels(link->channel_layout);
00170
00171 if (!s->buf_out) {
00172 s->buf_out = ff_get_audio_buffer(link, AV_PERM_WRITE,
00173 link->request_samples);
00174 if (!s->buf_out)
00175 return AVERROR(ENOMEM);
00176
00177 s->buf_out->audio->nb_samples = 0;
00178 s->buf_out->pts = head->pts;
00179 s->allocated_samples = link->request_samples;
00180 } else if (link->request_samples != s->allocated_samples) {
00181 av_log(ctx, AV_LOG_ERROR, "request_samples changed before the "
00182 "buffer was returned.\n");
00183 return AVERROR(EINVAL);
00184 }
00185
00186 while (s->buf_out->audio->nb_samples < s->allocated_samples) {
00187 int len = FFMIN(s->allocated_samples - s->buf_out->audio->nb_samples,
00188 head->audio->nb_samples);
00189
00190 av_samples_copy(s->buf_out->extended_data, head->extended_data,
00191 s->buf_out->audio->nb_samples, 0, len, nb_channels,
00192 link->format);
00193 s->buf_out->audio->nb_samples += len;
00194
00195 if (len == head->audio->nb_samples) {
00196 avfilter_unref_buffer(head);
00197 queue_pop(s);
00198
00199 if (!s->root.next &&
00200 (ret = ff_request_frame(ctx->inputs[0])) < 0) {
00201 if (ret == AVERROR_EOF) {
00202 av_samples_set_silence(s->buf_out->extended_data,
00203 s->buf_out->audio->nb_samples,
00204 s->allocated_samples -
00205 s->buf_out->audio->nb_samples,
00206 nb_channels, link->format);
00207 s->buf_out->audio->nb_samples = s->allocated_samples;
00208 break;
00209 }
00210 return ret;
00211 }
00212 head = s->root.next->buf;
00213 } else {
00214 buffer_offset(link, head, len);
00215 }
00216 }
00217 buf_out = s->buf_out;
00218 s->buf_out = NULL;
00219 }
00220 return ff_filter_frame(link, buf_out);
00221 }
00222
00223 static int request_frame(AVFilterLink *outlink)
00224 {
00225 FifoContext *fifo = outlink->src->priv;
00226 int ret = 0;
00227
00228 if (!fifo->root.next) {
00229 if ((ret = ff_request_frame(outlink->src->inputs[0])) < 0)
00230 return ret;
00231 av_assert0(fifo->root.next);
00232 }
00233
00234 if (outlink->request_samples) {
00235 return return_audio_frame(outlink->src);
00236 } else {
00237 ret = ff_filter_frame(outlink, fifo->root.next->buf);
00238 queue_pop(fifo);
00239 }
00240
00241 return ret;
00242 }
00243
00244 static const AVFilterPad avfilter_vf_fifo_inputs[] = {
00245 {
00246 .name = "default",
00247 .type = AVMEDIA_TYPE_VIDEO,
00248 .get_video_buffer = ff_null_get_video_buffer,
00249 .filter_frame = add_to_queue,
00250 .min_perms = AV_PERM_PRESERVE,
00251 },
00252 { NULL }
00253 };
00254
00255 static const AVFilterPad avfilter_vf_fifo_outputs[] = {
00256 {
00257 .name = "default",
00258 .type = AVMEDIA_TYPE_VIDEO,
00259 .request_frame = request_frame,
00260 },
00261 { NULL }
00262 };
00263
00264 AVFilter avfilter_vf_fifo = {
00265 .name = "fifo",
00266 .description = NULL_IF_CONFIG_SMALL("Buffer input images and send them when they are requested."),
00267
00268 .init = init,
00269 .uninit = uninit,
00270
00271 .priv_size = sizeof(FifoContext),
00272
00273 .inputs = avfilter_vf_fifo_inputs,
00274 .outputs = avfilter_vf_fifo_outputs,
00275 };
00276
00277 static const AVFilterPad avfilter_af_afifo_inputs[] = {
00278 {
00279 .name = "default",
00280 .type = AVMEDIA_TYPE_AUDIO,
00281 .get_audio_buffer = ff_null_get_audio_buffer,
00282 .filter_frame = add_to_queue,
00283 .min_perms = AV_PERM_PRESERVE,
00284 },
00285 { NULL }
00286 };
00287
00288 static const AVFilterPad avfilter_af_afifo_outputs[] = {
00289 {
00290 .name = "default",
00291 .type = AVMEDIA_TYPE_AUDIO,
00292 .request_frame = request_frame,
00293 },
00294 { NULL }
00295 };
00296
00297 AVFilter avfilter_af_afifo = {
00298 .name = "afifo",
00299 .description = NULL_IF_CONFIG_SMALL("Buffer input frames and send them when they are requested."),
00300
00301 .init = init,
00302 .uninit = uninit,
00303
00304 .priv_size = sizeof(FifoContext),
00305
00306 .inputs = avfilter_af_afifo_inputs,
00307 .outputs = avfilter_af_afifo_outputs,
00308 };