| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373 | #!/usr/bin/env perl# generate_code.pl## This file is part of mbed TLS (https://tls.mbed.org)## Copyright (c) 2009-2016, ARM Limited, All Rights Reserved## Purpose## Generates the test suite code given inputs of the test suite directory that# contain the test suites, and the test suite file names for the test code and# test data.## Usage: generate_code.pl <suite dir> <code file> <data file> [main code file]## Structure of files##   - main code file - 'main_test.function'#       Template file that contains the main() function for the test suite,#       test dispatch code as well as support functions. It contains the#       following symbols which are substituted by this script during#       processing:#           TESTCASE_FILENAME#           TESTCODE_FILENAME#           SUITE_PRE_DEP#           MAPPING_CODE#           FUNCTION CODE#           SUITE_POST_DEP#           DEP_CHECK_CODE#           DISPATCH_FUNCTION#           !LINE_NO!##   - common helper code file - 'helpers.function'#       Common helper functions##   - test suite code file - file name in the form 'test_suite_xxx.function'#       Code file that contains the actual test cases. The file contains a#       series of code sequences delimited by the following:#           BEGIN_HEADER / END_HEADER - list of headers files#           BEGIN_SUITE_HELPERS / END_SUITE_HELPERS - helper functions common to#               the test suite#           BEGIN_CASE / END_CASE - the test cases in the test suite. Each test#               case contains at least one function that is used to create the#               dispatch code.##   - test data file - file name in the form 'test_suite_xxxx.data'#       The test case parameters to to be used in execution of the test. The#       file name is used to replace the symbol 'TESTCASE_FILENAME' in the main#       code file above.#use strict;my $suite_dir = shift or die "Missing suite directory";my $suite_name = shift or die "Missing suite name";my $data_name = shift or die "Missing data name";my $test_main_file = do { my $arg = shift; defined($arg) ? $arg :  $suite_dir."/main_test.function" };my $test_file = $data_name.".c";my $test_common_helper_file = $suite_dir."/helpers.function";my $test_case_file = $suite_dir."/".$suite_name.".function";my $test_case_data = $suite_dir."/".$data_name.".data";my $line_separator = $/;undef $/;## Open and read in the input files#open(TEST_HELPERS, "$test_common_helper_file") or die "Opening test helpers'$test_common_helper_file': $!";my $test_common_helpers = <TEST_HELPERS>;close(TEST_HELPERS);open(TEST_MAIN, "$test_main_file") or die "Opening test main '$test_main_file': $!";my @test_main_lines = split/^/,  <TEST_MAIN>;my $test_main;my $index = 2;for my $line (@test_main_lines) {    $line =~ s/!LINE_NO!/$index/;    $test_main = $test_main.$line;    $index++;}close(TEST_MAIN);open(TEST_CASES, "$test_case_file") or die "Opening test cases '$test_case_file': $!";my @test_cases_lines = split/^/,  <TEST_CASES>;my $test_cases;my $index = 2;for my $line (@test_cases_lines) {    if ($line =~ /^\/\* BEGIN_SUITE_HELPERS .*\*\//)    {        $line = $line."#line $index \"$test_case_file\"\n";    }    if ($line =~ /^\/\* BEGIN_CASE .*\*\//)    {        $line = $line."#line $index \"$test_case_file\"\n";    }    $line =~ s/!LINE_NO!/$index/;    $test_cases = $test_cases.$line;    $index++;}close(TEST_CASES);open(TEST_DATA, "$test_case_data") or die "Opening test data '$test_case_data': $!";my $test_data = <TEST_DATA>;close(TEST_DATA);## Find the headers, dependencies, and suites in the test cases file#my ( $suite_header ) = $test_cases =~ /\/\* BEGIN_HEADER \*\/\n(.*?)\n\/\* END_HEADER \*\//s;my ( $suite_defines ) = $test_cases =~ /\/\* BEGIN_DEPENDENCIES\n \* (.*?)\n \* END_DEPENDENCIES/s;my ( $suite_helpers ) = $test_cases =~ /\/\* BEGIN_SUITE_HELPERS \*\/\n(.*?)\n\/\* END_SUITE_HELPERS \*\//s;my $requirements;if ($suite_defines =~ /^depends_on:/){    ( $requirements ) = $suite_defines =~ /^depends_on:(.*)$/;}my @var_req_arr = split(/:/, $requirements);my $suite_pre_code;my $suite_post_code;my $dispatch_code;my $mapping_code;my %mapping_values;while (@var_req_arr){    my $req = shift @var_req_arr;    $req =~ s/(!?)(.*)/$1defined($2)/;    $suite_pre_code .= "#if $req\n";    $suite_post_code .= "#endif /* $req */\n";}$/ = $line_separator;open(TEST_FILE, ">$test_file") or die "Opening destination file '$test_file': $!";print TEST_FILE << "END";/* * *** THIS FILE HAS BEEN MACHINE GENERATED *** * * This file has been machine generated using the script: $0 * * Test file      : $test_file * * The following files were used to create this file. * *      Main code file  : $test_main_file *      Helper file     : $test_common_helper_file *      Test suite file : $test_case_file *      Test suite data : $test_case_data * * *  This file is part of mbed TLS (https://tls.mbed.org) */#if !defined(MBEDTLS_CONFIG_FILE)#include <mbedtls/config.h>#else#include MBEDTLS_CONFIG_FILE#endif/*----------------------------------------------------------------------------*//* Common helper code */$test_common_helpers/*----------------------------------------------------------------------------*//* Test Suite Code */$suite_pre_code$suite_header$suite_helpers$suite_post_codeEND$test_main =~ s/SUITE_PRE_DEP/$suite_pre_code/;$test_main =~ s/SUITE_POST_DEP/$suite_post_code/;while($test_cases =~ /\/\* BEGIN_CASE *([\w:]*) \*\/\n(.*?)\n\/\* END_CASE \*\//msg){    my $function_deps = $1;    my $function_decl = $2;    # Sanity checks of function    if ($function_decl !~ /^#line\s*.*\nvoid /)    {        die "Test function does not have 'void' as return type.\n" .            "Function declaration:\n" .            $function_decl;    }    if ($function_decl !~ /^(#line\s*.*)\nvoid (\w+)\(\s*(.*?)\s*\)\s*{(.*)}/ms)    {        die "Function declaration not in expected format\n";    }    my $line_directive = $1;    my $function_name = $2;    my $function_params = $3;    my $function_pre_code;    my $function_post_code;    my $param_defs;    my $param_checks;    my @dispatch_params;    my @var_def_arr = split(/,\s*/, $function_params);    my $i = 1;    my $mapping_regex = "".$function_name;    my $mapping_count = 0;    $function_decl =~ s/(^#line\s*.*)\nvoid /$1\nvoid test_suite_/;    # Add exit label if not present    if ($function_decl !~ /^exit:$/m)    {        $function_decl =~ s/}\s*$/\nexit:\n    return;\n}/;    }    if ($function_deps =~ /^depends_on:/)    {        ( $function_deps ) = $function_deps =~ /^depends_on:(.*)$/;    }    foreach my $req (split(/:/, $function_deps))    {        $function_pre_code .= "#ifdef $req\n";        $function_post_code .= "#endif /* $req */\n";    }    foreach my $def (@var_def_arr)    {        # Handle the different parameter types        if( substr($def, 0, 4) eq "int " )        {            $param_defs .= "    int param$i;\n";            $param_checks .= "    if( verify_int( params[$i], ¶m$i ) != 0 ) return( DISPATCH_INVALID_TEST_DATA );\n";            push @dispatch_params, "param$i";            $mapping_regex .= ":([\\d\\w |\\+\\-\\(\\)]+)";            $mapping_count++;        }        elsif( substr($def, 0, 6) eq "char *" )        {            $param_defs .= "    char *param$i = params[$i];\n";            $param_checks .= "    if( verify_string( ¶m$i ) != 0 ) return( DISPATCH_INVALID_TEST_DATA );\n";            push @dispatch_params, "param$i";            $mapping_regex .= ":[^:\n]+";        }        else        {            die "Parameter declaration not of supported type (int, char *)\n";        }        $i++;    }    # Find non-integer values we should map for this function    if( $mapping_count)    {        my @res = $test_data =~ /^$mapping_regex/msg;        foreach my $value (@res)        {            next unless ($value !~ /^\d+$/);            if ( $mapping_values{$value} ) {                ${ $mapping_values{$value} }{$function_pre_code} = 1;            } else {                $mapping_values{$value} = { $function_pre_code => 1 };            }        }    }    my $call_params = join ", ", @dispatch_params;    my $param_count = @var_def_arr + 1;    $dispatch_code .= << "END";if( strcmp( params[0], "$function_name" ) == 0 ){$function_pre_code$param_defs    if( cnt != $param_count )    {        mbedtls_fprintf( stderr, "\\nIncorrect argument count (%d != %d)\\n", cnt, $param_count );        return( DISPATCH_INVALID_TEST_DATA );    }$param_checks    test_suite_$function_name( $call_params );    return ( DISPATCH_TEST_SUCCESS );$function_post_code    return ( DISPATCH_UNSUPPORTED_SUITE );}elseEND    my $function_code = $function_pre_code . $function_decl . "\n" .                        $function_post_code;    $test_main =~ s/FUNCTION_CODE/$function_code\nFUNCTION_CODE/;}# Find specific case dependencies that we should be able to check# and make check codemy $dep_check_code;my @res = $test_data =~ /^depends_on:([\w:]+)/msg;my %case_deps;foreach my $deps (@res){    foreach my $dep (split(/:/, $deps))    {        $case_deps{$dep} = 1;    }}while( my ($key, $value) = each(%case_deps) ){    $dep_check_code .= << "END";    if( strcmp( str, "$key" ) == 0 )    {#if defined($key)        return( DEPENDENCY_SUPPORTED );#else        return( DEPENDENCY_NOT_SUPPORTED );#endif    }END}# Make mapping codewhile( my ($key, $value) = each(%mapping_values) ){    my $key_mapping_code = << "END";    if( strcmp( str, "$key" ) == 0 )    {        *value = ( $key );        return( KEY_VALUE_MAPPING_FOUND );    }END    # handle depenencies, unless used at least one without depends    if ($value->{""}) {        $mapping_code .= $key_mapping_code;        next;    }    for my $ifdef ( keys %$value ) {        (my $endif = $ifdef) =~ s!ifdef!endif //!g;        $mapping_code .= $ifdef . $key_mapping_code . $endif;    }}$dispatch_code =~ s/^(.+)/    $1/mg;$test_main =~ s/TESTCASE_FILENAME/$test_case_data/g;$test_main =~ s/TESTCODE_FILENAME/$test_case_file/g;$test_main =~ s/FUNCTION_CODE//;$test_main =~ s/DEP_CHECK_CODE/$dep_check_code/;$test_main =~ s/DISPATCH_FUNCTION/$dispatch_code/;$test_main =~ s/MAPPING_CODE/$mapping_code/;print TEST_FILE << "END";$test_mainENDclose(TEST_FILE);
 |