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