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