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