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