]> git.ipfire.org Git - thirdparty/openssl.git/blame - Configure
check the return value of BIO_new_file()
[thirdparty/openssl.git] / Configure
CommitLineData
de17db91 1#! /usr/bin/env perl
f4d8f037 2# -*- mode: perl; -*-
4333b89f 3# Copyright 2016-2021 The OpenSSL Project Authors. All Rights Reserved.
ac3d0e13 4#
402dd558 5# Licensed under the Apache License 2.0 (the "License"). You may not use
ac3d0e13
RS
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;
7f73eafe 18use File::Spec::Functions qw/:DEFAULT abs2rel rel2abs splitdir/;
dca99383 19use File::Path qw/mkpath/;
1935a586 20use OpenSSL::fallback "$FindBin::Bin/external/perl/MODULES.txt";
8d2214c0 21use OpenSSL::Glob;
1f86b822 22use OpenSSL::Template;
e39795af 23use OpenSSL::config;
1641cb60 24
f828ba03 25# see INSTALL.md for instructions.
462ba4f6 26
8937a4ed
RL
27my $orig_death_handler = $SIG{__DIE__};
28$SIG{__DIE__} = \&death_handler;
29
31b6ed76 30my $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-egd] [sctp] [386] [--prefix=DIR] [--openssldir=OPENSSLDIR] [--with-xxx[=vvv]] [--config=FILE] os/compiler[:flags]\n";
462ba4f6 31
d0364dcc
RS
32my $banner = <<"EOF";
33
34**********************************************************************
35*** ***
36*** OpenSSL has been successfully configured ***
37*** ***
38*** If you encounter a problem while building, please open an ***
39*** issue on GitHub <https://github.com/openssl/openssl/issues> ***
40*** and include the output from the following command: ***
41*** ***
42*** perl configdata.pm --dump ***
43*** ***
44*** (If you are new to OpenSSL, you might want to consult the ***
45*** 'Troubleshooting' section in the INSTALL.md file first) ***
46*** ***
47**********************************************************************
48EOF
49
434c5dd3 50# Options:
e5f3045f 51#
f09e7ca9
RS
52# --config add the given configuration file, which will be read after
53# any "Configurations*" files that are found in the same
54# directory as this script.
d74dfafd
RL
55# --prefix prefix for the OpenSSL installation, which includes the
56# directories bin, lib, include, share/man, share/doc/openssl
57# This becomes the value of INSTALLTOP in Makefile
58# (Default: /usr/local)
59# --openssldir OpenSSL data area, such as openssl.cnf, certificates and keys.
60# If it's a relative directory, it will be added on the directory
61# given with --prefix.
62# This becomes the value of OPENSSLDIR in Makefile and in C.
63# (Default: PREFIX/ssl)
d0364dcc 64# --banner=".." Output specified text instead of default completion banner
e5f3045f 65#
ecb09baf
RS
66# -w Don't wait after showing a Configure warning
67#
cbfb39d1
AP
68# --cross-compile-prefix Add specified prefix to binutils components.
69#
a6a4d0ac
RL
70# --api One of 0.9.8, 1.0.0, 1.0.1, 1.0.2, 1.1.0, 1.1.1, or 3.0
71# Define the public APIs as they were for that version
72# including patch releases. If 'no-deprecated' is also
73# given, do not compile support for interfaces deprecated
74# up to and including the specified OpenSSL version.
98186eb4 75#
5270e702
RL
76# no-hw-xxx do not compile support for specific crypto hardware.
77# Generic OpenSSL-style methods relating to this support
78# are always compiled but return NULL if the hardware
79# support isn't compiled.
80# no-hw do not compile support for any crypto hardware.
5f8d5c96
BM
81# [no-]threads [don't] try to create a library that is suitable for
82# multithreaded applications (default is "threads" if we
83# know how to do it)
84f32c84 84# [no-]shared [don't] try to create shared libraries when supported.
ae48242c 85# [no-]pic [don't] try to build position independent code when supported.
45b71abe 86# If disabled, it also disables shared and dynamic-engine.
a723979d 87# no-asm do not use assembler
0423f812 88# no-egd do not compile support for the entropy-gathering daemon APIs
e452de9d 89# [no-]zlib [don't] compile support for zlib compression.
84f32c84
DMSP
90# zlib-dynamic Like "zlib", but the zlib library is expected to be a shared
91# library and will be loaded in run-time by the OpenSSL library.
7e159e01 92# sctp include SCTP support
5ded1ca6 93# no-uplink Don't build support for UPLINK interface.
8b1a5af3 94# enable-weak-ssl-ciphers
edcdf38b 95# Enable weak ciphers that are disabled by default.
5ae5dc96
AP
96# 386 generate 80386 code in assembly modules
97# no-sse2 disables IA-32 SSE2 code in assembly modules, the above
98# mentioned '386' option implies this one
3a1ee3c1 99# no-<cipher> build without specified algorithm (dsa, idea, rc5, ...)
f246f54f
DMSP
100# -<xxx> +<xxx> All options which are unknown to the 'Configure' script are
101# /<xxx> passed through to the compiler. Unix-style options beginning
102# with a '-' or '+' are recognized, as well as Windows-style
103# options beginning with a '/'. If the option contains arguments
104# separated by spaces, then the URL-style notation %20 can be
105# used for the space character in order to avoid having to quote
106# the option. For example, -opt%20arg gets expanded to -opt arg.
107# In fact, any ASCII character can be encoded as %xx using its
108# hexadecimal encoding.
047d97af
AP
109# -static while -static is also a pass-through compiler option (and
110# as such is limited to environments where it's actually
111# meaningful), it triggers a number configuration options,
31b6ed76 112# namely no-pic, no-shared and no-threads. It is
047d97af
AP
113# argued that the only reason to produce statically linked
114# binaries (and in context it means executables linked with
115# -static flag, and not just executables linked with static
116# libcrypto.a) is to eliminate dependency on specific run-time,
117# a.k.a. libc version. The mentioned config options are meant
118# to achieve just that. Unfortunately on Linux it's impossible
119# to eliminate the dependency completely for openssl executable
120# because of getaddrinfo and gethostbyname calls, which can
121# invoke dynamically loadable library facility anyway to meet
122# the lookup requests. For this reason on Linux statically
123# linked openssl executable has rather debugging value than
124# production quality.
e41c8d6a 125#
84f32c84
DMSP
126# BN_LLONG use the type 'long long' in crypto/bn/bn.h
127# RC4_CHAR use 'char' instead of 'int' for RC4_INT in crypto/rc4/rc4.h
d0590fe6
AP
128# Following are set automatically by this script
129#
84f32c84
DMSP
130# MD5_ASM use some extra md5 assembler,
131# SHA1_ASM use some extra sha1 assembler, must define L_ENDIAN for x86
132# RMD160_ASM use some extra ripemd160 assembler,
133# SHA256_ASM sha256_block is implemented in assembler
134# SHA512_ASM sha512_block is implemented in assembler
135# AES_ASM AES_[en|de]crypt is implemented in assembler
d02b48c6 136
3b437400
RL
137# Minimum warning options... any contributions to OpenSSL should at least
138# get past these. Note that we only use these with C compilers, not with
139# C++ compilers.
363bd0b4 140
77305338
RS
141# -DPEDANTIC complements -pedantic and is meant to mask code that
142# is not strictly standard-compliant and/or implementation-specific,
143# e.g. inline assembly, disregards to alignment requirements, such
144# that -pedantic would complain about. Incidentally -DPEDANTIC has
145# to be used even in sanitized builds, because sanitizer too is
146# supposed to and does take notice of non-standard behaviour. Then
147# -pedantic with pre-C9x compiler would also complain about 'long
148# long' not being supported. As 64-bit algorithms are common now,
149# it grew impossible to resolve this without sizeable additional
150# code, so we just tell compiler to be pedantic about everything
151# but 'long long' type.
152
3b437400 153my @gcc_devteam_warn = qw(
a935791d 154 -DPEDANTIC -pedantic -Wno-long-long -DUNUSEDRESULT_DEBUG
3b437400 155 -Wall
a935791d 156 -Wmissing-declarations
3b437400
RL
157 -Wextra
158 -Wno-unused-parameter
159 -Wno-missing-field-initializers
160 -Wswitch
161 -Wsign-compare
162 -Wshadow
163 -Wformat
164 -Wtype-limits
165 -Wundef
166 -Werror
167 -Wmissing-prototypes
168 -Wstrict-prototypes
169);
363bd0b4 170
190c8c60
BL
171# These are used in addition to $gcc_devteam_warn when the compiler is clang.
172# TODO(openssl-team): fix problems and investigate if (at least) the
480405e4 173# following warnings can also be enabled:
8bccbce5 174# -Wcast-align
77305338 175# -Wunreachable-code -- no, too ugly/compiler-specific
a773b52a
RS
176# -Wlanguage-extension-token -- no, we use asm()
177# -Wunused-macros -- no, too tricky for BN and _XOPEN_SOURCE etc
178# -Wextended-offsetof -- no, needed in CMS ASN1 code
3b437400 179my @clang_devteam_warn = qw(
03e56683 180 -Wno-unknown-warning-option
3b437400
RL
181 -Wswitch-default
182 -Wno-parentheses-equality
183 -Wno-language-extension-token
184 -Wno-extended-offsetof
185 -Wconditional-uninitialized
186 -Wincompatible-pointer-types-discards-qualifiers
3b437400
RL
187 -Wmissing-variable-declarations
188);
cb2bc054 189
b4a7b4ec
RL
190my @cl_devteam_warn = qw(
191 /WX
192);
193
0c28f277
DSH
194my $strict_warnings = 0;
195
b7efa56a 196# As for $BSDthreads. Idea is to maintain "collective" set of flags,
fce0ba5f 197# which would cover all BSD flavors. -pthread applies to them all,
b7efa56a
AP
198# but is treated differently. OpenBSD expands is as -D_POSIX_THREAD
199# -lc_r, which is sufficient. FreeBSD 4.x expands it as -lc_r,
200# which has to be accompanied by explicit -D_THREAD_SAFE and
201# sometimes -D_REENTRANT. FreeBSD 5.x expands it as -lc_r, which
202# seems to be sufficient?
9c62a279 203our $BSDthreads="-pthread -D_THREAD_SAFE -D_REENTRANT";
d02b48c6 204
98186eb4 205#
f430ba31 206# API compatibility name to version number mapping.
98186eb4 207#
98186eb4 208my $apitable = {
a6a4d0ac
RL
209 # This table expresses when API additions or changes can occur.
210 # The numbering used changes from 3.0 and on because we updated
211 # (solidified) our version numbering scheme at that point.
212
55affcad 213 # From 3.0 and on, we internalise the given version number in decimal
a6a4d0ac
RL
214 # as MAJOR * 10000 + MINOR * 100 + 0
215 "3.0.0" => 30000,
216 "3.0" => 30000,
217
218 # Note that before 3.0, we didn't have the same version number scheme.
219 # Still, the numbering we use here covers what we need.
220 "1.1.1" => 10101,
221 "1.1.0" => 10100,
222 "1.0.2" => 10002,
223 "1.0.1" => 10001,
224 "1.0.0" => 10000,
225 "0.9.8" => 908,
98186eb4
VD
226};
227
18062615
RL
228# For OpenSSL::config::get_platform
229my %guess_opts = ();
230
231my $dryrun = 0;
232
9e0724a1 233our %table = ();
291e94df 234our %config = ();
98fdbce0 235our %withargs = ();
f770d75b
AP
236our $now_printing; # set to current entry's name in print_table_entry
237 # (todo: right thing would be to encapsulate name
238 # into %target [class] and make print_table_entry
239 # a method)
3e83e686 240
bd5192b1 241# Forward declarations ###############################################
7ead0c89 242
bd5192b1
RL
243# read_config(filename)
244#
245# Reads a configuration file and populates %table with the contents
246# (which the configuration file places in %targets).
247sub read_config;
7d46b942 248
bd5192b1
RL
249# resolve_config(target)
250#
8483a003 251# Resolves all the late evaluations, inheritances and so on for the
bd5192b1
RL
252# chosen target and any target it inherits from.
253sub resolve_config;
7d46b942 254
15c7adb0 255
107b5792
RL
256# Information collection #############################################
257
9fe2bb77 258# Unified build supports separate build dir
ec182ef0
RL
259my $srcdir = catdir(absolutedir(dirname($0))); # catdir ensures local syntax
260my $blddir = catdir(absolutedir(".")); # catdir ensures local syntax
610e2b3b
RL
261
262# File::Spec::Unix doesn't detect case insensitivity, so we make sure to
263# check if the source and build directory are really the same, and make
264# them so. This avoids all kinds of confusion later on.
265# We must check @File::Spec::ISA rather than using File::Spec->isa() to
266# know if File::Spec ended up loading File::Spec::Unix.
267$srcdir = $blddir
268 if (grep(/::Unix$/, @File::Spec::ISA)
269 && samedir($srcdir, $blddir));
270
9fe2bb77
RL
271my $dofile = abs2rel(catfile($srcdir, "util/dofile.pl"));
272
b5293d4c
RL
273my $local_config_envname = 'OPENSSL_LOCAL_CONFIG_DIR';
274
610e2b3b
RL
275$config{sourcedir} = abs2rel($srcdir, $blddir);
276$config{builddir} = abs2rel($blddir, $blddir);
31214258
RS
277# echo -n 'holy hand grenade of antioch' | openssl sha256
278$config{FIPSKEY} =
279 'f4556650ac31d35461610bac4ed81b1a181b2d8a43ea2854cbae22ca74560813';
9fe2bb77 280
ee4cdb7f
RL
281# Collect reconfiguration information if needed
282my @argvcopy=@ARGV;
283
284if (grep /^reconf(igure)?$/, @argvcopy) {
99aeeecb
RL
285 die "reconfiguring with other arguments present isn't supported"
286 if scalar @argvcopy > 1;
ee4cdb7f 287 if (-f "./configdata.pm") {
84f32c84
DMSP
288 my $file = "./configdata.pm";
289 unless (my $return = do $file) {
290 die "couldn't parse $file: $@" if $@;
ee4cdb7f
RL
291 die "couldn't do $file: $!" unless defined $return;
292 die "couldn't run $file" unless $return;
84f32c84 293 }
ee4cdb7f 294
84f32c84
DMSP
295 @argvcopy = defined($configdata::config{perlargv}) ?
296 @{$configdata::config{perlargv}} : ();
297 die "Incorrect data to reconfigure, please do a normal configuration\n"
298 if (grep(/^reconf/,@argvcopy));
299 $config{perlenv} = $configdata::config{perlenv} // {};
ee4cdb7f 300 } else {
84f32c84 301 die "Insufficient data to reconfigure, please do a normal configuration\n";
ee4cdb7f
RL
302 }
303}
304
305$config{perlargv} = [ @argvcopy ];
306
b16654a2
RL
307# Historical: if known directories in crypto/ have been removed, it means
308# that those sub-systems are disabled.
309# (the other option would be to removed them from the SUBDIRS statement in
310# crypto/build.info)
311# We reverse the input list for cosmetic purely reasons, to compensate that
312# 'unshift' adds at the front of the list (i.e. in reverse input order).
313foreach ( reverse sort( 'aes', 'aria', 'bf', 'camellia', 'cast', 'des', 'dh',
314 'dsa', 'ec', 'hmac', 'idea', 'md2', 'md5', 'mdc2',
3a1ee3c1 315 'rc2', 'rc4', 'rc5', 'ripemd', 'seed', 'sha',
b16654a2
RL
316 'sm2', 'sm3', 'sm4') ) {
317 unshift @argvcopy, "no-$_" if ! -d catdir($srcdir, 'crypto', $_);
318}
319
107b5792 320# Collect version numbers
20551b2e 321my %version = ();
107b5792
RL
322
323collect_information(
036cbb6b 324 collect_from_file(catfile($srcdir,'VERSION.dat')),
20551b2e
RL
325 qr/\s*(\w+)\s*=\s*(.*?)\s*$/ =>
326 sub {
327 # Only define it if there is a value at all
6a960a94
RL
328 if ($2 ne '') {
329 my $k = $1;
330 my $v = $2;
331 # Some values are quoted. Trim the quotes
332 $v = $1 if $v =~ /^"(.*)"$/;
333 $version{uc $k} = $v;
334 }
20551b2e
RL
335 },
336 "OTHERWISE" =>
036cbb6b 337 sub { die "Something wrong with this line:\n$_\nin $srcdir/VERSION.dat" },
107b5792 338 );
20551b2e
RL
339
340$config{major} = $version{MAJOR} // 'unknown';
341$config{minor} = $version{MINOR} // 'unknown';
342$config{patch} = $version{PATCH} // 'unknown';
343$config{prerelease} =
344 defined $version{PRE_RELEASE_TAG} ? "-$version{PRE_RELEASE_TAG}" : '';
345$config{build_metadata} =
346 defined $version{BUILD_METADATA} ? "+$version{BUILD_METADATA}" : '';
347$config{shlib_version} = $version{SHLIB_VERSION} // 'unknown';
348$config{release_date} = $version{RELEASE_DATE} // 'xx XXX xxxx';
107b5792 349
16942e08
DMSP
350$config{version} = "$config{major}.$config{minor}.$config{patch}";
351$config{full_version} = "$config{version}$config{prerelease}$config{build_metadata}";
352
036cbb6b 353die "erroneous version information in VERSION.dat: ",
20551b2e
RL
354 "$config{version}, $config{shlib_version}\n"
355 unless (defined $version{MAJOR}
356 && defined $version{MINOR}
357 && defined $version{PATCH}
358 && defined $version{SHLIB_VERSION});
359
107b5792
RL
360# Collect target configurations
361
85152ca4 362my $pattern = catfile(dirname($0), "Configurations", "*.conf");
97855556 363foreach (sort glob($pattern)) {
f09e7ca9
RS
364 &read_config($_);
365}
d02b48c6 366
7ecdf18d 367if (defined env($local_config_envname)) {
b5293d4c
RL
368 if ($^O eq 'VMS') {
369 # VMS environment variables are logical names,
370 # which can be used as is
371 $pattern = $local_config_envname . ':' . '*.conf';
372 } else {
7ecdf18d 373 $pattern = catfile(env($local_config_envname), '*.conf');
b5293d4c
RL
374 }
375
97855556 376 foreach (sort glob($pattern)) {
b5293d4c
RL
377 &read_config($_);
378 }
379}
380
d5fa7035
RL
381# Save away perl command information
382$config{perl_cmd} = $^X;
383$config{perl_version} = $Config{version};
384$config{perl_archname} = $Config{archname};
385
291e94df
RL
386$config{prefix}="";
387$config{openssldir}="";
7d130f68 388$config{processor}="";
107b5792 389$config{libdir}="";
9c62a279 390my $auto_threads=1; # enable threads automatically? true by default
0396479d 391my $default_ranlib;
107b5792 392
6b01bed2 393# Known TLS and DTLS protocols
84a68336 394my @tls = qw(ssl3 tls1 tls1_1 tls1_2 tls1_3);
6b01bed2
VD
395my @dtls = qw(dtls1 dtls1_2);
396
8483a003 397# Explicitly known options that are possible to disable. They can
8b527be2
RL
398# be regexps, and will be used like this: /^no-${option}$/
399# For developers: keep it sorted alphabetically
400
401my @disablables = (
38230e30 402 "acvp-tests",
c91a0a83 403 "afalgeng",
d42d0a4d 404 "aria",
c38bb727 405 "asan",
8b527be2 406 "asm",
52739e40 407 "async",
b184e3ef 408 "autoalginit",
498abff0 409 "autoerrinit",
dbabc862 410 "autoload-config",
8b527be2 411 "bf",
2d0b4412 412 "blake2",
ac4033d6 413 "buildtest-c++",
06f81af8 414 "bulk",
1eaf1fc3 415 "cached-fetch",
8b527be2
RL
416 "camellia",
417 "capieng",
418 "cast",
48f14845 419 "chacha",
8b527be2 420 "cmac",
538f38db 421 "cmp",
8b527be2
RL
422 "cms",
423 "comp",
3e45d393 424 "crypto-mdebug",
8b527be2
RL
425 "ct",
426 "deprecated",
427 "des",
619eb33a 428 "devcryptoeng",
8b527be2
RL
429 "dgram",
430 "dh",
431 "dsa",
3b2f8c77 432 "dso",
a5ecdc6a 433 "dtls",
343ec2b0 434 "dynamic-engine",
8b527be2
RL
435 "ec",
436 "ec2m",
afa0a13c 437 "ec_nistp_64_gcc_128",
6b01bed2
VD
438 "ecdh",
439 "ecdsa",
b31feae6 440 "egd",
8b527be2 441 "engine",
1288f26f 442 "err",
ce2596d4 443 "external-tests",
02f7114a 444 "filenames",
e7545517 445 "fips",
7a810fac 446 "fips-securitychecks",
f59d0131 447 "fuzz-afl",
afa0a13c 448 "fuzz-libfuzzer",
168c3b73 449 "gost",
8b527be2 450 "idea",
742ccab3 451 "ktls",
d0308923 452 "legacy",
4b2981f1 453 "loadereng",
09aa263a 454 "makedepend",
8b527be2
RL
455 "md2",
456 "md4",
8b527be2 457 "mdc2",
34786bde 458 "module",
29df3061 459 "msan",
fa22f98f 460 "multiblock",
8b527be2
RL
461 "nextprotoneg",
462 "ocb",
463 "ocsp",
469ce8ff 464 "padlockeng",
ae48242c 465 "pic",
afa0a13c 466 "pinshared",
48f14845 467 "poly1305",
8b527be2
RL
468 "posix-io",
469 "psk",
470 "rc2",
471 "rc4",
472 "rc5",
473 "rdrand",
474 "rfc3779",
8b527be2 475 "rmd160",
8b527be2 476 "scrypt",
8b527be2 477 "sctp",
6943335e 478 "secure-memory",
8b527be2 479 "seed",
8b527be2 480 "shared",
3f5616d7 481 "siphash",
b1ceb439 482 "siv",
1bf2cc23 483 "sm2",
a0c3e4fa 484 "sm3",
f19a5ff9 485 "sm4",
8b527be2
RL
486 "sock",
487 "srp",
488 "srtp",
489 "sse2",
490 "ssl",
8b527be2
RL
491 "ssl-trace",
492 "static-engine",
493 "stdio",
93880ce1 494 "tests",
8b527be2
RL
495 "threads",
496 "tls",
16a9d374 497 "trace",
1288f26f 498 "ts",
c38bb727 499 "ubsan",
48feaceb 500 "ui-console",
8b527be2 501 "unit-test",
5ded1ca6 502 "uplink",
8b1a5af3 503 "weak-ssl-ciphers",
afa0a13c 504 "whirlpool",
8b527be2
RL
505 "zlib",
506 "zlib-dynamic",
507 );
6b01bed2 508foreach my $proto ((@tls, @dtls))
84f32c84
DMSP
509 {
510 push(@disablables, $proto);
511 push(@disablables, "$proto-method") unless $proto eq "tls1_3";
512 }
8b527be2 513
538f38db
RL
514# Internal disablables, for aliasing purposes. They serve no special
515# purpose here, but allow scripts to get to know them through configdata.pm,
516# where these are merged with @disablables.
517# The actual aliasing mechanism is done via %disable_cascades
518my @disablables_int = qw(
519 crmf
520 );
521
2b1343b9
MC
522my %deprecated_disablables = (
523 "ssl2" => undef,
524 "buf-freelists" => undef,
14611382 525 "crypto-mdebug-backtrace" => undef,
469ce8ff
RL
526 "hw" => "hw", # causes cascade, but no macro
527 "hw-padlock" => "padlockeng",
48feaceb
RL
528 "ripemd" => "rmd160",
529 "ui" => "ui-console",
0b45d8ee 530 "heartbeats" => undef,
e80381e1
RL
531 );
532
094925de 533# All of the following are disabled by default:
c9a112f5 534
9e04edf2 535our %disabled = ( # "what" => "comment"
b2d8c7b6 536 "fips" => "default",
84f32c84
DMSP
537 "asan" => "default",
538 "buildtest-c++" => "default",
539 "crypto-mdebug" => "default",
540 "crypto-mdebug-backtrace" => "default",
541 "devcryptoeng" => "default",
542 "ec_nistp_64_gcc_128" => "default",
543 "egd" => "default",
544 "external-tests" => "default",
84f32c84 545 "fuzz-afl" => "default",
afa0a13c
DMSP
546 "fuzz-libfuzzer" => "default",
547 "ktls" => "default",
84f32c84 548 "md2" => "default",
29df3061 549 "msan" => "default",
84f32c84
DMSP
550 "rc5" => "default",
551 "sctp" => "default",
84f32c84
DMSP
552 "ssl3" => "default",
553 "ssl3-method" => "default",
554 "trace" => "default",
555 "ubsan" => "default",
556 "unit-test" => "default",
557 "weak-ssl-ciphers" => "default",
558 "zlib" => "default",
559 "zlib-dynamic" => "default",
84f32c84 560 );
c9a112f5 561
c569e206
RL
562# Note: => pair form used for aesthetics, not to truly make a hash table
563my @disable_cascades = (
84f32c84 564 # "what" => [ "cascade", ... ]
e3577add 565 "bulk" => [ "shared", "dso",
06f81af8
DDO
566 "aria", "async", "autoload-config",
567 "blake2", "bf", "camellia", "cast", "chacha",
568 "cmac", "cms", "cmp", "comp", "ct",
569 "des", "dgram", "dh", "dsa",
570 "ec", "engine",
571 "filenames",
572 "idea", "ktls",
573 "md4", "multiblock", "nextprotoneg",
574 "ocsp", "ocb", "poly1305", "psk",
575 "rc2", "rc4", "rmd160",
576 "seed", "siphash", "siv",
577 "sm3", "sm4", "srp",
726f92e0 578 "srtp", "ssl3-method", "ssl-trace",
06f81af8
DDO
579 "ts", "ui-console", "whirlpool",
580 "fips-securitychecks" ],
7d130f68 581 sub { $config{processor} eq "386" }
84f32c84
DMSP
582 => [ "sse2" ],
583 "ssl" => [ "ssl3" ],
584 "ssl3-method" => [ "ssl3" ],
585 "zlib" => [ "zlib-dynamic" ],
586 "des" => [ "mdc2" ],
06f81af8 587 "ec" => [ "ec2m", "ecdsa", "ecdh", "sm2", "gost" ],
84f32c84
DMSP
588 "dgram" => [ "dtls", "sctp" ],
589 "sock" => [ "dgram" ],
590 "dtls" => [ @dtls ],
343a7467 591 sub { 0 == scalar grep { !$disabled{$_} } @dtls }
84f32c84 592 => [ "dtls" ],
c569e206 593
84f32c84 594 "tls" => [ @tls ],
343a7467 595 sub { 0 == scalar grep { !$disabled{$_} } @tls }
84f32c84 596 => [ "tls" ],
c569e206 597
ef8ca6bd 598 "crypto-mdebug" => [ "crypto-mdebug-backtrace" ],
343ec2b0 599
34786bde
RL
600 # If no modules, then no dynamic engines either
601 "module" => [ "dynamic-engine" ],
602
603 # Without shared libraries, dynamic engines aren't possible.
604 # This is due to them having to link with libcrypto and register features
605 # using the ENGINE functionality, and since that relies on global tables,
606 # those *have* to be exacty the same as the ones accessed from the app,
607 # which cannot be guaranteed if shared libraries aren't present.
608 # (note that even with shared libraries, both the app and dynamic engines
609 # must be linked with the same library)
5ded1ca6 610 "shared" => [ "dynamic-engine", "uplink" ],
3b2f8c77 611 "dso" => [ "dynamic-engine", "module" ],
34786bde
RL
612 # Other modules don't necessarily have to link with libcrypto, so shared
613 # libraries do not have to be a condition to produce those.
614
615 # Without position independent code, there can be no shared libraries
616 # or modules.
617 "pic" => [ "shared", "module" ],
469ce8ff 618
e1f5a92d 619 "module" => [ "fips", "dso" ],
e7545517 620
b8fa02e8 621 "engine" => [ "dynamic-engine", grep(/eng$/, @disablables) ],
86ff7cf2 622 "dynamic-engine" => [ "loadereng" ],
469ce8ff 623 "hw" => [ "padlockeng" ],
d90a6beb
MC
624
625 # no-autoalginit is only useful when building non-shared
9c11e8ec 626 "autoalginit" => [ "shared", "apps", "fips" ],
d90a6beb 627
15a1bd0a 628 "stdio" => [ "apps", "capieng", "egd" ],
d90a6beb 629 "apps" => [ "tests" ],
302eba3f 630 "tests" => [ "external-tests" ],
3cf96e88 631 "comp" => [ "zlib" ],
98020023 632 "sm3" => [ "sm2" ],
b612799a 633 sub { !$disabled{"unit-test"} } => [ "heartbeats" ],
29df3061
EK
634
635 sub { !$disabled{"msan"} } => [ "asm" ],
b1ceb439 636
6d2a1eff
MC
637 "cmac" => [ "siv" ],
638 "legacy" => [ "md2" ],
538f38db
RL
639
640 "cmp" => [ "crmf" ],
c72fa255 641
c3c00c7a 642 "fips" => [ "fips-securitychecks", "acvp-tests" ],
ccdfcf07 643
6d2a1eff 644 "deprecated-3.0" => [ "engine", "srp" ]
c569e206
RL
645 );
646
647# Avoid protocol support holes. Also disable all versions below N, if version
648# N is disabled while N+1 is enabled.
649#
650my @list = (reverse @tls);
651while ((my $first, my $second) = (shift @list, shift @list)) {
652 last unless @list;
653 push @disable_cascades, ( sub { !$disabled{$first} && $disabled{$second} }
84f32c84 654 => [ @list ] );
c569e206
RL
655 unshift @list, $second;
656}
657my @list = (reverse @dtls);
658while ((my $first, my $second) = (shift @list, shift @list)) {
659 last unless @list;
660 push @disable_cascades, ( sub { !$disabled{$first} && $disabled{$second} }
84f32c84 661 => [ @list ] );
c569e206
RL
662 unshift @list, $second;
663}
664
7a762197 665# Explicit "no-..." options will be collected in %disabled along with the defaults.
e4ef2e25 666# To remove something from %disabled, use "enable-foo".
7a762197
BM
667# For symmetry, "disable-foo" is a synonym for "no-foo".
668
bbe486cf 669# For the "make variables" CPPINCLUDES and CPPDEFINES, we support lists with
5b18235a
RL
670# platform specific list separators. Users from those platforms should
671# recognise those separators from how you set up the PATH to find executables.
672# The default is the Unix like separator, :, but as an exception, we also
673# support the space as separator.
674my $list_separator_re =
675 { VMS => qr/(?<!\^),/,
676 MSWin32 => qr/(?<!\\);/ } -> {$^O} // qr/(?<!\\)[:\s]/;
677# All the "make variables" we support
f5846179
RL
678# Some get pre-populated for the sake of backward compatibility
679# (we supported those before the change to "make variable" support.
5b18235a 680my %user = (
f5846179 681 AR => env('AR'),
5b18235a
RL
682 ARFLAGS => [],
683 AS => undef,
684 ASFLAGS => [],
f5846179 685 CC => env('CC'),
8e7984e5 686 CFLAGS => [ env('CFLAGS') || () ],
f5846179 687 CXX => env('CXX'),
8e7984e5 688 CXXFLAGS => [ env('CXXFLAGS') || () ],
5b18235a 689 CPP => undef,
8e7984e5 690 CPPFLAGS => [ env('CPPFLAGS') || () ], # -D, -I, -Wp,
5b18235a
RL
691 CPPDEFINES => [], # Alternative for -D
692 CPPINCLUDES => [], # Alternative for -I
f5846179
RL
693 CROSS_COMPILE => env('CROSS_COMPILE'),
694 HASHBANGPERL=> env('HASHBANGPERL') || env('PERL'),
5b18235a 695 LD => undef,
8e7984e5
RL
696 LDFLAGS => [ env('LDFLAGS') || () ], # -L, -Wl,
697 LDLIBS => [ env('LDLIBS') || () ], # -l
5b18235a
RL
698 MT => undef,
699 MTFLAGS => [],
9e265322 700 PERL => env('PERL') || ($^O ne "VMS" ? $^X : "perl"),
f5846179
RL
701 RANLIB => env('RANLIB'),
702 RC => env('RC') || env('WINDRES'),
0c4e984d 703 RCFLAGS => [ env('RCFLAGS') || () ],
5b18235a
RL
704 RM => undef,
705 );
f729ba55
RL
706# Info about what "make variables" may be prefixed with the cross compiler
707# prefix. This should NEVER mention any such variable with a list for value.
708my @user_crossable = qw ( AR AS CC CXX CPP LD MT RANLIB RC );
5b18235a
RL
709# The same but for flags given as Configure options. These are *additional*
710# input, as opposed to the VAR=string option that override the corresponding
711# config target attributes
712my %useradd = (
713 CPPDEFINES => [],
714 CPPINCLUDES => [],
715 CPPFLAGS => [],
716 CFLAGS => [],
717 CXXFLAGS => [],
718 LDFLAGS => [],
719 LDLIBS => [],
0c4e984d 720 RCFLAGS => [],
5b18235a
RL
721 );
722
723my %user_synonyms = (
724 HASHBANGPERL=> 'PERL',
725 RC => 'WINDRES',
726 );
abe256e7
RL
727
728# Some target attributes have been renamed, this is the translation table
729my %target_attr_translate =(
730 ar => 'AR',
731 as => 'AS',
732 cc => 'CC',
733 cxx => 'CXX',
734 cpp => 'CPP',
735 hashbangperl => 'HASHBANGPERL',
736 ld => 'LD',
737 mt => 'MT',
738 ranlib => 'RANLIB',
739 rc => 'RC',
740 rm => 'RM',
5b18235a 741 );
5b18235a 742
2ab92ae9 743# Initialisers coming from 'config' scripts
ff455d99
AP
744$config{defines} = [ split(/$list_separator_re/, env('__CNF_CPPDEFINES')) ];
745$config{includes} = [ split(/$list_separator_re/, env('__CNF_CPPINCLUDES')) ];
746$config{cppflags} = [ env('__CNF_CPPFLAGS') || () ];
747$config{cflags} = [ env('__CNF_CFLAGS') || () ];
748$config{cxxflags} = [ env('__CNF_CXXFLAGS') || () ];
749$config{lflags} = [ env('__CNF_LDFLAGS') || () ];
750$config{ex_libs} = [ env('__CNF_LDLIBS') || () ];
2ab92ae9 751
7d130f68 752$config{openssl_api_defines}=[];
7d130f68 753$config{openssl_sys_defines}=[];
e0bf7c01 754$config{openssl_feature_defines}=[];
3fa04f0d 755$config{options}="";
8864f0de 756$config{build_type} = "release";
5b18235a 757my $target="";
c59cb511 758
ac6ae8a9 759my %cmdvars = (); # Stores FOO='blah' type arguments
fe05264e 760my %unsupported_options = ();
e80381e1 761my %deprecated_options = ();
8389ec4b
RS
762# If you change this, update apps/version.c
763my @known_seed_sources = qw(getrandom devrandom os egd none rdcpu librandom);
764my @seed_sources = ();
fad599f7 765while (@argvcopy)
84f32c84
DMSP
766 {
767 $_ = shift @argvcopy;
768
769 # Support env variable assignments among the options
770 if (m|^(\w+)=(.+)?$|)
771 {
772 $cmdvars{$1} = $2;
773 # Every time a variable is given as a configuration argument,
774 # it acts as a reset if the variable.
775 if (exists $user{$1})
776 {
777 $user{$1} = ref $user{$1} eq "ARRAY" ? [] : undef;
778 }
779 #if (exists $useradd{$1})
780 # {
781 # $useradd{$1} = [];
782 # }
783 next;
784 }
785
786 # VMS is a case insensitive environment, and depending on settings
787 # out of our control, we may receive options uppercased. Let's
788 # downcase at least the part before any equal sign.
789 if ($^O eq "VMS")
790 {
791 s/^([^=]*)/lc($1)/e;
792 }
793
794 # some people just can't read the instructions, clang people have to...
795 s/^-no-(?!integrated-as)/no-/;
796
797 # rewrite some options in "enable-..." form
798 s /^-?-?shared$/enable-shared/;
799 s /^sctp$/enable-sctp/;
800 s /^threads$/enable-threads/;
801 s /^zlib$/enable-zlib/;
802 s /^zlib-dynamic$/enable-zlib-dynamic/;
cc9f9b98 803 s /^fips$/enable-fips/;
c9a112f5 804
e4ef2e25 805 if (/^(no|disable|enable)-(.+)$/)
2b1343b9
MC
806 {
807 my $word = $2;
469ce8ff
RL
808 if ($word !~ m|hw(?:-.+)| # special treatment for hw regexp opt
809 && !exists $deprecated_disablables{$word}
810 && !grep { $word eq $_ } @disablables)
2b1343b9
MC
811 {
812 $unsupported_options{$_} = 1;
813 next;
814 }
815 }
816 if (/^no-(.+)$/ || /^disable-(.+)$/)
817 {
e4ef2e25
RS
818 foreach my $proto ((@tls, @dtls))
819 {
820 if ($1 eq "$proto-method")
821 {
822 $disabled{"$proto"} = "option($proto-method)";
823 last;
824 }
825 }
826 if ($1 eq "dtls")
827 {
828 foreach my $proto (@dtls)
829 {
830 $disabled{$proto} = "option(dtls)";
831 }
c5c7700c 832 $disabled{"dtls"} = "option(dtls)";
e4ef2e25
RS
833 }
834 elsif ($1 eq "ssl")
835 {
836 # Last one of its kind
837 $disabled{"ssl3"} = "option(ssl)";
838 }
839 elsif ($1 eq "tls")
840 {
841 # XXX: Tests will fail if all SSL/TLS
842 # protocols are disabled.
843 foreach my $proto (@tls)
844 {
845 $disabled{$proto} = "option(tls)";
846 }
847 }
343ec2b0
RL
848 elsif ($1 eq "static-engine")
849 {
19ab5790 850 delete $disabled{"dynamic-engine"};
343ec2b0
RL
851 }
852 elsif ($1 eq "dynamic-engine")
853 {
19ab5790 854 $disabled{"dynamic-engine"} = "option";
343ec2b0 855 }
2b1343b9
MC
856 elsif (exists $deprecated_disablables{$1})
857 {
858 $deprecated_options{$_} = 1;
859 if (defined $deprecated_disablables{$1})
860 {
861 $disabled{$deprecated_disablables{$1}} = "option";
862 }
863 }
469ce8ff
RL
864 elsif ($1 =~ m|hw(?:-.+)|) # deprecate hw options in regexp form
865 {
866 $deprecated_options{$_} = 1;
867 }
e4ef2e25
RS
868 else
869 {
870 $disabled{$1} = "option";
871 }
84f32c84
DMSP
872 # No longer an automatic choice
873 $auto_threads = 0 if ($1 eq "threads");
874 }
875 elsif (/^enable-(.+)$/)
876 {
343ec2b0
RL
877 if ($1 eq "static-engine")
878 {
19ab5790 879 $disabled{"dynamic-engine"} = "option";
343ec2b0
RL
880 }
881 elsif ($1 eq "dynamic-engine")
882 {
19ab5790 883 delete $disabled{"dynamic-engine"};
343ec2b0 884 }
25004db7
RL
885 elsif ($1 eq "zlib-dynamic")
886 {
887 delete $disabled{"zlib"};
888 }
84f32c84
DMSP
889 my $algo = $1;
890 delete $disabled{$algo};
891
892 # No longer an automatic choice
893 $auto_threads = 0 if ($1 eq "threads");
894 }
18062615
RL
895 elsif (/^-d$/) # From older 'config'
896 {
897 $config{build_type} = "debug";
898 }
899 elsif (/^-v$/) # From older 'config'
900 {
901 $guess_opts{verbose} = 1;
902 }
ecb09baf 903 elsif (/^-w$/)
18062615
RL
904 {
905 $guess_opts{nowait} = 1;
906 }
907 elsif (/^-t$/) # From older 'config'
908 {
909 $dryrun = 1;
910 }
84f32c84
DMSP
911 elsif (/^--strict-warnings$/)
912 {
913 # Pretend that our strict flags is a C flag, and replace it
914 # with the proper flags later on
915 push @{$useradd{CFLAGS}}, '--ossl-strict-warnings';
84f32c84
DMSP
916 $strict_warnings=1;
917 }
918 elsif (/^--debug$/)
919 {
920 $config{build_type} = "debug";
921 }
922 elsif (/^--release$/)
923 {
924 $config{build_type} = "release";
925 }
926 elsif (/^386$/)
927 { $config{processor}=386; }
84f32c84
DMSP
928 elsif (/^rsaref$/)
929 {
930 # No RSAref support any more since it's not needed.
931 # The check for the option is there so scripts aren't
932 # broken
933 }
f246f54f 934 elsif (m|^[-+/]|)
84f32c84
DMSP
935 {
936 if (/^--prefix=(.*)$/)
937 {
938 $config{prefix}=$1;
939 die "Directory given with --prefix MUST be absolute\n"
940 unless file_name_is_absolute($config{prefix});
941 }
942 elsif (/^--api=(.*)$/)
943 {
a6a4d0ac
RL
944 my $api = $1;
945 die "Unknown API compatibility level $api"
946 unless defined $apitable->{$api};
947 $config{api}=$apitable->{$api};
84f32c84
DMSP
948 }
949 elsif (/^--libdir=(.*)$/)
950 {
951 $config{libdir}=$1;
952 }
953 elsif (/^--openssldir=(.*)$/)
954 {
955 $config{openssldir}=$1;
956 }
957 elsif (/^--with-zlib-lib=(.*)$/)
958 {
959 $withargs{zlib_lib}=$1;
960 }
961 elsif (/^--with-zlib-include=(.*)$/)
962 {
963 $withargs{zlib_include}=$1;
964 }
965 elsif (/^--with-fuzzer-lib=(.*)$/)
966 {
967 $withargs{fuzzer_lib}=$1;
968 }
969 elsif (/^--with-fuzzer-include=(.*)$/)
970 {
971 $withargs{fuzzer_include}=$1;
972 }
973 elsif (/^--with-rand-seed=(.*)$/)
974 {
975 foreach my $x (split(m|,|, $1))
976 {
977 die "Unknown --with-rand-seed choice $x\n"
978 if ! grep { $x eq $_ } @known_seed_sources;
979 push @seed_sources, $x;
980 }
981 }
31214258
RS
982 elsif (/^--fips-key=(.*)$/)
983 {
984 $user{FIPSKEY}=lc($1);
985 die "Non-hex character in FIPS key\n"
986 if $user{FIPSKEY} =~ /[^a-f0-9]/;
987 die "FIPS key must have even number of characters\n"
988 if length $1 & 1;
989 die "FIPS key too long (64 bytes max)\n"
990 if length $1 > 64;
991 }
d0364dcc
RS
992 elsif (/^--banner=(.*)$/)
993 {
994 $banner = $1 . "\n";
995 }
84f32c84
DMSP
996 elsif (/^--cross-compile-prefix=(.*)$/)
997 {
998 $user{CROSS_COMPILE}=$1;
999 }
1000 elsif (/^--config=(.*)$/)
1001 {
1002 read_config $1;
1003 }
1004 elsif (/^-l(.*)$/)
1005 {
1006 push @{$useradd{LDLIBS}}, $_;
1007 }
1008 elsif (/^-framework$/)
1009 {
1010 push @{$useradd{LDLIBS}}, $_, shift(@argvcopy);
1011 }
1012 elsif (/^-L(.*)$/ or /^-Wl,/)
1013 {
1014 push @{$useradd{LDFLAGS}}, $_;
1015 }
1016 elsif (/^-rpath$/ or /^-R$/)
1017 # -rpath is the OSF1 rpath flag
1018 # -R is the old Solaris rpath flag
1019 {
1020 my $rpath = shift(@argvcopy) || "";
1021 $rpath .= " " if $rpath ne "";
1022 push @{$useradd{LDFLAGS}}, $_, $rpath;
1023 }
1024 elsif (/^-static$/)
1025 {
1026 push @{$useradd{LDFLAGS}}, $_;
84f32c84 1027 }
f246f54f 1028 elsif (m|^[-/]D(.*)$|)
84f32c84
DMSP
1029 {
1030 push @{$useradd{CPPDEFINES}}, $1;
1031 }
f246f54f 1032 elsif (m|^[-/]I(.*)$|)
84f32c84
DMSP
1033 {
1034 push @{$useradd{CPPINCLUDES}}, $1;
1035 }
1036 elsif (/^-Wp,$/)
1037 {
1038 push @{$useradd{CPPFLAGS}}, $1;
1039 }
1040 else # common if (/^[-+]/), just pass down...
1041 {
f246f54f
DMSP
1042 # Treat %xx as an ASCII code (e.g. replace %20 by a space character).
1043 # This provides a simple way to pass options with arguments separated
1044 # by spaces without quoting (e.g. -opt%20arg translates to -opt arg).
84f32c84
DMSP
1045 $_ =~ s/%([0-9a-f]{1,2})/chr(hex($1))/gei;
1046 push @{$useradd{CFLAGS}}, $_;
1047 push @{$useradd{CXXFLAGS}}, $_;
8389ec4b 1048 }
84f32c84 1049 }
f246f54f
DMSP
1050 elsif (m|^/|)
1051 {
1052 # Treat %xx as an ASCII code (e.g. replace %20 by a space character).
1053 # This provides a simple way to pass options with arguments separated
1054 # by spaces without quoting (e.g. /opt%20arg translates to /opt arg).
1055 $_ =~ s/%([0-9a-f]{1,2})/chr(hex($1))/gei;
1056 push @{$useradd{CFLAGS}}, $_;
1057 push @{$useradd{CXXFLAGS}}, $_;
1058 }
84f32c84
DMSP
1059 else
1060 {
1061 die "target already defined - $target (offending arg: $_)\n" if ($target ne "");
1062 $target=$_;
1063 }
1064 unless ($_ eq $target || /^no-/ || /^disable-/)
1065 {
1066 # "no-..." follows later after implied deactivations
1067 # have been derived. (Don't take this too seriously,
1068 # we really only write OPTIONS to the Makefile out of
1069 # nostalgia.)
1070
1071 if ($config{options} eq "")
1072 { $config{options} = $_; }
1073 else
1074 { $config{options} .= " ".$_; }
1075 }
1076 }
489eb740 1077
ddbe700e 1078if (keys %deprecated_options)
84f32c84
DMSP
1079 {
1080 warn "***** Deprecated options: ",
1081 join(", ", keys %deprecated_options), "\n";
1082 }
ddbe700e 1083if (keys %unsupported_options)
84f32c84
DMSP
1084 {
1085 die "***** Unsupported options: ",
1086 join(", ", keys %unsupported_options), "\n";
1087 }
b6e4dac2 1088
ac6ae8a9
RL
1089# If any %useradd entry has been set, we must check that the "make
1090# variables" haven't been set. We start by checking of any %useradd entry
fb174faa 1091# is set.
b9201360 1092if (grep { scalar @$_ > 0 } values %useradd) {
fb174faa 1093 # Hash of env / make variables names. The possible values are:
ac6ae8a9 1094 # 1 - "make vars"
fb174faa
RL
1095 # 2 - %useradd entry set
1096 # 3 - both set
ac6ae8a9 1097 my %detected_vars =
fb174faa 1098 map { my $v = 0;
ac6ae8a9 1099 $v += 1 if $cmdvars{$_};
fb174faa
RL
1100 $v += 2 if @{$useradd{$_}};
1101 $_ => $v }
1102 keys %useradd;
1103
ac6ae8a9
RL
1104 # If any of the corresponding "make variables" is set, we error
1105 if (grep { $_ & 1 } values %detected_vars) {
1106 my $names = join(', ', grep { $detected_vars{$_} > 0 }
1107 sort keys %detected_vars);
b9201360 1108 die <<"_____";
ac6ae8a9 1109***** Mixing make variables and additional compiler/linker flags as
b9201360 1110***** configure command line option is not permitted.
ac6ae8a9 1111***** Affected make variables: $names
b9201360
RL
1112_____
1113 }
1114}
1115
ac6ae8a9
RL
1116# Check through all supported command line variables to see if any of them
1117# were set, and canonicalise the values we got. If no compiler or linker
1118# flag or anything else that affects %useradd was set, we also check the
1119# environment for values.
1120my $anyuseradd =
1121 grep { defined $_ && (ref $_ ne 'ARRAY' || @$_) } values %useradd;
5b18235a 1122foreach (keys %user) {
ac6ae8a9
RL
1123 my $value = $cmdvars{$_};
1124 $value //= env($_) unless $anyuseradd;
1125 $value //=
1126 defined $user_synonyms{$_} ? $cmdvars{$user_synonyms{$_}} : undef;
1127 $value //= defined $user_synonyms{$_} ? env($user_synonyms{$_}) : undef
1128 unless $anyuseradd;
5b18235a
RL
1129
1130 if (defined $value) {
1131 if (ref $user{$_} eq 'ARRAY') {
bbe486cf
RL
1132 if ($_ eq 'CPPDEFINES' || $_ eq 'CPPINCLUDES') {
1133 $user{$_} = [ split /$list_separator_re/, $value ];
1134 } else {
1135 $user{$_} = [ $value ];
1136 }
5b18235a
RL
1137 } elsif (!defined $user{$_}) {
1138 $user{$_} = $value;
1139 }
1140 }
1141}
1142
07e4dc34 1143if (grep { /-rpath\b/ } ($user{LDFLAGS} ? @{$user{LDFLAGS}} : ())
342a1a23
RL
1144 && !$disabled{shared}
1145 && !($disabled{asan} && $disabled{msan} && $disabled{ubsan})) {
1146 die "***** Cannot simultaneously use -rpath, shared libraries, and\n",
84f32c84 1147 "***** any of asan, msan or ubsan\n";
342a1a23
RL
1148}
1149
e39795af
RL
1150# If no target was given, try guessing.
1151unless ($target) {
18062615 1152 my %system_config = OpenSSL::config::get_platform(%guess_opts, %user);
e39795af
RL
1153
1154 # The $system_config{disable} is used to populate %disabled with
1155 # entries that aren't already there.
1156 foreach ( @{$system_config{disable} // []} ) {
1157 $disabled{$_} = 'system' unless defined $disabled{$_};
1158 }
1159 delete $system_config{disable};
1160
1161 # Override config entries with stuff from the guesser.
1162 # It's assumed that this really is nothing new.
1163 %config = ( %config, %system_config );
1164 $target = $system_config{target};
1165}
1166
71ef78d7
RL
1167sub disable {
1168 my $disable_type = shift;
1169
1170 for (@_) {
1171 $disabled{$_} = $disable_type;
1172 }
1173
1174 my @tocheckfor = (@_ ? @_ : keys %disabled);
1175 while (@tocheckfor) {
1176 my %new_tocheckfor = ();
1177 my @cascade_copy = (@disable_cascades);
1178 while (@cascade_copy) {
1179 my ($test, $descendents) =
1180 (shift @cascade_copy, shift @cascade_copy);
1181 if (ref($test) eq "CODE" ? $test->() : defined($disabled{$test})) {
1182 foreach (grep { !defined($disabled{$_}) } @$descendents) {
1183 $new_tocheckfor{$_} = 1; $disabled{$_} = "cascade";
1184 }
84f32c84
DMSP
1185 }
1186 }
71ef78d7 1187 @tocheckfor = (keys %new_tocheckfor);
c569e206 1188 }
c569e206 1189}
71ef78d7 1190disable(); # First cascade run
edc032b5 1191
d63c12c6 1192our $die = sub { die @_; };
436a376b 1193if ($target eq "TABLE") {
d63c12c6 1194 local $die = sub { warn @_; };
00ae96ca 1195 foreach (sort keys %table) {
84f32c84 1196 print_table_entry($_, "TABLE");
00ae96ca
RL
1197 }
1198 exit 0;
436a376b
BM
1199}
1200
10a926c1 1201if ($target eq "LIST") {
00ae96ca 1202 foreach (sort keys %table) {
84f32c84 1203 print $_,"\n" unless $table{$_}->{template};
00ae96ca
RL
1204 }
1205 exit 0;
10a926c1
UM
1206}
1207
aaf878cc 1208if ($target eq "HASH") {
d63c12c6 1209 local $die = sub { warn @_; };
00ae96ca
RL
1210 print "%table = (\n";
1211 foreach (sort keys %table) {
84f32c84 1212 print_table_entry($_, "HASH");
00ae96ca
RL
1213 }
1214 exit 0;
aaf878cc
RL
1215}
1216
16942e08
DMSP
1217print "Configuring OpenSSL version $config{full_version} ";
1218print "for target $target\n";
64119271 1219
51cf8e0b
RL
1220if (scalar(@seed_sources) == 0) {
1221 print "Using os-specific seed configuration\n";
1222 push @seed_sources, 'os';
1223}
ddec332f
BE
1224if (scalar(grep { $_ eq 'egd' } @seed_sources) > 0) {
1225 delete $disabled{'egd'};
1226}
2805ee1e
RL
1227if (scalar(grep { $_ eq 'none' } @seed_sources) > 0) {
1228 die "Cannot seed with none and anything else" if scalar(@seed_sources) > 1;
1229 warn <<_____ if scalar(@seed_sources) == 1;
2805ee1e 1230
caa85952
DMSP
1231============================== WARNING ===============================
1232You have selected the --with-rand-seed=none option, which effectively
1233disables automatic reseeding of the OpenSSL random generator.
1234All operations depending on the random generator such as creating keys
1235will not work unless the random generator is seeded manually by the
1236application.
1237
1238Please read the 'Note on random number generation' section in the
f828ba03
RL
1239INSTALL.md instructions and the RAND_DRBG(7) manual page for more
1240details.
caa85952
DMSP
1241============================== WARNING ===============================
1242
2805ee1e
RL
1243_____
1244}
e0bf7c01 1245push @{$config{openssl_feature_defines}},
51cf8e0b 1246 map { (my $x = $_) =~ tr|[\-a-z]|[_A-Z]|; "OPENSSL_RAND_SEED_$x" }
84f32c84 1247 @seed_sources;
51cf8e0b 1248
00ae96ca 1249# Backward compatibility?
49e04548 1250if ($target =~ m/^CygWin32(-.*)$/) {
00ae96ca 1251 $target = "Cygwin".$1;
49e04548
RL
1252}
1253
906eb3d0
RL
1254# Support for legacy targets having a name starting with 'debug-'
1255my ($d, $t) = $target =~ m/^(debug-)?(.*)$/;
1256if ($d) {
1257 $config{build_type} = "debug";
1258
1259 # If we do not find debug-foo in the table, the target is set to foo.
1260 if (!$table{$target}) {
84f32c84 1261 $target = $t;
906eb3d0
RL
1262 }
1263}
4e360445 1264
081436bf
RL
1265if ($target) {
1266 # It's possible that we have different config targets for specific
1267 # toolchains, so we try to detect them, and go for the plain config
1268 # target if not.
1269 my $found;
1270 foreach ( ( "$target-$user{CC}", "$target", undef ) ) {
1271 $found=$_ if $table{$_} && !$table{$_}->{template};
1272 last if $found;
1273 }
1274 $target = $found;
1275} else {
1276 # If we don't have a config target now, we try the C compiler as we
1277 # fallback
1278 my $cc = $user{CC} // 'cc';
1279 $target = $cc if $table{$cc} && !$table{$cc}->{template};
1280}
1281
1282&usage unless $target;
4e360445 1283
18062615
RL
1284exit 0 if $dryrun; # From older 'config'
1285
906eb3d0
RL
1286$config{target} = $target;
1287my %target = resolve_config($target);
1288
abe256e7
RL
1289foreach (keys %target_attr_translate) {
1290 $target{$target_attr_translate{$_}} = $target{$_}
1291 if $target{$_};
1292 delete $target{$_};
1293}
1294
793077d0
RL
1295%target = ( %{$table{DEFAULTS}}, %target );
1296
906eb3d0
RL
1297my %conf_files = map { $_ => 1 } (@{$target{_conf_fname_int}});
1298$config{conf_files} = [ sort keys %conf_files ];
906eb3d0 1299
71ef78d7
RL
1300# Using sub disable within these loops may prove fragile, so we run
1301# a cascade afterwards
906eb3d0
RL
1302foreach my $feature (@{$target{disable}}) {
1303 if (exists $deprecated_disablables{$feature}) {
1304 warn "***** config $target disables deprecated feature $feature\n";
1305 } elsif (!grep { $feature eq $_ } @disablables) {
1306 die "***** config $target disables unknown feature $feature\n";
1307 }
1308 $disabled{$feature} = 'config';
1309}
1310foreach my $feature (@{$target{enable}}) {
7a8a35ff 1311 if ("default" eq ($disabled{$feature} // "")) {
906eb3d0
RL
1312 if (exists $deprecated_disablables{$feature}) {
1313 warn "***** config $target enables deprecated feature $feature\n";
1314 } elsif (!grep { $feature eq $_ } @disablables) {
1315 die "***** config $target enables unknown feature $feature\n";
1316 }
7a8a35ff 1317 delete $disabled{$feature};
906eb3d0
RL
1318 }
1319}
b19fe714
RL
1320
1321# If uplink_arch isn't defined, disable uplink
1322$disabled{uplink} = 'no uplink_arch' unless (defined $target{uplink_arch});
e6f98ae4
RL
1323# If asm_arch isn't defined, disable asm
1324$disabled{asm} = 'no asm_arch' unless (defined $target{asm_arch});
b19fe714 1325
71ef78d7 1326disable(); # Run a cascade now
906eb3d0 1327
abe256e7
RL
1328$target{CXXFLAGS}//=$target{CFLAGS} if $target{CXX};
1329$target{cxxflags}//=$target{cflags} if $target{CXX};
9dd4ed28 1330$target{exe_extension}=".exe" if ($config{target} eq "DJGPP");
107b5792 1331$target{exe_extension}=".pm" if ($config{target} =~ /vos/);
e987f9f2 1332
9e265322
RL
1333# Fill %config with values from %user, and in case those are undefined or
1334# empty, use values from %target (acting as a default).
5b18235a 1335foreach (keys %user) {
5b18235a
RL
1336 my $ref_type = ref $user{$_};
1337
1338 # Temporary function. Takes an intended ref type (empty string or "ARRAY")
1339 # and a value that's to be coerced into that type.
1340 my $mkvalue = sub {
1341 my $type = shift;
1342 my $value = shift;
1343 my $undef_p = shift;
1344
1345 die "Too many arguments for \$mkvalue" if @_;
1346
1347 while (ref $value eq 'CODE') {
1348 $value = $value->();
1349 }
1350
1351 if ($type eq 'ARRAY') {
1352 return undef unless defined $value;
1353 return undef if ref $value ne 'ARRAY' && !$value;
1354 return undef if ref $value eq 'ARRAY' && !@$value;
1355 return [ $value ] unless ref $value eq 'ARRAY';
1356 }
1357 return undef unless $value;
1358 return $value;
1359 };
1360
abe256e7 1361 $config{$_} =
5b18235a 1362 $mkvalue->($ref_type, $user{$_})
abe256e7
RL
1363 || $mkvalue->($ref_type, $target{$_});
1364 delete $config{$_} unless defined $config{$_};
5b18235a 1365}
aaf878cc 1366
c1a09254
RL
1367# Finish up %config by appending things the user gave us on the command line
1368# apart from "make variables"
1369foreach (keys %useradd) {
1370 # The must all be lists, so we assert that here
1371 die "internal error: \$useradd{$_} isn't an ARRAY\n"
1372 unless ref $useradd{$_} eq 'ARRAY';
1373
1374 if (defined $config{$_}) {
1375 push @{$config{$_}}, @{$useradd{$_}};
1376 } else {
1377 $config{$_} = [ @{$useradd{$_}} ];
1378 }
1379}
1380# At this point, we can forget everything about %user and %useradd,
1381# because it's now all been merged into the corresponding $config entry
1382
0221b080
TC
1383if (grep { $_ =~ /(?:^|\s)-static(?:\s|$)/ } @{$config{LDFLAGS}}) {
1384 disable('static', 'pic', 'threads');
1385}
1386
8b5156d1 1387# Allow overriding the build file name
5b18235a 1388$config{build_file} = env('BUILDFILE') || $target{build_file} || "Makefile";
bd5192b1 1389
291e94df
RL
1390# Make sure build_scheme is consistent.
1391$target{build_scheme} = [ $target{build_scheme} ]
1392 if ref($target{build_scheme}) ne "ARRAY";
1393
ddf1847d
RL
1394my ($builder, $builder_platform, @builder_opts) =
1395 @{$target{build_scheme}};
1396
d192a3aa
RL
1397foreach my $checker (($builder_platform."-".$target{build_file}."-checker.pm",
1398 $builder_platform."-checker.pm")) {
1399 my $checker_path = catfile($srcdir, "Configurations", $checker);
1400 if (-f $checker_path) {
1401 my $fn = $ENV{CONFIGURE_CHECKER_WARN}
1402 ? sub { warn $@; } : sub { die $@; };
1403 if (! do $checker_path) {
1404 if ($@) {
1405 $fn->($@);
1406 } elsif ($!) {
1407 $fn->($!);
1408 } else {
1409 $fn->("The detected tools didn't match the platform\n");
1410 }
1411 }
1412 last;
1413 }
1414}
1415
488e2b0f
RL
1416push @{$config{defines}}, "NDEBUG" if $config{build_type} eq "release";
1417
abe256e7 1418if ($target =~ /^mingw/ && `$config{CC} --target-help 2>&1` =~ m/-mno-cygwin/m)
84f32c84
DMSP
1419 {
1420 push @{$config{cflags}}, "-mno-cygwin";
1421 push @{$config{cxxflags}}, "-mno-cygwin" if $config{CXX};
1422 push @{$config{shared_ldflag}}, "-mno-cygwin";
1423 }
cbecd29a 1424
5b18235a 1425if ($target =~ /linux.*-mips/ && !$disabled{asm}
c1a09254 1426 && !grep { $_ !~ /-m(ips|arch=)/ } (@{$config{CFLAGS}})) {
84f32c84
DMSP
1427 # minimally required architecture flags for assembly modules
1428 my $value;
1429 $value = '-mips2' if ($target =~ /mips32/);
1430 $value = '-mips3' if ($target =~ /mips64/);
1431 unshift @{$config{cflags}}, $value;
1432 unshift @{$config{cxxflags}}, $value if $config{CXX};
63d8834c
AP
1433}
1434
9c62a279
RL
1435# If threads aren't disabled, check how possible they are
1436unless ($disabled{threads}) {
1437 if ($auto_threads) {
1438 # Enabled by default, disable it forcibly if unavailable
1439 if ($target{thread_scheme} eq "(unknown)") {
71ef78d7 1440 disable("unavailable", 'threads');
9c62a279
RL
1441 }
1442 } else {
8483a003 1443 # The user chose to enable threads explicitly, let's see
9c62a279
RL
1444 # if there's a chance that's possible
1445 if ($target{thread_scheme} eq "(unknown)") {
1446 # If the user asked for "threads" and we don't have internal
1447 # knowledge how to do it, [s]he is expected to provide any
1448 # system-dependent compiler options that are necessary. We
1449 # can't truly check that the given options are correct, but
1450 # we expect the user to know what [s]He is doing.
c1a09254 1451 if (!@{$config{CFLAGS}} && !@{$config{CPPDEFINES}}) {
9c62a279
RL
1452 die "You asked for multi-threading support, but didn't\n"
1453 ,"provide any system-specific compiler options\n";
1454 }
1455 }
1456 }
1457}
1458
bacc3081
RL
1459# Find out if clang's sanitizers have been enabled with -fsanitize
1460# flags and ensure that the corresponding %disabled elements area
1461# removed to reflect that the sanitizers are indeed enabled.
1462my %detected_sanitizers = ();
1463foreach (grep /^-fsanitize=/, @{$config{CFLAGS} || []}) {
1464 (my $checks = $_) =~ s/^-fsanitize=//;
1465 foreach (split /,/, $checks) {
1466 my $d = { address => 'asan',
1467 undefined => 'ubsan',
1468 memory => 'msan' } -> {$_};
1469 next unless defined $d;
1470
1471 $detected_sanitizers{$d} = 1;
1472 if (defined $disabled{$d}) {
1473 die "***** Conflict between disabling $d and enabling $_ sanitizer"
1474 if $disabled{$d} ne "default";
1475 delete $disabled{$d};
1476 }
1477 }
1478}
1479
9c62a279
RL
1480# If threads still aren't disabled, add a C macro to ensure the source
1481# code knows about it. Any other flag is taken care of by the configs.
1482unless($disabled{threads}) {
e0bf7c01 1483 push @{$config{openssl_feature_defines}}, "OPENSSL_THREADS";
9c62a279 1484}
e452de9d 1485
8c3bc594 1486my $no_shared_warn=0;
e1f5a92d 1487if (($target{shared_target} // '') eq "")
84f32c84
DMSP
1488 {
1489 $no_shared_warn = 1
1490 if (!$disabled{shared} || !$disabled{"dynamic-engine"});
71ef78d7 1491 disable('no-shared-target', 'pic');
84f32c84 1492 }
b436a982 1493
19ab5790 1494if ($disabled{"dynamic-engine"}) {
343ec2b0 1495 $config{dynamic_engines} = 0;
19ab5790 1496} else {
19ab5790 1497 $config{dynamic_engines} = 1;
343ec2b0 1498}
ecd45314 1499
bacc3081 1500unless ($disabled{asan} || defined $detected_sanitizers{asan}) {
5b18235a 1501 push @{$config{cflags}}, "-fsanitize=address";
c38bb727
BL
1502}
1503
bacc3081 1504unless ($disabled{ubsan} || defined $detected_sanitizers{ubsan}) {
f430ba31 1505 # -DPEDANTIC or -fnosanitize=alignment may also be required on some
c38bb727 1506 # platforms.
5b18235a 1507 push @{$config{cflags}}, "-fsanitize=undefined", "-fno-sanitize-recover=all";
c38bb727
BL
1508}
1509
bacc3081 1510unless ($disabled{msan} || defined $detected_sanitizers{msan}) {
5b18235a 1511 push @{$config{cflags}}, "-fsanitize=memory";
29df3061
EK
1512}
1513
65cc6d5c 1514unless ($disabled{"fuzz-libfuzzer"} && $disabled{"fuzz-afl"}
29df3061 1515 && $disabled{asan} && $disabled{ubsan} && $disabled{msan}) {
5b18235a 1516 push @{$config{cflags}}, "-fno-omit-frame-pointer", "-g";
abe256e7 1517 push @{$config{cxxflags}}, "-fno-omit-frame-pointer", "-g" if $config{CXX};
c38bb727 1518}
c313e32a
AP
1519#
1520# Platform fix-ups
1521#
ae48242c
RL
1522
1523# This saves the build files from having to check
1524if ($disabled{pic})
84f32c84
DMSP
1525 {
1526 foreach (qw(shared_cflag shared_cxxflag shared_cppflag
1527 shared_defines shared_includes shared_ldflag
1528 module_cflags module_cxxflags module_cppflags
1529 module_defines module_includes module_lflags))
1530 {
1531 delete $config{$_};
1532 $target{$_} = "";
1533 }
1534 }
4f16039e 1535else
84f32c84
DMSP
1536 {
1537 push @{$config{lib_defines}}, "OPENSSL_PIC";
1538 }
ae48242c 1539
291e94df 1540if ($target{sys_id} ne "")
84f32c84
DMSP
1541 {
1542 push @{$config{openssl_sys_defines}}, "OPENSSL_SYS_$target{sys_id}";
1543 }
cf1b7d96 1544
e373c70a
RL
1545my %predefined_C = compiler_predefined($config{CROSS_COMPILE}.$config{CC});
1546my %predefined_CXX = $config{CXX}
1547 ? compiler_predefined($config{CROSS_COMPILE}.$config{CXX})
1548 : ();
54cf3b98 1549
09840412
AP
1550unless ($disabled{asm}) {
1551 # big endian systems can use ELFv2 ABI
1552 if ($target eq "linux-ppc64") {
1553 $target{perlasm_scheme} = "linux64v2" if ($predefined_C{_CALL_ELF} == 2);
1554 }
1555}
1556
fe191b49 1557# Check for makedepend capabilities.
6d75a83c 1558if (!$disabled{makedepend}) {
2e535eb5
RL
1559 # If the attribute makedep_scheme is defined, then we assume that the
1560 # config target and its associated build file are programmed to deal
1561 # with it.
1562 # If makedep_scheme is undefined, we go looking for GCC compatible
1563 # dependency making, and if that's not available, we try to fall back
1564 # on 'makedepend'.
1565 if ($target{makedep_scheme}) {
1566 $config{makedep_scheme} = $target{makedep_scheme};
1567 # If the makedepcmd attribute is defined, copy it. If not, the
1568 # build files will have to fend for themselves.
1569 $config{makedepcmd} = $target{makedepcmd} if $target{makedepcmd};
e373c70a 1570 } elsif (($predefined_C{__GNUC__} // -1) >= 3
84f32c84 1571 && !($predefined_C{__APPLE_CC__} && !$predefined_C{__clang__})) {
fe191b49 1572 # We know that GNU C version 3 and up as well as all clang
717f308e
TS
1573 # versions support dependency generation, but Xcode did not
1574 # handle $cc -M before clang support (but claims __GNUC__ = 3)
2e535eb5 1575 $config{makedep_scheme} = 'gcc';
6d75a83c 1576 } else {
2e535eb5
RL
1577 # In all other cases, we look for 'makedepend', and set the
1578 # makedep_scheme value if we found it.
1579 $config{makedepcmd} = which('makedepend');
1580 $config{makedep_scheme} = 'makedepend' if $config{makedepcmd};
54cf3b98 1581 }
2e535eb5
RL
1582
1583 # If no depend scheme is set, we disable makedepend
1584 disable('unavailable', 'makedepend') unless $config{makedep_scheme};
f1f07a23 1585}
8ed40b83 1586
e373c70a 1587if (!$disabled{asm} && !$predefined_C{__MACH__} && $^O ne 'VMS') {
0ad4078c 1588 # probe for -Wa,--noexecstack option...
e373c70a 1589 if ($predefined_C{__clang__}) {
0ad4078c
AP
1590 # clang has builtin assembler, which doesn't recognize --help,
1591 # but it apparently recognizes the option in question on all
1592 # supported platforms even when it's meaningless. In other words
1593 # probe would fail, but probed option always accepted...
1594 push @{$config{cflags}}, "-Wa,--noexecstack", "-Qunused-arguments";
8e5da579 1595 } else {
0ad4078c
AP
1596 my $cc = $config{CROSS_COMPILE}.$config{CC};
1597 open(PIPE, "$cc -Wa,--help -c -o null.$$.o -x assembler /dev/null 2>&1 |");
1598 while(<PIPE>) {
1599 if (m/--noexecstack/) {
1600 push @{$config{cflags}}, "-Wa,--noexecstack";
1601 last;
1602 }
1603 }
1604 close(PIPE);
1605 unlink("null.$$.o");
1606 }
1607}
7d130f68
RL
1608
1609# Deal with bn_ops ###################################################
1610
84f32c84 1611$config{bn_ll} =0;
7d130f68 1612my $def_int="unsigned int";
84f32c84 1613$config{rc4_int} =$def_int;
b4f35e5e 1614($config{b64l},$config{b64},$config{b32})=(0,0,1);
7d130f68 1615
94af0cd7 1616my $count = 0;
7d130f68 1617foreach (sort split(/\s+/,$target{bn_ops})) {
94af0cd7 1618 $count++ if /SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT/;
84f32c84
DMSP
1619 $config{bn_ll}=1 if $_ eq 'BN_LLONG';
1620 $config{rc4_int}="unsigned char" if $_ eq 'RC4_CHAR';
94af0cd7 1621 ($config{b64l},$config{b64},$config{b32})
84f32c84 1622 =(0,1,0) if $_ eq 'SIXTY_FOUR_BIT';
94af0cd7 1623 ($config{b64l},$config{b64},$config{b32})
84f32c84 1624 =(1,0,0) if $_ eq 'SIXTY_FOUR_BIT_LONG';
94af0cd7 1625 ($config{b64l},$config{b64},$config{b32})
84f32c84 1626 =(0,0,1) if $_ eq 'THIRTY_TWO_BIT';
7d130f68 1627}
94af0cd7
RS
1628die "Exactly one of SIXTY_FOUR_BIT|SIXTY_FOUR_BIT_LONG|THIRTY_TWO_BIT can be set in bn_ops\n"
1629 if $count > 1;
7d130f68 1630
a6a4d0ac
RL
1631$config{api} = $config{major} * 10000 + $config{minor} * 100
1632 unless $config{api};
b05d6327
RL
1633foreach (keys %$apitable) {
1634 $disabled{"deprecated-$_"} = "deprecation"
1635 if $disabled{deprecated} && $config{api} >= $apitable->{$_};
1636}
1637
1638disable(); # Run a cascade now
7d130f68
RL
1639
1640# Hack cflags for better warnings (dev option) #######################
1641
fa153b57
RL
1642# "Stringify" the C and C++ flags string. This permits it to be made part of
1643# a string and works as well on command lines.
5b18235a
RL
1644$config{cflags} = [ map { (my $x = $_) =~ s/([\\\"])/\\$1/g; $x }
1645 @{$config{cflags}} ];
fa153b57 1646$config{cxxflags} = [ map { (my $x = $_) =~ s/([\\\"])/\\$1/g; $x }
abe256e7 1647 @{$config{cxxflags}} ] if $config{CXX};
b436a982 1648
fcd2d5a6 1649$config{openssl_api_defines} = [
b05d6327 1650 "OPENSSL_CONFIGURED_API=".$config{api},
fcd2d5a6 1651];
98186eb4 1652
3b437400 1653my @strict_warnings_collection=();
0c28f277 1654if ($strict_warnings)
84f32c84
DMSP
1655 {
1656 my $wopt;
1657 my $gccver = $predefined_C{__GNUC__} // -1;
6d50589c 1658
b4a7b4ec
RL
1659 if ($gccver >= 4)
1660 {
1661 push @strict_warnings_collection, @gcc_devteam_warn;
1662 push @strict_warnings_collection, @clang_devteam_warn
1663 if (defined($predefined_C{__clang__}));
1664 }
1665 elsif ($config{target} =~ /^VC-/)
1666 {
1667 push @strict_warnings_collection, @cl_devteam_warn;
1668 }
1669 else
1670 {
1671 warn "WARNING --strict-warnings requires gcc[>=4] or gcc-alike, or MSVC"
1672 }
84f32c84 1673 }
4650d10f 1674
3b437400
RL
1675$config{CFLAGS} = [ map { $_ eq '--ossl-strict-warnings'
1676 ? @strict_warnings_collection
1677 : ( $_ ) }
1678 @{$config{CFLAGS}} ];
ef8ca6bd 1679
c91a0a83
EK
1680unless ($disabled{afalgeng}) {
1681 $config{afalgeng}="";
9e381e8a 1682 if (grep { $_ eq 'afalgeng' } @{$target{enable}}) {
79fff39d 1683 my $minver = 4*10000 + 1*100 + 0;
abe256e7 1684 if ($config{CROSS_COMPILE} eq "") {
79fff39d
RL
1685 my $verstr = `uname -r`;
1686 my ($ma, $mi1, $mi2) = split("\\.", $verstr);
1687 ($mi2) = $mi2 =~ /(\d+)/;
1688 my $ver = $ma*10000 + $mi1*100 + $mi2;
1689 if ($ver < $minver) {
71ef78d7 1690 disable('too-old-kernel', 'afalgeng');
79fff39d
RL
1691 } else {
1692 push @{$config{engdirs}}, "afalg";
1693 }
68dc37c1 1694 } else {
71ef78d7 1695 disable('cross-compiling', 'afalgeng');
6cba4a66 1696 }
79fff39d 1697 } else {
71ef78d7 1698 disable('not-linux', 'afalgeng');
7f458a48 1699 }
1700}
8da00a38 1701
76d0a74b
RL
1702unless ($disabled{devcryptoeng}) {
1703 if ($target =~ m/^BSD/) {
1704 my $maxver = 5*100 + 7;
1705 my $sysstr = `uname -s`;
1706 my $verstr = `uname -r`;
1707 $sysstr =~ s|\R$||;
1708 $verstr =~ s|\R$||;
1709 my ($ma, $mi, @rest) = split m|\.|, $verstr;
1710 my $ver = $ma*100 + $mi;
1711 if ($sysstr eq 'OpenBSD' && $ver >= $maxver) {
1712 disable('too-new-kernel', 'devcryptoeng');
1713 }
1714 }
1715}
1716
69495e3d
BP
1717unless ($disabled{ktls}) {
1718 $config{ktls}="";
1719 if ($target =~ m/^linux/) {
1720 my $usr = "/usr/$config{cross_compile_prefix}";
1721 chop($usr);
1722 if ($config{cross_compile_prefix} eq "") {
1723 $usr = "/usr";
1724 }
1725 my $minver = (4 << 16) + (13 << 8) + 0;
1726 my @verstr = split(" ",`cat $usr/include/linux/version.h | grep LINUX_VERSION_CODE`);
1727
1728 if ($verstr[2] < $minver) {
71ef78d7 1729 disable('too-old-kernel', 'ktls');
69495e3d 1730 }
2111f5c2
AG
1731 } elsif ($target =~ m/^BSD/) {
1732 my $cc = $config{CROSS_COMPILE}.$config{CC};
1733 system("printf '#include <sys/types.h>\n#include <sys/ktls.h>' | $cc -E - >/dev/null 2>&1");
1734 if ($? != 0) {
1735 disable('too-old-freebsd', 'ktls');
1736 }
69495e3d 1737 } else {
2111f5c2 1738 disable('not-linux-or-freebsd', 'ktls');
69495e3d
BP
1739 }
1740}
1741
1742push @{$config{openssl_other_defines}}, "OPENSSL_NO_KTLS" if ($disabled{ktls});
1743
8f0dd6d9
RL
1744# Get the extra flags used when building shared libraries and modules. We
1745# do this late because some of them depend on %disabled.
1746
1747# Make the flags to build DSOs the same as for shared libraries unless they
1748# are already defined
1749$target{module_cflags} = $target{shared_cflag} unless defined $target{module_cflags};
1750$target{module_cxxflags} = $target{shared_cxxflag} unless defined $target{module_cxxflags};
1751$target{module_ldflags} = $target{shared_ldflag} unless defined $target{module_ldflags};
1752{
1753 my $shared_info_pl =
1754 catfile(dirname($0), "Configurations", "shared-info.pl");
1755 my %shared_info = read_eval_file($shared_info_pl);
1756 push @{$target{_conf_fname_int}}, $shared_info_pl;
1757 my $si = $target{shared_target};
1758 while (ref $si ne "HASH") {
1759 last if ! defined $si;
1760 if (ref $si eq "CODE") {
1761 $si = $si->();
1762 } else {
1763 $si = $shared_info{$si};
1764 }
1765 }
1766
1767 # Some of the 'shared_target' values don't have any entries in
1768 # %shared_info. That's perfectly fine, AS LONG AS the build file
1769 # template knows how to handle this. That is currently the case for
1770 # Windows and VMS.
1771 if (defined $si) {
1772 # Just as above, copy certain shared_* attributes to the corresponding
1773 # module_ attribute unless the latter is already defined
1774 $si->{module_cflags} = $si->{shared_cflag} unless defined $si->{module_cflags};
1775 $si->{module_cxxflags} = $si->{shared_cxxflag} unless defined $si->{module_cxxflags};
1776 $si->{module_ldflags} = $si->{shared_ldflag} unless defined $si->{module_ldflags};
1777 foreach (sort keys %$si) {
1778 $target{$_} = defined $target{$_}
1779 ? add($si->{$_})->($target{$_})
1780 : $si->{$_};
1781 }
1782 }
1783}
1784
1785# ALL MODIFICATIONS TO %disabled, %config and %target MUST BE DONE FROM HERE ON
5b18235a 1786
8c06d719
RL
1787######################################################################
1788# Build up information for skipping certain directories depending on disabled
1789# features, as well as setting up macros for disabled features.
1790
1791# This is a tentative database of directories to skip. Some entries may not
1792# correspond to anything real, but that's ok, they will simply be ignored.
1793# The actual processing of these entries is done in the build.info lookup
1794# loop further down.
1795#
1796# The key is a Unix formatted path in the source tree, the value is an index
1797# into %disabled_info, so any existing path gets added to a corresponding
1798# 'skipped' entry in there with the list of skipped directories.
1799my %skipdir = ();
1800my %disabled_info = (); # For configdata.pm
1801foreach my $what (sort keys %disabled) {
1802 # There are deprecated disablables that translate to themselves.
1803 # They cause disabling cascades, but should otherwise not regiter.
1804 next if $deprecated_disablables{$what};
b05d6327
RL
1805 # The generated $disabled{"deprecated-x.y"} entries are special
1806 # and treated properly elsewhere
1807 next if $what =~ m|^deprecated-|;
8c06d719
RL
1808
1809 $config{options} .= " no-$what";
1810
1811 if (!grep { $what eq $_ } ( 'buildtest-c++', 'fips', 'threads', 'shared',
1812 'module', 'pic', 'dynamic-engine', 'makedepend',
1813 'zlib-dynamic', 'zlib', 'sse2', 'legacy' )) {
1814 (my $WHAT = uc $what) =~ s|-|_|g;
1815 my $skipdir = $what;
1816
1817 # fix-up crypto/directory name(s)
1818 $skipdir = "ripemd" if $what eq "rmd160";
1819 $skipdir = "whrlpool" if $what eq "whirlpool";
1820
1821 my $macro = $disabled_info{$what}->{macro} = "OPENSSL_NO_$WHAT";
1822 push @{$config{openssl_feature_defines}}, $macro;
1823
1824 $skipdir{engines} = $what if $what eq 'engine';
1825 $skipdir{"crypto/$skipdir"} = $what
1826 unless $what eq 'async' || $what eq 'err' || $what eq 'dso';
1827 }
1828}
1829
1830if ($disabled{"dynamic-engine"}) {
1831 push @{$config{openssl_feature_defines}}, "OPENSSL_NO_DYNAMIC_ENGINE";
1832} else {
1833 push @{$config{openssl_feature_defines}}, "OPENSSL_NO_STATIC_ENGINE";
1834}
1835
9fe2bb77
RL
1836# If we use the unified build, collect information from build.info files
1837my %unified_info = ();
1838
2b6b606c 1839my $buildinfo_debug = defined($ENV{CONFIGURE_DEBUG_BUILDINFO});
ddf1847d 1840if ($builder eq "unified") {
1935a586 1841 use Text::Template 1.46;
9fe2bb77 1842
9fe2bb77 1843 sub cleandir {
2e963849 1844 my $base = shift;
9fe2bb77 1845 my $dir = shift;
2e963849
RL
1846 my $relativeto = shift || ".";
1847
1848 $dir = catdir($base,$dir) unless isabsolute($dir);
9fe2bb77 1849
ec182ef0
RL
1850 # Make sure the directories we're building in exists
1851 mkpath($dir);
1852
2e963849 1853 my $res = abs2rel(absolutedir($dir), rel2abs($relativeto));
9fe2bb77
RL
1854 #print STDERR "DEBUG[cleandir]: $dir , $base => $res\n";
1855 return $res;
1856 }
1857
1858 sub cleanfile {
2e963849 1859 my $base = shift;
9fe2bb77 1860 my $file = shift;
2e963849
RL
1861 my $relativeto = shift || ".";
1862
1863 $file = catfile($base,$file) unless isabsolute($file);
1864
9fe2bb77
RL
1865 my $d = dirname($file);
1866 my $f = basename($file);
1867
ec182ef0
RL
1868 # Make sure the directories we're building in exists
1869 mkpath($d);
1870
2e963849 1871 my $res = abs2rel(catfile(absolutedir($d), $f), rel2abs($relativeto));
9fe2bb77
RL
1872 #print STDERR "DEBUG[cleanfile]: $d , $f => $res\n";
1873 return $res;
1874 }
1875
1967a42e
RL
1876 # Store the name of the template file we will build the build file from
1877 # in %config. This may be useful for the build file itself.
1878 my @build_file_template_names =
84f32c84
DMSP
1879 ( $builder_platform."-".$target{build_file}.".tmpl",
1880 $target{build_file}.".tmpl" );
1967a42e
RL
1881 my @build_file_templates = ();
1882
1883 # First, look in the user provided directory, if given
7ecdf18d 1884 if (defined env($local_config_envname)) {
84f32c84
DMSP
1885 @build_file_templates =
1886 map {
1887 if ($^O eq 'VMS') {
1888 # VMS environment variables are logical names,
1889 # which can be used as is
1890 $local_config_envname . ':' . $_;
1891 } else {
1892 catfile(env($local_config_envname), $_);
1893 }
1894 }
1895 @build_file_template_names;
1967a42e
RL
1896 }
1897 # Then, look in our standard directory
1898 push @build_file_templates,
84f32c84
DMSP
1899 ( map { cleanfile($srcdir, catfile("Configurations", $_), $blddir) }
1900 @build_file_template_names );
1967a42e
RL
1901
1902 my $build_file_template;
1903 for $_ (@build_file_templates) {
84f32c84 1904 $build_file_template = $_;
1967a42e
RL
1905 last if -f $build_file_template;
1906
1907 $build_file_template = undef;
1908 }
1909 if (!defined $build_file_template) {
84f32c84 1910 die "*** Couldn't find any of:\n", join("\n", @build_file_templates), "\n";
1967a42e
RL
1911 }
1912 $config{build_file_templates}
8258975c
RL
1913 = [ cleanfile($srcdir, catfile("Configurations", "common0.tmpl"),
1914 $blddir),
2660b7cf 1915 $build_file_template ];
1967a42e 1916
7f73eafe 1917 my @build_dirs = ( [ ] ); # current directory
9fe2bb77 1918
2e0956ba
RL
1919 $config{build_infos} = [ ];
1920
507f8380
RL
1921 # We want to detect configdata.pm in the source tree, so we
1922 # don't use it if the build tree is different.
1923 my $src_configdata = cleanfile($srcdir, "configdata.pm", $blddir);
1924
1925 # Any source file that we recognise is placed in this hash table, with
1926 # the list of its intended destinations as value. When everything has
1927 # been collected, there's a routine that checks that these source files
1928 # exist, or if they are generated, that the generator exists.
1929 my %check_exist = ();
1930 my %check_generate = ();
1931
d201dbc9 1932 my %ordinals = ();
7f73eafe
RL
1933 while (@build_dirs) {
1934 my @curd = @{shift @build_dirs};
1935 my $sourced = catdir($srcdir, @curd);
1936 my $buildd = catdir($blddir, @curd);
9fe2bb77 1937
75d47db4
RL
1938 my $unixdir = join('/', @curd);
1939 if (exists $skipdir{$unixdir}) {
1940 my $what = $skipdir{$unixdir};
1941 push @{$disabled_info{$what}->{skipped}}, catdir(@curd);
1942 next;
1943 }
1944
dca99383 1945 mkpath($buildd);
9fe2bb77 1946
7f73eafe 1947 my $f = 'build.info';
9fe2bb77
RL
1948 # The basic things we're trying to build
1949 my @programs = ();
1950 my @libraries = ();
1842f369 1951 my @modules = ();
9fe2bb77 1952 my @scripts = ();
9fe2bb77 1953
9fe2bb77 1954 my %sources = ();
2a08d1a0 1955 my %shared_sources = ();
9fe2bb77 1956 my %includes = ();
b96ab5e6 1957 my %defines = ();
9fe2bb77 1958 my %depends = ();
ae4c7450 1959 my %generate = ();
d594d2e1 1960 my %imagedocs = ();
829f86bb
RL
1961 my %htmldocs = ();
1962 my %mandocs = ();
9fe2bb77 1963
26fe9b07
RL
1964 # Support for $variablename in build.info files.
1965 # Embedded perl code is the ultimate master, still. If its output
1966 # contains a dollar sign, it had better be escaped, or it will be
1967 # taken for a variable name prefix.
1968 my %variables = ();
e4292179
RL
1969 # Variable name syntax
1970 my $variable_name_re = qr/(?P<VARIABLE>[[:alpha:]][[:alnum:]_]*)/;
1971 # Value modifier syntaxes
1972 my $variable_subst_re = qr/\/(?P<RE>(?:\\\/|.)*?)\/(?P<SUBST>.*?)/;
4975e8b4
RL
1973 # Variable reference
1974 my $variable_simple_re = qr/(?<!\\)\$${variable_name_re}/;
1975 my $variable_w_mod_re =
1976 qr/(?<!\\)\$\{${variable_name_re}(?P<MOD>(?:\\\/|.)*?)\}/;
1977 # Tie it all together
1978 my $variable_re = qr/${variable_simple_re}|${variable_w_mod_re}/;
1979
26fe9b07
RL
1980 my $expand_variables = sub {
1981 my $value = '';
1982 my $value_rest = shift;
1983
03f30c55
RL
1984 if ($ENV{CONFIGURE_DEBUG_VARIABLE_EXPAND}) {
1985 print STDERR
e4292179 1986 "DEBUG[\$expand_variables] Parsed '$value_rest' ...\n"
03f30c55 1987 }
e4292179 1988
4975e8b4
RL
1989 while ($value_rest =~ /${variable_re}/) {
1990 # We must save important regexp values, because the next
1991 # regexp clears them
1992 my $mod = $+{MOD};
e4292179
RL
1993 my $variable_value = $variables{$+{VARIABLE}};
1994
4975e8b4
RL
1995 $value_rest = $';
1996 $value .= $`;
1997
e4292179 1998 # Process modifier expressions, if present
4975e8b4
RL
1999 if (defined $mod) {
2000 if ($mod =~ /^${variable_subst_re}$/) {
2001 my $re = $+{RE};
2002 my $subst = $+{SUBST};
2003
2004 $variable_value =~ s/\Q$re\E/$subst/g;
2005
2006 if ($ENV{CONFIGURE_DEBUG_VARIABLE_EXPAND}) {
2007 print STDERR
2008 "DEBUG[\$expand_variables] ... and substituted ",
2009 "'$re' with '$subst'\n";
2010 }
e4292179
RL
2011 }
2012 }
2013
2014 $value .= $variable_value;
26fe9b07 2015 }
03f30c55
RL
2016 if ($ENV{CONFIGURE_DEBUG_VARIABLE_EXPAND}) {
2017 print STDERR
e4292179 2018 "DEBUG[\$expand_variables] ... into: '$value$value_rest'\n";
03f30c55 2019 }
26fe9b07
RL
2020 return $value . $value_rest;
2021 };
2022
285daccd
RL
2023 # Support for attributes in build.info files
2024 my %attributes = ();
2025 my $handle_attributes = sub {
2026 my $attr_str = shift;
2027 my $ref = shift;
2028 my @goals = @_;
2029
2030 return unless defined $attr_str;
2031
2032 my @a = tokenize($attr_str, qr|\s*,\s*|);
2033 foreach my $a (@a) {
2034 my $ac = 1;
2035 my $ak = $a;
2036 my $av = 1;
ea4ee152 2037 if ($a =~ m|^(!)?(.*?)\s* = \s*(.*?)$|x) {
285daccd 2038 $ac = ! $1;
ea4ee152
RL
2039 $ak = $2;
2040 $av = $3;
285daccd
RL
2041 }
2042 foreach my $g (@goals) {
2043 if ($ac) {
2044 $$ref->{$g}->{$ak} = $av;
2045 } else {
2046 delete $$ref->{$g}->{$ak};
2047 }
2048 }
2049 }
2050 };
2051
ad5be194
RL
2052 # Support for pushing values on multiple indexes of a given hash
2053 # array.
2054 my $push_to = sub {
2055 my $valueref = shift;
2056 my $index_str = shift; # May be undef or empty
2057 my $attrref = shift; # May be undef
2058 my $attr_str = shift;
2059 my @values = @_;
2060
2061 if (defined $index_str) {
2062 my @indexes = ( '' );
2063 if ($index_str !~ m|^\s*$|) {
2064 @indexes = tokenize($index_str);
2065 }
2066 foreach (@indexes) {
2067 push @{$valueref->{$_}}, @values;
2068 if (defined $attrref) {
2069 $handle_attributes->($attr_str, \$$attrref->{$_},
2070 @values);
2071 }
2072 }
2073 } else {
2074 push @$valueref, @values;
2075 $handle_attributes->($attr_str, $attrref, @values)
2076 if defined $attrref;
2077 }
2078 };
2079
829f86bb
RL
2080 if ($buildinfo_debug) {
2081 print STDERR "DEBUG: Reading ",catfile($sourced, $f),"\n";
2082 }
2e0956ba 2083 push @{$config{build_infos}}, catfile(abs2rel($sourced, $blddir), $f);
cb6afcd6
RL
2084 my $template =
2085 Text::Template->new(TYPE => 'FILE',
2086 SOURCE => catfile($sourced, $f),
2087 PREPEND => qq{use lib "$FindBin::Bin/util/perl";});
9fe2bb77
RL
2088 die "Something went wrong with $sourced/$f: $!\n" unless $template;
2089 my @text =
2090 split /^/m,
2091 $template->fill_in(HASH => { config => \%config,
2092 target => \%target,
9e04edf2 2093 disabled => \%disabled,
f59d0131 2094 withargs => \%withargs,
9fe2bb77
RL
2095 builddir => abs2rel($buildd, $blddir),
2096 sourcedir => abs2rel($sourced, $blddir),
2097 buildtop => abs2rel($blddir, $blddir),
2098 sourcetop => abs2rel($srcdir, $blddir) },
2099 DELIMITERS => [ "{-", "-}" ]);
2100
2101 # The top item of this stack has the following values
2102 # -2 positive already run and we found ELSE (following ELSIF should fail)
2103 # -1 positive already run (skip until ENDIF)
2104 # 0 negatives so far (if we're at a condition, check it)
2105 # 1 last was positive (don't skip lines until next ELSE, ELSIF or ENDIF)
2106 # 2 positive ELSE (following ELSIF should fail)
2107 my @skip = ();
285daccd
RL
2108
2109 # A few useful generic regexps
2110 my $index_re = qr/\[\s*(?P<INDEX>(?:\\.|.)*?)\s*\]/;
2111 my $cond_re = qr/\[\s*(?P<COND>(?:\\.|.)*?)\s*\]/;
2112 my $attribs_re = qr/(?:\{\s*(?P<ATTRIBS>(?:\\.|.)*?)\s*\})?/;
4a739387 2113 my $value_re = qr/(?P<VALUE>.*?)/;
9fe2bb77
RL
2114 collect_information(
2115 collect_from_array([ @text ],
2116 qr/\\$/ => sub { my $l1 = shift; my $l2 = shift;
2117 $l1 =~ s/\\$//; $l1.$l2 }),
2118 # Info we're looking for
285daccd 2119 qr/^\s* IF ${cond_re} \s*$/x
635bd409 2120 => sub {
c5798e0e 2121 if (! @skip || $skip[$#skip] > 0) {
285daccd 2122 push @skip, !! $expand_variables->($+{COND});
635bd409
RL
2123 } else {
2124 push @skip, -1;
2125 }
2126 },
285daccd 2127 qr/^\s* ELSIF ${cond_re} \s*$/x
9fe2bb77
RL
2128 => sub { die "ELSIF out of scope" if ! @skip;
2129 die "ELSIF following ELSE" if abs($skip[$#skip]) == 2;
2130 $skip[$#skip] = -1 if $skip[$#skip] != 0;
285daccd 2131 $skip[$#skip] = !! $expand_variables->($+{COND})
9fe2bb77 2132 if $skip[$#skip] == 0; },
285daccd 2133 qr/^\s* ELSE \s*$/x
9fe2bb77
RL
2134 => sub { die "ELSE out of scope" if ! @skip;
2135 $skip[$#skip] = -2 if $skip[$#skip] != 0;
2136 $skip[$#skip] = 2 if $skip[$#skip] == 0; },
285daccd 2137 qr/^\s* ENDIF \s*$/x
9fe2bb77
RL
2138 => sub { die "ENDIF out of scope" if ! @skip;
2139 pop @skip; },
4a739387 2140 qr/^\s* ${variable_re} \s* = \s* ${value_re} \s* $/x
26fe9b07
RL
2141 => sub {
2142 if (!@skip || $skip[$#skip] > 0) {
285daccd 2143 $variables{$+{VARIABLE}} = $expand_variables->($+{VALUE});
26fe9b07
RL
2144 }
2145 },
4a739387 2146 qr/^\s* SUBDIRS \s* = \s* ${value_re} \s* $/x
7f73eafe
RL
2147 => sub {
2148 if (!@skip || $skip[$#skip] > 0) {
285daccd 2149 foreach (tokenize($expand_variables->($+{VALUE}))) {
7f73eafe
RL
2150 push @build_dirs, [ @curd, splitdir($_, 1) ];
2151 }
2152 }
2153 },
4a739387 2154 qr/^\s* PROGRAMS ${attribs_re} \s* = \s* ${value_re} \s* $/x
ad5be194
RL
2155 => sub { $push_to->(\@programs, undef,
2156 \$attributes{programs}, $+{ATTRIBS},
2157 tokenize($expand_variables->($+{VALUE})))
2158 if !@skip || $skip[$#skip] > 0; },
4a739387 2159 qr/^\s* LIBS ${attribs_re} \s* = \s* ${value_re} \s* $/x
ad5be194
RL
2160 => sub { $push_to->(\@libraries, undef,
2161 \$attributes{libraries}, $+{ATTRIBS},
2162 tokenize($expand_variables->($+{VALUE})))
2163 if !@skip || $skip[$#skip] > 0; },
4a739387 2164 qr/^\s* MODULES ${attribs_re} \s* = \s* ${value_re} \s* $/x
ad5be194
RL
2165 => sub { $push_to->(\@modules, undef,
2166 \$attributes{modules}, $+{ATTRIBS},
2167 tokenize($expand_variables->($+{VALUE})))
2168 if !@skip || $skip[$#skip] > 0; },
4a739387 2169 qr/^\s* SCRIPTS ${attribs_re} \s* = \s* ${value_re} \s* $/x
ad5be194
RL
2170 => sub { $push_to->(\@scripts, undef,
2171 \$attributes{scripts}, $+{ATTRIBS},
2172 tokenize($expand_variables->($+{VALUE})))
2173 if !@skip || $skip[$#skip] > 0; },
4a739387 2174 qr/^\s* IMAGEDOCS ${index_re} \s* = \s* ${value_re} \s* $/x
d594d2e1
P
2175 => sub { $push_to->(\%imagedocs, $expand_variables->($+{INDEX}),
2176 undef, undef,
2177 tokenize($expand_variables->($+{VALUE})))
2178 if !@skip || $skip[$#skip] > 0; },
4a739387 2179 qr/^\s* HTMLDOCS ${index_re} \s* = \s* ${value_re} \s* $/x
ad5be194
RL
2180 => sub { $push_to->(\%htmldocs, $expand_variables->($+{INDEX}),
2181 undef, undef,
2182 tokenize($expand_variables->($+{VALUE})))
2183 if !@skip || $skip[$#skip] > 0; },
4a739387 2184 qr/^\s* MANDOCS ${index_re} \s* = \s* ${value_re} \s* $/x
ad5be194
RL
2185 => sub { $push_to->(\%mandocs, $expand_variables->($+{INDEX}),
2186 undef, undef,
2187 tokenize($expand_variables->($+{VALUE})))
2188 if !@skip || $skip[$#skip] > 0; },
4a739387 2189 qr/^\s* SOURCE ${index_re} ${attribs_re} \s* = \s* ${value_re} \s* $/x
ad5be194 2190 => sub { $push_to->(\%sources, $expand_variables->($+{INDEX}),
dd05c793 2191 \$attributes{sources}, $+{ATTRIBS},
ad5be194
RL
2192 tokenize($expand_variables->($+{VALUE})))
2193 if !@skip || $skip[$#skip] > 0; },
4a739387 2194 qr/^\s* SHARED_SOURCE ${index_re} ${attribs_re} \s* = \s* ${value_re} \s* $/x
ad5be194 2195 => sub { $push_to->(\%shared_sources, $expand_variables->($+{INDEX}),
dd05c793 2196 \$attributes{sources}, $+{ATTRIBS},
ad5be194
RL
2197 tokenize($expand_variables->($+{VALUE})))
2198 if !@skip || $skip[$#skip] > 0; },
4a739387 2199 qr/^\s* INCLUDE ${index_re} \s* = \s* ${value_re} \s* $/x
ad5be194
RL
2200 => sub { $push_to->(\%includes, $expand_variables->($+{INDEX}),
2201 undef, undef,
2202 tokenize($expand_variables->($+{VALUE})))
2203 if !@skip || $skip[$#skip] > 0; },
4a739387 2204 qr/^\s* DEFINE ${index_re} \s* = \s* ${value_re} \s* $/x
ad5be194
RL
2205 => sub { $push_to->(\%defines, $expand_variables->($+{INDEX}),
2206 undef, undef,
2207 tokenize($expand_variables->($+{VALUE})))
2208 if !@skip || $skip[$#skip] > 0; },
4a739387 2209 qr/^\s* DEPEND ${index_re} ${attribs_re} \s* = \s* ${value_re} \s* $/x
ad5be194
RL
2210 => sub { $push_to->(\%depends, $expand_variables->($+{INDEX}),
2211 \$attributes{depends}, $+{ATTRIBS},
2212 tokenize($expand_variables->($+{VALUE})))
2213 if !@skip || $skip[$#skip] > 0; },
4a739387 2214 qr/^\s* GENERATE ${index_re} \s* = \s* ${value_re} \s* $/x
ad5be194 2215 => sub { $push_to->(\%generate, $expand_variables->($+{INDEX}),
7058b4db 2216 undef, undef, $expand_variables->($+{VALUE}))
ad5be194 2217 if !@skip || $skip[$#skip] > 0; },
285daccd 2218 qr/^\s* (?:\#.*)? $/x => sub { },
2b6b606c
RL
2219 "OTHERWISE" => sub { die "Something wrong with this line:\n$_\nat $sourced/$f" },
2220 "BEFORE" => sub {
2221 if ($buildinfo_debug) {
2222 print STDERR "DEBUG: Parsing ",join(" ", @_),"\n";
2223 print STDERR "DEBUG: ... before parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
2224 }
2225 },
2226 "AFTER" => sub {
2227 if ($buildinfo_debug) {
2228 print STDERR "DEBUG: .... after parsing, skip stack is ",join(" ", map { int($_) } @skip),"\n";
2229 }
2230 },
9fe2bb77
RL
2231 );
2232 die "runaway IF?" if (@skip);
2233
285daccd 2234 if (grep { defined $attributes{modules}->{$_}->{engine} } keys %attributes
1842f369
RL
2235 and !$config{dynamic_engines}) {
2236 die <<"EOF"
19ab5790 2237ENGINES can only be used if configured with 'dynamic-engine'.
9fe2bb77
RL
2238This is usually a fault in a build.info file.
2239EOF
1842f369 2240 }
7f5af797 2241
c91f24d4
RL
2242 {
2243 my %infos = ( programs => [ @programs ],
2244 libraries => [ @libraries ],
1842f369 2245 modules => [ @modules ],
da7e31e0 2246 scripts => [ @scripts ] );
c91f24d4
RL
2247 foreach my $k (keys %infos) {
2248 foreach (@{$infos{$k}}) {
2249 my $item = cleanfile($buildd, $_, $blddir);
2250 $unified_info{$k}->{$item} = 1;
285daccd
RL
2251
2252 # Fix up associated attributes
2253 $unified_info{attributes}->{$k}->{$item} =
2254 $attributes{$k}->{$_}
2255 if defined $attributes{$k}->{$_};
c91f24d4
RL
2256 }
2257 }
8a67946e
RL
2258 }
2259
f5fb6f05
RL
2260 # Check that we haven't defined any library as both shared and
2261 # explicitly static. That is forbidden.
2262 my @doubles = ();
2263 foreach (grep /\.a$/, keys %{$unified_info{libraries}}) {
2264 (my $l = $_) =~ s/\.a$//;
2265 push @doubles, $l if defined $unified_info{libraries}->{$l};
9fe2bb77 2266 }
f5fb6f05
RL
2267 die "these libraries are both explicitly static and shared:\n ",
2268 join(" ", @doubles), "\n"
2269 if @doubles;
9fe2bb77 2270
9fe2bb77
RL
2271 foreach (keys %sources) {
2272 my $dest = $_;
2e963849 2273 my $ddest = cleanfile($buildd, $_, $blddir);
9fe2bb77 2274 foreach (@{$sources{$dest}}) {
2e963849 2275 my $s = cleanfile($sourced, $_, $blddir);
9fe2bb77 2276
19cf4404
RL
2277 # If it's generated or we simply don't find it in the source
2278 # tree, we assume it's in the build tree.
2279 if ($s eq $src_configdata || $generate{$_} || ! -f $s) {
2e963849 2280 $s = cleanfile($buildd, $_, $blddir);
9fe2bb77 2281 }
dd05c793 2282 my $o = $_;
ea241958
RL
2283 # We recognise C++, C and asm files
2284 if ($s =~ /\.(cc|cpp|c|s|S)$/) {
507f8380 2285 push @{$check_exist{$s}}, $ddest;
ea241958
RL
2286 $o =~ s/\.[csS]$/.o/; # C and assembler
2287 $o =~ s/\.(cc|cpp)$/_cc.o/; # C++
2e963849 2288 $o = cleanfile($buildd, $o, $blddir);
bec2db18
RL
2289 $unified_info{sources}->{$ddest}->{$o} = -1;
2290 $unified_info{sources}->{$o}->{$s} = -1;
83900628
RS
2291 } elsif ($s =~ /\.rc$/) {
2292 # We also recognise resource files
507f8380 2293 push @{$check_exist{$s}}, $ddest;
83900628 2294 $o =~ s/\.rc$/.res/; # Resource configuration
5e16ac14 2295 $o = cleanfile($buildd, $o, $blddir);
bec2db18
RL
2296 $unified_info{sources}->{$ddest}->{$o} = -1;
2297 $unified_info{sources}->{$o}->{$s} = -1;
9fe2bb77 2298 } else {
507f8380 2299 push @{$check_exist{$s}}, $ddest;
9fe2bb77
RL
2300 $unified_info{sources}->{$ddest}->{$s} = 1;
2301 }
dd05c793
RL
2302 # Fix up associated attributes
2303 if ($o ne $_) {
2304 $unified_info{attributes}->{sources}->{$ddest}->{$o} =
2305 $unified_info{attributes}->{sources}->{$o}->{$s} =
2306 $attributes{sources}->{$dest}->{$_}
2307 if defined $attributes{sources}->{$dest}->{$_};
2308 } else {
2309 $unified_info{attributes}->{sources}->{$ddest}->{$s} =
2310 $attributes{sources}->{$dest}->{$_}
2311 if defined $attributes{sources}->{$dest}->{$_};
2312 }
9fe2bb77
RL
2313 }
2314 }
2315
2a08d1a0
RL
2316 foreach (keys %shared_sources) {
2317 my $dest = $_;
2318 my $ddest = cleanfile($buildd, $_, $blddir);
2a08d1a0
RL
2319 foreach (@{$shared_sources{$dest}}) {
2320 my $s = cleanfile($sourced, $_, $blddir);
2321
19cf4404
RL
2322 # If it's generated or we simply don't find it in the source
2323 # tree, we assume it's in the build tree.
2324 if ($s eq $src_configdata || $generate{$_} || ! -f $s) {
2a08d1a0
RL
2325 $s = cleanfile($buildd, $_, $blddir);
2326 }
ccce3e1d 2327
dd05c793 2328 my $o = $_;
ea241958 2329 if ($s =~ /\.(cc|cpp|c|s|S)$/) {
ccce3e1d 2330 # We recognise C++, C and asm files
507f8380 2331 push @{$check_exist{$s}}, $ddest;
ea241958
RL
2332 $o =~ s/\.[csS]$/.o/; # C and assembler
2333 $o =~ s/\.(cc|cpp)$/_cc.o/; # C++
2a08d1a0 2334 $o = cleanfile($buildd, $o, $blddir);
bec2db18
RL
2335 $unified_info{shared_sources}->{$ddest}->{$o} = -1;
2336 $unified_info{sources}->{$o}->{$s} = -1;
ccce3e1d
RL
2337 } elsif ($s =~ /\.rc$/) {
2338 # We also recognise resource files
507f8380 2339 push @{$check_exist{$s}}, $ddest;
ccce3e1d 2340 $o =~ s/\.rc$/.res/; # Resource configuration
5e16ac14 2341 $o = cleanfile($buildd, $o, $blddir);
bec2db18
RL
2342 $unified_info{shared_sources}->{$ddest}->{$o} = -1;
2343 $unified_info{sources}->{$o}->{$s} = -1;
ef2dfc99
RL
2344 } elsif ($s =~ /\.ld$/) {
2345 # We also recognise linker scripts (or corresponding)
ccce3e1d 2346 # We know they are generated files
507f8380 2347 push @{$check_exist{$s}}, $ddest;
dd05c793
RL
2348 $o = cleanfile($buildd, $_, $blddir);
2349 $unified_info{shared_sources}->{$ddest}->{$o} = 1;
2a08d1a0
RL
2350 } else {
2351 die "unrecognised source file type for shared library: $s\n";
2352 }
dd05c793
RL
2353 # Fix up associated attributes
2354 if ($o ne $_) {
2355 $unified_info{attributes}->{shared_sources}->{$ddest}->{$o} =
2356 $unified_info{attributes}->{sources}->{$o}->{$s} =
2357 $attributes{sources}->{$dest}->{$_}
2358 if defined $attributes{sources}->{$dest}->{$_};
2359 } else {
2360 $unified_info{attributes}->{shared_sources}->{$ddest}->{$o} =
2361 $attributes{sources}->{$dest}->{$_}
2362 if defined $attributes{sources}->{$dest}->{$_};
2363 }
2a08d1a0
RL
2364 }
2365 }
2366
ae4c7450
RL
2367 foreach (keys %generate) {
2368 my $dest = $_;
2369 my $ddest = cleanfile($buildd, $_, $blddir);
ae4c7450 2370 die "more than one generator for $dest: "
cff64af5
RL
2371 ,join(" ", @{$generate{$_}}),"\n"
2372 if scalar @{$generate{$_}} > 1;
ae4c7450 2373 my @generator = split /\s+/, $generate{$dest}->[0];
cff64af5
RL
2374 my $gen = $generator[0];
2375 $generator[0] = cleanfile($sourced, $gen, $blddir);
2376
19cf4404 2377 # If the generator is itself generated, it's in the build tree
79f47ef5 2378 if ($generate{$gen} || ! -f $generator[0]) {
cff64af5
RL
2379 $generator[0] = cleanfile($buildd, $gen, $blddir);
2380 }
2497e2e7 2381 $check_generate{$ddest}->{$generator[0]}++;
cff64af5 2382
ae4c7450
RL
2383 $unified_info{generate}->{$ddest} = [ @generator ];
2384 }
2385
9fe2bb77
RL
2386 foreach (keys %depends) {
2387 my $dest = $_;
3f399e37
RL
2388 my $ddest = $dest;
2389
2390 if ($dest =~ /^\|(.*)\|$/) {
2391 # Collect the raw target
2392 $unified_info{targets}->{$1} = 1;
2393 $ddest = $1;
2394 } elsif ($dest eq '') {
2395 $ddest = '';
2396 } else {
2397 $ddest = cleanfile($sourced, $_, $blddir);
8d34daf0 2398
3f399e37
RL
2399 # If the destination doesn't exist in source, it can only be
2400 # a generated file in the build tree.
2401 if ($ddest eq $src_configdata || ! -f $ddest) {
2402 $ddest = cleanfile($buildd, $_, $blddir);
2403 }
9fe2bb77
RL
2404 }
2405 foreach (@{$depends{$dest}}) {
2e963849 2406 my $d = cleanfile($sourced, $_, $blddir);
2d101b0f 2407 my $d2 = cleanfile($buildd, $_, $blddir);
9fe2bb77 2408
e737d7b1
RL
2409 # If we know it's generated, or assume it is because we can't
2410 # find it in the source tree, we set file we depend on to be
19cf4404 2411 # in the build tree rather than the source tree.
846e4c4d 2412 if ($d eq $src_configdata
2d101b0f
RL
2413 || (grep { $d2 eq $_ }
2414 keys %{$unified_info{generate}})
19cf4404 2415 || ! -f $d) {
2d101b0f 2416 $d = $d2;
9fe2bb77 2417 }
9fe2bb77 2418 $unified_info{depends}->{$ddest}->{$d} = 1;
9eba5933
RL
2419
2420 # Fix up associated attributes
2421 $unified_info{attributes}->{depends}->{$ddest}->{$d} =
2422 $attributes{depends}->{$dest}->{$_}
2423 if defined $attributes{depends}->{$dest}->{$_};
9fe2bb77
RL
2424 }
2425 }
2426
2427 foreach (keys %includes) {
2428 my $dest = $_;
8d34daf0
RL
2429 my $ddest = cleanfile($sourced, $_, $blddir);
2430
2431 # If the destination doesn't exist in source, it can only be
2432 # a generated file in the build tree.
846e4c4d 2433 if ($ddest eq $src_configdata || ! -f $ddest) {
8d34daf0 2434 $ddest = cleanfile($buildd, $_, $blddir);
9fe2bb77
RL
2435 }
2436 foreach (@{$includes{$dest}}) {
4748f890
RL
2437 my $is = cleandir($sourced, $_, $blddir);
2438 my $ib = cleandir($buildd, $_, $blddir);
2439 push @{$unified_info{includes}->{$ddest}->{source}}, $is
2440 unless grep { $_ eq $is } @{$unified_info{includes}->{$ddest}->{source}};
2441 push @{$unified_info{includes}->{$ddest}->{build}}, $ib
2442 unless grep { $_ eq $ib } @{$unified_info{includes}->{$ddest}->{build}};
9fe2bb77
RL
2443 }
2444 }
b96ab5e6 2445
ef57f799
RL
2446 foreach my $dest (keys %defines) {
2447 my $ddest;
b96ab5e6 2448
ef57f799
RL
2449 if ($dest ne "") {
2450 $ddest = cleanfile($sourced, $dest, $blddir);
2451
2452 # If the destination doesn't exist in source, it can only
2453 # be a generated file in the build tree.
2454 if (! -f $ddest) {
2455 $ddest = cleanfile($buildd, $dest, $blddir);
b96ab5e6
RL
2456 }
2457 }
ef57f799
RL
2458 foreach my $v (@{$defines{$dest}}) {
2459 $v =~ m|^([^=]*)(=.*)?$|;
b96ab5e6 2460 die "0 length macro name not permitted\n" if $1 eq "";
ef57f799
RL
2461 if ($dest ne "") {
2462 die "$1 defined more than once\n"
2463 if defined $unified_info{defines}->{$ddest}->{$1};
2464 $unified_info{defines}->{$ddest}->{$1} = $2;
2465 } else {
2466 die "$1 defined more than once\n"
2467 if grep { $v eq $_ } @{$config{defines}};
2468 push @{$config{defines}}, $v;
2469 }
b96ab5e6
RL
2470 }
2471 }
829f86bb 2472
d594d2e1
P
2473 foreach my $section (keys %imagedocs) {
2474 foreach (@{$imagedocs{$section}}) {
2475 my $imagedocs = cleanfile($buildd, $_, $blddir);
2476 $unified_info{imagedocs}->{$section}->{$imagedocs} = 1;
2477 }
2478 }
2479
829f86bb
RL
2480 foreach my $section (keys %htmldocs) {
2481 foreach (@{$htmldocs{$section}}) {
2482 my $htmldocs = cleanfile($buildd, $_, $blddir);
2483 $unified_info{htmldocs}->{$section}->{$htmldocs} = 1;
2484 }
2485 }
2486
2487 foreach my $section (keys %mandocs) {
2488 foreach (@{$mandocs{$section}}) {
2489 my $mandocs = cleanfile($buildd, $_, $blddir);
2490 $unified_info{mandocs}->{$section}->{$mandocs} = 1;
2491 }
2492 }
9fe2bb77
RL
2493 }
2494
d201dbc9
RL
2495 my $ordinals_text = join(', ', sort keys %ordinals);
2496 warn <<"EOF" if $ordinals_text;
2497
2498WARNING: ORDINALS were specified for $ordinals_text
2499They are ignored and should be replaced with a combination of GENERATE,
2500DEPEND and SHARED_SOURCE.
2501EOF
2502
507f8380
RL
2503 # Check that each generated file is only generated once
2504 my $ambiguous_generation = 0;
2505 foreach (sort keys %check_generate) {
2506 my @generators = sort keys %{$check_generate{$_}};
2507 my $generators_txt = join(', ', @generators);
2508 if (scalar @generators > 1) {
2509 warn "$_ is GENERATEd by more than one generator ($generators_txt)\n";
2510 $ambiguous_generation++;
2511 }
2497e2e7
RL
2512 if ($check_generate{$_}->{$generators[0]} > 1) {
2513 warn "INFO: $_ has more than one GENERATE declaration (same generator)\n"
2514 }
507f8380
RL
2515 }
2516 die "There are ambiguous source file generations\n"
2517 if $ambiguous_generation > 0;
2518
2519 # All given source files should exist, or if generated, their
2520 # generator should exist. This loop ensures this is true.
2521 my $missing = 0;
2522 foreach my $orig (sort keys %check_exist) {
2523 foreach my $dest (@{$check_exist{$orig}}) {
2524 if ($orig ne $src_configdata) {
2525 if ($orig =~ /\.a$/) {
2526 # Static library names may be used as sources, so we
2527 # need to detect those and give them special treatment.
2528 unless (grep { $_ eq $orig }
2529 keys %{$unified_info{libraries}}) {
2530 warn "$orig is given as source for $dest, but no such library is built\n";
2531 $missing++;
2532 }
2533 } else {
2534 # A source may be generated, and its generator may be
2535 # generated as well. We therefore loop to dig out the
2536 # first generator.
2537 my $gen = $orig;
2538
2539 while (my @next = keys %{$check_generate{$gen}}) {
2540 $gen = $next[0];
2541 }
2542
2543 if (! -f $gen) {
2544 if ($gen ne $orig) {
2545 $missing++;
2546 warn "$orig is given as source for $dest, but its generator (leading to $gen) is missing\n";
2547 } else {
2548 $missing++;
2549 warn "$orig is given as source for $dest, but is missing\n";
2550 }
2551 }
2552 }
2553 }
2554 }
2555 }
2556 die "There are files missing\n" if $missing > 0;
51583cb8
RL
2557
2558 # Go through the sources of all libraries and check that the same basename
2559 # doesn't appear more than once. Some static library archivers depend on
2560 # them being unique.
2561 {
2562 my $err = 0;
2563 foreach my $prod (keys %{$unified_info{libraries}}) {
2564 my @prod_sources =
2565 map { keys %{$unified_info{sources}->{$_}} }
2566 keys %{$unified_info{sources}->{$prod}};
2567 my %srccnt = ();
2568
2569 # Count how many times a given each source basename
2570 # appears for each product.
2571 foreach my $src (@prod_sources) {
2572 $srccnt{basename $src}++;
2573 }
2574
2575 foreach my $src (keys %srccnt) {
2576 if ((my $cnt = $srccnt{$src}) > 1) {
2577 print STDERR "$src appears $cnt times for the product $prod\n";
2578 $err++
2579 }
2580 }
2581 }
2582 die if $err > 0;
2583 }
2584
1b5ad51f
RL
2585 # Massage the result
2586
e431bcfa
RL
2587 # If we depend on a header file or a perl module, add an inclusion of
2588 # its directory to allow smoothe inclusion
2589 foreach my $dest (keys %{$unified_info{depends}}) {
2590 next if $dest eq "";
2591 foreach my $d (keys %{$unified_info{depends}->{$dest}}) {
2592 next unless $d =~ /\.(h|pm)$/;
906032d5
RL
2593 my $i = dirname($d);
2594 my $spot =
2595 $d eq "configdata.pm" || defined($unified_info{generate}->{$d})
2596 ? 'build' : 'source';
2597 push @{$unified_info{includes}->{$dest}->{$spot}}, $i
2598 unless grep { $_ eq $i } @{$unified_info{includes}->{$dest}->{$spot}};
e431bcfa
RL
2599 }
2600 }
2601
ef2dfc99 2602 # Go through all intermediary files and change their names to something that
bec2db18
RL
2603 # reflects what they will be built for. Note that for some source files,
2604 # this leads to duplicate object files because they are used multiple times.
2605 # the goal is to rename all object files according to this scheme:
2606 # {productname}-{midfix}-{origobjname}.[o|res]
2607 # the {midfix} is a keyword indicating the type of product, which is mostly
2608 # valuable for libraries since they come in two forms.
2609 #
2610 # This also reorganises the {sources} and {shared_sources} so that the
2611 # former only contains ALL object files that are supposed to end up in
2612 # static libraries and programs, while the latter contains ALL object files
2613 # that are supposed to end up in shared libraries and DSOs.
2614 # The main reason for having two different source structures is to allow
2615 # the same name to be used for the static and the shared variants of a
2616 # library.
2617 {
2618 # Take copies so we don't get interference from added stuff
2619 my %unified_copy = ();
2620 foreach (('sources', 'shared_sources')) {
2621 $unified_copy{$_} = { %{$unified_info{$_}} }
2622 if defined($unified_info{$_});
2623 delete $unified_info{$_};
2624 }
1842f369 2625 foreach my $prodtype (('programs', 'libraries', 'modules', 'scripts')) {
bec2db18
RL
2626 # $intent serves multi purposes:
2627 # - give a prefix for the new object files names
2628 # - in the case of libraries, rearrange the object files so static
2629 # libraries use the 'sources' structure exclusively, while shared
2630 # libraries use the 'shared_sources' structure exclusively.
2631 my $intent = {
2632 programs => { bin => { src => [ 'sources' ],
2633 dst => 'sources' } },
2634 libraries => { lib => { src => [ 'sources' ],
2635 dst => 'sources' },
2636 shlib => { prodselect =>
2637 sub { grep !/\.a$/, @_ },
2638 src => [ 'sources',
2639 'shared_sources' ],
2640 dst => 'shared_sources' } },
22b41467
RL
2641 modules => { dso => { src => [ 'sources' ],
2642 dst => 'sources' } },
bec2db18
RL
2643 scripts => { script => { src => [ 'sources' ],
2644 dst => 'sources' } }
2645 } -> {$prodtype};
2646 foreach my $kind (keys %$intent) {
856b1b65
RL
2647 next if ($intent->{$kind}->{dst} eq 'shared_sources'
2648 && $disabled{shared});
2649
bec2db18
RL
2650 my @src = @{$intent->{$kind}->{src}};
2651 my $dst = $intent->{$kind}->{dst};
2652 my $prodselect = $intent->{$kind}->{prodselect} // sub { @_ };
2653 foreach my $prod ($prodselect->(keys %{$unified_info{$prodtype}})) {
2654 # %prod_sources has all applicable objects as keys, and
2655 # their corresponding sources as values
2656 my %prod_sources =
2657 map { $_ => [ keys %{$unified_copy{sources}->{$_}} ] }
2658 map { keys %{$unified_copy{$_}->{$prod}} }
2659 @src;
2660 foreach (keys %prod_sources) {
ef2dfc99
RL
2661 # Only affect object files and resource files,
2662 # the others simply get a new value
2663 # (+1 instead of -1)
bec2db18
RL
2664 if ($_ =~ /\.(o|res)$/) {
2665 (my $prodname = $prod) =~ s|\.a$||;
2666 my $newobj =
2667 catfile(dirname($_),
2668 basename($prodname)
2669 . '-' . $kind
2670 . '-' . basename($_));
2671 $unified_info{$dst}->{$prod}->{$newobj} = 1;
2672 foreach my $src (@{$prod_sources{$_}}) {
2673 $unified_info{sources}->{$newobj}->{$src} = 1;
dd05c793
RL
2674 # Adjust source attributes
2675 my $attrs = $unified_info{attributes}->{sources};
2676 if (defined $attrs->{$prod}
2677 && defined $attrs->{$prod}->{$_}) {
2678 $attrs->{$prod}->{$newobj} =
2679 $attrs->{$prod}->{$_};
2680 delete $attrs->{$prod}->{$_};
2681 }
2682 foreach my $objsrc (keys %{$attrs->{$_} // {}}) {
2683 $attrs->{$newobj}->{$objsrc} =
2684 $attrs->{$_}->{$objsrc};
2685 delete $attrs->{$_}->{$objsrc};
2686 }
bec2db18
RL
2687 }
2688 # Adjust dependencies
2689 foreach my $deps (keys %{$unified_info{depends}->{$_}}) {
2690 $unified_info{depends}->{$_}->{$deps} = -1;
2691 $unified_info{depends}->{$newobj}->{$deps} = 1;
2692 }
2693 # Adjust includes
2694 foreach my $k (('source', 'build')) {
2695 next unless
2696 defined($unified_info{includes}->{$_}->{$k});
2697 my @incs = @{$unified_info{includes}->{$_}->{$k}};
2698 $unified_info{includes}->{$newobj}->{$k} = [ @incs ];
2699 }
2700 } else {
2701 $unified_info{$dst}->{$prod}->{$_} = 1;
2702 }
2703 }
2704 }
2705 }
2706 }
2707 }
51583cb8 2708
bec2db18
RL
2709 # At this point, we have a number of sources with the value -1. They
2710 # aren't part of the local build and are probably meant for a different
2711 # platform, and can therefore be cleaned away. That happens when making
2712 # %unified_info more efficient below.
2713
9fe2bb77
RL
2714 ### Make unified_info a bit more efficient
2715 # One level structures
3f399e37 2716 foreach (("programs", "libraries", "modules", "scripts", "targets")) {
9fe2bb77
RL
2717 $unified_info{$_} = [ sort keys %{$unified_info{$_}} ];
2718 }
2719 # Two level structures
829f86bb 2720 foreach my $l1 (("sources", "shared_sources", "ldadd", "depends",
d594d2e1 2721 "imagedocs", "htmldocs", "mandocs")) {
9fe2bb77 2722 foreach my $l2 (sort keys %{$unified_info{$l1}}) {
bec2db18
RL
2723 my @items =
2724 sort
2725 grep { $unified_info{$l1}->{$l2}->{$_} > 0 }
2726 keys %{$unified_info{$l1}->{$l2}};
2727 if (@items) {
2728 $unified_info{$l1}->{$l2} = [ @items ];
2729 } else {
2730 delete $unified_info{$l1}->{$l2};
2731 }
9fe2bb77
RL
2732 }
2733 }
b96ab5e6
RL
2734 # Defines
2735 foreach my $dest (sort keys %{$unified_info{defines}}) {
2736 $unified_info{defines}->{$dest}
2737 = [ map { $_.$unified_info{defines}->{$dest}->{$_} }
2738 sort keys %{$unified_info{defines}->{$dest}} ];
2739 }
4748f890
RL
2740 # Includes
2741 foreach my $dest (sort keys %{$unified_info{includes}}) {
2742 if (defined($unified_info{includes}->{$dest}->{build})) {
e431bcfa
RL
2743 my @source_includes = ();
2744 @source_includes = ( @{$unified_info{includes}->{$dest}->{source}} )
2745 if defined($unified_info{includes}->{$dest}->{source});
4748f890
RL
2746 $unified_info{includes}->{$dest} =
2747 [ @{$unified_info{includes}->{$dest}->{build}} ];
2748 foreach my $inc (@source_includes) {
2749 push @{$unified_info{includes}->{$dest}}, $inc
2750 unless grep { $_ eq $inc } @{$unified_info{includes}->{$dest}};
2751 }
609e4be8 2752 } elsif (defined($unified_info{includes}->{$dest}->{source})) {
4748f890
RL
2753 $unified_info{includes}->{$dest} =
2754 [ @{$unified_info{includes}->{$dest}->{source}} ];
609e4be8
RL
2755 } else {
2756 delete $unified_info{includes}->{$dest};
4748f890
RL
2757 }
2758 }
b6e66075
RL
2759
2760 # For convenience collect information regarding directories where
2761 # files are generated, those generated files and the end product
2762 # they end up in where applicable. Then, add build rules for those
2763 # directories
2764 my %loopinfo = ( "lib" => [ @{$unified_info{libraries}} ],
1842f369 2765 "dso" => [ @{$unified_info{modules}} ],
b6e66075 2766 "bin" => [ @{$unified_info{programs}} ],
829f86bb 2767 "script" => [ @{$unified_info{scripts}} ],
d594d2e1
P
2768 "docs" => [ (map { @{$unified_info{imagedocs}->{$_} // []} }
2769 keys %{$unified_info{imagedocs} // {}}),
2770 (map { @{$unified_info{htmldocs}->{$_} // []} }
829f86bb
RL
2771 keys %{$unified_info{htmldocs} // {}}),
2772 (map { @{$unified_info{mandocs}->{$_} // []} }
2773 keys %{$unified_info{mandocs} // {}}) ] );
b6e66075
RL
2774 foreach my $type (keys %loopinfo) {
2775 foreach my $product (@{$loopinfo{$type}}) {
2776 my %dirs = ();
2777 my $pd = dirname($product);
2778
3bed01a0 2779 foreach (@{$unified_info{sources}->{$product} // []},
b6e66075
RL
2780 @{$unified_info{shared_sources}->{$product} // []}) {
2781 my $d = dirname($_);
2782
2783 # We don't want to create targets for source directories
2784 # when building out of source
2785 next if ($config{sourcedir} ne $config{builddir}
2786 && $d =~ m|^\Q$config{sourcedir}\E|);
2787 # We already have a "test" target, and the current directory
2788 # is just silly to make a target for
2789 next if $d eq "test" || $d eq ".";
2790
2791 $dirs{$d} = 1;
2792 push @{$unified_info{dirinfo}->{$d}->{deps}}, $_
2793 if $d ne $pd;
2794 }
2795 foreach (keys %dirs) {
2796 push @{$unified_info{dirinfo}->{$_}->{products}->{$type}},
2797 $product;
2798 }
2799 }
2800 }
9fe2bb77
RL
2801}
2802
2803# For the schemes that need it, we provide the old *_obj configs
2804# from the *_asm_obj ones
3a55c92b 2805foreach (grep /_(asm|aux)_src$/, keys %target) {
9fe2bb77 2806 my $src = $_;
3a55c92b 2807 (my $obj = $_) =~ s/_(asm|aux)_src$/_obj/;
ea241958
RL
2808 $target{$obj} = $target{$src};
2809 $target{$obj} =~ s/\.[csS]\b/.o/g; # C and assembler
2810 $target{$obj} =~ s/\.(cc|cpp)\b/_cc.o/g; # C++
9fe2bb77
RL
2811}
2812
291e94df
RL
2813# Write down our configuration where it fits #########################
2814
1f86b822
RL
2815my %template_vars = (
2816 config => \%config,
2817 target => \%target,
2818 disablables => \@disablables,
2819 disablables_int => \@disablables_int,
2820 disabled => \%disabled,
2821 withargs => \%withargs,
2822 unified_info => \%unified_info,
2823 tls => \@tls,
2824 dtls => \@dtls,
2825 makevars => [ sort keys %user ],
2826 disabled_info => \%disabled_info,
2827 user_crossable => \@user_crossable,
291e94df 2828);
1f86b822
RL
2829my $configdata_outname = 'configdata.pm';
2830print "Creating $configdata_outname\n";
2831open CONFIGDATA, ">$configdata_outname.new"
2660b7cf 2832 or die "Trying to create $configdata_outname.new: $!";
1f86b822
RL
2833my $configdata_tmplname = cleanfile($srcdir, "configdata.pm.in", $blddir);
2834my $configdata_tmpl =
2835 OpenSSL::Template->new(TYPE => 'FILE', SOURCE => $configdata_tmplname);
2836$configdata_tmpl->fill_in(
2837 FILENAME => $configdata_tmplname,
2838 OUTPUT => \*CONFIGDATA,
2839 HASH => { %template_vars,
2840 autowarntext => [
2841 'WARNING: do not edit!',
2842 "Generated by Configure from $configdata_tmplname",
2843 ] }
2844) or die $Text::Template::ERROR;
2845close CONFIGDATA;
2846rename "$configdata_outname.new", $configdata_outname;
b1fafff6
RL
2847if ($builder_platform eq 'unix') {
2848 my $mode = (0755 & ~umask);
2849 chmod $mode, 'configdata.pm'
2850 or warn sprintf("WARNING: Couldn't change mode for 'configdata.pm' to 0%03o: %s\n",$mode,$!);
2851}
cba5068d 2852
1f86b822
RL
2853print "Running $configdata_outname\n";
2854my $perlcmd = (quotify("maybeshell", $config{PERL}))[0];
2855my $cmd = "$perlcmd $configdata_outname";
2856#print STDERR "DEBUG[run_dofile]: \$cmd = $cmd\n";
2857system($cmd);
2858exit 1 if $? != 0;
fce0ba5f 2859
8937a4ed
RL
2860$SIG{__DIE__} = $orig_death_handler;
2861
9c62a279 2862print <<"EOF" if ($disabled{threads} eq "unavailable");
5f8d5c96
BM
2863
2864The library could not be configured for supporting multi-threaded
2865applications as the compiler options required on this system are not known.
f828ba03 2866See file INSTALL.md for details if you need multi-threading.
ec577822
BM
2867EOF
2868
76ffb43d 2869print <<"EOF" if ($no_shared_warn);
2964ba8c 2870
ae48242c
RL
2871The options 'shared', 'pic' and 'dynamic-engine' aren't supported on this
2872platform, so we will pretend you gave the option 'no-pic', which also disables
2873'shared' and 'dynamic-engine'. If you know how to implement shared libraries
2874or position independent code, please let us know (but please first make sure
2875you have tried with a current version of OpenSSL).
2e31ef03
RS
2876EOF
2877
d0364dcc 2878print $banner;
820e414d 2879
d02b48c6
RE
2880exit(0);
2881
bd5192b1
RL
2882######################################################################
2883#
2884# Helpers and utility functions
2885#
2886
8937a4ed
RL
2887# Death handler, to print a helpful message in case of failure #######
2888#
2889sub death_handler {
eb807d53 2890 die @_ if $^S; # To prevent the added message in eval blocks
8937a4ed 2891 my $build_file = $target{build_file} // "build file";
eb807d53 2892 my @message = ( <<"_____", @_ );
8937a4ed
RL
2893
2894Failure! $build_file wasn't produced.
1dc1ea18 2895Please read INSTALL.md and associated NOTES-* files. You may also have to
f828ba03 2896look over your available compiler tool chain or change your configuration.
8937a4ed
RL
2897
2898_____
eb807d53
RL
2899
2900 # Dying is terminal, so it's ok to reset the signal handler here.
2901 $SIG{__DIE__} = $orig_death_handler;
2902 die @message;
8937a4ed
RL
2903}
2904
bd5192b1
RL
2905# Configuration file reading #########################################
2906
1f2e1cd5
RL
2907# Note: All of the helper functions are for lazy evaluation. They all
2908# return a CODE ref, which will return the intended value when evaluated.
2909# Thus, whenever there's mention of a returned value, it's about that
2910# intended value.
2911
1f2e1cd5
RL
2912# Helper function to implement conditional value variants, with a default
2913# plus additional values based on the value of $config{build_type}.
2914# Arguments are given in hash table form:
2915#
2916# picker(default => "Basic string: ",
2917# debug => "debug",
2918# release => "release")
2919#
2920# When configuring with --debug, the resulting string will be
2921# "Basic string: debug", and when not, it will be "Basic string: release"
2922#
2923# This can be used to create variants of sets of flags according to the
2924# build type:
2925#
2926# cflags => picker(default => "-Wall",
2927# debug => "-g -O0",
2928# release => "-O3")
2929#
2930sub picker {
2931 my %opts = @_;
2932 return sub { add($opts{default} || (),
2933 $opts{$config{build_type}} || ())->(); }
2934}
2935
2936# Helper function to combine several values of different types into one.
2937# This is useful if you want to combine a string with the result of a
2938# lazy function, such as:
2939#
2940# cflags => combine("-Wall", sub { $disabled{zlib} ? () : "-DZLIB" })
2941#
2942sub combine {
2943 my @stuff = @_;
2944 return sub { add(@stuff)->(); }
2945}
2946
2947# Helper function to implement conditional values depending on the value
2948# of $disabled{threads}. Can be used as follows:
2949#
2950# cflags => combine("-Wall", threads("-pthread"))
2951#
2952sub threads {
2953 my @flags = @_;
2954 return sub { add($disabled{threads} ? () : @flags)->(); }
2955}
2956
60aa6c1a
AP
2957sub shared {
2958 my @flags = @_;
2959 return sub { add($disabled{shared} ? () : @flags)->(); }
2960}
1f2e1cd5 2961
9c62a279 2962our $add_called = 0;
88087414
RL
2963# Helper function to implement adding values to already existing configuration
2964# values. It handles elements that are ARRAYs, CODEs and scalars
2965sub _add {
2966 my $separator = shift;
2967
bcb1977b
RL
2968 # If there's any ARRAY in the collection of values OR the separator
2969 # is undef, we will return an ARRAY of combined values, otherwise a
2970 # string of joined values with $separator as the separator.
2971 my $found_array = !defined($separator);
88087414
RL
2972
2973 my @values =
84f32c84
DMSP
2974 map {
2975 my $res = $_;
2976 while (ref($res) eq "CODE") {
2977 $res = $res->();
2978 }
2979 if (defined($res)) {
2980 if (ref($res) eq "ARRAY") {
2981 $found_array = 1;
2982 @$res;
2983 } else {
2984 $res;
2985 }
2986 } else {
2987 ();
2988 }
88087414
RL
2989 } (@_);
2990
9c62a279
RL
2991 $add_called = 1;
2992
88087414 2993 if ($found_array) {
84f32c84 2994 [ @values ];
88087414 2995 } else {
84f32c84 2996 join($separator, grep { defined($_) && $_ ne "" } @values);
88087414
RL
2997 }
2998}
2999sub add_before {
bdcd83e1
RL
3000 my $separator = " ";
3001 if (ref($_[$#_]) eq "HASH") {
3002 my $opts = pop;
3003 $separator = $opts->{separator};
3004 }
88087414
RL
3005 my @x = @_;
3006 sub { _add($separator, @x, @_) };
3007}
3008sub add {
bdcd83e1
RL
3009 my $separator = " ";
3010 if (ref($_[$#_]) eq "HASH") {
3011 my $opts = pop;
3012 $separator = $opts->{separator};
3013 }
88087414
RL
3014 my @x = @_;
3015 sub { _add($separator, @_, @x) };
3016}
3017
3b6c4b07
RL
3018sub read_eval_file {
3019 my $fname = shift;
3020 my $content;
3021 my @result;
3022
3023 open F, "< $fname" or die "Can't open '$fname': $!\n";
3024 {
3025 undef local $/;
3026 $content = <F>;
3027 }
3028 close F;
3029 {
3030 local $@;
3031
3032 @result = ( eval $content );
3033 warn $@ if $@;
3034 }
3035 return wantarray ? @result : $result[0];
3036}
3037
bd5192b1
RL
3038# configuration reader, evaluates the input file as a perl script and expects
3039# it to fill %targets with target configurations. Those are then added to
3040# %table.
3041sub read_config {
3042 my $fname = shift;
3b6c4b07
RL
3043 my %targets;
3044
bd5192b1 3045 {
84f32c84
DMSP
3046 # Protect certain tables from tampering
3047 local %table = ();
bd5192b1 3048
84f32c84 3049 %targets = read_eval_file($fname);
bd5192b1 3050 }
225f980d
RL
3051 my %preexisting = ();
3052 foreach (sort keys %targets) {
3053 $preexisting{$_} = 1 if $table{$_};
3054 }
3055 die <<"EOF",
3056The following config targets from $fname
3057shadow pre-existing config targets with the same name:
3058EOF
3059 map { " $_\n" } sort keys %preexisting
3060 if %preexisting;
3061
bd5192b1
RL
3062
3063 # For each target, check that it's configured with a hash table.
3064 foreach (keys %targets) {
84f32c84
DMSP
3065 if (ref($targets{$_}) ne "HASH") {
3066 if (ref($targets{$_}) eq "") {
3067 warn "Deprecated target configuration for $_, ignoring...\n";
3068 } else {
3069 warn "Misconfigured target configuration for $_ (should be a hash table), ignoring...\n";
3070 }
3071 delete $targets{$_};
3072 } else {
ee9b0bbb
RL
3073 $targets{$_}->{_conf_fname_int} = add([ $fname ]);
3074 }
bd5192b1
RL
3075 }
3076
3077 %table = (%table, %targets);
3078
3079}
3080
8483a003
F
3081# configuration resolver. Will only resolve all the lazy evaluation
3082# codeblocks for the chosen target and all those it inherits from,
bd5192b1
RL
3083# recursively
3084sub resolve_config {
3085 my $target = shift;
3086 my @breadcrumbs = @_;
3087
c4718849 3088# my $extra_checks = defined($ENV{CONFIGURE_EXTRA_CHECKS});
9c62a279 3089
bd5192b1 3090 if (grep { $_ eq $target } @breadcrumbs) {
84f32c84
DMSP
3091 die "inherit_from loop! target backtrace:\n "
3092 ,$target,"\n ",join("\n ", @breadcrumbs),"\n";
bd5192b1
RL
3093 }
3094
3095 if (!defined($table{$target})) {
84f32c84
DMSP
3096 warn "Warning! target $target doesn't exist!\n";
3097 return ();
bd5192b1
RL
3098 }
3099 # Recurse through all inheritances. They will be resolved on the
3100 # fly, so when this operation is done, they will all just be a
3101 # bunch of attributes with string values.
3102 # What we get here, though, are keys with references to lists of
3103 # the combined values of them all. We will deal with lists after
3104 # this stage is done.
3105 my %combined_inheritance = ();
3106 if ($table{$target}->{inherit_from}) {
84f32c84
DMSP
3107 my @inherit_from =
3108 map { ref($_) eq "CODE" ? $_->() : $_ } @{$table{$target}->{inherit_from}};
3109 foreach (@inherit_from) {
3110 my %inherited_config = resolve_config($_, $target, @breadcrumbs);
3111
3112 # 'template' is a marker that's considered private to
3113 # the config that had it.
3114 delete $inherited_config{template};
3115
3116 foreach (keys %inherited_config) {
3117 if (!$combined_inheritance{$_}) {
3118 $combined_inheritance{$_} = [];
3119 }
3120 push @{$combined_inheritance{$_}}, $inherited_config{$_};
3121 }
3122 }
bd5192b1
RL
3123 }
3124
3125 # We won't need inherit_from in this target any more, since we've
3126 # resolved all the inheritances that lead to this
3127 delete $table{$target}->{inherit_from};
3128
3129 # Now is the time to deal with those lists. Here's the place to
3130 # decide what shall be done with those lists, all based on the
3131 # values of the target we're currently dealing with.
3132 # - If a value is a coderef, it will be executed with the list of
3133 # inherited values as arguments.
3134 # - If the corresponding key doesn't have a value at all or is the
8483a003 3135 # empty string, the inherited value list will be run through the
bd5192b1
RL
3136 # default combiner (below), and the result becomes this target's
3137 # value.
3138 # - Otherwise, this target's value is assumed to be a string that
3139 # will simply override the inherited list of values.
a26d8be9 3140 my $default_combiner = add();
bd5192b1
RL
3141
3142 my %all_keys =
84f32c84
DMSP
3143 map { $_ => 1 } (keys %combined_inheritance,
3144 keys %{$table{$target}});
b0b92a5b
RL
3145
3146 sub process_values {
84f32c84
DMSP
3147 my $object = shift;
3148 my $inherited = shift; # Always a [ list ]
3149 my $target = shift;
3150 my $entry = shift;
b0b92a5b 3151
9c62a279
RL
3152 $add_called = 0;
3153
b0b92a5b
RL
3154 while(ref($object) eq "CODE") {
3155 $object = $object->(@$inherited);
3156 }
3157 if (!defined($object)) {
3158 return ();
3159 }
3160 elsif (ref($object) eq "ARRAY") {
9c62a279 3161 local $add_called; # To make sure recursive calls don't affect it
b0b92a5b
RL
3162 return [ map { process_values($_, $inherited, $target, $entry) }
3163 @$object ];
3164 } elsif (ref($object) eq "") {
3165 return $object;
3166 } else {
3167 die "cannot handle reference type ",ref($object)
3168 ," found in target ",$target," -> ",$entry,"\n";
3169 }
3170 }
3171
317bedd6
AR
3172 foreach my $key (sort keys %all_keys) {
3173 my $previous = $combined_inheritance{$key};
bd5192b1 3174
84f32c84
DMSP
3175 # Current target doesn't have a value for the current key?
3176 # Assign it the default combiner, the rest of this loop body
3177 # will handle it just like any other coderef.
317bedd6
AR
3178 if (!exists $table{$target}->{$key}) {
3179 $table{$target}->{$key} = $default_combiner;
84f32c84 3180 }
bd5192b1 3181
317bedd6
AR
3182 $table{$target}->{$key} = process_values($table{$target}->{$key},
3183 $combined_inheritance{$key},
3184 $target, $key);
3185 unless(defined($table{$target}->{$key})) {
3186 delete $table{$target}->{$key};
b0b92a5b 3187 }
c4718849 3188# if ($extra_checks &&
317bedd6
AR
3189# $previous && !($add_called || $previous ~~ $table{$target}->{$key})) {
3190# warn "$key got replaced in $target\n";
c4718849 3191# }
bd5192b1
RL
3192 }
3193
3194 # Finally done, return the result.
3195 return %{$table{$target}};
3196}
3197
462ba4f6 3198sub usage
84f32c84
DMSP
3199 {
3200 print STDERR $usage;
3201 print STDERR "\npick os/compiler from:\n";
3202 my $j=0;
3203 my $i;
10a926c1 3204 my $k=0;
84f32c84
DMSP
3205 foreach $i (sort keys %table)
3206 {
3207 next if $table{$i}->{template};
3208 next if $i =~ /^debug/;
3209 $k += length($i) + 1;
3210 if ($k > 78)
3211 {
3212 print STDERR "\n";
3213 $k=length($i);
3214 }
3215 print STDERR $i . " ";
3216 }
3217 foreach $i (sort keys %table)
3218 {
3219 next if $table{$i}->{template};
3220 next if $i !~ /^debug/;
3221 $k += length($i) + 1;
3222 if ($k > 78)
3223 {
3224 print STDERR "\n";
3225 $k=length($i);
3226 }
3227 print STDERR $i . " ";
3228 }
84f32c84
DMSP
3229 exit(1);
3230 }
d02b48c6 3231
6d75a83c
RL
3232sub compiler_predefined {
3233 state %predefined;
41d6e0f3 3234 my $cc = shift;
6d75a83c
RL
3235
3236 return () if $^O eq 'VMS';
3237
41d6e0f3
AP
3238 die 'compiler_predefined called without a compiler command'
3239 unless $cc;
6d75a83c 3240
41d6e0f3 3241 if (! $predefined{$cc}) {
6d75a83c 3242
41d6e0f3 3243 $predefined{$cc} = {};
6d75a83c
RL
3244
3245 # collect compiler pre-defines from gcc or gcc-alike...
3246 open(PIPE, "$cc -dM -E -x c /dev/null 2>&1 |");
3247 while (my $l = <PIPE>) {
3248 $l =~ m/^#define\s+(\w+(?:\(\w+\))?)(?:\s+(.+))?/ or last;
41d6e0f3 3249 $predefined{$cc}->{$1} = $2 // '';
6d75a83c
RL
3250 }
3251 close(PIPE);
3252 }
3253
41d6e0f3 3254 return %{$predefined{$cc}};
6d75a83c
RL
3255}
3256
656bbdc6
AP
3257sub which
3258{
3259 my ($name)=@_;
3260
3261 if (eval { require IPC::Cmd; 1; }) {
3262 IPC::Cmd->import();
3263 return scalar IPC::Cmd::can_run($name);
3264 } else {
3265 # if there is $directories component in splitpath,
3266 # then it's not something to test with $PATH...
3267 return $name if (File::Spec->splitpath($name))[1];
3268
3269 foreach (File::Spec->path()) {
3270 my $fullpath = catfile($_, "$name$target{exe_extension}");
3271 if (-f $fullpath and -x $fullpath) {
3272 return $fullpath;
3273 }
3274 }
3275 }
3276}
3277
7ecdf18d
RL
3278sub env
3279{
3280 my $name = shift;
ac6ae8a9 3281 my %opts = @_;
7ecdf18d 3282
ac6ae8a9
RL
3283 unless ($opts{cacheonly}) {
3284 # Note that if $ENV{$name} doesn't exist or is undefined,
3285 # $config{perlenv}->{$name} will be created with the value
3286 # undef. This is intentional.
89bea083 3287
ac6ae8a9
RL
3288 $config{perlenv}->{$name} = $ENV{$name}
3289 if ! exists $config{perlenv}->{$name};
3290 }
7ecdf18d
RL
3291 return $config{perlenv}->{$name};
3292}
3293
00ae96ca
RL
3294# Configuration printer ##############################################
3295
3296sub print_table_entry
3297{
f770d75b
AP
3298 local $now_printing = shift;
3299 my %target = resolve_config($now_printing);
00ae96ca
RL
3300 my $type = shift;
3301
3302 # Don't print the templates
3303 return if $target{template};
3304
3305 my @sequence = (
84f32c84
DMSP
3306 "sys_id",
3307 "cpp",
3308 "cppflags",
3309 "defines",
3310 "includes",
3311 "cc",
3312 "cflags",
84f32c84
DMSP
3313 "ld",
3314 "lflags",
3315 "loutflag",
3316 "ex_libs",
3317 "bn_ops",
37fe90ad
RL
3318 "enable",
3319 "disable",
84f32c84
DMSP
3320 "poly1035_asm_src",
3321 "thread_scheme",
3322 "perlasm_scheme",
3323 "dso_scheme",
3324 "shared_target",
3325 "shared_cflag",
3326 "shared_defines",
3327 "shared_ldflag",
3328 "shared_rcflag",
3329 "shared_extension",
3330 "dso_extension",
3331 "obj_extension",
3332 "exe_extension",
3333 "ranlib",
3334 "ar",
3335 "arflags",
3336 "aroutflag",
3337 "rc",
3338 "rcflags",
3339 "rcoutflag",
3340 "mt",
3341 "mtflags",
3342 "mtinflag",
3343 "mtoutflag",
3344 "multilib",
3345 "build_scheme",
3346 );
00ae96ca
RL
3347
3348 if ($type eq "TABLE") {
84f32c84
DMSP
3349 print "\n";
3350 print "*** $now_printing\n";
cb212f23
RL
3351 foreach (@sequence) {
3352 if (ref($target{$_}) eq "ARRAY") {
3353 printf "\$%-12s = %s\n", $_, join(" ", @{$target{$_}});
3354 } else {
3355 printf "\$%-12s = %s\n", $_, $target{$_};
3356 }
3357 }
00ae96ca 3358 } elsif ($type eq "HASH") {
84f32c84
DMSP
3359 my $largest =
3360 length((sort { length($a) <=> length($b) } @sequence)[-1]);
3361 print " '$now_printing' => {\n";
3362 foreach (@sequence) {
3363 if ($target{$_}) {
cb212f23
RL
3364 if (ref($target{$_}) eq "ARRAY") {
3365 print " '",$_,"'"," " x ($largest - length($_))," => [ ",join(", ", map { "'$_'" } @{$target{$_}})," ],\n";
3366 } else {
3367 print " '",$_,"'"," " x ($largest - length($_))," => '",$target{$_},"',\n";
3368 }
84f32c84
DMSP
3369 }
3370 }
3371 print " },\n";
00ae96ca
RL
3372 }
3373}
3374
3375# Utility routines ###################################################
3376
2e963849
RL
3377# On VMS, if the given file is a logical name, File::Spec::Functions
3378# will consider it an absolute path. There are cases when we want a
3379# purely syntactic check without checking the environment.
3380sub isabsolute {
3381 my $file = shift;
3382
3383 # On non-platforms, we just use file_name_is_absolute().
3384 return file_name_is_absolute($file) unless $^O eq "VMS";
3385
69687aa8 3386 # If the file spec includes a device or a directory spec,
2e963849
RL
3387 # file_name_is_absolute() is perfectly safe.
3388 return file_name_is_absolute($file) if $file =~ m|[:\[]|;
3389
3390 # Here, we know the given file spec isn't absolute
3391 return 0;
3392}
3393
ec182ef0
RL
3394# Makes a directory absolute and cleans out /../ in paths like foo/../bar
3395# On some platforms, this uses rel2abs(), while on others, realpath() is used.
3396# realpath() requires that at least all path components except the last is an
3397# existing directory. On VMS, the last component of the directory spec must
3398# exist.
3399sub absolutedir {
3400 my $dir = shift;
3401
3402 # realpath() is quite buggy on VMS. It uses LIB$FID_TO_NAME, which
3403 # will return the volume name for the device, no matter what. Also,
3404 # it will return an incorrect directory spec if the argument is a
3405 # directory that doesn't exist.
3406 if ($^O eq "VMS") {
3407 return rel2abs($dir);
3408 }
3409
3410 # We use realpath() on Unix, since no other will properly clean out
3411 # a directory spec.
3412 use Cwd qw/realpath/;
3413
3414 return realpath($dir);
3415}
3416
610e2b3b
RL
3417# Check if all paths are one and the same, using stat. They must both exist
3418# We need this for the cases when File::Spec doesn't detect case insensitivity
3419# (File::Spec::Unix assumes case sensitivity)
3420sub samedir {
3421 die "samedir expects two arguments\n" unless scalar @_ == 2;
3422
3423 my @stat0 = stat($_[0]); # First argument
3424 my @stat1 = stat($_[1]); # Second argument
3425
3426 die "Couldn't stat $_[0]" unless @stat0;
3427 die "Couldn't stat $_[1]" unless @stat1;
3428
3429 # Compare device number
3430 return 0 unless ($stat0[0] == $stat1[0]);
3431 # Compare "inode". The perl manual recommends comparing as
3432 # string rather than as number.
3433 return 0 unless ($stat0[1] eq $stat1[1]);
3434
3435 return 1; # All the same
3436}
3437
fe05264e
RL
3438sub quotify {
3439 my %processors = (
84f32c84
DMSP
3440 perl => sub { my $x = shift;
3441 $x =~ s/([\\\$\@"])/\\$1/g;
3442 return '"'.$x.'"'; },
3443 maybeshell => sub { my $x = shift;
3444 (my $y = $x) =~ s/([\\\"])/\\$1/g;
3445 if ($x ne $y || $x =~ m|\s|) {
3446 return '"'.$y.'"';
3447 } else {
3448 return $x;
3449 }
3450 },
3451 );
fe05264e
RL
3452 my $for = shift;
3453 my $processor =
84f32c84 3454 defined($processors{$for}) ? $processors{$for} : sub { shift; };
fe05264e 3455
2110febb 3456 return map { $processor->($_); } @_;
fe05264e 3457}
107b5792 3458
9fe2bb77
RL
3459# collect_from_file($filename, $line_concat_cond_re, $line_concat)
3460# $filename is a file name to read from
3461# $line_concat_cond_re is a regexp detecting a line continuation ending
3462# $line_concat is a CODEref that takes care of concatenating two lines
3463sub collect_from_file {
3464 my $filename = shift;
3465 my $line_concat_cond_re = shift;
3466 my $line_concat = shift;
3467
3468 open my $fh, $filename || die "unable to read $filename: $!\n";
3469 return sub {
3470 my $saved_line = "";
3471 $_ = "";
3472 while (<$fh>) {
04f171c0 3473 s|\R$||;
9fe2bb77
RL
3474 if (defined $line_concat) {
3475 $_ = $line_concat->($saved_line, $_);
3476 $saved_line = "";
3477 }
3478 if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
3479 $saved_line = $_;
3480 next;
3481 }
3482 return $_;
3483 }
3484 die "$filename ending with continuation line\n" if $_;
3485 close $fh;
3486 return undef;
3487 }
3488}
3489
3490# collect_from_array($array, $line_concat_cond_re, $line_concat)
3491# $array is an ARRAYref of lines
3492# $line_concat_cond_re is a regexp detecting a line continuation ending
3493# $line_concat is a CODEref that takes care of concatenating two lines
3494sub collect_from_array {
3495 my $array = shift;
3496 my $line_concat_cond_re = shift;
3497 my $line_concat = shift;
3498 my @array = (@$array);
3499
3500 return sub {
3501 my $saved_line = "";
3502 $_ = "";
3503 while (defined($_ = shift @array)) {
04f171c0 3504 s|\R$||;
9fe2bb77
RL
3505 if (defined $line_concat) {
3506 $_ = $line_concat->($saved_line, $_);
3507 $saved_line = "";
3508 }
3509 if (defined $line_concat_cond_re && /$line_concat_cond_re/) {
3510 $saved_line = $_;
3511 next;
3512 }
3513 return $_;
3514 }
3515 die "input text ending with continuation line\n" if $_;
3516 return undef;
3517 }
3518}
3519
3520# collect_information($lineiterator, $line_continue, $regexp => $CODEref, ...)
3521# $lineiterator is a CODEref that delivers one line at a time.
107b5792
RL
3522# All following arguments are regex/CODEref pairs, where the regexp detects a
3523# line and the CODEref does something with the result of the regexp.
3524sub collect_information {
9fe2bb77 3525 my $lineiterator = shift;
107b5792
RL
3526 my %collectors = @_;
3527
9fe2bb77 3528 while(defined($_ = $lineiterator->())) {
04f171c0 3529 s|\R$||;
9fe2bb77 3530 my $found = 0;
2b6b606c
RL
3531 if ($collectors{"BEFORE"}) {
3532 $collectors{"BEFORE"}->($_);
3533 }
9fe2bb77 3534 foreach my $re (keys %collectors) {
2b6b606c 3535 if ($re !~ /^OTHERWISE|BEFORE|AFTER$/ && /$re/) {
9fe2bb77
RL
3536 $collectors{$re}->($lineiterator);
3537 $found = 1;
3538 };
3539 }
3540 if ($collectors{"OTHERWISE"}) {
3541 $collectors{"OTHERWISE"}->($lineiterator, $_)
3542 unless $found || !defined $collectors{"OTHERWISE"};
3543 }
2b6b606c
RL
3544 if ($collectors{"AFTER"}) {
3545 $collectors{"AFTER"}->($_);
3546 }
107b5792 3547 }
107b5792 3548}
ce959812
RL
3549
3550# tokenize($line)
5d3af259 3551# tokenize($line,$separator)
ce959812 3552# $line is a line of text to split up into tokens
5d3af259
RL
3553# $separator [optional] is a regular expression that separates the tokens,
3554# the default being spaces. Do not use quotes of any kind as separators,
3555# that will give undefined results.
3556# Returns a list of tokens.
ce959812 3557#
5d3af259
RL
3558# Tokens are divided by separator (spaces by default). If the tokens include
3559# the separators, they have to be quoted with single or double quotes.
3560# Double quotes inside a double quoted token must be escaped. Escaping is done
ce959812
RL
3561# with backslash.
3562# Basically, the same quoting rules apply for " and ' as in any
3563# Unix shell.
3564sub tokenize {
3565 my $line = my $debug_line = shift;
5d3af259 3566 my $separator = shift // qr|\s+|;
ce959812
RL
3567 my @result = ();
3568
5d3af259
RL
3569 if ($ENV{CONFIGURE_DEBUG_TOKENIZE}) {
3570 print STDERR "DEBUG[tokenize]: \$separator = $separator\n";
3571 }
3572
3573 while ($line =~ s|^${separator}||, $line ne "") {
ce959812 3574 my $token = "";
5d3af259
RL
3575 again:
3576 $line =~ m/^(.*?)(${separator}|"|'|$)/;
3577 $token .= $1;
3578 $line = $2.$';
3579
3580 if ($line =~ m/^"((?:[^"\\]+|\\.)*)"/) {
3581 $token .= $1;
3582 $line = $';
3583 goto again;
3584 } elsif ($line =~ m/^'([^']*)'/) {
3585 $token .= $1;
3586 $line = $';
3587 goto again;
ce959812
RL
3588 }
3589 push @result, $token;
3590 }
3591
3592 if ($ENV{CONFIGURE_DEBUG_TOKENIZE}) {
5d3af259
RL
3593 print STDERR "DEBUG[tokenize]: Parsed '$debug_line' into:\n";
3594 print STDERR "DEBUG[tokenize]: ('", join("', '", @result), "')\n";
ce959812
RL
3595 }
3596 return @result;
3597}