2 * vf_fstep.c - filter to ouput only 1 every n frame, or only the I (key)
9 * if you call the filter with I (uppercase) as the parameter
10 * ... -vf framestep=I ...
11 * then ONLY the keyframes are outputted.
12 * For DVD it means, generally, one every 15 frames (IBBPBBPBBPBBPBB), for avi it means
13 * every scene change or every keyint value (see -lavcopts).
15 * if you call the filter with the i (lowercase)
16 * ... -vf framestep=i ...
17 * then a I! followed by a cr is printed when a key frame (eg Intra frame) is
18 * found, leaving the current line of mplayer/mencoder, where you got the
19 * time, in seconds, and frame of the key. Use this information to split the
22 * After the i or alone you can put a positive number and only one frame every
23 * x (the number you set) is passed on the filter chain, limiting the output
27 * ... -vf framestep=i20 ...
28 * Dump one every 20 frames, printing on the console when a I-Frame is encounter.
30 * ... -vf framestep=25
31 * Dump one every 25 frames.
33 * If you call the filter without parameter it does nothing (except using memory
34 * and resource of your system,. of course).
36 * This filter doesn' t work like the option -sstep seconds.
38 * The -sstep seek to the new position, without decoding all frames but,
39 * expecially on avi file coded whith mpeg4 (lavc or xvid or divx), the
40 * seek is not always too much precise.
42 * This filter simply discard the unwanted frames, so you are very precise in
43 * counting the frame but sometime you use a lot of CPU for nothing.
45 * As usual it depends on what you're doing.
47 * Daniele Forghieri ( guru@digitalfantasy.it )
57 #include "cpudetect.h"
59 #include "img_format.h"
63 /* Uncomment if you want to print some info on the format */
64 // #define DUMP_FORMAT_DATA
70 /* Frame output step, 0 = all */
72 /* Only I-Frame (2), print on I-Frame (1) */
77 static int put_image(struct vf_instance_s
* vf
, mp_image_t
*mpi
, double pts
)
80 struct vf_priv_s
*priv
;
85 /* Print the 'I' if is a intra frame. The \n advance the current line so you got the
86 * current file time (in second) and the frame number on the console ;-)
88 if (priv
->dump_iframe
) {
89 if (mpi
->pict_type
== 1) {
90 mp_msg(MSGT_VFILTER
, MSGL_INFO
, "I!\n");
94 /* decide if frame must be shown */
95 if (priv
->dump_iframe
== 2) {
97 skip
= mpi
->pict_type
== 1 ? 0 : 1;
100 /* Only 1 every frame_step */
102 if ((priv
->frame_step
!= 0) && ((priv
->frame_cur
% priv
->frame_step
) != 0)) {
106 /* Increment current frame */
110 /* Get image, export type (we don't modify tghe image) */
111 dmpi
=vf_get_image(vf
->next
, mpi
->imgfmt
,
112 MP_IMGTYPE_EXPORT
, 0,
114 /* Copy only the pointer ( MP_IMGTYPE_EXPORT ! ) */
115 dmpi
->planes
[0] = mpi
->planes
[0];
116 dmpi
->planes
[1] = mpi
->planes
[1];
117 dmpi
->planes
[2] = mpi
->planes
[2];
119 dmpi
->stride
[0] = mpi
->stride
[0];
120 dmpi
->stride
[1] = mpi
->stride
[1];
121 dmpi
->stride
[2] = mpi
->stride
[2];
123 dmpi
->width
= mpi
->width
;
124 dmpi
->height
= mpi
->height
;
126 /* Chain to next filter / output ... */
127 return vf_next_put_image(vf
, dmpi
, pts
);
134 static void uninit(struct vf_instance_s
* vf
)
136 /* Free private data */
140 /* Main entry funct for the filter */
141 static int open(vf_instance_t
*vf
, char* args
)
145 vf
->put_image
= put_image
;
147 vf
->default_reqs
= VFCAP_ACCEPT_STRIDE
;
148 vf
->priv
= p
= calloc(1, sizeof(struct vf_priv_s
));
154 #ifdef DUMP_FORMAT_DATA
161 /* Dump only KEY (ie INTRA) frame */
166 /* Print a 'I!' when a i-frame is encounter */
172 p
->frame_step
= atoi(args
);
173 if (p
->frame_step
<= 0) {
174 mp_msg(MSGT_VFILTER
, MSGL_WARN
, MSGTR_MPCODECS_ErrorParsingArgument
);
183 const vf_info_t vf_info_framestep
= {
184 "Dump one every n / key frames",