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