Configure: clarify and refine -static.
[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                 }
867         else
868                 {
869                 my ($ALGO, $algo);
870                 ($ALGO = $algo = $_) =~ tr/[\-a-z]/[_A-Z]/;
871
872                 if (/^asm$/ || /^err$/ || /^hw$/ || /^hw-/ || /^async$/
873                                 || /^autoalginit/ || /^autoerrinit/)
874                         {
875                         push @{$config{openssl_other_defines}}, "OPENSSL_NO_$ALGO";
876                         print " OPENSSL_NO_$ALGO";
877
878                         if (/^err$/)    { push @user_defines, "OPENSSL_NO_ERR"; }
879                         }
880                 else
881                         {
882                         ($ALGO,$algo) = ("RMD160","rmd160") if ($algo eq "ripemd");
883
884                         push @{$config{openssl_algorithm_defines}}, "OPENSSL_NO_$ALGO";
885                         print " OPENSSL_NO_$ALGO";
886
887                         # fix-up crypto/directory name(s)
888                         $algo="whrlpool" if $algo eq "whirlpool";
889                         $algo="ripemd" if $algo eq "rmd160";
890                         @{$config{sdirs}} = grep { $_ ne $algo} @{$config{sdirs}};
891
892                         print " (skip dir)";
893                         }
894                 }
895
896         print "\n";
897         }
898
899 print "Configuring for $target\n";
900
901 # Support for legacy targets having a name starting with 'debug-'
902 my ($d, $t) = $target =~ m/^(debug-)?(.*)$/;
903 if ($d) {
904     $config{build_type} = "debug";
905
906     # If we do not find debug-foo in the table, the target is set to foo.
907     if (!$table{$target}) {
908         $target = $t;
909     }
910 }
911 $config{target} = $target;
912 my %target = resolve_config($target);
913
914 &usage if (!%target || $target{template});
915
916 %target = ( %{$table{DEFAULTS}}, %target );
917
918 $target{exe_extension}="";
919 $target{exe_extension}=".exe" if ($config{target} eq "DJGPP"
920                                   || $config{target} =~ /^(?:Cygwin|mingw)/);
921 $target{exe_extension}=".pm"  if ($config{target} =~ /vos/);
922
923 ($target{shared_extension_simple}=$target{shared_extension})
924     =~ s|\.\$\(SHLIB_MAJOR\)\.\$\(SHLIB_MINOR\)||;
925 $target{dso_extension}=$target{shared_extension_simple};
926 ($target{shared_import_extension}=$target{shared_extension_simple}.".a")
927     if ($config{target} =~ /^(?:Cygwin|mingw)/);
928
929
930 $config{cross_compile_prefix} = $ENV{'CROSS_COMPILE'}
931     if $config{cross_compile_prefix} eq "";
932
933 # Allow overriding the names of some tools.  USE WITH CARE
934 # Note: only Unix cares about HASHBANGPERL...  that explains
935 # the default string.
936 $config{perl} =    $ENV{'PERL'}    || ($^O ne "VMS" ? $^X : "perl");
937 $config{hashbangperl} =
938     $ENV{'HASHBANGPERL'}           || $ENV{'PERL'}     || "/usr/bin/env perl";
939 $target{cc} =      $ENV{'CC'}      || $target{cc}      || "cc";
940 $target{ranlib} =  $ENV{'RANLIB'}  || $target{ranlib}  ||
941                    (which("$config{cross_compile_prefix}ranlib") ?
942                           "\$(CROSS_COMPILE)ranlib" : "true");
943 $target{ar} =      $ENV{'AR'}      || $target{ar}      || "ar";
944 $target{nm} =      $ENV{'NM'}      || $target{nm}      || "nm";
945 $target{rc} =
946     $ENV{'RC'}  || $ENV{'WINDRES'} || $target{rc}      || "windres";
947
948 # Allow overriding the build file name
949 $target{build_file} = $ENV{BUILDFILE} || $target{build_file} || "Makefile";
950
951 # Cache information necessary for reconfiguration
952 $config{cc} = $target{cc};
953 $config{build_file} = $target{build_file};
954
955 # For cflags, lflags, plib_lflags, ex_libs and defines, add the debug_
956 # or release_ attributes.
957 # Do it in such a way that no spurious space is appended (hence the grep).
958 $config{defines} = [];
959 $config{cflags} = "";
960 $config{ex_libs} = "";
961 $config{shared_ldflag} = "";
962
963 # Make sure build_scheme is consistent.
964 $target{build_scheme} = [ $target{build_scheme} ]
965     if ref($target{build_scheme}) ne "ARRAY";
966
967 my ($builder, $builder_platform, @builder_opts) =
968     @{$target{build_scheme}};
969
970 push @{$config{defines}}, "NDEBUG"    if $config{build_type} eq "release";
971
972 if ($target =~ /^mingw/ && `$target{cc} --target-help 2>&1` =~ m/-mno-cygwin/m)
973         {
974         $config{cflags} .= " -mno-cygwin";
975         $config{shared_ldflag} .= " -mno-cygwin";
976         }
977
978 if ($target =~ /linux.*-mips/ && !$disabled{asm} && $user_cflags !~ /-m(ips|arch=)/) {
979         # minimally required architecture flags for assembly modules
980         $config{cflags}="-mips2 $config{cflags}" if ($target =~ /mips32/);
981         $config{cflags}="-mips3 $config{cflags}" if ($target =~ /mips64/);
982 }
983
984 my $no_shared_warn=0;
985 my $no_user_cflags=0;
986 my $no_user_defines=0;
987
988 # The DSO code currently always implements all functions so that no
989 # applications will have to worry about that from a compilation point
990 # of view. However, the "method"s may return zero unless that platform
991 # has support compiled in for them. Currently each method is enabled
992 # by a define "DSO_<name>" ... we translate the "dso_scheme" config
993 # string entry into using the following logic;
994 if (!$disabled{dso} && $target{dso_scheme} ne "")
995         {
996         $target{dso_scheme} =~ tr/[a-z]/[A-Z]/;
997         if ($target{dso_scheme} eq "DLFCN")
998                 {
999                 unshift @{$config{defines}}, "DSO_DLFCN", "HAVE_DLFCN_H";
1000                 }
1001         elsif ($target{dso_scheme} eq "DLFCN_NO_H")
1002                 {
1003                 unshift @{$config{defines}}, "DSO_DLFCN";
1004                 }
1005         else
1006                 {
1007                 unshift @{$config{defines}}, "DSO_$target{dso_scheme}";
1008                 }
1009         }
1010
1011 $config{ex_libs}="$libs$config{ex_libs}" if ($libs ne "");
1012
1013 if ($disabled{asm})
1014         {
1015         if ($config{fips})
1016                 {
1017                 @{$config{defines}} = grep !/^[BL]_ENDIAN$/, @{$config{defines}};
1018                 @{$target{defines}} = grep !/^[BL]_ENDIAN$/, @{$target{defines}};
1019                 }
1020         }
1021
1022 # If threads aren't disabled, check how possible they are
1023 unless ($disabled{threads}) {
1024     if ($auto_threads) {
1025         # Enabled by default, disable it forcibly if unavailable
1026         if ($target{thread_scheme} eq "(unknown)") {
1027             $disabled{threads} = "unavailable";
1028         }
1029     } else {
1030         # The user chose to enable threads explicitly, let's see
1031         # if there's a chance that's possible
1032         if ($target{thread_scheme} eq "(unknown)") {
1033             # If the user asked for "threads" and we don't have internal
1034             # knowledge how to do it, [s]he is expected to provide any
1035             # system-dependent compiler options that are necessary.  We
1036             # can't truly check that the given options are correct, but
1037             # we expect the user to know what [s]He is doing.
1038             if ($no_user_cflags && $no_user_defines) {
1039                 die "You asked for multi-threading support, but didn't\n"
1040                     ,"provide any system-specific compiler options\n";
1041             }
1042         }
1043     }
1044 }
1045
1046 # If threads still aren't disabled, add a C macro to ensure the source
1047 # code knows about it.  Any other flag is taken care of by the configs.
1048 unless($disabled{threads}) {
1049     foreach (("defines", "openssl_thread_defines")) {
1050         push @{$config{$_}}, "OPENSSL_THREADS";
1051     }
1052 }
1053
1054 # With "deprecated" disable all deprecated features.
1055 if (defined($disabled{"deprecated"})) {
1056         $config{api} = $maxapi;
1057 }
1058
1059 if ($target{shared_target} eq "")
1060         {
1061         $no_shared_warn = 1
1062             if ((!$disabled{shared} || !$disabled{"dynamic-engine"})
1063                 && !$config{fips});
1064         $disabled{shared} = "no-shared-target";
1065         $disabled{pic} = $disabled{shared} = $disabled{"dynamic-engine"} =
1066             "no-shared-target";
1067         }
1068
1069 if ($disabled{"dynamic-engine"}) {
1070         push @{$config{defines}}, "OPENSSL_NO_DYNAMIC_ENGINE";
1071         $config{dynamic_engines} = 0;
1072 } else {
1073         push @{$config{defines}}, "OPENSSL_NO_STATIC_ENGINE";
1074         $config{dynamic_engines} = 1;
1075 }
1076
1077 unless ($disabled{"fuzz-libfuzzer"}) {
1078     $config{cflags} .= "-fsanitize-coverage=edge,indirect-calls ";
1079 }
1080
1081 unless ($disabled{asan}) {
1082     $config{cflags} .= "-fsanitize=address ";
1083 }
1084
1085 unless ($disabled{ubsan}) {
1086     # -DPEDANTIC or -fnosanitize=alignment may also be required on some
1087     # platforms.
1088     $config{cflags} .= "-fsanitize=undefined -fno-sanitize-recover=all ";
1089 }
1090
1091 unless ($disabled{msan}) {
1092   $config{cflags} .= "-fsanitize=memory ";
1093 }
1094
1095 unless ($disabled{"fuzz-libfuzzer"} && $disabled{"fuzz-afl"}
1096         && $disabled{asan} && $disabled{ubsan} && $disabled{msan}) {
1097     $config{cflags} .= "-fno-omit-frame-pointer -g ";
1098 }
1099 #
1100 # Platform fix-ups
1101 #
1102
1103 # This saves the build files from having to check
1104 if ($disabled{pic})
1105         {
1106         $target{shared_cflag} = $target{shared_ldflag} =
1107                 $target{shared_rcflag} = "";
1108         }
1109 else
1110         {
1111         push @{$config{defines}}, "OPENSSL_PIC";
1112         }
1113
1114 if ($target{sys_id} ne "")
1115         {
1116         push @{$config{openssl_sys_defines}}, "OPENSSL_SYS_$target{sys_id}";
1117         }
1118
1119 unless ($disabled{asm}) {
1120     $target{cpuid_asm_src}=$table{DEFAULTS}->{cpuid_asm_src} if ($config{processor} eq "386");
1121     $target{bn_asm_src} =~ s/\w+-gf2m.c// if (defined($disabled{ec2m}));
1122
1123     # bn-586 is the only one implementing bn_*_part_words
1124     push @{$config{defines}}, "OPENSSL_BN_ASM_PART_WORDS" if ($target{bn_asm_src} =~ /bn-586/);
1125     push @{$config{defines}}, "OPENSSL_IA32_SSE2" if (!$no_sse2 && $target{bn_asm_src} =~ /86/);
1126
1127     push @{$config{defines}}, "OPENSSL_BN_ASM_MONT" if ($target{bn_asm_src} =~ /-mont/);
1128     push @{$config{defines}}, "OPENSSL_BN_ASM_MONT5" if ($target{bn_asm_src} =~ /-mont5/);
1129     push @{$config{defines}}, "OPENSSL_BN_ASM_GF2m" if ($target{bn_asm_src} =~ /-gf2m/);
1130
1131     if ($config{fips}) {
1132         push @{$config{openssl_other_defines}}, "OPENSSL_FIPS";
1133     }
1134
1135     if ($target{sha1_asm_src}) {
1136         push @{$config{defines}}, "SHA1_ASM"   if ($target{sha1_asm_src} =~ /sx86/ || $target{sha1_asm_src} =~ /sha1/);
1137         push @{$config{defines}}, "SHA256_ASM" if ($target{sha1_asm_src} =~ /sha256/);
1138         push @{$config{defines}}, "SHA512_ASM" if ($target{sha1_asm_src} =~ /sha512/);
1139     }
1140     if ($target{rc4_asm_src} ne $table{DEFAULTS}->{rc4_asm_src}) {
1141         push @{$config{defines}}, "RC4_ASM";
1142     }
1143     if ($target{md5_asm_src}) {
1144         push @{$config{defines}}, "MD5_ASM";
1145     }
1146     $target{cast_asm_src}=$table{DEFAULTS}->{cast_asm_src} unless $disabled{pic}; # CAST assembler is not PIC
1147     if ($target{rmd160_asm_src}) {
1148         push @{$config{defines}}, "RMD160_ASM";
1149     }
1150     if ($target{aes_asm_src}) {
1151         push @{$config{defines}}, "AES_ASM" if ($target{aes_asm_src} =~ m/\baes-/);;
1152         # aes-ctr.fake is not a real file, only indication that assembler
1153         # module implements AES_ctr32_encrypt...
1154         push @{$config{defines}}, "AES_CTR_ASM" if ($target{aes_asm_src} =~ s/\s*aes-ctr\.fake//);
1155         # aes-xts.fake indicates presence of AES_xts_[en|de]crypt...
1156         push @{$config{defines}}, "AES_XTS_ASM" if ($target{aes_asm_src} =~ s/\s*aes-xts\.fake//);
1157         $target{aes_asm_src} =~ s/\s*(vpaes|aesni)-x86\.s//g if ($no_sse2);
1158         push @{$config{defines}}, "VPAES_ASM" if ($target{aes_asm_src} =~ m/vpaes/);
1159         push @{$config{defines}}, "BSAES_ASM" if ($target{aes_asm_src} =~ m/bsaes/);
1160     }
1161     if ($target{wp_asm_src} =~ /mmx/) {
1162         if ($config{processor} eq "386") {
1163             $target{wp_asm_src}=$table{DEFAULTS}->{wp_asm_src};
1164         } elsif (!$disabled{"whirlpool"}) {
1165             push @{$config{defines}}, "WHIRLPOOL_ASM";
1166         }
1167     }
1168     if ($target{modes_asm_src} =~ /ghash-/) {
1169         push @{$config{defines}}, "GHASH_ASM";
1170     }
1171     if ($target{ec_asm_src} =~ /ecp_nistz256/) {
1172         push @{$config{defines}}, "ECP_NISTZ256_ASM";
1173     }
1174     if ($target{poly1305_asm_src} ne "") {
1175         push @{$config{defines}}, "POLY1305_ASM";
1176     }
1177 }
1178
1179 my $ecc = $target{cc};
1180 if ($^O ne "VMS" && !$disabled{makedepend}) {
1181     # Is the compiler gcc or clang?  $ecc is used below to see if
1182     # error-checking can be turned on.
1183     my $ccpcc = "$config{cross_compile_prefix}$target{cc}";
1184     open(PIPE, "$ccpcc --version 2>&1 |");
1185     my $lines = 2;
1186     while ( <PIPE> ) {
1187         # Find the version number and save the major.
1188         m|(?:.*)\b(\d+)\.\d+\.\d+\b(?:.*)|;
1189         my $compiler_major = $1;
1190         # We know that GNU C version 3 and up as well as all clang
1191         # versions support dependency generation
1192         $config{makedepprog} = $ccpcc
1193             if (/clang/ || (/gcc/ && $compiler_major > 3));
1194         $ecc = "clang" if /clang/;
1195         $ecc = "gcc" if /gcc/;
1196         last if ($config{makedepprog} || !$lines--);
1197     }
1198     close(PIPE);
1199
1200     $config{makedepprog} = which('makedepend') unless $config{makedepprog};
1201     $disabled{makedepend} = "unavailable" unless $config{makedepprog};
1202 }
1203
1204
1205
1206 # Deal with bn_ops ###################################################
1207
1208 $config{bn_ll}                  =0;
1209 $config{export_var_as_fn}       =0;
1210 my $def_int="unsigned int";
1211 $config{rc4_int}                =$def_int;
1212 ($config{b64l},$config{b64},$config{b32})=(0,0,1);
1213
1214 my $count = 0;
1215 foreach (sort split(/\s+/,$target{bn_ops})) {
1216     $count++ if /SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT/;
1217     $config{export_var_as_fn}=1                 if $_ eq 'EXPORT_VAR_AS_FN';
1218     $config{bn_ll}=1                            if $_ eq 'BN_LLONG';
1219     $config{rc4_int}="unsigned char"            if $_ eq 'RC4_CHAR';
1220     ($config{b64l},$config{b64},$config{b32})
1221         =(0,1,0)                                if $_ eq 'SIXTY_FOUR_BIT';
1222     ($config{b64l},$config{b64},$config{b32})
1223         =(1,0,0)                                if $_ eq 'SIXTY_FOUR_BIT_LONG';
1224     ($config{b64l},$config{b64},$config{b32})
1225         =(0,0,1)                                if $_ eq 'THIRTY_TWO_BIT';
1226 }
1227 die "Exactly one of SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT can be set in bn_ops\n"
1228     if $count > 1;
1229
1230
1231 # Hack cflags for better warnings (dev option) #######################
1232
1233 # "Stringify" the C flags string.  This permits it to be made part of a string
1234 # and works as well on command lines.
1235 $config{cflags} =~ s/([\\\"])/\\$1/g;
1236
1237 if (defined($config{api})) {
1238     $config{openssl_api_defines} = [ "OPENSSL_MIN_API=".$apitable->{$config{api}} ];
1239     my $apiflag = sprintf("OPENSSL_API_COMPAT=%s", $apitable->{$config{api}});
1240     push @{$config{defines}}, $apiflag;
1241 }
1242
1243 if ($strict_warnings)
1244         {
1245         my $wopt;
1246         die "ERROR --strict-warnings requires gcc or clang"
1247             unless $ecc eq 'gcc' || $ecc eq 'clang';
1248         foreach $wopt (split /\s+/, $gcc_devteam_warn)
1249                 {
1250                 $config{cflags} .= " $wopt" unless ($config{cflags} =~ /(?:^|\s)$wopt(?:\s|$)/)
1251                 }
1252         if ($ecc eq "clang")
1253                 {
1254                 foreach $wopt (split /\s+/, $clang_devteam_warn)
1255                         {
1256                         $config{cflags} .= " $wopt" unless ($config{cflags} =~ /(?:^|\s)$wopt(?:\s|$)/)
1257                         }
1258                 }
1259         }
1260
1261 unless ($disabled{"crypto-mdebug-backtrace"})
1262         {
1263         foreach my $wopt (split /\s+/, $memleak_devteam_backtrace)
1264                 {
1265                 $config{cflags} .= " $wopt" unless ($config{cflags} =~ /(?:^|\s)$wopt(?:\s|$)/)
1266                 }
1267         if ($target =~ /^BSD-/)
1268                 {
1269                 $config{ex_libs} .= " -lexecinfo";
1270                 }
1271         }
1272
1273 if ($user_cflags ne "") { $config{cflags}="$config{cflags}$user_cflags"; }
1274 else                    { $no_user_cflags=1;  }
1275 if (@user_defines) { $config{defines}=[ @{$config{defines}}, @user_defines ]; }
1276 else               { $no_user_defines=1;    }
1277
1278 # ALL MODIFICATIONS TO %config and %target MUST BE DONE FROM HERE ON
1279
1280 unless ($disabled{afalgeng}) {
1281     $config{afalgeng}="";
1282     if ($target =~ m/^linux/) {
1283         my $minver = 4*10000 + 1*100 + 0;
1284         if ($config{cross_compile_prefix} eq "") {
1285             my $verstr = `uname -r`;
1286             my ($ma, $mi1, $mi2) = split("\\.", $verstr);
1287             ($mi2) = $mi2 =~ /(\d+)/;
1288             my $ver = $ma*10000 + $mi1*100 + $mi2;
1289             if ($ver < $minver) {
1290                 $disabled{afalgeng} = "too-old-kernel";
1291             } else {
1292                 push @{$config{engdirs}}, "afalg";
1293             }
1294         } else {
1295             $disabled{afalgeng} = "cross-compiling";
1296         }
1297     } else {
1298         $disabled{afalgeng}  = "not-linux";
1299     }
1300 }
1301
1302 push @{$config{openssl_other_defines}}, "OPENSSL_NO_AFALGENG" if ($disabled{afalgeng});
1303
1304 # If we use the unified build, collect information from build.info files
1305 my %unified_info = ();
1306
1307 my $buildinfo_debug = defined($ENV{CONFIGURE_DEBUG_BUILDINFO});
1308 if ($builder eq "unified") {
1309     # Store the name of the template file we will build the build file from
1310     # in %config.  This may be useful for the build file itself.
1311     my @build_file_template_names =
1312         ( $builder_platform."-".$target{build_file}.".tmpl",
1313           $target{build_file}.".tmpl" );
1314     my @build_file_templates = ();
1315
1316     # First, look in the user provided directory, if given
1317     if (defined $ENV{$local_config_envname}) {
1318         @build_file_templates =
1319             map {
1320                 if ($^O eq 'VMS') {
1321                     # VMS environment variables are logical names,
1322                     # which can be used as is
1323                     $local_config_envname . ':' . $_;
1324                 } else {
1325                     catfile($ENV{$local_config_envname}, $_);
1326                 }
1327             }
1328             @build_file_template_names;
1329     }
1330     # Then, look in our standard directory
1331     push @build_file_templates,
1332         ( map { catfile($srcdir, "Configurations", $_) }
1333           @build_file_template_names );
1334
1335     my $build_file_template;
1336     for $_ (@build_file_templates) {
1337         $build_file_template = $_;
1338         last if -f $build_file_template;
1339
1340         $build_file_template = undef;
1341     }
1342     if (!defined $build_file_template) {
1343         die "*** Couldn't find any of:\n", join("\n", @build_file_templates), "\n";
1344     }
1345     $config{build_file_template} = $build_file_template;
1346
1347     use lib catdir(dirname(__FILE__),"util");
1348     use with_fallback qw(Text::Template);
1349
1350     sub cleandir {
1351         my $base = shift;
1352         my $dir = shift;
1353         my $relativeto = shift || ".";
1354
1355         $dir = catdir($base,$dir) unless isabsolute($dir);
1356
1357         # Make sure the directories we're building in exists
1358         mkpath($dir);
1359
1360         my $res = abs2rel(absolutedir($dir), rel2abs($relativeto));
1361         #print STDERR "DEBUG[cleandir]: $dir , $base => $res\n";
1362         return $res;
1363     }
1364
1365     sub cleanfile {
1366         my $base = shift;
1367         my $file = shift;
1368         my $relativeto = shift || ".";
1369
1370         $file = catfile($base,$file) unless isabsolute($file);
1371
1372         my $d = dirname($file);
1373         my $f = basename($file);
1374
1375         # Make sure the directories we're building in exists
1376         mkpath($d);
1377
1378         my $res = abs2rel(catfile(absolutedir($d), $f), rel2abs($relativeto));
1379         #print STDERR "DEBUG[cleanfile]: $d , $f => $res\n";
1380         return $res;
1381     }
1382
1383     my @build_infos = ( [ ".", "build.info" ] );
1384     foreach (@{$config{dirs}}) {
1385         push @build_infos, [ $_, "build.info" ]
1386             if (-f catfile($srcdir, $_, "build.info"));
1387     }
1388     foreach (@{$config{sdirs}}) {
1389         push @build_infos, [ catdir("crypto", $_), "build.info" ]
1390             if (-f catfile($srcdir, "crypto", $_, "build.info"));
1391     }
1392     foreach (@{$config{engdirs}}) {
1393         push @build_infos, [ catdir("engines", $_), "build.info" ]
1394             if (-f catfile($srcdir, "engines", $_, "build.info"));
1395     }
1396
1397     $config{build_infos} = [ ];
1398
1399     foreach (@build_infos) {
1400         my $sourced = catdir($srcdir, $_->[0]);
1401         my $buildd = catdir($blddir, $_->[0]);
1402
1403         mkpath($buildd);
1404
1405         my $f = $_->[1];
1406         # The basic things we're trying to build
1407         my @programs = ();
1408         my @programs_install = ();
1409         my @libraries = ();
1410         my @libraries_install = ();
1411         my @engines = ();
1412         my @engines_install = ();
1413         my @scripts = ();
1414         my @scripts_install = ();
1415         my @extra = ();
1416         my @overrides = ();
1417         my @intermediates = ();
1418         my @rawlines = ();
1419
1420         my %ordinals = ();
1421         my %sources = ();
1422         my %shared_sources = ();
1423         my %includes = ();
1424         my %depends = ();
1425         my %renames = ();
1426         my %sharednames = ();
1427         my %generate = ();
1428
1429         push @{$config{build_infos}}, catfile(abs2rel($sourced, $blddir), $f);
1430         my $template = Text::Template->new(TYPE => 'FILE',
1431                                            SOURCE => catfile($sourced, $f));
1432         die "Something went wrong with $sourced/$f: $!\n" unless $template;
1433         my @text =
1434             split /^/m,
1435             $template->fill_in(HASH => { config => \%config,
1436                                          target => \%target,
1437                                          disabled => \%disabled,
1438                                          withargs => \%withargs,
1439                                          builddir => abs2rel($buildd, $blddir),
1440                                          sourcedir => abs2rel($sourced, $blddir),
1441                                          buildtop => abs2rel($blddir, $blddir),
1442                                          sourcetop => abs2rel($srcdir, $blddir) },
1443                                DELIMITERS => [ "{-", "-}" ]);
1444
1445         # The top item of this stack has the following values
1446         # -2 positive already run and we found ELSE (following ELSIF should fail)
1447         # -1 positive already run (skip until ENDIF)
1448         # 0 negatives so far (if we're at a condition, check it)
1449         # 1 last was positive (don't skip lines until next ELSE, ELSIF or ENDIF)
1450         # 2 positive ELSE (following ELSIF should fail)
1451         my @skip = ();
1452         collect_information(
1453             collect_from_array([ @text ],
1454                                qr/\\$/ => sub { my $l1 = shift; my $l2 = shift;
1455                                                 $l1 =~ s/\\$//; $l1.$l2 }),
1456             # Info we're looking for
1457             qr/^\s*IF\[((?:\\.|[^\\\]])*)\]\s*$/
1458             => sub {
1459                 if (! @skip || $skip[$#skip] > 0) {
1460                     push @skip, !! $1;
1461                 } else {
1462                     push @skip, -1;
1463                 }
1464             },
1465             qr/^\s*ELSIF\[((?:\\.|[^\\\]])*)\]\s*$/
1466             => sub { die "ELSIF out of scope" if ! @skip;
1467                      die "ELSIF following ELSE" if abs($skip[$#skip]) == 2;
1468                      $skip[$#skip] = -1 if $skip[$#skip] != 0;
1469                      $skip[$#skip] = !! $1
1470                          if $skip[$#skip] == 0; },
1471             qr/^\s*ELSE\s*$/
1472             => sub { die "ELSE out of scope" if ! @skip;
1473                      $skip[$#skip] = -2 if $skip[$#skip] != 0;
1474                      $skip[$#skip] = 2 if $skip[$#skip] == 0; },
1475             qr/^\s*ENDIF\s*$/
1476             => sub { die "ENDIF out of scope" if ! @skip;
1477                      pop @skip; },
1478             qr/^\s*PROGRAMS(_NO_INST)?\s*=\s*(.*)\s*$/
1479             => sub {
1480                 if (!@skip || $skip[$#skip] > 0) {
1481                     my $install = $1;
1482                     my @x = tokenize($2);
1483                     push @programs, @x;
1484                     push @programs_install, @x unless $install;
1485                 }
1486             },
1487             qr/^\s*LIBS(_NO_INST)?\s*=\s*(.*)\s*$/
1488             => sub {
1489                 if (!@skip || $skip[$#skip] > 0) {
1490                     my $install = $1;
1491                     my @x = tokenize($2);
1492                     push @libraries, @x;
1493                     push @libraries_install, @x unless $install;
1494                 }
1495             },
1496             qr/^\s*ENGINES(_NO_INST)?\s*=\s*(.*)\s*$/
1497             => sub {
1498                 if (!@skip || $skip[$#skip] > 0) {
1499                     my $install = $1;
1500                     my @x = tokenize($2);
1501                     push @engines, @x;
1502                     push @engines_install, @x unless $install;
1503                 }
1504             },
1505             qr/^\s*SCRIPTS(_NO_INST)?\s*=\s*(.*)\s*$/
1506             => sub {
1507                 if (!@skip || $skip[$#skip] > 0) {
1508                     my $install = $1;
1509                     my @x = tokenize($2);
1510                     push @scripts, @x;
1511                     push @scripts_install, @x unless $install;
1512                 }
1513             },
1514             qr/^\s*EXTRA\s*=\s*(.*)\s*$/
1515             => sub { push @extra, tokenize($1)
1516                          if !@skip || $skip[$#skip] > 0 },
1517             qr/^\s*OVERRIDES\s*=\s*(.*)\s*$/
1518             => sub { push @overrides, tokenize($1)
1519                          if !@skip || $skip[$#skip] > 0 },
1520
1521             qr/^\s*ORDINALS\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/,
1522             => sub { push @{$ordinals{$1}}, tokenize($2)
1523                          if !@skip || $skip[$#skip] > 0 },
1524             qr/^\s*SOURCE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1525             => sub { push @{$sources{$1}}, tokenize($2)
1526                          if !@skip || $skip[$#skip] > 0 },
1527             qr/^\s*SHARED_SOURCE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1528             => sub { push @{$shared_sources{$1}}, tokenize($2)
1529                          if !@skip || $skip[$#skip] > 0 },
1530             qr/^\s*INCLUDE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1531             => sub { push @{$includes{$1}}, tokenize($2)
1532                          if !@skip || $skip[$#skip] > 0 },
1533             qr/^\s*DEPEND\[((?:\\.|[^\\\]])*)\]\s*=\s*(.*)\s*$/
1534             => sub { push @{$depends{$1}}, tokenize($2)
1535                          if !@skip || $skip[$#skip] > 0 },
1536             qr/^\s*GENERATE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1537             => sub { push @{$generate{$1}}, $2
1538                          if !@skip || $skip[$#skip] > 0 },
1539             qr/^\s*RENAME\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1540             => sub { push @{$renames{$1}}, tokenize($2)
1541                          if !@skip || $skip[$#skip] > 0 },
1542             qr/^\s*SHARED_NAME\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1543             => sub { push @{$sharednames{$1}}, tokenize($2)
1544                          if !@skip || $skip[$#skip] > 0 },
1545             qr/^\s*BEGINRAW\[((?:\\.|[^\\\]])+)\]\s*$/
1546             => sub {
1547                 my $lineiterator = shift;
1548                 my $target_kind = $1;
1549                 while (defined $lineiterator->()) {
1550                     s|\R$||;
1551                     if (/^\s*ENDRAW\[((?:\\.|[^\\\]])+)\]\s*$/) {
1552                         die "ENDRAW doesn't match BEGINRAW"
1553                             if $1 ne $target_kind;
1554                         last;
1555                     }
1556                     next if @skip && $skip[$#skip] <= 0;
1557                     push @rawlines,  $_
1558                         if ($target_kind eq $target{build_file}
1559                             || $target_kind eq $target{build_file}."(".$builder_platform.")");
1560                 }
1561             },
1562             qr/^(?:#.*|\s*)$/ => sub { },
1563             "OTHERWISE" => sub { die "Something wrong with this line:\n$_\nat $sourced/$f" },
1564             "BEFORE" => sub {
1565                 if ($buildinfo_debug) {
1566                     print STDERR "DEBUG: Parsing ",join(" ", @_),"\n";
1567                     print STDERR "DEBUG: ... before parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
1568                 }
1569             },
1570             "AFTER" => sub {
1571                 if ($buildinfo_debug) {
1572                     print STDERR "DEBUG: .... after parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
1573                 }
1574             },
1575             );
1576         die "runaway IF?" if (@skip);
1577
1578         foreach (keys %renames) {
1579             die "$_ renamed to more than one thing: "
1580                 ,join(" ", @{$renames{$_}}),"\n"
1581                 if scalar @{$renames{$_}} > 1;
1582             my $dest = cleanfile($buildd, $_, $blddir);
1583             my $to = cleanfile($buildd, $renames{$_}->[0], $blddir);
1584             die "$dest renamed to more than one thing: "
1585                 ,$unified_info{rename}->{$dest}, $to
1586                 unless !defined($unified_info{rename}->{$dest})
1587                 or $unified_info{rename}->{$dest} eq $to;
1588             $unified_info{rename}->{$dest} = $to;
1589         }
1590
1591         foreach (@programs) {
1592             my $program = cleanfile($buildd, $_, $blddir);
1593             if ($unified_info{rename}->{$program}) {
1594                 $program = $unified_info{rename}->{$program};
1595             }
1596             $unified_info{programs}->{$program} = 1;
1597         }
1598
1599         foreach (@programs_install) {
1600             my $program = cleanfile($buildd, $_, $blddir);
1601             if ($unified_info{rename}->{$program}) {
1602                 $program = $unified_info{rename}->{$program};
1603             }
1604             $unified_info{install}->{programs}->{$program} = 1;
1605         }
1606
1607         foreach (@libraries) {
1608             my $library = cleanfile($buildd, $_, $blddir);
1609             if ($unified_info{rename}->{$library}) {
1610                 $library = $unified_info{rename}->{$library};
1611             }
1612             $unified_info{libraries}->{$library} = 1;
1613         }
1614
1615         foreach (@libraries_install) {
1616             my $library = cleanfile($buildd, $_, $blddir);
1617             if ($unified_info{rename}->{$library}) {
1618                 $library = $unified_info{rename}->{$library};
1619             }
1620             $unified_info{install}->{libraries}->{$library} = 1;
1621         }
1622
1623         die <<"EOF" if scalar @engines and !$config{dynamic_engines};
1624 ENGINES can only be used if configured with 'dynamic-engine'.
1625 This is usually a fault in a build.info file.
1626 EOF
1627         foreach (@engines) {
1628             my $library = cleanfile($buildd, $_, $blddir);
1629             if ($unified_info{rename}->{$library}) {
1630                 $library = $unified_info{rename}->{$library};
1631             }
1632             $unified_info{engines}->{$library} = 1;
1633         }
1634
1635         foreach (@engines_install) {
1636             my $library = cleanfile($buildd, $_, $blddir);
1637             if ($unified_info{rename}->{$library}) {
1638                 $library = $unified_info{rename}->{$library};
1639             }
1640             $unified_info{install}->{engines}->{$library} = 1;
1641         }
1642
1643         foreach (@scripts) {
1644             my $script = cleanfile($buildd, $_, $blddir);
1645             if ($unified_info{rename}->{$script}) {
1646                 $script = $unified_info{rename}->{$script};
1647             }
1648             $unified_info{scripts}->{$script} = 1;
1649         }
1650
1651         foreach (@scripts_install) {
1652             my $script = cleanfile($buildd, $_, $blddir);
1653             if ($unified_info{rename}->{$script}) {
1654                 $script = $unified_info{rename}->{$script};
1655             }
1656             $unified_info{install}->{scripts}->{$script} = 1;
1657         }
1658
1659         foreach (@extra) {
1660             my $extra = cleanfile($buildd, $_, $blddir);
1661             $unified_info{extra}->{$extra} = 1;
1662         }
1663
1664         foreach (@overrides) {
1665             my $override = cleanfile($buildd, $_, $blddir);
1666             $unified_info{overrides}->{$override} = 1;
1667         }
1668
1669         push @{$unified_info{rawlines}}, @rawlines;
1670
1671         unless ($disabled{shared}) {
1672             # Check sharednames.
1673             foreach (keys %sharednames) {
1674                 my $dest = cleanfile($buildd, $_, $blddir);
1675                 if ($unified_info{rename}->{$dest}) {
1676                     $dest = $unified_info{rename}->{$dest};
1677                 }
1678                 die "shared_name for $dest with multiple values: "
1679                     ,join(" ", @{$sharednames{$_}}),"\n"
1680                     if scalar @{$sharednames{$_}} > 1;
1681                 my $to = cleanfile($buildd, $sharednames{$_}->[0], $blddir);
1682                 die "shared_name found for a library $dest that isn't defined\n"
1683                     unless $unified_info{libraries}->{$dest};
1684                 die "shared_name for $dest with multiple values: "
1685                     ,$unified_info{sharednames}->{$dest}, ", ", $to
1686                     unless !defined($unified_info{sharednames}->{$dest})
1687                     or $unified_info{sharednames}->{$dest} eq $to;
1688                 $unified_info{sharednames}->{$dest} = $to;
1689             }
1690
1691             # Additionally, we set up sharednames for libraries that don't
1692             # have any, as themselves.
1693             foreach (keys %{$unified_info{libraries}}) {
1694                 if (!defined $unified_info{sharednames}->{$_}) {
1695                     $unified_info{sharednames}->{$_} = $_
1696                 }
1697             }
1698         }
1699
1700         foreach (keys %ordinals) {
1701             my $dest = $_;
1702             my $ddest = cleanfile($buildd, $_, $blddir);
1703             if ($unified_info{rename}->{$ddest}) {
1704                 $ddest = $unified_info{rename}->{$ddest};
1705             }
1706             foreach (@{$ordinals{$dest}}) {
1707                 my %known_ordinals =
1708                     (
1709                      crypto =>
1710                      cleanfile($sourced, catfile("util", "libcrypto.num"), $blddir),
1711                      ssl =>
1712                      cleanfile($sourced, catfile("util", "libssl.num"), $blddir)
1713                     );
1714                 my $o = $known_ordinals{$_};
1715                 die "Ordinals for $ddest defined more than once\n"
1716                     if $unified_info{ordinals}->{$ddest};
1717                 $unified_info{ordinals}->{$ddest} = [ $_, $o ];
1718             }
1719         }
1720
1721         foreach (keys %sources) {
1722             my $dest = $_;
1723             my $ddest = cleanfile($buildd, $_, $blddir);
1724             if ($unified_info{rename}->{$ddest}) {
1725                 $ddest = $unified_info{rename}->{$ddest};
1726             }
1727             foreach (@{$sources{$dest}}) {
1728                 my $s = cleanfile($sourced, $_, $blddir);
1729
1730                 # If it isn't in the source tree, we assume it's generated
1731                 # in the build tree
1732                 if (! -f $s) {
1733                     $s = cleanfile($buildd, $_, $blddir);
1734                 }
1735                 # We recognise C and asm files
1736                 if ($s =~ /\.[csS]\b$/) {
1737                     (my $o = $_) =~ s/\.[csS]\b$/.o/;
1738                     $o = cleanfile($buildd, $o, $blddir);
1739                     $unified_info{sources}->{$ddest}->{$o} = 1;
1740                     $unified_info{sources}->{$o}->{$s} = 1;
1741                 } else {
1742                     $unified_info{sources}->{$ddest}->{$s} = 1;
1743                 }
1744             }
1745         }
1746
1747         foreach (keys %shared_sources) {
1748             my $dest = $_;
1749             my $ddest = cleanfile($buildd, $_, $blddir);
1750             if ($unified_info{rename}->{$ddest}) {
1751                 $ddest = $unified_info{rename}->{$ddest};
1752             }
1753             foreach (@{$shared_sources{$dest}}) {
1754                 my $s = cleanfile($sourced, $_, $blddir);
1755
1756                 # If it isn't in the source tree, we assume it's generated
1757                 # in the build tree
1758                 if (! -f $s) {
1759                     $s = cleanfile($buildd, $_, $blddir);
1760                 }
1761                 # We recognise C and asm files
1762                 if ($s =~ /\.[csS]\b$/) {
1763                     (my $o = $_) =~ s/\.[csS]\b$/.o/;
1764                     $o = cleanfile($buildd, $o, $blddir);
1765                     $unified_info{shared_sources}->{$ddest}->{$o} = 1;
1766                     $unified_info{sources}->{$o}->{$s} = 1;
1767                 } else {
1768                     die "unrecognised source file type for shared library: $s\n";
1769                 }
1770             }
1771         }
1772
1773         foreach (keys %generate) {
1774             my $dest = $_;
1775             my $ddest = cleanfile($buildd, $_, $blddir);
1776             if ($unified_info{rename}->{$ddest}) {
1777                 $ddest = $unified_info{rename}->{$ddest};
1778             }
1779             die "more than one generator for $dest: "
1780                     ,join(" ", @{$generate{$_}}),"\n"
1781                     if scalar @{$generate{$_}} > 1;
1782             my @generator = split /\s+/, $generate{$dest}->[0];
1783             $generator[0] = cleanfile($sourced, $generator[0], $blddir),
1784             $unified_info{generate}->{$ddest} = [ @generator ];
1785         }
1786
1787         foreach (keys %depends) {
1788             my $dest = $_;
1789             my $ddest = $dest eq "" ? "" : cleanfile($sourced, $_, $blddir);
1790
1791             # If the destination doesn't exist in source, it can only be
1792             # a generated file in the build tree.
1793             if ($ddest ne "" && ! -f $ddest) {
1794                 $ddest = cleanfile($buildd, $_, $blddir);
1795                 if ($unified_info{rename}->{$ddest}) {
1796                     $ddest = $unified_info{rename}->{$ddest};
1797                 }
1798             }
1799             foreach (@{$depends{$dest}}) {
1800                 my $d = cleanfile($sourced, $_, $blddir);
1801
1802                 # If we know it's generated, or assume it is because we can't
1803                 # find it in the source tree, we set file we depend on to be
1804                 # in the build tree rather than the source tree, and assume
1805                 # and that there are lines to build it in a BEGINRAW..ENDRAW
1806                 # section or in the Makefile template.
1807                 if (! -f $d
1808                     || (grep { $d eq $_ }
1809                         map { cleanfile($srcdir, $_, $blddir) }
1810                         grep { /\.h$/ } keys %{$unified_info{generate}})) {
1811                     $d = cleanfile($buildd, $_, $blddir);
1812                 }
1813                 # Take note if the file to depend on is being renamed
1814                 if ($unified_info{rename}->{$d}) {
1815                     $d = $unified_info{rename}->{$d};
1816                 }
1817                 $unified_info{depends}->{$ddest}->{$d} = 1;
1818                 # If we depend on a header file or a perl module, let's make
1819                 # sure it can get included
1820                 if ($dest ne "" && $d =~ /\.(h|pm)$/) {
1821                     my $i = dirname($d);
1822                     push @{$unified_info{includes}->{$ddest}->{source}}, $i
1823                         unless grep { $_ eq $i } @{$unified_info{includes}->{$ddest}->{source}};
1824                 }
1825             }
1826         }
1827
1828         foreach (keys %includes) {
1829             my $dest = $_;
1830             my $ddest = cleanfile($sourced, $_, $blddir);
1831
1832             # If the destination doesn't exist in source, it can only be
1833             # a generated file in the build tree.
1834             if (! -f $ddest) {
1835                 $ddest = cleanfile($buildd, $_, $blddir);
1836                 if ($unified_info{rename}->{$ddest}) {
1837                     $ddest = $unified_info{rename}->{$ddest};
1838                 }
1839             }
1840             foreach (@{$includes{$dest}}) {
1841                 my $is = cleandir($sourced, $_, $blddir);
1842                 my $ib = cleandir($buildd, $_, $blddir);
1843                 push @{$unified_info{includes}->{$ddest}->{source}}, $is
1844                     unless grep { $_ eq $is } @{$unified_info{includes}->{$ddest}->{source}};
1845                 push @{$unified_info{includes}->{$ddest}->{build}}, $ib
1846                     unless grep { $_ eq $ib } @{$unified_info{includes}->{$ddest}->{build}};
1847             }
1848         }
1849     }
1850
1851     ### Make unified_info a bit more efficient
1852     # One level structures
1853     foreach (("programs", "libraries", "engines", "scripts", "extra", "overrides")) {
1854         $unified_info{$_} = [ sort keys %{$unified_info{$_}} ];
1855     }
1856     # Two level structures
1857     foreach my $l1 (("install", "sources", "shared_sources", "ldadd", "depends")) {
1858         foreach my $l2 (sort keys %{$unified_info{$l1}}) {
1859             $unified_info{$l1}->{$l2} =
1860                 [ sort keys %{$unified_info{$l1}->{$l2}} ];
1861         }
1862     }
1863     # Includes
1864     foreach my $dest (sort keys %{$unified_info{includes}}) {
1865         if (defined($unified_info{includes}->{$dest}->{build})) {
1866             my @source_includes =
1867                 ( @{$unified_info{includes}->{$dest}->{source}} );
1868             $unified_info{includes}->{$dest} =
1869                 [ @{$unified_info{includes}->{$dest}->{build}} ];
1870             foreach my $inc (@source_includes) {
1871                 push @{$unified_info{includes}->{$dest}}, $inc
1872                     unless grep { $_ eq $inc } @{$unified_info{includes}->{$dest}};
1873             }
1874         } else {
1875             $unified_info{includes}->{$dest} =
1876                 [ @{$unified_info{includes}->{$dest}->{source}} ];
1877         }
1878     }
1879 }
1880
1881 # For the schemes that need it, we provide the old *_obj configs
1882 # from the *_asm_obj ones
1883 foreach (grep /_(asm|aux)_src$/, keys %target) {
1884     my $src = $_;
1885     (my $obj = $_) =~ s/_(asm|aux)_src$/_obj/;
1886     ($target{$obj} = $target{$src}) =~ s/\.[csS]\b/.o/g;
1887 }
1888
1889 # Write down our configuration where it fits #########################
1890
1891 open(OUT,">configdata.pm") || die "unable to create configdata.pm: $!\n";
1892 print OUT <<"EOF";
1893 package configdata;
1894
1895 use strict;
1896 use warnings;
1897
1898 use Exporter;
1899 #use vars qw(\@ISA \@EXPORT);
1900 our \@ISA = qw(Exporter);
1901 our \@EXPORT = qw(\%config \%target \%disabled \%withargs \%unified_info \@disablables);
1902
1903 EOF
1904 print OUT "our %config = (\n";
1905 foreach (sort keys %config) {
1906     if (ref($config{$_}) eq "ARRAY") {
1907         print OUT "  ", $_, " => [ ", join(", ",
1908                                            map { quotify("perl", $_) }
1909                                            @{$config{$_}}), " ],\n";
1910     } else {
1911         print OUT "  ", $_, " => ", quotify("perl", $config{$_}), ",\n"
1912     }
1913 }
1914 print OUT <<"EOF";
1915 );
1916
1917 EOF
1918 print OUT "our %target = (\n";
1919 foreach (sort keys %target) {
1920     if (ref($target{$_}) eq "ARRAY") {
1921         print OUT "  ", $_, " => [ ", join(", ",
1922                                            map { quotify("perl", $_) }
1923                                            @{$target{$_}}), " ],\n";
1924     } else {
1925         print OUT "  ", $_, " => ", quotify("perl", $target{$_}), ",\n"
1926     }
1927 }
1928 print OUT <<"EOF";
1929 );
1930
1931 EOF
1932 print OUT "our \%available_protocols = (\n";
1933 print OUT "  tls => [ ", join(", ", map { quotify("perl", $_) } @tls), " ],\n";
1934 print OUT "  dtls => [ ", join(", ", map { quotify("perl", $_) } @dtls), " ],\n";
1935 print OUT <<"EOF";
1936 );
1937
1938 EOF
1939 print OUT "our \@disablables = (\n";
1940 foreach (@disablables) {
1941     print OUT "  ", quotify("perl", $_), ",\n";
1942 }
1943 print OUT <<"EOF";
1944 );
1945
1946 EOF
1947 print OUT "our \%disabled = (\n";
1948 foreach (sort keys %disabled) {
1949     print OUT "  ", quotify("perl", $_), " => ", quotify("perl", $disabled{$_}), ",\n";
1950 }
1951 print OUT <<"EOF";
1952 );
1953
1954 EOF
1955 print OUT "our %withargs = (\n";
1956 foreach (sort keys %withargs) {
1957     if (ref($withargs{$_}) eq "ARRAY") {
1958         print OUT "  ", $_, " => [ ", join(", ",
1959                                            map { quotify("perl", $_) }
1960                                            @{$withargs{$_}}), " ],\n";
1961     } else {
1962         print OUT "  ", $_, " => ", quotify("perl", $withargs{$_}), ",\n"
1963     }
1964 }
1965 print OUT <<"EOF";
1966 );
1967
1968 EOF
1969 if ($builder eq "unified") {
1970     my $recurse;
1971     $recurse = sub {
1972         my $indent = shift;
1973         foreach (@_) {
1974             if (ref $_ eq "ARRAY") {
1975                 print OUT " "x$indent, "[\n";
1976                 foreach (@$_) {
1977                     $recurse->($indent + 4, $_);
1978                 }
1979                 print OUT " "x$indent, "],\n";
1980             } elsif (ref $_ eq "HASH") {
1981                 my %h = %$_;
1982                 print OUT " "x$indent, "{\n";
1983                 foreach (sort keys %h) {
1984                     if (ref $h{$_} eq "") {
1985                         print OUT " "x($indent + 4), quotify("perl", $_), " => ", quotify("perl", $h{$_}), ",\n";
1986                     } else {
1987                         print OUT " "x($indent + 4), quotify("perl", $_), " =>\n";
1988                         $recurse->($indent + 8, $h{$_});
1989                     }
1990                 }
1991                 print OUT " "x$indent, "},\n";
1992             } else {
1993                 print OUT " "x$indent, quotify("perl", $_), ",\n";
1994             }
1995         }
1996     };
1997     print OUT "our %unified_info = (\n";
1998     foreach (sort keys %unified_info) {
1999         if (ref $unified_info{$_} eq "") {
2000             print OUT " "x4, quotify("perl", $_), " => ", quotify("perl", $unified_info{$_}), ",\n";
2001         } else {
2002             print OUT " "x4, quotify("perl", $_), " =>\n";
2003             $recurse->(8, $unified_info{$_});
2004         }
2005     }
2006     print OUT <<"EOF";
2007 );
2008
2009 EOF
2010 }
2011 print OUT "1;\n";
2012 close(OUT);
2013
2014
2015 print "CC            =$config{cross_compile_prefix}$target{cc}\n";
2016 print "CFLAG         =$target{cflags} $config{cflags}\n";
2017 print "SHARED_CFLAG  =$target{shared_cflag}\n";
2018 print "DEFINES       =",join(" ", @{$target{defines}}, @{$config{defines}}),"\n";
2019 print "LFLAG         =$target{lflags}\n";
2020 print "PLIB_LFLAG    =$target{plib_lflags}\n";
2021 print "EX_LIBS       =$target{ex_libs} $config{ex_libs}\n";
2022 print "APPS_OBJ      =$target{apps_obj}\n";
2023 print "CPUID_OBJ     =$target{cpuid_obj}\n";
2024 print "UPLINK_OBJ    =$target{uplink_obj}\n";
2025 print "BN_ASM        =$target{bn_obj}\n";
2026 print "EC_ASM        =$target{ec_obj}\n";
2027 print "DES_ENC       =$target{des_obj}\n";
2028 print "AES_ENC       =$target{aes_obj}\n";
2029 print "BF_ENC        =$target{bf_obj}\n";
2030 print "CAST_ENC      =$target{cast_obj}\n";
2031 print "RC4_ENC       =$target{rc4_obj}\n";
2032 print "RC5_ENC       =$target{rc5_obj}\n";
2033 print "MD5_OBJ_ASM   =$target{md5_obj}\n";
2034 print "SHA1_OBJ_ASM  =$target{sha1_obj}\n";
2035 print "RMD160_OBJ_ASM=$target{rmd160_obj}\n";
2036 print "CMLL_ENC      =$target{cmll_obj}\n";
2037 print "MODES_OBJ     =$target{modes_obj}\n";
2038 print "PADLOCK_OBJ   =$target{padlock_obj}\n";
2039 print "CHACHA_ENC    =$target{chacha_obj}\n";
2040 print "POLY1305_OBJ  =$target{poly1305_obj}\n";
2041 print "BLAKE2_OBJ    =$target{blake2_obj}\n";
2042 print "PROCESSOR     =$config{processor}\n";
2043 print "RANLIB        =", $target{ranlib} eq '$(CROSS_COMPILE)ranlib' ?
2044                              "$config{cross_compile_prefix}ranlib" :
2045                              "$target{ranlib}", "\n";
2046 print "ARFLAGS       =$target{arflags}\n";
2047 print "PERL          =$config{perl}\n";
2048 print "\n";
2049 print "SIXTY_FOUR_BIT_LONG mode\n" if $config{b64l};
2050 print "SIXTY_FOUR_BIT mode\n" if $config{b64};
2051 print "THIRTY_TWO_BIT mode\n" if $config{b32};
2052 print "BN_LLONG mode\n" if $config{bn_ll};
2053 print "RC4 uses $config{rc4_int}\n" if $config{rc4_int} ne $def_int;
2054
2055 my %builders = (
2056     unified => sub {
2057         run_dofile(catfile($blddir, $target{build_file}),
2058                    $config{build_file_template},
2059                    catfile($srcdir, "Configurations", "common.tmpl"));
2060     },
2061     );
2062
2063 $builders{$builder}->($builder_platform, @builder_opts);
2064
2065 print <<"EOF";
2066
2067 Configured for $target.
2068 EOF
2069
2070 print <<"EOF" if ($disabled{threads} eq "unavailable");
2071
2072 The library could not be configured for supporting multi-threaded
2073 applications as the compiler options required on this system are not known.
2074 See file INSTALL for details if you need multi-threading.
2075 EOF
2076
2077 print <<"EOF" if ($no_shared_warn);
2078
2079 The options 'shared', 'pic' and 'dynamic-engine' aren't supported on this
2080 platform, so we will pretend you gave the option 'no-pic', which also disables
2081 'shared' and 'dynamic-engine'.  If you know how to implement shared libraries
2082 or position independent code, please let us know (but please first make sure
2083 you have tried with a current version of OpenSSL).
2084 EOF
2085
2086 print <<"EOF" if (-f catfile($srcdir, "configdata.pm") && $srcdir ne $blddir);
2087
2088 WARNING: there are indications that another build was made in the source
2089 directory.  This build may have picked up artifacts from that build, the
2090 safest course of action is to clean the source directory and redo this
2091 configuration.
2092 EOF
2093
2094 exit(0);
2095
2096 ######################################################################
2097 #
2098 # Helpers and utility functions
2099 #
2100
2101 # Configuration file reading #########################################
2102
2103 # Note: All of the helper functions are for lazy evaluation.  They all
2104 # return a CODE ref, which will return the intended value when evaluated.
2105 # Thus, whenever there's mention of a returned value, it's about that
2106 # intended value.
2107
2108 # Helper function to implement conditional inheritance depending on the
2109 # value of $disabled{asm}.  Used in inherit_from values as follows:
2110 #
2111 #      inherit_from => [ "template", asm("asm_tmpl") ]
2112 #
2113 sub asm {
2114     my @x = @_;
2115     sub {
2116         $disabled{asm} ? () : @x;
2117     }
2118 }
2119
2120 # Helper function to implement conditional value variants, with a default
2121 # plus additional values based on the value of $config{build_type}.
2122 # Arguments are given in hash table form:
2123 #
2124 #       picker(default => "Basic string: ",
2125 #              debug   => "debug",
2126 #              release => "release")
2127 #
2128 # When configuring with --debug, the resulting string will be
2129 # "Basic string: debug", and when not, it will be "Basic string: release"
2130 #
2131 # This can be used to create variants of sets of flags according to the
2132 # build type:
2133 #
2134 #       cflags => picker(default => "-Wall",
2135 #                        debug   => "-g -O0",
2136 #                        release => "-O3")
2137 #
2138 sub picker {
2139     my %opts = @_;
2140     return sub { add($opts{default} || (),
2141                      $opts{$config{build_type}} || ())->(); }
2142 }
2143
2144 # Helper function to combine several values of different types into one.
2145 # This is useful if you want to combine a string with the result of a
2146 # lazy function, such as:
2147 #
2148 #       cflags => combine("-Wall", sub { $disabled{zlib} ? () : "-DZLIB" })
2149 #
2150 sub combine {
2151     my @stuff = @_;
2152     return sub { add(@stuff)->(); }
2153 }
2154
2155 # Helper function to implement conditional values depending on the value
2156 # of $disabled{threads}.  Can be used as follows:
2157 #
2158 #       cflags => combine("-Wall", threads("-pthread"))
2159 #
2160 sub threads {
2161     my @flags = @_;
2162     return sub { add($disabled{threads} ? () : @flags)->(); }
2163 }
2164
2165
2166
2167 our $add_called = 0;
2168 # Helper function to implement adding values to already existing configuration
2169 # values.  It handles elements that are ARRAYs, CODEs and scalars
2170 sub _add {
2171     my $separator = shift;
2172
2173     # If there's any ARRAY in the collection of values OR the separator
2174     # is undef, we will return an ARRAY of combined values, otherwise a
2175     # string of joined values with $separator as the separator.
2176     my $found_array = !defined($separator);
2177
2178     my @values =
2179         map {
2180             my $res = $_;
2181             while (ref($res) eq "CODE") {
2182                 $res = $res->();
2183             }
2184             if (defined($res)) {
2185                 if (ref($res) eq "ARRAY") {
2186                     $found_array = 1;
2187                     @$res;
2188                 } else {
2189                     $res;
2190                 }
2191             } else {
2192                 ();
2193             }
2194     } (@_);
2195
2196     $add_called = 1;
2197
2198     if ($found_array) {
2199         [ @values ];
2200     } else {
2201         join($separator, grep { defined($_) && $_ ne "" } @values);
2202     }
2203 }
2204 sub add_before {
2205     my $separator = " ";
2206     if (ref($_[$#_]) eq "HASH") {
2207         my $opts = pop;
2208         $separator = $opts->{separator};
2209     }
2210     my @x = @_;
2211     sub { _add($separator, @x, @_) };
2212 }
2213 sub add {
2214     my $separator = " ";
2215     if (ref($_[$#_]) eq "HASH") {
2216         my $opts = pop;
2217         $separator = $opts->{separator};
2218     }
2219     my @x = @_;
2220     sub { _add($separator, @_, @x) };
2221 }
2222
2223 # configuration reader, evaluates the input file as a perl script and expects
2224 # it to fill %targets with target configurations.  Those are then added to
2225 # %table.
2226 sub read_config {
2227     my $fname = shift;
2228     open(CONFFILE, "< $fname")
2229         or die "Can't open configuration file '$fname'!\n";
2230     my $x = $/;
2231     undef $/;
2232     my $content = <CONFFILE>;
2233     $/ = $x;
2234     close(CONFFILE);
2235     my %targets = ();
2236     {
2237         local %table = %::table;    # Protect %table from tampering
2238
2239         eval $content;
2240         warn $@ if $@;
2241     }
2242
2243     # For each target, check that it's configured with a hash table.
2244     foreach (keys %targets) {
2245         if (ref($targets{$_}) ne "HASH") {
2246             if (ref($targets{$_}) eq "") {
2247                 warn "Deprecated target configuration for $_, ignoring...\n";
2248             } else {
2249                 warn "Misconfigured target configuration for $_ (should be a hash table), ignoring...\n";
2250             }
2251             delete $targets{$_};
2252         }
2253     }
2254
2255     %table = (%table, %targets);
2256
2257 }
2258
2259 # configuration resolver.  Will only resolve all the lazy evaluation
2260 # codeblocks for the chosen target and all those it inherits from,
2261 # recursively
2262 sub resolve_config {
2263     my $target = shift;
2264     my @breadcrumbs = @_;
2265
2266 #    my $extra_checks = defined($ENV{CONFIGURE_EXTRA_CHECKS});
2267
2268     if (grep { $_ eq $target } @breadcrumbs) {
2269         die "inherit_from loop!  target backtrace:\n  "
2270             ,$target,"\n  ",join("\n  ", @breadcrumbs),"\n";
2271     }
2272
2273     if (!defined($table{$target})) {
2274         warn "Warning! target $target doesn't exist!\n";
2275         return ();
2276     }
2277     # Recurse through all inheritances.  They will be resolved on the
2278     # fly, so when this operation is done, they will all just be a
2279     # bunch of attributes with string values.
2280     # What we get here, though, are keys with references to lists of
2281     # the combined values of them all.  We will deal with lists after
2282     # this stage is done.
2283     my %combined_inheritance = ();
2284     if ($table{$target}->{inherit_from}) {
2285         my @inherit_from =
2286             map { ref($_) eq "CODE" ? $_->() : $_ } @{$table{$target}->{inherit_from}};
2287         foreach (@inherit_from) {
2288             my %inherited_config = resolve_config($_, $target, @breadcrumbs);
2289
2290             # 'template' is a marker that's considered private to
2291             # the config that had it.
2292             delete $inherited_config{template};
2293
2294             foreach (keys %inherited_config) {
2295                 if (!$combined_inheritance{$_}) {
2296                     $combined_inheritance{$_} = [];
2297                 }
2298                 push @{$combined_inheritance{$_}}, $inherited_config{$_};
2299             }
2300         }
2301     }
2302
2303     # We won't need inherit_from in this target any more, since we've
2304     # resolved all the inheritances that lead to this
2305     delete $table{$target}->{inherit_from};
2306
2307     # Now is the time to deal with those lists.  Here's the place to
2308     # decide what shall be done with those lists, all based on the
2309     # values of the target we're currently dealing with.
2310     # - If a value is a coderef, it will be executed with the list of
2311     #   inherited values as arguments.
2312     # - If the corresponding key doesn't have a value at all or is the
2313     #   empty string, the inherited value list will be run through the
2314     #   default combiner (below), and the result becomes this target's
2315     #   value.
2316     # - Otherwise, this target's value is assumed to be a string that
2317     #   will simply override the inherited list of values.
2318     my $default_combiner = add();
2319
2320     my %all_keys =
2321         map { $_ => 1 } (keys %combined_inheritance,
2322                          keys %{$table{$target}});
2323
2324     sub process_values {
2325         my $object    = shift;
2326         my $inherited = shift;  # Always a [ list ]
2327         my $target    = shift;
2328         my $entry     = shift;
2329
2330         $add_called = 0;
2331
2332         while(ref($object) eq "CODE") {
2333             $object = $object->(@$inherited);
2334         }
2335         if (!defined($object)) {
2336             return ();
2337         }
2338         elsif (ref($object) eq "ARRAY") {
2339             local $add_called;  # To make sure recursive calls don't affect it
2340             return [ map { process_values($_, $inherited, $target, $entry) }
2341                      @$object ];
2342         } elsif (ref($object) eq "") {
2343             return $object;
2344         } else {
2345             die "cannot handle reference type ",ref($object)
2346                 ," found in target ",$target," -> ",$entry,"\n";
2347         }
2348     }
2349
2350     foreach (sort keys %all_keys) {
2351         my $previous = $combined_inheritance{$_};
2352
2353         # Current target doesn't have a value for the current key?
2354         # Assign it the default combiner, the rest of this loop body
2355         # will handle it just like any other coderef.
2356         if (!exists $table{$target}->{$_}) {
2357             $table{$target}->{$_} = $default_combiner;
2358         }
2359
2360         $table{$target}->{$_} = process_values($table{$target}->{$_},
2361                                                $combined_inheritance{$_},
2362                                                $target, $_);
2363         unless(defined($table{$target}->{$_})) {
2364             delete $table{$target}->{$_};
2365         }
2366 #        if ($extra_checks &&
2367 #            $previous && !($add_called ||  $previous ~~ $table{$target}->{$_})) {
2368 #            warn "$_ got replaced in $target\n";
2369 #        }
2370     }
2371
2372     # Finally done, return the result.
2373     return %{$table{$target}};
2374 }
2375
2376 sub usage
2377         {
2378         print STDERR $usage;
2379         print STDERR "\npick os/compiler from:\n";
2380         my $j=0;
2381         my $i;
2382         my $k=0;
2383         foreach $i (sort keys %table)
2384                 {
2385                 next if $table{$i}->{template};
2386                 next if $i =~ /^debug/;
2387                 $k += length($i) + 1;
2388                 if ($k > 78)
2389                         {
2390                         print STDERR "\n";
2391                         $k=length($i);
2392                         }
2393                 print STDERR $i . " ";
2394                 }
2395         foreach $i (sort keys %table)
2396                 {
2397                 next if $table{$i}->{template};
2398                 next if $i !~ /^debug/;
2399                 $k += length($i) + 1;
2400                 if ($k > 78)
2401                         {
2402                         print STDERR "\n";
2403                         $k=length($i);
2404                         }
2405                 print STDERR $i . " ";
2406                 }
2407         print STDERR "\n\nNOTE: If in doubt, on Unix-ish systems use './config'.\n";
2408         exit(1);
2409         }
2410
2411 sub run_dofile
2412 {
2413     my $out = shift;
2414     my @templates = @_;
2415
2416     unlink $out || warn "Can't remove $out, $!"
2417         if -f $out;
2418     foreach (@templates) {
2419         die "Can't open $_, $!" unless -f $_;
2420     }
2421     my $perlcmd = (quotify("maybeshell", $config{perl}))[0];
2422     my $cmd = "$perlcmd \"-I.\" \"-Mconfigdata\" \"$dofile\" -o\"Configure\" \"".join("\" \"",@templates)."\" > \"$out.new\"";
2423     #print STDERR "DEBUG[run_dofile]: \$cmd = $cmd\n";
2424     system($cmd);
2425     exit 1 if $? != 0;
2426     rename("$out.new", $out) || die "Can't rename $out.new, $!";
2427 }
2428
2429 sub which
2430 {
2431     my ($name)=@_;
2432
2433     if (eval { require IPC::Cmd; 1; }) {
2434         IPC::Cmd->import();
2435         return scalar IPC::Cmd::can_run($name);
2436     } else {
2437         # if there is $directories component in splitpath,
2438         # then it's not something to test with $PATH...
2439         return $name if (File::Spec->splitpath($name))[1];
2440
2441         foreach (File::Spec->path()) {
2442             my $fullpath = catfile($_, "$name$target{exe_extension}");
2443             if (-f $fullpath and -x $fullpath) {
2444                 return $fullpath;
2445             }
2446         }
2447     }
2448 }
2449
2450 # Configuration printer ##############################################
2451
2452 sub print_table_entry
2453 {
2454     my $target = shift;
2455     my %target = resolve_config($target);
2456     my $type = shift;
2457
2458     # Don't print the templates
2459     return if $target{template};
2460
2461     my @sequence = (
2462         "sys_id",
2463         "cc",
2464         "cflags",
2465         "defines",
2466         "unistd",
2467         "ld",
2468         "lflags",
2469         "loutflag",
2470         "plib_lflags",
2471         "ex_libs",
2472         "bn_ops",
2473         "apps_aux_src",
2474         "cpuid_asm_src",
2475         "uplink_aux_src",
2476         "bn_asm_src",
2477         "ec_asm_src",
2478         "des_asm_src",
2479         "aes_asm_src",
2480         "bf_asm_src",
2481         "md5_asm_src",
2482         "cast_asm_src",
2483         "sha1_asm_src",
2484         "rc4_asm_src",
2485         "rmd160_asm_src",
2486         "rc5_asm_src",
2487         "wp_asm_src",
2488         "cmll_asm_src",
2489         "modes_asm_src",
2490         "padlock_asm_src",
2491         "chacha_asm_src",
2492         "poly1035_asm_src",
2493         "thread_scheme",
2494         "perlasm_scheme",
2495         "dso_scheme",
2496         "shared_target",
2497         "shared_cflag",
2498         "shared_defines",
2499         "shared_ldflag",
2500         "shared_rcflag",
2501         "shared_extension",
2502         "dso_extension",
2503         "obj_extension",
2504         "exe_extension",
2505         "ranlib",
2506         "ar",
2507         "arflags",
2508         "aroutflag",
2509         "rc",
2510         "rcflags",
2511         "rcoutflag",
2512         "mt",
2513         "mtflags",
2514         "mtinflag",
2515         "mtoutflag",
2516         "multilib",
2517         "build_scheme",
2518         );
2519
2520     if ($type eq "TABLE") {
2521         print "\n";
2522         print "*** $target\n";
2523         foreach (@sequence) {
2524             if (ref($target{$_}) eq "ARRAY") {
2525                 printf "\$%-12s = %s\n", $_, join(" ", @{$target{$_}});
2526             } else {
2527                 printf "\$%-12s = %s\n", $_, $target{$_};
2528             }
2529         }
2530     } elsif ($type eq "HASH") {
2531         my $largest =
2532             length((sort { length($a) <=> length($b) } @sequence)[-1]);
2533         print "    '$target' => {\n";
2534         foreach (@sequence) {
2535             if ($target{$_}) {
2536                 if (ref($target{$_}) eq "ARRAY") {
2537                     print "      '",$_,"'"," " x ($largest - length($_))," => [ ",join(", ", map { "'$_'" } @{$target{$_}})," ],\n";
2538                 } else {
2539                     print "      '",$_,"'"," " x ($largest - length($_))," => '",$target{$_},"',\n";
2540                 }
2541             }
2542         }
2543         print "    },\n";
2544     }
2545 }
2546
2547 # Utility routines ###################################################
2548
2549 # On VMS, if the given file is a logical name, File::Spec::Functions
2550 # will consider it an absolute path.  There are cases when we want a
2551 # purely syntactic check without checking the environment.
2552 sub isabsolute {
2553     my $file = shift;
2554
2555     # On non-platforms, we just use file_name_is_absolute().
2556     return file_name_is_absolute($file) unless $^O eq "VMS";
2557
2558     # If the file spec includes a device or a directpry spec,
2559     # file_name_is_absolute() is perfectly safe.
2560     return file_name_is_absolute($file) if $file =~ m|[:\[]|;
2561
2562     # Here, we know the given file spec isn't absolute
2563     return 0;
2564 }
2565
2566 # Makes a directory absolute and cleans out /../ in paths like foo/../bar
2567 # On some platforms, this uses rel2abs(), while on others, realpath() is used.
2568 # realpath() requires that at least all path components except the last is an
2569 # existing directory.  On VMS, the last component of the directory spec must
2570 # exist.
2571 sub absolutedir {
2572     my $dir = shift;
2573
2574     # realpath() is quite buggy on VMS.  It uses LIB$FID_TO_NAME, which
2575     # will return the volume name for the device, no matter what.  Also,
2576     # it will return an incorrect directory spec if the argument is a
2577     # directory that doesn't exist.
2578     if ($^O eq "VMS") {
2579         return rel2abs($dir);
2580     }
2581
2582     # We use realpath() on Unix, since no other will properly clean out
2583     # a directory spec.
2584     use Cwd qw/realpath/;
2585
2586     return realpath($dir);
2587 }
2588
2589 sub quotify {
2590     my %processors = (
2591         perl    => sub { my $x = shift;
2592                          $x =~ s/([\\\$\@"])/\\$1/g;
2593                          return '"'.$x.'"'; },
2594         maybeshell => sub { my $x = shift;
2595                             (my $y = $x) =~ s/([\\\"])/\\$1/g;
2596                             if ($x ne $y || $x =~ m|\s|) {
2597                                 return '"'.$y.'"';
2598                             } else {
2599                                 return $x;
2600                             }
2601                         },
2602         );
2603     my $for = shift;
2604     my $processor =
2605         defined($processors{$for}) ? $processors{$for} : sub { shift; };
2606
2607     return map { $processor->($_); } @_;
2608 }
2609
2610 # collect_from_file($filename, $line_concat_cond_re, $line_concat)
2611 # $filename is a file name to read from
2612 # $line_concat_cond_re is a regexp detecting a line continuation ending
2613 # $line_concat is a CODEref that takes care of concatenating two lines
2614 sub collect_from_file {
2615     my $filename = shift;
2616     my $line_concat_cond_re = shift;
2617     my $line_concat = shift;
2618
2619     open my $fh, $filename || die "unable to read $filename: $!\n";
2620     return sub {
2621         my $saved_line = "";
2622         $_ = "";
2623         while (<$fh>) {
2624             s|\R$||;
2625             if (defined $line_concat) {
2626                 $_ = $line_concat->($saved_line, $_);
2627                 $saved_line = "";
2628             }
2629             if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
2630                 $saved_line = $_;
2631                 next;
2632             }
2633             return $_;
2634         }
2635         die "$filename ending with continuation line\n" if $_;
2636         close $fh;
2637         return undef;
2638     }
2639 }
2640
2641 # collect_from_array($array, $line_concat_cond_re, $line_concat)
2642 # $array is an ARRAYref of lines
2643 # $line_concat_cond_re is a regexp detecting a line continuation ending
2644 # $line_concat is a CODEref that takes care of concatenating two lines
2645 sub collect_from_array {
2646     my $array = shift;
2647     my $line_concat_cond_re = shift;
2648     my $line_concat = shift;
2649     my @array = (@$array);
2650
2651     return sub {
2652         my $saved_line = "";
2653         $_ = "";
2654         while (defined($_ = shift @array)) {
2655             s|\R$||;
2656             if (defined $line_concat) {
2657                 $_ = $line_concat->($saved_line, $_);
2658                 $saved_line = "";
2659             }
2660             if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
2661                 $saved_line = $_;
2662                 next;
2663             }
2664             return $_;
2665         }
2666         die "input text ending with continuation line\n" if $_;
2667         return undef;
2668     }
2669 }
2670
2671 # collect_information($lineiterator, $line_continue, $regexp => $CODEref, ...)
2672 # $lineiterator is a CODEref that delivers one line at a time.
2673 # All following arguments are regex/CODEref pairs, where the regexp detects a
2674 # line and the CODEref does something with the result of the regexp.
2675 sub collect_information {
2676     my $lineiterator = shift;
2677     my %collectors = @_;
2678
2679     while(defined($_ = $lineiterator->())) {
2680         s|\R$||;
2681         my $found = 0;
2682         if ($collectors{"BEFORE"}) {
2683             $collectors{"BEFORE"}->($_);
2684         }
2685         foreach my $re (keys %collectors) {
2686             if ($re !~ /^OTHERWISE|BEFORE|AFTER$/ && /$re/) {
2687                 $collectors{$re}->($lineiterator);
2688                 $found = 1;
2689             };
2690         }
2691         if ($collectors{"OTHERWISE"}) {
2692             $collectors{"OTHERWISE"}->($lineiterator, $_)
2693                 unless $found || !defined $collectors{"OTHERWISE"};
2694         }
2695         if ($collectors{"AFTER"}) {
2696             $collectors{"AFTER"}->($_);
2697         }
2698     }
2699 }
2700
2701 # tokenize($line)
2702 # $line is a line of text to split up into tokens
2703 # returns a list of tokens
2704 #
2705 # Tokens are divided by spaces.  If the tokens include spaces, they
2706 # have to be quoted with single or double quotes.  Double quotes
2707 # inside a double quoted token must be escaped.  Escaping is done
2708 # with backslash.
2709 # Basically, the same quoting rules apply for " and ' as in any
2710 # Unix shell.
2711 sub tokenize {
2712     my $line = my $debug_line = shift;
2713     my @result = ();
2714
2715     while ($line =~ s|^\s+||, $line ne "") {
2716         my $token = "";
2717         while ($line ne "" && $line !~ m|^\s|) {
2718             if ($line =~ m/^"((?:[^"\\]+|\\.)*)"/) {
2719                 $token .= $1;
2720                 $line = $';
2721             } elsif ($line =~ m/^'([^']*)'/) {
2722                 $token .= $1;
2723                 $line = $';
2724             } elsif ($line =~ m/^(\S+)/) {
2725                 $token .= $1;
2726                 $line = $';
2727             }
2728         }
2729         push @result, $token;
2730     }
2731
2732     if ($ENV{CONFIGURE_DEBUG_TOKENIZE}) {
2733         print STDERR "DEBUG[tokenize]: Parsed '$debug_line' into:\n";
2734         print STDERR "DEBUG[tokenize]: ('", join("', '", @result), "')\n";
2735     }
2736     return @result;
2737 }