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