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