933b9e6affbcc91e119c8e7ff117043cf214e2ea
[oweals/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             }
1851         }
1852
1853         foreach (keys %includes) {
1854             my $dest = $_;
1855             my $ddest = cleanfile($sourced, $_, $blddir);
1856
1857             # If the destination doesn't exist in source, it can only be
1858             # a generated file in the build tree.
1859             if (! -f $ddest) {
1860                 $ddest = cleanfile($buildd, $_, $blddir);
1861                 if ($unified_info{rename}->{$ddest}) {
1862                     $ddest = $unified_info{rename}->{$ddest};
1863                 }
1864             }
1865             foreach (@{$includes{$dest}}) {
1866                 my $is = cleandir($sourced, $_, $blddir);
1867                 my $ib = cleandir($buildd, $_, $blddir);
1868                 push @{$unified_info{includes}->{$ddest}->{source}}, $is
1869                     unless grep { $_ eq $is } @{$unified_info{includes}->{$ddest}->{source}};
1870                 push @{$unified_info{includes}->{$ddest}->{build}}, $ib
1871                     unless grep { $_ eq $ib } @{$unified_info{includes}->{$ddest}->{build}};
1872             }
1873         }
1874     }
1875
1876     # Massage the result
1877
1878     # If we depend on a header file or a perl module, add an inclusion of
1879     # its directory to allow smoothe inclusion
1880     foreach my $dest (keys %{$unified_info{depends}}) {
1881         next if $dest eq "";
1882         foreach my $d (keys %{$unified_info{depends}->{$dest}}) {
1883             next unless $d =~ /\.(h|pm)$/;
1884             if ($d eq "configdata.pm"
1885                     || defined($unified_info{generate}->{$d})) {
1886                 my $i = cleandir($blddir, dirname($d));
1887                 push @{$unified_info{includes}->{$dest}->{build}}, $i
1888                     unless grep { $_ eq $i } @{$unified_info{includes}->{$dest}->{build}};
1889             } else {
1890                 my $i = cleandir($srcdir, dirname($d));
1891                 push @{$unified_info{includes}->{$dest}->{source}}, $i
1892                     unless grep { $_ eq $i } @{$unified_info{includes}->{$dest}->{source}};
1893             }
1894         }
1895     }
1896
1897     # Trickle down includes placed on libraries, engines and programs to
1898     # their sources (i.e. object files)
1899     foreach my $dest (keys %{$unified_info{engines}},
1900                       keys %{$unified_info{libraries}},
1901                       keys %{$unified_info{programs}}) {
1902         foreach my $k (("source", "build")) {
1903             next unless defined($unified_info{includes}->{$dest}->{$k});
1904             my @incs = reverse @{$unified_info{includes}->{$dest}->{$k}};
1905             foreach my $obj (grep /\.o$/,
1906                              (keys %{$unified_info{sources}->{$dest}},
1907                               keys %{$unified_info{shared_sources}->{$dest}})) {
1908                 foreach my $inc (@incs) {
1909                     unshift @{$unified_info{includes}->{$obj}->{$k}}, $inc
1910                         unless grep { $_ eq $inc } @{$unified_info{includes}->{$obj}->{$k}};
1911                 }
1912             }
1913         }
1914         delete $unified_info{includes}->{$dest};
1915     }
1916
1917     ### Make unified_info a bit more efficient
1918     # One level structures
1919     foreach (("programs", "libraries", "engines", "scripts", "extra", "overrides")) {
1920         $unified_info{$_} = [ sort keys %{$unified_info{$_}} ];
1921     }
1922     # Two level structures
1923     foreach my $l1 (("install", "sources", "shared_sources", "ldadd", "depends")) {
1924         foreach my $l2 (sort keys %{$unified_info{$l1}}) {
1925             $unified_info{$l1}->{$l2} =
1926                 [ sort keys %{$unified_info{$l1}->{$l2}} ];
1927         }
1928     }
1929     # Includes
1930     foreach my $dest (sort keys %{$unified_info{includes}}) {
1931         if (defined($unified_info{includes}->{$dest}->{build})) {
1932             my @source_includes = ();
1933             @source_includes = ( @{$unified_info{includes}->{$dest}->{source}} )
1934                 if defined($unified_info{includes}->{$dest}->{source});
1935             $unified_info{includes}->{$dest} =
1936                 [ @{$unified_info{includes}->{$dest}->{build}} ];
1937             foreach my $inc (@source_includes) {
1938                 push @{$unified_info{includes}->{$dest}}, $inc
1939                     unless grep { $_ eq $inc } @{$unified_info{includes}->{$dest}};
1940             }
1941         } else {
1942             $unified_info{includes}->{$dest} =
1943                 [ @{$unified_info{includes}->{$dest}->{source}} ];
1944         }
1945     }
1946 }
1947
1948 # For the schemes that need it, we provide the old *_obj configs
1949 # from the *_asm_obj ones
1950 foreach (grep /_(asm|aux)_src$/, keys %target) {
1951     my $src = $_;
1952     (my $obj = $_) =~ s/_(asm|aux)_src$/_obj/;
1953     ($target{$obj} = $target{$src}) =~ s/\.[csS]\b/.o/g;
1954 }
1955
1956 # Write down our configuration where it fits #########################
1957
1958 open(OUT,">configdata.pm") || die "unable to create configdata.pm: $!\n";
1959 print OUT <<"EOF";
1960 package configdata;
1961
1962 use strict;
1963 use warnings;
1964
1965 use Exporter;
1966 #use vars qw(\@ISA \@EXPORT);
1967 our \@ISA = qw(Exporter);
1968 our \@EXPORT = qw(\%config \%target \%disabled \%withargs \%unified_info \@disablables);
1969
1970 EOF
1971 print OUT "our %config = (\n";
1972 foreach (sort keys %config) {
1973     if (ref($config{$_}) eq "ARRAY") {
1974         print OUT "  ", $_, " => [ ", join(", ",
1975                                            map { quotify("perl", $_) }
1976                                            @{$config{$_}}), " ],\n";
1977     } else {
1978         print OUT "  ", $_, " => ", quotify("perl", $config{$_}), ",\n"
1979     }
1980 }
1981 print OUT <<"EOF";
1982 );
1983
1984 EOF
1985 print OUT "our %target = (\n";
1986 foreach (sort keys %target) {
1987     if (ref($target{$_}) eq "ARRAY") {
1988         print OUT "  ", $_, " => [ ", join(", ",
1989                                            map { quotify("perl", $_) }
1990                                            @{$target{$_}}), " ],\n";
1991     } else {
1992         print OUT "  ", $_, " => ", quotify("perl", $target{$_}), ",\n"
1993     }
1994 }
1995 print OUT <<"EOF";
1996 );
1997
1998 EOF
1999 print OUT "our \%available_protocols = (\n";
2000 print OUT "  tls => [ ", join(", ", map { quotify("perl", $_) } @tls), " ],\n";
2001 print OUT "  dtls => [ ", join(", ", map { quotify("perl", $_) } @dtls), " ],\n";
2002 print OUT <<"EOF";
2003 );
2004
2005 EOF
2006 print OUT "our \@disablables = (\n";
2007 foreach (@disablables) {
2008     print OUT "  ", quotify("perl", $_), ",\n";
2009 }
2010 print OUT <<"EOF";
2011 );
2012
2013 EOF
2014 print OUT "our \%disabled = (\n";
2015 foreach (sort keys %disabled) {
2016     print OUT "  ", quotify("perl", $_), " => ", quotify("perl", $disabled{$_}), ",\n";
2017 }
2018 print OUT <<"EOF";
2019 );
2020
2021 EOF
2022 print OUT "our %withargs = (\n";
2023 foreach (sort keys %withargs) {
2024     if (ref($withargs{$_}) eq "ARRAY") {
2025         print OUT "  ", $_, " => [ ", join(", ",
2026                                            map { quotify("perl", $_) }
2027                                            @{$withargs{$_}}), " ],\n";
2028     } else {
2029         print OUT "  ", $_, " => ", quotify("perl", $withargs{$_}), ",\n"
2030     }
2031 }
2032 print OUT <<"EOF";
2033 );
2034
2035 EOF
2036 if ($builder eq "unified") {
2037     my $recurse;
2038     $recurse = sub {
2039         my $indent = shift;
2040         foreach (@_) {
2041             if (ref $_ eq "ARRAY") {
2042                 print OUT " "x$indent, "[\n";
2043                 foreach (@$_) {
2044                     $recurse->($indent + 4, $_);
2045                 }
2046                 print OUT " "x$indent, "],\n";
2047             } elsif (ref $_ eq "HASH") {
2048                 my %h = %$_;
2049                 print OUT " "x$indent, "{\n";
2050                 foreach (sort keys %h) {
2051                     if (ref $h{$_} eq "") {
2052                         print OUT " "x($indent + 4), quotify("perl", $_), " => ", quotify("perl", $h{$_}), ",\n";
2053                     } else {
2054                         print OUT " "x($indent + 4), quotify("perl", $_), " =>\n";
2055                         $recurse->($indent + 8, $h{$_});
2056                     }
2057                 }
2058                 print OUT " "x$indent, "},\n";
2059             } else {
2060                 print OUT " "x$indent, quotify("perl", $_), ",\n";
2061             }
2062         }
2063     };
2064     print OUT "our %unified_info = (\n";
2065     foreach (sort keys %unified_info) {
2066         if (ref $unified_info{$_} eq "") {
2067             print OUT " "x4, quotify("perl", $_), " => ", quotify("perl", $unified_info{$_}), ",\n";
2068         } else {
2069             print OUT " "x4, quotify("perl", $_), " =>\n";
2070             $recurse->(8, $unified_info{$_});
2071         }
2072     }
2073     print OUT <<"EOF";
2074 );
2075
2076 EOF
2077 }
2078 print OUT "1;\n";
2079 close(OUT);
2080
2081
2082 print "CC            =$config{cross_compile_prefix}$target{cc}\n";
2083 print "CFLAG         =$target{cflags} $config{cflags}\n";
2084 print "SHARED_CFLAG  =$target{shared_cflag}\n";
2085 print "DEFINES       =",join(" ", @{$target{defines}}, @{$config{defines}}),"\n";
2086 print "LFLAG         =$target{lflags}\n";
2087 print "PLIB_LFLAG    =$target{plib_lflags}\n";
2088 print "EX_LIBS       =$target{ex_libs} $config{ex_libs}\n";
2089 print "APPS_OBJ      =$target{apps_obj}\n";
2090 print "CPUID_OBJ     =$target{cpuid_obj}\n";
2091 print "UPLINK_OBJ    =$target{uplink_obj}\n";
2092 print "BN_ASM        =$target{bn_obj}\n";
2093 print "EC_ASM        =$target{ec_obj}\n";
2094 print "DES_ENC       =$target{des_obj}\n";
2095 print "AES_ENC       =$target{aes_obj}\n";
2096 print "BF_ENC        =$target{bf_obj}\n";
2097 print "CAST_ENC      =$target{cast_obj}\n";
2098 print "RC4_ENC       =$target{rc4_obj}\n";
2099 print "RC5_ENC       =$target{rc5_obj}\n";
2100 print "MD5_OBJ_ASM   =$target{md5_obj}\n";
2101 print "SHA1_OBJ_ASM  =$target{sha1_obj}\n";
2102 print "RMD160_OBJ_ASM=$target{rmd160_obj}\n";
2103 print "CMLL_ENC      =$target{cmll_obj}\n";
2104 print "MODES_OBJ     =$target{modes_obj}\n";
2105 print "PADLOCK_OBJ   =$target{padlock_obj}\n";
2106 print "CHACHA_ENC    =$target{chacha_obj}\n";
2107 print "POLY1305_OBJ  =$target{poly1305_obj}\n";
2108 print "BLAKE2_OBJ    =$target{blake2_obj}\n";
2109 print "PROCESSOR     =$config{processor}\n";
2110 print "RANLIB        =", $target{ranlib} eq '$(CROSS_COMPILE)ranlib' ?
2111                              "$config{cross_compile_prefix}ranlib" :
2112                              "$target{ranlib}", "\n";
2113 print "ARFLAGS       =$target{arflags}\n";
2114 print "PERL          =$config{perl}\n";
2115 print "\n";
2116 print "SIXTY_FOUR_BIT_LONG mode\n" if $config{b64l};
2117 print "SIXTY_FOUR_BIT mode\n" if $config{b64};
2118 print "THIRTY_TWO_BIT mode\n" if $config{b32};
2119 print "BN_LLONG mode\n" if $config{bn_ll};
2120 print "RC4 uses $config{rc4_int}\n" if $config{rc4_int} ne $def_int;
2121
2122 my %builders = (
2123     unified => sub {
2124         run_dofile(catfile($blddir, $target{build_file}),
2125                    @{$config{build_file_templates}});
2126     },
2127     );
2128
2129 $builders{$builder}->($builder_platform, @builder_opts);
2130
2131 print <<"EOF";
2132
2133 Configured for $target.
2134 EOF
2135
2136 print <<"EOF" if ($disabled{threads} eq "unavailable");
2137
2138 The library could not be configured for supporting multi-threaded
2139 applications as the compiler options required on this system are not known.
2140 See file INSTALL for details if you need multi-threading.
2141 EOF
2142
2143 print <<"EOF" if ($no_shared_warn);
2144
2145 The options 'shared', 'pic' and 'dynamic-engine' aren't supported on this
2146 platform, so we will pretend you gave the option 'no-pic', which also disables
2147 'shared' and 'dynamic-engine'.  If you know how to implement shared libraries
2148 or position independent code, please let us know (but please first make sure
2149 you have tried with a current version of OpenSSL).
2150 EOF
2151
2152 print <<"EOF" if (-f catfile($srcdir, "configdata.pm") && $srcdir ne $blddir);
2153
2154 WARNING: there are indications that another build was made in the source
2155 directory.  This build may have picked up artifacts from that build, the
2156 safest course of action is to clean the source directory and redo this
2157 configuration.
2158 EOF
2159
2160 exit(0);
2161
2162 ######################################################################
2163 #
2164 # Helpers and utility functions
2165 #
2166
2167 # Configuration file reading #########################################
2168
2169 # Note: All of the helper functions are for lazy evaluation.  They all
2170 # return a CODE ref, which will return the intended value when evaluated.
2171 # Thus, whenever there's mention of a returned value, it's about that
2172 # intended value.
2173
2174 # Helper function to implement conditional inheritance depending on the
2175 # value of $disabled{asm}.  Used in inherit_from values as follows:
2176 #
2177 #      inherit_from => [ "template", asm("asm_tmpl") ]
2178 #
2179 sub asm {
2180     my @x = @_;
2181     sub {
2182         $disabled{asm} ? () : @x;
2183     }
2184 }
2185
2186 # Helper function to implement conditional value variants, with a default
2187 # plus additional values based on the value of $config{build_type}.
2188 # Arguments are given in hash table form:
2189 #
2190 #       picker(default => "Basic string: ",
2191 #              debug   => "debug",
2192 #              release => "release")
2193 #
2194 # When configuring with --debug, the resulting string will be
2195 # "Basic string: debug", and when not, it will be "Basic string: release"
2196 #
2197 # This can be used to create variants of sets of flags according to the
2198 # build type:
2199 #
2200 #       cflags => picker(default => "-Wall",
2201 #                        debug   => "-g -O0",
2202 #                        release => "-O3")
2203 #
2204 sub picker {
2205     my %opts = @_;
2206     return sub { add($opts{default} || (),
2207                      $opts{$config{build_type}} || ())->(); }
2208 }
2209
2210 # Helper function to combine several values of different types into one.
2211 # This is useful if you want to combine a string with the result of a
2212 # lazy function, such as:
2213 #
2214 #       cflags => combine("-Wall", sub { $disabled{zlib} ? () : "-DZLIB" })
2215 #
2216 sub combine {
2217     my @stuff = @_;
2218     return sub { add(@stuff)->(); }
2219 }
2220
2221 # Helper function to implement conditional values depending on the value
2222 # of $disabled{threads}.  Can be used as follows:
2223 #
2224 #       cflags => combine("-Wall", threads("-pthread"))
2225 #
2226 sub threads {
2227     my @flags = @_;
2228     return sub { add($disabled{threads} ? () : @flags)->(); }
2229 }
2230
2231
2232
2233 our $add_called = 0;
2234 # Helper function to implement adding values to already existing configuration
2235 # values.  It handles elements that are ARRAYs, CODEs and scalars
2236 sub _add {
2237     my $separator = shift;
2238
2239     # If there's any ARRAY in the collection of values OR the separator
2240     # is undef, we will return an ARRAY of combined values, otherwise a
2241     # string of joined values with $separator as the separator.
2242     my $found_array = !defined($separator);
2243
2244     my @values =
2245         map {
2246             my $res = $_;
2247             while (ref($res) eq "CODE") {
2248                 $res = $res->();
2249             }
2250             if (defined($res)) {
2251                 if (ref($res) eq "ARRAY") {
2252                     $found_array = 1;
2253                     @$res;
2254                 } else {
2255                     $res;
2256                 }
2257             } else {
2258                 ();
2259             }
2260     } (@_);
2261
2262     $add_called = 1;
2263
2264     if ($found_array) {
2265         [ @values ];
2266     } else {
2267         join($separator, grep { defined($_) && $_ ne "" } @values);
2268     }
2269 }
2270 sub add_before {
2271     my $separator = " ";
2272     if (ref($_[$#_]) eq "HASH") {
2273         my $opts = pop;
2274         $separator = $opts->{separator};
2275     }
2276     my @x = @_;
2277     sub { _add($separator, @x, @_) };
2278 }
2279 sub add {
2280     my $separator = " ";
2281     if (ref($_[$#_]) eq "HASH") {
2282         my $opts = pop;
2283         $separator = $opts->{separator};
2284     }
2285     my @x = @_;
2286     sub { _add($separator, @_, @x) };
2287 }
2288
2289 # configuration reader, evaluates the input file as a perl script and expects
2290 # it to fill %targets with target configurations.  Those are then added to
2291 # %table.
2292 sub read_config {
2293     my $fname = shift;
2294     open(CONFFILE, "< $fname")
2295         or die "Can't open configuration file '$fname'!\n";
2296     my $x = $/;
2297     undef $/;
2298     my $content = <CONFFILE>;
2299     $/ = $x;
2300     close(CONFFILE);
2301     my %targets = ();
2302     {
2303         # Protect certain tables from tampering
2304         local %table = %::table;
2305
2306         eval $content;
2307         warn $@ if $@;
2308     }
2309     my %preexisting = ();
2310     foreach (sort keys %targets) {
2311         $preexisting{$_} = 1 if $table{$_};
2312     }
2313     die <<"EOF",
2314 The following config targets from $fname
2315 shadow pre-existing config targets with the same name:
2316 EOF
2317         map { "  $_\n" } sort keys %preexisting
2318         if %preexisting;
2319
2320
2321     # For each target, check that it's configured with a hash table.
2322     foreach (keys %targets) {
2323         if (ref($targets{$_}) ne "HASH") {
2324             if (ref($targets{$_}) eq "") {
2325                 warn "Deprecated target configuration for $_, ignoring...\n";
2326             } else {
2327                 warn "Misconfigured target configuration for $_ (should be a hash table), ignoring...\n";
2328             }
2329             delete $targets{$_};
2330         } else {
2331             $targets{$_}->{_conf_fname_int} = add([ $fname ]);
2332         }
2333     }
2334
2335     %table = (%table, %targets);
2336
2337 }
2338
2339 # configuration resolver.  Will only resolve all the lazy evaluation
2340 # codeblocks for the chosen target and all those it inherits from,
2341 # recursively
2342 sub resolve_config {
2343     my $target = shift;
2344     my @breadcrumbs = @_;
2345
2346 #    my $extra_checks = defined($ENV{CONFIGURE_EXTRA_CHECKS});
2347
2348     if (grep { $_ eq $target } @breadcrumbs) {
2349         die "inherit_from loop!  target backtrace:\n  "
2350             ,$target,"\n  ",join("\n  ", @breadcrumbs),"\n";
2351     }
2352
2353     if (!defined($table{$target})) {
2354         warn "Warning! target $target doesn't exist!\n";
2355         return ();
2356     }
2357     # Recurse through all inheritances.  They will be resolved on the
2358     # fly, so when this operation is done, they will all just be a
2359     # bunch of attributes with string values.
2360     # What we get here, though, are keys with references to lists of
2361     # the combined values of them all.  We will deal with lists after
2362     # this stage is done.
2363     my %combined_inheritance = ();
2364     if ($table{$target}->{inherit_from}) {
2365         my @inherit_from =
2366             map { ref($_) eq "CODE" ? $_->() : $_ } @{$table{$target}->{inherit_from}};
2367         foreach (@inherit_from) {
2368             my %inherited_config = resolve_config($_, $target, @breadcrumbs);
2369
2370             # 'template' is a marker that's considered private to
2371             # the config that had it.
2372             delete $inherited_config{template};
2373
2374             foreach (keys %inherited_config) {
2375                 if (!$combined_inheritance{$_}) {
2376                     $combined_inheritance{$_} = [];
2377                 }
2378                 push @{$combined_inheritance{$_}}, $inherited_config{$_};
2379             }
2380         }
2381     }
2382
2383     # We won't need inherit_from in this target any more, since we've
2384     # resolved all the inheritances that lead to this
2385     delete $table{$target}->{inherit_from};
2386
2387     # Now is the time to deal with those lists.  Here's the place to
2388     # decide what shall be done with those lists, all based on the
2389     # values of the target we're currently dealing with.
2390     # - If a value is a coderef, it will be executed with the list of
2391     #   inherited values as arguments.
2392     # - If the corresponding key doesn't have a value at all or is the
2393     #   empty string, the inherited value list will be run through the
2394     #   default combiner (below), and the result becomes this target's
2395     #   value.
2396     # - Otherwise, this target's value is assumed to be a string that
2397     #   will simply override the inherited list of values.
2398     my $default_combiner = add();
2399
2400     my %all_keys =
2401         map { $_ => 1 } (keys %combined_inheritance,
2402                          keys %{$table{$target}});
2403
2404     sub process_values {
2405         my $object    = shift;
2406         my $inherited = shift;  # Always a [ list ]
2407         my $target    = shift;
2408         my $entry     = shift;
2409
2410         $add_called = 0;
2411
2412         while(ref($object) eq "CODE") {
2413             $object = $object->(@$inherited);
2414         }
2415         if (!defined($object)) {
2416             return ();
2417         }
2418         elsif (ref($object) eq "ARRAY") {
2419             local $add_called;  # To make sure recursive calls don't affect it
2420             return [ map { process_values($_, $inherited, $target, $entry) }
2421                      @$object ];
2422         } elsif (ref($object) eq "") {
2423             return $object;
2424         } else {
2425             die "cannot handle reference type ",ref($object)
2426                 ," found in target ",$target," -> ",$entry,"\n";
2427         }
2428     }
2429
2430     foreach (sort keys %all_keys) {
2431         my $previous = $combined_inheritance{$_};
2432
2433         # Current target doesn't have a value for the current key?
2434         # Assign it the default combiner, the rest of this loop body
2435         # will handle it just like any other coderef.
2436         if (!exists $table{$target}->{$_}) {
2437             $table{$target}->{$_} = $default_combiner;
2438         }
2439
2440         $table{$target}->{$_} = process_values($table{$target}->{$_},
2441                                                $combined_inheritance{$_},
2442                                                $target, $_);
2443         unless(defined($table{$target}->{$_})) {
2444             delete $table{$target}->{$_};
2445         }
2446 #        if ($extra_checks &&
2447 #            $previous && !($add_called ||  $previous ~~ $table{$target}->{$_})) {
2448 #            warn "$_ got replaced in $target\n";
2449 #        }
2450     }
2451
2452     # Finally done, return the result.
2453     return %{$table{$target}};
2454 }
2455
2456 sub usage
2457         {
2458         print STDERR $usage;
2459         print STDERR "\npick os/compiler from:\n";
2460         my $j=0;
2461         my $i;
2462         my $k=0;
2463         foreach $i (sort keys %table)
2464                 {
2465                 next if $table{$i}->{template};
2466                 next if $i =~ /^debug/;
2467                 $k += length($i) + 1;
2468                 if ($k > 78)
2469                         {
2470                         print STDERR "\n";
2471                         $k=length($i);
2472                         }
2473                 print STDERR $i . " ";
2474                 }
2475         foreach $i (sort keys %table)
2476                 {
2477                 next if $table{$i}->{template};
2478                 next if $i !~ /^debug/;
2479                 $k += length($i) + 1;
2480                 if ($k > 78)
2481                         {
2482                         print STDERR "\n";
2483                         $k=length($i);
2484                         }
2485                 print STDERR $i . " ";
2486                 }
2487         print STDERR "\n\nNOTE: If in doubt, on Unix-ish systems use './config'.\n";
2488         exit(1);
2489         }
2490
2491 sub run_dofile
2492 {
2493     my $out = shift;
2494     my @templates = @_;
2495
2496     unlink $out || warn "Can't remove $out, $!"
2497         if -f $out;
2498     foreach (@templates) {
2499         die "Can't open $_, $!" unless -f $_;
2500     }
2501     my $perlcmd = (quotify("maybeshell", $config{perl}))[0];
2502     my $cmd = "$perlcmd \"-I.\" \"-Mconfigdata\" \"$dofile\" -o\"Configure\" \"".join("\" \"",@templates)."\" > \"$out.new\"";
2503     #print STDERR "DEBUG[run_dofile]: \$cmd = $cmd\n";
2504     system($cmd);
2505     exit 1 if $? != 0;
2506     rename("$out.new", $out) || die "Can't rename $out.new, $!";
2507 }
2508
2509 sub which
2510 {
2511     my ($name)=@_;
2512
2513     if (eval { require IPC::Cmd; 1; }) {
2514         IPC::Cmd->import();
2515         return scalar IPC::Cmd::can_run($name);
2516     } else {
2517         # if there is $directories component in splitpath,
2518         # then it's not something to test with $PATH...
2519         return $name if (File::Spec->splitpath($name))[1];
2520
2521         foreach (File::Spec->path()) {
2522             my $fullpath = catfile($_, "$name$target{exe_extension}");
2523             if (-f $fullpath and -x $fullpath) {
2524                 return $fullpath;
2525             }
2526         }
2527     }
2528 }
2529
2530 # Configuration printer ##############################################
2531
2532 sub print_table_entry
2533 {
2534     my $target = shift;
2535     my %target = resolve_config($target);
2536     my $type = shift;
2537
2538     # Don't print the templates
2539     return if $target{template};
2540
2541     my @sequence = (
2542         "sys_id",
2543         "cc",
2544         "cflags",
2545         "defines",
2546         "unistd",
2547         "ld",
2548         "lflags",
2549         "loutflag",
2550         "plib_lflags",
2551         "ex_libs",
2552         "bn_ops",
2553         "apps_aux_src",
2554         "cpuid_asm_src",
2555         "uplink_aux_src",
2556         "bn_asm_src",
2557         "ec_asm_src",
2558         "des_asm_src",
2559         "aes_asm_src",
2560         "bf_asm_src",
2561         "md5_asm_src",
2562         "cast_asm_src",
2563         "sha1_asm_src",
2564         "rc4_asm_src",
2565         "rmd160_asm_src",
2566         "rc5_asm_src",
2567         "wp_asm_src",
2568         "cmll_asm_src",
2569         "modes_asm_src",
2570         "padlock_asm_src",
2571         "chacha_asm_src",
2572         "poly1035_asm_src",
2573         "thread_scheme",
2574         "perlasm_scheme",
2575         "dso_scheme",
2576         "shared_target",
2577         "shared_cflag",
2578         "shared_defines",
2579         "shared_ldflag",
2580         "shared_rcflag",
2581         "shared_extension",
2582         "dso_extension",
2583         "obj_extension",
2584         "exe_extension",
2585         "ranlib",
2586         "ar",
2587         "arflags",
2588         "aroutflag",
2589         "rc",
2590         "rcflags",
2591         "rcoutflag",
2592         "mt",
2593         "mtflags",
2594         "mtinflag",
2595         "mtoutflag",
2596         "multilib",
2597         "build_scheme",
2598         );
2599
2600     if ($type eq "TABLE") {
2601         print "\n";
2602         print "*** $target\n";
2603         foreach (@sequence) {
2604             if (ref($target{$_}) eq "ARRAY") {
2605                 printf "\$%-12s = %s\n", $_, join(" ", @{$target{$_}});
2606             } else {
2607                 printf "\$%-12s = %s\n", $_, $target{$_};
2608             }
2609         }
2610     } elsif ($type eq "HASH") {
2611         my $largest =
2612             length((sort { length($a) <=> length($b) } @sequence)[-1]);
2613         print "    '$target' => {\n";
2614         foreach (@sequence) {
2615             if ($target{$_}) {
2616                 if (ref($target{$_}) eq "ARRAY") {
2617                     print "      '",$_,"'"," " x ($largest - length($_))," => [ ",join(", ", map { "'$_'" } @{$target{$_}})," ],\n";
2618                 } else {
2619                     print "      '",$_,"'"," " x ($largest - length($_))," => '",$target{$_},"',\n";
2620                 }
2621             }
2622         }
2623         print "    },\n";
2624     }
2625 }
2626
2627 # Utility routines ###################################################
2628
2629 # On VMS, if the given file is a logical name, File::Spec::Functions
2630 # will consider it an absolute path.  There are cases when we want a
2631 # purely syntactic check without checking the environment.
2632 sub isabsolute {
2633     my $file = shift;
2634
2635     # On non-platforms, we just use file_name_is_absolute().
2636     return file_name_is_absolute($file) unless $^O eq "VMS";
2637
2638     # If the file spec includes a device or a directory spec,
2639     # file_name_is_absolute() is perfectly safe.
2640     return file_name_is_absolute($file) if $file =~ m|[:\[]|;
2641
2642     # Here, we know the given file spec isn't absolute
2643     return 0;
2644 }
2645
2646 # Makes a directory absolute and cleans out /../ in paths like foo/../bar
2647 # On some platforms, this uses rel2abs(), while on others, realpath() is used.
2648 # realpath() requires that at least all path components except the last is an
2649 # existing directory.  On VMS, the last component of the directory spec must
2650 # exist.
2651 sub absolutedir {
2652     my $dir = shift;
2653
2654     # realpath() is quite buggy on VMS.  It uses LIB$FID_TO_NAME, which
2655     # will return the volume name for the device, no matter what.  Also,
2656     # it will return an incorrect directory spec if the argument is a
2657     # directory that doesn't exist.
2658     if ($^O eq "VMS") {
2659         return rel2abs($dir);
2660     }
2661
2662     # We use realpath() on Unix, since no other will properly clean out
2663     # a directory spec.
2664     use Cwd qw/realpath/;
2665
2666     return realpath($dir);
2667 }
2668
2669 sub quotify {
2670     my %processors = (
2671         perl    => sub { my $x = shift;
2672                          $x =~ s/([\\\$\@"])/\\$1/g;
2673                          return '"'.$x.'"'; },
2674         maybeshell => sub { my $x = shift;
2675                             (my $y = $x) =~ s/([\\\"])/\\$1/g;
2676                             if ($x ne $y || $x =~ m|\s|) {
2677                                 return '"'.$y.'"';
2678                             } else {
2679                                 return $x;
2680                             }
2681                         },
2682         );
2683     my $for = shift;
2684     my $processor =
2685         defined($processors{$for}) ? $processors{$for} : sub { shift; };
2686
2687     return map { $processor->($_); } @_;
2688 }
2689
2690 # collect_from_file($filename, $line_concat_cond_re, $line_concat)
2691 # $filename is a file name to read from
2692 # $line_concat_cond_re is a regexp detecting a line continuation ending
2693 # $line_concat is a CODEref that takes care of concatenating two lines
2694 sub collect_from_file {
2695     my $filename = shift;
2696     my $line_concat_cond_re = shift;
2697     my $line_concat = shift;
2698
2699     open my $fh, $filename || die "unable to read $filename: $!\n";
2700     return sub {
2701         my $saved_line = "";
2702         $_ = "";
2703         while (<$fh>) {
2704             s|\R$||;
2705             if (defined $line_concat) {
2706                 $_ = $line_concat->($saved_line, $_);
2707                 $saved_line = "";
2708             }
2709             if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
2710                 $saved_line = $_;
2711                 next;
2712             }
2713             return $_;
2714         }
2715         die "$filename ending with continuation line\n" if $_;
2716         close $fh;
2717         return undef;
2718     }
2719 }
2720
2721 # collect_from_array($array, $line_concat_cond_re, $line_concat)
2722 # $array is an ARRAYref of lines
2723 # $line_concat_cond_re is a regexp detecting a line continuation ending
2724 # $line_concat is a CODEref that takes care of concatenating two lines
2725 sub collect_from_array {
2726     my $array = shift;
2727     my $line_concat_cond_re = shift;
2728     my $line_concat = shift;
2729     my @array = (@$array);
2730
2731     return sub {
2732         my $saved_line = "";
2733         $_ = "";
2734         while (defined($_ = shift @array)) {
2735             s|\R$||;
2736             if (defined $line_concat) {
2737                 $_ = $line_concat->($saved_line, $_);
2738                 $saved_line = "";
2739             }
2740             if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
2741                 $saved_line = $_;
2742                 next;
2743             }
2744             return $_;
2745         }
2746         die "input text ending with continuation line\n" if $_;
2747         return undef;
2748     }
2749 }
2750
2751 # collect_information($lineiterator, $line_continue, $regexp => $CODEref, ...)
2752 # $lineiterator is a CODEref that delivers one line at a time.
2753 # All following arguments are regex/CODEref pairs, where the regexp detects a
2754 # line and the CODEref does something with the result of the regexp.
2755 sub collect_information {
2756     my $lineiterator = shift;
2757     my %collectors = @_;
2758
2759     while(defined($_ = $lineiterator->())) {
2760         s|\R$||;
2761         my $found = 0;
2762         if ($collectors{"BEFORE"}) {
2763             $collectors{"BEFORE"}->($_);
2764         }
2765         foreach my $re (keys %collectors) {
2766             if ($re !~ /^OTHERWISE|BEFORE|AFTER$/ && /$re/) {
2767                 $collectors{$re}->($lineiterator);
2768                 $found = 1;
2769             };
2770         }
2771         if ($collectors{"OTHERWISE"}) {
2772             $collectors{"OTHERWISE"}->($lineiterator, $_)
2773                 unless $found || !defined $collectors{"OTHERWISE"};
2774         }
2775         if ($collectors{"AFTER"}) {
2776             $collectors{"AFTER"}->($_);
2777         }
2778     }
2779 }
2780
2781 # tokenize($line)
2782 # $line is a line of text to split up into tokens
2783 # returns a list of tokens
2784 #
2785 # Tokens are divided by spaces.  If the tokens include spaces, they
2786 # have to be quoted with single or double quotes.  Double quotes
2787 # inside a double quoted token must be escaped.  Escaping is done
2788 # with backslash.
2789 # Basically, the same quoting rules apply for " and ' as in any
2790 # Unix shell.
2791 sub tokenize {
2792     my $line = my $debug_line = shift;
2793     my @result = ();
2794
2795     while ($line =~ s|^\s+||, $line ne "") {
2796         my $token = "";
2797         while ($line ne "" && $line !~ m|^\s|) {
2798             if ($line =~ m/^"((?:[^"\\]+|\\.)*)"/) {
2799                 $token .= $1;
2800                 $line = $';
2801             } elsif ($line =~ m/^'([^']*)'/) {
2802                 $token .= $1;
2803                 $line = $';
2804             } elsif ($line =~ m/^(\S+)/) {
2805                 $token .= $1;
2806                 $line = $';
2807             }
2808         }
2809         push @result, $token;
2810     }
2811
2812     if ($ENV{CONFIGURE_DEBUG_TOKENIZE}) {
2813         print STDERR "DEBUG[tokenize]: Parsed '$debug_line' into:\n";
2814         print STDERR "DEBUG[tokenize]: ('", join("', '", @result), "')\n";
2815     }
2816     return @result;
2817 }