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