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