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