]> granicus.if.org Git - check/commitdiff
* add make_macros perl script from somebody (who?)
authorcpickett <cpickett@64e312b2-a51f-0410-8e61-82d0ca0eb02a>
Mon, 23 Feb 2009 03:32:58 +0000 (03:32 +0000)
committercpickett <cpickett@64e312b2-a51f-0410-8e61-82d0ca0eb02a>
Mon, 23 Feb 2009 03:32:58 +0000 (03:32 +0000)
git-svn-id: svn+ssh://svn.code.sf.net/p/check/code/trunk@536 64e312b2-a51f-0410-8e61-82d0ca0eb02a

contrib/make_macros [new file with mode: 0644]

diff --git a/contrib/make_macros b/contrib/make_macros
new file mode 100644 (file)
index 0000000..7f25679
--- /dev/null
@@ -0,0 +1,302 @@
+#!/usr/bin/perl
+
+# Purpose: 
+#   Generates Assert/Fail macros for use with check.
+#
+# Usage:
+#   make_macros macros.in > macros.h
+# 
+# macros.in: one or more lines containing...
+#   Assert
+#   Assert_(|not)(Success|Error|Failure|NULL|True|False|Same)
+#   Fail_(if|unless)(Success|Error|Failure|NULL|True|False|Same)
+#   Assert_(|not)(0[0-7]+|0x[0-9A-Fa-f]+|0b[01]+|[1-9][0-9]+)
+#   Fail_(if|unless)(0[0-7]+|0x[0-9A-Fa-f]+|0b[01]+|[1-9][0-9]+)
+#   Assert_(|not)(LT|LE|EQ|GE|GT|NE)
+#   Fail_(if|unless)(LT|LE|EQ|GE|GT|NE)
+#   Assert_(|not){Type}(LT|LE|EQ|GE|GT|NE)
+#   Fail_(if|unless){Type}(LT|LE|EQ|GE|GT|NE)
+#   {Type}_compare
+#
+# Example:
+#   echo -e 'Assert_notNULL\nAssertStringEQ' | make_macros > macros.h
+#
+#   #include "macros.h"
+#          ...
+#   char *foo;
+#   Assert_notNULL( foo = strdup( "foo" ) , NULL );
+#   Assert_StringEQ( foo, "foo", NULL );
+#
+# author: unknown... if you know, please correct or email check-devel
+
+use warnings;
+use strict;
+
+use Switch 'Perl6';
+use Text::Wrap;
+
+our %need;
+our %include;
+our %macros;
+
+sub make_macro {
+    my ($name, $fh) = @_;
+    my ($condition) = $name;
+    
+    my ($failIf, $failUnless, $assertNot, $assert);
+    for ($condition) {
+        $failIf     = 1, next if s/^ Fail_ if     ([A-Z0-9]?) /$1/x;
+        $failUnless = 1, next if s/^ Fail_ unless ([A-Z0-9]?) /$1/x;
+        $assertNot  = 1, next if s/^ Assert_ not  ([A-Z0-9] ) /$1/x;
+        $assert     = 1, next if s/^ Assert_      (.        ) /$1/x;
+        $assert     = 1, next if s/^ Assert                  $//x;
+        print $fh "#error don't know how to make ${name}\n\n";
+        return;
+    }
+    
+    my ($invert) = 0;
+    foreach ($failIf, $assertNot) {
+       $invert = !$invert if $_;
+    }
+    
+    my $type = "";
+    if ($condition =~ s/(LT|LE|EQ|GE|GT|NE)$//) {
+        $type = $condition;
+        $condition = $1;
+    }
+    $type = "Number" unless length $type;
+    
+    my (@args) = qw( expr );
+    my ($expr) = qw( (expr) );
+    my ($msg);
+    my ($doc);
+    my (%argdocs) = (
+        expr => "The expression to test."
+    );
+    
+    my ($not)  = sub { $invert ? "not " :     "" };
+    my ($Not)  = sub { $invert ?     "" : "not " };
+    my ($EQ)   = sub { $invert ? "!=" : "==" };
+    my ($NE)   = sub { $invert ? "==" : "!=" };
+
+    given ($condition) {
+        when "" {
+            unless ( $invert ) {
+                $doc = "Asserts that \@p expr is true.";
+                $msg = "Assertion '\"#expr\"' failed.";
+            } else {
+                $doc = "Asserts that \@p expr is false.";
+                $msg = "Negative assertion '\"#expr\"' failed";
+            }
+        }
+        when [qw(Error error Failure failure Success success)] {
+            $invert = !$invert when [qw(Success success)];
+            unless ( $invert ) {
+                $doc = "Asserts that \@p expr failed.";
+                $msg = "'\"#expr\"' did not fail";
+            } else {
+                $doc = "Asserts that \@p expr did not fail.";
+                $msg = "'\"#expr\"' failed";
+            }
+            $invert = !$invert;
+        }
+        when [qw(True true False false)] {
+            $invert = !$invert when "false";
+            $doc = "Asserts that \@p expr is ".( $invert ? "false" : "true" ).".";
+            $msg = "'\"#expr\"' was " . ( $invert ? "true" : "false" );
+        }
+        when "NULL" {
+            $msg = "'\"#expr\"' was ".&$Not."NULL";
+            $doc = "Asserts that \@p expr is ".&$not."NULL";
+            $expr = "(expr) ".&$EQ." NULL";
+            $invert = 0;
+            $include{"stddef.h"} = 1;
+        }
+        when m/^(?:[1-9]\d*|0x[0-9A-Fa-f]+|0b[01]+|0[0-7]+)$/x {
+            $msg = "'\"#expr\"' was ".&$Not."equal to $condition";
+            $doc = "Asserts that \@p expr is ".&$not."equal to $condition.";
+            $condition = oct($condition) if $condition =~ /^0/;
+            $expr = "(expr) ".&$EQ." $condition";
+            $invert = 0;
+        }
+        when [qw(GT GE LT LE EQ NE Equal equal)] {
+            my $op;
+            given ( $condition ) {
+              $op = [ "less than"                 , "<"  ] when "LT";
+              $op = [ "less than that or equal to", "<=" ] when "LE";
+              $op = [ "equal to"                  , "==" ] when ["EQ", "Equal", "equal", "NE"];
+              $op = [ "greater than or equal to"  , ">=" ] when "GE";
+              $op = [ "greater than"              , ">"  ] when "GT";
+            };
+            $invert = !$invert when "NE";
+            @args = ( $type."1", $type."2" );
+            %argdocs = (
+                $type."1" => "The expression to test.",
+                $type."2" => "The test value."
+            );
+            $msg = "'\"#$args[0]\"' is ".&$Not.$op->[0]." '\"#$args[1]\"'";
+            $doc = "Asserts that \@p $args[0] is ".&$not.$op->[0]." \@p $args[1].";
+            $expr = "(${type}_compare( $args[0], $args[1] ) ".$op->[1]." 0)";
+            $need{$type."_compare"} = 1;
+        }
+        when [qw(Same same)] {
+            @args = qw( arg1 arg2 );
+            %argdocs = (
+                arg1 => "The expression to test.",
+                arg2 => "The expected value."
+            );
+            $msg = "'\"#arg1\"' and '\"#arg2\"' are ".&$Not."the same";
+            $msg = "Asserts that \@p arg1 and \@p arg2 are ".&$not."the same";
+            $expr = "(arg1) ".&$EQ." (arg2)";
+            $invert = 0;
+        }
+        default {
+            print $fh "#error don't know how to make ${name}\n\n";
+            return;
+        }
+    }
+
+    if ($invert) {
+        $expr = "!$expr";
+    }
+
+    if ($doc) {
+        print $fh "/**\n";
+        print $fh wrap(" * "," * ","$doc\n");
+        my ($maxlength) = 3;
+        foreach (keys %argdocs) {
+            $maxlength = length $_ if length $_ > $maxlength;
+        }
+        my ($indent) = " " x ($maxlength + 1 + 7);
+        foreach (keys %argdocs) {
+            print $fh wrap(" * ",
+                           " * $indent",
+                          "\@param $_ ".(" " x ($maxlength - length $_)).
+                          "$argdocs{$_}\n");
+        }
+        print $fh wrap(" * ",
+                       " * $indent",
+                       "\@param ... ".(" " x ($maxlength - 3)).
+                       "An optional message to indicate the assertion failed; ".
+                       "Omit for a default message.\n");
+
+        print $fh " */\n";
+    }
+    print $fh "#define ${name}(",(join ", ", @args, '...'),") \\\n";
+    print $fh "    _fail_unless( $expr, __FILE__, __LINE__, \\\n";
+    print $fh "                  \"$msg\", ## __VA_ARGS__, NULL)\n";
+    print $fh "\n";
+    
+    $macros{$name} = 1;
+}
+
+sub make_compare {
+    my ($name, $fh) = @_;
+    my ($type) = $name;
+    $type =~ s/_compare$//;
+
+    my ($expr);
+    my ($warning);
+    my ($ord);
+    my (@name) = ( "", "" );
+
+    given ($type) {
+        when "Number" {
+            $expr = "( ${type}1 - ${type}2 )";
+            $ord = "is";
+        }
+        when "String" {
+            $expr = "strcmp( ${type}1, ${type}2 )";
+            $include{"string.h"} = 1;
+            $ord = "is";
+        }
+        default {
+            $expr = "memcmp( ${type}1, ${type}2, sizeof( ${type} ) )";
+            $include{"string.h"} = 1;
+            $ord = "are";
+            @name = ( "'s bytes ", "'s" );
+            $warning = "This compares \@p ${type}1 and \@p ${type}2 "
+                     . "byte-for-byte, this is probably not what you want "
+                     . "for all but the simplest of structures.";
+        }
+    }
+
+    print $fh "/**\n";
+    print $fh " * Compares \@p ${type}1 and \@p ${type}2.\n";
+    print $fh wrap(" * "," * ",$warning . "\n") if $warning;
+    print $fh " * \@param ${type}1 The first ${type}.\n";
+    print $fh " * \@param ${type}2 The second ${type}.\n";
+    my (%val_name) = (
+        ">0" => "greater than",
+        "0 " => "equal to",
+        "<0" => "less than",
+    );
+    while (my ($val, $name) = each %val_name) {
+        print $fh " * \@retval $val If \@p ${type}1 $name[0]$ord $name \@p ${type}2 $name[1].\n";
+    }
+    print $fh " */\n";
+    print $fh "#define ${type}_compare(${type}1, ${type}2) \\\n";
+    print $fh "    $expr\n";
+    print $fh "\n";
+    $macros{$name} = 1;
+}
+
+my $INCLUDES = '';
+open my $includes, ">", \$INCLUDES or die;
+
+my $MACROS = '';
+open my $macros, ">", \$MACROS or die;
+
+my $MACROS2 = '';
+open my $macros2, ">", \$MACROS2 or die;
+
+while (<>) {
+    chomp;
+    make_macro( $_, $macros2 ) if /^(?:Assert|Fail)/;
+    make_compare( $_, $macros ) if /_compare$/;
+}
+
+foreach my $need (keys %need) {
+    given ($need) {
+        when /^(?:String|Number)_compare$/ {
+            make_compare( $need, $macros );
+            delete $need{$need};
+        }
+    }
+}
+
+foreach my $macro (keys %macros) {
+    delete $need{$macro};
+}
+
+if (scalar %need) {
+    print $includes "/* needed functions/macros:\n";
+    foreach my $need (keys %need) {
+        if ($need =~ /^(.*)_compare$/) {
+            print $includes " * int $need(${1}1,${1}2);\n";
+        } else {
+            print $includes " * $need\n";
+        }
+    }
+    print $includes " */\n\n";
+}
+
+foreach my $include (keys %include) {
+    print $includes "#include <$include>\n";
+}
+
+print <<'END';
+#ifndef _CHECK_MACROS_H
+#define _CHECK_MACROS_H
+
+END
+print $INCLUDES;
+print "\n";
+print $MACROS;
+print $MACROS2;
+print <<'END';
+
+#endif /* _CHECK_MACROS_H */
+END
+
+