libavfilter/vf_idet.c
e3e89b6d
 /*
  * Copyright (C) 2012 Michael Niedermayer <michaelni@gmx.at>
  *
  * 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
  */
 
43cbd440
 #include <float.h> /* FLT_MAX */
 
e3e89b6d
 #include "libavutil/cpu.h"
 #include "libavutil/common.h"
43cbd440
 #include "libavutil/opt.h"
e3e89b6d
 #include "libavutil/pixdesc.h"
 #include "avfilter.h"
c9e183b4
 #include "internal.h"
e3e89b6d
 
fa1f92a4
 #define HIST_SIZE 4
 
ca2b450c
 typedef enum {
     TFF,
     BFF,
     PROGRSSIVE,
     UNDETERMINED,
 } Type;
 
e3e89b6d
 typedef struct {
43cbd440
     const AVClass *class;
e3e89b6d
     float interlace_threshold;
     float progressive_threshold;
 
1a5c08ee
     Type last_type;
a2349dc3
     int prestat[4];
     int poststat[4];
5d9cfd87
 
fa1f92a4
     uint8_t history[HIST_SIZE];
 
a05a44e2
     AVFrame *cur;
     AVFrame *next;
     AVFrame *prev;
648dbae5
     int (*filter_line)(const uint8_t *prev, const uint8_t *cur, const uint8_t *next, int w);
e3e89b6d
 
     const AVPixFmtDescriptor *csp;
 } IDETContext;
 
43cbd440
 #define OFFSET(x) offsetof(IDETContext, x)
 #define FLAGS AV_OPT_FLAG_VIDEO_PARAM|AV_OPT_FLAG_FILTERING_PARAM
 
 static const AVOption idet_options[] = {
cdb7a1ac
     { "intl_thres", "set interlacing threshold", OFFSET(interlace_threshold),   AV_OPT_TYPE_FLOAT, {.dbl = 1.04}, -1, FLT_MAX, FLAGS },
     { "prog_thres", "set progressive threshold", OFFSET(progressive_threshold), AV_OPT_TYPE_FLOAT, {.dbl = 1.5},  -1, FLT_MAX, FLAGS },
43cbd440
     { NULL }
 };
 
 AVFILTER_DEFINE_CLASS(idet);
 
0477254d
 static const char *type2str(Type type)
 {
     switch(type) {
         case TFF         : return "Top Field First   ";
         case BFF         : return "Bottom Field First";
         case PROGRSSIVE  : return "Progressive       ";
         case UNDETERMINED: return "Undetermined      ";
     }
     return NULL;
 }
e3e89b6d
 
 static int filter_line_c(const uint8_t *a, const uint8_t *b, const uint8_t *c, int w)
 {
     int x;
     int ret=0;
 
     for(x=0; x<w; x++){
dc9edb06
         int v = (*a++ + *c++) - 2 * *b++;
         ret += FFABS(v);
e3e89b6d
     }
 
     return ret;
 }
 
 static int filter_line_c_16bit(const uint16_t *a, const uint16_t *b, const uint16_t *c, int w)
 {
     int x;
     int ret=0;
 
     for(x=0; x<w; x++){
dc9edb06
         int v = (*a++ + *c++) - 2 * *b++;
         ret += FFABS(v);
e3e89b6d
     }
 
     return ret;
 }
 
 static void filter(AVFilterContext *ctx)
 {
     IDETContext *idet = ctx->priv;
     int y, i;
     int64_t alpha[2]={0};
     int64_t delta=0;
fa1f92a4
     Type type, best_type;
     int match = 0;
e3e89b6d
 
     for (i = 0; i < idet->csp->nb_components; i++) {
a05a44e2
         int w = idet->cur->width;
         int h = idet->cur->height;
e3e89b6d
         int refs = idet->cur->linesize[i];
 
         if (i && i<3) {
61b268ee
             w = FF_CEIL_RSHIFT(w, idet->csp->log2_chroma_w);
             h = FF_CEIL_RSHIFT(h, idet->csp->log2_chroma_h);
e3e89b6d
         }
 
         for (y = 2; y < h - 2; y++) {
             uint8_t *prev = &idet->prev->data[i][y*refs];
             uint8_t *cur  = &idet->cur ->data[i][y*refs];
             uint8_t *next = &idet->next->data[i][y*refs];
             alpha[ y   &1] += idet->filter_line(cur-refs, prev, cur+refs, w);
             alpha[(y^1)&1] += idet->filter_line(cur-refs, next, cur+refs, w);
             delta          += idet->filter_line(cur-refs,  cur, cur+refs, w);
         }
     }
 
31fdf306
     if      (alpha[0] > idet->interlace_threshold * alpha[1]){
ca2b450c
         type = TFF;
31fdf306
     }else if(alpha[1] > idet->interlace_threshold * alpha[0]){
ca2b450c
         type = BFF;
31fdf306
     }else if(alpha[1] > idet->progressive_threshold * delta){
ca2b450c
         type = PROGRSSIVE;
     }else{
         type = UNDETERMINED;
     }
 
fa1f92a4
     memmove(idet->history+1, idet->history, HIST_SIZE-1);
     idet->history[0] = type;
     best_type = UNDETERMINED;
     for(i=0; i<HIST_SIZE; i++){
         if(idet->history[i] != UNDETERMINED){
             if(best_type == UNDETERMINED)
                 best_type = idet->history[i];
 
             if(idet->history[i] == best_type) {
                 match++;
             }else{
                 match=0;
                 break;
             }
         }
     }
     if(idet->last_type == UNDETERMINED){
         if(match  ) idet->last_type = best_type;
     }else{
         if(match>2) idet->last_type = best_type;
     }
1a5c08ee
 
     if      (idet->last_type == TFF){
a05a44e2
         idet->cur->top_field_first = 1;
         idet->cur->interlaced_frame = 1;
1a5c08ee
     }else if(idet->last_type == BFF){
a05a44e2
         idet->cur->top_field_first = 0;
         idet->cur->interlaced_frame = 1;
1a5c08ee
     }else if(idet->last_type == PROGRSSIVE){
a05a44e2
         idet->cur->interlaced_frame = 0;
e3e89b6d
     }
0477254d
 
4d4df310
     idet->prestat [           type] ++;
     idet->poststat[idet->last_type] ++;
c59e73d2
     av_log(ctx, AV_LOG_DEBUG, "Single frame:%s, Multi frame:%s\n", type2str(type), type2str(idet->last_type));
e3e89b6d
 }
 
a05a44e2
 static int filter_frame(AVFilterLink *link, AVFrame *picref)
e3e89b6d
 {
     AVFilterContext *ctx = link->dst;
     IDETContext *idet = ctx->priv;
 
     if (idet->prev)
a05a44e2
         av_frame_free(&idet->prev);
e3e89b6d
     idet->prev = idet->cur;
     idet->cur  = idet->next;
     idet->next = picref;
 
     if (!idet->cur)
88beb2df
         return 0;
e3e89b6d
 
     if (!idet->prev)
a05a44e2
         idet->prev = av_frame_clone(idet->cur);
e3e89b6d
 
     if (!idet->csp)
79393a83
         idet->csp = av_pix_fmt_desc_get(link->format);
e3e89b6d
     if (idet->csp->comp[0].depth_minus1 / 8 == 1)
         idet->filter_line = (void*)filter_line_c_16bit;
 
     filter(ctx);
 
a05a44e2
     return ff_filter_frame(ctx->outputs[0], av_frame_clone(idet->cur));
e3e89b6d
 }
 
 static av_cold void uninit(AVFilterContext *ctx)
 {
     IDETContext *idet = ctx->priv;
 
4d4df310
     av_log(ctx, AV_LOG_INFO, "Single frame detection: TFF:%d BFF:%d Progressive:%d Undetermined:%d\n",
ca2b450c
            idet->prestat[TFF],
            idet->prestat[BFF],
            idet->prestat[PROGRSSIVE],
            idet->prestat[UNDETERMINED]
6fb35dba
     );
4d4df310
     av_log(ctx, AV_LOG_INFO, "Multi frame detection: TFF:%d BFF:%d Progressive:%d Undetermined:%d\n",
            idet->poststat[TFF],
            idet->poststat[BFF],
            idet->poststat[PROGRSSIVE],
            idet->poststat[UNDETERMINED]
     );
6fb35dba
 
a05a44e2
     av_frame_free(&idet->prev);
     av_frame_free(&idet->cur );
     av_frame_free(&idet->next);
e3e89b6d
 }
 
 static int query_formats(AVFilterContext *ctx)
 {
ac627b3d
     static const enum AVPixelFormat pix_fmts[] = {
         AV_PIX_FMT_YUV420P,
         AV_PIX_FMT_YUV422P,
         AV_PIX_FMT_YUV444P,
         AV_PIX_FMT_YUV410P,
         AV_PIX_FMT_YUV411P,
         AV_PIX_FMT_GRAY8,
         AV_PIX_FMT_YUVJ420P,
         AV_PIX_FMT_YUVJ422P,
         AV_PIX_FMT_YUVJ444P,
         AV_NE( AV_PIX_FMT_GRAY16BE, AV_PIX_FMT_GRAY16LE ),
         AV_PIX_FMT_YUV440P,
         AV_PIX_FMT_YUVJ440P,
         AV_NE( AV_PIX_FMT_YUV420P10BE, AV_PIX_FMT_YUV420P10LE ),
         AV_NE( AV_PIX_FMT_YUV422P10BE, AV_PIX_FMT_YUV422P10LE ),
         AV_NE( AV_PIX_FMT_YUV444P10BE, AV_PIX_FMT_YUV444P10LE ),
         AV_NE( AV_PIX_FMT_YUV420P16BE, AV_PIX_FMT_YUV420P16LE ),
         AV_NE( AV_PIX_FMT_YUV422P16BE, AV_PIX_FMT_YUV422P16LE ),
         AV_NE( AV_PIX_FMT_YUV444P16BE, AV_PIX_FMT_YUV444P16LE ),
         AV_PIX_FMT_YUVA420P,
         AV_PIX_FMT_NONE
e3e89b6d
     };
 
c9e183b4
     ff_set_common_formats(ctx, ff_make_format_list(pix_fmts));
e3e89b6d
 
     return 0;
 }
 
c63e4e65
 static int config_output(AVFilterLink *outlink)
 {
     outlink->flags |= FF_LINK_FLAG_REQUEST_LOOP;
     return 0;
 }
 
fd6228e6
 static av_cold int init(AVFilterContext *ctx)
e3e89b6d
 {
     IDETContext *idet = ctx->priv;
 
1a5c08ee
     idet->last_type = UNDETERMINED;
fa1f92a4
     memset(idet->history, UNDETERMINED, HIST_SIZE);
1a5c08ee
 
e3e89b6d
     idet->filter_line = filter_line_c;
 
     return 0;
 }
 
 
2d9d4440
 static const AVFilterPad idet_inputs[] = {
     {
         .name         = "default",
         .type         = AVMEDIA_TYPE_VIDEO,
4cd40ef3
         .filter_frame = filter_frame,
2d9d4440
     },
     { NULL }
 };
 
 static const AVFilterPad idet_outputs[] = {
     {
         .name          = "default",
         .type          = AVMEDIA_TYPE_VIDEO,
c63e4e65
         .config_props  = config_output,
2d9d4440
     },
     { NULL }
 };
 
e3e89b6d
 AVFilter avfilter_vf_idet = {
     .name          = "idet",
     .description   = NULL_IF_CONFIG_SMALL("Interlace detect Filter."),
 
     .priv_size     = sizeof(IDETContext),
     .init          = init,
     .uninit        = uninit,
     .query_formats = query_formats,
2d9d4440
     .inputs        = idet_inputs,
     .outputs       = idet_outputs,
43cbd440
     .priv_class    = &idet_class,
e3e89b6d
 };