| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:40 +00:00
										 |  |  |  * filter layer | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |  * copyright (c) 2007 Bobby Bingham | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This file is part of FFmpeg. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * FFmpeg is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License as published by the Free Software Foundation; either | 
					
						
							|  |  |  |  * version 2.1 of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * FFmpeg is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
					
						
							|  |  |  |  * Lesser General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU Lesser General Public | 
					
						
							|  |  |  |  * License along with FFmpeg; if not, write to the Free Software | 
					
						
							|  |  |  |  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  | /* #define DEBUG */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-05-09 11:56:36 +00:00
										 |  |  | #include "libavcodec/imgconvert.h"
 | 
					
						
							| 
									
										
										
										
											2010-01-31 16:33:29 +00:00
										 |  |  | #include "libavutil/pixdesc.h"
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | #include "avfilter.h"
 | 
					
						
							| 
									
										
										
										
											2010-07-18 22:37:39 +00:00
										 |  |  | #include "internal.h"
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-08-08 18:43:38 +00:00
										 |  |  | unsigned avfilter_version(void) { | 
					
						
							|  |  |  |     return LIBAVFILTER_VERSION_INT; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-03 14:31:25 +00:00
										 |  |  | const char *avfilter_configuration(void) | 
					
						
							| 
									
										
										
										
											2009-11-18 17:15:17 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     return FFMPEG_CONFIGURATION; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-03 14:31:25 +00:00
										 |  |  | const char *avfilter_license(void) | 
					
						
							| 
									
										
										
										
											2009-11-18 17:15:17 +00:00
										 |  |  | { | 
					
						
							|  |  |  | #define LICENSE_PREFIX "libavfilter license: "
 | 
					
						
							|  |  |  |     return LICENSE_PREFIX FFMPEG_LICENSE + sizeof(LICENSE_PREFIX) - 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:54 +00:00
										 |  |  | /** helper macros to get the in/out pad on the dst/src filter */ | 
					
						
							|  |  |  | #define link_dpad(link)     link->dst-> input_pads[link->dstpad]
 | 
					
						
							|  |  |  | #define link_spad(link)     link->src->output_pads[link->srcpad]
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:27 +00:00
										 |  |  | AVFilterBufferRef *avfilter_ref_buffer(AVFilterBufferRef *ref, int pmask) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:19 +00:00
										 |  |  |     AVFilterBufferRef *ret = av_malloc(sizeof(AVFilterBufferRef)); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:38:13 +00:00
										 |  |  |     *ret = *ref; | 
					
						
							| 
									
										
										
										
											2010-08-11 11:06:04 +00:00
										 |  |  |     if (ref->type == AVMEDIA_TYPE_VIDEO) { | 
					
						
							|  |  |  |         ret->video = av_malloc(sizeof(AVFilterBufferRefVideoProps)); | 
					
						
							|  |  |  |         *ret->video = *ref->video; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-02-15 21:36:06 +00:00
										 |  |  |     ret->perms &= pmask; | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:27 +00:00
										 |  |  |     ret->buf->refcount ++; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:27 +00:00
										 |  |  | void avfilter_unref_buffer(AVFilterBufferRef *ref) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:27 +00:00
										 |  |  |     if(!(--ref->buf->refcount)) | 
					
						
							|  |  |  |         ref->buf->free(ref->buf); | 
					
						
							| 
									
										
										
										
											2010-08-11 11:06:04 +00:00
										 |  |  |     av_free(ref->video); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     av_free(ref); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:22 +00:00
										 |  |  | void avfilter_insert_pad(unsigned idx, unsigned *count, size_t padidx_off, | 
					
						
							|  |  |  |                          AVFilterPad **pads, AVFilterLink ***links, | 
					
						
							|  |  |  |                          AVFilterPad *newpad) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     unsigned i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     idx = FFMIN(idx, *count); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     *pads  = av_realloc(*pads,  sizeof(AVFilterPad)   * (*count + 1)); | 
					
						
							|  |  |  |     *links = av_realloc(*links, sizeof(AVFilterLink*) * (*count + 1)); | 
					
						
							|  |  |  |     memmove(*pads +idx+1, *pads +idx, sizeof(AVFilterPad)   * (*count-idx)); | 
					
						
							|  |  |  |     memmove(*links+idx+1, *links+idx, sizeof(AVFilterLink*) * (*count-idx)); | 
					
						
							|  |  |  |     memcpy(*pads+idx, newpad, sizeof(AVFilterPad)); | 
					
						
							|  |  |  |     (*links)[idx] = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     (*count) ++; | 
					
						
							|  |  |  |     for(i = idx+1; i < *count; i ++) | 
					
						
							|  |  |  |         if(*links[i]) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:15 +00:00
										 |  |  |             (*(unsigned *)((uint8_t *) *links[i] + padidx_off)) ++; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:22 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | int avfilter_link(AVFilterContext *src, unsigned srcpad, | 
					
						
							|  |  |  |                   AVFilterContext *dst, unsigned dstpad) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     AVFilterLink *link; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:36:57 +00:00
										 |  |  |     if(src->output_count <= srcpad || dst->input_count <= dstpad || | 
					
						
							|  |  |  |        src->outputs[srcpad]        || dst->inputs[dstpad]) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |         return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     src->outputs[srcpad] = | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:19 +00:00
										 |  |  |     dst-> inputs[dstpad] = link = av_mallocz(sizeof(AVFilterLink)); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:36:29 +00:00
										 |  |  |     link->src     = src; | 
					
						
							|  |  |  |     link->dst     = dst; | 
					
						
							|  |  |  |     link->srcpad  = srcpad; | 
					
						
							|  |  |  |     link->dstpad  = dstpad; | 
					
						
							| 
									
										
										
										
											2010-07-22 11:12:47 +00:00
										 |  |  |     link->type    = src->output_pads[srcpad].type; | 
					
						
							|  |  |  |     assert(PIX_FMT_NONE == -1 && SAMPLE_FMT_NONE == -1); | 
					
						
							|  |  |  |     link->format  = -1; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:18 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:19 +00:00
										 |  |  | int avfilter_insert_filter(AVFilterLink *link, AVFilterContext *filt, | 
					
						
							|  |  |  |                            unsigned in, unsigned out) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-01-13 00:16:52 +00:00
										 |  |  |     av_log(link->dst, AV_LOG_INFO, "auto-inserting filter '%s' " | 
					
						
							|  |  |  |            "between the filter '%s' and the filter '%s'\n", | 
					
						
							|  |  |  |            filt->name, link->src->name, link->dst->name); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:19 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     link->dst->inputs[link->dstpad] = NULL; | 
					
						
							|  |  |  |     if(avfilter_link(filt, out, link->dst, link->dstpad)) { | 
					
						
							|  |  |  |         /* failed to link output filter to new filter */ | 
					
						
							|  |  |  |         link->dst->inputs[link->dstpad] = link; | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* re-hookup the link to the new destination filter we inserted */ | 
					
						
							|  |  |  |     link->dst = filt; | 
					
						
							|  |  |  |     link->dstpad = in; | 
					
						
							|  |  |  |     filt->inputs[in] = link; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-22 11:12:47 +00:00
										 |  |  |     /* if any information on supported media formats already exists on the
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:28 +00:00
										 |  |  |      * link, we need to preserve that */ | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:08 +00:00
										 |  |  |     if(link->out_formats) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:32 +00:00
										 |  |  |         avfilter_formats_changeref(&link->out_formats, | 
					
						
							|  |  |  |                                    &filt->outputs[out]->out_formats); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:19 +00:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:41 +00:00
										 |  |  | int avfilter_config_links(AVFilterContext *filter) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:18 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     int (*config_link)(AVFilterLink *); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:41 +00:00
										 |  |  |     unsigned i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for(i = 0; i < filter->input_count; i ++) { | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:46 +00:00
										 |  |  |         AVFilterLink *link = filter->inputs[i]; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:38:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:46 +00:00
										 |  |  |         if(!link) continue; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         switch(link->init_state) { | 
					
						
							|  |  |  |         case AVLINK_INIT: | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         case AVLINK_STARTINIT: | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:11 +00:00
										 |  |  |             av_log(filter, AV_LOG_INFO, "circular filter chain detected\n"); | 
					
						
							|  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:41 +00:00
										 |  |  |         case AVLINK_UNINIT: | 
					
						
							|  |  |  |             link->init_state = AVLINK_STARTINIT; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if(avfilter_config_links(link->src)) | 
					
						
							|  |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:47 +00:00
										 |  |  |             if(!(config_link = link_spad(link).config_props)) | 
					
						
							|  |  |  |                 config_link  = avfilter_default_config_output_link; | 
					
						
							|  |  |  |             if(config_link(link)) | 
					
						
							|  |  |  |                 return -1; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:35:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:47 +00:00
										 |  |  |             if((config_link = link_dpad(link).config_props)) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:23 +00:00
										 |  |  |                 if(config_link(link)) | 
					
						
							|  |  |  |                     return -1; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:35:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:41 +00:00
										 |  |  |             link->init_state = AVLINK_INIT; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:19 +00:00
										 |  |  | void ff_dprintf_picref(void *ctx, AVFilterBufferRef *picref, int end) | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     dprintf(ctx, | 
					
						
							| 
									
										
										
										
											2010-03-13 11:27:05 +00:00
										 |  |  |             "picref[%p data[%p, %p, %p, %p] linesize[%d, %d, %d, %d] pts:%"PRId64" pos:%"PRId64" a:%d/%d s:%dx%d]%s", | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  |             picref, | 
					
						
							|  |  |  |             picref->data    [0], picref->data    [1], picref->data    [2], picref->data    [3], | 
					
						
							|  |  |  |             picref->linesize[0], picref->linesize[1], picref->linesize[2], picref->linesize[3], | 
					
						
							| 
									
										
										
										
											2010-03-13 11:27:05 +00:00
										 |  |  |             picref->pts, picref->pos, | 
					
						
							| 
									
										
										
										
											2010-08-11 11:06:04 +00:00
										 |  |  |             picref->video->pixel_aspect.num, picref->video->pixel_aspect.den, picref->video->w, picref->video->h, | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  |             end ? "\n" : ""); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-18 22:37:39 +00:00
										 |  |  | void ff_dprintf_link(void *ctx, AVFilterLink *link, int end) | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  | { | 
					
						
							|  |  |  |     dprintf(ctx, | 
					
						
							| 
									
										
										
										
											2009-10-25 21:06:00 +00:00
										 |  |  |             "link[%p s:%dx%d fmt:%-16s %-16s->%-16s]%s", | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  |             link, link->w, link->h, | 
					
						
							| 
									
										
										
										
											2010-01-31 16:33:29 +00:00
										 |  |  |             av_pix_fmt_descriptors[link->format].name, | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  |             link->src ? link->src->filter->name : "", | 
					
						
							|  |  |  |             link->dst ? link->dst->filter->name : "", | 
					
						
							|  |  |  |             end ? "\n" : ""); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:19 +00:00
										 |  |  | AVFilterBufferRef *avfilter_get_video_buffer(AVFilterLink *link, int perms, int w, int h) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:19 +00:00
										 |  |  |     AVFilterBufferRef *ret = NULL; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-18 22:37:39 +00:00
										 |  |  |     FF_DPRINTF_START(NULL, get_video_buffer); ff_dprintf_link(NULL, link, 0); dprintf(NULL, " perms:%d w:%d h:%d\n", perms, w, h); | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:54 +00:00
										 |  |  |     if(link_dpad(link).get_video_buffer) | 
					
						
							| 
									
										
										
										
											2009-10-18 08:16:18 +00:00
										 |  |  |         ret = link_dpad(link).get_video_buffer(link, perms, w, h); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     if(!ret) | 
					
						
							| 
									
										
										
										
											2009-10-18 08:16:18 +00:00
										 |  |  |         ret = avfilter_default_get_video_buffer(link, perms, w, h); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-18 22:37:39 +00:00
										 |  |  |     FF_DPRINTF_START(NULL, get_video_buffer); ff_dprintf_link(NULL, link, 0); dprintf(NULL, " returning "); ff_dprintf_picref(NULL, ret, 1); | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:31 +00:00
										 |  |  | int avfilter_request_frame(AVFilterLink *link) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-07-18 22:37:39 +00:00
										 |  |  |     FF_DPRINTF_START(NULL, request_frame); ff_dprintf_link(NULL, link, 1); | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:54 +00:00
										 |  |  |     if(link_spad(link).request_frame) | 
					
						
							|  |  |  |         return link_spad(link).request_frame(link); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:36:17 +00:00
										 |  |  |     else if(link->src->inputs[0]) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:31 +00:00
										 |  |  |         return avfilter_request_frame(link->src->inputs[0]); | 
					
						
							|  |  |  |     else return -1; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:26 +00:00
										 |  |  | int avfilter_poll_frame(AVFilterLink *link) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int i, min=INT_MAX; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if(link_spad(link).poll_frame) | 
					
						
							|  |  |  |         return link_spad(link).poll_frame(link); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:31 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (i=0; i<link->src->input_count; i++) { | 
					
						
							| 
									
										
										
										
											2009-10-18 22:33:37 +00:00
										 |  |  |         int val; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:31 +00:00
										 |  |  |         if(!link->src->inputs[i]) | 
					
						
							|  |  |  |             return -1; | 
					
						
							| 
									
										
										
										
											2009-10-18 22:33:37 +00:00
										 |  |  |         val = avfilter_poll_frame(link->src->inputs[i]); | 
					
						
							|  |  |  |         min = FFMIN(min, val); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:31 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return min; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | /* XXX: should we do the duplicating of the picture ref here, instead of
 | 
					
						
							|  |  |  |  * forcing the source filter to do it? */ | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:19 +00:00
										 |  |  | void avfilter_start_frame(AVFilterLink *link, AVFilterBufferRef *picref) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:19 +00:00
										 |  |  |     void (*start_frame)(AVFilterLink *, AVFilterBufferRef *); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:03 +00:00
										 |  |  |     AVFilterPad *dst = &link_dpad(link); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-18 22:37:39 +00:00
										 |  |  |     FF_DPRINTF_START(NULL, start_frame); ff_dprintf_link(NULL, link, 0); dprintf(NULL, " "); ff_dprintf_picref(NULL, picref, 1); | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:03 +00:00
										 |  |  |     if(!(start_frame = dst->start_frame)) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |         start_frame = avfilter_default_start_frame; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |     /* prepare to copy the picture if it has insufficient permissions */ | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:03 +00:00
										 |  |  |     if((dst->min_perms & picref->perms) != dst->min_perms || | 
					
						
							|  |  |  |         dst->rej_perms & picref->perms) { | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:30 +00:00
										 |  |  |         /*
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |         av_log(link->dst, AV_LOG_INFO, | 
					
						
							|  |  |  |                 "frame copy needed (have perms %x, need %x, reject %x)\n", | 
					
						
							| 
									
										
										
										
											2008-02-15 21:38:10 +00:00
										 |  |  |                 picref->perms, | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |                 link_dpad(link).min_perms, link_dpad(link).rej_perms); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:30 +00:00
										 |  |  |         */ | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:34 +00:00
										 |  |  |         link->cur_buf = avfilter_default_get_video_buffer(link, dst->min_perms, link->w, link->h); | 
					
						
							|  |  |  |         link->src_buf = picref; | 
					
						
							|  |  |  |         avfilter_copy_buffer_ref_props(link->cur_buf, link->src_buf); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     else | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:34 +00:00
										 |  |  |         link->cur_buf = picref; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:34 +00:00
										 |  |  |     start_frame(link, link->cur_buf); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void avfilter_end_frame(AVFilterLink *link) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     void (*end_frame)(AVFilterLink *); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:30 +00:00
										 |  |  |     if(!(end_frame = link_dpad(link).end_frame)) | 
					
						
							|  |  |  |         end_frame = avfilter_default_end_frame; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     end_frame(link); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |     /* unreference the source picture if we're feeding the destination filter
 | 
					
						
							|  |  |  |      * a copied version dues to permission issues */ | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:34 +00:00
										 |  |  |     if(link->src_buf) { | 
					
						
							|  |  |  |         avfilter_unref_buffer(link->src_buf); | 
					
						
							|  |  |  |         link->src_buf = NULL; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-04 23:26:13 +00:00
										 |  |  | void avfilter_draw_slice(AVFilterLink *link, int y, int h, int slice_dir) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |     uint8_t *src[4], *dst[4]; | 
					
						
							| 
									
										
										
										
											2010-04-03 15:29:48 +00:00
										 |  |  |     int i, j, vsub; | 
					
						
							| 
									
										
										
										
											2009-12-04 23:26:13 +00:00
										 |  |  |     void (*draw_slice)(AVFilterLink *, int, int, int); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-07-18 22:37:39 +00:00
										 |  |  |     FF_DPRINTF_START(NULL, draw_slice); ff_dprintf_link(NULL, link, 0); dprintf(NULL, " y:%d h:%d dir:%d\n", y, h, slice_dir); | 
					
						
							| 
									
										
										
										
											2009-10-06 22:07:32 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |     /* copy the slice if needed for permission reasons */ | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:34 +00:00
										 |  |  |     if(link->src_buf) { | 
					
						
							| 
									
										
										
										
											2010-01-31 16:33:29 +00:00
										 |  |  |         vsub = av_pix_fmt_descriptors[link->format].log2_chroma_h; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:41:38 +00:00
										 |  |  |         for(i = 0; i < 4; i ++) { | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:34 +00:00
										 |  |  |             if(link->src_buf->data[i]) { | 
					
						
							|  |  |  |                 src[i] = link->src_buf-> data[i] + | 
					
						
							|  |  |  |                     (y >> (i==0 ? 0 : vsub)) * link->src_buf-> linesize[i]; | 
					
						
							|  |  |  |                 dst[i] = link->cur_buf->data[i] + | 
					
						
							|  |  |  |                     (y >> (i==0 ? 0 : vsub)) * link->cur_buf->linesize[i]; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |             } else | 
					
						
							|  |  |  |                 src[i] = dst[i] = NULL; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         for(i = 0; i < 4; i ++) { | 
					
						
							| 
									
										
										
										
											2008-02-15 21:42:01 +00:00
										 |  |  |             int planew = | 
					
						
							| 
									
										
										
										
											2010-08-11 11:06:04 +00:00
										 |  |  |                 ff_get_plane_bytewidth(link->format, link->cur_buf->video->w, i); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:42:01 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |             if(!src[i]) continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:55 +00:00
										 |  |  |             for(j = 0; j < h >> (i==0 ? 0 : vsub); j ++) { | 
					
						
							| 
									
										
										
										
											2008-02-15 21:42:01 +00:00
										 |  |  |                 memcpy(dst[i], src[i], planew); | 
					
						
							| 
									
										
										
										
											2010-08-07 01:15:34 +00:00
										 |  |  |                 src[i] += link->src_buf ->linesize[i]; | 
					
						
							|  |  |  |                 dst[i] += link->cur_buf->linesize[i]; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:57 +00:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-11 22:05:48 +00:00
										 |  |  |     if(!(draw_slice = link_dpad(link).draw_slice)) | 
					
						
							|  |  |  |         draw_slice = avfilter_default_draw_slice; | 
					
						
							| 
									
										
										
										
											2009-12-04 23:26:13 +00:00
										 |  |  |     draw_slice(link, y, h, slice_dir); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-24 23:47:33 +00:00
										 |  |  | #define MAX_REGISTERED_AVFILTERS_NB 64
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static AVFilter *registered_avfilters[MAX_REGISTERED_AVFILTERS_NB + 1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int next_registered_avfilter_idx = 0; | 
					
						
							| 
									
										
										
										
											2009-10-27 19:52:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:33 +00:00
										 |  |  | AVFilter *avfilter_get_by_name(const char *name) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-24 23:47:33 +00:00
										 |  |  |     int i; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:38:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-24 23:47:33 +00:00
										 |  |  |     for (i = 0; registered_avfilters[i]; i++) | 
					
						
							|  |  |  |         if (!strcmp(registered_avfilters[i]->name, name)) | 
					
						
							|  |  |  |             return registered_avfilters[i]; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-24 23:47:33 +00:00
										 |  |  | int avfilter_register(AVFilter *filter) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-24 23:47:33 +00:00
										 |  |  |     if (next_registered_avfilter_idx == MAX_REGISTERED_AVFILTERS_NB) | 
					
						
							|  |  |  |         return -1; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:38:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-24 23:47:33 +00:00
										 |  |  |     registered_avfilters[next_registered_avfilter_idx++] = filter; | 
					
						
							|  |  |  |     return 0; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-24 23:58:48 +00:00
										 |  |  | AVFilter **av_filter_next(AVFilter **filter) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return filter ? ++filter : ®istered_avfilters[0]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | void avfilter_uninit(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-24 23:47:33 +00:00
										 |  |  |     memset(registered_avfilters, 0, sizeof(registered_avfilters)); | 
					
						
							|  |  |  |     next_registered_avfilter_idx = 0; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int pad_count(const AVFilterPad *pads) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int count; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:33 +00:00
										 |  |  |     for(count = 0; pads->name; count ++) pads ++; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     return count; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const char *filter_name(void *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     AVFilterContext *filter = p; | 
					
						
							|  |  |  |     return filter->filter->name; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-07 09:08:28 +00:00
										 |  |  | static const AVClass avfilter_class = { | 
					
						
							|  |  |  |     "AVFilter", | 
					
						
							| 
									
										
										
										
											2010-04-28 20:00:23 +00:00
										 |  |  |     filter_name, | 
					
						
							|  |  |  |     NULL, | 
					
						
							|  |  |  |     LIBAVUTIL_VERSION_INT, | 
					
						
							| 
									
										
										
										
											2008-03-07 09:08:28 +00:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:33 +00:00
										 |  |  | AVFilterContext *avfilter_open(AVFilter *filter, const char *inst_name) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-02-15 21:39:12 +00:00
										 |  |  |     AVFilterContext *ret; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!filter) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 20:30:57 +00:00
										 |  |  |     ret = av_mallocz(sizeof(AVFilterContext)); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-03-07 09:08:28 +00:00
										 |  |  |     ret->av_class = &avfilter_class; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     ret->filter   = filter; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:36:40 +00:00
										 |  |  |     ret->name     = inst_name ? av_strdup(inst_name) : NULL; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     ret->priv     = av_mallocz(filter->priv_size); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:36:57 +00:00
										 |  |  |     ret->input_count  = pad_count(filter->inputs); | 
					
						
							| 
									
										
										
										
											2009-01-12 20:30:57 +00:00
										 |  |  |     if (ret->input_count) { | 
					
						
							| 
									
										
										
										
											2009-01-12 20:34:34 +00:00
										 |  |  |         ret->input_pads   = av_malloc(sizeof(AVFilterPad) * ret->input_count); | 
					
						
							|  |  |  |         memcpy(ret->input_pads, filter->inputs, sizeof(AVFilterPad) * ret->input_count); | 
					
						
							|  |  |  |         ret->inputs       = av_mallocz(sizeof(AVFilterLink*) * ret->input_count); | 
					
						
							| 
									
										
										
										
											2009-01-12 20:30:57 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:07 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:36:57 +00:00
										 |  |  |     ret->output_count = pad_count(filter->outputs); | 
					
						
							| 
									
										
										
										
											2009-01-12 20:30:57 +00:00
										 |  |  |     if (ret->output_count) { | 
					
						
							| 
									
										
										
										
											2009-01-12 20:34:34 +00:00
										 |  |  |         ret->output_pads  = av_malloc(sizeof(AVFilterPad) * ret->output_count); | 
					
						
							|  |  |  |         memcpy(ret->output_pads, filter->outputs, sizeof(AVFilterPad) * ret->output_count); | 
					
						
							|  |  |  |         ret->outputs      = av_mallocz(sizeof(AVFilterLink*) * ret->output_count); | 
					
						
							| 
									
										
										
										
											2009-01-12 20:30:57 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-02-15 21:36:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void avfilter_destroy(AVFilterContext *filter) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if(filter->filter->uninit) | 
					
						
							|  |  |  |         filter->filter->uninit(filter); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:07 +00:00
										 |  |  |     for(i = 0; i < filter->input_count; i ++) { | 
					
						
							| 
									
										
										
										
											2010-08-06 22:25:33 +00:00
										 |  |  |         if(filter->inputs[i]) { | 
					
						
							| 
									
										
										
										
											2010-08-06 22:25:36 +00:00
										 |  |  |             if (filter->inputs[i]->src) | 
					
						
							| 
									
										
										
										
											2010-08-06 22:25:39 +00:00
										 |  |  |                 filter->inputs[i]->src->outputs[filter->inputs[i]->srcpad] = NULL; | 
					
						
							| 
									
										
										
										
											2010-08-06 22:25:33 +00:00
										 |  |  |             avfilter_formats_unref(&filter->inputs[i]->in_formats); | 
					
						
							|  |  |  |             avfilter_formats_unref(&filter->inputs[i]->out_formats); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-02-15 21:38:29 +00:00
										 |  |  |         av_freep(&filter->inputs[i]); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-02-15 21:37:07 +00:00
										 |  |  |     for(i = 0; i < filter->output_count; i ++) { | 
					
						
							| 
									
										
										
										
											2010-08-06 22:25:33 +00:00
										 |  |  |         if(filter->outputs[i]) { | 
					
						
							| 
									
										
										
										
											2010-08-06 22:25:36 +00:00
										 |  |  |             if (filter->outputs[i]->dst) | 
					
						
							| 
									
										
										
										
											2010-08-06 22:25:39 +00:00
										 |  |  |                 filter->outputs[i]->dst->inputs[filter->outputs[i]->dstpad] = NULL; | 
					
						
							| 
									
										
										
										
											2010-08-06 22:25:33 +00:00
										 |  |  |             avfilter_formats_unref(&filter->outputs[i]->in_formats); | 
					
						
							|  |  |  |             avfilter_formats_unref(&filter->outputs[i]->out_formats); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-02-15 21:38:29 +00:00
										 |  |  |         av_freep(&filter->outputs[i]); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:38:29 +00:00
										 |  |  |     av_freep(&filter->name); | 
					
						
							|  |  |  |     av_freep(&filter->input_pads); | 
					
						
							|  |  |  |     av_freep(&filter->output_pads); | 
					
						
							|  |  |  |     av_freep(&filter->inputs); | 
					
						
							|  |  |  |     av_freep(&filter->outputs); | 
					
						
							|  |  |  |     av_freep(&filter->priv); | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  |     av_free(filter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-02-15 21:36:50 +00:00
										 |  |  | int avfilter_init_filter(AVFilterContext *filter, const char *args, void *opaque) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:46 +00:00
										 |  |  |     int ret=0; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if(filter->filter->init) | 
					
						
							| 
									
										
										
										
											2008-02-15 21:40:46 +00:00
										 |  |  |         ret = filter->filter->init(filter, args, opaque); | 
					
						
							|  |  |  |     return ret; | 
					
						
							| 
									
										
										
										
											2008-02-15 21:33:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 |