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