Configure: make disabling stuff easier and safer
[oweals/openssl.git] / Configure
1 #! /usr/bin/env perl
2 # -*- mode: perl; -*-
3 # Copyright 2016-2019 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 FindBin;
16 use lib "$FindBin::Bin/util/perl";
17 use File::Basename;
18 use File::Spec::Functions qw/:DEFAULT abs2rel rel2abs/;
19 use File::Path qw/mkpath/;
20 use OpenSSL::Glob;
21
22 # see INSTALL for instructions.
23
24 my $orig_death_handler = $SIG{__DIE__};
25 $SIG{__DIE__} = \&death_handler;
26
27 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-egd] [sctp] [386] [--prefix=DIR] [--openssldir=OPENSSLDIR] [--with-xxx[=vvv]] [--config=FILE] os/compiler[:flags]\n";
28
29 # Options:
30 #
31 # --config      add the given configuration file, which will be read after
32 #               any "Configurations*" files that are found in the same
33 #               directory as this script.
34 # --prefix      prefix for the OpenSSL installation, which includes the
35 #               directories bin, lib, include, share/man, share/doc/openssl
36 #               This becomes the value of INSTALLTOP in Makefile
37 #               (Default: /usr/local)
38 # --openssldir  OpenSSL data area, such as openssl.cnf, certificates and keys.
39 #               If it's a relative directory, it will be added on the directory
40 #               given with --prefix.
41 #               This becomes the value of OPENSSLDIR in Makefile and in C.
42 #               (Default: PREFIX/ssl)
43 #
44 # --cross-compile-prefix Add specified prefix to binutils components.
45 #
46 # --api         One of 0.9.8, 1.0.0 or 1.1.0.  Do not compile support for
47 #               interfaces deprecated as of the specified OpenSSL version.
48 #
49 # no-hw-xxx     do not compile support for specific crypto hardware.
50 #               Generic OpenSSL-style methods relating to this support
51 #               are always compiled but return NULL if the hardware
52 #               support isn't compiled.
53 # no-hw         do not compile support for any crypto hardware.
54 # [no-]threads  [don't] try to create a library that is suitable for
55 #               multithreaded applications (default is "threads" if we
56 #               know how to do it)
57 # [no-]shared   [don't] try to create shared libraries when supported.
58 # [no-]pic      [don't] try to build position independent code when supported.
59 #               If disabled, it also disables shared and dynamic-engine.
60 # no-asm        do not use assembler
61 # no-egd        do not compile support for the entropy-gathering daemon APIs
62 # [no-]zlib     [don't] compile support for zlib compression.
63 # zlib-dynamic  Like "zlib", but the zlib library is expected to be a shared
64 #               library and will be loaded in run-time by the OpenSSL library.
65 # sctp          include SCTP support
66 # enable-weak-ssl-ciphers
67 #               Enable weak ciphers that are disabled by default.
68 # 386           generate 80386 code in assembly modules
69 # no-sse2       disables IA-32 SSE2 code in assembly modules, the above
70 #               mentioned '386' option implies this one
71 # no-<cipher>   build without specified algorithm (rsa, idea, rc5, ...)
72 # -<xxx> +<xxx> compiler options are passed through
73 # -static       while -static is also a pass-through compiler option (and
74 #               as such is limited to environments where it's actually
75 #               meaningful), it triggers a number configuration options,
76 #               namely no-pic, no-shared and no-threads. It is
77 #               argued that the only reason to produce statically linked
78 #               binaries (and in context it means executables linked with
79 #               -static flag, and not just executables linked with static
80 #               libcrypto.a) is to eliminate dependency on specific run-time,
81 #               a.k.a. libc version. The mentioned config options are meant
82 #               to achieve just that. Unfortunately on Linux it's impossible
83 #               to eliminate the dependency completely for openssl executable
84 #               because of getaddrinfo and gethostbyname calls, which can
85 #               invoke dynamically loadable library facility anyway to meet
86 #               the lookup requests. For this reason on Linux statically
87 #               linked openssl executable has rather debugging value than
88 #               production quality.
89 #
90 # DEBUG_SAFESTACK use type-safe stacks to enforce type-safety on stack items
91 #               provided to stack calls. Generates unique stack functions for
92 #               each possible stack type.
93 # BN_LLONG      use the type 'long long' in crypto/bn/bn.h
94 # RC4_CHAR      use 'char' instead of 'int' for RC4_INT in crypto/rc4/rc4.h
95 # Following are set automatically by this script
96 #
97 # MD5_ASM       use some extra md5 assembler,
98 # SHA1_ASM      use some extra sha1 assembler, must define L_ENDIAN for x86
99 # RMD160_ASM    use some extra ripemd160 assembler,
100 # SHA256_ASM    sha256_block is implemented in assembler
101 # SHA512_ASM    sha512_block is implemented in assembler
102 # AES_ASM       AES_[en|de]crypt is implemented in assembler
103
104 # Minimum warning options... any contributions to OpenSSL should at least get
105 # past these.
106
107 # DEBUG_UNUSED enables __owur (warn unused result) checks.
108 # -DPEDANTIC complements -pedantic and is meant to mask code that
109 # is not strictly standard-compliant and/or implementation-specific,
110 # e.g. inline assembly, disregards to alignment requirements, such
111 # that -pedantic would complain about. Incidentally -DPEDANTIC has
112 # to be used even in sanitized builds, because sanitizer too is
113 # supposed to and does take notice of non-standard behaviour. Then
114 # -pedantic with pre-C9x compiler would also complain about 'long
115 # long' not being supported. As 64-bit algorithms are common now,
116 # it grew impossible to resolve this without sizeable additional
117 # code, so we just tell compiler to be pedantic about everything
118 # but 'long long' type.
119
120 my %gcc_devteam_warn = ();
121 {
122     my @common = qw( -DDEBUG_UNUSED
123                      -DPEDANTIC -pedantic -Wno-long-long
124                      -Wall
125                      -Wextra
126                      -Wno-unused-parameter
127                      -Wno-missing-field-initializers
128                      -Wswitch
129                      -Wsign-compare
130                      -Wshadow
131                      -Wformat
132                      -Wtype-limits
133                      -Wundef
134                      -Werror );
135     %gcc_devteam_warn = (
136         CFLAGS          => [ @common, qw( -Wmissing-prototypes
137                                           -Wstrict-prototypes ) ],
138         CXXFLAGS        => [ @common ]
139     );
140 }
141
142 # These are used in addition to $gcc_devteam_warn when the compiler is clang.
143 # TODO(openssl-team): fix problems and investigate if (at least) the
144 # following warnings can also be enabled:
145 #       -Wcast-align
146 #       -Wunreachable-code -- no, too ugly/compiler-specific
147 #       -Wlanguage-extension-token -- no, we use asm()
148 #       -Wunused-macros -- no, too tricky for BN and _XOPEN_SOURCE etc
149 #       -Wextended-offsetof -- no, needed in CMS ASN1 code
150 my %clang_devteam_warn = ();
151 {
152     my @common = qw( -Wswitch-default
153                      -Wno-parentheses-equality
154                      -Wno-language-extension-token
155                      -Wno-extended-offsetof
156                      -Wconditional-uninitialized
157                      -Wincompatible-pointer-types-discards-qualifiers
158                      -Wno-unknown-warning-option );
159     %clang_devteam_warn = (
160         CFLAGS          => [ @common, qw( -Wmissing-variable-declarations ) ],
161         CXXFLAGS        => [ @common ]
162     );
163 }
164
165 # This adds backtrace information to the memory leak info.  Is only used
166 # when crypto-mdebug-backtrace is enabled.
167 my $memleak_devteam_backtrace = "-rdynamic";
168
169 my $strict_warnings = 0;
170
171 # As for $BSDthreads. Idea is to maintain "collective" set of flags,
172 # which would cover all BSD flavors. -pthread applies to them all,
173 # but is treated differently. OpenBSD expands is as -D_POSIX_THREAD
174 # -lc_r, which is sufficient. FreeBSD 4.x expands it as -lc_r,
175 # which has to be accompanied by explicit -D_THREAD_SAFE and
176 # sometimes -D_REENTRANT. FreeBSD 5.x expands it as -lc_r, which
177 # seems to be sufficient?
178 our $BSDthreads="-pthread -D_THREAD_SAFE -D_REENTRANT";
179
180 #
181 # API compatibility name to version number mapping.
182 #
183 my $maxapi = "1.1.0";           # API for "no-deprecated" builds
184 my $apitable = {
185     "1.1.0" => "0x10100000L",
186     "1.0.0" => "0x10000000L",
187     "0.9.8" => "0x00908000L",
188 };
189
190 our %table = ();
191 our %config = ();
192 our %withargs = ();
193 our $now_printing;      # set to current entry's name in print_table_entry
194                         # (todo: right thing would be to encapsulate name
195                         # into %target [class] and make print_table_entry
196                         # a method)
197
198 # Forward declarations ###############################################
199
200 # read_config(filename)
201 #
202 # Reads a configuration file and populates %table with the contents
203 # (which the configuration file places in %targets).
204 sub read_config;
205
206 # resolve_config(target)
207 #
208 # Resolves all the late evaluations, inheritances and so on for the
209 # chosen target and any target it inherits from.
210 sub resolve_config;
211
212
213 # Information collection #############################################
214
215 # Unified build supports separate build dir
216 my $srcdir = catdir(absolutedir(dirname($0))); # catdir ensures local syntax
217 my $blddir = catdir(absolutedir("."));         # catdir ensures local syntax
218 my $dofile = abs2rel(catfile($srcdir, "util/dofile.pl"));
219
220 my $local_config_envname = 'OPENSSL_LOCAL_CONFIG_DIR';
221
222 $config{sourcedir} = abs2rel($srcdir);
223 $config{builddir} = abs2rel($blddir);
224
225 # Collect reconfiguration information if needed
226 my @argvcopy=@ARGV;
227
228 if (grep /^reconf(igure)?$/, @argvcopy) {
229     die "reconfiguring with other arguments present isn't supported"
230         if scalar @argvcopy > 1;
231     if (-f "./configdata.pm") {
232         my $file = "./configdata.pm";
233         unless (my $return = do $file) {
234             die "couldn't parse $file: $@" if $@;
235             die "couldn't do $file: $!"    unless defined $return;
236             die "couldn't run $file"       unless $return;
237         }
238
239         @argvcopy = defined($configdata::config{perlargv}) ?
240             @{$configdata::config{perlargv}} : ();
241         die "Incorrect data to reconfigure, please do a normal configuration\n"
242             if (grep(/^reconf/,@argvcopy));
243         $config{perlenv} = $configdata::config{perlenv} // {};
244     } else {
245         die "Insufficient data to reconfigure, please do a normal configuration\n";
246     }
247 }
248
249 $config{perlargv} = [ @argvcopy ];
250
251 # Collect version numbers
252 $config{version} = "unknown";
253 $config{version_num} = "unknown";
254 $config{shlib_version_number} = "unknown";
255 $config{shlib_version_history} = "unknown";
256
257 collect_information(
258     collect_from_file(catfile($srcdir,'include/openssl/opensslv.h')),
259     qr/OPENSSL.VERSION.TEXT.*OpenSSL (\S+) / => sub { $config{version} = $1; },
260     qr/OPENSSL.VERSION.NUMBER.*(0x\S+)/      => sub { $config{version_num}=$1 },
261     qr/SHLIB_VERSION_NUMBER *"([^"]+)"/      => sub { $config{shlib_version_number}=$1 },
262     qr/SHLIB_VERSION_HISTORY *"([^"]*)"/     => sub { $config{shlib_version_history}=$1 }
263     );
264 if ($config{shlib_version_history} ne "") { $config{shlib_version_history} .= ":"; }
265
266 ($config{major}, $config{minor})
267     = ($config{version} =~ /^([0-9]+)\.([0-9\.]+)/);
268 ($config{shlib_major}, $config{shlib_minor})
269     = ($config{shlib_version_number} =~ /^([0-9]+)\.([0-9\.]+)/);
270 die "erroneous version information in opensslv.h: ",
271     "$config{major}, $config{minor}, $config{shlib_major}, $config{shlib_minor}\n"
272     if ($config{major} eq "" || $config{minor} eq ""
273         || $config{shlib_major} eq "" ||  $config{shlib_minor} eq "");
274
275 # Collect target configurations
276
277 my $pattern = catfile(dirname($0), "Configurations", "*.conf");
278 foreach (sort glob($pattern)) {
279     &read_config($_);
280 }
281
282 if (defined env($local_config_envname)) {
283     if ($^O eq 'VMS') {
284         # VMS environment variables are logical names,
285         # which can be used as is
286         $pattern = $local_config_envname . ':' . '*.conf';
287     } else {
288         $pattern = catfile(env($local_config_envname), '*.conf');
289     }
290
291     foreach (sort glob($pattern)) {
292         &read_config($_);
293     }
294 }
295
296 # Save away perl command information
297 $config{perl_cmd} = $^X;
298 $config{perl_version} = $Config{version};
299 $config{perl_archname} = $Config{archname};
300
301 $config{prefix}="";
302 $config{openssldir}="";
303 $config{processor}="";
304 $config{libdir}="";
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", "sm3",
314     "des", "aes", "rc2", "rc4", "rc5", "idea", "aria", "bf", "cast", "camellia", "seed", "sm4", "chacha", "modes",
315     "bn", "ec", "rsa", "dsa", "dh", "sm2", "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", "store"
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     "autoload-config",
340     "bf",
341     "blake2",
342     "camellia",
343     "capieng",
344     "cast",
345     "chacha",
346     "cmac",
347     "cms",
348     "comp",
349     "crypto-mdebug",
350     "crypto-mdebug-backtrace",
351     "ct",
352     "deprecated",
353     "des",
354     "devcryptoeng",
355     "dgram",
356     "dh",
357     "dsa",
358     "dtls",
359     "dynamic-engine",
360     "ec",
361     "ec2m",
362     "ecdh",
363     "ecdsa",
364     "ec_nistp_64_gcc_128",
365     "egd",
366     "engine",
367     "err",
368     "external-tests",
369     "filenames",
370     "fuzz-libfuzzer",
371     "fuzz-afl",
372     "gost",
373     "heartbeats",
374     "hw(-.+)?",
375     "idea",
376     "makedepend",
377     "md2",
378     "md4",
379     "mdc2",
380     "msan",
381     "multiblock",
382     "nextprotoneg",
383     "pinshared",
384     "ocb",
385     "ocsp",
386     "pic",
387     "poly1305",
388     "posix-io",
389     "psk",
390     "rc2",
391     "rc4",
392     "rc5",
393     "rdrand",
394     "rfc3779",
395     "rmd160",
396     "scrypt",
397     "sctp",
398     "seed",
399     "shared",
400     "siphash",
401     "sm2",
402     "sm3",
403     "sm4",
404     "sock",
405     "srp",
406     "srtp",
407     "sse2",
408     "ssl",
409     "ssl-trace",
410     "static-engine",
411     "stdio",
412     "tests",
413     "threads",
414     "tls",
415     "ts",
416     "ubsan",
417     "ui-console",
418     "unit-test",
419     "whirlpool",
420     "weak-ssl-ciphers",
421     "zlib",
422     "zlib-dynamic",
423     );
424 foreach my $proto ((@tls, @dtls))
425         {
426         push(@disablables, $proto);
427         push(@disablables, "$proto-method") unless $proto eq "tls1_3";
428         }
429
430 my %deprecated_disablables = (
431     "ssl2" => undef,
432     "buf-freelists" => undef,
433     "ripemd" => "rmd160",
434     "ui" => "ui-console",
435     "dso" => "",                # Empty string means we're silent about it
436     );
437
438 # All of the following are disabled by default:
439
440 our %disabled = ( # "what"         => "comment"
441                   "asan"                => "default",
442                   "crypto-mdebug"       => "default",
443                   "crypto-mdebug-backtrace" => "default",
444                   "devcryptoeng"        => "default",
445                   "ec_nistp_64_gcc_128" => "default",
446                   "egd"                 => "default",
447                   "external-tests"      => "default",
448                   "fuzz-libfuzzer"      => "default",
449                   "fuzz-afl"            => "default",
450                   "heartbeats"          => "default",
451                   "md2"                 => "default",
452                   "msan"                => "default",
453                   "rc5"                 => "default",
454                   "sctp"                => "default",
455                   "ssl-trace"           => "default",
456                   "ssl3"                => "default",
457                   "ssl3-method"         => "default",
458                   "ubsan"               => "default",
459                   "unit-test"           => "default",
460                   "weak-ssl-ciphers"    => "default",
461                   "zlib"                => "default",
462                   "zlib-dynamic"        => "default",
463                 );
464
465 # Note: => pair form used for aesthetics, not to truly make a hash table
466 my @disable_cascades = (
467     # "what"            => [ "cascade", ... ]
468     sub { $config{processor} eq "386" }
469                         => [ "sse2" ],
470     "ssl"               => [ "ssl3" ],
471     "ssl3-method"       => [ "ssl3" ],
472     "zlib"              => [ "zlib-dynamic" ],
473     "des"               => [ "mdc2" ],
474     "ec"                => [ "ecdsa", "ecdh" ],
475
476     "dgram"             => [ "dtls", "sctp" ],
477     "sock"              => [ "dgram" ],
478     "dtls"              => [ @dtls ],
479     sub { 0 == scalar grep { !$disabled{$_} } @dtls }
480                         => [ "dtls" ],
481
482     "tls"               => [ @tls ],
483     sub { 0 == scalar grep { !$disabled{$_} } @tls }
484                         => [ "tls" ],
485
486     "crypto-mdebug"     => [ "crypto-mdebug-backtrace" ],
487
488     # Without position independent code, there can be no shared libraries or DSOs
489     "pic"               => [ "shared" ],
490     "shared"            => [ "dynamic-engine" ],
491     "engine"            => [ "afalgeng", "devcryptoeng" ],
492
493     # no-autoalginit is only useful when building non-shared
494     "autoalginit"       => [ "shared", "apps" ],
495
496     "stdio"             => [ "apps", "capieng", "egd" ],
497     "apps"              => [ "tests" ],
498     "tests"             => [ "external-tests" ],
499     "comp"              => [ "zlib" ],
500     "ec"                => [ "tls1_3", "sm2" ],
501     "sm3"               => [ "sm2" ],
502     sub { !$disabled{"unit-test"} } => [ "heartbeats" ],
503
504     sub { !$disabled{"msan"} } => [ "asm" ],
505     );
506
507 # Avoid protocol support holes.  Also disable all versions below N, if version
508 # N is disabled while N+1 is enabled.
509 #
510 my @list = (reverse @tls);
511 while ((my $first, my $second) = (shift @list, shift @list)) {
512     last unless @list;
513     push @disable_cascades, ( sub { !$disabled{$first} && $disabled{$second} }
514                               => [ @list ] );
515     unshift @list, $second;
516 }
517 my @list = (reverse @dtls);
518 while ((my $first, my $second) = (shift @list, shift @list)) {
519     last unless @list;
520     push @disable_cascades, ( sub { !$disabled{$first} && $disabled{$second} }
521                               => [ @list ] );
522     unshift @list, $second;
523 }
524
525 # Explicit "no-..." options will be collected in %disabled along with the defaults.
526 # To remove something from %disabled, use "enable-foo".
527 # For symmetry, "disable-foo" is a synonym for "no-foo".
528
529 &usage if ($#ARGV < 0);
530
531 # For the "make variables" CINCLUDES and CDEFINES, we support lists with
532 # platform specific list separators.  Users from those platforms should
533 # recognise those separators from how you set up the PATH to find executables.
534 # The default is the Unix like separator, :, but as an exception, we also
535 # support the space as separator.
536 my $list_separator_re =
537     { VMS           => qr/(?<!\^),/,
538       MSWin32       => qr/(?<!\\);/ } -> {$^O} // qr/(?<!\\)[:\s]/;
539 # All the "make variables" we support
540 # Some get pre-populated for the sake of backward compatibility
541 # (we supported those before the change to "make variable" support.
542 my %user = (
543     AR          => env('AR'),
544     ARFLAGS     => [],
545     AS          => undef,
546     ASFLAGS     => [],
547     CC          => env('CC'),
548     CFLAGS      => [ env('CFLAGS') || () ],
549     CXX         => env('CXX'),
550     CXXFLAGS    => [ env('CXXFLAGS') || () ],
551     CPP         => undef,
552     CPPFLAGS    => [ env('CPPFLAGS') || () ],  # -D, -I, -Wp,
553     CPPDEFINES  => [],  # Alternative for -D
554     CPPINCLUDES => [],  # Alternative for -I
555     CROSS_COMPILE => env('CROSS_COMPILE'),
556     HASHBANGPERL=> env('HASHBANGPERL') || env('PERL'),
557     LD          => undef,
558     LDFLAGS     => [ env('LDFLAGS') || () ],  # -L, -Wl,
559     LDLIBS      => [ env('LDLIBS') || () ],  # -l
560     MT          => undef,
561     MTFLAGS     => [],
562     PERL        => env('PERL') || ($^O ne "VMS" ? $^X : "perl"),
563     RANLIB      => env('RANLIB'),
564     RC          => env('RC') || env('WINDRES'),
565     RCFLAGS     => [],
566     RM          => undef,
567    );
568 # Info about what "make variables" may be prefixed with the cross compiler
569 # prefix.  This should NEVER mention any such variable with a list for value.
570 my @user_crossable = qw ( AR AS CC CXX CPP LD MT RANLIB RC );
571 # The same but for flags given as Configure options.  These are *additional*
572 # input, as opposed to the VAR=string option that override the corresponding
573 # config target attributes
574 my %useradd = (
575     CPPDEFINES  => [],
576     CPPINCLUDES => [],
577     CPPFLAGS    => [],
578     CFLAGS      => [],
579     CXXFLAGS    => [],
580     LDFLAGS     => [],
581     LDLIBS      => [],
582    );
583
584 my %user_synonyms = (
585     HASHBANGPERL=> 'PERL',
586     RC          => 'WINDRES',
587    );
588
589 # Some target attributes have been renamed, this is the translation table
590 my %target_attr_translate =(
591     ar          => 'AR',
592     as          => 'AS',
593     cc          => 'CC',
594     cxx         => 'CXX',
595     cpp         => 'CPP',
596     hashbangperl => 'HASHBANGPERL',
597     ld          => 'LD',
598     mt          => 'MT',
599     ranlib      => 'RANLIB',
600     rc          => 'RC',
601     rm          => 'RM',
602    );
603
604 # Initialisers coming from 'config' scripts
605 $config{defines} = [ split(/$list_separator_re/, env('__CNF_CPPDEFINES')) ];
606 $config{includes} = [ split(/$list_separator_re/, env('__CNF_CPPINCLUDES')) ];
607 $config{cppflags} = [ env('__CNF_CPPFLAGS') || () ];
608 $config{cflags} = [ env('__CNF_CFLAGS') || () ];
609 $config{cxxflags} = [ env('__CNF_CXXFLAGS') || () ];
610 $config{lflags} = [ env('__CNF_LDFLAGS') || () ];
611 $config{ex_libs} = [ env('__CNF_LDLIBS') || () ];
612
613 $config{openssl_api_defines}=[];
614 $config{openssl_algorithm_defines}=[];
615 $config{openssl_thread_defines}=[];
616 $config{openssl_sys_defines}=[];
617 $config{openssl_other_defines}=[];
618 $config{options}="";
619 $config{build_type} = "release";
620 my $target="";
621
622 my %cmdvars = ();               # Stores FOO='blah' type arguments
623 my %unsupported_options = ();
624 my %deprecated_options = ();
625 # If you change this, update apps/version.c
626 my @known_seed_sources = qw(getrandom devrandom os egd none rdcpu librandom);
627 my @seed_sources = ();
628 while (@argvcopy)
629         {
630         $_ = shift @argvcopy;
631
632         # Support env variable assignments among the options
633         if (m|^(\w+)=(.+)?$|)
634                 {
635                 $cmdvars{$1} = $2;
636                 # Every time a variable is given as a configuration argument,
637                 # it acts as a reset if the variable.
638                 if (exists $user{$1})
639                         {
640                         $user{$1} = ref $user{$1} eq "ARRAY" ? [] : undef;
641                         }
642                 #if (exists $useradd{$1})
643                 #       {
644                 #       $useradd{$1} = [];
645                 #       }
646                 next;
647                 }
648
649         # VMS is a case insensitive environment, and depending on settings
650         # out of our control, we may receive options uppercased.  Let's
651         # downcase at least the part before any equal sign.
652         if ($^O eq "VMS")
653                 {
654                 s/^([^=]*)/lc($1)/e;
655                 }
656
657         # some people just can't read the instructions, clang people have to...
658         s/^-no-(?!integrated-as)/no-/;
659
660         # rewrite some options in "enable-..." form
661         s /^-?-?shared$/enable-shared/;
662         s /^sctp$/enable-sctp/;
663         s /^threads$/enable-threads/;
664         s /^zlib$/enable-zlib/;
665         s /^zlib-dynamic$/enable-zlib-dynamic/;
666
667         if (/^(no|disable|enable)-(.+)$/)
668                 {
669                 my $word = $2;
670                 if (!exists $deprecated_disablables{$word}
671                         && !grep { $word =~ /^${_}$/ } @disablables)
672                         {
673                         $unsupported_options{$_} = 1;
674                         next;
675                         }
676                 }
677         if (/^no-(.+)$/ || /^disable-(.+)$/)
678                 {
679                 foreach my $proto ((@tls, @dtls))
680                         {
681                         if ($1 eq "$proto-method")
682                                 {
683                                 $disabled{"$proto"} = "option($proto-method)";
684                                 last;
685                                 }
686                         }
687                 if ($1 eq "dtls")
688                         {
689                         foreach my $proto (@dtls)
690                                 {
691                                 $disabled{$proto} = "option(dtls)";
692                                 }
693                         $disabled{"dtls"} = "option(dtls)";
694                         }
695                 elsif ($1 eq "ssl")
696                         {
697                         # Last one of its kind
698                         $disabled{"ssl3"} = "option(ssl)";
699                         }
700                 elsif ($1 eq "tls")
701                         {
702                         # XXX: Tests will fail if all SSL/TLS
703                         # protocols are disabled.
704                         foreach my $proto (@tls)
705                                 {
706                                 $disabled{$proto} = "option(tls)";
707                                 }
708                         }
709                 elsif ($1 eq "static-engine")
710                         {
711                         delete $disabled{"dynamic-engine"};
712                         }
713                 elsif ($1 eq "dynamic-engine")
714                         {
715                         $disabled{"dynamic-engine"} = "option";
716                         }
717                 elsif (exists $deprecated_disablables{$1})
718                         {
719                         if ($deprecated_disablables{$1} ne "")
720                                 {
721                                 $deprecated_options{$_} = 1;
722                                 if (defined $deprecated_disablables{$1})
723                                         {
724                                         $disabled{$deprecated_disablables{$1}} = "option";
725                                         }
726                                 }
727                         }
728                 else
729                         {
730                         $disabled{$1} = "option";
731                         }
732                 # No longer an automatic choice
733                 $auto_threads = 0 if ($1 eq "threads");
734                 }
735         elsif (/^enable-(.+)$/)
736                 {
737                 if ($1 eq "static-engine")
738                         {
739                         $disabled{"dynamic-engine"} = "option";
740                         }
741                 elsif ($1 eq "dynamic-engine")
742                         {
743                         delete $disabled{"dynamic-engine"};
744                         }
745                 elsif ($1 eq "zlib-dynamic")
746                         {
747                         delete $disabled{"zlib"};
748                         }
749                 my $algo = $1;
750                 delete $disabled{$algo};
751
752                 # No longer an automatic choice
753                 $auto_threads = 0 if ($1 eq "threads");
754                 }
755         elsif (/^--strict-warnings$/)
756                 {
757                 # Pretend that our strict flags is a C flag, and replace it
758                 # with the proper flags later on
759                 push @{$useradd{CFLAGS}}, '--ossl-strict-warnings';
760                 push @{$useradd{CXXFLAGS}}, '--ossl-strict-warnings';
761                 $strict_warnings=1;
762                 }
763         elsif (/^--debug$/)
764                 {
765                 $config{build_type} = "debug";
766                 }
767         elsif (/^--release$/)
768                 {
769                 $config{build_type} = "release";
770                 }
771         elsif (/^386$/)
772                 { $config{processor}=386; }
773         elsif (/^fips$/)
774                 {
775                 die "FIPS mode not supported\n";
776                 }
777         elsif (/^rsaref$/)
778                 {
779                 # No RSAref support any more since it's not needed.
780                 # The check for the option is there so scripts aren't
781                 # broken
782                 }
783         elsif (/^nofipscanistercheck$/)
784                 {
785                 die "FIPS mode not supported\n";
786                 }
787         elsif (/^[-+]/)
788                 {
789                 if (/^--prefix=(.*)$/)
790                         {
791                         $config{prefix}=$1;
792                         die "Directory given with --prefix MUST be absolute\n"
793                                 unless file_name_is_absolute($config{prefix});
794                         }
795                 elsif (/^--api=(.*)$/)
796                         {
797                         $config{api}=$1;
798                         }
799                 elsif (/^--libdir=(.*)$/)
800                         {
801                         $config{libdir}=$1;
802                         }
803                 elsif (/^--openssldir=(.*)$/)
804                         {
805                         $config{openssldir}=$1;
806                         }
807                 elsif (/^--with-zlib-lib=(.*)$/)
808                         {
809                         $withargs{zlib_lib}=$1;
810                         }
811                 elsif (/^--with-zlib-include=(.*)$/)
812                         {
813                         $withargs{zlib_include}=$1;
814                         }
815                 elsif (/^--with-fuzzer-lib=(.*)$/)
816                         {
817                         $withargs{fuzzer_lib}=$1;
818                         }
819                 elsif (/^--with-fuzzer-include=(.*)$/)
820                         {
821                         $withargs{fuzzer_include}=$1;
822                         }
823                 elsif (/^--with-rand-seed=(.*)$/)
824                         {
825                         foreach my $x (split(m|,|, $1))
826                             {
827                             die "Unknown --with-rand-seed choice $x\n"
828                                 if ! grep { $x eq $_ } @known_seed_sources;
829                             push @seed_sources, $x;
830                             }
831                         }
832                 elsif (/^--cross-compile-prefix=(.*)$/)
833                         {
834                         $user{CROSS_COMPILE}=$1;
835                         }
836                 elsif (/^--config=(.*)$/)
837                         {
838                         read_config $1;
839                         }
840                 elsif (/^-l(.*)$/)
841                         {
842                         push @{$useradd{LDLIBS}}, $_;
843                         }
844                 elsif (/^-framework$/)
845                         {
846                         push @{$useradd{LDLIBS}}, $_, shift(@argvcopy);
847                         }
848                 elsif (/^-L(.*)$/ or /^-Wl,/)
849                         {
850                         push @{$useradd{LDFLAGS}}, $_;
851                         }
852                 elsif (/^-rpath$/ or /^-R$/)
853                         # -rpath is the OSF1 rpath flag
854                         # -R is the old Solaris rpath flag
855                         {
856                         my $rpath = shift(@argvcopy) || "";
857                         $rpath .= " " if $rpath ne "";
858                         push @{$useradd{LDFLAGS}}, $_, $rpath;
859                         }
860                 elsif (/^-static$/)
861                         {
862                         push @{$useradd{LDFLAGS}}, $_;
863                         }
864                 elsif (/^-D(.*)$/)
865                         {
866                         push @{$useradd{CPPDEFINES}}, $1;
867                         }
868                 elsif (/^-I(.*)$/)
869                         {
870                         push @{$useradd{CPPINCLUDES}}, $1;
871                         }
872                 elsif (/^-Wp,$/)
873                         {
874                         push @{$useradd{CPPFLAGS}}, $1;
875                         }
876                 else    # common if (/^[-+]/), just pass down...
877                         {
878                         $_ =~ s/%([0-9a-f]{1,2})/chr(hex($1))/gei;
879                         push @{$useradd{CFLAGS}}, $_;
880                         push @{$useradd{CXXFLAGS}}, $_;
881                         }
882                 }
883         else
884                 {
885                 die "target already defined - $target (offending arg: $_)\n" if ($target ne "");
886                 $target=$_;
887                 }
888         unless ($_ eq $target || /^no-/ || /^disable-/)
889                 {
890                 # "no-..." follows later after implied deactivations
891                 # have been derived.  (Don't take this too seriously,
892                 # we really only write OPTIONS to the Makefile out of
893                 # nostalgia.)
894
895                 if ($config{options} eq "")
896                         { $config{options} = $_; }
897                 else
898                         { $config{options} .= " ".$_; }
899                 }
900         }
901
902 if (defined($config{api}) && !exists $apitable->{$config{api}}) {
903         die "***** Unsupported api compatibility level: $config{api}\n",
904 }
905
906 if (keys %deprecated_options)
907         {
908         warn "***** Deprecated options: ",
909                 join(", ", keys %deprecated_options), "\n";
910         }
911 if (keys %unsupported_options)
912         {
913         die "***** Unsupported options: ",
914                 join(", ", keys %unsupported_options), "\n";
915         }
916
917 # If any %useradd entry has been set, we must check that the "make
918 # variables" haven't been set.  We start by checking of any %useradd entry
919 # is set.
920 if (grep { scalar @$_ > 0 } values %useradd) {
921     # Hash of env / make variables names.  The possible values are:
922     # 1 - "make vars"
923     # 2 - %useradd entry set
924     # 3 - both set
925     my %detected_vars =
926         map { my $v = 0;
927               $v += 1 if $cmdvars{$_};
928               $v += 2 if @{$useradd{$_}};
929               $_ => $v }
930         keys %useradd;
931
932     # If any of the corresponding "make variables" is set, we error
933     if (grep { $_ & 1 } values %detected_vars) {
934         my $names = join(', ', grep { $detected_vars{$_} > 0 }
935                                sort keys %detected_vars);
936         die <<"_____";
937 ***** Mixing make variables and additional compiler/linker flags as
938 ***** configure command line option is not permitted.
939 ***** Affected make variables: $names
940 _____
941     }
942 }
943
944 # Check through all supported command line variables to see if any of them
945 # were set, and canonicalise the values we got.  If no compiler or linker
946 # flag or anything else that affects %useradd was set, we also check the
947 # environment for values.
948 my $anyuseradd =
949     grep { defined $_ && (ref $_ ne 'ARRAY' || @$_) } values %useradd;
950 foreach (keys %user) {
951     my $value = $cmdvars{$_};
952     $value //= env($_) unless $anyuseradd;
953     $value //=
954         defined $user_synonyms{$_} ? $cmdvars{$user_synonyms{$_}} : undef;
955     $value //= defined $user_synonyms{$_} ? env($user_synonyms{$_}) : undef
956         unless $anyuseradd;
957
958     if (defined $value) {
959         if (ref $user{$_} eq 'ARRAY') {
960             $user{$_} = [ split /$list_separator_re/, $value ];
961         } elsif (!defined $user{$_}) {
962             $user{$_} = $value;
963         }
964     }
965 }
966
967 if (grep { /-rpath\b/ } ($user{LDFLAGS} ? @{$user{LDFLAGS}} : ())
968     && !$disabled{shared}
969     && !($disabled{asan} && $disabled{msan} && $disabled{ubsan})) {
970     die "***** Cannot simultaneously use -rpath, shared libraries, and\n",
971         "***** any of asan, msan or ubsan\n";
972 }
973
974 sub disable {
975     my $disable_type = shift;
976
977     for (@_) {
978         $disabled{$_} = $disable_type;
979     }
980
981     my @tocheckfor = (@_ ? @_ : keys %disabled);
982     while (@tocheckfor) {
983         my %new_tocheckfor = ();
984         my @cascade_copy = (@disable_cascades);
985         while (@cascade_copy) {
986             my ($test, $descendents) =
987                 (shift @cascade_copy, shift @cascade_copy);
988             if (ref($test) eq "CODE" ? $test->() : defined($disabled{$test})) {
989                 foreach (grep { !defined($disabled{$_}) } @$descendents) {
990                     $new_tocheckfor{$_} = 1; $disabled{$_} = "cascade";
991                 }
992             }
993         }
994         @tocheckfor = (keys %new_tocheckfor);
995     }
996 }
997 disable();                     # First cascade run
998
999 our $die = sub { die @_; };
1000 if ($target eq "TABLE") {
1001     local $die = sub { warn @_; };
1002     foreach (sort keys %table) {
1003         print_table_entry($_, "TABLE");
1004     }
1005     exit 0;
1006 }
1007
1008 if ($target eq "LIST") {
1009     foreach (sort keys %table) {
1010         print $_,"\n" unless $table{$_}->{template};
1011     }
1012     exit 0;
1013 }
1014
1015 if ($target eq "HASH") {
1016     local $die = sub { warn @_; };
1017     print "%table = (\n";
1018     foreach (sort keys %table) {
1019         print_table_entry($_, "HASH");
1020     }
1021     exit 0;
1022 }
1023
1024 print "Configuring OpenSSL version $config{version} ($config{version_num}) ";
1025 print "for $target\n";
1026
1027 if (scalar(@seed_sources) == 0) {
1028     print "Using os-specific seed configuration\n";
1029     push @seed_sources, 'os';
1030 }
1031 if (scalar(grep { $_ eq 'none' } @seed_sources) > 0) {
1032     die "Cannot seed with none and anything else" if scalar(@seed_sources) > 1;
1033     warn <<_____ if scalar(@seed_sources) == 1;
1034
1035 ============================== WARNING ===============================
1036 You have selected the --with-rand-seed=none option, which effectively
1037 disables automatic reseeding of the OpenSSL random generator.
1038 All operations depending on the random generator such as creating keys
1039 will not work unless the random generator is seeded manually by the
1040 application.
1041
1042 Please read the 'Note on random number generation' section in the
1043 INSTALL instructions and the RAND_DRBG(7) manual page for more details.
1044 ============================== WARNING ===============================
1045
1046 _____
1047 }
1048 push @{$config{openssl_other_defines}},
1049      map { (my $x = $_) =~ tr|[\-a-z]|[_A-Z]|; "OPENSSL_RAND_SEED_$x" }
1050         @seed_sources;
1051
1052 # Backward compatibility?
1053 if ($target =~ m/^CygWin32(-.*)$/) {
1054     $target = "Cygwin".$1;
1055 }
1056
1057 # Support for legacy targets having a name starting with 'debug-'
1058 my ($d, $t) = $target =~ m/^(debug-)?(.*)$/;
1059 if ($d) {
1060     $config{build_type} = "debug";
1061
1062     # If we do not find debug-foo in the table, the target is set to foo.
1063     if (!$table{$target}) {
1064         $target = $t;
1065     }
1066 }
1067
1068 &usage if !$table{$target} || $table{$target}->{template};
1069
1070 $config{target} = $target;
1071 my %target = resolve_config($target);
1072
1073 foreach (keys %target_attr_translate) {
1074     $target{$target_attr_translate{$_}} = $target{$_}
1075         if $target{$_};
1076     delete $target{$_};
1077 }
1078
1079 %target = ( %{$table{DEFAULTS}}, %target );
1080
1081 # Make the flags to build DSOs the same as for shared libraries unless they
1082 # are already defined
1083 $target{module_cflags} = $target{shared_cflag} unless defined $target{module_cflags};
1084 $target{module_cxxflags} = $target{shared_cxxflag} unless defined $target{module_cxxflags};
1085 $target{module_ldflags} = $target{shared_ldflag} unless defined $target{module_ldflags};
1086 {
1087     my $shared_info_pl =
1088         catfile(dirname($0), "Configurations", "shared-info.pl");
1089     my %shared_info = read_eval_file($shared_info_pl);
1090     push @{$target{_conf_fname_int}}, $shared_info_pl;
1091     my $si = $target{shared_target};
1092     while (ref $si ne "HASH") {
1093         last if ! defined $si;
1094         if (ref $si eq "CODE") {
1095             $si = $si->();
1096         } else {
1097             $si = $shared_info{$si};
1098         }
1099     }
1100
1101     # Some of the 'shared_target' values don't have any entried in
1102     # %shared_info.  That's perfectly fine, AS LONG AS the build file
1103     # template knows how to handle this.  That is currently the case for
1104     # Windows and VMS.
1105     if (defined $si) {
1106         # Just as above, copy certain shared_* attributes to the corresponding
1107         # module_ attribute unless the latter is already defined
1108         $si->{module_cflags} = $si->{shared_cflag} unless defined $si->{module_cflags};
1109         $si->{module_cxxflags} = $si->{shared_cxxflag} unless defined $si->{module_cxxflags};
1110         $si->{module_ldflags} = $si->{shared_ldflag} unless defined $si->{module_ldflags};
1111         foreach (sort keys %$si) {
1112             $target{$_} = defined $target{$_}
1113                 ? add($si->{$_})->($target{$_})
1114                 : $si->{$_};
1115         }
1116     }
1117 }
1118
1119 my %conf_files = map { $_ => 1 } (@{$target{_conf_fname_int}});
1120 $config{conf_files} = [ sort keys %conf_files ];
1121
1122 # Using sub disable within these loops may prove fragile, so we run
1123 # a cascade afterwards
1124 foreach my $feature (@{$target{disable}}) {
1125     if (exists $deprecated_disablables{$feature}) {
1126         warn "***** config $target disables deprecated feature $feature\n";
1127     } elsif (!grep { $feature eq $_ } @disablables) {
1128         die "***** config $target disables unknown feature $feature\n";
1129     }
1130     $disabled{$feature} = 'config';
1131 }
1132 foreach my $feature (@{$target{enable}}) {
1133     if ("default" eq ($disabled{$feature} // "")) {
1134         if (exists $deprecated_disablables{$feature}) {
1135             warn "***** config $target enables deprecated feature $feature\n";
1136         } elsif (!grep { $feature eq $_ } @disablables) {
1137             die "***** config $target enables unknown feature $feature\n";
1138         }
1139         delete $disabled{$feature};
1140     }
1141 }
1142 disable();                      # Run a cascade now
1143
1144 $target{CXXFLAGS}//=$target{CFLAGS} if $target{CXX};
1145 $target{cxxflags}//=$target{cflags} if $target{CXX};
1146 $target{exe_extension}="";
1147 $target{exe_extension}=".exe" if ($config{target} eq "DJGPP"
1148                                   || $config{target} =~ /^(?:Cygwin|mingw)/);
1149 $target{exe_extension}=".pm"  if ($config{target} =~ /vos/);
1150
1151 ($target{shared_extension_simple}=$target{shared_extension})
1152     =~ s|\.\$\(SHLIB_VERSION_NUMBER\)||
1153     unless defined($target{shared_extension_simple});
1154 $target{dso_extension}//=$target{shared_extension_simple};
1155 ($target{shared_import_extension}=$target{shared_extension_simple}.".a")
1156     if ($config{target} =~ /^(?:Cygwin|mingw)/);
1157
1158 # Fill %config with values from %user, and in case those are undefined or
1159 # empty, use values from %target (acting as a default).
1160 foreach (keys %user) {
1161     my $ref_type = ref $user{$_};
1162
1163     # Temporary function.  Takes an intended ref type (empty string or "ARRAY")
1164     # and a value that's to be coerced into that type.
1165     my $mkvalue = sub {
1166         my $type = shift;
1167         my $value = shift;
1168         my $undef_p = shift;
1169
1170         die "Too many arguments for \$mkvalue" if @_;
1171
1172         while (ref $value eq 'CODE') {
1173             $value = $value->();
1174         }
1175
1176         if ($type eq 'ARRAY') {
1177             return undef unless defined $value;
1178             return undef if ref $value ne 'ARRAY' && !$value;
1179             return undef if ref $value eq 'ARRAY' && !@$value;
1180             return [ $value ] unless ref $value eq 'ARRAY';
1181         }
1182         return undef unless $value;
1183         return $value;
1184     };
1185
1186     $config{$_} =
1187         $mkvalue->($ref_type, $user{$_})
1188         || $mkvalue->($ref_type, $target{$_});
1189     delete $config{$_} unless defined $config{$_};
1190 }
1191
1192 # Finish up %config by appending things the user gave us on the command line
1193 # apart from "make variables"
1194 foreach (keys %useradd) {
1195     # The must all be lists, so we assert that here
1196     die "internal error: \$useradd{$_} isn't an ARRAY\n"
1197         unless ref $useradd{$_} eq 'ARRAY';
1198
1199     if (defined $config{$_}) {
1200         push @{$config{$_}}, @{$useradd{$_}};
1201     } else {
1202         $config{$_} = [ @{$useradd{$_}} ];
1203     }
1204 }
1205 # At this point, we can forget everything about %user and %useradd,
1206 # because it's now all been merged into the corresponding $config entry
1207
1208 # Allow overriding the build file name
1209 $config{build_file} = env('BUILDFILE') || $target{build_file} || "Makefile";
1210
1211 my %disabled_info = ();         # For configdata.pm
1212 foreach my $what (sort keys %disabled) {
1213     $config{options} .= " no-$what";
1214
1215     if (!grep { $what eq $_ } ( 'threads', 'shared', 'pic',
1216                                 'dynamic-engine', 'makedepend',
1217                                 'zlib-dynamic', 'zlib', 'sse2' )) {
1218         (my $WHAT = uc $what) =~ s|-|_|g;
1219
1220         # Fix up C macro end names
1221         $WHAT = "RMD160" if $what eq "ripemd";
1222
1223         # fix-up crypto/directory name(s)
1224         $what = "ripemd" if $what eq "rmd160";
1225         $what = "whrlpool" if $what eq "whirlpool";
1226
1227         my $macro = $disabled_info{$what}->{macro} = "OPENSSL_NO_$WHAT";
1228
1229         if ((grep { $what eq $_ } @{$config{sdirs}})
1230                 && $what ne 'async' && $what ne 'err') {
1231             @{$config{sdirs}} = grep { $what ne $_} @{$config{sdirs}};
1232             $disabled_info{$what}->{skipped} = [ catdir('crypto', $what) ];
1233
1234             if ($what ne 'engine') {
1235                 push @{$config{openssl_algorithm_defines}}, $macro;
1236             } else {
1237                 @{$config{dirs}} = grep !/^engines$/, @{$config{dirs}};
1238                 push @{$disabled_info{engine}->{skipped}}, catdir('engines');
1239                 push @{$config{openssl_other_defines}}, $macro;
1240             }
1241         } else {
1242             push @{$config{openssl_other_defines}}, $macro;
1243         }
1244
1245     }
1246 }
1247
1248 # Make sure build_scheme is consistent.
1249 $target{build_scheme} = [ $target{build_scheme} ]
1250     if ref($target{build_scheme}) ne "ARRAY";
1251
1252 my ($builder, $builder_platform, @builder_opts) =
1253     @{$target{build_scheme}};
1254
1255 foreach my $checker (($builder_platform."-".$target{build_file}."-checker.pm",
1256                       $builder_platform."-checker.pm")) {
1257     my $checker_path = catfile($srcdir, "Configurations", $checker);
1258     if (-f $checker_path) {
1259         my $fn = $ENV{CONFIGURE_CHECKER_WARN}
1260             ? sub { warn $@; } : sub { die $@; };
1261         if (! do $checker_path) {
1262             if ($@) {
1263                 $fn->($@);
1264             } elsif ($!) {
1265                 $fn->($!);
1266             } else {
1267                 $fn->("The detected tools didn't match the platform\n");
1268             }
1269         }
1270         last;
1271     }
1272 }
1273
1274 push @{$config{defines}}, "NDEBUG"    if $config{build_type} eq "release";
1275
1276 if ($target =~ /^mingw/ && `$config{CC} --target-help 2>&1` =~ m/-mno-cygwin/m)
1277         {
1278         push @{$config{cflags}}, "-mno-cygwin";
1279         push @{$config{cxxflags}}, "-mno-cygwin" if $config{CXX};
1280         push @{$config{shared_ldflag}}, "-mno-cygwin";
1281         }
1282
1283 if ($target =~ /linux.*-mips/ && !$disabled{asm}
1284         && !grep { $_ !~ /-m(ips|arch=)/ } (@{$config{CFLAGS}})) {
1285         # minimally required architecture flags for assembly modules
1286         my $value;
1287         $value = '-mips2' if ($target =~ /mips32/);
1288         $value = '-mips3' if ($target =~ /mips64/);
1289         unshift @{$config{cflags}}, $value;
1290         unshift @{$config{cxxflags}}, $value if $config{CXX};
1291 }
1292
1293 # If threads aren't disabled, check how possible they are
1294 unless ($disabled{threads}) {
1295     if ($auto_threads) {
1296         # Enabled by default, disable it forcibly if unavailable
1297         if ($target{thread_scheme} eq "(unknown)") {
1298             disable("unavailable", 'threads');
1299         }
1300     } else {
1301         # The user chose to enable threads explicitly, let's see
1302         # if there's a chance that's possible
1303         if ($target{thread_scheme} eq "(unknown)") {
1304             # If the user asked for "threads" and we don't have internal
1305             # knowledge how to do it, [s]he is expected to provide any
1306             # system-dependent compiler options that are necessary.  We
1307             # can't truly check that the given options are correct, but
1308             # we expect the user to know what [s]He is doing.
1309             if (!@{$config{CFLAGS}} && !@{$config{CPPDEFINES}}) {
1310                 die "You asked for multi-threading support, but didn't\n"
1311                     ,"provide any system-specific compiler options\n";
1312             }
1313         }
1314     }
1315 }
1316
1317 # If threads still aren't disabled, add a C macro to ensure the source
1318 # code knows about it.  Any other flag is taken care of by the configs.
1319 unless($disabled{threads}) {
1320     push @{$config{openssl_thread_defines}}, "OPENSSL_THREADS";
1321 }
1322
1323 # With "deprecated" disable all deprecated features.
1324 if (defined($disabled{"deprecated"})) {
1325         $config{api} = $maxapi;
1326 }
1327
1328 my $no_shared_warn=0;
1329 if ($target{shared_target} eq "")
1330         {
1331         $no_shared_warn = 1
1332             if (!$disabled{shared} || !$disabled{"dynamic-engine"});
1333         disable('no-shared-target', 'pic');
1334         }
1335
1336 if ($disabled{"dynamic-engine"}) {
1337         push @{$config{openssl_other_defines}}, "OPENSSL_NO_DYNAMIC_ENGINE";
1338         $config{dynamic_engines} = 0;
1339 } else {
1340         push @{$config{openssl_other_defines}}, "OPENSSL_NO_STATIC_ENGINE";
1341         $config{dynamic_engines} = 1;
1342 }
1343
1344 unless ($disabled{asan}) {
1345     push @{$config{cflags}}, "-fsanitize=address";
1346     push @{$config{cxxflags}}, "-fsanitize=address" if $config{CXX};
1347 }
1348
1349 unless ($disabled{ubsan}) {
1350     # -DPEDANTIC or -fnosanitize=alignment may also be required on some
1351     # platforms.
1352     push @{$config{cflags}}, "-fsanitize=undefined", "-fno-sanitize-recover=all";
1353     push @{$config{cxxflags}}, "-fsanitize=undefined", "-fno-sanitize-recover=all"
1354         if $config{CXX};
1355 }
1356
1357 unless ($disabled{msan}) {
1358   push @{$config{cflags}}, "-fsanitize=memory";
1359   push @{$config{cxxflags}}, "-fsanitize=memory" if $config{CXX};
1360 }
1361
1362 unless ($disabled{"fuzz-libfuzzer"} && $disabled{"fuzz-afl"}
1363         && $disabled{asan} && $disabled{ubsan} && $disabled{msan}) {
1364     push @{$config{cflags}}, "-fno-omit-frame-pointer", "-g";
1365     push @{$config{cxxflags}}, "-fno-omit-frame-pointer", "-g" if $config{CXX};
1366 }
1367 #
1368 # Platform fix-ups
1369 #
1370
1371 # This saves the build files from having to check
1372 if ($disabled{pic})
1373         {
1374         foreach (qw(shared_cflag shared_cxxflag shared_cppflag
1375                     shared_defines shared_includes shared_ldflag
1376                     module_cflags module_cxxflags module_cppflags
1377                     module_defines module_includes module_lflags))
1378                 {
1379                 delete $config{$_};
1380                 $target{$_} = "";
1381                 }
1382         }
1383 else
1384         {
1385         push @{$config{lib_defines}}, "OPENSSL_PIC";
1386         }
1387
1388 if ($target{sys_id} ne "")
1389         {
1390         push @{$config{openssl_sys_defines}}, "OPENSSL_SYS_$target{sys_id}";
1391         }
1392
1393 unless ($disabled{asm}) {
1394     $target{cpuid_asm_src}=$table{DEFAULTS}->{cpuid_asm_src} if ($config{processor} eq "386");
1395     push @{$config{lib_defines}}, "OPENSSL_CPUID_OBJ" if ($target{cpuid_asm_src} ne "mem_clr.c");
1396
1397     $target{bn_asm_src} =~ s/\w+-gf2m.c// if (defined($disabled{ec2m}));
1398
1399     # bn-586 is the only one implementing bn_*_part_words
1400     push @{$config{lib_defines}}, "OPENSSL_BN_ASM_PART_WORDS" if ($target{bn_asm_src} =~ /bn-586/);
1401     push @{$config{lib_defines}}, "OPENSSL_IA32_SSE2" if (!$disabled{sse2} && $target{bn_asm_src} =~ /86/);
1402
1403     push @{$config{lib_defines}}, "OPENSSL_BN_ASM_MONT" if ($target{bn_asm_src} =~ /-mont/);
1404     push @{$config{lib_defines}}, "OPENSSL_BN_ASM_MONT5" if ($target{bn_asm_src} =~ /-mont5/);
1405     push @{$config{lib_defines}}, "OPENSSL_BN_ASM_GF2m" if ($target{bn_asm_src} =~ /-gf2m/);
1406     push @{$config{lib_defines}}, "BN_DIV3W" if ($target{bn_asm_src} =~ /-div3w/);
1407
1408     if ($target{sha1_asm_src}) {
1409         push @{$config{lib_defines}}, "SHA1_ASM"   if ($target{sha1_asm_src} =~ /sx86/ || $target{sha1_asm_src} =~ /sha1/);
1410         push @{$config{lib_defines}}, "SHA256_ASM" if ($target{sha1_asm_src} =~ /sha256/);
1411         push @{$config{lib_defines}}, "SHA512_ASM" if ($target{sha1_asm_src} =~ /sha512/);
1412     }
1413     if ($target{keccak1600_asm_src} ne $table{DEFAULTS}->{keccak1600_asm_src}) {
1414         push @{$config{lib_defines}}, "KECCAK1600_ASM";
1415     }
1416     if ($target{rc4_asm_src} ne $table{DEFAULTS}->{rc4_asm_src}) {
1417         push @{$config{lib_defines}}, "RC4_ASM";
1418     }
1419     if ($target{md5_asm_src}) {
1420         push @{$config{lib_defines}}, "MD5_ASM";
1421     }
1422     $target{cast_asm_src}=$table{DEFAULTS}->{cast_asm_src} unless $disabled{pic}; # CAST assembler is not PIC
1423     if ($target{rmd160_asm_src}) {
1424         push @{$config{lib_defines}}, "RMD160_ASM";
1425     }
1426     if ($target{aes_asm_src}) {
1427         push @{$config{lib_defines}}, "AES_ASM" if ($target{aes_asm_src} =~ m/\baes-/);;
1428         # aes-ctr.fake is not a real file, only indication that assembler
1429         # module implements AES_ctr32_encrypt...
1430         push @{$config{lib_defines}}, "AES_CTR_ASM" if ($target{aes_asm_src} =~ s/\s*aes-ctr\.fake//);
1431         # aes-xts.fake indicates presence of AES_xts_[en|de]crypt...
1432         push @{$config{lib_defines}}, "AES_XTS_ASM" if ($target{aes_asm_src} =~ s/\s*aes-xts\.fake//);
1433         $target{aes_asm_src} =~ s/\s*(vpaes|aesni)-x86\.s//g if ($disabled{sse2});
1434         push @{$config{lib_defines}}, "VPAES_ASM" if ($target{aes_asm_src} =~ m/vpaes/);
1435         push @{$config{lib_defines}}, "BSAES_ASM" if ($target{aes_asm_src} =~ m/bsaes/);
1436     }
1437     if ($target{wp_asm_src} =~ /mmx/) {
1438         if ($config{processor} eq "386") {
1439             $target{wp_asm_src}=$table{DEFAULTS}->{wp_asm_src};
1440         } elsif (!$disabled{"whirlpool"}) {
1441             push @{$config{lib_defines}}, "WHIRLPOOL_ASM";
1442         }
1443     }
1444     if ($target{modes_asm_src} =~ /ghash-/) {
1445         push @{$config{lib_defines}}, "GHASH_ASM";
1446     }
1447     if ($target{ec_asm_src} =~ /ecp_nistz256/) {
1448         push @{$config{lib_defines}}, "ECP_NISTZ256_ASM";
1449     }
1450     if ($target{ec_asm_src} =~ /x25519/) {
1451         push @{$config{lib_defines}}, "X25519_ASM";
1452     }
1453     if ($target{padlock_asm_src} ne $table{DEFAULTS}->{padlock_asm_src}) {
1454         push @{$config{dso_defines}}, "PADLOCK_ASM";
1455     }
1456     if ($target{poly1305_asm_src} ne "") {
1457         push @{$config{lib_defines}}, "POLY1305_ASM";
1458     }
1459 }
1460
1461 my %predefined_C = compiler_predefined($config{CROSS_COMPILE}.$config{CC});
1462 my %predefined_CXX = $config{CXX}
1463     ? compiler_predefined($config{CROSS_COMPILE}.$config{CXX})
1464     : ();
1465
1466 # Check for makedepend capabilities.
1467 if (!$disabled{makedepend}) {
1468     if ($config{target} =~ /^(VC|vms)-/) {
1469         # For VC- and vms- targets, there's nothing more to do here.  The
1470         # functionality is hard coded in the corresponding build files for
1471         # cl (Windows) and CC/DECC (VMS).
1472     } elsif (($predefined_C{__GNUC__} // -1) >= 3
1473              && !($predefined_C{__APPLE_CC__} && !$predefined_C{__clang__})) {
1474         # We know that GNU C version 3 and up as well as all clang
1475         # versions support dependency generation, but Xcode did not
1476         # handle $cc -M before clang support (but claims __GNUC__ = 3)
1477         $config{makedepprog} = "\$(CROSS_COMPILE)$config{CC}";
1478     } else {
1479         # In all other cases, we look for 'makedepend', and disable the
1480         # capability if not found.
1481         $config{makedepprog} = which('makedepend');
1482         disable('unavailable', 'makedepend') unless $config{makedepprog};
1483     }
1484 }
1485
1486 if (!$disabled{asm} && !$predefined_C{__MACH__} && $^O ne 'VMS') {
1487     # probe for -Wa,--noexecstack option...
1488     if ($predefined_C{__clang__}) {
1489         # clang has builtin assembler, which doesn't recognize --help,
1490         # but it apparently recognizes the option in question on all
1491         # supported platforms even when it's meaningless. In other words
1492         # probe would fail, but probed option always accepted...
1493         push @{$config{cflags}}, "-Wa,--noexecstack", "-Qunused-arguments";
1494     } else {
1495         my $cc = $config{CROSS_COMPILE}.$config{CC};
1496         open(PIPE, "$cc -Wa,--help -c -o null.$$.o -x assembler /dev/null 2>&1 |");
1497         while(<PIPE>) {
1498             if (m/--noexecstack/) {
1499                 push @{$config{cflags}}, "-Wa,--noexecstack";
1500                 last;
1501             }
1502         }
1503         close(PIPE);
1504         unlink("null.$$.o");
1505     }
1506 }
1507
1508 # Deal with bn_ops ###################################################
1509
1510 $config{bn_ll}                  =0;
1511 $config{export_var_as_fn}       =0;
1512 my $def_int="unsigned int";
1513 $config{rc4_int}                =$def_int;
1514 ($config{b64l},$config{b64},$config{b32})=(0,0,1);
1515
1516 my $count = 0;
1517 foreach (sort split(/\s+/,$target{bn_ops})) {
1518     $count++ if /SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT/;
1519     $config{export_var_as_fn}=1                 if $_ eq 'EXPORT_VAR_AS_FN';
1520     $config{bn_ll}=1                            if $_ eq 'BN_LLONG';
1521     $config{rc4_int}="unsigned char"            if $_ eq 'RC4_CHAR';
1522     ($config{b64l},$config{b64},$config{b32})
1523         =(0,1,0)                                if $_ eq 'SIXTY_FOUR_BIT';
1524     ($config{b64l},$config{b64},$config{b32})
1525         =(1,0,0)                                if $_ eq 'SIXTY_FOUR_BIT_LONG';
1526     ($config{b64l},$config{b64},$config{b32})
1527         =(0,0,1)                                if $_ eq 'THIRTY_TWO_BIT';
1528 }
1529 die "Exactly one of SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT can be set in bn_ops\n"
1530     if $count > 1;
1531
1532
1533 # Hack cflags for better warnings (dev option) #######################
1534
1535 # "Stringify" the C and C++ flags string.  This permits it to be made part of
1536 # a string and works as well on command lines.
1537 $config{cflags} = [ map { (my $x = $_) =~ s/([\\\"])/\\$1/g; $x }
1538                         @{$config{cflags}} ];
1539 $config{cxxflags} = [ map { (my $x = $_) =~ s/([\\\"])/\\$1/g; $x }
1540                           @{$config{cxxflags}} ] if $config{CXX};
1541
1542 if (defined($config{api})) {
1543     $config{openssl_api_defines} = [ "OPENSSL_MIN_API=".$apitable->{$config{api}} ];
1544     my $apiflag = sprintf("OPENSSL_API_COMPAT=%s", $apitable->{$config{api}});
1545     push @{$config{defines}}, $apiflag;
1546 }
1547
1548 my %strict_warnings_collection=( CFLAGS => [], CXXFLAGS => []);
1549 if ($strict_warnings)
1550         {
1551         my $wopt;
1552         my $gccver = $predefined_C{__GNUC__} // -1;
1553         my $gxxver = $predefined_CXX{__GNUC__} // -1;
1554
1555         warn "WARNING --strict-warnings requires gcc[>=4] or gcc-alike"
1556             unless $gccver >= 4;
1557         warn "WARNING --strict-warnings requires g++[>=4] or g++-alike"
1558             unless $gxxver >= 4;
1559         foreach (qw(CFLAGS CXXFLAGS))
1560                 {
1561                 push @{$strict_warnings_collection{$_}},
1562                         @{$gcc_devteam_warn{$_}};
1563                 }
1564         push @{$strict_warnings_collection{CFLAGS}},
1565                 @{$clang_devteam_warn{CFLAGS}}
1566                         if (defined($predefined_C{__clang__}));
1567         push @{$strict_warnings_collection{CXXFLAGS}},
1568                 @{$clang_devteam_warn{CXXFLAGS}}
1569                         if (defined($predefined_CXX{__clang__}));
1570         }
1571
1572 if (grep { $_ eq '-static' } @{$config{LDFLAGS}}) {
1573     disable('static', 'pic', 'threads');
1574 }
1575
1576 foreach my $idx (qw(CFLAGS CXXFLAGS))
1577         {
1578         $config{$idx} = [ map { $_ eq '--ossl-strict-warnings'
1579                                     ? @{$strict_warnings_collection{$idx}}
1580                                 : ( $_ ) }
1581                           @{$config{$idx}} ];
1582         }
1583
1584 unless ($disabled{"crypto-mdebug-backtrace"})
1585         {
1586         foreach my $wopt (split /\s+/, $memleak_devteam_backtrace)
1587                 {
1588                 push @{$config{cflags}}, $wopt
1589                         unless grep { $_ eq $wopt } @{$config{cflags}};
1590                 push @{$config{cxxflags}}, $wopt
1591                         if ($config{CXX}
1592                             && !grep { $_ eq $wopt } @{$config{cxxflags}});
1593                 }
1594         if ($target =~ /^BSD-/)
1595                 {
1596                 push @{$config{ex_libs}}, "-lexecinfo";
1597                 }
1598         }
1599
1600 unless ($disabled{afalgeng}) {
1601     $config{afalgeng}="";
1602     if (grep { $_ eq 'afalgeng' } @{$target{enable}}) {
1603         my $minver = 4*10000 + 1*100 + 0;
1604         if ($config{CROSS_COMPILE} eq "") {
1605             my $verstr = `uname -r`;
1606             my ($ma, $mi1, $mi2) = split("\\.", $verstr);
1607             ($mi2) = $mi2 =~ /(\d+)/;
1608             my $ver = $ma*10000 + $mi1*100 + $mi2;
1609             if ($ver < $minver) {
1610                 disable('too-old-kernel', 'afalgeng');
1611             } else {
1612                 push @{$config{engdirs}}, "afalg";
1613             }
1614         } else {
1615             disable('cross-compiling', 'afalgeng');
1616         }
1617     } else {
1618         disable('not-linux', 'afalgeng');
1619     }
1620 }
1621
1622 push @{$config{openssl_other_defines}}, "OPENSSL_NO_AFALGENG" if ($disabled{afalgeng});
1623
1624 # ALL MODIFICATIONS TO %config and %target MUST BE DONE FROM HERE ON
1625
1626 # If we use the unified build, collect information from build.info files
1627 my %unified_info = ();
1628
1629 my $buildinfo_debug = defined($ENV{CONFIGURE_DEBUG_BUILDINFO});
1630 if ($builder eq "unified") {
1631     use with_fallback qw(Text::Template);
1632
1633     sub cleandir {
1634         my $base = shift;
1635         my $dir = shift;
1636         my $relativeto = shift || ".";
1637
1638         $dir = catdir($base,$dir) unless isabsolute($dir);
1639
1640         # Make sure the directories we're building in exists
1641         mkpath($dir);
1642
1643         my $res = abs2rel(absolutedir($dir), rel2abs($relativeto));
1644         #print STDERR "DEBUG[cleandir]: $dir , $base => $res\n";
1645         return $res;
1646     }
1647
1648     sub cleanfile {
1649         my $base = shift;
1650         my $file = shift;
1651         my $relativeto = shift || ".";
1652
1653         $file = catfile($base,$file) unless isabsolute($file);
1654
1655         my $d = dirname($file);
1656         my $f = basename($file);
1657
1658         # Make sure the directories we're building in exists
1659         mkpath($d);
1660
1661         my $res = abs2rel(catfile(absolutedir($d), $f), rel2abs($relativeto));
1662         #print STDERR "DEBUG[cleanfile]: $d , $f => $res\n";
1663         return $res;
1664     }
1665
1666     # Store the name of the template file we will build the build file from
1667     # in %config.  This may be useful for the build file itself.
1668     my @build_file_template_names =
1669         ( $builder_platform."-".$target{build_file}.".tmpl",
1670           $target{build_file}.".tmpl" );
1671     my @build_file_templates = ();
1672
1673     # First, look in the user provided directory, if given
1674     if (defined env($local_config_envname)) {
1675         @build_file_templates =
1676             map {
1677                 if ($^O eq 'VMS') {
1678                     # VMS environment variables are logical names,
1679                     # which can be used as is
1680                     $local_config_envname . ':' . $_;
1681                 } else {
1682                     catfile(env($local_config_envname), $_);
1683                 }
1684             }
1685             @build_file_template_names;
1686     }
1687     # Then, look in our standard directory
1688     push @build_file_templates,
1689         ( map { cleanfile($srcdir, catfile("Configurations", $_), $blddir) }
1690           @build_file_template_names );
1691
1692     my $build_file_template;
1693     for $_ (@build_file_templates) {
1694         $build_file_template = $_;
1695         last if -f $build_file_template;
1696
1697         $build_file_template = undef;
1698     }
1699     if (!defined $build_file_template) {
1700         die "*** Couldn't find any of:\n", join("\n", @build_file_templates), "\n";
1701     }
1702     $config{build_file_templates}
1703       = [ cleanfile($srcdir, catfile("Configurations", "common0.tmpl"),
1704                     $blddir),
1705           $build_file_template,
1706           cleanfile($srcdir, catfile("Configurations", "common.tmpl"),
1707                     $blddir) ];
1708
1709     my @build_infos = ( [ ".", "build.info" ] );
1710     foreach (@{$config{dirs}}) {
1711         push @build_infos, [ $_, "build.info" ]
1712             if (-f catfile($srcdir, $_, "build.info"));
1713     }
1714     foreach (@{$config{sdirs}}) {
1715         push @build_infos, [ catdir("crypto", $_), "build.info" ]
1716             if (-f catfile($srcdir, "crypto", $_, "build.info"));
1717     }
1718     foreach (@{$config{engdirs}}) {
1719         push @build_infos, [ catdir("engines", $_), "build.info" ]
1720             if (-f catfile($srcdir, "engines", $_, "build.info"));
1721     }
1722     foreach (@{$config{tdirs}}) {
1723         push @build_infos, [ catdir("test", $_), "build.info" ]
1724             if (-f catfile($srcdir, "test", $_, "build.info"));
1725     }
1726
1727     $config{build_infos} = [ ];
1728
1729     my %ordinals = ();
1730     foreach (@build_infos) {
1731         my $sourced = catdir($srcdir, $_->[0]);
1732         my $buildd = catdir($blddir, $_->[0]);
1733
1734         mkpath($buildd);
1735
1736         my $f = $_->[1];
1737         # The basic things we're trying to build
1738         my @programs = ();
1739         my @programs_install = ();
1740         my @libraries = ();
1741         my @libraries_install = ();
1742         my @engines = ();
1743         my @engines_install = ();
1744         my @scripts = ();
1745         my @scripts_install = ();
1746         my @extra = ();
1747         my @overrides = ();
1748         my @intermediates = ();
1749         my @rawlines = ();
1750
1751         my %sources = ();
1752         my %shared_sources = ();
1753         my %includes = ();
1754         my %depends = ();
1755         my %renames = ();
1756         my %sharednames = ();
1757         my %generate = ();
1758
1759         # We want to detect configdata.pm in the source tree, so we
1760         # don't use it if the build tree is different.
1761         my $src_configdata = cleanfile($srcdir, "configdata.pm", $blddir);
1762
1763         push @{$config{build_infos}}, catfile(abs2rel($sourced, $blddir), $f);
1764         my $template =
1765             Text::Template->new(TYPE => 'FILE',
1766                                 SOURCE => catfile($sourced, $f),
1767                                 PREPEND => qq{use lib "$FindBin::Bin/util/perl";});
1768         die "Something went wrong with $sourced/$f: $!\n" unless $template;
1769         my @text =
1770             split /^/m,
1771             $template->fill_in(HASH => { config => \%config,
1772                                          target => \%target,
1773                                          disabled => \%disabled,
1774                                          withargs => \%withargs,
1775                                          builddir => abs2rel($buildd, $blddir),
1776                                          sourcedir => abs2rel($sourced, $blddir),
1777                                          buildtop => abs2rel($blddir, $blddir),
1778                                          sourcetop => abs2rel($srcdir, $blddir) },
1779                                DELIMITERS => [ "{-", "-}" ]);
1780
1781         # The top item of this stack has the following values
1782         # -2 positive already run and we found ELSE (following ELSIF should fail)
1783         # -1 positive already run (skip until ENDIF)
1784         # 0 negatives so far (if we're at a condition, check it)
1785         # 1 last was positive (don't skip lines until next ELSE, ELSIF or ENDIF)
1786         # 2 positive ELSE (following ELSIF should fail)
1787         my @skip = ();
1788         collect_information(
1789             collect_from_array([ @text ],
1790                                qr/\\$/ => sub { my $l1 = shift; my $l2 = shift;
1791                                                 $l1 =~ s/\\$//; $l1.$l2 }),
1792             # Info we're looking for
1793             qr/^\s*IF\[((?:\\.|[^\\\]])*)\]\s*$/
1794             => sub {
1795                 if (! @skip || $skip[$#skip] > 0) {
1796                     push @skip, !! $1;
1797                 } else {
1798                     push @skip, -1;
1799                 }
1800             },
1801             qr/^\s*ELSIF\[((?:\\.|[^\\\]])*)\]\s*$/
1802             => sub { die "ELSIF out of scope" if ! @skip;
1803                      die "ELSIF following ELSE" if abs($skip[$#skip]) == 2;
1804                      $skip[$#skip] = -1 if $skip[$#skip] != 0;
1805                      $skip[$#skip] = !! $1
1806                          if $skip[$#skip] == 0; },
1807             qr/^\s*ELSE\s*$/
1808             => sub { die "ELSE out of scope" if ! @skip;
1809                      $skip[$#skip] = -2 if $skip[$#skip] != 0;
1810                      $skip[$#skip] = 2 if $skip[$#skip] == 0; },
1811             qr/^\s*ENDIF\s*$/
1812             => sub { die "ENDIF out of scope" if ! @skip;
1813                      pop @skip; },
1814             qr/^\s*PROGRAMS(_NO_INST)?\s*=\s*(.*)\s*$/
1815             => sub {
1816                 if (!@skip || $skip[$#skip] > 0) {
1817                     my $install = $1;
1818                     my @x = tokenize($2);
1819                     push @programs, @x;
1820                     push @programs_install, @x unless $install;
1821                 }
1822             },
1823             qr/^\s*LIBS(_NO_INST)?\s*=\s*(.*)\s*$/
1824             => sub {
1825                 if (!@skip || $skip[$#skip] > 0) {
1826                     my $install = $1;
1827                     my @x = tokenize($2);
1828                     push @libraries, @x;
1829                     push @libraries_install, @x unless $install;
1830                 }
1831             },
1832             qr/^\s*ENGINES(_NO_INST)?\s*=\s*(.*)\s*$/
1833             => sub {
1834                 if (!@skip || $skip[$#skip] > 0) {
1835                     my $install = $1;
1836                     my @x = tokenize($2);
1837                     push @engines, @x;
1838                     push @engines_install, @x unless $install;
1839                 }
1840             },
1841             qr/^\s*SCRIPTS(_NO_INST)?\s*=\s*(.*)\s*$/
1842             => sub {
1843                 if (!@skip || $skip[$#skip] > 0) {
1844                     my $install = $1;
1845                     my @x = tokenize($2);
1846                     push @scripts, @x;
1847                     push @scripts_install, @x unless $install;
1848                 }
1849             },
1850             qr/^\s*EXTRA\s*=\s*(.*)\s*$/
1851             => sub { push @extra, tokenize($1)
1852                          if !@skip || $skip[$#skip] > 0 },
1853             qr/^\s*OVERRIDES\s*=\s*(.*)\s*$/
1854             => sub { push @overrides, tokenize($1)
1855                          if !@skip || $skip[$#skip] > 0 },
1856
1857             qr/^\s*ORDINALS\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/,
1858             => sub { push @{$ordinals{$1}}, tokenize($2)
1859                          if !@skip || $skip[$#skip] > 0 },
1860             qr/^\s*SOURCE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1861             => sub { push @{$sources{$1}}, tokenize($2)
1862                          if !@skip || $skip[$#skip] > 0 },
1863             qr/^\s*SHARED_SOURCE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1864             => sub { push @{$shared_sources{$1}}, tokenize($2)
1865                          if !@skip || $skip[$#skip] > 0 },
1866             qr/^\s*INCLUDE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1867             => sub { push @{$includes{$1}}, tokenize($2)
1868                          if !@skip || $skip[$#skip] > 0 },
1869             qr/^\s*DEPEND\[((?:\\.|[^\\\]])*)\]\s*=\s*(.*)\s*$/
1870             => sub { push @{$depends{$1}}, tokenize($2)
1871                          if !@skip || $skip[$#skip] > 0 },
1872             qr/^\s*GENERATE\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1873             => sub { push @{$generate{$1}}, $2
1874                          if !@skip || $skip[$#skip] > 0 },
1875             qr/^\s*RENAME\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1876             => sub { push @{$renames{$1}}, tokenize($2)
1877                          if !@skip || $skip[$#skip] > 0 },
1878             qr/^\s*SHARED_NAME\[((?:\\.|[^\\\]])+)\]\s*=\s*(.*)\s*$/
1879             => sub { push @{$sharednames{$1}}, tokenize($2)
1880                          if !@skip || $skip[$#skip] > 0 },
1881             qr/^\s*BEGINRAW\[((?:\\.|[^\\\]])+)\]\s*$/
1882             => sub {
1883                 my $lineiterator = shift;
1884                 my $target_kind = $1;
1885                 while (defined $lineiterator->()) {
1886                     s|\R$||;
1887                     if (/^\s*ENDRAW\[((?:\\.|[^\\\]])+)\]\s*$/) {
1888                         die "ENDRAW doesn't match BEGINRAW"
1889                             if $1 ne $target_kind;
1890                         last;
1891                     }
1892                     next if @skip && $skip[$#skip] <= 0;
1893                     push @rawlines,  $_
1894                         if ($target_kind eq $target{build_file}
1895                             || $target_kind eq $target{build_file}."(".$builder_platform.")");
1896                 }
1897             },
1898             qr/^\s*(?:#.*)?$/ => sub { },
1899             "OTHERWISE" => sub { die "Something wrong with this line:\n$_\nat $sourced/$f" },
1900             "BEFORE" => sub {
1901                 if ($buildinfo_debug) {
1902                     print STDERR "DEBUG: Parsing ",join(" ", @_),"\n";
1903                     print STDERR "DEBUG: ... before parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
1904                 }
1905             },
1906             "AFTER" => sub {
1907                 if ($buildinfo_debug) {
1908                     print STDERR "DEBUG: .... after parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
1909                 }
1910             },
1911             );
1912         die "runaway IF?" if (@skip);
1913
1914         foreach (keys %renames) {
1915             die "$_ renamed to more than one thing: "
1916                 ,join(" ", @{$renames{$_}}),"\n"
1917                 if scalar @{$renames{$_}} > 1;
1918             my $dest = cleanfile($buildd, $_, $blddir);
1919             my $to = cleanfile($buildd, $renames{$_}->[0], $blddir);
1920             die "$dest renamed to more than one thing: "
1921                 ,$unified_info{rename}->{$dest}, $to
1922                 unless !defined($unified_info{rename}->{$dest})
1923                 or $unified_info{rename}->{$dest} eq $to;
1924             $unified_info{rename}->{$dest} = $to;
1925         }
1926
1927         foreach (@programs) {
1928             my $program = cleanfile($buildd, $_, $blddir);
1929             if ($unified_info{rename}->{$program}) {
1930                 $program = $unified_info{rename}->{$program};
1931             }
1932             $unified_info{programs}->{$program} = 1;
1933         }
1934
1935         foreach (@programs_install) {
1936             my $program = cleanfile($buildd, $_, $blddir);
1937             if ($unified_info{rename}->{$program}) {
1938                 $program = $unified_info{rename}->{$program};
1939             }
1940             $unified_info{install}->{programs}->{$program} = 1;
1941         }
1942
1943         foreach (@libraries) {
1944             my $library = cleanfile($buildd, $_, $blddir);
1945             if ($unified_info{rename}->{$library}) {
1946                 $library = $unified_info{rename}->{$library};
1947             }
1948             $unified_info{libraries}->{$library} = 1;
1949         }
1950
1951         foreach (@libraries_install) {
1952             my $library = cleanfile($buildd, $_, $blddir);
1953             if ($unified_info{rename}->{$library}) {
1954                 $library = $unified_info{rename}->{$library};
1955             }
1956             $unified_info{install}->{libraries}->{$library} = 1;
1957         }
1958
1959         die <<"EOF" if scalar @engines and !$config{dynamic_engines};
1960 ENGINES can only be used if configured with 'dynamic-engine'.
1961 This is usually a fault in a build.info file.
1962 EOF
1963         foreach (@engines) {
1964             my $library = cleanfile($buildd, $_, $blddir);
1965             if ($unified_info{rename}->{$library}) {
1966                 $library = $unified_info{rename}->{$library};
1967             }
1968             $unified_info{engines}->{$library} = 1;
1969         }
1970
1971         foreach (@engines_install) {
1972             my $library = cleanfile($buildd, $_, $blddir);
1973             if ($unified_info{rename}->{$library}) {
1974                 $library = $unified_info{rename}->{$library};
1975             }
1976             $unified_info{install}->{engines}->{$library} = 1;
1977         }
1978
1979         foreach (@scripts) {
1980             my $script = cleanfile($buildd, $_, $blddir);
1981             if ($unified_info{rename}->{$script}) {
1982                 $script = $unified_info{rename}->{$script};
1983             }
1984             $unified_info{scripts}->{$script} = 1;
1985         }
1986
1987         foreach (@scripts_install) {
1988             my $script = cleanfile($buildd, $_, $blddir);
1989             if ($unified_info{rename}->{$script}) {
1990                 $script = $unified_info{rename}->{$script};
1991             }
1992             $unified_info{install}->{scripts}->{$script} = 1;
1993         }
1994
1995         foreach (@extra) {
1996             my $extra = cleanfile($buildd, $_, $blddir);
1997             $unified_info{extra}->{$extra} = 1;
1998         }
1999
2000         foreach (@overrides) {
2001             my $override = cleanfile($buildd, $_, $blddir);
2002             $unified_info{overrides}->{$override} = 1;
2003         }
2004
2005         push @{$unified_info{rawlines}}, @rawlines;
2006
2007         unless ($disabled{shared}) {
2008             # Check sharednames.
2009             foreach (keys %sharednames) {
2010                 my $dest = cleanfile($buildd, $_, $blddir);
2011                 if ($unified_info{rename}->{$dest}) {
2012                     $dest = $unified_info{rename}->{$dest};
2013                 }
2014                 die "shared_name for $dest with multiple values: "
2015                     ,join(" ", @{$sharednames{$_}}),"\n"
2016                     if scalar @{$sharednames{$_}} > 1;
2017                 my $to = cleanfile($buildd, $sharednames{$_}->[0], $blddir);
2018                 die "shared_name found for a library $dest that isn't defined\n"
2019                     unless $unified_info{libraries}->{$dest};
2020                 die "shared_name for $dest with multiple values: "
2021                     ,$unified_info{sharednames}->{$dest}, ", ", $to
2022                     unless !defined($unified_info{sharednames}->{$dest})
2023                     or $unified_info{sharednames}->{$dest} eq $to;
2024                 $unified_info{sharednames}->{$dest} = $to;
2025             }
2026
2027             # Additionally, we set up sharednames for libraries that don't
2028             # have any, as themselves.  Only for libraries that aren't
2029             # explicitly static.
2030             foreach (grep !/\.a$/, keys %{$unified_info{libraries}}) {
2031                 if (!defined $unified_info{sharednames}->{$_}) {
2032                     $unified_info{sharednames}->{$_} = $_
2033                 }
2034             }
2035
2036             # Check that we haven't defined any library as both shared and
2037             # explicitly static.  That is forbidden.
2038             my @doubles = ();
2039             foreach (grep /\.a$/, keys %{$unified_info{libraries}}) {
2040                 (my $l = $_) =~ s/\.a$//;
2041                 push @doubles, $l if defined $unified_info{sharednames}->{$l};
2042             }
2043             die "these libraries are both explicitly static and shared:\n  ",
2044                 join(" ", @doubles), "\n"
2045                 if @doubles;
2046         }
2047
2048         foreach (keys %sources) {
2049             my $dest = $_;
2050             my $ddest = cleanfile($buildd, $_, $blddir);
2051             if ($unified_info{rename}->{$ddest}) {
2052                 $ddest = $unified_info{rename}->{$ddest};
2053             }
2054             foreach (@{$sources{$dest}}) {
2055                 my $s = cleanfile($sourced, $_, $blddir);
2056
2057                 # If it isn't in the source tree, we assume it's generated
2058                 # in the build tree
2059                 if ($s eq $src_configdata || ! -f $s || $generate{$_}) {
2060                     $s = cleanfile($buildd, $_, $blddir);
2061                 }
2062                 # We recognise C++, C and asm files
2063                 if ($s =~ /\.(cc|cpp|c|s|S)$/) {
2064                     my $o = $_;
2065                     $o =~ s/\.[csS]$/.o/; # C and assembler
2066                     $o =~ s/\.(cc|cpp)$/_cc.o/; # C++
2067                     $o = cleanfile($buildd, $o, $blddir);
2068                     $unified_info{sources}->{$ddest}->{$o} = 1;
2069                     $unified_info{sources}->{$o}->{$s} = 1;
2070                 } elsif ($s =~ /\.rc$/) {
2071                     # We also recognise resource files
2072                     my $o = $_;
2073                     $o =~ s/\.rc$/.res/; # Resource configuration
2074                     my $o = cleanfile($buildd, $o, $blddir);
2075                     $unified_info{sources}->{$ddest}->{$o} = 1;
2076                     $unified_info{sources}->{$o}->{$s} = 1;
2077                 } else {
2078                     $unified_info{sources}->{$ddest}->{$s} = 1;
2079                 }
2080             }
2081         }
2082
2083         foreach (keys %shared_sources) {
2084             my $dest = $_;
2085             my $ddest = cleanfile($buildd, $_, $blddir);
2086             if ($unified_info{rename}->{$ddest}) {
2087                 $ddest = $unified_info{rename}->{$ddest};
2088             }
2089             foreach (@{$shared_sources{$dest}}) {
2090                 my $s = cleanfile($sourced, $_, $blddir);
2091
2092                 # If it isn't in the source tree, we assume it's generated
2093                 # in the build tree
2094                 if ($s eq $src_configdata || ! -f $s || $generate{$_}) {
2095                     $s = cleanfile($buildd, $_, $blddir);
2096                 }
2097
2098                 if ($s =~ /\.(cc|cpp|c|s|S)$/) {
2099                     # We recognise C++, C and asm files
2100                     my $o = $_;
2101                     $o =~ s/\.[csS]$/.o/; # C and assembler
2102                     $o =~ s/\.(cc|cpp)$/_cc.o/; # C++
2103                     $o = cleanfile($buildd, $o, $blddir);
2104                     $unified_info{shared_sources}->{$ddest}->{$o} = 1;
2105                     $unified_info{sources}->{$o}->{$s} = 1;
2106                 } elsif ($s =~ /\.rc$/) {
2107                     # We also recognise resource files
2108                     my $o = $_;
2109                     $o =~ s/\.rc$/.res/; # Resource configuration
2110                     my $o = cleanfile($buildd, $o, $blddir);
2111                     $unified_info{shared_sources}->{$ddest}->{$o} = 1;
2112                     $unified_info{sources}->{$o}->{$s} = 1;
2113                 } elsif ($s =~ /\.(def|map|opt)$/) {
2114                     # We also recognise .def / .map / .opt files
2115                     # We know they are generated files
2116                     my $def = cleanfile($buildd, $s, $blddir);
2117                     $unified_info{shared_sources}->{$ddest}->{$def} = 1;
2118                 } else {
2119                     die "unrecognised source file type for shared library: $s\n";
2120                 }
2121             }
2122         }
2123
2124         foreach (keys %generate) {
2125             my $dest = $_;
2126             my $ddest = cleanfile($buildd, $_, $blddir);
2127             if ($unified_info{rename}->{$ddest}) {
2128                 $ddest = $unified_info{rename}->{$ddest};
2129             }
2130             die "more than one generator for $dest: "
2131                     ,join(" ", @{$generate{$_}}),"\n"
2132                     if scalar @{$generate{$_}} > 1;
2133             my @generator = split /\s+/, $generate{$dest}->[0];
2134             $generator[0] = cleanfile($sourced, $generator[0], $blddir),
2135             $unified_info{generate}->{$ddest} = [ @generator ];
2136         }
2137
2138         foreach (keys %depends) {
2139             my $dest = $_;
2140             my $ddest = $dest eq "" ? "" : cleanfile($sourced, $_, $blddir);
2141
2142             # If the destination doesn't exist in source, it can only be
2143             # a generated file in the build tree.
2144             if ($ddest ne "" && ($ddest eq $src_configdata || ! -f $ddest)) {
2145                 $ddest = cleanfile($buildd, $_, $blddir);
2146                 if ($unified_info{rename}->{$ddest}) {
2147                     $ddest = $unified_info{rename}->{$ddest};
2148                 }
2149             }
2150             foreach (@{$depends{$dest}}) {
2151                 my $d = cleanfile($sourced, $_, $blddir);
2152
2153                 # If we know it's generated, or assume it is because we can't
2154                 # find it in the source tree, we set file we depend on to be
2155                 # in the build tree rather than the source tree, and assume
2156                 # and that there are lines to build it in a BEGINRAW..ENDRAW
2157                 # section or in the Makefile template.
2158                 if ($d eq $src_configdata
2159                     || ! -f $d
2160                     || (grep { $d eq $_ }
2161                         map { cleanfile($srcdir, $_, $blddir) }
2162                         grep { /\.h$/ } keys %{$unified_info{generate}})) {
2163                     $d = cleanfile($buildd, $_, $blddir);
2164                 }
2165                 # Take note if the file to depend on is being renamed
2166                 # Take extra care with files ending with .a, they should
2167                 # be treated without that extension, and the extension
2168                 # should be added back after treatment.
2169                 $d =~ /(\.a)?$/;
2170                 my $e = $1 // "";
2171                 $d = $`;
2172                 if ($unified_info{rename}->{$d}) {
2173                     $d = $unified_info{rename}->{$d};
2174                 }
2175                 $d .= $e;
2176                 $unified_info{depends}->{$ddest}->{$d} = 1;
2177             }
2178         }
2179
2180         foreach (keys %includes) {
2181             my $dest = $_;
2182             my $ddest = cleanfile($sourced, $_, $blddir);
2183
2184             # If the destination doesn't exist in source, it can only be
2185             # a generated file in the build tree.
2186             if ($ddest eq $src_configdata || ! -f $ddest) {
2187                 $ddest = cleanfile($buildd, $_, $blddir);
2188                 if ($unified_info{rename}->{$ddest}) {
2189                     $ddest = $unified_info{rename}->{$ddest};
2190                 }
2191             }
2192             foreach (@{$includes{$dest}}) {
2193                 my $is = cleandir($sourced, $_, $blddir);
2194                 my $ib = cleandir($buildd, $_, $blddir);
2195                 push @{$unified_info{includes}->{$ddest}->{source}}, $is
2196                     unless grep { $_ eq $is } @{$unified_info{includes}->{$ddest}->{source}};
2197                 push @{$unified_info{includes}->{$ddest}->{build}}, $ib
2198                     unless grep { $_ eq $ib } @{$unified_info{includes}->{$ddest}->{build}};
2199             }
2200         }
2201     }
2202
2203     my $ordinals_text = join(', ', sort keys %ordinals);
2204     warn <<"EOF" if $ordinals_text;
2205
2206 WARNING: ORDINALS were specified for $ordinals_text
2207 They are ignored and should be replaced with a combination of GENERATE,
2208 DEPEND and SHARED_SOURCE.
2209 EOF
2210
2211     # Massage the result
2212
2213     # If the user configured no-shared, we allow no shared sources
2214     if ($disabled{shared}) {
2215         foreach (keys %{$unified_info{shared_sources}}) {
2216             foreach (keys %{$unified_info{shared_sources}->{$_}}) {
2217                 delete $unified_info{sources}->{$_};
2218             }
2219         }
2220         $unified_info{shared_sources} = {};
2221     }
2222
2223     # If we depend on a header file or a perl module, add an inclusion of
2224     # its directory to allow smoothe inclusion
2225     foreach my $dest (keys %{$unified_info{depends}}) {
2226         next if $dest eq "";
2227         foreach my $d (keys %{$unified_info{depends}->{$dest}}) {
2228             next unless $d =~ /\.(h|pm)$/;
2229             my $i = dirname($d);
2230             my $spot =
2231                 $d eq "configdata.pm" || defined($unified_info{generate}->{$d})
2232                 ? 'build' : 'source';
2233             push @{$unified_info{includes}->{$dest}->{$spot}}, $i
2234                 unless grep { $_ eq $i } @{$unified_info{includes}->{$dest}->{$spot}};
2235         }
2236     }
2237
2238     # Trickle down includes placed on libraries, engines and programs to
2239     # their sources (i.e. object files)
2240     foreach my $dest (keys %{$unified_info{engines}},
2241                       keys %{$unified_info{libraries}},
2242                       keys %{$unified_info{programs}}) {
2243         foreach my $k (("source", "build")) {
2244             next unless defined($unified_info{includes}->{$dest}->{$k});
2245             my @incs = reverse @{$unified_info{includes}->{$dest}->{$k}};
2246             foreach my $obj (grep /\.o$/,
2247                              (keys %{$unified_info{sources}->{$dest} // {}},
2248                               keys %{$unified_info{shared_sources}->{$dest} // {}})) {
2249                 foreach my $inc (@incs) {
2250                     unshift @{$unified_info{includes}->{$obj}->{$k}}, $inc
2251                         unless grep { $_ eq $inc } @{$unified_info{includes}->{$obj}->{$k}};
2252                 }
2253             }
2254         }
2255         delete $unified_info{includes}->{$dest};
2256     }
2257
2258     ### Make unified_info a bit more efficient
2259     # One level structures
2260     foreach (("programs", "libraries", "engines", "scripts", "extra", "overrides")) {
2261         $unified_info{$_} = [ sort keys %{$unified_info{$_}} ];
2262     }
2263     # Two level structures
2264     foreach my $l1 (("install", "sources", "shared_sources", "ldadd", "depends")) {
2265         foreach my $l2 (sort keys %{$unified_info{$l1}}) {
2266             $unified_info{$l1}->{$l2} =
2267                 [ sort keys %{$unified_info{$l1}->{$l2}} ];
2268         }
2269     }
2270     # Includes
2271     foreach my $dest (sort keys %{$unified_info{includes}}) {
2272         if (defined($unified_info{includes}->{$dest}->{build})) {
2273             my @source_includes = ();
2274             @source_includes = ( @{$unified_info{includes}->{$dest}->{source}} )
2275                 if defined($unified_info{includes}->{$dest}->{source});
2276             $unified_info{includes}->{$dest} =
2277                 [ @{$unified_info{includes}->{$dest}->{build}} ];
2278             foreach my $inc (@source_includes) {
2279                 push @{$unified_info{includes}->{$dest}}, $inc
2280                     unless grep { $_ eq $inc } @{$unified_info{includes}->{$dest}};
2281             }
2282         } else {
2283             $unified_info{includes}->{$dest} =
2284                 [ @{$unified_info{includes}->{$dest}->{source}} ];
2285         }
2286     }
2287
2288     # For convenience collect information regarding directories where
2289     # files are generated, those generated files and the end product
2290     # they end up in where applicable.  Then, add build rules for those
2291     # directories
2292     my %loopinfo = ( "lib" => [ @{$unified_info{libraries}} ],
2293                      "dso" => [ @{$unified_info{engines}} ],
2294                      "bin" => [ @{$unified_info{programs}} ],
2295                      "script" => [ @{$unified_info{scripts}} ] );
2296     foreach my $type (keys %loopinfo) {
2297         foreach my $product (@{$loopinfo{$type}}) {
2298             my %dirs = ();
2299             my $pd = dirname($product);
2300
2301             foreach (@{$unified_info{sources}->{$product} // []},
2302                      @{$unified_info{shared_sources}->{$product} // []}) {
2303                 my $d = dirname($_);
2304
2305                 # We don't want to create targets for source directories
2306                 # when building out of source
2307                 next if ($config{sourcedir} ne $config{builddir}
2308                              && $d =~ m|^\Q$config{sourcedir}\E|);
2309                 # We already have a "test" target, and the current directory
2310                 # is just silly to make a target for
2311                 next if $d eq "test" || $d eq ".";
2312
2313                 $dirs{$d} = 1;
2314                 push @{$unified_info{dirinfo}->{$d}->{deps}}, $_
2315                     if $d ne $pd;
2316             }
2317             foreach (keys %dirs) {
2318                 push @{$unified_info{dirinfo}->{$_}->{products}->{$type}},
2319                     $product;
2320             }
2321         }
2322     }
2323 }
2324
2325 # For the schemes that need it, we provide the old *_obj configs
2326 # from the *_asm_obj ones
2327 foreach (grep /_(asm|aux)_src$/, keys %target) {
2328     my $src = $_;
2329     (my $obj = $_) =~ s/_(asm|aux)_src$/_obj/;
2330     $target{$obj} = $target{$src};
2331     $target{$obj} =~ s/\.[csS]\b/.o/g; # C and assembler
2332     $target{$obj} =~ s/\.(cc|cpp)\b/_cc.o/g; # C++
2333 }
2334
2335 # Write down our configuration where it fits #########################
2336
2337 print "Creating configdata.pm\n";
2338 open(OUT,">configdata.pm") || die "unable to create configdata.pm: $!\n";
2339 print OUT <<"EOF";
2340 #! $config{HASHBANGPERL}
2341
2342 package configdata;
2343
2344 use strict;
2345 use warnings;
2346
2347 use Exporter;
2348 #use vars qw(\@ISA \@EXPORT);
2349 our \@ISA = qw(Exporter);
2350 our \@EXPORT = qw(\%config \%target \%disabled \%withargs \%unified_info \@disablables);
2351
2352 EOF
2353 print OUT "our %config = (\n";
2354 foreach (sort keys %config) {
2355     if (ref($config{$_}) eq "ARRAY") {
2356         print OUT "  ", $_, " => [ ", join(", ",
2357                                            map { quotify("perl", $_) }
2358                                            @{$config{$_}}), " ],\n";
2359     } elsif (ref($config{$_}) eq "HASH") {
2360         print OUT "  ", $_, " => {";
2361         if (scalar keys %{$config{$_}} > 0) {
2362             print OUT "\n";
2363             foreach my $key (sort keys %{$config{$_}}) {
2364                 print OUT "      ",
2365                     join(" => ",
2366                          quotify("perl", $key),
2367                          defined $config{$_}->{$key}
2368                              ? quotify("perl", $config{$_}->{$key})
2369                              : "undef");
2370                 print OUT ",\n";
2371             }
2372             print OUT "  ";
2373         }
2374         print OUT "},\n";
2375     } else {
2376         print OUT "  ", $_, " => ", quotify("perl", $config{$_}), ",\n"
2377     }
2378 }
2379 print OUT <<"EOF";
2380 );
2381
2382 EOF
2383 print OUT "our %target = (\n";
2384 foreach (sort keys %target) {
2385     if (ref($target{$_}) eq "ARRAY") {
2386         print OUT "  ", $_, " => [ ", join(", ",
2387                                            map { quotify("perl", $_) }
2388                                            @{$target{$_}}), " ],\n";
2389     } else {
2390         print OUT "  ", $_, " => ", quotify("perl", $target{$_}), ",\n"
2391     }
2392 }
2393 print OUT <<"EOF";
2394 );
2395
2396 EOF
2397 print OUT "our \%available_protocols = (\n";
2398 print OUT "  tls => [ ", join(", ", map { quotify("perl", $_) } @tls), " ],\n";
2399 print OUT "  dtls => [ ", join(", ", map { quotify("perl", $_) } @dtls), " ],\n";
2400 print OUT <<"EOF";
2401 );
2402
2403 EOF
2404 print OUT "our \@disablables = (\n";
2405 foreach (@disablables) {
2406     print OUT "  ", quotify("perl", $_), ",\n";
2407 }
2408 print OUT <<"EOF";
2409 );
2410
2411 EOF
2412 print OUT "our \%disabled = (\n";
2413 foreach (sort keys %disabled) {
2414     print OUT "  ", quotify("perl", $_), " => ", quotify("perl", $disabled{$_}), ",\n";
2415 }
2416 print OUT <<"EOF";
2417 );
2418
2419 EOF
2420 print OUT "our %withargs = (\n";
2421 foreach (sort keys %withargs) {
2422     if (ref($withargs{$_}) eq "ARRAY") {
2423         print OUT "  ", $_, " => [ ", join(", ",
2424                                            map { quotify("perl", $_) }
2425                                            @{$withargs{$_}}), " ],\n";
2426     } else {
2427         print OUT "  ", $_, " => ", quotify("perl", $withargs{$_}), ",\n"
2428     }
2429 }
2430 print OUT <<"EOF";
2431 );
2432
2433 EOF
2434 if ($builder eq "unified") {
2435     my $recurse;
2436     $recurse = sub {
2437         my $indent = shift;
2438         foreach (@_) {
2439             if (ref $_ eq "ARRAY") {
2440                 print OUT " "x$indent, "[\n";
2441                 foreach (@$_) {
2442                     $recurse->($indent + 4, $_);
2443                 }
2444                 print OUT " "x$indent, "],\n";
2445             } elsif (ref $_ eq "HASH") {
2446                 my %h = %$_;
2447                 print OUT " "x$indent, "{\n";
2448                 foreach (sort keys %h) {
2449                     if (ref $h{$_} eq "") {
2450                         print OUT " "x($indent + 4), quotify("perl", $_), " => ", quotify("perl", $h{$_}), ",\n";
2451                     } else {
2452                         print OUT " "x($indent + 4), quotify("perl", $_), " =>\n";
2453                         $recurse->($indent + 8, $h{$_});
2454                     }
2455                 }
2456                 print OUT " "x$indent, "},\n";
2457             } else {
2458                 print OUT " "x$indent, quotify("perl", $_), ",\n";
2459             }
2460         }
2461     };
2462     print OUT "our %unified_info = (\n";
2463     foreach (sort keys %unified_info) {
2464         if (ref $unified_info{$_} eq "") {
2465             print OUT " "x4, quotify("perl", $_), " => ", quotify("perl", $unified_info{$_}), ",\n";
2466         } else {
2467             print OUT " "x4, quotify("perl", $_), " =>\n";
2468             $recurse->(8, $unified_info{$_});
2469         }
2470     }
2471     print OUT <<"EOF";
2472 );
2473
2474 EOF
2475 }
2476 print OUT
2477     "# The following data is only used when this files is use as a script\n";
2478 print OUT "my \@makevars = (\n";
2479 foreach (sort keys %user) {
2480     print OUT "    '",$_,"',\n";
2481 }
2482 print OUT ");\n";
2483 print OUT "my \%disabled_info = (\n";
2484 foreach my $what (sort keys %disabled_info) {
2485     print OUT "    '$what' => {\n";
2486     foreach my $info (sort keys %{$disabled_info{$what}}) {
2487         if (ref $disabled_info{$what}->{$info} eq 'ARRAY') {
2488             print OUT "        $info => [ ",
2489                 join(', ', map { "'$_'" } @{$disabled_info{$what}->{$info}}),
2490                 " ],\n";
2491         } else {
2492             print OUT "        $info => '", $disabled_info{$what}->{$info},
2493                 "',\n";
2494         }
2495     }
2496     print OUT "    },\n";
2497 }
2498 print OUT ");\n";
2499 print OUT 'my @user_crossable = qw( ', join (' ', @user_crossable), " );\n";
2500 print OUT << 'EOF';
2501 # If run directly, we can give some answers, and even reconfigure
2502 unless (caller) {
2503     use Getopt::Long;
2504     use File::Spec::Functions;
2505     use File::Basename;
2506     use Pod::Usage;
2507
2508     my $here = dirname($0);
2509
2510     my $dump = undef;
2511     my $cmdline = undef;
2512     my $options = undef;
2513     my $target = undef;
2514     my $envvars = undef;
2515     my $makevars = undef;
2516     my $buildparams = undef;
2517     my $reconf = undef;
2518     my $verbose = undef;
2519     my $help = undef;
2520     my $man = undef;
2521     GetOptions('dump|d'                 => \$dump,
2522                'command-line|c'         => \$cmdline,
2523                'options|o'              => \$options,
2524                'target|t'               => \$target,
2525                'environment|e'          => \$envvars,
2526                'make-variables|m'       => \$makevars,
2527                'build-parameters|b'     => \$buildparams,
2528                'reconfigure|reconf|r'   => \$reconf,
2529                'verbose|v'              => \$verbose,
2530                'help'                   => \$help,
2531                'man'                    => \$man)
2532         or die "Errors in command line arguments\n";
2533
2534     unless ($dump || $cmdline || $options || $target || $envvars || $makevars
2535             || $buildparams || $reconf || $verbose || $help || $man) {
2536         print STDERR <<"_____";
2537 You must give at least one option.
2538 For more information, do '$0 --help'
2539 _____
2540         exit(2);
2541     }
2542
2543     if ($help) {
2544         pod2usage(-exitval => 0,
2545                   -verbose => 1);
2546     }
2547     if ($man) {
2548         pod2usage(-exitval => 0,
2549                   -verbose => 2);
2550     }
2551     if ($dump || $cmdline) {
2552         print "\nCommand line (with current working directory = $here):\n\n";
2553         print '    ',join(' ',
2554                           $config{PERL},
2555                           catfile($config{sourcedir}, 'Configure'),
2556                           @{$config{perlargv}}), "\n";
2557         print "\nPerl information:\n\n";
2558         print '    ',$config{perl_cmd},"\n";
2559         print '    ',$config{perl_version},' for ',$config{perl_archname},"\n";
2560     }
2561     if ($dump || $options) {
2562         my $longest = 0;
2563         my $longest2 = 0;
2564         foreach my $what (@disablables) {
2565             $longest = length($what) if $longest < length($what);
2566             $longest2 = length($disabled{$what})
2567                 if $disabled{$what} && $longest2 < length($disabled{$what});
2568         }
2569         print "\nEnabled features:\n\n";
2570         foreach my $what (@disablables) {
2571             print "    $what\n" unless $disabled{$what};
2572         }
2573         print "\nDisabled features:\n\n";
2574         foreach my $what (@disablables) {
2575             if ($disabled{$what}) {
2576                 print "    $what", ' ' x ($longest - length($what) + 1),
2577                     "[$disabled{$what}]", ' ' x ($longest2 - length($disabled{$what}) + 1);
2578                 print $disabled_info{$what}->{macro}
2579                     if $disabled_info{$what}->{macro};
2580                 print ' (skip ',
2581                     join(', ', @{$disabled_info{$what}->{skipped}}),
2582                     ')'
2583                     if $disabled_info{$what}->{skipped};
2584                 print "\n";
2585             }
2586         }
2587     }
2588     if ($dump || $target) {
2589         print "\nConfig target attributes:\n\n";
2590         foreach (sort keys %target) {
2591             next if $_ =~ m|^_| || $_ eq 'template';
2592             my $quotify = sub {
2593                 map { (my $x = $_) =~ s|([\\\$\@"])|\\$1|g; "\"$x\""} @_;
2594             };
2595             print '    ', $_, ' => ';
2596             if (ref($target{$_}) eq "ARRAY") {
2597                 print '[ ', join(', ', $quotify->(@{$target{$_}})), " ],\n";
2598             } else {
2599                 print $quotify->($target{$_}), ",\n"
2600             }
2601         }
2602     }
2603     if ($dump || $envvars) {
2604         print "\nRecorded environment:\n\n";
2605         foreach (sort keys %{$config{perlenv}}) {
2606             print '    ',$_,' = ',($config{perlenv}->{$_} || ''),"\n";
2607         }
2608     }
2609     if ($dump || $makevars) {
2610         print "\nMakevars:\n\n";
2611         foreach my $var (@makevars) {
2612             my $prefix = '';
2613             $prefix = $config{CROSS_COMPILE}
2614                 if grep { $var eq $_ } @user_crossable;
2615             $prefix //= '';
2616             print '    ',$var,' ' x (16 - length $var),'= ',
2617                 (ref $config{$var} eq 'ARRAY'
2618                  ? join(' ', @{$config{$var}})
2619                  : $prefix.$config{$var}),
2620                 "\n"
2621                 if defined $config{$var};
2622         }
2623
2624         my @buildfile = ($config{builddir}, $config{build_file});
2625         unshift @buildfile, $here
2626             unless file_name_is_absolute($config{builddir});
2627         my $buildfile = canonpath(catdir(@buildfile));
2628         print <<"_____";
2629
2630 NOTE: These variables only represent the configuration view.  The build file
2631 template may have processed these variables further, please have a look at the
2632 build file for more exact data:
2633     $buildfile
2634 _____
2635     }
2636     if ($dump || $buildparams) {
2637         my @buildfile = ($config{builddir}, $config{build_file});
2638         unshift @buildfile, $here
2639             unless file_name_is_absolute($config{builddir});
2640         print "\nbuild file:\n\n";
2641         print "    ", canonpath(catfile(@buildfile)),"\n";
2642
2643         print "\nbuild file templates:\n\n";
2644         foreach (@{$config{build_file_templates}}) {
2645             my @tmpl = ($_);
2646             unshift @tmpl, $here
2647                 unless file_name_is_absolute($config{sourcedir});
2648             print '    ',canonpath(catfile(@tmpl)),"\n";
2649         }
2650     }
2651     if ($reconf) {
2652         if ($verbose) {
2653             print 'Reconfiguring with: ', join(' ',@{$config{perlargv}}), "\n";
2654             foreach (sort keys %{$config{perlenv}}) {
2655                 print '    ',$_,' = ',($config{perlenv}->{$_} || ""),"\n";
2656             }
2657         }
2658
2659         chdir $here;
2660         exec $^X,catfile($config{sourcedir}, 'Configure'),'reconf';
2661     }
2662 }
2663
2664 1;
2665
2666 __END__
2667
2668 =head1 NAME
2669
2670 configdata.pm - configuration data for OpenSSL builds
2671
2672 =head1 SYNOPSIS
2673
2674 Interactive:
2675
2676   perl configdata.pm [options]
2677
2678 As data bank module:
2679
2680   use configdata;
2681
2682 =head1 DESCRIPTION
2683
2684 This module can be used in two modes, interactively and as a module containing
2685 all the data recorded by OpenSSL's Configure script.
2686
2687 When used interactively, simply run it as any perl script, with at least one
2688 option, and you will get the information you ask for.  See L</OPTIONS> below.
2689
2690 When loaded as a module, you get a few databanks with useful information to
2691 perform build related tasks.  The databanks are:
2692
2693     %config             Configured things.
2694     %target             The OpenSSL config target with all inheritances
2695                         resolved.
2696     %disabled           The features that are disabled.
2697     @disablables        The list of features that can be disabled.
2698     %withargs           All data given through --with-THING options.
2699     %unified_info       All information that was computed from the build.info
2700                         files.
2701
2702 =head1 OPTIONS
2703
2704 =over 4
2705
2706 =item B<--help>
2707
2708 Print a brief help message and exit.
2709
2710 =item B<--man>
2711
2712 Print the manual page and exit.
2713
2714 =item B<--dump> | B<-d>
2715
2716 Print all relevant configuration data.  This is equivalent to B<--command-line>
2717 B<--options> B<--target> B<--environment> B<--make-variables>
2718 B<--build-parameters>.
2719
2720 =item B<--command-line> | B<-c>
2721
2722 Print the current configuration command line.
2723
2724 =item B<--options> | B<-o>
2725
2726 Print the features, both enabled and disabled, and display defined macro and
2727 skipped directories where applicable.
2728
2729 =item B<--target> | B<-t>
2730
2731 Print the config attributes for this config target.
2732
2733 =item B<--environment> | B<-e>
2734
2735 Print the environment variables and their values at the time of configuration.
2736
2737 =item B<--make-variables> | B<-m>
2738
2739 Print the main make variables generated in the current configuration
2740
2741 =item B<--build-parameters> | B<-b>
2742
2743 Print the build parameters, i.e. build file and build file templates.
2744
2745 =item B<--reconfigure> | B<--reconf> | B<-r>
2746
2747 Redo the configuration.
2748
2749 =item B<--verbose> | B<-v>
2750
2751 Verbose output.
2752
2753 =back
2754
2755 =cut
2756
2757 EOF
2758 close(OUT);
2759 if ($builder_platform eq 'unix') {
2760     my $mode = (0755 & ~umask);
2761     chmod $mode, 'configdata.pm'
2762         or warn sprintf("WARNING: Couldn't change mode for 'configdata.pm' to 0%03o: %s\n",$mode,$!);
2763 }
2764
2765 my %builders = (
2766     unified => sub {
2767         print 'Creating ',$target{build_file},"\n";
2768         run_dofile(catfile($blddir, $target{build_file}),
2769                    @{$config{build_file_templates}});
2770     },
2771     );
2772
2773 $builders{$builder}->($builder_platform, @builder_opts);
2774
2775 $SIG{__DIE__} = $orig_death_handler;
2776
2777 print <<"EOF" if ($disabled{threads} eq "unavailable");
2778
2779 The library could not be configured for supporting multi-threaded
2780 applications as the compiler options required on this system are not known.
2781 See file INSTALL for details if you need multi-threading.
2782 EOF
2783
2784 print <<"EOF" if ($no_shared_warn);
2785
2786 The options 'shared', 'pic' and 'dynamic-engine' aren't supported on this
2787 platform, so we will pretend you gave the option 'no-pic', which also disables
2788 'shared' and 'dynamic-engine'.  If you know how to implement shared libraries
2789 or position independent code, please let us know (but please first make sure
2790 you have tried with a current version of OpenSSL).
2791 EOF
2792
2793 print <<"EOF";
2794
2795 **********************************************************************
2796 ***                                                                ***
2797 ***   OpenSSL has been successfully configured                     ***
2798 ***                                                                ***
2799 ***   If you encounter a problem while building, please open an    ***
2800 ***   issue on GitHub <https://github.com/openssl/openssl/issues>  ***
2801 ***   and include the output from the following command:           ***
2802 ***                                                                ***
2803 ***       perl configdata.pm --dump                                ***
2804 ***                                                                ***
2805 ***   (If you are new to OpenSSL, you might want to consult the    ***
2806 ***   'Troubleshooting' section in the INSTALL file first)         ***
2807 ***                                                                ***
2808 **********************************************************************
2809 EOF
2810
2811 exit(0);
2812
2813 ######################################################################
2814 #
2815 # Helpers and utility functions
2816 #
2817
2818 # Death handler, to print a helpful message in case of failure #######
2819 #
2820 sub death_handler {
2821     die @_ if $^S;              # To prevent the added message in eval blocks
2822     my $build_file = $target{build_file} // "build file";
2823     my @message = ( <<"_____", @_ );
2824
2825 Failure!  $build_file wasn't produced.
2826 Please read INSTALL and associated NOTES files.  You may also have to look over
2827 your available compiler tool chain or change your configuration.
2828
2829 _____
2830
2831     # Dying is terminal, so it's ok to reset the signal handler here.
2832     $SIG{__DIE__} = $orig_death_handler;
2833     die @message;
2834 }
2835
2836 # Configuration file reading #########################################
2837
2838 # Note: All of the helper functions are for lazy evaluation.  They all
2839 # return a CODE ref, which will return the intended value when evaluated.
2840 # Thus, whenever there's mention of a returned value, it's about that
2841 # intended value.
2842
2843 # Helper function to implement conditional inheritance depending on the
2844 # value of $disabled{asm}.  Used in inherit_from values as follows:
2845 #
2846 #      inherit_from => [ "template", asm("asm_tmpl") ]
2847 #
2848 sub asm {
2849     my @x = @_;
2850     sub {
2851         $disabled{asm} ? () : @x;
2852     }
2853 }
2854
2855 # Helper function to implement conditional value variants, with a default
2856 # plus additional values based on the value of $config{build_type}.
2857 # Arguments are given in hash table form:
2858 #
2859 #       picker(default => "Basic string: ",
2860 #              debug   => "debug",
2861 #              release => "release")
2862 #
2863 # When configuring with --debug, the resulting string will be
2864 # "Basic string: debug", and when not, it will be "Basic string: release"
2865 #
2866 # This can be used to create variants of sets of flags according to the
2867 # build type:
2868 #
2869 #       cflags => picker(default => "-Wall",
2870 #                        debug   => "-g -O0",
2871 #                        release => "-O3")
2872 #
2873 sub picker {
2874     my %opts = @_;
2875     return sub { add($opts{default} || (),
2876                      $opts{$config{build_type}} || ())->(); }
2877 }
2878
2879 # Helper function to combine several values of different types into one.
2880 # This is useful if you want to combine a string with the result of a
2881 # lazy function, such as:
2882 #
2883 #       cflags => combine("-Wall", sub { $disabled{zlib} ? () : "-DZLIB" })
2884 #
2885 sub combine {
2886     my @stuff = @_;
2887     return sub { add(@stuff)->(); }
2888 }
2889
2890 # Helper function to implement conditional values depending on the value
2891 # of $disabled{threads}.  Can be used as follows:
2892 #
2893 #       cflags => combine("-Wall", threads("-pthread"))
2894 #
2895 sub threads {
2896     my @flags = @_;
2897     return sub { add($disabled{threads} ? () : @flags)->(); }
2898 }
2899
2900 sub shared {
2901     my @flags = @_;
2902     return sub { add($disabled{shared} ? () : @flags)->(); }
2903 }
2904
2905 our $add_called = 0;
2906 # Helper function to implement adding values to already existing configuration
2907 # values.  It handles elements that are ARRAYs, CODEs and scalars
2908 sub _add {
2909     my $separator = shift;
2910
2911     # If there's any ARRAY in the collection of values OR the separator
2912     # is undef, we will return an ARRAY of combined values, otherwise a
2913     # string of joined values with $separator as the separator.
2914     my $found_array = !defined($separator);
2915
2916     my @values =
2917         map {
2918             my $res = $_;
2919             while (ref($res) eq "CODE") {
2920                 $res = $res->();
2921             }
2922             if (defined($res)) {
2923                 if (ref($res) eq "ARRAY") {
2924                     $found_array = 1;
2925                     @$res;
2926                 } else {
2927                     $res;
2928                 }
2929             } else {
2930                 ();
2931             }
2932     } (@_);
2933
2934     $add_called = 1;
2935
2936     if ($found_array) {
2937         [ @values ];
2938     } else {
2939         join($separator, grep { defined($_) && $_ ne "" } @values);
2940     }
2941 }
2942 sub add_before {
2943     my $separator = " ";
2944     if (ref($_[$#_]) eq "HASH") {
2945         my $opts = pop;
2946         $separator = $opts->{separator};
2947     }
2948     my @x = @_;
2949     sub { _add($separator, @x, @_) };
2950 }
2951 sub add {
2952     my $separator = " ";
2953     if (ref($_[$#_]) eq "HASH") {
2954         my $opts = pop;
2955         $separator = $opts->{separator};
2956     }
2957     my @x = @_;
2958     sub { _add($separator, @_, @x) };
2959 }
2960
2961 sub read_eval_file {
2962     my $fname = shift;
2963     my $content;
2964     my @result;
2965
2966     open F, "< $fname" or die "Can't open '$fname': $!\n";
2967     {
2968         undef local $/;
2969         $content = <F>;
2970     }
2971     close F;
2972     {
2973         local $@;
2974
2975         @result = ( eval $content );
2976         warn $@ if $@;
2977     }
2978     return wantarray ? @result : $result[0];
2979 }
2980
2981 # configuration reader, evaluates the input file as a perl script and expects
2982 # it to fill %targets with target configurations.  Those are then added to
2983 # %table.
2984 sub read_config {
2985     my $fname = shift;
2986     my %targets;
2987
2988     {
2989         # Protect certain tables from tampering
2990         local %table = ();
2991
2992         %targets = read_eval_file($fname);
2993     }
2994     my %preexisting = ();
2995     foreach (sort keys %targets) {
2996         $preexisting{$_} = 1 if $table{$_};
2997     }
2998     die <<"EOF",
2999 The following config targets from $fname
3000 shadow pre-existing config targets with the same name:
3001 EOF
3002         map { "  $_\n" } sort keys %preexisting
3003         if %preexisting;
3004
3005
3006     # For each target, check that it's configured with a hash table.
3007     foreach (keys %targets) {
3008         if (ref($targets{$_}) ne "HASH") {
3009             if (ref($targets{$_}) eq "") {
3010                 warn "Deprecated target configuration for $_, ignoring...\n";
3011             } else {
3012                 warn "Misconfigured target configuration for $_ (should be a hash table), ignoring...\n";
3013             }
3014             delete $targets{$_};
3015         } else {
3016             $targets{$_}->{_conf_fname_int} = add([ $fname ]);
3017         }
3018     }
3019
3020     %table = (%table, %targets);
3021
3022 }
3023
3024 # configuration resolver.  Will only resolve all the lazy evaluation
3025 # codeblocks for the chosen target and all those it inherits from,
3026 # recursively
3027 sub resolve_config {
3028     my $target = shift;
3029     my @breadcrumbs = @_;
3030
3031 #    my $extra_checks = defined($ENV{CONFIGURE_EXTRA_CHECKS});
3032
3033     if (grep { $_ eq $target } @breadcrumbs) {
3034         die "inherit_from loop!  target backtrace:\n  "
3035             ,$target,"\n  ",join("\n  ", @breadcrumbs),"\n";
3036     }
3037
3038     if (!defined($table{$target})) {
3039         warn "Warning! target $target doesn't exist!\n";
3040         return ();
3041     }
3042     # Recurse through all inheritances.  They will be resolved on the
3043     # fly, so when this operation is done, they will all just be a
3044     # bunch of attributes with string values.
3045     # What we get here, though, are keys with references to lists of
3046     # the combined values of them all.  We will deal with lists after
3047     # this stage is done.
3048     my %combined_inheritance = ();
3049     if ($table{$target}->{inherit_from}) {
3050         my @inherit_from =
3051             map { ref($_) eq "CODE" ? $_->() : $_ } @{$table{$target}->{inherit_from}};
3052         foreach (@inherit_from) {
3053             my %inherited_config = resolve_config($_, $target, @breadcrumbs);
3054
3055             # 'template' is a marker that's considered private to
3056             # the config that had it.
3057             delete $inherited_config{template};
3058
3059             foreach (keys %inherited_config) {
3060                 if (!$combined_inheritance{$_}) {
3061                     $combined_inheritance{$_} = [];
3062                 }
3063                 push @{$combined_inheritance{$_}}, $inherited_config{$_};
3064             }
3065         }
3066     }
3067
3068     # We won't need inherit_from in this target any more, since we've
3069     # resolved all the inheritances that lead to this
3070     delete $table{$target}->{inherit_from};
3071
3072     # Now is the time to deal with those lists.  Here's the place to
3073     # decide what shall be done with those lists, all based on the
3074     # values of the target we're currently dealing with.
3075     # - If a value is a coderef, it will be executed with the list of
3076     #   inherited values as arguments.
3077     # - If the corresponding key doesn't have a value at all or is the
3078     #   empty string, the inherited value list will be run through the
3079     #   default combiner (below), and the result becomes this target's
3080     #   value.
3081     # - Otherwise, this target's value is assumed to be a string that
3082     #   will simply override the inherited list of values.
3083     my $default_combiner = add();
3084
3085     my %all_keys =
3086         map { $_ => 1 } (keys %combined_inheritance,
3087                          keys %{$table{$target}});
3088
3089     sub process_values {
3090         my $object    = shift;
3091         my $inherited = shift;  # Always a [ list ]
3092         my $target    = shift;
3093         my $entry     = shift;
3094
3095         $add_called = 0;
3096
3097         while(ref($object) eq "CODE") {
3098             $object = $object->(@$inherited);
3099         }
3100         if (!defined($object)) {
3101             return ();
3102         }
3103         elsif (ref($object) eq "ARRAY") {
3104             local $add_called;  # To make sure recursive calls don't affect it
3105             return [ map { process_values($_, $inherited, $target, $entry) }
3106                      @$object ];
3107         } elsif (ref($object) eq "") {
3108             return $object;
3109         } else {
3110             die "cannot handle reference type ",ref($object)
3111                 ," found in target ",$target," -> ",$entry,"\n";
3112         }
3113     }
3114
3115     foreach (sort keys %all_keys) {
3116         my $previous = $combined_inheritance{$_};
3117
3118         # Current target doesn't have a value for the current key?
3119         # Assign it the default combiner, the rest of this loop body
3120         # will handle it just like any other coderef.
3121         if (!exists $table{$target}->{$_}) {
3122             $table{$target}->{$_} = $default_combiner;
3123         }
3124
3125         $table{$target}->{$_} = process_values($table{$target}->{$_},
3126                                                $combined_inheritance{$_},
3127                                                $target, $_);
3128         unless(defined($table{$target}->{$_})) {
3129             delete $table{$target}->{$_};
3130         }
3131 #        if ($extra_checks &&
3132 #            $previous && !($add_called ||  $previous ~~ $table{$target}->{$_})) {
3133 #            warn "$_ got replaced in $target\n";
3134 #        }
3135     }
3136
3137     # Finally done, return the result.
3138     return %{$table{$target}};
3139 }
3140
3141 sub usage
3142         {
3143         print STDERR $usage;
3144         print STDERR "\npick os/compiler from:\n";
3145         my $j=0;
3146         my $i;
3147         my $k=0;
3148         foreach $i (sort keys %table)
3149                 {
3150                 next if $table{$i}->{template};
3151                 next if $i =~ /^debug/;
3152                 $k += length($i) + 1;
3153                 if ($k > 78)
3154                         {
3155                         print STDERR "\n";
3156                         $k=length($i);
3157                         }
3158                 print STDERR $i . " ";
3159                 }
3160         foreach $i (sort keys %table)
3161                 {
3162                 next if $table{$i}->{template};
3163                 next if $i !~ /^debug/;
3164                 $k += length($i) + 1;
3165                 if ($k > 78)
3166                         {
3167                         print STDERR "\n";
3168                         $k=length($i);
3169                         }
3170                 print STDERR $i . " ";
3171                 }
3172         print STDERR "\n\nNOTE: If in doubt, on Unix-ish systems use './config'.\n";
3173         exit(1);
3174         }
3175
3176 sub run_dofile
3177 {
3178     my $out = shift;
3179     my @templates = @_;
3180
3181     unlink $out || warn "Can't remove $out, $!"
3182         if -f $out;
3183     foreach (@templates) {
3184         die "Can't open $_, $!" unless -f $_;
3185     }
3186     my $perlcmd = (quotify("maybeshell", $config{PERL}))[0];
3187     my $cmd = "$perlcmd \"-I.\" \"-Mconfigdata\" \"$dofile\" -o\"Configure\" \"".join("\" \"",@templates)."\" > \"$out.new\"";
3188     #print STDERR "DEBUG[run_dofile]: \$cmd = $cmd\n";
3189     system($cmd);
3190     exit 1 if $? != 0;
3191     rename("$out.new", $out) || die "Can't rename $out.new, $!";
3192 }
3193
3194 sub compiler_predefined {
3195     state %predefined;
3196     my $cc = shift;
3197
3198     return () if $^O eq 'VMS';
3199
3200     die 'compiler_predefined called without a compiler command'
3201         unless $cc;
3202
3203     if (! $predefined{$cc}) {
3204
3205         $predefined{$cc} = {};
3206
3207         # collect compiler pre-defines from gcc or gcc-alike...
3208         open(PIPE, "$cc -dM -E -x c /dev/null 2>&1 |");
3209         while (my $l = <PIPE>) {
3210             $l =~ m/^#define\s+(\w+(?:\(\w+\))?)(?:\s+(.+))?/ or last;
3211             $predefined{$cc}->{$1} = $2 // '';
3212         }
3213         close(PIPE);
3214     }
3215
3216     return %{$predefined{$cc}};
3217 }
3218
3219 sub which
3220 {
3221     my ($name)=@_;
3222
3223     if (eval { require IPC::Cmd; 1; }) {
3224         IPC::Cmd->import();
3225         return scalar IPC::Cmd::can_run($name);
3226     } else {
3227         # if there is $directories component in splitpath,
3228         # then it's not something to test with $PATH...
3229         return $name if (File::Spec->splitpath($name))[1];
3230
3231         foreach (File::Spec->path()) {
3232             my $fullpath = catfile($_, "$name$target{exe_extension}");
3233             if (-f $fullpath and -x $fullpath) {
3234                 return $fullpath;
3235             }
3236         }
3237     }
3238 }
3239
3240 sub env
3241 {
3242     my $name = shift;
3243     my %opts = @_;
3244
3245     unless ($opts{cacheonly}) {
3246         # Note that if $ENV{$name} doesn't exist or is undefined,
3247         # $config{perlenv}->{$name} will be created with the value
3248         # undef.  This is intentional.
3249
3250         $config{perlenv}->{$name} = $ENV{$name}
3251             if ! exists $config{perlenv}->{$name};
3252     }
3253     return $config{perlenv}->{$name};
3254 }
3255
3256 # Configuration printer ##############################################
3257
3258 sub print_table_entry
3259 {
3260     local $now_printing = shift;
3261     my %target = resolve_config($now_printing);
3262     my $type = shift;
3263
3264     # Don't print the templates
3265     return if $target{template};
3266
3267     my @sequence = (
3268         "sys_id",
3269         "cpp",
3270         "cppflags",
3271         "defines",
3272         "includes",
3273         "cc",
3274         "cflags",
3275         "unistd",
3276         "ld",
3277         "lflags",
3278         "loutflag",
3279         "ex_libs",
3280         "bn_ops",
3281         "apps_aux_src",
3282         "cpuid_asm_src",
3283         "uplink_aux_src",
3284         "bn_asm_src",
3285         "ec_asm_src",
3286         "des_asm_src",
3287         "aes_asm_src",
3288         "bf_asm_src",
3289         "md5_asm_src",
3290         "cast_asm_src",
3291         "sha1_asm_src",
3292         "rc4_asm_src",
3293         "rmd160_asm_src",
3294         "rc5_asm_src",
3295         "wp_asm_src",
3296         "cmll_asm_src",
3297         "modes_asm_src",
3298         "padlock_asm_src",
3299         "chacha_asm_src",
3300         "poly1035_asm_src",
3301         "thread_scheme",
3302         "perlasm_scheme",
3303         "dso_scheme",
3304         "shared_target",
3305         "shared_cflag",
3306         "shared_defines",
3307         "shared_ldflag",
3308         "shared_rcflag",
3309         "shared_extension",
3310         "dso_extension",
3311         "obj_extension",
3312         "exe_extension",
3313         "ranlib",
3314         "ar",
3315         "arflags",
3316         "aroutflag",
3317         "rc",
3318         "rcflags",
3319         "rcoutflag",
3320         "mt",
3321         "mtflags",
3322         "mtinflag",
3323         "mtoutflag",
3324         "multilib",
3325         "build_scheme",
3326         );
3327
3328     if ($type eq "TABLE") {
3329         print "\n";
3330         print "*** $now_printing\n";
3331         foreach (@sequence) {
3332             if (ref($target{$_}) eq "ARRAY") {
3333                 printf "\$%-12s = %s\n", $_, join(" ", @{$target{$_}});
3334             } else {
3335                 printf "\$%-12s = %s\n", $_, $target{$_};
3336             }
3337         }
3338     } elsif ($type eq "HASH") {
3339         my $largest =
3340             length((sort { length($a) <=> length($b) } @sequence)[-1]);
3341         print "    '$now_printing' => {\n";
3342         foreach (@sequence) {
3343             if ($target{$_}) {
3344                 if (ref($target{$_}) eq "ARRAY") {
3345                     print "      '",$_,"'"," " x ($largest - length($_))," => [ ",join(", ", map { "'$_'" } @{$target{$_}})," ],\n";
3346                 } else {
3347                     print "      '",$_,"'"," " x ($largest - length($_))," => '",$target{$_},"',\n";
3348                 }
3349             }
3350         }
3351         print "    },\n";
3352     }
3353 }
3354
3355 # Utility routines ###################################################
3356
3357 # On VMS, if the given file is a logical name, File::Spec::Functions
3358 # will consider it an absolute path.  There are cases when we want a
3359 # purely syntactic check without checking the environment.
3360 sub isabsolute {
3361     my $file = shift;
3362
3363     # On non-platforms, we just use file_name_is_absolute().
3364     return file_name_is_absolute($file) unless $^O eq "VMS";
3365
3366     # If the file spec includes a device or a directory spec,
3367     # file_name_is_absolute() is perfectly safe.
3368     return file_name_is_absolute($file) if $file =~ m|[:\[]|;
3369
3370     # Here, we know the given file spec isn't absolute
3371     return 0;
3372 }
3373
3374 # Makes a directory absolute and cleans out /../ in paths like foo/../bar
3375 # On some platforms, this uses rel2abs(), while on others, realpath() is used.
3376 # realpath() requires that at least all path components except the last is an
3377 # existing directory.  On VMS, the last component of the directory spec must
3378 # exist.
3379 sub absolutedir {
3380     my $dir = shift;
3381
3382     # realpath() is quite buggy on VMS.  It uses LIB$FID_TO_NAME, which
3383     # will return the volume name for the device, no matter what.  Also,
3384     # it will return an incorrect directory spec if the argument is a
3385     # directory that doesn't exist.
3386     if ($^O eq "VMS") {
3387         return rel2abs($dir);
3388     }
3389
3390     # We use realpath() on Unix, since no other will properly clean out
3391     # a directory spec.
3392     use Cwd qw/realpath/;
3393
3394     return realpath($dir);
3395 }
3396
3397 sub quotify {
3398     my %processors = (
3399         perl    => sub { my $x = shift;
3400                          $x =~ s/([\\\$\@"])/\\$1/g;
3401                          return '"'.$x.'"'; },
3402         maybeshell => sub { my $x = shift;
3403                             (my $y = $x) =~ s/([\\\"])/\\$1/g;
3404                             if ($x ne $y || $x =~ m|\s|) {
3405                                 return '"'.$y.'"';
3406                             } else {
3407                                 return $x;
3408                             }
3409                         },
3410         );
3411     my $for = shift;
3412     my $processor =
3413         defined($processors{$for}) ? $processors{$for} : sub { shift; };
3414
3415     return map { $processor->($_); } @_;
3416 }
3417
3418 # collect_from_file($filename, $line_concat_cond_re, $line_concat)
3419 # $filename is a file name to read from
3420 # $line_concat_cond_re is a regexp detecting a line continuation ending
3421 # $line_concat is a CODEref that takes care of concatenating two lines
3422 sub collect_from_file {
3423     my $filename = shift;
3424     my $line_concat_cond_re = shift;
3425     my $line_concat = shift;
3426
3427     open my $fh, $filename || die "unable to read $filename: $!\n";
3428     return sub {
3429         my $saved_line = "";
3430         $_ = "";
3431         while (<$fh>) {
3432             s|\R$||;
3433             if (defined $line_concat) {
3434                 $_ = $line_concat->($saved_line, $_);
3435                 $saved_line = "";
3436             }
3437             if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
3438                 $saved_line = $_;
3439                 next;
3440             }
3441             return $_;
3442         }
3443         die "$filename ending with continuation line\n" if $_;
3444         close $fh;
3445         return undef;
3446     }
3447 }
3448
3449 # collect_from_array($array, $line_concat_cond_re, $line_concat)
3450 # $array is an ARRAYref of lines
3451 # $line_concat_cond_re is a regexp detecting a line continuation ending
3452 # $line_concat is a CODEref that takes care of concatenating two lines
3453 sub collect_from_array {
3454     my $array = shift;
3455     my $line_concat_cond_re = shift;
3456     my $line_concat = shift;
3457     my @array = (@$array);
3458
3459     return sub {
3460         my $saved_line = "";
3461         $_ = "";
3462         while (defined($_ = shift @array)) {
3463             s|\R$||;
3464             if (defined $line_concat) {
3465                 $_ = $line_concat->($saved_line, $_);
3466                 $saved_line = "";
3467             }
3468             if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
3469                 $saved_line = $_;
3470                 next;
3471             }
3472             return $_;
3473         }
3474         die "input text ending with continuation line\n" if $_;
3475         return undef;
3476     }
3477 }
3478
3479 # collect_information($lineiterator, $line_continue, $regexp => $CODEref, ...)
3480 # $lineiterator is a CODEref that delivers one line at a time.
3481 # All following arguments are regex/CODEref pairs, where the regexp detects a
3482 # line and the CODEref does something with the result of the regexp.
3483 sub collect_information {
3484     my $lineiterator = shift;
3485     my %collectors = @_;
3486
3487     while(defined($_ = $lineiterator->())) {
3488         s|\R$||;
3489         my $found = 0;
3490         if ($collectors{"BEFORE"}) {
3491             $collectors{"BEFORE"}->($_);
3492         }
3493         foreach my $re (keys %collectors) {
3494             if ($re !~ /^OTHERWISE|BEFORE|AFTER$/ && /$re/) {
3495                 $collectors{$re}->($lineiterator);
3496                 $found = 1;
3497             };
3498         }
3499         if ($collectors{"OTHERWISE"}) {
3500             $collectors{"OTHERWISE"}->($lineiterator, $_)
3501                 unless $found || !defined $collectors{"OTHERWISE"};
3502         }
3503         if ($collectors{"AFTER"}) {
3504             $collectors{"AFTER"}->($_);
3505         }
3506     }
3507 }
3508
3509 # tokenize($line)
3510 # $line is a line of text to split up into tokens
3511 # returns a list of tokens
3512 #
3513 # Tokens are divided by spaces.  If the tokens include spaces, they
3514 # have to be quoted with single or double quotes.  Double quotes
3515 # inside a double quoted token must be escaped.  Escaping is done
3516 # with backslash.
3517 # Basically, the same quoting rules apply for " and ' as in any
3518 # Unix shell.
3519 sub tokenize {
3520     my $line = my $debug_line = shift;
3521     my @result = ();
3522
3523     while ($line =~ s|^\s+||, $line ne "") {
3524         my $token = "";
3525         while ($line ne "" && $line !~ m|^\s|) {
3526             if ($line =~ m/^"((?:[^"\\]+|\\.)*)"/) {
3527                 $token .= $1;
3528                 $line = $';
3529             } elsif ($line =~ m/^'([^']*)'/) {
3530                 $token .= $1;
3531                 $line = $';
3532             } elsif ($line =~ m/^(\S+)/) {
3533                 $token .= $1;
3534                 $line = $';
3535             }
3536         }
3537         push @result, $token;
3538     }
3539
3540     if ($ENV{CONFIGURE_DEBUG_TOKENIZE}) {
3541         print STDERR "DEBUG[tokenize]: Parsed '$debug_line' into:\n";
3542         print STDERR "DEBUG[tokenize]: ('", join("', '", @result), "')\n";
3543     }
3544     return @result;
3545 }