#!/usr/bin/perl -w require 5.006; use strict; use DirHandle; use Getopt::Long; use File::Spec; use File::Copy; my $MAJOR_VERSION = 1; my $MINOR_VERSION = 5; my $PATCH_VERSION = 11; # If 1, then we always copy include files, without attempting to make symlinks # or hardlinks. It seems that link("","") will succeed on Windows, but the # actual operation will fail (FAT32 doesn't support any kind of links). # This is automatically forced if $^O is 'dos', 'MSWin32', or 'cygwin' my $FORCE_COPY = 0; my $INCLUDE_DIR = 'include'; my $SRC_DIR = 'src'; my $MOD_DIR = 'modules'; my $CHECK_DIR = 'checks'; my $DOC_DIR = 'doc'; my $BUILD_DIR = 'build'; my $BUILD_DIR_LIB = 'lib'; my $BUILD_DIR_CHECKS = 'checks'; my $MAKE_FILE = 'Makefile'; my $BUILD_INCLUDE_DIR = 'build/include'; my $ARCH_DIR = 'misc/config/arch'; my $OS_DIR = 'misc/config/os'; my $CC_DIR = 'misc/config/cc'; my $CONFIG_HEADER = 'build.h'; my $CPP_INCLUDE_DIR_DIRNAME = 'botan'; my %MODULE_SETS = ( 'unix' => [ 'alloc_mmap', 'es_egd', 'es_ftw', 'es_unix', 'fd_unix', 'tm_unix' ], 'beos' => [ 'es_beos', 'es_unix', 'fd_unix', 'tm_unix' ], 'win32' => ['es_capi', 'es_win32', 'mux_win32', 'tm_win32' ], ); my %DOCS = ( 'readme.txt' => undef, # undef = file is in top level directory 'pgpkeys.asc' => $DOC_DIR, 'api.pdf' => $DOC_DIR, 'tutorial.pdf' => $DOC_DIR, 'fips140.pdf' => $DOC_DIR, 'api.tex' => $DOC_DIR, 'tutorial.tex' => $DOC_DIR, 'fips140.tex' => $DOC_DIR, 'botan.rc' => $DOC_DIR, 'credits.txt' => $DOC_DIR, 'info.txt' => $DOC_DIR, 'license.txt' => $DOC_DIR, 'log.txt' => $DOC_DIR, 'thanks.txt' => $DOC_DIR, 'todo.txt' => $DOC_DIR ); my (%REALNAME,%MODULES); my(%SUBMODEL_ALIAS, %DEFAULT_SUBMODEL, %ARCH, %ARCH_ALIAS, %OS_SUPPORTS_ARCH, %OS_SUPPORTS_SHARED, %OS_TYPE, %INSTALL_INFO, %OS_OBJ_SUFFIX, %OS_SHARED_SUFFIX, %OS_STATIC_SUFFIX, %OS_AR_COMMAND, %OS_AR_NEEDS_RANLIB, %OS_ALIAS, %CC_BINARY_NAME, %CC_LIB_OPT_FLAGS, %CC_CHECK_OPT_FLAGS, %CC_WARN_FLAGS, %CC_LANG_FLAGS, %CC_SO_OBJ_FLAGS, %CC_SO_LINK_FLAGS, %CC_DEBUG_FLAGS, %CC_NO_DEBUG_FLAGS, %CC_MACHINE_OPT_FLAGS, %CC_MACHINE_OPT_FLAGS_RE, %CC_ABI_FLAGS, %CC_SUPPORTS_OS, %CC_SUPPORTS_ARCH, %CC_AR_COMMAND, %MAKEFILE_STYLE); my $user_set_root = ''; my ($doc_dir, $lib_dir); my (%ignored_src, %ignored_include, %added_src, %added_include, %lib_src, %check_src, %include); my ($CPP_INCLUDE_DIR, $BUILD_LIB_DIR, $BUILD_CHECK_DIR); sub main() { %MODULES = get_modules_list($MOD_DIR); set_arch_defines($ARCH_DIR); set_os_defines($OS_DIR); set_cc_defines($CC_DIR); my ($debug, $dumb_gcc, $no_shared) = (0, 0, 0); my ($make_style, $build_dir, $module_set, $local_config) = ('', '', '', ''); my $autoconfig = 1; my @using_mods; GetOptions('debug' => sub { $debug = 1; }, 'disable-shared' => sub { $no_shared = 1; }, 'noauto' => sub { $autoconfig = 0 }, 'gcc295x' => sub { $dumb_gcc = 1; }, 'dumb-gcc' => sub { $dumb_gcc = 1; }, 'make-style=s' => \$make_style, 'modules=s' => \@using_mods, 'module-set=s' => \$module_set, 'prefix=s' => \$user_set_root, 'docdir=s' => \$doc_dir, 'libdir=s' => \$lib_dir, 'build-dir=s' => \$build_dir, 'local-config=s' => \$local_config, 'help' => sub { help(); } ); if($^O eq 'MSWin32' or $^O eq 'dos' or $^O eq 'cygwin') { print "Disabling use of symlink()/link() due to Win FS limitations\n"; $FORCE_COPY = 1; } my $cc_os_cpu_set = ''; if($#ARGV == 0) { $cc_os_cpu_set = $ARGV[0]; } elsif($autoconfig) { $cc_os_cpu_set = guess_triple(); print "(autoconfig): Guessing your system config is $cc_os_cpu_set\n"; } else { help(); } my ($cc,$os,$submodel) = split(/-/,$cc_os_cpu_set,3); if(!defined($cc) or !defined($os) or !defined($submodel)) { help(); } if($build_dir ne '') { $BUILD_DIR = $build_dir; $BUILD_INCLUDE_DIR = $build_dir . '/include'; } # hacks if($cc eq 'gcc' && $dumb_gcc != 1) { my $gcc_version = ''; # Stupid Apple. At least they fixed it after 10.2 if($os eq 'darwin') { $gcc_version = `c++ -v 2>&1`; } else { $gcc_version = `g++ -v 2>&1`; } $gcc_version = '' if not defined $gcc_version; # GCC 2.95.x and 3.[34] are busted in handling long long in # C++. The third check is because on Cygwin (at least for me) # $gcc_version doesn't get the output from g++, not sure # what's up with that. If it's Cygwin and we didn't get # output, assume it's a buggy GCC. There is no reduction in # code quality, etc, so even if we're wrong it's no big deal. if(($gcc_version =~ /4\.[01]/) || ($gcc_version =~ /3\.[34]/) || ($gcc_version =~ /2\.95\.[0-4]/) || ($gcc_version eq '' && $^O eq 'cygwin')) { warn "(note): Enabling -fpermissive to work around " . "possible GCC issues\n"; $dumb_gcc = 1; } if($gcc_version =~ /2\.95\.[0-4]/) { print "(note): GCC 2.95.x issues a lot of warnings for \n" . " Botan; either ignore the warnings or upgrade to 3.x\n"; } } $os = $OS_ALIAS{$os} if(defined $OS_ALIAS{$os}); die "(error): Compiler $cc isn't known\n" unless defined($CC_BINARY_NAME{$cc}); die "(error): OS $os isn't known\n" unless (defined($OS_SUPPORTS_ARCH{$os}) or $os eq 'generic'); # Get the canonical submodel name (like r8k -> r8000) $submodel = $SUBMODEL_ALIAS{$submodel} if(defined($SUBMODEL_ALIAS{$submodel})); my $arch = undef; # Convert an arch alias to it's real name (like axp -> alpha) if(defined($ARCH_ALIAS{$submodel})) { $arch = $ARCH_ALIAS{$submodel}; $submodel = $arch; } # If it's a regular submodel type, figure out what arch it is elsif(defined($ARCH{$submodel})) { $arch = $ARCH{$submodel}; } elsif($submodel eq 'generic') { $arch = 'generic'; } else { die "(error): Arch $submodel isn't known\n"; } # If we got a generic family name as the model type if($submodel eq $arch and $submodel ne 'generic') { $submodel = $DEFAULT_SUBMODEL{$arch}; warn "(note): Using $submodel as default type for family ", $REALNAME{$arch},"\n" if($submodel ne $arch); } $make_style = $MAKEFILE_STYLE{$cc} unless($make_style); die "(error): Module set $module_set isn't known\n" if($module_set && !defined($MODULE_SETS{$module_set})); if($module_set) { foreach (@{ $MODULE_SETS{$module_set} }) { push @using_mods,$_; } } @using_mods = grep {/./} split(/,/,join(',',@using_mods)); if($autoconfig) { foreach (guess_mods($cc,$os,$arch,$submodel)) { # Print a notice, unless it was enabled explicitly (or in a set) my $picked_by_user = 0; foreach my $x (@using_mods) { $picked_by_user = 1 if($_ eq $x); } print " (autoconfig): Enabling module $_\n" if(!$picked_by_user); push @using_mods, $_; } } # Uniqify @using_mods my %uniqed_mods; foreach my $mod (@using_mods) { $uniqed_mods{$mod} = 0; } @using_mods = sort keys %uniqed_mods; foreach (@using_mods) { die "(error): Module $_ isn't known (try --help)\n" unless(exists($MODULES{$_})); } die "(error): $REALNAME{$os} doesn't run on $arch ($submodel)\n" unless($arch eq 'generic' or $os eq 'generic' or in_array($OS_SUPPORTS_ARCH{$os}, $arch)); die "(error): $REALNAME{$cc} doesn't run on $arch ($submodel)\n" unless($arch eq 'generic' or (in_array($CC_SUPPORTS_ARCH{$cc}, $arch))); die "(error): $REALNAME{$cc} doesn't run on $REALNAME{$os}\n" unless($os eq 'generic' or (in_array($CC_SUPPORTS_OS{$cc}, $os))); check_for_conflicts(@using_mods); foreach (@using_mods) { load_module($_, $cc, $os, $arch, $submodel, %{ $MODULES{$_} }); } print_pkg_config($os, $MAJOR_VERSION, $MINOR_VERSION, $PATCH_VERSION, using_libs($os, @using_mods)); $CPP_INCLUDE_DIR = File::Spec->catdir($BUILD_INCLUDE_DIR, $CPP_INCLUDE_DIR_DIRNAME); $BUILD_LIB_DIR = File::Spec->catdir($BUILD_DIR, $BUILD_DIR_LIB); $BUILD_CHECK_DIR = File::Spec->catdir($BUILD_DIR, $BUILD_DIR_CHECKS); %lib_src = list_dir($SRC_DIR, \%ignored_src); %check_src = list_dir($CHECK_DIR, undef); %include = list_dir($INCLUDE_DIR, \%ignored_include); mkdirs(($BUILD_DIR, $BUILD_INCLUDE_DIR, $CPP_INCLUDE_DIR, $BUILD_LIB_DIR, $BUILD_CHECK_DIR)); clean_out_dirs(($CPP_INCLUDE_DIR)); my $config_h = File::Spec->catfile($BUILD_DIR, $CONFIG_HEADER); print_config_h($MAJOR_VERSION, $MINOR_VERSION, $PATCH_VERSION, $config_h, $local_config, $os, $arch, $submodel, find_mp_bits(@using_mods), defines(@using_mods), defines_base(@using_mods)); $added_include{$CONFIG_HEADER} = $BUILD_DIR; copy_files($CPP_INCLUDE_DIR, \%include, \%added_include); my %all_includes = list_dir($CPP_INCLUDE_DIR); generate_makefile($make_style, $cc, $os, $submodel, $arch, $debug, $no_shared, $dumb_gcc, \%lib_src, \%check_src, \%all_includes, \%added_src, using_libs($os, @using_mods)); } # Run stuff, quit main(); exit; sub process { my $l = $_[0]; chomp($l); $l =~ s/#.*//; $l =~ s/^\s*//; $l =~ s/\s*$//; $l =~ s/\s\s*/ /; $l =~ s/\t/ /; $l; } sub check_for_file { my ($file,$mod) = @_; die "(error): Module $mod requires that file $file exist. This error\n", "should never occur; please contact the maintainers with details.\n" unless(-e $file); } sub using_libs { my ($os,@using) = @_; my %libs; foreach my $mod (@using) { my %MOD_LIBS = %{ $MODULES{$mod}{'libs'} }; foreach my $mod_os (keys %MOD_LIBS) { next if($mod_os =~ /^all!$os$/); next if($mod_os =~ /^all!$os,/); next if($mod_os =~ /^all!.*,${os}$/); next if($mod_os =~ /^all!.*,$os,.*/); next unless($mod_os eq $os or ($mod_os =~ /^all.*/)); my @liblist = split(/,/, $MOD_LIBS{$mod_os}); foreach my $lib (@liblist) { $libs{$lib} = 1; } } } my @libarray; foreach (sort keys %libs) { push @libarray , $_; } return @libarray; } sub defines { my @defarray; foreach (@_) { foreach my $define (sort keys %{ $MODULES{$_}{'define'} }) { push @defarray , $define; } } return \@defarray; } sub defines_base { my @defarray; foreach (@_) { foreach my $define (sort keys %{ $MODULES{$_}{'define_base'} }) { push @defarray , $define; } } return \@defarray; } # Any other alternatives here? sub portable_symlink { my ($from, $to_dir, $to_fname) = @_; my $can_symlink = eval { symlink("",""); 1 }; my $can_link = eval { link("",""); 1 }; if($FORCE_COPY) { $can_symlink = 0; $can_link = 0; } chdir $to_dir or die "Can't chdir to $to_dir ($!)\n"; if($can_symlink) { symlink $from, $to_fname or die "Can't symlink $from to $to_fname ($!)"; } elsif($can_link) { link $from, $to_fname or die "Can't link $from to $to_fname ($!)"; } else { copy ($from, $to_fname) or die "Can't copy $from to $to_fname ($!)"; } my $go_up = File::Spec->splitdir($to_dir); for(my $j = 0; $j != $go_up; $j++) { chdir File::Spec->updir(); } } sub copy_files { my ($include_dir, $mainline, $modules) = @_; my $updir = File::Spec->updir(); foreach (keys %{ $mainline }) { portable_symlink(File::Spec->catfile($updir, $updir, $updir, 'include', $_), $include_dir, $_); } foreach (keys %{ $modules }) { portable_symlink(File::Spec->catfile($updir, $updir, $updir, $$modules{$_}, $_), $include_dir, $_); } } sub list_dir { my ($dir, $ignore) = @_; opendir DIR, $dir or die "Couldn't open directory $dir ($!)\n"; my @list = grep { !/^\./ } readdir DIR; if($dir eq $CHECK_DIR) { @list = grep { !/\.dat$/ } grep { !/^keys$/ } grep { !/\.h$/ } @list; } # If $ignore is set, pull everything in @list that's in $ignore out of it if(defined($ignore)) { @list = grep { !exists($$ignore{$_}) } @list; } close DIR; my %list = map { $_ => $dir } @list; return %list; } sub clean_out_dirs { my (@dirs) = @_; foreach my $dir (@dirs) { my %files = list_dir($dir); foreach my $file (keys %files) { my $path = File::Spec->catfile($dir, $file); unlink $path or die "Could not unlink $path ($!)\n"; } } } sub mkdirs { my (@dirs) = @_; foreach my $dir (@dirs) { next if( -e $dir and -d $dir ); # skip it if it's already there mkdir($dir, 0777) or die "(error): Could not create directory $dir ($!)\n"; } } sub in_array { my($array, $target) = @_; return 0 unless defined($array); foreach (@$array) { return 1 if($_ eq $target); } return 0; } sub find_mp_bits { my(@modules_list) = @_; my $mp_bits = 32; # default, good for most systems my $seen_mp_module = 0; foreach my $modname (@modules_list) { my %modinfo = %{ $MODULES{$modname} }; if($modinfo{'mp_bits'}) { die "(error): Inconsistent mp_bits requests from modules\n" if($seen_mp_module && $modinfo{'mp_bits'} != $mp_bits); $seen_mp_module = 1; $mp_bits = $modinfo{'mp_bits'}; } } return $mp_bits; } sub print_config_h { my ($major, $minor, $patch, $config_h, $local_config, $os, $arch, $cpu, $mp_bits, $defines_ref, $defines_base_ref) = @_; my @defines = @{ $defines_ref }; my @defines_base = @{ $defines_base_ref }; chomp($patch); my $defines = ''; foreach (sort @defines_base) { next if not defined $_ or not $_; $defines .= "#define BOTAN_$_\n"; } if(scalar @defines_base) { $defines .= "\n"; } foreach (sort @defines) { next if not defined $_ or not $_; $defines .= "#define BOTAN_EXT_$_\n"; } chomp($defines); if($defines) { $defines = "\n" . $defines . "\n"; } open CONFIG_H, ">$config_h" or die "Couldn't write $config_h ($!)\n"; print CONFIG_H <) { print CONFIG_H; } } print CONFIG_H "\n#endif\n"; close CONFIG_H; } sub check_for_conflicts { my @mods = @_; my (%ignored, %added, %replaced, %defines); foreach my $mod (@mods) { sub check_hash { my ($mod, $do_what, $hashref) = @_; foreach (keys %{ $MODULES{$mod}{$do_what} }) { ${$hashref}{conflicts($mod, $_, $do_what, $hashref)} = $mod; } } check_hash($mod, 'define', \%defines); check_hash($mod, 'replace', \%replaced); check_hash($mod, 'add', \%added); check_hash($mod, 'ignore', \%ignored); } } sub conflicts { my ($mod, $item, $do_what, $hashref) = @_; return if(!defined($item)); if(defined($$hashref{$item})) { my $other_mod = $$hashref{$item}; die "(error): Both $mod and $other_mod $do_what $item\n"; } return $item; } sub get_modules_list { my $MOD_DIR = $_[0]; my $dir = new DirHandle $MOD_DIR; if(!defined $dir) { return (); } my %MODULES; while(defined($_ = $dir->read)) { next if($_ eq '.' or $_ eq '..'); my $MODULE = $_; my %modinfo = get_module_info($MODULE, $MOD_DIR); foreach (keys %modinfo) { $MODULES{$MODULE}{$_} = $modinfo{$_}; } } return %MODULES; } sub get_module_info { my ($MODULE, $MOD_DIR) = @_; my %modinfo; my $desc_file = File::Spec->catfile($MOD_DIR,$MODULE, 'modinfo.txt'); die "(error): Module $MODULE does not seem to have a description file\n" unless(-e $desc_file); open MODFILE, "<$desc_file" or die "(error): Couldn't open file $desc_file, ($!)\n"; $modinfo{'libs'} = {}; $modinfo{'add'} = {}; $modinfo{'replace'} = {}; $modinfo{'ignore'} = {}; $modinfo{'define'} = {}; $modinfo{'define_base'} = {}; $modinfo{'external_libs'} = 0; while() { $modinfo{'name'} = $1 if(/^realname \"(.*)\"/); $modinfo{'notes'} = $1 if(/^note \"(.*)\"/); $modinfo{'define'}{$1} = undef if(/^define (\w*)/); $modinfo{'define_base'}{$1} = undef if(/^define_base (\w*)/); $modinfo{'mp_bits'} = $1 if(/^mp_bits ([0-9]*)/); $modinfo{'external_libs'} = 1 if(/^uses_external_libs/); if(/^require_version /) { if(/^require_version (\d)\.(\d)\.(\d)$/) { my $version = "$1.$2.$3"; my $needed_version = 100*$1 + 10*$2 + $3; my $have_version = 100*$MAJOR_VERSION + 10*$MINOR_VERSION + $PATCH_VERSION; if($needed_version > $have_version) { warn "Module $MODULE requires Botan version $version\n"; %modinfo = (); close MODFILE; return %modinfo; } } else { warn "In module $MODULE, bad version code in require_version\n"; } } # Read in a list of supported CPU types (archs and/or submodels) if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if (m@^$@); $modinfo{'arch'}{$_} = undef; } } # Read in a list of supported OSes if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if (m@^$@); $modinfo{'os'}{$_} = undef; } } if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if (m@^$@); $modinfo{'add'}{$_} = undef; } } if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if (m@^$@); $modinfo{'ignore'}{$_} = undef; } } if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if (m@^$@); $modinfo{'replace'}{$_} = undef; } } # Read in a set of os->extra library mappings if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if (m@^$@); m/^([\w!,]*) -> ([\w,-]*)$/; $modinfo{'libs'}{$1} = $2; } } # Read in a list of supported compilers if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if (m@^$@); $modinfo{'cc'}{$_} = undef; } } } close MODFILE; return %modinfo; } sub load_module { my ($modname,$cc,$os,$arch,$sub,%module) = @_; # Check to see if everything is OK WRT system requirements if(defined($module{'os'}) and !exists($module{'os'}{$os}) and $os ne 'generic') { die "(error): Module '$modname' does not run on $REALNAME{$os}\n"; } if(defined($module{'arch'}) and $arch ne 'generic' and !exists($module{'arch'}{$arch}) and !exists($module{'arch'}{$sub})) { die "(error): Module '$modname' does not run on ". "$REALNAME{$arch}/$sub\n"; } if(defined($module{'cc'}) and !exists($module{'cc'}{$cc})) { die "(error): Module '$modname' does not work with $REALNAME{$cc}\n"; } handle_files($modname, $module{'replace'}, \&replace_file); handle_files($modname, $module{'ignore'}, \&ignore_file); handle_files($modname, $module{'add'}, \&add_file); if(defined($module{'notes'})) { my $realname = $module{'name'}; my $note = $module{'notes'}; print STDERR "(note): $modname (\"$realname\"): $note\n"; } } sub handle_files { my($modname, $hash_scalar, $func) = @_; return unless defined($hash_scalar); my %hash = %{ $hash_scalar }; foreach (sort keys %hash) { if(defined($hash{$_})) { &$func($modname, $_, $hash{$_}); } else { &$func($modname, $_); } } } sub full_path { my ($file,$modname) = @_; if(defined($modname)) { return File::Spec->catfile ($MOD_DIR, $modname, $file); } else { if($file =~ /\.h$/) { return File::Spec->catfile ($INCLUDE_DIR, $file); } elsif($file =~ /\.cpp$/ or $file =~ /\.s$/ or $file =~ /\.S$/) { return File::Spec->catfile ($SRC_DIR, $file); } else { die "(internal error): Not sure where to put $file\n"; } } } sub add_file { my ($modname,$file) = @_; check_for_file(full_path($file, $modname), $modname); if($file =~ /\.cpp$/ or $file =~ /\.s$/ or $file =~ /\.S$/) { $added_src{$file} = File::Spec->catdir($MOD_DIR, $modname); } elsif($file =~ /\.h$/) { $added_include{$file} = File::Spec->catdir($MOD_DIR, $modname); } else { die "Not sure where to put $file\n"; } } sub ignore_file { my ($modname,$file) = @_; check_for_file(full_path($file), $modname); if($file =~ /\.cpp$/ or $file =~ /\.s$/ or $file =~ /\.S$/) { $ignored_src{$file} = 1; } elsif($file =~ /\.h$/) { $ignored_include{$file} = 1; } else { die "Not sure where to put $file\n"; } } # This works because ignore file always runs on files in the main source tree, # and add always works on the file in the modules directory. sub replace_file { my ($modname,$file) = @_; ignore_file($modname, $file); add_file($modname, $file); } sub help { print < 71) { print "\n "; $len = 3; } print "$_ "; $len += length "$_ "; } print "\n"; } sub which { my $file = $_[0]; my @paths = split(/:/, $ENV{PATH}); foreach my $path (@paths) { my $file_path = File::Spec->catfile($path, $file); return $file_path if(guess_check_for_file($file_path)); } return ''; } sub guess_cpu_from_this { my $cpuinfo = lc $_[0]; my $cpu = ''; $cpu = 'athlon' if($cpuinfo =~ /athlon/); $cpu = 'pentium4' if($cpuinfo =~ /pentium 4/); $cpu = 'pentium4' if($cpuinfo =~ /pentium\(r\) 4/); $cpu = 'pentium3' if($cpuinfo =~ /pentium iii/); $cpu = 'pentium2' if($cpuinfo =~ /pentium ii/); $cpu = 'pentium3' if($cpuinfo =~ /pentium 3/); $cpu = 'pentium2' if($cpuinfo =~ /pentium 2/); # The 32-bit SPARC stuff is impossible to match to arch type easily, and # anyway the uname stuff will pick up that it's a SPARC so it doesn't # matter. If it's an Ultra, assume a 32-bit userspace, no 64-bit code # possible; that's the most common setup right now anyway $cpu = 'sparc32-v9' if($cpuinfo =~ /ultrasparc/); # 64-bit PowerPC $cpu = 'rs64a' if($cpuinfo =~ /rs64-/); $cpu = 'power3' if($cpuinfo =~ /power3/); $cpu = 'power4' if($cpuinfo =~ /power4/); $cpu = 'power5' if($cpuinfo =~ /power5/); $cpu = 'ppc970' if($cpuinfo =~ /ppc970/); # Ooh, an Alpha. Try to figure out what kind if($cpuinfo =~ /alpha/) { $cpu = 'alpha-ev4' if($cpuinfo =~ /ev4/); $cpu = 'alpha-ev5' if($cpuinfo =~ /ev5/); $cpu = 'alpha-ev56' if($cpuinfo =~ /ev56/); $cpu = 'alpha-pca56' if($cpuinfo =~ /pca56/); $cpu = 'alpha-ev6' if($cpuinfo =~ /ev6/); $cpu = 'alpha-ev67' if($cpuinfo =~ /ev67/); $cpu = 'alpha-ev68' if($cpuinfo =~ /ev68/); $cpu = 'alpha-ev7' if($cpuinfo =~ /ev7/); } return $cpu; } # Do some WAGing and see if we can figure out what system we are. Think about # this as a really moronic config.guess sub guess_triple { # /bin/sh, good bet we're on something Unix-y (at least it'll have uname) if(-f '/bin/sh') { my $os = lc `uname -s 2>/dev/null`; chomp $os; # Let the crappy hacks commence! # Cygwin's uname -s is cygwin_ $os = 'cygwin' if($os =~ /^cygwin/); if(!defined $OS_TYPE{$os} && !defined $OS_ALIAS{$os}) { print "Unknown uname -s output: $os, falling back to 'generic'\n"; $os = 'generic'; } $os = $OS_ALIAS{$os} if(defined($OS_ALIAS{$os})); my $cpu = ''; # If we have /proc/cpuinfo, try to get nice specific information about # what kind of CPU we're running on. if(guess_check_for_file('/proc/cpuinfo')) { open CPUINFO, '/proc/cpuinfo' or die "Hey, guess_check_for_file said /proc/cpuinfo was readable!\n"; my $cpuinfo = join('', ); close CPUINFO; $cpu = guess_cpu_from_this($cpuinfo); } # `umame -p` is sometimes something stupid like unknown, but in some # cases it can be more specific (useful) than `uname -m` if($cpu eq '') # no guess so far { my $uname_p = `uname -p 2>/dev/null`; chomp $uname_p; $cpu = guess_cpu_from_this($uname_p); # If guess_cpu_from_this didn't figure it out, try it plain if($cpu eq '') { $cpu = lc $uname_p; } if(!defined $ARCH{$cpu} && !defined $SUBMODEL_ALIAS{$cpu} && !defined $ARCH_ALIAS{$cpu}) { # Nope, couldn't figure out uname -p $cpu = lc `uname -m 2>/dev/null`; chomp $cpu; if(!defined $ARCH{$cpu} && !defined $SUBMODEL_ALIAS{$cpu} && !defined $ARCH_ALIAS{$cpu}) { $cpu = 'generic'; } } } my @CCS = ('gcc', 'icc', 'compaq', 'kai'); # Skips several, oh well... # First try the CC enviornmental variable, if it's set if(defined($ENV{CC})) { my @new_CCS = ($ENV{CC}); foreach my $cc (@CCS) { push @new_CCS, $cc; } @CCS = @new_CCS; } my $cc = ''; foreach (@CCS) { my $bin_name = $CC_BINARY_NAME{$_}; $cc = $_ if(which($bin_name) ne ''); last if($cc ne ''); } if($cc eq '') { warn "Can't find a usable C++ compiler, is your PATH right?\n"; warn "You might need to run with explicit compiler/system flags;\n"; warn " run '$0 --help' for more information\n"; exit 1; } return "$cc-$os-$cpu"; } elsif($^O eq 'MSWin32' or $^O eq 'dos') { my $os = 'windows'; # obviously # Suggestions on this? The Win32 'shell' env is not so hot. We could # try using cpuinfo, except that will crash hard on NT/Alpha (like what # we're doing now won't!). In my defense of choosing i686: # a) There are maybe a few hundred Alpha/MIPS boxes running NT4 today # b) Anyone running Windows on < Pentium Pro deserves to lose. my $cpu = 'i686'; # No /bin/sh, so not cygwin. Assume VC++; again, this could be much # smarter my $cc = 'msvc'; return "$cc-$os-$cpu"; } else { print "Sorry, you don't seem to be on Unix or Windows;\n" . " autoconfig failed (try running me with --help)\n"; exit 1; } } sub guess_check_for_file { my $file = $_[0]; return 1 if(-f $file and -r $file); return 0; } sub guess_mods { my ($cc, $os, $arch, $submodel) = @_; my @usable_modules; foreach my $mod (sort keys %MODULES) { my %modinfo = %{ $MODULES{$mod} }; # If it uses external libs, the user has to request it specifically next if($modinfo{'external_libs'}); my @cc_list = (); if($modinfo{'cc'}) { @cc_list = keys %{ $modinfo{'cc'} }; } my @os_list = (); if($modinfo{'os'}) { @os_list = keys %{ $modinfo{'os'} }; } my @arch_list = (); if($modinfo{'arch'}) { @arch_list = keys %{ $modinfo{'arch'} }; } next if(scalar @cc_list > 0 && !in_array(\@cc_list, $cc)); next if(scalar @os_list > 0 && !in_array(\@os_list, $os)); next if(scalar @arch_list > 0 && !in_array(\@arch_list, $arch) && !in_array(\@arch_list, $submodel)); push @usable_modules, $mod; } return @usable_modules; } sub os_info_for { my ($os,$hashref) = @_; my %hash = %{ $hashref }; die "Internal error: os_info_for called with undef hash\n" unless %hash; die "Internal error: os_info_for called with an os of defaults\n" if($os eq 'defaults'); return $hash{$os} if(defined($hash{$os}) && $hash{$os} ne ''); return $hash{'defaults'}; } sub os_static_suffix { my $os = $_[0]; return os_info_for($os, \%OS_STATIC_SUFFIX); } sub os_shared_suffix { my $os = $_[0]; return os_info_for($os, \%OS_SHARED_SUFFIX); } sub os_obj_suffix { my $os = $_[0]; return os_info_for($os, \%OS_OBJ_SUFFIX); } sub os_ar_command { my $os = $_[0]; return os_info_for($os, \%OS_AR_COMMAND); } sub os_ar_needs_ranlib { my $os = $_[0]; return os_info_for($os, \%OS_AR_NEEDS_RANLIB); } sub os_install_info { my ($os,$what) = @_; return $doc_dir if($what eq 'docs' && $doc_dir); return $lib_dir if($what eq 'libs' && $lib_dir); return $user_set_root if($what eq 'root' && $user_set_root); return $INSTALL_INFO{$os}{$what} if(defined($INSTALL_INFO{$os}) && defined($INSTALL_INFO{$os}{$what})); return $INSTALL_INFO{'defaults'}{$what}; } sub generate_makefile { my($make_style, $cc, $os, $submodel, $arch, $debug, $no_shared, $dumb_gcc, $lib_src, $check_src, $all_includes, $added_src, @libs_used) = @_; my %all_lib_srcs = (%{ $lib_src }, %{ $added_src }); my $lang_flags = $CC_LANG_FLAGS{$cc}; $lang_flags = "$lang_flags -fpermissive" if($dumb_gcc); my $lib_opt_flags = $CC_LIB_OPT_FLAGS{$cc}; if(!$debug and ($CC_NO_DEBUG_FLAGS{$cc})) { $lib_opt_flags .= ' '.$CC_NO_DEBUG_FLAGS{$cc}; } if($debug and ($CC_DEBUG_FLAGS{$cc})) { $lib_opt_flags .= ' '.$CC_DEBUG_FLAGS{$cc}; } my $mach_opt_flags = ''; if(defined($CC_MACHINE_OPT_FLAGS{$cc}{$submodel})) { $mach_opt_flags = $CC_MACHINE_OPT_FLAGS{$cc}{$submodel}; } elsif(defined($CC_MACHINE_OPT_FLAGS{$cc}{$arch})) { $mach_opt_flags = $CC_MACHINE_OPT_FLAGS{$cc}{$arch}; my $processed_modelname = $submodel; if(defined($CC_MACHINE_OPT_FLAGS_RE{$cc}{$arch})) { $processed_modelname =~ s/$CC_MACHINE_OPT_FLAGS_RE{$cc}{$arch}//; } $mach_opt_flags =~ s/SUBMODEL/$processed_modelname/g; } # This is a default that works on most Unix and Unix-like systems my $ar_command = "ar crs"; my $ar_needs_ranlib = 0; # almost no systems need it anymore # See if there are any over-riding methods. We presume if CC is creating # the static libs, it knows how to create the index itself. if($CC_AR_COMMAND{$cc}) { $ar_command = $CC_AR_COMMAND{$cc}; } elsif(os_ar_command($os)) { $ar_command = os_ar_command($os); $ar_needs_ranlib = 1 if(os_ar_needs_ranlib($os)); } my $so_link_flags = ''; my $so_obj_flags = $CC_SO_OBJ_FLAGS{$cc}; if($no_shared or (!in_array($OS_SUPPORTS_SHARED{$os}, 'all') and !in_array($OS_SUPPORTS_SHARED{$os}, $arch))) { $so_obj_flags = ''; } elsif(defined($CC_SO_LINK_FLAGS{$cc}{$os})) { $so_link_flags = $CC_SO_LINK_FLAGS{$cc}{$os}; } elsif(defined($CC_SO_LINK_FLAGS{$cc}{'default'})) { $so_link_flags = $CC_SO_LINK_FLAGS{$cc}{'default'}; } my $make_shared = 0; $make_shared = 1 if(($so_obj_flags or $so_link_flags) and $OS_SUPPORTS_SHARED{$os}); my $check_opt_flags = $CC_CHECK_OPT_FLAGS{$cc}; my $ccopts = ''; $ccopts .= ' '.$CC_ABI_FLAGS{$cc}{$arch} if(defined($CC_ABI_FLAGS{$cc}{$arch})); $ccopts .= ' '.$CC_ABI_FLAGS{$cc}{$os} if(defined($CC_ABI_FLAGS{$cc}{$os})); $ccopts .= ' '.$CC_ABI_FLAGS{$cc}{'all'} if(defined($CC_ABI_FLAGS{$cc}{'all'})); my $install_root = os_install_info($os, 'root'); open MAKEFILE, ">$MAKE_FILE" or die "Couldn't write $MAKE_FILE ($!)\n"; ################################################## # Ready, set, print! # ################################################## my $cc_bin = $CC_BINARY_NAME{$cc}; # Hack for 10.1, 10.2+ is fixed. Don't have a 10.0.x machine anymore if($os eq "darwin" and $cc eq "gcc") { $cc_bin = "c++"; } my $obj_suffix = os_obj_suffix($os); my $static_suffix = os_static_suffix($os); my @arguments = (\*MAKEFILE, $os, $cc_bin . $ccopts, $lib_opt_flags, $check_opt_flags, $mach_opt_flags, $lang_flags, $CC_WARN_FLAGS{$cc}, $make_shared, $so_obj_flags, $so_link_flags, $obj_suffix, os_shared_suffix($os), $static_suffix, $ar_command, $ar_needs_ranlib, \%all_lib_srcs, $check_src, $all_includes, \%DOCS, $install_root, os_install_info($os, 'headers'), os_install_info($os, 'libs'), os_install_info($os, 'docs'), \@libs_used); if($make_style eq 'unix') { print_unix_makefile(@arguments); } elsif($make_style eq 'nmake') { print_nmake_makefile(@arguments); } else { die "(error): This configure script does not know how to make ", "a makefile for makefile style \"$make_style\"\n"; } close MAKEFILE; } sub print_header { my ($fh, $string) = @_; print $fh '#' x 50, "\n", "# $string", ' 'x(47-length($string)), "#\n", '#' x 50, "\n"; } ################################################## # Print a Unix style makefile # ################################################## sub print_unix_makefile { my ($makefile, $os, $cc, $lib_opt, $check_opt, $mach_opt, $lang_flags, $warn_flags, $make_shared, $so_obj, $so_link, $obj_suffix, $so_suffix, $static_lib_suffix, $ar_command, $use_ranlib, $src, $check, $include_r, $docs, $install_root, $header_dir, $lib_dir, $doc_dir, $lib_list) = @_; my $__TAB__ = "\t"; my $link_to = "-lm"; foreach my $lib (@{ $lib_list }) { $link_to .= " -l" . $lib; } my $lib_flags = '$(LIB_OPT) $(MACH_OPT) $(LANG_FLAGS) $(WARN_FLAGS)'; my $libs = '$(STATIC_LIB)'; if($make_shared) { $lib_flags .= ' $(SO_OBJ_FLAGS)'; $libs .= ' $(SHARED_LIB)'; } my $install_user = os_install_info($os, 'user'); my $install_group = os_install_info($os, 'group'); my $install_cmd_exec = os_install_info($os, 'command'); my $install_cmd_data = os_install_info($os, 'command'); $install_cmd_exec =~ s/OWNER/\$(OWNER)/; $install_cmd_data =~ s/OWNER/\$(OWNER)/; $install_cmd_exec =~ s/GROUP/\$(GROUP)/; $install_cmd_data =~ s/GROUP/\$(GROUP)/; $install_cmd_exec =~ s/MODE/\$(EXEC_MODE)/; $install_cmd_data =~ s/MODE/\$(DATA_MODE)/; ##################### COMMON CODE (PARTIALLY) ###################### sub file_list { my ($spaces, $put_in, $from, $to, %files) = @_; my $len = $spaces; my $list; foreach (sort keys %files) { my $file = $_; my $dir = $put_in; if(!defined($dir)) { $dir = $files{$_}; } if($len > 60) { $list .= "\\\n" . ' 'x$spaces; $len = $spaces; } if(defined($from) and defined($to)) { $file =~ s/$from/$to/; } if(defined($dir)) { $list .= File::Spec->catfile ($dir, $file) . ' '; $len += length($file) + length($dir); } else { $list .= $file . ' '; $len += length($file); } } return $list; } my $includes = file_list(16, undef, undef, undef, %$include_r); my $lib_obj = file_list(16, $BUILD_LIB_DIR, '(\.cpp$|\.s$|\.S$)', '.'.$obj_suffix, %$src, %added_src); my $check_obj = file_list(16, $BUILD_CHECK_DIR, '.cpp', '.'.$obj_suffix, %$check); my $doc_list = file_list(16, undef, undef, undef, %$docs); ##################### / COMMON CODE (PARTIALLY) ###################### print_header($makefile, 'Compiler Options'); print $makefile <catfile ($files{$_}, $_); my $obj_file = File::Spec->catfile ($dir, $_); $obj_file =~ s/\.cpp$/.$obj_suffix/; $obj_file =~ s/\.s$/.$obj_suffix/; $obj_file =~ s/\.S$/.$obj_suffix/; print $fh "$obj_file: $src_file\n", "\t\$(CXX) -I$BUILD_INCLUDE_DIR $flags -c \$? -o \$@\n\n"; } } print_build_cmds($makefile, $BUILD_LIB_DIR, '$(LIB_FLAGS)', $obj_suffix, %$src, %added_src); print_build_cmds($makefile, $BUILD_CHECK_DIR, '$(CHECK_FLAGS)', $obj_suffix, %$check); print_header($makefile, 'Link Commands'); print $makefile <catfile ($files{$_}, $_); my $obj_file = File::Spec->catfile ($dir, $_); $obj_file =~ s/.cpp/.$obj_suffix/; print $fh "$obj_file: $src_file\n", "\t\$(CXX) -I$BUILD_INCLUDE_DIR $flags /c \$? /Fo\$@\n\n"; } } print_build_cmds_nmake($makefile, $BUILD_LIB_DIR, '$(LIB_FLAGS)', $obj_suffix, %$src, %added_src); print_build_cmds_nmake($makefile, $BUILD_CHECK_DIR, '$(CHECK_FLAGS)', $obj_suffix, %$check); print_header($makefile, 'Link Commands'); print $makefile <botan-config" or die "Couldn't write to botan-config ($!)"; print PKGCONFIG <read)) { next if($_ eq '.' or $_ eq '..'); my $arch = $_; my $filename = File::Spec->catfile($_[0], $arch); open ARCHFILE, "<$filename" or die "Couldn't open $filename, ($!)"; $ARCH{$arch} = $arch; while() { $_ = process($_); next unless $_; $REALNAME{$arch} = $1 if(/^realname \"(.*)\"/); $DEFAULT_SUBMODEL{$arch} = $1 if(/^default_submodel (.*)$/); # Read in a list of aliases and add them to ARCH_ALIAS if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); $ARCH_ALIAS{$_} = $arch; } } # Read in a list of submodels and add them to ARCH if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); $ARCH{$_} = $arch; } } # Read in a list of submodel aliases and add them to SUBMODEL_ALIAS if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); m/^(\S*) -> (\S*)$/; $SUBMODEL_ALIAS{$1} = $2; } } } } undef $dir; } sub set_os_defines { my $dir = new DirHandle $_[0]; if(!defined $dir) { die "Couldn't open directory $_[0] ($!)"; } while(defined($_ = $dir->read)) { next if($_ eq '.' or $_ eq '..'); my $os = $_; my $filename = File::Spec->catfile($_[0], $os); open OSFILE, "<$filename" or die "Couldn't open $filename, ($!)"; $OS_SHARED_SUFFIX{$os} = ''; $OS_AR_COMMAND{$os} = ''; # Default values while() { $_ = process($_); next unless $_; $REALNAME{$os} = $1 if(/^realname \"(.*)\"/); $OS_TYPE{$os} = $1 if(/^os_type (.*)/); $OS_AR_COMMAND{$os} = $1 if(/^ar_command \"(.*)\"/); $OS_AR_NEEDS_RANLIB{$os} = 1 if(/^ar_needs_ranlib yes$/); $OS_AR_NEEDS_RANLIB{$os} = 0 if(/^ar_needs_ranlib no$/); $OS_OBJ_SUFFIX{$os} = $1 if(/^obj_suffix (.*)/); $OS_SHARED_SUFFIX{$os} = $1 if(/^so_suffix (.*)/); $OS_STATIC_SUFFIX{$os} = $1 if(/^static_suffix (.*)/); $INSTALL_INFO{$os}{'root'} = $1 if(/^install_root (.*)/); $INSTALL_INFO{$os}{'headers'} = $1 if(/^header_dir (.*)/); $INSTALL_INFO{$os}{'libs'} = $1 if(/^lib_dir (.*)/); $INSTALL_INFO{$os}{'docs'} = $1 if(/^doc_dir (.*)/); $INSTALL_INFO{$os}{'user'} = $1 if(/^install_user (.*)/); $INSTALL_INFO{$os}{'group'} = $1 if(/^install_group (.*)/); $INSTALL_INFO{$os}{'command'} = $1 if(/^install_cmd (.*)/); if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); $OS_ALIAS{$_} = $os; } } if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); push @{$OS_SUPPORTS_SHARED{$os}}, $_; } } # Read in a list of architectures and add them to OS_SUPPORTS_ARCH if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); push @{$OS_SUPPORTS_ARCH{$os}}, $_; } } } } undef $dir; } sub set_cc_defines { my $dir = new DirHandle $_[0]; if(!defined $dir) { die "Couldn't open directory $_[0] ($!)"; } while(defined($_ = $dir->read)) { next if($_ eq '.' or $_ eq '..'); my $cc = $_; my $filename = File::Spec->catfile($_[0], $cc); open CCFILE, "<$filename" or die "Couldn't open $filename, ($!)"; # Default to empty values, so they don't have to be explicitly set $CC_LIB_OPT_FLAGS{$cc} = $CC_CHECK_OPT_FLAGS{$cc} = $CC_LANG_FLAGS{$cc} = $CC_WARN_FLAGS{$cc} = $CC_SO_OBJ_FLAGS{$cc} = $CC_DEBUG_FLAGS{$cc} = $CC_AR_COMMAND{$cc} = $CC_NO_DEBUG_FLAGS{$cc} = ''; while() { $_ = process($_); next unless $_; $REALNAME{$cc} = $1 if(/^realname \"(.*)\"/); $CC_BINARY_NAME{$cc} = $1 if(/^binary_name \"(.*)\"/); $CC_LIB_OPT_FLAGS{$cc} = $1 if(/^lib_opt_flags \"(.*)\"/); $CC_CHECK_OPT_FLAGS{$cc} = $1 if(/^check_opt_flags \"(.*)\"/); $CC_AR_COMMAND{$cc} = $1 if(/^ar_command \"(.*)\"/); $CC_LANG_FLAGS{$cc} = $1 if(/^lang_flags \"(.*)\"/); $CC_WARN_FLAGS{$cc} = $1 if(/^warning_flags \"(.*)\"/); $CC_SO_OBJ_FLAGS{$cc} = $1 if(/^so_obj_flags \"(.*)\"/); $CC_DEBUG_FLAGS{$cc} = $1 if(/^debug_flags \"(.*)\"/); $CC_NO_DEBUG_FLAGS{$cc} = $1 if(/^no_debug_flags \"(.*)\"/); $MAKEFILE_STYLE{$cc} = $1 if(/^makefile_style (.*)/); # Read in a list of supported CPU types if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); push @{$CC_SUPPORTS_ARCH{$cc}}, $_; } } # Read in a list of supported OSes if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); push @{$CC_SUPPORTS_OS{$cc}}, $_; } } # Read in a list of machine optimization flags if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); m/^(\S*) -> \"(.*)\" ?(.*)?$/; $CC_MACHINE_OPT_FLAGS{$cc}{$1} = $2; if($3 ne '') { $CC_MACHINE_OPT_FLAGS_RE{$cc}{$1} = $3; } } } # Some systems need certain flags passed for linking as well # (usually these change the ABI somehow). We just append this # value to the CXX variable, so it's used for all ops. if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); m/^(\S*) -> \"(.*)\"$/; $CC_ABI_FLAGS{$cc}{$1} = $2; } } # Read in a list of flags to created a shared lib (and set soname) if(/^$/) { while(1) { $_ = process($_ = ); next unless $_; last if(m@^$@); m/^(\S*) -> \"(.*)\"$/; $CC_SO_LINK_FLAGS{$cc}{$1} = $2; } } } } undef $dir; }