• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2013 Paul B Mahol
3  *
4  * This file is part of FFmpeg.
5  *
6  * FFmpeg is free software; you can redistribute it and/or
7  * modify it under the terms of the GNU Lesser General Public
8  * License as published by the Free Software Foundation; either
9  * version 2.1 of the License, or (at your option) any later version.
10  *
11  * FFmpeg is distributed in the hope that it will be useful,
12  * but WITHOUT ANY WARRANTY; without even the implied warranty of
13  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
14  * Lesser General Public License for more details.
15  *
16  * You should have received a copy of the GNU Lesser General Public
17  * License along with FFmpeg; if not, write to the Free Software
18  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
19  */
20 
21 #include "libavutil/imgutils.h"
22 #include "libavutil/opt.h"
23 #include "libavutil/pixdesc.h"
24 #include "avfilter.h"
25 #include "internal.h"
26 
27 typedef struct WeaveContext {
28     const AVClass *class;
29     int first_field;
30     int double_weave;
31     int nb_planes;
32     int planeheight[4];
33     int linesize[4];
34 
35     AVFrame *prev;
36 } WeaveContext;
37 
38 #define OFFSET(x) offsetof(WeaveContext, x)
39 #define FLAGS AV_OPT_FLAG_FILTERING_PARAM|AV_OPT_FLAG_VIDEO_PARAM
40 
41 static const AVOption weave_options[] = {
42     { "first_field", "set first field", OFFSET(first_field), AV_OPT_TYPE_INT,   {.i64=0}, 0, 1, FLAGS, "field"},
43         { "top",     "set top field first",               0, AV_OPT_TYPE_CONST, {.i64=0}, 0, 0, FLAGS, "field"},
44         { "t",       "set top field first",               0, AV_OPT_TYPE_CONST, {.i64=0}, 0, 0, FLAGS, "field"},
45         { "bottom",  "set bottom field first",            0, AV_OPT_TYPE_CONST, {.i64=1}, 0, 0, FLAGS, "field"},
46         { "b",       "set bottom field first",            0, AV_OPT_TYPE_CONST, {.i64=1}, 0, 0, FLAGS, "field"},
47     { NULL }
48 };
49 
50 AVFILTER_DEFINE_CLASS(weave);
51 
query_formats(AVFilterContext * ctx)52 static int query_formats(AVFilterContext *ctx)
53 {
54     AVFilterFormats *formats = NULL;
55     int ret;
56 
57     ret = ff_formats_pixdesc_filter(&formats, 0,
58                                     AV_PIX_FMT_FLAG_PAL |
59                                     AV_PIX_FMT_FLAG_HWACCEL);
60     if (ret < 0)
61         return ret;
62     return ff_set_common_formats(ctx, formats);
63 }
64 
config_props_output(AVFilterLink * outlink)65 static int config_props_output(AVFilterLink *outlink)
66 {
67     AVFilterContext *ctx = outlink->src;
68     WeaveContext *s = ctx->priv;
69     AVFilterLink *inlink = ctx->inputs[0];
70     const AVPixFmtDescriptor *desc = av_pix_fmt_desc_get(inlink->format);
71     int ret;
72 
73     if (!s->double_weave) {
74         outlink->time_base.num = inlink->time_base.num * 2;
75         outlink->time_base.den = inlink->time_base.den;
76         outlink->frame_rate.num = inlink->frame_rate.num;
77         outlink->frame_rate.den = inlink->frame_rate.den * 2;
78     }
79     outlink->w = inlink->w;
80     outlink->h = inlink->h * 2;
81 
82     if ((ret = av_image_fill_linesizes(s->linesize, inlink->format, inlink->w)) < 0)
83         return ret;
84 
85     s->planeheight[1] = s->planeheight[2] = AV_CEIL_RSHIFT(inlink->h, desc->log2_chroma_h);
86     s->planeheight[0] = s->planeheight[3] = inlink->h;
87 
88     s->nb_planes = av_pix_fmt_count_planes(inlink->format);
89 
90     return 0;
91 }
92 
93 typedef struct ThreadData {
94     AVFrame *in, *out;
95 } ThreadData;
96 
weave_slice(AVFilterContext * ctx,void * arg,int jobnr,int nb_jobs)97 static int weave_slice(AVFilterContext *ctx, void *arg, int jobnr, int nb_jobs)
98 {
99     AVFilterLink *inlink = ctx->inputs[0];
100     WeaveContext *s = ctx->priv;
101     ThreadData *td = arg;
102     AVFrame *in = td->in;
103     AVFrame *out = td->out;
104 
105     const int weave = (s->double_weave && !(inlink->frame_count_out & 1));
106     const int field1 = weave ? s->first_field : (!s->first_field);
107     const int field2 = weave ? (!s->first_field) : s->first_field;
108 
109     for (int i = 0; i < s->nb_planes; i++) {
110         const int height = s->planeheight[i];
111         const int start = (height * jobnr) / nb_jobs;
112         const int end = (height * (jobnr+1)) / nb_jobs;
113 
114         av_image_copy_plane(out->data[i] + out->linesize[i] * field1 +
115                             out->linesize[i] * start * 2,
116                             out->linesize[i] * 2,
117                             in->data[i] + start * in->linesize[i],
118                             in->linesize[i],
119                             s->linesize[i], end - start);
120         av_image_copy_plane(out->data[i] + out->linesize[i] * field2 +
121                             out->linesize[i] * start * 2,
122                             out->linesize[i] * 2,
123                             s->prev->data[i] + start * s->prev->linesize[i],
124                             s->prev->linesize[i],
125                             s->linesize[i], end - start);
126     }
127 
128     return 0;
129 }
130 
filter_frame(AVFilterLink * inlink,AVFrame * in)131 static int filter_frame(AVFilterLink *inlink, AVFrame *in)
132 {
133     AVFilterContext *ctx = inlink->dst;
134     WeaveContext *s = ctx->priv;
135     AVFilterLink *outlink = ctx->outputs[0];
136     ThreadData td;
137     AVFrame *out;
138 
139     if (!s->prev) {
140         s->prev = in;
141         return 0;
142     }
143 
144     out = ff_get_video_buffer(outlink, outlink->w, outlink->h);
145     if (!out) {
146         av_frame_free(&in);
147         av_frame_free(&s->prev);
148         return AVERROR(ENOMEM);
149     }
150     av_frame_copy_props(out, in);
151 
152     td.out = out, td.in = in;
153     ctx->internal->execute(ctx, weave_slice, &td, NULL, FFMIN(s->planeheight[1],
154                                                               ff_filter_get_nb_threads(ctx)));
155 
156     out->pts = s->double_weave ? s->prev->pts : in->pts / 2;
157     out->interlaced_frame = 1;
158     out->top_field_first = !s->first_field;
159 
160     if (!s->double_weave)
161         av_frame_free(&in);
162     av_frame_free(&s->prev);
163     if (s->double_weave)
164         s->prev = in;
165     return ff_filter_frame(outlink, out);
166 }
167 
uninit(AVFilterContext * ctx)168 static av_cold void uninit(AVFilterContext *ctx)
169 {
170     WeaveContext *s = ctx->priv;
171 
172     av_frame_free(&s->prev);
173 }
174 
175 static const AVFilterPad weave_inputs[] = {
176     {
177         .name             = "default",
178         .type             = AVMEDIA_TYPE_VIDEO,
179         .filter_frame     = filter_frame,
180     },
181     { NULL }
182 };
183 
184 static const AVFilterPad weave_outputs[] = {
185     {
186         .name          = "default",
187         .type          = AVMEDIA_TYPE_VIDEO,
188         .config_props  = config_props_output,
189     },
190     { NULL }
191 };
192 
193 AVFilter ff_vf_weave = {
194     .name          = "weave",
195     .description   = NULL_IF_CONFIG_SMALL("Weave input video fields into frames."),
196     .priv_size     = sizeof(WeaveContext),
197     .priv_class    = &weave_class,
198     .query_formats = query_formats,
199     .uninit        = uninit,
200     .inputs        = weave_inputs,
201     .outputs       = weave_outputs,
202     .flags         = AVFILTER_FLAG_SLICE_THREADS,
203 };
204 
init(AVFilterContext * ctx)205 static av_cold int init(AVFilterContext *ctx)
206 {
207     WeaveContext *s = ctx->priv;
208 
209     if (!strcmp(ctx->filter->name, "doubleweave"))
210         s->double_weave = 1;
211 
212     return 0;
213 }
214 
215 #define doubleweave_options weave_options
216 AVFILTER_DEFINE_CLASS(doubleweave);
217 
218 AVFilter ff_vf_doubleweave = {
219     .name          = "doubleweave",
220     .description   = NULL_IF_CONFIG_SMALL("Weave input video fields into double number of frames."),
221     .priv_size     = sizeof(WeaveContext),
222     .priv_class    = &doubleweave_class,
223     .query_formats = query_formats,
224     .init          = init,
225     .uninit        = uninit,
226     .inputs        = weave_inputs,
227     .outputs       = weave_outputs,
228     .flags         = AVFILTER_FLAG_SLICE_THREADS,
229 };
230