Configure: let INCLUDEs set on binaries "trickle down" to the objects
[openssl.git] / Configure
1 #! /usr/bin/env perl
2 # -*- mode: perl; -*-
3 # Copyright 2016 The OpenSSL Project Authors. All Rights Reserved.
4 #
5 # Licensed under the OpenSSL license (the "License").  You may not use
6 # this file except in compliance with the License.  You can obtain a copy
7 # in the file LICENSE in the source distribution or at
8 # https://www.openssl.org/source/license.html
9
10 ##  Configure -- OpenSSL source tree configuration script
11
12 use 5.10.0;
13 use strict;
14 use FindBin;
15 use lib "$FindBin::Bin/util/perl";
16 use File::Basename;
17 use File::Spec::Functions qw/:DEFAULT abs2rel rel2abs/;
18 use File::Path qw/mkpath/;
19 use OpenSSL::Glob;
20
21 # see INSTALL for instructions.
22
23 my $usage="Usage: Configure [no-<cipher> ...] [enable-<cipher> ...] [-Dxxx] [-lxxx] [-Lxxx] [-fxxx] [-Kxxx] [no-hw-xxx|no-hw] [[no-]threads] [[no-]shared] [[no-]zlib|zlib-dynamic] [no-asm] [no-dso] [no-egd] [sctp] [386] [--prefix=DIR] [--openssldir=OPENSSLDIR] [--with-xxx[=vvv]] [--config=FILE] os/compiler[:flags]\n";
24
25 # Options:
26 #
27 # --config      add the given configuration file, which will be read after
28 #               any "Configurations*" files that are found in the same
29 #               directory as this script.
30 # --prefix      prefix for the OpenSSL installation, which includes the
31 #               directories bin, lib, include, share/man, share/doc/openssl
32 #               This becomes the value of INSTALLTOP in Makefile
33 #               (Default: /usr/local)
34 # --openssldir  OpenSSL data area, such as openssl.cnf, certificates and keys.
35 #               If it's a relative directory, it will be added on the directory
36 #               given with --prefix.
37 #               This becomes the value of OPENSSLDIR in Makefile and in C.
38 #               (Default: PREFIX/ssl)
39 #
40 # --cross-compile-prefix Add specified prefix to binutils components.
41 #
42 # --api         One of 0.9.8, 1.0.0 or 1.1.0.  Do not compile support for
43 #               interfaces deprecated as of the specified OpenSSL version.
44 #
45 # no-hw-xxx     do not compile support for specific crypto hardware.
46 #               Generic OpenSSL-style methods relating to this support
47 #               are always compiled but return NULL if the hardware
48 #               support isn't compiled.
49 # no-hw         do not compile support for any crypto hardware.
50 # [no-]threads  [don't] try to create a library that is suitable for
51 #               multithreaded applications (default is "threads" if we
52 #               know how to do it)
53 # [no-]shared   [don't] try to create shared libraries when supported.
54 # [no-]pic      [don't] try to build position independent code when supported.
55 #               If disabled, it also disables shared and dynamic-engine.
56 # no-asm        do not use assembler
57 # no-dso        do not compile in any native shared-library methods. This
58 #               will ensure that all methods just return NULL.
59 # no-egd        do not compile support for the entropy-gathering daemon APIs
60 # [no-]zlib     [don't] compile support for zlib compression.
61 # zlib-dynamic  Like "zlib", but the zlib library is expected to be a shared
62 #               library and will be loaded in run-time by the OpenSSL library.
63 # sctp          include SCTP support
64 # enable-weak-ssl-ciphers
65 #               Enable weak ciphers that are disabled by default.
66 # 386           generate 80386 code in assembly modules
67 # no-sse2       disables IA-32 SSE2 code in assembly modules, the above
68 #               mentioned '386' option implies this one
69 # no-<cipher>   build without specified algorithm (rsa, idea, rc5, ...)
70 # -<xxx> +<xxx> compiler options are passed through
71 # -static       while -static is also a pass-through compiler option (and
72 #               as such is limited to environments where it's actually
73 #               meaningful), it triggers a number configuration options,
74 #               namely no-dso, no-pic, no-shared and no-threads. It is
75 #               argued that the only reason to produce statically linked
76 #               binaries (and in context it means executables linked with
77 #               -static flag, and not just executables linked with static
78 #               libcrypto.a) is to eliminate dependency on specific run-time,
79 #               a.k.a. libc version. The mentioned config options are meant
80 #               to achieve just that. Unfortunately on Linux it's impossible
81 #               to eliminate the dependency completely for openssl executable
82 #               because of getaddrinfo and gethostbyname calls, which can
83 #               invoke dynamically loadable library facility anyway to meet
84 #               the lookup requests. For this reason on Linux statically
85 #               linked openssl executable has rather debugging value than
86 #               production quality.
87 #
88 # DEBUG_SAFESTACK use type-safe stacks to enforce type-safety on stack items
89 #               provided to stack calls. Generates unique stack functions for
90 #               each possible stack type.
91 # BN_LLONG      use the type 'long long' in crypto/bn/bn.h
92 # RC4_CHAR      use 'char' instead of 'int' for RC4_INT in crypto/rc4/rc4.h
93 # Following are set automatically by this script
94 #
95 # MD5_ASM       use some extra md5 assembler,
96 # SHA1_ASM      use some extra sha1 assembler, must define L_ENDIAN for x86
97 # RMD160_ASM    use some extra ripemd160 assembler,
98 # SHA256_ASM    sha256_block is implemented in assembler
99 # SHA512_ASM    sha512_block is implemented in assembler
100 # AES_ASM       AES_[en|de]crypt is implemented in assembler
101
102 # Minimum warning options... any contributions to OpenSSL should at least get
103 # past these.
104
105 # DEBUG_UNUSED enables __owur (warn unused result) checks.
106 my $gcc_devteam_warn = "-DDEBUG_UNUSED"
107         # -DPEDANTIC complements -pedantic and is meant to mask code that
108         # is not strictly standard-compliant and/or implementation-specific,
109         # e.g. inline assembly, disregards to alignment requirements, such
110         # that -pedantic would complain about. Incidentally -DPEDANTIC has
111         # to be used even in sanitized builds, because sanitizer too is
112         # supposed to and does take notice of non-standard behaviour. Then
113         # -pedantic with pre-C9x compiler would also complain about 'long
114         # long' not being supported. As 64-bit algorithms are common now,
115         # it grew impossible to resolve this without sizeable additional
116         # code, so we just tell compiler to be pedantic about everything
117         # but 'long long' type.
118         . " -DPEDANTIC -pedantic -Wno-long-long"
119         . " -Wall"
120         . " -Wextra"
121         . " -Wno-unused-parameter"
122         . " -Wno-missing-field-initializers"
123         . " -Wsign-compare"
124         . " -Wmissing-prototypes"
125         . " -Wshadow"
126         . " -Wformat"
127         . " -Wtype-limits"
128         . " -Wundef"
129         . " -Werror"
130         ;
131
132 # These are used in addition to $gcc_devteam_warn when the compiler is clang.
133 # TODO(openssl-team): fix problems and investigate if (at least) the
134 # following warnings can also be enabled:
135 #       -Wswitch-enum
136 #       -Wcast-align
137 #       -Wunreachable-code
138 #       -Wlanguage-extension-token -- no, we use asm()
139 #       -Wunused-macros -- no, too tricky for BN and _XOPEN_SOURCE etc
140 #       -Wextended-offsetof -- no, needed in CMS ASN1 code
141 my $clang_devteam_warn = ""
142         . " -Qunused-arguments"
143         . " -Wno-language-extension-token"
144         . " -Wno-extended-offsetof"
145         . " -Wconditional-uninitialized"
146         . " -Wincompatible-pointer-types-discards-qualifiers"
147         . " -Wmissing-variable-declarations"
148         ;
149
150 # This adds backtrace information to the memory leak info.  Is only used
151 # when crypto-mdebug-backtrace is enabled.
152 my $memleak_devteam_backtrace = "-rdynamic";
153
154 my $strict_warnings = 0;
155
156 # As for $BSDthreads. Idea is to maintain "collective" set of flags,
157 # which would cover all BSD flavors. -pthread applies to them all,
158 # but is treated differently. OpenBSD expands is as -D_POSIX_THREAD
159 # -lc_r, which is sufficient. FreeBSD 4.x expands it as -lc_r,
160 # which has to be accompanied by explicit -D_THREAD_SAFE and
161 # sometimes -D_REENTRANT. FreeBSD 5.x expands it as -lc_r, which
162 # seems to be sufficient?
163 our $BSDthreads="-pthread -D_THREAD_SAFE -D_REENTRANT";
164
165 #
166 # API compatibility name to version number mapping.
167 #
168 my $maxapi = "1.1.0";           # API for "no-deprecated" builds
169 my $apitable = {
170     "1.1.0" => "0x10100000L",
171     "1.0.0" => "0x10000000L",
172     "0.9.8" => "0x00908000L",
173 };
174
175 our %table = ();
176 our %config = ();
177 our %withargs = ();
178
179 # Forward declarations ###############################################
180
181 # read_config(filename)
182 #
183 # Reads a configuration file and populates %table with the contents
184 # (which the configuration file places in %targets).
185 sub read_config;
186
187 # resolve_config(target)
188 #
189 # Resolves all the late evaluations, inheritances and so on for the
190 # chosen target and any target it inherits from.
191 sub resolve_config;
192
193
194 # Information collection #############################################
195
196 # Unified build supports separate build dir
197 my $srcdir = catdir(absolutedir(dirname($0))); # catdir ensures local syntax
198 my $blddir = catdir(absolutedir("."));         # catdir ensures local syntax
199 my $dofile = abs2rel(catfile($srcdir, "util/dofile.pl"));
200
201 my $local_config_envname = 'OPENSSL_LOCAL_CONFIG_DIR';
202
203 $config{sourcedir} = abs2rel($srcdir);
204 $config{builddir} = abs2rel($blddir);
205
206 # Collect reconfiguration information if needed
207 my @argvcopy=@ARGV;
208
209 if (grep /^reconf(igure)?$/, @argvcopy) {
210     if (-f "./configdata.pm") {
211         my $file = "./configdata.pm";
212         unless (my $return = do $file) {
213             die "couldn't parse $file: $@" if $@;
214             die "couldn't do $file: $!"    unless defined $return;
215             die "couldn't run $file"       unless $return;
216         }
217
218         @argvcopy = defined($configdata::config{perlargv}) ?
219             @{$configdata::config{perlargv}} : ();
220         die "Incorrect data to reconfigure, please do a normal configuration\n"
221             if (grep(/^reconf/,@argvcopy));
222         $ENV{CROSS_COMPILE} = $configdata::config{cross_compile_prefix}
223             if defined($configdata::config{cross_compile_prefix});
224         $ENV{CC} = $configdata::config{cc}
225             if defined($configdata::config{cc});
226         $ENV{BUILDFILE} = $configdata::config{build_file}
227             if defined($configdata::config{build_file});
228         $ENV{$local_config_envname} = $configdata::config{local_config_dir}
229             if defined($configdata::config{local_config_dir});
230
231         print "Reconfiguring with: ", join(" ",@argvcopy), "\n";
232         print "    CROSS_COMPILE = ",$ENV{CROSS_COMPILE},"\n"
233             if $ENV{CROSS_COMPILE};
234         print "    CC = ",$ENV{CC},"\n" if $ENV{CC};
235         print "    BUILDFILE = ",$ENV{BUILDFILE},"\n" if $ENV{BUILDFILE};
236         print "    $local_config_envname = ",$ENV{$local_config_envname},"\n"
237             if $ENV{$local_config_envname};
238     } else {
239         die "Insufficient data to reconfigure, please do a normal configuration\n";
240     }
241 }
242
243 $config{perlargv} = [ @argvcopy ];
244
245 # Collect version numbers
246 $config{version} = "unknown";
247 $config{version_num} = "unknown";
248 $config{shlib_version_number} = "unknown";
249 $config{shlib_version_history} = "unknown";
250
251 collect_information(
252     collect_from_file(catfile($srcdir,'include/openssl/opensslv.h')),
253     qr/OPENSSL.VERSION.TEXT.*OpenSSL (\S+) / => sub { $config{version} = $1; },
254     qr/OPENSSL.VERSION.NUMBER.*(0x\S+)/      => sub { $config{version_num}=$1 },
255     qr/SHLIB_VERSION_NUMBER *"([^"]+)"/      => sub { $config{shlib_version_number}=$1 },
256     qr/SHLIB_VERSION_HISTORY *"([^"]*)"/     => sub { $config{shlib_version_history}=$1 }
257     );
258 if ($config{shlib_version_history} ne "") { $config{shlib_version_history} .= ":"; }
259
260 ($config{major}, $config{minor})
261     = ($config{version} =~ /^([0-9]+)\.([0-9\.]+)/);
262 ($config{shlib_major}, $config{shlib_minor})
263     = ($config{shlib_version_number} =~ /^([0-9]+)\.([0-9\.]+)/);
264 die "erroneous version information in opensslv.h: ",
265     "$config{major}, $config{minor}, $config{shlib_major}, $config{shlib_minor}\n"
266     if ($config{major} eq "" || $config{minor} eq ""
267         || $config{shlib_major} eq "" ||  $config{shlib_minor} eq "");
268
269 # Collect target configurations
270
271 my $pattern = catfile(dirname($0), "Configurations", "*.conf");
272 foreach (sort glob($pattern)) {
273     &read_config($_);
274 }
275
276 if (defined $ENV{$local_config_envname}) {
277     if ($^O eq 'VMS') {
278         # VMS environment variables are logical names,
279         # which can be used as is
280         $pattern = $local_config_envname . ':' . '*.conf';
281     } else {
282         $pattern = catfile($ENV{$local_config_envname}, '*.conf');
283     }
284
285     foreach (sort glob($pattern)) {
286         &read_config($_);
287     }
288 }
289
290
291 print "Configuring OpenSSL version $config{version} ($config{version_num})\n";
292
293 $config{prefix}="";
294 $config{openssldir}="";
295 $config{processor}="";
296 $config{libdir}="";
297 $config{cross_compile_prefix}="";
298 $config{fipslibdir}="/usr/local/ssl/fips-2.0/lib/";
299 my $nofipscanistercheck=0;
300 $config{baseaddr}="0xFB00000";
301 my $auto_threads=1;    # enable threads automatically? true by default
302 my $default_ranlib;
303 $config{fips}=0;
304
305 # Top level directories to build
306 $config{dirs} = [ "crypto", "ssl", "engines", "apps", "test", "util", "tools", "fuzz" ];
307 # crypto/ subdirectories to build
308 $config{sdirs} = [
309     "objects",
310     "md2", "md4", "md5", "sha", "mdc2", "hmac", "ripemd", "whrlpool", "poly1305", "blake2",
311     "des", "aes", "rc2", "rc4", "rc5", "idea", "bf", "cast", "camellia", "seed", "chacha", "modes",
312     "bn", "ec", "rsa", "dsa", "dh", "dso", "engine",
313     "buffer", "bio", "stack", "lhash", "rand", "err",
314     "evp", "asn1", "pem", "x509", "x509v3", "conf", "txt_db", "pkcs7", "pkcs12", "comp", "ocsp", "ui",
315     "cms", "ts", "srp", "cmac", "ct", "async", "kdf"
316     ];
317
318 # Known TLS and DTLS protocols
319 my @tls = qw(ssl3 tls1 tls1_1 tls1_2);
320 my @dtls = qw(dtls1 dtls1_2);
321
322 # Explicitly known options that are possible to disable.  They can
323 # be regexps, and will be used like this: /^no-${option}$/
324 # For developers: keep it sorted alphabetically
325
326 my @disablables = (
327     "afalgeng",
328     "asan",
329     "asm",
330     "async",
331     "autoalginit",
332     "autoerrinit",
333     "bf",
334     "blake2",
335     "camellia",
336     "capieng",
337     "cast",
338     "chacha",
339     "cmac",
340     "cms",
341     "comp",
342     "crypto-mdebug",
343     "crypto-mdebug-backtrace",
344     "ct",
345     "deprecated",
346     "des",
347     "dgram",
348     "dh",
349     "dsa",
350     "dso",
351     "dtls",
352     "dynamic-engine",
353     "ec",
354     "ec2m",
355     "ecdh",
356     "ecdsa",
357     "ec_nistp_64_gcc_128",
358     "egd",
359     "engine",
360     "err",
361     "filenames",
362     "fuzz-libfuzzer",
363     "fuzz-afl",
364     "gost",
365     "heartbeats",
366     "hw(-.+)?",
367     "idea",
368     "makedepend",
369     "md2",
370     "md4",
371     "mdc2",
372     "msan",
373     "multiblock",
374     "nextprotoneg",
375     "ocb",
376     "ocsp",
377     "pic",
378     "poly1305",
379     "posix-io",
380     "psk",
381     "rc2",
382     "rc4",
383     "rc5",
384     "rdrand",
385     "rfc3779",
386     "rmd160",
387     "scrypt",
388     "sctp",
389     "seed",
390     "shared",
391     "sock",
392     "srp",
393     "srtp",
394     "sse2",
395     "ssl",
396     "ssl-trace",
397     "static-engine",
398     "stdio",
399     "threads",
400     "tls",
401     "ts",
402     "ubsan",
403     "ui",
404     "unit-test",
405     "whirlpool",
406     "weak-ssl-ciphers",
407     "zlib",
408     "zlib-dynamic",
409     );
410 foreach my $proto ((@tls, @dtls))
411         {
412         push(@disablables, $proto);
413         push(@disablables, "$proto-method");
414         }
415
416 my %deprecated_disablables = (
417     "ssl2" => undef,
418     "buf-freelists" => undef,
419     "ripemd" => "rmd160"
420     );
421
422 # All of the following is disabled by default (RC5 was enabled before 0.9.8):
423
424 our %disabled = ( # "what"         => "comment"
425                   "asan"                => "default",
426                   "crypto-mdebug"       => "default",
427                   "crypto-mdebug-backtrace" => "default",
428                   "ec_nistp_64_gcc_128" => "default",
429                   "egd"                 => "default",
430                   "fuzz-libfuzzer"      => "default",
431                   "fuzz-afl"            => "default",
432                   "heartbeats"          => "default",
433                   "md2"                 => "default",
434                   "msan"                => "default",
435                   "rc5"                 => "default",
436                   "sctp"                => "default",
437                   "ssl-trace"           => "default",
438                   "ssl3"                => "default",
439                   "ssl3-method"         => "default",
440                   "ubsan"               => "default",
441                   "unit-test"           => "default",
442                   "weak-ssl-ciphers"    => "default",
443                   "zlib"                => "default",
444                   "zlib-dynamic"        => "default",
445                 );
446
447 # Note: => pair form used for aesthetics, not to truly make a hash table
448 my @disable_cascades = (
449     # "what"            => [ "cascade", ... ]
450     sub { $config{processor} eq "386" }
451                         => [ "sse2" ],
452     "ssl"               => [ "ssl3" ],
453     "ssl3-method"       => [ "ssl3" ],
454     "zlib"              => [ "zlib-dynamic" ],
455     "des"               => [ "mdc2" ],
456     "ec"                => [ "ecdsa", "ecdh" ],
457
458     "dgram"             => [ "dtls", "sctp" ],
459     "sock"              => [ "dgram" ],
460     "dtls"              => [ @dtls ],
461     sub { 0 == scalar grep { !$disabled{$_} } @dtls }
462                         => [ "dtls" ],
463
464     "tls"               => [ @tls ],
465     sub { 0 == scalar grep { !$disabled{$_} } @tls }
466                         => [ "tls" ],
467
468     "crypto-mdebug"     => [ "crypto-mdebug-backtrace" ],
469
470     # Without DSO, we can't load dynamic engines, so don't build them dynamic
471     "dso"               => [ "dynamic-engine" ],
472
473     # Without position independent code, there can be no shared libraries or DSOs
474     "pic"               => [ "shared" ],
475     "shared"            => [ "dynamic-engine" ],
476     "engine"            => [ "afalgeng" ],
477
478     # no-autoalginit is only useful when building non-shared
479     "autoalginit"       => [ "shared", "apps" ],
480
481     "stdio"             => [ "apps", "capieng", "egd" ],
482     "apps"              => [ "tests" ],
483     "comp"              => [ "zlib" ],
484     sub { !$disabled{"unit-test"} } => [ "heartbeats" ],
485
486     sub { !$disabled{"msan"} } => [ "asm" ],
487     );
488
489 # Avoid protocol support holes.  Also disable all versions below N, if version
490 # N is disabled while N+1 is enabled.
491 #
492 my @list = (reverse @tls);
493 while ((my $first, my $second) = (shift @list, shift @list)) {
494     last unless @list;
495     push @disable_cascades, ( sub { !$disabled{$first} && $disabled{$second} }
496                               => [ @list ] );
497     unshift @list, $second;
498 }
499 my @list = (reverse @dtls);
500 while ((my $first, my $second) = (shift @list, shift @list)) {
501     last unless @list;
502     push @disable_cascades, ( sub { !$disabled{$first} && $disabled{$second} }
503                               => [ @list ] );
504     unshift @list, $second;
505 }
506
507 # Explicit "no-..." options will be collected in %disabled along with the defaults.
508 # To remove something from %disabled, use "enable-foo".
509 # For symmetry, "disable-foo" is a synonym for "no-foo".
510
511 my $no_sse2=0;
512
513 &usage if ($#ARGV < 0);
514
515 my $user_cflags="";
516 my @user_defines=();
517 $config{openssl_api_defines}=[];
518 $config{openssl_algorithm_defines}=[];
519 $config{openssl_thread_defines}=[];
520 $config{openssl_sys_defines}=[];
521 $config{openssl_other_defines}=[];
522 my $libs="";
523 my $target="";
524 $config{options}="";
525 $config{build_type} = "release";
526
527 my %unsupported_options = ();
528 my %deprecated_options = ();
529 while (@argvcopy)
530         {
531         $_ = shift @argvcopy;
532         # VMS is a case insensitive environment, and depending on settings
533         # out of our control, we may receive options uppercased.  Let's
534         # downcase at least the part before any equal sign.
535         if ($^O eq "VMS")
536                 {
537                 s/^([^=]*)/lc($1)/e;
538                 }
539         s /^-no-/no-/; # some people just can't read the instructions
540
541         # rewrite some options in "enable-..." form
542         s /^-?-?shared$/enable-shared/;
543         s /^sctp$/enable-sctp/;
544         s /^threads$/enable-threads/;
545         s /^zlib$/enable-zlib/;
546         s /^zlib-dynamic$/enable-zlib-dynamic/;
547
548         if (/^(no|disable|enable)-(.+)$/)
549                 {
550                 my $word = $2;
551                 if (!exists $deprecated_disablables{$word}
552                         && !grep { $word =~ /^${_}$/ } @disablables)
553                         {
554                         $unsupported_options{$_} = 1;
555                         next;
556                         }
557                 }
558         if (/^no-(.+)$/ || /^disable-(.+)$/)
559                 {
560                 foreach my $proto ((@tls, @dtls))
561                         {
562                         if ($1 eq "$proto-method")
563                                 {
564                                 $disabled{"$proto"} = "option($proto-method)";
565                                 last;
566                                 }
567                         }
568                 if ($1 eq "dtls")
569                         {
570                         foreach my $proto (@dtls)
571                                 {
572                                 $disabled{$proto} = "option(dtls)";
573                                 }
574                         $disabled{"dtls"} = "option(dtls)";
575                         }
576                 elsif ($1 eq "ssl")
577                         {
578                         # Last one of its kind
579                         $disabled{"ssl3"} = "option(ssl)";
580                         }
581                 elsif ($1 eq "tls")
582                         {
583                         # XXX: Tests will fail if all SSL/TLS
584                         # protocols are disabled.
585                         foreach my $proto (@tls)
586                                 {
587                                 $disabled{$proto} = "option(tls)";
588                                 }
589                         }
590                 elsif ($1 eq "static-engine")
591                         {
592                         delete $disabled{"dynamic-engine"};
593                         }
594                 elsif ($1 eq "dynamic-engine")
595                         {
596                         $disabled{"dynamic-engine"} = "option";
597                         }
598                 elsif (exists $deprecated_disablables{$1})
599                         {
600                         $deprecated_options{$_} = 1;
601                         if (defined $deprecated_disablables{$1})
602                                 {
603                                 $disabled{$deprecated_disablables{$1}} = "option";
604                                 }
605                         }
606                 else
607                         {
608                         $disabled{$1} = "option";
609                         }
610                 # No longer an automatic choice
611                 $auto_threads = 0 if ($1 eq "threads");
612                 }
613         elsif (/^enable-(.+)$/)
614                 {
615                 if ($1 eq "static-engine")
616                         {
617                         $disabled{"dynamic-engine"} = "option";
618                         }
619                 elsif ($1 eq "dynamic-engine")
620                         {
621                         delete $disabled{"dynamic-engine"};
622                         }
623                 elsif ($1 eq "zlib-dynamic")
624                         {
625                         delete $disabled{"zlib"};
626                         }
627                 my $algo = $1;
628                 delete $disabled{$algo};
629
630                 # No longer an automatic choice
631                 $auto_threads = 0 if ($1 eq "threads");
632                 }
633         elsif (/^--strict-warnings$/)
634                 {
635                 $strict_warnings = 1;
636                 }
637         elsif (/^--debug$/)
638                 {
639                 $config{build_type} = "debug";
640                 }
641         elsif (/^--release$/)
642                 {
643                 $config{build_type} = "release";
644                 }
645         elsif (/^386$/)
646                 { $config{processor}=386; }
647         elsif (/^fips$/)
648                 {
649                 $config{fips}=1;
650                 }
651         elsif (/^rsaref$/)
652                 {
653                 # No RSAref support any more since it's not needed.
654                 # The check for the option is there so scripts aren't
655                 # broken
656                 }
657         elsif (/^nofipscanistercheck$/)
658                 {
659                 $config{fips} = 1;
660                 $nofipscanistercheck = 1;
661                 }
662         elsif (/^[-+]/)
663                 {
664                 if (/^--prefix=(.*)$/)
665                         {
666                         $config{prefix}=$1;
667                         die "Directory given with --prefix MUST be absolute\n"
668                                 unless file_name_is_absolute($config{prefix});
669                         }
670                 elsif (/^--api=(.*)$/)
671                         {
672                         $config{api}=$1;
673                         }
674                 elsif (/^--libdir=(.*)$/)
675                         {
676                         $config{libdir}=$1;
677                         }
678                 elsif (/^--openssldir=(.*)$/)
679                         {
680                         $config{openssldir}=$1;
681                         }
682                 elsif (/^--with-zlib-lib=(.*)$/)
683                         {
684                         $withargs{zlib_lib}=$1;
685                         }
686                 elsif (/^--with-zlib-include=(.*)$/)
687                         {
688                         $withargs{zlib_include}=$1;
689                         }
690                 elsif (/^--with-fuzzer-lib=(.*)$/)
691                         {
692                         $withargs{fuzzer_lib}=$1;
693                         }
694                 elsif (/^--with-fuzzer-include=(.*)$/)
695                         {
696                         $withargs{fuzzer_include}=$1;
697                         }
698                 elsif (/^--with-fipslibdir=(.*)$/)
699                         {
700                         $config{fipslibdir}="$1/";
701                         }
702                 elsif (/^--with-baseaddr=(.*)$/)
703                         {
704                         $config{baseaddr}="$1";
705                         }
706                 elsif (/^--cross-compile-prefix=(.*)$/)
707                         {
708                         $config{cross_compile_prefix}=$1;
709                         }
710                 elsif (/^--config=(.*)$/)
711                         {
712                         read_config $1;
713                         }
714                 elsif (/^-[lL](.*)$/ or /^-Wl,/)
715                         {
716                         $libs.=$_." ";
717                         }
718                 elsif (/^-rpath$/ or /^-R$/)
719                         # -rpath is the OSF1 rpath flag
720                         # -R is the old Solaris rpath flag
721                         {
722                         my $rpath = shift(@argvcopy) || "";
723                         $rpath .= " " if $rpath ne "";
724                         $libs.=$_." ".$rpath;
725                         }
726                 elsif (/^-static$/)
727                         {
728                         $libs.=$_." ";
729                         $disabled{"dso"} = "forced";
730                         $disabled{"pic"} = "forced";
731                         $disabled{"shared"} = "forced";
732                         $disabled{"threads"} = "forced";
733                         }
734                 elsif (/^-D(.*)$/)
735                         {
736                         push @user_defines, $1;
737                         }
738                 else    # common if (/^[-+]/), just pass down...
739                         {
740                         $_ =~ s/%([0-9a-f]{1,2})/chr(hex($1))/gei;
741                         $user_cflags.=" ".$_;
742                         }
743                 }
744         else
745                 {
746                 die "target already defined - $target (offending arg: $_)\n" if ($target ne "");
747                 $target=$_;
748                 }
749         unless ($_ eq $target || /^no-/ || /^disable-/)
750                 {
751                 # "no-..." follows later after implied disactivations
752                 # have been derived.  (Don't take this too seriously,
753                 # we really only write OPTIONS to the Makefile out of
754                 # nostalgia.)
755
756                 if ($config{options} eq "")
757                         { $config{options} = $_; }
758                 else
759                         { $config{options} .= " ".$_; }
760                 }
761
762         if (defined($config{api}) && !exists $apitable->{$config{api}}) {
763                 die "***** Unsupported api compatibility level: $config{api}\n",
764         }
765
766         if (keys %deprecated_options)
767                 {
768                 warn "***** Deprecated options: ",
769                         join(", ", keys %deprecated_options), "\n";
770                 }
771         if (keys %unsupported_options)
772                 {
773                 die "***** Unsupported options: ",
774                         join(", ", keys %unsupported_options), "\n";
775                 }
776         }
777
778 if ($libs =~ /(^|\s)-Wl,-rpath,/
779     && !$disabled{shared}
780     && !($disabled{asan} && $disabled{msan} && $disabled{ubsan})) {
781     die "***** Cannot simultaneously use -rpath, shared libraries, and\n",
782         "***** any of asan, msan or ubsan\n";
783 }
784
785 if ($config{fips})
786         {
787         delete $disabled{"shared"} if ($disabled{"shared"} =~ /^default/);
788         }
789 else
790         {
791         @{$config{dirs}} = grep !/^fips$/, @{$config{dirs}};
792         }
793
794 my @tocheckfor = (keys %disabled);
795 while (@tocheckfor) {
796     my %new_tocheckfor = ();
797     my @cascade_copy = (@disable_cascades);
798     while (@cascade_copy) {
799         my ($test, $descendents) = (shift @cascade_copy, shift @cascade_copy);
800         if (ref($test) eq "CODE" ? $test->() : defined($disabled{$test})) {
801             foreach(grep { !defined($disabled{$_}) } @$descendents) {
802                 $new_tocheckfor{$_} = 1; $disabled{$_} = "forced";
803             }
804         }
805     }
806     @tocheckfor = (keys %new_tocheckfor);
807 }
808
809 our $die = sub { die @_; };
810 if ($target eq "TABLE") {
811     local $die = sub { warn @_; };
812     foreach (sort keys %table) {
813         print_table_entry($_, "TABLE");
814     }
815     exit 0;
816 }
817
818 if ($target eq "LIST") {
819     foreach (sort keys %table) {
820         print $_,"\n" unless $table{$_}->{template};
821     }
822     exit 0;
823 }
824
825 if ($target eq "HASH") {
826     local $die = sub { warn @_; };
827     print "%table = (\n";
828     foreach (sort keys %table) {
829         print_table_entry($_, "HASH");
830     }
831     exit 0;
832 }
833
834 # Backward compatibility?
835 if ($target =~ m/^CygWin32(-.*)$/) {
836     $target = "Cygwin".$1;
837 }
838
839 foreach (sort (keys %disabled))
840         {
841         $config{options} .= " no-$_";
842
843         printf "    no-%-12s %-10s", $_, "[$disabled{$_}]";
844
845         if (/^dso$/)
846                 { }
847         elsif (/^threads$/)
848                 { }
849         elsif (/^shared$/)
850                 { }
851         elsif (/^pic$/)
852                 { }
853         elsif (/^zlib$/)
854                 { }
855         elsif (/^dynamic-engine$/)
856                 { }
857         elsif (/^makedepend$/)
858                 { }
859         elsif (/^zlib-dynamic$/)
860                 { }
861         elsif (/^sse2$/)
862                 { $no_sse2 = 1; }
863         elsif (/^engine$/)
864                 {
865                 @{$config{dirs}} = grep !/^engines$/, @{$config{dirs}};
866                 @{$config{sdirs}} = grep !/^engine$/, @{$config{sdirs}};
867                 push @{$config{openssl_other_defines}}, "OPENSSL_NO_ENGINE";
868                 print " OPENSSL_NO_ENGINE (skip engines)";
869                 }
870         else
871                 {
872                 my ($WHAT, $what);
873
874                 ($WHAT = $what = $_) =~ tr/[\-a-z]/[_A-Z]/;
875
876                 # Fix up C macro end names
877                 $WHAT = "RMD160" if $what eq "ripemd";
878
879                 # fix-up crypto/directory name(s)
880                 $what = "ripemd" if $what eq "rmd160";
881                 $what = "whrlpool" if $what eq "whirlpool";
882
883                 if ($what ne "async" && $what ne "err"
884                     && grep { $_ eq $what } @{$config{sdirs}})
885                         {
886                         push @{$config{openssl_algorithm_defines}}, "OPENSSL_NO_$WHAT";
887                         @{$config{sdirs}} = grep { $_ ne $what} @{$config{sdirs}};
888
889                         print " OPENSSL_NO_$WHAT (skip dir)";
890                         }
891                 else
892                         {
893                         push @{$config{openssl_other_defines}}, "OPENSSL_NO_$WHAT";
894                         print " OPENSSL_NO_$WHAT";
895
896                         if (/^err$/)    { push @user_defines, "OPENSSL_NO_ERR"; }
897                         }
898                 }
899
900         print "\n";
901         }
902
903 print "Configuring for $target\n";
904
905 # Support for legacy targets having a name starting with 'debug-'
906 my ($d, $t) = $target =~ m/^(debug-)?(.*)$/;
907 if ($d) {
908     $config{build_type} = "debug";
909
910     # If we do not find debug-foo in the table, the target is set to foo.
911     if (!$table{$target}) {
912         $target = $t;
913     }
914 }
915 $config{target} = $target;
916 my %target = resolve_config($target);
917
918 &usage if (!%target || $target{template});
919
920 my %conf_files = map { $_ => 1 } (@{$target{_conf_fname_int}});
921 $config{conf_files} = [ sort keys %conf_files ];
922 %target = ( %{$table{DEFAULTS}}, %target );
923
924 $target{exe_extension}="";
925 $target{exe_extension}=".exe" if ($config{target} eq "DJGPP"
926                                   || $config{target} =~ /^(?:Cygwin|mingw)/);
927 $target{exe_extension}=".pm"  if ($config{target} =~ /vos/);
928
929 ($target{shared_extension_simple}=$target{shared_extension})
930     =~ s|\.\$\(SHLIB_MAJOR\)\.\$\(SHLIB_MINOR\)||;
931 $target{dso_extension}=$target{shared_extension_simple};
932 ($target{shared_import_extension}=$target{shared_extension_simple}.".a")
933     if ($config{target} =~ /^(?:Cygwin|mingw)/);
934
935
936 $config{cross_compile_prefix} = $ENV{'CROSS_COMPILE'}
937     if $config{cross_compile_prefix} eq "";
938
939 # Allow overriding the names of some tools.  USE WITH CARE
940 # Note: only Unix cares about HASHBANGPERL...  that explains
941 # the default string.
942 $config{perl} =    $ENV{'PERL'}    || ($^O ne "VMS" ? $^X : "perl");
943 $config{hashbangperl} =
944     $ENV{'HASHBANGPERL'}           || $ENV{'PERL'}     || "/usr/bin/env perl";
945 $target{cc} =      $ENV{'CC'}      || $target{cc}      || "cc";
946 $target{ranlib} =  $ENV{'RANLIB'}  || $target{ranlib}  ||
947                    (which("$config{cross_compile_prefix}ranlib") ?
948                           "\$(CROSS_COMPILE)ranlib" : "true");
949 $target{ar} =      $ENV{'AR'}      || $target{ar}      || "ar";
950 $target{nm} =      $ENV{'NM'}      || $target{nm}      || "nm";
951 $target{rc} =
952     $ENV{'RC'}  || $ENV{'WINDRES'} || $target{rc}      || "windres";
953
954 # Allow overriding the build file name
955 $target{build_file} = $ENV{BUILDFILE} || $target{build_file} || "Makefile";
956
957 # Cache information necessary for reconfiguration
958 $config{cc} = $target{cc};
959 $config{build_file} = $target{build_file};
960
961 # For cflags, lflags, plib_lflags, ex_libs and defines, add the debug_
962 # or release_ attributes.
963 # Do it in such a way that no spurious space is appended (hence the grep).
964 $config{defines} = [];
965 $config{cflags} = "";
966 $config{ex_libs} = "";
967 $config{shared_ldflag} = "";
968
969 # Make sure build_scheme is consistent.
970 $target{build_scheme} = [ $target{build_scheme} ]
971     if ref($target{build_scheme}) ne "ARRAY";
972
973 my ($builder, $builder_platform, @builder_opts) =
974     @{$target{build_scheme}};
975
976 foreach my $checker (($builder_platform."-".$target{build_file}."-checker.pm",
977                       $builder_platform."-checker.pm")) {
978     my $checker_path = catfile($srcdir, "Configurations", $checker);
979     if (-f $checker_path) {
980         my $fn = $ENV{CONFIGURE_CHECKER_WARN}
981             ? sub { warn $@; } : sub { die $@; };
982         if (! do $checker_path) {
983             if ($@) {
984                 $fn->($@);
985             } elsif ($!) {
986                 $fn->($!);
987             } else {
988                 $fn->("The detected tools didn't match the platform\n");
989             }
990         }
991         last;
992     }
993 }
994
995 push @{$config{defines}}, "NDEBUG"    if $config{build_type} eq "release";
996
997 if ($target =~ /^mingw/ && `$target{cc} --target-help 2>&1` =~ m/-mno-cygwin/m)
998         {
999         $config{cflags} .= " -mno-cygwin";
1000         $config{shared_ldflag} .= " -mno-cygwin";
1001         }
1002
1003 if ($target =~ /linux.*-mips/ && !$disabled{asm} && $user_cflags !~ /-m(ips|arch=)/) {
1004         # minimally required architecture flags for assembly modules
1005         $config{cflags}="-mips2 $config{cflags}" if ($target =~ /mips32/);
1006         $config{cflags}="-mips3 $config{cflags}" if ($target =~ /mips64/);
1007 }
1008
1009 my $no_shared_warn=0;
1010 my $no_user_cflags=0;
1011 my $no_user_defines=0;
1012
1013 # The DSO code currently always implements all functions so that no
1014 # applications will have to worry about that from a compilation point
1015 # of view. However, the "method"s may return zero unless that platform
1016 # has support compiled in for them. Currently each method is enabled
1017 # by a define "DSO_<name>" ... we translate the "dso_scheme" config
1018 # string entry into using the following logic;
1019 if (!$disabled{dso} && $target{dso_scheme} ne "")
1020         {
1021         $target{dso_scheme} =~ tr/[a-z]/[A-Z]/;
1022         if ($target{dso_scheme} eq "DLFCN")
1023                 {
1024                 unshift @{$config{defines}}, "DSO_DLFCN", "HAVE_DLFCN_H";
1025                 }
1026         elsif ($target{dso_scheme} eq "DLFCN_NO_H")
1027                 {
1028                 unshift @{$config{defines}}, "DSO_DLFCN";
1029                 }
1030         else
1031                 {
1032                 unshift @{$config{defines}}, "DSO_$target{dso_scheme}";
1033                 }
1034         }
1035
1036 $config{ex_libs}="$libs$config{ex_libs}" if ($libs ne "");
1037
1038 if ($disabled{asm})
1039         {
1040         if ($config{fips})
1041                 {
1042                 @{$config{defines}} = grep !/^[BL]_ENDIAN$/, @{$config{defines}};
1043                 @{$target{defines}} = grep !/^[BL]_ENDIAN$/, @{$target{defines}};
1044                 }
1045         }
1046
1047 # If threads aren't disabled, check how possible they are
1048 unless ($disabled{threads}) {
1049     if ($auto_threads) {
1050         # Enabled by default, disable it forcibly if unavailable
1051         if ($target{thread_scheme} eq "(unknown)") {
1052             $disabled{threads} = "unavailable";
1053         }
1054     } else {
1055         # The user chose to enable threads explicitly, let's see
1056         # if there's a chance that's possible
1057         if ($target{thread_scheme} eq "(unknown)") {
1058             # If the user asked for "threads" and we don't have internal
1059             # knowledge how to do it, [s]he is expected to provide any
1060             # system-dependent compiler options that are necessary.  We
1061             # can't truly check that the given options are correct, but
1062             # we expect the user to know what [s]He is doing.
1063             if ($no_user_cflags && $no_user_defines) {
1064                 die "You asked for multi-threading support, but didn't\n"
1065                     ,"provide any system-specific compiler options\n";
1066             }
1067         }
1068     }
1069 }
1070
1071 # If threads still aren't disabled, add a C macro to ensure the source
1072 # code knows about it.  Any other flag is taken care of by the configs.
1073 unless($disabled{threads}) {
1074     foreach (("defines", "openssl_thread_defines")) {
1075         push @{$config{$_}}, "OPENSSL_THREADS";
1076     }
1077 }
1078
1079 # With "deprecated" disable all deprecated features.
1080 if (defined($disabled{"deprecated"})) {
1081         $config{api} = $maxapi;
1082 }
1083
1084 if ($target{shared_target} eq "")
1085         {
1086         $no_shared_warn = 1
1087             if ((!$disabled{shared} || !$disabled{"dynamic-engine"})
1088                 && !$config{fips});
1089         $disabled{shared} = "no-shared-target";
1090         $disabled{pic} = $disabled{shared} = $disabled{"dynamic-engine"} =
1091             "no-shared-target";
1092         }
1093
1094 if ($disabled{"dynamic-engine"}) {
1095         push @{$config{defines}}, "OPENSSL_NO_DYNAMIC_ENGINE";
1096         $config{dynamic_engines} = 0;
1097 } else {
1098         push @{$config{defines}}, "OPENSSL_NO_STATIC_ENGINE";
1099         $config{dynamic_engines} = 1;
1100 }
1101
1102 unless ($disabled{"fuzz-libfuzzer"}) {
1103     $config{cflags} .= "-fsanitize-coverage=edge,indirect-calls ";
1104 }
1105
1106 unless ($disabled{asan}) {
1107     $config{cflags} .= "-fsanitize=address ";
1108 }
1109
1110 unless ($disabled{ubsan}) {
1111     # -DPEDANTIC or -fnosanitize=alignment may also be required on some
1112     # platforms.
1113     $config{cflags} .= "-fsanitize=undefined -fno-sanitize-recover=all ";
1114 }
1115
1116 unless ($disabled{msan}) {
1117   $config{cflags} .= "-fsanitize=memory ";
1118 }
1119
1120 unless ($disabled{"fuzz-libfuzzer"} && $disabled{"fuzz-afl"}
1121         && $disabled{asan} && $disabled{ubsan} && $disabled{msan}) {
1122     $config{cflags} .= "-fno-omit-frame-pointer -g ";
1123 }
1124 #
1125 # Platform fix-ups
1126 #
1127
1128 # This saves the build files from having to check
1129 if ($disabled{pic})
1130         {
1131         $target{shared_cflag} = $target{shared_ldflag} =
1132                 $target{shared_rcflag} = "";
1133         }
1134 else
1135         {
1136         push @{$config{defines}}, "OPENSSL_PIC";
1137         }
1138
1139 if ($target{sys_id} ne "")
1140         {
1141         push @{$config{openssl_sys_defines}}, "OPENSSL_SYS_$target{sys_id}";
1142         }
1143
1144 unless ($disabled{asm}) {
1145     $target{cpuid_asm_src}=$table{DEFAULTS}->{cpuid_asm_src} if ($config{processor} eq "386");
1146     $target{bn_asm_src} =~ s/\w+-gf2m.c// if (defined($disabled{ec2m}));
1147
1148     # bn-586 is the only one implementing bn_*_part_words
1149     push @{$config{defines}}, "OPENSSL_BN_ASM_PART_WORDS" if ($target{bn_asm_src} =~ /bn-586/);
1150     push @{$config{defines}}, "OPENSSL_IA32_SSE2" if (!$no_sse2 && $target{bn_asm_src} =~ /86/);
1151
1152     push @{$config{defines}}, "OPENSSL_BN_ASM_MONT" if ($target{bn_asm_src} =~ /-mont/);
1153     push @{$config{defines}}, "OPENSSL_BN_ASM_MONT5" if ($target{bn_asm_src} =~ /-mont5/);
1154     push @{$config{defines}}, "OPENSSL_BN_ASM_GF2m" if ($target{bn_asm_src} =~ /-gf2m/);
1155
1156     if ($config{fips}) {
1157         push @{$config{openssl_other_defines}}, "OPENSSL_FIPS";
1158     }
1159
1160     if ($target{sha1_asm_src}) {
1161         push @{$config{defines}}, "SHA1_ASM"   if ($target{sha1_asm_src} =~ /sx86/ || $target{sha1_asm_src} =~ /sha1/);
1162         push @{$config{defines}}, "SHA256_ASM" if ($target{sha1_asm_src} =~ /sha256/);
1163         push @{$config{defines}}, "SHA512_ASM" if ($target{sha1_asm_src} =~ /sha512/);
1164     }
1165     if ($target{rc4_asm_src} ne $table{DEFAULTS}->{rc4_asm_src}) {
1166         push @{$config{defines}}, "RC4_ASM";
1167     }
1168     if ($target{md5_asm_src}) {
1169         push @{$config{defines}}, "MD5_ASM";
1170     }
1171     $target{cast_asm_src}=$table{DEFAULTS}->{cast_asm_src} unless $disabled{pic}; # CAST assembler is not PIC
1172     if ($target{rmd160_asm_src}) {
1173         push @{$config{defines}}, "RMD160_ASM";
1174     }
1175     if ($target{aes_asm_src}) {
1176         push @{$config{defines}}, "AES_ASM" if ($target{aes_asm_src} =~ m/\baes-/);;
1177         # aes-ctr.fake is not a real file, only indication that assembler
1178         # module implements AES_ctr32_encrypt...
1179         push @{$config{defines}}, "AES_CTR_ASM" if ($target{aes_asm_src} =~ s/\s*aes-ctr\.fake//);
1180         # aes-xts.fake indicates presence of AES_xts_[en|de]crypt...
1181         push @{$config{defines}}, "AES_XTS_ASM" if ($target{aes_asm_src} =~ s/\s*aes-xts\.fake//);
1182         $target{aes_asm_src} =~ s/\s*(vpaes|aesni)-x86\.s//g if ($no_sse2);
1183         push @{$config{defines}}, "VPAES_ASM" if ($target{aes_asm_src} =~ m/vpaes/);
1184         push @{$config{defines}}, "BSAES_ASM" if ($target{aes_asm_src} =~ m/bsaes/);
1185     }
1186     if ($target{wp_asm_src} =~ /mmx/) {
1187         if ($config{processor} eq "386") {
1188             $target{wp_asm_src}=$table{DEFAULTS}->{wp_asm_src};
1189         } elsif (!$disabled{"whirlpool"}) {
1190             push @{$config{defines}}, "WHIRLPOOL_ASM";
1191         }
1192     }
1193     if ($target{modes_asm_src} =~ /ghash-/) {
1194         push @{$config{defines}}, "GHASH_ASM";
1195     }
1196     if ($target{ec_asm_src} =~ /ecp_nistz256/) {
1197         push @{$config{defines}}, "ECP_NISTZ256_ASM";
1198     }
1199     if ($target{padlock_asm_src} ne $table{DEFAULTS}->{padlock_asm_src}) {
1200         push @{$config{defines}}, "PADLOCK_ASM";
1201     }
1202     if ($target{poly1305_asm_src} ne "") {
1203         push @{$config{defines}}, "POLY1305_ASM";
1204     }
1205 }
1206
1207 my %predefined;
1208
1209 if ($^O ne "VMS") {
1210     my $cc = "$config{cross_compile_prefix}$target{cc}";
1211
1212     # collect compiler pre-defines from gcc or gcc-alike...
1213     open(PIPE, "$cc -dM -E -x c /dev/null 2>&1 |");
1214     while (<PIPE>) {
1215         m/^#define\s+(\w+(?:\(\w+\))?)(?:\s+(.+))?/ or last;
1216         $predefined{$1} = $2 // "";
1217     }
1218     close(PIPE);
1219
1220     if (!$disabled{makedepend}) {
1221         # We know that GNU C version 3 and up as well as all clang
1222         # versions support dependency generation
1223         if ($predefined{__GNUC__} >= 3) {
1224             $config{makedepprog} = $cc;
1225         } else {
1226             $config{makedepprog} = which('makedepend');
1227             $disabled{makedepend} = "unavailable" unless $config{makedepprog};
1228         }
1229     }
1230 }
1231
1232
1233
1234 # Deal with bn_ops ###################################################
1235
1236 $config{bn_ll}                  =0;
1237 $config{export_var_as_fn}       =0;
1238 my $def_int="unsigned int";
1239 $config{rc4_int}                =$def_int;
1240 ($config{b64l},$config{b64},$config{b32})=(0,0,1);
1241
1242 my $count = 0;
1243 foreach (sort split(/\s+/,$target{bn_ops})) {
1244     $count++ if /SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT/;
1245     $config{export_var_as_fn}=1                 if $_ eq 'EXPORT_VAR_AS_FN';
1246     $config{bn_ll}=1                            if $_ eq 'BN_LLONG';
1247     $config{rc4_int}="unsigned char"            if $_ eq 'RC4_CHAR';
1248     ($config{b64l},$config{b64},$config{b32})
1249         =(0,1,0)                                if $_ eq 'SIXTY_FOUR_BIT';
1250     ($config{b64l},$config{b64},$config{b32})
1251         =(1,0,0)                                if $_ eq 'SIXTY_FOUR_BIT_LONG';
1252     ($config{b64l},$config{b64},$config{b32})
1253         =(0,0,1)                                if $_ eq 'THIRTY_TWO_BIT';
1254 }
1255 die "Exactly one of SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT can be set in bn_ops\n"
1256     if $count > 1;
1257
1258
1259 # Hack cflags for better warnings (dev option) #######################
1260
1261 # "Stringify" the C flags string.  This permits it to be made part of a string
1262 # and works as well on command lines.
1263 $config{cflags} =~ s/([\\\"])/\\$1/g;
1264
1265 if (defined($config{api})) {
1266     $config{openssl_api_defines} = [ "OPENSSL_MIN_API=".$apitable->{$config{api}} ];
1267     my $apiflag = sprintf("OPENSSL_API_COMPAT=%s", $apitable->{$config{api}});
1268     push @{$config{defines}}, $apiflag;
1269 }
1270
1271 if ($strict_warnings)
1272         {
1273         my $wopt;
1274         die "ERROR --strict-warnings requires gcc or gcc-alike"
1275             unless defined($predefined{__GNUC__});
1276         foreach $wopt (split /\s+/, $gcc_devteam_warn)
1277                 {
1278                 $config{cflags} .= " $wopt" unless ($config{cflags} =~ /(?:^|\s)$wopt(?:\s|$)/)
1279                 }
1280         if (defined($predefined{__clang__}))
1281                 {
1282                 foreach $wopt (split /\s+/, $clang_devteam_warn)
1283                         {
1284                         $config{cflags} .= " $wopt" unless ($config{cflags} =~ /(?:^|\s)$wopt(?:\s|$)/)
1285                         }
1286                 }
1287         }
1288
1289 unless ($disabled{"crypto-mdebug-backtrace"})
1290         {
1291         foreach my $wopt (split /\s+/, $memleak_devteam_backtrace)
1292                 {
1293                 $config{cflags} .= " $wopt" unless ($config{cflags} =~ /(?:^|\s)$wopt(?:\s|$)/)
1294                 }
1295         if ($target =~ /^BSD-/)
1296                 {
1297                 $config{ex_libs} .= " -lexecinfo";
1298                 }
1299         }
1300
1301 if ($user_cflags ne "") { $config{cflags}="$config{cflags}$user_cflags"; }
1302 else                    { $no_user_cflags=1;  }
1303 if (@user_defines) { $config{defines}=[ @{$config{defines}}, @user_defines ]; }
1304 else               { $no_user_defines=1;    }
1305
1306 # ALL MODIFICATIONS TO %config and %target MUST BE DONE FROM HERE ON
1307
1308 unless ($disabled{afalgeng}) {
1309     $config{afalgeng}="";
1310     if ($target =~ m/^linux/) {
1311         my $minver = 4*10000 + 1*100 + 0;
1312         if ($config{cross_compile_prefix} eq "") {
1313             my $verstr = `uname -r`;
1314             my ($ma, $mi1, $mi2) = split("\\.", $verstr);
1315             ($mi2) = $mi2 =~ /(\d+)/;
1316             my $ver = $ma*10000 + $mi1*100 + $mi2;
1317             if ($ver < $minver) {
1318                 $disabled{afalgeng} = "too-old-kernel";
1319             } else {
1320                 push @{$config{engdirs}}, "afalg";
1321             }
1322         } else {
1323             $disabled{afalgeng} = "cross-compiling";
1324         }
1325     } else {
1326         $disabled{afalgeng}  = "not-linux";
1327     }
1328 }
1329
1330 push @{$config{openssl_other_defines}}, "OPENSSL_NO_AFALGENG" if ($disabled{afalgeng});
1331
1332 # If we use the unified build, collect information from build.info files
1333 my %unified_info = ();
1334
1335 my $buildinfo_debug = defined($ENV{CONFIGURE_DEBUG_BUILDINFO});
1336 if ($builder eq "unified") {
1337     use with_fallback qw(Text::Template);
1338
1339     sub cleandir {
1340         my $base = shift;
1341         my $dir = shift;
1342         my $relativeto = shift || ".";
1343
1344         $dir = catdir($base,$dir) unless isabsolute($dir);
1345
1346         # Make sure the directories we're building in exists
1347         mkpath($dir);
1348
1349         my $res = abs2rel(absolutedir($dir), rel2abs($relativeto));
1350         #print STDERR "DEBUG[cleandir]: $dir , $base => $res\n";
1351         return $res;
1352     }
1353
1354     sub cleanfile {
1355         my $base = shift;
1356         my $file = shift;
1357         my $relativeto = shift || ".";
1358
1359         $file = catfile($base,$file) unless isabsolute($file);
1360
1361         my $d = dirname($file);
1362         my $f = basename($file);
1363
1364         # Make sure the directories we're building in exists
1365         mkpath($d);
1366
1367         my $res = abs2rel(catfile(absolutedir($d), $f), rel2abs($relativeto));
1368         #print STDERR "DEBUG[cleanfile]: $d , $f => $res\n";
1369         return $res;
1370     }
1371
1372     # Store the name of the template file we will build the build file from
1373     # in %config.  This may be useful for the build file itself.
1374     my @build_file_template_names =
1375         ( $builder_platform."-".$target{build_file}.".tmpl",
1376           $target{build_file}.".tmpl" );
1377     my @build_file_templates = ();
1378
1379     # First, look in the user provided directory, if given
1380     if (defined $ENV{$local_config_envname}) {
1381         @build_file_templates =
1382             map {
1383                 if ($^O eq 'VMS') {
1384                     # VMS environment variables are logical names,
1385                     # which can be used as is
1386                     $local_config_envname . ':' . $_;
1387                 } else {
1388                     catfile($ENV{$local_config_envname}, $_);
1389                 }
1390             }
1391             @build_file_template_names;
1392     }
1393     # Then, look in our standard directory
1394     push @build_file_templates,
1395         ( map { cleanfile($srcdir, catfile("Configurations", $_), $blddir) }
1396           @build_file_template_names );
1397
1398     my $build_file_template;
1399     for $_ (@build_file_templates) {
1400         $build_file_template = $_;
1401         last if -f $build_file_template;
1402
1403         $build_file_template = undef;
1404     }
1405     if (!defined $build_file_template) {
1406         die "*** Couldn't find any of:\n", join("\n", @build_file_templates), "\n";
1407     }
1408     $config{build_file_templates}
1409       = [ $build_file_template,
1410           cleanfile($srcdir, catfile("Configurations", "common.tmpl"),
1411                     $blddir) ];
1412
1413     my @build_infos = ( [ ".", "build.info" ] );
1414     foreach (@{$config{dirs}}) {
1415         push @build_infos, [ $_, "build.info" ]
1416             if (-f catfile($srcdir, $_, "build.info"));
1417     }
1418     foreach (@{$config{sdirs}}) {
1419         push @build_infos, [ catdir("crypto", $_), "build.info" ]
1420             if (-f catfile($srcdir, "crypto", $_, "build.info"));
1421     }
1422     foreach (@{$config{engdirs}}) {
1423         push @build_infos, [ catdir("engines", $_), "build.info" ]
1424             if (-f catfile($srcdir, "engines", $_, "build.info"));
1425     }
1426
1427     $config{build_infos} = [ ];
1428
1429     foreach (@build_infos) {
1430         my $sourced = catdir($srcdir, $_->[0]);
1431         my $buildd = catdir($blddir, $_->[0]);
1432
1433         mkpath($buildd);
1434
1435         my $f = $_->[1];
1436         # The basic things we're trying to build
1437         my @programs = ();
1438         my @programs_install = ();
1439         my @libraries = ();
1440         my @libraries_install = ();
1441         my @engines = ();
1442         my @engines_install = ();
1443         my @scripts = ();
1444         my @scripts_install = ();
1445         my @extra = ();
1446         my @overrides = ();
1447         my @intermediates = ();
1448         my @rawlines = ();
1449
1450         my %ordinals = ();
1451         my %sources = ();
1452         my %shared_sources = ();
1453         my %includes = ();
1454         my %depends = ();
1455         my %renames = ();
1456         my %sharednames = ();
1457         my %generate = ();
1458
1459         push @{$config{build_infos}}, catfile(abs2rel($sourced, $blddir), $f);
1460         my $template =
1461             Text::Template->new(TYPE => 'FILE',
1462                                 SOURCE => catfile($sourced, $f),
1463                                 PREPEND => qq{use lib "$FindBin::Bin/util/perl";});
1464         die "Something went wrong with $sourced/$f: $!\n" unless $template;
1465         my @text =
1466             split /^/m,
1467             $template->fill_in(HASH => { config => \%config,
1468                                          target => \%target,
1469                                          disabled => \%disabled,
1470                                          withargs => \%withargs,
1471                                          builddir => abs2rel($buildd, $blddir),
1472                                          sourcedir => abs2rel($sourced, $blddir),
1473                                          buildtop => abs2rel($blddir, $blddir),
1474                                          sourcetop => abs2rel($srcdir, $blddir) },
1475                                DELIMITERS => [ "{-", "-}" ]);
1476
1477         # The top item of this stack has the following values
1478         # -2 positive already run and we found ELSE (following ELSIF should fail)
1479         # -1 positive already run (skip until ENDIF)
1480         # 0 negatives so far (if we're at a condition, check it)
1481         # 1 last was positive (don't skip lines until next ELSE, ELSIF or ENDIF)
1482         # 2 positive ELSE (following ELSIF should fail)
1483         my @skip = ();
1484         collect_information(
1485             collect_from_array([ @text ],
1486                                qr/\\$/ => sub { my $l1 = shift; my $l2 = shift;
1487                                                 $l1 =~ s/\\$//; $l1.$l2 }),
1488             # Info we're looking for
1489             qr/^\s*IF\[((?:\\.|[^\\\]])*)\]\s*$/
1490             => sub {
1491                 if (! @skip || $skip[$#skip] > 0) {
1492                     push @skip, !! $1;
1493                 } else {
1494                     push @skip, -1;
1495                 }
1496             },
1497             qr/^\s*ELSIF\[((?:\\.|[^\\\]])*)\]\s*$/
1498             => sub { die "ELSIF out of scope" if ! @skip;
1499                      die "ELSIF following ELSE" if abs($skip[$#skip]) == 2;
1500                      $skip[$#skip] = -1 if $skip[$#skip] != 0;
1501                      $skip[$#skip] = !! $1
1502                          if $skip[$#skip] == 0; },
1503             qr/^\s*ELSE\s*$/
1504             => sub { die "ELSE out of scope" if ! @skip;
1505                      $skip[$#skip] = -2 if $skip[$#skip] != 0;
1506                      $skip[$#skip] = 2 if $skip[$#skip] == 0; },
1507             qr/^\s*ENDIF\s*$/
1508             => sub { die "ENDIF out of scope" if ! @skip;
1509                      pop @skip; },
1510             qr/^\s*PROGRAMS(_NO_INST)?\s*=\s*(.*)\s*$/
1511             => sub {
1512                 if (!@skip || $skip[$#skip] > 0) {
1513                     my $install = $1;
1514                     my @x = tokenize($2);
1515                     push @programs, @x;
1516                     push @programs_install, @x unless $install;
1517                 }
1518             },
1519             qr/^\s*LIBS(_NO_INST)?\s*=\s*(.*)\s*$/
1520             => sub {
1521                 if (!@skip || $skip[$#skip] > 0) {
1522                     my $install = $1;
1523                     my @x = tokenize($2);
1524                     push @libraries, @x;
1525                     push @libraries_install, @x unless $install;
1526                 }
1527             },
1528             qr/^\s*ENGINES(_NO_INST)?\s*=\s*(.*)\s*$/
1529             => sub {
1530                 if (!@skip || $skip[$#skip] > 0) {
1531                     my $install = $1;
1532                     my @x = tokenize($2);
1533                     push @engines, @x;
1534                     push @engines_install, @x unless $install;
1535                 }
1536             },
1537             qr/^\s*SCRIPTS(_NO_INST)?\s*=\s*(.*)\s*$/
1538             => sub {
1539                 if (!@skip || $skip[$#skip] > 0) {
1540                     my $install = $1;
1541                     my @x = tokenize($2);
1542                     push @scripts, @x;
1543                     push @scripts_install, @x unless $install;
1544                 }
1545             },
1546             qr/^\s*EXTRA\s*=\s*(.*)\s*$/
1547             => sub { push @extra, tokenize($1)
1548                          if !@skip || $skip[$#skip] > 0 },
1549             qr/^\s*OVERRIDES\s*=\s*(.*)\s*$/
1550             => sub { push @overrides, tokenize($1)
1551                          if !@skip || $skip[$#skip] > 0 },
1552
1553             qr/^\s*ORDINALS\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/,
1554             => sub { push @{$ordinals{$1}}, tokenize($2)
1555                          if !@skip || $skip[$#skip] > 0 },
1556             qr/^\s*SOURCE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1557             => sub { push @{$sources{$1}}, tokenize($2)
1558                          if !@skip || $skip[$#skip] > 0 },
1559             qr/^\s*SHARED_SOURCE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1560             => sub { push @{$shared_sources{$1}}, tokenize($2)
1561                          if !@skip || $skip[$#skip] > 0 },
1562             qr/^\s*INCLUDE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1563             => sub { push @{$includes{$1}}, tokenize($2)
1564                          if !@skip || $skip[$#skip] > 0 },
1565             qr/^\s*DEPEND\[((?:\\.|[^\\\]])*)\]\s*=\s*(.*)\s*$/
1566             => sub { push @{$depends{$1}}, tokenize($2)
1567                          if !@skip || $skip[$#skip] > 0 },
1568             qr/^\s*GENERATE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1569             => sub { push @{$generate{$1}}, $2
1570                          if !@skip || $skip[$#skip] > 0 },
1571             qr/^\s*RENAME\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1572             => sub { push @{$renames{$1}}, tokenize($2)
1573                          if !@skip || $skip[$#skip] > 0 },
1574             qr/^\s*SHARED_NAME\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1575             => sub { push @{$sharednames{$1}}, tokenize($2)
1576                          if !@skip || $skip[$#skip] > 0 },
1577             qr/^\s*BEGINRAW\[((?:\\.|[^\\\]])+)\]\s*$/
1578             => sub {
1579                 my $lineiterator = shift;
1580                 my $target_kind = $1;
1581                 while (defined $lineiterator->()) {
1582                     s|\R$||;
1583                     if (/^\s*ENDRAW\[((?:\\.|[^\\\]])+)\]\s*$/) {
1584                         die "ENDRAW doesn't match BEGINRAW"
1585                             if $1 ne $target_kind;
1586                         last;
1587                     }
1588                     next if @skip && $skip[$#skip] <= 0;
1589                     push @rawlines,  $_
1590                         if ($target_kind eq $target{build_file}
1591                             || $target_kind eq $target{build_file}."(".$builder_platform.")");
1592                 }
1593             },
1594             qr/^(?:#.*|\s*)$/ => sub { },
1595             "OTHERWISE" => sub { die "Something wrong with this line:\n$_\nat $sourced/$f" },
1596             "BEFORE" => sub {
1597                 if ($buildinfo_debug) {
1598                     print STDERR "DEBUG: Parsing ",join(" ", @_),"\n";
1599                     print STDERR "DEBUG: ... before parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
1600                 }
1601             },
1602             "AFTER" => sub {
1603                 if ($buildinfo_debug) {
1604                     print STDERR "DEBUG: .... after parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
1605                 }
1606             },
1607             );
1608         die "runaway IF?" if (@skip);
1609
1610         foreach (keys %renames) {
1611             die "$_ renamed to more than one thing: "
1612                 ,join(" ", @{$renames{$_}}),"\n"
1613                 if scalar @{$renames{$_}} > 1;
1614             my $dest = cleanfile($buildd, $_, $blddir);
1615             my $to = cleanfile($buildd, $renames{$_}->[0], $blddir);
1616             die "$dest renamed to more than one thing: "
1617                 ,$unified_info{rename}->{$dest}, $to
1618                 unless !defined($unified_info{rename}->{$dest})
1619                 or $unified_info{rename}->{$dest} eq $to;
1620             $unified_info{rename}->{$dest} = $to;
1621         }
1622
1623         foreach (@programs) {
1624             my $program = cleanfile($buildd, $_, $blddir);
1625             if ($unified_info{rename}->{$program}) {
1626                 $program = $unified_info{rename}->{$program};
1627             }
1628             $unified_info{programs}->{$program} = 1;
1629         }
1630
1631         foreach (@programs_install) {
1632             my $program = cleanfile($buildd, $_, $blddir);
1633             if ($unified_info{rename}->{$program}) {
1634                 $program = $unified_info{rename}->{$program};
1635             }
1636             $unified_info{install}->{programs}->{$program} = 1;
1637         }
1638
1639         foreach (@libraries) {
1640             my $library = cleanfile($buildd, $_, $blddir);
1641             if ($unified_info{rename}->{$library}) {
1642                 $library = $unified_info{rename}->{$library};
1643             }
1644             $unified_info{libraries}->{$library} = 1;
1645         }
1646
1647         foreach (@libraries_install) {
1648             my $library = cleanfile($buildd, $_, $blddir);
1649             if ($unified_info{rename}->{$library}) {
1650                 $library = $unified_info{rename}->{$library};
1651             }
1652             $unified_info{install}->{libraries}->{$library} = 1;
1653         }
1654
1655         die <<"EOF" if scalar @engines and !$config{dynamic_engines};
1656 ENGINES can only be used if configured with 'dynamic-engine'.
1657 This is usually a fault in a build.info file.
1658 EOF
1659         foreach (@engines) {
1660             my $library = cleanfile($buildd, $_, $blddir);
1661             if ($unified_info{rename}->{$library}) {
1662                 $library = $unified_info{rename}->{$library};
1663             }
1664             $unified_info{engines}->{$library} = 1;
1665         }
1666
1667         foreach (@engines_install) {
1668             my $library = cleanfile($buildd, $_, $blddir);
1669             if ($unified_info{rename}->{$library}) {
1670                 $library = $unified_info{rename}->{$library};
1671             }
1672             $unified_info{install}->{engines}->{$library} = 1;
1673         }
1674
1675         foreach (@scripts) {
1676             my $script = cleanfile($buildd, $_, $blddir);
1677             if ($unified_info{rename}->{$script}) {
1678                 $script = $unified_info{rename}->{$script};
1679             }
1680             $unified_info{scripts}->{$script} = 1;
1681         }
1682
1683         foreach (@scripts_install) {
1684             my $script = cleanfile($buildd, $_, $blddir);
1685             if ($unified_info{rename}->{$script}) {
1686                 $script = $unified_info{rename}->{$script};
1687             }
1688             $unified_info{install}->{scripts}->{$script} = 1;
1689         }
1690
1691         foreach (@extra) {
1692             my $extra = cleanfile($buildd, $_, $blddir);
1693             $unified_info{extra}->{$extra} = 1;
1694         }
1695
1696         foreach (@overrides) {
1697             my $override = cleanfile($buildd, $_, $blddir);
1698             $unified_info{overrides}->{$override} = 1;
1699         }
1700
1701         push @{$unified_info{rawlines}}, @rawlines;
1702
1703         unless ($disabled{shared}) {
1704             # Check sharednames.
1705             foreach (keys %sharednames) {
1706                 my $dest = cleanfile($buildd, $_, $blddir);
1707                 if ($unified_info{rename}->{$dest}) {
1708                     $dest = $unified_info{rename}->{$dest};
1709                 }
1710                 die "shared_name for $dest with multiple values: "
1711                     ,join(" ", @{$sharednames{$_}}),"\n"
1712                     if scalar @{$sharednames{$_}} > 1;
1713                 my $to = cleanfile($buildd, $sharednames{$_}->[0], $blddir);
1714                 die "shared_name found for a library $dest that isn't defined\n"
1715                     unless $unified_info{libraries}->{$dest};
1716                 die "shared_name for $dest with multiple values: "
1717                     ,$unified_info{sharednames}->{$dest}, ", ", $to
1718                     unless !defined($unified_info{sharednames}->{$dest})
1719                     or $unified_info{sharednames}->{$dest} eq $to;
1720                 $unified_info{sharednames}->{$dest} = $to;
1721             }
1722
1723             # Additionally, we set up sharednames for libraries that don't
1724             # have any, as themselves.
1725             foreach (keys %{$unified_info{libraries}}) {
1726                 if (!defined $unified_info{sharednames}->{$_}) {
1727                     $unified_info{sharednames}->{$_} = $_
1728                 }
1729             }
1730         }
1731
1732         foreach (keys %ordinals) {
1733             my $dest = $_;
1734             my $ddest = cleanfile($buildd, $_, $blddir);
1735             if ($unified_info{rename}->{$ddest}) {
1736                 $ddest = $unified_info{rename}->{$ddest};
1737             }
1738             foreach (@{$ordinals{$dest}}) {
1739                 my %known_ordinals =
1740                     (
1741                      crypto =>
1742                      cleanfile($sourced, catfile("util", "libcrypto.num"), $blddir),
1743                      ssl =>
1744                      cleanfile($sourced, catfile("util", "libssl.num"), $blddir)
1745                     );
1746                 my $o = $known_ordinals{$_};
1747                 die "Ordinals for $ddest defined more than once\n"
1748                     if $unified_info{ordinals}->{$ddest};
1749                 $unified_info{ordinals}->{$ddest} = [ $_, $o ];
1750             }
1751         }
1752
1753         foreach (keys %sources) {
1754             my $dest = $_;
1755             my $ddest = cleanfile($buildd, $_, $blddir);
1756             if ($unified_info{rename}->{$ddest}) {
1757                 $ddest = $unified_info{rename}->{$ddest};
1758             }
1759             foreach (@{$sources{$dest}}) {
1760                 my $s = cleanfile($sourced, $_, $blddir);
1761
1762                 # If it isn't in the source tree, we assume it's generated
1763                 # in the build tree
1764                 if (! -f $s) {
1765                     $s = cleanfile($buildd, $_, $blddir);
1766                 }
1767                 # We recognise C and asm files
1768                 if ($s =~ /\.[csS]\b$/) {
1769                     (my $o = $_) =~ s/\.[csS]\b$/.o/;
1770                     $o = cleanfile($buildd, $o, $blddir);
1771                     $unified_info{sources}->{$ddest}->{$o} = 1;
1772                     $unified_info{sources}->{$o}->{$s} = 1;
1773                 } else {
1774                     $unified_info{sources}->{$ddest}->{$s} = 1;
1775                 }
1776             }
1777         }
1778
1779         foreach (keys %shared_sources) {
1780             my $dest = $_;
1781             my $ddest = cleanfile($buildd, $_, $blddir);
1782             if ($unified_info{rename}->{$ddest}) {
1783                 $ddest = $unified_info{rename}->{$ddest};
1784             }
1785             foreach (@{$shared_sources{$dest}}) {
1786                 my $s = cleanfile($sourced, $_, $blddir);
1787
1788                 # If it isn't in the source tree, we assume it's generated
1789                 # in the build tree
1790                 if (! -f $s) {
1791                     $s = cleanfile($buildd, $_, $blddir);
1792                 }
1793                 # We recognise C and asm files
1794                 if ($s =~ /\.[csS]\b$/) {
1795                     (my $o = $_) =~ s/\.[csS]\b$/.o/;
1796                     $o = cleanfile($buildd, $o, $blddir);
1797                     $unified_info{shared_sources}->{$ddest}->{$o} = 1;
1798                     $unified_info{sources}->{$o}->{$s} = 1;
1799                 } else {
1800                     die "unrecognised source file type for shared library: $s\n";
1801                 }
1802             }
1803         }
1804
1805         foreach (keys %generate) {
1806             my $dest = $_;
1807             my $ddest = cleanfile($buildd, $_, $blddir);
1808             if ($unified_info{rename}->{$ddest}) {
1809                 $ddest = $unified_info{rename}->{$ddest};
1810             }
1811             die "more than one generator for $dest: "
1812                     ,join(" ", @{$generate{$_}}),"\n"
1813                     if scalar @{$generate{$_}} > 1;
1814             my @generator = split /\s+/, $generate{$dest}->[0];
1815             $generator[0] = cleanfile($sourced, $generator[0], $blddir),
1816             $unified_info{generate}->{$ddest} = [ @generator ];
1817         }
1818
1819         foreach (keys %depends) {
1820             my $dest = $_;
1821             my $ddest = $dest eq "" ? "" : cleanfile($sourced, $_, $blddir);
1822
1823             # If the destination doesn't exist in source, it can only be
1824             # a generated file in the build tree.
1825             if ($ddest ne "" && ! -f $ddest) {
1826                 $ddest = cleanfile($buildd, $_, $blddir);
1827                 if ($unified_info{rename}->{$ddest}) {
1828                     $ddest = $unified_info{rename}->{$ddest};
1829                 }
1830             }
1831             foreach (@{$depends{$dest}}) {
1832                 my $d = cleanfile($sourced, $_, $blddir);
1833
1834                 # If we know it's generated, or assume it is because we can't
1835                 # find it in the source tree, we set file we depend on to be
1836                 # in the build tree rather than the source tree, and assume
1837                 # and that there are lines to build it in a BEGINRAW..ENDRAW
1838                 # section or in the Makefile template.
1839                 if (! -f $d
1840                     || (grep { $d eq $_ }
1841                         map { cleanfile($srcdir, $_, $blddir) }
1842                         grep { /\.h$/ } keys %{$unified_info{generate}})) {
1843                     $d = cleanfile($buildd, $_, $blddir);
1844                 }
1845                 # Take note if the file to depend on is being renamed
1846                 if ($unified_info{rename}->{$d}) {
1847                     $d = $unified_info{rename}->{$d};
1848                 }
1849                 $unified_info{depends}->{$ddest}->{$d} = 1;
1850                 # If we depend on a header file or a perl module, let's make
1851                 # sure it can get included
1852                 if ($dest ne "" && $d =~ /\.(h|pm)$/) {
1853                     my $i = dirname($d);
1854                     push @{$unified_info{includes}->{$ddest}->{source}}, $i
1855                         unless grep { $_ eq $i } @{$unified_info{includes}->{$ddest}->{source}};
1856                 }
1857             }
1858         }
1859
1860         foreach (keys %includes) {
1861             my $dest = $_;
1862             my $ddest = cleanfile($sourced, $_, $blddir);
1863
1864             # If the destination doesn't exist in source, it can only be
1865             # a generated file in the build tree.
1866             if (! -f $ddest) {
1867                 $ddest = cleanfile($buildd, $_, $blddir);
1868                 if ($unified_info{rename}->{$ddest}) {
1869                     $ddest = $unified_info{rename}->{$ddest};
1870                 }
1871             }
1872             foreach (@{$includes{$dest}}) {
1873                 my $is = cleandir($sourced, $_, $blddir);
1874                 my $ib = cleandir($buildd, $_, $blddir);
1875                 push @{$unified_info{includes}->{$ddest}->{source}}, $is
1876                     unless grep { $_ eq $is } @{$unified_info{includes}->{$ddest}->{source}};
1877                 push @{$unified_info{includes}->{$ddest}->{build}}, $ib
1878                     unless grep { $_ eq $ib } @{$unified_info{includes}->{$ddest}->{build}};
1879             }
1880         }
1881     }
1882
1883     # Massage the result
1884
1885     # Trickle down includes placed on libraries, engines and programs to
1886     # their sources (i.e. object files)
1887     foreach my $dest (keys %{$unified_info{engines}},
1888                       keys %{$unified_info{libraries}},
1889                       keys %{$unified_info{programs}}) {
1890         foreach my $k (("source", "build")) {
1891             next unless defined($unified_info{includes}->{$dest}->{$k});
1892             my @incs = reverse @{$unified_info{includes}->{$dest}->{$k}};
1893             foreach my $obj (grep /\.o$/,
1894                              (keys %{$unified_info{sources}->{$dest}},
1895                               keys %{$unified_info{shared_sources}->{$dest}})) {
1896                 foreach my $inc (@incs) {
1897                     unshift @{$unified_info{includes}->{$obj}->{$k}}, $inc
1898                         unless grep { $_ eq $inc } @{$unified_info{includes}->{$obj}->{$k}};
1899                 }
1900             }
1901         }
1902         delete $unified_info{includes}->{$dest};
1903     }
1904
1905     ### Make unified_info a bit more efficient
1906     # One level structures
1907     foreach (("programs", "libraries", "engines", "scripts", "extra", "overrides")) {
1908         $unified_info{$_} = [ sort keys %{$unified_info{$_}} ];
1909     }
1910     # Two level structures
1911     foreach my $l1 (("install", "sources", "shared_sources", "ldadd", "depends")) {
1912         foreach my $l2 (sort keys %{$unified_info{$l1}}) {
1913             $unified_info{$l1}->{$l2} =
1914                 [ sort keys %{$unified_info{$l1}->{$l2}} ];
1915         }
1916     }
1917     # Includes
1918     foreach my $dest (sort keys %{$unified_info{includes}}) {
1919         if (defined($unified_info{includes}->{$dest}->{build})) {
1920             my @source_includes =
1921                 ( @{$unified_info{includes}->{$dest}->{source}} );
1922             $unified_info{includes}->{$dest} =
1923                 [ @{$unified_info{includes}->{$dest}->{build}} ];
1924             foreach my $inc (@source_includes) {
1925                 push @{$unified_info{includes}->{$dest}}, $inc
1926                     unless grep { $_ eq $inc } @{$unified_info{includes}->{$dest}};
1927             }
1928         } else {
1929             $unified_info{includes}->{$dest} =
1930                 [ @{$unified_info{includes}->{$dest}->{source}} ];
1931         }
1932     }
1933 }
1934
1935 # For the schemes that need it, we provide the old *_obj configs
1936 # from the *_asm_obj ones
1937 foreach (grep /_(asm|aux)_src$/, keys %target) {
1938     my $src = $_;
1939     (my $obj = $_) =~ s/_(asm|aux)_src$/_obj/;
1940     ($target{$obj} = $target{$src}) =~ s/\.[csS]\b/.o/g;
1941 }
1942
1943 # Write down our configuration where it fits #########################
1944
1945 open(OUT,">configdata.pm") || die "unable to create configdata.pm: $!\n";
1946 print OUT <<"EOF";
1947 package configdata;
1948
1949 use strict;
1950 use warnings;
1951
1952 use Exporter;
1953 #use vars qw(\@ISA \@EXPORT);
1954 our \@ISA = qw(Exporter);
1955 our \@EXPORT = qw(\%config \%target \%disabled \%withargs \%unified_info \@disablables);
1956
1957 EOF
1958 print OUT "our %config = (\n";
1959 foreach (sort keys %config) {
1960     if (ref($config{$_}) eq "ARRAY") {
1961         print OUT "  ", $_, " => [ ", join(", ",
1962                                            map { quotify("perl", $_) }
1963                                            @{$config{$_}}), " ],\n";
1964     } else {
1965         print OUT "  ", $_, " => ", quotify("perl", $config{$_}), ",\n"
1966     }
1967 }
1968 print OUT <<"EOF";
1969 );
1970
1971 EOF
1972 print OUT "our %target = (\n";
1973 foreach (sort keys %target) {
1974     if (ref($target{$_}) eq "ARRAY") {
1975         print OUT "  ", $_, " => [ ", join(", ",
1976                                            map { quotify("perl", $_) }
1977                                            @{$target{$_}}), " ],\n";
1978     } else {
1979         print OUT "  ", $_, " => ", quotify("perl", $target{$_}), ",\n"
1980     }
1981 }
1982 print OUT <<"EOF";
1983 );
1984
1985 EOF
1986 print OUT "our \%available_protocols = (\n";
1987 print OUT "  tls => [ ", join(", ", map { quotify("perl", $_) } @tls), " ],\n";
1988 print OUT "  dtls => [ ", join(", ", map { quotify("perl", $_) } @dtls), " ],\n";
1989 print OUT <<"EOF";
1990 );
1991
1992 EOF
1993 print OUT "our \@disablables = (\n";
1994 foreach (@disablables) {
1995     print OUT "  ", quotify("perl", $_), ",\n";
1996 }
1997 print OUT <<"EOF";
1998 );
1999
2000 EOF
2001 print OUT "our \%disabled = (\n";
2002 foreach (sort keys %disabled) {
2003     print OUT "  ", quotify("perl", $_), " => ", quotify("perl", $disabled{$_}), ",\n";
2004 }
2005 print OUT <<"EOF";
2006 );
2007
2008 EOF
2009 print OUT "our %withargs = (\n";
2010 foreach (sort keys %withargs) {
2011     if (ref($withargs{$_}) eq "ARRAY") {
2012         print OUT "  ", $_, " => [ ", join(", ",
2013                                            map { quotify("perl", $_) }
2014                                            @{$withargs{$_}}), " ],\n";
2015     } else {
2016         print OUT "  ", $_, " => ", quotify("perl", $withargs{$_}), ",\n"
2017     }
2018 }
2019 print OUT <<"EOF";
2020 );
2021
2022 EOF
2023 if ($builder eq "unified") {
2024     my $recurse;
2025     $recurse = sub {
2026         my $indent = shift;
2027         foreach (@_) {
2028             if (ref $_ eq "ARRAY") {
2029                 print OUT " "x$indent, "[\n";
2030                 foreach (@$_) {
2031                     $recurse->($indent + 4, $_);
2032                 }
2033                 print OUT " "x$indent, "],\n";
2034             } elsif (ref $_ eq "HASH") {
2035                 my %h = %$_;
2036                 print OUT " "x$indent, "{\n";
2037                 foreach (sort keys %h) {
2038                     if (ref $h{$_} eq "") {
2039                         print OUT " "x($indent + 4), quotify("perl", $_), " => ", quotify("perl", $h{$_}), ",\n";
2040                     } else {
2041                         print OUT " "x($indent + 4), quotify("perl", $_), " =>\n";
2042                         $recurse->($indent + 8, $h{$_});
2043                     }
2044                 }
2045                 print OUT " "x$indent, "},\n";
2046             } else {
2047                 print OUT " "x$indent, quotify("perl", $_), ",\n";
2048             }
2049         }
2050     };
2051     print OUT "our %unified_info = (\n";
2052     foreach (sort keys %unified_info) {
2053         if (ref $unified_info{$_} eq "") {
2054             print OUT " "x4, quotify("perl", $_), " => ", quotify("perl", $unified_info{$_}), ",\n";
2055         } else {
2056             print OUT " "x4, quotify("perl", $_), " =>\n";
2057             $recurse->(8, $unified_info{$_});
2058         }
2059     }
2060     print OUT <<"EOF";
2061 );
2062
2063 EOF
2064 }
2065 print OUT "1;\n";
2066 close(OUT);
2067
2068
2069 print "CC            =$config{cross_compile_prefix}$target{cc}\n";
2070 print "CFLAG         =$target{cflags} $config{cflags}\n";
2071 print "SHARED_CFLAG  =$target{shared_cflag}\n";
2072 print "DEFINES       =",join(" ", @{$target{defines}}, @{$config{defines}}),"\n";
2073 print "LFLAG         =$target{lflags}\n";
2074 print "PLIB_LFLAG    =$target{plib_lflags}\n";
2075 print "EX_LIBS       =$target{ex_libs} $config{ex_libs}\n";
2076 print "APPS_OBJ      =$target{apps_obj}\n";
2077 print "CPUID_OBJ     =$target{cpuid_obj}\n";
2078 print "UPLINK_OBJ    =$target{uplink_obj}\n";
2079 print "BN_ASM        =$target{bn_obj}\n";
2080 print "EC_ASM        =$target{ec_obj}\n";
2081 print "DES_ENC       =$target{des_obj}\n";
2082 print "AES_ENC       =$target{aes_obj}\n";
2083 print "BF_ENC        =$target{bf_obj}\n";
2084 print "CAST_ENC      =$target{cast_obj}\n";
2085 print "RC4_ENC       =$target{rc4_obj}\n";
2086 print "RC5_ENC       =$target{rc5_obj}\n";
2087 print "MD5_OBJ_ASM   =$target{md5_obj}\n";
2088 print "SHA1_OBJ_ASM  =$target{sha1_obj}\n";
2089 print "RMD160_OBJ_ASM=$target{rmd160_obj}\n";
2090 print "CMLL_ENC      =$target{cmll_obj}\n";
2091 print "MODES_OBJ     =$target{modes_obj}\n";
2092 print "PADLOCK_OBJ   =$target{padlock_obj}\n";
2093 print "CHACHA_ENC    =$target{chacha_obj}\n";
2094 print "POLY1305_OBJ  =$target{poly1305_obj}\n";
2095 print "BLAKE2_OBJ    =$target{blake2_obj}\n";
2096 print "PROCESSOR     =$config{processor}\n";
2097 print "RANLIB        =", $target{ranlib} eq '$(CROSS_COMPILE)ranlib' ?
2098                              "$config{cross_compile_prefix}ranlib" :
2099                              "$target{ranlib}", "\n";
2100 print "ARFLAGS       =$target{arflags}\n";
2101 print "PERL          =$config{perl}\n";
2102 print "\n";
2103 print "SIXTY_FOUR_BIT_LONG mode\n" if $config{b64l};
2104 print "SIXTY_FOUR_BIT mode\n" if $config{b64};
2105 print "THIRTY_TWO_BIT mode\n" if $config{b32};
2106 print "BN_LLONG mode\n" if $config{bn_ll};
2107 print "RC4 uses $config{rc4_int}\n" if $config{rc4_int} ne $def_int;
2108
2109 my %builders = (
2110     unified => sub {
2111         run_dofile(catfile($blddir, $target{build_file}),
2112                    @{$config{build_file_templates}});
2113     },
2114     );
2115
2116 $builders{$builder}->($builder_platform, @builder_opts);
2117
2118 print <<"EOF";
2119
2120 Configured for $target.
2121 EOF
2122
2123 print <<"EOF" if ($disabled{threads} eq "unavailable");
2124
2125 The library could not be configured for supporting multi-threaded
2126 applications as the compiler options required on this system are not known.
2127 See file INSTALL for details if you need multi-threading.
2128 EOF
2129
2130 print <<"EOF" if ($no_shared_warn);
2131
2132 The options 'shared', 'pic' and 'dynamic-engine' aren't supported on this
2133 platform, so we will pretend you gave the option 'no-pic', which also disables
2134 'shared' and 'dynamic-engine'.  If you know how to implement shared libraries
2135 or position independent code, please let us know (but please first make sure
2136 you have tried with a current version of OpenSSL).
2137 EOF
2138
2139 print <<"EOF" if (-f catfile($srcdir, "configdata.pm") && $srcdir ne $blddir);
2140
2141 WARNING: there are indications that another build was made in the source
2142 directory.  This build may have picked up artifacts from that build, the
2143 safest course of action is to clean the source directory and redo this
2144 configuration.
2145 EOF
2146
2147 exit(0);
2148
2149 ######################################################################
2150 #
2151 # Helpers and utility functions
2152 #
2153
2154 # Configuration file reading #########################################
2155
2156 # Note: All of the helper functions are for lazy evaluation.  They all
2157 # return a CODE ref, which will return the intended value when evaluated.
2158 # Thus, whenever there's mention of a returned value, it's about that
2159 # intended value.
2160
2161 # Helper function to implement conditional inheritance depending on the
2162 # value of $disabled{asm}.  Used in inherit_from values as follows:
2163 #
2164 #      inherit_from => [ "template", asm("asm_tmpl") ]
2165 #
2166 sub asm {
2167     my @x = @_;
2168     sub {
2169         $disabled{asm} ? () : @x;
2170     }
2171 }
2172
2173 # Helper function to implement conditional value variants, with a default
2174 # plus additional values based on the value of $config{build_type}.
2175 # Arguments are given in hash table form:
2176 #
2177 #       picker(default => "Basic string: ",
2178 #              debug   => "debug",
2179 #              release => "release")
2180 #
2181 # When configuring with --debug, the resulting string will be
2182 # "Basic string: debug", and when not, it will be "Basic string: release"
2183 #
2184 # This can be used to create variants of sets of flags according to the
2185 # build type:
2186 #
2187 #       cflags => picker(default => "-Wall",
2188 #                        debug   => "-g -O0",
2189 #                        release => "-O3")
2190 #
2191 sub picker {
2192     my %opts = @_;
2193     return sub { add($opts{default} || (),
2194                      $opts{$config{build_type}} || ())->(); }
2195 }
2196
2197 # Helper function to combine several values of different types into one.
2198 # This is useful if you want to combine a string with the result of a
2199 # lazy function, such as:
2200 #
2201 #       cflags => combine("-Wall", sub { $disabled{zlib} ? () : "-DZLIB" })
2202 #
2203 sub combine {
2204     my @stuff = @_;
2205     return sub { add(@stuff)->(); }
2206 }
2207
2208 # Helper function to implement conditional values depending on the value
2209 # of $disabled{threads}.  Can be used as follows:
2210 #
2211 #       cflags => combine("-Wall", threads("-pthread"))
2212 #
2213 sub threads {
2214     my @flags = @_;
2215     return sub { add($disabled{threads} ? () : @flags)->(); }
2216 }
2217
2218
2219
2220 our $add_called = 0;
2221 # Helper function to implement adding values to already existing configuration
2222 # values.  It handles elements that are ARRAYs, CODEs and scalars
2223 sub _add {
2224     my $separator = shift;
2225
2226     # If there's any ARRAY in the collection of values OR the separator
2227     # is undef, we will return an ARRAY of combined values, otherwise a
2228     # string of joined values with $separator as the separator.
2229     my $found_array = !defined($separator);
2230
2231     my @values =
2232         map {
2233             my $res = $_;
2234             while (ref($res) eq "CODE") {
2235                 $res = $res->();
2236             }
2237             if (defined($res)) {
2238                 if (ref($res) eq "ARRAY") {
2239                     $found_array = 1;
2240                     @$res;
2241                 } else {
2242                     $res;
2243                 }
2244             } else {
2245                 ();
2246             }
2247     } (@_);
2248
2249     $add_called = 1;
2250
2251     if ($found_array) {
2252         [ @values ];
2253     } else {
2254         join($separator, grep { defined($_) && $_ ne "" } @values);
2255     }
2256 }
2257 sub add_before {
2258     my $separator = " ";
2259     if (ref($_[$#_]) eq "HASH") {
2260         my $opts = pop;
2261         $separator = $opts->{separator};
2262     }
2263     my @x = @_;
2264     sub { _add($separator, @x, @_) };
2265 }
2266 sub add {
2267     my $separator = " ";
2268     if (ref($_[$#_]) eq "HASH") {
2269         my $opts = pop;
2270         $separator = $opts->{separator};
2271     }
2272     my @x = @_;
2273     sub { _add($separator, @_, @x) };
2274 }
2275
2276 # configuration reader, evaluates the input file as a perl script and expects
2277 # it to fill %targets with target configurations.  Those are then added to
2278 # %table.
2279 sub read_config {
2280     my $fname = shift;
2281     open(CONFFILE, "< $fname")
2282         or die "Can't open configuration file '$fname'!\n";
2283     my $x = $/;
2284     undef $/;
2285     my $content = <CONFFILE>;
2286     $/ = $x;
2287     close(CONFFILE);
2288     my %targets = ();
2289     {
2290         # Protect certain tables from tampering
2291         local %table = %::table;
2292
2293         eval $content;
2294         warn $@ if $@;
2295     }
2296     my %preexisting = ();
2297     foreach (sort keys %targets) {
2298         $preexisting{$_} = 1 if $table{$_};
2299     }
2300     die <<"EOF",
2301 The following config targets from $fname
2302 shadow pre-existing config targets with the same name:
2303 EOF
2304         map { "  $_\n" } sort keys %preexisting
2305         if %preexisting;
2306
2307
2308     # For each target, check that it's configured with a hash table.
2309     foreach (keys %targets) {
2310         if (ref($targets{$_}) ne "HASH") {
2311             if (ref($targets{$_}) eq "") {
2312                 warn "Deprecated target configuration for $_, ignoring...\n";
2313             } else {
2314                 warn "Misconfigured target configuration for $_ (should be a hash table), ignoring...\n";
2315             }
2316             delete $targets{$_};
2317         } else {
2318             $targets{$_}->{_conf_fname_int} = add([ $fname ]);
2319         }
2320     }
2321
2322     %table = (%table, %targets);
2323
2324 }
2325
2326 # configuration resolver.  Will only resolve all the lazy evaluation
2327 # codeblocks for the chosen target and all those it inherits from,
2328 # recursively
2329 sub resolve_config {
2330     my $target = shift;
2331     my @breadcrumbs = @_;
2332
2333 #    my $extra_checks = defined($ENV{CONFIGURE_EXTRA_CHECKS});
2334
2335     if (grep { $_ eq $target } @breadcrumbs) {
2336         die "inherit_from loop!  target backtrace:\n  "
2337             ,$target,"\n  ",join("\n  ", @breadcrumbs),"\n";
2338     }
2339
2340     if (!defined($table{$target})) {
2341         warn "Warning! target $target doesn't exist!\n";
2342         return ();
2343     }
2344     # Recurse through all inheritances.  They will be resolved on the
2345     # fly, so when this operation is done, they will all just be a
2346     # bunch of attributes with string values.
2347     # What we get here, though, are keys with references to lists of
2348     # the combined values of them all.  We will deal with lists after
2349     # this stage is done.
2350     my %combined_inheritance = ();
2351     if ($table{$target}->{inherit_from}) {
2352         my @inherit_from =
2353             map { ref($_) eq "CODE" ? $_->() : $_ } @{$table{$target}->{inherit_from}};
2354         foreach (@inherit_from) {
2355             my %inherited_config = resolve_config($_, $target, @breadcrumbs);
2356
2357             # 'template' is a marker that's considered private to
2358             # the config that had it.
2359             delete $inherited_config{template};
2360
2361             foreach (keys %inherited_config) {
2362                 if (!$combined_inheritance{$_}) {
2363                     $combined_inheritance{$_} = [];
2364                 }
2365                 push @{$combined_inheritance{$_}}, $inherited_config{$_};
2366             }
2367         }
2368     }
2369
2370     # We won't need inherit_from in this target any more, since we've
2371     # resolved all the inheritances that lead to this
2372     delete $table{$target}->{inherit_from};
2373
2374     # Now is the time to deal with those lists.  Here's the place to
2375     # decide what shall be done with those lists, all based on the
2376     # values of the target we're currently dealing with.
2377     # - If a value is a coderef, it will be executed with the list of
2378     #   inherited values as arguments.
2379     # - If the corresponding key doesn't have a value at all or is the
2380     #   empty string, the inherited value list will be run through the
2381     #   default combiner (below), and the result becomes this target's
2382     #   value.
2383     # - Otherwise, this target's value is assumed to be a string that
2384     #   will simply override the inherited list of values.
2385     my $default_combiner = add();
2386
2387     my %all_keys =
2388         map { $_ => 1 } (keys %combined_inheritance,
2389                          keys %{$table{$target}});
2390
2391     sub process_values {
2392         my $object    = shift;
2393         my $inherited = shift;  # Always a [ list ]
2394         my $target    = shift;
2395         my $entry     = shift;
2396
2397         $add_called = 0;
2398
2399         while(ref($object) eq "CODE") {
2400             $object = $object->(@$inherited);
2401         }
2402         if (!defined($object)) {
2403             return ();
2404         }
2405         elsif (ref($object) eq "ARRAY") {
2406             local $add_called;  # To make sure recursive calls don't affect it
2407             return [ map { process_values($_, $inherited, $target, $entry) }
2408                      @$object ];
2409         } elsif (ref($object) eq "") {
2410             return $object;
2411         } else {
2412             die "cannot handle reference type ",ref($object)
2413                 ," found in target ",$target," -> ",$entry,"\n";
2414         }
2415     }
2416
2417     foreach (sort keys %all_keys) {
2418         my $previous = $combined_inheritance{$_};
2419
2420         # Current target doesn't have a value for the current key?
2421         # Assign it the default combiner, the rest of this loop body
2422         # will handle it just like any other coderef.
2423         if (!exists $table{$target}->{$_}) {
2424             $table{$target}->{$_} = $default_combiner;
2425         }
2426
2427         $table{$target}->{$_} = process_values($table{$target}->{$_},
2428                                                $combined_inheritance{$_},
2429                                                $target, $_);
2430         unless(defined($table{$target}->{$_})) {
2431             delete $table{$target}->{$_};
2432         }
2433 #        if ($extra_checks &&
2434 #            $previous && !($add_called ||  $previous ~~ $table{$target}->{$_})) {
2435 #            warn "$_ got replaced in $target\n";
2436 #        }
2437     }
2438
2439     # Finally done, return the result.
2440     return %{$table{$target}};
2441 }
2442
2443 sub usage
2444         {
2445         print STDERR $usage;
2446         print STDERR "\npick os/compiler from:\n";
2447         my $j=0;
2448         my $i;
2449         my $k=0;
2450         foreach $i (sort keys %table)
2451                 {
2452                 next if $table{$i}->{template};
2453                 next if $i =~ /^debug/;
2454                 $k += length($i) + 1;
2455                 if ($k > 78)
2456                         {
2457                         print STDERR "\n";
2458                         $k=length($i);
2459                         }
2460                 print STDERR $i . " ";
2461                 }
2462         foreach $i (sort keys %table)
2463                 {
2464                 next if $table{$i}->{template};
2465                 next if $i !~ /^debug/;
2466                 $k += length($i) + 1;
2467                 if ($k > 78)
2468                         {
2469                         print STDERR "\n";
2470                         $k=length($i);
2471                         }
2472                 print STDERR $i . " ";
2473                 }
2474         print STDERR "\n\nNOTE: If in doubt, on Unix-ish systems use './config'.\n";
2475         exit(1);
2476         }
2477
2478 sub run_dofile
2479 {
2480     my $out = shift;
2481     my @templates = @_;
2482
2483     unlink $out || warn "Can't remove $out, $!"
2484         if -f $out;
2485     foreach (@templates) {
2486         die "Can't open $_, $!" unless -f $_;
2487     }
2488     my $perlcmd = (quotify("maybeshell", $config{perl}))[0];
2489     my $cmd = "$perlcmd \"-I.\" \"-Mconfigdata\" \"$dofile\" -o\"Configure\" \"".join("\" \"",@templates)."\" > \"$out.new\"";
2490     #print STDERR "DEBUG[run_dofile]: \$cmd = $cmd\n";
2491     system($cmd);
2492     exit 1 if $? != 0;
2493     rename("$out.new", $out) || die "Can't rename $out.new, $!";
2494 }
2495
2496 sub which
2497 {
2498     my ($name)=@_;
2499
2500     if (eval { require IPC::Cmd; 1; }) {
2501         IPC::Cmd->import();
2502         return scalar IPC::Cmd::can_run($name);
2503     } else {
2504         # if there is $directories component in splitpath,
2505         # then it's not something to test with $PATH...
2506         return $name if (File::Spec->splitpath($name))[1];
2507
2508         foreach (File::Spec->path()) {
2509             my $fullpath = catfile($_, "$name$target{exe_extension}");
2510             if (-f $fullpath and -x $fullpath) {
2511                 return $fullpath;
2512             }
2513         }
2514     }
2515 }
2516
2517 # Configuration printer ##############################################
2518
2519 sub print_table_entry
2520 {
2521     my $target = shift;
2522     my %target = resolve_config($target);
2523     my $type = shift;
2524
2525     # Don't print the templates
2526     return if $target{template};
2527
2528     my @sequence = (
2529         "sys_id",
2530         "cc",
2531         "cflags",
2532         "defines",
2533         "unistd",
2534         "ld",
2535         "lflags",
2536         "loutflag",
2537         "plib_lflags",
2538         "ex_libs",
2539         "bn_ops",
2540         "apps_aux_src",
2541         "cpuid_asm_src",
2542         "uplink_aux_src",
2543         "bn_asm_src",
2544         "ec_asm_src",
2545         "des_asm_src",
2546         "aes_asm_src",
2547         "bf_asm_src",
2548         "md5_asm_src",
2549         "cast_asm_src",
2550         "sha1_asm_src",
2551         "rc4_asm_src",
2552         "rmd160_asm_src",
2553         "rc5_asm_src",
2554         "wp_asm_src",
2555         "cmll_asm_src",
2556         "modes_asm_src",
2557         "padlock_asm_src",
2558         "chacha_asm_src",
2559         "poly1035_asm_src",
2560         "thread_scheme",
2561         "perlasm_scheme",
2562         "dso_scheme",
2563         "shared_target",
2564         "shared_cflag",
2565         "shared_defines",
2566         "shared_ldflag",
2567         "shared_rcflag",
2568         "shared_extension",
2569         "dso_extension",
2570         "obj_extension",
2571         "exe_extension",
2572         "ranlib",
2573         "ar",
2574         "arflags",
2575         "aroutflag",
2576         "rc",
2577         "rcflags",
2578         "rcoutflag",
2579         "mt",
2580         "mtflags",
2581         "mtinflag",
2582         "mtoutflag",
2583         "multilib",
2584         "build_scheme",
2585         );
2586
2587     if ($type eq "TABLE") {
2588         print "\n";
2589         print "*** $target\n";
2590         foreach (@sequence) {
2591             if (ref($target{$_}) eq "ARRAY") {
2592                 printf "\$%-12s = %s\n", $_, join(" ", @{$target{$_}});
2593             } else {
2594                 printf "\$%-12s = %s\n", $_, $target{$_};
2595             }
2596         }
2597     } elsif ($type eq "HASH") {
2598         my $largest =
2599             length((sort { length($a) <=> length($b) } @sequence)[-1]);
2600         print "    '$target' => {\n";
2601         foreach (@sequence) {
2602             if ($target{$_}) {
2603                 if (ref($target{$_}) eq "ARRAY") {
2604                     print "      '",$_,"'"," " x ($largest - length($_))," => [ ",join(", ", map { "'$_'" } @{$target{$_}})," ],\n";
2605                 } else {
2606                     print "      '",$_,"'"," " x ($largest - length($_))," => '",$target{$_},"',\n";
2607                 }
2608             }
2609         }
2610         print "    },\n";
2611     }
2612 }
2613
2614 # Utility routines ###################################################
2615
2616 # On VMS, if the given file is a logical name, File::Spec::Functions
2617 # will consider it an absolute path.  There are cases when we want a
2618 # purely syntactic check without checking the environment.
2619 sub isabsolute {
2620     my $file = shift;
2621
2622     # On non-platforms, we just use file_name_is_absolute().
2623     return file_name_is_absolute($file) unless $^O eq "VMS";
2624
2625     # If the file spec includes a device or a directory spec,
2626     # file_name_is_absolute() is perfectly safe.
2627     return file_name_is_absolute($file) if $file =~ m|[:\[]|;
2628
2629     # Here, we know the given file spec isn't absolute
2630     return 0;
2631 }
2632
2633 # Makes a directory absolute and cleans out /../ in paths like foo/../bar
2634 # On some platforms, this uses rel2abs(), while on others, realpath() is used.
2635 # realpath() requires that at least all path components except the last is an
2636 # existing directory.  On VMS, the last component of the directory spec must
2637 # exist.
2638 sub absolutedir {
2639     my $dir = shift;
2640
2641     # realpath() is quite buggy on VMS.  It uses LIB$FID_TO_NAME, which
2642     # will return the volume name for the device, no matter what.  Also,
2643     # it will return an incorrect directory spec if the argument is a
2644     # directory that doesn't exist.
2645     if ($^O eq "VMS") {
2646         return rel2abs($dir);
2647     }
2648
2649     # We use realpath() on Unix, since no other will properly clean out
2650     # a directory spec.
2651     use Cwd qw/realpath/;
2652
2653     return realpath($dir);
2654 }
2655
2656 sub quotify {
2657     my %processors = (
2658         perl    => sub { my $x = shift;
2659                          $x =~ s/([\\\$\@"])/\\$1/g;
2660                          return '"'.$x.'"'; },
2661         maybeshell => sub { my $x = shift;
2662                             (my $y = $x) =~ s/([\\\"])/\\$1/g;
2663                             if ($x ne $y || $x =~ m|\s|) {
2664                                 return '"'.$y.'"';
2665                             } else {
2666                                 return $x;
2667                             }
2668                         },
2669         );
2670     my $for = shift;
2671     my $processor =
2672         defined($processors{$for}) ? $processors{$for} : sub { shift; };
2673
2674     return map { $processor->($_); } @_;
2675 }
2676
2677 # collect_from_file($filename, $line_concat_cond_re, $line_concat)
2678 # $filename is a file name to read from
2679 # $line_concat_cond_re is a regexp detecting a line continuation ending
2680 # $line_concat is a CODEref that takes care of concatenating two lines
2681 sub collect_from_file {
2682     my $filename = shift;
2683     my $line_concat_cond_re = shift;
2684     my $line_concat = shift;
2685
2686     open my $fh, $filename || die "unable to read $filename: $!\n";
2687     return sub {
2688         my $saved_line = "";
2689         $_ = "";
2690         while (<$fh>) {
2691             s|\R$||;
2692             if (defined $line_concat) {
2693                 $_ = $line_concat->($saved_line, $_);
2694                 $saved_line = "";
2695             }
2696             if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
2697                 $saved_line = $_;
2698                 next;
2699             }
2700             return $_;
2701         }
2702         die "$filename ending with continuation line\n" if $_;
2703         close $fh;
2704         return undef;
2705     }
2706 }
2707
2708 # collect_from_array($array, $line_concat_cond_re, $line_concat)
2709 # $array is an ARRAYref of lines
2710 # $line_concat_cond_re is a regexp detecting a line continuation ending
2711 # $line_concat is a CODEref that takes care of concatenating two lines
2712 sub collect_from_array {
2713     my $array = shift;
2714     my $line_concat_cond_re = shift;
2715     my $line_concat = shift;
2716     my @array = (@$array);
2717
2718     return sub {
2719         my $saved_line = "";
2720         $_ = "";
2721         while (defined($_ = shift @array)) {
2722             s|\R$||;
2723             if (defined $line_concat) {
2724                 $_ = $line_concat->($saved_line, $_);
2725                 $saved_line = "";
2726             }
2727             if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
2728                 $saved_line = $_;
2729                 next;
2730             }
2731             return $_;
2732         }
2733         die "input text ending with continuation line\n" if $_;
2734         return undef;
2735     }
2736 }
2737
2738 # collect_information($lineiterator, $line_continue, $regexp => $CODEref, ...)
2739 # $lineiterator is a CODEref that delivers one line at a time.
2740 # All following arguments are regex/CODEref pairs, where the regexp detects a
2741 # line and the CODEref does something with the result of the regexp.
2742 sub collect_information {
2743     my $lineiterator = shift;
2744     my %collectors = @_;
2745
2746     while(defined($_ = $lineiterator->())) {
2747         s|\R$||;
2748         my $found = 0;
2749         if ($collectors{"BEFORE"}) {
2750             $collectors{"BEFORE"}->($_);
2751         }
2752         foreach my $re (keys %collectors) {
2753             if ($re !~ /^OTHERWISE|BEFORE|AFTER$/ && /$re/) {
2754                 $collectors{$re}->($lineiterator);
2755                 $found = 1;
2756             };
2757         }
2758         if ($collectors{"OTHERWISE"}) {
2759             $collectors{"OTHERWISE"}->($lineiterator, $_)
2760                 unless $found || !defined $collectors{"OTHERWISE"};
2761         }
2762         if ($collectors{"AFTER"}) {
2763             $collectors{"AFTER"}->($_);
2764         }
2765     }
2766 }
2767
2768 # tokenize($line)
2769 # $line is a line of text to split up into tokens
2770 # returns a list of tokens
2771 #
2772 # Tokens are divided by spaces.  If the tokens include spaces, they
2773 # have to be quoted with single or double quotes.  Double quotes
2774 # inside a double quoted token must be escaped.  Escaping is done
2775 # with backslash.
2776 # Basically, the same quoting rules apply for " and ' as in any
2777 # Unix shell.
2778 sub tokenize {
2779     my $line = my $debug_line = shift;
2780     my @result = ();
2781
2782     while ($line =~ s|^\s+||, $line ne "") {
2783         my $token = "";
2784         while ($line ne "" && $line !~ m|^\s|) {
2785             if ($line =~ m/^"((?:[^"\\]+|\\.)*)"/) {
2786                 $token .= $1;
2787                 $line = $';
2788             } elsif ($line =~ m/^'([^']*)'/) {
2789                 $token .= $1;
2790                 $line = $';
2791             } elsif ($line =~ m/^(\S+)/) {
2792                 $token .= $1;
2793                 $line = $';
2794             }
2795         }
2796         push @result, $token;
2797     }
2798
2799     if ($ENV{CONFIGURE_DEBUG_TOKENIZE}) {
2800         print STDERR "DEBUG[tokenize]: Parsed '$debug_line' into:\n";
2801         print STDERR "DEBUG[tokenize]: ('", join("', '", @result), "')\n";
2802     }
2803     return @result;
2804 }