mirror of https://github.com/FFmpeg/FFmpeg.git
164 lines
4.5 KiB
164 lines
4.5 KiB
#!/usr/bin/env perl |
|
|
|
# Copyright (c) 2007-2013 Stefano Sabatini |
|
# |
|
# 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 |
|
|
|
=head1 NAME |
|
|
|
plotframes - Plot video frame sizes using ffprobe and gnuplot |
|
|
|
=head1 SYNOPSIS |
|
|
|
plotframes [I<options>] [I<input>] |
|
|
|
=head1 DESCRIPTION |
|
|
|
plotframes reads a multimedia files with ffprobe, and plots the |
|
collected video sizes with gnuplot. |
|
|
|
=head1 OPTIONS |
|
|
|
=over 4 |
|
|
|
=item B<--input|-i> I<infile> |
|
|
|
Specify multimedia file to read. This is the file passed to the |
|
ffprobe command. If not specified it is the first argument passed to |
|
the script. |
|
|
|
=item B<--help|--usage|-h|-?> |
|
|
|
Print a brief help message and exit. |
|
|
|
=item B<--manpage|-m> |
|
|
|
Print the man page. |
|
|
|
=item B<--output|-o> I<outfile> |
|
|
|
Set the name of the output used by gnuplot. If not specified no output |
|
is created. Must be used in conjunction with the B<terminal> option. |
|
|
|
=item B<--stream|--s> I<stream_specifier> |
|
|
|
Specify stream. The value must be a string containing a stream |
|
specifier. Default value is "v". |
|
|
|
=item B<--terminal|-t> I<terminal> |
|
|
|
Set the name of the terminal used by gnuplot. By default it is |
|
"x11". Must be used in conjunction with the B<output> option. Check |
|
the gnuplot manual for the valid values. |
|
|
|
=back |
|
|
|
=cut |
|
|
|
=head1 SEE ALSO |
|
|
|
ffprobe(1), gnuplot(1) |
|
|
|
=cut |
|
|
|
use warnings; |
|
use strict; |
|
|
|
use File::Temp; |
|
use JSON -support_by_pp; |
|
use Getopt::Long; |
|
use Pod::Usage; |
|
|
|
my $input = $ARGV[0]; |
|
my $stream_specifier = "v"; |
|
my $gnuplot_terminal = "x11"; |
|
my $gnuplot_output; |
|
|
|
GetOptions ( |
|
'input|i=s' => \$input, |
|
'help|usage|?|h' => sub { pod2usage ( { -verbose => 1, -exitval => 0 }) }, |
|
'manpage|m' => sub { pod2usage ( { -verbose => 2, -exitval => 0 }) }, |
|
'stream|s=s' => \$stream_specifier, |
|
'terminal|t=s' => \$gnuplot_terminal, |
|
'output|o=s' => \$gnuplot_output, |
|
) or pod2usage( { -message=> "Parsing error", -verbose => 1, -exitval => 1 }); |
|
|
|
die "You must specify an input file\n" unless $input; |
|
|
|
# fetch data |
|
my @cmd = (qw{ffprobe -show_entries frame -select_streams}, $stream_specifier, "-of", "json", $input); |
|
print STDERR "Executing command: @cmd\n"; |
|
my $json_struct; |
|
{ |
|
open(FH, "-|", @cmd) or die "ffprobe command failed: $!\n"; |
|
local $/; |
|
my $json_text = <FH>; |
|
close FH; |
|
die "ffprobe command failed" if $?; |
|
eval { $json_struct = decode_json($json_text); }; |
|
die "JSON parsing error: $@\n" if $@; |
|
} |
|
|
|
# collect and print frame statistics per pict_type |
|
my %stats; |
|
my $frames = $json_struct->{frames}; |
|
my $frame_count = 0; |
|
foreach my $frame (@{$frames}) { |
|
my $type = $frame->{pict_type}; |
|
$frame->{count} = $frame_count++; |
|
if (not $stats{$type}) { |
|
$stats{$type}->{tmpfile} = File::Temp->new(SUFFIX => '.dat'); |
|
my $fn = $stats{$type}->{tmpfile}->filename; |
|
open($stats{$type}->{fh}, ">", $fn) or die "Can't open $fn"; |
|
} |
|
|
|
print { $stats{$type}->{fh} } |
|
"$frame->{count} ", $frame->{pkt_size} * 8 / 1000, "\n"; |
|
} |
|
foreach (keys %stats) { close $stats{$_}->{fh}; } |
|
|
|
# write gnuplot script |
|
my %type_color_map = ( |
|
"I" => "red", |
|
"P" => "green", |
|
"B" => "blue" |
|
); |
|
|
|
my $gnuplot_script_tmpfile = File::Temp->new(SUFFIX => '.gnuplot'); |
|
my $fn = $gnuplot_script_tmpfile->filename; |
|
open(FH, ">", $fn) or die "Couldn't open $fn: $!"; |
|
print FH << "EOF"; |
|
set title "video frame sizes" |
|
set xlabel "frame time" |
|
set ylabel "frame size (Kbits)" |
|
set grid |
|
set terminal "$gnuplot_terminal" |
|
EOF |
|
|
|
print FH "set output \"$gnuplot_output\"\n" if $gnuplot_output; |
|
print FH "plot"; |
|
my $sep = ""; |
|
foreach my $type (keys %stats) { |
|
my $fn = $stats{$type}->{tmpfile}->filename; |
|
print FH "$sep\"$fn\" title \"$type frames\" with impulses"; |
|
print FH " linecolor rgb \"$type_color_map{$type}\"" if $type_color_map{$type}; |
|
$sep = ", "; |
|
} |
|
close FH; |
|
|
|
# launch gnuplot with the generated script |
|
system ("gnuplot", "--persist", $gnuplot_script_tmpfile->filename);
|
|
|