mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-07 16:05:47 +00:00
6b55d8f1eb
Original commit message from CVS: * testsuite/gst-lint: Check for non-statically scoped parent_class variables. This won't be a problem once plugins are loaded with RTLD_LOCAL.
571 lines
12 KiB
Perl
Executable file
571 lines
12 KiB
Perl
Executable file
#!/usr/bin/perl -w
|
|
# vi: set ts=4:
|
|
#
|
|
|
|
#
|
|
# GStreamer developers: please add comments on any tests you think
|
|
# are dumb or have too many false positives.
|
|
#
|
|
|
|
#
|
|
# Future ideas:
|
|
# - spell check comments
|
|
# - check each function for at least one assertion (?)
|
|
# - check parameters that init/set/get have consistent types
|
|
# - check for gst_caps_set() without check for writeability
|
|
# - check .so files for stray symbols
|
|
#
|
|
|
|
#
|
|
# Random "other" testing ideas
|
|
# - load each plugin individually
|
|
#
|
|
|
|
sub check_copyright();
|
|
sub check_license();
|
|
sub check_buffer_alloc();
|
|
sub check_bad_includes();
|
|
sub check_begin_decls();
|
|
sub check_c99_comments();
|
|
sub check_carriage_returns();
|
|
sub check_printf_lld();
|
|
sub check_glibisms();
|
|
sub check_indentation();
|
|
sub check_no_ignore();
|
|
sub check_deprecated();
|
|
sub check_config_h();
|
|
sub check_varargs_functions();
|
|
sub check_debugging();
|
|
sub check_old_typefind();
|
|
sub check_bad_casts();
|
|
sub check_old_plugin();
|
|
sub check_signal_new();
|
|
sub check_gnuc_const();
|
|
sub check_caps();
|
|
sub check_lib_deprecated();
|
|
sub check_typo();
|
|
sub check_explicit_caps();
|
|
sub check_signals();
|
|
sub check_gettext();
|
|
sub check_padtemplate();
|
|
sub check_parent_class();
|
|
|
|
sub m_check_plugindir();
|
|
sub m_check_interfaces();
|
|
|
|
open FIND, "find . -name \"*.[ch]\" -print|";
|
|
|
|
foreach $filename (<FIND>) {
|
|
chomp $filename;
|
|
open FILE, "$filename";
|
|
@lines = <FILE>;
|
|
close FILE;
|
|
|
|
print "I: $filename\n";
|
|
|
|
# important stuff
|
|
check_bad_includes();
|
|
check_printf_lld();
|
|
check_no_ignore();
|
|
check_deprecated();
|
|
check_config_h();
|
|
check_old_typefind();
|
|
check_old_plugin();
|
|
check_caps();
|
|
check_lib_deprecated();
|
|
check_typo();
|
|
check_glibisms();
|
|
check_explicit_caps();
|
|
check_signals();
|
|
check_gettext();
|
|
check_padtemplate();
|
|
check_parent_class();
|
|
|
|
# less important stuff
|
|
check_license();
|
|
|
|
if (0) {
|
|
check_copyright();
|
|
|
|
check_gnuc_const();
|
|
check_begin_decls();
|
|
check_buffer_alloc();
|
|
check_c99_comments();
|
|
check_carriage_returns();
|
|
#check_indentation();
|
|
check_varargs_functions();
|
|
check_debugging();
|
|
check_bad_casts();
|
|
check_signal_new();
|
|
}
|
|
}
|
|
|
|
open FIND, "find . -name \"Makefile.am\" -print|";
|
|
|
|
foreach $filename (<FIND>) {
|
|
chomp $filename;
|
|
open FILE, "$filename";
|
|
@lines = <FILE>;
|
|
close FILE;
|
|
|
|
print "I: $filename\n";
|
|
|
|
m_check_plugindir();
|
|
m_check_interfaces();
|
|
}
|
|
|
|
#
|
|
# Every source file must have a copyright block
|
|
#
|
|
sub check_copyright()
|
|
{
|
|
if (! grep { /copyright/i; } @lines) {
|
|
print "E: no copyright block\n";
|
|
}
|
|
}
|
|
|
|
#
|
|
# Every source file should have a license statement
|
|
#
|
|
sub check_license()
|
|
{
|
|
if (grep { /Lesser General Public License/; } @lines) {
|
|
print "I: license is LGPL\n";
|
|
} elsif (grep { /Library General Public License/; } @lines) {
|
|
print "I: license is LGPL\n";
|
|
print "W: copyright header uses \"Library\" LGPL\n";
|
|
} elsif (grep { /General Public License/; } @lines) {
|
|
print "I: license is GPL\n";
|
|
} else {
|
|
print "E: unknown license or no copyright block\n";
|
|
}
|
|
}
|
|
|
|
#
|
|
# Suggest usage of gst_buffer_new_and_alloc()
|
|
#
|
|
sub check_buffer_alloc()
|
|
{
|
|
my $n = 0;
|
|
my $lineno = 1;
|
|
|
|
foreach $line (@lines){
|
|
if($line =~ /gst_buffer_new/){
|
|
$n=5;
|
|
}
|
|
if($n>0 && $line =~ /malloc/){
|
|
print "W: ($lineno) gst_buffer_new() followed by malloc(), suggest gst_buffer_new_and_alloc()\n";
|
|
return;
|
|
}
|
|
$n--;
|
|
$lineno++;
|
|
}
|
|
}
|
|
|
|
sub check_bad_includes()
|
|
{
|
|
#
|
|
# malloc.h is non-standard (and probably not what is indended)
|
|
#
|
|
if (grep { /^#include\s+<malloc.h>/; } @lines) {
|
|
print "E: bad header: malloc.h\n"
|
|
}
|
|
}
|
|
|
|
sub check_begin_decls()
|
|
{
|
|
#
|
|
# Prefer "G_BEGIN_DECLS" to 'extern "C" {'
|
|
#
|
|
if($filename =~ /\.h$/){
|
|
if (grep { /extern\s*\"C\"\s*/; } @lines) {
|
|
print "W: extern \"C\" { should be changed to G_BEGIN_DECLS,G_END_DECLS\n";
|
|
}elsif (!grep { /G_BEGIN_DECLS/; } @lines) {
|
|
print "E: header doesn't use G_BEGIN_DECLS\n";
|
|
}
|
|
}
|
|
}
|
|
|
|
#
|
|
# Prefer c89-style comments
|
|
#
|
|
sub check_c99_comments()
|
|
{
|
|
if (grep { /\/\//; } @lines) {
|
|
print "W: //-style comments should be converted to /* */\n"
|
|
}
|
|
}
|
|
|
|
#
|
|
# DOS end-of-line characters are just wrong
|
|
#
|
|
sub check_carriage_returns()
|
|
{
|
|
if (grep { /\r/; } @lines) {
|
|
print "E: source has carriage returns (DOS-style files)\n"
|
|
}
|
|
}
|
|
|
|
#
|
|
# Many uses of %lld are wrong. This could have a lot of false-positives
|
|
#
|
|
sub check_printf_lld()
|
|
{
|
|
if (grep { /\".*\%\d*ll[du].*\"/; } @lines) {
|
|
print "W: Possible \%lld or \%llu in printf format\n"
|
|
}
|
|
}
|
|
|
|
#
|
|
# Glib functions are preferred
|
|
#
|
|
sub check_glibisms()
|
|
{
|
|
if (grep { /\bcalloc\s*\(/; } @lines) {
|
|
print "E: use g_malloc0() instead of calloc()\n"
|
|
}
|
|
if (grep { /\bfree\s*\(/; } @lines) {
|
|
print "E: use g_free() instead of free()\n"
|
|
}
|
|
if (grep { /\bmalloc\s*\(/; } @lines) {
|
|
print "E: use g_malloc() instead of malloc()\n"
|
|
}
|
|
if (grep { /\bprintf\s*\(/; } @lines) {
|
|
print "E: use g_print() instead of printf()\n"
|
|
}
|
|
if (grep { /\brealloc\s*\(/; } @lines) {
|
|
print "E: use g_realloc() instead of realloc()\n"
|
|
}
|
|
if (grep { /^#include\s+<ctype.h>/; } @lines) {
|
|
print "E: ctype.h functions are not locale-independent and don't work in UTF-8 locales. Use g_ascii_is*()\n"
|
|
}
|
|
}
|
|
|
|
#
|
|
# I don't think that indentation necessarily needs to be fixed, since
|
|
# it causes problems with patching and cvs annotate.
|
|
#
|
|
# This takes forever and isn't very useful
|
|
#
|
|
sub check_indentation()
|
|
{
|
|
my $changed_lines;
|
|
my $percent;
|
|
|
|
`indent -br -bad -cbi0 -cli2 -bls -l80 -ut -ce $filename -o .check_plugin.tmp`;
|
|
$changed_lines = `diff $filename .check_plugin.tmp | grep '^>' | wc -l`;
|
|
`rm -f .check_plugin.tmp`;
|
|
|
|
$percent = int(100 * $changed_lines / $#lines);
|
|
|
|
if($percent < 10){
|
|
print "I: indent changed $percent % of the lines\n";
|
|
}elsif($percent <20){
|
|
print "W: indent changed $percent % of the lines\n";
|
|
}else{
|
|
print "E: indent changed $percent % of the lines\n";
|
|
}
|
|
}
|
|
|
|
|
|
#
|
|
# Check (roughly) for functions whose value should not be ignored
|
|
#
|
|
sub check_no_ignore()
|
|
{
|
|
if (grep { /^\s+gst_buffer_merge\s*\(/; } @lines) {
|
|
print "E: return value of gst_buffer_merge () possibly ignored\n";
|
|
}
|
|
if (grep { /^\s+malloc\s*\(/; } @lines) {
|
|
print "E: return value of malloc() possibly ignored\n";
|
|
}
|
|
if (grep { /^\s+gst_buffer_new\s*\(/; } @lines) {
|
|
print "E: return value of gst_buffer_new() possibly ignored\n";
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check for some deprecated stuff (that _shouldn't_ be around anymore)
|
|
#
|
|
sub check_deprecated()
|
|
{
|
|
#
|
|
# Check for old GST_DEBUG() usage
|
|
# (none found)
|
|
#
|
|
if (grep { /GST_DEBUG\s*\(\s+\d/; } @lines) {
|
|
print "E: old-style GST_DEBUG()\n";
|
|
}
|
|
if (grep { /GST_INFO\s*\(\s+\d/; } @lines) {
|
|
print "E: old-style GST_DEBUG()\n";
|
|
}
|
|
if (grep { /GstEventFlags/; } @lines) {
|
|
print "W: who uses GstEventFlags\n";
|
|
}
|
|
if (grep { /g_type_class_ref/ } @lines) {
|
|
print "W: g_type_class_ref should be changed to g_type_class_peek_parent\n";
|
|
}
|
|
|
|
}
|
|
|
|
#
|
|
# Every .c file should include config.h before any other headers
|
|
# No .h file should include config.h
|
|
#
|
|
sub check_config_h()
|
|
{
|
|
if($filename =~ /\.c$/){
|
|
#
|
|
# config.h should be wrapped
|
|
#
|
|
my @includes = grep { /^#include/; } @lines;
|
|
|
|
if (!grep { /^#include\s+["<]config.h[">]/; } @includes) {
|
|
print "E: #include <config.h> missing\n";
|
|
}else{
|
|
if (!($includes[0] =~ /^#include\s+["<]config.h[">]/)){
|
|
print "E: #include <config.h> is not first include\n";
|
|
}
|
|
if(!grep { /^#ifdef HAVE_CONFIG_H/; } @lines) {
|
|
print "E: #include <config.h> not surrounded by #ifdef HAVE_CONFIG_H\n";
|
|
}
|
|
}
|
|
}
|
|
|
|
if($filename =~ /\.h$/){
|
|
if (grep { /^#include\s+["<]config.h[">]/; } @lines) {
|
|
print "E: headers should not #include <config.h>\n";
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
#
|
|
# Check for functions that take varargs to make sure they are
|
|
# named correctly
|
|
#
|
|
sub check_varargs_functions()
|
|
{
|
|
if($filename =~ /\.h$/){
|
|
if (grep { /varargs/; } @lines) {
|
|
print "I: has varargs\n";
|
|
}
|
|
}
|
|
}
|
|
|
|
#
|
|
# Debugging checks
|
|
#
|
|
sub check_debugging()
|
|
{
|
|
if (grep { /\Wg_print\W/ || /\Wprintf\W/ && /\Wfprintf\W/; } @lines) {
|
|
print "W: friendly libraries don't print to stdio or stderr\n";
|
|
}
|
|
|
|
if (grep { /GST_DEBUG.*\\n"/; } @lines) {
|
|
print "W: possible newline in GST_DEBUG()\n";
|
|
}
|
|
|
|
}
|
|
|
|
#
|
|
# check for plugindir=
|
|
#
|
|
sub m_check_plugindir()
|
|
{
|
|
if (grep { /plugindir\s*=/; } @lines) {
|
|
print "E: plugindir= is no longer necessary\n";
|
|
}
|
|
}
|
|
|
|
#
|
|
# check for old typefinding code
|
|
#
|
|
sub check_old_typefind()
|
|
{
|
|
if (grep { /GstTypeDefinition/ || /GstTypeFactory/ } @lines) {
|
|
print "E: old typefind interface has been removed\n";
|
|
}
|
|
}
|
|
|
|
#
|
|
# check for casts that we've deemed incorrect (fix the prototype)
|
|
#
|
|
sub check_bad_casts()
|
|
{
|
|
if (grep { /GBaseInitFunc/ || /GBaseFinalizeFunc/ ||
|
|
/GClassInitFunc/ || /GClassFinalizeFunc/ ||
|
|
/GInstanceInitFunc/ || /GInterfaceInitFunc/ ||
|
|
/GInterfaceFinalizeFunc/ } @lines) {
|
|
print "W: bad casts (fix prototype)\n";
|
|
}
|
|
if (grep { /\(\s*Gst[A-Z][A-Za-z]*\s*\*\s*\)/ } @lines ) {
|
|
print "W: use GST_XXX() instead of (GstXxx *)\n";
|
|
}
|
|
|
|
}
|
|
|
|
#
|
|
# check for old plugin code
|
|
#
|
|
sub check_old_plugin()
|
|
{
|
|
if (grep { /plugin_init.*GModule.*GstPlugin/ } @lines) {
|
|
print "E: old plugin interface detected\n";
|
|
}
|
|
if (grep { /GstPluginDesc.*plugin_desc/ } @lines) {
|
|
print "W: should use GST_PLUGIN_DEFINE() instead of GstPluginDesc\n";
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check for calls to g_signal_new() with a callback type of G_TYPE_POINTER
|
|
#
|
|
sub check_signal_new()
|
|
{
|
|
my $n = 0;
|
|
my $lineno = 1;
|
|
|
|
foreach $line (@lines){
|
|
if($line =~ /g_signal_new/){
|
|
$n=5;
|
|
}
|
|
if($n>0 && $line =~ /G_TYPE_POINTER/){
|
|
print "W: ($lineno) g_signal_new() with callback type of G_TYPE_POINTER. Register and use a boxed type instead.\n";
|
|
return;
|
|
}
|
|
$n--;
|
|
$lineno++;
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check that libgstinterfaces is in LDADD
|
|
#
|
|
sub m_check_interfaces()
|
|
{
|
|
if (grep { /libgstinterfaces.la/ } @lines) {
|
|
if (! grep { /libgstinterfaces_la/ } @lines) {
|
|
if (! grep { /_LDADD.*libgstinterfaces.la/ } @lines) {
|
|
print "E: libgstinterfaces.la not in LDADD\n";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check that get_type() functions return G_CONST_RETURN GType
|
|
#
|
|
sub check_gnuc_const()
|
|
{
|
|
my $n = 0;
|
|
my $lineno = 1;
|
|
|
|
foreach $line (@lines){
|
|
if($line =~ /GType.*get_type.*/ &&
|
|
!($line =~ /GType.*get_type.*G_GNUC_CONST/)) {
|
|
|
|
print "E: get_type function does not have G_GNUC_CONST attribute\n";
|
|
}
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check caps usage
|
|
#
|
|
sub check_caps()
|
|
{
|
|
if (grep { /gst_pad_get_caps/ } @lines) {
|
|
print "E: elements should not call gst_pad_get_caps(), use gst_pad_get_allowed_caps()\n";
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check for use of deprecated functions
|
|
#
|
|
sub check_lib_deprecated()
|
|
{
|
|
if (grep { /bzero/ } @lines) {
|
|
print "E: change bzero() to memset()\n";
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check for typos
|
|
#
|
|
sub check_typo()
|
|
{
|
|
if (grep { /;\s*;\s*$/ } @lines) {
|
|
print "W: typo? \";;\"\n";
|
|
}
|
|
}
|
|
|
|
#
|
|
# set_explicit_caps() should preceed pad_add()
|
|
#
|
|
sub check_explicit_caps()
|
|
{
|
|
my $n = 0;
|
|
my $lineno = 1;
|
|
|
|
foreach $line (@lines){
|
|
if($line =~ /gst_element_add_pad/){
|
|
$n=10;
|
|
}
|
|
if($n>0 && $line =~ /gst_pad_set_explicit_caps/){
|
|
print "W: ($lineno) explicit caps should be set before adding pad\n";
|
|
return;
|
|
}
|
|
$n--;
|
|
$lineno++;
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check for - in signal names
|
|
#
|
|
sub check_signals()
|
|
{
|
|
if (grep { /g_signal_new.*\".*-.*\"/; } @lines) {
|
|
print "E: g_signal_new() with a signal name with a - in it (we prefer _)\n"
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check for things that gettext gets wrong
|
|
#
|
|
sub check_gettext()
|
|
{
|
|
if (grep { /\b_\(.*G_GU?INT64_FORMAT/ ||
|
|
/\b_\(.*GST_TIME_FORMAT/ ||
|
|
/\b_\(.*GST_FOURCC_FORMAT/ } @lines) {
|
|
print "E: gettext doesn't handle format strings that are defines\n"
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check that pad templates are statically scoped
|
|
#
|
|
sub check_padtemplate()
|
|
{
|
|
foreach $line (@lines){
|
|
if ($line =~ /GstStaticPadTemplate/ && !($line =~ /static/)) {
|
|
print "W: pad template definitions should be static\n";
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
#
|
|
# Check that parent_class is statically scoped
|
|
#
|
|
sub check_parent_class()
|
|
{
|
|
foreach $line (@lines){
|
|
if ($line =~ /Gst.*\*\s*parent_class/ && !($line =~ /static/)) {
|
|
print "E: parent_class definitions should be static\n";
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|