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