| #!/usr/bin/env perl |
| ## |
| ## Copyright (c) 2017, Alliance for Open Media. All rights reserved |
| ## |
| ## This source code is subject to the terms of the BSD 2 Clause License and |
| ## the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License |
| ## was not distributed with this source code in the LICENSE file, you can |
| ## obtain it at www.aomedia.org/license/software. If the Alliance for Open |
| ## Media Patent License 1.0 was not distributed with this source code in the |
| ## PATENTS file, you can obtain it at www.aomedia.org/license/patent. |
| ## |
| no strict 'refs'; |
| use warnings; |
| use Getopt::Long; |
| Getopt::Long::Configure("auto_help") if $Getopt::Long::VERSION > 2.32; |
| |
| my %ALL_FUNCS = (); |
| my @ALL_ARCHS; |
| my @ALL_FORWARD_DECLS; |
| my @REQUIRES; |
| |
| my %opts = (); |
| my %disabled = (); |
| my %required = (); |
| |
| my @argv; |
| foreach (@ARGV) { |
| $disabled{$1} = 1, next if /--disable-(.*)/; |
| $required{$1} = 1, next if /--require-(.*)/; |
| push @argv, $_; |
| } |
| |
| # NB: use GetOptions() instead of GetOptionsFromArray() for compatibility. |
| @ARGV = @argv; |
| GetOptions( |
| \%opts, |
| 'arch=s', |
| 'sym=s', |
| 'config=s', |
| ); |
| |
| foreach my $opt (qw/arch config/) { |
| if (!defined($opts{$opt})) { |
| warn "--$opt is required!\n"; |
| Getopt::Long::HelpMessage('-exit' => 1); |
| } |
| } |
| |
| foreach my $defs_file (@ARGV) { |
| if (!-f $defs_file) { |
| warn "$defs_file: $!\n"; |
| Getopt::Long::HelpMessage('-exit' => 1); |
| } |
| } |
| |
| open CONFIG_FILE, $opts{config} or |
| die "Error opening config file '$opts{config}': $!\n"; |
| |
| my %config = (); |
| while (<CONFIG_FILE>) { |
| next if !/^#define\s+(?:CONFIG_|HAVE_)/; |
| chomp; |
| my @line_components = split /\s/; |
| scalar @line_components > 2 or |
| die "Invalid input passed to rtcd.pl via $opts{config}."; |
| # $line_components[0] = #define |
| # $line_components[1] = flag name (CONFIG_SOMETHING or HAVE_SOMETHING) |
| # $line_components[2] = flag value (0 or 1) |
| $config{$line_components[1]} = "$line_components[2]" eq "1" ? "yes" : ""; |
| } |
| close CONFIG_FILE; |
| |
| # |
| # Routines for the RTCD DSL to call |
| # |
| sub aom_config($) { |
| return (defined $config{$_[0]}) ? $config{$_[0]} : ""; |
| } |
| |
| sub specialize { |
| if (@_ <= 1) { |
| die "'specialize' must be called with a function name and at least one ", |
| "architecture ('C' is implied): \n@_\n"; |
| } |
| my $fn=$_[0]; |
| shift; |
| foreach my $opt (@_) { |
| eval "\$${fn}_${opt}=${fn}_${opt}"; |
| } |
| } |
| |
| sub add_proto { |
| my $fn = splice(@_, -2, 1); |
| my @proto = @_; |
| foreach (@proto) { tr/\t/ / } |
| $ALL_FUNCS{$fn} = \@proto; |
| specialize $fn, "c"; |
| } |
| |
| sub require { |
| foreach my $fn (keys %ALL_FUNCS) { |
| foreach my $opt (@_) { |
| my $ofn = eval "\$${fn}_${opt}"; |
| next if !$ofn; |
| |
| # if we already have a default, then we can disable it, as we know |
| # we can do better. |
| my $best = eval "\$${fn}_default"; |
| if ($best) { |
| my $best_ofn = eval "\$${best}"; |
| if ($best_ofn && "$best_ofn" ne "$ofn") { |
| eval "\$${best}_link = 'false'"; |
| } |
| } |
| eval "\$${fn}_default=${fn}_${opt}"; |
| eval "\$${fn}_${opt}_link='true'"; |
| } |
| } |
| } |
| |
| sub forward_decls { |
| push @ALL_FORWARD_DECLS, @_; |
| } |
| |
| # |
| # Include the user's directives |
| # |
| foreach my $f (@ARGV) { |
| open FILE, "<", $f or die "cannot open $f: $!\n"; |
| my $contents = join('', <FILE>); |
| close FILE; |
| eval $contents or warn "eval failed: $@\n"; |
| } |
| |
| # |
| # Process the directives according to the command line |
| # |
| sub process_forward_decls() { |
| foreach (@ALL_FORWARD_DECLS) { |
| $_->(); |
| } |
| } |
| |
| sub determine_indirection { |
| aom_config("CONFIG_RUNTIME_CPU_DETECT") eq "yes" or &require(@ALL_ARCHS); |
| foreach my $fn (keys %ALL_FUNCS) { |
| my $n = ""; |
| my @val = @{$ALL_FUNCS{$fn}}; |
| my $args = pop @val; |
| my $rtyp = "@val"; |
| my $dfn = eval "\$${fn}_default"; |
| $dfn = eval "\$${dfn}"; |
| foreach my $opt (@_) { |
| my $ofn = eval "\$${fn}_${opt}"; |
| next if !$ofn; |
| my $link = eval "\$${fn}_${opt}_link"; |
| next if $link && $link eq "false"; |
| $n .= "x"; |
| } |
| if ($n eq "x") { |
| eval "\$${fn}_indirect = 'false'"; |
| } else { |
| eval "\$${fn}_indirect = 'true'"; |
| } |
| } |
| } |
| |
| sub declare_function_pointers { |
| foreach my $fn (sort keys %ALL_FUNCS) { |
| my @val = @{$ALL_FUNCS{$fn}}; |
| my $args = pop @val; |
| my $rtyp = "@val"; |
| my $dfn = eval "\$${fn}_default"; |
| $dfn = eval "\$${dfn}"; |
| foreach my $opt (@_) { |
| my $ofn = eval "\$${fn}_${opt}"; |
| next if !$ofn; |
| print "$rtyp ${ofn}($args);\n"; |
| } |
| if (eval "\$${fn}_indirect" eq "false") { |
| print "#define ${fn} ${dfn}\n"; |
| } else { |
| print "RTCD_EXTERN $rtyp (*${fn})($args);\n"; |
| } |
| print "\n"; |
| } |
| } |
| |
| sub set_function_pointers { |
| foreach my $fn (sort keys %ALL_FUNCS) { |
| my @val = @{$ALL_FUNCS{$fn}}; |
| my $args = pop @val; |
| my $rtyp = "@val"; |
| my $dfn = eval "\$${fn}_default"; |
| $dfn = eval "\$${dfn}"; |
| if (eval "\$${fn}_indirect" eq "true") { |
| print " $fn = $dfn;\n"; |
| foreach my $opt (@_) { |
| my $ofn = eval "\$${fn}_${opt}"; |
| next if !$ofn; |
| next if "$ofn" eq "$dfn"; |
| my $link = eval "\$${fn}_${opt}_link"; |
| next if $link && $link eq "false"; |
| my $cond = eval "\$have_${opt}"; |
| print " if (${cond}) $fn = $ofn;\n" |
| } |
| } |
| } |
| } |
| |
| sub filter { |
| my @filtered; |
| foreach (@_) { push @filtered, $_ unless $disabled{$_}; } |
| return @filtered; |
| } |
| |
| # |
| # Helper functions for generating the arch specific RTCD files |
| # |
| sub common_top() { |
| my $include_guard = uc($opts{sym})."_H_"; |
| print <<EOF; |
| // This file is generated. Do not edit. |
| #ifndef ${include_guard} |
| #define ${include_guard} |
| |
| #ifdef RTCD_C |
| #define RTCD_EXTERN |
| #else |
| #define RTCD_EXTERN extern |
| #endif |
| |
| EOF |
| |
| process_forward_decls(); |
| print <<EOF; |
| |
| #ifdef __cplusplus |
| extern "C" { |
| #endif |
| |
| EOF |
| declare_function_pointers("c", @ALL_ARCHS); |
| |
| print <<EOF; |
| void $opts{sym}(void); |
| |
| EOF |
| } |
| |
| sub common_bottom() { |
| print <<EOF; |
| |
| #ifdef __cplusplus |
| } // extern "C" |
| #endif |
| |
| #endif |
| EOF |
| } |
| |
| sub x86() { |
| determine_indirection("c", @ALL_ARCHS); |
| |
| # Assign the helper variable for each enabled extension |
| foreach my $opt (@ALL_ARCHS) { |
| my $opt_uc = uc $opt; |
| eval "\$have_${opt}=\"flags & HAS_${opt_uc}\""; |
| } |
| |
| common_top; |
| print <<EOF; |
| #ifdef RTCD_C |
| #include "aom_ports/x86.h" |
| static void setup_rtcd_internal(void) |
| { |
| int flags = x86_simd_caps(); |
| |
| (void)flags; |
| |
| EOF |
| |
| set_function_pointers("c", @ALL_ARCHS); |
| |
| print <<EOF; |
| } |
| #endif |
| EOF |
| common_bottom; |
| } |
| |
| sub arm() { |
| determine_indirection("c", @ALL_ARCHS); |
| |
| # Assign the helper variable for each enabled extension |
| foreach my $opt (@ALL_ARCHS) { |
| my $opt_uc = uc $opt; |
| eval "\$have_${opt}=\"flags & HAS_${opt_uc}\""; |
| } |
| |
| common_top; |
| print <<EOF; |
| #include "config/aom_config.h" |
| |
| #ifdef RTCD_C |
| #include "aom_ports/arm.h" |
| static void setup_rtcd_internal(void) |
| { |
| int flags = aom_arm_cpu_caps(); |
| |
| (void)flags; |
| |
| EOF |
| |
| set_function_pointers("c", @ALL_ARCHS); |
| |
| print <<EOF; |
| } |
| #endif |
| EOF |
| common_bottom; |
| } |
| |
| sub ppc() { |
| determine_indirection("c", @ALL_ARCHS); |
| |
| # Assign the helper variable for each enabled extension |
| foreach my $opt (@ALL_ARCHS) { |
| my $opt_uc = uc $opt; |
| eval "\$have_${opt}=\"flags & HAS_${opt_uc}\""; |
| } |
| |
| common_top; |
| |
| print <<EOF; |
| #include "config/aom_config.h" |
| |
| #ifdef RTCD_C |
| #include "aom_ports/ppc.h" |
| static void setup_rtcd_internal(void) |
| { |
| int flags = ppc_simd_caps(); |
| |
| (void)flags; |
| |
| EOF |
| |
| set_function_pointers("c", @ALL_ARCHS); |
| |
| print <<EOF; |
| } |
| #endif |
| EOF |
| common_bottom; |
| } |
| |
| sub unoptimized() { |
| determine_indirection "c"; |
| common_top; |
| print <<EOF; |
| #include "config/aom_config.h" |
| |
| #ifdef RTCD_C |
| static void setup_rtcd_internal(void) |
| { |
| EOF |
| |
| set_function_pointers "c"; |
| |
| print <<EOF; |
| } |
| #endif |
| EOF |
| common_bottom; |
| } |
| |
| # |
| # Main Driver |
| # |
| |
| &require("c"); |
| &require(keys %required); |
| if ($opts{arch} eq 'x86') { |
| @ALL_ARCHS = filter(qw/mmx sse sse2 sse3 ssse3 sse4_1 sse4_2 avx avx2/); |
| x86; |
| } elsif ($opts{arch} eq 'x86_64') { |
| @ALL_ARCHS = filter(qw/mmx sse sse2 sse3 ssse3 sse4_1 sse4_2 avx avx2/); |
| @REQUIRES = filter(qw/mmx sse sse2/); |
| &require(@REQUIRES); |
| x86; |
| } elsif ($opts{arch} =~ /armv[78]\w?/) { |
| @ALL_ARCHS = filter(qw/neon/); |
| arm; |
| } elsif ($opts{arch} eq 'arm64' ) { |
| @ALL_ARCHS = filter(qw/neon arm_crc32 neon_dotprod neon_i8mm sve sve2/); |
| @REQUIRES = filter(qw/neon/); |
| &require(@REQUIRES); |
| arm; |
| } elsif ($opts{arch} eq 'ppc') { |
| @ALL_ARCHS = filter(qw/vsx/); |
| ppc; |
| } else { |
| unoptimized; |
| } |
| |
| __END__ |
| |
| =head1 NAME |
| |
| rtcd - |
| |
| =head1 SYNOPSIS |
| |
| Usage: rtcd.pl [options] FILE |
| |
| See 'perldoc rtcd.pl' for more details. |
| |
| =head1 DESCRIPTION |
| |
| Reads the Run Time CPU Detections definitions from FILE and generates a |
| C header file on stdout. |
| |
| =head1 OPTIONS |
| |
| Options: |
| --arch=ARCH Architecture to generate defs for (required) |
| --disable-EXT Disable support for EXT extensions |
| --require-EXT Require support for EXT extensions |
| --sym=SYMBOL Unique symbol to use for RTCD initialization function |
| --config=FILE Path to file containing C preprocessor directives to parse |