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